forked from MoneroOcean/node-cryptonight-hashing
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
65 lines (56 loc) · 2.31 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
module.exports = require('bindings')('cryptonight-hashing.node');
const uint64be = require('uint64be');
const blake2 = require('blake2');
const BigIntBuffer = require('bigint-buffer');
const M = Buffer.concat(Array(1024).fill().map((_, i) => uint64be.encode(i)));
const NBase = BigInt(Math.pow(2, 26));
const IncreaseStart = 600 * 1024;
const IncreasePeriodForN = 50 * 1024;
const NIncreasementHeightMax = 9216000;
const N = height => {
height = Math.min(NIncreasementHeightMax, height);
if (height < IncreaseStart) {
return NBase;
} else if(height >= NIncreasementHeightMax){
return 2147387550;
} else {
let res = NBase;
let iterationsNumber = Math.floor((height - IncreaseStart) / IncreasePeriodForN) + 1;
for (let i = 0; i < iterationsNumber; i++) {
res = res / BigInt(100) * BigInt(105);
}
return res;
}
}
function blake2b(seed) {
const h = blake2.createHash('blake2b', {digestLength: 32});
h.update(seed);
return h.digest();
}
function genIndexes(seed, height) {
const hash = blake2b(seed);
const extendedHash = new Uint8Array(hash.length * 2);
extendedHash.__proto__ = hash.__proto__;
extendedHash.set(hash);
extendedHash.set(hash, hash.length);
return Array.from({length: 32}).map((_, index) => extendedHash.readUIntBE(index, 4) % parseInt(N(height)))
}
// function serializeCoinbase(extraNonce1, extraNonce2){
// return Buffer.concat([
// msg,
// extraNonce1,
// extraNonce2
// ]);
// };
// const extraNonce1Buffer = Buffer.from(extraNonce1, 'hex');
// const extraNonce2Buffer = Buffer.from(extraNonce2, 'hex');
// const coinbaseBuffer = serializeCoinbase(extraNonce1Buffer, extraNonce2Buffer);
module.exports.autolykos2_hashes = function(coinbaseBuffer, height) {
const h = BigIntBuffer.toBufferBE(BigInt(height), 4);
const i = BigIntBuffer.toBufferBE(BigIntBuffer.toBigIntBE(blake2b(coinbaseBuffer).slice(24, 32)) % N(height), 4);
const e = blake2b(Buffer.concat([i, h, M])).slice(1, 32);
const J = genIndexes(Buffer.concat([e, coinbaseBuffer]), height).map(item => BigIntBuffer.toBufferBE(BigInt(item), 4));
const f = J.map(item => BigIntBuffer.toBigIntBE(blake2b(Buffer.concat([item, h, M])).slice(1, 32))).reduce((a, b) => a + b);
const hash = BigIntBuffer.toBufferBE(f, 32);
return [ hash, blake2b(hash) ];
};