1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-12 00:10:58 +00:00

feat(legacy/ethereum): support access lists in EIP-1559 transactions

This commit is contained in:
Martin Milata 2021-10-21 17:29:04 +02:00
parent 076c5b9cbe
commit fa3c6140a6
3 changed files with 65 additions and 4 deletions

View File

@ -51,6 +51,12 @@ static uint64_t chain_id;
static bool eip1559; static bool eip1559;
struct SHA3_CTX keccak_ctx = {0}; struct SHA3_CTX keccak_ctx = {0};
static uint32_t signing_access_list_count;
static EthereumAccessList signing_access_list[8];
_Static_assert(sizeof(signing_access_list) ==
sizeof(((EthereumSignTxEIP1559 *)NULL)->access_list),
"access_list buffer size mismatch");
struct signing_params { struct signing_params {
bool pubkeyhash_set; bool pubkeyhash_set;
uint8_t pubkeyhash[20]; uint8_t pubkeyhash[20];
@ -192,6 +198,29 @@ static int rlp_calculate_number_length(uint64_t number) {
return length; return length;
} }
static uint32_t rlp_calculate_access_list_keys_length(
const EthereumAccessList_storage_keys_t *keys, uint32_t keys_count) {
uint32_t keys_length = 0;
for (size_t i = 0; i < keys_count; i++) {
keys_length += rlp_calculate_length(keys[i].size, keys[i].bytes[0]);
}
return keys_length;
}
static uint32_t rlp_calculate_access_list_length(
const EthereumAccessList access_list[8], uint32_t access_list_count) {
uint32_t length = 0;
for (size_t i = 0; i < access_list_count; i++) {
uint32_t address_length = rlp_calculate_length(20, 0xff);
uint32_t keys_length = rlp_calculate_access_list_keys_length(
access_list[i].storage_keys, access_list[i].storage_keys_count);
length += rlp_calculate_length(
address_length + rlp_calculate_length(keys_length, 0xff), 0xff);
}
return length;
}
static void send_request_chunk(void) { static void send_request_chunk(void) {
int progress = 1000 - (data_total > 1000000 ? data_left / (data_total / 800) int progress = 1000 - (data_total > 1000000 ? data_left / (data_total / 800)
: data_left * 800 / data_total); : data_left * 800 / data_total);
@ -212,7 +241,31 @@ static void send_signature(void) {
layoutProgress(_("Signing"), 1000); layoutProgress(_("Signing"), 1000);
if (eip1559) { if (eip1559) {
hash_rlp_list_length(0); hash_rlp_list_length(rlp_calculate_access_list_length(
signing_access_list, signing_access_list_count));
for (size_t i = 0; i < signing_access_list_count; i++) {
uint8_t address[20] = {0};
if (!ethereum_parse(signing_access_list[i].address, address)) {
fsm_sendFailure(FailureType_Failure_DataError, _("Malformed address"));
ethereum_signing_abort();
return;
}
uint32_t address_length =
rlp_calculate_length(sizeof(address), address[0]);
uint32_t keys_length = rlp_calculate_access_list_keys_length(
signing_access_list[i].storage_keys,
signing_access_list[i].storage_keys_count);
hash_rlp_list_length(address_length +
rlp_calculate_length(keys_length, 0xff));
hash_rlp_field(address, sizeof(address));
hash_rlp_list_length(keys_length);
for (size_t j = 0; j < signing_access_list[i].storage_keys_count; j++) {
hash_rlp_field(signing_access_list[i].storage_keys[j].bytes,
signing_access_list[i].storage_keys[j].size);
}
}
} else { } else {
/* eip-155 replay protection */ /* eip-155 replay protection */
/* hash v=chain_id, r=0, s=0 */ /* hash v=chain_id, r=0, s=0 */
@ -444,6 +497,8 @@ static bool ethereum_signing_init_common(struct signing_params *params) {
chain_id = 0; chain_id = 0;
memzero(&msg_tx_request, sizeof(EthereumTxRequest)); memzero(&msg_tx_request, sizeof(EthereumTxRequest));
memzero(signing_access_list, sizeof(signing_access_list));
signing_access_list_count = 0;
/* eip-155 chain id */ /* eip-155 chain id */
if (params->chain_id < 1) { if (params->chain_id < 1) {
@ -725,7 +780,10 @@ void ethereum_signing_init_eip1559(const EthereumSignTxEIP1559 *msg,
rlp_length += rlp_length +=
rlp_calculate_length(data_total, params.data_initial_chunk_bytes[0]); rlp_calculate_length(data_total, params.data_initial_chunk_bytes[0]);
rlp_length += rlp_calculate_length(0, 0xff); rlp_length +=
rlp_calculate_length(rlp_calculate_access_list_length(
msg->access_list, msg->access_list_count),
0xff);
/* Stage 2: Store header fields */ /* Stage 2: Store header fields */
hash_rlp_number(EIP1559_TX_TYPE); hash_rlp_number(EIP1559_TX_TYPE);
@ -744,6 +802,10 @@ void ethereum_signing_init_eip1559(const EthereumSignTxEIP1559 *msg,
hash_data(params.data_initial_chunk_bytes, params.data_initial_chunk_size); hash_data(params.data_initial_chunk_bytes, params.data_initial_chunk_size);
data_left = data_total - params.data_initial_chunk_size; data_left = data_total - params.data_initial_chunk_size;
/* make a copy of access_list, hash it after data is processed */
memcpy(signing_access_list, msg->access_list, sizeof(signing_access_list));
signing_access_list_count = msg->access_list_count;
memcpy(privkey, node->private_key, 32); memcpy(privkey, node->private_key, 32);
if (data_left > 0) { if (data_left > 0) {

View File

@ -16,7 +16,7 @@ EthereumSignTxEIP1559.value max_size:32
EthereumSignTxEIP1559.data_initial_chunk max_size:1024 EthereumSignTxEIP1559.data_initial_chunk max_size:1024
EthereumSignTxEIP1559.access_list max_count:8 EthereumSignTxEIP1559.access_list max_count:8
EthereumAccessList.address max_size:32 EthereumAccessList.address max_size:43
EthereumAccessList.storage_keys max_count:8 max_size:32 EthereumAccessList.storage_keys max_count:8 max_size:32
EthereumTxRequest.signature_r max_size:32 EthereumTxRequest.signature_r max_size:32

View File

@ -172,7 +172,6 @@ def test_data_streaming(client):
) )
@pytest.mark.skip_t1
def test_signtx_eip1559_access_list(client): def test_signtx_eip1559_access_list(client):
with client: with client: