Source: util/ByteBuffer.js

/*
 * Copyright 2003-2006, 2009, 2017, 2020 United States Government, as represented
 * by the Administrator of the National Aeronautics and Space Administration.
 * All rights reserved.
 *
 * The NASAWorldWind/WebWorldWind platform is licensed under the Apache License,
 * Version 2.0 (the "License"); you may not use this file except in compliance
 * with the License. You may obtain a copy of the License
 * at http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed
 * under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
 * CONDITIONS OF ANY KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations under the License.
 *
 * NASAWorldWind/WebWorldWind also contains the following 3rd party Open Source
 * software:
 *
 *    ES6-Promise – under MIT License
 *    libtess.js – SGI Free Software License B
 *    Proj4 – under MIT License
 *    JSZip – under MIT License
 *
 * A complete listing of 3rd Party software notices and licenses included in
 * WebWorldWind can be found in the WebWorldWind 3rd-party notices and licenses
 * PDF found in code  directory.
 */
/**
 * @exports ByteBuffer
 */
define(['../error/ArgumentError',
        '../util/Logger'
        ],
    function (ArgumentError,
              Logger) {
        "use strict";

        /**
         * Constructs a wrapper around an array buffer that enables byte-level access to its data.
         * This wrapper strives to minimize secondary allocations when subarrays are accessed.
         * The one exception is when double precision floating point data is access that is not properly aligned.
         * @alias ByteBuffer
         * @classdesc A structured wrapper around an array buffer that provides byte-level access to its data.
         * @param {ArrayBuffer} array An array buffer containing source data.
         * @constructor
         */
        var ByteBuffer = function(array) {
            if (!array) {
                throw new ArgumentError(
                    Logger.logMessage(Logger.LEVEL_SEVERE, "ByteBuffer", "constructor", "missingArray"));
            }

            /**
             * The raw data of the array buffer.
             * @type {ArrayBuffer}
             */
            this.array = array;

            /**
             * A data view on the array buffer.
             * This data view is used to extract integer and floating point data from that array buffer.
             * @type {DataView}
             */
            this.data = new DataView(array);

            /**
             * The current position in the array buffer.
             * This position is implicitly used to access all data.
             * @type {Number}
             */
            this.position = 0;

            /**
             * The byte order in which the data is encoded.
             * Byte order will either be big endian or little endian.
             * @type {Boolean}
             * @default ByteByffer.LITTLE_ENDIAN
             * @private
             */
            this._order = ByteBuffer.LITTLE_ENDIAN;
        };

        /**
         * Get a byte from the current position and advance the position.
         * @returns {Number}
         */
        ByteBuffer.prototype.getByte = function() {
            var result = this.data.getUint8(this.position);
            this.position += ByteBuffer.BYTE_SIZE;
            return result;
        };

        /**
         * Get a byte array from the current position and advance the position.
         * To avoid secondary allocation, a TypedArray shadows the underlying ArrayBuffer.
         * @param {Number} numBytes The number of bytes in the desired array.
         * @returns {Uint8Array}
         */
        ByteBuffer.prototype.getByteArray = function(numBytes) {
            var result = new Uint8Array(this.array, this.position, numBytes);
            this.position += ByteBuffer.BYTE_SIZE * numBytes;
            return result;
        };

        /**
         * Get a 16-bit integer from the current position and advance the position.
         * @returns {Number}
         */
        ByteBuffer.prototype.getInt16 = function() {
            var result = this.data.getInt16(this.position, this._order);
            this.position += ByteBuffer.INT16_SIZE;
            return result;
        };

        /**
         * Get a 16-bit integer array from the current position and advance the position.
         * To avoid secondary allocation, a TypedArray shadows the underlying ArrayBuffer.
         * @param {Number} numInt16s The number of 16-bit integers in the desired array.
         * @returns {Int16Array}
         */
        ByteBuffer.prototype.getInt16Array = function(numInt16s) {
            var result = new Int16Array(this.array, this.position, numInt16s);
            this.position += ByteBuffer.INT16_SIZE * numInt16s;
            return result;
        };

        /**
         * Get a 32-bit integer from the current position and advance the position.
         * @returns {Number}
         */
        ByteBuffer.prototype.getInt32 = function() {
            var result = this.data.getInt32(this.position, this._order);
            this.position += ByteBuffer.INT32_SIZE;
            return result;
        };

        /**
         * Get a single precision floating point array from the current position and advance the position.
         * To avoid secondary allocation, a TypedArray shadows the underlying ArrayBuffer.
         * @param {Number} numInt32s The number of 32-bit integers in the desired array.
         * @returns {Int32Array}
         */
        ByteBuffer.prototype.getInt32Array = function(numInt32s) {
            var result = new Int32Array(this.array, this.position, numInt32s);
            this.position += ByteBuffer.INT32_SIZE * numInt32s;
            return result;
        };

        /**
         * Get a single precision floating point number from the current position and advance the position.
         * @returns {Number}
         */
        ByteBuffer.prototype.getFloat = function() {
            var result = this.data.getFloat32(this.position, this._order);
            this.position += ByteBuffer.FLOAT_SIZE;
            return result;
        };

        /**
         * Get a single precision floating point array from the current position and advance the position.
         * To avoid secondary allocation, a TypedArray shadows the underlying ArrayBuffer.
         * @param {Number} numFloats The number of single precision floating point numbers in the desired array.
         * @returns {Float32Array}
         */
        ByteBuffer.prototype.getFloatArray = function(numFloats) {
            var result = new Float32Array(this.array, this.position, numFloats);
            this.position += ByteBuffer.FLOAT_SIZE * numFloats;
            return result;
        };

        /**
         * Get a double precision floating point number from the current position and advance the position.
         * @returns {Number}
         */
        ByteBuffer.prototype.getDouble = function() {
            var result = this.data.getFloat64(this.position, this._order);
            this.position += ByteBuffer.DOUBLE_SIZE;
            return result;
        };

        /**
         * Get a single precision floating point array from the current position and advance the position.
         * To avoid secondary allocation, a TypedArray shadows the underlying ArrayBuffer.
         * @param {Number} numDoubles The number of double precision floating point numbers in the desired array.
         * @returns {Float64Array}
         */
        ByteBuffer.prototype.getDoubleArray = function(numDoubles) {
            // Issue: Float64Array c'tor throws an exception if the starting offset is not a multiple of 8.
            // We see this in shapefiles.
            var result;
            // If the data is not DWORD aligned, ...
            if (this.position % 8 != 0) {
                var bytes = this.array.slice(this.position, this.position + numDoubles * ByteBuffer.DOUBLE_SIZE);
                result = new Float64Array(bytes);
            }
            else {
                result = new Float64Array(this.array, this.position, numDoubles);
            }
            this.position += ByteBuffer.DOUBLE_SIZE * numDoubles;
            return result;
        };

        /**
         * Skip over the specified number of bytes.
         * @param {Number} numBytes The number of bytes to skip.
         */
        ByteBuffer.prototype.skipBytes = function(numBytes) {
            this.position += numBytes * ByteBuffer.BYTE_SIZE;
        };

        /**
         * Skip over the specified number of 16-bit integers.
         * @param {Number} numInt16s The number of 16-bit integers to skip.
         */
        ByteBuffer.prototype.skipInt16s = function(numInt16s) {
            this.position += numInt16s * ByteBuffer.INT16_SIZE;
        };

        /**
         * Skip over the specified number of 32-bit integers.
         * @param {Number} numInt32s The number of 32-bit integers to skip.
         */
        ByteBuffer.prototype.skipInt32s = function(numInt32s) {
            this.position += numInt32s * ByteBuffer.INT32_SIZE;
        };

        /**
         * Skip over the specified number of single precision floating point numbers.
         * @param {Number} numFloats The number of single precision floating point numbers to skip.
         */
        ByteBuffer.prototype.skipFloats = function(numFloats) {
            this.position += numFloats * ByteBuffer.FLOAT_SIZE;
        };

        /**
         * Skip over the specified number of double precision floating point numbers.
         * @param {Number} numDoubles The number of double precision floating point numbers to skip.
         */
        ByteBuffer.prototype.skipDoubles = function(numDoubles) {
            this.position += numDoubles * ByteBuffer.DOUBLE_SIZE;
        };

        /**
         * Advance to a specific position.
         * @param {Number} position The specified position.
         */
        ByteBuffer.prototype.seek = function(position) {
            this.position = position;
        };

        /**
         * Set the byte order of the underlying data.
         * @param {Boolean} order The byte order of the underlying data.
         */
        ByteBuffer.prototype.order = function(order) {
            this._order = order;
        };

        /**
         * Return the total size of the underlying data.
         * @returns {Number} The size of the underlying data.
         */
        ByteBuffer.prototype.limit = function() {
            return this.data.byteLength;
        };

        /**
         * Indicates whether there remains any data to be accessed sequentially.
         * @returns {Boolean} True if more data can be accessed sequentially.
         */
        ByteBuffer.prototype.hasRemaining = function() {
            return this.position < this.data.byteLength;
        };

        /**
         * Access the underlying data in big endian order, where the most significant bits of the data are encountered first.
         * @type {Boolean}
         * @constant
         */
        ByteBuffer.BIG_ENDIAN = false;

        /**
         * Access the underlying data in little endian order, where the least significant bits of the data are encountered first.
         * @type {Boolean}
         * @constant
         */
        ByteBuffer.LITTLE_ENDIAN = true;

        /**
         * The size of a byte.
         * @type {Number}
         * @constant
         */
        ByteBuffer.BYTE_SIZE = 1;

        /**
         * The size of a 16-bit integer.
         * @type {Number}
         * @constant
         */
        ByteBuffer.INT16_SIZE = 2;

        /**
         * The size of a 32-bit integer.
         * @type {Number}
         * @constant
         */
        ByteBuffer.INT32_SIZE = 4;

        /**
         * The size of a single precision floating point number.
         * @type {Number}
         * @constant
         */
        ByteBuffer.FLOAT_SIZE = 4;

        /**
         * The size of a double precision floating point number.
         * @type {Number}
         * @constant
         */
        ByteBuffer.DOUBLE_SIZE = 8;

        return ByteBuffer;
    }
);