LCOV - code coverage report
Current view: top level - flamenco/runtime/tests - fd_dump_pb.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 0 958 0.0 %
Date: 2026-06-13 08:04:44 Functions: 0 31 0.0 %

          Line data    Source code
       1             : #include "fd_dump_pb.h"
       2             : #include "generated/block.pb.h"
       3             : #include "generated/instr.pb.h"
       4             : #include "generated/txn.pb.h"
       5             : #include "generated/vm.pb.h"
       6             : #include "../fd_system_ids.h"
       7             : #include "../fd_bank.h"
       8             : #include "../fd_runtime.h"
       9             : #include "../fd_alut.h"
      10             : #include "../program/fd_precompiles.h"
      11             : #include "../../../ballet/nanopb/pb_encode.h"
      12             : #include "../fd_runtime_stack.h"
      13             : 
      14             : #include <stdio.h> /* fopen */
      15             : #include <unistd.h> /* access */
      16             : 
      17             : 
      18             : struct fd_dump_account_key {
      19             :   fd_pubkey_t key;
      20             : };
      21             : typedef struct fd_dump_account_key fd_dump_account_t;
      22             : 
      23           0 : #define FD_DUMP_ACCOUNT_KEY_MAP_LG_SLOT_CNT (20)
      24             : 
      25             : #define MAP_NAME              fd_dump_account_key_map
      26           0 : #define MAP_T                 fd_dump_account_t
      27           0 : #define MAP_LG_SLOT_CNT       FD_DUMP_ACCOUNT_KEY_MAP_LG_SLOT_CNT
      28           0 : #define MAP_KEY_T             fd_pubkey_t
      29           0 : #define MAP_KEY_NULL          fd_solana_system_program_id
      30           0 : #define MAP_KEY_INVAL(k)      fd_pubkey_eq( &(k), &fd_solana_system_program_id )
      31           0 : #define MAP_KEY_EQUAL(k0,k1)  fd_pubkey_eq( &(k0), &(k1) )
      32             : #define MAP_KEY_EQUAL_IS_SLOW (0)
      33             : #define MAP_MEMOIZE           (0)
      34           0 : #define MAP_KEY_HASH(key)     ((uint)fd_hash( 0UL, (key).uc, sizeof(fd_pubkey_t) ))
      35             : #include "../../../util/tmpl/fd_map.c"
      36             : 
      37             : struct fd_dump_account_key_set {
      38             :   fd_dump_account_t * map;
      39             :   ulong               cnt;
      40             :   uchar               system_program_dumped;
      41             : };
      42             : typedef struct fd_dump_account_key_set fd_dump_account_key_set_t;
      43             : 
      44             : struct fd_dump_account_key_iter {
      45             :   fd_dump_account_key_set_t const * set;
      46             :   ulong                             slot_idx;
      47             :   uchar                             system_program;
      48             : };
      49             : typedef struct fd_dump_account_key_iter fd_dump_account_key_iter_t;
      50             : 
      51             : static void
      52           0 : fd_dump_account_key_iter_advance_map( fd_dump_account_key_iter_t * iter ) {
      53           0 :   while( iter->slot_idx<fd_dump_account_key_map_slot_cnt() &&
      54           0 :          fd_dump_account_key_map_key_inval( iter->set->map[iter->slot_idx].key ) ) {
      55           0 :     iter->slot_idx++;
      56           0 :   }
      57           0 : }
      58             : 
      59             : static fd_dump_account_key_iter_t *
      60             : fd_dump_account_key_iter_init( fd_dump_account_key_iter_t *      iter,
      61           0 :                                fd_dump_account_key_set_t const * set ) {
      62           0 :   iter->set            = set;
      63           0 :   iter->slot_idx       = 0UL;
      64           0 :   iter->system_program = set->system_program_dumped;
      65           0 :   if( !iter->system_program ) fd_dump_account_key_iter_advance_map( iter );
      66           0 :   return iter;
      67           0 : }
      68             : 
      69             : static int
      70           0 : fd_dump_account_key_iter_done( fd_dump_account_key_iter_t const * iter ) {
      71           0 :   return (!iter->system_program) & (iter->slot_idx>=fd_dump_account_key_map_slot_cnt());
      72           0 : }
      73             : 
      74             : static fd_pubkey_t const *
      75           0 : fd_dump_account_key_iter_ele( fd_dump_account_key_iter_t const * iter ) {
      76           0 :   return iter->system_program ? &fd_solana_system_program_id : &iter->set->map[iter->slot_idx].key;
      77           0 : }
      78             : 
      79             : static void
      80           0 : fd_dump_account_key_iter_next( fd_dump_account_key_iter_t * iter ) {
      81           0 :   if( iter->system_program ) {
      82           0 :     iter->system_program = 0;
      83           0 :   } else {
      84           0 :     iter->slot_idx++;
      85           0 :   }
      86           0 :   fd_dump_account_key_iter_advance_map( iter );
      87           0 : }
      88             : 
      89             : /***** CONSTANTS *****/
      90             : static fd_pubkey_t const * fd_dump_sysvar_ids[] = {
      91             :   &fd_sysvar_recent_block_hashes_id,
      92             :   &fd_sysvar_clock_id,
      93             :   &fd_sysvar_slot_history_id,
      94             :   &fd_sysvar_slot_hashes_id,
      95             :   &fd_sysvar_epoch_schedule_id,
      96             :   &fd_sysvar_epoch_rewards_id,
      97             :   &fd_sysvar_fees_id,
      98             :   &fd_sysvar_rent_id,
      99             :   &fd_sysvar_stake_history_id,
     100             :   &fd_sysvar_last_restart_slot_id,
     101             :   &fd_sysvar_instructions_id,
     102             : };
     103             : static ulong const num_sysvar_entries = (sizeof(fd_dump_sysvar_ids) / sizeof(fd_pubkey_t *));
     104             : 
     105             : static fd_pubkey_t const * fd_dump_builtin_ids[] = {
     106             :   &fd_solana_system_program_id,
     107             :   &fd_solana_vote_program_id,
     108             :   &fd_solana_stake_program_id,
     109             :   &fd_solana_bpf_loader_v4_program_id,
     110             :   &fd_solana_bpf_loader_deprecated_program_id,
     111             :   &fd_solana_bpf_loader_program_id,
     112             :   &fd_solana_bpf_loader_upgradeable_program_id,
     113             :   &fd_solana_compute_budget_program_id,
     114             :   &fd_solana_keccak_secp_256k_program_id,
     115             :   &fd_solana_secp256r1_program_id,
     116             :   &fd_solana_zk_elgamal_proof_program_id,
     117             :   &fd_solana_ed25519_sig_verify_program_id,
     118             : };
     119             : static ulong const num_loaded_builtins = (sizeof(fd_dump_builtin_ids) / sizeof(fd_pubkey_t *));
     120             : 
     121             : /***** UTILITY FUNCTIONS *****/
     122             : 
     123             : /** FEATURE DUMPING **/
     124             : static void
     125             : dump_sorted_features( fd_features_t const *        features,
     126             :                       fd_exec_test_feature_set_t * output_feature_set,
     127           0 :                       fd_spad_t *                  spad ) {
     128             :   /* NOTE: Caller must have a spad frame prepared */
     129           0 :   uint64_t * unsorted_features = fd_spad_alloc( spad, alignof(uint64_t), FD_FEATURE_ID_CNT * sizeof(uint64_t) );
     130           0 :   ulong num_features = 0;
     131           0 :   for( const fd_feature_id_t * current_feature = fd_feature_iter_init(); !fd_feature_iter_done( current_feature ); current_feature = fd_feature_iter_next( current_feature ) ) {
     132           0 :     if (features->f[current_feature->index] != FD_FEATURE_DISABLED) {
     133           0 :       unsorted_features[num_features++] = (uint64_t) current_feature->id.ul[0];
     134           0 :     }
     135           0 :   }
     136             : 
     137             :   // Set feature set in message
     138           0 :   output_feature_set->features_count = (pb_size_t)num_features;
     139           0 :   output_feature_set->features       = unsorted_features;
     140           0 : }
     141             : 
     142             : /** ACCOUNT DUMPING **/
     143             : static void
     144             : dump_account_state( fd_acc_t const *            acc,
     145             :                     fd_exec_test_acct_state_t * output_account,
     146           0 :                     fd_spad_t *                 spad ) {
     147             :     // Address
     148           0 :     fd_memcpy(output_account->address, acc->pubkey, sizeof(fd_pubkey_t));
     149             : 
     150             :     // Lamports
     151           0 :     output_account->lamports = (uint64_t)acc->lamports;
     152             : 
     153             :     // Data
     154           0 :     output_account->data = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE( acc->data_len ) );
     155           0 :     output_account->data->size = (pb_size_t) acc->data_len;
     156           0 :     fd_memcpy(output_account->data->bytes, acc->data, acc->data_len );
     157             : 
     158             :     // Executable
     159           0 :     output_account->executable = (bool)acc->executable;
     160             : 
     161             :     // Owner
     162           0 :     fd_memcpy(output_account->owner, acc->owner, sizeof(fd_pubkey_t));
     163           0 : }
     164             : 
     165             : static uchar
     166             : account_already_dumped( fd_exec_test_acct_state_t const * dumped_accounts,
     167             :                         ulong                             dumped_cnt,
     168           0 :                         fd_pubkey_t const *               account_key ) {
     169           0 :   for( ulong i=0UL; i<dumped_cnt; i++ ) {
     170           0 :     if( !memcmp( account_key, dumped_accounts[i].address, sizeof(fd_pubkey_t) ) ) {
     171           0 :       return 1;
     172           0 :     }
     173           0 :   }
     174           0 :   return 0;
     175           0 : }
     176             : 
     177             : /* Dumps a borrowed account if it exists and has not been dumped yet.
     178             :    Sets up the output borrowed account if it exists. Returns 0 if the
     179             :    account exists, 1 otherwise.
     180             :    TODO: This can be optimized by using a set. */
     181             : static uchar
     182             : dump_account_if_not_already_dumped( fd_accdb_t *                accdb,
     183             :                                     fd_accdb_fork_id_t          fork_id,
     184             :                                     fd_pubkey_t const *         account_key,
     185             :                                     fd_spad_t *                 spad,
     186             :                                     fd_exec_test_acct_state_t * out_acct_states,
     187             :                                     pb_size_t *                 out_acct_states_cnt,
     188           0 :                                     fd_acc_t *                  out_ro ) {
     189           0 :   fd_acc_t ro = fd_accdb_read_one( accdb, fork_id, account_key->uc );
     190           0 :   if( FD_UNLIKELY( !ro.lamports ) ) {
     191           0 :     fd_accdb_unread_one( accdb, &ro );
     192           0 :     return 1;
     193           0 :   }
     194             : 
     195           0 :   if( !account_already_dumped( out_acct_states, *out_acct_states_cnt, account_key ) ) {
     196           0 :     dump_account_state( &ro, &out_acct_states[*out_acct_states_cnt], spad );
     197           0 :     (*out_acct_states_cnt)++;
     198           0 :   }
     199             : 
     200           0 :   if( out_ro ) {
     201           0 :     *out_ro = ro;
     202           0 :   } else {
     203           0 :     fd_accdb_unread_one( accdb, &ro );
     204           0 :   }
     205           0 :   return 0;
     206           0 : }
     207             : 
     208             : static void
     209             : dump_executable_account_if_exists( fd_accdb_t *                      accdb,
     210             :                                    fd_accdb_fork_id_t                fork_id,
     211             :                                    fd_exec_test_acct_state_t const * program_account,
     212             :                                    fd_spad_t *                       spad,
     213             :                                    fd_exec_test_acct_state_t *       out_account_states,
     214           0 :                                    pb_size_t *                       out_account_states_count ) {
     215           0 :   if( FD_LIKELY( memcmp( program_account->owner, fd_solana_bpf_loader_upgradeable_program_id.key, sizeof(fd_pubkey_t) ) ) ) {
     216           0 :     return;
     217           0 :   }
     218             : 
     219           0 :   fd_bpf_state_t program_loader_state[1];
     220           0 :   if( FD_UNLIKELY( fd_bpf_state_decode(
     221           0 :       program_loader_state,
     222           0 :       program_account->data->bytes,
     223           0 :       program_account->data->size ) ) ) {
     224           0 :     return;
     225           0 :   }
     226           0 :   if( program_loader_state->discriminant!=FD_BPF_STATE_PROGRAM ) {
     227           0 :     return;
     228           0 :   }
     229             : 
     230           0 :   fd_pubkey_t * programdata_acc = &program_loader_state->inner.program.programdata_address;
     231           0 :   dump_account_if_not_already_dumped( accdb, fork_id, programdata_acc, spad, out_account_states, out_account_states_count, NULL );
     232           0 : }
     233             : 
     234             : static void
     235             : dump_sanitized_transaction( fd_accdb_t *                           accdb,
     236             :                             fd_accdb_fork_id_t                     fork_id,
     237             :                             fd_txn_t const *                       txn_descriptor,
     238             :                             uchar const *                          txn_payload,
     239             :                             fd_spad_t *                            spad,
     240           0 :                             fd_exec_test_sanitized_transaction_t * sanitized_transaction ) {
     241           0 :   fd_txn_acct_addr_lut_t const * address_lookup_tables = fd_txn_get_address_tables_const( txn_descriptor );
     242             : 
     243             :   /* Transaction Context -> tx -> message */
     244           0 :   sanitized_transaction->has_message = true;
     245           0 :   fd_exec_test_transaction_message_t * message = &sanitized_transaction->message;
     246             : 
     247             :   /* Transaction Context -> tx -> message -> is_legacy */
     248           0 :   message->is_legacy = txn_descriptor->transaction_version == FD_TXN_VLEGACY;
     249             : 
     250             :   /* Transaction Context -> tx -> message -> header */
     251           0 :   message->has_header = true;
     252           0 :   fd_exec_test_message_header_t * header = &message->header;
     253             : 
     254             :   /* Transaction Context -> tx -> message -> header -> num_required_signatures */
     255           0 :   header->num_required_signatures = txn_descriptor->signature_cnt;
     256             : 
     257             :   /* Transaction Context -> tx -> message -> header -> num_readonly_signed_accounts */
     258           0 :   header->num_readonly_signed_accounts = txn_descriptor->readonly_signed_cnt;
     259             : 
     260             :   /* Transaction Context -> tx -> message -> header -> num_readonly_unsigned_accounts */
     261           0 :   header->num_readonly_unsigned_accounts = txn_descriptor->readonly_unsigned_cnt;
     262             : 
     263             :   /* Transaction Context -> tx -> message -> account_keys */
     264           0 :   message->account_keys_count = txn_descriptor->acct_addr_cnt;
     265           0 :   message->account_keys = fd_spad_alloc( spad, alignof(pb_bytes_array_t *), PB_BYTES_ARRAY_T_ALLOCSIZE(txn_descriptor->acct_addr_cnt * sizeof(pb_bytes_array_t *)) );
     266           0 :   fd_acct_addr_t const * account_keys = fd_txn_get_acct_addrs( txn_descriptor, txn_payload );
     267           0 :   for( ulong i = 0; i < txn_descriptor->acct_addr_cnt; i++ ) {
     268           0 :     pb_bytes_array_t * account_key = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE(sizeof(fd_pubkey_t)) );
     269           0 :     account_key->size = sizeof(fd_pubkey_t);
     270           0 :     memcpy( account_key->bytes, &account_keys[i], sizeof(fd_pubkey_t) );
     271           0 :     message->account_keys[i] = account_key;
     272           0 :   }
     273             : 
     274             :   /* Transaction Context -> tx -> message -> recent_blockhash */
     275           0 :   uchar const * recent_blockhash = fd_txn_get_recent_blockhash( txn_descriptor, txn_payload );
     276           0 :   memcpy( message->recent_blockhash, recent_blockhash, sizeof(fd_hash_t) );
     277             : 
     278             :   /* Transaction Context -> tx -> message -> instructions */
     279           0 :   message->instructions_count = txn_descriptor->instr_cnt;
     280           0 :   message->instructions = fd_spad_alloc( spad, alignof(fd_exec_test_compiled_instruction_t), txn_descriptor->instr_cnt * sizeof(fd_exec_test_compiled_instruction_t) );
     281           0 :   for( ulong i = 0; i < txn_descriptor->instr_cnt; ++i ) {
     282           0 :     fd_txn_instr_t instr = txn_descriptor->instr[i];
     283           0 :     fd_exec_test_compiled_instruction_t * compiled_instruction = &message->instructions[i];
     284             : 
     285             :     // compiled instruction -> program_id_index
     286           0 :     compiled_instruction->program_id_index = instr.program_id;
     287             : 
     288             :     // compiled instruction -> accounts
     289           0 :     compiled_instruction->accounts_count = instr.acct_cnt;
     290           0 :     compiled_instruction->accounts = fd_spad_alloc( spad, alignof(uint32_t), instr.acct_cnt * sizeof(uint32_t) );
     291           0 :     uchar const * instr_accounts = fd_txn_get_instr_accts( &instr, txn_payload );
     292           0 :     for( ulong j = 0; j < instr.acct_cnt; ++j ) {
     293           0 :       uchar instr_acct_index = instr_accounts[j];
     294           0 :       compiled_instruction->accounts[j] = instr_acct_index;
     295           0 :     }
     296             : 
     297             :     // compiled instruction -> data
     298           0 :     uchar const * instr_data = fd_txn_get_instr_data( &instr, txn_payload );
     299           0 :     compiled_instruction->data = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE(instr.data_sz) );
     300           0 :     compiled_instruction->data->size = instr.data_sz;
     301           0 :     memcpy( compiled_instruction->data->bytes, instr_data, instr.data_sz );
     302           0 :   }
     303             : 
     304             :   /* ALUT stuff (non-legacy) */
     305           0 :   message->address_table_lookups_count = 0;
     306           0 :   if( !message->is_legacy ) {
     307             :     /* Transaction Context -> tx -> message -> address_table_lookups */
     308           0 :     message->address_table_lookups_count = txn_descriptor->addr_table_lookup_cnt;
     309           0 :     message->address_table_lookups = fd_spad_alloc( spad,
     310           0 :                                                     alignof(fd_exec_test_message_address_table_lookup_t),
     311           0 :                                                     txn_descriptor->addr_table_lookup_cnt * sizeof(fd_exec_test_message_address_table_lookup_t) );
     312           0 :     for( ulong i = 0; i < txn_descriptor->addr_table_lookup_cnt; ++i ) {
     313             :       // alut -> account_key
     314           0 :       fd_pubkey_t * alut_key = (fd_pubkey_t *) (txn_payload + address_lookup_tables[i].addr_off);
     315           0 :       memcpy( message->address_table_lookups[i].account_key, alut_key, sizeof(fd_pubkey_t) );
     316             : 
     317             :       // Access ALUT account data to access its keys
     318           0 :       fd_acc_t acc = fd_accdb_read_one( accdb, fork_id, alut_key->uc );
     319           0 :       if( FD_UNLIKELY( !acc.lamports ) ) FD_LOG_ERR(( "addr lut not found" ));
     320             : 
     321             :       // alut -> writable_indexes
     322           0 :       message->address_table_lookups[i].writable_indexes_count = address_lookup_tables[i].writable_cnt;
     323           0 :       message->address_table_lookups[i].writable_indexes = fd_spad_alloc( spad, alignof(uint32_t), address_lookup_tables[i].writable_cnt * sizeof(uint32_t) );
     324           0 :       uchar * writable_indexes = (uchar *) (txn_payload + address_lookup_tables[i].writable_off);
     325           0 :       for( ulong j = 0; j < address_lookup_tables[i].writable_cnt; ++j ) {
     326           0 :         message->address_table_lookups[i].writable_indexes[j] = writable_indexes[j];
     327           0 :       }
     328             : 
     329             :       // alut -> readonly_indexes
     330           0 :       message->address_table_lookups[i].readonly_indexes_count = address_lookup_tables[i].readonly_cnt;
     331           0 :       message->address_table_lookups[i].readonly_indexes = fd_spad_alloc( spad, alignof(uint32_t), address_lookup_tables[i].readonly_cnt * sizeof(uint32_t) );
     332           0 :       uchar * readonly_indexes = (uchar *) (txn_payload + address_lookup_tables[i].readonly_off);
     333           0 :       for( ulong j = 0; j < address_lookup_tables[i].readonly_cnt; ++j ) {
     334           0 :         message->address_table_lookups[i].readonly_indexes[j] = readonly_indexes[j];
     335           0 :       }
     336             : 
     337           0 :       fd_accdb_unread_one( accdb, &acc );
     338           0 :     }
     339           0 :   }
     340             : 
     341             :   /* Transaction Context -> tx -> message_hash */
     342             :   // Skip because it does not matter what's in here
     343             : 
     344             :   /* Transaction Context -> tx -> signatures */
     345           0 :   sanitized_transaction->signatures_count = txn_descriptor->signature_cnt;
     346           0 :   sanitized_transaction->signatures = fd_spad_alloc( spad, alignof(pb_bytes_array_t *), PB_BYTES_ARRAY_T_ALLOCSIZE(txn_descriptor->signature_cnt * sizeof(pb_bytes_array_t *)) );
     347           0 :   fd_ed25519_sig_t const * signatures = fd_txn_get_signatures( txn_descriptor, txn_payload );
     348           0 :   for( uchar i = 0; i < txn_descriptor->signature_cnt; ++i ) {
     349           0 :     pb_bytes_array_t * signature = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE(sizeof(fd_ed25519_sig_t)) );
     350           0 :     signature->size = sizeof(fd_ed25519_sig_t);
     351           0 :     memcpy( signature->bytes, &signatures[i], sizeof(fd_ed25519_sig_t) );
     352           0 :     sanitized_transaction->signatures[i] = signature;
     353           0 :   }
     354           0 : }
     355             : 
     356             : static void
     357             : dump_fee_rate_governor( fd_bank_t *                        bank,
     358           0 :                         fd_exec_test_fee_rate_governor_t * out ) {
     359           0 :   fd_fee_rate_governor_t const * frg = &bank->f.fee_rate_governor;
     360           0 :   *out = (fd_exec_test_fee_rate_governor_t){
     361           0 :     .target_lamports_per_signature = frg->target_lamports_per_signature,
     362           0 :     .target_signatures_per_slot    = frg->target_signatures_per_slot,
     363           0 :     .min_lamports_per_signature    = frg->min_lamports_per_signature,
     364           0 :     .max_lamports_per_signature    = frg->max_lamports_per_signature,
     365           0 :     .burn_percent                  = frg->burn_percent,
     366           0 :   };
     367           0 : }
     368             : 
     369             : static void
     370             : dump_blockhash_queue( fd_bank_t *                             bank,
     371             :                       fd_spad_t *                             spad,
     372             :                       fd_exec_test_blockhash_queue_entry_t ** entries_out,
     373           0 :                       pb_size_t *                             count_out ) {
     374           0 :   fd_blockhashes_t const * bhq      = &bank->f.block_hash_queue;
     375           0 :   ulong                    bhq_size = fd_ulong_min( FD_BLOCKHASHES_MAX, fd_blockhash_deq_cnt( bhq->d.deque ) );
     376             : 
     377           0 :   fd_exec_test_blockhash_queue_entry_t * entries = fd_spad_alloc( spad,
     378           0 :       alignof(fd_exec_test_blockhash_queue_entry_t),
     379           0 :       bhq_size * sizeof(fd_exec_test_blockhash_queue_entry_t) );
     380             : 
     381           0 :   ulong cnt = 0UL;
     382           0 :   for( fd_blockhash_deq_iter_t iter=fd_blockhash_deq_iter_init_rev( bhq->d.deque );
     383           0 :        !fd_blockhash_deq_iter_done_rev( bhq->d.deque, iter ) && cnt<bhq_size;
     384           0 :        iter=fd_blockhash_deq_iter_prev( bhq->d.deque, iter ), cnt++ ) {
     385           0 :     fd_blockhash_info_t const * ele   = fd_blockhash_deq_iter_ele_const( bhq->d.deque, iter );
     386           0 :     fd_exec_test_blockhash_queue_entry_t * entry = &entries[bhq_size-cnt-1UL];
     387           0 :     fd_memcpy( entry->blockhash, ele->hash.uc, sizeof(fd_hash_t) );
     388           0 :     entry->lamports_per_signature = ele->lamports_per_signature;
     389           0 :   }
     390             : 
     391           0 :   *entries_out = entries;
     392           0 :   *count_out   = (pb_size_t)bhq_size;
     393           0 : }
     394             : 
     395             : static void
     396             : dump_txn_bank( fd_bank_t *                  bank,
     397             :                fd_spad_t *                  spad,
     398           0 :                fd_exec_test_txn_context_t * txn_context ) {
     399           0 :   txn_context->has_bank              = true;
     400           0 :   fd_exec_test_txn_bank_t * txn_bank = &txn_context->bank;
     401             : 
     402             :   /* TxnBank -> blockhash_queue */
     403           0 :   dump_blockhash_queue( bank, spad, &txn_bank->blockhash_queue, &txn_bank->blockhash_queue_count );
     404             : 
     405             :   /* TxnBank -> rbh_lamports_per_signature */
     406           0 :   txn_bank->rbh_lamports_per_signature = (uint)bank->f.rbh_lamports_per_sig;
     407             : 
     408             :   /* TxnBank -> fee_rate_governor */
     409           0 :   txn_bank->has_fee_rate_governor = true;
     410           0 :   dump_fee_rate_governor( bank, &txn_bank->fee_rate_governor );
     411             : 
     412             :   /* TxnBank -> total_epoch_stake */
     413           0 :   txn_bank->total_epoch_stake = bank->f.total_epoch_stake;
     414             : 
     415             :   /* TxnBank -> features */
     416           0 :   txn_bank->has_features = true;
     417           0 :   dump_sorted_features( &bank->f.features, &txn_bank->features, spad );
     418           0 : }
     419             : 
     420             : /** SECONDARY FUNCTIONS **/
     421             : 
     422             : /* add_account_to_dumped_accounts adds an account to the dumped accounts
     423             :    set if it does not exist already. Returns 0 if the account already
     424             :    exists, and 1 if the account was added successfully.
     425             : 
     426             :    TODO: Txn dumping should be optimized to use these functions. */
     427             : static uchar
     428             : add_account_to_dumped_accounts( fd_dump_account_key_set_t * dumped_accounts,
     429           0 :                                 fd_pubkey_t const *         pubkey ) {
     430           0 :   if( fd_pubkey_eq( pubkey, &fd_solana_system_program_id ) ) {
     431           0 :     if( dumped_accounts->system_program_dumped ) return 0;
     432           0 :     dumped_accounts->system_program_dumped = 1;
     433           0 :     dumped_accounts->cnt++;
     434           0 :     return 1;
     435           0 :   }
     436             : 
     437             :   /* If the key already exists, return early. */
     438           0 :   if( fd_dump_account_key_map_query( dumped_accounts->map, *pubkey, NULL ) ) {
     439           0 :     return 0;
     440           0 :   }
     441             : 
     442           0 :   if( FD_UNLIKELY( dumped_accounts->cnt-dumped_accounts->system_program_dumped>=fd_dump_account_key_map_key_max() ) ) {
     443           0 :     FD_LOG_CRIT(( "too many dumped accounts for fd_dump_account_key_map" ));
     444           0 :   }
     445           0 :   fd_dump_account_t * new_ele = fd_dump_account_key_map_insert( dumped_accounts->map, *pubkey );
     446           0 :   if( FD_UNLIKELY( !new_ele ) ) FD_LOG_CRIT(( "fd_dump_account_key_map_insert failed" ));
     447           0 :   dumped_accounts->cnt++;
     448           0 :   return 1;
     449           0 : }
     450             : 
     451             : /* add_account_and_programdata_to_dumped_accounts adds an account and
     452             :    its programdata account (if the account is a v3 program) to the
     453             :    dumped accounts set if they do not exist already. */
     454             : static void
     455             : add_account_and_programdata_to_dumped_accounts( fd_accdb_t *                accdb,
     456             :                                                 fd_accdb_fork_id_t          fork_id,
     457             :                                                 fd_dump_account_key_set_t * dumped_accounts,
     458           0 :                                                 fd_pubkey_t const *         pubkey ) {
     459             :   /* Add the current account to the dumped accounts set. We can save
     460             :      some time by enforcing an invariant that "if current account was
     461             :      dumped, then programdata account was also dumped," so we save
     462             :      ourselves a call to accdb. */
     463           0 :   uchar ret = add_account_to_dumped_accounts( dumped_accounts, pubkey );
     464           0 :   if( ret==0 ) return;
     465             : 
     466             :   /* Read the account from accdb to see if its a program account and if
     467             :      it needs to be dumped. */
     468           0 :   fd_acc_t program_account = fd_accdb_read_one( accdb, fork_id, pubkey->uc );
     469           0 :   if( FD_UNLIKELY( !program_account.lamports ) ) {
     470           0 :     fd_accdb_unread_one( accdb, &program_account );
     471           0 :     return;
     472           0 :   }
     473             : 
     474             :   /* Return if its not owned by the v3 loader */
     475           0 :   if( FD_LIKELY( !fd_pubkey_eq( (fd_pubkey_t*)program_account.owner, &fd_solana_bpf_loader_upgradeable_program_id ) ) ) {
     476           0 :     fd_accdb_unread_one( accdb, &program_account );
     477           0 :     return;
     478           0 :   }
     479             : 
     480             :   /* Get the program account state */
     481           0 :   fd_bpf_state_t program_account_state[1];
     482           0 :   if( FD_UNLIKELY( fd_bpf_state_decode(
     483           0 :       program_account_state,
     484           0 :       program_account.data,
     485           0 :       program_account.data_len ) ) ) {
     486           0 :     fd_accdb_unread_one( accdb, &program_account );
     487           0 :     return;
     488           0 :   }
     489           0 :   if( program_account_state->discriminant!=FD_BPF_STATE_PROGRAM ) {
     490           0 :     fd_accdb_unread_one( accdb, &program_account );
     491           0 :     return;
     492           0 :   }
     493             : 
     494             :   /* Dump the programdata address */
     495           0 :   add_account_to_dumped_accounts( dumped_accounts, &program_account_state->inner.program.programdata_address );
     496           0 :   fd_accdb_unread_one( accdb, &program_account );
     497           0 : }
     498             : 
     499             : /* add_lut_account_to_dumped_accounts adds an address lookup table
     500             :    account AND all pubkeys in the lookup table to the dumped accounts
     501             :    set if they do not exist already. */
     502             : static void
     503             : add_lut_accounts_to_dumped_accounts( fd_accdb_t *                accdb,
     504             :                                      fd_accdb_fork_id_t          fork_id,
     505             :                                      fd_dump_account_key_set_t * dumped_accounts,
     506           0 :                                      fd_pubkey_t const *         pubkey ) {
     507             :   /* Add the current account to the dumped accounts set. */
     508           0 :   add_account_to_dumped_accounts( dumped_accounts, pubkey );
     509             : 
     510             :   /* Read the account and dump all pubkeys within the lookup table. */
     511           0 :   fd_acc_t lut_account = fd_accdb_read_one( accdb, fork_id, pubkey->uc );
     512           0 :   if( FD_UNLIKELY( !lut_account.lamports ) ) {
     513           0 :     fd_accdb_unread_one( accdb, &lut_account );
     514           0 :     return;
     515           0 :   }
     516             : 
     517             :   /* Decode the ALUT account and dump all pubkeys within the lookup
     518             :      table. */
     519           0 :   if( lut_account.data_len<FD_LOOKUP_TABLE_META_SIZE || (lut_account.data_len&0x1fUL) ) {
     520           0 :     fd_accdb_unread_one( accdb, &lut_account );
     521           0 :     return;
     522           0 :   }
     523             : 
     524             :   /* Copy the addresses out and release before nested acquires. */
     525           0 :   fd_pubkey_t lookup_addrs[ 256 ]; /* LOOKUP_TABLE_MAX_ADDRESSES */
     526           0 :   ulong       lookup_addrs_cnt = fd_ulong_min( (lut_account.data_len-FD_LOOKUP_TABLE_META_SIZE)>>5UL, 256UL );
     527           0 :   fd_memcpy( lookup_addrs, lut_account.data+FD_LOOKUP_TABLE_META_SIZE, lookup_addrs_cnt*sizeof(fd_pubkey_t) );
     528           0 :   fd_accdb_unread_one( accdb, &lut_account );
     529             : 
     530           0 :   for( ulong i=0UL; i<lookup_addrs_cnt; i++ ) {
     531           0 :     fd_pubkey_t const * referenced_pubkey = &lookup_addrs[i];
     532           0 :     add_account_and_programdata_to_dumped_accounts( accdb, fork_id, dumped_accounts, referenced_pubkey );
     533           0 :   }
     534           0 : }
     535             : 
     536             : static void
     537             : create_block_context_protobuf_from_block( fd_block_dump_ctx_t * dump_ctx,
     538             :                                           fd_banks_t *          banks,
     539             :                                           fd_bank_t *           bank,
     540             :                                           fd_accdb_t *          accdb,
     541           0 :                                           fd_runtime_stack_t *  runtime_stack ) {
     542             :   /* We should use the bank fields from the parent slot in order to
     543             :      capture the block context from before the current block was
     544             :      executed, since dumping is happening in the block finalize step. */
     545           0 :   fd_bank_t * parent_bank = fd_banks_get_parent( banks, bank );
     546           0 :   fd_exec_test_block_context_t * block_context  = &dump_ctx->block_context;
     547           0 :   ulong                          dump_txn_count = dump_ctx->txns_to_dump_cnt;
     548           0 :   fd_spad_t *                    spad           = dump_ctx->spad;
     549             : 
     550             :   /* Get vote and stake delegation infos */
     551           0 :   fd_vote_stakes_t * vote_stakes        = fd_bank_vote_stakes( parent_bank );
     552           0 :   ulong              vote_account_t_cnt = fd_vote_stakes_ele_cnt( vote_stakes, parent_bank->vote_stakes_fork_id );
     553             : 
     554           0 :   fd_stake_delegations_t const * stake_delegations = fd_bank_stake_delegations_frontier_query( banks, parent_bank );
     555             : 
     556             :   /* Collect account states in a temporary set before iterating over
     557             :      them and dumping them out. */
     558           0 :   void *              dumped_accounts_mem = fd_spad_alloc( spad, fd_dump_account_key_map_align(), fd_dump_account_key_map_footprint() );
     559           0 :   fd_dump_account_t * dumped_accounts_map = fd_dump_account_key_map_join( fd_dump_account_key_map_new( dumped_accounts_mem ) );
     560           0 :   fd_dump_account_key_set_t   dumped_accounts[1] = {{ dumped_accounts_map, 0UL, 0 }};
     561             : 
     562             :   /* BlockContext -> txns */
     563           0 :   block_context->txns_count = (pb_size_t)dump_txn_count;
     564           0 :   block_context->txns       = fd_spad_alloc( spad, alignof(fd_exec_test_sanitized_transaction_t), dump_ctx->txns_to_dump_cnt * sizeof(fd_exec_test_sanitized_transaction_t) );
     565           0 :   fd_memset( block_context->txns, 0, dump_ctx->txns_to_dump_cnt * sizeof(fd_exec_test_sanitized_transaction_t) );
     566             : 
     567             :   /* Dump sanitized transactions from the transaction descriptors */
     568           0 :   for( ulong i=0UL; i<dump_ctx->txns_to_dump_cnt; i++ ) {
     569           0 :     fd_txn_p_t const * txn_ptr        = &dump_ctx->txns_to_dump[i];
     570           0 :     fd_txn_t const *   txn_descriptor = TXN( txn_ptr );
     571           0 :     dump_sanitized_transaction( accdb, parent_bank->accdb_fork_id, txn_descriptor, txn_ptr->payload, spad, &block_context->txns[i] );
     572             : 
     573             :     /* Dump account + alut + programdata accounts (if applicable).
     574             :        1. Dump the raw txn account keys
     575             :        2. Dump the ALUT accounts
     576             :        3. Dump all referenced accounts in the ALUTs
     577             :        4. Dump any executable accounts */
     578             : 
     579             :     // 1 + 4. Dump any account keys that are referenced by transactions
     580             :     // + any programdata accounts (if applicable).
     581           0 :     fd_acct_addr_t const * account_keys = fd_txn_get_acct_addrs( txn_descriptor, txn_ptr->payload );
     582           0 :     for( ushort l=0; l<txn_descriptor->acct_addr_cnt; l++ ) {
     583           0 :       fd_pubkey_t const * account_key = fd_type_pun_const( &account_keys[l] );
     584           0 :       add_account_and_programdata_to_dumped_accounts( accdb, parent_bank->accdb_fork_id, dumped_accounts, account_key );
     585           0 :     }
     586             : 
     587             :     // 2 + 3 + 4. Dump any ALUT accounts + any accounts referenced in
     588             :     // the ALUTs + any programdata accounts (if applicable).
     589           0 :     fd_txn_acct_addr_lut_t const * txn_lookup_tables = fd_txn_get_address_tables_const( txn_descriptor );
     590           0 :     for( ushort l=0; l<txn_descriptor->addr_table_lookup_cnt; l++ ) {
     591           0 :       fd_txn_acct_addr_lut_t const * lookup_table = &txn_lookup_tables[l];
     592           0 :       fd_pubkey_t const *            lut_key      = fd_type_pun_const( txn_ptr->payload+lookup_table->addr_off );
     593           0 :       add_lut_accounts_to_dumped_accounts( accdb, parent_bank->accdb_fork_id, dumped_accounts, lut_key );
     594           0 :     }
     595           0 :   }
     596             : 
     597             :   /* Dump sysvars */
     598           0 :   for( ulong i=0UL; i<num_sysvar_entries; i++ ) {
     599           0 :     add_account_to_dumped_accounts( dumped_accounts, fd_dump_sysvar_ids[i] );
     600           0 :   }
     601             : 
     602             :   /* Dump builtins */
     603           0 :   for( ulong i=0UL; i<num_loaded_builtins; i++ ) {
     604           0 :     add_account_to_dumped_accounts( dumped_accounts, fd_dump_builtin_ids[i] );
     605           0 :   }
     606             : 
     607             :   /* Dump stake accounts for this epoch */
     608           0 :   fd_stake_delegations_iter_t iter_[1];
     609           0 :   for( fd_stake_delegations_iter_t * iter = fd_stake_delegations_iter_init( iter_, stake_delegations );
     610           0 :        !fd_stake_delegations_iter_done( iter );
     611           0 :        fd_stake_delegations_iter_next( iter ) ) {
     612           0 :     fd_stake_delegation_t const * stake_delegation = fd_stake_delegations_iter_ele( iter );
     613           0 :     add_account_to_dumped_accounts( dumped_accounts, &stake_delegation->stake_account );
     614           0 :   }
     615           0 :   fd_bank_stake_delegations_end_frontier_query( banks, parent_bank );
     616             : 
     617             : 
     618           0 :   ushort fork_idx = parent_bank->vote_stakes_fork_id;
     619           0 :   uchar __attribute__((aligned(FD_VOTE_STAKES_ITER_ALIGN))) iter_mem[ FD_VOTE_STAKES_ITER_FOOTPRINT ];
     620           0 :   for( fd_vote_stakes_iter_t * iter = fd_vote_stakes_fork_iter_init( vote_stakes, fork_idx, iter_mem );
     621           0 :        !fd_vote_stakes_fork_iter_done( vote_stakes, fork_idx, iter );
     622           0 :        fd_vote_stakes_fork_iter_next( vote_stakes, fork_idx, iter ) ) {
     623           0 :     fd_pubkey_t pubkey;
     624           0 :     fd_vote_stakes_fork_iter_ele( vote_stakes, fork_idx, iter, &pubkey, NULL, NULL, NULL, NULL, NULL, NULL );
     625           0 :     add_account_to_dumped_accounts( dumped_accounts, &pubkey );
     626           0 :   }
     627           0 :   fd_vote_stakes_fork_iter_fini( vote_stakes );
     628             : 
     629             :   /* BlockBank -> vote_accounts_t_1 and vote_accounts_t_2 */
     630           0 :   fd_exec_test_prev_vote_account_t * va_t1 = fd_spad_alloc( spad,
     631           0 :       alignof(fd_exec_test_prev_vote_account_t),
     632           0 :       vote_account_t_cnt * sizeof(fd_exec_test_prev_vote_account_t) );
     633           0 :   fd_exec_test_prev_vote_account_t * va_t2 = fd_spad_alloc( spad,
     634           0 :       alignof(fd_exec_test_prev_vote_account_t),
     635           0 :       vote_account_t_cnt * sizeof(fd_exec_test_prev_vote_account_t) );
     636           0 :   pb_size_t va_t1_cnt = 0U;
     637           0 :   pb_size_t va_t2_cnt = 0U;
     638             : 
     639           0 :   for( fd_vote_stakes_iter_t * iter = fd_vote_stakes_fork_iter_init( vote_stakes, fork_idx, iter_mem );
     640           0 :        !fd_vote_stakes_fork_iter_done( vote_stakes, fork_idx, iter );
     641           0 :        fd_vote_stakes_fork_iter_next( vote_stakes, fork_idx, iter ) ) {
     642           0 :     fd_pubkey_t pubkey;
     643           0 :     ulong       stake_t_1;
     644           0 :     ulong       stake_t_2;
     645           0 :     fd_pubkey_t node_t_1;
     646           0 :     fd_pubkey_t node_t_2;
     647           0 :     ushort      commission_t_1;
     648           0 :     ushort      commission_t_2;
     649           0 :     fd_vote_stakes_fork_iter_ele( vote_stakes, fork_idx, iter, &pubkey, &stake_t_1, &stake_t_2, &node_t_1, &node_t_2, &commission_t_1, &commission_t_2 );
     650             : 
     651           0 :     if( stake_t_1 ) {
     652           0 :       fd_exec_test_prev_vote_account_t * acc = &va_t1[ va_t1_cnt++ ];
     653           0 :       fd_memcpy( acc->address,     &pubkey, sizeof(fd_pubkey_t) );
     654           0 :       fd_memcpy( acc->node_pubkey, &node_t_1, sizeof(fd_pubkey_t) );
     655           0 :       acc->stake               = stake_t_1;
     656           0 :       acc->commission_bps      = commission_t_1;
     657           0 :       acc->version             = FD_EXEC_TEST_VOTE_ACCOUNT_VERSION_V3;
     658           0 :       acc->epoch_credits_count = 0U;
     659           0 :     }
     660             : 
     661           0 :     if( stake_t_2 ) {
     662           0 :       fd_exec_test_prev_vote_account_t * acc = &va_t2[ va_t2_cnt++ ];
     663           0 :       fd_memcpy( acc->address,     &pubkey, sizeof(fd_pubkey_t) );
     664           0 :       fd_memcpy( acc->node_pubkey, &node_t_2, sizeof(fd_pubkey_t) );
     665           0 :       acc->stake               = stake_t_2;
     666           0 :       acc->commission_bps      = commission_t_2;
     667           0 :       acc->version             = FD_EXEC_TEST_VOTE_ACCOUNT_VERSION_V3;
     668           0 :       acc->epoch_credits_count = 0U;
     669           0 :     }
     670           0 :   }
     671           0 :   fd_vote_stakes_fork_iter_fini( vote_stakes );
     672             : 
     673             :   /* Dump epoch_credits from runtime_stack->stakes.vote_ele if the
     674             :      vote_ele_map has been populated (happens after epoch boundary
     675             :      reward calculation).  Needed for the harness to correctly
     676             :      recalculate partitioned epoch rewards. */
     677           0 :   fd_vote_rewards_map_t * vote_ele_map = runtime_stack->stakes.vote_map;
     678           0 :   for( pb_size_t i=0U; i<va_t1_cnt; i++ ) {
     679           0 :     fd_pubkey_t va_pubkey = FD_LOAD( fd_pubkey_t, va_t1[i].address );
     680           0 :     uint idx = (uint)fd_vote_rewards_map_idx_query( vote_ele_map, &va_pubkey, UINT_MAX, runtime_stack->stakes.vote_ele );
     681           0 :     if( idx==UINT_MAX ) continue;
     682           0 :     fd_epoch_credits_t * ec = &fd_bank_epoch_credits( parent_bank )[idx];
     683           0 :     ulong cnt  = ec->cnt;
     684           0 :     ulong base = ec->base_credits;
     685           0 :     va_t1[i].epoch_credits_count = (pb_size_t)cnt;
     686           0 :     va_t1[i].epoch_credits = fd_spad_alloc( spad, alignof(fd_exec_test_epoch_credit_t), cnt * sizeof(fd_exec_test_epoch_credit_t) );
     687           0 :     for( ulong j=0; j<cnt; j++ ) {
     688           0 :       va_t1[i].epoch_credits[j].epoch        = ec->epoch[j];
     689           0 :       va_t1[i].epoch_credits[j].credits      = base + ec->credits_delta[j];
     690           0 :       va_t1[i].epoch_credits[j].prev_credits = base + ec->prev_credits_delta[j];
     691           0 :     }
     692           0 :   }
     693             : 
     694             :   /* BlockContext -> acct_states
     695             :      Iterate over the set and dump all the account keys in one pass. */
     696           0 :   block_context->acct_states_count = 0U;
     697           0 :   block_context->acct_states       = fd_spad_alloc(
     698           0 :       spad,
     699           0 :       alignof(fd_exec_test_acct_state_t),
     700           0 :       dumped_accounts->cnt*sizeof(fd_exec_test_acct_state_t) );
     701           0 :   fd_dump_account_key_iter_t dumped_accounts_iter_[1];
     702           0 :   for( fd_dump_account_key_iter_t * iter = fd_dump_account_key_iter_init( dumped_accounts_iter_, dumped_accounts );
     703           0 :        !fd_dump_account_key_iter_done( iter );
     704           0 :        fd_dump_account_key_iter_next( iter ) ) {
     705           0 :     fd_pubkey_t const * pubkey = fd_dump_account_key_iter_ele( iter );
     706           0 :     fd_acc_t acc = fd_accdb_read_one( accdb, parent_bank->accdb_fork_id, pubkey->uc );
     707           0 :     if( FD_UNLIKELY( !acc.lamports ) ) {
     708           0 :       fd_accdb_unread_one( accdb, &acc );
     709           0 :       continue;
     710           0 :     }
     711           0 :     dump_account_state(
     712           0 :         &acc,
     713           0 :         &block_context->acct_states[block_context->acct_states_count++],
     714           0 :         spad );
     715           0 :     fd_accdb_unread_one( accdb, &acc );
     716           0 :   }
     717             : 
     718             :   /* BlockContext -> bank */
     719           0 :   block_context->has_bank = true;
     720           0 :   fd_exec_test_block_bank_t * block_bank = &block_context->bank;
     721             : 
     722             :   /* BlockBank -> blockhash_queue */
     723           0 :   dump_blockhash_queue( parent_bank, spad, &block_bank->blockhash_queue, &block_bank->blockhash_queue_count );
     724             : 
     725             :   /* BlockBank -> rbh_lamports_per_signature */
     726           0 :   block_bank->rbh_lamports_per_signature = (uint)parent_bank->f.rbh_lamports_per_sig;
     727             : 
     728             :   /* BlockBank -> fee_rate_governor */
     729           0 :   block_bank->has_fee_rate_governor = true;
     730           0 :   dump_fee_rate_governor( parent_bank, &block_bank->fee_rate_governor );
     731             : 
     732             :   /* BlockBank -> slot */
     733           0 :   block_bank->slot = bank->f.slot;
     734             : 
     735             :   /* BlockBank -> parent_slot */
     736           0 :   block_bank->parent_slot = bank->f.parent_slot;
     737             : 
     738             :   /* BlockBank -> capitalization */
     739           0 :   block_bank->capitalization = parent_bank->f.capitalization;
     740             : 
     741             :   /* BlockBank -> ns_per_slot */
     742           0 :   fd_w_u128_t ns_per_slot = bank->f.ns_per_slot;
     743           0 :   fd_memcpy( block_bank->ns_per_slot, &ns_per_slot.ud, sizeof(uint128) );
     744             : 
     745             :   /* BlockBank -> inflation */
     746           0 :   block_bank->has_inflation = true;
     747           0 :   fd_inflation_t const * inflation = &parent_bank->f.inflation;
     748           0 :   block_bank->inflation = (fd_exec_test_inflation_t){
     749           0 :     .initial         = inflation->initial,
     750           0 :     .terminal        = inflation->terminal,
     751           0 :     .taper           = inflation->taper,
     752           0 :     .foundation      = inflation->foundation,
     753           0 :     .foundation_term = inflation->foundation_term,
     754           0 :   };
     755             : 
     756             :   /* BlockBank -> block_height */
     757           0 :   block_bank->block_height = bank->f.block_height;
     758             : 
     759             :   /* BlockBank -> poh */
     760           0 :   fd_memcpy( block_bank->poh, &bank->f.poh, sizeof(fd_hash_t) );
     761             : 
     762             :   /* BlockBank -> parent_bank_hash */
     763           0 :   fd_memcpy( block_bank->parent_bank_hash, &parent_bank->f.bank_hash, sizeof(fd_hash_t) );
     764             : 
     765             :   /* BlockBank -> parent_lt_hash */
     766           0 :   fd_lthash_value_t const * parent_lthash = fd_bank_lthash_locking_query( parent_bank );
     767           0 :   fd_memcpy( block_bank->parent_lt_hash, parent_lthash, sizeof(fd_lthash_value_t) );
     768           0 :   fd_bank_lthash_end_locking_query( parent_bank );
     769             : 
     770             :   /* BlockBank -> parent_signature_count */
     771           0 :   block_bank->parent_signature_count = parent_bank->f.parent_signature_cnt;
     772             : 
     773             :   /* BlockBank -> features */
     774           0 :   block_bank->has_features = true;
     775           0 :   dump_sorted_features( &parent_bank->f.features, &block_bank->features, spad );
     776             : 
     777             :   /* BlockBank -> vote_accounts_t_1 / vote_accounts_t_2 */
     778           0 :   block_bank->vote_accounts_t_1       = va_t1;
     779           0 :   block_bank->vote_accounts_t_1_count = va_t1_cnt;
     780           0 :   block_bank->vote_accounts_t_2       = va_t2;
     781           0 :   block_bank->vote_accounts_t_2_count = va_t2_cnt;
     782           0 : }
     783             : 
     784             : static void
     785             : create_txn_context_protobuf_from_txn( fd_exec_test_txn_context_t * txn_context_msg,
     786             :                                       fd_runtime_t *               runtime,
     787             :                                       fd_bank_t *                  bank,
     788             :                                       fd_txn_in_t const *          txn_in,
     789             :                                       fd_txn_out_t *               txn_out,
     790           0 :                                       fd_spad_t *                  spad ) {
     791           0 :   fd_txn_t const * txn_descriptor = TXN( txn_in->txn );
     792           0 :   uchar const *    txn_payload    = (uchar const *) txn_in->txn->payload;
     793           0 :   (void)txn_out;
     794             : 
     795             :   /* Transaction Context -> account_shared_data
     796             :      Contains:
     797             :       - Account data for regular accounts
     798             :       - Account data for LUT accounts
     799             :       - Account data for executable accounts
     800             :       - Account data for (almost) all sysvars */
     801           0 :   txn_context_msg->account_shared_data_count = 0;
     802           0 :   txn_context_msg->account_shared_data = fd_spad_alloc( spad,
     803           0 :                                                         alignof(fd_exec_test_acct_state_t),
     804           0 :                                                         (256UL*2UL + txn_descriptor->addr_table_lookup_cnt + num_sysvar_entries) * sizeof(fd_exec_test_acct_state_t) );
     805             : 
     806             :   /* Dump regular accounts first */
     807           0 :   fd_acct_addr_t const * account_keys = fd_txn_get_acct_addrs( txn_descriptor, txn_payload );
     808           0 :   for( ushort i=0; i<txn_descriptor->acct_addr_cnt; i++ ) {
     809           0 :     dump_account_if_not_already_dumped(
     810           0 :       runtime->accdb,
     811           0 :       bank->accdb_fork_id,
     812           0 :       fd_type_pun_const( &account_keys[i] ),
     813           0 :       spad,
     814           0 :       txn_context_msg->account_shared_data,
     815           0 :       &txn_context_msg->account_shared_data_count,
     816           0 :       NULL
     817           0 :     );
     818           0 :   }
     819             : 
     820             :   // Dump LUT accounts
     821           0 :   fd_txn_acct_addr_lut_t const * address_lookup_tables = fd_txn_get_address_tables_const( txn_descriptor );
     822           0 :   for( ulong i = 0; i < txn_descriptor->addr_table_lookup_cnt; ++i ) {
     823           0 :     fd_txn_acct_addr_lut_t const * addr_lut  = &address_lookup_tables[i];
     824           0 :     fd_pubkey_t * alut_key = (fd_pubkey_t *) (txn_payload + addr_lut->addr_off);
     825             : 
     826             :     // Dump the LUT account itself if not already dumped
     827           0 :     fd_acc_t acc;
     828           0 :     int ret = dump_account_if_not_already_dumped(
     829           0 :         runtime->accdb,
     830           0 :         bank->accdb_fork_id,
     831           0 :         alut_key,
     832           0 :         spad,
     833           0 :         txn_context_msg->account_shared_data,
     834           0 :         &txn_context_msg->account_shared_data_count,
     835           0 :         &acc
     836           0 :     );
     837           0 :     if( FD_UNLIKELY( ret ) ) continue;
     838             : 
     839           0 :     if( FD_UNLIKELY( acc.data_len < FD_LOOKUP_TABLE_META_SIZE ) ) {
     840             :       /* Skip over invalid address lookup tables */
     841           0 :       fd_accdb_unread_one( runtime->accdb, &acc );
     842           0 :       continue;
     843           0 :     }
     844             : 
     845             :     /* Copy the addresses out and release before nested acquires. */
     846           0 :     fd_pubkey_t lookup_addrs[ 256 ]; /* LOOKUP_TABLE_MAX_ADDRESSES */
     847           0 :     ulong       lookup_addrs_cnt = fd_ulong_min( (acc.data_len - FD_LOOKUP_TABLE_META_SIZE) / sizeof(fd_pubkey_t), 256UL );
     848           0 :     fd_memcpy( lookup_addrs, acc.data + FD_LOOKUP_TABLE_META_SIZE, lookup_addrs_cnt*sizeof(fd_pubkey_t) );
     849           0 :     fd_accdb_unread_one( runtime->accdb, &acc );
     850             : 
     851             :     /* Dump any account state refererenced in ALUTs */
     852           0 :     uchar const * writable_lut_idxs = txn_payload + addr_lut->writable_off;
     853           0 :     for( ulong j=0; j<addr_lut->writable_cnt; j++ ) {
     854           0 :       if( writable_lut_idxs[j] >= lookup_addrs_cnt ) {
     855           0 :         continue;
     856           0 :       }
     857           0 :       fd_pubkey_t const * referenced_addr = lookup_addrs + writable_lut_idxs[j];
     858           0 :       dump_account_if_not_already_dumped(
     859           0 :           runtime->accdb,
     860           0 :           bank->accdb_fork_id,
     861           0 :           referenced_addr,
     862           0 :           spad,
     863           0 :           txn_context_msg->account_shared_data,
     864           0 :           &txn_context_msg->account_shared_data_count,
     865           0 :           NULL
     866           0 :       );
     867           0 :     }
     868             : 
     869           0 :     uchar const * readonly_lut_idxs = txn_payload + addr_lut->readonly_off;
     870           0 :     for( ulong j = 0; j < addr_lut->readonly_cnt; j++ ) {
     871           0 :       if( readonly_lut_idxs[j] >= lookup_addrs_cnt ) {
     872           0 :         continue;
     873           0 :       }
     874           0 :       fd_pubkey_t const * referenced_addr = lookup_addrs + readonly_lut_idxs[j];
     875           0 :       dump_account_if_not_already_dumped(
     876           0 :           runtime->accdb,
     877           0 :           bank->accdb_fork_id,
     878           0 :           referenced_addr,
     879           0 :           spad,
     880           0 :           txn_context_msg->account_shared_data,
     881           0 :           &txn_context_msg->account_shared_data_count,
     882           0 :           NULL
     883           0 :       );
     884           0 :     }
     885           0 :   }
     886             : 
     887             :   /* Dump the programdata accounts for any potential v3-owned program accounts */
     888           0 :   uint accounts_dumped_so_far = txn_context_msg->account_shared_data_count;
     889           0 :   for( uint i=0U; i<accounts_dumped_so_far; i++ ) {
     890           0 :     fd_exec_test_acct_state_t const * maybe_program_account = &txn_context_msg->account_shared_data[i];
     891           0 :     dump_executable_account_if_exists( runtime->accdb, bank->accdb_fork_id, maybe_program_account, spad, txn_context_msg->account_shared_data, &txn_context_msg->account_shared_data_count );
     892           0 :   }
     893             : 
     894             :   /* Dump sysvars */
     895           0 :   for( ulong i = 0; i < num_sysvar_entries; i++ ) {
     896           0 :     dump_account_if_not_already_dumped(
     897           0 :         runtime->accdb,
     898           0 :         bank->accdb_fork_id,
     899           0 :         fd_dump_sysvar_ids[i],
     900           0 :         spad,
     901           0 :         txn_context_msg->account_shared_data,
     902           0 :         &txn_context_msg->account_shared_data_count,
     903           0 :         NULL
     904           0 :     );
     905           0 :   }
     906             : 
     907             :   /* Transaction Context -> tx */
     908           0 :   txn_context_msg->has_tx = true;
     909           0 :   fd_exec_test_sanitized_transaction_t * sanitized_transaction = &txn_context_msg->tx;
     910           0 :   dump_sanitized_transaction( runtime->accdb, bank->accdb_fork_id, txn_descriptor, txn_payload, spad, sanitized_transaction );
     911             : 
     912             :   /* Transaction Context -> bank */
     913           0 :   dump_txn_bank( bank, spad, txn_context_msg );
     914           0 : }
     915             : 
     916             : static void
     917             : create_instr_context_protobuf_from_instructions( fd_exec_test_instr_context_t * instr_context,
     918             :                                                  fd_runtime_t *                 runtime,
     919             :                                                  fd_bank_t *                    bank,
     920             :                                                  fd_txn_out_t *                 txn_out,
     921             :                                                  fd_instr_info_t const *        instr,
     922           0 :                                                  fd_spad_t *                    spad ) {
     923             :   /* Program ID */
     924           0 :   fd_memcpy( instr_context->program_id, txn_out->accounts.keys[ instr->program_id ].uc, sizeof(fd_pubkey_t) );
     925             : 
     926             :   /* Accounts */
     927           0 :   instr_context->accounts_count = (pb_size_t) txn_out->accounts.cnt;
     928           0 :   instr_context->accounts = fd_spad_alloc( spad, alignof(fd_exec_test_acct_state_t), (instr_context->accounts_count + num_sysvar_entries + txn_out->accounts.executable_cnt) * sizeof(fd_exec_test_acct_state_t));
     929           0 :   for( ulong i = 0; i < txn_out->accounts.cnt; i++ ) {
     930             :     // Copy account information over
     931           0 :     fd_exec_test_acct_state_t * output_account = &instr_context->accounts[i];
     932           0 :     dump_account_state( txn_out->accounts.account[ i ], output_account, spad );
     933           0 :   }
     934             : 
     935             :   /* Add sysvar cache variables */
     936           0 :   uchar * sysvar_data = fd_spad_alloc( spad, 1UL, FD_RUNTIME_ACC_SZ_MAX );
     937           0 :   for( ulong i = 0; i < num_sysvar_entries; i++ ) {
     938           0 :     if( account_already_dumped( instr_context->accounts, instr_context->accounts_count, fd_dump_sysvar_ids[i] ) ) continue;
     939             : 
     940           0 :     fd_acc_t acc = {0};
     941           0 :     fd_memcpy( acc.pubkey, fd_dump_sysvar_ids[i]->uc, sizeof(fd_pubkey_t) );
     942           0 :     acc.data = sysvar_data;
     943           0 :     fd_accdb_read_one_nocache(
     944           0 :       runtime->accdb,
     945           0 :       bank->accdb_fork_id,
     946           0 :       fd_dump_sysvar_ids[i]->uc,
     947           0 :       &acc.lamports,
     948           0 :       &acc.executable,
     949           0 :       acc.owner,
     950           0 :       acc.data,
     951           0 :       &acc.data_len
     952           0 :     );
     953           0 :     if( FD_UNLIKELY( !acc.lamports ) ) continue;
     954             : 
     955           0 :     fd_exec_test_acct_state_t * output_account = &instr_context->accounts[instr_context->accounts_count++];
     956           0 :     dump_account_state( &acc, output_account, spad );
     957           0 :   }
     958             : 
     959             :   /* Add executable accounts */
     960           0 :   for( ulong i = 0; i < txn_out->accounts.executable_cnt; i++ ) {
     961             :     // Make sure the account doesn't exist in the output accounts yet
     962           0 :     fd_acc_t const * ro = txn_out->accounts.executable[i];
     963           0 :     bool account_exists = false;
     964           0 :     for( ulong j = 0; j < instr_context->accounts_count; j++ ) {
     965           0 :       if( 0 == memcmp( instr_context->accounts[j].address, ro->pubkey, sizeof(fd_pubkey_t) ) ) {
     966           0 :         account_exists = true;
     967           0 :         break;
     968           0 :       }
     969           0 :     }
     970             :     // Copy it into output
     971           0 :     if( !account_exists ) {
     972           0 :       fd_exec_test_acct_state_t * output_account = &instr_context->accounts[instr_context->accounts_count++];
     973           0 :       dump_account_state( ro, output_account, spad );
     974           0 :     }
     975           0 :   }
     976             : 
     977             :   /* Instruction Accounts */
     978           0 :   instr_context->instr_accounts_count = (pb_size_t) instr->acct_cnt;
     979           0 :   instr_context->instr_accounts = fd_spad_alloc( spad, alignof(fd_exec_test_instr_acct_t), instr_context->instr_accounts_count * sizeof(fd_exec_test_instr_acct_t) );
     980           0 :   for( ushort i = 0; i < instr->acct_cnt; i++ ) {
     981           0 :     fd_exec_test_instr_acct_t * output_instr_account = &instr_context->instr_accounts[i];
     982             : 
     983           0 :     output_instr_account->index       = instr->accounts[i].index_in_transaction;
     984           0 :     output_instr_account->is_writable = instr->accounts[i].is_writable;
     985           0 :     output_instr_account->is_signer   = instr->accounts[i].is_signer;
     986           0 :   }
     987             : 
     988             :   /* Data */
     989           0 :   instr_context->data = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE( instr->data_sz ) );
     990           0 :   instr_context->data->size = (pb_size_t) instr->data_sz;
     991           0 :   fd_memcpy( instr_context->data->bytes, instr->data, instr->data_sz );
     992             : 
     993             :   /* Compute Units */
     994           0 :   instr_context->cu_avail = txn_out->details.compute_budget.compute_meter;
     995             : 
     996             :   /* Feature set */
     997           0 :   instr_context->has_features = true;
     998           0 :   dump_sorted_features( &bank->f.features, &instr_context->features, spad );
     999           0 : }
    1000             : 
    1001             : /***** PUBLIC APIs *****/
    1002             : 
    1003             : void
    1004             : fd_dump_instr_to_protobuf( fd_runtime_t *      runtime,
    1005             :                            fd_bank_t *         bank,
    1006             :                            fd_txn_in_t const * txn_in,
    1007             :                            fd_txn_out_t *      txn_out,
    1008             :                            fd_instr_info_t *   instr,
    1009           0 :                            ushort              instruction_idx ) {
    1010             :   /* Check program ID filter, if it exists */
    1011           0 :   if( runtime->log.dump_proto_ctx->has_dump_instr_program_id_filter &&
    1012           0 :       memcmp( txn_out->accounts.keys[ instr->program_id ].uc, runtime->log.dump_proto_ctx->dump_instr_program_id_filter, sizeof(fd_pubkey_t) ) ) {
    1013           0 :     return;
    1014           0 :   }
    1015             : 
    1016           0 :   fd_spad_t * spad = fd_spad_join( fd_spad_new( runtime->log.dumping_mem, 1UL<<28UL ) );
    1017             : 
    1018           0 :   FD_SPAD_FRAME_BEGIN( spad ) {
    1019             :     // Get base58-encoded tx signature
    1020           0 :     const fd_ed25519_sig_t * signatures = fd_txn_get_signatures( TXN( txn_in->txn ), txn_in->txn->payload );
    1021           0 :     char encoded_signature[FD_BASE58_ENCODED_64_SZ];
    1022           0 :     fd_base58_encode_64( signatures[0], NULL, encoded_signature );
    1023             : 
    1024           0 :     fd_exec_test_instr_context_t instr_context = FD_EXEC_TEST_INSTR_CONTEXT_INIT_DEFAULT;
    1025           0 :     create_instr_context_protobuf_from_instructions( &instr_context, runtime, bank, txn_out, instr, spad );
    1026             : 
    1027             :     /* Output to file */
    1028           0 :     ulong        out_buf_size = 100 * 1024 * 1024;
    1029           0 :     uint8_t *    out          = fd_spad_alloc( spad, alignof(uchar) , out_buf_size );
    1030           0 :     pb_ostream_t stream       = pb_ostream_from_buffer( out, out_buf_size );
    1031           0 :     if (pb_encode(&stream, FD_EXEC_TEST_INSTR_CONTEXT_FIELDS, &instr_context)) {
    1032           0 :       char output_filepath[ PATH_MAX ];
    1033           0 :       snprintf( output_filepath, PATH_MAX, "%s/instr-%s-%hu.instrctx", runtime->log.dump_proto_ctx->dump_proto_output_dir, encoded_signature, instruction_idx );
    1034           0 :       FILE * file = fopen(output_filepath, "wb");
    1035           0 :       if( file ) {
    1036           0 :         fwrite( out, 1, stream.bytes_written, file );
    1037           0 :         fclose( file );
    1038           0 :       }
    1039           0 :     }
    1040           0 :   } FD_SPAD_FRAME_END;
    1041           0 : }
    1042             : 
    1043             : /* Writes a single account state into the resulting_state field of a
    1044             :    TxnResult protobuf.  Sub-allocations for account data are bump-
    1045             :    allocated from the caller's scratch region via _l. */
    1046             : static void
    1047             : write_account_to_result( fd_acc_t const *            acc,
    1048             :                          fd_exec_test_acct_state_t * out_accounts,
    1049             :                          pb_size_t *                 out_accounts_cnt,
    1050             :                          ulong *                     scratch_cur,
    1051           0 :                          ulong                       scratch_end ) {
    1052           0 :   fd_exec_test_acct_state_t * out_acct = &out_accounts[ *out_accounts_cnt ];
    1053           0 :   (*out_accounts_cnt)++;
    1054             : 
    1055           0 :   memset( out_acct, 0, sizeof(fd_exec_test_acct_state_t) );
    1056           0 :   memcpy( out_acct->address, acc->pubkey, sizeof(fd_pubkey_t) );
    1057           0 :   out_acct->lamports = acc->lamports;
    1058             : 
    1059           0 :   if( acc->data_len>0UL ) {
    1060           0 :     pb_bytes_array_t * data = (pb_bytes_array_t *)fd_ulong_align_up( *scratch_cur, alignof(pb_bytes_array_t) );
    1061           0 :     *scratch_cur = (ulong)data + PB_BYTES_ARRAY_T_ALLOCSIZE( acc->data_len );
    1062           0 :     if( FD_UNLIKELY( *scratch_cur > scratch_end ) ) abort();
    1063           0 :     data->size = (pb_size_t)acc->data_len;
    1064           0 :     fd_memcpy( data->bytes, acc->data, acc->data_len );
    1065           0 :     out_acct->data = data;
    1066           0 :   }
    1067             : 
    1068           0 :   out_acct->executable = acc->executable;
    1069           0 :   memcpy( out_acct->owner, acc->owner, sizeof(fd_pubkey_t) );
    1070           0 : }
    1071             : 
    1072             : static void
    1073             : write_account_to_result1( uchar const *               pubkey,
    1074             :                           ulong                       lamports,
    1075             :                           uchar const *               owner,
    1076             :                           ulong                       data_len,
    1077             :                           uchar const *               _data,
    1078             :                           int                         executable,
    1079             :                           fd_exec_test_acct_state_t * out_accounts,
    1080             :                           pb_size_t *                 out_accounts_cnt,
    1081             :                           ulong *                     scratch_cur,
    1082           0 :                           ulong                       scratch_end ) {
    1083           0 :   fd_exec_test_acct_state_t * out_acct = &out_accounts[ *out_accounts_cnt ];
    1084           0 :   (*out_accounts_cnt)++;
    1085             : 
    1086           0 :   memset( out_acct, 0, sizeof(fd_exec_test_acct_state_t) );
    1087           0 :   memcpy( out_acct->address, pubkey, sizeof(fd_pubkey_t) );
    1088           0 :   out_acct->lamports = lamports;
    1089             : 
    1090           0 :   if( data_len>0UL ) {
    1091           0 :     pb_bytes_array_t * data = (pb_bytes_array_t *)fd_ulong_align_up( *scratch_cur, alignof(pb_bytes_array_t) );
    1092           0 :     *scratch_cur = (ulong)data + PB_BYTES_ARRAY_T_ALLOCSIZE( data_len );
    1093           0 :     if( FD_UNLIKELY( *scratch_cur > scratch_end ) ) abort();
    1094           0 :     data->size = (pb_size_t)data_len;
    1095           0 :     fd_memcpy( data->bytes, _data, data_len );
    1096           0 :     out_acct->data = data;
    1097           0 :   }
    1098             : 
    1099           0 :   out_acct->executable = executable;
    1100           0 :   memcpy( out_acct->owner, owner, sizeof(fd_pubkey_t) );
    1101           0 : }
    1102             : 
    1103             : ulong
    1104             : create_txn_result_protobuf_from_txn( fd_exec_test_txn_result_t ** txn_result_out,
    1105             :                                      void *                       out_buf,
    1106             :                                      ulong                        out_bufsz,
    1107             :                                      fd_txn_in_t const *          txn_in,
    1108             :                                      fd_txn_out_t *               txn_out,
    1109           0 :                                      int                          exec_res ) {
    1110           0 :   FD_SCRATCH_ALLOC_INIT( l, out_buf );
    1111           0 :   ulong out_end = (ulong)out_buf + out_bufsz;
    1112             : 
    1113           0 :   fd_exec_test_txn_result_t * txn_result =
    1114           0 :     FD_SCRATCH_ALLOC_APPEND( l, alignof(fd_exec_test_txn_result_t),
    1115           0 :                                 sizeof(fd_exec_test_txn_result_t) );
    1116           0 :   if( FD_UNLIKELY( _l > out_end ) ) abort();
    1117           0 :   fd_memset( txn_result, 0, sizeof(fd_exec_test_txn_result_t) );
    1118             : 
    1119             :   /* Map nonce errors into the agave expected ones. */
    1120           0 :   if( FD_UNLIKELY( exec_res==FD_RUNTIME_TXN_ERR_BLOCKHASH_NONCE_ALREADY_ADVANCED ||
    1121           0 :                    exec_res==FD_RUNTIME_TXN_ERR_BLOCKHASH_FAIL_ADVANCE_NONCE_INSTR ||
    1122           0 :                    exec_res==FD_RUNTIME_TXN_ERR_BLOCKHASH_FAIL_WRONG_NONCE )) {
    1123           0 :     exec_res = FD_RUNTIME_TXN_ERR_BLOCKHASH_NOT_FOUND;
    1124           0 :   }
    1125             : 
    1126             :   /* Basic result fields */
    1127           0 :   txn_result->executed                  = txn_out->err.is_committable;
    1128           0 :   txn_result->sanitization_error        = !txn_out->err.is_committable;
    1129           0 :   txn_result->modified_accounts_count   = 0;
    1130           0 :   txn_result->rollback_accounts_count   = 0;
    1131           0 :   txn_result->is_ok                     = !exec_res;
    1132           0 :   txn_result->status                    = (uint32_t) -exec_res;
    1133           0 :   txn_result->instruction_error         = 0;
    1134           0 :   txn_result->instruction_error_index   = 0;
    1135           0 :   txn_result->custom_error              = 0;
    1136           0 :   txn_result->has_fee_details           = false;
    1137           0 :   txn_result->loaded_accounts_data_size = txn_out->details.loaded_accounts_data_size;
    1138             : 
    1139           0 :   if( txn_result->sanitization_error ) {
    1140           0 :     if( txn_out->err.is_fees_only ) {
    1141           0 :       txn_result->has_fee_details                = true;
    1142           0 :       txn_result->fee_details.prioritization_fee = txn_out->details.priority_fee;
    1143           0 :       txn_result->fee_details.transaction_fee    = txn_out->details.execution_fee;
    1144           0 :     }
    1145             : 
    1146           0 :     if( exec_res==FD_RUNTIME_TXN_ERR_INSTRUCTION_ERROR ) {
    1147           0 :       txn_result->instruction_error       = (uint32_t) -txn_out->err.exec_err;
    1148           0 :       txn_result->instruction_error_index = (uint32_t) txn_out->err.exec_err_idx;
    1149           0 :       if( txn_out->err.exec_err==FD_EXECUTOR_INSTR_ERR_CUSTOM_ERR ) {
    1150           0 :         txn_result->custom_error = txn_out->err.custom_err;
    1151           0 :       }
    1152           0 :     }
    1153             : 
    1154           0 :     *txn_result_out = txn_result;
    1155           0 :     return FD_SCRATCH_ALLOC_FINI( l, 1UL ) - (ulong)out_buf;
    1156           0 :   }
    1157             : 
    1158             :   /* Capture instruction error code for executed transactions */
    1159           0 :   if( exec_res==FD_RUNTIME_TXN_ERR_INSTRUCTION_ERROR ) {
    1160           0 :     fd_txn_t const * txn            = TXN( txn_in->txn );
    1161           0 :     int              instr_err_idx  = txn_out->err.exec_err_idx;
    1162           0 :     int              program_id_idx = txn->instr[instr_err_idx].program_id;
    1163             : 
    1164           0 :     txn_result->instruction_error       = (uint32_t) -txn_out->err.exec_err;
    1165           0 :     txn_result->instruction_error_index = (uint32_t) instr_err_idx;
    1166             : 
    1167           0 :     if( txn_out->err.exec_err==FD_EXECUTOR_INSTR_ERR_CUSTOM_ERR &&
    1168           0 :         fd_executor_lookup_native_precompile_program( &txn_out->accounts.keys[ program_id_idx ] )==NULL ) {
    1169           0 :       txn_result->custom_error = txn_out->err.custom_err;
    1170           0 :     }
    1171           0 :   }
    1172             : 
    1173           0 :   txn_result->has_fee_details                = true;
    1174           0 :   txn_result->fee_details.transaction_fee    = txn_out->details.execution_fee;
    1175           0 :   txn_result->fee_details.prioritization_fee = txn_out->details.priority_fee;
    1176           0 :   txn_result->executed_units                 = txn_out->details.compute_budget.compute_unit_limit - txn_out->details.compute_budget.compute_meter;
    1177             : 
    1178             :   /* Return data */
    1179           0 :   if( txn_out->details.return_data.len>0 ) {
    1180           0 :     txn_result->return_data = FD_SCRATCH_ALLOC_APPEND( l, alignof(pb_bytes_array_t),
    1181           0 :                                                        PB_BYTES_ARRAY_T_ALLOCSIZE( txn_out->details.return_data.len ) );
    1182           0 :     if( FD_UNLIKELY( _l > out_end ) ) abort();
    1183           0 :     txn_result->return_data->size = (pb_size_t)txn_out->details.return_data.len;
    1184           0 :     fd_memcpy( txn_result->return_data->bytes, txn_out->details.return_data.data, txn_out->details.return_data.len );
    1185           0 :   }
    1186             : 
    1187             :   /* Modified accounts */
    1188           0 :   txn_result->modified_accounts = FD_SCRATCH_ALLOC_APPEND( l, alignof(fd_exec_test_acct_state_t), sizeof(fd_exec_test_acct_state_t) * txn_out->accounts.cnt );
    1189           0 :   txn_result->rollback_accounts = FD_SCRATCH_ALLOC_APPEND( l, alignof(fd_exec_test_acct_state_t), sizeof(fd_exec_test_acct_state_t) * 2UL );
    1190           0 :   if( FD_UNLIKELY( _l > out_end ) ) abort();
    1191             : 
    1192           0 :   if( txn_out->err.is_fees_only || exec_res!=FD_RUNTIME_EXECUTE_SUCCESS ) {
    1193             :     /* If the transaction errored, capture the rollback accounts (fee payer and nonce). */
    1194           0 :     if( FD_LIKELY( txn_out->accounts.nonce_idx_in_txn!=FD_FEE_PAYER_TXN_IDX ) ) {
    1195           0 :       write_account_to_result1(
    1196           0 :         txn_out->accounts.keys[FD_FEE_PAYER_TXN_IDX].uc,
    1197           0 :         txn_out->accounts.fee_payer_rollback_lamports,
    1198           0 :         txn_out->accounts.account[ FD_FEE_PAYER_TXN_IDX ]->prior_owner,
    1199           0 :         txn_out->accounts.account[ FD_FEE_PAYER_TXN_IDX ]->prior_data_len,
    1200           0 :         txn_out->accounts.account[ FD_FEE_PAYER_TXN_IDX ]->prior_data,
    1201           0 :         txn_out->accounts.account[ FD_FEE_PAYER_TXN_IDX ]->prior_executable,
    1202           0 :         txn_result->rollback_accounts,
    1203           0 :         &txn_result->rollback_accounts_count,
    1204           0 :         &_l,
    1205           0 :         out_end
    1206           0 :       );
    1207           0 :     }
    1208             : 
    1209           0 :     if( txn_out->accounts.nonce_idx_in_txn!=ULONG_MAX ) {
    1210             :       /* When the nonce account is also the fee payer, the rollback must
    1211             :          reflect the fee debit (the fee is still charged on a failed
    1212             :          nonce txn).  prior_lamports is the pre-fee balance, so use the
    1213             :          post-fee fee_payer_rollback_lamports for that account instead. */
    1214           0 :       ulong nonce_rollback_lamports =
    1215           0 :         ( txn_out->accounts.nonce_idx_in_txn==FD_FEE_PAYER_TXN_IDX )
    1216           0 :           ? txn_out->accounts.fee_payer_rollback_lamports
    1217           0 :           : txn_out->accounts.account[ txn_out->accounts.nonce_idx_in_txn ]->prior_lamports;
    1218           0 :       write_account_to_result1(
    1219           0 :         txn_out->accounts.keys[txn_out->accounts.nonce_idx_in_txn].uc,
    1220           0 :         nonce_rollback_lamports,
    1221           0 :         txn_out->accounts.account[ txn_out->accounts.nonce_idx_in_txn ]->prior_owner,
    1222           0 :         txn_out->accounts.nonce_rollback_data_len,
    1223           0 :         txn_out->accounts.nonce_rollback_data,
    1224           0 :         txn_out->accounts.account[ txn_out->accounts.nonce_idx_in_txn ]->prior_executable,
    1225           0 :         txn_result->rollback_accounts,
    1226           0 :         &txn_result->rollback_accounts_count,
    1227           0 :         &_l,
    1228           0 :         out_end
    1229           0 :       );
    1230           0 :     }
    1231           0 :   }
    1232             : 
    1233           0 :   if( !txn_out->err.is_fees_only ) {
    1234             :     /* Executed: capture fee payer and writable accounts. */
    1235           0 :     for( ulong j=0UL; j<txn_out->accounts.cnt; j++ ) {
    1236           0 :       if( !( fd_runtime_account_is_writable_idx( txn_in, txn_out, (ushort)j ) ||
    1237           0 :              j==FD_FEE_PAYER_TXN_IDX ) ) {
    1238           0 :         continue;
    1239           0 :       }
    1240             : 
    1241           0 :       write_account_to_result(
    1242           0 :         txn_out->accounts.account[j],
    1243           0 :         txn_result->modified_accounts,
    1244           0 :         &txn_result->modified_accounts_count,
    1245           0 :         &_l,
    1246           0 :         out_end
    1247           0 :       );
    1248           0 :     }
    1249           0 :   }
    1250             : 
    1251           0 :   *txn_result_out = txn_result;
    1252           0 :   return FD_SCRATCH_ALLOC_FINI( l, 1UL ) - (ulong)out_buf;
    1253           0 : }
    1254             : 
    1255             : void
    1256             : fd_dump_txn_to_protobuf( fd_runtime_t *      runtime,
    1257             :                          fd_bank_t *         bank,
    1258             :                          fd_txn_in_t const * txn_in,
    1259           0 :                          fd_txn_out_t *      txn_out ) {
    1260           0 :   fd_spad_t * spad = fd_spad_join( fd_spad_new( runtime->log.dumping_mem, 1UL<<28UL ) );
    1261             : 
    1262           0 :   FD_SPAD_FRAME_BEGIN( spad ) {
    1263             :     // Get base58-encoded tx signature
    1264           0 :     const fd_ed25519_sig_t * signatures = fd_txn_get_signatures( TXN( txn_in->txn ), txn_in->txn->payload );
    1265           0 :     char encoded_signature[FD_BASE58_ENCODED_64_SZ];
    1266           0 :     fd_base58_encode_64( signatures[0], NULL, encoded_signature );
    1267             : 
    1268           0 :     fd_exec_test_txn_context_t txn_context_msg = FD_EXEC_TEST_TXN_CONTEXT_INIT_DEFAULT;
    1269           0 :     create_txn_context_protobuf_from_txn( &txn_context_msg, runtime, bank, txn_in, txn_out, spad );
    1270             : 
    1271             :     /* Output to file */
    1272           0 :     ulong        out_buf_size = 100UL<<20UL; // 100 MB
    1273           0 :     uchar *      out          = fd_spad_alloc( spad, alignof(uchar), out_buf_size );
    1274           0 :     pb_ostream_t stream       = pb_ostream_from_buffer( out, out_buf_size );
    1275           0 :     if( pb_encode( &stream, FD_EXEC_TEST_TXN_CONTEXT_FIELDS, &txn_context_msg ) ) {
    1276           0 :       char output_filepath[ PATH_MAX ];
    1277           0 :       snprintf( output_filepath, PATH_MAX, "%s/txn-%s.txnctx", runtime->log.dump_proto_ctx->dump_proto_output_dir, encoded_signature );
    1278           0 :       FILE * file = fopen(output_filepath, "wb");
    1279           0 :       if( file ) {
    1280           0 :         fwrite( out, 1, stream.bytes_written, file );
    1281           0 :         fclose( file );
    1282           0 :       }
    1283           0 :     }
    1284           0 :   } FD_SPAD_FRAME_END;
    1285           0 : }
    1286             : 
    1287             : void
    1288             : fd_dump_txn_context_to_protobuf( fd_txn_dump_ctx_t * txn_dump_ctx,
    1289             :                                  fd_runtime_t *      runtime,
    1290             :                                  fd_bank_t *         bank,
    1291             :                                  fd_txn_in_t const * txn_in,
    1292           0 :                                  fd_txn_out_t *      txn_out ) {
    1293           0 :   fd_txn_dump_context_reset( txn_dump_ctx );
    1294             : 
    1295           0 :   txn_dump_ctx->fixture.has_metadata = true;
    1296           0 :   strncpy(
    1297           0 :       txn_dump_ctx->fixture.metadata.fn_entrypoint,
    1298           0 :       "sol_compat_txn_execute_v1",
    1299           0 :       sizeof(txn_dump_ctx->fixture.metadata.fn_entrypoint)-1UL
    1300           0 :   );
    1301             : 
    1302           0 :   txn_dump_ctx->fixture.has_input = true;
    1303           0 :   create_txn_context_protobuf_from_txn( &txn_dump_ctx->fixture.input,
    1304           0 :                                         runtime, bank, txn_in, txn_out,
    1305           0 :                                         txn_dump_ctx->spad );
    1306           0 : }
    1307             : 
    1308             : void
    1309             : fd_dump_txn_result_to_protobuf( fd_txn_dump_ctx_t * txn_dump_ctx,
    1310             :                                 fd_txn_in_t const * txn_in,
    1311             :                                 fd_txn_out_t *      txn_out,
    1312           0 :                                 int                 exec_res ) {
    1313           0 :   txn_dump_ctx->fixture.has_output = true;
    1314             : 
    1315           0 :   ulong  buf_sz = 100UL<<20UL;
    1316           0 :   void * buf    = fd_spad_alloc( txn_dump_ctx->spad, alignof(fd_exec_test_txn_result_t), buf_sz );
    1317           0 :   fd_exec_test_txn_result_t * result = NULL;
    1318           0 :   create_txn_result_protobuf_from_txn( &result, buf, buf_sz, txn_in, txn_out, exec_res );
    1319           0 :   txn_dump_ctx->fixture.output = *result;
    1320           0 : }
    1321             : 
    1322             : void
    1323             : fd_dump_txn_fixture_to_file( fd_txn_dump_ctx_t *         txn_dump_ctx,
    1324             :                              fd_dump_proto_ctx_t const * dump_proto_ctx,
    1325           0 :                              fd_txn_in_t const *         txn_in ) {
    1326           0 :   const fd_ed25519_sig_t * signatures = fd_txn_get_signatures( TXN( txn_in->txn ), txn_in->txn->payload );
    1327           0 :   char encoded_signature[FD_BASE58_ENCODED_64_SZ];
    1328           0 :   fd_base58_encode_64( signatures[0], NULL, encoded_signature );
    1329             : 
    1330           0 :   FD_SPAD_FRAME_BEGIN( txn_dump_ctx->spad ) {
    1331           0 :     ulong        out_buf_size = 100UL<<20UL;
    1332           0 :     uchar *      out          = fd_spad_alloc( txn_dump_ctx->spad, alignof(uchar), out_buf_size );
    1333           0 :     pb_ostream_t stream       = pb_ostream_from_buffer( out, out_buf_size );
    1334             : 
    1335           0 :     char output_filepath[ PATH_MAX ];
    1336             : 
    1337           0 :     if( dump_proto_ctx->dump_txn_as_fixture ) {
    1338           0 :       if( pb_encode( &stream, FD_EXEC_TEST_TXN_FIXTURE_FIELDS, &txn_dump_ctx->fixture ) ) {
    1339           0 :         snprintf( output_filepath, PATH_MAX, "%s/txn-%s.fix", dump_proto_ctx->dump_proto_output_dir, encoded_signature );
    1340           0 :         FILE * file = fopen( output_filepath, "wb" );
    1341           0 :         if( file ) {
    1342           0 :           fwrite( out, 1, stream.bytes_written, file );
    1343           0 :           fclose( file );
    1344           0 :         }
    1345           0 :       }
    1346           0 :     } else {
    1347           0 :       if( pb_encode( &stream, FD_EXEC_TEST_TXN_CONTEXT_FIELDS, &txn_dump_ctx->fixture.input ) ) {
    1348           0 :         snprintf( output_filepath, PATH_MAX, "%s/txn-%s.txnctx", dump_proto_ctx->dump_proto_output_dir, encoded_signature );
    1349           0 :         FILE * file = fopen( output_filepath, "wb" );
    1350           0 :         if( file ) {
    1351           0 :           fwrite( out, 1, stream.bytes_written, file );
    1352           0 :           fclose( file );
    1353           0 :         }
    1354           0 :       }
    1355           0 :     }
    1356           0 :   } FD_SPAD_FRAME_END;
    1357           0 : }
    1358             : 
    1359             : void
    1360             : fd_dump_block_to_protobuf_collect_tx( fd_block_dump_ctx_t * dump_block_ctx,
    1361           0 :                                       fd_txn_p_t const *    txn ) {
    1362           0 :   if( FD_UNLIKELY( dump_block_ctx->txns_to_dump_cnt>=FD_BLOCK_DUMP_CTX_MAX_TXN_CNT ) ) {
    1363           0 :     FD_LOG_ERR(( "Please increase FD_BLOCK_DUMP_CTX_MAX_TXN_CNT to dump more than %lu transactions.", FD_BLOCK_DUMP_CTX_MAX_TXN_CNT ));
    1364           0 :     return;
    1365           0 :   }
    1366           0 :   fd_memcpy( &dump_block_ctx->txns_to_dump[dump_block_ctx->txns_to_dump_cnt++], txn, sizeof(fd_txn_p_t) );
    1367           0 : }
    1368             : 
    1369             : void
    1370             : fd_dump_block_to_protobuf( fd_block_dump_ctx_t *       dump_block_ctx,
    1371             :                            fd_banks_t *                banks,
    1372             :                            fd_bank_t *                 bank,
    1373             :                            fd_accdb_t *                accdb,
    1374             :                            fd_dump_proto_ctx_t const * dump_proto_ctx,
    1375           0 :                            fd_runtime_stack_t *        runtime_stack ) {
    1376           0 :   if( FD_UNLIKELY( dump_block_ctx==NULL ) ) {
    1377           0 :     FD_LOG_WARNING(( "Block dumping context may not be NULL when dumping blocks." ));
    1378           0 :     return;
    1379           0 :   }
    1380             : 
    1381           0 : FD_SPAD_FRAME_BEGIN( dump_block_ctx->spad ) {
    1382           0 :   if( FD_UNLIKELY( dump_proto_ctx==NULL ) ) {
    1383           0 :     FD_LOG_WARNING(( "Protobuf dumping context may not be NULL when dumping blocks." ));
    1384           0 :     return;
    1385           0 :   }
    1386             : 
    1387             :   /* Dump the block context */
    1388           0 :   create_block_context_protobuf_from_block( dump_block_ctx, banks, bank, accdb, runtime_stack );
    1389             : 
    1390             :   /* Output to file */
    1391           0 :   ulong        out_buf_size = 1UL<<30UL; /* 1 GB */
    1392           0 :   uint8_t *    out          = fd_spad_alloc( dump_block_ctx->spad, alignof(uint8_t), out_buf_size );
    1393           0 :   pb_ostream_t stream       = pb_ostream_from_buffer( out, out_buf_size );
    1394           0 :   if( pb_encode( &stream, FD_EXEC_TEST_BLOCK_CONTEXT_FIELDS, &dump_block_ctx->block_context ) ) {
    1395           0 :     char output_filepath[ PATH_MAX ];
    1396           0 :     snprintf( output_filepath, PATH_MAX, "%s/block-%lu.blockctx", dump_proto_ctx->dump_proto_output_dir, bank->f.slot );
    1397           0 :     FILE * file = fopen(output_filepath, "wb");
    1398           0 :     if( file ) {
    1399           0 :       fwrite( out, 1, stream.bytes_written, file );
    1400           0 :       fclose( file );
    1401           0 :     }
    1402           0 :   }
    1403           0 : } FD_SPAD_FRAME_END;
    1404           0 : }
    1405             : 
    1406             : void
    1407             : fd_dump_vm_syscall_to_protobuf( fd_vm_t const * vm,
    1408           0 :                                 char const *    fn_name ) {
    1409           0 :   char const * syscall_name_filter = vm->instr_ctx->runtime->log.dump_proto_ctx->dump_syscall_name_filter;
    1410           0 :   if( syscall_name_filter && strlen( syscall_name_filter ) && strcmp( syscall_name_filter, fn_name ) ) {
    1411           0 :     return;
    1412           0 :   }
    1413             : 
    1414           0 :   fd_spad_t * spad = fd_spad_join( fd_spad_new( vm->instr_ctx->runtime->log.dumping_mem, 1UL<<28UL ) );
    1415             : 
    1416           0 :   FD_SPAD_FRAME_BEGIN( spad ) {
    1417             : 
    1418           0 :   fd_ed25519_sig_t signature;
    1419           0 :   memcpy( signature, (uchar const *)vm->instr_ctx->txn_in->txn->payload + TXN( vm->instr_ctx->txn_in->txn )->signature_off, sizeof(fd_ed25519_sig_t) );
    1420           0 :   char encoded_signature[FD_BASE58_ENCODED_64_SZ];
    1421           0 :   fd_base58_encode_64( signature, NULL, encoded_signature );
    1422             : 
    1423           0 :   char filename[ PATH_MAX ];
    1424           0 :   snprintf( filename,
    1425           0 :           PATH_MAX,
    1426           0 :           "%s/syscall-%s-%s-%d-%hhu-%lu.sysctx",
    1427           0 :           vm->instr_ctx->runtime->log.dump_proto_ctx->dump_proto_output_dir,
    1428           0 :           fn_name,
    1429           0 :           encoded_signature,
    1430           0 :           vm->instr_ctx->runtime->instr.current_idx,
    1431           0 :           vm->instr_ctx->runtime->instr.stack_sz,
    1432           0 :           vm->cu );
    1433             : 
    1434             :   /* The generated filename should be unique for every call. Silently return otherwise. */
    1435           0 :   if( FD_UNLIKELY( access( filename, F_OK )!=-1 ) ) {
    1436           0 :     return;
    1437           0 :   }
    1438             : 
    1439           0 :   fd_exec_test_syscall_context_t sys_ctx = FD_EXEC_TEST_SYSCALL_CONTEXT_INIT_ZERO;
    1440             : 
    1441             :   /* SyscallContext -> vm_ctx */
    1442           0 :   sys_ctx.has_vm_ctx = 1;
    1443             : 
    1444             :   /* SyscallContext -> vm_ctx -> heap_max */
    1445           0 :   sys_ctx.vm_ctx.heap_max = vm->heap_max; /* should be equiv. to txn_ctx->heap_sz */
    1446             : 
    1447             :   /* SyscallContext -> vm_ctx -> rodata */
    1448           0 :   sys_ctx.vm_ctx.rodata = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE( vm->rodata_sz ) );
    1449           0 :   sys_ctx.vm_ctx.rodata->size = (pb_size_t) vm->rodata_sz;
    1450           0 :   fd_memcpy( sys_ctx.vm_ctx.rodata->bytes, vm->rodata, vm->rodata_sz );
    1451             : 
    1452             :   /* SyscallContext -> vm_ctx -> r0-11 */
    1453           0 :   sys_ctx.vm_ctx.r0  = vm->reg[0];
    1454           0 :   sys_ctx.vm_ctx.r1  = vm->reg[1];
    1455           0 :   sys_ctx.vm_ctx.r2  = vm->reg[2];
    1456           0 :   sys_ctx.vm_ctx.r3  = vm->reg[3];
    1457           0 :   sys_ctx.vm_ctx.r4  = vm->reg[4];
    1458           0 :   sys_ctx.vm_ctx.r5  = vm->reg[5];
    1459           0 :   sys_ctx.vm_ctx.r6  = vm->reg[6];
    1460           0 :   sys_ctx.vm_ctx.r7  = vm->reg[7];
    1461           0 :   sys_ctx.vm_ctx.r8  = vm->reg[8];
    1462           0 :   sys_ctx.vm_ctx.r9  = vm->reg[9];
    1463           0 :   sys_ctx.vm_ctx.r10 = vm->reg[10];
    1464           0 :   sys_ctx.vm_ctx.r11 = vm->reg[11];
    1465             : 
    1466             :   /* SyscallContext -> vm_ctx -> entry_pc */
    1467           0 :   sys_ctx.vm_ctx.entry_pc = vm->entry_pc;
    1468             : 
    1469             :   /* SyscallContext -> vm_ctx -> return_data */
    1470           0 :   sys_ctx.vm_ctx.has_return_data = 1;
    1471             : 
    1472             :   /* SyscallContext -> vm_ctx -> return_data -> data */
    1473           0 :   sys_ctx.vm_ctx.return_data.data = fd_spad_alloc( spad, alignof(pb_bytes_array_t), PB_BYTES_ARRAY_T_ALLOCSIZE( vm->instr_ctx->txn_out->details.return_data.len ) );
    1474           0 :   sys_ctx.vm_ctx.return_data.data->size = (pb_size_t)vm->instr_ctx->txn_out->details.return_data.len;
    1475           0 :   fd_memcpy( sys_ctx.vm_ctx.return_data.data->bytes, vm->instr_ctx->txn_out->details.return_data.data, vm->instr_ctx->txn_out->details.return_data.len );
    1476             : 
    1477             :   /* SyscallContext -> vm_ctx -> return_data -> program_id */
    1478           0 :   sys_ctx.vm_ctx.return_data.program_id = fd_spad_alloc( spad, alignof(pb_bytes_array_t), sizeof(fd_pubkey_t) );
    1479           0 :   sys_ctx.vm_ctx.return_data.program_id->size = sizeof(fd_pubkey_t);
    1480           0 :   fd_memcpy( sys_ctx.vm_ctx.return_data.program_id->bytes, vm->instr_ctx->txn_out->details.return_data.program_id.key, sizeof(fd_pubkey_t) );
    1481             : 
    1482             :   /* SyscallContext -> vm_ctx -> sbpf_version */
    1483           0 :   sys_ctx.vm_ctx.sbpf_version = (uint)vm->sbpf_version;
    1484             : 
    1485             :   /* SyscallContext -> instr_ctx */
    1486           0 :   sys_ctx.has_instr_ctx = 1;
    1487           0 :   create_instr_context_protobuf_from_instructions( &sys_ctx.instr_ctx,
    1488           0 :                                                    vm->instr_ctx->runtime,
    1489           0 :                                                    vm->instr_ctx->bank,
    1490           0 :                                                    vm->instr_ctx->txn_out,
    1491           0 :                                                    vm->instr_ctx->instr,
    1492           0 :                                                    spad );
    1493             : 
    1494             :   /* SyscallContext -> syscall_invocation */
    1495           0 :   sys_ctx.has_syscall_invocation = 1;
    1496             : 
    1497             :   /* SyscallContext -> syscall_invocation -> function_name */
    1498           0 :   sys_ctx.syscall_invocation.function_name.size = fd_uint_min( (uint) strlen(fn_name), sizeof(sys_ctx.syscall_invocation.function_name.bytes) );
    1499           0 :   fd_memcpy( sys_ctx.syscall_invocation.function_name.bytes,
    1500           0 :              fn_name,
    1501           0 :              sys_ctx.syscall_invocation.function_name.size );
    1502             : 
    1503             :   /* SyscallContext -> syscall_invocation -> heap_prefix */
    1504           0 :   sys_ctx.syscall_invocation.heap_prefix = fd_spad_alloc( spad, 8UL, PB_BYTES_ARRAY_T_ALLOCSIZE( vm->heap_max ) );
    1505           0 :   sys_ctx.syscall_invocation.heap_prefix->size = (pb_size_t) vm->instr_ctx->txn_out->details.compute_budget.heap_size;
    1506           0 :   fd_memcpy( sys_ctx.syscall_invocation.heap_prefix->bytes, vm->heap, vm->instr_ctx->txn_out->details.compute_budget.heap_size );
    1507             : 
    1508             :   /* SyscallContext -> syscall_invocation -> stack_prefix */
    1509           0 :   pb_size_t stack_sz = (pb_size_t)FD_VM_STACK_MAX;
    1510           0 :   sys_ctx.syscall_invocation.stack_prefix = fd_spad_alloc( spad, 8UL, PB_BYTES_ARRAY_T_ALLOCSIZE( stack_sz ) );
    1511           0 :   sys_ctx.syscall_invocation.stack_prefix->size = stack_sz;
    1512           0 :   fd_memcpy( sys_ctx.syscall_invocation.stack_prefix->bytes, vm->stack, stack_sz );
    1513             : 
    1514             :   /* Output to file */
    1515           0 :   ulong out_buf_size = 1UL<<29UL; /* 128 MB */
    1516           0 :   uint8_t * out = fd_spad_alloc( spad, alignof(uint8_t), out_buf_size );
    1517           0 :   pb_ostream_t stream = pb_ostream_from_buffer( out, out_buf_size );
    1518           0 :   if( pb_encode( &stream, FD_EXEC_TEST_SYSCALL_CONTEXT_FIELDS, &sys_ctx ) ) {
    1519           0 :     FILE * file = fopen(filename, "wb");
    1520           0 :     if( file ) {
    1521           0 :       fwrite( out, 1, stream.bytes_written, file );
    1522           0 :       fclose( file );
    1523           0 :     }
    1524           0 :   }
    1525           0 :   } FD_SPAD_FRAME_END;
    1526           0 : }

Generated by: LCOV version 1.14