733ecd2681
Add root Cargo workspace with 9 crates: salvium-crypto (extended), salvium-types, salvium-consensus, salvium-wallet, salvium-tx, salvium-rpc, salvium-miner (extended), salvium-cli, salvium-multisig. New modules: chain_state, block_weight, alt_chain, validation, offline signing, stake lifecycle, wallet sync/query/encryption/utxo, randomx utilities, and full multisig crate with CARROT support. Delete 188 JS test/helper/debug files; archive integration test scripts to test/legacy-js/ for live testnet use. Testnet integration tests (transfer, stake, burn, convert, sweep) remain as #[ignore]- gated Rust tests runnable with --ignored against a live daemon.
313 lines
8.1 KiB
JavaScript
313 lines
8.1 KiB
JavaScript
/**
|
|
* Debug - compare pass 1 first iteration between JS and WASM
|
|
*/
|
|
|
|
import { blake2b } from '../src/blake2b.js';
|
|
import { readFileSync } from 'fs';
|
|
import { fileURLToPath } from 'url';
|
|
import { dirname, join } from 'path';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = dirname(__filename);
|
|
|
|
const ARGON2_BLOCK_SIZE = 1024;
|
|
const ARGON2_PREHASH_DIGEST_LENGTH = 64;
|
|
const ARGON2_PREHASH_SEED_LENGTH = 72;
|
|
const ARGON2_SYNC_POINTS = 4;
|
|
const ARGON2_VERSION = 0x13;
|
|
const RANDOMX_ARGON_MEMORY = 262144;
|
|
const RANDOMX_ARGON_ITERATIONS = 3;
|
|
const RANDOMX_ARGON_LANES = 1;
|
|
const RANDOMX_ARGON_SALT = new TextEncoder().encode("RandomX\x03");
|
|
|
|
function blake2bLong(outLen, input) {
|
|
const prefixed = new Uint8Array(4 + input.length);
|
|
prefixed[0] = outLen & 0xff;
|
|
prefixed[1] = (outLen >> 8) & 0xff;
|
|
prefixed[2] = (outLen >> 16) & 0xff;
|
|
prefixed[3] = (outLen >> 24) & 0xff;
|
|
prefixed.set(input, 4);
|
|
|
|
if (outLen <= 64) {
|
|
return blake2b(prefixed, outLen);
|
|
}
|
|
|
|
const result = new Uint8Array(outLen);
|
|
let v = blake2b(prefixed, 64);
|
|
result.set(v.subarray(0, 32), 0);
|
|
|
|
let pos = 32;
|
|
while (pos < outLen - 64) {
|
|
v = blake2b(v, 64);
|
|
result.set(v.subarray(0, 32), pos);
|
|
pos += 32;
|
|
}
|
|
|
|
const remaining = outLen - pos;
|
|
v = blake2b(v, remaining);
|
|
result.set(v, pos);
|
|
|
|
return result;
|
|
}
|
|
|
|
function initialHash(ctx) {
|
|
const parts = [];
|
|
const addU32 = (val) => {
|
|
const arr = new Uint8Array(4);
|
|
arr[0] = val & 0xff;
|
|
arr[1] = (val >> 8) & 0xff;
|
|
arr[2] = (val >> 16) & 0xff;
|
|
arr[3] = (val >> 24) & 0xff;
|
|
parts.push(arr);
|
|
};
|
|
|
|
addU32(ctx.lanes);
|
|
addU32(ctx.outLen);
|
|
addU32(ctx.mCost);
|
|
addU32(ctx.tCost);
|
|
addU32(ctx.version);
|
|
addU32(ctx.type);
|
|
addU32(ctx.password.length);
|
|
if (ctx.password.length > 0) parts.push(ctx.password);
|
|
addU32(ctx.salt.length);
|
|
if (ctx.salt.length > 0) parts.push(ctx.salt);
|
|
addU32(0);
|
|
addU32(0);
|
|
|
|
let totalLen = 0;
|
|
for (const p of parts) totalLen += p.length;
|
|
const input = new Uint8Array(totalLen);
|
|
let offset = 0;
|
|
for (const p of parts) {
|
|
input.set(p, offset);
|
|
offset += p.length;
|
|
}
|
|
|
|
return blake2b(input, ARGON2_PREHASH_DIGEST_LENGTH);
|
|
}
|
|
|
|
// Reference JS implementation of fill_block
|
|
function rotr64(x, n) {
|
|
const mask = (1n << 64n) - 1n;
|
|
return ((x >> BigInt(n)) | (x << BigInt(64 - n))) & mask;
|
|
}
|
|
|
|
function fBlaMka(x, y) {
|
|
const mask32 = 0xFFFFFFFFn;
|
|
const mask64 = (1n << 64n) - 1n;
|
|
const xy = (x & mask32) * (y & mask32);
|
|
return (x + y + (xy << 1n)) & mask64;
|
|
}
|
|
|
|
function G(v, a, b, c, d) {
|
|
v[a] = fBlaMka(v[a], v[b]);
|
|
v[d] = rotr64(v[d] ^ v[a], 32);
|
|
v[c] = fBlaMka(v[c], v[d]);
|
|
v[b] = rotr64(v[b] ^ v[c], 24);
|
|
v[a] = fBlaMka(v[a], v[b]);
|
|
v[d] = rotr64(v[d] ^ v[a], 16);
|
|
v[c] = fBlaMka(v[c], v[d]);
|
|
v[b] = rotr64(v[b] ^ v[c], 63);
|
|
}
|
|
|
|
function blake2RoundNoMsg(v) {
|
|
G(v, 0, 4, 8, 12);
|
|
G(v, 1, 5, 9, 13);
|
|
G(v, 2, 6, 10, 14);
|
|
G(v, 3, 7, 11, 15);
|
|
G(v, 0, 5, 10, 15);
|
|
G(v, 1, 6, 11, 12);
|
|
G(v, 2, 7, 8, 13);
|
|
G(v, 3, 4, 9, 14);
|
|
}
|
|
|
|
function jsFillBlock(prevBlock, refBlock, currBlock, withXor) {
|
|
// blockR = ref XOR prev
|
|
const blockR = new Array(128);
|
|
for (let i = 0; i < 128; i++) {
|
|
blockR[i] = prevBlock[i] ^ refBlock[i];
|
|
}
|
|
|
|
// blockTmp = blockR (or XOR with curr if withXor)
|
|
const blockTmp = new Array(128);
|
|
if (withXor) {
|
|
for (let i = 0; i < 128; i++) {
|
|
blockTmp[i] = blockR[i] ^ currBlock[i];
|
|
}
|
|
} else {
|
|
for (let i = 0; i < 128; i++) {
|
|
blockTmp[i] = blockR[i];
|
|
}
|
|
}
|
|
|
|
// Column rounds
|
|
for (let i = 0; i < 8; i++) {
|
|
const v = [];
|
|
for (let j = 0; j < 16; j++) {
|
|
v.push(blockR[i * 16 + j]);
|
|
}
|
|
blake2RoundNoMsg(v);
|
|
for (let j = 0; j < 16; j++) {
|
|
blockR[i * 16 + j] = v[j];
|
|
}
|
|
}
|
|
|
|
// Row rounds
|
|
for (let i = 0; i < 8; i++) {
|
|
const indices = [
|
|
i * 2, i * 2 + 1, i * 2 + 16, i * 2 + 17,
|
|
i * 2 + 32, i * 2 + 33, i * 2 + 48, i * 2 + 49,
|
|
i * 2 + 64, i * 2 + 65, i * 2 + 80, i * 2 + 81,
|
|
i * 2 + 96, i * 2 + 97, i * 2 + 112, i * 2 + 113
|
|
];
|
|
const v = indices.map(idx => blockR[idx]);
|
|
blake2RoundNoMsg(v);
|
|
for (let j = 0; j < 16; j++) {
|
|
blockR[indices[j]] = v[j];
|
|
}
|
|
}
|
|
|
|
// Result = blockTmp XOR blockR
|
|
const result = new Array(128);
|
|
for (let i = 0; i < 128; i++) {
|
|
result[i] = blockTmp[i] ^ blockR[i];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
console.log('=== Debug pass 1 first iteration ===\n');
|
|
|
|
// Load WASM
|
|
const wasmPath = join(__dirname, '../build/randomx.wasm');
|
|
const wasmBuffer = readFileSync(wasmPath);
|
|
|
|
const wasmMemory = new WebAssembly.Memory({
|
|
initial: 4096 + 512,
|
|
maximum: 8192
|
|
});
|
|
|
|
const imports = {
|
|
env: {
|
|
memory: wasmMemory,
|
|
abort: () => {}
|
|
}
|
|
};
|
|
|
|
const { instance } = await WebAssembly.instantiate(wasmBuffer, imports);
|
|
const wasm = instance.exports;
|
|
|
|
// Setup
|
|
const key = new TextEncoder().encode("test key 000");
|
|
const lanes = RANDOMX_ARGON_LANES;
|
|
const memoryBlocks = Math.floor(RANDOMX_ARGON_MEMORY / (ARGON2_BLOCK_SIZE / 1024));
|
|
const segmentLength = Math.floor(memoryBlocks / (lanes * ARGON2_SYNC_POINTS));
|
|
const laneLength = segmentLength * ARGON2_SYNC_POINTS;
|
|
const totalBlocks = lanes * laneLength;
|
|
const totalBytes = totalBlocks * ARGON2_BLOCK_SIZE;
|
|
|
|
// Initialize WASM
|
|
const memPtr = 0;
|
|
wasm.argon2d_init(memPtr, totalBlocks, laneLength, segmentLength);
|
|
|
|
// Create context
|
|
const ctx = {
|
|
password: key,
|
|
salt: RANDOMX_ARGON_SALT,
|
|
tCost: RANDOMX_ARGON_ITERATIONS,
|
|
mCost: RANDOMX_ARGON_MEMORY,
|
|
lanes,
|
|
outLen: 0,
|
|
version: ARGON2_VERSION,
|
|
type: 0
|
|
};
|
|
|
|
// Generate H0
|
|
const blockHash = initialHash(ctx);
|
|
|
|
// Build seed for block 0
|
|
const seed = new Uint8Array(ARGON2_PREHASH_SEED_LENGTH);
|
|
seed.set(blockHash);
|
|
|
|
// Fill first two blocks
|
|
const mem = new Uint8Array(wasmMemory.buffer);
|
|
const tempPtr = totalBytes + 1024;
|
|
|
|
seed[64] = 0; seed[65] = 0; seed[66] = 0; seed[67] = 0;
|
|
seed[68] = 0; seed[69] = 0; seed[70] = 0; seed[71] = 0;
|
|
const block0 = blake2bLong(ARGON2_BLOCK_SIZE, seed);
|
|
mem.set(block0, tempPtr);
|
|
wasm.argon2d_write_block(0, tempPtr);
|
|
|
|
seed[64] = 1;
|
|
const block1 = blake2bLong(ARGON2_BLOCK_SIZE, seed);
|
|
mem.set(block1, tempPtr);
|
|
wasm.argon2d_write_block(1, tempPtr);
|
|
|
|
const wasmView = new DataView(wasmMemory.buffer);
|
|
|
|
// Helper to read block as BigInt array
|
|
function readBlockFromWasm(blockIdx) {
|
|
const result = [];
|
|
for (let i = 0; i < 128; i++) {
|
|
result.push(wasmView.getBigUint64(blockIdx * 1024 + i * 8, true));
|
|
}
|
|
return result;
|
|
}
|
|
|
|
console.log('Running pass 0...');
|
|
for (let slice = 0; slice < ARGON2_SYNC_POINTS; slice++) {
|
|
wasm.argon2d_fill_segment(0, 0, slice);
|
|
}
|
|
console.log('Pass 0 completed');
|
|
|
|
// Read the blocks we need for pass 1 slice 0 first iteration
|
|
const prevBlockIdx = 262143;
|
|
const refBlockIdx = 259861; // Calculated from pseudoRand
|
|
const currBlockIdx = 0;
|
|
|
|
const prevBlock = readBlockFromWasm(prevBlockIdx);
|
|
const refBlock = readBlockFromWasm(refBlockIdx);
|
|
const currBlock = readBlockFromWasm(currBlockIdx);
|
|
|
|
console.log();
|
|
console.log('For pass 1, slice 0, iteration 0:');
|
|
console.log(` prevBlockIdx: ${prevBlockIdx}`);
|
|
console.log(` refBlockIdx: ${refBlockIdx}`);
|
|
console.log(` currBlockIdx: ${currBlockIdx}`);
|
|
console.log(` withXor: true (pass > 0)`);
|
|
console.log();
|
|
console.log(` prev block first qword: 0x${prevBlock[0].toString(16)}`);
|
|
console.log(` ref block first qword: 0x${refBlock[0].toString(16)}`);
|
|
console.log(` curr block first qword (before): 0x${currBlock[0].toString(16)}`);
|
|
console.log();
|
|
|
|
// Compute with JS reference
|
|
const jsResult = jsFillBlock(prevBlock, refBlock, currBlock, true);
|
|
console.log(`JS reference result first qword: 0x${jsResult[0].toString(16)}`);
|
|
|
|
// Run WASM pass 1 slice 0
|
|
wasm.argon2d_fill_segment(1, 0, 0);
|
|
|
|
const wasmResult = readBlockFromWasm(0);
|
|
console.log(`WASM result first qword: 0x${wasmResult[0].toString(16)}`);
|
|
|
|
// Compare
|
|
console.log();
|
|
console.log('Match:', jsResult[0] === wasmResult[0]);
|
|
|
|
// Compare all qwords
|
|
let mismatches = 0;
|
|
for (let i = 0; i < 128; i++) {
|
|
if (jsResult[i] !== wasmResult[i]) {
|
|
if (mismatches < 5) {
|
|
console.log(` Mismatch at qword ${i}: JS=0x${jsResult[i].toString(16)}, WASM=0x${wasmResult[i].toString(16)}`);
|
|
}
|
|
mismatches++;
|
|
}
|
|
}
|
|
if (mismatches > 0) {
|
|
console.log(`Total mismatches: ${mismatches}/128`);
|
|
} else {
|
|
console.log('All 128 qwords match between JS and WASM!');
|
|
}
|