| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243 | 
							- "use strict";
 
- Object.defineProperty(exports, "__esModule", { value: true });
 
- exports.blake3 = void 0;
 
- const _assert_js_1 = require("./_assert.js");
 
- const _u64_js_1 = require("./_u64.js");
 
- const _blake_js_1 = require("./_blake.js");
 
- const blake2s_js_1 = require("./blake2s.js");
 
- const utils_js_1 = require("./utils.js");
 
- const SIGMA = /* @__PURE__ */ (() => {
 
-     const Id = Array.from({ length: 16 }, (_, i) => i);
 
-     const permute = (arr) => [2, 6, 3, 10, 7, 0, 4, 13, 1, 11, 12, 5, 9, 14, 15, 8].map((i) => arr[i]);
 
-     const res = [];
 
-     for (let i = 0, v = Id; i < 7; i++, v = permute(v))
 
-         res.push(...v);
 
-     return Uint8Array.from(res);
 
- })();
 
- // Why is this so slow? It should be 6x faster than blake2b.
 
- // - There is only 30% reduction in number of rounds from blake2s
 
- // - This function uses tree mode to achive parallelisation via SIMD and threading,
 
- //   however in JS we don't have threads and SIMD, so we get only overhead from tree structure
 
- // - It is possible to speed it up via Web Workers, hovewer it will make code singnificantly more
 
- //   complicated, which we are trying to avoid, since this library is intended to be used
 
- //   for cryptographic purposes. Also, parallelization happens only on chunk level (1024 bytes),
 
- //   which won't really benefit small inputs.
 
