LCOV - code coverage report
Current view: top level - flamenco/vm - fd_vm.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 346 454 76.2 %
Date: 2026-04-08 06:13:01 Functions: 9 11 81.8 %

          Line data    Source code
       1             : #include "fd_vm_private.h"
       2             : #include "../../ballet/sbpf/fd_sbpf_instr.h"
       3             : #include "../../ballet/sbpf/fd_sbpf_opcodes.h"
       4             : 
       5             : /* fd_vm_syscall_strerror() returns the error message corresponding to err,
       6             :    intended to be logged by log_collector, or an empty string if the error code
       7             :    should be omitted in logs for whatever reason.  Omitted examples are success,
       8             :    panic (logged in place)...
       9             :    See also fd_log_collector_program_failure(). */
      10             : char const *
      11           0 : fd_vm_syscall_strerror( int err ) {
      12             : 
      13           0 :   switch( err ) {
      14             : 
      15           0 :   case FD_VM_SYSCALL_ERR_INVALID_STRING:                         return "invalid utf-8 sequence"; // truncated
      16           0 :   case FD_VM_SYSCALL_ERR_ABORT:                                  return "SBF program panicked";
      17           0 :   case FD_VM_SYSCALL_ERR_PANIC:                                  return "SBF program Panicked in..."; // truncated
      18           0 :   case FD_VM_SYSCALL_ERR_INVOKE_CONTEXT_BORROW_FAILED:           return "Cannot borrow invoke context";
      19           0 :   case FD_VM_SYSCALL_ERR_MALFORMED_SIGNER_SEED:                  return "Malformed signer seed"; // truncated
      20           0 :   case FD_VM_SYSCALL_ERR_BAD_SEEDS:                              return "Could not create program address with signer seeds"; // truncated
      21           0 :   case FD_VM_SYSCALL_ERR_PROGRAM_NOT_SUPPORTED:                  return "Program not supported by inner instructions"; // truncated
      22           0 :   case FD_VM_SYSCALL_ERR_UNALIGNED_POINTER:                      return "Unaligned pointer";
      23           0 :   case FD_VM_SYSCALL_ERR_TOO_MANY_SIGNERS:                       return "Too many signers";
      24           0 :   case FD_VM_SYSCALL_ERR_INSTRUCTION_TOO_LARGE:                  return "Instruction passed to inner instruction is too large"; // truncated
      25           0 :   case FD_VM_SYSCALL_ERR_TOO_MANY_ACCOUNTS:                      return "Too many accounts passed to inner instruction";
      26           0 :   case FD_VM_SYSCALL_ERR_COPY_OVERLAPPING:                       return "Overlapping copy";
      27           0 :   case FD_VM_SYSCALL_ERR_RETURN_DATA_TOO_LARGE:                  return "Return data too large"; // truncated
      28           0 :   case FD_VM_SYSCALL_ERR_TOO_MANY_SLICES:                        return "Hashing too many sequences";
      29           0 :   case FD_VM_SYSCALL_ERR_INVALID_LENGTH:                         return "InvalidLength";
      30           0 :   case FD_VM_SYSCALL_ERR_MAX_INSTRUCTION_DATA_LEN_EXCEEDED:      return "Invoked an instruction with data that is too large"; // truncated
      31           0 :   case FD_VM_SYSCALL_ERR_MAX_INSTRUCTION_ACCOUNTS_EXCEEDED:      return "Invoked an instruction with too many accounts"; // truncated
      32           0 :   case FD_VM_SYSCALL_ERR_MAX_INSTRUCTION_ACCOUNT_INFOS_EXCEEDED: return "Invoked an instruction with too many account info's"; // truncated
      33           0 :   case FD_VM_SYSCALL_ERR_INVALID_ATTRIBUTE:                      return "InvalidAttribute";
      34           0 :   case FD_VM_SYSCALL_ERR_INVALID_POINTER:                        return "Invalid pointer";
      35           0 :   case FD_VM_SYSCALL_ERR_ARITHMETIC_OVERFLOW:                    return "Arithmetic overflow";
      36             : 
      37           0 :   case FD_VM_SYSCALL_ERR_INSTR_ERR:                              return "Instruction error";
      38           0 :   case FD_VM_SYSCALL_ERR_INVALID_PDA:                            return "Invalid PDA";
      39           0 :   case FD_VM_SYSCALL_ERR_COMPUTE_BUDGET_EXCEEDED:                return "Compute budget exceeded";
      40           0 :   case FD_VM_SYSCALL_ERR_SEGFAULT:                               return "Segmentation fault";
      41           0 :   case FD_VM_SYSCALL_ERR_OUTSIDE_RUNTIME:                        return "Syscall executed outside runtime";
      42             : 
      43             : 
      44           0 :   case FD_VM_SYSCALL_ERR_POSEIDON_INVALID_PARAMS:                return "Invalid parameters.";
      45           0 :   case FD_VM_SYSCALL_ERR_POSEIDON_INVALID_ENDIANNESS:            return "Invalid endianness.";
      46             : 
      47           0 :   default: break;
      48           0 :   }
      49             : 
      50           0 :   return "";
      51           0 : }
      52             : 
      53             : /* fd_vm_ebpf_strerror() returns the error message corresponding to err,
      54             :    intended to be logged by log_collector, or an empty string if the error code
      55             :    should be omitted in logs for whatever reason.
      56             :    See also fd_log_collector_program_failure(). */
      57             : char const *
      58           0 : fd_vm_ebpf_strerror( int err ) {
      59             : 
      60           0 :   switch( err ) {
      61             : 
      62           0 :   case FD_VM_ERR_EBPF_ELF_ERROR:                   return "ELF error"; // truncated
      63           0 :   case FD_VM_ERR_EBPF_FUNCTION_ALREADY_REGISTERED: return "function was already registered"; // truncated
      64           0 :   case FD_VM_ERR_EBPF_CALL_DEPTH_EXCEEDED:         return "exceeded max BPF to BPF call depth";
      65           0 :   case FD_VM_ERR_EBPF_EXIT_ROOT_CALL_FRAME:        return "attempted to exit root call frame";
      66           0 :   case FD_VM_ERR_EBPF_DIVIDE_BY_ZERO:              return "divide by zero at BPF instruction";
      67           0 :   case FD_VM_ERR_EBPF_DIVIDE_OVERFLOW:             return "division overflow at BPF instruction";
      68           0 :   case FD_VM_ERR_EBPF_EXECUTION_OVERRUN:           return "attempted to execute past the end of the text segment at BPF instruction";
      69           0 :   case FD_VM_ERR_EBPF_CALL_OUTSIDE_TEXT_SEGMENT:   return "callx attempted to call outside of the text segment";
      70           0 :   case FD_VM_ERR_EBPF_EXCEEDED_MAX_INSTRUCTIONS:   return "exceeded CUs meter at BPF instruction";
      71           0 :   case FD_VM_ERR_EBPF_JIT_NOT_COMPILED:            return "program has not been JIT-compiled";
      72           0 :   case FD_VM_ERR_EBPF_INVALID_VIRTUAL_ADDRESS:     return "invalid virtual address"; // truncated
      73           0 :   case FD_VM_ERR_EBPF_INVALID_MEMORY_REGION:       return "Invalid memory region at index"; // truncated
      74           0 :   case FD_VM_ERR_EBPF_ACCESS_VIOLATION:            return "Access violation"; // truncated
      75           0 :   case FD_VM_ERR_EBPF_STACK_ACCESS_VIOLATION:      return "Access violation in stack frame"; // truncated
      76           0 :   case FD_VM_ERR_EBPF_INVALID_INSTRUCTION:         return "invalid BPF instruction";
      77           0 :   case FD_VM_ERR_EBPF_UNSUPPORTED_INSTRUCTION:     return "unsupported BPF instruction";
      78           0 :   case FD_VM_ERR_EBPF_EXHAUSTED_TEXT_SEGMENT:      return "Compilation exhausted text segment at BPF instruction"; // truncated
      79           0 :   case FD_VM_ERR_EBPF_LIBC_INVOCATION_FAILED:      return "Libc calling returned error code"; // truncated
      80           0 :   case FD_VM_ERR_EBPF_VERIFIER_ERROR:              return "Verifier error"; // truncated
      81           0 :   case FD_VM_ERR_EBPF_SYSCALL_ERROR:               return ""; // handled explicitly via fd_vm_syscall_strerror()
      82             : 
      83           0 :   default: break;
      84           0 :   }
      85             : 
      86           0 :   return "";
      87           0 : }
      88             : 
      89             : /* fd_vm_strerror() returns the error message corresponding to err, used internally
      90             :    for system logs, NOT for log_collector / transaction logs. */
      91             : char const *
      92          42 : fd_vm_strerror( int err ) {
      93             : 
      94          42 :   switch( err ) {
      95             : 
      96             :   /* "Standard" Firedancer error codes */
      97             : 
      98           3 :   case FD_VM_SUCCESS:   return "SUCCESS success";
      99           3 :   case FD_VM_ERR_INVAL: return "INVAL invalid request";
     100           3 :   case FD_VM_ERR_UNSUP: return "UNSUP unsupported request";
     101           3 :   case FD_VM_ERR_FULL:  return "FULL storage full";
     102           3 :   case FD_VM_ERR_EMPTY: return "EMPTY nothing to do";
     103           3 :   case FD_VM_ERR_IO:    return "IO input-output error";
     104             : 
     105             :   /* VM exec error codes */
     106             : 
     107           0 :   case FD_VM_ERR_SIGFPE:      return "SIGFPE division by zero";
     108             : 
     109             :   /* VM validate error codes */
     110             : 
     111           3 :   case FD_VM_ERR_INVALID_OPCODE:    return "INVALID_OPCODE detected an invalid opcode";
     112           3 :   case FD_VM_ERR_INVALID_SRC_REG:   return "INVALID_SRC_REG detected an invalid source register";
     113           3 :   case FD_VM_ERR_INVALID_DST_REG:   return "INVALID_DST_REG detected an invalid destination register";
     114           3 :   case FD_VM_ERR_JMP_OUT_OF_BOUNDS: return "JMP_OUT_OF_BOUNDS detected an out of bounds jump";
     115           3 :   case FD_VM_ERR_JMP_TO_ADDL_IMM:   return "JMP_TO_ADDL_IMM detected a jump to an addl imm";
     116           3 :   case FD_VM_ERR_INVALID_END_IMM:   return "INVALID_END_IMM detected an invalid immediate for an endianness conversion instruction";
     117           3 :   case FD_VM_ERR_INCOMPLETE_LDQ:    return "INCOMPLETE_LDQ detected an incomplete ldq at program end";
     118           3 :   case FD_VM_ERR_LDQ_NO_ADDL_IMM:   return "LDQ_NO_ADDL_IMM detected a ldq without an addl imm following it";
     119           0 :   case FD_VM_ERR_INVALID_REG:       return "INVALID_REG detected an invalid register number in a callx instruction";
     120           0 :   case FD_VM_ERR_BAD_TEXT:          return "BAD_TEXT detected a bad text section";
     121           0 :   case FD_VM_SH_OVERFLOW:           return "SH_OVERFLOW detected a shift overflow in an instruction";
     122           0 :   case FD_VM_TEXT_SZ_UNALIGNED:     return "TEXT_SZ_UNALIGNED detected an unaligned text section size (not a multiple of 8)";
     123             : 
     124           0 :   default: break;
     125          42 :   }
     126             : 
     127           0 :   return "UNKNOWN probably not a FD_VM_ERR code";
     128          42 : }
     129             : 
     130             : /* FIXME: add a pedantic version of this validation that does things
     131             :    like:
     132             :   - only 0 imms when the instruction does not use an imm
     133             :   - same as above but for src/dst reg, offset */
     134             : /* FIXME: HANDLING OF LDQ ON NEWER SBPF VERSUS OLDER SBPF (AND WITH CALL
     135             :    REG) */
     136             : /* FIXME: LINK TO SOLANA CODE VALIDATE AND DOUBLE CHECK THIS BEHAVES
     137             :    IDENTICALLY! */
     138             : 
     139             : int
     140        8994 : fd_vm_validate( fd_vm_t const * vm ) {
     141             : 
     142        8994 :   ulong sbpf_version = vm->sbpf_version;
     143             : 
     144             :   /* A mapping of all the possible 1-byte sBPF opcodes to their
     145             :      validation criteria. */
     146             : 
     147   645194469 : # define FD_VALID               ((uchar)0)  /* Valid opcode */
     148      261861 : # define FD_CHECK_JMP           ((uchar)1)  /* Validation should check that the instruction is a valid jump */
     149       13749 : # define FD_CHECK_END           ((uchar)2)  /* Validation should check that the instruction is a valid endianness conversion */
     150      246078 : # define FD_CHECK_ST            ((uchar)3)  /* Validation should check that the instruction is a valid store */
     151       15000 : # define FD_CHECK_LDQ           ((uchar)4)  /* Validation should check that the instruction is a valid load-quad */
     152    64568454 : # define FD_CHECK_DIV           ((uchar)5)  /* Validation should check that the instruction is a valid division by immediate */
     153    48335400 : # define FD_CHECK_SH32          ((uchar)6)  /* Validation should check that the immediate is a valid 32-bit shift exponent */
     154    48365871 : # define FD_CHECK_SH64          ((uchar)7)  /* Validation should check that the immediate is a valid 64-bit shift exponent */
     155     1877889 : # define FD_INVALID             ((uchar)8)  /* The opcode is invalid */
     156       13350 : # define FD_CHECK_CALL_REG_SRC  ((uchar)9)  /* Validation should check that callx has valid register number in src */
     157        8994 : # define FD_CHECK_CALL_REG_DST  ((uchar)13) /* Validation should check that callx has a valid register number in dst */
     158       17088 : # define FD_CHECK_CALL_REG_IMM  ((uchar)10) /* Validation should check that callx has a valid register number in imm */
     159             : 
     160        8994 :   uchar validation_map[ 256 ] = {
     161        8994 :     /* 0x00 */ FD_INVALID,    /* 0x01 */ FD_INVALID,    /* 0x02 */ FD_INVALID,    /* 0x03 */ FD_INVALID,
     162        8994 :     /* 0x04 */ FD_VALID,      /* 0x05 */ FD_CHECK_JMP,  /* 0x06 */ FD_INVALID,    /* 0x07 */ FD_VALID,
     163        8994 :     /* 0x08 */ FD_INVALID,    /* 0x09 */ FD_INVALID,    /* 0x0a */ FD_INVALID,    /* 0x0b */ FD_INVALID,
     164        8994 :     /* 0x0c */ FD_VALID,      /* 0x0d */ FD_INVALID,    /* 0x0e */ FD_INVALID,    /* 0x0f */ FD_VALID,
     165        8994 :     /* 0x10 */ FD_INVALID,    /* 0x11 */ FD_INVALID,    /* 0x12 */ FD_INVALID,    /* 0x13 */ FD_INVALID,
     166        8994 :     /* 0x14 */ FD_VALID,      /* 0x15 */ FD_CHECK_JMP,  /* 0x16 */ FD_INVALID,    /* 0x17 */ FD_VALID,
     167        8994 :     /* 0x18 */ FD_INVALID,    /* 0x19 */ FD_INVALID,    /* 0x1a */ FD_INVALID,    /* 0x1b */ FD_INVALID,
     168        8994 :     /* 0x1c */ FD_VALID,      /* 0x1d */ FD_CHECK_JMP,  /* 0x1e */ FD_INVALID,    /* 0x1f */ FD_VALID,
     169        8994 :     /* 0x20 */ FD_INVALID,    /* 0x21 */ FD_INVALID,    /* 0x22 */ FD_INVALID,    /* 0x23 */ FD_INVALID,
     170        8994 :     /* 0x24 */ FD_INVALID,    /* 0x25 */ FD_CHECK_JMP,  /* 0x26 */ FD_INVALID,    /* 0x27 */ FD_CHECK_ST,
     171        8994 :     /* 0x28 */ FD_INVALID,    /* 0x29 */ FD_INVALID,    /* 0x2a */ FD_INVALID,    /* 0x2b */ FD_INVALID,
     172        8994 :     /* 0x2c */ FD_VALID,      /* 0x2d */ FD_CHECK_JMP,  /* 0x2e */ FD_INVALID,    /* 0x2f */ FD_CHECK_ST,
     173        8994 :     /* 0x30 */ FD_INVALID,    /* 0x31 */ FD_INVALID,    /* 0x32 */ FD_INVALID,    /* 0x33 */ FD_INVALID,
     174        8994 :     /* 0x34 */ FD_INVALID,    /* 0x35 */ FD_CHECK_JMP,  /* 0x36 */ FD_VALID,      /* 0x37 */ FD_CHECK_ST,
     175        8994 :     /* 0x38 */ FD_INVALID,    /* 0x39 */ FD_INVALID,    /* 0x3a */ FD_INVALID,    /* 0x3b */ FD_INVALID,
     176        8994 :     /* 0x3c */ FD_VALID,      /* 0x3d */ FD_CHECK_JMP,  /* 0x3e */ FD_VALID,      /* 0x3f */ FD_CHECK_ST,
     177        8994 :     /* 0x40 */ FD_INVALID,    /* 0x41 */ FD_INVALID,    /* 0x42 */ FD_INVALID,    /* 0x43 */ FD_INVALID,
     178        8994 :     /* 0x44 */ FD_VALID,      /* 0x45 */ FD_CHECK_JMP,  /* 0x46 */ FD_CHECK_DIV,  /* 0x47 */ FD_VALID,
     179        8994 :     /* 0x48 */ FD_INVALID,    /* 0x49 */ FD_INVALID,    /* 0x4a */ FD_INVALID,    /* 0x4b */ FD_INVALID,
     180        8994 :     /* 0x4c */ FD_VALID,      /* 0x4d */ FD_CHECK_JMP,  /* 0x4e */ FD_VALID,      /* 0x4f */ FD_VALID,
     181        8994 :     /* 0x50 */ FD_INVALID,    /* 0x51 */ FD_INVALID,    /* 0x52 */ FD_INVALID,    /* 0x53 */ FD_INVALID,
     182        8994 :     /* 0x54 */ FD_VALID,      /* 0x55 */ FD_CHECK_JMP,  /* 0x56 */ FD_CHECK_DIV,  /* 0x57 */ FD_VALID,
     183        8994 :     /* 0x58 */ FD_INVALID,    /* 0x59 */ FD_INVALID,    /* 0x5a */ FD_INVALID,    /* 0x5b */ FD_INVALID,
     184        8994 :     /* 0x5c */ FD_VALID,      /* 0x5d */ FD_CHECK_JMP,  /* 0x5e */ FD_VALID,      /* 0x5f */ FD_VALID,
     185        8994 :     /* 0x60 */ FD_INVALID,    /* 0x61 */ FD_INVALID,    /* 0x62 */ FD_INVALID,    /* 0x63 */ FD_INVALID,
     186        8994 :     /* 0x64 */ FD_CHECK_SH32, /* 0x65 */ FD_CHECK_JMP,  /* 0x66 */ FD_CHECK_DIV,  /* 0x67 */ FD_CHECK_SH64,
     187        8994 :     /* 0x68 */ FD_INVALID,    /* 0x69 */ FD_INVALID,    /* 0x6a */ FD_INVALID,    /* 0x6b */ FD_INVALID,
     188        8994 :     /* 0x6c */ FD_VALID,      /* 0x6d */ FD_CHECK_JMP,  /* 0x6e */ FD_VALID,      /* 0x6f */ FD_VALID,
     189        8994 :     /* 0x70 */ FD_INVALID,    /* 0x71 */ FD_INVALID,    /* 0x72 */ FD_INVALID,    /* 0x73 */ FD_INVALID,
     190        8994 :     /* 0x74 */ FD_CHECK_SH32, /* 0x75 */ FD_CHECK_JMP,  /* 0x76 */ FD_CHECK_DIV,  /* 0x77 */ FD_CHECK_SH64,
     191        8994 :     /* 0x78 */ FD_INVALID,    /* 0x79 */ FD_INVALID,    /* 0x7a */ FD_INVALID,    /* 0x7b */ FD_INVALID,
     192        8994 :     /* 0x7c */ FD_VALID,      /* 0x7d */ FD_CHECK_JMP,  /* 0x7e */ FD_VALID,      /* 0x7f */ FD_VALID,
     193        8994 :     /* 0x80 */ FD_INVALID,    /* 0x81 */ FD_INVALID,    /* 0x82 */ FD_INVALID,    /* 0x83 */ FD_INVALID,
     194        8994 :     /* 0x84 */ FD_INVALID,    /* 0x85 */ FD_VALID,      /* 0x86 */ FD_VALID,      /* 0x87 */ FD_CHECK_ST,
     195        8994 :     /* 0x88 */ FD_INVALID,    /* 0x89 */ FD_INVALID,    /* 0x8a */ FD_INVALID,    /* 0x8b */ FD_INVALID,
     196        8994 :     /* 0x8c */ FD_VALID,      /* 0x8d */ FD_CHECK_CALL_REG_SRC,/*0x8e*/FD_VALID,      /* 0x8f */ FD_CHECK_ST,
     197        8994 :     /* 0x90 */ FD_INVALID,    /* 0x91 */ FD_INVALID,    /* 0x92 */ FD_INVALID,    /* 0x93 */ FD_INVALID,
     198        8994 :     /* 0x94 */ FD_INVALID,    /* 0x95 */ FD_VALID,      /* 0x96 */ FD_VALID,      /* 0x97 */ FD_CHECK_ST,
     199        8994 :     /* 0x98 */ FD_INVALID,    /* 0x99 */ FD_INVALID,    /* 0x9a */ FD_INVALID,    /* 0x9b */ FD_INVALID,
     200        8994 :     /* 0x9c */ FD_VALID,      /* 0x9d */ FD_INVALID,    /* 0x9e */ FD_VALID,      /* 0x9f */ FD_CHECK_ST,
     201        8994 :     /* 0xa0 */ FD_INVALID,    /* 0xa1 */ FD_INVALID,    /* 0xa2 */ FD_INVALID,    /* 0xa3 */ FD_INVALID,
     202        8994 :     /* 0xa4 */ FD_VALID,      /* 0xa5 */ FD_CHECK_JMP,  /* 0xa6 */ FD_INVALID,    /* 0xa7 */ FD_VALID,
     203        8994 :     /* 0xa8 */ FD_INVALID,    /* 0xa9 */ FD_INVALID,    /* 0xaa */ FD_INVALID,    /* 0xab */ FD_INVALID,
     204        8994 :     /* 0xac */ FD_VALID,      /* 0xad */ FD_CHECK_JMP,  /* 0xae */ FD_INVALID,    /* 0xaf */ FD_VALID,
     205        8994 :     /* 0xb0 */ FD_INVALID,    /* 0xb1 */ FD_INVALID,    /* 0xb2 */ FD_INVALID,    /* 0xb3 */ FD_INVALID,
     206        8994 :     /* 0xb4 */ FD_VALID,      /* 0xb5 */ FD_CHECK_JMP,  /* 0xb6 */ FD_VALID,      /* 0xb7 */ FD_VALID,
     207        8994 :     /* 0xb8 */ FD_INVALID,    /* 0xb9 */ FD_INVALID,    /* 0xba */ FD_INVALID,    /* 0xbb */ FD_INVALID,
     208        8994 :     /* 0xbc */ FD_VALID,      /* 0xbd */ FD_CHECK_JMP,  /* 0xbe */ FD_VALID,      /* 0xbf */ FD_VALID,
     209        8994 :     /* 0xc0 */ FD_INVALID,    /* 0xc1 */ FD_INVALID,    /* 0xc2 */ FD_INVALID,    /* 0xc3 */ FD_INVALID,
     210        8994 :     /* 0xc4 */ FD_CHECK_SH32, /* 0xc5 */ FD_CHECK_JMP,  /* 0xc6 */ FD_CHECK_DIV,  /* 0xc7 */ FD_CHECK_SH64,
     211        8994 :     /* 0xc8 */ FD_INVALID,    /* 0xc9 */ FD_INVALID,    /* 0xca */ FD_INVALID,    /* 0xcb */ FD_INVALID,
     212        8994 :     /* 0xcc */ FD_VALID,      /* 0xcd */ FD_CHECK_JMP,  /* 0xce */ FD_VALID,      /* 0xcf */ FD_VALID,
     213        8994 :     /* 0xd0 */ FD_INVALID,    /* 0xd1 */ FD_INVALID,    /* 0xd2 */ FD_INVALID,    /* 0xd3 */ FD_INVALID,
     214        8994 :     /* 0xd4 */ FD_INVALID,    /* 0xd5 */ FD_CHECK_JMP,  /* 0xd6 */ FD_CHECK_DIV,  /* 0xd7 */ FD_INVALID,
     215        8994 :     /* 0xd8 */ FD_INVALID,    /* 0xd9 */ FD_INVALID,    /* 0xda */ FD_INVALID,    /* 0xdb */ FD_INVALID,
     216        8994 :     /* 0xdc */ FD_CHECK_END,  /* 0xdd */ FD_CHECK_JMP,  /* 0xde */ FD_VALID,      /* 0xdf */ FD_INVALID,
     217        8994 :     /* 0xe0 */ FD_INVALID,    /* 0xe1 */ FD_INVALID,    /* 0xe2 */ FD_INVALID,    /* 0xe3 */ FD_INVALID,
     218        8994 :     /* 0xe4 */ FD_INVALID,    /* 0xe5 */ FD_INVALID,    /* 0xe6 */ FD_CHECK_DIV,  /* 0xe7 */ FD_INVALID,
     219        8994 :     /* 0xe8 */ FD_INVALID,    /* 0xe9 */ FD_INVALID,    /* 0xea */ FD_INVALID,    /* 0xeb */ FD_INVALID,
     220        8994 :     /* 0xec */ FD_INVALID,    /* 0xed */ FD_INVALID,    /* 0xee */ FD_VALID,      /* 0xef */ FD_INVALID,
     221        8994 :     /* 0xf0 */ FD_INVALID,    /* 0xf1 */ FD_INVALID,    /* 0xf2 */ FD_INVALID,    /* 0xf3 */ FD_INVALID,
     222        8994 :     /* 0xf4 */ FD_INVALID,    /* 0xf5 */ FD_INVALID,    /* 0xf6 */ FD_CHECK_DIV,  /* 0xf7 */ FD_VALID,
     223        8994 :     /* 0xf8 */ FD_INVALID,    /* 0xf9 */ FD_INVALID,    /* 0xfa */ FD_INVALID,    /* 0xfb */ FD_INVALID,
     224        8994 :     /* 0xfc */ FD_INVALID,    /* 0xfd */ FD_INVALID,    /* 0xfe */ FD_VALID,      /* 0xff */ FD_INVALID,
     225        8994 :   };
     226             : 
     227             :   /* SIMD-0173: LDDW
     228             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L232-L235 */
     229        8994 :   validation_map[ 0x18 ] = !FD_VM_SBPF_DISABLE_LDDW(sbpf_version) ? FD_CHECK_LDQ : FD_INVALID;
     230             :   /* HOR64 https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L325 */
     231        8994 :   validation_map[ 0xf7 ] = FD_VM_SBPF_DISABLE_LDDW(sbpf_version) ? FD_VALID   :    FD_INVALID;
     232             : 
     233             :   /* SIMD-0173: LE
     234             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L285 */
     235        8994 :   validation_map[ 0xd4 ] = !FD_VM_SBPF_DISABLE_LE(sbpf_version) ? FD_CHECK_END : FD_INVALID;
     236             : 
     237             :   /* SIMD-0173: LDXW, STW, STXW */
     238        8994 :   validation_map[ 0x61 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_VALID;
     239        8994 :   validation_map[ 0x62 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     240        8994 :   validation_map[ 0x63 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     241        8994 :   validation_map[ 0x8c ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_VALID     : FD_INVALID;
     242        8994 :   validation_map[ 0x87 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_VALID; /* VALID because it's NEG64 */
     243        8994 :   validation_map[ 0x8f ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_INVALID;
     244             : 
     245             :   /* SIMD-0173: LDXH, STH, STXH */
     246        8994 :   validation_map[ 0x69 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_VALID;
     247        8994 :   validation_map[ 0x6a ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     248        8994 :   validation_map[ 0x6b ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     249        8994 :   validation_map[ 0x3c ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_VALID     : FD_VALID;
     250        8994 :   validation_map[ 0x37 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_CHECK_DIV;
     251        8994 :   validation_map[ 0x3f ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_VALID;
     252             : 
     253             :   /* SIMD-0173: LDXB, STB, STXB */
     254        8994 :   validation_map[ 0x71 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_VALID;
     255        8994 :   validation_map[ 0x72 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     256        8994 :   validation_map[ 0x73 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     257        8994 :   validation_map[ 0x2c ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_VALID     : FD_VALID;
     258        8994 :   validation_map[ 0x27 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_VALID;
     259        8994 :   validation_map[ 0x2f ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_VALID;
     260             : 
     261             :   /* SIMD-0173: LDXDW, STDW, STXDW */
     262        8994 :   validation_map[ 0x79 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_VALID;
     263        8994 :   validation_map[ 0x7a ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     264        8994 :   validation_map[ 0x7b ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_INVALID   : FD_CHECK_ST;
     265        8994 :   validation_map[ 0x9c ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_VALID     : FD_VALID;
     266        8994 :   validation_map[ 0x97 ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_CHECK_DIV;
     267        8994 :   validation_map[ 0x9f ] = FD_VM_SBPF_MOVE_MEMORY_IX_CLASSES(sbpf_version) ? FD_CHECK_ST  : FD_VALID;
     268             : 
     269             :   /* SIMD-0173 / SIMD-0377: CALLX references valid register numbers
     270             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L198-L214 */
     271        8994 :   validation_map[ 0x8d ] = FD_VM_SBPF_CALLX_USES_SRC_REG(sbpf_version)  ? FD_CHECK_CALL_REG_SRC
     272        8994 :                           : FD_VM_SBPF_CALLX_USES_DST_REG(sbpf_version) ? FD_CHECK_CALL_REG_DST
     273        4659 :                           : FD_CHECK_CALL_REG_IMM;
     274             : 
     275             :   /* SIMD-0174: MUL, DIV, MOD */
     276        8994 :   validation_map[ 0x24 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_INVALID : FD_VALID;
     277        8994 :   validation_map[ 0x34 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_INVALID : FD_CHECK_DIV;
     278        8994 :   validation_map[ 0x94 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_INVALID : FD_CHECK_DIV;
     279             :   /* note: 0x?c, 0x?7, 0x?f should not be overwritten because they're now load/store ix */
     280             : 
     281             :   /* SIMD-0174: NEG
     282             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L274 */
     283        8994 :   validation_map[ 0x84 ] = !FD_VM_SBPF_DISABLE_NEG(sbpf_version) ? FD_VALID : FD_INVALID;
     284             :   /* note: 0x87 should not be overwritten because it was NEG64 and it becomes STW */
     285             : 
     286             :   /* SIMD-0174: MUL, DIV, MOD */
     287        8994 :   validation_map[ 0x36 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID; /* UHMUL64 */
     288        8994 :   validation_map[ 0x3e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     289        8994 :   validation_map[ 0x46 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* UDIV32 */
     290        8994 :   validation_map[ 0x4e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     291        8994 :   validation_map[ 0x56 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* UDIV64 */
     292        8994 :   validation_map[ 0x5e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     293        8994 :   validation_map[ 0x66 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* UREM32 */
     294        8994 :   validation_map[ 0x6e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     295        8994 :   validation_map[ 0x76 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* UREM64 */
     296        8994 :   validation_map[ 0x7e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     297        8994 :   validation_map[ 0x86 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID; /* LMUL32 */
     298        8994 :   validation_map[ 0x8e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     299        8994 :   validation_map[ 0x96 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID; /* LMUL64 */
     300        8994 :   validation_map[ 0x9e ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     301        8994 :   validation_map[ 0xb6 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID; /* SHMUL64 */
     302        8994 :   validation_map[ 0xbe ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     303        8994 :   validation_map[ 0xc6 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* SDIV32 */
     304        8994 :   validation_map[ 0xce ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     305        8994 :   validation_map[ 0xd6 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* SDIV64 */
     306        8994 :   validation_map[ 0xde ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     307        8994 :   validation_map[ 0xe6 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* SREM32 */
     308        8994 :   validation_map[ 0xee ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     309        8994 :   validation_map[ 0xf6 ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_CHECK_DIV : FD_INVALID; /* SREM64 */
     310        8994 :   validation_map[ 0xfe ] = FD_VM_SBPF_ENABLE_PQR (sbpf_version) ? FD_VALID     : FD_INVALID;
     311             : 
     312             :   /* SIMD-0377: JMP32
     313             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L354-L375
     314             : 
     315             :      Note that these override some opcodes that conflict with PQR, so
     316             :      these must come AFTER PQR. */
     317        8994 :   if( FD_VM_SBPF_ENABLE_JMP32( sbpf_version ) ) {
     318         633 :     validation_map[ 0x16 ] = FD_CHECK_JMP;  /* JEQ32_IMM  */
     319         633 :     validation_map[ 0x1e ] = FD_CHECK_JMP;  /* JEQ32_REG  */
     320         633 :     validation_map[ 0x26 ] = FD_CHECK_JMP;  /* JGT32_IMM  */
     321         633 :     validation_map[ 0x2e ] = FD_CHECK_JMP;  /* JGT32_REG  */
     322         633 :     validation_map[ 0x36 ] = FD_CHECK_JMP;  /* JGE32_IMM  */
     323         633 :     validation_map[ 0x3e ] = FD_CHECK_JMP;  /* JGE32_REG  */
     324         633 :     validation_map[ 0x46 ] = FD_CHECK_JMP;  /* JSET32_IMM */
     325         633 :     validation_map[ 0x4e ] = FD_CHECK_JMP;  /* JSET32_REG */
     326         633 :     validation_map[ 0x56 ] = FD_CHECK_JMP;  /* JNE32_IMM  */
     327         633 :     validation_map[ 0x5e ] = FD_CHECK_JMP;  /* JNE32_REG  */
     328         633 :     validation_map[ 0x66 ] = FD_CHECK_JMP;  /* JSGT32_IMM */
     329         633 :     validation_map[ 0x6e ] = FD_CHECK_JMP;  /* JSGT32_REG */
     330         633 :     validation_map[ 0x76 ] = FD_CHECK_JMP;  /* JSGE32_IMM */
     331         633 :     validation_map[ 0x7e ] = FD_CHECK_JMP;  /* JSGE32_REG */
     332         633 :     validation_map[ 0xa6 ] = FD_CHECK_JMP;  /* JLT32_IMM  */
     333         633 :     validation_map[ 0xae ] = FD_CHECK_JMP;  /* JLT32_REG  */
     334         633 :     validation_map[ 0xb6 ] = FD_CHECK_JMP;  /* JLE32_IMM  */
     335         633 :     validation_map[ 0xbe ] = FD_CHECK_JMP;  /* JLE32_REG  */
     336         633 :     validation_map[ 0xc6 ] = FD_CHECK_JMP;  /* JSLT32_IMM */
     337         633 :     validation_map[ 0xce ] = FD_CHECK_JMP;  /* JSLT32_REG */
     338         633 :     validation_map[ 0xd6 ] = FD_CHECK_JMP;  /* JSLE32_IMM */
     339         633 :     validation_map[ 0xde ] = FD_CHECK_JMP;  /* JSLE32_REG */
     340         633 :   }
     341             : 
     342             :   /* FIXME: These checks are not necessary assuming fd_vm_t is populated by metadata
     343             :      generated in fd_sbpf_elf_peek (which performs these checks). But there is no guarantee, and
     344             :      this non-guarantee is (rightfully) exploited by the fuzz harnesses.
     345             :      Agave doesn't perform these checks explicitly due to Rust's guarantees  */
     346        8994 :   if( FD_UNLIKELY( vm->text_sz / 8UL != vm->text_cnt ||
     347        8994 :                    (const uchar *)vm->text < vm->rodata ||
     348        8994 :                    (ulong)vm->text > (ulong)vm->text + vm->text_sz || /* Overflow chk */
     349        8994 :                    (const uchar *)vm->text + vm->text_sz > vm->rodata + vm->rodata_sz +
     350        8994 :                      ( FD_VM_SBPF_ENABLE_STRICTER_ELF_HEADERS( sbpf_version ) ? vm->text_sz : 0UL ) ) )
     351           0 :     return FD_VM_ERR_BAD_TEXT;
     352             : 
     353        8994 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( vm->text_sz, 8UL ) ) ) /* https://github.com/solana-labs/rbpf/blob/v0.8.0/src/verifier.rs#L109 */
     354           0 :     return FD_VM_TEXT_SZ_UNALIGNED;
     355             : 
     356        8994 :   if ( FD_UNLIKELY( vm->text_cnt == 0UL ) ) /* https://github.com/solana-labs/rbpf/blob/v0.8.0/src/verifier.rs#L112 */
     357           0 :     return FD_VM_ERR_EMPTY;
     358             : 
     359        8994 :   ulong const * text     = vm->text;
     360        8994 :   ulong         text_cnt = vm->text_cnt;
     361             : 
     362   805592361 :   for( ulong i=0UL; i<text_cnt; i++ ) {
     363   805586133 :     fd_sbpf_instr_t instr = fd_sbpf_instr( text[i] );
     364             : 
     365   805586133 :     uchar validation_code = validation_map[ instr.opcode.raw ];
     366   805586133 :     switch( validation_code ) {
     367             : 
     368   644444604 :     case FD_VALID: break;
     369             : 
     370             :     /* Store ops are special because they allow dreg==r10.
     371             :        We use a special validation_code, used later in the
     372             :        "Check registers" section.
     373             :        But there's nothing to do at this time. */
     374       30222 :     case FD_CHECK_ST: break;
     375             : 
     376       41073 :     case FD_CHECK_JMP: {
     377       41073 :       long jmp_dst = (long)i + (long)instr.offset + 1L;
     378       41073 :       if( FD_UNLIKELY( (jmp_dst<0) | (jmp_dst>=(long)text_cnt)                          ) ) return FD_VM_ERR_JMP_OUT_OF_BOUNDS;
     379             :       //FIXME: this shouldn't be here?
     380       40926 :       if( FD_UNLIKELY( fd_sbpf_instr( text[ jmp_dst ] ).opcode.raw==FD_SBPF_OP_ADDL_IMM ) ) return FD_VM_ERR_JMP_TO_ADDL_IMM;
     381       40926 :       break;
     382       40926 :     }
     383             : 
     384       40926 :     case FD_CHECK_END: {
     385          96 :       if( FD_UNLIKELY( !((instr.imm==16) | (instr.imm==32) | (instr.imm==64)) ) ) return FD_VM_ERR_INVALID_END_IMM;
     386          78 :       break;
     387          96 :     }
     388             : 
     389             :     /* https://github.com/solana-labs/rbpf/blob/b503a1867a9cfa13f93b4d99679a17fe219831de/src/verifier.rs#L244 */
     390       10341 :     case FD_CHECK_LDQ: {
     391             :       /* https://github.com/solana-labs/rbpf/blob/b503a1867a9cfa13f93b4d99679a17fe219831de/src/verifier.rs#L131 */
     392       10341 :       if( FD_UNLIKELY( (i+1UL)>=text_cnt ) ) return FD_VM_ERR_INCOMPLETE_LDQ;
     393             : 
     394             :       /* https://github.com/solana-labs/rbpf/blob/b503a1867a9cfa13f93b4d99679a17fe219831de/src/verifier.rs#L137-L139 */
     395       10341 :       fd_sbpf_instr_t addl_imm = fd_sbpf_instr( text[ i+1UL ] );
     396       10341 :       if( FD_UNLIKELY( addl_imm.opcode.raw!=FD_SBPF_OP_ADDL_IMM ) ) return FD_VM_ERR_LDQ_NO_ADDL_IMM;
     397             : 
     398             :       /* FIXME: SET A BIT MAP HERE OF ADDL_IMM TO DENOTE * AS FORBIDDEN
     399             :          BRANCH TARGETS OF CALL_REG?? */
     400             : 
     401       10341 :       i++; /* Skip the addl imm */
     402       10341 :       break;
     403       10341 :     }
     404             : 
     405    64407210 :     case FD_CHECK_DIV: {
     406    64407210 :       if( FD_UNLIKELY( instr.imm==0 ) ) return FD_VM_ERR_SIGFPE;
     407    64407150 :       break;
     408    64407210 :     }
     409             : 
     410    64407150 :     case FD_CHECK_SH32: {
     411    48308418 :       if( FD_UNLIKELY( instr.imm>=32 ) ) return FD_VM_SH_OVERFLOW;
     412    48308328 :       break;
     413    48308418 :     }
     414             : 
     415    48338889 :     case FD_CHECK_SH64: {
     416    48338889 :       if( FD_UNLIKELY( instr.imm>=64 ) ) return FD_VM_SH_OVERFLOW;
     417    48338781 :       break;
     418    48338889 :     }
     419             : 
     420             :     /* https://github.com/anza-xyz/sbpf/blob/v0.11.1/src/verifier.rs#L220 */
     421    48338781 :     case FD_CHECK_CALL_REG_SRC: {
     422          21 :       if( FD_UNLIKELY( instr.src_reg > 9 ) ) {
     423           9 :         return FD_VM_ERR_INVALID_REG;
     424           9 :       }
     425          12 :       break;
     426          21 :     }
     427        4068 :     case FD_CHECK_CALL_REG_IMM: {
     428        4068 :       if( FD_UNLIKELY( instr.imm > 9 ) ) {
     429           9 :         return FD_VM_ERR_INVALID_REG;
     430           9 :       }
     431        4059 :       break;
     432        4068 :     }
     433             :     /* https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/verifier.rs#L205-L206 */
     434        4059 :     case FD_CHECK_CALL_REG_DST: {
     435           9 :       if( FD_UNLIKELY( instr.dst_reg > 9 ) ) {
     436           3 :         return FD_VM_ERR_INVALID_REG;
     437           3 :       }
     438           6 :       break;
     439           9 :     }
     440             : 
     441        1182 :     case FD_INVALID: default: return FD_VM_ERR_INVALID_OPCODE;
     442   805586133 :     }
     443             : 
     444             :     /* Check registers
     445             :        https://github.com/solana-labs/rbpf/blob/v0.8.5/src/verifier.rs#L177 */
     446             : 
     447             :     /* Source register */
     448   805584507 :     if( FD_UNLIKELY( instr.src_reg>10 ) ) return FD_VM_ERR_INVALID_SRC_REG;
     449             : 
     450             :     /* Special R10 register allowed for ADD64_IMM */
     451   805584039 :     if( instr.dst_reg==10U
     452   805584039 :         && FD_VM_SBPF_MANUAL_STACK_FRAME_BUMP( sbpf_version )
     453   805584039 :         && instr.opcode.raw == 0x07
     454   805584039 :         && ( instr.imm % FD_VM_SBPF_DYNAMIC_STACK_FRAMES_ALIGN )==0 )
     455           9 :         continue;
     456             : 
     457             :     /* Destination register. */
     458   805584030 :     if( FD_UNLIKELY( instr.dst_reg==10U && validation_code != FD_CHECK_ST ) ) return FD_VM_ERR_INVALID_DST_REG;
     459   805583571 :     if( FD_UNLIKELY( instr.dst_reg > 10U ) ) return FD_VM_ERR_INVALID_DST_REG;
     460   805583571 :   }
     461             : 
     462        6228 :   return FD_VM_SUCCESS;
     463        8994 : }
     464             : 
     465             : FD_FN_CONST ulong
     466        1317 : fd_vm_align( void ) {
     467        1317 :   return FD_VM_ALIGN;
     468        1317 : }
     469             : 
     470             : FD_FN_CONST ulong
     471         648 : fd_vm_footprint( void ) {
     472         648 :   return FD_VM_FOOTPRINT;
     473         648 : }
     474             : 
     475             : void *
     476         600 : fd_vm_new( void * shmem ) {
     477             : 
     478         600 :   if( FD_UNLIKELY( !shmem ) ) {
     479           0 :     FD_LOG_WARNING(( "NULL shmem" ));
     480           0 :     return NULL;
     481           0 :   }
     482             : 
     483         600 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)shmem, fd_vm_align() ) ) ) {
     484           0 :     FD_LOG_WARNING(( "misaligned shmem" ));
     485           0 :     return NULL;
     486           0 :   }
     487             : 
     488         600 :   fd_vm_t * vm = (fd_vm_t *)shmem;
     489         600 :   fd_memset( vm, 0, fd_vm_footprint() );
     490             : 
     491         600 :   FD_COMPILER_MFENCE();
     492         600 :   FD_VOLATILE( vm->magic ) = FD_VM_MAGIC;
     493         600 :   FD_COMPILER_MFENCE();
     494             : 
     495         600 :   return shmem;
     496         600 : }
     497             : 
     498             : fd_vm_t *
     499         600 : fd_vm_join( void * shmem ) {
     500             : 
     501         600 :   if( FD_UNLIKELY( !shmem ) ) {
     502           0 :     FD_LOG_WARNING(( "NULL shmem" ));
     503           0 :     return NULL;
     504           0 :   }
     505             : 
     506         600 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)shmem, fd_vm_align() ) ) ) {
     507           0 :     FD_LOG_WARNING(( "misaligned shmem" ));
     508           0 :     return NULL;
     509           0 :   }
     510             : 
     511         600 :   fd_vm_t * vm = (fd_vm_t *)shmem;
     512             : 
     513         600 :   if( FD_UNLIKELY( vm->magic!=FD_VM_MAGIC ) ) {
     514           0 :     FD_LOG_WARNING(( "bad magic" ));
     515           0 :     return NULL;
     516           0 :   }
     517             : 
     518         600 :   return vm;
     519         600 : }
     520             : 
     521             : void *
     522          69 : fd_vm_leave( fd_vm_t * vm ) {
     523             : 
     524          69 :   if( FD_UNLIKELY( !vm ) ) {
     525           0 :     FD_LOG_WARNING(( "NULL vm" ));
     526           0 :     return NULL;
     527           0 :   }
     528             : 
     529          69 :   return (void *)vm;
     530          69 : }
     531             : 
     532             : void *
     533          69 : fd_vm_delete( void * shmem ) {
     534             : 
     535          69 :   if( FD_UNLIKELY( !shmem ) ) {
     536           0 :     FD_LOG_WARNING(( "NULL shmem" ));
     537           0 :     return NULL;
     538           0 :   }
     539             : 
     540          69 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)shmem, fd_vm_align() ) ) ) {
     541           0 :     FD_LOG_WARNING(( "misaligned shmem" ));
     542           0 :     return NULL;
     543           0 :   }
     544             : 
     545          69 :   fd_vm_t * vm = (fd_vm_t *)shmem;
     546             : 
     547          69 :   if( FD_UNLIKELY( vm->magic!=FD_VM_MAGIC ) ) {
     548           0 :     FD_LOG_WARNING(( "bad magic" ));
     549           0 :     return NULL;
     550           0 :   }
     551             : 
     552          69 :   FD_COMPILER_MFENCE();
     553          69 :   FD_VOLATILE( vm->magic ) = 0UL;
     554          69 :   FD_COMPILER_MFENCE();
     555             : 
     556          69 :   return (void *)vm;
     557          69 : }
     558             : 
     559             : fd_vm_t *
     560             : fd_vm_init(
     561             :    fd_vm_t *                 vm,
     562             :    fd_exec_instr_ctx_t *     instr_ctx,
     563             :    ulong                     heap_max,
     564             :    ulong                     entry_cu,
     565             :    uchar const *             rodata,
     566             :    ulong                     rodata_sz,
     567             :    ulong const *             text,
     568             :    ulong                     text_cnt,
     569             :    ulong                     text_off,
     570             :    ulong                     text_sz,
     571             :    ulong                     entry_pc,
     572             :    ulong const *             calldests,
     573             :    ulong                     sbpf_version,
     574             :    fd_sbpf_syscalls_t *      syscalls,
     575             :    fd_vm_trace_t *           trace,
     576             :    fd_sha256_t *             sha,
     577             :    fd_vm_input_region_t *    mem_regions,
     578             :    uint                      mem_regions_cnt,
     579             :    fd_vm_acc_region_meta_t * acc_region_metas,
     580             :    uchar                     is_deprecated,
     581             :    int                       direct_mapping,
     582             :    int                       syscall_parameter_address_restrictions,
     583             :    int                       virtual_address_space_adjustments,
     584             :    int                       dump_syscall_to_pb,
     585        9060 :    ulong                     r2_initial_value ) {
     586             : 
     587        9060 :   if ( FD_UNLIKELY( vm == NULL ) ) {
     588           0 :     FD_LOG_WARNING(( "NULL vm" ));
     589           0 :     return NULL;
     590           0 :   }
     591             : 
     592        9060 :   if ( FD_UNLIKELY( vm->magic != FD_VM_MAGIC ) ) {
     593           0 :     FD_LOG_WARNING(( "bad magic" ));
     594           0 :     return NULL;
     595           0 :   }
     596             : 
     597        9060 :   if ( FD_UNLIKELY( heap_max > FD_VM_HEAP_MAX ) ) {
     598           0 :     FD_LOG_WARNING(( "heap_max > FD_VM_HEAP_MAX" ));
     599           0 :     return NULL;
     600           0 :   }
     601             : 
     602             :   /* We do support calldests==NULL for tests that do not require
     603             :      program execution, e.g. just testing some interpreter functionality
     604             :      or syscalls.
     605             :      SBPF v3+ no longer needs calldests, so we enforce it to be NULL. */
     606        9060 :   if( FD_UNLIKELY( calldests && FD_VM_SBPF_ENABLE_STRICTER_ELF_HEADERS( sbpf_version ) ) ) {
     607           0 :     return NULL;
     608           0 :   }
     609             : 
     610             :   // Set the vm fields
     611        9060 :   vm->instr_ctx                              = instr_ctx;
     612        9060 :   vm->heap_max                               = heap_max;
     613        9060 :   vm->entry_cu                               = entry_cu;
     614        9060 :   vm->rodata                                 = rodata;
     615        9060 :   vm->rodata_sz                              = rodata_sz;
     616        9060 :   vm->text                                   = text;
     617        9060 :   vm->text_cnt                               = text_cnt;
     618             :   /* In SBPF V3, bytecode starts at vaddr 0x100000000 exactly, so
     619             :      text_off from the POV of the VM is 0.
     620             :      https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/interpreter.rs#L539 */
     621        9060 :   vm->text_off                               = FD_VM_SBPF_ENABLE_LOWER_RODATA_VADDR( sbpf_version ) ? 0UL : text_off;
     622        9060 :   vm->text_sz                                = text_sz;
     623        9060 :   vm->entry_pc                               = entry_pc;
     624        9060 :   vm->calldests                              = calldests;
     625        9060 :   vm->sbpf_version                           = sbpf_version;
     626        9060 :   vm->syscalls                               = syscalls;
     627        9060 :   vm->trace                                  = trace;
     628        9060 :   vm->sha                                    = sha;
     629        9060 :   vm->input_mem_regions                      = mem_regions;
     630        9060 :   vm->input_mem_regions_cnt                  = mem_regions_cnt;
     631        9060 :   vm->acc_region_metas                       = acc_region_metas;
     632        9060 :   vm->is_deprecated                          = is_deprecated;
     633        9060 :   vm->direct_mapping                         = direct_mapping;
     634        9060 :   vm->syscall_parameter_address_restrictions = syscall_parameter_address_restrictions;
     635        9060 :   vm->virtual_address_space_adjustments      = virtual_address_space_adjustments;
     636             :   /* https://github.com/anza-xyz/agave/blob/v4.0.0-beta.3/program-runtime/src/vm.rs#L99-L103 */
     637        9060 :   if( FD_UNLIKELY( FD_VM_SBPF_STACK_FRAME_GAPS( sbpf_version ) && !virtual_address_space_adjustments ) ) {
     638        4005 :     vm->stack_frame_sz                       = FD_VM_STACK_FRAME_SZ;
     639        4005 :     vm->stack_push_frame_count               = 2UL;
     640        5055 :   } else if ( FD_VM_SBPF_MANUAL_STACK_FRAME_BUMP( sbpf_version ) ) {
     641        4341 :     vm->stack_frame_sz                       = 0UL;
     642        4341 :     vm->stack_push_frame_count               = 0UL;
     643        4341 :   } else {
     644         714 :     vm->stack_frame_sz                       = FD_VM_STACK_FRAME_SZ;
     645         714 :     vm->stack_push_frame_count               = 1UL;
     646         714 :   }
     647             : 
     648        9060 :   vm->segv_vaddr                             = ULONG_MAX;
     649        9060 :   vm->segv_access_len                        = 0UL;
     650        9060 :   vm->segv_access_type                       = 0;
     651        9060 :   vm->dump_syscall_to_pb                     = dump_syscall_to_pb;
     652             : 
     653             :   /* Unpack input and rodata */
     654        9060 :   fd_vm_mem_cfg( vm );
     655             : 
     656             :   /* Initialize registers */
     657             :   /* FIXME: Zero out shadow, stack and heap here? */
     658        9060 :   fd_memset( vm->reg, 0, FD_VM_REG_MAX * sizeof(ulong) );
     659        9060 :   vm->reg[1]  = FD_VM_MEM_MAP_INPUT_REGION_START;
     660        9060 :   vm->reg[2]  = r2_initial_value;
     661             :   /* https://github.com/anza-xyz/sbpf/blob/v0.14.4/src/vm.rs#L325-L330 */
     662        9060 :   vm->reg[10] = FD_VM_MEM_MAP_STACK_REGION_START +
     663        9060 :     ( FD_VM_SBPF_MANUAL_STACK_FRAME_BUMP( vm->sbpf_version ) ? FD_VM_STACK_MAX : vm->stack_frame_sz );
     664             :   /* Note: Agave uses r11 as pc, we don't */
     665             : 
     666             :   /* Set execution state */
     667        9060 :   vm->pc        = vm->entry_pc;
     668        9060 :   vm->ic        = 0UL;
     669        9060 :   vm->cu        = vm->entry_cu;
     670        9060 :   vm->frame_cnt = 0UL;
     671             : 
     672        9060 :   vm->heap_sz = 0UL;
     673             : 
     674             :   /* Do NOT reset logs */
     675             : 
     676        9060 :   return vm;
     677        9060 : }

Generated by: LCOV version 1.14