| OLD | NEW |
| 1 /* | 1 /* |
| 2 * This file is part of Adblock Plus <https://adblockplus.org/>, | 2 * This file is part of Adblock Plus <https://adblockplus.org/>, |
| 3 * Copyright (C) 2006-2015 Eyeo GmbH | 3 * Copyright (C) 2006-2015 Eyeo GmbH |
| 4 * | 4 * |
| 5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
| 6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
| 7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
| 8 * | 8 * |
| 9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
| 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 12 * GNU General Public License for more details. | 12 * GNU General Public License for more details. |
| 13 * | 13 * |
| 14 * You should have received a copy of the GNU General Public License | 14 * You should have received a copy of the GNU General Public License |
| 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. | 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. |
| 16 */ | 16 */ |
| 17 | 17 |
| 18 "use strict"; | 18 "use strict"; |
| 19 | 19 |
| 20 let {ilog2, nextPow2, alloc, dealloc, addBuffer, removeBuffer} = require("typedO
bjects/utils"); | 20 let {ilog2, nextPow2, alloc, fixedPropertyDescriptor, dealloc, addBuffer, remove
Buffer} = require("typedObjects/utils"); |
| 21 | 21 |
| 22 function defaultArrayConstructor() | 22 function defaultArrayConstructor() |
| 23 { | 23 { |
| 24 this.arrayBufferIndex = -1; | 24 this.arrayBufferIndex = -1; |
| 25 }; | 25 }; |
| 26 | 26 |
| 27 function defaultArrayDestructor() | 27 function defaultArrayDestructor() |
| 28 { | 28 { |
| 29 this.length = 0; | 29 if (this._copy) |
| 30 this.size = 0; | 30 { |
| 31 // Carefully zero length + size if this is a slice of an array |
| 32 Object.defineProperties(this, { |
| 33 length: fixedPropertyDescriptor(0), |
| 34 size: fixedPropertyDescriptor(0) |
| 35 }); |
| 36 } |
| 37 else |
| 38 { |
| 39 // Remove read-only length property for previously sliced arrays |
| 40 if (this.hasOwnProperty("length")) |
| 41 delete this.length; |
| 42 |
| 43 this.length = 0; |
| 44 this.size = 0; |
| 45 } |
| 31 }; | 46 }; |
| 32 | 47 |
| 33 function createGetter(elementGetter, elementShift) | 48 function createGetter(elementGetter, elementShift) |
| 34 { | 49 { |
| 35 return function(index) | 50 return function(index) |
| 36 { | 51 { |
| 37 if (index < 0 || index >= this.length) | 52 if (index < 0 || index >= this.length) |
| 38 throw new Error("Array index out of bounds"); | 53 throw new Error("Array index out of bounds"); |
| 39 return elementGetter.call(this, this.arrayBufferIndex, this.arrayByteOffset
+ (index << elementShift)); | 54 return elementGetter.call(this, this.arrayBufferIndex, this.arrayByteOffset
+ (index << elementShift)); |
| 40 }; | 55 }; |
| (...skipping 164 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 205 { | 220 { |
| 206 let length = this.length | 0; | 221 let length = this.length | 0; |
| 207 if (length == 0) | 222 if (length == 0) |
| 208 throw new Error("No elements in the array"); | 223 throw new Error("No elements in the array"); |
| 209 | 224 |
| 210 let result = this.get(length - 1); | 225 let result = this.get(length - 1); |
| 211 this.length = this.length - 1; | 226 this.length = this.length - 1; |
| 212 return result; | 227 return result; |
| 213 } | 228 } |
| 214 | 229 |
| 230 function createSlicer(elementShift) |
| 231 { |
| 232 return function slice(start, end) |
| 233 { |
| 234 start = start | 0; |
| 235 end = end | 0; |
| 236 |
| 237 if (start < 0) |
| 238 start = Math.max(0, this.length + start); |
| 239 else |
| 240 start = Math.min(start, this.length); |
| 241 |
| 242 if (end > 0) |
| 243 end = Math.min(0, end - this.length); |
| 244 else |
| 245 end = Math.max(end, -1 * this.length); |
| 246 |
| 247 Object.defineProperties(this, { |
| 248 length: {value: this.length, configurable: true} |
| 249 }); |
| 250 this.retain(); |
| 251 |
| 252 let result = Object.create(this, { |
| 253 arrayByteOffset: fixedPropertyDescriptor( |
| 254 this.arrayByteOffset + start << elementShift |
| 255 ), |
| 256 length: {value: this.length - start + end, configurable: true}, |
| 257 _refCount: {value: 1, writable: true}, |
| 258 _copy: fixedPropertyDescriptor(true), |
| 259 _state: {value: 0, writable: true} |
| 260 }); |
| 261 |
| 262 // TODO |
| 263 // - A sliced array will have a read-only length even if all slices are |
| 264 // released. We could fix this by keeping a count of slices, but is it |
| 265 // worth it? |
| 266 // - A slice of an array still keeps a reference to the array in it's |
| 267 // __proto__ property after it's released. (We can't just replace it |
| 268 // with null as modifying an object's prototype is apparently very slow.) |
| 269 |
| 270 return result; |
| 271 }; |
| 272 } |
| 273 |
| 215 function splice(index, count) | 274 function splice(index, count) |
| 216 { | 275 { |
| 217 index = index | 0; | 276 index = index | 0; |
| 218 count = count | 0; | 277 count = count | 0; |
| 219 let length = this.length | 0; | 278 let length = this.length | 0; |
| 220 if (index < 0) | 279 if (index < 0) |
| 221 { | 280 { |
| 222 index += length; | 281 index += length; |
| 223 if (index < 0) | 282 if (index < 0) |
| 224 index = 0; | 283 index = 0; |
| (...skipping 93 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 318 }; | 377 }; |
| 319 typeDescriptor.length = uint32; | 378 typeDescriptor.length = uint32; |
| 320 typeDescriptor.size = uint32; | 379 typeDescriptor.size = uint32; |
| 321 typeDescriptor.push = push; | 380 typeDescriptor.push = push; |
| 322 typeDescriptor.pop = pop; | 381 typeDescriptor.pop = pop; |
| 323 typeDescriptor.splice = splice; | 382 typeDescriptor.splice = splice; |
| 324 typeDescriptor.unshift = unshift; | 383 typeDescriptor.unshift = unshift; |
| 325 typeDescriptor.shift = shift; | 384 typeDescriptor.shift = shift; |
| 326 | 385 |
| 327 let elementShift = ilog2(elementType.referenceLength | 0); | 386 let elementShift = ilog2(elementType.referenceLength | 0); |
| 387 typeDescriptor.slice = createSlicer(elementShift); |
| 328 typeDescriptor.get = createGetter(elementGetter, elementShift); | 388 typeDescriptor.get = createGetter(elementGetter, elementShift); |
| 329 typeDescriptor.set = createSetter(elementSetter, elementShift); | 389 typeDescriptor.set = createSetter(elementSetter, elementShift); |
| 330 | 390 |
| 331 if (meta.hasOwnProperty("constructor") && typeof meta.constructor == "function
") | 391 if (meta.hasOwnProperty("constructor") && typeof meta.constructor == "function
") |
| 332 meta.constructor = createCombinedConstructor(meta.constructor); | 392 meta.constructor = createCombinedConstructor(meta.constructor); |
| 333 else | 393 else |
| 334 meta.constructor = defaultArrayConstructor; | 394 meta.constructor = defaultArrayConstructor; |
| 335 | 395 |
| 336 if (meta.hasOwnProperty("destructor") && typeof meta.destructor == "function") | 396 if (meta.hasOwnProperty("destructor") && typeof meta.destructor == "function") |
| 337 meta.destructor = createCombinedDestructor(meta.destructor); | 397 meta.destructor = createCombinedDestructor(meta.destructor); |
| 338 else | 398 else |
| 339 meta.destructor = defaultArrayDestructor; | 399 meta.destructor = defaultArrayDestructor; |
| 340 | 400 |
| 341 if (!meta.watch || typeof meta.watch != "object") | 401 if (!meta.watch || typeof meta.watch != "object") |
| 342 meta.watch = {}; | 402 meta.watch = {}; |
| 343 | 403 |
| 344 meta.watch.length = createLengthWatcher(elementType, elementSetter); | 404 meta.watch.length = createLengthWatcher(elementType, elementSetter); |
| 345 meta.watch.size = createSizeWatcher(elementType, minElements, bufferSize, buff
ers, viewTypes, views, firstFree); | 405 meta.watch.size = createSizeWatcher(elementType, minElements, bufferSize, buff
ers, viewTypes, views, firstFree); |
| 346 | 406 |
| 347 let {ObjectBase} = require("typedObjects/objectTypes"); | 407 let {ObjectBase} = require("typedObjects/objectTypes"); |
| 348 return ObjectBase.extend(typeDescriptor, meta); | 408 return ObjectBase.extend(typeDescriptor, meta); |
| 349 } | 409 } |
| 350 | 410 |
| 351 exports.createArrayType = createArrayType; | 411 exports.createArrayType = createArrayType; |
| OLD | NEW |