diff --git a/src/Decoder.ts b/src/Decoder.ts index f3f2517..daccee8 100644 --- a/src/Decoder.ts +++ b/src/Decoder.ts @@ -707,8 +707,16 @@ export class Decoder { throw new DecodeError(`Max length exceeded: ext length (${size}) > maxExtLength (${this.maxExtLength})`); } - const extType = this.view.getInt8(this.pos + headOffset); - const data = this.decodeBinary(size, headOffset + 1 /* extType */); + let padding = 0; + let extType = this.view.getInt8(this.pos + headOffset); + + // 0xc1 => -63 (Int8) (noop byte) + while (extType === -63) { + padding++; + extType = this.view.getInt8(this.pos + headOffset + padding); + } + + const data = this.decodeBinary(size, headOffset + padding + 1 /* extType */); return this.extensionCodec.decode(data, extType, this.context); } diff --git a/src/Encoder.ts b/src/Encoder.ts index cda0822..df94109 100644 --- a/src/Encoder.ts +++ b/src/Encoder.ts @@ -427,6 +427,14 @@ export class Encoder { } else { throw new Error(`Too large extension object: ${size}`); } + if (ext.align && Number.isInteger(ext.align)) { + const align = ext.align; + const dataPos = this.pos + 1; // + extType size + const padding = (align - (dataPos % align)) % align; + for (let i = 0; i < padding; i++) { + this.writeU8(0xc1); // noop byte + } + } this.writeI8(ext.type); this.writeU8a(ext.data); } diff --git a/src/ExtData.ts b/src/ExtData.ts index facdddc..0fc5ff1 100644 --- a/src/ExtData.ts +++ b/src/ExtData.ts @@ -5,5 +5,6 @@ export class ExtData { constructor( readonly type: number, readonly data: Uint8Array, + readonly align: number | undefined | null = null, ) {} } diff --git a/src/ExtensionCodec.ts b/src/ExtensionCodec.ts index 6ca9495..7f340e8 100644 --- a/src/ExtensionCodec.ts +++ b/src/ExtensionCodec.ts @@ -34,6 +34,7 @@ export class ExtensionCodec implements ExtensionCodecTy // custom extensions private readonly encoders: Array | undefined | null> = []; private readonly decoders: Array | undefined | null> = []; + private readonly aligns: Array = []; public constructor() { this.register(timestampExtension); @@ -41,10 +42,12 @@ export class ExtensionCodec implements ExtensionCodecTy public register({ type, + align, encode, decode, }: { type: number; + align?: number; encode: ExtensionEncoderType; decode: ExtensionDecoderType; }): void { @@ -52,6 +55,7 @@ export class ExtensionCodec implements ExtensionCodecTy // custom extensions this.encoders[type] = encode; this.decoders[type] = decode; + this.aligns[type] = align; } else { // built-in extensions const index = 1 + type; @@ -80,7 +84,8 @@ export class ExtensionCodec implements ExtensionCodecTy const data = encodeExt(object, context); if (data != null) { const type = i; - return new ExtData(type, data); + const align = this.aligns[type]; + return new ExtData(type, data, align); } } } diff --git a/test/ExtensionCodec.test.ts b/test/ExtensionCodec.test.ts index 6949b17..2cd49c9 100644 --- a/test/ExtensionCodec.test.ts +++ b/test/ExtensionCodec.test.ts @@ -201,4 +201,32 @@ describe("ExtensionCodec", () => { ]); }); }); + + context("custom extensions with alignment", () => { + const extensionCodec = new ExtensionCodec(); + + extensionCodec.register({ + type: 0x01, + align: 4, + encode: (object: unknown): Uint8Array | null => { + if (object instanceof Float32Array) { + return new Uint8Array(object.buffer); + } + return null; + }, + decode: (data: Uint8Array) => { + return new Float32Array(data.buffer, data.byteOffset, data.byteLength / Float32Array.BYTES_PER_ELEMENT); + }, + }); + + it("encodes and decodes Float32Array type with zero-copy", () => { + const data = { + position: new Float32Array([1.1, 2.2, 3.3, 4.4, 5.5]), + }; + const encoded = encode(data, { extensionCodec }); + const decoded = decode(encoded, { extensionCodec }); + assert.deepStrictEqual(decoded, data); + assert.strictEqual(decoded.position.buffer, encoded.buffer); + }); + }); });