Skip to content

Commit e1f11ed

Browse files
Ruteriavalonche
authored andcommitted
Flashbots changes v0.4 to v0.5
* fix issue with geth not shutting down (flashbots#97) * Add eth_callBundle rpc method (flashbots#14) * flashbots: add eth_estimateGasBundle (flashbots#102) * feat(ethash): flashbots_getWork RPC with profit (flashbots#106) * Calculate megabundle as soon as it's received (flashbots#112) * Add v0.5 specification link (flashbots#118)
1 parent 15bd950 commit e1f11ed

22 files changed

+518
-49
lines changed

cmd/evm/internal/t8ntool/block.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -191,7 +191,7 @@ func (i *bbInput) sealEthash(block *types.Block) (*types.Block, error) {
191191
// If the testmode is used, the sealer will return quickly, and complain
192192
// "Sealing result is not read by miner" if it cannot write the result.
193193
results := make(chan *types.Block, 1)
194-
if err := engine.Seal(nil, block, results, nil); err != nil {
194+
if err := engine.Seal(nil, block, nil, results, nil); err != nil {
195195
panic(fmt.Sprintf("failed to seal block: %v", err))
196196
}
197197
found := <-results

consensus/beacon/consensus.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -374,9 +374,9 @@ func (beacon *Beacon) FinalizeAndAssemble(chain consensus.ChainHeaderReader, hea
374374
//
375375
// Note, the method returns immediately and will send the result async. More
376376
// than one result may also be returned depending on the consensus algorithm.
377-
func (beacon *Beacon) Seal(chain consensus.ChainHeaderReader, block *types.Block, results chan<- *types.Block, stop <-chan struct{}) error {
377+
func (beacon *Beacon) Seal(chain consensus.ChainHeaderReader, block *types.Block, profit *big.Int, results chan<- *types.Block, stop <-chan struct{}) error {
378378
if !beacon.IsPoSHeader(block.Header()) {
379-
return beacon.ethone.Seal(chain, block, results, stop)
379+
return beacon.ethone.Seal(chain, block, profit, results, stop)
380380
}
381381
// The seal verification is done by the external consensus engine,
382382
// return directly without pushing any block back. In another word

consensus/clique/clique.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -599,7 +599,7 @@ func (c *Clique) Authorize(signer common.Address, signFn SignerFn) {
599599

600600
// Seal implements consensus.Engine, attempting to create a sealed block using
601601
// the local signing credentials.
602-
func (c *Clique) Seal(chain consensus.ChainHeaderReader, block *types.Block, results chan<- *types.Block, stop <-chan struct{}) error {
602+
func (c *Clique) Seal(chain consensus.ChainHeaderReader, block *types.Block, profit *big.Int, results chan<- *types.Block, stop <-chan struct{}) error {
603603
header := block.Header()
604604

605605
// Sealing the genesis block is not supported

consensus/consensus.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@ type Engine interface {
105105
//
106106
// Note, the method returns immediately and will send the result async. More
107107
// than one result may also be returned depending on the consensus algorithm.
108-
Seal(chain ChainHeaderReader, block *types.Block, results chan<- *types.Block, stop <-chan struct{}) error
108+
Seal(chain ChainHeaderReader, block *types.Block, profit *big.Int, results chan<- *types.Block, stop <-chan struct{}) error
109109

110110
// SealHash returns the hash of a block prior to it being sealed.
111111
SealHash(header *types.Header) common.Hash

consensus/ethash/api.go

+5-2
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ func (api *API) GetWork() ([4]string, error) {
4545
}
4646

4747
var (
48-
workCh = make(chan [4]string, 1)
48+
workCh = make(chan [5]string, 1)
4949
errc = make(chan error, 1)
5050
)
5151
select {
@@ -54,7 +54,10 @@ func (api *API) GetWork() ([4]string, error) {
5454
return [4]string{}, errEthashStopped
5555
}
5656
select {
57-
case work := <-workCh:
57+
case fullWork := <-workCh:
58+
var work [4]string
59+
copy(work[:], fullWork[:4])
60+
5861
return work, nil
5962
case err := <-errc:
6063
return [4]string{}, err

consensus/ethash/ethash.go

+6
Original file line numberDiff line numberDiff line change
@@ -690,6 +690,12 @@ func (ethash *Ethash) APIs(chain consensus.ChainHeaderReader) []rpc.API {
690690
Namespace: "ethash",
691691
Service: &API{ethash},
692692
},
693+
{
694+
Namespace: "flashbots",
695+
Version: "1.0",
696+
Service: &FlashbotsAPI{ethash},
697+
Public: true,
698+
},
693699
}
694700
}
695701

consensus/ethash/ethash_test.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ func TestTestMode(t *testing.T) {
3737
defer ethash.Close()
3838

3939
results := make(chan *types.Block)
40-
err := ethash.Seal(nil, types.NewBlockWithHeader(header), results, nil)
40+
err := ethash.Seal(nil, types.NewBlockWithHeader(header), nil, results, nil)
4141
if err != nil {
4242
t.Fatalf("failed to seal block: %v", err)
4343
}
@@ -112,7 +112,7 @@ func TestRemoteSealer(t *testing.T) {
112112

113113
// Push new work.
114114
results := make(chan *types.Block)
115-
ethash.Seal(nil, block, results, nil)
115+
ethash.Seal(nil, block, nil, results, nil)
116116

117117
var (
118118
work [4]string
@@ -129,7 +129,7 @@ func TestRemoteSealer(t *testing.T) {
129129
header = &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(1000)}
130130
block = types.NewBlockWithHeader(header)
131131
sealhash = ethash.SealHash(header)
132-
ethash.Seal(nil, block, results, nil)
132+
ethash.Seal(nil, block, nil, results, nil)
133133

134134
if work, err = api.GetWork(); err != nil || work[0] != sealhash.Hex() {
135135
t.Error("expect to return the latest pushed work")

consensus/ethash/flashbots_api.go

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
package ethash
2+
3+
import "errors"
4+
5+
// FlashbotsAPI exposes Flashbots related methods for the RPC interface.
6+
type FlashbotsAPI struct {
7+
ethash *Ethash
8+
}
9+
10+
// GetWork returns a work package for external miner.
11+
//
12+
// The work package consists of 5 strings:
13+
// result[0] - 32 bytes hex encoded current block header pow-hash
14+
// result[1] - 32 bytes hex encoded seed hash used for DAG
15+
// result[2] - 32 bytes hex encoded boundary condition ("target"), 2^256/difficulty
16+
// result[3] - hex encoded block number
17+
// result[4] - hex encoded profit generated from this block
18+
func (api *FlashbotsAPI) GetWork() ([5]string, error) {
19+
if api.ethash.remote == nil {
20+
return [5]string{}, errors.New("not supported")
21+
}
22+
23+
var (
24+
workCh = make(chan [5]string, 1)
25+
errc = make(chan error, 1)
26+
)
27+
select {
28+
case api.ethash.remote.fetchWorkCh <- &sealWork{errc: errc, res: workCh}:
29+
case <-api.ethash.remote.exitCh:
30+
return [5]string{}, errEthashStopped
31+
}
32+
select {
33+
case work := <-workCh:
34+
return work, nil
35+
case err := <-errc:
36+
return [5]string{}, err
37+
}
38+
}

consensus/ethash/sealer.go

+13-8
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,7 @@ var (
4848

4949
// Seal implements consensus.Engine, attempting to find a nonce that satisfies
5050
// the block's difficulty requirements.
51-
func (ethash *Ethash) Seal(chain consensus.ChainHeaderReader, block *types.Block, results chan<- *types.Block, stop <-chan struct{}) error {
51+
func (ethash *Ethash) Seal(chain consensus.ChainHeaderReader, block *types.Block, profit *big.Int, results chan<- *types.Block, stop <-chan struct{}) error {
5252
// If we're running a fake PoW, simply return a 0 nonce immediately
5353
if ethash.config.PowMode == ModeFake || ethash.config.PowMode == ModeFullFake {
5454
header := block.Header()
@@ -62,7 +62,7 @@ func (ethash *Ethash) Seal(chain consensus.ChainHeaderReader, block *types.Block
6262
}
6363
// If we're running a shared PoW, delegate sealing to it
6464
if ethash.shared != nil {
65-
return ethash.shared.Seal(chain, block, results, stop)
65+
return ethash.shared.Seal(chain, block, profit, results, stop)
6666
}
6767
// Create a runner and the multiple search threads it directs
6868
abort := make(chan struct{})
@@ -86,7 +86,7 @@ func (ethash *Ethash) Seal(chain consensus.ChainHeaderReader, block *types.Block
8686
}
8787
// Push new work to remote sealer
8888
if ethash.remote != nil {
89-
ethash.remote.workCh <- &sealTask{block: block, results: results}
89+
ethash.remote.workCh <- &sealTask{block: block, profit: profit, results: results}
9090
}
9191
var (
9292
pend sync.WaitGroup
@@ -117,7 +117,7 @@ func (ethash *Ethash) Seal(chain consensus.ChainHeaderReader, block *types.Block
117117
case <-ethash.update:
118118
// Thread count was changed on user request, restart
119119
close(abort)
120-
if err := ethash.Seal(chain, block, results, stop); err != nil {
120+
if err := ethash.Seal(chain, block, profit, results, stop); err != nil {
121121
ethash.config.Log.Error("Failed to restart sealing after update", "err", err)
122122
}
123123
}
@@ -194,7 +194,7 @@ type remoteSealer struct {
194194
works map[common.Hash]*types.Block
195195
rates map[common.Hash]hashrate
196196
currentBlock *types.Block
197-
currentWork [4]string
197+
currentWork [5]string
198198
notifyCtx context.Context
199199
cancelNotify context.CancelFunc // cancels all notification requests
200200
reqWG sync.WaitGroup // tracks notification request goroutines
@@ -215,6 +215,7 @@ type remoteSealer struct {
215215
// sealTask wraps a seal block with relative result channel for remote sealer thread.
216216
type sealTask struct {
217217
block *types.Block
218+
profit *big.Int
218219
results chan<- *types.Block
219220
}
220221

@@ -239,7 +240,7 @@ type hashrate struct {
239240
// sealWork wraps a seal work package for remote sealer.
240241
type sealWork struct {
241242
errc chan error
242-
res chan [4]string
243+
res chan [5]string
243244
}
244245

245246
func startRemoteSealer(ethash *Ethash, urls []string, noverify bool) *remoteSealer {
@@ -281,7 +282,7 @@ func (s *remoteSealer) loop() {
281282
// Update current work with new received block.
282283
// Note same work can be past twice, happens when changing CPU threads.
283284
s.results = work.results
284-
s.makeWork(work.block)
285+
s.makeWork(work.block, work.profit)
285286
s.notifyWork()
286287

287288
case work := <-s.fetchWorkCh:
@@ -351,6 +352,10 @@ func (s *remoteSealer) makeWork(block *types.Block) {
351352
s.currentWork[2] = common.BytesToHash(new(big.Int).Div(two256, block.Difficulty()).Bytes()).Hex()
352353
s.currentWork[3] = hexutil.EncodeBig(block.Number())
353354

355+
if profit != nil {
356+
s.currentWork[4] = hexutil.EncodeBig(profit)
357+
}
358+
354359
// Trace the seal work fetched by remote sealer.
355360
s.currentBlock = block
356361
s.works[hash] = block
@@ -376,7 +381,7 @@ func (s *remoteSealer) notifyWork() {
376381
}
377382
}
378383

379-
func (s *remoteSealer) sendNotification(ctx context.Context, url string, json []byte, work [4]string) {
384+
func (s *remoteSealer) sendNotification(ctx context.Context, url string, json []byte, work [5]string) {
380385
defer s.reqWG.Done()
381386

382387
req, err := http.NewRequest(http.MethodPost, url, bytes.NewReader(json))

consensus/ethash/sealer_test.go

+5-5
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ func TestRemoteNotify(t *testing.T) {
5757
header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)}
5858
block := types.NewBlockWithHeader(header)
5959

60-
ethash.Seal(nil, block, nil, nil)
60+
ethash.Seal(nil, block, nil, nil, nil)
6161
select {
6262
case work := <-sink:
6363
if want := ethash.SealHash(header).Hex(); work[0] != want {
@@ -105,7 +105,7 @@ func TestRemoteNotifyFull(t *testing.T) {
105105
header := &types.Header{Number: big.NewInt(1), Difficulty: big.NewInt(100)}
106106
block := types.NewBlockWithHeader(header)
107107

108-
ethash.Seal(nil, block, nil, nil)
108+
ethash.Seal(nil, block, nil, nil, nil)
109109
select {
110110
case work := <-sink:
111111
if want := "0x" + strconv.FormatUint(header.Number.Uint64(), 16); work["number"] != want {
@@ -151,7 +151,7 @@ func TestRemoteMultiNotify(t *testing.T) {
151151
for i := 0; i < cap(sink); i++ {
152152
header := &types.Header{Number: big.NewInt(int64(i)), Difficulty: big.NewInt(100)}
153153
block := types.NewBlockWithHeader(header)
154-
ethash.Seal(nil, block, results, nil)
154+
ethash.Seal(nil, block, nil, results, nil)
155155
}
156156

157157
for i := 0; i < cap(sink); i++ {
@@ -200,7 +200,7 @@ func TestRemoteMultiNotifyFull(t *testing.T) {
200200
for i := 0; i < cap(sink); i++ {
201201
header := &types.Header{Number: big.NewInt(int64(i)), Difficulty: big.NewInt(100)}
202202
block := types.NewBlockWithHeader(header)
203-
ethash.Seal(nil, block, results, nil)
203+
ethash.Seal(nil, block, nil, results, nil)
204204
}
205205

206206
for i := 0; i < cap(sink); i++ {
@@ -266,7 +266,7 @@ func TestStaleSubmission(t *testing.T) {
266266

267267
for id, c := range testcases {
268268
for _, h := range c.headers {
269-
ethash.Seal(nil, types.NewBlockWithHeader(h), results, nil)
269+
ethash.Seal(nil, types.NewBlockWithHeader(h), nil, results, nil)
270270
}
271271
if res := api.SubmitWork(fakeNonce, ethash.SealHash(c.headers[c.submitIndex]), fakeDigest); res != c.submitRes {
272272
t.Errorf("case %d submit result mismatch, want %t, get %t", id+1, c.submitRes, res)

core/state_processor.go

+56
Original file line numberDiff line numberDiff line change
@@ -142,6 +142,51 @@ func applyTransaction(msg *Message, config *params.ChainConfig, gp *GasPool, sta
142142
return receipt, err
143143
}
144144

145+
func applyTransactionWithResult(msg types.Message, config *params.ChainConfig, bc ChainContext, author *common.Address, gp *GasPool, statedb *state.StateDB, header *types.Header, tx *types.Transaction, usedGas *uint64, evm *vm.EVM) (*types.Receipt, *ExecutionResult, error) {
146+
// Create a new context to be used in the EVM environment.
147+
txContext := NewEVMTxContext(msg)
148+
evm.Reset(txContext, statedb)
149+
150+
// Apply the transaction to the current state (included in the env).
151+
result, err := ApplyMessage(evm, msg, gp)
152+
if err != nil {
153+
return nil, nil, err
154+
}
155+
156+
// Update the state with pending changes.
157+
var root []byte
158+
if config.IsByzantium(header.Number) {
159+
statedb.Finalise(true)
160+
} else {
161+
root = statedb.IntermediateRoot(config.IsEIP158(header.Number)).Bytes()
162+
}
163+
*usedGas += result.UsedGas
164+
165+
// Create a new receipt for the transaction, storing the intermediate root and gas used
166+
// by the tx.
167+
receipt := &types.Receipt{Type: tx.Type(), PostState: root, CumulativeGasUsed: *usedGas}
168+
if result.Failed() {
169+
receipt.Status = types.ReceiptStatusFailed
170+
} else {
171+
receipt.Status = types.ReceiptStatusSuccessful
172+
}
173+
receipt.TxHash = tx.Hash()
174+
receipt.GasUsed = result.UsedGas
175+
176+
// If the transaction created a contract, store the creation address in the receipt.
177+
if msg.To() == nil {
178+
receipt.ContractAddress = crypto.CreateAddress(evm.TxContext.Origin, tx.Nonce())
179+
}
180+
181+
// Set the receipt logs and create the bloom filter.
182+
receipt.Logs = statedb.GetLogs(tx.Hash(), header.Hash())
183+
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
184+
receipt.BlockHash = header.Hash()
185+
receipt.BlockNumber = header.Number
186+
receipt.TransactionIndex = uint(statedb.TxIndex())
187+
return receipt, result, err
188+
}
189+
145190
// ApplyTransaction attempts to apply a transaction to the given state database
146191
// and uses the input parameters for its environment. It returns the receipt
147192
// for the transaction, gas used and an error if the transaction failed,
@@ -156,3 +201,14 @@ func ApplyTransaction(config *params.ChainConfig, bc ChainContext, author *commo
156201
vmenv := vm.NewEVM(blockContext, vm.TxContext{}, statedb, config, cfg)
157202
return applyTransaction(msg, config, gp, statedb, header.Number, header.Hash(), tx, usedGas, vmenv)
158203
}
204+
205+
func ApplyTransactionWithResult(config *params.ChainConfig, bc ChainContext, author *common.Address, gp *GasPool, statedb *state.StateDB, header *types.Header, tx *types.Transaction, usedGas *uint64, cfg vm.Config) (*types.Receipt, *ExecutionResult, error) {
206+
msg, err := tx.AsMessage(types.MakeSigner(config, header.Number), header.BaseFee)
207+
if err != nil {
208+
return nil, nil, err
209+
}
210+
// Create a new context to be used in the EVM environment
211+
blockContext := NewEVMBlockContext(header, bc, author)
212+
vmenv := vm.NewEVM(blockContext, vm.TxContext{}, statedb, config, cfg)
213+
return applyTransactionWithResult(msg, config, bc, author, gp, statedb, header, tx, usedGas, vmenv)
214+
}

core/txpool/txpool.go

+8-1
Original file line numberDiff line numberDiff line change
@@ -638,13 +638,20 @@ func (pool *TxPool) AddMegabundle(relayAddr common.Address, txs types.Transactio
638638
return errors.New("megabundle from non-trusted address")
639639
}
640640

641-
pool.megabundles[relayAddr] = types.MevBundle{
641+
megabundle := types.MevBundle{
642642
Txs: txs,
643643
BlockNumber: blockNumber,
644644
MinTimestamp: minTimestamp,
645645
MaxTimestamp: maxTimestamp,
646646
RevertingTxHashes: revertingTxHashes,
647647
}
648+
649+
pool.megabundles[relayAddr] = megabundle
650+
651+
for _, hook := range pool.NewMegabundleHooks {
652+
go hook(relayAddr, &megabundle)
653+
}
654+
648655
return nil
649656
}
650657

eth/backend.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -288,7 +288,7 @@ func makeExtraData(extra []byte) []byte {
288288
// APIs return the collection of RPC services the ethereum package offers.
289289
// NOTE, some of these services probably need to be moved to somewhere else.
290290
func (s *Ethereum) APIs() []rpc.API {
291-
apis := ethapi.GetAPIs(s.APIBackend)
291+
apis := ethapi.GetAPIs(s.APIBackend, s.BlockChain())
292292

293293
// Append any APIs exposed explicitly by the consensus engine
294294
apis = append(apis, s.engine.APIs(s.BlockChain())...)

infra/Dockerfile.node

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ FROM golang:1.15-alpine as builder
44
RUN apk add --no-cache make gcc musl-dev linux-headers git
55

66
ADD . /go-ethereum
7-
RUN cd /go-ethereum && make geth
7+
RUN cd /go-ethereum && GO111MODULE=on go run build/ci.go install ./cmd/geth
88

99
# Pull Geth into a second stage deploy alpine container
1010
FROM alpine:latest

infra/Dockerfile.updater

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ FROM golang:1.15-alpine as builder
44
RUN apk add --no-cache make gcc musl-dev linux-headers git
55

66
ADD . /go-ethereum
7-
RUN cd /go-ethereum && make geth
7+
RUN cd /go-ethereum && GO111MODULE=on go run build/ci.go install ./cmd/geth
88

99
# Pull Geth into a second stage deploy alpine container
1010
FROM alpine:latest

0 commit comments

Comments
 (0)