Skip to content

Commit

Permalink
feat: setup FEVM during genesis: create ETH0 actor
Browse files Browse the repository at this point in the history
  • Loading branch information
simlecode committed Dec 16, 2022
1 parent 0e96e94 commit 7b84fce
Show file tree
Hide file tree
Showing 2 changed files with 162 additions and 0 deletions.
124 changes: 124 additions & 0 deletions pkg/gen/genesis/fevm.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,124 @@
package genesis

import (
"context"
"encoding/hex"
"fmt"

"github.com/ipfs/go-cid"

"github.com/filecoin-project/go-state-types/abi"
actorstypes "github.com/filecoin-project/go-state-types/actors"
"github.com/filecoin-project/go-state-types/big"
builtintypes "github.com/filecoin-project/go-state-types/builtin"
evm10 "github.com/filecoin-project/go-state-types/builtin/v10/evm"
init10 "github.com/filecoin-project/go-state-types/builtin/v10/init"
"github.com/filecoin-project/go-state-types/network"
"github.com/filecoin-project/venus/pkg/chain"
"github.com/filecoin-project/venus/pkg/config"
"github.com/filecoin-project/venus/pkg/consensusfault"
"github.com/filecoin-project/venus/pkg/fork"
"github.com/filecoin-project/venus/pkg/fvm"
"github.com/filecoin-project/venus/pkg/state/tree"
"github.com/filecoin-project/venus/pkg/util/ffiwrapper/impl"
"github.com/filecoin-project/venus/pkg/vm"
"github.com/filecoin-project/venus/pkg/vm/gas"
"github.com/filecoin-project/venus/pkg/vm/vmcontext"
"github.com/filecoin-project/venus/pkg/vmsupport"
"github.com/filecoin-project/venus/venus-shared/actors"
)

