From 759dda71cec8a5d1353c67fa46bb81dce7ac94b8 Mon Sep 17 00:00:00 2001 From: mralj Date: Thu, 26 Sep 2024 13:22:58 +0200 Subject: [PATCH 1/6] added L1SLoad sekelton --- core/vm/contracts_rollup.go | 58 ++++++++++++++------------- core/vm/contracts_rollup_overrides.go | 17 +------- 2 files changed, 31 insertions(+), 44 deletions(-) diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index d18e5f92bf29..d633e368905d 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -11,14 +11,20 @@ import ( "github.com/ethereum/go-ethereum/params" ) -var rollupL1SloadAddress = common.BytesToAddress([]byte{0x10, 0x01}) +type RollupPrecompiledContractsOverrides struct { + l1SLoadGetLatestL1Block func() *big.Int +} -var PrecompiledContractsRollupR0 = PrecompiledContracts{ - rollupL1SloadAddress: &L1SLoad{}, +func GenerateRollupPrecompiledContractsOverrides(evm *EVM) RollupPrecompiledContractsOverrides { + return RollupPrecompiledContractsOverrides{ + l1SLoadGetLatestL1Block: getLatestL1BlockNumber(evm), + } } -type RollupPrecompileActivationConfig struct { - L1SLoad +var rollupL1SloadAddress = common.BytesToAddress([]byte{0x10, 0x01}) + +var PrecompiledContractsRollupR0 = PrecompiledContracts{ + rollupL1SloadAddress: &l1SLoad{}, } func activeRollupPrecompiledContracts(rules params.Rules) PrecompiledContracts { @@ -30,25 +36,24 @@ func activeRollupPrecompiledContracts(rules params.Rules) PrecompiledContracts { } } -func (pc *PrecompiledContracts) ActivateRollupPrecompiledContracts(config RollupPrecompileActivationConfig) { - // NOTE: if L1SLoad was not activated via chain rules this is no-op - pc.activateL1SLoad(config.L1RpcClient, config.GetLatestL1BlockNumber) -} - func (evm *EVM) activateRollupPrecompiledContracts() { - evm.precompiles.ActivateRollupPrecompiledContracts(RollupPrecompileActivationConfig{ - L1SLoad{L1RpcClient: evm.Config.L1RpcClient, GetLatestL1BlockNumber: evm.rollupPrecompileOverrides.l1SLoadGetLatestL1Block}, - }) + activeRollupPrecompiles := activeRollupPrecompiledContracts(evm.chainRules) + for k, v := range activeRollupPrecompiles { + evm.precompiles[k] = v + } + + // NOTE: if L1SLoad was not activated via chain rules this is no-op + evm.precompiles.activateL1SLoad(evm.Config.L1RpcClient, evm.rollupPrecompileOverrides.l1SLoadGetLatestL1Block) } -type L1SLoad struct { - L1RpcClient L1RpcClient - GetLatestL1BlockNumber func() *big.Int +type l1SLoad struct { + l1RpcClient L1Client + getLatestL1BlockNumber func() *big.Int } -func (c *L1SLoad) RequiredGas(input []byte) uint64 { return 0 } +func (c *l1SLoad) RequiredGas(input []byte) uint64 { return 0 } -func (c *L1SLoad) Run(input []byte) ([]byte, error) { +func (c *l1SLoad) Run(input []byte) ([]byte, error) { if !c.isL1SLoadActive() { return nil, errors.New("L1SLoad precompile not active") } @@ -56,18 +61,15 @@ func (c *L1SLoad) Run(input []byte) ([]byte, error) { return nil, nil } -func (c *L1SLoad) isL1SLoadActive() bool { - return c.GetLatestL1BlockNumber != nil && c.L1RpcClient != nil +func (c *l1SLoad) isL1SLoadActive() bool { + return c.getLatestL1BlockNumber != nil && c.l1RpcClient != nil } -func (pc *PrecompiledContracts) activateL1SLoad(l1RpcClient L1RpcClient, getLatestL1BlockNumber func() *big.Int) { - rulesSayContractShouldBeActive := (*pc)[rollupL1SloadAddress] != nil - paramsNotNil := l1RpcClient != nil && getLatestL1BlockNumber != nil - - if shouldActivateL1SLoad := rulesSayContractShouldBeActive && paramsNotNil; shouldActivateL1SLoad { - (*pc)[rollupL1SloadAddress] = &L1SLoad{ - L1RpcClient: l1RpcClient, - GetLatestL1BlockNumber: getLatestL1BlockNumber, +func (pc *PrecompiledContracts) activateL1SLoad(l1RpcClient L1Client, getLatestL1BlockNumber func() *big.Int) { + if (*pc)[rollupL1SloadAddress] != nil { + (*pc)[rollupL1SloadAddress] = &l1SLoad{ + l1RpcClient: l1RpcClient, + getLatestL1BlockNumber: getLatestL1BlockNumber, } } } diff --git a/core/vm/contracts_rollup_overrides.go b/core/vm/contracts_rollup_overrides.go index df7b4c3c6c30..56c70990b029 100644 --- a/core/vm/contracts_rollup_overrides.go +++ b/core/vm/contracts_rollup_overrides.go @@ -3,9 +3,7 @@ package vm -import ( - "math/big" -) +import "math/big" type RollupPrecompiledContractsOverrides struct { l1SLoadGetLatestL1Block func() *big.Int @@ -25,16 +23,3 @@ func getLatestL1BlockNumber(evm *EVM) func() *big.Int { return evm.Context.BlockNumber } } - -// [OVERRIDE] getLatestL1BlockNumber -// Each rollup should override this function so that it returns -// correct latest L1 block number -// -// EXAMPLE 2 -// func getLatestL1BlockNumber(evm *EVM) func() *big.Int { -// return func() *big.Int { -// addressOfL1BlockContract := common.Address{} -// slotInContractRepresentingL1BlockNumber := common.Hash{} -// return evm.StateDB.GetState(addressOfL1BlockContract, slotInContractRepresentingL1BlockNumber).Big() -// } -// } From f0dd2170c62121c7bdc1ba47389b6f1b790d9807 Mon Sep 17 00:00:00 2001 From: mralj Date: Fri, 27 Sep 2024 13:10:01 +0200 Subject: [PATCH 2/6] implements L1SLOAD contract --- core/vm/contracts_rollup.go | 105 +++++++++++++++++++++---------- core/vm/contracts_rollup_test.go | 32 ++++++++++ core/vm/contracts_test.go | 4 +- params/protocol_params_rollup.go | 7 +++ 4 files changed, 114 insertions(+), 34 deletions(-) create mode 100644 core/vm/contracts_rollup_test.go create mode 100644 params/protocol_params_rollup.go diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index d633e368905d..2b6140423ebc 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -4,27 +4,23 @@ package vm import ( + "context" "errors" "math/big" "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/log" "github.com/ethereum/go-ethereum/params" ) -type RollupPrecompiledContractsOverrides struct { - l1SLoadGetLatestL1Block func() *big.Int -} - -func GenerateRollupPrecompiledContractsOverrides(evm *EVM) RollupPrecompiledContractsOverrides { - return RollupPrecompiledContractsOverrides{ - l1SLoadGetLatestL1Block: getLatestL1BlockNumber(evm), - } -} - var rollupL1SloadAddress = common.BytesToAddress([]byte{0x10, 0x01}) var PrecompiledContractsRollupR0 = PrecompiledContracts{ - rollupL1SloadAddress: &l1SLoad{}, + rollupL1SloadAddress: &L1SLoad{}, +} + +type RollupPrecompileActivationConfig struct { + L1SLoad } func activeRollupPrecompiledContracts(rules params.Rules) PrecompiledContracts { @@ -36,40 +32,85 @@ func activeRollupPrecompiledContracts(rules params.Rules) PrecompiledContracts { } } -func (evm *EVM) activateRollupPrecompiledContracts() { - activeRollupPrecompiles := activeRollupPrecompiledContracts(evm.chainRules) - for k, v := range activeRollupPrecompiles { - evm.precompiles[k] = v - } - +func (pc *PrecompiledContracts) ActivateRollupPrecompiledContracts(config RollupPrecompileActivationConfig) { // NOTE: if L1SLoad was not activated via chain rules this is no-op - evm.precompiles.activateL1SLoad(evm.Config.L1RpcClient, evm.rollupPrecompileOverrides.l1SLoadGetLatestL1Block) + pc.activateL1SLoad(config.L1RpcClient, config.GetLatestL1BlockNumber) +} + +func (evm *EVM) activateRollupPrecompiledContracts() { + evm.precompiles.ActivateRollupPrecompiledContracts(RollupPrecompileActivationConfig{ + L1SLoad{L1RpcClient: evm.Config.L1RpcClient, GetLatestL1BlockNumber: evm.rollupPrecompileOverrides.l1SLoadGetLatestL1Block}, + }) } -type l1SLoad struct { - l1RpcClient L1Client - getLatestL1BlockNumber func() *big.Int +//INPUT SPECS: +//Byte range Name Description +//------------------------------------------------------------ +//[0: 19] (20 bytes) address The contract address +//[20: 51] (32 bytes) key1 The storage key +//... ... ... +//[k*32-12: k*32+19] (32 bytes) key_k The storage key + +type L1SLoad struct { + L1RpcClient L1RpcClient + GetLatestL1BlockNumber func() *big.Int } -func (c *l1SLoad) RequiredGas(input []byte) uint64 { return 0 } +func (c *L1SLoad) RequiredGas(input []byte) uint64 { + storageSlotsToLoad := len(input[common.AddressLength-1:]) / common.HashLength + storageSlotsToLoad = min(storageSlotsToLoad, params.L1SLoadMaxNumStorageSlots) -func (c *l1SLoad) Run(input []byte) ([]byte, error) { + return params.L1SLoadBaseGas + uint64(storageSlotsToLoad)*params.L1SLoadPerLoadGas +} + +func (c *L1SLoad) Run(input []byte) ([]byte, error) { if !c.isL1SLoadActive() { - return nil, errors.New("L1SLoad precompile not active") + log.Error("L1SLOAD called, but not activated", "client", c.L1RpcClient, "and latest block number function", c.GetLatestL1BlockNumber) + return nil, errors.New("L1SLOAD precompile not active") + } + + if len(input) < common.AddressLength+common.HashLength { + return nil, errors.New("L1SLOAD input too short") + } + + countOfStorageKeysToRead := (len(input) - common.AddressLength) / common.HashLength + thereIsAtLeast1StorageKeyToRead := countOfStorageKeysToRead > 0 + allStorageKeysAreExactly32Bytes := countOfStorageKeysToRead*common.HashLength == len(input)-common.AddressLength + + if inputIsValid := thereIsAtLeast1StorageKeyToRead && allStorageKeysAreExactly32Bytes; !inputIsValid { + return nil, errors.New("L1SLOAD input is malformed") + } + + contractAddress := common.BytesToAddress(input[:common.AddressLength]) + input = input[common.AddressLength-1:] + contractStorageKeys := make([]common.Hash, countOfStorageKeysToRead) + for k := 0; k < countOfStorageKeysToRead; k++ { + contractStorageKeys[k] = common.BytesToHash(input[k*common.HashLength : (k+1)*common.HashLength]) } - return nil, nil + // TODO: + // 1. Batch multiple storage slots + // 2. What about timeout strategy here? + res, err := c.L1RpcClient.StorageAt(context.Background(), contractAddress, contractStorageKeys[0], c.GetLatestL1BlockNumber()) + if err != nil { + return nil, err + } + + return res, nil } -func (c *l1SLoad) isL1SLoadActive() bool { - return c.getLatestL1BlockNumber != nil && c.l1RpcClient != nil +func (c *L1SLoad) isL1SLoadActive() bool { + return c.GetLatestL1BlockNumber != nil && c.L1RpcClient != nil } -func (pc *PrecompiledContracts) activateL1SLoad(l1RpcClient L1Client, getLatestL1BlockNumber func() *big.Int) { - if (*pc)[rollupL1SloadAddress] != nil { - (*pc)[rollupL1SloadAddress] = &l1SLoad{ - l1RpcClient: l1RpcClient, - getLatestL1BlockNumber: getLatestL1BlockNumber, +func (pc PrecompiledContracts) activateL1SLoad(l1RpcClient L1RpcClient, getLatestL1BlockNumber func() *big.Int) { + rulesSayContractShouldBeActive := pc[rollupL1SloadAddress] != nil + paramsNotNil := l1RpcClient != nil && getLatestL1BlockNumber != nil + + if shouldActivateL1SLoad := rulesSayContractShouldBeActive && paramsNotNil; shouldActivateL1SLoad { + pc[rollupL1SloadAddress] = &L1SLoad{ + L1RpcClient: l1RpcClient, + GetLatestL1BlockNumber: getLatestL1BlockNumber, } } } diff --git a/core/vm/contracts_rollup_test.go b/core/vm/contracts_rollup_test.go new file mode 100644 index 000000000000..cbca8188c9f6 --- /dev/null +++ b/core/vm/contracts_rollup_test.go @@ -0,0 +1,32 @@ +package vm + +import ( + "context" + "math/big" + "testing" + + "github.com/ethereum/go-ethereum/common" +) + +type MockL1RPCClient struct{} + +func (m MockL1RPCClient) StorageAt(ctx context.Context, account common.Address, key common.Hash, blockNumber *big.Int) ([]byte, error) { + return common.Hex2Bytes("abab"), nil +} + +func TestPrecompiledL1SLOAD(t *testing.T) { + mockL1RPCClient := MockL1RPCClient{} + + allPrecompiles[rollupL1SloadAddress] = &L1SLoad{} + allPrecompiles.activateL1SLoad(mockL1RPCClient, func() *big.Int { return big1 }) + + l1SLoadTestcase := precompiledTest{ + Name: "L1SLOAD", + Input: "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e864", + Expected: "abab", + Gas: 4000, + NoBenchmark: true, + } + + testPrecompiled(rollupL1SloadAddress.Hex(), l1SLoadTestcase, t) +} diff --git a/core/vm/contracts_test.go b/core/vm/contracts_test.go index fff5c966f34f..23035e0eeec2 100644 --- a/core/vm/contracts_test.go +++ b/core/vm/contracts_test.go @@ -45,7 +45,7 @@ type precompiledFailureTest struct { // allPrecompiles does not map to the actual set of precompiles, as it also contains // repriced versions of precompiles at certain slots -var allPrecompiles = map[common.Address]PrecompiledContract{ +var allPrecompiles = PrecompiledContracts{ common.BytesToAddress([]byte{1}): &ecrecover{}, common.BytesToAddress([]byte{2}): &sha256hash{}, common.BytesToAddress([]byte{3}): &ripemd160hash{}, @@ -181,7 +181,7 @@ func benchmarkPrecompiled(addr string, test precompiledTest, bench *testing.B) { // Keep it as uint64, multiply 100 to get two digit float later mgasps := (100 * 1000 * gasUsed) / elapsed bench.ReportMetric(float64(mgasps)/100, "mgas/s") - //Check if it is correct + // Check if it is correct if err != nil { bench.Error(err) return diff --git a/params/protocol_params_rollup.go b/params/protocol_params_rollup.go new file mode 100644 index 000000000000..3ac56c6f63d2 --- /dev/null +++ b/params/protocol_params_rollup.go @@ -0,0 +1,7 @@ +package params + +const ( + L1SLoadBaseGas uint64 = 2000 // Base price for L1Sload + L1SLoadPerLoadGas uint64 = 2000 // Per-load price for loading one storage slot + L1SLoadMaxNumStorageSlots = 5 // Max number of storage slots requested in L1Sload precompile +) From c4b24af69ac7940ca9c0fe09a12b55e52564367b Mon Sep 17 00:00:00 2001 From: mralj Date: Fri, 27 Sep 2024 13:32:52 +0200 Subject: [PATCH 3/6] added rpc call timeout strategy --- core/vm/contracts_rollup.go | 13 +++++++++++-- params/protocol_params_rollup.go | 2 ++ 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index 2b6140423ebc..8dee19ed6f4f 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -7,6 +7,7 @@ import ( "context" "errors" "math/big" + "time" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/log" @@ -90,8 +91,16 @@ func (c *L1SLoad) Run(input []byte) ([]byte, error) { // TODO: // 1. Batch multiple storage slots - // 2. What about timeout strategy here? - res, err := c.L1RpcClient.StorageAt(context.Background(), contractAddress, contractStorageKeys[0], c.GetLatestL1BlockNumber()) + var ctx context.Context + if params.L1SLoadRPCTimeoutInSec > 0 { + c, cancel := context.WithTimeout(context.Background(), time.Second*time.Duration(params.L1SLoadRPCTimeoutInSec)) + ctx = c + defer cancel() + } else { + ctx = context.Background() + } + + res, err := c.L1RpcClient.StorageAt(ctx, contractAddress, contractStorageKeys[0], c.GetLatestL1BlockNumber()) if err != nil { return nil, err } diff --git a/params/protocol_params_rollup.go b/params/protocol_params_rollup.go index 3ac56c6f63d2..46a0c9f570a9 100644 --- a/params/protocol_params_rollup.go +++ b/params/protocol_params_rollup.go @@ -5,3 +5,5 @@ const ( L1SLoadPerLoadGas uint64 = 2000 // Per-load price for loading one storage slot L1SLoadMaxNumStorageSlots = 5 // Max number of storage slots requested in L1Sload precompile ) + +var L1SLoadRPCTimeoutInSec = MainnetChainConfig.Clique.Period // After how many ms will RPC call timeout From bea23a3c5fca4f2bda610ec235e252373f3d762c Mon Sep 17 00:00:00 2001 From: mralj Date: Fri, 27 Sep 2024 22:40:37 +0200 Subject: [PATCH 4/6] added batch call for StoragesAt as well as tests --- core/vm/contracts_rollup.go | 2 +- core/vm/contracts_rollup_test.go | 22 +++++------ core/vm/interpreter.go | 2 +- .../vm/testdata/precompiles/fail-l1sload.json | 16 ++++++++ core/vm/testdata/precompiles/l1sload.json | 24 ++++++++++++ ethclient/ethclient_rollup.go | 38 +++++++++++++++++++ params/protocol_params_rollup.go | 3 +- 7 files changed, 92 insertions(+), 15 deletions(-) create mode 100644 core/vm/testdata/precompiles/fail-l1sload.json create mode 100644 core/vm/testdata/precompiles/l1sload.json create mode 100644 ethclient/ethclient_rollup.go diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index 8dee19ed6f4f..b1b2f17b4cc3 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -100,7 +100,7 @@ func (c *L1SLoad) Run(input []byte) ([]byte, error) { ctx = context.Background() } - res, err := c.L1RpcClient.StorageAt(ctx, contractAddress, contractStorageKeys[0], c.GetLatestL1BlockNumber()) + res, err := c.L1RpcClient.StoragesAt(ctx, contractAddress, contractStorageKeys, c.GetLatestL1BlockNumber()) if err != nil { return nil, err } diff --git a/core/vm/contracts_rollup_test.go b/core/vm/contracts_rollup_test.go index cbca8188c9f6..b798b5a24535 100644 --- a/core/vm/contracts_rollup_test.go +++ b/core/vm/contracts_rollup_test.go @@ -10,8 +10,15 @@ import ( type MockL1RPCClient struct{} -func (m MockL1RPCClient) StorageAt(ctx context.Context, account common.Address, key common.Hash, blockNumber *big.Int) ([]byte, error) { - return common.Hex2Bytes("abab"), nil +func (m MockL1RPCClient) StoragesAt(ctx context.Context, account common.Address, keys []common.Hash, blockNumber *big.Int) ([]byte, error) { + // testcase is in format "abab", this makes output lenght 2 bytes + const mockedRespValueSize = 2 + mockResp := make([]byte, mockedRespValueSize*len(keys)) + for i := range keys { + copy(mockResp[mockedRespValueSize*i:], common.Hex2Bytes("abab")) + } + + return mockResp, nil } func TestPrecompiledL1SLOAD(t *testing.T) { @@ -20,13 +27,6 @@ func TestPrecompiledL1SLOAD(t *testing.T) { allPrecompiles[rollupL1SloadAddress] = &L1SLoad{} allPrecompiles.activateL1SLoad(mockL1RPCClient, func() *big.Int { return big1 }) - l1SLoadTestcase := precompiledTest{ - Name: "L1SLOAD", - Input: "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e864", - Expected: "abab", - Gas: 4000, - NoBenchmark: true, - } - - testPrecompiled(rollupL1SloadAddress.Hex(), l1SLoadTestcase, t) + testJson("l1sload", rollupL1SloadAddress.Hex(), t) + testJsonFail("l1sload", rollupL1SloadAddress.Hex(), t) } diff --git a/core/vm/interpreter.go b/core/vm/interpreter.go index ebb07b8b8b5a..0872dfde4048 100644 --- a/core/vm/interpreter.go +++ b/core/vm/interpreter.go @@ -42,7 +42,7 @@ type Config struct { // [rollup-geth] type L1RpcClient interface { - StorageAt(ctx context.Context, account common.Address, key common.Hash, blockNumber *big.Int) ([]byte, error) + StoragesAt(ctx context.Context, account common.Address, keys []common.Hash, blockNumber *big.Int) ([]byte, error) } // ScopeContext contains the things that are per-call, such as stack and memory, diff --git a/core/vm/testdata/precompiles/fail-l1sload.json b/core/vm/testdata/precompiles/fail-l1sload.json new file mode 100644 index 000000000000..2128bec29b86 --- /dev/null +++ b/core/vm/testdata/precompiles/fail-l1sload.json @@ -0,0 +1,16 @@ +[ + { + "Name": "L1SLOAD FAIL: input contains only address", + "Input": "a83114A443dA1CecEFC50368531cACE9F37fCCcb", + "ExpectedError": "L1SLOAD input too short", + "Gas": 4000, + "NoBenchmark": true + }, + { + "Name": "L1SLOAD FAIL: input key not 32 bytes", + "Input": "a83114A443dA1CecEFC50368531cACE9F37fCCcb112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7122", + "ExpectedError": "L1SLOAD input is malformed", + "Gas": 4000, + "NoBenchmark": true + } +] diff --git a/core/vm/testdata/precompiles/l1sload.json b/core/vm/testdata/precompiles/l1sload.json new file mode 100644 index 000000000000..b0debfd3b49a --- /dev/null +++ b/core/vm/testdata/precompiles/l1sload.json @@ -0,0 +1,24 @@ +[ + { + "Name": "L1SLOAD: 1 key", + "Input": "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e864", + "Expected": "abab", + "Gas": 4000, + "NoBenchmark": true + }, + { + "Name": "L1SLOAD: 2 keys", + "Input": "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e8640112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a", + "Expected": "abababab", + "Gas": 6000, + "NoBenchmark": true + }, + + { + "Name": "L1SLOAD: 5 keys", + "Input": "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e8640112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a", + "Expected": "abababababababababab", + "Gas": 12000, + "NoBenchmark": true + } +] diff --git a/ethclient/ethclient_rollup.go b/ethclient/ethclient_rollup.go new file mode 100644 index 000000000000..4571cdca8509 --- /dev/null +++ b/ethclient/ethclient_rollup.go @@ -0,0 +1,38 @@ +package ethclient + +import ( + "context" + "math/big" + + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/common/hexutil" + "github.com/ethereum/go-ethereum/rpc" +) + +// StoragesAt returns the values of keys in the contract storage of the given account. +// The block number can be nil, in which case the value is taken from the latest known block. +func (ec *Client) StoragesAt(ctx context.Context, account common.Address, keys []common.Hash, blockNumber *big.Int) ([]byte, error) { + results := make([]hexutil.Bytes, len(keys)) + reqs := make([]rpc.BatchElem, len(keys)) + + for i := range reqs { + reqs[i] = rpc.BatchElem{ + Method: "eth_getStorageAt", + Args: []interface{}{account, keys[i], toBlockNumArg(blockNumber)}, + Result: &results[i], + } + } + if err := ec.c.BatchCallContext(ctx, reqs); err != nil { + return nil, err + } + + output := make([]byte, common.HashLength*len(keys)) + for i := range reqs { + if reqs[i].Error != nil { + return nil, reqs[i].Error + } + copy(output[i*common.HashLength:], results[i]) + } + + return output, nil +} diff --git a/params/protocol_params_rollup.go b/params/protocol_params_rollup.go index 46a0c9f570a9..d75663d81c78 100644 --- a/params/protocol_params_rollup.go +++ b/params/protocol_params_rollup.go @@ -4,6 +4,5 @@ const ( L1SLoadBaseGas uint64 = 2000 // Base price for L1Sload L1SLoadPerLoadGas uint64 = 2000 // Per-load price for loading one storage slot L1SLoadMaxNumStorageSlots = 5 // Max number of storage slots requested in L1Sload precompile + L1SLoadRPCTimeoutInSec = 3 ) - -var L1SLoadRPCTimeoutInSec = MainnetChainConfig.Clique.Period // After how many ms will RPC call timeout From e9a5c283a73f7775ae7b63eba3bcf355bd09eb9b Mon Sep 17 00:00:00 2001 From: mralj Date: Fri, 27 Sep 2024 22:44:44 +0200 Subject: [PATCH 5/6] added test for too long input edgecase --- core/vm/contracts_rollup.go | 2 -- core/vm/testdata/precompiles/fail-l1sload.json | 8 ++++++++ 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index b1b2f17b4cc3..7de2f59a9e04 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -89,8 +89,6 @@ func (c *L1SLoad) Run(input []byte) ([]byte, error) { contractStorageKeys[k] = common.BytesToHash(input[k*common.HashLength : (k+1)*common.HashLength]) } - // TODO: - // 1. Batch multiple storage slots var ctx context.Context if params.L1SLoadRPCTimeoutInSec > 0 { c, cancel := context.WithTimeout(context.Background(), time.Second*time.Duration(params.L1SLoadRPCTimeoutInSec)) diff --git a/core/vm/testdata/precompiles/fail-l1sload.json b/core/vm/testdata/precompiles/fail-l1sload.json index 2128bec29b86..43250bede45a 100644 --- a/core/vm/testdata/precompiles/fail-l1sload.json +++ b/core/vm/testdata/precompiles/fail-l1sload.json @@ -12,5 +12,13 @@ "ExpectedError": "L1SLOAD input is malformed", "Gas": 4000, "NoBenchmark": true + }, + { + "Name": "L1SLOAD FAIL: input too long", + "Input": "C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc22d2c7bb6fc06067df8b0223aec460d1ebb51febb9012bc2554141a4dca08e8640112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7a112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7aa83114A443dA1CecEFC50368531cACE9F37fCCcb112d016b65e9c617ad9ab60604f772a3620177bada4cdc773d9b6a982d3c2a7122", + "ExpectedError": "L1SLOAD input is malformed", + "Gas": 4000, + "NoBenchmark": true } + ] From 56c1f67d41e2cae55bba3e1a32a1f4bf5cabc88c Mon Sep 17 00:00:00 2001 From: mralj Date: Sat, 28 Sep 2024 13:00:06 +0200 Subject: [PATCH 6/6] added missing mocks for tests --- core/vm/contracts_rollup.go | 2 +- eth/tracers/api_rollup_test.go | 7 +++++++ internal/ethapi/api_rollup_test.go | 7 +++++++ internal/ethapi/transaction_args_rollup_test.go | 7 +++++++ 4 files changed, 22 insertions(+), 1 deletion(-) create mode 100644 eth/tracers/api_rollup_test.go create mode 100644 internal/ethapi/api_rollup_test.go create mode 100644 internal/ethapi/transaction_args_rollup_test.go diff --git a/core/vm/contracts_rollup.go b/core/vm/contracts_rollup.go index 7de2f59a9e04..d80bcd343782 100644 --- a/core/vm/contracts_rollup.go +++ b/core/vm/contracts_rollup.go @@ -78,7 +78,7 @@ func (c *L1SLoad) Run(input []byte) ([]byte, error) { thereIsAtLeast1StorageKeyToRead := countOfStorageKeysToRead > 0 allStorageKeysAreExactly32Bytes := countOfStorageKeysToRead*common.HashLength == len(input)-common.AddressLength - if inputIsValid := thereIsAtLeast1StorageKeyToRead && allStorageKeysAreExactly32Bytes; !inputIsValid { + if inputIsInvalid := !(thereIsAtLeast1StorageKeyToRead && allStorageKeysAreExactly32Bytes); inputIsInvalid { return nil, errors.New("L1SLOAD input is malformed") } diff --git a/eth/tracers/api_rollup_test.go b/eth/tracers/api_rollup_test.go new file mode 100644 index 000000000000..21581a015aef --- /dev/null +++ b/eth/tracers/api_rollup_test.go @@ -0,0 +1,7 @@ +package tracers + +import "github.com/ethereum/go-ethereum/core/vm" + +func (b *testBackend) GetL1RpcClient() vm.L1RpcClient { + return nil +} diff --git a/internal/ethapi/api_rollup_test.go b/internal/ethapi/api_rollup_test.go new file mode 100644 index 000000000000..337dca43e256 --- /dev/null +++ b/internal/ethapi/api_rollup_test.go @@ -0,0 +1,7 @@ +package ethapi + +import "github.com/ethereum/go-ethereum/core/vm" + +func (b *testBackend) GetL1RpcClient() vm.L1RpcClient { + return nil +} diff --git a/internal/ethapi/transaction_args_rollup_test.go b/internal/ethapi/transaction_args_rollup_test.go new file mode 100644 index 000000000000..5004799d10a9 --- /dev/null +++ b/internal/ethapi/transaction_args_rollup_test.go @@ -0,0 +1,7 @@ +package ethapi + +import "github.com/ethereum/go-ethereum/core/vm" + +func (b *backendMock) GetL1RpcClient() vm.L1RpcClient { + return nil +}