Skip to content

Commit

Permalink
Read intrinsic gas from FeeCurrencyDirectory
Browse files Browse the repository at this point in the history
  • Loading branch information
ezdac committed Jul 31, 2024
1 parent 5b788ef commit 107ba04
Show file tree
Hide file tree
Showing 24 changed files with 252 additions and 104 deletions.
7 changes: 6 additions & 1 deletion cmd/evm/internal/t8ntool/execution.go
Original file line number Diff line number Diff line change
Expand Up @@ -206,7 +206,12 @@ func (pre *Prestate) Apply(vmConfig vm.Config, chainConfig *params.ChainConfig,
rejectedTxs = append(rejectedTxs, &rejectedTx{i, errMsg})
continue
}
msg, err := core.TransactionToMessage(tx, signer, pre.Env.BaseFee, vmContext.ExchangeRates)
// NOTE: we can't provide exchange rates
// for fee-currencies here, since those are dynamically changing
// based on the oracle's exchange rates.
// When a Celo transaction with specified fee-currency is validated with this tool,
// this will thus result in a ErrNonWhitelistedFeeCurrency error for now.
msg, err := core.TransactionToMessage(tx, signer, pre.Env.BaseFee, vmContext.FeeCurrencyContext.ExchangeRates)
if err != nil {
log.Warn("rejected tx", "index", i, "hash", tx.Hash(), "error", err)
rejectedTxs = append(rejectedTxs, &rejectedTx{i, err.Error()})
Expand Down
26 changes: 26 additions & 0 deletions common/celo_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,32 @@ var (
)

type ExchangeRates = map[Address]*big.Rat
type IntrinsicGasCosts = map[Address]uint64

type FeeCurrencyContext struct {
ExchangeRates ExchangeRates
IntrinsicGasCosts IntrinsicGasCosts
}

func MaxAllowedIntrinsicGasCost(i IntrinsicGasCosts, feeCurrency *Address) (uint64, bool) {
intrinsicGas, ok := CurrencyIntrinsicGasCost(i, feeCurrency)
if !ok {
return 0, false
}
// allow the contract to overshoot 2 times the deducted intrinsic gas
// during execution
return intrinsicGas * 3, true
}
func CurrencyIntrinsicGasCost(i IntrinsicGasCosts, feeCurrency *Address) (uint64, bool) {
if feeCurrency == nil {
return 0, true
}
gasCost, ok := i[*feeCurrency]
if !ok {
return 0, false
}
return gasCost, true
}

func CurrencyWhitelist(exchangeRates ExchangeRates) []Address {
addrs := make([]Address, len(exchangeRates))
Expand Down
153 changes: 116 additions & 37 deletions contracts/fee_currencies.go
Original file line number Diff line number Diff line change
@@ -1,28 +1,22 @@
package contracts

import (
"errors"
"fmt"
"math"
"math/big"

"github.com/ethereum/go-ethereum/accounts/abi"
"github.com/ethereum/go-ethereum/accounts/abi/bind"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/exchange"
"github.com/ethereum/go-ethereum/contracts/addresses"
"github.com/ethereum/go-ethereum/contracts/celo/abigen"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/core/vm"
"github.com/ethereum/go-ethereum/log"
)

const (
Thousand = 1000

// Default intrinsic gas cost of transactions paying for gas in alternative currencies.
// Calculated to estimate 1 balance read, 1 debit, and 4 credit transactions.
IntrinsicGasForAlternativeFeeCurrency uint64 = 50 * Thousand
maxAllowedGasForDebitAndCredit uint64 = 3 * IntrinsicGasForAlternativeFeeCurrency
)

var feeCurrencyABI *abi.ABI

func init() {
Expand All @@ -34,31 +28,43 @@ func init() {
}

// Returns nil if debit is possible, used in tx pool validation
func TryDebitFees(tx *types.Transaction, from common.Address, backend *CeloBackend) error {
func TryDebitFees(tx *types.Transaction, from common.Address, backend *CeloBackend, feeContext common.FeeCurrencyContext) error {
amount := new(big.Int).SetUint64(tx.Gas())
amount.Mul(amount, tx.GasFeeCap())

snapshot := backend.State.Snapshot()
err := DebitFees(backend.NewEVM(), tx.FeeCurrency(), from, amount)
evm := backend.NewEVM()
evm.Context.FeeCurrencyContext = feeContext
_, err := DebitFees(evm, tx.FeeCurrency(), from, amount)
backend.State.RevertToSnapshot(snapshot)
return err
}

// Debits transaction fees from the transaction sender and stores them in the temporary address
func DebitFees(evm *vm.EVM, feeCurrency *common.Address, address common.Address, amount *big.Int) error {
func DebitFees(evm *vm.EVM, feeCurrency *common.Address, address common.Address, amount *big.Int) (uint64, error) {
if amount.Cmp(big.NewInt(0)) == 0 {
return nil
return 0, nil
}

maxIntrinsicGasCost, ok := common.MaxAllowedIntrinsicGasCost(evm.Context.FeeCurrencyContext.IntrinsicGasCosts, feeCurrency)
if !ok {
return 0, exchange.ErrNonWhitelistedFeeCurrency
}

leftoverGas, err := evm.CallWithABI(
feeCurrencyABI, "debitGasFees", *feeCurrency, maxAllowedGasForDebitAndCredit,
feeCurrencyABI, "debitGasFees", *feeCurrency, maxIntrinsicGasCost,
// debitGasFees(address from, uint256 value) parameters
address, amount,
)
gasUsed := maxAllowedGasForDebitAndCredit - leftoverGas
evm.Context.GasUsedForDebit = gasUsed
if errors.Is(err, vm.ErrOutOfGas) {
// This basically is a configuration / contract error, since
// the contract itself used way more gas than was expected (including grace limit)
return 0, fmt.Errorf("surpassed maximum allowed intrinsic gas for fee currency: %w", err)
}

gasUsed := maxIntrinsicGasCost - leftoverGas
log.Trace("DebitFees called", "feeCurrency", *feeCurrency, "gasUsed", gasUsed)
return err
return gasUsed, err
}

// Credits fees to the respective parties
Expand All @@ -71,6 +77,7 @@ func CreditFees(
feeCurrency *common.Address,
txSender, tipReceiver, baseFeeReceiver, l1DataFeeReceiver common.Address,
refund, feeTip, baseFee, l1DataFee *big.Int,
gasUsedDebit uint64,
) error {
// Our old `creditGasFees` function does not accept an l1DataFee and
// the fee currencies do not implement the new interface yet. Since tip
Expand All @@ -85,8 +92,12 @@ func CreditFees(
if tipReceiver.Cmp(common.ZeroAddress) == 0 {
tipReceiver = baseFeeReceiver
}
maxAllowedGasForDebitAndCredit, ok := common.MaxAllowedIntrinsicGasCost(evm.Context.FeeCurrencyContext.IntrinsicGasCosts, feeCurrency)
if !ok {
return exchange.ErrNonWhitelistedFeeCurrency
}

maxAllowedGasForCredit := maxAllowedGasForDebitAndCredit - evm.Context.GasUsedForDebit
maxAllowedGasForCredit := maxAllowedGasForDebitAndCredit - gasUsedDebit
leftoverGas, err := evm.CallWithABI(
feeCurrencyABI, "creditGasFees", *feeCurrency, maxAllowedGasForCredit,
// function creditGasFees(
Expand All @@ -101,43 +112,73 @@ func CreditFees(
// )
txSender, tipReceiver, common.ZeroAddress, baseFeeReceiver, refund, feeTip, common.Big0, baseFee,
)
if errors.Is(err, vm.ErrOutOfGas) {
// This is a configuration / contract error, since
// the contract itself used way more gas than was expected (including grace limit)
return fmt.Errorf("surpassed maximum allowed intrinsic gas for fee currency: %w", err)
}

gasUsed := maxAllowedGasForCredit - leftoverGas
log.Trace("CreditFees called", "feeCurrency", *feeCurrency, "gasUsed", gasUsed)

gasUsedForDebitAndCredit := evm.Context.GasUsedForDebit + gasUsed
if gasUsedForDebitAndCredit > IntrinsicGasForAlternativeFeeCurrency {
log.Info("Gas usage for debit+credit exceeds intrinsic gas!", "gasUsed", gasUsedForDebitAndCredit, "intrinsicGas", IntrinsicGasForAlternativeFeeCurrency, "feeCurrency", feeCurrency)
intrinsicGas, ok := common.CurrencyIntrinsicGasCost(evm.Context.FeeCurrencyContext.IntrinsicGasCosts, feeCurrency)
if !ok {
// this will never happen
return exchange.ErrNonWhitelistedFeeCurrency
}
gasUsedForDebitAndCredit := gasUsedDebit + gasUsed
if gasUsedForDebitAndCredit > intrinsicGas {
log.Info("Gas usage for debit+credit exceeds intrinsic gas!", "gasUsed", gasUsedForDebitAndCredit, "intrinsicGas", intrinsicGas, "feeCurrency", feeCurrency)
}
return err
}

// GetExchangeRates returns the exchange rates for all gas currencies from CELO
func GetRegisteredCurrencies(caller *abigen.FeeCurrencyDirectoryCaller) ([]common.Address, error) {
currencies, err := caller.GetCurrencies(&bind.CallOpts{})
if err != nil {
return currencies, fmt.Errorf("Failed to get registered tokens: %w", err)
}
return currencies, nil
}

// GetExchangeRates returns the exchange rates for the provided gas currencies from CELO
func GetExchangeRates(caller bind.ContractCaller) (common.ExchangeRates, error) {
exchangeRates := map[common.Address]*big.Rat{}
directory, err := abigen.NewFeeCurrencyDirectoryCaller(addresses.FeeCurrencyDirectoryAddress, caller)
if err != nil {
return exchangeRates, fmt.Errorf("Failed to access FeeCurrencyDirectory: %w", err)
}

registeredTokens, err := directory.GetCurrencies(&bind.CallOpts{})
currencies, err := GetRegisteredCurrencies(directory)
if err != nil {
return exchangeRates, fmt.Errorf("Failed to get whitelisted tokens: %w", err)
return map[common.Address]*big.Rat{}, err
}
for _, tokenAddress := range registeredTokens {
rate, err := directory.GetExchangeRate(&bind.CallOpts{}, tokenAddress)
if err != nil {
log.Error("Failed to get medianRate for gas currency!", "err", err, "tokenAddress", tokenAddress.Hex())
continue
}
if rate.Numerator.Sign() <= 0 || rate.Denominator.Sign() <= 0 {
log.Error("Bad exchange rate for fee currency", "tokenAddress", tokenAddress.Hex(), "numerator", rate.Numerator, "denominator", rate.Denominator)
continue
}
exchangeRates[tokenAddress] = new(big.Rat).SetFrac(rate.Numerator, rate.Denominator)
return getExchangeRatesForTokens(directory, currencies)
}

// GetFeeCurrencyContext returns the fee currency block context for all registered gas currencies from CELO
func GetFeeCurrencyContext(caller bind.ContractCaller) (common.FeeCurrencyContext, error) {
var feeContext common.FeeCurrencyContext
directory, err := abigen.NewFeeCurrencyDirectoryCaller(addresses.FeeCurrencyDirectoryAddress, caller)
if err != nil {
return feeContext, fmt.Errorf("Failed to access FeeCurrencyDirectory: %w", err)
}

return exchangeRates, nil
currencies, err := GetRegisteredCurrencies(directory)
if err != nil {
return feeContext, err
}
rates, err := getExchangeRatesForTokens(directory, currencies)
if err != nil {
return feeContext, err
}
intrinsicGas, err := getIntrinsicGasForTokens(directory, currencies)
if err != nil {
return feeContext, err
}
return common.FeeCurrencyContext{
ExchangeRates: rates,
IntrinsicGasCosts: intrinsicGas,
}, nil
}

// GetBalanceERC20 returns an account's balance on a given ERC20 currency
Expand Down Expand Up @@ -167,3 +208,41 @@ func GetFeeBalance(backend *CeloBackend, account common.Address, feeCurrency *co
}
return balance
}

// getIntrinsicGasForTokens returns the intrinsic gas costs for the provided gas currencies from CELO
func getIntrinsicGasForTokens(caller *abigen.FeeCurrencyDirectoryCaller, tokens []common.Address) (common.IntrinsicGasCosts, error) {
gasCosts := common.IntrinsicGasCosts{}
for _, tokenAddress := range tokens {
config, err := caller.GetCurrencyConfig(&bind.CallOpts{}, tokenAddress)
if err != nil {
log.Error("Failed to get intrinsic gas cost for gas currency!", "err", err, "tokenAddress", tokenAddress.Hex())
continue
}
if !config.IntrinsicGas.IsUint64() {
log.Error("Intrinsic gas cost exceeds MaxUint64 limit, capping at MaxUint64", "err", err, "tokenAddress", tokenAddress.Hex())
gasCosts[tokenAddress] = math.MaxUint64
} else {
gasCosts[tokenAddress] = config.IntrinsicGas.Uint64()
}
}
return gasCosts, nil
}

// getExchangeRatesForTokens returns the exchange rates for the provided gas currencies from CELO
func getExchangeRatesForTokens(caller *abigen.FeeCurrencyDirectoryCaller, tokens []common.Address) (common.ExchangeRates, error) {
exchangeRates := common.ExchangeRates{}
for _, tokenAddress := range tokens {
rate, err := caller.GetExchangeRate(&bind.CallOpts{}, tokenAddress)
if err != nil {
log.Error("Failed to get medianRate for gas currency!", "err", err, "tokenAddress", tokenAddress.Hex())
continue
}
if rate.Numerator.Sign() <= 0 || rate.Denominator.Sign() <= 0 {
log.Error("Bad exchange rate for fee currency", "tokenAddress", tokenAddress.Hex(), "numerator", rate.Numerator, "denominator", rate.Denominator)
continue
}
exchangeRates[tokenAddress] = new(big.Rat).SetFrac(rate.Numerator, rate.Denominator)
}

return exchangeRates, nil
}
4 changes: 2 additions & 2 deletions core/celo_evm.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,9 +15,9 @@ func setCeloFieldsInBlockContext(blockContext *vm.BlockContext, header *types.He

caller := &contracts.CeloBackend{ChainConfig: config, State: statedb}

// Add fee currency exchange rates
// Add fee currency context
var err error
blockContext.ExchangeRates, err = contracts.GetExchangeRates(caller)
blockContext.FeeCurrencyContext, err = contracts.GetFeeCurrencyContext(caller)
if err != nil {
log.Error("Error fetching exchange rates!", "err", err)
}
Expand Down
2 changes: 1 addition & 1 deletion core/state_prefetcher.go
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ func (p *statePrefetcher) Prefetch(block *types.Block, statedb *state.StateDB, c
return
}
// Convert the transaction into an executable message and pre-cache its sender
msg, err := TransactionToMessage(tx, signer, header.BaseFee, blockContext.ExchangeRates)
msg, err := TransactionToMessage(tx, signer, header.BaseFee, blockContext.FeeCurrencyContext.ExchangeRates)
if err != nil {
return // Also invalid block, bail out
}
Expand Down
6 changes: 3 additions & 3 deletions core/state_processor.go
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,7 @@ func (p *StateProcessor) Process(block *types.Block, statedb *state.StateDB, cfg
}
// Iterate over and process the individual transactions
for i, tx := range block.Transactions() {
msg, err := TransactionToMessage(tx, signer, header.BaseFee, context.ExchangeRates)
msg, err := TransactionToMessage(tx, signer, header.BaseFee, context.FeeCurrencyContext.ExchangeRates)
if err != nil {
return nil, nil, 0, fmt.Errorf("could not apply tx %d [%v]: %w", i, tx.Hash().Hex(), err)
}
Expand Down Expand Up @@ -156,7 +156,7 @@ func applyTransaction(msg *Message, config *params.ChainConfig, gp *GasPool, sta
if tx.Type() == types.CeloDynamicFeeTxV2Type {
alternativeBaseFee := evm.Context.BaseFee
if msg.FeeCurrency != nil {
alternativeBaseFee, err = exchange.ConvertCeloToCurrency(evm.Context.ExchangeRates, msg.FeeCurrency, evm.Context.BaseFee)
alternativeBaseFee, err = exchange.ConvertCeloToCurrency(evm.Context.FeeCurrencyContext.ExchangeRates, msg.FeeCurrency, evm.Context.BaseFee)
if err != nil {
return nil, err
}
Expand Down Expand Up @@ -189,7 +189,7 @@ func applyTransaction(msg *Message, config *params.ChainConfig, gp *GasPool, sta
func ApplyTransaction(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, error) {
// Create a new context to be used in the EVM environment
blockContext := NewEVMBlockContext(header, bc, author, config, statedb)
msg, err := TransactionToMessage(tx, types.MakeSigner(config, header.Number, header.Time), header.BaseFee, blockContext.ExchangeRates)
msg, err := TransactionToMessage(tx, types.MakeSigner(config, header.Number, header.Time), header.BaseFee, blockContext.FeeCurrencyContext.ExchangeRates)
if err != nil {
return nil, err
}
Expand Down
Loading

0 comments on commit 107ba04

Please sign in to comment.