func SetupFEVM(ctx context.Context, cs *chain.Store, sroot cid.Cid, nv network.Version, para *config.ForkUpgradeConfig) (cid.Cid, error) {
av, err := actorstypes.VersionForNetwork(nv)
if err != nil {
return cid.Undef, fmt.Errorf("failed to get actors version for network version %d: %w", nv, err)
}

if av < actorstypes.Version10 {
// Not defined before version 10; migration has to setup.
return sroot, nil
}

csc := func(context.Context, abi.ChainEpoch, tree.Tree) (abi.TokenAmount, error) {
return big.Zero(), nil
}
faultChecker := consensusfault.NewFaultChecker(cs, fork.NewMockFork())
syscalls := vmsupport.NewSyscalls(faultChecker, impl.ProofVerifier)
gasPirceSchedule := gas.NewPricesSchedule(para)

newVM := func(base cid.Cid) (vmcontext.Interface, error) {
vmopt := vmcontext.VmOption{
PRoot: base,
Epoch: 0,
Rnd: &fakeRand{},
Bsstore: cs.Blockstore(),
ActorCodeLoader: vm.GetDefaultActors(),
SysCallsImpl: mkFakedSigSyscalls(syscalls),
CircSupplyCalculator: csc,
NetworkVersion: nv,
BaseFee: big.Zero(),
GasPriceSchedule: gasPirceSchedule,
}

return fvm.NewVM(ctx, vmopt)
}

genesisVm, err := newVM(sroot)
if err != nil {
return cid.Undef, fmt.Errorf("creating vm: %w", err)
}

// The ETH0 address is occupied by an empty contract EVM actor
evmCodeCid, ok := actors.GetActorCodeID(av, actors.EvmKey)
if !ok {
return cid.Undef, fmt.Errorf("failed to get CodeCID for EVM during genesis")
}

// initcode:
// %push(code_end - code_begin)
// dup1
// %push(code_begin)
// push1 0x00
// codecopy
// push1 0x00
// return
// code_begin:
// push1 0x00
// push1 0x00
// return
// code_end:
initcode, err := hex.DecodeString("600580600b6000396000f360006000f3")
if err != nil {
return cid.Undef, fmt.Errorf("failed to parse ETH0 init code during genesis: %w", err)
}

ctorParams := &evm10.ConstructorParams{
Creator: make([]byte, 20), // self!
// TODO we have a bunch of bugs in the evm constructor around empty contracts
// - empty init code is not allowed
// - returning an empty contract is not allowed
// So this uses code that constructs a just return contract until that can be fixed
// and we can pass an empty byte array
Initcode: initcode,
}

params := &init10.Exec4Params{
CodeCID: evmCodeCid,
ConstructorParams: mustEnc(ctorParams),
SubAddress: make([]byte, 20),
}

// TODO method 3 is Exec4; we have to name the methods in go-state-types and avoid using the number
// directly.
if _, err := doExecValue(ctx, genesisVm, builtintypes.InitActorAddr, builtintypes.EthereumAddressManagerActorAddr, big.Zero(), 3, mustEnc(params)); err != nil {
return cid.Undef, fmt.Errorf("creating ETH0 actor: %w", err)
}

newroot, err := genesisVm.Flush(ctx)
if err != nil {
return cid.Undef, fmt.Errorf("flushing vm: %w", err)
}

return newroot, nil

}
38 changes: 38 additions & 0 deletions pkg/gen/genesis/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (

"github.com/filecoin-project/venus/pkg/config"
"github.com/filecoin-project/venus/pkg/vm/gas"
"github.com/filecoin-project/venus/pkg/vm/vmcontext"

"github.com/filecoin-project/venus/pkg/constants"
"github.com/filecoin-project/venus/pkg/state/tree"
Expand All @@ -26,6 +27,7 @@ import (
"github.com/filecoin-project/venus/venus-shared/actors/builtin/datacap"

actorstypes "github.com/filecoin-project/go-state-types/actors"
builtintypes "github.com/filecoin-project/go-state-types/builtin"
"github.com/filecoin-project/go-state-types/network"

"github.com/filecoin-project/venus/venus-shared/actors"
Expand Down Expand Up @@ -583,6 +585,11 @@ func MakeGenesisBlock(ctx context.Context, rep repo.Repo, bs bstore.Blockstore,
return nil, fmt.Errorf("make initial state tree failed: %w", err)
}

// Set up the Ethereum Address Manager
if err = SetupEAM(ctx, st, template.NetworkVersion); err != nil {
return nil, fmt.Errorf("failed to setup EAM: %w", err)
}

stateroot, err := st.Flush(ctx)
if err != nil {
return nil, fmt.Errorf("flush state tree failed: %w", err)
Expand All @@ -602,6 +609,12 @@ func MakeGenesisBlock(ctx context.Context, rep repo.Repo, bs bstore.Blockstore,
return nil, fmt.Errorf("setup miners failed: %w", err)
}

// setup FEVM
stateroot, err = SetupFEVM(ctx, cs, stateroot, template.NetworkVersion, para)
if err != nil {
return nil, fmt.Errorf("failed to setup FEVM functionality: %w", err)
}

store := adt.WrapStore(ctx, cbor.NewCborStore(bs))
emptyroot, err := adt0.MakeEmptyArray(store).Root()
if err != nil {
Expand Down Expand Up @@ -685,3 +698,28 @@ func MakeGenesisBlock(ctx context.Context, rep repo.Repo, bs bstore.Blockstore,
Genesis: b,
}, nil
}

func SetupEAM(ctx context.Context, nst tree.Tree, nv network.Version) error {
av, err := actorstypes.VersionForNetwork(nv)
if err != nil {
return fmt.Errorf("failed to get actors version for network version %d: %w", nv, err)
}

if av < actorstypes.Version10 {
// Not defined before version 10; migration has to create.
return nil
}

codecid, ok := actors.GetActorCodeID(av, actors.EamKey)
if !ok {
return fmt.Errorf("failed to get CodeCID for EAM during genesis")
}

header := &types.Actor{
Code: codecid,
Head: vmcontext.EmptyObjectCid,
Balance: big.Zero(),
Address: &builtintypes.EthereumAddressManagerActorAddr, // so that it can create ETH0
}
return nst.SetActor(ctx, builtintypes.EthereumAddressManagerActorAddr, header)
}

0 comments on commit 7b84fce

Please sign in to comment.