2019-07-09 05:56:05 +08:00
|
|
|
import {
|
|
|
|
ASCII_DECODER,
|
|
|
|
UTF_16BE_DECODER,
|
|
|
|
UTF_16BE_ENCODER,
|
|
|
|
UTF_16LE_DECODER,
|
|
|
|
UTF_16LE_ENCODER,
|
|
|
|
} from ".";
|
|
|
|
import { Endianness } from "..";
|
|
|
|
import { Cursor } from "./Cursor";
|
2019-07-12 19:10:51 +08:00
|
|
|
import { Vec3 } from "../vector";
|
2019-07-09 05:56:05 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A cursor for reading from an array buffer or part of an array buffer.
|
|
|
|
*/
|
|
|
|
export class ArrayBufferCursor implements Cursor {
|
|
|
|
get offset(): number {
|
|
|
|
return this.dv.byteOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
get size(): number {
|
|
|
|
return this.dv.byteLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
set size(size: number) {
|
|
|
|
this.dv = new DataView(this.buffer, this.offset, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected _position: number;
|
|
|
|
|
|
|
|
get position(): number {
|
|
|
|
return this._position;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected little_endian!: boolean;
|
|
|
|
|
|
|
|
get endianness(): Endianness {
|
|
|
|
return this.little_endian ? Endianness.Little : Endianness.Big;
|
|
|
|
}
|
|
|
|
|
|
|
|
set endianness(endianness: Endianness) {
|
|
|
|
this.little_endian = endianness === Endianness.Little;
|
|
|
|
this.utf16_decoder = this.little_endian ? UTF_16LE_DECODER : UTF_16BE_DECODER;
|
|
|
|
this.utf16_encoder = this.little_endian ? UTF_16LE_ENCODER : UTF_16BE_ENCODER;
|
|
|
|
}
|
|
|
|
|
|
|
|
get bytes_left(): number {
|
|
|
|
return this.size - this.position;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected buffer: ArrayBuffer;
|
|
|
|
protected dv: DataView;
|
|
|
|
|
|
|
|
private utf16_decoder: TextDecoder = UTF_16BE_DECODER;
|
|
|
|
private utf16_encoder: TextEncoder = UTF_16BE_ENCODER;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param buffer The buffer to read from.
|
|
|
|
* @param endianness Decides in which byte order multi-byte integers and floats will be interpreted.
|
|
|
|
* @param offset The start offset of the part that will be read from.
|
|
|
|
* @param size The size of the part that will be read from.
|
|
|
|
*/
|
|
|
|
constructor(
|
|
|
|
buffer: ArrayBuffer,
|
|
|
|
endianness: Endianness,
|
|
|
|
offset: number = 0,
|
|
|
|
size: number = buffer.byteLength
|
|
|
|
) {
|
|
|
|
this.buffer = buffer;
|
|
|
|
this.dv = new DataView(buffer, offset, size);
|
|
|
|
this.endianness = endianness;
|
|
|
|
this._position = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
seek(offset: number): this {
|
|
|
|
return this.seek_start(this.position + offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
seek_start(offset: number): this {
|
|
|
|
if (offset < 0 || offset > this.size) {
|
|
|
|
throw new Error(`Offset ${offset} is out of bounds.`);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._position = offset;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
seek_end(offset: number): this {
|
|
|
|
if (offset < 0 || offset > this.size) {
|
|
|
|
throw new Error(`Offset ${offset} is out of bounds.`);
|
|
|
|
}
|
|
|
|
|
|
|
|
this._position = this.size - offset;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8(): number {
|
|
|
|
return this.u8_at(this._position++);
|
|
|
|
}
|
|
|
|
|
|
|
|
u8_at(offset: number): number {
|
|
|
|
return this.dv.getUint8(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
u16(): number {
|
|
|
|
const r = this.u16_at(this.position);
|
|
|
|
this._position += 2;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16_at(offset: number): number {
|
|
|
|
return this.dv.getUint16(offset, this.little_endian);
|
|
|
|
}
|
|
|
|
|
|
|
|
u32(): number {
|
|
|
|
const r = this.u32_at(this.position);
|
|
|
|
this._position += 4;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32_at(offset: number): number {
|
|
|
|
return this.dv.getUint32(offset, this.little_endian);
|
|
|
|
}
|
|
|
|
|
|
|
|
i8(): number {
|
|
|
|
return this.i8_at(this._position++);
|
|
|
|
}
|
|
|
|
|
|
|
|
i8_at(offset: number): number {
|
|
|
|
return this.dv.getInt8(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
i16(): number {
|
|
|
|
const r = this.i16_at(this.position);
|
|
|
|
this._position += 2;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
i16_at(offset: number): number {
|
|
|
|
return this.dv.getInt16(offset, this.little_endian);
|
|
|
|
}
|
|
|
|
|
|
|
|
i32(): number {
|
|
|
|
const r = this.i32_at(this.position);
|
|
|
|
this._position += 4;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
i32_at(offset: number): number {
|
|
|
|
return this.dv.getInt32(offset, this.little_endian);
|
|
|
|
}
|
|
|
|
|
|
|
|
f32(): number {
|
|
|
|
const r = this.f32_at(this.position);
|
|
|
|
this._position += 4;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
f32_at(offset: number): number {
|
|
|
|
return this.dv.getFloat32(offset, this.little_endian);
|
|
|
|
}
|
|
|
|
|
|
|
|
u8_array(n: number): number[] {
|
|
|
|
const array = [];
|
|
|
|
for (let i = 0; i < n; ++i) array.push(this.dv.getUint8(this._position++));
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16_array(n: number): number[] {
|
|
|
|
const array = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < n; ++i) {
|
|
|
|
array.push(this.dv.getUint16(this.position, this.little_endian));
|
|
|
|
this._position += 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32_array(n: number): number[] {
|
|
|
|
const array = [];
|
|
|
|
|
|
|
|
for (let i = 0; i < n; ++i) {
|
|
|
|
array.push(this.dv.getUint32(this.position, this.little_endian));
|
|
|
|
this._position += 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2019-07-10 02:22:18 +08:00
|
|
|
vec3(): Vec3 {
|
|
|
|
return new Vec3(this.f32(), this.f32(), this.f32());
|
|
|
|
}
|
|
|
|
|
2019-07-09 05:56:05 +08:00
|
|
|
take(size: number): ArrayBufferCursor {
|
|
|
|
const offset = this.offset + this.position;
|
|
|
|
const wrapper = new ArrayBufferCursor(this.buffer, this.endianness, offset, size);
|
|
|
|
this._position += size;
|
|
|
|
return wrapper;
|
|
|
|
}
|
|
|
|
|
|
|
|
string_ascii(
|
|
|
|
max_byte_length: number,
|
|
|
|
null_terminated: boolean,
|
|
|
|
drop_remaining: boolean
|
|
|
|
): string {
|
|
|
|
const string_length = null_terminated
|
|
|
|
? this.index_of_u8(0, max_byte_length) - this.position
|
|
|
|
: max_byte_length;
|
|
|
|
|
|
|
|
const view = new DataView(this.buffer, this.offset + this.position, string_length);
|
|
|
|
const r = ASCII_DECODER.decode(view);
|
|
|
|
|
|
|
|
this._position += drop_remaining
|
|
|
|
? max_byte_length
|
|
|
|
: Math.min(string_length + 1, max_byte_length);
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
string_utf16(
|
|
|
|
max_byte_length: number,
|
|
|
|
null_terminated: boolean,
|
|
|
|
drop_remaining: boolean
|
|
|
|
): string {
|
|
|
|
const string_length = null_terminated
|
|
|
|
? this.index_of_u16(0, max_byte_length) - this.position
|
|
|
|
: Math.floor(max_byte_length / 2) * 2;
|
|
|
|
|
|
|
|
const view = new DataView(this.buffer, this.offset + this.position, string_length);
|
|
|
|
const r = this.utf16_decoder.decode(view);
|
|
|
|
|
|
|
|
this._position += drop_remaining
|
|
|
|
? max_byte_length
|
|
|
|
: Math.min(string_length + 2, max_byte_length);
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
array_buffer(size: number = this.size - this.position): ArrayBuffer {
|
2019-07-11 23:30:23 +08:00
|
|
|
const r = this.buffer.slice(
|
|
|
|
this.offset + this.position,
|
|
|
|
this.offset + this.position + size
|
|
|
|
);
|
2019-07-09 05:56:05 +08:00
|
|
|
this._position += size;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
copy_to_uint8_array(array: Uint8Array, size: number = this.size - this.position): this {
|
|
|
|
array.set(new Uint8Array(this.buffer, this.offset + this.position, size));
|
|
|
|
this._position += size;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
private index_of_u8(value: number, max_byte_length: number): number {
|
|
|
|
const max_pos = Math.min(this.position + max_byte_length, this.size);
|
|
|
|
|
|
|
|
for (let i = this.position; i < max_pos; ++i) {
|
|
|
|
if (this.dv.getUint8(i) === value) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.position + max_byte_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
private index_of_u16(value: number, max_byte_length: number): number {
|
|
|
|
const max_pos = Math.min(this.position + max_byte_length, this.size);
|
|
|
|
|
|
|
|
for (let i = this.position; i < max_pos; i += 2) {
|
|
|
|
if (this.dv.getUint16(i, this.little_endian) === value) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.position + max_byte_length;
|
|
|
|
}
|
|
|
|
}
|