evm: benchmark evm keeper (#586)
* adds more erc20 token benchmark tests * add more benchmark for statedb * adds benchmark for state_transition * fix typo * refactor state_transition_benchmark_test * update newSignedEthTx input argument * revises tx template type * modify tests for both LegacyTx and AccessListTx * update changelog
This commit is contained in:
parent
692fd9548a
commit
2088297e3d
@ -64,6 +64,7 @@ Ref: https://keepachangelog.com/en/1.0.0/
|
||||
|
||||
* (evm) [tharsis#461](https://github.com/tharsis/ethermint/pull/461) Increase performance of `StateDB` transaction log storage (r/w).
|
||||
* (evm) [tharsis#566](https://github.com/tharsis/ethermint/pull/566) Introduce `stateErr` store in `StateDB` to avoid meaningless operations if any error happened before
|
||||
* (evm) [tharsis#586](https://github.com/tharsis/ethermint/pull/586) Benchmark evm keeper
|
||||
|
||||
## [v0.5.0] - 2021-08-20
|
||||
|
||||
|
@ -77,3 +77,21 @@ func BenchmarkEmitLogs(b *testing.B) {
|
||||
return types.NewTx(suite.app.EvmKeeper.ChainID(), nonce, &contract, big.NewInt(0), 4100000, big.NewInt(1), input, nil)
|
||||
})
|
||||
}
|
||||
|
||||
func BenchmarkTokenTransferFrom(b *testing.B) {
|
||||
DoBenchmark(b, func(suite *KeeperTestSuite, contract common.Address) *types.MsgEthereumTx {
|
||||
input, err := ContractABI.Pack("transferFrom", suite.address, common.HexToAddress("0x378c50D9264C63F3F92B806d4ee56E9D86FfB3Ec"), big.NewInt(0))
|
||||
require.NoError(b, err)
|
||||
nonce := suite.app.EvmKeeper.GetNonce(suite.address)
|
||||
return types.NewTx(suite.app.EvmKeeper.ChainID(), nonce, &contract, big.NewInt(0), 410000, big.NewInt(1), input, nil)
|
||||
})
|
||||
}
|
||||
|
||||
func BenchmarkTokenMint(b *testing.B) {
|
||||
DoBenchmark(b, func(suite *KeeperTestSuite, contract common.Address) *types.MsgEthereumTx {
|
||||
input, err := ContractABI.Pack("mint", common.HexToAddress("0x378c50D9264C63F3F92B806d4ee56E9D86FfB3Ec"), big.NewInt(1000))
|
||||
require.NoError(b, err)
|
||||
nonce := suite.app.EvmKeeper.GetNonce(suite.address)
|
||||
return types.NewTx(suite.app.EvmKeeper.ChainID(), nonce, &contract, big.NewInt(0), 410000, big.NewInt(1), input, nil)
|
||||
})
|
||||
}
|
||||
|
220
x/evm/keeper/state_transition_benchmark_test.go
Normal file
220
x/evm/keeper/state_transition_benchmark_test.go
Normal file
@ -0,0 +1,220 @@
|
||||
package keeper_test
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"math/big"
|
||||
"testing"
|
||||
|
||||
"github.com/cosmos/cosmos-sdk/crypto/keyring"
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/ethereum/go-ethereum/common"
|
||||
"github.com/ethereum/go-ethereum/core"
|
||||
ethtypes "github.com/ethereum/go-ethereum/core/types"
|
||||
"github.com/ethereum/go-ethereum/params"
|
||||
"github.com/stretchr/testify/require"
|
||||
evmtypes "github.com/tharsis/ethermint/x/evm/types"
|
||||
)
|
||||
|
||||
var templateAccessListTx = ðtypes.AccessListTx{
|
||||
GasPrice: big.NewInt(1),
|
||||
Gas: 21000,
|
||||
To: &common.Address{},
|
||||
Value: big.NewInt(0),
|
||||
Data: []byte{},
|
||||
}
|
||||
|
||||
var templateLegacyTx = ðtypes.LegacyTx{
|
||||
GasPrice: big.NewInt(1),
|
||||
Gas: 21000,
|
||||
To: &common.Address{},
|
||||
Value: big.NewInt(0),
|
||||
Data: []byte{},
|
||||
}
|
||||
|
||||
func newSignedEthTx(
|
||||
txData ethtypes.TxData,
|
||||
nonce uint64,
|
||||
addr sdk.Address,
|
||||
krSigner keyring.Signer,
|
||||
ethSigner ethtypes.Signer,
|
||||
) (*ethtypes.Transaction, error) {
|
||||
var ethTx *ethtypes.Transaction
|
||||
switch txData := txData.(type) {
|
||||
case *ethtypes.AccessListTx:
|
||||
txData.Nonce = nonce
|
||||
ethTx = ethtypes.NewTx(txData)
|
||||
case *ethtypes.LegacyTx:
|
||||
txData.Nonce = nonce
|
||||
ethTx = ethtypes.NewTx(txData)
|
||||
default:
|
||||
return nil, errors.New("unknown transaction type!")
|
||||
}
|
||||
|
||||
sig, _, err := krSigner.SignByAddress(addr, ethTx.Hash().Bytes())
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ethTx, err = ethTx.WithSignature(ethSigner, sig)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return ethTx, nil
|
||||
}
|
||||
|
||||
func newNativeMessage(
|
||||
nonce uint64,
|
||||
blockHeight int64,
|
||||
address common.Address,
|
||||
cfg *params.ChainConfig,
|
||||
krSigner keyring.Signer,
|
||||
ethSigner ethtypes.Signer,
|
||||
isLegacy bool,
|
||||
) (core.Message, error) {
|
||||
msgSigner := ethtypes.MakeSigner(cfg, big.NewInt(blockHeight))
|
||||
|
||||
var ethTx *ethtypes.Transaction
|
||||
if isLegacy {
|
||||
templateLegacyTx.Nonce = nonce
|
||||
ethTx = ethtypes.NewTx(templateLegacyTx)
|
||||
} else {
|
||||
templateAccessListTx.Nonce = nonce
|
||||
ethTx = ethtypes.NewTx(templateAccessListTx)
|
||||
}
|
||||
|
||||
msg := &evmtypes.MsgEthereumTx{}
|
||||
msg.FromEthereumTx(ethTx)
|
||||
msg.From = address.Hex()
|
||||
|
||||
if err := msg.Sign(ethSigner, krSigner); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
m, err := msg.AsMessage(msgSigner)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return m, nil
|
||||
}
|
||||
|
||||
func BenchmarkApplyTransaction(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
ethSigner := ethtypes.LatestSignerForChainID(suite.app.EvmKeeper.ChainID())
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
for i := 0; i < b.N; i++ {
|
||||
b.StopTimer()
|
||||
tx, err := newSignedEthTx(templateAccessListTx,
|
||||
suite.app.EvmKeeper.GetNonce(suite.address),
|
||||
sdk.AccAddress(suite.address.Bytes()),
|
||||
suite.signer,
|
||||
ethSigner,
|
||||
)
|
||||
require.NoError(b, err)
|
||||
|
||||
b.StartTimer()
|
||||
resp, err := suite.app.EvmKeeper.ApplyTransaction(tx)
|
||||
b.StopTimer()
|
||||
|
||||
require.NoError(b, err)
|
||||
require.False(b, resp.Failed())
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkApplyTransactionWithLegacyTx(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
ethSigner := ethtypes.LatestSignerForChainID(suite.app.EvmKeeper.ChainID())
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
for i := 0; i < b.N; i++ {
|
||||
b.StopTimer()
|
||||
tx, err := newSignedEthTx(templateLegacyTx,
|
||||
suite.app.EvmKeeper.GetNonce(suite.address),
|
||||
sdk.AccAddress(suite.address.Bytes()),
|
||||
suite.signer,
|
||||
ethSigner,
|
||||
)
|
||||
require.NoError(b, err)
|
||||
|
||||
b.StartTimer()
|
||||
resp, err := suite.app.EvmKeeper.ApplyTransaction(tx)
|
||||
b.StopTimer()
|
||||
|
||||
require.NoError(b, err)
|
||||
require.False(b, resp.Failed())
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkApplyNativeMessage(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
params := suite.app.EvmKeeper.GetParams(suite.ctx)
|
||||
ethCfg := params.ChainConfig.EthereumConfig(suite.app.EvmKeeper.ChainID())
|
||||
signer := ethtypes.LatestSignerForChainID(suite.app.EvmKeeper.ChainID())
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
for i := 0; i < b.N; i++ {
|
||||
b.StopTimer()
|
||||
|
||||
m, err := newNativeMessage(
|
||||
suite.app.EvmKeeper.GetNonce(suite.address),
|
||||
suite.ctx.BlockHeight(),
|
||||
suite.address,
|
||||
ethCfg,
|
||||
suite.signer,
|
||||
signer,
|
||||
false,
|
||||
)
|
||||
require.NoError(b, err)
|
||||
|
||||
b.StartTimer()
|
||||
resp, err := suite.app.EvmKeeper.ApplyNativeMessage(m)
|
||||
b.StopTimer()
|
||||
|
||||
require.NoError(b, err)
|
||||
require.False(b, resp.Failed())
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkApplyNativeMessageWithLegacyTx(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
params := suite.app.EvmKeeper.GetParams(suite.ctx)
|
||||
ethCfg := params.ChainConfig.EthereumConfig(suite.app.EvmKeeper.ChainID())
|
||||
signer := ethtypes.LatestSignerForChainID(suite.app.EvmKeeper.ChainID())
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
for i := 0; i < b.N; i++ {
|
||||
b.StopTimer()
|
||||
|
||||
m, err := newNativeMessage(
|
||||
suite.app.EvmKeeper.GetNonce(suite.address),
|
||||
suite.ctx.BlockHeight(),
|
||||
suite.address,
|
||||
ethCfg,
|
||||
suite.signer,
|
||||
signer,
|
||||
true,
|
||||
)
|
||||
require.NoError(b, err)
|
||||
|
||||
b.StartTimer()
|
||||
resp, err := suite.app.EvmKeeper.ApplyNativeMessage(m)
|
||||
b.StopTimer()
|
||||
|
||||
require.NoError(b, err)
|
||||
require.False(b, resp.Failed())
|
||||
}
|
||||
}
|
@ -126,3 +126,57 @@ func BenchmarkSnapshot(b *testing.B) {
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkSubBalance(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
amt := big.NewInt(10)
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
|
||||
for i := 0; i < b.N; i++ {
|
||||
suite.app.EvmKeeper.SubBalance(suite.address, amt)
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkSetNonce(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
|
||||
for i := 0; i < b.N; i++ {
|
||||
suite.app.EvmKeeper.SetNonce(suite.address, 1)
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkAddRefund(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
|
||||
for i := 0; i < b.N; i++ {
|
||||
suite.app.EvmKeeper.AddRefund(1)
|
||||
}
|
||||
}
|
||||
|
||||
func BenchmarkSuicide(b *testing.B) {
|
||||
suite := KeeperTestSuite{}
|
||||
suite.DoSetupTest(b)
|
||||
|
||||
b.ResetTimer()
|
||||
b.ReportAllocs()
|
||||
for i := 0; i < b.N; i++ {
|
||||
b.StopTimer()
|
||||
addr := tests.GenerateAddress()
|
||||
suite.app.EvmKeeper.CreateAccount(addr)
|
||||
b.StartTimer()
|
||||
|
||||
suite.app.EvmKeeper.Suicide(addr)
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user