This is an automated email from Gerrit. Peter Stuge (pe...@stuge.se) just uploaded a new patch set to Gerrit, which you can find at http://openocd.zylin.com/895
-- gerrit commit 269e81c491703ad46f4e97dc667df2839ae3d8bc Author: Peter Stuge <pe...@stuge.se> Date: Thu Oct 4 15:17:53 2012 +0200 rtos: Rewrite rtos_qsymbol() and fix false positive in auto-detection Matthias Blaicher submitted a patch at http://openocd.zylin.com/#/c/891/ to fix the false positive; when no RTOS matched the target OpenOCD would still use the last RTOS in the list. While reviewing the code affected by the patch a rewrite seemed appropriate, to make the code readable. Matthias has abandoned his change and this change also fixes the false positive. Change-Id: Ic3327ccd036da52ba0a7e21ef93018205e74149c Signed-off-by: Peter Stuge <pe...@stuge.se> diff --git a/src/rtos/linux.c b/src/rtos/linux.c index 15d5236..ba65558 100644 --- a/src/rtos/linux.c +++ b/src/rtos/linux.c @@ -1398,8 +1398,6 @@ static int linux_thread_packet(struct connection *connection, char *packet, if ((strstr(packet, "qSymbol"))) { if (rtos_qsymbol(connection, packet, packet_size) == 1) { - gdb_put_packet(connection, "OK", 2); - linux_compute_virt2phys(target, target->rtos-> symbols[INIT_TASK]. diff --git a/src/rtos/rtos.c b/src/rtos/rtos.c index 80f0938..cb208b3 100644 --- a/src/rtos/rtos.c +++ b/src/rtos/rtos.c @@ -133,90 +133,96 @@ int gdb_thread_packet(struct connection *connection, char *packet, int packet_si *found*/ return target->rtos->gdb_thread_packet(connection, packet, packet_size); } -/* return -1 if no rtos defined, 0 if rtos and symbol to be asked, 1 if all - * symbol have been asked*/ + +static char *next_symbol(struct rtos *os, char *cur_symbol, uint64_t cur_addr) +{ + symbol_table_elem_t *s; + + if (!os->symbols) + os->type->get_symbol_list_to_lookup(&os->symbols); + + if (!cur_symbol[0]) + return os->symbols[0].symbol_name; + + for (s = os->symbols; s->symbol_name; s++) + if (!strcmp(s->symbol_name, cur_symbol)) { + s->address = cur_addr; + s++; + return s->symbol_name; + } + + return NULL; +} + +/* rtos_qsymbol() processes and replies to all qSymbol packets from GDB. + * + * GDB sends a qSymbol:: packet (empty address, empty name) to notify + * that it can now answer qSymbol::hexcodedname queries, to look up symbols. + * + * If the qSymbol packet has no address that means GDB did not find the + * symbol, in which case auto-detect will move on to try the next RTOS. + * + * rtos_qsymbol() then calls the next_symbol() helper function, which + * iterates over symbol names for the current RTOS until it finds the + * symbol in the received GDB packet, and then returns the next entry + * in the list of symbols. + * + * If GDB replied about the last symbol for the RTOS and the RTOS was + * specified explicitly, then no further symbol lookup is done. When + * auto-detecting, the RTOS driver _detect() function must return success. + * + * rtos_qsymbol() returns 1 if an RTOS has been detected, or 0 otherwise. + */ int rtos_qsymbol(struct connection *connection, char *packet, int packet_size) { + int rtos_detected = 0; + uint64_t addr; + size_t reply_len; + char reply[GDB_BUFFER_SIZE], cur_sym[GDB_BUFFER_SIZE / 2] = "", *next_sym; struct target *target = get_target_from_connection(connection); - if (target->rtos != NULL) { - int next_symbol_num = -1; - if (target->rtos->symbols == NULL) - target->rtos->type->get_symbol_list_to_lookup(&target->rtos->symbols); - if (0 == strcmp("qSymbol::", packet)) - /* first query - */ - next_symbol_num = 0; - else { - int64_t value = 0; - char *hex_name_str = malloc(strlen(packet)); - char *name_str; - int symbol_num; - - char *found = strstr(packet, "qSymbol::"); - if (0 == found) - sscanf(packet, "qSymbol:%" SCNx64 ":%s", &value, hex_name_str); - else - /* No value returned by GDB - symbol was not found*/ - sscanf(packet, "qSymbol::%s", hex_name_str); - name_str = (char *) malloc(1 + strlen(hex_name_str) / 2); - - hex_to_str(name_str, hex_name_str); - symbol_num = 0; - while ((target->rtos->symbols[symbol_num].symbol_name != NULL) && - (0 != strcmp(target->rtos->symbols[symbol_num].symbol_name, name_str))) - symbol_num++; - - if (target->rtos->symbols[symbol_num].symbol_name == NULL) { - LOG_OUTPUT("ERROR: unknown symbol\r\n"); - gdb_put_packet(connection, "OK", 2); - free(hex_name_str); - free(name_str); - return ERROR_OK; - } + struct rtos *os = target->rtos; - target->rtos->symbols[symbol_num].address = value; + reply_len = sprintf(reply, "OK"); - next_symbol_num = symbol_num+1; - free(hex_name_str); - free(name_str); - } + if (sscanf(packet, "qSymbol:%" SCNx64 ":", &addr)) + hex_to_str(cur_sym, strchr(packet + 8, ':') + 1); + else if (target->rtos_auto_detect && !rtos_try_next(target)) + goto done; - int symbols_done = 0; - if (target->rtos->symbols[next_symbol_num].symbol_name == NULL) { - if ((target->rtos_auto_detect == false) || - (1 == target->rtos->type->detect_rtos(target))) { - /* Found correct RTOS or not autodetecting */ - if (target->rtos_auto_detect == true) - LOG_OUTPUT("Auto-detected RTOS: %s\r\n", - target->rtos->type->name); - symbols_done = 1; - } else { - /* Auto detecting RTOS and currently not found */ - if (1 != rtos_try_next(target)) - /* No more RTOS's to try */ - symbols_done = 1; - else { - next_symbol_num = 0; - target->rtos->type->get_symbol_list_to_lookup( - &target->rtos->symbols); - } - } + next_sym = next_symbol(os, cur_sym, addr); + if (!next_sym) { + if (!target->rtos_auto_detect) { + rtos_detected = 1; + goto done; } - if (symbols_done == 1) - return symbols_done; - else { - char *symname = target->rtos->symbols[next_symbol_num].symbol_name; - char qsymstr[] = "qSymbol:"; - char *opstring = (char *)malloc(sizeof(qsymstr)+strlen(symname)*2+1); - char *posptr = opstring; - posptr += sprintf(posptr, "%s", qsymstr); - str_to_hex(posptr, symname); - gdb_put_packet(connection, opstring, strlen(opstring)); - free(opstring); - return symbols_done; + + if (os->type->detect_rtos(target)) { + LOG_OUTPUT("Auto-detected RTOS: %s\r\n", os->type->name); + rtos_detected = 1; + goto done; } + + if (!rtos_try_next(target)) + goto done; + + os->type->get_symbol_list_to_lookup(&os->symbols); + + next_sym = os->symbols[0].symbol_name; + if (!next_sym) + goto done; + } + + if (8 + (strlen(next_sym) * 2) + 1 > sizeof(reply)) { + LOG_OUTPUT("ERROR: RTOS symbol '%s' name is too long for GDB!", next_sym); + goto done; } - gdb_put_packet(connection, "OK", 2); - return -1; + + reply_len = sprintf(reply, "qSymbol:"); + reply_len += str_to_hex(reply + reply_len, next_sym); + +done: + gdb_put_packet(connection, reply, reply_len); + return rtos_detected; } int rtos_thread_packet(struct connection *connection, char *packet, int packet_size) @@ -290,8 +296,6 @@ int rtos_thread_packet(struct connection *connection, char *packet, int packet_s target->rtos_auto_detect = false; target->rtos->type->create(target); target->rtos->type->update_threads(target->rtos); - /* No more symbols needed */ - gdb_put_packet(connection, "OK", 2); } return ERROR_OK; } else if (strstr(packet, "qfThreadInfo")) { -- ------------------------------------------------------------------------------ Don't let slow site performance ruin your business. Deploy New Relic APM Deploy New Relic app performance management and know exactly what is happening inside your Ruby, Python, PHP, Java, and .NET app Try New Relic at no cost today and get our sweet Data Nerd shirt too! http://p.sf.net/sfu/newrelic-dev2dev _______________________________________________ OpenOCD-devel mailing list OpenOCD-devel@lists.sourceforge.net https://lists.sourceforge.net/lists/listinfo/openocd-devel