Line data Source code
1 : #include "fd_sysvar_cache.h" 2 : #include "../fd_acc_mgr.h" 3 : #include "../fd_executor.h" 4 : #include "../fd_system_ids.h" 5 : #include "../context/fd_exec_instr_ctx.h" 6 : #include "../context/fd_exec_txn_ctx.h" 7 : #include "../context/fd_exec_slot_ctx.h" 8 : 9 407859 : #define FD_SYSVAR_CACHE_MAGIC (0x195a0e78828cacd5UL) 10 : 11 : ulong 12 407859 : fd_sysvar_cache_align( void ) { 13 407859 : return alignof(fd_sysvar_cache_t); 14 407859 : } 15 : 16 : ulong 17 407859 : fd_sysvar_cache_footprint( void ) { 18 407859 : return sizeof(fd_sysvar_cache_t); 19 407859 : } 20 : 21 : fd_sysvar_cache_t * 22 : fd_sysvar_cache_new( void * mem, 23 407859 : fd_valloc_t valloc ) { 24 : 25 407859 : if( FD_UNLIKELY( !mem ) ) { 26 0 : FD_LOG_WARNING(( "NULL mem" )); 27 0 : return NULL; 28 0 : } 29 : 30 407859 : fd_sysvar_cache_t * cache = (fd_sysvar_cache_t *)mem; 31 407859 : fd_memset( cache, 0, sizeof(fd_sysvar_cache_t) ); 32 : 33 407859 : cache->valloc = valloc; 34 : 35 407859 : FD_COMPILER_MFENCE(); 36 407859 : cache->magic = FD_SYSVAR_CACHE_MAGIC; 37 407859 : FD_COMPILER_MFENCE(); 38 407859 : return cache; 39 407859 : } 40 : 41 : void * 42 407859 : fd_sysvar_cache_delete( fd_sysvar_cache_t * cache ) { 43 : 44 407859 : if( FD_UNLIKELY( !cache ) ) { 45 0 : FD_LOG_WARNING(( "NULL cache" )); 46 0 : return NULL; 47 0 : } 48 : 49 407859 : if( FD_UNLIKELY( cache->magic != FD_SYSVAR_CACHE_MAGIC ) ) { 50 0 : FD_LOG_WARNING(( "bad magic" )); 51 0 : return NULL; 52 0 : } 53 : 54 407859 : if( FD_UNLIKELY( !cache->valloc.vt ) ) { 55 0 : FD_LOG_WARNING(( "NULL alloc" )); 56 0 : return NULL; 57 0 : } 58 : 59 407859 : FD_COMPILER_MFENCE(); 60 407859 : cache->magic = 0UL; 61 407859 : FD_COMPILER_MFENCE(); 62 : 63 407859 : fd_bincode_destroy_ctx_t ctx = { .valloc = cache->valloc }; 64 : 65 : /* Call destroy on all objects. 66 : This is safe even if these objects logically don't exist 67 : (destory is safe on zero-initialized values and is idempotent) */ 68 407859 : # define X( type, name ) \ 69 3670731 : type##_destroy( cache->val_##name, &ctx ); 70 3670731 : FD_SYSVAR_CACHE_ITER(X) 71 407859 : # undef X 72 : 73 407859 : return (void *)cache; 74 407859 : } 75 : 76 : /* Provide accessor methods */ 77 : 78 : #define X( type, name ) \ 79 : type##_t const * \ 80 1306995 : fd_sysvar_cache_##name( fd_sysvar_cache_t const * cache ) { \ 81 1306995 : type##_t const * val = cache->val_##name; \ 82 1306995 : return (cache->has_##name) ? val : NULL; \ 83 1306995 : } 84 : FD_SYSVAR_CACHE_ITER(X) 85 : #undef X 86 : 87 : /* Restore sysvars */ 88 : 89 : # define X( type, name ) \ 90 : void \ 91 : fd_sysvar_cache_restore_##name( \ 92 : fd_sysvar_cache_t * cache, \ 93 : fd_acc_mgr_t * acc_mgr, \ 94 3785400 : fd_funk_txn_t * funk_txn) { \ 95 3785400 : do { \ 96 3785400 : fd_pubkey_t const * pubkey = &fd_sysvar_##name##_id; \ 97 3785400 : FD_BORROWED_ACCOUNT_DECL( account ); \ 98 3785400 : int view_err = fd_acc_mgr_view( acc_mgr, funk_txn, pubkey, account ); \ 99 3785400 : if( view_err==FD_ACC_MGR_ERR_UNKNOWN_ACCOUNT ) break; \ 100 3785400 : \ 101 3785400 : if( view_err!=FD_ACC_MGR_SUCCESS ) { \ 102 0 : char pubkey_cstr[ FD_BASE58_ENCODED_32_SZ ]; \ 103 0 : FD_LOG_ERR(( "fd_acc_mgr_view(%s) failed (%d-%s)", \ 104 0 : fd_acct_addr_cstr( pubkey_cstr, pubkey->key ), \ 105 0 : view_err, fd_acc_mgr_strerror( view_err ) )); \ 106 0 : } \ 107 173478 : \ 108 173478 : if( account->const_meta->info.lamports == 0UL ) break; \ 109 173478 : \ 110 173478 : /* Destroy previous value */ \ 111 173478 : \ 112 173478 : fd_bincode_destroy_ctx_t destroy = { .valloc = cache->valloc }; \ 113 172722 : type##_destroy( cache->val_##name, &destroy ); \ 114 172722 : \ 115 172722 : /* Decode new value \ 116 172722 : type##_decode() does not do heap allocations on failure */ \ 117 172722 : \ 118 172722 : fd_bincode_decode_ctx_t decode = \ 119 172722 : { .data = account->const_data, \ 120 172722 : .dataend = account->const_data + account->const_meta->dlen, \ 121 172722 : .valloc = cache->valloc }; \ 122 172722 : int err = type##_decode( cache->val_##name, &decode ); \ 123 172722 : cache->has_##name = (err==FD_BINCODE_SUCCESS); \ 124 172722 : } while(0); \ 125 3785400 : } 126 : FD_SYSVAR_CACHE_ITER(X) 127 : # undef X 128 : 129 : void 130 : fd_sysvar_cache_restore( fd_sysvar_cache_t * cache, 131 : fd_acc_mgr_t * acc_mgr, 132 420600 : fd_funk_txn_t * funk_txn ) { 133 3785400 : # define X( type, name ) \ 134 3785400 : fd_sysvar_cache_restore_##name( cache, acc_mgr, funk_txn ); 135 3785400 : FD_SYSVAR_CACHE_ITER(X) 136 420600 : # undef X 137 420600 : } 138 : 139 : # define X( type, name ) \ 140 : type##_t const * \ 141 : fd_sysvar_from_instr_acct_##name( fd_exec_instr_ctx_t const * ctx, \ 142 : ulong idx, \ 143 15222 : int * err ) {\ 144 15222 : \ 145 15222 : if( FD_UNLIKELY( idx >= ctx->instr->acct_cnt ) ) { \ 146 45 : *err = FD_EXECUTOR_INSTR_ERR_NOT_ENOUGH_ACC_KEYS; \ 147 45 : return NULL; \ 148 45 : } \ 149 15222 : \ 150 15222 : fd_sysvar_cache_t const * cache = ctx->slot_ctx->sysvar_cache; \ 151 15177 : type##_t const * val = fd_sysvar_cache_##name ( cache ); \ 152 15177 : \ 153 15177 : fd_pubkey_t const * addr_have = &ctx->instr->acct_pubkeys[idx]; \ 154 15177 : fd_pubkey_t const * addr_want = &fd_sysvar_##name##_id; \ 155 15177 : if( 0!=memcmp( addr_have, addr_want, sizeof(fd_pubkey_t) ) ) { \ 156 1170 : *err = FD_EXECUTOR_INSTR_ERR_INVALID_ARG; \ 157 1170 : return NULL; \ 158 1170 : } \ 159 15177 : \ 160 15177 : *err = val ? \ 161 14007 : FD_EXECUTOR_INSTR_SUCCESS : \ 162 14007 : FD_EXECUTOR_INSTR_ERR_UNSUPPORTED_SYSVAR; \ 163 14007 : return val; \ 164 15177 : \ 165 15177 : } 166 : FD_SYSVAR_CACHE_ITER(X) 167 : # undef X 168 : 169 : /* https://github.com/anza-xyz/agave/blob/77daab497df191ef485a7ad36ed291c1874596e5/program-runtime/src/sysvar_cache.rs#L223-L234 */ 170 : int 171 : fd_check_sysvar_account( fd_exec_instr_ctx_t const * ctx, 172 : ulong insn_acc_idx, 173 1686 : fd_pubkey_t const * expected_id ) { 174 1686 : uchar const * instr_acc_idxs = ctx->instr->acct_txn_idxs; 175 1686 : fd_pubkey_t const * txn_accs = ctx->txn_ctx->accounts; 176 : 177 1686 : if( insn_acc_idx>=ctx->instr->acct_cnt ) { 178 0 : return FD_EXECUTOR_INSTR_ERR_NOT_ENOUGH_ACC_KEYS; 179 0 : } 180 : 181 1686 : fd_pubkey_t const * insn_acc_key = &txn_accs[ instr_acc_idxs[ insn_acc_idx ] ]; 182 : 183 1686 : if( memcmp( expected_id, insn_acc_key, sizeof(fd_pubkey_t) ) ) { 184 0 : return FD_EXECUTOR_INSTR_ERR_INVALID_ARG; 185 0 : } 186 1686 : return FD_EXECUTOR_INSTR_SUCCESS; 187 1686 : }