Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
G
Geth-Modification
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
张蕾
Geth-Modification
Commits
8d9be18b
Commit
8d9be18b
authored
Mar 06, 2015
by
obscuren
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Queued approach to delivering chain events
parent
b72ca577
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
86 additions
and
36 deletions
+86
-36
chain_manager.go
core/chain_manager.go
+73
-34
events.go
core/events.go
+10
-0
miner.go
miner/miner.go
+1
-0
worker.go
miner/worker.go
+2
-2
No files found.
core/chain_manager.go
View file @
8d9be18b
...
...
@@ -19,11 +19,6 @@ var (
jsonlogger
=
logger
.
NewJsonLogger
()
)
type
ChainEvent
struct
{
Block
*
types
.
Block
Td
*
big
.
Int
}
type
StateQuery
interface
{
GetAccount
(
addr
[]
byte
)
*
state
.
StateObject
}
...
...
@@ -93,13 +88,16 @@ type ChainManager struct {
transState
*
state
.
StateDB
txState
*
state
.
StateDB
quit
chan
struct
{}
}
func
NewChainManager
(
db
ethutil
.
Database
,
mux
*
event
.
TypeMux
)
*
ChainManager
{
bc
:=
&
ChainManager
{
db
:
db
,
genesisBlock
:
GenesisBlock
(
db
),
eventMux
:
mux
}
bc
:=
&
ChainManager
{
db
:
db
,
genesisBlock
:
GenesisBlock
(
db
),
eventMux
:
mux
,
quit
:
make
(
chan
struct
{})
}
bc
.
setLastBlock
()
bc
.
transState
=
bc
.
State
()
.
Copy
()
bc
.
txState
=
bc
.
State
()
.
Copy
()
go
bc
.
update
()
return
bc
}
...
...
@@ -388,16 +386,24 @@ func (self *ChainManager) CalcTotalDiff(block *types.Block) (*big.Int, error) {
}
func
(
bc
*
ChainManager
)
Stop
()
{
if
bc
.
CurrentBlock
!=
nil
{
chainlogger
.
Infoln
(
"Stopped"
)
}
close
(
bc
.
quit
)
}
type
queueEvent
struct
{
queue
[]
interface
{}
canonicalCount
int
sideCount
int
splitCount
int
}
func
(
self
*
ChainManager
)
InsertChain
(
chain
types
.
Blocks
)
error
{
self
.
tsmu
.
Lock
()
defer
self
.
tsmu
.
Unlock
()
//
self.tsmu.Lock()
//
defer self.tsmu.Unlock()
for
_
,
block
:=
range
chain
{
// A queued approach to delivering events. This is generally faster than direct delivery and requires much less mutex acquiring.
var
queue
=
make
([]
interface
{},
len
(
chain
))
var
queueEvent
=
queueEvent
{
queue
:
queue
}
for
i
,
block
:=
range
chain
{
// Call in to the block processor and check for errors. It's likely that if one block fails
// all others will fail too (unless a known block is returned).
td
,
err
:=
self
.
processor
.
Process
(
block
)
...
...
@@ -414,7 +420,6 @@ func (self *ChainManager) InsertChain(chain types.Blocks) error {
}
block
.
Td
=
td
var
canonical
,
split
bool
self
.
mu
.
Lock
()
cblock
:=
self
.
currentBlock
{
...
...
@@ -426,41 +431,75 @@ func (self *ChainManager) InsertChain(chain types.Blocks) error {
if
td
.
Cmp
(
self
.
td
)
>
0
{
if
block
.
Header
()
.
Number
.
Cmp
(
new
(
big
.
Int
)
.
Add
(
cblock
.
Header
()
.
Number
,
ethutil
.
Big1
))
<
0
{
chainlogger
.
Infof
(
"Split detected. New head #%v (%x) TD=%v, was #%v (%x) TD=%v
\n
"
,
block
.
Header
()
.
Number
,
block
.
Hash
()[
:
4
],
td
,
cblock
.
Header
()
.
Number
,
cblock
.
Hash
()[
:
4
],
self
.
td
)
split
=
true
queue
[
i
]
=
ChainSplitEvent
{
block
}
queueEvent
.
splitCount
++
}
self
.
setTotalDifficulty
(
td
)
self
.
insert
(
block
)
canonical
=
true
/*
jsonlogger.LogJson(&logger.EthChainNewHead{
BlockHash: ethutil.Bytes2Hex(block.Hash()),
BlockNumber: block.Number(),
ChainHeadHash: ethutil.Bytes2Hex(cblock.Hash()),
BlockPrevHash: ethutil.Bytes2Hex(block.ParentHash()),
})
*/
self
.
setTransState
(
state
.
New
(
block
.
Root
(),
self
.
db
))
queue
[
i
]
=
ChainEvent
{
block
}
queueEvent
.
canonicalCount
++
}
else
{
queue
[
i
]
=
ChainSideEvent
{
block
}
queueEvent
.
sideCount
++
}
}
self
.
mu
.
Unlock
()
if
canonical
{
/*
jsonlogger.LogJson(&logger.EthChainNewHead{
BlockHash: ethutil.Bytes2Hex(block.Hash()),
BlockNumber: block.Number(),
ChainHeadHash: ethutil.Bytes2Hex(cblock.Hash()),
BlockPrevHash: ethutil.Bytes2Hex(block.ParentHash()),
})
*/
self
.
setTransState
(
state
.
New
(
block
.
Root
(),
self
.
db
))
self
.
eventMux
.
Post
(
ChainEvent
{
block
,
td
})
}
else
{
//self.eventMux.
}
if
split
{
self
.
setTxState
(
state
.
New
(
block
.
Root
(),
self
.
db
))
self
.
eventMux
.
Post
(
ChainSplitEvent
{
block
})
}
}
// XXX put this in a goroutine?
go
self
.
eventMux
.
Post
(
queueEvent
)
return
nil
}
func
(
self
*
ChainManager
)
update
()
{
events
:=
self
.
eventMux
.
Subscribe
(
queueEvent
{})
out
:
for
{
select
{
case
ev
:=
<-
events
.
Chan
()
:
switch
ev
:=
ev
.
(
type
)
{
case
queueEvent
:
for
i
,
event
:=
range
ev
.
queue
{
switch
event
:=
event
.
(
type
)
{
case
ChainEvent
:
// We need some control over the mining operation. Acquiring locks and waiting for the miner to create new block takes too long
// and in most cases isn't even necessary.
if
i
==
ev
.
canonicalCount
{
self
.
eventMux
.
Post
(
ChainHeadEvent
{
event
.
Block
})
}
case
ChainSplitEvent
:
// On chain splits we need to reset the transaction state. We can't be sure whether the actual
// state of the accounts are still valid.
if
i
==
ev
.
splitCount
{
self
.
setTxState
(
state
.
New
(
event
.
Block
.
Root
(),
self
.
db
))
}
}
self
.
eventMux
.
Post
(
event
)
}
}
case
<-
self
.
quit
:
break
out
}
}
}
// Satisfy state query interface
func
(
self
*
ChainManager
)
GetAccount
(
addr
[]
byte
)
*
state
.
StateObject
{
return
self
.
State
()
.
GetAccount
(
addr
)
...
...
core/events.go
View file @
8d9be18b
...
...
@@ -16,3 +16,13 @@ type NewMinedBlockEvent struct{ Block *types.Block }
// ChainSplit is posted when a new head is detected
type
ChainSplitEvent
struct
{
Block
*
types
.
Block
}
type
ChainEvent
struct
{
Block
*
types
.
Block
}
type
ChainSideEvent
struct
{
Block
*
types
.
Block
}
type
ChainHeadEvent
struct
{
Block
*
types
.
Block
}
// Mining operation events
type
StartMining
struct
{}
type
TopMining
struct
{}
miner/miner.go
View file @
8d9be18b
...
...
@@ -30,6 +30,7 @@ func New(coinbase []byte, eth core.Backend, pow pow.PoW, minerThreads int) *Mine
pow
:
pow
,
}
minerThreads
=
1
for
i
:=
0
;
i
<
minerThreads
;
i
++
{
miner
.
worker
.
register
(
NewCpuMiner
(
i
,
miner
.
pow
))
}
...
...
miner/worker.go
View file @
8d9be18b
...
...
@@ -116,7 +116,7 @@ func (self *worker) register(agent Agent) {
}
func
(
self
*
worker
)
update
()
{
events
:=
self
.
mux
.
Subscribe
(
core
.
ChainEvent
{},
core
.
NewMinedBlockEvent
{})
events
:=
self
.
mux
.
Subscribe
(
core
.
Chain
Head
Event
{},
core
.
NewMinedBlockEvent
{})
timer
:=
time
.
NewTicker
(
2
*
time
.
Second
)
...
...
@@ -125,7 +125,7 @@ out:
select
{
case
event
:=
<-
events
.
Chan
()
:
switch
ev
:=
event
.
(
type
)
{
case
core
.
ChainEvent
:
case
core
.
Chain
Head
Event
:
if
self
.
current
.
block
!=
ev
.
Block
{
self
.
commitNewWork
()
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment