Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: hashInto() and digest64Into() api #480

Open
wants to merge 4 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 27 additions & 0 deletions packages/as-sha256/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,19 @@ export function digest64(data: Uint8Array): Uint8Array {
throw new Error("InvalidLengthForDigest64");
}

export function digest64Into(data: Uint8Array, output: Uint8Array): void {
if (data.length !== 64) {
throw new Error(`InvalidLengthForDigest64, got ${data.length}`);
}
if (output.length !== 32) {
throw new Error(`InvalidLengthForOutput32, got ${output.length}`);
}

inputUint8Array.set(data);
ctx.digest64(wasmInputValue, wasmOutputValue);
output.set(outputUint8Array32);
}

export function digest2Bytes32(bytes1: Uint8Array, bytes2: Uint8Array): Uint8Array {
if (bytes1.length === 32 && bytes2.length === 32) {
inputUint8Array.set(bytes1);
Expand All @@ -64,6 +77,20 @@ export function digest2Bytes32(bytes1: Uint8Array, bytes2: Uint8Array): Uint8Arr
throw new Error("InvalidLengthForDigest64");
}

export function digest2Bytes32Into(bytes1: Uint8Array, bytes2: Uint8Array, output: Uint8Array): void {
if (bytes1.length !== 32 || bytes2.length !== 32) {
throw new Error("InvalidLengthForDigest64");
}
if (output.length !== 32) {
throw new Error("InvalidLengthForOutput32");
}

inputUint8Array.set(bytes1);
inputUint8Array.set(bytes2, 32);
ctx.digest64(wasmInputValue, wasmOutputValue);
output.set(outputUint8Array32);
}

/**
* Digest 2 objects, each has 8 properties from h0 to h7.
* The performance is a little bit better than digest64 due to the use of Uint32Array
Expand Down
14 changes: 12 additions & 2 deletions packages/as-sha256/test/unit/index.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@ import {
byteArrayToHashObject,
digest,
digest2Bytes32,
digest2Bytes32Into,
digest64,
digest64HashObjects,
digest64Into,
hashObjectToByteArray,
} from "../../src/index.js";

Expand Down Expand Up @@ -74,11 +76,15 @@ describe("as-sha256 non-SIMD enabled methods", () => {
}
});

it("digest64()", () => {
it("digest64() and digest64Into()", () => {
const input = Buffer.alloc(64, "lodestar");
const output = Buffer.from(digest64(input)).toString("hex");
const expected = createHash("sha256").update(input).digest("hex");
expect(output).to.equal(expected);

const output2 = Buffer.alloc(32);
digest64Into(input, output2);
expect(output2.toString("hex")).to.equal(expected);
});

it("digest() and digest64() output matches", () => {
Expand All @@ -88,14 +94,18 @@ describe("as-sha256 non-SIMD enabled methods", () => {
expect(output).to.be.equal(output64);
});

it("digest2Bytes32()", () => {
it("digest2Bytes32() and digest2Bytes32Into()", () => {
const input1 = randomBytes(32);
const input2 = randomBytes(32);
const output = Buffer.from(digest2Bytes32(input1, input2)).toString("hex");
const expectedOutput = createHash("sha256")
.update(Buffer.of(...input1, ...input2))
.digest("hex");
expect(output).to.equal(expectedOutput);

const output2 = Buffer.alloc(32);
digest2Bytes32Into(input1, input2, output2);
expect(output2.toString("hex")).to.equal(expectedOutput);
});

it("digest2Bytes32() matches digest64()", () => {
Expand Down
3 changes: 3 additions & 0 deletions packages/persistent-merkle-tree/src/hasher/as-sha256.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import {
batchHash4HashObjectInputs,
digest2Bytes32,
digest2Bytes32Into,
digest64HashObjects,
digest64HashObjectsInto,
hashInto,
Expand All @@ -18,7 +19,9 @@ const buffer = new Uint8Array(4 * BLOCK_SIZE);

export const hasher: Hasher = {
name: "as-sha256",
hashInto,
digest64: digest2Bytes32,
digest64Into: digest2Bytes32Into,
digest64HashObjects: digest64HashObjectsInto,
merkleizeBlocksBytes(blocksBytes: Uint8Array, padFor: number, output: Uint8Array, offset: number): void {
doMerkleizeBlocksBytes(blocksBytes, padFor, output, offset, hashInto);
Expand Down
13 changes: 13 additions & 0 deletions packages/persistent-merkle-tree/src/hasher/hashtree.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ const destNodes: Node[] = new Array<Node>(PARALLEL_FACTOR);

export const hasher: Hasher = {
name: "hashtree",
hashInto,
digest64(obj1: Uint8Array, obj2: Uint8Array): Uint8Array {
if (obj1.length !== 32 || obj2.length !== 32) {
throw new Error("Invalid input length");
Expand All @@ -35,6 +36,18 @@ export const hasher: Hasher = {
hashInto(hash64Input, hash64Output);
return hash64Output.slice();
},
digest64Into: (obj1: Uint8Array, obj2: Uint8Array, output: Uint8Array): void => {
if (obj1.length !== 32 || obj2.length !== 32) {
throw new Error("Invalid input length");
}
if (output.length !== 32) {
throw new Error("Invalid output length");
}

hash64Input.set(obj1, 0);
hash64Input.set(obj2, 32);
hashInto(hash64Input, output);
},
digest64HashObjects(left: HashObject, right: HashObject, parent: HashObject): void {
hashObjectsToUint32Array(left, right, uint32Input);
hashInto(hash64Input, hash64Output);
Expand Down
8 changes: 8 additions & 0 deletions packages/persistent-merkle-tree/src/hasher/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,18 @@ export function setHasher(newHasher: Hasher): void {
hasher = newHasher;
}

export function hashInto(data: Uint8Array, output: Uint8Array): void {
hasher.hashInto(data, output);
}

export function digest64(a: Uint8Array, b: Uint8Array): Uint8Array {
return hasher.digest64(a, b);
}

export function digest64Into(a: Uint8Array, b: Uint8Array, output: Uint8Array): void {
hasher.digest64Into(a, b, output);
}

export function digestNLevel(data: Uint8Array, nLevel: number): Uint8Array {
return hasher.digestNLevel(data, nLevel);
}
Expand Down
15 changes: 15 additions & 0 deletions packages/persistent-merkle-tree/src/hasher/noble.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ import {
hashObjectToUint8Array,
} from "./util.js";

const hash64Input = new Uint8Array(64);

const digest64 = (a: Uint8Array, b: Uint8Array): Uint8Array => sha256.create().update(a).update(b).digest();
const hashInto = (input: Uint8Array, output: Uint8Array): void => {
if (input.length % 64 !== 0) {
Expand All @@ -33,7 +35,20 @@ const buffer = new Uint8Array(4 * BLOCK_SIZE);

export const hasher: Hasher = {
name: "noble",
hashInto,
digest64,
digest64Into: (a, b, output) => {
if (a.length !== 32 || b.length !== 32) {
throw new Error("Invalid input length");
}
if (output.length !== 32) {
throw new Error("Invalid output length");
}

hash64Input.set(a, 0);
hash64Input.set(b, 32);
hashInto(hash64Input, output);
},
digest64HashObjects: (left, right, parent) => {
byteArrayIntoHashObject(digest64(hashObjectToUint8Array(left), hashObjectToUint8Array(right)), 0, parent);
},
Expand Down
9 changes: 9 additions & 0 deletions packages/persistent-merkle-tree/src/hasher/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,19 @@ export type {HashObject};
export type Hasher = {
// name of the hashing library
name: string;
/**
* Hash input Uint8Array into output Uint8Array
* output.length = input.length / 2
*/
hashInto(input: Uint8Array, output: Uint8Array): void;
/**
* Hash two 32-byte Uint8Arrays
*/
digest64(a32Bytes: Uint8Array, b32Bytes: Uint8Array): Uint8Array;
/**
* The same to digest64, but output is passed as argument
*/
digest64Into(a32Bytes: Uint8Array, b32Bytes: Uint8Array, output: Uint8Array): void;
/**
* Hash two 32-byte HashObjects
*/
Expand Down
27 changes: 25 additions & 2 deletions packages/persistent-merkle-tree/test/perf/hasher.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import {hasher as nobleHasher} from "../../src/hasher/noble.js";
import {HashComputationLevel, getHashComputations} from "../../src/index.js";
import {buildComparisonTrees} from "../utils/tree.js";

const PARALLEL_FACTOR = 16;

describe("hasher", () => {
const iterations = 500_000;

Expand All @@ -18,16 +20,34 @@ describe("hasher", () => {
root2[i] = 2;
}

const batchInput = new Uint8Array(PARALLEL_FACTOR * 64).fill(1);
const batchOutput = new Uint8Array(PARALLEL_FACTOR * 32);
const hashers: Hasher[] = [hashtreeHasher, asSha256Hasher, nobleHasher];

const runsFactor = 10;
for (const hasher of hashers) {
describe(hasher.name, () => {
bench({
id: `hash 2 Uint8Array ${iterations} times - ${hasher.name}`,
id: `hash 2 32 bytes Uint8Array ${iterations} times - ${hasher.name}`,
fn: () => {
const output = new Uint8Array(32);
for (let i = 0; i < runsFactor; i++) {
// should not use `hasher.digest64` here because of memory allocation, and it's not comparable
// to the batch hash test below and `digest64HashObjects`
for (let j = 0; j < iterations; j++) hasher.digest64Into(root1, root2, output);
}
},
runsFactor,
});

// use this test to see how faster batch hash is compared to single hash in `digest64`
bench({
id: `batch hash ${PARALLEL_FACTOR} x 64 Uint8Array ${iterations / PARALLEL_FACTOR} times - ${hasher.name}`,
fn: () => {
for (let i = 0; i < runsFactor; i++) {
for (let j = 0; j < iterations; j++) hasher.digest64(root1, root2);
for (let j = 0; j < iterations / PARALLEL_FACTOR; j++) {
hasher.hashInto(batchInput, batchOutput);
}
}
},
runsFactor,
Expand All @@ -49,6 +69,7 @@ describe("hasher", () => {
runsFactor,
});

// use to compare performance between hashers
bench({
id: `executeHashComputations - ${hasher.name}`,
beforeEach: () => {
Expand Down Expand Up @@ -78,6 +99,7 @@ describe("hashtree", () => {
},
});

// compare this to "get root" to see how efficient the hash computation is
bench({
id: "executeHashComputations",
beforeEach: () => {
Expand All @@ -91,6 +113,7 @@ describe("hashtree", () => {
},
});

// the traditional/naive way of getting the root
bench({
id: "get root",
beforeEach: async () => {
Expand Down