2015-02-28 19:58:37 +00:00
|
|
|
package ethash
|
|
|
|
|
|
|
|
/*
|
|
|
|
#cgo CFLAGS: -std=gnu99 -Wall
|
2015-03-06 12:27:11 +00:00
|
|
|
#include "src/libethash/util.c"
|
|
|
|
#include "src/libethash/internal.c"
|
|
|
|
#include "src/libethash/sha3.c"
|
2015-02-28 19:58:37 +00:00
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
2015-03-06 12:27:11 +00:00
|
|
|
"io/ioutil"
|
2015-02-28 19:58:37 +00:00
|
|
|
"log"
|
|
|
|
"math/big"
|
|
|
|
"math/rand"
|
2015-03-06 12:27:11 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
2015-02-28 19:58:37 +00:00
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
"unsafe"
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
"github.com/ethereum/go-ethereum/ethutil"
|
2015-02-28 19:58:37 +00:00
|
|
|
"github.com/ethereum/go-ethereum/logger"
|
|
|
|
"github.com/ethereum/go-ethereum/pow"
|
|
|
|
)
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
var tt256 = new(big.Int).Exp(big.NewInt(2), big.NewInt(256), big.NewInt(0))
|
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
var powlogger = logger.NewLogger("POW")
|
|
|
|
|
|
|
|
type DAG struct {
|
|
|
|
SeedBlockNum uint64
|
|
|
|
dag unsafe.Pointer // full GB of memory for dag
|
2015-03-06 12:27:11 +00:00
|
|
|
file bool
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type ParamsAndCache struct {
|
|
|
|
params *C.ethash_params
|
|
|
|
cache *C.ethash_cache
|
|
|
|
SeedBlockNum uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
type Ethash struct {
|
|
|
|
turbo bool
|
|
|
|
HashRate int64
|
|
|
|
chainManager pow.ChainManager
|
|
|
|
dag *DAG
|
|
|
|
paramsAndCache *ParamsAndCache
|
|
|
|
nextdag unsafe.Pointer
|
|
|
|
ret *C.ethash_return_value
|
|
|
|
dagMutex *sync.Mutex
|
|
|
|
cacheMutex *sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseNonce(nonce []byte) (uint64, error) {
|
|
|
|
nonceBuf := bytes.NewBuffer(nonce)
|
|
|
|
nonceInt, err := binary.ReadUvarint(nonceBuf)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return nonceInt, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
const epochLength uint64 = 30000
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
func GetSeedBlockNum(blockNum uint64) uint64 {
|
|
|
|
var seedBlockNum uint64 = 0
|
|
|
|
if blockNum > epochLength {
|
|
|
|
seedBlockNum = ((blockNum - 1) / epochLength) * epochLength
|
|
|
|
}
|
|
|
|
return seedBlockNum
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
XXX THIS DOESN'T WORK!! NEEDS FIXING
|
|
|
|
blockEpoch will underflow and wrap around causing massive issues
|
|
|
|
func GetSeedBlockNum(blockNum uint64) uint64 {
|
2015-02-28 19:58:37 +00:00
|
|
|
var seedBlockNum uint64 = 0
|
2015-03-06 12:27:11 +00:00
|
|
|
if blockNum > epochLength {
|
|
|
|
seedBlockNum = ((blockNum - 1) / epochLength) * epochLength
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
return seedBlockNum
|
|
|
|
}
|
2015-03-06 12:27:11 +00:00
|
|
|
*/
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
func makeParamsAndCache(chainManager pow.ChainManager, blockNum uint64) *ParamsAndCache {
|
2015-03-06 12:27:11 +00:00
|
|
|
seedBlockNum := GetSeedBlockNum(blockNum)
|
2015-02-28 19:58:37 +00:00
|
|
|
paramsAndCache := &ParamsAndCache{
|
|
|
|
params: new(C.ethash_params),
|
|
|
|
cache: new(C.ethash_cache),
|
|
|
|
SeedBlockNum: seedBlockNum,
|
|
|
|
}
|
|
|
|
C.ethash_params_init(paramsAndCache.params, C.uint32_t(seedBlockNum))
|
|
|
|
paramsAndCache.cache.mem = C.malloc(paramsAndCache.params.cache_size)
|
2015-03-06 12:27:11 +00:00
|
|
|
seedHash := chainManager.GetBlockByNumber(seedBlockNum).SeedHash()
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
log.Println("Making Cache")
|
|
|
|
start := time.Now()
|
2015-03-06 12:27:11 +00:00
|
|
|
C.ethash_mkcache(paramsAndCache.cache, paramsAndCache.params, (*C.uint8_t)(unsafe.Pointer(&seedHash[0])))
|
2015-02-28 19:58:37 +00:00
|
|
|
log.Println("Took:", time.Since(start))
|
2015-03-06 12:27:11 +00:00
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
return paramsAndCache
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) updateCache() {
|
|
|
|
pow.cacheMutex.Lock()
|
2015-03-06 12:27:11 +00:00
|
|
|
seedNum := GetSeedBlockNum(pow.chainManager.CurrentBlock().NumberU64())
|
2015-02-28 19:58:37 +00:00
|
|
|
if pow.paramsAndCache.SeedBlockNum != seedNum {
|
|
|
|
pow.paramsAndCache = makeParamsAndCache(pow.chainManager, pow.chainManager.CurrentBlock().NumberU64())
|
|
|
|
}
|
|
|
|
pow.cacheMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
func makeDAG(p *ParamsAndCache) *DAG {
|
|
|
|
d := &DAG{
|
|
|
|
dag: C.malloc(p.params.full_size),
|
|
|
|
SeedBlockNum: p.SeedBlockNum,
|
|
|
|
}
|
|
|
|
C.ethash_compute_full_data(d.dag, p.params, p.cache)
|
|
|
|
return d
|
|
|
|
}
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
func (pow *Ethash) writeDagToDisk(dag *DAG, seedNum uint64) *os.File {
|
|
|
|
data := C.GoBytes(unsafe.Pointer(dag.dag), C.int(pow.paramsAndCache.params.full_size))
|
|
|
|
file, err := os.Create("/tmp/dag")
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
num := make([]byte, 8)
|
|
|
|
binary.BigEndian.PutUint64(num, seedNum)
|
|
|
|
|
|
|
|
file.Write(num)
|
|
|
|
file.Write(data)
|
|
|
|
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) UpdateDAG() {
|
2015-02-28 19:58:37 +00:00
|
|
|
pow.cacheMutex.Lock()
|
|
|
|
pow.dagMutex.Lock()
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
seedNum := GetSeedBlockNum(pow.chainManager.CurrentBlock().NumberU64())
|
2015-02-28 19:58:37 +00:00
|
|
|
if pow.dag == nil || pow.dag.SeedBlockNum != seedNum {
|
2015-03-06 12:27:11 +00:00
|
|
|
if pow.dag != nil && pow.dag.dag != nil {
|
|
|
|
C.free(pow.dag.dag)
|
|
|
|
pow.dag.dag = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
path := path.Join("/", "tmp", "dag")
|
2015-02-28 19:58:37 +00:00
|
|
|
pow.dag = nil
|
2015-03-06 12:27:11 +00:00
|
|
|
log.Println("Generating dag")
|
2015-02-28 19:58:37 +00:00
|
|
|
start := time.Now()
|
2015-03-06 12:27:11 +00:00
|
|
|
|
|
|
|
file, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("No dag found in '%s'. Generating new dago(takes a while)...", path)
|
|
|
|
pow.dag = makeDAG(pow.paramsAndCache)
|
|
|
|
file = pow.writeDagToDisk(pow.dag, seedNum)
|
|
|
|
} else {
|
|
|
|
data, err := ioutil.ReadAll(file)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
num := binary.BigEndian.Uint64(data[0:8])
|
|
|
|
if num < seedNum {
|
|
|
|
log.Printf("Old found. Generating new dag (takes a while)...")
|
|
|
|
pow.dag = makeDAG(pow.paramsAndCache)
|
|
|
|
file = pow.writeDagToDisk(pow.dag, seedNum)
|
|
|
|
} else {
|
|
|
|
data = data[8:]
|
|
|
|
pow.dag = &DAG{
|
|
|
|
dag: unsafe.Pointer(&data[0]),
|
|
|
|
file: true,
|
|
|
|
SeedBlockNum: pow.paramsAndCache.SeedBlockNum,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-28 19:58:37 +00:00
|
|
|
log.Println("Took:", time.Since(start))
|
2015-03-06 12:27:11 +00:00
|
|
|
|
|
|
|
file.Close()
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pow.dagMutex.Unlock()
|
|
|
|
pow.cacheMutex.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
func New(chainManager pow.ChainManager) *Ethash {
|
|
|
|
return &Ethash{
|
2015-03-06 12:27:11 +00:00
|
|
|
turbo: true,
|
2015-02-28 19:58:37 +00:00
|
|
|
paramsAndCache: makeParamsAndCache(chainManager, chainManager.CurrentBlock().NumberU64()),
|
|
|
|
chainManager: chainManager,
|
|
|
|
dag: nil,
|
|
|
|
cacheMutex: new(sync.Mutex),
|
|
|
|
dagMutex: new(sync.Mutex),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) DAGSize() uint64 {
|
|
|
|
return uint64(pow.paramsAndCache.params.full_size)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) CacheSize() uint64 {
|
|
|
|
return uint64(pow.paramsAndCache.params.cache_size)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) GetSeedHash(blockNum uint64) []byte {
|
2015-03-06 12:27:11 +00:00
|
|
|
seednum := GetSeedBlockNum(blockNum)
|
|
|
|
return pow.chainManager.GetBlockByNumber(seednum).SeedHash()
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) Stop() {
|
|
|
|
pow.cacheMutex.Lock()
|
|
|
|
pow.dagMutex.Lock()
|
2015-03-06 12:27:11 +00:00
|
|
|
defer pow.dagMutex.Unlock()
|
|
|
|
defer pow.cacheMutex.Unlock()
|
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
if pow.paramsAndCache.cache != nil {
|
|
|
|
C.free(pow.paramsAndCache.cache.mem)
|
|
|
|
}
|
2015-03-06 12:27:11 +00:00
|
|
|
if pow.dag.dag != nil && !pow.dag.file {
|
2015-02-28 19:58:37 +00:00
|
|
|
C.free(pow.dag.dag)
|
|
|
|
}
|
2015-03-06 12:27:11 +00:00
|
|
|
pow.dag.dag = nil
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
func (pow *Ethash) Search(block pow.Block, stop <-chan struct{}) (uint64, []byte, []byte) {
|
|
|
|
//pow.UpdateDAG()
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
// Not very elegant, multiple mining instances are not supported
|
2015-03-06 12:27:11 +00:00
|
|
|
//pow.dagMutex.Lock()
|
|
|
|
//pow.cacheMutex.Lock()
|
|
|
|
//defer pow.cacheMutex.Unlock()
|
|
|
|
//defer pow.dagMutex.Unlock()
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
r := rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
|
|
miningHash := block.HashNoNonce()
|
|
|
|
diff := block.Difficulty()
|
2015-03-06 12:27:11 +00:00
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
i := int64(0)
|
2015-03-06 12:27:11 +00:00
|
|
|
starti := i
|
2015-02-28 19:58:37 +00:00
|
|
|
start := time.Now().UnixNano()
|
|
|
|
|
|
|
|
nonce := uint64(r.Int63())
|
2015-03-06 12:27:11 +00:00
|
|
|
cMiningHash := (*C.uint8_t)(unsafe.Pointer(&miningHash[0]))
|
|
|
|
target := new(big.Int).Div(tt256, diff)
|
2015-02-28 19:58:37 +00:00
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
var ret C.ethash_return_value
|
2015-02-28 19:58:37 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-stop:
|
|
|
|
powlogger.Infoln("Breaking from mining")
|
|
|
|
pow.HashRate = 0
|
2015-03-06 12:27:11 +00:00
|
|
|
return 0, nil, nil
|
2015-02-28 19:58:37 +00:00
|
|
|
default:
|
|
|
|
i++
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
elapsed := time.Now().UnixNano() - start
|
|
|
|
hashes := ((float64(1e9) / float64(elapsed)) * float64(i-starti)) / 1000
|
|
|
|
pow.HashRate = int64(hashes)
|
2015-02-28 19:58:37 +00:00
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
C.ethash_full(&ret, pow.dag.dag, pow.paramsAndCache.params, cMiningHash, C.uint64_t(nonce))
|
|
|
|
result := ethutil.Bytes2Big(C.GoBytes(unsafe.Pointer(&ret.result[0]), C.int(32)))
|
|
|
|
|
|
|
|
if result.Cmp(target) <= 0 {
|
|
|
|
mixDigest := C.GoBytes(unsafe.Pointer(&ret.mix_hash[0]), C.int(32))
|
|
|
|
|
|
|
|
return nonce, mixDigest, pow.GetSeedHash(block.NumberU64())
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
}
|
2015-03-06 12:27:11 +00:00
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
nonce += 1
|
|
|
|
}
|
|
|
|
|
|
|
|
if !pow.turbo {
|
|
|
|
time.Sleep(20 * time.Microsecond)
|
|
|
|
}
|
|
|
|
}
|
2015-03-06 12:27:11 +00:00
|
|
|
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) Verify(block pow.Block) bool {
|
|
|
|
// Make sure the SeedHash is set correctly
|
|
|
|
if bytes.Compare(block.SeedHash(), pow.GetSeedHash(block.NumberU64())) != 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
return pow.verify(block.HashNoNonce(), block.MixDigest(), block.Difficulty(), block.NumberU64(), block.Nonce())
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) verify(hash []byte, mixDigest []byte, difficulty *big.Int, blockNum uint64, nonce uint64) bool {
|
|
|
|
// First check: make sure header, mixDigest, nonce are correct without hitting the DAG
|
|
|
|
// This is to prevent DOS attacks
|
2015-03-06 12:27:11 +00:00
|
|
|
chash := (*C.uint8_t)(unsafe.Pointer(&hash[0]))
|
2015-02-28 19:58:37 +00:00
|
|
|
cnonce := C.uint64_t(nonce)
|
2015-03-06 12:27:11 +00:00
|
|
|
target := new(big.Int).Div(tt256, difficulty)
|
2015-02-28 19:58:37 +00:00
|
|
|
|
|
|
|
var pAc *ParamsAndCache
|
|
|
|
// If its an old block (doesn't use the current cache)
|
|
|
|
// get the cache for it but don't update (so we don't need the mutex)
|
|
|
|
// Otherwise, it's the current block or a future.
|
|
|
|
// If current, updateCache will do nothing.
|
2015-03-06 12:27:11 +00:00
|
|
|
if GetSeedBlockNum(blockNum) < pow.paramsAndCache.SeedBlockNum {
|
2015-02-28 19:58:37 +00:00
|
|
|
pAc = makeParamsAndCache(pow.chainManager, blockNum)
|
|
|
|
} else {
|
|
|
|
pow.updateCache()
|
|
|
|
pow.cacheMutex.Lock()
|
|
|
|
defer pow.cacheMutex.Unlock()
|
|
|
|
pAc = pow.paramsAndCache
|
|
|
|
}
|
|
|
|
|
2015-03-06 12:27:11 +00:00
|
|
|
ret := new(C.ethash_return_value)
|
|
|
|
|
|
|
|
C.ethash_light(ret, pAc.cache, pAc.params, chash, cnonce)
|
|
|
|
|
|
|
|
result := ethutil.Bytes2Big(C.GoBytes(unsafe.Pointer(&ret.result[0]), C.int(32)))
|
|
|
|
return result.Cmp(target) <= 0
|
2015-02-28 19:58:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) GetHashrate() int64 {
|
|
|
|
return pow.HashRate
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) Turbo(on bool) {
|
|
|
|
pow.turbo = on
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) FullHash(nonce uint64, miningHash []byte) []byte {
|
2015-03-06 12:27:11 +00:00
|
|
|
pow.UpdateDAG()
|
2015-02-28 19:58:37 +00:00
|
|
|
pow.dagMutex.Lock()
|
|
|
|
defer pow.dagMutex.Unlock()
|
2015-03-06 12:27:11 +00:00
|
|
|
cMiningHash := (*C.uint8_t)(unsafe.Pointer(&miningHash[0]))
|
2015-02-28 19:58:37 +00:00
|
|
|
cnonce := C.uint64_t(nonce)
|
2015-03-06 12:27:11 +00:00
|
|
|
ret := new(C.ethash_return_value)
|
2015-02-28 19:58:37 +00:00
|
|
|
// pow.hash is the output/return of ethash_full
|
2015-03-06 12:27:11 +00:00
|
|
|
C.ethash_full(ret, pow.dag.dag, pow.paramsAndCache.params, cMiningHash, cnonce)
|
|
|
|
ghash_full := C.GoBytes(unsafe.Pointer(&ret.result), 32)
|
2015-02-28 19:58:37 +00:00
|
|
|
return ghash_full
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pow *Ethash) LightHash(nonce uint64, miningHash []byte) []byte {
|
2015-03-06 12:27:11 +00:00
|
|
|
cMiningHash := (*C.uint8_t)(unsafe.Pointer(&miningHash[0]))
|
2015-02-28 19:58:37 +00:00
|
|
|
cnonce := C.uint64_t(nonce)
|
2015-03-06 12:27:11 +00:00
|
|
|
ret := new(C.ethash_return_value)
|
|
|
|
C.ethash_light(ret, pow.paramsAndCache.cache, pow.paramsAndCache.params, cMiningHash, cnonce)
|
|
|
|
ghash_light := C.GoBytes(unsafe.Pointer(&ret.result), 32)
|
2015-02-28 19:58:37 +00:00
|
|
|
return ghash_light
|
|
|
|
}
|