fix gdbstub support for multiple threads in usermode, v3
When debugging multi-threaded programs, QEMU's gdb stub would report the correct number of threads (the qfThreadInfo and qsThreadInfo packets). However, the stub was unable to actually switch between threads (the T packet), since it would report every thread except the first as being dead. Furthermore, the stub relied upon cpu_index as a reliable means of assigning IDs to the threads. This was a bad idea; if you have this sequence of events: initial thread created new thread #1 new thread #2 thread #1 exits new thread #3 thread #3 will have the same cpu_index as thread #1, which would confuse GDB. (This problem is partly due to the remote protocol not having a good way to send thread creation/destruction events.) We fix this by using the host thread ID for the identifier passed to GDB when debugging a multi-threaded userspace program. The thread ID might wrap, but the same sort of problems with wrapping thread IDs would come up with debugging programs natively, so this doesn't represent a problem. Signed-off-by: Nathan Froyd <froydnj@codesourcery.com>
This commit is contained in:
		
							parent
							
								
									4548eaea13
								
							
						
					
					
						commit
						1e9fa73016
					
				|  | @ -184,6 +184,7 @@ typedef struct CPUWatchpoint { | ||||||
|                                                                         \ |                                                                         \ | ||||||
|     CPUState *next_cpu; /* next CPU sharing TB cache */                 \ |     CPUState *next_cpu; /* next CPU sharing TB cache */                 \ | ||||||
|     int cpu_index; /* CPU index (informative) */                        \ |     int cpu_index; /* CPU index (informative) */                        \ | ||||||
|  |     uint32_t host_tid; /* host thread ID */                             \ | ||||||
|     int numa_node; /* NUMA node this cpu is belonging to  */            \ |     int numa_node; /* NUMA node this cpu is belonging to  */            \ | ||||||
|     int running; /* Nonzero if cpu is currently running(usermode).  */  \ |     int running; /* Nonzero if cpu is currently running(usermode).  */  \ | ||||||
|     /* user data */                                                     \ |     /* user data */                                                     \ | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								exec.c
								
								
								
								
							
							
						
						
									
										2
									
								
								exec.c
								
								
								
								
							|  | @ -553,7 +553,7 @@ void cpu_exec_init(CPUState *env) | ||||||
|     penv = &first_cpu; |     penv = &first_cpu; | ||||||
|     cpu_index = 0; |     cpu_index = 0; | ||||||
|     while (*penv != NULL) { |     while (*penv != NULL) { | ||||||
|         penv = (CPUState **)&(*penv)->next_cpu; |         penv = &(*penv)->next_cpu; | ||||||
|         cpu_index++; |         cpu_index++; | ||||||
|     } |     } | ||||||
|     env->cpu_index = cpu_index; |     env->cpu_index = cpu_index; | ||||||
|  |  | ||||||
							
								
								
									
										69
									
								
								gdbstub.c
								
								
								
								
							
							
						
						
									
										69
									
								
								gdbstub.c
								
								
								
								
							|  | @ -1567,11 +1567,34 @@ static void gdb_set_cpu_pc(GDBState *s, target_ulong pc) | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static inline int gdb_id(CPUState *env) | ||||||
|  | { | ||||||
|  | #if defined(CONFIG_USER_ONLY) && defined(USE_NPTL) | ||||||
|  |     return env->host_tid; | ||||||
|  | #else | ||||||
|  |     return env->cpu_index + 1; | ||||||
|  | #endif | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static CPUState *find_cpu(uint32_t thread_id) | ||||||
|  | { | ||||||
|  |     CPUState *env; | ||||||
|  | 
 | ||||||
|  |     for (env = first_cpu; env != NULL; env = env->next_cpu) { | ||||||
|  |         if (gdb_id(env) == thread_id) { | ||||||
|  |             return env; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int gdb_handle_packet(GDBState *s, const char *line_buf) | static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
| { | { | ||||||
|     CPUState *env; |     CPUState *env; | ||||||
|     const char *p; |     const char *p; | ||||||
|     int ch, reg_size, type, res, thread; |     uint32_t thread; | ||||||
|  |     int ch, reg_size, type, res; | ||||||
|     char buf[MAX_PACKET_LENGTH]; |     char buf[MAX_PACKET_LENGTH]; | ||||||
|     uint8_t mem_buf[MAX_PACKET_LENGTH]; |     uint8_t mem_buf[MAX_PACKET_LENGTH]; | ||||||
|     uint8_t *registers; |     uint8_t *registers; | ||||||
|  | @ -1586,7 +1609,7 @@ static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
|     case '?': |     case '?': | ||||||
|         /* TODO: Make this return the correct value for user-mode.  */ |         /* TODO: Make this return the correct value for user-mode.  */ | ||||||
|         snprintf(buf, sizeof(buf), "T%02xthread:%02x;", GDB_SIGNAL_TRAP, |         snprintf(buf, sizeof(buf), "T%02xthread:%02x;", GDB_SIGNAL_TRAP, | ||||||
|                  s->c_cpu->cpu_index+1); |                  gdb_id(s->c_cpu)); | ||||||
|         put_packet(s, buf); |         put_packet(s, buf); | ||||||
|         /* Remove all the breakpoints when this query is issued,
 |         /* Remove all the breakpoints when this query is issued,
 | ||||||
|          * because gdb is doing and initial connect and the state |          * because gdb is doing and initial connect and the state | ||||||
|  | @ -1750,9 +1773,7 @@ static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
|             put_packet(s, "OK"); |             put_packet(s, "OK"); | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
|         for (env = first_cpu; env != NULL; env = env->next_cpu) |         env = find_cpu(thread); | ||||||
|             if (env->cpu_index + 1 == thread) |  | ||||||
|                 break; |  | ||||||
|         if (env == NULL) { |         if (env == NULL) { | ||||||
|             put_packet(s, "E22"); |             put_packet(s, "E22"); | ||||||
|             break; |             break; | ||||||
|  | @ -1773,14 +1794,13 @@ static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
|         break; |         break; | ||||||
|     case 'T': |     case 'T': | ||||||
|         thread = strtoull(p, (char **)&p, 16); |         thread = strtoull(p, (char **)&p, 16); | ||||||
| #ifndef CONFIG_USER_ONLY |         env = find_cpu(thread); | ||||||
|         if (thread > 0 && thread < smp_cpus + 1) | 
 | ||||||
| #else |         if (env != NULL) { | ||||||
|         if (thread == 1) |             put_packet(s, "OK"); | ||||||
| #endif |         } else { | ||||||
|              put_packet(s, "OK"); |  | ||||||
|         else |  | ||||||
|             put_packet(s, "E22"); |             put_packet(s, "E22"); | ||||||
|  |         } | ||||||
|         break; |         break; | ||||||
|     case 'q': |     case 'q': | ||||||
|     case 'Q': |     case 'Q': | ||||||
|  | @ -1818,7 +1838,7 @@ static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
|         } else if (strcmp(p,"sThreadInfo") == 0) { |         } else if (strcmp(p,"sThreadInfo") == 0) { | ||||||
|         report_cpuinfo: |         report_cpuinfo: | ||||||
|             if (s->query_cpu) { |             if (s->query_cpu) { | ||||||
|                 snprintf(buf, sizeof(buf), "m%x", s->query_cpu->cpu_index+1); |                 snprintf(buf, sizeof(buf), "m%x", gdb_id(s->query_cpu)); | ||||||
|                 put_packet(s, buf); |                 put_packet(s, buf); | ||||||
|                 s->query_cpu = s->query_cpu->next_cpu; |                 s->query_cpu = s->query_cpu->next_cpu; | ||||||
|             } else |             } else | ||||||
|  | @ -1826,16 +1846,15 @@ static int gdb_handle_packet(GDBState *s, const char *line_buf) | ||||||
|             break; |             break; | ||||||
|         } else if (strncmp(p,"ThreadExtraInfo,", 16) == 0) { |         } else if (strncmp(p,"ThreadExtraInfo,", 16) == 0) { | ||||||
|             thread = strtoull(p+16, (char **)&p, 16); |             thread = strtoull(p+16, (char **)&p, 16); | ||||||
|             for (env = first_cpu; env != NULL; env = env->next_cpu) |             env = find_cpu(thread); | ||||||
|                 if (env->cpu_index + 1 == thread) { |             if (env != NULL) { | ||||||
|                     cpu_synchronize_state(env, 0); |                 cpu_synchronize_state(env, 0); | ||||||
|                     len = snprintf((char *)mem_buf, sizeof(mem_buf), |                 len = snprintf((char *)mem_buf, sizeof(mem_buf), | ||||||
|                                    "CPU#%d [%s]", env->cpu_index, |                                "CPU#%d [%s]", env->cpu_index, | ||||||
|                                    env->halted ? "halted " : "running"); |                                env->halted ? "halted " : "running"); | ||||||
|                     memtohex(buf, mem_buf, len); |                 memtohex(buf, mem_buf, len); | ||||||
|                     put_packet(s, buf); |                 put_packet(s, buf); | ||||||
|                     break; |             } | ||||||
|                 } |  | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
| #ifdef CONFIG_USER_ONLY | #ifdef CONFIG_USER_ONLY | ||||||
|  | @ -1965,7 +1984,7 @@ static void gdb_vm_state_change(void *opaque, int running, int reason) | ||||||
|             } |             } | ||||||
|             snprintf(buf, sizeof(buf), |             snprintf(buf, sizeof(buf), | ||||||
|                      "T%02xthread:%02x;%swatch:" TARGET_FMT_lx ";", |                      "T%02xthread:%02x;%swatch:" TARGET_FMT_lx ";", | ||||||
|                      GDB_SIGNAL_TRAP, env->cpu_index+1, type, |                      GDB_SIGNAL_TRAP, gdb_id(env), type, | ||||||
|                      env->watchpoint_hit->vaddr); |                      env->watchpoint_hit->vaddr); | ||||||
|             put_packet(s, buf); |             put_packet(s, buf); | ||||||
|             env->watchpoint_hit = NULL; |             env->watchpoint_hit = NULL; | ||||||
|  | @ -1976,7 +1995,7 @@ static void gdb_vm_state_change(void *opaque, int running, int reason) | ||||||
|     } else { |     } else { | ||||||
|         ret = GDB_SIGNAL_INT; |         ret = GDB_SIGNAL_INT; | ||||||
|     } |     } | ||||||
|     snprintf(buf, sizeof(buf), "T%02xthread:%02x;", ret, env->cpu_index+1); |     snprintf(buf, sizeof(buf), "T%02xthread:%02x;", ret, gdb_id(env)); | ||||||
|     put_packet(s, buf); |     put_packet(s, buf); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -3202,6 +3202,7 @@ static void *clone_func(void *arg) | ||||||
|     env = info->env; |     env = info->env; | ||||||
|     thread_env = env; |     thread_env = env; | ||||||
|     info->tid = gettid(); |     info->tid = gettid(); | ||||||
|  |     env->host_tid = info->tid; | ||||||
|     if (info->child_tidptr) |     if (info->child_tidptr) | ||||||
|         put_user_u32(info->tid, info->child_tidptr); |         put_user_u32(info->tid, info->child_tidptr); | ||||||
|     if (info->parent_tidptr) |     if (info->parent_tidptr) | ||||||
|  | @ -3792,6 +3793,7 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1, | ||||||
|       /* FIXME: This probably breaks if a signal arrives.  We should probably
 |       /* FIXME: This probably breaks if a signal arrives.  We should probably
 | ||||||
|          be disabling signals.  */ |          be disabling signals.  */ | ||||||
|       if (first_cpu->next_cpu) { |       if (first_cpu->next_cpu) { | ||||||
|  |           TaskState *ts; | ||||||
|           CPUState **lastp; |           CPUState **lastp; | ||||||
|           CPUState *p; |           CPUState *p; | ||||||
| 
 | 
 | ||||||
|  | @ -3809,7 +3811,7 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1, | ||||||
|           /* Remove the CPU from the list.  */ |           /* Remove the CPU from the list.  */ | ||||||
|           *lastp = p->next_cpu; |           *lastp = p->next_cpu; | ||||||
|           cpu_list_unlock(); |           cpu_list_unlock(); | ||||||
|           TaskState *ts = ((CPUState *)cpu_env)->opaque; |           ts = ((CPUState *)cpu_env)->opaque; | ||||||
|           if (ts->child_tidptr) { |           if (ts->child_tidptr) { | ||||||
|               put_user_u32(0, ts->child_tidptr); |               put_user_u32(0, ts->child_tidptr); | ||||||
|               sys_futex(g2h(ts->child_tidptr), FUTEX_WAKE, INT_MAX, |               sys_futex(g2h(ts->child_tidptr), FUTEX_WAKE, INT_MAX, | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Nathan Froyd
						Nathan Froyd