src/lj_dispatch.h - luajit-2.0-src

Data types defined

Macros defined

Source code

  1. /*
  2. ** Instruction dispatch handling.
  3. ** Copyright (C) 2005-2015 Mike Pall. See Copyright Notice in luajit.h
  4. */

  5. #ifndef _LJ_DISPATCH_H
  6. #define _LJ_DISPATCH_H

  7. #include "lj_obj.h"
  8. #include "lj_bc.h"
  9. #if LJ_HASJIT
  10. #include "lj_jit.h"
  11. #endif

  12. #if LJ_TARGET_MIPS
  13. /* Need our own global offset table for the dreaded MIPS calling conventions. */
  14. #if LJ_HASJIT
  15. #define JITGOTDEF(_)        _(lj_trace_exit) _(lj_trace_hot)
  16. #else
  17. #define JITGOTDEF(_)
  18. #endif
  19. #if LJ_HASFFI
  20. #define FFIGOTDEF(_) \
  21.   _(lj_meta_equal_cd) _(lj_ccallback_enter) _(lj_ccallback_leave)
  22. #else
  23. #define FFIGOTDEF(_)
  24. #endif
  25. #define GOTDEF(_) \
  26.   _(floor) _(ceil) _(trunc) _(log) _(log10) _(exp) _(sin) _(cos) _(tan) \
  27.   _(asin) _(acos) _(atan) _(sinh) _(cosh) _(tanh) _(frexp) _(modf) _(atan2) \
  28.   _(pow) _(fmod) _(ldexp) \
  29.   _(lj_dispatch_call) _(lj_dispatch_ins) _(lj_dispatch_stitch) \
  30.   _(lj_dispatch_profile) _(lj_err_throw) \
  31.   _(lj_ffh_coroutine_wrap_err) _(lj_func_closeuv) _(lj_func_newL_gc) \
  32.   _(lj_gc_barrieruv) _(lj_gc_step) _(lj_gc_step_fixtop) _(lj_meta_arith) \
  33.   _(lj_meta_call) _(lj_meta_cat) _(lj_meta_comp) _(lj_meta_equal) \
  34.   _(lj_meta_for) _(lj_meta_istype) _(lj_meta_len) _(lj_meta_tget) \
  35.   _(lj_meta_tset) _(lj_state_growstack) _(lj_strfmt_num) \
  36.   _(lj_str_new) _(lj_tab_dup) _(lj_tab_get) _(lj_tab_getinth) _(lj_tab_len) \
  37.   _(lj_tab_new) _(lj_tab_newkey) _(lj_tab_next) _(lj_tab_reasize) \
  38.   _(lj_tab_setinth) _(lj_buf_putstr_reverse) _(lj_buf_putstr_lower) \
  39.   _(lj_buf_putstr_upper) _(lj_buf_tostr) JITGOTDEF(_) FFIGOTDEF(_)

  40. enum {
  41. #define GOTENUM(name) LJ_GOT_##name,
  42. GOTDEF(GOTENUM)
  43. #undef GOTENUM
  44.   LJ_GOT__MAX
  45. };
  46. #endif

  47. /* Type of hot counter. Must match the code in the assembler VM. */
  48. /* 16 bits are sufficient. Only 0.0015% overhead with maximum slot penalty. */
  49. typedef uint16_t HotCount;

  50. /* Number of hot counter hash table entries (must be a power of two). */
  51. #define HOTCOUNT_SIZE                64
  52. #define HOTCOUNT_PCMASK                ((HOTCOUNT_SIZE-1)*sizeof(HotCount))

  53. /* Hotcount decrements. */
  54. #define HOTCOUNT_LOOP                2
  55. #define HOTCOUNT_CALL                1

  56. /* This solves a circular dependency problem -- bump as needed. Sigh. */
  57. #define GG_NUM_ASMFF        57

  58. #define GG_LEN_DDISP        (BC__MAX + GG_NUM_ASMFF)
  59. #define GG_LEN_SDISP        BC_FUNCF
  60. #define GG_LEN_DISP        (GG_LEN_DDISP + GG_LEN_SDISP)

  61. /* Global state, main thread and extra fields are allocated together. */
  62. typedef struct GG_State {
  63.   lua_State L;                                /* Main thread. */
  64.   global_State g;                        /* Global state. */
  65. #if LJ_TARGET_MIPS
  66.   ASMFunction got[LJ_GOT__MAX];                /* Global offset table. */
  67. #endif
  68. #if LJ_HASJIT
  69.   jit_State J;                                /* JIT state. */
  70.   HotCount hotcount[HOTCOUNT_SIZE];        /* Hot counters. */
  71. #endif
  72.   ASMFunction dispatch[GG_LEN_DISP];        /* Instruction dispatch tables. */
  73.   BCIns bcff[GG_NUM_ASMFF];                /* Bytecode for ASM fast functions. */
  74. } GG_State;

  75. #define GG_OFS(field)        ((int)offsetof(GG_State, field))
  76. #define G2GG(gl)        ((GG_State *)((char *)(gl) - GG_OFS(g)))
  77. #define J2GG(j)                ((GG_State *)((char *)(j) - GG_OFS(J)))
  78. #define L2GG(L)                (G2GG(G(L)))
  79. #define J2G(J)                (&J2GG(J)->g)
  80. #define G2J(gl)                (&G2GG(gl)->J)
  81. #define L2J(L)                (&L2GG(L)->J)
  82. #define GG_G2DISP        (GG_OFS(dispatch) - GG_OFS(g))
  83. #define GG_DISP2G        (GG_OFS(g) - GG_OFS(dispatch))
  84. #define GG_DISP2J        (GG_OFS(J) - GG_OFS(dispatch))
  85. #define GG_DISP2HOT        (GG_OFS(hotcount) - GG_OFS(dispatch))
  86. #define GG_DISP2STATIC        (GG_LEN_DDISP*(int)sizeof(ASMFunction))

  87. #define hotcount_get(gg, pc) \
  88.   (gg)->hotcount[(u32ptr(pc)>>2) & (HOTCOUNT_SIZE-1)]
  89. #define hotcount_set(gg, pc, val) \
  90.   (hotcount_get((gg), (pc)) = (HotCount)(val))

  91. /* Dispatch table management. */
  92. LJ_FUNC void lj_dispatch_init(GG_State *GG);
  93. #if LJ_HASJIT
  94. LJ_FUNC void lj_dispatch_init_hotcount(global_State *g);
  95. #endif
  96. LJ_FUNC void lj_dispatch_update(global_State *g);

  97. /* Instruction dispatch callback for hooks or when recording. */
  98. LJ_FUNCA void LJ_FASTCALL lj_dispatch_ins(lua_State *L, const BCIns *pc);
  99. LJ_FUNCA ASMFunction LJ_FASTCALL lj_dispatch_call(lua_State *L, const BCIns*pc);
  100. #if LJ_HASJIT
  101. LJ_FUNCA void LJ_FASTCALL lj_dispatch_stitch(jit_State *J, const BCIns *pc);
  102. #endif
  103. #if LJ_HASPROFILE
  104. LJ_FUNCA void LJ_FASTCALL lj_dispatch_profile(lua_State *L, const BCIns *pc);
  105. #endif

  106. #if LJ_HASFFI && !defined(_BUILDVM_H)
  107. /* Save/restore errno and GetLastError() around hooks, exits and recording. */
  108. #include <errno.h>
  109. #if LJ_TARGET_WINDOWS
  110. #define WIN32_LEAN_AND_MEAN
  111. #include <windows.h>
  112. #define ERRNO_SAVE        int olderr = errno; DWORD oldwerr = GetLastError();
  113. #define ERRNO_RESTORE        errno = olderr; SetLastError(oldwerr);
  114. #else
  115. #define ERRNO_SAVE        int olderr = errno;
  116. #define ERRNO_RESTORE        errno = olderr;
  117. #endif
  118. #else
  119. #define ERRNO_SAVE
  120. #define ERRNO_RESTORE
  121. #endif

  122. #endif