7f01cafc62
Crypto Backend Refactoring
──────────────────────────
- Delete src/ed25519.js — all scalar/point operations now route through
the crypto provider (WASM/FFI/JSI backends only)
- Remove duplicate JS BigInt implementations of scReduce32, scReduce64,
scalarAdd, scalarMul from scanning.js, carrot.js, and carrot-scanning.js;
delegate to Rust backend via crypto/index.js
- Remove debug/test exports from index.js (randomPoint, testDouble,
getBasePoint, isOnCurve, etc.) that were only used during development
- Rebuild WASM binary (476KB → 487KB) with updated Rust crate
Consolidated CryptoNote Scanner (cn_scan)
─────────────────────────────────────────
- New Rust module crates/salvium-crypto/src/cn_scan.rs replaces 5-12
individual FFI round-trips per output with a single native call:
view tag check → derive subaddress pubkey → subaddress map lookup →
amount decryption → commitment mask → key image generation
- FFI wrapper salvium_cn_scan_output in ffi.rs + C header declaration
- FFI backend scanCnOutput() with full subaddress map marshaling
(32-byte key + u32 major/minor LE per entry) and JSON result parsing
- JSI backend delegation via this.native.cnScanOutput()
- wallet-sync.js _scanCNOutput() tries native path first when available
(FFI/JSI), falls through to existing JS pipeline for WASM/JS backends
- Change pub(crate) visibility on subaddress.rs cn_subaddress_secret_key
- 8 Rust unit tests covering view tag, amount, commitment mask,
subaddress matching, and key image generation
- Verified identical results: WASM (JS fallback) and FFI (native cn_scan)
produce same 964 outputs at same chain height; FFI is 3x faster sync
(0.8s vs 2.5s) with 12x less heap (12MB vs 150MB)
RCT Batch Signature Verification
─────────────────────────────────
- New Rust module crates/salvium-crypto/src/rct_verify.rs — single-call
verification of all ring signatures in a transaction (CLSAG + TCLSAG),
avoiding N individual JS↔Rust boundary crossings
- Computes pre-MLSAG message hash matching C++ get_pre_mlsag_hash
- FFI export salvium_verify_rct_signatures with flat byte array interface
- FFI backend verifyRctSignatures() method
- JS backend stub returns null (validation.js handles JS fallback)
- validation.js: 200+ lines of RCT verification logic including
flattenKeyImages, packTclsagSigsFlat, packClsagSigsFlat helpers
Transaction Expansion
─────────────────────
- transaction.js: add expandTransaction() matching C++ expand_transaction_2
(copies key images from prefix inputs into TCLSAG/CLSAG signature structs)
- New test/expand-transaction.test.js (634 lines)
- New test/rct-verify-testnet.test.js (430 lines)
Mining Resilience
─────────────────
- salvium-miner main.rs: retry get_info and get_block_template up to 5
times with 2s delay for transient daemon errors
- full-testnet.js mineTo(): retry miner up to 3 times with 3s delay,
check for partial progress between attempts
Testnet Tooling
───────────────
- sync-only.js: CRYPTO_BACKEND env var for A/B testing (wasm vs ffi)
- full-testnet.js: daemon URL update (node12.whiskymine.io)
- Debug scripts for cn_scan development (debug-cn-scan/marshal/match/wasm)
- Android .gitignore and build-bundle.sh for mobile builds
188 lines
6.0 KiB
JavaScript
188 lines
6.0 KiB
JavaScript
#!/usr/bin/env bun
|
|
/**
|
|
* Local RandomX miner against remote Salvium daemon
|
|
*
|
|
* Uses our WASM RandomX (full mode, 2GB dataset) + daemon RPC.
|
|
* Single-threaded but with full dataset for maximum per-thread hashrate.
|
|
*
|
|
* Usage:
|
|
* bun mine-testnet.js [mode]
|
|
* mode: "full" (default, 2GB dataset, ~30-50 H/s) or "light" (256MB, ~10 H/s)
|
|
*/
|
|
|
|
import { DaemonRPC } from './src/rpc/daemon.js';
|
|
import { RandomXContext } from './src/randomx/index.js';
|
|
import { RandomXFullMode } from './src/randomx/full-mode.js';
|
|
import {
|
|
parseBlockTemplate,
|
|
findNonceOffset,
|
|
setNonce,
|
|
checkHash,
|
|
formatBlockForSubmission,
|
|
formatHashrate,
|
|
} from './src/mining.js';
|
|
import { hexToBytes, bytesToHex } from './src/address.js';
|
|
|
|
// ============================================================================
|
|
// Configuration
|
|
// ============================================================================
|
|
|
|
const DAEMON_URL = process.env.DAEMON_URL || 'http://node12.whiskymine.io:29081';
|
|
const WALLET_FILE = process.env.WALLET_FILE || `${process.env.HOME}/testnet-wallet/wallet.json`;
|
|
const MODE = process.argv[2] || 'full';
|
|
const TEMPLATE_REFRESH_MS = 5000;
|
|
|
|
// ============================================================================
|
|
// Main
|
|
// ============================================================================
|
|
|
|
const wallet = JSON.parse(await Bun.file(WALLET_FILE).text());
|
|
const daemon = new DaemonRPC({ url: DAEMON_URL });
|
|
|
|
console.log('Salvium Local RandomX Miner');
|
|
console.log('===========================');
|
|
console.log(`Daemon: ${DAEMON_URL}`);
|
|
console.log(`Wallet: ${wallet.address.slice(0, 20)}...`);
|
|
console.log(`Mode: ${MODE} (${MODE === 'full' ? '2GB dataset' : '256MB cache'})`);
|
|
console.log('');
|
|
|
|
// Verify daemon
|
|
const info = await daemon.getInfo();
|
|
if (!info.success) {
|
|
console.error('Cannot connect to daemon');
|
|
process.exit(1);
|
|
}
|
|
console.log(`Daemon height: ${info.result.height}, difficulty: ${info.result.difficulty}`);
|
|
console.log('');
|
|
|
|
// Get block template
|
|
let template = await daemon.getBlockTemplate(wallet.address, 8);
|
|
if (!template.success) {
|
|
console.error('Failed to get block template:', template.error);
|
|
process.exit(1);
|
|
}
|
|
|
|
let parsed = parseBlockTemplate(template.result);
|
|
console.log(`Template: height=${parsed.height}, difficulty=${parsed.difficulty}`);
|
|
|
|
// Initialize RandomX
|
|
const seedBytes = parsed.seedHash ? hexToBytes(parsed.seedHash) : new Uint8Array(32);
|
|
let rx;
|
|
|
|
if (MODE === 'full') {
|
|
console.log('Initializing RandomX full mode (256MB cache + 2GB dataset)...');
|
|
console.log('This will take 30-60 seconds...');
|
|
rx = new RandomXFullMode();
|
|
await rx.init(seedBytes, {
|
|
onProgress: (pct, phase) => {
|
|
process.stdout.write(`\r ${phase}: ${pct}% `);
|
|
}
|
|
});
|
|
console.log('');
|
|
} else {
|
|
console.log('Initializing RandomX light mode (256MB cache)...');
|
|
rx = new RandomXContext();
|
|
await rx.init(seedBytes);
|
|
}
|
|
|
|
console.log('RandomX ready.\n');
|
|
|
|
// ============================================================================
|
|
// Mining loop
|
|
// ============================================================================
|
|
|
|
let totalHashes = 0;
|
|
let blocksFound = 0;
|
|
const startTime = Date.now();
|
|
let lastTemplateFetch = Date.now();
|
|
let lastStatsPrint = Date.now();
|
|
let currentSeedHash = parsed.seedHash;
|
|
let running = true;
|
|
|
|
process.on('SIGINT', () => {
|
|
running = false;
|
|
console.log('\nShutting down...');
|
|
const elapsed = (Date.now() - startTime) / 1000;
|
|
console.log(`Total hashes: ${totalHashes.toLocaleString()}`);
|
|
console.log(`Blocks found: ${blocksFound}`);
|
|
console.log(`Avg hashrate: ${formatHashrate(totalHashes / elapsed)}`);
|
|
process.exit(0);
|
|
});
|
|
|
|
console.log('Mining started. Press Ctrl+C to stop.\n');
|
|
|
|
while (running) {
|
|
const hashingBlob = parsed.blockhashingBytes;
|
|
const nonceOffset = findNonceOffset(hashingBlob);
|
|
const difficulty = parsed.difficulty;
|
|
const templateBlob = parsed.blocktemplateBytes;
|
|
|
|
// Random starting nonce
|
|
const startNonce = Math.floor(Math.random() * 0xFFFFFFFF);
|
|
|
|
for (let i = 0; i < 256 && running; i++) {
|
|
const nonce = (startNonce + i) & 0xFFFFFFFF;
|
|
const blob = setNonce(hashingBlob, nonce, nonceOffset);
|
|
const hash = rx.hash(blob);
|
|
totalHashes++;
|
|
|
|
if (checkHash(hash, difficulty)) {
|
|
const blockHex = formatBlockForSubmission(templateBlob, nonce,
|
|
findNonceOffset(templateBlob));
|
|
|
|
console.log(`\n*** BLOCK FOUND at height ${parsed.height}! nonce=${nonce} ***`);
|
|
|
|
const submitResult = await daemon.submitBlock([blockHex]);
|
|
if (submitResult.success) {
|
|
blocksFound++;
|
|
console.log(`Block accepted! Total: ${blocksFound}`);
|
|
} else {
|
|
console.log(`Rejected: ${JSON.stringify(submitResult.error || submitResult.result)}`);
|
|
}
|
|
|
|
// New template immediately
|
|
lastTemplateFetch = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Refresh template
|
|
const now = Date.now();
|
|
if (now - lastTemplateFetch > TEMPLATE_REFRESH_MS) {
|
|
try {
|
|
const newTemplate = await daemon.getBlockTemplate(wallet.address, 8);
|
|
if (newTemplate.success) {
|
|
const newParsed = parseBlockTemplate(newTemplate.result);
|
|
if (newParsed.seedHash !== currentSeedHash) {
|
|
console.log(`\nSeed changed, re-initializing RandomX...`);
|
|
if (MODE === 'full') {
|
|
await rx.init(hexToBytes(newParsed.seedHash), {});
|
|
} else {
|
|
await rx.init(hexToBytes(newParsed.seedHash));
|
|
}
|
|
currentSeedHash = newParsed.seedHash;
|
|
}
|
|
parsed = newParsed;
|
|
}
|
|
lastTemplateFetch = now;
|
|
} catch (e) {
|
|
// Keep mining
|
|
}
|
|
}
|
|
|
|
// Stats every 10s
|
|
if (now - lastStatsPrint > 10000) {
|
|
const elapsed = (now - startTime) / 1000;
|
|
const hr = totalHashes / elapsed;
|
|
const estBlockTime = Number(parsed.difficulty) / hr;
|
|
process.stdout.write(
|
|
`\r[H=${parsed.height}] ${formatHashrate(hr)} | ` +
|
|
`Hashes: ${totalHashes.toLocaleString()} | ` +
|
|
`Blocks: ${blocksFound} | ` +
|
|
`Diff: ${parsed.difficulty} | ` +
|
|
`Est: ${isFinite(estBlockTime) ? estBlockTime.toFixed(0) + 's' : '...'} `
|
|
);
|
|
lastStatsPrint = now;
|
|
}
|
|
}
|