gdbstub: Rework configuration via command line and monitor (Jan Kiszka)
Introduce a more canonical gdbstub configuration (system emulation only) via the new switch '-gdb dev'. Keep '-s' as shorthand for '-gdb tcp::1234'. Use the same syntax also for the corresponding monitor command 'gdbserver'. Its default remains to listen on TCP port 1234. Changes in v4: - Rebased over new command line switches meta file Changes in v3: - Fix documentation Changes in v2: - Support for pipe-based like to gdb (target remote | qemu -gdb stdio) - Properly update the qemu-doc Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@6992 c046a42c-6fe2-441c-8c8c-71466251a162
This commit is contained in:
parent
bc14ca2453
commit
59030a8cd4
41
gdbstub.c
41
gdbstub.c
@ -2337,27 +2337,40 @@ static int gdb_monitor_write(CharDriverState *chr, const uint8_t *buf, int len)
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
int gdbserver_start(const char *port)
|
#ifndef _WIN32
|
||||||
|
static void gdb_sigterm_handler(int signal)
|
||||||
|
{
|
||||||
|
if (vm_running)
|
||||||
|
vm_stop(EXCP_INTERRUPT);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int gdbserver_start(const char *device)
|
||||||
{
|
{
|
||||||
GDBState *s;
|
GDBState *s;
|
||||||
char gdbstub_port_name[128];
|
char gdbstub_device_name[128];
|
||||||
int port_num;
|
|
||||||
char *p;
|
|
||||||
CharDriverState *chr = NULL;
|
CharDriverState *chr = NULL;
|
||||||
CharDriverState *mon_chr;
|
CharDriverState *mon_chr;
|
||||||
|
|
||||||
if (!port || !*port)
|
if (!device)
|
||||||
return -1;
|
return -1;
|
||||||
if (strcmp(port, "none") != 0) {
|
if (strcmp(device, "none") != 0) {
|
||||||
port_num = strtol(port, &p, 10);
|
if (strstart(device, "tcp:", NULL)) {
|
||||||
if (*p == 0) {
|
/* enforce required TCP attributes */
|
||||||
/* A numeric value is interpreted as a port number. */
|
snprintf(gdbstub_device_name, sizeof(gdbstub_device_name),
|
||||||
snprintf(gdbstub_port_name, sizeof(gdbstub_port_name),
|
"%s,nowait,nodelay,server", device);
|
||||||
"tcp::%d,nowait,nodelay,server", port_num);
|
device = gdbstub_device_name;
|
||||||
port = gdbstub_port_name;
|
|
||||||
}
|
}
|
||||||
|
#ifndef _WIN32
|
||||||
|
else if (strcmp(device, "stdio") == 0) {
|
||||||
|
struct sigaction act;
|
||||||
|
|
||||||
chr = qemu_chr_open("gdb", port, NULL);
|
memset(&act, 0, sizeof(act));
|
||||||
|
act.sa_handler = gdb_sigterm_handler;
|
||||||
|
sigaction(SIGINT, &act, NULL);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
chr = qemu_chr_open("gdb", device, NULL);
|
||||||
if (!chr)
|
if (!chr)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
|
17
monitor.c
17
monitor.c
@ -570,17 +570,18 @@ static void encrypted_bdrv_it(void *opaque, BlockDriverState *bs)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_GDBSTUB
|
#ifdef CONFIG_GDBSTUB
|
||||||
static void do_gdbserver(Monitor *mon, const char *port)
|
static void do_gdbserver(Monitor *mon, const char *device)
|
||||||
{
|
{
|
||||||
if (!port)
|
if (!device)
|
||||||
port = DEFAULT_GDBSTUB_PORT;
|
device = "tcp::" DEFAULT_GDBSTUB_PORT;
|
||||||
if (gdbserver_start(port) < 0) {
|
if (gdbserver_start(device) < 0) {
|
||||||
monitor_printf(mon, "Could not open gdbserver socket on port '%s'\n",
|
monitor_printf(mon, "Could not open gdbserver on device '%s'\n",
|
||||||
port);
|
device);
|
||||||
} else if (strcmp(port, "none") == 0) {
|
} else if (strcmp(device, "none") == 0) {
|
||||||
monitor_printf(mon, "Disabled gdbserver\n");
|
monitor_printf(mon, "Disabled gdbserver\n");
|
||||||
} else {
|
} else {
|
||||||
monitor_printf(mon, "Waiting gdb connection on port '%s'\n", port);
|
monitor_printf(mon, "Waiting for gdb connection on device '%s'\n",
|
||||||
|
device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -1216,19 +1216,25 @@ STEXI
|
|||||||
Do not start CPU at startup (you must type 'c' in the monitor).
|
Do not start CPU at startup (you must type 'c' in the monitor).
|
||||||
ETEXI
|
ETEXI
|
||||||
|
|
||||||
DEF("s", 0, QEMU_OPTION_s, \
|
DEF("gdb", HAS_ARG, QEMU_OPTION_gdb, \
|
||||||
"-s wait gdb connection to port\n")
|
"-gdb dev wait for gdb connection on 'dev'\n")
|
||||||
STEXI
|
STEXI
|
||||||
@item -s
|
@item -gdb @var{dev}
|
||||||
Wait gdb connection to port 1234 (@pxref{gdb_usage}).
|
Wait for gdb connection on device @var{dev} (@pxref{gdb_usage}). Typical
|
||||||
|
connections will likely be TCP-based, but also UDP, pseudo TTY, or even
|
||||||
|
stdio are reasonable use case. The latter is allowing to start qemu from
|
||||||
|
within gdb and establish the connection via a pipe:
|
||||||
|
@example
|
||||||
|
(gdb) target remote | exec qemu -gdb stdio ...
|
||||||
|
@end example
|
||||||
ETEXI
|
ETEXI
|
||||||
|
|
||||||
DEF("p", HAS_ARG, QEMU_OPTION_p, \
|
DEF("s", 0, QEMU_OPTION_s, \
|
||||||
"-p port set gdb connection port [default=%s]\n")
|
"-s shorthand for -gdb tcp::%s\n")
|
||||||
STEXI
|
STEXI
|
||||||
@item -p @var{port}
|
@item -s
|
||||||
Change gdb connection port. @var{port} can be either a decimal number
|
Shorthand for -gdb tcp::1234, i.e. open a gdbserver on TCP port 1234
|
||||||
to specify a TCP port, or a host device (same devices as the serial port).
|
(@pxref{gdb_usage}).
|
||||||
ETEXI
|
ETEXI
|
||||||
|
|
||||||
DEF("d", HAS_ARG, QEMU_OPTION_d, \
|
DEF("d", HAS_ARG, QEMU_OPTION_d, \
|
||||||
|
25
vl.c
25
vl.c
@ -4233,8 +4233,7 @@ static void termsig_setup(void)
|
|||||||
int main(int argc, char **argv, char **envp)
|
int main(int argc, char **argv, char **envp)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_GDBSTUB
|
#ifdef CONFIG_GDBSTUB
|
||||||
int use_gdbstub;
|
const char *gdbstub_dev = NULL;
|
||||||
const char *gdbstub_port;
|
|
||||||
#endif
|
#endif
|
||||||
uint32_t boot_devices_bitmap = 0;
|
uint32_t boot_devices_bitmap = 0;
|
||||||
int i;
|
int i;
|
||||||
@ -4317,10 +4316,6 @@ int main(int argc, char **argv, char **envp)
|
|||||||
initrd_filename = NULL;
|
initrd_filename = NULL;
|
||||||
ram_size = 0;
|
ram_size = 0;
|
||||||
vga_ram_size = VGA_RAM_SIZE;
|
vga_ram_size = VGA_RAM_SIZE;
|
||||||
#ifdef CONFIG_GDBSTUB
|
|
||||||
use_gdbstub = 0;
|
|
||||||
gdbstub_port = DEFAULT_GDBSTUB_PORT;
|
|
||||||
#endif
|
|
||||||
snapshot = 0;
|
snapshot = 0;
|
||||||
nographic = 0;
|
nographic = 0;
|
||||||
curses = 0;
|
curses = 0;
|
||||||
@ -4653,10 +4648,10 @@ int main(int argc, char **argv, char **envp)
|
|||||||
break;
|
break;
|
||||||
#ifdef CONFIG_GDBSTUB
|
#ifdef CONFIG_GDBSTUB
|
||||||
case QEMU_OPTION_s:
|
case QEMU_OPTION_s:
|
||||||
use_gdbstub = 1;
|
gdbstub_dev = "tcp::" DEFAULT_GDBSTUB_PORT;
|
||||||
break;
|
break;
|
||||||
case QEMU_OPTION_p:
|
case QEMU_OPTION_gdb:
|
||||||
gdbstub_port = optarg;
|
gdbstub_dev = optarg;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
case QEMU_OPTION_L:
|
case QEMU_OPTION_L:
|
||||||
@ -5370,14 +5365,10 @@ int main(int argc, char **argv, char **envp)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_GDBSTUB
|
#ifdef CONFIG_GDBSTUB
|
||||||
if (use_gdbstub) {
|
if (gdbstub_dev && gdbserver_start(gdbstub_dev) < 0) {
|
||||||
/* XXX: use standard host:port notation and modify options
|
fprintf(stderr, "qemu: could not open gdbserver on device '%s'\n",
|
||||||
accordingly. */
|
gdbstub_dev);
|
||||||
if (gdbserver_start(gdbstub_port) < 0) {
|
exit(1);
|
||||||
fprintf(stderr, "qemu: could not open gdbstub device on port '%s'\n",
|
|
||||||
gdbstub_port);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user