core: compute less transaction hashes in TxPool
This commit is contained in:
parent
770a0e7839
commit
08befff8f1
@ -61,7 +61,7 @@ type TxPool struct {
|
|||||||
txs map[common.Hash]*types.Transaction
|
txs map[common.Hash]*types.Transaction
|
||||||
invalidHashes *set.Set
|
invalidHashes *set.Set
|
||||||
|
|
||||||
queue map[common.Address]types.Transactions
|
queue map[common.Address]map[common.Hash]*types.Transaction
|
||||||
|
|
||||||
subscribers []chan TxMsg
|
subscribers []chan TxMsg
|
||||||
|
|
||||||
@ -71,7 +71,7 @@ type TxPool struct {
|
|||||||
func NewTxPool(eventMux *event.TypeMux, currentStateFn stateFn, gasLimitFn func() *big.Int) *TxPool {
|
func NewTxPool(eventMux *event.TypeMux, currentStateFn stateFn, gasLimitFn func() *big.Int) *TxPool {
|
||||||
txPool := &TxPool{
|
txPool := &TxPool{
|
||||||
txs: make(map[common.Hash]*types.Transaction),
|
txs: make(map[common.Hash]*types.Transaction),
|
||||||
queue: make(map[common.Address]types.Transactions),
|
queue: make(map[common.Address]map[common.Hash]*types.Transaction),
|
||||||
queueChan: make(chan *types.Transaction, txPoolQueueSize),
|
queueChan: make(chan *types.Transaction, txPoolQueueSize),
|
||||||
quit: make(chan bool),
|
quit: make(chan bool),
|
||||||
eventMux: eventMux,
|
eventMux: eventMux,
|
||||||
@ -157,9 +157,9 @@ func (self *TxPool) add(tx *types.Transaction) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
self.queueTx(hash, tx)
|
||||||
|
|
||||||
self.queueTx(tx)
|
if glog.V(logger.Debug) {
|
||||||
|
|
||||||
var toname string
|
var toname string
|
||||||
if to := tx.To(); to != nil {
|
if to := tx.To(); to != nil {
|
||||||
toname = common.Bytes2Hex(to[:4])
|
toname = common.Bytes2Hex(to[:4])
|
||||||
@ -170,9 +170,7 @@ func (self *TxPool) add(tx *types.Transaction) error {
|
|||||||
// verified in ValidateTransaction.
|
// verified in ValidateTransaction.
|
||||||
f, _ := tx.From()
|
f, _ := tx.From()
|
||||||
from := common.Bytes2Hex(f[:4])
|
from := common.Bytes2Hex(f[:4])
|
||||||
|
glog.Infof("(t) %x => %s (%v) %x\n", from, toname, tx.Value, hash)
|
||||||
if glog.V(logger.Debug) {
|
|
||||||
glog.Infof("(t) %x => %s (%v) %x\n", from, toname, tx.Value, tx.Hash())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
@ -211,16 +209,12 @@ func (tp *TxPool) GetTransaction(hash common.Hash) *types.Transaction {
|
|||||||
if tx, ok := tp.txs[hash]; ok {
|
if tx, ok := tp.txs[hash]; ok {
|
||||||
return tx
|
return tx
|
||||||
}
|
}
|
||||||
|
|
||||||
// check queue
|
// check queue
|
||||||
for _, txs := range tp.queue {
|
for _, txs := range tp.queue {
|
||||||
for _, tx := range txs {
|
if tx, ok := txs[hash]; ok {
|
||||||
if tx.Hash() == hash {
|
|
||||||
return tx
|
return tx
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -234,26 +228,26 @@ func (self *TxPool) GetTransactions() (txs types.Transactions) {
|
|||||||
txs[i] = tx
|
txs[i] = tx
|
||||||
i++
|
i++
|
||||||
}
|
}
|
||||||
|
return txs
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *TxPool) GetQueuedTransactions() types.Transactions {
|
func (self *TxPool) GetQueuedTransactions() types.Transactions {
|
||||||
self.mu.RLock()
|
self.mu.RLock()
|
||||||
defer self.mu.RUnlock()
|
defer self.mu.RUnlock()
|
||||||
|
|
||||||
var txs types.Transactions
|
var ret types.Transactions
|
||||||
for _, ts := range self.queue {
|
for _, txs := range self.queue {
|
||||||
txs = append(txs, ts...)
|
for _, tx := range txs {
|
||||||
|
ret = append(ret, tx)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return txs
|
sort.Sort(types.TxByNonce{ret})
|
||||||
|
return ret
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *TxPool) RemoveTransactions(txs types.Transactions) {
|
func (self *TxPool) RemoveTransactions(txs types.Transactions) {
|
||||||
self.mu.Lock()
|
self.mu.Lock()
|
||||||
defer self.mu.Unlock()
|
defer self.mu.Unlock()
|
||||||
|
|
||||||
for _, tx := range txs {
|
for _, tx := range txs {
|
||||||
self.removeTx(tx.Hash())
|
self.removeTx(tx.Hash())
|
||||||
}
|
}
|
||||||
@ -270,14 +264,17 @@ func (pool *TxPool) Stop() {
|
|||||||
glog.V(logger.Info).Infoln("TX Pool stopped")
|
glog.V(logger.Info).Infoln("TX Pool stopped")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *TxPool) queueTx(tx *types.Transaction) {
|
func (self *TxPool) queueTx(hash common.Hash, tx *types.Transaction) {
|
||||||
from, _ := tx.From() // already validated
|
from, _ := tx.From() // already validated
|
||||||
self.queue[from] = append(self.queue[from], tx)
|
if self.queue[from] == nil {
|
||||||
|
self.queue[from] = make(map[common.Hash]*types.Transaction)
|
||||||
|
}
|
||||||
|
self.queue[from][hash] = tx
|
||||||
}
|
}
|
||||||
|
|
||||||
func (pool *TxPool) addTx(tx *types.Transaction) {
|
func (pool *TxPool) addTx(hash common.Hash, tx *types.Transaction) {
|
||||||
if _, ok := pool.txs[tx.Hash()]; !ok {
|
if _, ok := pool.txs[hash]; !ok {
|
||||||
pool.txs[tx.Hash()] = tx
|
pool.txs[hash] = tx
|
||||||
// Notify the subscribers. This event is posted in a goroutine
|
// Notify the subscribers. This event is posted in a goroutine
|
||||||
// because it's possible that somewhere during the post "Remove transaction"
|
// because it's possible that somewhere during the post "Remove transaction"
|
||||||
// gets called which will then wait for the global tx pool lock and deadlock.
|
// gets called which will then wait for the global tx pool lock and deadlock.
|
||||||
@ -291,36 +288,33 @@ func (pool *TxPool) checkQueue() {
|
|||||||
defer pool.mu.Unlock()
|
defer pool.mu.Unlock()
|
||||||
|
|
||||||
statedb := pool.currentState()
|
statedb := pool.currentState()
|
||||||
|
var addq txQueue
|
||||||
for address, txs := range pool.queue {
|
for address, txs := range pool.queue {
|
||||||
sort.Sort(types.TxByNonce{txs})
|
curnonce := statedb.GetNonce(address)
|
||||||
|
addq := addq[:0]
|
||||||
var (
|
for hash, tx := range txs {
|
||||||
nonce = statedb.GetNonce(address)
|
if tx.AccountNonce < curnonce {
|
||||||
start int
|
// Drop queued transactions whose nonce is lower than
|
||||||
)
|
// the account nonce because they have been processed.
|
||||||
// Clean up the transactions first and determine the start of the nonces
|
delete(txs, hash)
|
||||||
for _, tx := range txs {
|
} else {
|
||||||
if tx.Nonce() >= nonce {
|
// Collect the remaining transactions for the next pass.
|
||||||
|
addq = append(addq, txQueueEntry{hash, tx})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Find the next consecutive nonce range starting at the
|
||||||
|
// current account nonce.
|
||||||
|
sort.Sort(addq)
|
||||||
|
for _, e := range addq {
|
||||||
|
if e.AccountNonce != curnonce {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
start++
|
curnonce++
|
||||||
|
delete(txs, e.hash)
|
||||||
|
pool.addTx(e.hash, e.Transaction)
|
||||||
}
|
}
|
||||||
pool.queue[address] = txs[start:]
|
// Delete the entire queue entry if it became empty.
|
||||||
|
if len(txs) == 0 {
|
||||||
// expected nonce
|
|
||||||
enonce := nonce
|
|
||||||
for _, tx := range pool.queue[address] {
|
|
||||||
// If the expected nonce does not match up with the next one
|
|
||||||
// (i.e. a nonce gap), we stop the loop
|
|
||||||
if enonce != tx.Nonce() {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
enonce++
|
|
||||||
|
|
||||||
pool.addTx(tx)
|
|
||||||
}
|
|
||||||
// delete the entire queue entry if it's empty. There's no need to keep it
|
|
||||||
if len(pool.queue[address]) == 0 {
|
|
||||||
delete(pool.queue, address)
|
delete(pool.queue, address)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -329,20 +323,16 @@ func (pool *TxPool) checkQueue() {
|
|||||||
func (pool *TxPool) removeTx(hash common.Hash) {
|
func (pool *TxPool) removeTx(hash common.Hash) {
|
||||||
// delete from pending pool
|
// delete from pending pool
|
||||||
delete(pool.txs, hash)
|
delete(pool.txs, hash)
|
||||||
|
|
||||||
// delete from queue
|
// delete from queue
|
||||||
out:
|
|
||||||
for address, txs := range pool.queue {
|
for address, txs := range pool.queue {
|
||||||
for i, tx := range txs {
|
if _, ok := txs[hash]; ok {
|
||||||
if tx.Hash() == hash {
|
|
||||||
if len(txs) == 1 {
|
if len(txs) == 1 {
|
||||||
// if only one tx, remove entire address entry
|
// if only one tx, remove entire address entry.
|
||||||
delete(pool.queue, address)
|
delete(pool.queue, address)
|
||||||
} else {
|
} else {
|
||||||
pool.queue[address][len(txs)-1], pool.queue[address] = nil, append(txs[:i], txs[i+1:]...)
|
delete(txs, hash)
|
||||||
}
|
|
||||||
break out
|
|
||||||
}
|
}
|
||||||
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -356,8 +346,18 @@ func (pool *TxPool) validatePool() {
|
|||||||
if glog.V(logger.Info) {
|
if glog.V(logger.Info) {
|
||||||
glog.Infof("removed tx (%x) from pool: %v\n", hash[:4], err)
|
glog.Infof("removed tx (%x) from pool: %v\n", hash[:4], err)
|
||||||
}
|
}
|
||||||
|
delete(pool.txs, hash)
|
||||||
pool.removeTx(hash)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type txQueue []txQueueEntry
|
||||||
|
|
||||||
|
type txQueueEntry struct {
|
||||||
|
hash common.Hash
|
||||||
|
*types.Transaction
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q txQueue) Len() int { return len(q) }
|
||||||
|
func (q txQueue) Swap(i, j int) { q[i], q[j] = q[j], q[i] }
|
||||||
|
func (q txQueue) Less(i, j int) bool { return q[i].AccountNonce < q[j].AccountNonce }
|
||||||
|
@ -68,7 +68,7 @@ func TestTransactionQueue(t *testing.T) {
|
|||||||
tx.SignECDSA(key)
|
tx.SignECDSA(key)
|
||||||
from, _ := tx.From()
|
from, _ := tx.From()
|
||||||
pool.currentState().AddBalance(from, big.NewInt(1))
|
pool.currentState().AddBalance(from, big.NewInt(1))
|
||||||
pool.queueTx(tx)
|
pool.queueTx(tx.Hash(), tx)
|
||||||
|
|
||||||
pool.checkQueue()
|
pool.checkQueue()
|
||||||
if len(pool.txs) != 1 {
|
if len(pool.txs) != 1 {
|
||||||
@ -80,7 +80,7 @@ func TestTransactionQueue(t *testing.T) {
|
|||||||
from, _ = tx.From()
|
from, _ = tx.From()
|
||||||
pool.currentState().SetNonce(from, 10)
|
pool.currentState().SetNonce(from, 10)
|
||||||
tx.SetNonce(1)
|
tx.SetNonce(1)
|
||||||
pool.queueTx(tx)
|
pool.queueTx(tx.Hash(), tx)
|
||||||
pool.checkQueue()
|
pool.checkQueue()
|
||||||
if _, ok := pool.txs[tx.Hash()]; ok {
|
if _, ok := pool.txs[tx.Hash()]; ok {
|
||||||
t.Error("expected transaction to be in tx pool")
|
t.Error("expected transaction to be in tx pool")
|
||||||
@ -97,18 +97,18 @@ func TestTransactionQueue(t *testing.T) {
|
|||||||
tx1.SignECDSA(key)
|
tx1.SignECDSA(key)
|
||||||
tx2.SignECDSA(key)
|
tx2.SignECDSA(key)
|
||||||
tx3.SignECDSA(key)
|
tx3.SignECDSA(key)
|
||||||
pool.queueTx(tx1)
|
pool.queueTx(tx1.Hash(), tx1)
|
||||||
pool.queueTx(tx2)
|
pool.queueTx(tx2.Hash(), tx2)
|
||||||
pool.queueTx(tx3)
|
pool.queueTx(tx3.Hash(), tx3)
|
||||||
from, _ = tx1.From()
|
from, _ = tx1.From()
|
||||||
|
|
||||||
pool.checkQueue()
|
pool.checkQueue()
|
||||||
|
|
||||||
if len(pool.txs) != 1 {
|
if len(pool.txs) != 1 {
|
||||||
t.Error("expected tx pool to be 1 =")
|
t.Error("expected tx pool to be 1 =")
|
||||||
}
|
}
|
||||||
|
if len(pool.queue[from]) != 2 {
|
||||||
if len(pool.queue[from]) != 3 {
|
t.Error("expected len(queue) == 2, got", len(pool.queue[from]))
|
||||||
t.Error("expected transaction queue to be empty. is", len(pool.queue[from]))
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -118,8 +118,8 @@ func TestRemoveTx(t *testing.T) {
|
|||||||
tx.SignECDSA(key)
|
tx.SignECDSA(key)
|
||||||
from, _ := tx.From()
|
from, _ := tx.From()
|
||||||
pool.currentState().AddBalance(from, big.NewInt(1))
|
pool.currentState().AddBalance(from, big.NewInt(1))
|
||||||
pool.queueTx(tx)
|
pool.queueTx(tx.Hash(), tx)
|
||||||
pool.addTx(tx)
|
pool.addTx(tx.Hash(), tx)
|
||||||
if len(pool.queue) != 1 {
|
if len(pool.queue) != 1 {
|
||||||
t.Error("expected queue to be 1, got", len(pool.queue))
|
t.Error("expected queue to be 1, got", len(pool.queue))
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user