Commit b4608c0455a410994b31888b458a552d640b9765
1 parent
d5a0b50c
added gdb support to vl
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@288 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
3 changed files
with
497 additions
and
65 deletions
Makefile
| ... | ... | @@ -98,7 +98,7 @@ SRCS:= $(OBJS:.o=.c) |
| 98 | 98 | OBJS+= libqemu.a |
| 99 | 99 | |
| 100 | 100 | # cpu emulator library |
| 101 | -LIBOBJS=thunk.o exec.o translate.o cpu-exec.o | |
| 101 | +LIBOBJS=thunk.o exec.o translate.o cpu-exec.o gdbstub.o | |
| 102 | 102 | |
| 103 | 103 | ifeq ($(TARGET_ARCH), i386) |
| 104 | 104 | LIBOBJS+=translate-i386.o op-i386.o helper-i386.o | ... | ... |
gdbstub.c
0 → 100644
| 1 | +/* | |
| 2 | + * gdb server stub | |
| 3 | + * | |
| 4 | + * Copyright (c) 2003 Fabrice Bellard | |
| 5 | + * | |
| 6 | + * This library is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU Lesser General Public | |
| 8 | + * License as published by the Free Software Foundation; either | |
| 9 | + * version 2 of the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This library is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | |
| 14 | + * Lesser General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU Lesser General Public | |
| 17 | + * License along with this library; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | |
| 19 | + */ | |
| 20 | +#include <stdlib.h> | |
| 21 | +#include <stdio.h> | |
| 22 | +#include <string.h> | |
| 23 | +#include <unistd.h> | |
| 24 | +#include <errno.h> | |
| 25 | +#include <sys/socket.h> | |
| 26 | +#include <netinet/in.h> | |
| 27 | +#include <netinet/tcp.h> | |
| 28 | +#include <signal.h> | |
| 29 | + | |
| 30 | +#include "config.h" | |
| 31 | +#ifdef TARGET_I386 | |
| 32 | +#include "cpu-i386.h" | |
| 33 | +#endif | |
| 34 | +#ifdef TARGET_ARM | |
| 35 | +#include "cpu-arm.h" | |
| 36 | +#endif | |
| 37 | +#include "thunk.h" | |
| 38 | +#include "exec.h" | |
| 39 | + | |
| 40 | +//#define DEBUG_GDB | |
| 41 | + | |
| 42 | +int gdbstub_fd = -1; | |
| 43 | + | |
| 44 | +/* return 0 if OK */ | |
| 45 | +static int gdbstub_open(int port) | |
| 46 | +{ | |
| 47 | + struct sockaddr_in sockaddr; | |
| 48 | + socklen_t len; | |
| 49 | + int fd, val, ret; | |
| 50 | + | |
| 51 | + fd = socket(PF_INET, SOCK_STREAM, 0); | |
| 52 | + if (fd < 0) { | |
| 53 | + perror("socket"); | |
| 54 | + return -1; | |
| 55 | + } | |
| 56 | + | |
| 57 | + /* allow fast reuse */ | |
| 58 | + val = 1; | |
| 59 | + setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)); | |
| 60 | + | |
| 61 | + sockaddr.sin_family = AF_INET; | |
| 62 | + sockaddr.sin_port = htons(port); | |
| 63 | + sockaddr.sin_addr.s_addr = 0; | |
| 64 | + ret = bind(fd, (struct sockaddr *)&sockaddr, sizeof(sockaddr)); | |
| 65 | + if (ret < 0) { | |
| 66 | + perror("bind"); | |
| 67 | + return -1; | |
| 68 | + } | |
| 69 | + ret = listen(fd, 0); | |
| 70 | + if (ret < 0) { | |
| 71 | + perror("listen"); | |
| 72 | + return -1; | |
| 73 | + } | |
| 74 | + | |
| 75 | + /* now wait for one connection */ | |
| 76 | + for(;;) { | |
| 77 | + len = sizeof(sockaddr); | |
| 78 | + gdbstub_fd = accept(fd, (struct sockaddr *)&sockaddr, &len); | |
| 79 | + if (gdbstub_fd < 0 && errno != EINTR) { | |
| 80 | + perror("accept"); | |
| 81 | + return -1; | |
| 82 | + } else if (gdbstub_fd >= 0) { | |
| 83 | + break; | |
| 84 | + } | |
| 85 | + } | |
| 86 | + | |
| 87 | + /* set short latency */ | |
| 88 | + val = 1; | |
| 89 | + setsockopt(gdbstub_fd, SOL_TCP, TCP_NODELAY, &val, sizeof(val)); | |
| 90 | + return 0; | |
| 91 | +} | |
| 92 | + | |
| 93 | +static int get_char(void) | |
| 94 | +{ | |
| 95 | + uint8_t ch; | |
| 96 | + int ret; | |
| 97 | + | |
| 98 | + for(;;) { | |
| 99 | + ret = read(gdbstub_fd, &ch, 1); | |
| 100 | + if (ret < 0) { | |
| 101 | + if (errno != EINTR && errno != EAGAIN) | |
| 102 | + return -1; | |
| 103 | + } else if (ret == 0) { | |
| 104 | + return -1; | |
| 105 | + } else { | |
| 106 | + break; | |
| 107 | + } | |
| 108 | + } | |
| 109 | + return ch; | |
| 110 | +} | |
| 111 | + | |
| 112 | +static void put_buffer(const uint8_t *buf, int len) | |
| 113 | +{ | |
| 114 | + int ret; | |
| 115 | + | |
| 116 | + while (len > 0) { | |
| 117 | + ret = write(gdbstub_fd, buf, len); | |
| 118 | + if (ret < 0) { | |
| 119 | + if (errno != EINTR && errno != EAGAIN) | |
| 120 | + return; | |
| 121 | + } else { | |
| 122 | + buf += ret; | |
| 123 | + len -= ret; | |
| 124 | + } | |
| 125 | + } | |
| 126 | +} | |
| 127 | + | |
| 128 | +static inline int fromhex(int v) | |
| 129 | +{ | |
| 130 | + if (v >= '0' && v <= '9') | |
| 131 | + return v - '0'; | |
| 132 | + else if (v >= 'A' && v <= 'F') | |
| 133 | + return v - 'A' + 10; | |
| 134 | + else if (v >= 'a' && v <= 'f') | |
| 135 | + return v - 'a' + 10; | |
| 136 | + else | |
| 137 | + return 0; | |
| 138 | +} | |
| 139 | + | |
| 140 | +static inline int tohex(int v) | |
| 141 | +{ | |
| 142 | + if (v < 10) | |
| 143 | + return v + '0'; | |
| 144 | + else | |
| 145 | + return v - 10 + 'a'; | |
| 146 | +} | |
| 147 | + | |
| 148 | +static void memtohex(char *buf, const uint8_t *mem, int len) | |
| 149 | +{ | |
| 150 | + int i, c; | |
| 151 | + char *q; | |
| 152 | + q = buf; | |
| 153 | + for(i = 0; i < len; i++) { | |
| 154 | + c = mem[i]; | |
| 155 | + *q++ = tohex(c >> 4); | |
| 156 | + *q++ = tohex(c & 0xf); | |
| 157 | + } | |
| 158 | + *q = '\0'; | |
| 159 | +} | |
| 160 | + | |
| 161 | +static void hextomem(uint8_t *mem, const char *buf, int len) | |
| 162 | +{ | |
| 163 | + int i; | |
| 164 | + | |
| 165 | + for(i = 0; i < len; i++) { | |
| 166 | + mem[i] = (fromhex(buf[0]) << 4) | fromhex(buf[1]); | |
| 167 | + buf += 2; | |
| 168 | + } | |
| 169 | +} | |
| 170 | + | |
| 171 | +/* return -1 if error or EOF */ | |
| 172 | +static int get_packet(char *buf, int buf_size) | |
| 173 | +{ | |
| 174 | + int ch, len, csum, csum1; | |
| 175 | + char reply[1]; | |
| 176 | + | |
| 177 | + for(;;) { | |
| 178 | + for(;;) { | |
| 179 | + ch = get_char(); | |
| 180 | + if (ch < 0) | |
| 181 | + return -1; | |
| 182 | + if (ch == '$') | |
| 183 | + break; | |
| 184 | + } | |
| 185 | + len = 0; | |
| 186 | + csum = 0; | |
| 187 | + for(;;) { | |
| 188 | + ch = get_char(); | |
| 189 | + if (ch < 0) | |
| 190 | + return -1; | |
| 191 | + if (ch == '#') | |
| 192 | + break; | |
| 193 | + if (len > buf_size - 1) | |
| 194 | + return -1; | |
| 195 | + buf[len++] = ch; | |
| 196 | + csum += ch; | |
| 197 | + } | |
| 198 | + buf[len] = '\0'; | |
| 199 | + ch = get_char(); | |
| 200 | + if (ch < 0) | |
| 201 | + return -1; | |
| 202 | + csum1 = fromhex(ch) << 4; | |
| 203 | + ch = get_char(); | |
| 204 | + if (ch < 0) | |
| 205 | + return -1; | |
| 206 | + csum1 |= fromhex(ch); | |
| 207 | + if ((csum & 0xff) != csum1) { | |
| 208 | + reply[0] = '-'; | |
| 209 | + put_buffer(reply, 1); | |
| 210 | + } else { | |
| 211 | + reply[0] = '+'; | |
| 212 | + put_buffer(reply, 1); | |
| 213 | + break; | |
| 214 | + } | |
| 215 | + } | |
| 216 | +#ifdef DEBUG_GDB | |
| 217 | + printf("command='%s'\n", buf); | |
| 218 | +#endif | |
| 219 | + return len; | |
| 220 | +} | |
| 221 | + | |
| 222 | +/* return -1 if error, 0 if OK */ | |
| 223 | +static int put_packet(char *buf) | |
| 224 | +{ | |
| 225 | + char buf1[3]; | |
| 226 | + int len, csum, ch, i; | |
| 227 | + | |
| 228 | +#ifdef DEBUG_GDB | |
| 229 | + printf("reply='%s'\n", buf); | |
| 230 | +#endif | |
| 231 | + | |
| 232 | + for(;;) { | |
| 233 | + buf1[0] = '$'; | |
| 234 | + put_buffer(buf1, 1); | |
| 235 | + len = strlen(buf); | |
| 236 | + put_buffer(buf, len); | |
| 237 | + csum = 0; | |
| 238 | + for(i = 0; i < len; i++) { | |
| 239 | + csum += buf[i]; | |
| 240 | + } | |
| 241 | + buf1[0] = '#'; | |
| 242 | + buf1[1] = tohex((csum >> 4) & 0xf); | |
| 243 | + buf1[2] = tohex((csum) & 0xf); | |
| 244 | + | |
| 245 | + put_buffer(buf1, 3); | |
| 246 | + | |
| 247 | + ch = get_char(); | |
| 248 | + if (ch < 0) | |
| 249 | + return -1; | |
| 250 | + if (ch == '+') | |
| 251 | + break; | |
| 252 | + } | |
| 253 | + return 0; | |
| 254 | +} | |
| 255 | + | |
| 256 | +static int memory_rw(uint8_t *buf, uint32_t addr, int len, int is_write) | |
| 257 | +{ | |
| 258 | + int l, flags; | |
| 259 | + uint32_t page; | |
| 260 | + | |
| 261 | + while (len > 0) { | |
| 262 | + page = addr & TARGET_PAGE_MASK; | |
| 263 | + l = (page + TARGET_PAGE_SIZE) - addr; | |
| 264 | + if (l > len) | |
| 265 | + l = len; | |
| 266 | + flags = page_get_flags(page); | |
| 267 | + if (!(flags & PAGE_VALID)) | |
| 268 | + return -1; | |
| 269 | + if (is_write) { | |
| 270 | + if (!(flags & PAGE_WRITE)) | |
| 271 | + return -1; | |
| 272 | + memcpy((uint8_t *)addr, buf, l); | |
| 273 | + } else { | |
| 274 | + if (!(flags & PAGE_READ)) | |
| 275 | + return -1; | |
| 276 | + memcpy(buf, (uint8_t *)addr, l); | |
| 277 | + } | |
| 278 | + len -= l; | |
| 279 | + buf += l; | |
| 280 | + addr += l; | |
| 281 | + } | |
| 282 | + return 0; | |
| 283 | +} | |
| 284 | + | |
| 285 | +/* port = 0 means default port */ | |
| 286 | +int cpu_gdbstub(void *opaque, void (*main_loop)(void *opaque), int port) | |
| 287 | +{ | |
| 288 | + CPUState *env; | |
| 289 | + const char *p; | |
| 290 | + int ret, ch, nb_regs, i; | |
| 291 | + char buf[4096]; | |
| 292 | + uint8_t mem_buf[2000]; | |
| 293 | + uint32_t *registers; | |
| 294 | + uint32_t addr, len; | |
| 295 | + | |
| 296 | + printf("Waiting gdb connection on port %d\n", port); | |
| 297 | + if (gdbstub_open(port) < 0) | |
| 298 | + return -1; | |
| 299 | + printf("Connected\n"); | |
| 300 | + for(;;) { | |
| 301 | + ret = get_packet(buf, sizeof(buf)); | |
| 302 | + if (ret < 0) | |
| 303 | + break; | |
| 304 | + p = buf; | |
| 305 | + ch = *p++; | |
| 306 | + switch(ch) { | |
| 307 | + case '?': | |
| 308 | + snprintf(buf, sizeof(buf), "S%02x", SIGTRAP); | |
| 309 | + put_packet(buf); | |
| 310 | + break; | |
| 311 | + case 'c': | |
| 312 | + main_loop(opaque); | |
| 313 | + snprintf(buf, sizeof(buf), "S%02x", 0); | |
| 314 | + put_packet(buf); | |
| 315 | + break; | |
| 316 | + case 'g': | |
| 317 | + env = cpu_gdbstub_get_env(opaque); | |
| 318 | + registers = (void *)mem_buf; | |
| 319 | +#if defined(TARGET_I386) | |
| 320 | + for(i = 0; i < 8; i++) { | |
| 321 | + registers[i] = tswapl(env->regs[i]); | |
| 322 | + } | |
| 323 | + registers[8] = env->eip; | |
| 324 | + registers[9] = env->eflags; | |
| 325 | + registers[10] = env->segs[R_CS].selector; | |
| 326 | + registers[11] = env->segs[R_SS].selector; | |
| 327 | + registers[12] = env->segs[R_DS].selector; | |
| 328 | + registers[13] = env->segs[R_ES].selector; | |
| 329 | + registers[14] = env->segs[R_FS].selector; | |
| 330 | + registers[15] = env->segs[R_GS].selector; | |
| 331 | + nb_regs = 16; | |
| 332 | +#endif | |
| 333 | + memtohex(buf, (const uint8_t *)registers, | |
| 334 | + sizeof(registers[0]) * nb_regs); | |
| 335 | + put_packet(buf); | |
| 336 | + break; | |
| 337 | + case 'G': | |
| 338 | + env = cpu_gdbstub_get_env(opaque); | |
| 339 | + registers = (void *)mem_buf; | |
| 340 | +#if defined(TARGET_I386) | |
| 341 | + hextomem((uint8_t *)registers, p, 16 * 4); | |
| 342 | + for(i = 0; i < 8; i++) { | |
| 343 | + env->regs[i] = tswapl(registers[i]); | |
| 344 | + } | |
| 345 | + env->eip = registers[8]; | |
| 346 | + env->eflags = registers[9]; | |
| 347 | +#define LOAD_SEG(index, sreg)\ | |
| 348 | + if (tswapl(registers[index]) != env->segs[sreg].selector)\ | |
| 349 | + cpu_x86_load_seg(env, sreg, tswapl(registers[index])); | |
| 350 | + LOAD_SEG(10, R_CS); | |
| 351 | + LOAD_SEG(11, R_SS); | |
| 352 | + LOAD_SEG(12, R_DS); | |
| 353 | + LOAD_SEG(13, R_ES); | |
| 354 | + LOAD_SEG(14, R_FS); | |
| 355 | + LOAD_SEG(15, R_GS); | |
| 356 | +#endif | |
| 357 | + put_packet("OK"); | |
| 358 | + break; | |
| 359 | + case 'm': | |
| 360 | + addr = strtoul(p, (char **)&p, 16); | |
| 361 | + if (*p == ',') | |
| 362 | + p++; | |
| 363 | + len = strtoul(p, NULL, 16); | |
| 364 | + if (memory_rw(mem_buf, addr, len, 0) != 0) | |
| 365 | + memset(mem_buf, 0, len); | |
| 366 | + memtohex(buf, mem_buf, len); | |
| 367 | + put_packet(buf); | |
| 368 | + break; | |
| 369 | + case 'M': | |
| 370 | + addr = strtoul(p, (char **)&p, 16); | |
| 371 | + if (*p == ',') | |
| 372 | + p++; | |
| 373 | + len = strtoul(p, (char **)&p, 16); | |
| 374 | + if (*p == ',') | |
| 375 | + p++; | |
| 376 | + hextomem(mem_buf, p, len); | |
| 377 | + if (memory_rw(mem_buf, addr, len, 1) != 0) | |
| 378 | + put_packet("ENN"); | |
| 379 | + else | |
| 380 | + put_packet("OK"); | |
| 381 | + break; | |
| 382 | + default: | |
| 383 | + /* put empty packet */ | |
| 384 | + buf[0] = '\0'; | |
| 385 | + put_packet(buf); | |
| 386 | + break; | |
| 387 | + } | |
| 388 | + } | |
| 389 | + return 0; | |
| 390 | +} | ... | ... |
vl.c
| ... | ... | @@ -1783,27 +1783,116 @@ static void host_alarm_handler(int host_signum, siginfo_t *info, |
| 1783 | 1783 | } |
| 1784 | 1784 | } |
| 1785 | 1785 | |
| 1786 | +/* main execution loop */ | |
| 1787 | + | |
| 1788 | +CPUState *cpu_gdbstub_get_env(void *opaque) | |
| 1789 | +{ | |
| 1790 | + return global_env; | |
| 1791 | +} | |
| 1792 | + | |
| 1793 | +void main_loop(void *opaque) | |
| 1794 | +{ | |
| 1795 | + struct pollfd ufds[2], *pf, *serial_ufd, *net_ufd, *gdb_ufd; | |
| 1796 | + int ret, n, timeout; | |
| 1797 | + uint8_t ch; | |
| 1798 | + CPUState *env = global_env; | |
| 1799 | + | |
| 1800 | + for(;;) { | |
| 1801 | + | |
| 1802 | + ret = cpu_x86_exec(env); | |
| 1803 | + | |
| 1804 | + /* if hlt instruction, we wait until the next IRQ */ | |
| 1805 | + if (ret == EXCP_HLT) | |
| 1806 | + timeout = 10; | |
| 1807 | + else | |
| 1808 | + timeout = 0; | |
| 1809 | + /* poll any events */ | |
| 1810 | + serial_ufd = NULL; | |
| 1811 | + pf = ufds; | |
| 1812 | + if (!(serial_ports[0].lsr & UART_LSR_DR)) { | |
| 1813 | + serial_ufd = pf; | |
| 1814 | + pf->fd = 0; | |
| 1815 | + pf->events = POLLIN; | |
| 1816 | + pf++; | |
| 1817 | + } | |
| 1818 | + net_ufd = NULL; | |
| 1819 | + if (net_fd > 0 && ne2000_can_receive(&ne2000_state)) { | |
| 1820 | + net_ufd = pf; | |
| 1821 | + pf->fd = net_fd; | |
| 1822 | + pf->events = POLLIN; | |
| 1823 | + pf++; | |
| 1824 | + } | |
| 1825 | + gdb_ufd = NULL; | |
| 1826 | + if (gdbstub_fd > 0) { | |
| 1827 | + gdb_ufd = pf; | |
| 1828 | + pf->fd = gdbstub_fd; | |
| 1829 | + pf->events = POLLIN; | |
| 1830 | + pf++; | |
| 1831 | + } | |
| 1832 | + | |
| 1833 | + ret = poll(ufds, pf - ufds, timeout); | |
| 1834 | + if (ret > 0) { | |
| 1835 | + if (serial_ufd && (serial_ufd->revents & POLLIN)) { | |
| 1836 | + n = read(0, &ch, 1); | |
| 1837 | + if (n == 1) { | |
| 1838 | + serial_received_byte(&serial_ports[0], ch); | |
| 1839 | + } | |
| 1840 | + } | |
| 1841 | + if (net_ufd && (net_ufd->revents & POLLIN)) { | |
| 1842 | + uint8_t buf[MAX_ETH_FRAME_SIZE]; | |
| 1843 | + | |
| 1844 | + n = read(net_fd, buf, MAX_ETH_FRAME_SIZE); | |
| 1845 | + if (n > 0) { | |
| 1846 | + if (n < 60) { | |
| 1847 | + memset(buf + n, 0, 60 - n); | |
| 1848 | + n = 60; | |
| 1849 | + } | |
| 1850 | + ne2000_receive(&ne2000_state, buf, n); | |
| 1851 | + } | |
| 1852 | + } | |
| 1853 | + if (gdb_ufd && (gdb_ufd->revents & POLLIN)) { | |
| 1854 | + uint8_t buf[1]; | |
| 1855 | + /* stop emulation if requested by gdb */ | |
| 1856 | + n = read(gdbstub_fd, buf, 1); | |
| 1857 | + if (n == 1) | |
| 1858 | + break; | |
| 1859 | + } | |
| 1860 | + } | |
| 1861 | + | |
| 1862 | + /* timer IRQ */ | |
| 1863 | + if (timer_irq_pending) { | |
| 1864 | + pic_set_irq(0, 1); | |
| 1865 | + pic_set_irq(0, 0); | |
| 1866 | + timer_irq_pending = 0; | |
| 1867 | + } | |
| 1868 | + | |
| 1869 | + pic_handle_irq(); | |
| 1870 | + } | |
| 1871 | +} | |
| 1872 | + | |
| 1786 | 1873 | void help(void) |
| 1787 | 1874 | { |
| 1788 | 1875 | printf("Virtual Linux version " QEMU_VERSION ", Copyright (c) 2003 Fabrice Bellard\n" |
| 1789 | - "usage: vl [-h] bzImage initrd [kernel parameters...]\n" | |
| 1876 | + "usage: vl [options] bzImage initrd [kernel parameters...]\n" | |
| 1790 | 1877 | "\n" |
| 1791 | 1878 | "'bzImage' is a Linux kernel image (PAGE_OFFSET must be defined\n" |
| 1792 | 1879 | "to 0x90000000 in asm/page.h and arch/i386/vmlinux.lds)\n" |
| 1793 | 1880 | "'initrd' is an initrd image\n" |
| 1794 | 1881 | "-m megs set virtual RAM size to megs MB\n" |
| 1795 | 1882 | "-n script set network init script [default=%s]\n" |
| 1883 | + "-s wait gdb connection to port %d\n" | |
| 1884 | + "-p port change gdb connection port\n" | |
| 1796 | 1885 | "-d output log in /tmp/vl.log\n" |
| 1797 | 1886 | "\n" |
| 1798 | 1887 | "During emulation, use C-a h to get terminal commands:\n", |
| 1799 | - DEFAULT_NETWORK_SCRIPT); | |
| 1888 | + DEFAULT_NETWORK_SCRIPT, DEFAULT_GDBSTUB_PORT); | |
| 1800 | 1889 | term_print_help(); |
| 1801 | 1890 | exit(1); |
| 1802 | 1891 | } |
| 1803 | 1892 | |
| 1804 | 1893 | int main(int argc, char **argv) |
| 1805 | 1894 | { |
| 1806 | - int c, ret, initrd_size, i; | |
| 1895 | + int c, ret, initrd_size, i, use_gdbstub, gdbstub_port; | |
| 1807 | 1896 | struct linux_params *params; |
| 1808 | 1897 | struct sigaction act; |
| 1809 | 1898 | struct itimerval itv; |
| ... | ... | @@ -1815,8 +1904,10 @@ int main(int argc, char **argv) |
| 1815 | 1904 | |
| 1816 | 1905 | phys_ram_size = 32 * 1024 * 1024; |
| 1817 | 1906 | pstrcpy(network_script, sizeof(network_script), DEFAULT_NETWORK_SCRIPT); |
| 1907 | + use_gdbstub = 0; | |
| 1908 | + gdbstub_port = DEFAULT_GDBSTUB_PORT; | |
| 1818 | 1909 | for(;;) { |
| 1819 | - c = getopt(argc, argv, "hm:dn:"); | |
| 1910 | + c = getopt(argc, argv, "hm:dn:sp:"); | |
| 1820 | 1911 | if (c == -1) |
| 1821 | 1912 | break; |
| 1822 | 1913 | switch(c) { |
| ... | ... | @@ -1834,6 +1925,12 @@ int main(int argc, char **argv) |
| 1834 | 1925 | case 'n': |
| 1835 | 1926 | pstrcpy(network_script, sizeof(network_script), optarg); |
| 1836 | 1927 | break; |
| 1928 | + case 's': | |
| 1929 | + use_gdbstub = 1; | |
| 1930 | + break; | |
| 1931 | + case 'p': | |
| 1932 | + gdbstub_port = atoi(optarg); | |
| 1933 | + break; | |
| 1837 | 1934 | } |
| 1838 | 1935 | } |
| 1839 | 1936 | if (optind + 1 >= argc) |
| ... | ... | @@ -1974,66 +2071,11 @@ int main(int argc, char **argv) |
| 1974 | 2071 | getitimer(ITIMER_REAL, &itv); |
| 1975 | 2072 | pit_min_timer_count = ((uint64_t)itv.it_interval.tv_usec * PIT_FREQ) / |
| 1976 | 2073 | 1000000; |
| 1977 | - | |
| 1978 | - for(;;) { | |
| 1979 | - struct pollfd ufds[2], *pf, *serial_ufd, *net_ufd; | |
| 1980 | - int ret, n, timeout; | |
| 1981 | - uint8_t ch; | |
| 1982 | - | |
| 1983 | - ret = cpu_x86_exec(env); | |
| 1984 | - | |
| 1985 | - /* if hlt instruction, we wait until the next IRQ */ | |
| 1986 | - if (ret == EXCP_HLT) | |
| 1987 | - timeout = 10; | |
| 1988 | - else | |
| 1989 | - timeout = 0; | |
| 1990 | - /* poll any events */ | |
| 1991 | - serial_ufd = NULL; | |
| 1992 | - net_ufd = NULL; | |
| 1993 | - pf = ufds; | |
| 1994 | - if (!(serial_ports[0].lsr & UART_LSR_DR)) { | |
| 1995 | - serial_ufd = pf; | |
| 1996 | - pf->fd = 0; | |
| 1997 | - pf->events = POLLIN; | |
| 1998 | - pf++; | |
| 1999 | - } | |
| 2000 | - if (net_fd > 0 && ne2000_can_receive(&ne2000_state)) { | |
| 2001 | - net_ufd = pf; | |
| 2002 | - pf->fd = net_fd; | |
| 2003 | - pf->events = POLLIN; | |
| 2004 | - pf++; | |
| 2005 | - } | |
| 2006 | - ret = poll(ufds, pf - ufds, timeout); | |
| 2007 | - if (ret > 0) { | |
| 2008 | - if (serial_ufd && (serial_ufd->revents & POLLIN)) { | |
| 2009 | - n = read(0, &ch, 1); | |
| 2010 | - if (n == 1) { | |
| 2011 | - serial_received_byte(&serial_ports[0], ch); | |
| 2012 | - } | |
| 2013 | - } | |
| 2014 | - if (net_ufd && (net_ufd->revents & POLLIN)) { | |
| 2015 | - uint8_t buf[MAX_ETH_FRAME_SIZE]; | |
| 2016 | - | |
| 2017 | - n = read(net_fd, buf, MAX_ETH_FRAME_SIZE); | |
| 2018 | - if (n > 0) { | |
| 2019 | - if (n < 60) { | |
| 2020 | - memset(buf + n, 0, 60 - n); | |
| 2021 | - n = 60; | |
| 2022 | - } | |
| 2023 | - ne2000_receive(&ne2000_state, buf, n); | |
| 2024 | - } | |
| 2025 | - } | |
| 2026 | - } | |
| 2027 | - | |
| 2028 | - /* timer IRQ */ | |
| 2029 | - if (timer_irq_pending) { | |
| 2030 | - pic_set_irq(0, 1); | |
| 2031 | - pic_set_irq(0, 0); | |
| 2032 | - timer_irq_pending = 0; | |
| 2033 | - } | |
| 2034 | - | |
| 2035 | - pic_handle_irq(); | |
| 2074 | + | |
| 2075 | + if (use_gdbstub) { | |
| 2076 | + cpu_gdbstub(NULL, main_loop, gdbstub_port); | |
| 2077 | + } else { | |
| 2078 | + main_loop(NULL); | |
| 2036 | 2079 | } |
| 2037 | - | |
| 2038 | 2080 | return 0; |
| 2039 | 2081 | } | ... | ... |