diff options
Diffstat (limited to 'miner/agent.go')
-rw-r--r-- | miner/agent.go | 98 |
1 files changed, 98 insertions, 0 deletions
diff --git a/miner/agent.go b/miner/agent.go new file mode 100644 index 000000000..c650fa2f3 --- /dev/null +++ b/miner/agent.go @@ -0,0 +1,98 @@ +package miner + +import ( + "sync" + + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/core/types" + "github.com/ethereum/go-ethereum/pow" +) + +type CpuMiner struct { + chMu sync.Mutex + c chan *types.Block + quit chan struct{} + quitCurrentOp chan struct{} + returnCh chan<- *types.Block + + index int + pow pow.PoW +} + +func NewCpuMiner(index int, pow pow.PoW) *CpuMiner { + miner := &CpuMiner{ + pow: pow, + index: index, + } + + return miner +} + +func (self *CpuMiner) Work() chan<- *types.Block { return self.c } +func (self *CpuMiner) Pow() pow.PoW { return self.pow } +func (self *CpuMiner) SetReturnCh(ch chan<- *types.Block) { self.returnCh = ch } + +func (self *CpuMiner) Stop() { + close(self.quit) + close(self.quitCurrentOp) +} + +func (self *CpuMiner) Start() { + self.quit = make(chan struct{}) + self.quitCurrentOp = make(chan struct{}, 1) + self.c = make(chan *types.Block, 1) + + go self.update() +} + +func (self *CpuMiner) update() { +out: + for { + select { + case block := <-self.c: + self.chMu.Lock() + self.quitCurrentOp <- struct{}{} + self.chMu.Unlock() + + go self.mine(block) + case <-self.quit: + break out + } + } + + close(self.quitCurrentOp) +done: + // Empty channel + for { + select { + case <-self.c: + default: + close(self.c) + + break done + } + } +} + +func (self *CpuMiner) mine(block *types.Block) { + minerlogger.Debugf("(re)started agent[%d]. mining...\n", self.index) + + // Reset the channel + self.chMu.Lock() + self.quitCurrentOp = make(chan struct{}, 1) + self.chMu.Unlock() + + // Mine + nonce, mixDigest, _ := self.pow.Search(block, self.quitCurrentOp) + if nonce != 0 { + block.SetNonce(nonce) + block.Header().MixDigest = common.BytesToHash(mixDigest) + self.returnCh <- block + } else { + self.returnCh <- nil + } +} + +func (self *CpuMiner) GetHashRate() int64 { + return self.pow.GetHashrate() +} |