mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Update EVMC to 7.1.0
This commit is contained in:
parent
7a8d0c9250
commit
aba2ee0f68
@ -530,7 +530,7 @@ typedef size_t (*evmc_get_code_size_fn)(struct evmc_host_context* context,
|
|||||||
const evmc_address* address);
|
const evmc_address* address);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get code size callback function.
|
* Get code hash callback function.
|
||||||
*
|
*
|
||||||
* This callback function is used by a VM to get the keccak256 hash of the code stored
|
* This callback function is used by a VM to get the keccak256 hash of the code stored
|
||||||
* in the account at the given address. For existing accounts not having a code, this
|
* in the account at the given address. For existing accounts not having a code, this
|
||||||
|
@ -294,7 +294,7 @@ public:
|
|||||||
/// Destructor responsible for automatically releasing attached resources.
|
/// Destructor responsible for automatically releasing attached resources.
|
||||||
~result() noexcept
|
~result() noexcept
|
||||||
{
|
{
|
||||||
if (release)
|
if (release != nullptr)
|
||||||
release(this);
|
release(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -342,10 +342,10 @@ public:
|
|||||||
virtual ~HostInterface() noexcept = default;
|
virtual ~HostInterface() noexcept = default;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::account_exists
|
/// @copydoc evmc_host_interface::account_exists
|
||||||
virtual bool account_exists(const address& addr) noexcept = 0;
|
virtual bool account_exists(const address& addr) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_storage
|
/// @copydoc evmc_host_interface::get_storage
|
||||||
virtual bytes32 get_storage(const address& addr, const bytes32& key) noexcept = 0;
|
virtual bytes32 get_storage(const address& addr, const bytes32& key) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::set_storage
|
/// @copydoc evmc_host_interface::set_storage
|
||||||
virtual evmc_storage_status set_storage(const address& addr,
|
virtual evmc_storage_status set_storage(const address& addr,
|
||||||
@ -353,19 +353,19 @@ public:
|
|||||||
const bytes32& value) noexcept = 0;
|
const bytes32& value) noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_balance
|
/// @copydoc evmc_host_interface::get_balance
|
||||||
virtual uint256be get_balance(const address& addr) noexcept = 0;
|
virtual uint256be get_balance(const address& addr) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_code_size
|
/// @copydoc evmc_host_interface::get_code_size
|
||||||
virtual size_t get_code_size(const address& addr) noexcept = 0;
|
virtual size_t get_code_size(const address& addr) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_code_hash
|
/// @copydoc evmc_host_interface::get_code_hash
|
||||||
virtual bytes32 get_code_hash(const address& addr) noexcept = 0;
|
virtual bytes32 get_code_hash(const address& addr) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::copy_code
|
/// @copydoc evmc_host_interface::copy_code
|
||||||
virtual size_t copy_code(const address& addr,
|
virtual size_t copy_code(const address& addr,
|
||||||
size_t code_offset,
|
size_t code_offset,
|
||||||
uint8_t* buffer_data,
|
uint8_t* buffer_data,
|
||||||
size_t buffer_size) noexcept = 0;
|
size_t buffer_size) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::selfdestruct
|
/// @copydoc evmc_host_interface::selfdestruct
|
||||||
virtual void selfdestruct(const address& addr, const address& beneficiary) noexcept = 0;
|
virtual void selfdestruct(const address& addr, const address& beneficiary) noexcept = 0;
|
||||||
@ -374,10 +374,10 @@ public:
|
|||||||
virtual result call(const evmc_message& msg) noexcept = 0;
|
virtual result call(const evmc_message& msg) noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_tx_context
|
/// @copydoc evmc_host_interface::get_tx_context
|
||||||
virtual evmc_tx_context get_tx_context() noexcept = 0;
|
virtual evmc_tx_context get_tx_context() const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::get_block_hash
|
/// @copydoc evmc_host_interface::get_block_hash
|
||||||
virtual bytes32 get_block_hash(int64_t block_number) noexcept = 0;
|
virtual bytes32 get_block_hash(int64_t block_number) const noexcept = 0;
|
||||||
|
|
||||||
/// @copydoc evmc_host_interface::emit_log
|
/// @copydoc evmc_host_interface::emit_log
|
||||||
virtual void emit_log(const address& addr,
|
virtual void emit_log(const address& addr,
|
||||||
@ -395,7 +395,7 @@ class HostContext : public HostInterface
|
|||||||
{
|
{
|
||||||
const evmc_host_interface* host = nullptr;
|
const evmc_host_interface* host = nullptr;
|
||||||
evmc_host_context* context = nullptr;
|
evmc_host_context* context = nullptr;
|
||||||
evmc_tx_context tx_context = {};
|
mutable evmc_tx_context tx_context = {};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// Default constructor for null Host context.
|
/// Default constructor for null Host context.
|
||||||
@ -408,12 +408,12 @@ public:
|
|||||||
: host{&interface}, context{ctx}
|
: host{&interface}, context{ctx}
|
||||||
{}
|
{}
|
||||||
|
|
||||||
bool account_exists(const address& address) noexcept final
|
bool account_exists(const address& address) const noexcept final
|
||||||
{
|
{
|
||||||
return host->account_exists(context, &address);
|
return host->account_exists(context, &address);
|
||||||
}
|
}
|
||||||
|
|
||||||
bytes32 get_storage(const address& address, const bytes32& key) noexcept final
|
bytes32 get_storage(const address& address, const bytes32& key) const noexcept final
|
||||||
{
|
{
|
||||||
return host->get_storage(context, &address, &key);
|
return host->get_storage(context, &address, &key);
|
||||||
}
|
}
|
||||||
@ -425,17 +425,17 @@ public:
|
|||||||
return host->set_storage(context, &address, &key, &value);
|
return host->set_storage(context, &address, &key, &value);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint256be get_balance(const address& address) noexcept final
|
uint256be get_balance(const address& address) const noexcept final
|
||||||
{
|
{
|
||||||
return host->get_balance(context, &address);
|
return host->get_balance(context, &address);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t get_code_size(const address& address) noexcept final
|
size_t get_code_size(const address& address) const noexcept final
|
||||||
{
|
{
|
||||||
return host->get_code_size(context, &address);
|
return host->get_code_size(context, &address);
|
||||||
}
|
}
|
||||||
|
|
||||||
bytes32 get_code_hash(const address& address) noexcept final
|
bytes32 get_code_hash(const address& address) const noexcept final
|
||||||
{
|
{
|
||||||
return host->get_code_hash(context, &address);
|
return host->get_code_hash(context, &address);
|
||||||
}
|
}
|
||||||
@ -443,7 +443,7 @@ public:
|
|||||||
size_t copy_code(const address& address,
|
size_t copy_code(const address& address,
|
||||||
size_t code_offset,
|
size_t code_offset,
|
||||||
uint8_t* buffer_data,
|
uint8_t* buffer_data,
|
||||||
size_t buffer_size) noexcept final
|
size_t buffer_size) const noexcept final
|
||||||
{
|
{
|
||||||
return host->copy_code(context, &address, code_offset, buffer_data, buffer_size);
|
return host->copy_code(context, &address, code_offset, buffer_data, buffer_size);
|
||||||
}
|
}
|
||||||
@ -464,14 +464,14 @@ public:
|
|||||||
/// by assuming that the block timestamp should never be zero.
|
/// by assuming that the block timestamp should never be zero.
|
||||||
///
|
///
|
||||||
/// @return The cached transaction context.
|
/// @return The cached transaction context.
|
||||||
evmc_tx_context get_tx_context() noexcept final
|
evmc_tx_context get_tx_context() const noexcept final
|
||||||
{
|
{
|
||||||
if (tx_context.block_timestamp == 0)
|
if (tx_context.block_timestamp == 0)
|
||||||
tx_context = host->get_tx_context(context);
|
tx_context = host->get_tx_context(context);
|
||||||
return tx_context;
|
return tx_context;
|
||||||
}
|
}
|
||||||
|
|
||||||
bytes32 get_block_hash(int64_t number) noexcept final
|
bytes32 get_block_hash(int64_t number) const noexcept final
|
||||||
{
|
{
|
||||||
return host->get_block_hash(context, number);
|
return host->get_block_hash(context, number);
|
||||||
}
|
}
|
||||||
@ -534,7 +534,7 @@ public:
|
|||||||
/// Destructor responsible for automatically destroying the VM instance.
|
/// Destructor responsible for automatically destroying the VM instance.
|
||||||
~VM() noexcept
|
~VM() noexcept
|
||||||
{
|
{
|
||||||
if (m_instance)
|
if (m_instance != nullptr)
|
||||||
m_instance->destroy(m_instance);
|
m_instance->destroy(m_instance);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -570,6 +570,12 @@ public:
|
|||||||
/// @copydoc evmc_vm::version
|
/// @copydoc evmc_vm::version
|
||||||
char const* version() const noexcept { return m_instance->version; }
|
char const* version() const noexcept { return m_instance->version; }
|
||||||
|
|
||||||
|
/// Checks if the VM has the given capability.
|
||||||
|
bool has_capability(evmc_capabilities capability) const noexcept
|
||||||
|
{
|
||||||
|
return (get_capabilities() & static_cast<evmc_capabilities_flagset>(capability)) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
/// @copydoc evmc::vm::get_capabilities
|
/// @copydoc evmc::vm::get_capabilities
|
||||||
evmc_capabilities_flagset get_capabilities() const noexcept
|
evmc_capabilities_flagset get_capabilities() const noexcept
|
||||||
{
|
{
|
||||||
|
@ -297,6 +297,12 @@ struct evmc_vm* evmc_load_and_configure(const char* config, enum evmc_loader_err
|
|||||||
"%s (%s): unsupported value '%s' for option '%s'", vm->name, path,
|
"%s (%s): unsupported value '%s' for option '%s'", vm->name, path,
|
||||||
option, name);
|
option, name);
|
||||||
goto exit;
|
goto exit;
|
||||||
|
|
||||||
|
default:
|
||||||
|
ec = set_error(EVMC_LOADER_INVALID_OPTION_VALUE,
|
||||||
|
"%s (%s): unknown error when setting value '%s' for option '%s'",
|
||||||
|
vm->name, path, option, name);
|
||||||
|
goto exit;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
315
test/evmc/mocked_host.hpp
Normal file
315
test/evmc/mocked_host.hpp
Normal file
@ -0,0 +1,315 @@
|
|||||||
|
// EVMC: Ethereum Client-VM Connector API.
|
||||||
|
// Copyright 2019 The EVMC Authors.
|
||||||
|
// Licensed under the Apache License, Version 2.0.
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <evmc/evmc.hpp>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <string>
|
||||||
|
#include <unordered_map>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace evmc
|
||||||
|
{
|
||||||
|
/// The string of bytes.
|
||||||
|
using bytes = std::basic_string<uint8_t>;
|
||||||
|
|
||||||
|
/// Extended value (by dirty flag) for account storage.
|
||||||
|
struct storage_value
|
||||||
|
{
|
||||||
|
/// The storage value.
|
||||||
|
bytes32 value;
|
||||||
|
|
||||||
|
/// True means this value has been modified already by the current transaction.
|
||||||
|
bool dirty{false};
|
||||||
|
|
||||||
|
/// Default constructor.
|
||||||
|
storage_value() noexcept = default;
|
||||||
|
|
||||||
|
/// Constructor.
|
||||||
|
storage_value(const bytes32& _value, bool _dirty = false) noexcept // NOLINT
|
||||||
|
: value{_value}, dirty{_dirty}
|
||||||
|
{}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Mocked account.
|
||||||
|
struct MockedAccount
|
||||||
|
{
|
||||||
|
/// The account nonce.
|
||||||
|
int nonce = 0;
|
||||||
|
|
||||||
|
/// The account code.
|
||||||
|
bytes code;
|
||||||
|
|
||||||
|
/// The code hash. Can be a value not related to the actual code.
|
||||||
|
bytes32 codehash;
|
||||||
|
|
||||||
|
/// The account balance.
|
||||||
|
uint256be balance;
|
||||||
|
|
||||||
|
/// The account storage map.
|
||||||
|
std::unordered_map<bytes32, storage_value> storage;
|
||||||
|
|
||||||
|
/// Helper method for setting balance by numeric type.
|
||||||
|
void set_balance(uint64_t x) noexcept
|
||||||
|
{
|
||||||
|
balance = uint256be{};
|
||||||
|
for (std::size_t i = 0; i < sizeof(x); ++i)
|
||||||
|
balance.bytes[sizeof(balance) - 1 - i] = static_cast<uint8_t>(x >> (8 * i));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Mocked EVMC Host implementation.
|
||||||
|
class MockedHost : public Host
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// LOG record.
|
||||||
|
struct log_record
|
||||||
|
{
|
||||||
|
/// The address of the account which created the log.
|
||||||
|
address creator;
|
||||||
|
|
||||||
|
/// The data attached to the log.
|
||||||
|
bytes data;
|
||||||
|
|
||||||
|
/// The log topics.
|
||||||
|
std::vector<bytes32> topics;
|
||||||
|
|
||||||
|
/// Equal operator.
|
||||||
|
bool operator==(const log_record& other) const noexcept
|
||||||
|
{
|
||||||
|
return creator == other.creator && data == other.data && topics == other.topics;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// SELFDESTRUCT record.
|
||||||
|
struct selfdestuct_record
|
||||||
|
{
|
||||||
|
/// The address of the account which has self-destructed.
|
||||||
|
address selfdestructed;
|
||||||
|
|
||||||
|
/// The address of the beneficiary account.
|
||||||
|
address beneficiary;
|
||||||
|
|
||||||
|
/// Equal operator.
|
||||||
|
bool operator==(const selfdestuct_record& other) const noexcept
|
||||||
|
{
|
||||||
|
return selfdestructed == other.selfdestructed && beneficiary == other.beneficiary;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/// The set of all accounts in the Host, organized by their addresses.
|
||||||
|
std::unordered_map<address, MockedAccount> accounts;
|
||||||
|
|
||||||
|
/// The EVMC transaction context to be returned by get_tx_context().
|
||||||
|
evmc_tx_context tx_context = {};
|
||||||
|
|
||||||
|
/// The block header hash value to be returned by get_block_hash().
|
||||||
|
bytes32 block_hash = {};
|
||||||
|
|
||||||
|
/// The call result to be returned by the call() method.
|
||||||
|
evmc_result call_result = {};
|
||||||
|
|
||||||
|
/// The record of all block numbers for which get_block_hash() was called.
|
||||||
|
mutable std::vector<int64_t> recorded_blockhashes;
|
||||||
|
|
||||||
|
/// The record of all account accesses.
|
||||||
|
mutable std::vector<address> recorded_account_accesses;
|
||||||
|
|
||||||
|
/// The maximum number of entries in recorded_account_accesses record.
|
||||||
|
/// This is arbitrary value useful in fuzzing when we don't want the record to explode.
|
||||||
|
static constexpr auto max_recorded_account_accesses = 200;
|
||||||
|
|
||||||
|
/// The record of all call messages requested in the call() method.
|
||||||
|
std::vector<evmc_message> recorded_calls;
|
||||||
|
|
||||||
|
/// The maximum number of entries in recorded_calls record.
|
||||||
|
/// This is arbitrary value useful in fuzzing when we don't want the record to explode.
|
||||||
|
static constexpr auto max_recorded_calls = 100;
|
||||||
|
|
||||||
|
/// The record of all LOGs passed to the emit_log() method.
|
||||||
|
std::vector<log_record> recorded_logs;
|
||||||
|
|
||||||
|
/// The record of all SELFDESTRUCTs from the selfdestruct() method.
|
||||||
|
std::vector<selfdestuct_record> recorded_selfdestructs;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/// The copy of call inputs for the recorded_calls record.
|
||||||
|
std::vector<bytes> m_recorded_calls_inputs;
|
||||||
|
|
||||||
|
/// Record an account access.
|
||||||
|
/// @param addr The address of the accessed account.
|
||||||
|
void record_account_access(const address& addr) const
|
||||||
|
{
|
||||||
|
if (recorded_account_accesses.empty())
|
||||||
|
recorded_account_accesses.reserve(max_recorded_account_accesses);
|
||||||
|
|
||||||
|
if (recorded_account_accesses.size() < max_recorded_account_accesses)
|
||||||
|
recorded_account_accesses.emplace_back(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if an account exists (EVMC Host method).
|
||||||
|
bool account_exists(const address& addr) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
return accounts.count(addr) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the account's storage value at the given key (EVMC Host method).
|
||||||
|
bytes32 get_storage(const address& addr, const bytes32& key) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
|
||||||
|
const auto account_iter = accounts.find(addr);
|
||||||
|
if (account_iter == accounts.end())
|
||||||
|
return {};
|
||||||
|
|
||||||
|
const auto storage_iter = account_iter->second.storage.find(key);
|
||||||
|
if (storage_iter != account_iter->second.storage.end())
|
||||||
|
return storage_iter->second.value;
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Set the account's storage value (EVMC Host method).
|
||||||
|
evmc_storage_status set_storage(const address& addr,
|
||||||
|
const bytes32& key,
|
||||||
|
const bytes32& value) noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
const auto it = accounts.find(addr);
|
||||||
|
if (it == accounts.end())
|
||||||
|
return EVMC_STORAGE_UNCHANGED;
|
||||||
|
|
||||||
|
auto& old = it->second.storage[key];
|
||||||
|
|
||||||
|
// Follow https://eips.ethereum.org/EIPS/eip-1283 specification.
|
||||||
|
// WARNING! This is not complete implementation as refund is not handled here.
|
||||||
|
|
||||||
|
if (old.value == value)
|
||||||
|
return EVMC_STORAGE_UNCHANGED;
|
||||||
|
|
||||||
|
evmc_storage_status status{};
|
||||||
|
if (!old.dirty)
|
||||||
|
{
|
||||||
|
old.dirty = true;
|
||||||
|
if (!old.value)
|
||||||
|
status = EVMC_STORAGE_ADDED;
|
||||||
|
else if (value)
|
||||||
|
status = EVMC_STORAGE_MODIFIED;
|
||||||
|
else
|
||||||
|
status = EVMC_STORAGE_DELETED;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
status = EVMC_STORAGE_MODIFIED_AGAIN;
|
||||||
|
|
||||||
|
old.value = value;
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the account's balance (EVMC Host method).
|
||||||
|
uint256be get_balance(const address& addr) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
const auto it = accounts.find(addr);
|
||||||
|
if (it == accounts.end())
|
||||||
|
return {};
|
||||||
|
|
||||||
|
return it->second.balance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the account's code size (EVMC host method).
|
||||||
|
size_t get_code_size(const address& addr) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
const auto it = accounts.find(addr);
|
||||||
|
if (it == accounts.end())
|
||||||
|
return 0;
|
||||||
|
return it->second.code.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get the account's code hash (EVMC host method).
|
||||||
|
bytes32 get_code_hash(const address& addr) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
const auto it = accounts.find(addr);
|
||||||
|
if (it == accounts.end())
|
||||||
|
return {};
|
||||||
|
return it->second.codehash;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Copy the account's code to the given buffer (EVMC host method).
|
||||||
|
size_t copy_code(const address& addr,
|
||||||
|
size_t code_offset,
|
||||||
|
uint8_t* buffer_data,
|
||||||
|
size_t buffer_size) const noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
const auto it = accounts.find(addr);
|
||||||
|
if (it == accounts.end())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
const auto& code = it->second.code;
|
||||||
|
|
||||||
|
if (code_offset >= code.size())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
const auto n = std::min(buffer_size, code.size() - code_offset);
|
||||||
|
|
||||||
|
if (n > 0)
|
||||||
|
std::copy_n(&code[code_offset], n, buffer_data);
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Selfdestruct the account (EVMC host method).
|
||||||
|
void selfdestruct(const address& addr, const address& beneficiary) noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(addr);
|
||||||
|
recorded_selfdestructs.push_back({addr, beneficiary});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Call/create other contract (EVMC host method).
|
||||||
|
result call(const evmc_message& msg) noexcept override
|
||||||
|
{
|
||||||
|
record_account_access(msg.destination);
|
||||||
|
|
||||||
|
if (recorded_calls.empty())
|
||||||
|
{
|
||||||
|
recorded_calls.reserve(max_recorded_calls);
|
||||||
|
m_recorded_calls_inputs.reserve(max_recorded_calls); // Iterators will not invalidate.
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recorded_calls.size() < max_recorded_calls)
|
||||||
|
{
|
||||||
|
recorded_calls.emplace_back(msg);
|
||||||
|
auto& call_msg = recorded_calls.back();
|
||||||
|
if (call_msg.input_size > 0)
|
||||||
|
{
|
||||||
|
m_recorded_calls_inputs.emplace_back(call_msg.input_data, call_msg.input_size);
|
||||||
|
const auto& input_copy = m_recorded_calls_inputs.back();
|
||||||
|
call_msg.input_data = input_copy.data();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result{call_result};
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get transaction context (EVMC host method).
|
||||||
|
evmc_tx_context get_tx_context() const noexcept override { return tx_context; }
|
||||||
|
|
||||||
|
/// Get the block header hash (EVMC host method).
|
||||||
|
bytes32 get_block_hash(int64_t block_number) const noexcept override
|
||||||
|
{
|
||||||
|
recorded_blockhashes.emplace_back(block_number);
|
||||||
|
return block_hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Emit LOG (EVMC host method).
|
||||||
|
void emit_log(const address& addr,
|
||||||
|
const uint8_t* data,
|
||||||
|
size_t data_size,
|
||||||
|
const bytes32 topics[],
|
||||||
|
size_t topics_count) noexcept override
|
||||||
|
{
|
||||||
|
recorded_logs.push_back({addr, {data, data_size}, {topics, topics + topics_count}});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} // namespace evmc
|
Loading…
Reference in New Issue
Block a user