123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- /*
- * gdbstub internals
- *
- * Copyright (c) 2022 Linaro Ltd
- *
- * SPDX-License-Identifier: GPL-2.0-or-later
- */
- #ifndef GDBSTUB_INTERNALS_H
- #define GDBSTUB_INTERNALS_H
- #include "exec/cpu-common.h"
- #define MAX_PACKET_LENGTH 4096
- /*
- * Shared structures and definitions
- */
- enum {
- GDB_SIGNAL_0 = 0,
- GDB_SIGNAL_INT = 2,
- GDB_SIGNAL_QUIT = 3,
- GDB_SIGNAL_TRAP = 5,
- GDB_SIGNAL_ABRT = 6,
- GDB_SIGNAL_ALRM = 14,
- GDB_SIGNAL_IO = 23,
- GDB_SIGNAL_XCPU = 24,
- GDB_SIGNAL_UNKNOWN = 143
- };
- typedef struct GDBProcess {
- uint32_t pid;
- bool attached;
- /* If gdb sends qXfer:features:read:target.xml this will be populated */
- char *target_xml;
- } GDBProcess;
- enum RSState {
- RS_INACTIVE,
- RS_IDLE,
- RS_GETLINE,
- RS_GETLINE_ESC,
- RS_GETLINE_RLE,
- RS_CHKSUM1,
- RS_CHKSUM2,
- };
- typedef struct GDBState {
- bool init; /* have we been initialised? */
- CPUState *c_cpu; /* current CPU for step/continue ops */
- CPUState *g_cpu; /* current CPU for other ops */
- CPUState *query_cpu; /* for q{f|s}ThreadInfo */
- enum RSState state; /* parsing state */
- char line_buf[MAX_PACKET_LENGTH];
- int line_buf_index;
- int line_sum; /* running checksum */
- int line_csum; /* checksum at the end of the packet */
- GByteArray *last_packet;
- int signal;
- bool multiprocess;
- GDBProcess *processes;
- int process_num;
- GString *str_buf;
- GByteArray *mem_buf;
- int sstep_flags;
- int supported_sstep_flags;
- /*
- * Whether we are allowed to send a stop reply packet at this moment.
- * Must be set off after sending the stop reply itself.
- */
- bool allow_stop_reply;
- } GDBState;
- /* lives in main gdbstub.c */
- extern GDBState gdbserver_state;
- /*
- * Inline utility function, convert from int to hex and back
- */
- static inline int fromhex(int v)
- {
- if (v >= '0' && v <= '9') {
- return v - '0';
- } else if (v >= 'A' && v <= 'F') {
- return v - 'A' + 10;
- } else if (v >= 'a' && v <= 'f') {
- return v - 'a' + 10;
- } else {
- return 0;
- }
- }
- static inline int tohex(int v)
- {
- if (v < 10) {
- return v + '0';
- } else {
- return v - 10 + 'a';
- }
- }
- /*
- * Connection helpers for both softmmu and user backends
- */
- void gdb_put_strbuf(void);
- int gdb_put_packet(const char *buf);
- int gdb_put_packet_binary(const char *buf, int len, bool dump);
- void gdb_hextomem(GByteArray *mem, const char *buf, int len);
- void gdb_memtohex(GString *buf, const uint8_t *mem, int len);
- void gdb_memtox(GString *buf, const char *mem, int len);
- void gdb_read_byte(uint8_t ch);
- /*
- * Packet acknowledgement - we handle this slightly differently
- * between user and softmmu mode, mainly to deal with the differences
- * between the flexible chardev and the direct fd approaches.
- *
- * We currently don't support a negotiated QStartNoAckMode
- */
- /**
- * gdb_got_immediate_ack() - check ok to continue
- *
- * Returns true to continue, false to re-transmit for user only, the
- * softmmu stub always returns true.
- */
- bool gdb_got_immediate_ack(void);
- /* utility helpers */
- GDBProcess *gdb_get_process(uint32_t pid);
- CPUState *gdb_get_first_cpu_in_process(GDBProcess *process);
- CPUState *gdb_first_attached_cpu(void);
- void gdb_append_thread_id(CPUState *cpu, GString *buf);
- int gdb_get_cpu_index(CPUState *cpu);
- unsigned int gdb_get_max_cpus(void); /* both */
- bool gdb_can_reverse(void); /* softmmu, stub for user */
- void gdb_create_default_process(GDBState *s);
- /* signal mapping, common for softmmu, specialised for user-mode */
- int gdb_signal_to_target(int sig);
- int gdb_target_signal_to_gdb(int sig);
- int gdb_get_char(void); /* user only */
- /**
- * gdb_continue() - handle continue in mode specific way.
- */
- void gdb_continue(void);
- /**
- * gdb_continue_partial() - handle partial continue in mode specific way.
- */
- int gdb_continue_partial(char *newstates);
- /*
- * Helpers with separate softmmu and user implementations
- */
- void gdb_put_buffer(const uint8_t *buf, int len);
- /*
- * Command handlers - either specialised or softmmu or user only
- */
- void gdb_init_gdbserver_state(void);
- typedef enum GDBThreadIdKind {
- GDB_ONE_THREAD = 0,
- GDB_ALL_THREADS, /* One process, all threads */
- GDB_ALL_PROCESSES,
- GDB_READ_THREAD_ERR
- } GDBThreadIdKind;
- typedef union GdbCmdVariant {
- const char *data;
- uint8_t opcode;
- unsigned long val_ul;
- unsigned long long val_ull;
- struct {
- GDBThreadIdKind kind;
- uint32_t pid;
- uint32_t tid;
- } thread_id;
- } GdbCmdVariant;
- #define get_param(p, i) (&g_array_index(p, GdbCmdVariant, i))
- void gdb_handle_query_rcmd(GArray *params, void *user_ctx); /* softmmu */
- void gdb_handle_query_offsets(GArray *params, void *user_ctx); /* user */
- void gdb_handle_query_xfer_auxv(GArray *params, void *user_ctx); /*user */
- void gdb_handle_v_file_open(GArray *params, void *user_ctx); /* user */
- void gdb_handle_v_file_close(GArray *params, void *user_ctx); /* user */
- void gdb_handle_v_file_pread(GArray *params, void *user_ctx); /* user */
- void gdb_handle_v_file_readlink(GArray *params, void *user_ctx); /* user */
- void gdb_handle_query_xfer_exec_file(GArray *params, void *user_ctx); /* user */
- void gdb_handle_query_attached(GArray *params, void *user_ctx); /* both */
- /* softmmu only */
- void gdb_handle_query_qemu_phy_mem_mode(GArray *params, void *user_ctx);
- void gdb_handle_set_qemu_phy_mem_mode(GArray *params, void *user_ctx);
- /* sycall handling */
- void gdb_handle_file_io(GArray *params, void *user_ctx);
- bool gdb_handled_syscall(void);
- void gdb_disable_syscalls(void);
- void gdb_syscall_reset(void);
- /* user/softmmu specific syscall handling */
- void gdb_syscall_handling(const char *syscall_packet);
- /*
- * Break/Watch point support - there is an implementation for softmmu
- * and user mode.
- */
- bool gdb_supports_guest_debug(void);
- int gdb_breakpoint_insert(CPUState *cs, int type, vaddr addr, vaddr len);
- int gdb_breakpoint_remove(CPUState *cs, int type, vaddr addr, vaddr len);
- void gdb_breakpoint_remove_all(CPUState *cs);
- /**
- * gdb_target_memory_rw_debug() - handle debug access to memory
- * @cs: CPUState
- * @addr: nominal address, could be an entire physical address
- * @buf: data
- * @len: length of access
- * @is_write: is it a write operation
- *
- * This function is specialised depending on the mode we are running
- * in. For softmmu guests we can switch the interpretation of the
- * address to a physical address.
- */
- int gdb_target_memory_rw_debug(CPUState *cs, hwaddr addr,
- uint8_t *buf, int len, bool is_write);
- #endif /* GDBSTUB_INTERNALS_H */
|