LCOV - code coverage report
Current view: top level - flamenco/runtime/sysvar - fd_sysvar_cache.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 99 120 82.5 %
Date: 2025-01-08 12:08:44 Functions: 27 33 81.8 %

          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 : }

Generated by: LCOV version 1.14