mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
All unit tests pass again. Removed dependency on text-encoding. Refactored cursors to reuse more code.
This commit is contained in:
parent
4eeda19837
commit
f95e7ea220
@ -24,7 +24,6 @@
|
|||||||
"react-dom": "^16.8.6",
|
"react-dom": "^16.8.6",
|
||||||
"react-virtualized": "^9.21.1",
|
"react-virtualized": "^9.21.1",
|
||||||
"react-virtualized-select": "^3.1.3",
|
"react-virtualized-select": "^3.1.3",
|
||||||
"text-encoding": "^0.7.0",
|
|
||||||
"three": "^0.106.2"
|
"three": "^0.106.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
import { WritableResizableBufferCursor } from "../../cursor/WritableResizableBufferCursor";
|
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
|
|
||||||
export function prs_compress(src: Cursor): Cursor {
|
export function prs_compress(src: Cursor): Cursor {
|
||||||
@ -117,10 +117,7 @@ class Context {
|
|||||||
|
|
||||||
constructor(cursor: Cursor) {
|
constructor(cursor: Cursor) {
|
||||||
this.src = cursor;
|
this.src = cursor;
|
||||||
this.dst = new WritableResizableBufferCursor(
|
this.dst = new ResizableBufferCursor(new ResizableBuffer(cursor.size), cursor.endianness);
|
||||||
new ResizableBuffer(cursor.size),
|
|
||||||
cursor.endianness
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
set_bit(bit: number): void {
|
set_bit(bit: number): void {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
import { WritableResizableBufferCursor } from "../../cursor/WritableResizableBufferCursor";
|
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
|
|
||||||
const logger = Logger.get("data_formats/compression/prs/decompress");
|
const logger = Logger.get("data_formats/compression/prs/decompress");
|
||||||
@ -63,7 +63,7 @@ class Context {
|
|||||||
|
|
||||||
constructor(cursor: Cursor) {
|
constructor(cursor: Cursor) {
|
||||||
this.src = cursor;
|
this.src = cursor;
|
||||||
this.dst = new WritableResizableBufferCursor(
|
this.dst = new ResizableBufferCursor(
|
||||||
new ResizableBuffer(Math.floor(1.5 * cursor.size)),
|
new ResizableBuffer(Math.floor(1.5 * cursor.size)),
|
||||||
cursor.endianness
|
cursor.endianness
|
||||||
);
|
);
|
||||||
|
253
src/data_formats/cursor/AbstractCursor.ts
Normal file
253
src/data_formats/cursor/AbstractCursor.ts
Normal file
@ -0,0 +1,253 @@
|
|||||||
|
import { Endianness } from "..";
|
||||||
|
import { Vec2, Vec3 } from "../vector";
|
||||||
|
import { Cursor } from "./Cursor";
|
||||||
|
|
||||||
|
export abstract class AbstractCursor implements Cursor {
|
||||||
|
abstract readonly size: number;
|
||||||
|
|
||||||
|
protected _position: number = 0;
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
get bytes_left(): number {
|
||||||
|
return this.size - this.position;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected readonly offset: number;
|
||||||
|
|
||||||
|
protected abstract readonly backing_buffer: ArrayBuffer;
|
||||||
|
protected abstract readonly dv: DataView;
|
||||||
|
|
||||||
|
constructor(endianness: Endianness, offset: number) {
|
||||||
|
this.endianness = endianness;
|
||||||
|
this.offset = offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
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 {
|
||||||
|
this.check_offset(offset, 1);
|
||||||
|
return this.dv.getUint8(this.offset + offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
u16(): number {
|
||||||
|
const r = this.u16_at(this.position);
|
||||||
|
this._position += 2;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
u16_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 2);
|
||||||
|
return this.dv.getUint16(this.offset + offset, this.little_endian);
|
||||||
|
}
|
||||||
|
|
||||||
|
u32(): number {
|
||||||
|
const r = this.u32_at(this.position);
|
||||||
|
this._position += 4;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 4);
|
||||||
|
return this.dv.getUint32(this.offset + offset, this.little_endian);
|
||||||
|
}
|
||||||
|
|
||||||
|
i8(): number {
|
||||||
|
return this.i8_at(this._position++);
|
||||||
|
}
|
||||||
|
|
||||||
|
i8_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 1);
|
||||||
|
return this.dv.getInt8(this.offset + offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
i16(): number {
|
||||||
|
const r = this.i16_at(this.position);
|
||||||
|
this._position += 2;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
i16_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 2);
|
||||||
|
return this.dv.getInt16(this.offset + offset, this.little_endian);
|
||||||
|
}
|
||||||
|
|
||||||
|
i32(): number {
|
||||||
|
const r = this.i32_at(this.position);
|
||||||
|
this._position += 4;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
i32_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 4);
|
||||||
|
return this.dv.getInt32(this.offset + offset, this.little_endian);
|
||||||
|
}
|
||||||
|
|
||||||
|
f32(): number {
|
||||||
|
const r = this.f32_at(this.position);
|
||||||
|
this._position += 4;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
f32_at(offset: number): number {
|
||||||
|
this.check_offset(offset, 4);
|
||||||
|
return this.dv.getFloat32(this.offset + offset, this.little_endian);
|
||||||
|
}
|
||||||
|
|
||||||
|
u8_array(n: number): number[] {
|
||||||
|
this.check_size("n", n, n);
|
||||||
|
|
||||||
|
const array = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < n; ++i) {
|
||||||
|
array.push(this.dv.getUint8(this.offset + this._position++));
|
||||||
|
}
|
||||||
|
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
u16_array(n: number): number[] {
|
||||||
|
this.check_size("n", n, 2 * n);
|
||||||
|
|
||||||
|
const array = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < n; ++i) {
|
||||||
|
array.push(this.dv.getUint16(this.offset + this.position, this.little_endian));
|
||||||
|
this._position += 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32_array(n: number): number[] {
|
||||||
|
this.check_size("n", n, 4 * n);
|
||||||
|
|
||||||
|
const array = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < n; ++i) {
|
||||||
|
array.push(this.dv.getUint32(this.offset + this.position, this.little_endian));
|
||||||
|
this._position += 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
return array;
|
||||||
|
}
|
||||||
|
|
||||||
|
vec2_f32(): Vec2 {
|
||||||
|
return new Vec2(this.f32(), this.f32());
|
||||||
|
}
|
||||||
|
|
||||||
|
vec3_f32(): Vec3 {
|
||||||
|
return new Vec3(this.f32(), this.f32(), this.f32());
|
||||||
|
}
|
||||||
|
|
||||||
|
string_ascii(max_byte_length: number, null_terminated: boolean, drop_remaining: boolean) {
|
||||||
|
let code_points: number[] = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < max_byte_length; i++) {
|
||||||
|
const code_point = this.u8();
|
||||||
|
|
||||||
|
if (null_terminated && code_point === 0) {
|
||||||
|
if (drop_remaining) {
|
||||||
|
this.seek(max_byte_length - i - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
code_points.push(code_point);
|
||||||
|
}
|
||||||
|
|
||||||
|
return String.fromCodePoint(...code_points);
|
||||||
|
}
|
||||||
|
|
||||||
|
string_utf16(max_byte_length: number, null_terminated: boolean, drop_remaining: boolean) {
|
||||||
|
let code_points: number[] = [];
|
||||||
|
let len = Math.floor(max_byte_length / 2);
|
||||||
|
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
|
const code_point = this.u16();
|
||||||
|
|
||||||
|
if (null_terminated && code_point === 0) {
|
||||||
|
if (drop_remaining) {
|
||||||
|
this.seek(2 * (len - i - 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
code_points.push(code_point);
|
||||||
|
}
|
||||||
|
|
||||||
|
return String.fromCodePoint(...code_points);
|
||||||
|
}
|
||||||
|
|
||||||
|
array_buffer(size: number = this.size - this.position): ArrayBuffer {
|
||||||
|
this.check_size("size", size, size);
|
||||||
|
const r = this.backing_buffer.slice(
|
||||||
|
this.offset + this.position,
|
||||||
|
this.offset + this.position + size
|
||||||
|
);
|
||||||
|
this._position += size;
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
copy_to_uint8_array(array: Uint8Array, size: number = this.size - this.position): this {
|
||||||
|
this.check_size("size", size, size);
|
||||||
|
array.set(new Uint8Array(this.backing_buffer, this.offset + this.position, size));
|
||||||
|
this._position += size;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract take(size: number): Cursor;
|
||||||
|
|
||||||
|
protected check_size(name: string, value: number, byte_size: number): void {
|
||||||
|
if (byte_size < 0 || byte_size > this.size - this.position) {
|
||||||
|
throw new Error(`${name} ${value} is out of bounds.`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks whether we can read size bytes at offset.
|
||||||
|
*/
|
||||||
|
private check_offset(offset: number, size: number): void {
|
||||||
|
if (offset < 0 || offset + size > this.size) {
|
||||||
|
throw new Error(`Offset ${offset} is out of bounds.`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,21 +1,10 @@
|
|||||||
import { WritableCursor } from "./WritableCursor";
|
import { Vec2, Vec3 } from "../vector";
|
||||||
import { ResizableBufferCursor } from "./ResizableBufferCursor";
|
import { AbstractCursor } from "./AbstractCursor";
|
||||||
import { Cursor } from "./Cursor";
|
import { Cursor } from "./Cursor";
|
||||||
import { ASCII_ENCODER } from ".";
|
import { WritableCursor } from "./WritableCursor";
|
||||||
import { Vec3, Vec2 } from "../vector";
|
|
||||||
|
|
||||||
export class WritableResizableBufferCursor extends ResizableBufferCursor implements WritableCursor {
|
export abstract class AbstractWritableCursor extends AbstractCursor implements WritableCursor {
|
||||||
get size(): number {
|
abstract size: number;
|
||||||
return this._size;
|
|
||||||
}
|
|
||||||
|
|
||||||
set size(size: number) {
|
|
||||||
if (size > this._size) {
|
|
||||||
this.ensure_size(size - this._size);
|
|
||||||
} else {
|
|
||||||
this._size = size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u8(value: number): this {
|
write_u8(value: number): this {
|
||||||
this.ensure_size(1);
|
this.ensure_size(1);
|
||||||
@ -53,9 +42,7 @@ export class WritableResizableBufferCursor extends ResizableBufferCursor impleme
|
|||||||
|
|
||||||
write_u8_array(array: number[]): this {
|
write_u8_array(array: number[]): this {
|
||||||
this.ensure_size(array.length);
|
this.ensure_size(array.length);
|
||||||
new Uint8Array(this.buffer.backing_buffer, this.offset + this.position).set(
|
new Uint8Array(this.backing_buffer, this.offset + this.position).set(new Uint8Array(array));
|
||||||
new Uint8Array(array)
|
|
||||||
);
|
|
||||||
this._position += array.length;
|
this._position += array.length;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
@ -104,7 +91,7 @@ export class WritableResizableBufferCursor extends ResizableBufferCursor impleme
|
|||||||
this.ensure_size(size);
|
this.ensure_size(size);
|
||||||
|
|
||||||
other.copy_to_uint8_array(
|
other.copy_to_uint8_array(
|
||||||
new Uint8Array(this.buffer.backing_buffer, this.offset + this.position, size),
|
new Uint8Array(this.backing_buffer, this.offset + this.position, size),
|
||||||
size
|
size
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -115,15 +102,15 @@ export class WritableResizableBufferCursor extends ResizableBufferCursor impleme
|
|||||||
write_string_ascii(str: string, byte_length: number): this {
|
write_string_ascii(str: string, byte_length: number): this {
|
||||||
this.ensure_size(byte_length);
|
this.ensure_size(byte_length);
|
||||||
|
|
||||||
const encoded = ASCII_ENCODER.encode(str);
|
const len = Math.min(byte_length, str.length);
|
||||||
const encoded_length = Math.min(encoded.byteLength, byte_length);
|
|
||||||
let i = 0;
|
|
||||||
|
|
||||||
while (i < encoded_length) {
|
for (let i = 0; i < len; i++) {
|
||||||
this.write_u8(encoded[i++]);
|
this.write_u8(str.codePointAt(i)!);
|
||||||
}
|
}
|
||||||
|
|
||||||
while (i++ < byte_length) {
|
let pad_len = byte_length - len;
|
||||||
|
|
||||||
|
for (let i = 0; i < pad_len; i++) {
|
||||||
this.write_u8(0);
|
this.write_u8(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,30 +120,25 @@ export class WritableResizableBufferCursor extends ResizableBufferCursor impleme
|
|||||||
write_string_utf16(str: string, byte_length: number): this {
|
write_string_utf16(str: string, byte_length: number): this {
|
||||||
this.ensure_size(byte_length);
|
this.ensure_size(byte_length);
|
||||||
|
|
||||||
const encoded = this.utf16_encoder.encode(str);
|
const max_len = Math.floor(byte_length / 2);
|
||||||
const encoded_length = Math.min(encoded.byteLength, byte_length);
|
const len = Math.min(max_len, str.length);
|
||||||
let i = 0;
|
|
||||||
|
|
||||||
while (i < encoded_length) {
|
for (let i = 0; i < len; i++) {
|
||||||
this.write_u8(encoded[i++]);
|
this.write_u16(str.codePointAt(i)!);
|
||||||
}
|
}
|
||||||
|
|
||||||
while (i++ < byte_length) {
|
let pad_len = max_len - len;
|
||||||
this.write_u8(0);
|
|
||||||
|
for (let i = 0; i < pad_len; i++) {
|
||||||
|
this.write_u16(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
private ensure_size(size: number): void {
|
protected ensure_size(size: number): void {
|
||||||
const needed = this.position + size - this._size;
|
if (size > this.bytes_left) {
|
||||||
|
throw new Error(`${size} Bytes required but only ${this.bytes_left} available.`);
|
||||||
if (needed > 0) {
|
|
||||||
this._size += needed;
|
|
||||||
|
|
||||||
if (this.buffer.size < this.offset + this._size) {
|
|
||||||
this.buffer.size = this.offset + this._size;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,56 +1,27 @@
|
|||||||
import {
|
|
||||||
ASCII_DECODER,
|
|
||||||
UTF_16BE_DECODER,
|
|
||||||
UTF_16BE_ENCODER,
|
|
||||||
UTF_16LE_DECODER,
|
|
||||||
UTF_16LE_ENCODER,
|
|
||||||
} from ".";
|
|
||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
import { Vec2, Vec3 } from "../vector";
|
import { AbstractWritableCursor } from "./AbstractWritableCursor";
|
||||||
import { Cursor } from "./Cursor";
|
import { WritableCursor } from "./WritableCursor";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A cursor for reading from an array buffer or part of an array buffer.
|
* A cursor for reading from an array buffer or part of an array buffer.
|
||||||
*/
|
*/
|
||||||
export class ArrayBufferCursor implements Cursor {
|
export class ArrayBufferCursor extends AbstractWritableCursor implements WritableCursor {
|
||||||
get offset(): number {
|
private _size: number;
|
||||||
return this.dv.byteOffset;
|
|
||||||
}
|
|
||||||
|
|
||||||
get size(): number {
|
get size(): number {
|
||||||
return this.dv.byteLength;
|
return this._size;
|
||||||
}
|
}
|
||||||
|
|
||||||
set size(size: number) {
|
set size(size: number) {
|
||||||
this.dv = new DataView(this.buffer, this.offset, size);
|
if (size > this.backing_buffer.byteLength - this.offset) {
|
||||||
|
throw new Error(`Size ${size} is out of bounds.`);
|
||||||
|
}
|
||||||
|
|
||||||
|
this._size = size;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected _position: number;
|
protected backing_buffer: ArrayBuffer;
|
||||||
|
|
||||||
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;
|
protected dv: DataView;
|
||||||
protected utf16_decoder!: TextDecoder;
|
|
||||||
protected utf16_encoder!: TextEncoder;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param buffer The buffer to read from.
|
* @param buffer The buffer to read from.
|
||||||
@ -62,219 +33,18 @@ export class ArrayBufferCursor implements Cursor {
|
|||||||
buffer: ArrayBuffer,
|
buffer: ArrayBuffer,
|
||||||
endianness: Endianness,
|
endianness: Endianness,
|
||||||
offset: number = 0,
|
offset: number = 0,
|
||||||
size: number = buffer.byteLength
|
size: number = buffer.byteLength - offset
|
||||||
) {
|
) {
|
||||||
this.buffer = buffer;
|
super(endianness, offset);
|
||||||
this.dv = new DataView(buffer, offset, size);
|
this._size = size;
|
||||||
this.endianness = endianness;
|
this.backing_buffer = buffer;
|
||||||
this._position = 0;
|
this.dv = new DataView(buffer, 0, buffer.byteLength);
|
||||||
}
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
vec2_f32(): Vec2 {
|
|
||||||
return new Vec2(this.f32(), this.f32());
|
|
||||||
}
|
|
||||||
|
|
||||||
vec3_f32(): Vec3 {
|
|
||||||
return new Vec3(this.f32(), this.f32(), this.f32());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
take(size: number): ArrayBufferCursor {
|
take(size: number): ArrayBufferCursor {
|
||||||
const offset = this.offset + this.position;
|
const offset = this.offset + this.position;
|
||||||
const wrapper = new ArrayBufferCursor(this.buffer, this.endianness, offset, size);
|
const wrapper = new ArrayBufferCursor(this.backing_buffer, this.endianness, offset, size);
|
||||||
this._position += size;
|
this._position += size;
|
||||||
return wrapper;
|
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 {
|
|
||||||
const r = this.buffer.slice(
|
|
||||||
this.offset + this.position,
|
|
||||||
this.offset + this.position + size
|
|
||||||
);
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,18 @@
|
|||||||
import { ArrayBufferCursor } from "./ArrayBufferCursor";
|
|
||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
|
import { AbstractCursor } from "./AbstractCursor";
|
||||||
|
import { Cursor } from "./Cursor";
|
||||||
|
|
||||||
|
export class BufferCursor extends AbstractCursor implements Cursor {
|
||||||
|
readonly size: number;
|
||||||
|
|
||||||
|
protected buffer: Buffer;
|
||||||
|
|
||||||
|
protected get backing_buffer(): ArrayBuffer {
|
||||||
|
return this.buffer.buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected dv: DataView;
|
||||||
|
|
||||||
export class BufferCursor extends ArrayBufferCursor {
|
|
||||||
/**
|
/**
|
||||||
* @param buffer The buffer to read from.
|
* @param buffer The buffer to read from.
|
||||||
* @param endianness Decides in which byte order multi-byte integers and floats will be interpreted.
|
* @param endianness Decides in which byte order multi-byte integers and floats will be interpreted.
|
||||||
@ -12,7 +23,7 @@ export class BufferCursor extends ArrayBufferCursor {
|
|||||||
buffer: Buffer,
|
buffer: Buffer,
|
||||||
endianness: Endianness,
|
endianness: Endianness,
|
||||||
offset: number = 0,
|
offset: number = 0,
|
||||||
size: number = buffer.byteLength
|
size: number = buffer.byteLength - offset
|
||||||
) {
|
) {
|
||||||
if (offset < 0 || offset > buffer.byteLength) {
|
if (offset < 0 || offset > buffer.byteLength) {
|
||||||
throw new Error(`Offset ${offset} is out of bounds.`);
|
throw new Error(`Offset ${offset} is out of bounds.`);
|
||||||
@ -22,6 +33,17 @@ export class BufferCursor extends ArrayBufferCursor {
|
|||||||
throw new Error(`Size ${size} is out of bounds.`);
|
throw new Error(`Size ${size} is out of bounds.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
super(buffer.buffer, endianness, buffer.byteOffset + offset, size);
|
super(endianness, buffer.byteOffset + offset);
|
||||||
|
|
||||||
|
this.buffer = buffer;
|
||||||
|
this.size = size;
|
||||||
|
this.dv = new DataView(buffer.buffer, 0, buffer.buffer.byteLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
take(size: number): BufferCursor {
|
||||||
|
const offset = this.offset + this.position;
|
||||||
|
const wrapper = new BufferCursor(this.buffer, this.endianness, offset, size);
|
||||||
|
this._position += size;
|
||||||
|
return wrapper;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,8 +5,6 @@ import { ArrayBufferCursor } from "./ArrayBufferCursor";
|
|||||||
import { BufferCursor } from "./BufferCursor";
|
import { BufferCursor } from "./BufferCursor";
|
||||||
import { Cursor } from "./Cursor";
|
import { Cursor } from "./Cursor";
|
||||||
import { ResizableBufferCursor } from "./ResizableBufferCursor";
|
import { ResizableBufferCursor } from "./ResizableBufferCursor";
|
||||||
import { WritableArrayBufferCursor } from "./WritableArrayBufferCursor";
|
|
||||||
import { WritableResizableBufferCursor } from "./WritableResizableBufferCursor";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Run a test on every cursor implementation with every endianness.
|
* Run a test on every cursor implementation with every endianness.
|
||||||
@ -50,16 +48,6 @@ function test_all(
|
|||||||
endianness,
|
endianness,
|
||||||
new ResizableBufferCursor(rbuf(endianness), endianness),
|
new ResizableBufferCursor(rbuf(endianness), endianness),
|
||||||
]),
|
]),
|
||||||
...endiannesses.map(endianness => [
|
|
||||||
WritableArrayBufferCursor.name,
|
|
||||||
endianness,
|
|
||||||
new WritableArrayBufferCursor(new Uint8Array(bytes(endianness)).buffer, endianness),
|
|
||||||
]),
|
|
||||||
...endiannesses.map(endianness => [
|
|
||||||
WritableResizableBufferCursor.name,
|
|
||||||
endianness,
|
|
||||||
new WritableResizableBufferCursor(rbuf(endianness), endianness),
|
|
||||||
]),
|
|
||||||
] as any;
|
] as any;
|
||||||
|
|
||||||
for (const [cursor_name, endianness, cursor] of cursors) {
|
for (const [cursor_name, endianness, cursor] of cursors) {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { WritableResizableBufferCursor } from "./WritableResizableBufferCursor";
|
|
||||||
import { ResizableBuffer } from "../ResizableBuffer";
|
|
||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
|
import { ResizableBuffer } from "../ResizableBuffer";
|
||||||
|
import { ResizableBufferCursor } from "./ResizableBufferCursor";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes two integers to a cursor backed with a buffer of size 0.
|
* Writes two integers to a cursor backed with a buffer of size 0.
|
||||||
@ -13,7 +13,7 @@ function test_integer_write(method_name: string): void {
|
|||||||
let expected_number_2 = 7348942;
|
let expected_number_2 = 7348942;
|
||||||
|
|
||||||
const buf = new ResizableBuffer(8);
|
const buf = new ResizableBuffer(8);
|
||||||
const cursor = new WritableResizableBufferCursor(buf, Endianness.Little);
|
const cursor = new ResizableBufferCursor(buf, Endianness.Little);
|
||||||
|
|
||||||
expect(buf.size).toBe(0);
|
expect(buf.size).toBe(0);
|
||||||
expect(cursor.size).toBe(0);
|
expect(cursor.size).toBe(0);
|
||||||
@ -33,7 +33,7 @@ test_integer_write("write_u32");
|
|||||||
test_integer_write("write_i32");
|
test_integer_write("write_i32");
|
||||||
|
|
||||||
test("write, seek backwards then take", () => {
|
test("write, seek backwards then take", () => {
|
||||||
const cursor = new WritableResizableBufferCursor(new ResizableBuffer(0), Endianness.Little);
|
const cursor = new ResizableBufferCursor(new ResizableBuffer(0), Endianness.Little);
|
||||||
cursor
|
cursor
|
||||||
.write_u32(1)
|
.write_u32(1)
|
||||||
.write_u32(2)
|
.write_u32(2)
|
@ -1,59 +1,33 @@
|
|||||||
import {
|
|
||||||
ASCII_DECODER,
|
|
||||||
UTF_16BE_DECODER,
|
|
||||||
UTF_16BE_ENCODER,
|
|
||||||
UTF_16LE_DECODER,
|
|
||||||
UTF_16LE_ENCODER,
|
|
||||||
} from ".";
|
|
||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
import { ResizableBuffer } from "../ResizableBuffer";
|
import { ResizableBuffer } from "../ResizableBuffer";
|
||||||
import { Vec2, Vec3 } from "../vector";
|
import { AbstractWritableCursor } from "./AbstractWritableCursor";
|
||||||
import { Cursor } from "./Cursor";
|
import { WritableCursor } from "./WritableCursor";
|
||||||
|
|
||||||
export class ResizableBufferCursor implements Cursor {
|
|
||||||
private _offset: number;
|
|
||||||
|
|
||||||
get offset(): number {
|
|
||||||
return this._offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
export class ResizableBufferCursor extends AbstractWritableCursor implements WritableCursor {
|
||||||
protected _size: number;
|
protected _size: number;
|
||||||
|
|
||||||
get size(): number {
|
get size(): number {
|
||||||
return this._size;
|
return this._size;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected _position: number;
|
set size(size: number) {
|
||||||
|
if (size > this._size) {
|
||||||
get position(): number {
|
this.ensure_size(size - this._size);
|
||||||
return this._position;
|
} else {
|
||||||
}
|
this._size = size;
|
||||||
|
}
|
||||||
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: ResizableBuffer;
|
protected buffer: ResizableBuffer;
|
||||||
|
|
||||||
|
protected get backing_buffer(): ArrayBuffer {
|
||||||
|
return this.buffer.backing_buffer;
|
||||||
|
}
|
||||||
|
|
||||||
protected get dv(): DataView {
|
protected get dv(): DataView {
|
||||||
return this.buffer.view;
|
return this.buffer.view;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected utf16_decoder: TextDecoder = UTF_16BE_DECODER;
|
|
||||||
protected utf16_encoder: TextEncoder = UTF_16BE_ENCODER;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param buffer The buffer to read from.
|
* @param buffer The buffer to read from.
|
||||||
* @param endianness Decides in which byte order multi-byte integers and floats will be interpreted.
|
* @param endianness Decides in which byte order multi-byte integers and floats will be interpreted.
|
||||||
@ -64,7 +38,7 @@ export class ResizableBufferCursor implements Cursor {
|
|||||||
buffer: ResizableBuffer,
|
buffer: ResizableBuffer,
|
||||||
endianness: Endianness,
|
endianness: Endianness,
|
||||||
offset: number = 0,
|
offset: number = 0,
|
||||||
size: number = buffer.size
|
size: number = buffer.size - offset
|
||||||
) {
|
) {
|
||||||
if (offset < 0 || offset > buffer.size) {
|
if (offset < 0 || offset > buffer.size) {
|
||||||
throw new Error(`Offset ${offset} is out of bounds.`);
|
throw new Error(`Offset ${offset} is out of bounds.`);
|
||||||
@ -74,152 +48,10 @@ export class ResizableBufferCursor implements Cursor {
|
|||||||
throw new Error(`Size ${size} is out of bounds.`);
|
throw new Error(`Size ${size} is out of bounds.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
super(endianness, offset);
|
||||||
|
|
||||||
this.buffer = buffer;
|
this.buffer = buffer;
|
||||||
this.endianness = endianness;
|
|
||||||
this._offset = offset;
|
|
||||||
this._size = size;
|
this._size = size;
|
||||||
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 {
|
|
||||||
this.check_offset(offset, 1);
|
|
||||||
return this.dv.getUint8(this.offset + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
u16(): number {
|
|
||||||
const r = this.u16_at(this.position);
|
|
||||||
this._position += 2;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
u16_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 2);
|
|
||||||
return this.dv.getUint16(this.offset + offset, this.little_endian);
|
|
||||||
}
|
|
||||||
|
|
||||||
u32(): number {
|
|
||||||
const r = this.u32_at(this.position);
|
|
||||||
this._position += 4;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 4);
|
|
||||||
return this.dv.getUint32(this.offset + offset, this.little_endian);
|
|
||||||
}
|
|
||||||
|
|
||||||
i8(): number {
|
|
||||||
return this.i8_at(this._position++);
|
|
||||||
}
|
|
||||||
|
|
||||||
i8_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 1);
|
|
||||||
return this.dv.getInt8(this.offset + offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
i16(): number {
|
|
||||||
const r = this.i16_at(this.position);
|
|
||||||
this._position += 2;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
i16_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 2);
|
|
||||||
return this.dv.getInt16(this.offset + offset, this.little_endian);
|
|
||||||
}
|
|
||||||
|
|
||||||
i32(): number {
|
|
||||||
const r = this.i32_at(this.position);
|
|
||||||
this._position += 4;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
i32_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 4);
|
|
||||||
return this.dv.getInt32(this.offset + offset, this.little_endian);
|
|
||||||
}
|
|
||||||
|
|
||||||
f32(): number {
|
|
||||||
const r = this.f32_at(this.position);
|
|
||||||
this._position += 4;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
f32_at(offset: number): number {
|
|
||||||
this.check_offset(offset, 4);
|
|
||||||
return this.dv.getFloat32(this.offset + offset, this.little_endian);
|
|
||||||
}
|
|
||||||
|
|
||||||
u8_array(n: number): number[] {
|
|
||||||
this.check_size("n", n, n);
|
|
||||||
|
|
||||||
const array = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < n; ++i) {
|
|
||||||
array.push(this.dv.getUint8(this.offset + this._position++));
|
|
||||||
}
|
|
||||||
|
|
||||||
return array;
|
|
||||||
}
|
|
||||||
|
|
||||||
u16_array(n: number): number[] {
|
|
||||||
this.check_size("n", n, 2 * n);
|
|
||||||
|
|
||||||
const array = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < n; ++i) {
|
|
||||||
array.push(this.dv.getUint16(this.offset + this.position, this.little_endian));
|
|
||||||
this._position += 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
return array;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32_array(n: number): number[] {
|
|
||||||
this.check_size("n", n, 4 * n);
|
|
||||||
|
|
||||||
const array = [];
|
|
||||||
|
|
||||||
for (let i = 0; i < n; ++i) {
|
|
||||||
array.push(this.dv.getUint32(this.offset + this.position, this.little_endian));
|
|
||||||
this._position += 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
return array;
|
|
||||||
}
|
|
||||||
|
|
||||||
vec2_f32(): Vec2 {
|
|
||||||
return new Vec2(this.f32(), this.f32());
|
|
||||||
}
|
|
||||||
|
|
||||||
vec3_f32(): Vec3 {
|
|
||||||
return new Vec3(this.f32(), this.f32(), this.f32());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
take(size: number): ResizableBufferCursor {
|
take(size: number): ResizableBufferCursor {
|
||||||
@ -231,101 +63,15 @@ export class ResizableBufferCursor implements Cursor {
|
|||||||
return wrapper;
|
return wrapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
string_ascii(
|
protected ensure_size(size: number): void {
|
||||||
max_byte_length: number,
|
const needed = this.position + size - this._size;
|
||||||
null_terminated: boolean,
|
|
||||||
drop_remaining: boolean
|
|
||||||
): string {
|
|
||||||
this.check_size("max_byte_length", max_byte_length, max_byte_length);
|
|
||||||
|
|
||||||
const string_length = null_terminated
|
if (needed > 0) {
|
||||||
? this.index_of_u8(0, max_byte_length) - this.position
|
this._size += needed;
|
||||||
: max_byte_length;
|
|
||||||
|
|
||||||
const view = this.buffer.sub_view(this.offset + this.position, string_length);
|
if (this.buffer.size < this.offset + this._size) {
|
||||||
const r = ASCII_DECODER.decode(view);
|
this.buffer.size = this.offset + this._size;
|
||||||
|
|
||||||
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 {
|
|
||||||
this.check_size("max_byte_length", max_byte_length, max_byte_length);
|
|
||||||
|
|
||||||
const string_length = null_terminated
|
|
||||||
? this.index_of_u16(0, max_byte_length) - this.position
|
|
||||||
: Math.floor(max_byte_length / 2) * 2;
|
|
||||||
|
|
||||||
const view = this.buffer.sub_view(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 {
|
|
||||||
this.check_size("size", size, size);
|
|
||||||
const r = this.buffer.backing_buffer.slice(
|
|
||||||
this.offset + this.position,
|
|
||||||
this.offset + this.position + size
|
|
||||||
);
|
|
||||||
this._position += size;
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
copy_to_uint8_array(array: Uint8Array, size: number = this.size - this.position): this {
|
|
||||||
this.check_size("size", size, size);
|
|
||||||
array.set(new Uint8Array(this.buffer.backing_buffer, this.offset + this.position, size));
|
|
||||||
this._position += size;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
private check_size(name: string, value: number, byte_size: number): void {
|
|
||||||
if (byte_size < 0 || byte_size > this.size - this.position) {
|
|
||||||
throw new Error(`${name} ${value} is out of bounds.`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks whether we can read size bytes at offset.
|
|
||||||
*/
|
|
||||||
protected check_offset(offset: number, size: number): void {
|
|
||||||
if (offset < 0 || offset + size > this.size) {
|
|
||||||
throw new Error(`Offset ${offset} is out of bounds.`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
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(this.offset + 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(this.offset + i, this.little_endian) === value) {
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.position + max_byte_length;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,122 +0,0 @@
|
|||||||
import { ASCII_ENCODER } from ".";
|
|
||||||
import { Vec2, Vec3 } from "../vector";
|
|
||||||
import { ArrayBufferCursor } from "./ArrayBufferCursor";
|
|
||||||
import { Cursor } from "./Cursor";
|
|
||||||
import { WritableCursor } from "./WritableCursor";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A cursor for reading and writing from an array buffer or part of an array buffer.
|
|
||||||
*/
|
|
||||||
export class WritableArrayBufferCursor extends ArrayBufferCursor implements WritableCursor {
|
|
||||||
write_u8(value: number): this {
|
|
||||||
this.dv.setUint8(this._position++, value);
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u16(value: number): this {
|
|
||||||
this.dv.setUint16(this.position, value, this.little_endian);
|
|
||||||
this._position += 2;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u32(value: number): this {
|
|
||||||
this.dv.setUint32(this.position, value, this.little_endian);
|
|
||||||
this._position += 4;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_i32(value: number): this {
|
|
||||||
this.dv.setInt32(this.position, value, this.little_endian);
|
|
||||||
this._position += 4;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_f32(value: number): this {
|
|
||||||
this.dv.setFloat32(this.position, value, this.little_endian);
|
|
||||||
this._position += 4;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u8_array(array: number[]): this {
|
|
||||||
new Uint8Array(this.buffer, this.offset + this.position).set(new Uint8Array(array));
|
|
||||||
this._position += array.length;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u16_array(array: number[]): this {
|
|
||||||
const len = array.length;
|
|
||||||
|
|
||||||
for (let i = 0; i < len; i++) {
|
|
||||||
this.write_u16(array[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_u32_array(array: number[]): this {
|
|
||||||
const len = array.length;
|
|
||||||
|
|
||||||
for (let i = 0; i < len; i++) {
|
|
||||||
this.write_u32(array[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_vec2_f32(value: Vec2): this {
|
|
||||||
this.dv.setFloat32(this.position, value.x, this.little_endian);
|
|
||||||
this.dv.setFloat32(this.position + 4, value.y, this.little_endian);
|
|
||||||
this._position += 8;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_vec3_f32(value: Vec3): this {
|
|
||||||
this.dv.setFloat32(this.position, value.x, this.little_endian);
|
|
||||||
this.dv.setFloat32(this.position + 4, value.y, this.little_endian);
|
|
||||||
this.dv.setFloat32(this.position + 8, value.z, this.little_endian);
|
|
||||||
this._position += 12;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_cursor(other: Cursor): this {
|
|
||||||
const size = other.size - other.position;
|
|
||||||
other.copy_to_uint8_array(
|
|
||||||
new Uint8Array(this.buffer, this.offset + this.position, size),
|
|
||||||
size
|
|
||||||
);
|
|
||||||
this._position += size;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_string_ascii(str: string, byte_length: number): this {
|
|
||||||
const encoded = ASCII_ENCODER.encode(str);
|
|
||||||
const encoded_length = Math.min(encoded.byteLength, byte_length);
|
|
||||||
let i = 0;
|
|
||||||
|
|
||||||
while (i < encoded_length) {
|
|
||||||
this.write_u8(encoded[i++]);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (i++ < byte_length) {
|
|
||||||
this.write_u8(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_string_utf16(str: string, byte_length: number): this {
|
|
||||||
const encoded = this.utf16_encoder.encode(str);
|
|
||||||
const encoded_length = Math.min(encoded.byteLength, byte_length);
|
|
||||||
let i = 0;
|
|
||||||
|
|
||||||
while (i < encoded_length) {
|
|
||||||
this.write_u8(encoded[i++]);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (i++ < byte_length) {
|
|
||||||
this.write_u8(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,9 +1,9 @@
|
|||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
import { enum_values } from "../../enums";
|
import { enum_values } from "../../enums";
|
||||||
import { ResizableBuffer } from "../ResizableBuffer";
|
import { ResizableBuffer } from "../ResizableBuffer";
|
||||||
import { WritableArrayBufferCursor } from "./WritableArrayBufferCursor";
|
import { ArrayBufferCursor } from "./ArrayBufferCursor";
|
||||||
|
import { ResizableBufferCursor } from "./ResizableBufferCursor";
|
||||||
import { WritableCursor } from "./WritableCursor";
|
import { WritableCursor } from "./WritableCursor";
|
||||||
import { WritableResizableBufferCursor } from "./WritableResizableBufferCursor";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Run a test on every writable cursor implementation with every endianness.
|
* Run a test on every writable cursor implementation with every endianness.
|
||||||
@ -33,14 +33,14 @@ function test_all(
|
|||||||
|
|
||||||
const cursors: [string, Endianness, WritableCursor][] = [
|
const cursors: [string, Endianness, WritableCursor][] = [
|
||||||
...endiannesses.map(endianness => [
|
...endiannesses.map(endianness => [
|
||||||
WritableArrayBufferCursor.name,
|
ArrayBufferCursor.name,
|
||||||
endianness,
|
endianness,
|
||||||
new WritableArrayBufferCursor(new Uint8Array(bytes(endianness)).buffer, endianness),
|
new ArrayBufferCursor(new Uint8Array(bytes(endianness)).buffer, endianness),
|
||||||
]),
|
]),
|
||||||
...endiannesses.map(endianness => [
|
...endiannesses.map(endianness => [
|
||||||
WritableResizableBufferCursor.name,
|
ResizableBufferCursor.name,
|
||||||
endianness,
|
endianness,
|
||||||
new WritableResizableBufferCursor(rbuf(endianness), endianness),
|
new ResizableBufferCursor(rbuf(endianness), endianness),
|
||||||
]),
|
]),
|
||||||
] as any;
|
] as any;
|
||||||
|
|
||||||
|
@ -1,14 +0,0 @@
|
|||||||
// TODO: remove dependency on text-encoding because it is no longer maintained.
|
|
||||||
import { TextDecoder, TextEncoder } from "text-encoding";
|
|
||||||
|
|
||||||
export const ASCII_DECODER = new TextDecoder("ascii");
|
|
||||||
export const UTF_16BE_DECODER = new TextDecoder("utf-16be");
|
|
||||||
export const UTF_16LE_DECODER = new TextDecoder("utf-16le");
|
|
||||||
|
|
||||||
export const ASCII_ENCODER = new TextEncoder("ascii");
|
|
||||||
export const UTF_16BE_ENCODER = new TextEncoder("utf-16be", {
|
|
||||||
NONSTANDARD_allowLegacyEncoding: true,
|
|
||||||
});
|
|
||||||
export const UTF_16LE_ENCODER = new TextEncoder("utf-16le", {
|
|
||||||
NONSTANDARD_allowLegacyEncoding: true,
|
|
||||||
});
|
|
@ -1,6 +1,6 @@
|
|||||||
import { Cursor } from "../cursor/Cursor";
|
|
||||||
import { WritableArrayBufferCursor } from "../cursor/WritableArrayBufferCursor";
|
|
||||||
import { Endianness } from "..";
|
import { Endianness } from "..";
|
||||||
|
import { ArrayBufferCursor } from "../cursor/ArrayBufferCursor";
|
||||||
|
import { Cursor } from "../cursor/Cursor";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decrypts the bytes left in cursor.
|
* Decrypts the bytes left in cursor.
|
||||||
@ -21,10 +21,7 @@ class PrcDecryptor {
|
|||||||
// Size should be divisible by 4.
|
// Size should be divisible by 4.
|
||||||
const actual_size = cursor.bytes_left;
|
const actual_size = cursor.bytes_left;
|
||||||
const size = Math.ceil(actual_size / 4) * 4;
|
const size = Math.ceil(actual_size / 4) * 4;
|
||||||
const out_cursor = new WritableArrayBufferCursor(
|
const out_cursor = new ArrayBufferCursor(new ArrayBuffer(actual_size), cursor.endianness);
|
||||||
new ArrayBuffer(actual_size),
|
|
||||||
cursor.endianness
|
|
||||||
);
|
|
||||||
|
|
||||||
for (let pos = 0; pos < size; pos += 4) {
|
for (let pos = 0; pos < size; pos += 4) {
|
||||||
let u32;
|
let u32;
|
||||||
|
@ -128,9 +128,14 @@ function parse_ninja<M extends NjModel>(
|
|||||||
context: any
|
context: any
|
||||||
): NjObject<M>[] {
|
): NjObject<M>[] {
|
||||||
// POF0 and other chunks types are ignored.
|
// POF0 and other chunks types are ignored.
|
||||||
return parse_iff(cursor)
|
const njcm_chunks = parse_iff(cursor).filter(chunk => chunk.type === NJCM);
|
||||||
.filter(chunk => chunk.type === NJCM)
|
const objects: NjObject<M>[] = [];
|
||||||
.flatMap(chunk => parse_sibling_objects(chunk.data, parse_model, context));
|
|
||||||
|
for (const chunk of njcm_chunks) {
|
||||||
|
objects.push(...parse_sibling_objects(chunk.data, parse_model, context));
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: cache model and object offsets so we don't reparse the same data.
|
// TODO: cache model and object offsets so we don't reparse the same data.
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
import { Endianness } from "../..";
|
import { Endianness } from "../..";
|
||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
import { WritableResizableBufferCursor } from "../../cursor/WritableResizableBufferCursor";
|
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { Opcode, OPCODES, Type } from "./opcodes";
|
import { Opcode, OPCODES, Type } from "./opcodes";
|
||||||
|
|
||||||
@ -195,7 +195,7 @@ export function write_bin(bin: BinFile): ArrayBuffer {
|
|||||||
const buffer = new ResizableBuffer(
|
const buffer = new ResizableBuffer(
|
||||||
object_code_offset + 10 * bin.instructions.length + 4 * labels.length
|
object_code_offset + 10 * bin.instructions.length + 4 * labels.length
|
||||||
);
|
);
|
||||||
const cursor = new WritableResizableBufferCursor(buffer, Endianness.Little);
|
const cursor = new ResizableBufferCursor(buffer, Endianness.Little);
|
||||||
|
|
||||||
cursor.write_u32(object_code_offset);
|
cursor.write_u32(object_code_offset);
|
||||||
cursor.write_u32(0); // Placeholder for the labels offset.
|
cursor.write_u32(0); // Placeholder for the labels offset.
|
||||||
|
@ -2,7 +2,7 @@ import Logger from "js-logger";
|
|||||||
import { groupBy } from "lodash";
|
import { groupBy } from "lodash";
|
||||||
import { Endianness } from "../..";
|
import { Endianness } from "../..";
|
||||||
import { Cursor } from "../../cursor/Cursor";
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
import { WritableResizableBufferCursor } from "../../cursor/WritableResizableBufferCursor";
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
import { Vec3 } from "../../vector";
|
import { Vec3 } from "../../vector";
|
||||||
|
|
||||||
@ -159,7 +159,7 @@ export function write_dat({ objs, npcs, unknowns }: DatFile): ResizableBuffer {
|
|||||||
npcs.length * (16 + NPC_SIZE) +
|
npcs.length * (16 + NPC_SIZE) +
|
||||||
unknowns.reduce((a, b) => a + b.total_size, 0)
|
unknowns.reduce((a, b) => a + b.total_size, 0)
|
||||||
);
|
);
|
||||||
const cursor = new WritableResizableBufferCursor(buffer, Endianness.Little);
|
const cursor = new ResizableBufferCursor(buffer, Endianness.Little);
|
||||||
|
|
||||||
const grouped_objs = groupBy(objs, obj => obj.area_id);
|
const grouped_objs = groupBy(objs, obj => obj.area_id);
|
||||||
const obj_area_ids = Object.keys(grouped_objs)
|
const obj_area_ids = Object.keys(grouped_objs)
|
||||||
|
@ -1,11 +1,10 @@
|
|||||||
import Logger from "js-logger";
|
import Logger from "js-logger";
|
||||||
import { Cursor } from "../../cursor/Cursor";
|
|
||||||
import { WritableArrayBufferCursor } from "../../cursor/WritableArrayBufferCursor";
|
|
||||||
import { Endianness } from "../..";
|
import { Endianness } from "../..";
|
||||||
import { WritableCursor } from "../../cursor/WritableCursor";
|
|
||||||
import { WritableResizableBufferCursor } from "../../cursor/WritableResizableBufferCursor";
|
|
||||||
import { ResizableBuffer } from "../../ResizableBuffer";
|
|
||||||
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
import { ArrayBufferCursor } from "../../cursor/ArrayBufferCursor";
|
||||||
|
import { Cursor } from "../../cursor/Cursor";
|
||||||
|
import { ResizableBufferCursor } from "../../cursor/ResizableBufferCursor";
|
||||||
|
import { WritableCursor } from "../../cursor/WritableCursor";
|
||||||
|
import { ResizableBuffer } from "../../ResizableBuffer";
|
||||||
|
|
||||||
const logger = Logger.get("data_formats/parsing/quest/qst");
|
const logger = Logger.get("data_formats/parsing/quest/qst");
|
||||||
|
|
||||||
@ -92,7 +91,7 @@ export function write_qst(params: WriteQstParams): ArrayBuffer {
|
|||||||
.map(f => 88 + Math.ceil(f.data.byteLength / 1024) * 1056)
|
.map(f => 88 + Math.ceil(f.data.byteLength / 1024) * 1056)
|
||||||
.reduce((a, b) => a + b);
|
.reduce((a, b) => a + b);
|
||||||
const buffer = new ArrayBuffer(total_size);
|
const buffer = new ArrayBuffer(total_size);
|
||||||
const cursor = new WritableArrayBufferCursor(buffer, Endianness.Little);
|
const cursor = new ArrayBufferCursor(buffer, Endianness.Little);
|
||||||
|
|
||||||
write_file_headers(cursor, files);
|
write_file_headers(cursor, files);
|
||||||
write_file_chunks(cursor, files);
|
write_file_chunks(cursor, files);
|
||||||
@ -166,7 +165,7 @@ function parse_files(cursor: Cursor, expected_sizes: Map<string, number>): QstCo
|
|||||||
(file = {
|
(file = {
|
||||||
name: file_name,
|
name: file_name,
|
||||||
expected_size,
|
expected_size,
|
||||||
cursor: new WritableResizableBufferCursor(
|
cursor: new ResizableBufferCursor(
|
||||||
new ResizableBuffer(expected_size || 10 * 1024),
|
new ResizableBuffer(expected_size || 10 * 1024),
|
||||||
Endianness.Little
|
Endianness.Little
|
||||||
),
|
),
|
||||||
|
@ -4,7 +4,7 @@
|
|||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
"module": "es6",
|
"module": "es6",
|
||||||
"target": "es6",
|
"target": "es6",
|
||||||
"lib": ["dom", "dom.iterable", "esnext"],
|
"lib": ["es6", "dom", "dom.iterable"],
|
||||||
"allowJs": true,
|
"allowJs": true,
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"allowSyntheticDefaultImports": true,
|
"allowSyntheticDefaultImports": true,
|
||||||
|
@ -8133,11 +8133,6 @@ test-exclude@^5.2.3:
|
|||||||
read-pkg-up "^4.0.0"
|
read-pkg-up "^4.0.0"
|
||||||
require-main-filename "^2.0.0"
|
require-main-filename "^2.0.0"
|
||||||
|
|
||||||
text-encoding@^0.7.0:
|
|
||||||
version "0.7.0"
|
|
||||||
resolved "https://registry.yarnpkg.com/text-encoding/-/text-encoding-0.7.0.tgz#f895e836e45990624086601798ea98e8f36ee643"
|
|
||||||
integrity sha512-oJQ3f1hrOnbRLOcwKz0Liq2IcrvDeZRHXhd9RgLrsT+DjWY/nty1Hi7v3dtkaEYbPYe0mUoOfzRrMwfXXwgPUA==
|
|
||||||
|
|
||||||
text-table@^0.2.0:
|
text-table@^0.2.0:
|
||||||
version "0.2.0"
|
version "0.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
|
resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4"
|
||||||
|
Loading…
Reference in New Issue
Block a user