| 
									
										
										
										
											2005-04-17 19:16:13 +00:00
										 |  |  | #ifndef GDBSTUB_H
 | 
					
						
							|  |  |  | #define GDBSTUB_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-22 01:48:01 +00:00
										 |  |  | #define DEFAULT_GDBSTUB_PORT "1234"
 | 
					
						
							| 
									
										
										
										
											2005-04-17 19:16:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 20:12:48 +00:00
										 |  |  | /* GDB breakpoint/watchpoint types */ | 
					
						
							|  |  |  | #define GDB_BREAKPOINT_SW        0
 | 
					
						
							|  |  |  | #define GDB_BREAKPOINT_HW        1
 | 
					
						
							|  |  |  | #define GDB_WATCHPOINT_WRITE     2
 | 
					
						
							|  |  |  | #define GDB_WATCHPOINT_READ      3
 | 
					
						
							|  |  |  | #define GDB_WATCHPOINT_ACCESS    4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:47 +00:00
										 |  |  | #ifdef NEED_CPU_H
 | 
					
						
							| 
									
										
										
										
											2016-03-15 16:58:45 +01:00
										 |  |  | #include "cpu.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-27 17:45:01 +02:00
										 |  |  | typedef void (*gdb_syscall_complete_cb)(CPUState *cpu, | 
					
						
							| 
									
										
										
										
											2007-01-28 03:10:55 +00:00
										 |  |  |                                         target_ulong ret, target_ulong err); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-07 10:39:27 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_do_syscall: | 
					
						
							|  |  |  |  * @cb: function to call when the system call has completed | 
					
						
							|  |  |  |  * @fmt: gdb syscall format string | 
					
						
							|  |  |  |  * ...: list of arguments to interpolate into @fmt | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Send a GDB syscall request. This function will return immediately; | 
					
						
							|  |  |  |  * the callback function will be called later when the remote system | 
					
						
							|  |  |  |  * call has completed. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @fmt should be in the 'call-id,parameter,parameter...' format documented | 
					
						
							|  |  |  |  * for the F request packet in the GDB remote protocol. A limited set of | 
					
						
							|  |  |  |  * printf-style format specifiers is supported: | 
					
						
							|  |  |  |  *   %x  - target_ulong argument printed in hex | 
					
						
							|  |  |  |  *   %lx - 64-bit argument printed in hex | 
					
						
							|  |  |  |  *   %s  - string pointer (target_ulong) and length (int) pair | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-09-14 06:45:34 +00:00
										 |  |  | void gdb_do_syscall(gdb_syscall_complete_cb cb, const char *fmt, ...); | 
					
						
							| 
									
										
										
										
											2015-09-07 10:39:27 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_do_syscallv: | 
					
						
							|  |  |  |  * @cb: function to call when the system call has completed | 
					
						
							|  |  |  |  * @fmt: gdb syscall format string | 
					
						
							|  |  |  |  * @va: arguments to interpolate into @fmt | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * As gdb_do_syscall, but taking a va_list rather than a variable | 
					
						
							|  |  |  |  * argument list. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void gdb_do_syscallv(gdb_syscall_complete_cb cb, const char *fmt, va_list va); | 
					
						
							| 
									
										
										
										
											2007-01-28 03:10:55 +00:00
										 |  |  | int use_gdb_syscalls(void); | 
					
						
							| 
									
										
										
										
											2021-01-08 22:42:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-17 19:16:13 +00:00
										 |  |  | #ifdef CONFIG_USER_ONLY
 | 
					
						
							| 
									
										
										
										
											2018-05-15 19:19:58 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_handlesig: yield control to gdb | 
					
						
							|  |  |  |  * @cpu: CPU | 
					
						
							|  |  |  |  * @sig: if non-zero, the signal number which caused us to stop | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function yields control to gdb, when a user-mode-only target | 
					
						
							|  |  |  |  * needs to stop execution. If @sig is non-zero, then we will send a | 
					
						
							|  |  |  |  * stop packet to tell gdb that we have stopped because of this signal. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function will block (handling protocol requests from gdb) | 
					
						
							|  |  |  |  * until gdb tells us to continue target execution. When it does | 
					
						
							|  |  |  |  * return, the return value is a signal to deliver to the target, | 
					
						
							|  |  |  |  * or 0 if no signal should be delivered, ie the signal that caused | 
					
						
							|  |  |  |  * us to stop should be ignored. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2013-06-27 19:49:31 +02:00
										 |  |  | int gdb_handlesig(CPUState *, int); | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  | void gdb_signalled(CPUArchState *, int); | 
					
						
							| 
									
										
										
										
											2015-06-23 19:31:16 -07:00
										 |  |  | void gdbserver_fork(CPUState *); | 
					
						
							| 
									
										
										
										
											2007-01-28 01:53:16 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-10-11 17:55:29 +00:00
										 |  |  | /* Get or set a register.  Returns the size of the register.  */ | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | typedef int (*gdb_get_reg_cb)(CPUArchState *env, GByteArray *buf, int reg); | 
					
						
							|  |  |  | typedef int (*gdb_set_reg_cb)(CPUArchState *env, uint8_t *buf, int reg); | 
					
						
							| 
									
										
										
										
											2013-06-28 21:27:39 +02:00
										 |  |  | void gdb_register_coprocessor(CPUState *cpu, | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |                               gdb_get_reg_cb get_reg, gdb_set_reg_cb set_reg, | 
					
						
							| 
									
										
										
										
											2008-10-11 17:55:29 +00:00
										 |  |  |                               int num_regs, const char *xml, int g_pos); | 
					
						
							| 
									
										
										
										
											2005-04-17 19:16:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * The GDB remote protocol transfers values in target byte order. As | 
					
						
							|  |  |  |  * the gdbstub may be batching up several register values we always | 
					
						
							|  |  |  |  * append to the array. | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | static inline int gdb_get_reg8(GByteArray *buf, uint8_t val) | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     g_byte_array_append(buf, &val, 1); | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  |     return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | static inline int gdb_get_reg16(GByteArray *buf, uint16_t val) | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     uint16_t to_word = tswap16(val); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_word, 2); | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  |     return 2; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | static inline int gdb_get_reg32(GByteArray *buf, uint32_t val) | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     uint32_t to_long = tswap32(val); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_long, 4); | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  |     return 4; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | static inline int gdb_get_reg64(GByteArray *buf, uint64_t val) | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     uint64_t to_quad = tswap64(val); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_quad, 8); | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  |     return 8; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | static inline int gdb_get_reg128(GByteArray *buf, uint64_t val_hi, | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:37 +00:00
										 |  |  |                                  uint64_t val_lo) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     uint64_t to_quad; | 
					
						
							| 
									
										
										
										
											2022-03-23 19:57:18 +04:00
										 |  |  | #if TARGET_BIG_ENDIAN
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     to_quad = tswap64(val_hi); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_quad, 8); | 
					
						
							|  |  |  |     to_quad = tswap64(val_lo); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_quad, 8); | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:37 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  |     to_quad = tswap64(val_lo); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_quad, 8); | 
					
						
							|  |  |  |     to_quad = tswap64(val_hi); | 
					
						
							|  |  |  |     g_byte_array_append(buf, (uint8_t *) &to_quad, 8); | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:37 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     return 16; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-14 21:06:26 +01:00
										 |  |  | static inline int gdb_get_zeroes(GByteArray *array, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     guint oldlen = array->len; | 
					
						
							|  |  |  |     g_byte_array_set_size(array, oldlen + len); | 
					
						
							|  |  |  |     memset(array->data + oldlen, 0, len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 17:21:41 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_get_reg_ptr: get pointer to start of last element | 
					
						
							|  |  |  |  * @len: length of element | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This is a helper function to extract the pointer to the last | 
					
						
							|  |  |  |  * element for additional processing. Some front-ends do additional | 
					
						
							|  |  |  |  * dynamic swapping of the elements based on CPU state. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static inline uint8_t * gdb_get_reg_ptr(GByteArray *buf, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return buf->data + buf->len - len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-07 13:05:05 +02:00
										 |  |  | #if TARGET_LONG_BITS == 64
 | 
					
						
							|  |  |  | #define gdb_get_regl(buf, val) gdb_get_reg64(buf, val)
 | 
					
						
							|  |  |  | #define ldtul_p(addr) ldq_p(addr)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define gdb_get_regl(buf, val) gdb_get_reg32(buf, val)
 | 
					
						
							|  |  |  | #define ldtul_p(addr) ldl_p(addr)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 19:56:55 +02:00
										 |  |  | #endif /* NEED_CPU_H */
 | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-30 20:01:19 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdbserver_start: start the gdb server | 
					
						
							|  |  |  |  * @port_or_device: connection spec for gdb | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For CONFIG_USER this is either a tcp port or a path to a fifo. For | 
					
						
							|  |  |  |  * system emulation you can use a full chardev spec for your gdbserver | 
					
						
							|  |  |  |  * port. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int gdbserver_start(const char *port_or_device); | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 19:56:55 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_exit: exit gdb session, reporting inferior status | 
					
						
							|  |  |  |  * @code: exit code reported | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This closes the session and sends a final packet to GDB reporting | 
					
						
							|  |  |  |  * the exit status of the program. It also cleans up any connections | 
					
						
							|  |  |  |  * detritus before returning. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void gdb_exit(int code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void gdb_set_stop_cpu(CPUState *cpu); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-29 04:18:45 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * gdb_has_xml: | 
					
						
							|  |  |  |  * This is an ugly hack to cope with both new and old gdb. | 
					
						
							|  |  |  |  * If gdb sends qXfer:features:read then assume we're talking to a newish | 
					
						
							|  |  |  |  * gdb that understands target descriptions. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | extern bool gdb_has_xml; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-20 20:54:21 +00:00
										 |  |  | /* in gdbstub-xml.c, generated by scripts/feature_to_c.sh */ | 
					
						
							| 
									
										
										
										
											2010-09-18 07:02:16 +00:00
										 |  |  | extern const char *const xml_builtin[][2]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-29 19:23:47 +00:00
										 |  |  | #endif
 |