- class BLAKE3 extends _blake_js_1.BLAKE {
 
-     constructor(opts = {}, flags = 0) {
 
-         super(64, opts.dkLen === undefined ? 32 : opts.dkLen, {}, Number.MAX_SAFE_INTEGER, 0, 0);
 
-         this.flags = 0 | 0;
 
-         this.chunkPos = 0; // Position of current block in chunk
 
-         this.chunksDone = 0; // How many chunks we already have
 
-         this.stack = [];
 
-         // Output
 
-         this.posOut = 0;
 
-         this.bufferOut32 = new Uint32Array(16);
 
-         this.chunkOut = 0; // index of output chunk
 
-         this.enableXOF = true;
 
-         this.outputLen = opts.dkLen === undefined ? 32 : opts.dkLen;
 
-         (0, _assert_js_1.number)(this.outputLen);
 
-         if (opts.key !== undefined && opts.context !== undefined)
 
-             throw new Error('Blake3: only key or context can be specified at same time');
 
-         else if (opts.key !== undefined) {
 
-             const key = (0, utils_js_1.toBytes)(opts.key).slice();
 
-             if (key.length !== 32)
 
-                 throw new Error('Blake3: key should be 32 byte');
 
-             this.IV = (0, utils_js_1.u32)(key);
 
-             if (!utils_js_1.isLE)
 
-                 (0, utils_js_1.byteSwap32)(this.IV);
 
-             this.flags = flags | 16 /* B3_Flags.KEYED_HASH */;
 
-         }
 
-         else if (opts.context !== undefined) {
 
-             const context_key = new BLAKE3({ dkLen: 32 }, 32 /* B3_Flags.DERIVE_KEY_CONTEXT */)
 
-                 .update(opts.context)
 
-                 .digest();
 
-             this.IV = (0, utils_js_1.u32)(context_key);
 
-             if (!utils_js_1.isLE)
 
-                 (0, utils_js_1.byteSwap32)(this.IV);
 
-             this.flags = flags | 64 /* B3_Flags.DERIVE_KEY_MATERIAL */;
 
-         }
 
-         else {
 
-             this.IV = blake2s_js_1.B2S_IV.slice();
 
-             this.flags = flags;
 
-         }
 
-         this.state = this.IV.slice();
 
-         this.bufferOut = (0, utils_js_1.u8)(this.bufferOut32);
 
-     }
 
-     // Unused
 
-     get() {
 
-         return [];
 
-     }
 
-     set() { }
 
-     b2Compress(counter, flags, buf, bufPos = 0) {
 
-         const { state: s, pos } = this;
 
-         const { h, l } = (0, _u64_js_1.fromBig)(BigInt(counter), true);
 
-         // prettier-ignore
 
-         const { v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15 } = (0, blake2s_js_1.compress)(SIGMA, bufPos, buf, 7, s[0], s[1], s[2], s[3], s[4], s[5], s[6], s[7], blake2s_js_1.B2S_IV[0], blake2s_js_1.B2S_IV[1], blake2s_js_1.B2S_IV[2], blake2s_js_1.B2S_IV[3], h, l, pos, flags);
 
-         s[0] = v0 ^ v8;
 
-         s[1] = v1 ^ v9;
 
-         s[2] = v2 ^ v10;
 
-         s[3] = v3 ^ v11;
 
-         s[4] = v4 ^ v12;
 
-         s[5] = v5 ^ v13;
 
-         s[6] = v6 ^ v14;
 
-         s[7] = v7 ^ v15;
 
-     }
 
-     compress(buf, bufPos = 0, isLast = false) {
 
-         // Compress last block
 
-         let flags = this.flags;
 
-         if (!this.chunkPos)
 
-             flags |= 1 /* B3_Flags.CHUNK_START */;
 
-         if (this.chunkPos === 15 || isLast)
 
-             flags |= 2 /* B3_Flags.CHUNK_END */;
 
-         if (!isLast)
 
-             this.pos = this.blockLen;
 
-         this.b2Compress(this.chunksDone, flags, buf, bufPos);
 
-         this.chunkPos += 1;
 
-         // If current block is last in chunk (16 blocks), then compress chunks
 
-         if (this.chunkPos === 16 || isLast) {
 
-             let chunk = this.state;
 
-             this.state = this.IV.slice();
 
-             // If not the last one, compress only when there are trailing zeros in chunk counter
 
-             // chunks used as binary tree where current stack is path. Zero means current leaf is finished and can be compressed.
 
-             // 1 (001) - leaf not finished (just push current chunk to stack)
 
-             // 2 (010) - leaf finished at depth=1 (merge with last elm on stack and push back)
 
-             // 3 (011) - last leaf not finished
 
-             // 4 (100) - leafs finished at depth=1 and depth=2
 
-             for (let last, chunks = this.chunksDone + 1; isLast || !(chunks & 1); chunks >>= 1) {
 
-                 if (!(last = this.stack.pop()))
 
-                     break;
 
-                 this.buffer32.set(last, 0);
 
-                 this.buffer32.set(chunk, 8);
 
-                 this.pos = this.blockLen;
 
-                 this.b2Compress(0, this.flags | 4 /* B3_Flags.PARENT */, this.buffer32, 0);
 
-                 chunk = this.state;
 
-                 this.state = this.IV.slice();
 
-             }
 
-             this.chunksDone++;
 
-             this.chunkPos = 0;
 
-             this.stack.push(chunk);
 
-         }
 
-         this.pos = 0;
 
-     }
 
-     _cloneInto(to) {
 
-         to = super._cloneInto(to);
 
-         const { IV, flags, state, chunkPos, posOut, chunkOut, stack, chunksDone } = this;
 
-         to.state.set(state.slice());
 
-         to.stack = stack.map((i) => Uint32Array.from(i));
 
-         to.IV.set(IV);
 
-         to.flags = flags;
 
-         to.chunkPos = chunkPos;
 
-         to.chunksDone = chunksDone;
 
-         to.posOut = posOut;
 
-         to.chunkOut = chunkOut;
 
-         to.enableXOF = this.enableXOF;
 
-         to.bufferOut32.set(this.bufferOut32);
 
-         return to;
 
-     }
 
-     destroy() {
 
-         this.destroyed = true;
 
-         this.state.fill(0);
 
-         this.buffer32.fill(0);
 
-         this.IV.fill(0);
 
-         this.bufferOut32.fill(0);
 
-         for (let i of this.stack)
 
-             i.fill(0);
 
-     }
 
-     // Same as b2Compress, but doesn't modify state and returns 16 u32 array (instead of 8)
 
-     b2CompressOut() {
 
-         const { state: s, pos, flags, buffer32, bufferOut32: out32 } = this;
 
-         const { h, l } = (0, _u64_js_1.fromBig)(BigInt(this.chunkOut++));
 
-         if (!utils_js_1.isLE)
 
-             (0, utils_js_1.byteSwap32)(buffer32);
 
-         // prettier-ignore
 
-         const { v0, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15 } = (0, blake2s_js_1.compress)(SIGMA, 0, buffer32, 7, s[0], s[1], s[2], s[3], s[4], s[5], s[6], s[7], blake2s_js_1.B2S_IV[0], blake2s_js_1.B2S_IV[1], blake2s_js_1.B2S_IV[2], blake2s_js_1.B2S_IV[3], l, h, pos, flags);
 
-         out32[0] = v0 ^ v8;
 
-         out32[1] = v1 ^ v9;
 
-         out32[2] = v2 ^ v10;
 
-         out32[3] = v3 ^ v11;
 
-         out32[4] = v4 ^ v12;
 
-         out32[5] = v5 ^ v13;
 
-         out32[6] = v6 ^ v14;
 
-         out32[7] = v7 ^ v15;
 
-         out32[8] = s[0] ^ v8;
 
-         out32[9] = s[1] ^ v9;
 
-         out32[10] = s[2] ^ v10;
 
-         out32[11] = s[3] ^ v11;
 
-         out32[12] = s[4] ^ v12;
 
-         out32[13] = s[5] ^ v13;
 
-         out32[14] = s[6] ^ v14;
 
-         out32[15] = s[7] ^ v15;
 
-         if (!utils_js_1.isLE) {
 
-             (0, utils_js_1.byteSwap32)(buffer32);
 
-             (0, utils_js_1.byteSwap32)(out32);
 
-         }
 
-         this.posOut = 0;
 
-     }
 
-     finish() {
 
-         if (this.finished)
 
-             return;
 
-         this.finished = true;
 
-         // Padding
 
-         this.buffer.fill(0, this.pos);
 
-         // Process last chunk
 
-         let flags = this.flags | 8 /* B3_Flags.ROOT */;
 
-         if (this.stack.length) {
 
-             flags |= 4 /* B3_Flags.PARENT */;
 
-             if (!utils_js_1.isLE)
 
-                 (0, utils_js_1.byteSwap32)(this.buffer32);
 
-             this.compress(this.buffer32, 0, true);
 
-             if (!utils_js_1.isLE)
 
-                 (0, utils_js_1.byteSwap32)(this.buffer32);
 
-             this.chunksDone = 0;
 
-             this.pos = this.blockLen;
 
-         }
 
-         else {
 
-             flags |= (!this.chunkPos ? 1 /* B3_Flags.CHUNK_START */ : 0) | 2 /* B3_Flags.CHUNK_END */;
 
-         }
 
-         this.flags = flags;
 
-         this.b2CompressOut();
 
-     }
 
-     writeInto(out) {
 
-         (0, _assert_js_1.exists)(this, false);
 
-         (0, _assert_js_1.bytes)(out);
 
-         this.finish();
 
-         const { blockLen, bufferOut } = this;
 
-         for (let pos = 0, len = out.length; pos < len;) {
 
-             if (this.posOut >= blockLen)
 
-                 this.b2CompressOut();
 
-             const take = Math.min(blockLen - this.posOut, len - pos);
 
-             out.set(bufferOut.subarray(this.posOut, this.posOut + take), pos);
 
-             this.posOut += take;
 
-             pos += take;
 
-         }
 
-         return out;
 
-     }
 
-     xofInto(out) {
 
-         if (!this.enableXOF)
 
-             throw new Error('XOF is not possible after digest call');
 
-         return this.writeInto(out);
 
-     }
 
-     xof(bytes) {
 
-         (0, _assert_js_1.number)(bytes);
 
-         return this.xofInto(new Uint8Array(bytes));
 
-     }
 
-     digestInto(out) {
 
-         (0, _assert_js_1.output)(out, this);
 
-         if (this.finished)
 
-             throw new Error('digest() was already called');
 
-         this.enableXOF = false;
 
-         this.writeInto(out);
 
-         this.destroy();
 
-         return out;
 
-     }
 
-     digest() {
 
-         return this.digestInto(new Uint8Array(this.outputLen));
 
-     }
 
- }
 
- /**
 
-  * BLAKE3 hash function.
 
-  * @param msg - message that would be hashed
 
-  * @param opts - dkLen, key, context
 
-  */
 
- exports.blake3 = (0, utils_js_1.wrapXOFConstructorWithOpts)((opts) => new BLAKE3(opts));
 
- //# sourceMappingURL=blake3.js.map
 
 
  |