LCOV - code coverage report
Current view: top level - flamenco/vm/jit - fd_jit_compiler.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 37 579 6.4 %
Date: 2025-01-08 12:08:44 Functions: 2 4 50.0 %

          Line data    Source code
       1             : /*
       2             : ** This file has been pre-processed with DynASM.
       3             : ** https://luajit.org/dynasm.html
       4             : ** DynASM version 1.5.0, DynASM x64 version 1.5.0
       5             : ** DO NOT EDIT! The original file is in "fd_jit_compiler.dasc".
       6             : */
       7             : 
       8             : #line 1 "fd_jit_compiler.dasc"
       9             : /* WARNING: WORK IN PROGRESS!
      10             :    This is an experimental version of the Firedancer JIT compiler.
      11             :    It is disabled in production.  There are known security issues in
      12             :    this code.  It is not covered by the Firedancer bug bounty policy.
      13             : 
      14             :    ### fd_vm_interp compatibility
      15             : 
      16             :    fd_jit aims for exact compatibility with fd_vm_interp unless
      17             :    otherwise noted.
      18             : 
      19             :    ### Optimizations
      20             : 
      21             :    The fd_jit transpiler does not implement any optimizations.
      22             : 
      23             :    ### Error Handling
      24             : 
      25             :    VM faults (e.g. segfault, divide by zero, CU overrun, invalid call
      26             :    destination, program EOF, syscall error) are detected reliably and
      27             :    deterministically.
      28             : 
      29             :    However, after a fault happens, any VM state becomes unreliable
      30             :    (non-deterministic or undefined).  Unreliable fault state include CU
      31             :    left, current PC, and the error code.
      32             : 
      33             :    A VM fault usually triggers ->longjmp, transitioning back into the
      34             :    fd_jit_execute call frame. */
      35             : 
      36             : #include "../../../util/fd_util_base.h"
      37             : 
      38             : /* Include dynasm headers.  These fail to compile when some strict
      39             :    checks are enabled. */
      40             : 
      41             : void fd_dasm_grow_check( void * ptr, ulong min_sz );
      42           0 : #define DASM_M_GROW(ctx, t, p, sz, need) (fd_dasm_grow_check( (p), (need) ))
      43           0 : #define DASM_M_FREE(ctx, p, sz) do{}while(0)
      44             : 
      45             : #pragma GCC diagnostic push
      46             : #pragma GCC diagnostic ignored "-Wconversion"
      47             : #pragma GCC diagnostic ignored "-Wsign-conversion"
      48             : #pragma GCC diagnostic ignored "-Wimplicit-fallthrough"
      49             : #include "dasm_proto.h"
      50             : #include "dasm_x86.h"
      51             : #pragma GCC diagnostic pop
      52             : 
      53             : #include "fd_jit_private.h"
      54             : #include "../fd_vm_private.h"
      55             : 
      56             : //| .arch x64
      57             : #if DASM_VERSION != 10500
      58             : #error "Version mismatch between DynASM and included encoding engine"
      59             : #endif
      60             : #line 49 "fd_jit_compiler.dasc"
      61             : //| .actionlist actions
      62             : static const unsigned char actions[1657] = {
      63             :   254,0,248,10,100,72,139,4,37,237,72,137,176,233,76,137,152,233,76,137,160,
      64             :   233,76,137,168,233,76,137,176,233,76,137,184,233,72,137,152,233,72,137,136,
      65             :   233,76,137,128,233,76,137,136,233,76,137,144,233,195,255,248,11,100,72,139,
      66             :   4,37,237,72,139,176,233,76,139,152,233,76,139,160,233,76,139,168,233,76,139,
      67             :   176,233,76,139,184,233,72,139,152,233,72,139,136,233,76,139,128,233,76,139,
      68             :   136,233,76,139,144,233,195,255,248,12,137,252,250,72,193,252,239,32,209,231,
      69             :   252,233,244,13,255,248,14,137,252,250,72,193,252,239,32,141,60,125,1,0,0,
      70             :   0,255,248,13,255,252,233,244,15,255,252,233,244,16,255,248,15,137,252,250,
      71             :   72,193,252,239,32,100,59,60,37,237,15,131,244,17,1,213,15,130,244,17,100,
      72             :   59,20,189,237,15,131,244,17,49,252,237,131,252,255,2,184,0,16,0,0,15,68,232,
      73             :   33,252,234,15,131,244,17,100,72,3,20,252,253,237,195,255,248,16,100,59,60,
      74             :   37,237,15,131,244,17,1,213,15,130,244,17,100,59,20,189,237,15,131,244,17,
      75             :   100,72,3,20,252,253,237,195,255,248,17,72,193,231,32,72,9,215,100,72,137,
      76             :   60,37,237,72,139,60,36,100,72,137,60,37,237,252,233,244,18,255,248,19,73,
      77             :   187,237,237,73,137,155,233,73,137,171,233,77,137,99,16,77,137,107,24,77,137,
      78             :   115,32,77,137,123,40,72,141,84,36,8,73,137,83,48,72,139,20,36,73,137,83,56,
      79             :   49,192,49,210,195,255,248,20,72,137,252,248,186,1,0,0,0,72,191,237,237,72,
      80             :   139,159,233,72,139,175,233,76,139,103,16,76,139,111,24,76,139,119,32,76,139,
      81             :   127,40,72,139,103,48,252,255,119,56,195,255,248,21,232,244,10,232,244,10,
      82             :   72,137,229,72,131,228,252,240,72,131,252,236,16,100,76,139,28,37,237,100,
      83             :   76,139,151,233,76,137,223,72,139,176,233,72,139,144,233,72,139,136,233,76,
      84             :   139,128,233,76,139,136,233,76,141,152,233,65,83,65,252,255,210,133,252,255,
      85             :   15,133,244,18,72,137,252,236,232,244,11,195,255,248,22,100,72,139,60,37,237,
      86             :   72,252,255,135,233,95,83,81,65,80,65,81,73,129,194,0,32,0,0,252,255,231,255,
      87             :   248,23,100,72,139,60,37,237,72,252,255,143,233,95,65,89,65,88,89,91,73,129,
      88             :   252,234,0,32,0,0,252,255,231,255,248,18,191,237,252,233,244,20,255,248,24,
      89             :   255,232,244,19,133,210,15,133,244,247,255,232,244,11,72,131,252,236,32,252,
      90             :   255,215,72,137,252,247,232,244,20,248,1,195,255,249,255,64,129,192,240,43,
      91             :   239,255,252,233,245,255,72,129,192,240,35,239,255,64,1,192,240,131,240,51,
      92             :   255,72,1,192,240,131,240,35,255,64,129,232,240,43,239,255,72,129,252,248,
      93             :   240,35,239,255,15,132,245,255,72,129,232,240,35,239,255,64,49,192,240,131,
      94             :   240,51,255,72,199,192,240,35,237,255,64,41,192,240,131,240,51,255,72,57,192,
      95             :   240,131,240,35,15,132,245,255,72,41,192,240,131,240,35,255,64,105,192,240,
      96             :   131,240,51,239,255,72,129,252,248,240,35,239,15,135,245,255,72,105,192,240,
      97             :   131,240,35,239,255,64,15,175,192,240,132,240,52,255,72,57,192,240,131,240,
      98             :   35,15,135,245,255,72,15,175,192,240,132,240,36,255,64,144,240,42,49,210,191,
      99             :   237,252,247,252,247,64,144,240,42,255,72,129,252,248,240,35,239,15,131,245,
     100             :   255,72,144,240,34,49,210,72,199,199,237,72,252,247,252,247,72,144,240,34,
     101             :   255,64,133,192,240,131,240,51,15,132,244,18,255,64,184,240,42,1,0,0,0,255,
     102             :   64,144,240,42,49,210,64,252,247,252,240,240,43,64,144,240,42,255,72,57,192,
     103             :   240,131,240,35,15,131,245,255,72,133,192,240,131,240,35,15,132,244,18,255,
     104             :   72,144,240,34,49,210,72,252,247,252,240,240,35,72,144,240,34,255,64,129,200,
     105             :   240,43,239,255,72,252,247,192,240,35,237,15,133,245,255,72,129,200,240,35,
     106             :   239,255,64,9,192,240,131,240,51,255,72,133,192,240,131,240,35,15,133,245,
     107             :   255,72,9,192,240,131,240,35,255,64,129,224,240,43,239,255,72,129,252,248,
     108             :   240,35,239,15,133,245,255,72,129,224,240,35,239,255,64,33,192,240,131,240,
     109             :   51,255,72,57,192,240,131,240,35,15,133,245,255,72,33,192,240,131,240,35,255,
     110             :   72,141,184,253,240,3,233,189,3,0,0,0,232,244,12,64,139,2,240,139,255,72,141,
     111             :   184,253,240,3,233,189,3,0,0,0,232,244,14,199,2,237,255,72,141,184,253,240,
     112             :   3,233,189,3,0,0,0,232,244,14,64,137,2,240,139,255,64,193,224,240,43,235,255,
     113             :   72,129,252,248,240,35,239,15,143,245,255,72,193,224,240,35,235,255,72,141,
     114             :   184,253,240,3,233,189,1,0,0,0,232,244,12,64,49,192,240,131,240,51,102,64,
     115             :   139,2,240,139,255,72,141,184,253,240,3,233,189,1,0,0,0,232,244,14,102,199,
     116             :   2,236,255,72,141,184,253,240,3,233,189,1,0,0,0,232,244,14,64,137,2,240,139,
     117             :   255,64,136,193,240,131,64,211,224,240,43,255,72,57,192,240,131,240,35,15,
     118             :   143,245,255,64,136,193,240,131,72,211,224,240,35,255,72,141,184,253,240,3,
     119             :   233,49,252,237,232,244,12,255,64,49,192,240,131,240,51,64,138,2,240,131,255,
     120             :   72,141,184,253,240,3,233,49,252,237,232,244,14,198,2,235,255,72,141,184,253,
     121             :   240,3,233,49,252,237,232,244,14,64,136,2,240,131,255,64,193,232,240,43,235,
     122             :   255,72,129,252,248,240,35,239,15,141,245,255,72,193,232,240,35,235,255,72,
     123             :   141,184,253,240,3,233,189,7,0,0,0,232,244,12,72,139,2,240,131,255,72,141,
     124             :   184,253,240,3,233,189,7,0,0,0,232,244,14,199,2,237,199,66,4,0,0,0,0,255,72,
     125             :   141,184,253,240,3,233,189,7,0,0,0,232,244,14,72,137,2,240,131,255,64,136,
     126             :   193,240,131,64,211,232,240,43,255,72,57,192,240,131,240,35,15,141,245,255,
     127             :   64,136,193,240,131,72,211,232,240,35,255,64,252,247,216,240,43,255,232,244,
     128             :   22,232,245,255,72,199,199,237,232,244,21,255,72,252,247,216,240,35,255,64,
     129             :   144,240,42,49,210,191,237,252,247,252,247,64,135,208,240,43,255,232,244,23,
     130             :   195,255,64,184,240,42,0,0,0,0,255,64,144,240,42,49,210,64,252,247,252,240,
     131             :   240,43,64,135,208,240,43,255,72,144,240,34,49,210,72,252,247,252,240,240,
     132             :   35,72,135,208,240,35,255,64,129,252,240,240,43,239,255,72,129,252,248,240,
     133             :   35,239,15,130,245,255,72,129,252,240,240,35,239,255,72,57,192,240,131,240,
     134             :   35,15,130,245,255,72,49,192,240,131,240,35,255,64,184,240,42,237,255,72,129,
     135             :   252,248,240,35,239,15,134,245,255,64,137,192,240,131,240,51,255,72,57,192,
     136             :   240,131,240,35,15,134,245,255,72,137,192,240,131,240,35,255,64,193,252,248,
     137             :   240,43,235,255,72,129,252,248,240,35,239,15,140,245,255,72,193,252,248,240,
     138             :   35,235,255,64,136,193,240,131,64,211,252,248,240,43,255,72,57,192,240,131,
     139             :   240,35,15,140,245,255,64,136,193,240,131,72,211,252,248,240,35,255,72,129,
     140             :   252,248,240,35,239,15,142,245,255,64,15,183,192,240,132,240,52,102,64,193,
     141             :   200,240,43,8,255,64,15,200,240,43,255,72,15,200,240,35,255,72,57,192,240,
     142             :   131,240,35,15,142,245,255,248,25,252,233,244,18,255
     143             : };
     144             : 
     145             : #line 50 "fd_jit_compiler.dasc"
     146             : //| .globals fd_jit_lbl_
     147             : enum {
     148             :   fd_jit_lbl_save_regs,
     149             :   fd_jit_lbl_restore_regs,
     150             :   fd_jit_lbl_fd_jit_vm_translate_ro,
     151             :   fd_jit_lbl_fd_jit_vm_translate,
     152             :   fd_jit_lbl_fd_jit_vm_translate_rw,
     153             :   fd_jit_lbl_fd_jit_vm_translate_abiv1,
     154             :   fd_jit_lbl_fd_jit_vm_translate_abiv2,
     155             :   fd_jit_lbl_translate_fail,
     156             :   fd_jit_lbl_vm_fault,
     157             :   fd_jit_lbl_setjmp,
     158             :   fd_jit_lbl_longjmp,
     159             :   fd_jit_lbl_emulate_syscall,
     160             :   fd_jit_lbl_call_stack_push,
     161             :   fd_jit_lbl_call_stack_pop,
     162             :   fd_jit_lbl_entrypoint,
     163             :   fd_jit_lbl_overrun,
     164             :   fd_jit_lbl__MAX
     165             : };
     166             : #line 51 "fd_jit_compiler.dasc"
     167             : //| .section code
     168             : #define DASM_SECTION_CODE       0
     169           3 : #define DASM_MAXSECTION         1
     170             : #line 52 "fd_jit_compiler.dasc"
     171             : 
     172             : fd_jit_scratch_layout_t *
     173             : fd_jit_scratch_layout( fd_jit_scratch_layout_t * layout,
     174           6 :                        ulong                     bpf_sz ) {
     175             : 
     176           6 :   if( FD_UNLIKELY( bpf_sz > (1UL<<24) ) ) return NULL;
     177           6 :   ulong text_cnt = bpf_sz / 8UL;
     178             : 
     179             :   /* These "magic" values are taken from dasm_x86.h */
     180             : 
     181           6 :   ulong dasm_sz     = DASM_PSZ( DASM_MAXSECTION );       /* dasm_x86.h(89) */
     182           6 :   ulong lglabels_sz = (10+fd_jit_lbl__MAX)*sizeof(int);  /* dasm_x86.h(119) */
     183           6 :   ulong pclabels_sz = text_cnt*sizeof(int);              /* dasm_x86.h(127) */
     184           6 :   ulong code_sz     = fd_jit_est_code_sz( bpf_sz );
     185             : 
     186           6 :   memset( layout, 0, sizeof(fd_jit_scratch_layout_t) );
     187           6 :   FD_SCRATCH_ALLOC_INIT( l, 0 );
     188           6 :   layout->dasm_off     = (ulong)FD_SCRATCH_ALLOC_APPEND( l, 16UL, dasm_sz     );
     189           0 :   layout->dasm_sz      = dasm_sz;
     190           6 :   layout->lglabels_off = (ulong)FD_SCRATCH_ALLOC_APPEND( l, 16UL, lglabels_sz );
     191           0 :   layout->lglabels_sz  = lglabels_sz;
     192           6 :   layout->pclabels_off = (ulong)FD_SCRATCH_ALLOC_APPEND( l, 16UL, pclabels_sz );
     193           0 :   layout->pclabels_sz  = pclabels_sz;
     194           6 :   layout->code_off     = (ulong)FD_SCRATCH_ALLOC_APPEND( l, 16UL, code_sz     );
     195           0 :   layout->code_sz      = code_sz;
     196           6 :   layout->sz           = (ulong)FD_SCRATCH_ALLOC_FINI( l, 16UL );
     197           6 :   return layout;
     198           6 : }
     199             : 
     200             : dasm_State *
     201             : fd_jit_prepare( void *                          scratch,
     202           3 :                 fd_jit_scratch_layout_t const * layout ) {
     203             : 
     204             :   /* Custom dasm_init */
     205           3 :   dasm_State * d = (void *)( (ulong)scratch + layout->dasm_off );
     206           3 :   fd_memset( d, 0, layout->dasm_sz );
     207           3 :   d->psize      = layout->dasm_sz;
     208           3 :   d->maxsection = DASM_MAXSECTION;
     209             : 
     210             :   /* Custom dasm_setupglobal */
     211           3 :   d->globals  = fd_jit_labels;
     212           3 :   d->lglabels = (void *)( (ulong)scratch + layout->lglabels_off );
     213           3 :   d->lgsize   = layout->lglabels_sz;
     214             : 
     215             :   /* Custom dasm_growpc */
     216           3 :   d->pcsize   = layout->pclabels_sz;
     217           3 :   d->pclabels = (void *)( (ulong)scratch + layout->pclabels_off );
     218             : 
     219             :   /* Setup encoder. Zeros lglabels and pclabels. */
     220           3 :   dasm_setup( &d, actions );
     221             : 
     222             :   /* Preallocate space for .code section
     223             :      See dasm_x86.h(172) */
     224           3 :   dasm_Section * code = d->sections + 0;
     225           3 :   code->buf   = (void *)( (ulong)scratch + layout->code_off );
     226           3 :   code->bsize = layout->code_sz;
     227           3 :   code->pos   = 0;
     228           3 :   code->rbuf  = code->buf;
     229           3 :   code->epos  = (int)( ((ulong)code->bsize / sizeof(int)) - DASM_MAXSECPOS );
     230           3 :   code->ofs   = 0;
     231             : 
     232           3 :   return d;
     233           3 : }
     234             : 
     235             : /* Compile time thread locals */
     236             : 
     237             : FD_TL void * fd_jit_labels[ FD_JIT_LABEL_CNT ];
     238             : FD_STATIC_ASSERT( sizeof(fd_jit_labels)==fd_jit_lbl__MAX*8, label_cnt );
     239             : 
     240             : /* Mapping between sBPF registers and x86_64 registers *****************
     241             : 
     242             :    This mapping is valid just before a translated sBPF instruction is
     243             :    about to be executed.  (At the `=>next_label` token in the code gen
     244             :    loop)
     245             : 
     246             :    BPF | r0  | r1  | r2  | r3  | r4  | r5  | r6  | r7  | r8  | r9  | r10
     247             :    X86 | rsi | r11 | r12 | r13 | r14 | r15 | rbx | rcx | r8  | r9  | r10
     248             : 
     249             :    x86_64 GPRs rax, rdi, rdx, rbp do not map to sBPF registers.  Those can
     250             :    be used as scratch registers for complex opcodes.
     251             : 
     252             :    Note that this mapping cannot be trivially changed.  Certain x86
     253             :    instructions (like div) have hardcoded register accesses which the
     254             :    JIT code works around.
     255             : 
     256             :    dynasm macros bpf_r{...} resolve to 64-bit register names.
     257             : 
     258             :    reg_bpf2x86 is indexed by sBPF register numbers and resolves to the
     259             :    x86_64 dynasm register index. */
     260             : 
     261             : static uchar const reg_bpf2x86[11] = {
     262             :   [ 0] =     FD_DASM_RSI,
     263             :   //| .define bpf_r0,  rsi
     264             :   [ 1] =     FD_DASM_R11,
     265             :   //| .define bpf_r1,  r11
     266             :   [ 2] =     FD_DASM_R12,
     267             :   //| .define bpf_r2,  r12
     268             :   [ 3] =     FD_DASM_R13,
     269             :   //| .define bpf_r3,  r13
     270             :   [ 4] =     FD_DASM_R14,
     271             :   //| .define bpf_r4,  r14
     272             :   [ 5] =     FD_DASM_R15,
     273             :   //| .define bpf_r5,  r15
     274             :   [ 6] =     FD_DASM_RBX,
     275             :   //| .define bpf_r6,  rbx
     276             :   [ 7] =     FD_DASM_RCX,
     277             :   //| .define bpf_r7,  rcx
     278             :   [ 8] =     FD_DASM_R8,
     279             :   //| .define bpf_r8,  r8
     280             :   [ 9] =     FD_DASM_R9,
     281             :   //| .define bpf_r9,  r9
     282             :   [10] =     FD_DASM_R10
     283             :   //| .define bpf_r10, r10
     284             : };
     285             : 
     286             : 
     287             : /* JIT compiler *******************************************************/
     288             : 
     289             : void
     290             : fd_jit_compile( struct dasm_State **       Dst,
     291             :                 fd_sbpf_program_t const *  prog,
     292           0 :                 fd_sbpf_syscalls_t const * syscalls ) {
     293             : 
     294           0 :   int const abiv2 = 0;  /* For now, only support ABIv1 */
     295             : 
     296             :   //| .code
     297           0 :   dasm_put(Dst, 0);
     298           0 : #line 178 "fd_jit_compiler.dasc"
     299             : 
     300             :   /* Derive offsets of thread locals in FS "segment" */
     301             : 
     302           0 : # if defined(__FSGSBASE__)
     303           0 :   ulong fs_base = __builtin_ia32_rdfsbase64();
     304             : # else
     305             :   ulong fs_base; __asm__( "mov %%fs:0, %0" : "=r"(fs_base) );
     306             : # endif
     307           0 : # define FS_RELATIVE(ptr) ((uint)( (ulong)(ptr) - fs_base ))
     308           0 :   uint  fd_jit_vm_tpoff             = FS_RELATIVE( &fd_jit_vm             );
     309           0 :   uint  fd_jit_syscalls_tpoff       = FS_RELATIVE( &fd_jit_syscalls       );
     310           0 :   uint  fd_jit_segment_cnt_tpoff    = FS_RELATIVE( &fd_jit_segment_cnt    );
     311           0 :   uint  fd_jit_mem_sz_tpoff         = FS_RELATIVE( fd_jit_mem_sz          );
     312           0 :   uint  fd_jit_mem_haddr_tpoff      = FS_RELATIVE( fd_jit_mem_haddr       );
     313           0 :   uint  fd_jit_segfault_vaddr_tpoff = FS_RELATIVE( &fd_jit_segfault_vaddr );
     314           0 :   uint  fd_jit_segfault_rip_tpoff   = FS_RELATIVE( &fd_jit_segfault_rip   );
     315           0 : # undef FD_RELATIVE
     316             : 
     317             :   //|->save_regs:
     318             :   //| fs
     319             :   //| mov rax, [fd_jit_vm_tpoff]
     320             :   //| mov [rax + offsetof(fd_vm_t, reg[ 0])], bpf_r0
     321             :   //| mov [rax + offsetof(fd_vm_t, reg[ 1])], bpf_r1
     322             :   //| mov [rax + offsetof(fd_vm_t, reg[ 2])], bpf_r2
     323             :   //| mov [rax + offsetof(fd_vm_t, reg[ 3])], bpf_r3
     324             :   //| mov [rax + offsetof(fd_vm_t, reg[ 4])], bpf_r4
     325             :   //| mov [rax + offsetof(fd_vm_t, reg[ 5])], bpf_r5
     326             :   //| mov [rax + offsetof(fd_vm_t, reg[ 6])], bpf_r6
     327             :   //| mov [rax + offsetof(fd_vm_t, reg[ 7])], bpf_r7
     328             :   //| mov [rax + offsetof(fd_vm_t, reg[ 8])], bpf_r8
     329             :   //| mov [rax + offsetof(fd_vm_t, reg[ 9])], bpf_r9
     330             :   //| mov [rax + offsetof(fd_vm_t, reg[10])], bpf_r10
     331             :   //| ret
     332           0 :   dasm_put(Dst, 2, fd_jit_vm_tpoff, offsetof(fd_vm_t, reg[ 0]), offsetof(fd_vm_t, reg[ 1]), offsetof(fd_vm_t, reg[ 2]), offsetof(fd_vm_t, reg[ 3]), offsetof(fd_vm_t, reg[ 4]), offsetof(fd_vm_t, reg[ 5]), offsetof(fd_vm_t, reg[ 6]), offsetof(fd_vm_t, reg[ 7]), offsetof(fd_vm_t, reg[ 8]), offsetof(fd_vm_t, reg[ 9]), offsetof(fd_vm_t, reg[10]));
     333           0 : #line 211 "fd_jit_compiler.dasc"
     334             : 
     335             :   //|->restore_regs:
     336             :   //| fs
     337             :   //| mov rax, [fd_jit_vm_tpoff]
     338             :   //| mov bpf_r0,  [rax + offsetof(fd_vm_t, reg[ 0])]
     339             :   //| mov bpf_r1,  [rax + offsetof(fd_vm_t, reg[ 1])]
     340             :   //| mov bpf_r2,  [rax + offsetof(fd_vm_t, reg[ 2])]
     341             :   //| mov bpf_r3,  [rax + offsetof(fd_vm_t, reg[ 3])]
     342             :   //| mov bpf_r4,  [rax + offsetof(fd_vm_t, reg[ 4])]
     343             :   //| mov bpf_r5,  [rax + offsetof(fd_vm_t, reg[ 5])]
     344             :   //| mov bpf_r6,  [rax + offsetof(fd_vm_t, reg[ 6])]
     345             :   //| mov bpf_r7,  [rax + offsetof(fd_vm_t, reg[ 7])]
     346             :   //| mov bpf_r8,  [rax + offsetof(fd_vm_t, reg[ 8])]
     347             :   //| mov bpf_r9,  [rax + offsetof(fd_vm_t, reg[ 9])]
     348             :   //| mov bpf_r10, [rax + offsetof(fd_vm_t, reg[10])]
     349             :   //| ret
     350           0 :   dasm_put(Dst, 56, fd_jit_vm_tpoff, offsetof(fd_vm_t, reg[ 0]), offsetof(fd_vm_t, reg[ 1]), offsetof(fd_vm_t, reg[ 2]), offsetof(fd_vm_t, reg[ 3]), offsetof(fd_vm_t, reg[ 4]), offsetof(fd_vm_t, reg[ 5]), offsetof(fd_vm_t, reg[ 6]), offsetof(fd_vm_t, reg[ 7]), offsetof(fd_vm_t, reg[ 8]), offsetof(fd_vm_t, reg[ 9]), offsetof(fd_vm_t, reg[10]));
     351           0 : #line 227 "fd_jit_compiler.dasc"
     352             : 
     353             :   /* Helper macros for BPF-to-x86 function calls */
     354             : 
     355             :   //|.macro enter_x86_frame
     356             :   //| call ->save_regs
     357             :   //| mov rbp, rsp
     358             :   //| and rsp, -16
     359             :   //|.endmacro
     360             : 
     361             :   //|.macro leave_x86_frame
     362             :   //| mov rsp, rbp
     363             :   //| call ->restore_regs
     364             :   //|.endmacro
     365             : 
     366             :   /* Address translation macros
     367             : 
     368             :      The translate_{rw,ro}_{1,2,4,8} macros perform address translation
     369             :      and access permission checks for {read-write,read-only} accesses of
     370             :      {1,2,4,8} bytes.  The compiler may inline this macro for each
     371             :      translated sBPF instruction, so these should be optimized for small
     372             :      size.
     373             : 
     374             :      Prior to the macro, rdi holds an address in the virtual address
     375             :      space (untrusted in [0,2^64)).  If translation and permission
     376             :      checks succeed, rdx holds the translated address in the host
     377             :      address space.  On failure jumps to sigsegv.  Reasons for failure
     378             :      include access to out-of-bounds memory, unaligned address, access
     379             :      permission error. */
     380             : 
     381             :   //| .define translate_in,  rdi
     382             :   //| .define translate_out, rdx
     383             :   /* FIXME make output argument rdi instead of rdx */
     384             : 
     385             :   //|->fd_jit_vm_translate_ro:
     386             :   //| mov edx, edi // segment offset
     387             :   //| shr rdi, 32  // segment index
     388             :   //| shl edi, 1
     389             :   //| jmp ->fd_jit_vm_translate
     390           0 :   dasm_put(Dst, 110);
     391           0 : #line 265 "fd_jit_compiler.dasc"
     392             : 
     393             :   //|->fd_jit_vm_translate_rw:
     394             :   //| mov edx, edi // segment offset
     395             :   //| shr rdi, 32  // segment index
     396             :   //| lea edi, [edi*2+1]
     397           0 :   dasm_put(Dst, 127);
     398           0 : #line 270 "fd_jit_compiler.dasc"
     399             :   /* fallthrough */
     400             : 
     401             :   //|->fd_jit_vm_translate:
     402           0 :   dasm_put(Dst, 145);
     403           0 : #line 273 "fd_jit_compiler.dasc"
     404           0 :   if( abiv2 ) {
     405             :     //| jmp ->fd_jit_vm_translate_abiv1
     406           0 :     dasm_put(Dst, 148);
     407           0 : #line 275 "fd_jit_compiler.dasc"
     408           0 :   } else {
     409             :     //| jmp ->fd_jit_vm_translate_abiv2
     410           0 :     dasm_put(Dst, 153);
     411           0 : #line 277 "fd_jit_compiler.dasc"
     412           0 :   }
     413             : 
     414             :   /* ABIv1 virtual memory overview
     415             : 
     416             :      - There are 6 pages indexed in [0,6)
     417             :      - The number of readable bytes in a page is variable in [0,2^31)
     418             :      - The number of writable bytes in a page is equal to the number of
     419             :        readable bytes or zero
     420             :      - The first byte of a page in virtual memory is at (index<<32)
     421             :      - Page 0 is always empty (all bytes are unaddressable)
     422             :      - Page 2 is 'striped': Virtual addresses with bit 12 set are not
     423             :        addressable */
     424             : 
     425             :   //|->fd_jit_vm_translate_abiv1:
     426             :   //| // rdi := virtual address
     427             :   //| // ebp := size of the access minus 1
     428             :   //|
     429             :   //| // edx := segment offset
     430             :   //| mov edx, edi
     431             :   //|
     432             :   //| // edi := segment index
     433             :   //| shr rdi, 32
     434             :   //|
     435             :   //| // segment index in bounds?
     436             :   //| fs
     437             :   //| cmp edi, [fd_jit_segment_cnt_tpoff]
     438             :   //| jae ->translate_fail
     439             :   //|
     440             :   //| // no multi segment overlap?
     441             :   //| add ebp, edx
     442             :   //| jc ->translate_fail
     443             :   //|
     444             :   //| // segment offset in bounds?
     445             :   //| fs
     446             :   //| cmp edx, [rdi*4 + fd_jit_mem_sz_tpoff]
     447             :   //| jae ->translate_fail
     448             :   //|
     449             :   //| // stack gap?
     450             :   //| xor ebp, ebp
     451             :   //| cmp edi, 2
     452             :   //| mov eax, 0x1000
     453             :   //| cmove ebp, eax
     454             :   //| and edx, ebp
     455             :   //| jae ->translate_fail
     456             :   //|
     457             :   //| // rdx := host address
     458             :   //| fs
     459             :   //| add rdx, [rdi*8 + fd_jit_mem_haddr_tpoff]
     460             :   //| ret
     461           0 :   dasm_put(Dst, 158, fd_jit_segment_cnt_tpoff, fd_jit_mem_sz_tpoff, fd_jit_mem_haddr_tpoff);
     462           0 : #line 326 "fd_jit_compiler.dasc"
     463             : 
     464             :   /* ABIv2 virtual memory overview:
     465             : 
     466             :      - Virtual memory is described by a page table
     467             :      - There can be up to 2^16 pages
     468             :      - Each page is 4 GiB (2^32 byte) aligned in virtual memory
     469             :      - The number of readable bytes in a page is variable in [0,2^31)
     470             :      - The number of writable bytes in a page is less or equal to the
     471             :        number of readable bytes */
     472             : 
     473             :   //|->fd_jit_vm_translate_abiv2:
     474             :   //| // rdi := virtual address
     475             :   //| // ebp := size of the access minus 1
     476             :   //|
     477             :   //| // segment index in bounds?
     478             :   //| fs
     479             :   //| cmp edi, [fd_jit_segment_cnt_tpoff]
     480             :   //| jae ->translate_fail
     481             :   //|
     482             :   //| //// aligned access?
     483             :   //| // FIXME do we need an alignment check?
     484             :   //| //mov eax, edx
     485             :   //| //and eax, ebp
     486             :   //| //test eax, eax
     487             :   //| //jnz ->translate_fail
     488             :   //|
     489             :   //| // no multi segment overlap?
     490             :   //| add ebp, edx
     491             :   //| jc ->translate_fail
     492             :   //|
     493             :   //| // segment offset in bounds?
     494             :   //| fs
     495             :   //| cmp edx, [rdi*4 + fd_jit_mem_sz_tpoff]
     496             :   //| jae ->translate_fail
     497             :   //|
     498             :   //| // rdx := host address
     499             :   //| fs
     500             :   //| add rdx, [rdi*8 + fd_jit_mem_haddr_tpoff]
     501             :   //| ret
     502           0 :   dasm_put(Dst, 223, fd_jit_segment_cnt_tpoff, fd_jit_mem_sz_tpoff, fd_jit_mem_haddr_tpoff);
     503           0 : #line 365 "fd_jit_compiler.dasc"
     504             : 
     505             :   //|->translate_fail:
     506             :   //| shl rdi, 32
     507             :   //| or rdi, rdx
     508             :   //| fs
     509             :   //| mov [fd_jit_segfault_vaddr_tpoff], rdi
     510             :   //| mov rdi, [rsp]
     511             :   //| fs
     512             :   //| mov [fd_jit_segfault_rip_tpoff], rdi
     513             :   //| jmp ->vm_fault
     514           0 :   dasm_put(Dst, 258, fd_jit_segfault_vaddr_tpoff, fd_jit_segfault_rip_tpoff);
     515           0 : #line 375 "fd_jit_compiler.dasc"
     516             : 
     517             :   //|.macro translate_rw_1
     518             :   //| xor ebp, ebp
     519             :   //| call ->fd_jit_vm_translate_rw
     520             :   //|.endmacro
     521             : 
     522             :   //|.macro translate_rw_2
     523             :   //| mov ebp, 1
     524             :   //| call ->fd_jit_vm_translate_rw
     525             :   //|.endmacro
     526             : 
     527             :   //|.macro translate_rw_4
     528             :   //| mov ebp, 3
     529             :   //| call ->fd_jit_vm_translate_rw
     530             :   //|.endmacro
     531             : 
     532             :   //|.macro translate_rw_8
     533             :   //| mov ebp, 7
     534             :   //| call ->fd_jit_vm_translate_rw
     535             :   //|.endmacro
     536             : 
     537             :   //|.macro translate_ro_1
     538             :   //| xor ebp, ebp
     539             :   //| call ->fd_jit_vm_translate_ro
     540             :   //|.endmacro
     541             : 
     542             :   //|.macro translate_ro_2
     543             :   //| mov ebp, 1
     544             :   //| call ->fd_jit_vm_translate_ro
     545             :   //|.endmacro
     546             : 
     547             :   //|.macro translate_ro_4
     548             :   //| mov ebp, 3
     549             :   //| call ->fd_jit_vm_translate_ro
     550             :   //|.endmacro
     551             : 
     552             :   //|.macro translate_ro_8
     553             :   //| mov ebp, 7
     554             :   //| call ->fd_jit_vm_translate_ro
     555             :   //|.endmacro
     556             : 
     557             :   /* Generate setjmp/longjmp subroutines.  These can be called from any
     558             :      execution state with a valid stack.  The JIT uses them to restore a
     559             :      sane SystemV-ABI context when exiting JIT code.
     560             : 
     561             :      These are based on musl libc's setjmp/longjmp implementation.
     562             :      Copyright 2011-2012 Nicholas J. Kain, licensed under standard MIT license
     563             : 
     564             :      setjmp takes no arguments.  longjmp takes a 64-bit value in rdi.
     565             :      When setjmp returns from setjmp, sets rax=0 and rdx=0.  When setjmp
     566             :      returns from longjmp, sets rax to the rdi argument of longjmp, and
     567             :      sets rdx=1.  setjmp preserves rdi. */
     568             : 
     569             :   //|->setjmp:
     570             :   //| mov64 r11, (ulong)fd_jit_jmp_buf
     571             :   //| mov [r11+ 0], rbx
     572             :   //| mov [r11+ 8], rbp
     573             :   //| mov [r11+16], r12
     574             :   //| mov [r11+24], r13
     575             :   //| mov [r11+32], r14
     576             :   //| mov [r11+40], r15
     577             :   //| // save callee's stack pointer
     578             :   //| // derived by removing our 8 byte stack frame (only return address)
     579             :   //| lea rdx, [rsp+8]
     580             :   //| mov [r11+48], rdx
     581             :   //| // save return address
     582             :   //| mov rdx, [rsp]
     583             :   //| mov [r11+56], rdx
     584             :   //| // normal return
     585             :   //| xor eax, eax
     586             :   //| xor edx, edx
     587             :   //| ret
     588           0 :   dasm_put(Dst, 288, (unsigned int)((ulong)fd_jit_jmp_buf), (unsigned int)(((ulong)fd_jit_jmp_buf)>>32), 0, 8);
     589           0 : #line 447 "fd_jit_compiler.dasc"
     590             : 
     591             :   //|->longjmp:
     592             :   //| mov rax, rdi // move first argument to first output register
     593             :   //| mov edx, 1   // set second output register to 1
     594             :   //| mov64 rdi, (ulong)fd_jit_jmp_buf
     595             :   //| // restore execution state to callee of setjmp
     596             :   //| mov rbx, [rdi+ 0]
     597             :   //| mov rbp, [rdi+ 8]
     598             :   //| mov r12, [rdi+16]
     599             :   //| mov r13, [rdi+24]
     600             :   //| mov r14, [rdi+32]
     601             :   //| mov r15, [rdi+40]
     602             :   //| mov rsp, [rdi+48]
     603             :   //| push qword [rdi+56]
     604             :   //| ret // retpoline
     605           0 :   dasm_put(Dst, 341, (unsigned int)((ulong)fd_jit_jmp_buf), (unsigned int)(((ulong)fd_jit_jmp_buf)>>32), 0, 8);
     606           0 : #line 462 "fd_jit_compiler.dasc"
     607             : 
     608             :   /* The emulate_syscall function switches from a JIT to an interpreter (C)
     609             :      execution context and invokes a syscall handler.  Register edi is
     610             :      assumed to hold the byte offset into the vm->syscalls table of the
     611             :      fd_sbpf_syscalls_t entry to invoke.
     612             :      On syscall return, switches back to the JIT execution context and
     613             :      resumes execution after the syscall instruction. */
     614             : 
     615             :   //|->emulate_syscall:
     616             :   //| call ->save_regs
     617             :   //| enter_x86_frame
     618             :   //| sub rsp, 16
     619             :   //| fs
     620             :   //| mov r11, [fd_jit_vm_tpoff]
     621             :   //| fs
     622             :   //| mov r10, [rdi + fd_jit_syscalls_tpoff + offsetof(fd_sbpf_syscalls_t, func)]
     623             :   //| mov rdi, r11
     624             :   //| // load BPF r1 through r5 into function arguments
     625             :   //| // FIXME could avoid spill to memory by shuffling registers
     626             :   //| mov rsi, [rax + offsetof(fd_vm_t, reg[1])]
     627             :   //| mov rdx, [rax + offsetof(fd_vm_t, reg[2])]
     628             :   //| mov rcx, [rax + offsetof(fd_vm_t, reg[3])]
     629             :   //| mov r8,  [rax + offsetof(fd_vm_t, reg[4])]
     630             :   //| mov r9,  [rax + offsetof(fd_vm_t, reg[5])]
     631             :   //| lea r11, [rax + offsetof(fd_vm_t, reg[0])]
     632             :   //| push r11
     633             :   //| call r10
     634             :   //| test edi, edi
     635             :   //| jnz ->vm_fault
     636             :   //| leave_x86_frame
     637             :   //| ret
     638           0 :   dasm_put(Dst, 390, fd_jit_vm_tpoff, fd_jit_syscalls_tpoff + offsetof(fd_sbpf_syscalls_t, func), offsetof(fd_vm_t, reg[1]), offsetof(fd_vm_t, reg[2]), offsetof(fd_vm_t, reg[3]), offsetof(fd_vm_t, reg[4]), offsetof(fd_vm_t, reg[5]), offsetof(fd_vm_t, reg[0]));
     639           0 : #line 493 "fd_jit_compiler.dasc"
     640             : 
     641             :   /* The call_stack_push function pushes the current program counter and
     642             :      eBPF registers r6, r7, r8, r9 to the shadow stack.  The frame register
     643             :      (r10) grows upwards.  FIXME implement shadow stack overflow. */
     644             : 
     645           0 : # define REG(n) (offsetof(fd_vm_t, shadow[0].r##n))
     646             : 
     647             :   //|->call_stack_push:
     648             :   //| fs
     649             :   //| mov rdi, [fd_jit_vm_tpoff]
     650             :   //| // vm->frame_cnt++
     651             :   //| inc qword [rdi + offsetof(fd_vm_t, frame_cnt)]
     652             :   //| // save registers
     653             :   //| pop rdi
     654             :   //| push bpf_r6
     655             :   //| push bpf_r7
     656             :   //| push bpf_r8
     657             :   //| push bpf_r9
     658             :   //| add bpf_r10, 0x2000
     659             :   //| jmp rdi
     660           0 :   dasm_put(Dst, 471, fd_jit_vm_tpoff, offsetof(fd_vm_t, frame_cnt));
     661           0 : #line 513 "fd_jit_compiler.dasc"
     662             : 
     663             :   /* The call_stack_pop function undoes the effects of call_stack_push. */
     664             : 
     665             :   //|->call_stack_pop:
     666             :   //| fs
     667             :   //| mov rdi, [fd_jit_vm_tpoff]
     668             :   //| // vm->frame_cnt--
     669             :   //| dec qword [rdi + offsetof(fd_vm_t, frame_cnt)]
     670             :   //| // restore registers
     671             :   //| pop rdi
     672             :   //| pop bpf_r9
     673             :   //| pop bpf_r8
     674             :   //| pop bpf_r7
     675             :   //| pop bpf_r6
     676             :   //| sub bpf_r10, 0x2000
     677             :   //| jmp rdi
     678           0 :   dasm_put(Dst, 502, fd_jit_vm_tpoff, offsetof(fd_vm_t, frame_cnt));
     679           0 : #line 529 "fd_jit_compiler.dasc"
     680             : 
     681           0 : # undef REG
     682             :   /* Exception handlers */
     683             : 
     684             :   //|->vm_fault:
     685             :   //| mov edi, FD_VM_ERR_ABORT
     686             :   //| jmp ->longjmp
     687           0 :   dasm_put(Dst, 534, FD_VM_ERR_SIGABORT);
     688           0 : #line 536 "fd_jit_compiler.dasc"
     689             : 
     690             :   /* JIT entrypoint from C code */
     691             : 
     692             :   //|->entrypoint:
     693           0 :   dasm_put(Dst, 543);
     694           0 : #line 540 "fd_jit_compiler.dasc"
     695             : 
     696             :   /* Create setjmp anchor used to return from JIT */
     697             : 
     698             :   //| call ->setjmp // preserves rdi
     699             :   //| test edx, edx
     700             :   //| jnz >1
     701           0 :   dasm_put(Dst, 546);
     702           0 : #line 546 "fd_jit_compiler.dasc"
     703             : 
     704             :   /* Enter JIT execution context */
     705             : 
     706             :   //| call ->restore_regs
     707             :   //| sub rsp, 0x20 // balance call_stack_push
     708             :   //| call rdi
     709             :   //| mov rdi, bpf_r0
     710             :   //| call ->longjmp
     711             :   //|1:
     712             :   //| ret
     713           0 :   dasm_put(Dst, 556);
     714           0 : #line 556 "fd_jit_compiler.dasc"
     715             : 
     716             :   /* Start translating user code */
     717             : 
     718           0 :   ulong * const text_start = prog->text;
     719           0 :   ulong *       text_end   = prog->text + prog->text_cnt;
     720             : 
     721           0 :   for( ulong * cur=text_start; cur<text_end; cur++ ) {
     722           0 :     ulong instr = *cur;
     723             : 
     724           0 :     ulong opcode  = fd_vm_instr_opcode( instr ); /* in [0,256) even if malformed */
     725           0 :     ulong dst     = fd_vm_instr_dst   ( instr ); /* in [0, 16) even if malformed */
     726           0 :     ulong src     = fd_vm_instr_src   ( instr ); /* in [0, 16) even if malformed */
     727           0 :     ulong offset  = fd_vm_instr_offset( instr ); /* in [-2^15,2^15) even if malformed */
     728           0 :     uint  imm     = fd_vm_instr_imm   ( instr ); /* in [0,2^32) even if malformed */
     729             : 
     730             :     /* Macros for translating register accesses */
     731             : 
     732           0 :     uint x86_dst = reg_bpf2x86[ dst ];
     733           0 :     uint x86_src = reg_bpf2x86[ src ];
     734             : 
     735             :     //| .define dst64, Rq(x86_dst)
     736             :     //| .define src64, Rq(x86_src)
     737             :     //| .define dst32, Rd(x86_dst)
     738             :     //| .define src32, Rd(x86_src)
     739             :     //| .define src8,  Rb(x86_src)
     740             : 
     741             :     /* Macro for translating jumps */
     742             : 
     743           0 :     ulong * jmp_dst       = cur + 1 + offset; /* may be OOB, FIXME validate */
     744           0 :     int     jmp_dst_lbl   = (int)( jmp_dst - text_start );
     745             :     //int     jmp_bounds_ok = jmp_dst>=text_start && jmp<text_end;
     746             :     /* FIXME do a bounds check */
     747             :     /* FIXME what happens if the label is not set? */
     748             : 
     749             :     /* FIXME CU accounting */
     750             : 
     751             :     /* Create a dynamic label for each instruction */
     752             : 
     753           0 :     uint cur_pc = (uint)( cur - text_start );
     754           0 :     int next_label = (int)cur_pc;
     755             :     //|=>next_label:
     756           0 :     dasm_put(Dst, 578, next_label);
     757           0 : #line 597 "fd_jit_compiler.dasc"
     758             : 
     759             :     /* Translate instruction */
     760             : 
     761           0 :     switch( opcode ) {
     762             : 
     763             :     /* 0x00 - 0x0f ******************************************************/
     764             : 
     765           0 :     case 0x04:  /* FD_SBPF_OP_ADD_IMM */
     766             :       //| add dst32, imm
     767           0 :       dasm_put(Dst, 580, (x86_dst), imm);
     768           0 : #line 606 "fd_jit_compiler.dasc"
     769           0 :       break;
     770             : 
     771           0 :     case 0x05:  /* FD_SBPF_OP_JA */
     772             :       //| jmp =>jmp_dst_lbl
     773           0 :       dasm_put(Dst, 587, jmp_dst_lbl);
     774           0 : #line 610 "fd_jit_compiler.dasc"
     775           0 :       break;
     776             : 
     777           0 :     case 0x07:  /* FD_SBPF_OP_ADD64_IMM */
     778             :       //| add dst64, imm
     779           0 :       dasm_put(Dst, 591, (x86_dst), imm);
     780           0 : #line 614 "fd_jit_compiler.dasc"
     781           0 :       break;
     782             : 
     783           0 :     case 0x0c:  /* FD_SBPF_OP_ADD_REG */
     784             :       //| add dst32, src32
     785           0 :       dasm_put(Dst, 598, (x86_src), (x86_dst));
     786           0 : #line 618 "fd_jit_compiler.dasc"
     787           0 :       break;
     788             : 
     789           0 :     case 0x0f:  /* FD_SBPF_OP_ADD64_REG */
     790             :       //| add dst64, src64
     791           0 :       dasm_put(Dst, 606, (x86_src), (x86_dst));
     792           0 : #line 622 "fd_jit_compiler.dasc"
     793           0 :       break;
     794             : 
     795             :     /* 0x10 - 0x1f ******************************************************/
     796             : 
     797           0 :     case 0x14:  /* FD_SBPF_OP_SUB_IMM */
     798             :       //| sub dst32, imm
     799           0 :       dasm_put(Dst, 614, (x86_dst), imm);
     800           0 : #line 628 "fd_jit_compiler.dasc"
     801           0 :       break;
     802             : 
     803           0 :     case 0x15:  /* FD_SBPF_OP_JEQ_IMM */
     804             :       //| cmp dst64, imm
     805           0 :       dasm_put(Dst, 621, (x86_dst), imm);
     806           0 : #line 632 "fd_jit_compiler.dasc"
     807             :       /* pre branch check here ... branchless cu update? */
     808             :       //| je =>jmp_dst_lbl
     809           0 :       dasm_put(Dst, 629, jmp_dst_lbl);
     810           0 : #line 634 "fd_jit_compiler.dasc"
     811           0 :       break;
     812             : 
     813           0 :     case 0x17:  /* FD_SBPF_OP_SUB64_IMM */
     814             :       //| sub dst64, imm
     815           0 :       dasm_put(Dst, 633, (x86_dst), imm);
     816           0 : #line 638 "fd_jit_compiler.dasc"
     817           0 :       break;
     818             : 
     819           0 :     case 0x18:  /* FD_SBPF_OP_LDQ */
     820           0 :       cur++; {
     821           0 :       ulong imm64 = (ulong)imm | ( (ulong)fd_vm_instr_imm( *cur ) << 32 );
     822           0 :       if( imm64==0 ) {
     823             :         //| xor dst32, dst32
     824           0 :         dasm_put(Dst, 640, (x86_dst), (x86_dst));
     825           0 : #line 645 "fd_jit_compiler.dasc"
     826           0 :       } else {
     827             :         //| mov dst64, imm64
     828           0 :         dasm_put(Dst, 648, (x86_dst), imm64);
     829           0 : #line 647 "fd_jit_compiler.dasc"
     830           0 :       }
     831           0 :       break;
     832           0 :     }
     833             : 
     834           0 :     case 0x1c:  /* FD_SBPF_OP_SUB_REG */
     835             :       //| sub dst32, src32
     836           0 :       dasm_put(Dst, 655, (x86_src), (x86_dst));
     837           0 : #line 653 "fd_jit_compiler.dasc"
     838           0 :       break;
     839             : 
     840           0 :     case 0x1d:  /* FD_SBPF_OP_JEQ_REG */
     841             :       //| cmp dst64, src64
     842             :       //| je =>jmp_dst_lbl
     843           0 :       dasm_put(Dst, 663, (x86_src), (x86_dst), jmp_dst_lbl);
     844           0 : #line 658 "fd_jit_compiler.dasc"
     845           0 :       break;
     846             : 
     847           0 :     case 0x1f:  /* FD_SBPF_OP_SUB64_REG */
     848             :       //| sub dst64, src64
     849           0 :       dasm_put(Dst, 674, (x86_src), (x86_dst));
     850           0 : #line 662 "fd_jit_compiler.dasc"
     851           0 :       break;
     852             : 
     853             :     /* 0x20 - 0x2f ******************************************************/
     854             : 
     855           0 :     case 0x24:  /* FD_SBPF_OP_MUL_IMM */
     856             :       /* TODO strength reduction? */
     857             :       //| imul dst32, imm
     858           0 :       dasm_put(Dst, 682, (x86_dst), (x86_dst), imm);
     859           0 : #line 669 "fd_jit_compiler.dasc"
     860           0 :       break;
     861             : 
     862           0 :     case 0x25:  /* FD_SBPF_OP_JGT_IMM */
     863             :       //| cmp dst64, imm
     864             :       //| ja =>jmp_dst_lbl
     865           0 :       dasm_put(Dst, 691, (x86_dst), imm, jmp_dst_lbl);
     866           0 : #line 674 "fd_jit_compiler.dasc"
     867           0 :       break;
     868             : 
     869           0 :     case 0x27:  /* FD_SBPF_OP_MUL64_IMM */
     870             :       /* TODO strength reduction? */
     871             :       //| imul dst64, imm
     872           0 :       dasm_put(Dst, 702, (x86_dst), (x86_dst), imm);
     873           0 : #line 679 "fd_jit_compiler.dasc"
     874           0 :       break;
     875             : 
     876           0 :     case 0x2c:  /* FD_SBPF_OP_MUL_REG */
     877             :       //| imul dst32, src32
     878           0 :       dasm_put(Dst, 711, (x86_dst), (x86_src));
     879           0 : #line 683 "fd_jit_compiler.dasc"
     880           0 :       break;
     881             : 
     882           0 :     case 0x2d:  /* FD_SBPF_OP_JGT_REG */
     883             :       //| cmp dst64, src64
     884             :       //| ja =>jmp_dst_lbl
     885           0 :       dasm_put(Dst, 720, (x86_src), (x86_dst), jmp_dst_lbl);
     886           0 : #line 688 "fd_jit_compiler.dasc"
     887           0 :       break;
     888             : 
     889           0 :     case 0x2f:  /* FD_SBPF_OP_MUL64_REG */
     890             :       //| imul dst64, src64
     891           0 :       dasm_put(Dst, 731, (x86_dst), (x86_src));
     892           0 : #line 692 "fd_jit_compiler.dasc"
     893           0 :       break;
     894             : 
     895             :     /* 0x30 - 0x3f ******************************************************/
     896             : 
     897           0 :     case 0x34:  /* FD_SBPF_OP_DIV_IMM */
     898           0 :       if( FD_UNLIKELY( imm==0 ) ) {
     899             :         //| jmp ->vm_fault
     900           0 :         dasm_put(Dst, 283);
     901           0 : #line 699 "fd_jit_compiler.dasc"
     902           0 :         break;
     903           0 :       }
     904             :       //| xchg eax, dst32
     905             :       //| xor edx, edx
     906             :       //| mov edi, imm
     907             :       //| div edi
     908             :       //| xchg eax, dst32
     909           0 :       dasm_put(Dst, 740, (x86_dst), imm, (x86_dst));
     910           0 : #line 706 "fd_jit_compiler.dasc"
     911           0 :       break;
     912             : 
     913           0 :     case 0x35:  /* FD_SBPF_OP_JGE_IMM */
     914             :       //| cmp dst64, imm
     915             :       //| jae =>jmp_dst_lbl
     916           0 :       dasm_put(Dst, 757, (x86_dst), imm, jmp_dst_lbl);
     917           0 : #line 711 "fd_jit_compiler.dasc"
     918           0 :       break;
     919             : 
     920           0 :     case 0x37:  /* FD_SBPF_OP_DIV64_IMM */
     921           0 :       if( FD_UNLIKELY( imm==0 ) ) {
     922             :         //| jmp ->vm_fault
     923           0 :         dasm_put(Dst, 283);
     924           0 : #line 716 "fd_jit_compiler.dasc"
     925           0 :         break;
     926           0 :       }
     927             :       //| xchg rax, dst64
     928             :       //| xor edx, edx
     929             :       //| mov rdi, imm
     930             :       //| div rdi
     931             :       //| xchg rax, dst64
     932           0 :       dasm_put(Dst, 768, (x86_dst), imm, (x86_dst));
     933           0 : #line 723 "fd_jit_compiler.dasc"
     934           0 :       break;
     935             : 
     936           0 :     case 0x3c:  /* FD_SBPF_OP_DIV_REG */
     937             :       //| test src32, src32
     938             :       //| jz ->vm_fault
     939           0 :       dasm_put(Dst, 788, (x86_src), (x86_src));
     940           0 : #line 728 "fd_jit_compiler.dasc"
     941           0 :       if( x86_dst==x86_src ) {
     942             :         //| mov dst32, 1
     943           0 :         dasm_put(Dst, 800, (x86_dst));
     944           0 : #line 730 "fd_jit_compiler.dasc"
     945           0 :         break;
     946           0 :       }
     947             :       //| xchg eax, dst32
     948             :       //| xor edx, edx
     949             :       //| div src32
     950             :       //| xchg eax, dst32
     951           0 :       dasm_put(Dst, 809, (x86_dst), (x86_src), (x86_dst));
     952           0 : #line 736 "fd_jit_compiler.dasc"
     953           0 :       break;
     954             : 
     955           0 :     case 0x3d:  /* FD_SBPF_OP_JGE_REG */
     956             :       //| cmp dst64, src64
     957             :       //| jae =>jmp_dst_lbl
     958           0 :       dasm_put(Dst, 827, (x86_src), (x86_dst), jmp_dst_lbl);
     959           0 : #line 741 "fd_jit_compiler.dasc"
     960           0 :       break;
     961             : 
     962           0 :     case 0x3f:  /* FD_SBPF_OP_DIV64_REG */
     963             :       //| test src64, src64
     964             :       //| jz ->vm_fault
     965           0 :       dasm_put(Dst, 838, (x86_src), (x86_src));
     966           0 : #line 746 "fd_jit_compiler.dasc"
     967           0 :       if( x86_dst==x86_src ) {
     968             :         //| mov dst32, 1
     969           0 :         dasm_put(Dst, 800, (x86_dst));
     970           0 : #line 748 "fd_jit_compiler.dasc"
     971           0 :         break;
     972           0 :       }
     973             :       //| xchg rax, dst64
     974             :       //| xor edx, edx
     975             :       //| div src64
     976             :       //| xchg rax, dst64
     977           0 :       dasm_put(Dst, 850, (x86_dst), (x86_src), (x86_dst));
     978           0 : #line 754 "fd_jit_compiler.dasc"
     979           0 :       break;
     980             : 
     981             :     /* 0x40 - 0x4f ******************************************************/
     982             : 
     983           0 :     case 0x44:  /* FD_SBPF_OP_OR_IMM */
     984             :       //| or dst32, imm
     985           0 :       dasm_put(Dst, 868, (x86_dst), imm);
     986           0 : #line 760 "fd_jit_compiler.dasc"
     987           0 :       break;
     988             : 
     989           0 :     case 0x45:  /* FD_SBPF_OP_JSET_IMM */
     990             :       //| test dst64, imm
     991             :       //| jnz =>jmp_dst_lbl
     992           0 :       dasm_put(Dst, 875, (x86_dst), imm, jmp_dst_lbl);
     993           0 : #line 765 "fd_jit_compiler.dasc"
     994           0 :       break;
     995             : 
     996           0 :     case 0x47:  /* FD_SBPF_OP_OR64_IMM */
     997             :       //| or dst64, imm
     998           0 :       dasm_put(Dst, 886, (x86_dst), imm);
     999           0 : #line 769 "fd_jit_compiler.dasc"
    1000           0 :       break;
    1001             : 
    1002           0 :     case 0x4c:  /* FD_SBPF_OP_OR_REG */
    1003             :       //| or dst32, src32
    1004           0 :       dasm_put(Dst, 893, (x86_src), (x86_dst));
    1005           0 : #line 773 "fd_jit_compiler.dasc"
    1006           0 :       break;
    1007             : 
    1008           0 :     case 0x4d:  /* FD_SBPF_OP_JSET_REG */
    1009             :       //| test dst64, src64
    1010             :       //| jnz =>jmp_dst_lbl
    1011           0 :       dasm_put(Dst, 901, (x86_src), (x86_dst), jmp_dst_lbl);
    1012           0 : #line 778 "fd_jit_compiler.dasc"
    1013           0 :       break;
    1014             : 
    1015           0 :     case 0x4f:  /* FD_SBPF_OP_OR64_REG */
    1016             :       //| or dst64, src64
    1017           0 :       dasm_put(Dst, 912, (x86_src), (x86_dst));
    1018           0 : #line 782 "fd_jit_compiler.dasc"
    1019           0 :       break;
    1020             : 
    1021             :     /* 0x50 - 0x5f ******************************************************/
    1022             : 
    1023           0 :     case 0x54:  /* FD_SBPF_OP_AND_IMM */
    1024             :       //| and dst32, imm
    1025           0 :       dasm_put(Dst, 920, (x86_dst), imm);
    1026           0 : #line 788 "fd_jit_compiler.dasc"
    1027           0 :       break;
    1028             : 
    1029           0 :     case 0x55:  /* FD_SBPF_OP_JNE_IMM */
    1030             :       //| cmp dst64, imm
    1031             :       //| jne =>jmp_dst_lbl
    1032           0 :       dasm_put(Dst, 927, (x86_dst), imm, jmp_dst_lbl);
    1033           0 : #line 793 "fd_jit_compiler.dasc"
    1034           0 :       break;
    1035             : 
    1036           0 :     case 0x57:  /* FD_SBPF_OP_AND64_IMM */
    1037             :       //| and dst64, imm
    1038           0 :       dasm_put(Dst, 938, (x86_dst), imm);
    1039           0 : #line 797 "fd_jit_compiler.dasc"
    1040           0 :       break;
    1041             : 
    1042           0 :     case 0x5c:  /* FD_SBPF_OP_AND_REG */
    1043             :       //| and dst32, src32
    1044           0 :       dasm_put(Dst, 945, (x86_src), (x86_dst));
    1045           0 : #line 801 "fd_jit_compiler.dasc"
    1046           0 :       break;
    1047             : 
    1048           0 :     case 0x5d:  /* FD_SBPF_OP_JNE_REG */
    1049             :       //| cmp dst64, src64
    1050             :       //| jne =>jmp_dst_lbl
    1051           0 :       dasm_put(Dst, 953, (x86_src), (x86_dst), jmp_dst_lbl);
    1052           0 : #line 806 "fd_jit_compiler.dasc"
    1053           0 :       break;
    1054             : 
    1055           0 :     case 0x5f:  /* FD_SBPF_OP_AND64_REG */
    1056             :       //| and dst64, src64
    1057           0 :       dasm_put(Dst, 964, (x86_src), (x86_dst));
    1058           0 : #line 810 "fd_jit_compiler.dasc"
    1059           0 :       break;
    1060             : 
    1061             :     /* 0x60 - 0x6f ******************************************************/
    1062             : 
    1063           0 :     case 0x61:  /* FD_SBPF_OP_LDXW */
    1064             :       //| lea translate_in, [src64+offset]
    1065             :       //| translate_ro_4
    1066             :       //| mov dst32, [translate_out]
    1067           0 :       dasm_put(Dst, 972, (x86_src), offset, (x86_dst));
    1068           0 : #line 818 "fd_jit_compiler.dasc"
    1069           0 :       break;
    1070             : 
    1071           0 :     case 0x62:  /* FD_SBPF_OP_STW */
    1072             :       //| lea translate_in, [dst64+offset]
    1073             :       //| translate_rw_4
    1074             :       //| mov dword [translate_out], imm
    1075           0 :       dasm_put(Dst, 993, (x86_dst), offset, imm);
    1076           0 : #line 824 "fd_jit_compiler.dasc"
    1077           0 :       break;
    1078             : 
    1079           0 :     case 0x63:  /* FD_SBPF_OP_STXW */
    1080             :       //| lea translate_in, [dst64+offset]
    1081             :       //| translate_rw_4
    1082             :       //| mov [translate_out], src32
    1083           0 :       dasm_put(Dst, 1012, (x86_dst), offset, (x86_src));
    1084           0 : #line 830 "fd_jit_compiler.dasc"
    1085           0 :       break;
    1086             : 
    1087           0 :     case 0x64:  /* FD_SBPF_OP_LSH_IMM */
    1088             :       //| shl dst32, imm
    1089           0 :       dasm_put(Dst, 1033, (x86_dst), imm);
    1090           0 : #line 834 "fd_jit_compiler.dasc"
    1091           0 :       break;
    1092             : 
    1093           0 :     case 0x65:  /* FD_SBPF_OP_JSGT_IMM */
    1094             :       //| cmp dst64, imm
    1095             :       //| jg =>jmp_dst_lbl
    1096           0 :       dasm_put(Dst, 1040, (x86_dst), imm, jmp_dst_lbl);
    1097           0 : #line 839 "fd_jit_compiler.dasc"
    1098           0 :       break;
    1099             : 
    1100           0 :     case 0x67:  /* FD_SBPF_OP_LSH64_IMM */
    1101             :       //| shl dst64, imm
    1102           0 :       dasm_put(Dst, 1051, (x86_dst), imm);
    1103           0 : #line 843 "fd_jit_compiler.dasc"
    1104           0 :       break;
    1105             : 
    1106           0 :     case 0x69:  /* FD_SBPF_OP_LDXH */
    1107             :       //| lea translate_in, [src64+offset]
    1108             :       //| translate_ro_2
    1109             :       //| xor dst32, dst32
    1110             :       //| mov Rw(x86_dst), [translate_out]
    1111           0 :       dasm_put(Dst, 1058, (x86_src), offset, (x86_dst), (x86_dst), (x86_dst));
    1112           0 : #line 850 "fd_jit_compiler.dasc"
    1113           0 :       break;
    1114             : 
    1115           0 :     case 0x6a:  /* FD_SBPF_OP_STH */
    1116             :       //| lea translate_in, [dst64+offset]
    1117             :       //| translate_rw_2
    1118             :       //| mov word [translate_out], imm
    1119           0 :       dasm_put(Dst, 1087, (x86_dst), offset, imm);
    1120           0 : #line 856 "fd_jit_compiler.dasc"
    1121           0 :       break;
    1122             : 
    1123           0 :     case 0x6b:  /* FD_SBPF_OP_STXH */
    1124             :       //| lea translate_in, [dst64+offset]
    1125             :       //| translate_rw_2
    1126             :       //| mov [translate_out], src32
    1127           0 :       dasm_put(Dst, 1107, (x86_dst), offset, (x86_src));
    1128           0 : #line 862 "fd_jit_compiler.dasc"
    1129           0 :       break;
    1130             : 
    1131           0 :     case 0x6c:  /* FD_SBPF_OP_LSH_REG */
    1132             :       //| mov cl, src8
    1133             :       //| shl dst32, cl
    1134           0 :       dasm_put(Dst, 1128, (x86_src), (x86_dst));
    1135           0 : #line 867 "fd_jit_compiler.dasc"
    1136           0 :       break;
    1137             : 
    1138           0 :     case 0x6d:  /* FD_SBPF_OP_JSGT_REG */
    1139             :       //| cmp dst64, src64
    1140             :       //| jg =>jmp_dst_lbl
    1141           0 :       dasm_put(Dst, 1139, (x86_src), (x86_dst), jmp_dst_lbl);
    1142           0 : #line 872 "fd_jit_compiler.dasc"
    1143           0 :       break;
    1144             : 
    1145           0 :     case 0x6f:  /* FD_SBPF_OP_LSH64_REG */
    1146             :       //| mov cl, src8
    1147             :       //| shl dst64, cl
    1148           0 :       dasm_put(Dst, 1150, (x86_src), (x86_dst));
    1149           0 : #line 877 "fd_jit_compiler.dasc"
    1150           0 :       break;
    1151             : 
    1152             :     /* 0x70 - 0x7f ******************************************************/
    1153             : 
    1154           0 :     case 0x71:  /* FD_SBPF_OP_LDXB */
    1155             :       //| lea translate_in, [src64+offset]
    1156             :       //| translate_ro_1
    1157           0 :       dasm_put(Dst, 1161, (x86_src), offset);
    1158           0 : #line 884 "fd_jit_compiler.dasc"
    1159             :       /* TODO is there a better way to zero upper and mov byte? */
    1160             :       //| xor dst32, dst32
    1161             :       //| mov Rb(x86_dst), [translate_out]
    1162           0 :       dasm_put(Dst, 1175, (x86_dst), (x86_dst), (x86_dst));
    1163           0 : #line 887 "fd_jit_compiler.dasc"
    1164           0 :       break;
    1165             : 
    1166           0 :     case 0x72:  /* FD_SBPF_OP_STB */
    1167             :       //| lea translate_in, [src64+offset]
    1168             :       //| translate_rw_1
    1169             :       //| mov byte [translate_out], imm
    1170           0 :       dasm_put(Dst, 1188, (x86_src), offset, imm);
    1171           0 : #line 893 "fd_jit_compiler.dasc"
    1172           0 :       break;
    1173             : 
    1174           0 :     case 0x73:  /* FD_SBPF_OP_STXB */
    1175             :       //| lea translate_in, [dst64+offset]
    1176             :       //| translate_rw_1
    1177             :       //| mov byte [translate_out], Rb(x86_src)
    1178           0 :       dasm_put(Dst, 1205, (x86_dst), offset, (x86_src));
    1179           0 : #line 899 "fd_jit_compiler.dasc"
    1180           0 :       break;
    1181             : 
    1182           0 :     case 0x74:  /* FD_SBPF_OP_RSH_IMM */
    1183             :       //| shr dst32, imm
    1184           0 :       dasm_put(Dst, 1224, (x86_dst), imm);
    1185           0 : #line 903 "fd_jit_compiler.dasc"
    1186           0 :       break;
    1187             : 
    1188           0 :     case 0x75:  /* FD_SBPF_OP_JSGE_IMM */
    1189             :       //| cmp dst64, imm
    1190             :       //| jge =>jmp_dst_lbl
    1191           0 :       dasm_put(Dst, 1231, (x86_dst), imm, jmp_dst_lbl);
    1192           0 : #line 908 "fd_jit_compiler.dasc"
    1193           0 :       break;
    1194             : 
    1195           0 :     case 0x77:  /* FD_SBPF_OP_RSH64_IMM */
    1196             :       //| shr dst64, imm
    1197           0 :       dasm_put(Dst, 1242, (x86_dst), imm);
    1198           0 : #line 912 "fd_jit_compiler.dasc"
    1199           0 :       break;
    1200             : 
    1201           0 :     case 0x79:  /* FD_SBPF_OP_LDXQ */
    1202             :       //| lea translate_in, [src64+offset]
    1203             :       //| translate_ro_8
    1204             :       //| mov dst64, [translate_out]
    1205           0 :       dasm_put(Dst, 1249, (x86_src), offset, (x86_dst));
    1206           0 : #line 918 "fd_jit_compiler.dasc"
    1207           0 :       break;
    1208             : 
    1209           0 :     case 0x7a:  /* FD_SBPF_OP_STQ */
    1210             :       //| lea translate_in, [dst64+offset]
    1211             :       //| translate_rw_8
    1212             :       //| mov dword [translate_out], imm
    1213             :       //| mov dword [translate_out+4], 0
    1214           0 :       dasm_put(Dst, 1270, (x86_dst), offset, imm);
    1215           0 : #line 925 "fd_jit_compiler.dasc"
    1216           0 :       break;
    1217             : 
    1218           0 :     case 0x7b:  /* FD_SBPF_OP_STXQ */
    1219             :       //| lea translate_in, [dst64+offset]
    1220             :       //| translate_rw_8
    1221             :       //| mov [translate_out], src64
    1222           0 :       dasm_put(Dst, 1296, (x86_dst), offset, (x86_src));
    1223           0 : #line 931 "fd_jit_compiler.dasc"
    1224           0 :       break;
    1225             : 
    1226           0 :     case 0x7c:  /* FD_SBPF_OP_RSH_REG */
    1227             :       //| mov cl, src8
    1228             :       //| shr dst32, cl
    1229           0 :       dasm_put(Dst, 1317, (x86_src), (x86_dst));
    1230           0 : #line 936 "fd_jit_compiler.dasc"
    1231           0 :       break;
    1232             : 
    1233           0 :     case 0x7d:  /* FD_SBPF_OP_JSGE_REG */
    1234             :       //| cmp dst64, src64
    1235             :       //| jge =>jmp_dst_lbl
    1236           0 :       dasm_put(Dst, 1328, (x86_src), (x86_dst), jmp_dst_lbl);
    1237           0 : #line 941 "fd_jit_compiler.dasc"
    1238           0 :       break;
    1239             : 
    1240           0 :     case 0x7f:  /* FD_SBPF_OP_RSH64_REG */
    1241             :       //| mov cl, src8
    1242             :       //| shr dst64, cl
    1243           0 :       dasm_put(Dst, 1339, (x86_src), (x86_dst));
    1244           0 : #line 946 "fd_jit_compiler.dasc"
    1245           0 :       break;
    1246             : 
    1247             :     /* 0x80-0x8f ********************************************************/
    1248             : 
    1249           0 :     case 0x84:  /* FD_SBPF_OP_NEG */
    1250             :       //| neg dst32
    1251           0 :       dasm_put(Dst, 1350, (x86_dst));
    1252           0 : #line 952 "fd_jit_compiler.dasc"
    1253           0 :       break;
    1254             : 
    1255           0 :     case 0x85: { /* FD_SBPF_OP_CALL_IMM */
    1256           0 :       fd_sbpf_syscalls_t const * syscall = fd_sbpf_syscalls_query_const( syscalls, imm, NULL );
    1257           0 :       if( !syscall ) {
    1258           0 :         ulong target_pc = (ulong)fd_pchash_inverse( imm );
    1259             :         //| call ->call_stack_push
    1260             :         //| call =>target_pc
    1261           0 :         dasm_put(Dst, 1357, target_pc);
    1262           0 : #line 960 "fd_jit_compiler.dasc"
    1263           0 :       } else {
    1264             :         /* Optimize for code footprint: Generate an offset into the
    1265             :            syscall table (32-bit) instead of the syscall address (64-bit) */
    1266             :         //| mov rdi, (uint)( (ulong)syscall - (ulong)syscalls );
    1267             :         //| call ->emulate_syscall
    1268           0 :         dasm_put(Dst, 1363, (uint)( (ulong)syscall - (ulong)syscalls ));
    1269           0 : #line 965 "fd_jit_compiler.dasc"
    1270           0 :       }
    1271           0 :       break;
    1272           0 :     }
    1273             : 
    1274           0 :     case 0x87:  /* FD_SBPF_OP_NEG64 */
    1275             :       //| neg dst64
    1276           0 :       dasm_put(Dst, 1371, (x86_dst));
    1277           0 : #line 971 "fd_jit_compiler.dasc"
    1278           0 :       break;
    1279             : 
    1280           0 :     case 0x8d:  /* FD_SBPF_OP_CALL_REG */
    1281           0 :       FD_LOG_WARNING(( "TODO: CALLX" ));
    1282           0 :       break;
    1283             : 
    1284             :     /* 0x90 - 0x9f ******************************************************/
    1285             : 
    1286           0 :     case 0x94:  /* FD_SBPF_OP_MOD_IMM */
    1287           0 :       if( FD_UNLIKELY( imm==0 ) ) {
    1288             :         //| jmp ->vm_fault
    1289           0 :         dasm_put(Dst, 283);
    1290           0 : #line 982 "fd_jit_compiler.dasc"
    1291           0 :         break;
    1292           0 :       }
    1293             :       //| xchg eax, dst32
    1294             :       //| xor edx, edx
    1295             :       //| mov edi, imm
    1296             :       //| div edi
    1297             :       //| xchg edx, dst32
    1298           0 :       dasm_put(Dst, 1378, (x86_dst), imm, (x86_dst));
    1299           0 : #line 989 "fd_jit_compiler.dasc"
    1300           0 :       break;
    1301             : 
    1302           0 :     case 0x95:  /* FD_SBPF_OP_EXIT */
    1303             :       //| call ->call_stack_pop
    1304             :       //| ret
    1305           0 :       dasm_put(Dst, 1396);
    1306           0 : #line 994 "fd_jit_compiler.dasc"
    1307           0 :       break;
    1308             : 
    1309           0 :     case 0x97:  /* FD_SBPF_OP_MOD64_IMM */
    1310           0 :       if( FD_UNLIKELY( imm==0 ) ) {
    1311             :         //| jmp ->vm_fault
    1312           0 :         dasm_put(Dst, 283);
    1313           0 : #line 999 "fd_jit_compiler.dasc"
    1314           0 :         break;
    1315           0 :       }
    1316             :       //| xchg rax, dst64
    1317             :       //| xor edx, edx
    1318             :       //| mov rdi, imm
    1319             :       //| div rdi
    1320             :       //| xchg rax, dst64
    1321           0 :       dasm_put(Dst, 768, (x86_dst), imm, (x86_dst));
    1322           0 : #line 1006 "fd_jit_compiler.dasc"
    1323           0 :       break;
    1324             : 
    1325           0 :     case 0x9c:  /* FD_SBPF_OP_MOD_REG */
    1326             :       //| test src32, src32
    1327             :       //| jz ->vm_fault
    1328           0 :       dasm_put(Dst, 788, (x86_src), (x86_src));
    1329           0 : #line 1011 "fd_jit_compiler.dasc"
    1330           0 :       if( x86_dst==x86_src ) {
    1331             :         //| mov dst32, 0
    1332           0 :         dasm_put(Dst, 1401, (x86_dst));
    1333           0 : #line 1013 "fd_jit_compiler.dasc"
    1334           0 :         break;
    1335           0 :       }
    1336             :       //| xchg eax, dst32
    1337             :       //| xor edx, edx
    1338             :       //| div src32
    1339             :       //| xchg edx, dst32
    1340           0 :       dasm_put(Dst, 1410, (x86_dst), (x86_src), (x86_dst));
    1341           0 : #line 1019 "fd_jit_compiler.dasc"
    1342           0 :       break;
    1343             : 
    1344           0 :     case 0x9f:  /* FD_SBPF_OP_MOD64_REG */
    1345             :       //| test src64, src64
    1346             :       //| jz ->vm_fault
    1347           0 :       dasm_put(Dst, 838, (x86_src), (x86_src));
    1348           0 : #line 1024 "fd_jit_compiler.dasc"
    1349           0 :       if( x86_dst==x86_src ) {
    1350             :         //| mov dst32, 0
    1351           0 :         dasm_put(Dst, 1401, (x86_dst));
    1352           0 : #line 1026 "fd_jit_compiler.dasc"
    1353           0 :         break;
    1354           0 :       }
    1355             :       //| xchg rax, dst64
    1356             :       //| xor edx, edx
    1357             :       //| div src64
    1358             :       //| xchg rdx, dst64
    1359           0 :       dasm_put(Dst, 1429, (x86_dst), (x86_src), (x86_dst));
    1360           0 : #line 1032 "fd_jit_compiler.dasc"
    1361           0 :       break;
    1362             : 
    1363             :     /* 0xa0 - 0xaf ******************************************************/
    1364             : 
    1365           0 :     case 0xa4:  /* FD_SBPF_OP_XOR_IMM */
    1366             :       //| xor dst32, imm
    1367           0 :       dasm_put(Dst, 1448, (x86_dst), imm);
    1368           0 : #line 1038 "fd_jit_compiler.dasc"
    1369           0 :       break;
    1370             : 
    1371           0 :     case 0xa5:  /* FD_SBPF_OP_JLT_IMM */
    1372             :       //| cmp dst64, imm
    1373             :       //| jb =>jmp_dst_lbl
    1374           0 :       dasm_put(Dst, 1456, (x86_dst), imm, jmp_dst_lbl);
    1375           0 : #line 1043 "fd_jit_compiler.dasc"
    1376           0 :       break;
    1377             : 
    1378           0 :     case 0xa7:  /* FD_SBPF_OP_XOR64_IMM */
    1379             :       // TODO sign extension
    1380             :       //| xor dst64, imm
    1381           0 :       dasm_put(Dst, 1467, (x86_dst), imm);
    1382           0 : #line 1048 "fd_jit_compiler.dasc"
    1383           0 :       break;
    1384             : 
    1385           0 :     case 0xac:  /* FD_SBPF_OP_XOR_REG */
    1386             :       //| xor dst32, src32
    1387           0 :       dasm_put(Dst, 640, (x86_src), (x86_dst));
    1388           0 : #line 1052 "fd_jit_compiler.dasc"
    1389           0 :       break;
    1390             : 
    1391           0 :     case 0xad:  /* FD_SBPF_OP_JLT_REG */
    1392             :       //| cmp dst64, src64
    1393             :       //| jb =>jmp_dst_lbl
    1394           0 :       dasm_put(Dst, 1475, (x86_src), (x86_dst), jmp_dst_lbl);
    1395           0 : #line 1057 "fd_jit_compiler.dasc"
    1396           0 :       break;
    1397             : 
    1398           0 :     case 0xaf:  /* FD_SBPF_OP_XOR64_REG */
    1399             :       //| xor dst64, src64
    1400           0 :       dasm_put(Dst, 1486, (x86_src), (x86_dst));
    1401           0 : #line 1061 "fd_jit_compiler.dasc"
    1402           0 :       break;
    1403             : 
    1404             :     /* 0xb0 - 0xbf ******************************************************/
    1405             : 
    1406           0 :     case 0xb4:  /* FD_SBPF_OP_MOV_IMM */
    1407             :       //| mov dst32, imm
    1408           0 :       dasm_put(Dst, 1494, (x86_dst), imm);
    1409           0 : #line 1067 "fd_jit_compiler.dasc"
    1410           0 :       break;
    1411             : 
    1412           0 :     case 0xb5:  /* FD_SBPF_OP_JLE_IMM */
    1413             :       //| cmp dst64, imm
    1414             :       //| jbe =>jmp_dst_lbl
    1415           0 :       dasm_put(Dst, 1500, (x86_dst), imm, jmp_dst_lbl);
    1416           0 : #line 1072 "fd_jit_compiler.dasc"
    1417           0 :       break;
    1418             : 
    1419           0 :     case 0xb7:  /* FD_SBPF_OP_MOV64_IMM */
    1420           0 :       if( imm==0 ) {
    1421             :         //| xor dst32, dst32
    1422           0 :         dasm_put(Dst, 640, (x86_dst), (x86_dst));
    1423           0 : #line 1077 "fd_jit_compiler.dasc"
    1424           0 :       } else {
    1425             :         //| mov dst64, imm
    1426           0 :         dasm_put(Dst, 648, (x86_dst), imm);
    1427           0 : #line 1079 "fd_jit_compiler.dasc"
    1428           0 :       }
    1429           0 :       break;
    1430             : 
    1431           0 :     case 0xbc:  /* FD_SBPF_OP_MOV_REG */
    1432             :       //| mov dst32, src32
    1433           0 :       dasm_put(Dst, 1511, (x86_src), (x86_dst));
    1434           0 : #line 1084 "fd_jit_compiler.dasc"
    1435           0 :       break;
    1436             : 
    1437           0 :     case 0xbd:  /* FD_SBPF_OP_JLE_REG */
    1438             :       //| cmp dst64, src64
    1439             :       //| jbe =>jmp_dst_lbl
    1440           0 :       dasm_put(Dst, 1519, (x86_src), (x86_dst), jmp_dst_lbl);
    1441           0 : #line 1089 "fd_jit_compiler.dasc"
    1442           0 :       break;
    1443             : 
    1444           0 :     case 0xbf:  /* FD_SBPF_OP_MOV64_REG */
    1445             :       //| mov dst64, src64
    1446           0 :       dasm_put(Dst, 1530, (x86_src), (x86_dst));
    1447           0 : #line 1093 "fd_jit_compiler.dasc"
    1448           0 :       break;
    1449             : 
    1450             :     /* 0xc0 - 0xcf ******************************************************/
    1451             : 
    1452           0 :     case 0xc4:  /* FD_SBPF_OP_ARSH_IMM */
    1453             :       //| sar dst32, imm
    1454           0 :       dasm_put(Dst, 1538, (x86_dst), imm);
    1455           0 : #line 1099 "fd_jit_compiler.dasc"
    1456           0 :       break;
    1457             : 
    1458           0 :     case 0xc5:  /* FD_SBPF_OP_JSLT_IMM */
    1459             :       //| cmp dst64, imm
    1460             :       //| jl =>jmp_dst_lbl
    1461           0 :       dasm_put(Dst, 1546, (x86_dst), imm, jmp_dst_lbl);
    1462           0 : #line 1104 "fd_jit_compiler.dasc"
    1463           0 :       break;
    1464             : 
    1465           0 :     case 0xc7:  /* FD_SBPF_OP_ARSH64_IMM */
    1466             :       //| sar dst64, imm
    1467           0 :       dasm_put(Dst, 1557, (x86_dst), imm);
    1468           0 : #line 1108 "fd_jit_compiler.dasc"
    1469           0 :       break;
    1470             : 
    1471           0 :     case 0xcc:  /* FD_SBPF_OP_ARSH_REG */
    1472             :       //| mov cl, src8
    1473             :       //| sar dst32, cl
    1474           0 :       dasm_put(Dst, 1565, (x86_src), (x86_dst));
    1475           0 : #line 1113 "fd_jit_compiler.dasc"
    1476           0 :       break;
    1477             : 
    1478           0 :     case 0xcd:  /* FD_SBPF_OP_JSLT_REG */
    1479             :       //| cmp dst64, src64
    1480             :       //| jl =>jmp_dst_lbl
    1481           0 :       dasm_put(Dst, 1577, (x86_src), (x86_dst), jmp_dst_lbl);
    1482           0 : #line 1118 "fd_jit_compiler.dasc"
    1483           0 :       break;
    1484             : 
    1485           0 :     case 0xcf:  /* FD_SBPF_OP_ARSH64_REG */
    1486             :       //| mov cl, src8
    1487             :       //| sar dst64, cl
    1488           0 :       dasm_put(Dst, 1588, (x86_src), (x86_dst));
    1489           0 : #line 1123 "fd_jit_compiler.dasc"
    1490           0 :       break;
    1491             : 
    1492             :     /* 0xd0 - 0xdf ******************************************************/
    1493             : 
    1494           0 :     case 0xd4:  /* FD_SBPF_OP_END_LE */
    1495             :       /* nop */
    1496           0 :       break;
    1497             : 
    1498           0 :     case 0xd5:  /* FD_SBPF_OP_JSLE_IMM */
    1499             :       //| cmp dst64, imm
    1500             :       //| jle =>jmp_dst_lbl
    1501           0 :       dasm_put(Dst, 1600, (x86_dst), imm, jmp_dst_lbl);
    1502           0 : #line 1134 "fd_jit_compiler.dasc"
    1503           0 :       break;
    1504             : 
    1505           0 :     case 0xdc:  /* FD_SBPF_OP_END_BE */
    1506           0 :       switch( imm ) {
    1507           0 :       case 16U:
    1508             :         //| movzx dst32, Rw(x86_dst)
    1509             :         //| ror Rw(x86_dst), 8
    1510           0 :         dasm_put(Dst, 1611, (x86_dst), (x86_dst), (x86_dst));
    1511           0 : #line 1141 "fd_jit_compiler.dasc"
    1512           0 :         break;
    1513           0 :       case 32U:
    1514             :         //| bswap dst32
    1515           0 :         dasm_put(Dst, 1627, (x86_dst));
    1516           0 : #line 1144 "fd_jit_compiler.dasc"
    1517           0 :         break;
    1518           0 :       case 64U:
    1519             :         //| bswap dst64
    1520           0 :         dasm_put(Dst, 1633, (x86_dst));
    1521           0 : #line 1147 "fd_jit_compiler.dasc"
    1522           0 :         break;
    1523           0 :       default:
    1524           0 :         break;
    1525             :         // TODO sigill
    1526           0 :       }
    1527           0 :       break;
    1528             : 
    1529           0 :     case 0xdd:  /* FD_SBPF_OP_JSLE_REG */
    1530             :       //| cmp dst64, src64
    1531             :       //| jle =>jmp_dst_lbl
    1532           0 :       dasm_put(Dst, 1639, (x86_src), (x86_dst), jmp_dst_lbl);
    1533           0 : #line 1157 "fd_jit_compiler.dasc"
    1534           0 :       break;
    1535             : 
    1536           0 :     default:
    1537           0 :       FD_LOG_WARNING(( "Unsupported opcode %lx", opcode ));
    1538           0 :       cur = text_end;
    1539           0 :       break;
    1540             : 
    1541           0 :     }
    1542             : 
    1543           0 :   }
    1544             : 
    1545             :   //|->overrun:
    1546             :   //| jmp ->vm_fault
    1547           0 :   dasm_put(Dst, 1650);
    1548           0 : #line 1170 "fd_jit_compiler.dasc"
    1549             : 
    1550           0 : }
    1551             : 
    1552             : fd_jit_entrypoint_t
    1553           0 : fd_jit_get_entrypoint( void ) {
    1554           0 :   return (fd_jit_entrypoint_t)(ulong)fd_jit_labels[ fd_jit_lbl_entrypoint ];
    1555           0 : }

Generated by: LCOV version 1.14