-
Notifications
You must be signed in to change notification settings - Fork 11
/
cryptonight.go
45 lines (38 loc) · 1.39 KB
/
cryptonight.go
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
// Package cryptonight implements CryptoNight hash function and some of its
// variant. Original CryptoNight algorithm is defined in CNS008 at
// https://cryptonote.org/cns/cns008.txt
package cryptonight // import "ekyu.moe/cryptonight"
import (
"sync"
)
type cache struct {
// DO NOT change the order of these fields in this struct!
// They are carefully placed in this order to keep at least 16-byte aligned
// for some fields.
//
// In the future the alignment may be set explicitly, see
// https://github.com/golang/go/issues/19057
scratchpad [2 * 1024 * 1024 / 8]uint64 // 2 MiB scratchpad for memhard loop
finalState [25]uint64 // state of keccak1600
_ [8]byte // padded to keep 16-byte align (0x2000d0)
blocks [16]uint64 // temporary chunk/pointer of data
rkeys [40]uint32 // 10 rounds, instead of 14 as in standard AES-256
}
// cachePool is a pool of cache.
var cachePool = sync.Pool{
New: func() interface{} {
return new(cache)
},
}
// Sum calculate a CryptoNight hash digest. The return value is exactly 32 bytes
// long.
//
// When variant is 1, data is required to have at least 43 bytes.
// This is assumed and not checked by Sum. If this condition doesn't meet, Sum
// will panic straightforward.
func Sum(data []byte, variant int) []byte {
cc := cachePool.Get().(*cache)
sum := cc.sum(data, variant)
cachePool.Put(cc)
return sum
}