go-ethereum/ethminer/miner.go

202 lines
5.9 KiB
Go
Raw Normal View History

2014-03-20 05:20:29 -05:00
package ethminer
import (
"bytes"
"github.com/ethereum/eth-go/ethchain"
2014-06-26 12:45:57 -05:00
"github.com/ethereum/eth-go/ethlog"
"github.com/ethereum/eth-go/ethreact"
2014-03-20 05:20:29 -05:00
"github.com/ethereum/eth-go/ethwire"
"sort"
2014-03-20 05:20:29 -05:00
)
var logger = ethlog.NewLogger("MINER")
2014-03-20 05:20:29 -05:00
type Miner struct {
2014-05-21 17:25:34 -05:00
pow ethchain.PoW
ethereum ethchain.EthManager
coinbase []byte
reactChan chan ethreact.Event
txs ethchain.Transactions
2014-05-21 17:25:34 -05:00
uncles []*ethchain.Block
block *ethchain.Block
powChan chan []byte
powQuitChan chan ethreact.Event
2014-07-21 07:26:29 -05:00
quitChan chan chan error
2014-03-20 05:20:29 -05:00
}
func (self *Miner) GetPow() ethchain.PoW {
return self.pow
2014-07-18 05:01:08 -05:00
}
func NewDefaultMiner(coinbase []byte, ethereum ethchain.EthManager) *Miner {
2014-03-20 05:20:29 -05:00
miner := Miner{
pow: &ethchain.EasyPow{},
ethereum: ethereum,
coinbase: coinbase,
2014-03-20 05:20:29 -05:00
}
return &miner
2014-03-20 05:20:29 -05:00
}
2014-06-26 12:45:57 -05:00
2014-03-20 05:20:29 -05:00
func (miner *Miner) Start() {
miner.reactChan = make(chan ethreact.Event, 1) // This is the channel that receives 'updates' when ever a new transaction or block comes in
miner.powChan = make(chan []byte, 1) // This is the channel that receives valid sha hashes for a given block
miner.powQuitChan = make(chan ethreact.Event, 1) // This is the channel that can exit the miner thread
2014-07-21 07:26:29 -05:00
miner.quitChan = make(chan chan error, 1)
// Insert initial TXs in our little miner 'pool'
miner.txs = miner.ethereum.TxPool().Flush()
miner.block = miner.ethereum.BlockChain().NewBlock(miner.coinbase)
2014-03-20 05:20:29 -05:00
// Prepare inital block
//miner.ethereum.StateManager().Prepare(miner.block.State(), miner.block.State())
go miner.listener()
reactor := miner.ethereum.Reactor()
reactor.Subscribe("newBlock", miner.reactChan)
reactor.Subscribe("newTx:pre", miner.reactChan)
// We need the quit chan to be a Reactor event.
// The POW search method is actually blocking and if we don't
// listen to the reactor events inside of the pow itself
// The miner overseer will never get the reactor events themselves
// Only after the miner will find the sha
reactor.Subscribe("newBlock", miner.powQuitChan)
reactor.Subscribe("newTx:pre", miner.powQuitChan)
2014-06-26 12:45:57 -05:00
logger.Infoln("Started")
2014-07-21 07:26:29 -05:00
reactor.Post("miner:start", miner)
2014-03-20 05:20:29 -05:00
}
2014-06-26 12:45:57 -05:00
2014-03-20 05:20:29 -05:00
func (miner *Miner) listener() {
for {
select {
2014-07-21 07:26:29 -05:00
case status := <-miner.quitChan:
2014-06-26 12:45:57 -05:00
logger.Infoln("Stopped")
2014-07-21 07:26:29 -05:00
status <- nil
return
2014-03-20 05:20:29 -05:00
case chanMessage := <-miner.reactChan:
2014-06-26 12:45:57 -05:00
2014-03-20 05:20:29 -05:00
if block, ok := chanMessage.Resource.(*ethchain.Block); ok {
//logger.Infoln("Got new block via Reactor")
2014-03-20 05:20:29 -05:00
if bytes.Compare(miner.ethereum.BlockChain().CurrentBlock.Hash(), block.Hash()) == 0 {
// TODO: Perhaps continue mining to get some uncle rewards
//logger.Infoln("New top block found resetting state")
2014-03-20 05:20:29 -05:00
// Filter out which Transactions we have that were not in this block
var newtxs []*ethchain.Transaction
for _, tx := range miner.txs {
found := false
for _, othertx := range block.Transactions() {
if bytes.Compare(tx.Hash(), othertx.Hash()) == 0 {
found = true
}
}
if found == false {
newtxs = append(newtxs, tx)
}
}
miner.txs = newtxs
// Setup a fresh state to mine on
//miner.block = miner.ethereum.BlockChain().NewBlock(miner.coinbase, miner.txs)
2014-03-20 05:20:29 -05:00
} else {
if bytes.Compare(block.PrevHash, miner.ethereum.BlockChain().CurrentBlock.PrevHash) == 0 {
logger.Infoln("Adding uncle block")
2014-03-20 05:20:29 -05:00
miner.uncles = append(miner.uncles, block)
}
}
}
if tx, ok := chanMessage.Resource.(*ethchain.Transaction); ok {
found := false
for _, ctx := range miner.txs {
if found = bytes.Compare(ctx.Hash(), tx.Hash()) == 0; found {
break
}
}
if found == false {
// Undo all previous commits
miner.block.Undo()
// Apply new transactions
2014-03-20 05:20:29 -05:00
miner.txs = append(miner.txs, tx)
}
}
default:
miner.mineNewBlock()
}
}
}
func (miner *Miner) Stop() {
2014-06-26 12:45:57 -05:00
logger.Infoln("Stopping...")
2014-07-21 07:26:29 -05:00
status := make(chan error)
miner.quitChan <- status
<-status
reactor := miner.ethereum.Reactor()
reactor.Unsubscribe("newBlock", miner.powQuitChan)
reactor.Unsubscribe("newTx:pre", miner.powQuitChan)
reactor.Unsubscribe("newBlock", miner.reactChan)
reactor.Unsubscribe("newTx:pre", miner.reactChan)
2014-07-01 04:55:50 -05:00
close(miner.powQuitChan)
close(miner.quitChan)
2014-07-21 07:26:29 -05:00
reactor.Post("miner:stop", miner)
2014-05-21 17:25:34 -05:00
}
func (self *Miner) mineNewBlock() {
stateManager := self.ethereum.StateManager()
self.block = self.ethereum.BlockChain().NewBlock(self.coinbase)
2014-03-20 05:20:29 -05:00
// Apply uncles
if len(self.uncles) > 0 {
self.block.SetUncles(self.uncles)
}
2014-03-20 05:20:29 -05:00
// Sort the transactions by nonce in case of odd network propagation
sort.Sort(ethchain.TxByNonce{self.txs})
2014-06-14 04:46:09 -05:00
// Accumulate all valid transactions and apply them to the new state
2014-06-14 04:46:09 -05:00
// Error may be ignored. It's not important during mining
2014-06-16 04:51:16 -05:00
parent := self.ethereum.BlockChain().GetBlock(self.block.PrevHash)
2014-06-16 04:14:01 -05:00
coinbase := self.block.State().GetOrNewStateObject(self.block.Coinbase)
2014-06-16 04:51:16 -05:00
coinbase.SetGasPool(self.block.CalcGasLimit(parent))
2014-06-16 04:14:01 -05:00
receipts, txs, unhandledTxs, err := stateManager.ProcessTransactions(coinbase, self.block.State(), self.block, self.block, self.txs)
2014-06-14 04:46:09 -05:00
if err != nil {
logger.Debugln(err)
2014-06-14 04:46:09 -05:00
}
self.txs = append(txs, unhandledTxs...)
2014-07-21 05:21:34 -05:00
self.block.SetTxHash(receipts)
2014-06-14 04:46:09 -05:00
// Set the transactions to the block so the new SHA3 can be calculated
self.block.SetReceipts(receipts, txs)
2014-06-14 04:46:09 -05:00
// Accumulate the rewards included for this block
stateManager.AccumelateRewards(self.block.State(), self.block)
self.block.State().Update()
2014-06-25 12:43:35 -05:00
logger.Infof("Mining on block. Includes %v transactions", len(self.txs))
// Find a valid nonce
2014-05-21 17:25:34 -05:00
self.block.Nonce = self.pow.Search(self.block, self.powQuitChan)
if self.block.Nonce != nil {
err := self.ethereum.StateManager().Process(self.block, false)
if err != nil {
logger.Infoln(err)
} else {
self.ethereum.Broadcast(ethwire.MsgBlockTy, []interface{}{self.block.Value().Val})
logger.Infof("🔨 Mined block %x\n", self.block.Hash())
logger.Infoln(self.block)
// Gather the new batch of transactions currently in the tx pool
self.txs = self.ethereum.TxPool().CurrentTransactions()
2014-03-20 05:20:29 -05:00
}
}
}