import { BLOCK_MAXSIZE } from "./rt/common"; import { ArrayBuffer } from "./arraybuffer"; import { E_INDEXOUTOFRANGE, E_INVALIDLENGTH } from "./util/error"; // TODO: there is probably a smarter way to check byteOffset for accesses larger than 1 byte export class DataView { private data: ArrayBuffer; private dataStart: usize; private dataLength: i32; constructor( buffer: ArrayBuffer, byteOffset: i32 = 0, byteLength: i32 = buffer.byteLength ) { if ( i32(byteLength > BLOCK_MAXSIZE) | i32(byteOffset + byteLength > buffer.byteLength) ) throw new RangeError(E_INVALIDLENGTH); this.data = buffer; // retains var dataStart = changetype(buffer) + byteOffset; this.dataStart = dataStart; this.dataLength = byteLength; } get buffer(): ArrayBuffer { return this.data; } get byteOffset(): i32 { return (this.dataStart - changetype(this.data)); } get byteLength(): i32 { return this.dataLength; } getFloat32(byteOffset: i32, littleEndian: boolean = false): f32 { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); return littleEndian ? load(this.dataStart + byteOffset) : reinterpret( bswap( load(this.dataStart + byteOffset) ) ); } getFloat64(byteOffset: i32, littleEndian: boolean = false): f64 { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); return littleEndian ? load(this.dataStart + byteOffset) : reinterpret( bswap( load(this.dataStart + byteOffset) ) ); } getInt8(byteOffset: i32): i8 { if (byteOffset >= this.dataLength) throw new RangeError(E_INDEXOUTOFRANGE); return load(this.dataStart + byteOffset); } getInt16(byteOffset: i32, littleEndian: boolean = false): i16 { if ( i32(byteOffset < 0) | i32(byteOffset + 2 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result: i16 = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } getInt32(byteOffset: i32, littleEndian: boolean = false): i32 { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result: i32 = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } getUint8(byteOffset: i32): u8 { if (byteOffset >= this.dataLength) throw new RangeError(E_INDEXOUTOFRANGE); return load(this.dataStart + byteOffset); } getUint16(byteOffset: i32, littleEndian: boolean = false): u16 { if ( i32(byteOffset < 0) | i32(byteOffset + 2 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result: u16 = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } getUint32(byteOffset: i32, littleEndian: boolean = false): u32 { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result: u32 = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } setFloat32(byteOffset: i32, value: f32, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); if (littleEndian) store(this.dataStart + byteOffset, value); else store(this.dataStart + byteOffset, bswap(reinterpret(value))); } setFloat64(byteOffset: i32, value: f64, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); if (littleEndian) store(this.dataStart + byteOffset, value); else store(this.dataStart + byteOffset, bswap(reinterpret(value))); } setInt8(byteOffset: i32, value: i8): void { if (byteOffset >= this.dataLength) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, value); } setInt16(byteOffset: i32, value: i16, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 2 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } setInt32(byteOffset: i32, value: i32, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } setUint8(byteOffset: i32, value: u8): void { if (byteOffset >= this.dataLength) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, value); } setUint16(byteOffset: i32, value: u16, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 2 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } setUint32(byteOffset: i32, value: u32, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 4 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } // Non-standard additions that make sense in WebAssembly, but won't work in JS: getInt64(byteOffset: i32, littleEndian: boolean = false): i64 { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result: i64 = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } getUint64(byteOffset: i32, littleEndian: boolean = false): u64 { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); var result = load(this.dataStart + byteOffset); return littleEndian ? result : bswap(result); } setInt64(byteOffset: i32, value: i64, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } setUint64(byteOffset: i32, value: u64, littleEndian: boolean = false): void { if ( i32(byteOffset < 0) | i32(byteOffset + 8 > this.dataLength) ) throw new RangeError(E_INDEXOUTOFRANGE); store(this.dataStart + byteOffset, littleEndian ? value : bswap(value)); } toString(): string { return "[object DataView]"; } }