bluebird
Version:
Full featured Promises/A+ implementation with exceptionally good performance
136 lines (120 loc) • 4.2 kB
JavaScript
/**
* Copyright (c) 2013 Petka Antonov
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:</p>
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
"use strict";
var ASSERT = require("./assert.js");
function arrayCopy( src, srcIndex, dst, dstIndex, len ) {
for( var j = 0; j < len; ++j ) {
dst[ j + dstIndex ] = src[ j + srcIndex ];
}
}
function pow2AtLeast( n ) {
n = n >>> 0;
n = n - 1;
n = n | (n >> 1);
n = n | (n >> 2);
n = n | (n >> 4);
n = n | (n >> 8);
n = n | (n >> 16);
return n + 1;
}
function getCapacity( capacity ) {
if( typeof capacity !== "number" ) return 16;
return pow2AtLeast(
Math.min(
Math.max( 16, capacity ), 1073741824 )
);
}
function Queue( capacity ) {
this._capacity = getCapacity( capacity );
this._length = 0;
this._front = 0;
this._makeCapacity();
}
Queue.prototype._willBeOverCapacity =
function Queue$_willBeOverCapacity( size ) {
return this._capacity < size;
};
Queue.prototype._pushOne = function Queue$_pushOne( arg ) {
var length = this.length();
this._checkCapacity( length + 1 );
var i = ( this._front + length ) & ( this._capacity - 1 );
this[i] = arg;
this._length = length + 1;
};
Queue.prototype.push = function Queue$push( fn, receiver, arg ) {
var length = this.length() + 3;
if( this._willBeOverCapacity( length ) ) {
this._pushOne( fn );
this._pushOne( receiver );
this._pushOne( arg );
return;
}
var j = this._front + length - 3;
this._checkCapacity( length );
var wrapMask = this._capacity - 1;
this[ ( j + 0 ) & wrapMask ] = fn;
this[ ( j + 1 ) & wrapMask ] = receiver;
this[ ( j + 2 ) & wrapMask ] = arg;
this._length = length;
};
Queue.prototype.shift = function Queue$shift() {
var front = this._front,
ret = this[ front ];
this[ front ] = void 0;
this._front = ( front + 1 ) & ( this._capacity - 1 );
this._length--;
return ret;
};
Queue.prototype.length = function Queue$length() {
return this._length;
};
Queue.prototype._makeCapacity = function Queue$_makeCapacity() {
var len = this._capacity;
for( var i = 0; i < len; ++i ) {
this[i] = void 0;
}
};
Queue.prototype._checkCapacity = function Queue$_checkCapacity( size ) {
if( this._capacity < size ) {
this._resizeTo( this._capacity << 3 );
}
};
Queue.prototype._resizeTo = function Queue$_resizeTo( capacity ) {
var oldFront = this._front;
var oldCapacity = this._capacity;
var oldQueue = new Array( oldCapacity );
var length = this.length();
arrayCopy( this, 0, oldQueue, 0, oldCapacity );
this._capacity = capacity;
this._makeCapacity();
this._front = 0;
if( oldFront + length <= oldCapacity ) {
arrayCopy( oldQueue, oldFront, this, 0, length );
}
else { var lengthBeforeWrapping =
length - ( ( oldFront + length ) & ( oldCapacity - 1 ) );
arrayCopy( oldQueue, oldFront, this, 0, lengthBeforeWrapping );
arrayCopy( oldQueue, 0, this, lengthBeforeWrapping,
length - lengthBeforeWrapping );
}
};
module.exports = Queue;