| 1 | /* | 
|---|
| 2 | ** Client for the GDB JIT API. | 
|---|
| 3 | ** Copyright (C) 2005-2021 Mike Pall. See Copyright Notice in luajit.h | 
|---|
| 4 | */ | 
|---|
| 5 |  | 
|---|
| 6 | #define lj_gdbjit_c | 
|---|
| 7 | #define LUA_CORE | 
|---|
| 8 |  | 
|---|
| 9 | #include "lj_obj.h" | 
|---|
| 10 |  | 
|---|
| 11 | #if LJ_HASJIT | 
|---|
| 12 |  | 
|---|
| 13 | #include "lj_gc.h" | 
|---|
| 14 | #include "lj_err.h" | 
|---|
| 15 | #include "lj_debug.h" | 
|---|
| 16 | #include "lj_frame.h" | 
|---|
| 17 | #include "lj_buf.h" | 
|---|
| 18 | #include "lj_strfmt.h" | 
|---|
| 19 | #include "lj_jit.h" | 
|---|
| 20 | #include "lj_dispatch.h" | 
|---|
| 21 |  | 
|---|
| 22 | /* This is not compiled in by default. | 
|---|
| 23 | ** Enable with -DLUAJIT_USE_GDBJIT in the Makefile and recompile everything. | 
|---|
| 24 | */ | 
|---|
| 25 | #ifdef LUAJIT_USE_GDBJIT | 
|---|
| 26 |  | 
|---|
| 27 | /* The GDB JIT API allows JIT compilers to pass debug information about | 
|---|
| 28 | ** JIT-compiled code back to GDB. You need at least GDB 7.0 or higher | 
|---|
| 29 | ** to see it in action. | 
|---|
| 30 | ** | 
|---|
| 31 | ** This is a passive API, so it works even when not running under GDB | 
|---|
| 32 | ** or when attaching to an already running process. Alas, this implies | 
|---|
| 33 | ** enabling it always has a non-negligible overhead -- do not use in | 
|---|
| 34 | ** release mode! | 
|---|
| 35 | ** | 
|---|
| 36 | ** The LuaJIT GDB JIT client is rather minimal at the moment. It gives | 
|---|
| 37 | ** each trace a symbol name and adds a source location and frame unwind | 
|---|
| 38 | ** information. Obviously LuaJIT itself and any embedding C application | 
|---|
| 39 | ** should be compiled with debug symbols, too (see the Makefile). | 
|---|
| 40 | ** | 
|---|
| 41 | ** Traces are named TRACE_1, TRACE_2, ... these correspond to the trace | 
|---|
| 42 | ** numbers from -jv or -jdump. Use "break TRACE_1" or "tbreak TRACE_1" etc. | 
|---|
| 43 | ** to set breakpoints on specific traces (even ahead of their creation). | 
|---|
| 44 | ** | 
|---|
| 45 | ** The source location for each trace allows listing the corresponding | 
|---|
| 46 | ** source lines with the GDB command "list" (but only if the Lua source | 
|---|
| 47 | ** has been loaded from a file). Currently this is always set to the | 
|---|
| 48 | ** location where the trace has been started. | 
|---|
| 49 | ** | 
|---|
| 50 | ** Frame unwind information can be inspected with the GDB command | 
|---|
| 51 | ** "info frame". This also allows proper backtraces across JIT-compiled | 
|---|
| 52 | ** code with the GDB command "bt". | 
|---|
| 53 | ** | 
|---|
| 54 | ** You probably want to add the following settings to a .gdbinit file | 
|---|
| 55 | ** (or add them to ~/.gdbinit): | 
|---|
| 56 | **   set disassembly-flavor intel | 
|---|
| 57 | **   set breakpoint pending on | 
|---|
| 58 | ** | 
|---|
| 59 | ** Here's a sample GDB session: | 
|---|
| 60 | ** ------------------------------------------------------------------------ | 
|---|
| 61 |  | 
|---|
| 62 | $ cat >x.lua | 
|---|
| 63 | for outer=1,100 do | 
|---|
| 64 | for inner=1,100 do end | 
|---|
| 65 | end | 
|---|
| 66 | ^D | 
|---|
| 67 |  | 
|---|
| 68 | $ luajit -jv x.lua | 
|---|
| 69 | [TRACE   1 x.lua:2] | 
|---|
| 70 | [TRACE   2 (1/3) x.lua:1 -> 1] | 
|---|
| 71 |  | 
|---|
| 72 | $ gdb --quiet --args luajit x.lua | 
|---|
| 73 | (gdb) tbreak TRACE_1 | 
|---|
| 74 | Function "TRACE_1" not defined. | 
|---|
| 75 | Temporary breakpoint 1 (TRACE_1) pending. | 
|---|
| 76 | (gdb) run | 
|---|
| 77 | Starting program: luajit x.lua | 
|---|
| 78 |  | 
|---|
| 79 | Temporary breakpoint 1, TRACE_1 () at x.lua:2 | 
|---|
| 80 | 2	  for inner=1,100 do end | 
|---|
| 81 | (gdb) list | 
|---|
| 82 | 1	for outer=1,100 do | 
|---|
| 83 | 2	  for inner=1,100 do end | 
|---|
| 84 | 3	end | 
|---|
| 85 | (gdb) bt | 
|---|
| 86 | #0  TRACE_1 () at x.lua:2 | 
|---|
| 87 | #1  0x08053690 in lua_pcall [...] | 
|---|
| 88 | [...] | 
|---|
| 89 | #7  0x0806ff90 in main [...] | 
|---|
| 90 | (gdb) disass TRACE_1 | 
|---|
| 91 | Dump of assembler code for function TRACE_1: | 
|---|
| 92 | 0xf7fd9fba <TRACE_1+0>:	mov    DWORD PTR ds:0xf7e0e2a0,0x1 | 
|---|
| 93 | 0xf7fd9fc4 <TRACE_1+10>:	movsd  xmm7,QWORD PTR [edx+0x20] | 
|---|
| 94 | [...] | 
|---|
| 95 | 0xf7fd9ff8 <TRACE_1+62>:	jmp    0xf7fd2014 | 
|---|
| 96 | End of assembler dump. | 
|---|
| 97 | (gdb) tbreak TRACE_2 | 
|---|
| 98 | Function "TRACE_2" not defined. | 
|---|
| 99 | Temporary breakpoint 2 (TRACE_2) pending. | 
|---|
| 100 | (gdb) cont | 
|---|
| 101 | Continuing. | 
|---|
| 102 |  | 
|---|
| 103 | Temporary breakpoint 2, TRACE_2 () at x.lua:1 | 
|---|
| 104 | 1	for outer=1,100 do | 
|---|
| 105 | (gdb) info frame | 
|---|
| 106 | Stack level 0, frame at 0xffffd7c0: | 
|---|
| 107 | eip = 0xf7fd9f60 in TRACE_2 (x.lua:1); saved eip 0x8053690 | 
|---|
| 108 | called by frame at 0xffffd7e0 | 
|---|
| 109 | source language unknown. | 
|---|
| 110 | Arglist at 0xffffd78c, args: | 
|---|
| 111 | Locals at 0xffffd78c, Previous frame's sp is 0xffffd7c0 | 
|---|
| 112 | Saved registers: | 
|---|
| 113 | ebx at 0xffffd7ac, ebp at 0xffffd7b8, esi at 0xffffd7b0, edi at 0xffffd7b4, | 
|---|
| 114 | eip at 0xffffd7bc | 
|---|
| 115 | (gdb) | 
|---|
| 116 |  | 
|---|
| 117 | ** ------------------------------------------------------------------------ | 
|---|
| 118 | */ | 
|---|
| 119 |  | 
|---|
| 120 | /* -- GDB JIT API --------------------------------------------------------- */ | 
|---|
| 121 |  | 
|---|
| 122 | /* GDB JIT actions. */ | 
|---|
| 123 | enum { | 
|---|
| 124 | GDBJIT_NOACTION = 0, | 
|---|
| 125 | GDBJIT_REGISTER, | 
|---|
| 126 | GDBJIT_UNREGISTER | 
|---|
| 127 | }; | 
|---|
| 128 |  | 
|---|
| 129 | /* GDB JIT entry. */ | 
|---|
| 130 | typedef struct GDBJITentry { | 
|---|
| 131 | struct GDBJITentry *next_entry; | 
|---|
| 132 | struct GDBJITentry *prev_entry; | 
|---|
| 133 | const char *symfile_addr; | 
|---|
| 134 | uint64_t symfile_size; | 
|---|
| 135 | } GDBJITentry; | 
|---|
| 136 |  | 
|---|
| 137 | /* GDB JIT descriptor. */ | 
|---|
| 138 | typedef struct GDBJITdesc { | 
|---|
| 139 | uint32_t version; | 
|---|
| 140 | uint32_t action_flag; | 
|---|
| 141 | GDBJITentry *relevant_entry; | 
|---|
| 142 | GDBJITentry *first_entry; | 
|---|
| 143 | } GDBJITdesc; | 
|---|
| 144 |  | 
|---|
| 145 | GDBJITdesc __jit_debug_descriptor = { | 
|---|
| 146 | 1, GDBJIT_NOACTION, NULL, NULL | 
|---|
| 147 | }; | 
|---|
| 148 |  | 
|---|
| 149 | /* GDB sets a breakpoint at this function. */ | 
|---|
| 150 | void LJ_NOINLINE __jit_debug_register_code() | 
|---|
| 151 | { | 
|---|
| 152 | __asm__ __volatile__( ""); | 
|---|
| 153 | }; | 
|---|
| 154 |  | 
|---|
| 155 | /* -- In-memory ELF object definitions ------------------------------------ */ | 
|---|
| 156 |  | 
|---|
| 157 | /* ELF definitions. */ | 
|---|
| 158 | typedef struct ELFheader { | 
|---|
| 159 | uint8_t emagic[4]; | 
|---|
| 160 | uint8_t eclass; | 
|---|
| 161 | uint8_t eendian; | 
|---|
| 162 | uint8_t eversion; | 
|---|
| 163 | uint8_t eosabi; | 
|---|
| 164 | uint8_t eabiversion; | 
|---|
| 165 | uint8_t epad[7]; | 
|---|
| 166 | uint16_t type; | 
|---|
| 167 | uint16_t machine; | 
|---|
| 168 | uint32_t version; | 
|---|
| 169 | uintptr_t entry; | 
|---|
| 170 | uintptr_t phofs; | 
|---|
| 171 | uintptr_t shofs; | 
|---|
| 172 | uint32_t flags; | 
|---|
| 173 | uint16_t ehsize; | 
|---|
| 174 | uint16_t phentsize; | 
|---|
| 175 | uint16_t phnum; | 
|---|
| 176 | uint16_t shentsize; | 
|---|
| 177 | uint16_t shnum; | 
|---|
| 178 | uint16_t shstridx; | 
|---|
| 179 | } ELFheader; | 
|---|
| 180 |  | 
|---|
| 181 | typedef struct ELFsectheader { | 
|---|
| 182 | uint32_t name; | 
|---|
| 183 | uint32_t type; | 
|---|
| 184 | uintptr_t flags; | 
|---|
| 185 | uintptr_t addr; | 
|---|
| 186 | uintptr_t ofs; | 
|---|
| 187 | uintptr_t size; | 
|---|
| 188 | uint32_t link; | 
|---|
| 189 | uint32_t info; | 
|---|
| 190 | uintptr_t align; | 
|---|
| 191 | uintptr_t entsize; | 
|---|
| 192 | } ELFsectheader; | 
|---|
| 193 |  | 
|---|
| 194 | #define ELFSECT_IDX_ABS		0xfff1 | 
|---|
| 195 |  | 
|---|
| 196 | enum { | 
|---|
| 197 | ELFSECT_TYPE_PROGBITS = 1, | 
|---|
| 198 | ELFSECT_TYPE_SYMTAB = 2, | 
|---|
| 199 | ELFSECT_TYPE_STRTAB = 3, | 
|---|
| 200 | ELFSECT_TYPE_NOBITS = 8 | 
|---|
| 201 | }; | 
|---|
| 202 |  | 
|---|
| 203 | #define ELFSECT_FLAGS_WRITE	1 | 
|---|
| 204 | #define ELFSECT_FLAGS_ALLOC	2 | 
|---|
| 205 | #define ELFSECT_FLAGS_EXEC	4 | 
|---|
| 206 |  | 
|---|
| 207 | typedef struct ELFsymbol { | 
|---|
| 208 | #if LJ_64 | 
|---|
| 209 | uint32_t name; | 
|---|
| 210 | uint8_t info; | 
|---|
| 211 | uint8_t other; | 
|---|
| 212 | uint16_t sectidx; | 
|---|
| 213 | uintptr_t value; | 
|---|
| 214 | uint64_t size; | 
|---|
| 215 | #else | 
|---|
| 216 | uint32_t name; | 
|---|
| 217 | uintptr_t value; | 
|---|
| 218 | uint32_t size; | 
|---|
| 219 | uint8_t info; | 
|---|
| 220 | uint8_t other; | 
|---|
| 221 | uint16_t sectidx; | 
|---|
| 222 | #endif | 
|---|
| 223 | } ELFsymbol; | 
|---|
| 224 |  | 
|---|
| 225 | enum { | 
|---|
| 226 | ELFSYM_TYPE_FUNC = 2, | 
|---|
| 227 | ELFSYM_TYPE_FILE = 4, | 
|---|
| 228 | ELFSYM_BIND_LOCAL = 0 << 4, | 
|---|
| 229 | ELFSYM_BIND_GLOBAL = 1 << 4, | 
|---|
| 230 | }; | 
|---|
| 231 |  | 
|---|
| 232 | /* DWARF definitions. */ | 
|---|
| 233 | #define DW_CIE_VERSION	1 | 
|---|
| 234 |  | 
|---|
| 235 | enum { | 
|---|
| 236 | DW_CFA_nop = 0x0, | 
|---|
| 237 | DW_CFA_offset_extended = 0x5, | 
|---|
| 238 | DW_CFA_def_cfa = 0xc, | 
|---|
| 239 | DW_CFA_def_cfa_offset = 0xe, | 
|---|
| 240 | DW_CFA_offset_extended_sf = 0x11, | 
|---|
| 241 | DW_CFA_advance_loc = 0x40, | 
|---|
| 242 | DW_CFA_offset = 0x80 | 
|---|
| 243 | }; | 
|---|
| 244 |  | 
|---|
| 245 | enum { | 
|---|
| 246 | DW_EH_PE_udata4 = 3, | 
|---|
| 247 | DW_EH_PE_textrel = 0x20 | 
|---|
| 248 | }; | 
|---|
| 249 |  | 
|---|
| 250 | enum { | 
|---|
| 251 | DW_TAG_compile_unit = 0x11 | 
|---|
| 252 | }; | 
|---|
| 253 |  | 
|---|
| 254 | enum { | 
|---|
| 255 | DW_children_no = 0, | 
|---|
| 256 | DW_children_yes = 1 | 
|---|
| 257 | }; | 
|---|
| 258 |  | 
|---|
| 259 | enum { | 
|---|
| 260 | DW_AT_name = 0x03, | 
|---|
| 261 | DW_AT_stmt_list = 0x10, | 
|---|
| 262 | DW_AT_low_pc = 0x11, | 
|---|
| 263 | DW_AT_high_pc = 0x12 | 
|---|
| 264 | }; | 
|---|
| 265 |  | 
|---|
| 266 | enum { | 
|---|
| 267 | DW_FORM_addr = 0x01, | 
|---|
| 268 | DW_FORM_data4 = 0x06, | 
|---|
| 269 | DW_FORM_string = 0x08 | 
|---|
| 270 | }; | 
|---|
| 271 |  | 
|---|
| 272 | enum { | 
|---|
| 273 | DW_LNS_extended_op = 0, | 
|---|
| 274 | DW_LNS_copy = 1, | 
|---|
| 275 | DW_LNS_advance_pc = 2, | 
|---|
| 276 | DW_LNS_advance_line = 3 | 
|---|
| 277 | }; | 
|---|
| 278 |  | 
|---|
| 279 | enum { | 
|---|
| 280 | DW_LNE_end_sequence = 1, | 
|---|
| 281 | DW_LNE_set_address = 2 | 
|---|
| 282 | }; | 
|---|
| 283 |  | 
|---|
| 284 | enum { | 
|---|
| 285 | #if LJ_TARGET_X86 | 
|---|
| 286 | DW_REG_AX, DW_REG_CX, DW_REG_DX, DW_REG_BX, | 
|---|
| 287 | DW_REG_SP, DW_REG_BP, DW_REG_SI, DW_REG_DI, | 
|---|
| 288 | DW_REG_RA, | 
|---|
| 289 | #elif LJ_TARGET_X64 | 
|---|
| 290 | /* Yes, the order is strange, but correct. */ | 
|---|
| 291 | DW_REG_AX, DW_REG_DX, DW_REG_CX, DW_REG_BX, | 
|---|
| 292 | DW_REG_SI, DW_REG_DI, DW_REG_BP, DW_REG_SP, | 
|---|
| 293 | DW_REG_8, DW_REG_9, DW_REG_10, DW_REG_11, | 
|---|
| 294 | DW_REG_12, DW_REG_13, DW_REG_14, DW_REG_15, | 
|---|
| 295 | DW_REG_RA, | 
|---|
| 296 | #elif LJ_TARGET_ARM | 
|---|
| 297 | DW_REG_SP = 13, | 
|---|
| 298 | DW_REG_RA = 14, | 
|---|
| 299 | #elif LJ_TARGET_ARM64 | 
|---|
| 300 | DW_REG_SP = 31, | 
|---|
| 301 | DW_REG_RA = 30, | 
|---|
| 302 | #elif LJ_TARGET_PPC | 
|---|
| 303 | DW_REG_SP = 1, | 
|---|
| 304 | DW_REG_RA = 65, | 
|---|
| 305 | DW_REG_CR = 70, | 
|---|
| 306 | #elif LJ_TARGET_MIPS | 
|---|
| 307 | DW_REG_SP = 29, | 
|---|
| 308 | DW_REG_RA = 31, | 
|---|
| 309 | #else | 
|---|
| 310 | #error "Unsupported target architecture" | 
|---|
| 311 | #endif | 
|---|
| 312 | }; | 
|---|
| 313 |  | 
|---|
| 314 | /* Minimal list of sections for the in-memory ELF object. */ | 
|---|
| 315 | enum { | 
|---|
| 316 | GDBJIT_SECT_NULL, | 
|---|
| 317 | GDBJIT_SECT_text, | 
|---|
| 318 | GDBJIT_SECT_eh_frame, | 
|---|
| 319 | GDBJIT_SECT_shstrtab, | 
|---|
| 320 | GDBJIT_SECT_strtab, | 
|---|
| 321 | GDBJIT_SECT_symtab, | 
|---|
| 322 | GDBJIT_SECT_debug_info, | 
|---|
| 323 | GDBJIT_SECT_debug_abbrev, | 
|---|
| 324 | GDBJIT_SECT_debug_line, | 
|---|
| 325 | GDBJIT_SECT__MAX | 
|---|
| 326 | }; | 
|---|
| 327 |  | 
|---|
| 328 | enum { | 
|---|
| 329 | GDBJIT_SYM_UNDEF, | 
|---|
| 330 | GDBJIT_SYM_FILE, | 
|---|
| 331 | GDBJIT_SYM_FUNC, | 
|---|
| 332 | GDBJIT_SYM__MAX | 
|---|
| 333 | }; | 
|---|
| 334 |  | 
|---|
| 335 | /* In-memory ELF object. */ | 
|---|
| 336 | typedef struct GDBJITobj { | 
|---|
| 337 | ELFheader hdr;			/* ELF header. */ | 
|---|
| 338 | ELFsectheader sect[GDBJIT_SECT__MAX];	/* ELF sections. */ | 
|---|
| 339 | ELFsymbol sym[GDBJIT_SYM__MAX];	/* ELF symbol table. */ | 
|---|
| 340 | uint8_t space[4096];			/* Space for various section data. */ | 
|---|
| 341 | } GDBJITobj; | 
|---|
| 342 |  | 
|---|
| 343 | /* Combined structure for GDB JIT entry and ELF object. */ | 
|---|
| 344 | typedef struct GDBJITentryobj { | 
|---|
| 345 | GDBJITentry entry; | 
|---|
| 346 | size_t sz; | 
|---|
| 347 | GDBJITobj obj; | 
|---|
| 348 | } GDBJITentryobj; | 
|---|
| 349 |  | 
|---|
| 350 | /* Template for in-memory ELF header. */ | 
|---|
| 351 | static const ELFheader elfhdr_template = { | 
|---|
| 352 | .emagic = { 0x7f, 'E', 'L', 'F' }, | 
|---|
| 353 | .eclass = LJ_64 ? 2 : 1, | 
|---|
| 354 | .eendian = LJ_ENDIAN_SELECT(1, 2), | 
|---|
| 355 | .eversion = 1, | 
|---|
| 356 | #if LJ_TARGET_LINUX | 
|---|
| 357 | .eosabi = 0,  /* Nope, it's not 3. */ | 
|---|
| 358 | #elif defined(__FreeBSD__) | 
|---|
| 359 | .eosabi = 9, | 
|---|
| 360 | #elif defined(__NetBSD__) | 
|---|
| 361 | .eosabi = 2, | 
|---|
| 362 | #elif defined(__OpenBSD__) | 
|---|
| 363 | .eosabi = 12, | 
|---|
| 364 | #elif defined(__DragonFly__) | 
|---|
| 365 | .eosabi = 0, | 
|---|
| 366 | #elif LJ_TARGET_SOLARIS | 
|---|
| 367 | .eosabi = 6, | 
|---|
| 368 | #else | 
|---|
| 369 | .eosabi = 0, | 
|---|
| 370 | #endif | 
|---|
| 371 | .eabiversion = 0, | 
|---|
| 372 | .epad = { 0, 0, 0, 0, 0, 0, 0 }, | 
|---|
| 373 | .type = 1, | 
|---|
| 374 | #if LJ_TARGET_X86 | 
|---|
| 375 | .machine = 3, | 
|---|
| 376 | #elif LJ_TARGET_X64 | 
|---|
| 377 | .machine = 62, | 
|---|
| 378 | #elif LJ_TARGET_ARM | 
|---|
| 379 | .machine = 40, | 
|---|
| 380 | #elif LJ_TARGET_ARM64 | 
|---|
| 381 | .machine = 183, | 
|---|
| 382 | #elif LJ_TARGET_PPC | 
|---|
| 383 | .machine = 20, | 
|---|
| 384 | #elif LJ_TARGET_MIPS | 
|---|
| 385 | .machine = 8, | 
|---|
| 386 | #else | 
|---|
| 387 | #error "Unsupported target architecture" | 
|---|
| 388 | #endif | 
|---|
| 389 | .version = 1, | 
|---|
| 390 | .entry = 0, | 
|---|
| 391 | .phofs = 0, | 
|---|
| 392 | .shofs = offsetof(GDBJITobj, sect), | 
|---|
| 393 | .flags = 0, | 
|---|
| 394 | .ehsize = sizeof(ELFheader), | 
|---|
| 395 | .phentsize = 0, | 
|---|
| 396 | .phnum = 0, | 
|---|
| 397 | .shentsize = sizeof(ELFsectheader), | 
|---|
| 398 | .shnum = GDBJIT_SECT__MAX, | 
|---|
| 399 | .shstridx = GDBJIT_SECT_shstrtab | 
|---|
| 400 | }; | 
|---|
| 401 |  | 
|---|
| 402 | /* -- In-memory ELF object generation ------------------------------------- */ | 
|---|
| 403 |  | 
|---|
| 404 | /* Context for generating the ELF object for the GDB JIT API. */ | 
|---|
| 405 | typedef struct GDBJITctx { | 
|---|
| 406 | uint8_t *p;		/* Pointer to next address in obj.space. */ | 
|---|
| 407 | uint8_t *startp;	/* Pointer to start address in obj.space. */ | 
|---|
| 408 | GCtrace *T;		/* Generate symbols for this trace. */ | 
|---|
| 409 | uintptr_t mcaddr;	/* Machine code address. */ | 
|---|
| 410 | MSize szmcode;	/* Size of machine code. */ | 
|---|
| 411 | MSize spadjp;		/* Stack adjustment for parent trace or interpreter. */ | 
|---|
| 412 | MSize spadj;		/* Stack adjustment for trace itself. */ | 
|---|
| 413 | BCLine lineno;	/* Starting line number. */ | 
|---|
| 414 | const char *filename;	/* Starting file name. */ | 
|---|
| 415 | size_t objsize;	/* Final size of ELF object. */ | 
|---|
| 416 | GDBJITobj obj;	/* In-memory ELF object. */ | 
|---|
| 417 | } GDBJITctx; | 
|---|
| 418 |  | 
|---|
| 419 | /* Add a zero-terminated string. */ | 
|---|
| 420 | static uint32_t gdbjit_strz(GDBJITctx *ctx, const char *str) | 
|---|
| 421 | { | 
|---|
| 422 | uint8_t *p = ctx->p; | 
|---|
| 423 | uint32_t ofs = (uint32_t)(p - ctx->startp); | 
|---|
| 424 | do { | 
|---|
| 425 | *p++ = (uint8_t)*str; | 
|---|
| 426 | } while (*str++); | 
|---|
| 427 | ctx->p = p; | 
|---|
| 428 | return ofs; | 
|---|
| 429 | } | 
|---|
| 430 |  | 
|---|
| 431 | /* Append a decimal number. */ | 
|---|
| 432 | static void gdbjit_catnum(GDBJITctx *ctx, uint32_t n) | 
|---|
| 433 | { | 
|---|
| 434 | if (n >= 10) { uint32_t m = n / 10; n = n % 10; gdbjit_catnum(ctx, m); } | 
|---|
| 435 | *ctx->p++ = '0' + n; | 
|---|
| 436 | } | 
|---|
| 437 |  | 
|---|
| 438 | /* Add a SLEB128 value. */ | 
|---|
| 439 | static void gdbjit_sleb128(GDBJITctx *ctx, int32_t v) | 
|---|
| 440 | { | 
|---|
| 441 | uint8_t *p = ctx->p; | 
|---|
| 442 | for (; (uint32_t)(v+0x40) >= 0x80; v >>= 7) | 
|---|
| 443 | *p++ = (uint8_t)((v & 0x7f) | 0x80); | 
|---|
| 444 | *p++ = (uint8_t)(v & 0x7f); | 
|---|
| 445 | ctx->p = p; | 
|---|
| 446 | } | 
|---|
| 447 |  | 
|---|
| 448 | /* Shortcuts to generate DWARF structures. */ | 
|---|
| 449 | #define DB(x)		(*p++ = (x)) | 
|---|
| 450 | #define DI8(x)		(*(int8_t *)p = (x), p++) | 
|---|
| 451 | #define DU16(x)		(*(uint16_t *)p = (x), p += 2) | 
|---|
| 452 | #define DU32(x)		(*(uint32_t *)p = (x), p += 4) | 
|---|
| 453 | #define DADDR(x)	(*(uintptr_t *)p = (x), p += sizeof(uintptr_t)) | 
|---|
| 454 | #define DUV(x)		(p = (uint8_t *)lj_strfmt_wuleb128((char *)p, (x))) | 
|---|
| 455 | #define DSV(x)		(ctx->p = p, gdbjit_sleb128(ctx, (x)), p = ctx->p) | 
|---|
| 456 | #define DSTR(str)	(ctx->p = p, gdbjit_strz(ctx, (str)), p = ctx->p) | 
|---|
| 457 | #define DALIGNNOP(s)	while ((uintptr_t)p & ((s)-1)) *p++ = DW_CFA_nop | 
|---|
| 458 | #define DSECT(name, stmt) \ | 
|---|
| 459 | { uint32_t *szp_##name = (uint32_t *)p; p += 4; stmt \ | 
|---|
| 460 | *szp_##name = (uint32_t)((p-(uint8_t *)szp_##name)-4); } \ | 
|---|
| 461 |  | 
|---|
| 462 | /* Initialize ELF section headers. */ | 
|---|
| 463 | static void LJ_FASTCALL gdbjit_secthdr(GDBJITctx *ctx) | 
|---|
| 464 | { | 
|---|
| 465 | ELFsectheader *sect; | 
|---|
| 466 |  | 
|---|
| 467 | *ctx->p++ = '\0';  /* Empty string at start of string table. */ | 
|---|
| 468 |  | 
|---|
| 469 | #define SECTDEF(id, tp, al) \ | 
|---|
| 470 | sect = &ctx->obj.sect[GDBJIT_SECT_##id]; \ | 
|---|
| 471 | sect->name = gdbjit_strz(ctx, "." #id); \ | 
|---|
| 472 | sect->type = ELFSECT_TYPE_##tp; \ | 
|---|
| 473 | sect->align = (al) | 
|---|
| 474 |  | 
|---|
| 475 | SECTDEF(text, NOBITS, 16); | 
|---|
| 476 | sect->flags = ELFSECT_FLAGS_ALLOC|ELFSECT_FLAGS_EXEC; | 
|---|
| 477 | sect->addr = ctx->mcaddr; | 
|---|
| 478 | sect->ofs = 0; | 
|---|
| 479 | sect->size = ctx->szmcode; | 
|---|
| 480 |  | 
|---|
| 481 | SECTDEF(eh_frame, PROGBITS, sizeof(uintptr_t)); | 
|---|
| 482 | sect->flags = ELFSECT_FLAGS_ALLOC; | 
|---|
| 483 |  | 
|---|
| 484 | SECTDEF(shstrtab, STRTAB, 1); | 
|---|
| 485 | SECTDEF(strtab, STRTAB, 1); | 
|---|
| 486 |  | 
|---|
| 487 | SECTDEF(symtab, SYMTAB, sizeof(uintptr_t)); | 
|---|
| 488 | sect->ofs = offsetof(GDBJITobj, sym); | 
|---|
| 489 | sect->size = sizeof(ctx->obj.sym); | 
|---|
| 490 | sect->link = GDBJIT_SECT_strtab; | 
|---|
| 491 | sect->entsize = sizeof(ELFsymbol); | 
|---|
| 492 | sect->info = GDBJIT_SYM_FUNC; | 
|---|
| 493 |  | 
|---|
| 494 | SECTDEF(debug_info, PROGBITS, 1); | 
|---|
| 495 | SECTDEF(debug_abbrev, PROGBITS, 1); | 
|---|
| 496 | SECTDEF(debug_line, PROGBITS, 1); | 
|---|
| 497 |  | 
|---|
| 498 | #undef SECTDEF | 
|---|
| 499 | } | 
|---|
| 500 |  | 
|---|
| 501 | /* Initialize symbol table. */ | 
|---|
| 502 | static void LJ_FASTCALL gdbjit_symtab(GDBJITctx *ctx) | 
|---|
| 503 | { | 
|---|
| 504 | ELFsymbol *sym; | 
|---|
| 505 |  | 
|---|
| 506 | *ctx->p++ = '\0';  /* Empty string at start of string table. */ | 
|---|
| 507 |  | 
|---|
| 508 | sym = &ctx->obj.sym[GDBJIT_SYM_FILE]; | 
|---|
| 509 | sym->name = gdbjit_strz(ctx, "JIT mcode"); | 
|---|
| 510 | sym->sectidx = ELFSECT_IDX_ABS; | 
|---|
| 511 | sym->info = ELFSYM_TYPE_FILE|ELFSYM_BIND_LOCAL; | 
|---|
| 512 |  | 
|---|
| 513 | sym = &ctx->obj.sym[GDBJIT_SYM_FUNC]; | 
|---|
| 514 | sym->name = gdbjit_strz(ctx, "TRACE_"); ctx->p--; | 
|---|
| 515 | gdbjit_catnum(ctx, ctx->T->traceno); *ctx->p++ = '\0'; | 
|---|
| 516 | sym->sectidx = GDBJIT_SECT_text; | 
|---|
| 517 | sym->value = 0; | 
|---|
| 518 | sym->size = ctx->szmcode; | 
|---|
| 519 | sym->info = ELFSYM_TYPE_FUNC|ELFSYM_BIND_GLOBAL; | 
|---|
| 520 | } | 
|---|
| 521 |  | 
|---|
| 522 | /* Initialize .eh_frame section. */ | 
|---|
| 523 | static void LJ_FASTCALL gdbjit_ehframe(GDBJITctx *ctx) | 
|---|
| 524 | { | 
|---|
| 525 | uint8_t *p = ctx->p; | 
|---|
| 526 | uint8_t *framep = p; | 
|---|
| 527 |  | 
|---|
| 528 | /* Emit DWARF EH CIE. */ | 
|---|
| 529 | DSECT(CIE, | 
|---|
| 530 | DU32(0);			/* Offset to CIE itself. */ | 
|---|
| 531 | DB(DW_CIE_VERSION); | 
|---|
| 532 | DSTR( "zR");			/* Augmentation. */ | 
|---|
| 533 | DUV(1);			/* Code alignment factor. */ | 
|---|
| 534 | DSV(-(int32_t)sizeof(uintptr_t));  /* Data alignment factor. */ | 
|---|
| 535 | DB(DW_REG_RA);		/* Return address register. */ | 
|---|
| 536 | DB(1); DB(DW_EH_PE_textrel|DW_EH_PE_udata4);  /* Augmentation data. */ | 
|---|
| 537 | DB(DW_CFA_def_cfa); DUV(DW_REG_SP); DUV(sizeof(uintptr_t)); | 
|---|
| 538 | #if LJ_TARGET_PPC | 
|---|
| 539 | DB(DW_CFA_offset_extended_sf); DB(DW_REG_RA); DSV(-1); | 
|---|
| 540 | #else | 
|---|
| 541 | DB(DW_CFA_offset|DW_REG_RA); DUV(1); | 
|---|
| 542 | #endif | 
|---|
| 543 | DALIGNNOP(sizeof(uintptr_t)); | 
|---|
| 544 | ) | 
|---|
| 545 |  | 
|---|
| 546 | /* Emit DWARF EH FDE. */ | 
|---|
| 547 | DSECT(FDE, | 
|---|
| 548 | DU32((uint32_t)(p-framep));	/* Offset to CIE. */ | 
|---|
| 549 | DU32(0);			/* Machine code offset relative to .text. */ | 
|---|
| 550 | DU32(ctx->szmcode);		/* Machine code length. */ | 
|---|
| 551 | DB(0);			/* Augmentation data. */ | 
|---|
| 552 | /* Registers saved in CFRAME. */ | 
|---|
| 553 | #if LJ_TARGET_X86 | 
|---|
| 554 | DB(DW_CFA_offset|DW_REG_BP); DUV(2); | 
|---|
| 555 | DB(DW_CFA_offset|DW_REG_DI); DUV(3); | 
|---|
| 556 | DB(DW_CFA_offset|DW_REG_SI); DUV(4); | 
|---|
| 557 | DB(DW_CFA_offset|DW_REG_BX); DUV(5); | 
|---|
| 558 | #elif LJ_TARGET_X64 | 
|---|
| 559 | DB(DW_CFA_offset|DW_REG_BP); DUV(2); | 
|---|
| 560 | DB(DW_CFA_offset|DW_REG_BX); DUV(3); | 
|---|
| 561 | DB(DW_CFA_offset|DW_REG_15); DUV(4); | 
|---|
| 562 | DB(DW_CFA_offset|DW_REG_14); DUV(5); | 
|---|
| 563 | /* Extra registers saved for JIT-compiled code. */ | 
|---|
| 564 | DB(DW_CFA_offset|DW_REG_13); DUV(LJ_GC64 ? 10 : 9); | 
|---|
| 565 | DB(DW_CFA_offset|DW_REG_12); DUV(LJ_GC64 ? 11 : 10); | 
|---|
| 566 | #elif LJ_TARGET_ARM | 
|---|
| 567 | { | 
|---|
| 568 | int i; | 
|---|
| 569 | for (i = 11; i >= 4; i--) { DB(DW_CFA_offset|i); DUV(2+(11-i)); } | 
|---|
| 570 | } | 
|---|
| 571 | #elif LJ_TARGET_ARM64 | 
|---|
| 572 | { | 
|---|
| 573 | int i; | 
|---|
| 574 | DB(DW_CFA_offset|31); DUV(2); | 
|---|
| 575 | for (i = 28; i >= 19; i--) { DB(DW_CFA_offset|i); DUV(3+(28-i)); } | 
|---|
| 576 | for (i = 15; i >= 8; i--) { DB(DW_CFA_offset|32|i); DUV(28-i); } | 
|---|
| 577 | } | 
|---|
| 578 | #elif LJ_TARGET_PPC | 
|---|
| 579 | { | 
|---|
| 580 | int i; | 
|---|
| 581 | DB(DW_CFA_offset_extended); DB(DW_REG_CR); DUV(55); | 
|---|
| 582 | for (i = 14; i <= 31; i++) { | 
|---|
| 583 | DB(DW_CFA_offset|i); DUV(37+(31-i)); | 
|---|
| 584 | DB(DW_CFA_offset|32|i); DUV(2+2*(31-i)); | 
|---|
| 585 | } | 
|---|
| 586 | } | 
|---|
| 587 | #elif LJ_TARGET_MIPS | 
|---|
| 588 | { | 
|---|
| 589 | int i; | 
|---|
| 590 | DB(DW_CFA_offset|30); DUV(2); | 
|---|
| 591 | for (i = 23; i >= 16; i--) { DB(DW_CFA_offset|i); DUV(26-i); } | 
|---|
| 592 | for (i = 30; i >= 20; i -= 2) { DB(DW_CFA_offset|32|i); DUV(42-i); } | 
|---|
| 593 | } | 
|---|
| 594 | #else | 
|---|
| 595 | #error "Unsupported target architecture" | 
|---|
| 596 | #endif | 
|---|
| 597 | if (ctx->spadjp != ctx->spadj) {  /* Parent/interpreter stack frame size. */ | 
|---|
| 598 | DB(DW_CFA_def_cfa_offset); DUV(ctx->spadjp); | 
|---|
| 599 | DB(DW_CFA_advance_loc|1);  /* Only an approximation. */ | 
|---|
| 600 | } | 
|---|
| 601 | DB(DW_CFA_def_cfa_offset); DUV(ctx->spadj);  /* Trace stack frame size. */ | 
|---|
| 602 | DALIGNNOP(sizeof(uintptr_t)); | 
|---|
| 603 | ) | 
|---|
| 604 |  | 
|---|
| 605 | ctx->p = p; | 
|---|
| 606 | } | 
|---|
| 607 |  | 
|---|
| 608 | /* Initialize .debug_info section. */ | 
|---|
| 609 | static void LJ_FASTCALL gdbjit_debuginfo(GDBJITctx *ctx) | 
|---|
| 610 | { | 
|---|
| 611 | uint8_t *p = ctx->p; | 
|---|
| 612 |  | 
|---|
| 613 | DSECT(info, | 
|---|
| 614 | DU16(2);			/* DWARF version. */ | 
|---|
| 615 | DU32(0);			/* Abbrev offset. */ | 
|---|
| 616 | DB(sizeof(uintptr_t));	/* Pointer size. */ | 
|---|
| 617 |  | 
|---|
| 618 | DUV(1);			/* Abbrev #1: DW_TAG_compile_unit. */ | 
|---|
| 619 | DSTR(ctx->filename);	/* DW_AT_name. */ | 
|---|
| 620 | DADDR(ctx->mcaddr);		/* DW_AT_low_pc. */ | 
|---|
| 621 | DADDR(ctx->mcaddr + ctx->szmcode);  /* DW_AT_high_pc. */ | 
|---|
| 622 | DU32(0);			/* DW_AT_stmt_list. */ | 
|---|
| 623 | ) | 
|---|
| 624 |  | 
|---|
| 625 | ctx->p = p; | 
|---|
| 626 | } | 
|---|
| 627 |  | 
|---|
| 628 | /* Initialize .debug_abbrev section. */ | 
|---|
| 629 | static void LJ_FASTCALL gdbjit_debugabbrev(GDBJITctx *ctx) | 
|---|
| 630 | { | 
|---|
| 631 | uint8_t *p = ctx->p; | 
|---|
| 632 |  | 
|---|
| 633 | /* Abbrev #1: DW_TAG_compile_unit. */ | 
|---|
| 634 | DUV(1); DUV(DW_TAG_compile_unit); | 
|---|
| 635 | DB(DW_children_no); | 
|---|
| 636 | DUV(DW_AT_name);	DUV(DW_FORM_string); | 
|---|
| 637 | DUV(DW_AT_low_pc);	DUV(DW_FORM_addr); | 
|---|
| 638 | DUV(DW_AT_high_pc);	DUV(DW_FORM_addr); | 
|---|
| 639 | DUV(DW_AT_stmt_list);	DUV(DW_FORM_data4); | 
|---|
| 640 | DB(0); DB(0); | 
|---|
| 641 |  | 
|---|
| 642 | ctx->p = p; | 
|---|
| 643 | } | 
|---|
| 644 |  | 
|---|
| 645 | #define DLNE(op, s)	(DB(DW_LNS_extended_op), DUV(1+(s)), DB((op))) | 
|---|
| 646 |  | 
|---|
| 647 | /* Initialize .debug_line section. */ | 
|---|
| 648 | static void LJ_FASTCALL gdbjit_debugline(GDBJITctx *ctx) | 
|---|
| 649 | { | 
|---|
| 650 | uint8_t *p = ctx->p; | 
|---|
| 651 |  | 
|---|
| 652 | DSECT(line, | 
|---|
| 653 | DU16(2);			/* DWARF version. */ | 
|---|
| 654 | DSECT(header, | 
|---|
| 655 | DB(1);			/* Minimum instruction length. */ | 
|---|
| 656 | DB(1);			/* is_stmt. */ | 
|---|
| 657 | DI8(0);			/* Line base for special opcodes. */ | 
|---|
| 658 | DB(2);			/* Line range for special opcodes. */ | 
|---|
| 659 | DB(3+1);			/* Opcode base at DW_LNS_advance_line+1. */ | 
|---|
| 660 | DB(0); DB(1); DB(1);	/* Standard opcode lengths. */ | 
|---|
| 661 | /* Directory table. */ | 
|---|
| 662 | DB(0); | 
|---|
| 663 | /* File name table. */ | 
|---|
| 664 | DSTR(ctx->filename); DUV(0); DUV(0); DUV(0); | 
|---|
| 665 | DB(0); | 
|---|
| 666 | ) | 
|---|
| 667 |  | 
|---|
| 668 | DLNE(DW_LNE_set_address, sizeof(uintptr_t)); DADDR(ctx->mcaddr); | 
|---|
| 669 | if (ctx->lineno) { | 
|---|
| 670 | DB(DW_LNS_advance_line); DSV(ctx->lineno-1); | 
|---|
| 671 | } | 
|---|
| 672 | DB(DW_LNS_copy); | 
|---|
| 673 | DB(DW_LNS_advance_pc); DUV(ctx->szmcode); | 
|---|
| 674 | DLNE(DW_LNE_end_sequence, 0); | 
|---|
| 675 | ) | 
|---|
| 676 |  | 
|---|
| 677 | ctx->p = p; | 
|---|
| 678 | } | 
|---|
| 679 |  | 
|---|
| 680 | #undef DLNE | 
|---|
| 681 |  | 
|---|
| 682 | /* Undef shortcuts. */ | 
|---|
| 683 | #undef DB | 
|---|
| 684 | #undef DI8 | 
|---|
| 685 | #undef DU16 | 
|---|
| 686 | #undef DU32 | 
|---|
| 687 | #undef DADDR | 
|---|
| 688 | #undef DUV | 
|---|
| 689 | #undef DSV | 
|---|
| 690 | #undef DSTR | 
|---|
| 691 | #undef DALIGNNOP | 
|---|
| 692 | #undef DSECT | 
|---|
| 693 |  | 
|---|
| 694 | /* Type of a section initializer callback. */ | 
|---|
| 695 | typedef void (LJ_FASTCALL *GDBJITinitf)(GDBJITctx *ctx); | 
|---|
| 696 |  | 
|---|
| 697 | /* Call section initializer and set the section offset and size. */ | 
|---|
| 698 | static void gdbjit_initsect(GDBJITctx *ctx, int sect, GDBJITinitf initf) | 
|---|
| 699 | { | 
|---|
| 700 | ctx->startp = ctx->p; | 
|---|
| 701 | ctx->obj.sect[sect].ofs = (uintptr_t)((char *)ctx->p - (char *)&ctx->obj); | 
|---|
| 702 | initf(ctx); | 
|---|
| 703 | ctx->obj.sect[sect].size = (uintptr_t)(ctx->p - ctx->startp); | 
|---|
| 704 | } | 
|---|
| 705 |  | 
|---|
| 706 | #define SECTALIGN(p, a) \ | 
|---|
| 707 | ((p) = (uint8_t *)(((uintptr_t)(p) + ((a)-1)) & ~(uintptr_t)((a)-1))) | 
|---|
| 708 |  | 
|---|
| 709 | /* Build in-memory ELF object. */ | 
|---|
| 710 | static void gdbjit_buildobj(GDBJITctx *ctx) | 
|---|
| 711 | { | 
|---|
| 712 | GDBJITobj *obj = &ctx->obj; | 
|---|
| 713 | /* Fill in ELF header and clear structures. */ | 
|---|
| 714 | memcpy(&obj->hdr, &elfhdr_template, sizeof(ELFheader)); | 
|---|
| 715 | memset(&obj->sect, 0, sizeof(ELFsectheader)*GDBJIT_SECT__MAX); | 
|---|
| 716 | memset(&obj->sym, 0, sizeof(ELFsymbol)*GDBJIT_SYM__MAX); | 
|---|
| 717 | /* Initialize sections. */ | 
|---|
| 718 | ctx->p = obj->space; | 
|---|
| 719 | gdbjit_initsect(ctx, GDBJIT_SECT_shstrtab, gdbjit_secthdr); | 
|---|
| 720 | gdbjit_initsect(ctx, GDBJIT_SECT_strtab, gdbjit_symtab); | 
|---|
| 721 | gdbjit_initsect(ctx, GDBJIT_SECT_debug_info, gdbjit_debuginfo); | 
|---|
| 722 | gdbjit_initsect(ctx, GDBJIT_SECT_debug_abbrev, gdbjit_debugabbrev); | 
|---|
| 723 | gdbjit_initsect(ctx, GDBJIT_SECT_debug_line, gdbjit_debugline); | 
|---|
| 724 | SECTALIGN(ctx->p, sizeof(uintptr_t)); | 
|---|
| 725 | gdbjit_initsect(ctx, GDBJIT_SECT_eh_frame, gdbjit_ehframe); | 
|---|
| 726 | ctx->objsize = (size_t)((char *)ctx->p - (char *)obj); | 
|---|
| 727 | lj_assertX(ctx->objsize < sizeof(GDBJITobj), "GDBJITobj overflow"); | 
|---|
| 728 | } | 
|---|
| 729 |  | 
|---|
| 730 | #undef SECTALIGN | 
|---|
| 731 |  | 
|---|
| 732 | /* -- Interface to GDB JIT API -------------------------------------------- */ | 
|---|
| 733 |  | 
|---|
| 734 | static int gdbjit_lock; | 
|---|
| 735 |  | 
|---|
| 736 | static void gdbjit_lock_acquire() | 
|---|
| 737 | { | 
|---|
| 738 | while (__sync_lock_test_and_set(&gdbjit_lock, 1)) { | 
|---|
| 739 | /* Just spin; futexes or pthreads aren't worth the portability cost. */ | 
|---|
| 740 | } | 
|---|
| 741 | } | 
|---|
| 742 |  | 
|---|
| 743 | static void gdbjit_lock_release() | 
|---|
| 744 | { | 
|---|
| 745 | __sync_lock_release(&gdbjit_lock); | 
|---|
| 746 | } | 
|---|
| 747 |  | 
|---|
| 748 | /* Add new entry to GDB JIT symbol chain. */ | 
|---|
| 749 | static void gdbjit_newentry(lua_State *L, GDBJITctx *ctx) | 
|---|
| 750 | { | 
|---|
| 751 | /* Allocate memory for GDB JIT entry and ELF object. */ | 
|---|
| 752 | MSize sz = (MSize)(sizeof(GDBJITentryobj) - sizeof(GDBJITobj) + ctx->objsize); | 
|---|
| 753 | GDBJITentryobj *eo = lj_mem_newt(L, sz, GDBJITentryobj); | 
|---|
| 754 | memcpy(&eo->obj, &ctx->obj, ctx->objsize);  /* Copy ELF object. */ | 
|---|
| 755 | eo->sz = sz; | 
|---|
| 756 | ctx->T->gdbjit_entry = (void *)eo; | 
|---|
| 757 | /* Link new entry to chain and register it. */ | 
|---|
| 758 | eo->entry.prev_entry = NULL; | 
|---|
| 759 | gdbjit_lock_acquire(); | 
|---|
| 760 | eo->entry.next_entry = __jit_debug_descriptor.first_entry; | 
|---|
| 761 | if (eo->entry.next_entry) | 
|---|
| 762 | eo->entry.next_entry->prev_entry = &eo->entry; | 
|---|
| 763 | eo->entry.symfile_addr = (const char *)&eo->obj; | 
|---|
| 764 | eo->entry.symfile_size = ctx->objsize; | 
|---|
| 765 | __jit_debug_descriptor.first_entry = &eo->entry; | 
|---|
| 766 | __jit_debug_descriptor.relevant_entry = &eo->entry; | 
|---|
| 767 | __jit_debug_descriptor.action_flag = GDBJIT_REGISTER; | 
|---|
| 768 | __jit_debug_register_code(); | 
|---|
| 769 | gdbjit_lock_release(); | 
|---|
| 770 | } | 
|---|
| 771 |  | 
|---|
| 772 | /* Add debug info for newly compiled trace and notify GDB. */ | 
|---|
| 773 | void lj_gdbjit_addtrace(jit_State *J, GCtrace *T) | 
|---|
| 774 | { | 
|---|
| 775 | GDBJITctx ctx; | 
|---|
| 776 | GCproto *pt = &gcref(T->startpt)->pt; | 
|---|
| 777 | TraceNo parent = T->ir[REF_BASE].op1; | 
|---|
| 778 | const BCIns *startpc = mref(T->startpc, const BCIns); | 
|---|
| 779 | ctx.T = T; | 
|---|
| 780 | ctx.mcaddr = (uintptr_t)T->mcode; | 
|---|
| 781 | ctx.szmcode = T->szmcode; | 
|---|
| 782 | ctx.spadjp = CFRAME_SIZE_JIT + | 
|---|
| 783 | (MSize)(parent ? traceref(J, parent)->spadjust : 0); | 
|---|
| 784 | ctx.spadj = CFRAME_SIZE_JIT + T->spadjust; | 
|---|
| 785 | lj_assertJ(startpc >= proto_bc(pt) && startpc < proto_bc(pt) + pt->sizebc, | 
|---|
| 786 | "start PC out of range"); | 
|---|
| 787 | ctx.lineno = lj_debug_line(pt, proto_bcpos(pt, startpc)); | 
|---|
| 788 | ctx.filename = proto_chunknamestr(pt); | 
|---|
| 789 | if (*ctx.filename == '@' || *ctx.filename == '=') | 
|---|
| 790 | ctx.filename++; | 
|---|
| 791 | else | 
|---|
| 792 | ctx.filename = "(string)"; | 
|---|
| 793 | gdbjit_buildobj(&ctx); | 
|---|
| 794 | gdbjit_newentry(J->L, &ctx); | 
|---|
| 795 | } | 
|---|
| 796 |  | 
|---|
| 797 | /* Delete debug info for trace and notify GDB. */ | 
|---|
| 798 | void lj_gdbjit_deltrace(jit_State *J, GCtrace *T) | 
|---|
| 799 | { | 
|---|
| 800 | GDBJITentryobj *eo = (GDBJITentryobj *)T->gdbjit_entry; | 
|---|
| 801 | if (eo) { | 
|---|
| 802 | gdbjit_lock_acquire(); | 
|---|
| 803 | if (eo->entry.prev_entry) | 
|---|
| 804 | eo->entry.prev_entry->next_entry = eo->entry.next_entry; | 
|---|
| 805 | else | 
|---|
| 806 | __jit_debug_descriptor.first_entry = eo->entry.next_entry; | 
|---|
| 807 | if (eo->entry.next_entry) | 
|---|
| 808 | eo->entry.next_entry->prev_entry = eo->entry.prev_entry; | 
|---|
| 809 | __jit_debug_descriptor.relevant_entry = &eo->entry; | 
|---|
| 810 | __jit_debug_descriptor.action_flag = GDBJIT_UNREGISTER; | 
|---|
| 811 | __jit_debug_register_code(); | 
|---|
| 812 | gdbjit_lock_release(); | 
|---|
| 813 | lj_mem_free(J2G(J), eo, eo->sz); | 
|---|
| 814 | } | 
|---|
| 815 | } | 
|---|
| 816 |  | 
|---|
| 817 | #endif | 
|---|
| 818 | #endif | 
|---|
| 819 |  | 
|---|