Skip to content

Commit 133d2a7

Browse files
committed
Support EIP-7702 set code transactions
1 parent ff3559a commit 133d2a7

File tree

3 files changed

+65
-2
lines changed

3 files changed

+65
-2
lines changed

lib/evmone/constants.hpp

+6
Original file line numberDiff line numberDiff line change
@@ -12,4 +12,10 @@ constexpr auto MAX_CODE_SIZE = 0x6000;
1212
/// The limit of the size of init codes for contract creation
1313
/// defined by [EIP-3860](https://eips.ethereum.org/EIPS/eip-3860)
1414
constexpr auto MAX_INITCODE_SIZE = 2 * MAX_CODE_SIZE;
15+
16+
/// Prefix of code for delegated accounts
17+
/// defined by [EIP-7702](https://eips.ethereum.org/EIPS/eip-7702)
18+
constexpr uint8_t DELEGATION_MAGIC_BYTES[] = {0xef, 0x01, 0x00};
19+
constexpr bytes_view DELEGATION_MAGIC{DELEGATION_MAGIC_BYTES, std::size(DELEGATION_MAGIC_BYTES)};
20+
1521
} // namespace evmone

test/state/state.cpp

+42-1
Original file line numberDiff line numberDiff line change
@@ -56,13 +56,16 @@ int64_t compute_tx_intrinsic_cost(evmc_revision rev, const Transaction& tx) noex
5656
{
5757
static constexpr auto call_tx_cost = 21000;
5858
static constexpr auto create_tx_cost = 53000;
59+
static constexpr auto per_auth_base_cost = 2500;
5960
static constexpr auto initcode_word_cost = 2;
6061
const auto is_create = !tx.to.has_value(); // Covers also EOF creation txs.
62+
const auto auth_list_cost =
63+
static_cast<int64_t>(per_auth_base_cost * tx.authorization_list.size());
6164
const auto initcode_cost =
6265
is_create && rev >= EVMC_SHANGHAI ? initcode_word_cost * num_words(tx.data.size()) : 0;
6366
const auto tx_cost = is_create && rev >= EVMC_HOMESTEAD ? create_tx_cost : call_tx_cost;
6467
return tx_cost + compute_tx_data_cost(rev, tx.data) + compute_access_list_cost(tx.access_list) +
65-
compute_initcode_list_cost(rev, tx.initcodes) + initcode_cost;
68+
compute_initcode_list_cost(rev, tx.initcodes) + auth_list_cost + initcode_cost;
6669
}
6770

6871
evmc_message build_message(
@@ -283,6 +286,17 @@ std::variant<int64_t, std::error_code> validate_transaction(const Account& sende
283286
return make_error_code(BLOB_GAS_LIMIT_EXCEEDED);
284287
break;
285288

289+
case Transaction::Type::set_code:
290+
if (rev < EVMC_PRAGUE)
291+
return make_error_code(TX_TYPE_NOT_SUPPORTED);
292+
293+
// TODO is empty auth list valid?
294+
// if (tx.authorization_list.empty())
295+
// return make_error_code(EMPTY_BLOB_AUTHORIZATION_LIST);
296+
297+
// TODO is empty destination valid?
298+
break;
299+
286300
case Transaction::Type::initcodes:
287301
if (rev < EVMC_OSAKA)
288302
return make_error_code(TX_TYPE_NOT_SUPPORTED);
@@ -305,6 +319,7 @@ std::variant<int64_t, std::error_code> validate_transaction(const Account& sende
305319
{
306320
case Transaction::Type::blob:
307321
case Transaction::Type::initcodes:
322+
case Transaction::Type::set_code:
308323
case Transaction::Type::eip1559:
309324
if (rev < EVMC_LONDON)
310325
return make_error_code(TX_TYPE_NOT_SUPPORTED);
@@ -329,6 +344,7 @@ std::variant<int64_t, std::error_code> validate_transaction(const Account& sende
329344
if (tx.max_gas_price < block.base_fee)
330345
return make_error_code(FEE_CAP_LESS_THEN_BLOCKS);
331346

347+
// TODO this is relaxed for 7702
332348
if (!sender_acc.code.empty())
333349
return make_error_code(SENDER_NOT_EOA); // Origin must not be a contract (EIP-3607).
334350

@@ -442,6 +458,11 @@ void finalize(State& state, evmc_revision rev, const address& coinbase,
442458
delete_empty_accounts(state);
443459
}
444460

461+
constexpr bool is_code_delegated(bytes_view code) noexcept
462+
{
463+
return code.starts_with(DELEGATION_MAGIC);
464+
}
465+
445466
std::variant<TransactionReceipt, std::error_code> transition(State& state, const BlockInfo& block,
446467
const Transaction& tx, evmc_revision rev, evmc::VM& vm, int64_t block_gas_left,
447468
int64_t blob_gas_left)
@@ -458,6 +479,26 @@ std::variant<TransactionReceipt, std::error_code> transition(State& state, const
458479
if (holds_alternative<std::error_code>(validation_result))
459480
return get<std::error_code>(validation_result);
460481

482+
for (const auto& auth : tx.authorization_list)
483+
{
484+
// TODO check chain_id
485+
486+
auto& acc = state.get_or_insert(auth.signer);
487+
if (!acc.code.empty() && !is_code_delegated(acc.code))
488+
continue;
489+
490+
if (acc.nonce != auth.nonce)
491+
continue;
492+
493+
acc.code.reserve(std::size(DELEGATION_MAGIC) + std::size(auth.addr.bytes));
494+
acc.code = DELEGATION_MAGIC;
495+
acc.code += auth.addr;
496+
497+
++acc.nonce;
498+
499+
acc.access_status = EVMC_ACCESS_WARM;
500+
}
501+
461502
// Once the transaction is valid, create new sender account.
462503
// The account won't be empty because its nonce will be bumped.
463504
auto& sender_acc = (sender_ptr != nullptr) ? *sender_ptr : state.insert(tx.sender);

test/state/state.hpp

+17-1
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,17 @@ struct BlockInfo
174174

175175
using AccessList = std::vector<std::pair<address, std::vector<bytes32>>>;
176176

177+
struct Authorization
178+
{
179+
uint64_t chain_id = 0;
180+
address addr;
181+
uint64_t nonce = 0;
182+
address signer;
183+
intx::uint256 r;
184+
intx::uint256 s;
185+
bool y_parity = false;
186+
};
187+
177188
struct Transaction
178189
{
179190
/// The type of the transaction.
@@ -197,8 +208,12 @@ struct Transaction
197208
/// Introduced by EIP-4844 https://eips.ethereum.org/EIPS/eip-4844.
198209
blob = 3,
199210

211+
/// The typed set code transaction (with authorization list).
212+
/// Introduced by EIP-7702 https://eips.ethereum.org/EIPS/eip-7702.
213+
set_code = 4,
214+
200215
/// The typed transaction with initcode list.
201-
initcodes = 4,
216+
initcodes = 5,
202217
};
203218

204219
/// Returns amount of blob gas used by this transaction
@@ -224,6 +239,7 @@ struct Transaction
224239
intx::uint256 r;
225240
intx::uint256 s;
226241
uint8_t v = 0;
242+
std::vector<Authorization> authorization_list;
227243
std::vector<bytes> initcodes;
228244
};
229245

0 commit comments

Comments
 (0)