[79a0317] | 1 | // Copyright 2012 The Obvious Corporation.
|
---|
| 2 |
|
---|
| 3 | /*
|
---|
| 4 | * bufs: Buffer utilities.
|
---|
| 5 | */
|
---|
| 6 |
|
---|
| 7 | /*
|
---|
| 8 | * Module variables
|
---|
| 9 | */
|
---|
| 10 |
|
---|
| 11 | /** Pool of buffers, where `bufPool[x].length === x`. */
|
---|
| 12 | var bufPool = [];
|
---|
| 13 | /** Maximum length of kept temporary buffers. */
|
---|
| 14 |
|
---|
| 15 | var TEMP_BUF_MAXIMUM_LENGTH = 20;
|
---|
| 16 | /** Minimum exactly-representable 64-bit int. */
|
---|
| 17 |
|
---|
| 18 | var MIN_EXACT_INT64 = -0x8000000000000000;
|
---|
| 19 | /** Maximum exactly-representable 64-bit int. */
|
---|
| 20 |
|
---|
| 21 | var MAX_EXACT_INT64 = 0x7ffffffffffffc00;
|
---|
| 22 | /** Maximum exactly-representable 64-bit uint. */
|
---|
| 23 |
|
---|
| 24 | var MAX_EXACT_UINT64 = 0xfffffffffffff800;
|
---|
| 25 | /**
|
---|
| 26 | * The int value consisting just of a 1 in bit #32 (that is, one more
|
---|
| 27 | * than the maximum 32-bit unsigned value).
|
---|
| 28 | */
|
---|
| 29 |
|
---|
| 30 | var BIT_32 = 0x100000000;
|
---|
| 31 | /**
|
---|
| 32 | * The int value consisting just of a 1 in bit #64 (that is, one more
|
---|
| 33 | * than the maximum 64-bit unsigned value).
|
---|
| 34 | */
|
---|
| 35 |
|
---|
| 36 | var BIT_64 = 0x10000000000000000;
|
---|
| 37 | /*
|
---|
| 38 | * Helper functions
|
---|
| 39 | */
|
---|
| 40 |
|
---|
| 41 | /**
|
---|
| 42 | * Masks off all but the lowest bit set of the given number.
|
---|
| 43 | */
|
---|
| 44 |
|
---|
| 45 | function lowestBit(num) {
|
---|
| 46 | return num & -num;
|
---|
| 47 | }
|
---|
| 48 | /**
|
---|
| 49 | * Gets whether trying to add the second number to the first is lossy
|
---|
| 50 | * (inexact). The first number is meant to be an accumulated result.
|
---|
| 51 | */
|
---|
| 52 |
|
---|
| 53 |
|
---|
| 54 | function isLossyToAdd(accum, num) {
|
---|
| 55 | if (num === 0) {
|
---|
| 56 | return false;
|
---|
| 57 | }
|
---|
| 58 |
|
---|
| 59 | var lowBit = lowestBit(num);
|
---|
| 60 | var added = accum + lowBit;
|
---|
| 61 |
|
---|
| 62 | if (added === accum) {
|
---|
| 63 | return true;
|
---|
| 64 | }
|
---|
| 65 |
|
---|
| 66 | if (added - lowBit !== accum) {
|
---|
| 67 | return true;
|
---|
| 68 | }
|
---|
| 69 |
|
---|
| 70 | return false;
|
---|
| 71 | }
|
---|
| 72 | /*
|
---|
| 73 | * Exported functions
|
---|
| 74 | */
|
---|
| 75 |
|
---|
| 76 | /**
|
---|
| 77 | * Allocates a buffer of the given length, which is initialized
|
---|
| 78 | * with all zeroes. This returns a buffer from the pool if it is
|
---|
| 79 | * available, or a freshly-allocated buffer if not.
|
---|
| 80 | */
|
---|
| 81 |
|
---|
| 82 |
|
---|
| 83 | export function alloc(length) {
|
---|
| 84 | var result = bufPool[length];
|
---|
| 85 |
|
---|
| 86 | if (result) {
|
---|
| 87 | bufPool[length] = undefined;
|
---|
| 88 | } else {
|
---|
| 89 | result = new Uint8Array(length);
|
---|
| 90 | }
|
---|
| 91 |
|
---|
| 92 | result.fill(0);
|
---|
| 93 | return result;
|
---|
| 94 | }
|
---|
| 95 | /**
|
---|
| 96 | * Releases a buffer back to the pool.
|
---|
| 97 | */
|
---|
| 98 |
|
---|
| 99 | export function free(buffer) {
|
---|
| 100 | var length = buffer.length;
|
---|
| 101 |
|
---|
| 102 | if (length < TEMP_BUF_MAXIMUM_LENGTH) {
|
---|
| 103 | bufPool[length] = buffer;
|
---|
| 104 | }
|
---|
| 105 | }
|
---|
| 106 | /**
|
---|
| 107 | * Resizes a buffer, returning a new buffer. Returns the argument if
|
---|
| 108 | * the length wouldn't actually change. This function is only safe to
|
---|
| 109 | * use if the given buffer was allocated within this module (since
|
---|
| 110 | * otherwise the buffer might possibly be shared externally).
|
---|
| 111 | */
|
---|
| 112 |
|
---|
| 113 | export function resize(buffer, length) {
|
---|
| 114 | if (length === buffer.length) {
|
---|
| 115 | return buffer;
|
---|
| 116 | }
|
---|
| 117 |
|
---|
| 118 | var newBuf = alloc(length);
|
---|
| 119 |
|
---|
| 120 | for (var i = 0; i <= buffer.length; i++) {
|
---|
| 121 | newBuf[i] = buffer[i];
|
---|
| 122 | }
|
---|
| 123 |
|
---|
| 124 | free(buffer);
|
---|
| 125 | return newBuf;
|
---|
| 126 | }
|
---|
| 127 | /**
|
---|
| 128 | * Reads an arbitrary signed int from a buffer.
|
---|
| 129 | */
|
---|
| 130 |
|
---|
| 131 | export function readInt(buffer) {
|
---|
| 132 | var length = buffer.length;
|
---|
| 133 | var positive = buffer[length - 1] < 0x80;
|
---|
| 134 | var result = positive ? 0 : -1;
|
---|
| 135 | var lossy = false; // Note: We can't use bit manipulation here, since that stops
|
---|
| 136 | // working if the result won't fit in a 32-bit int.
|
---|
| 137 |
|
---|
| 138 | if (length < 7) {
|
---|
| 139 | // Common case which can't possibly be lossy (because the result has
|
---|
| 140 | // no more than 48 bits, and loss only happens with 54 or more).
|
---|
| 141 | for (var i = length - 1; i >= 0; i--) {
|
---|
| 142 | result = result * 0x100 + buffer[i];
|
---|
| 143 | }
|
---|
| 144 | } else {
|
---|
| 145 | for (var _i = length - 1; _i >= 0; _i--) {
|
---|
| 146 | var one = buffer[_i];
|
---|
| 147 | result *= 0x100;
|
---|
| 148 |
|
---|
| 149 | if (isLossyToAdd(result, one)) {
|
---|
| 150 | lossy = true;
|
---|
| 151 | }
|
---|
| 152 |
|
---|
| 153 | result += one;
|
---|
| 154 | }
|
---|
| 155 | }
|
---|
| 156 |
|
---|
| 157 | return {
|
---|
| 158 | value: result,
|
---|
| 159 | lossy: lossy
|
---|
| 160 | };
|
---|
| 161 | }
|
---|
| 162 | /**
|
---|
| 163 | * Reads an arbitrary unsigned int from a buffer.
|
---|
| 164 | */
|
---|
| 165 |
|
---|
| 166 | export function readUInt(buffer) {
|
---|
| 167 | var length = buffer.length;
|
---|
| 168 | var result = 0;
|
---|
| 169 | var lossy = false; // Note: See above in re bit manipulation.
|
---|
| 170 |
|
---|
| 171 | if (length < 7) {
|
---|
| 172 | // Common case which can't possibly be lossy (see above).
|
---|
| 173 | for (var i = length - 1; i >= 0; i--) {
|
---|
| 174 | result = result * 0x100 + buffer[i];
|
---|
| 175 | }
|
---|
| 176 | } else {
|
---|
| 177 | for (var _i2 = length - 1; _i2 >= 0; _i2--) {
|
---|
| 178 | var one = buffer[_i2];
|
---|
| 179 | result *= 0x100;
|
---|
| 180 |
|
---|
| 181 | if (isLossyToAdd(result, one)) {
|
---|
| 182 | lossy = true;
|
---|
| 183 | }
|
---|
| 184 |
|
---|
| 185 | result += one;
|
---|
| 186 | }
|
---|
| 187 | }
|
---|
| 188 |
|
---|
| 189 | return {
|
---|
| 190 | value: result,
|
---|
| 191 | lossy: lossy
|
---|
| 192 | };
|
---|
| 193 | }
|
---|
| 194 | /**
|
---|
| 195 | * Writes a little-endian 64-bit signed int into a buffer.
|
---|
| 196 | */
|
---|
| 197 |
|
---|
| 198 | export function writeInt64(value, buffer) {
|
---|
| 199 | if (value < MIN_EXACT_INT64 || value > MAX_EXACT_INT64) {
|
---|
| 200 | throw new Error("Value out of range.");
|
---|
| 201 | }
|
---|
| 202 |
|
---|
| 203 | if (value < 0) {
|
---|
| 204 | value += BIT_64;
|
---|
| 205 | }
|
---|
| 206 |
|
---|
| 207 | writeUInt64(value, buffer);
|
---|
| 208 | }
|
---|
| 209 | /**
|
---|
| 210 | * Writes a little-endian 64-bit unsigned int into a buffer.
|
---|
| 211 | */
|
---|
| 212 |
|
---|
| 213 | export function writeUInt64(value, buffer) {
|
---|
| 214 | if (value < 0 || value > MAX_EXACT_UINT64) {
|
---|
| 215 | throw new Error("Value out of range.");
|
---|
| 216 | }
|
---|
| 217 |
|
---|
| 218 | var lowWord = value % BIT_32;
|
---|
| 219 | var highWord = Math.floor(value / BIT_32);
|
---|
| 220 | buffer[0] = lowWord & 0xff;
|
---|
| 221 | buffer[1] = lowWord >> 8 & 0xff;
|
---|
| 222 | buffer[2] = lowWord >> 16 & 0xff;
|
---|
| 223 | buffer[3] = lowWord >> 24 & 0xff;
|
---|
| 224 | buffer[4] = highWord & 0xff;
|
---|
| 225 | buffer[5] = highWord >> 8 & 0xff;
|
---|
| 226 | buffer[6] = highWord >> 16 & 0xff;
|
---|
| 227 | buffer[7] = highWord >> 24 & 0xff;
|
---|
| 228 | } |
---|