-
Notifications
You must be signed in to change notification settings - Fork 31
/
parallel_tx_dag.go
100 lines (89 loc) · 2.6 KB
/
parallel_tx_dag.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
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
package core
import (
"github.com/AlayaNetwork/Alaya-Go/common"
dag3 "github.com/AlayaNetwork/Alaya-Go/core/dag"
"github.com/AlayaNetwork/Alaya-Go/core/state"
"github.com/AlayaNetwork/Alaya-Go/core/types"
"github.com/AlayaNetwork/Alaya-Go/log"
)
type TxDag struct {
dag *dag3.Dag
signer types.Signer
contracts map[int]struct{}
}
func NewTxDag(signer types.Signer) *TxDag {
txDag := &TxDag{
signer: signer,
contracts: make(map[int]struct{}),
}
return txDag
}
func (txDag *TxDag) MakeDagGraph(blockNumber uint64, state *state.StateDB, txs []*types.Transaction, exe *Executor) error {
txDag.dag = dag3.NewDag(len(txs))
//save all transfer addresses between two contracts(precompiled and user defined)
transferAddressMap := make(map[common.Address]int, 0)
latestPrecompiledIndex := -1
for index, tx := range txs {
if tx.FromAddr(txDag.signer) == (common.Address{}) {
log.Error("The from of the transaction cannot be resolved", "number", blockNumber, "index", index)
continue
}
if exe.isContract(tx.To(), state) {
txDag.contracts[index] = struct{}{}
if index > 0 {
if index-latestPrecompiledIndex > 1 {
for begin := latestPrecompiledIndex + 1; begin < index; begin++ {
txDag.dag.AddEdge(begin, index)
}
} else if index-latestPrecompiledIndex == 1 {
txDag.dag.AddEdge(latestPrecompiledIndex, index)
}
}
latestPrecompiledIndex = index
//reset transferAddressMap
if len(transferAddressMap) > 0 {
transferAddressMap = make(map[common.Address]int, 0)
}
} else {
dependFound := 0
if dependIdx, ok := transferAddressMap[tx.FromAddr(txDag.signer)]; ok {
txDag.dag.AddEdge(dependIdx, index)
dependFound++
}
if dependIdx, ok := transferAddressMap[*tx.To()]; ok {
txDag.dag.AddEdge(dependIdx, index)
dependFound++
}
if dependFound == 0 && latestPrecompiledIndex >= 0 {
txDag.dag.AddEdge(latestPrecompiledIndex, index)
}
transferAddressMap[tx.FromAddr(txDag.signer)] = index
transferAddressMap[*tx.To()] = index
}
}
/*
// dag print info
logVerbosity := debug.GetLogVerbosity()
if logVerbosity == log.LvlTrace {
buff, err := txDag.dag.Print()
if err != nil {
log.Error("print DAG Graph error!", "blockNumber", blockNumber, "err", err)
return nil
}
log.Trace("DAG Graph", "blockNumber", blockNumber, "info", buff.String())
}
*/
return nil
}
func (txDag *TxDag) HasNext() bool {
return txDag.dag.HasNext()
}
func (txDag *TxDag) Next() []int {
return txDag.dag.Next()
}
func (txDag *TxDag) IsContract(idx int) bool {
if _, ok := txDag.contracts[idx]; ok {
return true
}
return false
}