forked from ethereum/evmc
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathexample_host.cpp
219 lines (178 loc) · 6.25 KB
/
example_host.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
// EVMC: Ethereum Client-VM Connector API.
// Copyright 2016 The EVMC Authors.
// Licensed under the Apache License, Version 2.0.
/// @file
/// Example implementation of an EVMC Host.
#include "example_host.h"
#include <evmc/evmc.hpp>
#include <algorithm>
#include <map>
#include <vector>
using namespace evmc::literals;
namespace evmc
{
struct account
{
virtual ~account() = default;
evmc::uint256be balance = {};
std::vector<uint8_t> code;
std::map<evmc::bytes32, evmc::bytes32> storage;
std::map<evmc::bytes32, evmc::bytes32> transient_storage;
virtual evmc::bytes32 code_hash() const
{
// Extremely dumb "hash" function.
evmc::bytes32 ret{};
for (const auto v : code)
ret.bytes[v % sizeof(ret.bytes)] ^= v;
return ret;
}
};
using accounts = std::map<evmc::address, account>;
} // namespace evmc
class ExampleHost : public evmc::Host
{
evmc::accounts accounts;
evmc_tx_context tx_context{};
public:
ExampleHost() = default;
explicit ExampleHost(evmc_tx_context& _tx_context) noexcept : tx_context{_tx_context} {}
ExampleHost(evmc_tx_context& _tx_context, evmc::accounts& _accounts) noexcept
: accounts{_accounts}, tx_context{_tx_context}
{}
bool account_exists(const evmc::address& addr) const noexcept final
{
return accounts.find(addr) != accounts.end();
}
evmc::bytes32 get_storage(const evmc::address& addr,
const evmc::bytes32& key) const noexcept final
{
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;
return {};
}
evmc_storage_status set_storage(const evmc::address& addr,
const evmc::bytes32& key,
const evmc::bytes32& value) noexcept final
{
auto& account = accounts[addr];
auto prev_value = account.storage[key];
account.storage[key] = value;
return (prev_value == value) ? EVMC_STORAGE_ASSIGNED : EVMC_STORAGE_MODIFIED;
}
evmc::uint256be get_balance(const evmc::address& addr) const noexcept final
{
auto it = accounts.find(addr);
if (it != accounts.end())
return it->second.balance;
return {};
}
size_t get_code_size(const evmc::address& addr) const noexcept final
{
auto it = accounts.find(addr);
if (it != accounts.end())
return it->second.code.size();
return 0;
}
evmc::bytes32 get_code_hash(const evmc::address& addr) const noexcept final
{
auto it = accounts.find(addr);
if (it != accounts.end())
return it->second.code_hash();
return {};
}
size_t copy_code(const evmc::address& addr,
size_t code_offset,
uint8_t* buffer_data,
size_t buffer_size) const noexcept final
{
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;
}
bool selfdestruct(const evmc::address& addr, const evmc::address& beneficiary) noexcept final
{
(void)addr;
(void)beneficiary;
return false;
}
evmc::Result call(const evmc_message& msg) noexcept final
{
return evmc::Result{EVMC_REVERT, msg.gas, 0, msg.input_data, msg.input_size};
}
evmc_tx_context get_tx_context() const noexcept final { return tx_context; }
// NOLINTNEXTLINE(bugprone-exception-escape)
evmc::bytes32 get_block_hash(int64_t number) const noexcept final
{
const int64_t current_block_number = get_tx_context().block_number;
return (number < current_block_number && number >= current_block_number - 256) ?
0xb10c8a5fb10c8a5fb10c8a5fb10c8a5fb10c8a5fb10c8a5fb10c8a5fb10c8a5f_bytes32 :
0x0000000000000000000000000000000000000000000000000000000000000000_bytes32;
}
void emit_log(const evmc::address& addr,
const uint8_t* data,
size_t data_size,
const evmc::bytes32 topics[],
size_t topics_count) noexcept final
{
(void)addr;
(void)data;
(void)data_size;
(void)topics;
(void)topics_count;
}
evmc_access_status access_account(const evmc::address& addr) noexcept final
{
(void)addr;
return EVMC_ACCESS_COLD;
}
evmc_access_status access_storage(const evmc::address& addr,
const evmc::bytes32& key) noexcept final
{
(void)addr;
(void)key;
return EVMC_ACCESS_COLD;
}
evmc::bytes32 get_transient_storage(const evmc::address& addr,
const evmc::bytes32& key) const noexcept override
{
const auto account_iter = accounts.find(addr);
if (account_iter == accounts.end())
return {};
const auto transient_storage_iter = account_iter->second.transient_storage.find(key);
if (transient_storage_iter != account_iter->second.transient_storage.end())
return transient_storage_iter->second;
return {};
}
void set_transient_storage(const evmc::address& addr,
const evmc::bytes32& key,
const evmc::bytes32& value) noexcept override
{
accounts[addr].transient_storage[key] = value;
}
};
extern "C" {
const evmc_host_interface* example_host_get_interface()
{
return &evmc::Host::get_interface();
}
evmc_host_context* example_host_create_context(evmc_tx_context tx_context)
{
auto host = new ExampleHost{tx_context};
return host->to_context();
}
void example_host_destroy_context(evmc_host_context* context)
{
delete evmc::Host::from_context<ExampleHost>(context);
}
}