mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-25 02:53:48 +08:00
6a271cae85
2010-06-14 Pedro Alves <pedro@codesourcery.com> * NEWS: Mention GDBserver's JIT compilation of tracepoint bytecode. gdb/gdbserver/ 2010-06-14 Stan Shebs <stan@codesourcery.com> Pedro Alves <pedro@codesourcery.com> Bytecode compiler. * linux-x86-low.c: Include limits.h. (add_insns): New. (always_true): New. (EMIT_ASM): New. (EMIT_ASM32): New. (amd64_emit_prologue, amd64_emit_epilogue, amd64_emit_add) (amd64_emit_sub, amd64_emit_mul, amd64_emit_lsh) (amd64_emit_rsh_signed, amd64_emit_rsh_unsigned, amd64_emit_ext, (amd64_emit_log_not, amd64_emit_bit_and, amd64_emit_bit_or) (amd64_emit_bit_xor, amd64_emit_bit_not, amd64_emit_equal, (amd64_emit_less_signed, amd64_emit_less_unsigned, amd64_emit_ref, (amd64_emit_if_goto, amd64_emit_goto, amd64_write_goto_address) (amd64_emit_const, amd64_emit_call, amd64_emit_reg) (amd64_emit_pop, amd64_emit_stack_flush, amd64_emit_zero_ext) (amd64_emit_swap, amd64_emit_stack_adjust, amd64_emit_int_call_1) (amd64_emit_void_call_2): New. (amd64_emit_ops): New. (i386_emit_prologue, i386_emit_epilogue, i386_emit_add) (i386_emit_sub,i386_emit_mul, i386_emit_lsh, i386_emit_rsh_signed) (i386_emit_rsh_unsigned, i386_emit_ext, i386_emit_log_not) (i386_emit_bit_and, i386_emit_bit_or, i386_emit_bit_xor) (i386_emit_bit_not, i386_emit_equal, i386_emit_less_signed) (i386_emit_less_unsigned, i386_emit_ref, i386_emit_if_goto) (i386_emit_goto, i386_write_goto_address, i386_emit_const) (i386_emit_call, i386_emit_reg, i386_emit_pop) (i386_emit_stack_flush, i386_emit_zero_ext, i386_emit_swap) (i386_emit_stack_adjust, i386_emit_int_call_1) (i386_emit_void_call_2): New. (i386_emit_ops): New. (x86_emit_ops): New. (the_low_target): Install x86_emit_ops. * server.h (struct emit_ops): New. (get_raw_reg_func_addr): Declare. (current_insn_ptr, emit_error): Declare. * tracepoint.c (get_raw_reg, get_trace_state_variable_value) (set_trace_state_variable_value): New defines. (struct ipa_sym_addresses): New fields addr_get_raw_reg, addr_get_trace_state_variable_value and addr_set_trace_state_variable_value. (symbol_list): New fields for get_raw_reg, get_trace_state_variable_value and set_trace_state_variable_value. (condfn): New typedef. (struct tracepoint): New field `compiled_cond'. (do_action_at_tracepoint): Clear compiled_cond. (get_trace_state_variable_value, set_trace_state_variable_value): Export in the IPA. (condition_true_at_tracepoint): If there's a compiled condition, run that. (current_insn_ptr, emit_error): New globals. (struct bytecode_address): New. (get_raw_reg_func_addr): New. (emit_prologue, emit_epilogue, emit_add, emit_sub, emit_mul) (emit_lsh, emit_rsh_signed, emit_rsh_unsigned, emit_ext) (emit_log_not, emit_bit_and, emit_bit_or, emit_bit_xor) (emit_bit_not, emit_equal, emit_less_signed, emit_less_unsigned) (emit_ref, emit_if_goto, emit_goto, write_goto_address, emit_const) (emit_reg, emit_pop, emit_stack_flush, emit_zero_ext, emit_swap) (emit_stack_adjust, emit_int_call_1, emit_void_call_2): New. (compile_tracepoint_condition, compile_bytecodes): New. * target.h (emit_ops): Forward declare. (struct target_ops): New field emit_ops. (target_emit_ops): New. * linux-amd64-ipa.c (gdb_agent_get_raw_reg): New. * linux-i386-ipa.c (gdb_agent_get_raw_reg): New. * linux-low.c (linux_emit_ops): New. (linux_target_ops): Install it. * linux-low.h (struct linux_target_ops): New field emit_ops.
122 lines
3.4 KiB
C
122 lines
3.4 KiB
C
/* GNU/Linux/x86 specific low level interface, for the in-process
|
|
agent library for GDB.
|
|
|
|
Copyright (C) 2010 Free Software Foundation, Inc.
|
|
|
|
This file is part of GDB.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#include "server.h"
|
|
|
|
/* GDB register numbers. */
|
|
|
|
enum i386_gdb_regnum
|
|
{
|
|
I386_EAX_REGNUM, /* %eax */
|
|
I386_ECX_REGNUM, /* %ecx */
|
|
I386_EDX_REGNUM, /* %edx */
|
|
I386_EBX_REGNUM, /* %ebx */
|
|
I386_ESP_REGNUM, /* %esp */
|
|
I386_EBP_REGNUM, /* %ebp */
|
|
I386_ESI_REGNUM, /* %esi */
|
|
I386_EDI_REGNUM, /* %edi */
|
|
I386_EIP_REGNUM, /* %eip */
|
|
I386_EFLAGS_REGNUM, /* %eflags */
|
|
I386_CS_REGNUM, /* %cs */
|
|
I386_SS_REGNUM, /* %ss */
|
|
I386_DS_REGNUM, /* %ds */
|
|
I386_ES_REGNUM, /* %es */
|
|
I386_FS_REGNUM, /* %fs */
|
|
I386_GS_REGNUM, /* %gs */
|
|
I386_ST0_REGNUM /* %st(0) */
|
|
};
|
|
|
|
#define i386_num_regs 16
|
|
|
|
/* Defined in auto-generated file i386-linux.c. */
|
|
void init_registers_i386_linux (void);
|
|
|
|
#define FT_CR_EAX 15
|
|
#define FT_CR_ECX 14
|
|
#define FT_CR_EDX 13
|
|
#define FT_CR_EBX 12
|
|
#define FT_CR_UESP 11
|
|
#define FT_CR_EBP 10
|
|
#define FT_CR_ESI 9
|
|
#define FT_CR_EDI 8
|
|
#define FT_CR_EIP 7
|
|
#define FT_CR_EFL 6
|
|
#define FT_CR_DS 5
|
|
#define FT_CR_ES 4
|
|
#define FT_CR_FS 3
|
|
#define FT_CR_GS 2
|
|
#define FT_CR_SS 1
|
|
#define FT_CR_CS 0
|
|
|
|
/* Mapping between the general-purpose registers in jump tracepoint
|
|
format and GDB's register array layout. */
|
|
|
|
static const int i386_ft_collect_regmap[] =
|
|
{
|
|
FT_CR_EAX * 4, FT_CR_ECX * 4, FT_CR_EDX * 4, FT_CR_EBX * 4,
|
|
FT_CR_UESP * 4, FT_CR_EBP * 4, FT_CR_ESI * 4, FT_CR_EDI * 4,
|
|
FT_CR_EIP * 4, FT_CR_EFL * 4, FT_CR_CS * 4, FT_CR_SS * 4,
|
|
FT_CR_DS * 4, FT_CR_ES * 4, FT_CR_FS * 4, FT_CR_GS * 4
|
|
};
|
|
|
|
void
|
|
supply_fast_tracepoint_registers (struct regcache *regcache,
|
|
const unsigned char *buf)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < i386_num_regs; i++)
|
|
{
|
|
int regval;
|
|
|
|
if (i >= I386_CS_REGNUM && i <= I386_GS_REGNUM)
|
|
regval = *(short *) (((char *) buf) + i386_ft_collect_regmap[i]);
|
|
else
|
|
regval = *(int *) (((char *) buf) + i386_ft_collect_regmap[i]);
|
|
|
|
supply_register (regcache, i, ®val);
|
|
}
|
|
}
|
|
|
|
ULONGEST __attribute__ ((visibility("default"), used))
|
|
gdb_agent_get_raw_reg (unsigned char *raw_regs, int regnum)
|
|
{
|
|
/* This should maybe be allowed to return an error code, or perhaps
|
|
better, have the emit_reg detect this, and emit a constant zero,
|
|
or something. */
|
|
|
|
if (regnum > i386_num_regs)
|
|
return 0;
|
|
else if (regnum >= I386_CS_REGNUM && regnum <= I386_GS_REGNUM)
|
|
return *(short *) (raw_regs + i386_ft_collect_regmap[regnum]);
|
|
else
|
|
return *(int *) (raw_regs + i386_ft_collect_regmap[regnum]);
|
|
}
|
|
|
|
/* This is only needed because reg-i386-linux-lib.o references it. We
|
|
may use it proper at some point. */
|
|
const char *gdbserver_xmltarget;
|
|
|
|
void
|
|
initialize_low_tracepoint (void)
|
|
{
|
|
init_registers_i386_linux ();
|
|
}
|