- Introduce a significant design decision.
[wine/wine-kai.git] / debugger / hash.c
blobd987b8dcee004b0e38d332557342a6555106bb8f
1 /*
2 * File hash.c - generate hash tables for Wine debugger symbols
4 * Copyright (C) 1993, Eric Youngdale.
5 */
8 #include "config.h"
9 #include <stdlib.h>
10 #include <stdio.h>
11 #include <string.h>
12 #include <limits.h>
13 #include <sys/types.h>
14 #include "debugger.h"
16 #define NR_NAME_HASH 16384
17 #ifndef PATH_MAX
18 #define PATH_MAX _MAX_PATH
19 #endif
21 #ifdef __i386__
22 static char * reg_name[] =
24 "eax", "ecx", "edx", "ebx", "esp", "ebp", "esi", "edi"
27 static unsigned reg_ofs[] =
29 FIELD_OFFSET(CONTEXT, Eax), FIELD_OFFSET(CONTEXT, Ecx),
30 FIELD_OFFSET(CONTEXT, Edx), FIELD_OFFSET(CONTEXT, Ebx),
31 FIELD_OFFSET(CONTEXT, Esp), FIELD_OFFSET(CONTEXT, Ebp),
32 FIELD_OFFSET(CONTEXT, Esi), FIELD_OFFSET(CONTEXT, Edi)
34 #else
35 static char * reg_name[] = { NULL }; /* FIXME */
36 static unsigned reg_ofs[] = { 0 };
37 #endif
40 struct name_hash
42 struct name_hash * next; /* Used to look up within name hash */
43 char * name;
44 char * sourcefile;
46 int n_locals;
47 int locals_alloc;
48 WineLocals * local_vars;
50 int n_lines;
51 int lines_alloc;
52 WineLineNo * linetab;
54 DBG_VALUE value;
55 unsigned short flags;
56 unsigned short breakpoint_offset;
57 unsigned int symbol_size;
61 static BOOL DEBUG_GetStackSymbolValue( const char * name, DBG_VALUE *value );
62 static int sortlist_valid = FALSE;
64 static int sorttab_nsym;
65 static struct name_hash ** addr_sorttab = NULL;
67 static struct name_hash * name_hash_table[NR_NAME_HASH];
69 static unsigned int name_hash( const char * name )
71 unsigned int hash = 0;
72 unsigned int tmp;
73 const char * p;
75 p = name;
77 while (*p)
79 hash = (hash << 4) + *p++;
81 if( (tmp = (hash & 0xf0000000)) )
83 hash ^= tmp >> 24;
85 hash &= ~tmp;
87 return hash % NR_NAME_HASH;
90 int
91 DEBUG_cmp_sym(const void * p1, const void * p2)
93 struct name_hash ** name1 = (struct name_hash **) p1;
94 struct name_hash ** name2 = (struct name_hash **) p2;
96 if( ((*name1)->flags & SYM_INVALID) != 0 )
98 return -1;
101 if( ((*name2)->flags & SYM_INVALID) != 0 )
103 return 1;
106 if( (*name1)->value.addr.seg > (*name2)->value.addr.seg )
108 return 1;
111 if( (*name1)->value.addr.seg < (*name2)->value.addr.seg )
113 return -1;
116 if( (*name1)->value.addr.off > (*name2)->value.addr.off )
118 return 1;
121 if( (*name1)->value.addr.off < (*name2)->value.addr.off )
123 return -1;
126 return 0;
129 /***********************************************************************
130 * DEBUG_ResortSymbols
132 * Rebuild sorted list of symbols.
134 static
135 void
136 DEBUG_ResortSymbols(void)
138 struct name_hash *nh;
139 int nsym = 0;
140 int i;
142 for(i=0; i<NR_NAME_HASH; i++)
144 for (nh = name_hash_table[i]; nh; nh = nh->next)
146 if( (nh->flags & SYM_INVALID) == 0 )
147 nsym++;
148 else
149 DEBUG_Printf( DBG_CHN_MESG, "Symbol %s is invalid\n", nh->name );
153 sorttab_nsym = nsym;
154 if( nsym == 0 )
156 return;
159 addr_sorttab = (struct name_hash **) DBG_realloc(addr_sorttab,
160 nsym * sizeof(struct name_hash *));
162 nsym = 0;
163 for(i=0; i<NR_NAME_HASH; i++)
165 for (nh = name_hash_table[i]; nh; nh = nh->next)
167 if( (nh->flags & SYM_INVALID) == 0 )
168 addr_sorttab[nsym++] = nh;
172 qsort(addr_sorttab, nsym,
173 sizeof(struct name_hash *), DEBUG_cmp_sym);
174 sortlist_valid = TRUE;
178 /***********************************************************************
179 * DEBUG_AddSymbol
181 * Add a symbol to the table.
183 struct name_hash *
184 DEBUG_AddSymbol( const char * name, const DBG_VALUE *value,
185 const char * source, int flags)
187 struct name_hash * new;
188 struct name_hash *nh;
189 static char prev_source[PATH_MAX] = {'\0', };
190 static char * prev_duped_source = NULL;
191 int hash;
193 assert(value->cookie == DV_TARGET || value->cookie == DV_HOST);
195 hash = name_hash(name);
196 for (nh = name_hash_table[hash]; nh; nh = nh->next)
198 if( ((nh->flags & SYM_INVALID) != 0) && strcmp(name, nh->name) == 0 )
200 #if 0
201 DEBUG_Printf(DBG_CHN_MESG, "Changing address for symbol %s (%08lx:%08lx => %08lx:%08lx)\n",
202 name, nh->value.addr.seg, nh->value.addr.off, value->addr.seg, value->addr.off);
203 #endif
204 nh->value.addr = value->addr;
205 if( nh->value.type == NULL && value->type != NULL )
207 nh->value.type = value->type;
208 nh->value.cookie = value->cookie;
210 /* it may happen that the same symbol is defined in several compilation
211 * units, but the linker decides to merge it into a single instance.
212 * in that case, we don't clear the invalid flag for all the compilation
213 * units (N_GSYM), and wait to get the symbol from the symtab
215 if ((flags & SYM_INVALID) == 0)
216 nh->flags &= ~SYM_INVALID;
218 return nh;
220 if (nh->value.addr.seg == value->addr.seg &&
221 nh->value.addr.off == value->addr.off &&
222 strcmp(name, nh->name) == 0 )
224 return nh;
228 #if 0
229 DEBUG_Printf(DBG_CHN_TRACE, "adding symbol (%s) from file '%s' at 0x%04lx:%08lx\n",
230 name, source, value->addr.seg, value->addr.off);
231 #endif
234 * First see if we already have an entry for this symbol. If so
235 * return it, so we don't end up with duplicates.
238 new = (struct name_hash *) DBG_alloc(sizeof(struct name_hash));
239 new->value = *value;
240 new->name = DBG_strdup(name);
242 if( source != NULL )
245 * This is an enhancement to reduce memory consumption. The idea
246 * is that we duplicate a given string only once. This is a big
247 * win if there are lots of symbols defined in a given source file.
249 if( strcmp(source, prev_source) == 0 )
251 new->sourcefile = prev_duped_source;
253 else
255 strcpy(prev_source, source);
256 prev_duped_source = new->sourcefile = DBG_strdup(source);
259 else
261 new->sourcefile = NULL;
264 new->n_lines = 0;
265 new->lines_alloc = 0;
266 new->linetab = NULL;
268 new->n_locals = 0;
269 new->locals_alloc = 0;
270 new->local_vars = NULL;
272 new->flags = flags;
273 new->next = NULL;
275 /* Now insert into the hash table */
276 new->next = name_hash_table[hash];
277 name_hash_table[hash] = new;
280 * Check some heuristics based upon the file name to see whether
281 * we want to step through this guy or not. These are machine generated
282 * assembly files that are used to translate between the MS way of
283 * calling things and the GCC way of calling things. In general we
284 * always want to step through.
286 if ( source != NULL ) {
287 int len = strlen(source);
289 if (len > 2 && source[len-2] == '.' && source[len-1] == 's') {
290 char* c = strrchr(source - 2, '/');
291 if (c != NULL) {
292 if (strcmp(c + 1, "asmrelay.s") == 0)
293 new->flags |= SYM_TRAMPOLINE;
298 sortlist_valid = FALSE;
299 return new;
302 BOOL DEBUG_Normalize(struct name_hash * nh )
306 * We aren't adding any more locals or linenumbers to this function.
307 * Free any spare memory that we might have allocated.
309 if( nh == NULL )
311 return TRUE;
314 if( nh->n_locals != nh->locals_alloc )
316 nh->locals_alloc = nh->n_locals;
317 nh->local_vars = DBG_realloc(nh->local_vars,
318 nh->locals_alloc * sizeof(WineLocals));
321 if( nh->n_lines != nh->lines_alloc )
323 nh->lines_alloc = nh->n_lines;
324 nh->linetab = DBG_realloc(nh->linetab,
325 nh->lines_alloc * sizeof(WineLineNo));
328 return TRUE;
331 /***********************************************************************
332 * DEBUG_GetSymbolValue
334 * Get the address of a named symbol.
336 static int DEBUG_GSV_Helper(const char* name, const int lineno,
337 DBG_VALUE* value, int num, int bp_flag)
339 struct name_hash* nh;
340 int i = 0;
341 DBG_ADDR addr;
343 for (nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
345 if ((nh->flags & SYM_INVALID) != 0) continue;
346 if (!strcmp(nh->name, name) && DEBUG_GetLineNumberAddr( nh, lineno, &addr, bp_flag ))
348 if (i >= num) return num + 1;
349 value[i].addr = addr;
350 value[i].type = nh->value.type;
351 value[i].cookie = nh->value.cookie;
352 i++;
355 return i;
358 BOOL DEBUG_GetSymbolValue( const char * name, const int lineno,
359 DBG_VALUE *rtn, int bp_flag )
361 #define NUMDBGV 10
362 /* FIXME: NUMDBGV should be made variable */
363 DBG_VALUE value[NUMDBGV];
364 DBG_VALUE vtmp;
365 int num, i, local = -1;
367 num = DEBUG_GSV_Helper(name, lineno, value, NUMDBGV, bp_flag);
368 if (!num && (name[0] != '_'))
370 char buffer[512];
372 assert(strlen(name) < sizeof(buffer) - 2); /* one for '_', one for '\0' */
373 buffer[0] = '_';
374 strcpy(buffer + 1, name);
375 num = DEBUG_GSV_Helper(buffer, lineno, value, NUMDBGV, bp_flag);
378 /* now get the local symbols if any */
379 if (DEBUG_GetStackSymbolValue(name, &vtmp) && num < NUMDBGV)
381 value[num] = vtmp;
382 local = num;
383 num++;
386 if (num == 0) {
387 return FALSE;
388 } else if (!DEBUG_interactiveP || num == 1) {
389 i = 0;
390 } else {
391 char buffer[256];
393 if (num == NUMDBGV+1) {
394 DEBUG_Printf(DBG_CHN_MESG, "Too many addresses for symbol '%s', limiting the first %d\n", name, NUMDBGV);
395 num = NUMDBGV;
397 DEBUG_Printf(DBG_CHN_MESG, "Many symbols with name '%s', choose the one you want (<cr> to abort):\n", name);
398 for (i = 0; i < num; i++) {
399 DEBUG_Printf(DBG_CHN_MESG, "[%d]: ", i + 1);
400 if (i == local) {
401 struct name_hash*func;
402 unsigned int ebp;
403 unsigned int eip;
405 if (DEBUG_GetCurrentFrame(&func, &eip, &ebp))
406 DEBUG_Printf(DBG_CHN_MESG, "local variable of %s in %s\n", func->name, func->sourcefile);
407 else
408 DEBUG_Printf(DBG_CHN_MESG, "local variable\n");
409 } else {
410 DEBUG_PrintAddress( &value[i].addr, DEBUG_GetSelectorType(value[i].addr.seg), TRUE);
411 DEBUG_Printf(DBG_CHN_MESG, "\n");
414 do {
415 i = 0;
416 if (DEBUG_ReadLine("=> ", buffer, sizeof(buffer), FALSE))
418 i = atoi(buffer);
419 if (i < 1 || i > num)
420 DEBUG_Printf(DBG_CHN_MESG, "Invalid choice %d\n", i);
422 } while (i < 1 || i > num);
424 /* The array is 0-based, but the choices are 1..n, so we have to subtract one before returning. */
425 i--;
427 *rtn = value[i];
428 return TRUE;
431 /***********************************************************************
432 * DEBUG_GetLineNumberAddr
434 * Get the address of a named symbol.
436 BOOL DEBUG_GetLineNumberAddr( const struct name_hash * nh, const int lineno,
437 DBG_ADDR *addr, int bp_flag )
439 int i;
441 if( lineno == -1 )
443 *addr = nh->value.addr;
444 if( bp_flag )
446 addr->off += nh->breakpoint_offset;
449 else
452 * Search for the specific line number. If we don't find it,
453 * then return FALSE.
455 if( nh->linetab == NULL )
457 return FALSE;
460 for(i=0; i < nh->n_lines; i++ )
462 if( nh->linetab[i].line_number == lineno )
464 *addr = nh->linetab[i].pc_offset;
465 return TRUE;
470 * This specific line number not found.
472 return FALSE;
475 return TRUE;
479 /***********************************************************************
480 * DEBUG_SetSymbolValue
482 * Set the address of a named symbol.
484 BOOL DEBUG_SetSymbolValue( const char * name, const DBG_VALUE *value )
486 char buffer[256];
487 struct name_hash *nh;
489 assert(value->cookie == DV_TARGET || value->cookie == DV_HOST);
491 for(nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
492 if (!strcmp(nh->name, name)) break;
494 if (!nh && (name[0] != '_'))
496 buffer[0] = '_';
497 strcpy(buffer+1, name);
498 for(nh = name_hash_table[name_hash(buffer)]; nh; nh = nh->next)
499 if (!strcmp(nh->name, buffer)) break;
502 if (!nh) return FALSE;
503 nh->value = *value;
504 nh->flags &= ~SYM_INVALID;
506 #ifdef __i386__
507 DEBUG_FixAddress( &nh->value.addr, DEBUG_context.SegDs );
508 #endif
510 return TRUE;
514 /***********************************************************************
515 * DEBUG_FindNearestSymbol
517 * Find the symbol nearest to a given address.
518 * If ebp is specified as non-zero, it means we should dump the argument
519 * list into the string we return as well.
521 const char * DEBUG_FindNearestSymbol( const DBG_ADDR *addr, int flag,
522 struct name_hash ** rtn,
523 unsigned int ebp,
524 struct list_id * source)
526 static char name_buffer[MAX_PATH + 256];
527 static char arglist[1024];
528 static char argtmp[256];
529 struct name_hash * nearest = NULL;
530 int mid, high, low;
531 unsigned int * ptr;
532 int lineno;
533 char * lineinfo, *sourcefile;
534 int i;
535 char linebuff[16];
536 unsigned val;
537 DBG_MODULE* module;
538 char modbuf[256];
540 if( rtn != NULL )
542 *rtn = NULL;
545 if( source != NULL )
547 source->sourcefile = NULL;
548 source->line = -1;
551 if( sortlist_valid == FALSE )
553 DEBUG_ResortSymbols();
556 if( sortlist_valid == FALSE )
558 return NULL;
562 * FIXME - use the binary search that we added to
563 * the function DEBUG_CheckLinenoStatus. Better yet, we should
564 * probably keep some notion of the current function so we don't
565 * have to search every time.
568 * Binary search to find closest symbol.
570 low = 0;
571 high = sorttab_nsym;
572 if( addr_sorttab[0]->value.addr.seg > addr->seg
573 || ( addr_sorttab[0]->value.addr.seg == addr->seg
574 && addr_sorttab[0]->value.addr.off > addr->off) )
576 nearest = NULL;
578 else if( addr_sorttab[high - 1]->value.addr.seg < addr->seg
579 || ( addr_sorttab[high - 1]->value.addr.seg == addr->seg
580 && addr_sorttab[high - 1]->value.addr.off < addr->off) )
582 nearest = addr_sorttab[high - 1];
584 else
586 while(1==1)
588 mid = (high + low)/2;
589 if( mid == low )
592 * See if there are any other entries that might also
593 * have the same address, and would also have a line
594 * number table.
596 if( mid > 0 && addr_sorttab[mid]->linetab == NULL )
598 if( (addr_sorttab[mid - 1]->value.addr.seg ==
599 addr_sorttab[mid]->value.addr.seg)
600 && (addr_sorttab[mid - 1]->value.addr.off ==
601 addr_sorttab[mid]->value.addr.off)
602 && (addr_sorttab[mid - 1]->linetab != NULL) )
604 mid--;
608 if( (mid < sorttab_nsym - 1)
609 && (addr_sorttab[mid]->linetab == NULL) )
611 if( (addr_sorttab[mid + 1]->value.addr.seg ==
612 addr_sorttab[mid]->value.addr.seg)
613 && (addr_sorttab[mid + 1]->value.addr.off ==
614 addr_sorttab[mid]->value.addr.off)
615 && (addr_sorttab[mid + 1]->linetab != NULL) )
617 mid++;
620 nearest = addr_sorttab[mid];
621 #if 0
622 DEBUG_Printf(DBG_CHN_MESG, "Found %x:%x when looking for %x:%x %x %s\n",
623 addr_sorttab[mid ]->value.addr.seg,
624 addr_sorttab[mid ]->value.addr.off,
625 addr->seg, addr->off,
626 addr_sorttab[mid ]->linetab,
627 addr_sorttab[mid ]->name);
628 #endif
629 break;
631 if( (addr_sorttab[mid]->value.addr.seg < addr->seg)
632 || ( addr_sorttab[mid]->value.addr.seg == addr->seg
633 && addr_sorttab[mid]->value.addr.off <= addr->off) )
635 low = mid;
637 else
639 high = mid;
644 if (!nearest) return NULL;
646 if( rtn != NULL )
648 *rtn = nearest;
652 * Fill in the relevant bits to the structure so that we can
653 * locate the source and line for this bit of code.
655 if( source != NULL )
657 source->sourcefile = nearest->sourcefile;
658 if( nearest->linetab == NULL )
660 source->line = -1;
662 else
664 source->line = nearest->linetab[0].line_number;
668 lineinfo = "";
669 lineno = -1;
672 * Prepare to display the argument list. If ebp is specified, it is
673 * the framepointer for the function in question. If not specified,
674 * we don't want the arglist.
676 memset(arglist, '\0', sizeof(arglist));
677 if( ebp != 0 )
679 for(i=0; i < nearest->n_locals; i++ )
682 * If this is a register (offset == 0) or a local
683 * variable, we don't want to know about it.
685 if( nearest->local_vars[i].offset <= 0 )
687 continue;
690 ptr = (unsigned int *) (ebp + nearest->local_vars[i].offset);
691 if( arglist[0] == '\0' )
693 arglist[0] = '(';
695 else
697 strcat(arglist, ", ");
699 DEBUG_READ_MEM_VERBOSE(ptr, &val, sizeof(val));
700 sprintf(argtmp, "%s=0x%x", nearest->local_vars[i].name, val);
702 strcat(arglist, argtmp);
704 if( arglist[0] == '(' )
706 strcat(arglist, ")");
710 module = DEBUG_FindModuleByAddr((void*)DEBUG_ToLinear(addr), DMT_UNKNOWN);
711 if (module) {
712 char* ptr = strrchr(module->module_name, '/');
714 if (!ptr++) ptr = module->module_name;
715 sprintf( modbuf, " in %s", ptr);
717 else
718 modbuf[0] = '\0';
720 if( (nearest->sourcefile != NULL) && (flag == TRUE)
721 && (addr->off - nearest->value.addr.off < 0x100000) )
725 * Try and find the nearest line number to the current offset.
727 if( nearest->linetab != NULL )
729 low = 0;
730 high = nearest->n_lines;
731 while ((high - low) > 1)
733 mid = (high + low) / 2;
734 if (addr->off < nearest->linetab[mid].pc_offset.off)
735 high = mid;
736 else
737 low = mid;
739 lineno = nearest->linetab[low].line_number;
742 if( lineno != -1 )
744 sprintf(linebuff, ":%d", lineno);
745 lineinfo = linebuff;
746 if( source != NULL )
748 source->line = lineno;
752 /* Remove the path from the file name */
753 sourcefile = strrchr( nearest->sourcefile, '/' );
754 if (!sourcefile) sourcefile = nearest->sourcefile;
755 else sourcefile++;
757 if (addr->off == nearest->value.addr.off)
758 sprintf( name_buffer, "%s%s [%s%s]%s", nearest->name,
759 arglist, sourcefile, lineinfo, modbuf);
760 else
761 sprintf( name_buffer, "%s+0x%lx%s [%s%s]%s", nearest->name,
762 addr->off - nearest->value.addr.off,
763 arglist, sourcefile, lineinfo, modbuf );
765 else
767 if (addr->off == nearest->value.addr.off)
768 sprintf( name_buffer, "%s%s%s", nearest->name, arglist, modbuf);
769 else {
770 if (addr->seg && (nearest->value.addr.seg!=addr->seg))
771 return NULL;
772 else
773 sprintf( name_buffer, "%s+0x%lx%s%s", nearest->name,
774 addr->off - nearest->value.addr.off, arglist, modbuf);
777 return name_buffer;
781 /***********************************************************************
782 * DEBUG_ReadSymbolTable
784 * Read a symbol file into the hash table.
786 void DEBUG_ReadSymbolTable( const char* filename )
788 FILE * symbolfile;
789 DBG_VALUE value;
790 char type;
791 char * cpnt;
792 char buffer[256];
793 char name[256];
795 if (!(symbolfile = fopen(filename, "r")))
797 DEBUG_Printf( DBG_CHN_WARN, "Unable to open symbol table %s\n", filename );
798 return;
801 DEBUG_Printf( DBG_CHN_MESG, "Reading symbols from file %s\n", filename );
803 value.type = NULL;
804 value.addr.seg = 0;
805 value.addr.off = 0;
806 value.cookie = DV_TARGET;
808 while (1)
810 fgets( buffer, sizeof(buffer), symbolfile );
811 if (feof(symbolfile)) break;
813 /* Strip any text after a # sign (i.e. comments) */
814 cpnt = buffer;
815 while (*cpnt)
816 if(*cpnt++ == '#') { *cpnt = 0; break; }
818 /* Quietly ignore any lines that have just whitespace */
819 cpnt = buffer;
820 while(*cpnt)
822 if(*cpnt != ' ' && *cpnt != '\t') break;
823 cpnt++;
825 if (!(*cpnt) || *cpnt == '\n') continue;
827 if (sscanf(buffer, "%lx %c %s", &value.addr.off, &type, name) == 3)
828 DEBUG_AddSymbol( name, &value, NULL, SYM_WINE );
830 fclose(symbolfile);
834 void
835 DEBUG_AddLineNumber( struct name_hash * func, int line_num,
836 unsigned long offset )
838 if( func == NULL )
840 return;
843 if( func->n_lines + 1 >= func->lines_alloc )
845 func->lines_alloc += 64;
846 func->linetab = DBG_realloc(func->linetab,
847 func->lines_alloc * sizeof(WineLineNo));
850 func->linetab[func->n_lines].line_number = line_num;
851 func->linetab[func->n_lines].pc_offset.seg = func->value.addr.seg;
852 func->linetab[func->n_lines].pc_offset.off = func->value.addr.off + offset;
853 func->n_lines++;
857 struct wine_locals *
858 DEBUG_AddLocal( struct name_hash * func, int regno,
859 int offset,
860 int pc_start,
861 int pc_end,
862 char * name)
864 if( func == NULL )
866 return NULL;
869 if( func->n_locals + 1 >= func->locals_alloc )
871 func->locals_alloc += 32;
872 func->local_vars = DBG_realloc(func->local_vars,
873 func->locals_alloc * sizeof(WineLocals));
876 func->local_vars[func->n_locals].regno = regno;
877 func->local_vars[func->n_locals].offset = offset;
878 func->local_vars[func->n_locals].pc_start = pc_start;
879 func->local_vars[func->n_locals].pc_end = pc_end;
880 func->local_vars[func->n_locals].name = DBG_strdup(name);
881 func->local_vars[func->n_locals].type = NULL;
882 func->n_locals++;
884 return &func->local_vars[func->n_locals - 1];
887 void
888 DEBUG_DumpHashInfo(void)
890 int i;
891 int depth;
892 struct name_hash *nh;
895 * Utility function to dump stats about the hash table.
897 for(i=0; i<NR_NAME_HASH; i++)
899 depth = 0;
900 for (nh = name_hash_table[i]; nh; nh = nh->next)
902 depth++;
904 DEBUG_Printf(DBG_CHN_MESG, "Bucket %d: %d\n", i, depth);
908 /***********************************************************************
909 * DEBUG_CheckLinenoStatus
911 * Find the symbol nearest to a given address.
912 * If ebp is specified as non-zero, it means we should dump the argument
913 * list into the string we return as well.
915 int DEBUG_CheckLinenoStatus( const DBG_ADDR *addr)
917 struct name_hash * nearest = NULL;
918 int mid, high, low;
920 if( sortlist_valid == FALSE )
922 DEBUG_ResortSymbols();
926 * Binary search to find closest symbol.
928 low = 0;
929 high = sorttab_nsym;
930 if( addr_sorttab[0]->value.addr.seg > addr->seg
931 || ( addr_sorttab[0]->value.addr.seg == addr->seg
932 && addr_sorttab[0]->value.addr.off > addr->off) )
934 nearest = NULL;
936 else if( addr_sorttab[high - 1]->value.addr.seg < addr->seg
937 || ( addr_sorttab[high - 1]->value.addr.seg == addr->seg
938 && addr_sorttab[high - 1]->value.addr.off < addr->off) )
940 nearest = addr_sorttab[high - 1];
942 else
944 while(1==1)
946 mid = (high + low)/2;
947 if( mid == low )
950 * See if there are any other entries that might also
951 * have the same address, and would also have a line
952 * number table.
954 if( mid > 0 && addr_sorttab[mid]->linetab == NULL )
956 if( (addr_sorttab[mid - 1]->value.addr.seg ==
957 addr_sorttab[mid]->value.addr.seg)
958 && (addr_sorttab[mid - 1]->value.addr.off ==
959 addr_sorttab[mid]->value.addr.off)
960 && (addr_sorttab[mid - 1]->linetab != NULL) )
962 mid--;
966 if( (mid < sorttab_nsym - 1)
967 && (addr_sorttab[mid]->linetab == NULL) )
969 if( (addr_sorttab[mid + 1]->value.addr.seg ==
970 addr_sorttab[mid]->value.addr.seg)
971 && (addr_sorttab[mid + 1]->value.addr.off ==
972 addr_sorttab[mid]->value.addr.off)
973 && (addr_sorttab[mid + 1]->linetab != NULL) )
975 mid++;
978 nearest = addr_sorttab[mid];
979 #if 0
980 DEBUG_Printf(DBG_CHN_MESG, "Found %x:%x when looking for %x:%x %x %s\n",
981 addr_sorttab[mid ]->value.addr.seg,
982 addr_sorttab[mid ]->value.addr.off,
983 addr->seg, addr->off,
984 addr_sorttab[mid ]->linetab,
985 addr_sorttab[mid ]->name);
986 #endif
987 break;
989 if( (addr_sorttab[mid]->value.addr.seg < addr->seg)
990 || ( addr_sorttab[mid]->value.addr.seg == addr->seg
991 && addr_sorttab[mid]->value.addr.off <= addr->off) )
993 low = mid;
995 else
997 high = mid;
1002 if (!nearest) return FUNC_HAS_NO_LINES;
1004 if( nearest->flags & SYM_STEP_THROUGH )
1007 * This will cause us to keep single stepping until
1008 * we get to the other side somewhere.
1010 return NOT_ON_LINENUMBER;
1013 if( (nearest->flags & SYM_TRAMPOLINE) )
1016 * This will cause us to keep single stepping until
1017 * we get to the other side somewhere.
1019 return FUNC_IS_TRAMPOLINE;
1022 if( nearest->linetab == NULL )
1024 return FUNC_HAS_NO_LINES;
1029 * We never want to stop on the first instruction of a function
1030 * even if it has it's own linenumber. Let the thing keep running
1031 * until it gets past the function prologue. We only do this if there
1032 * is more than one line number for the function, of course.
1034 if( nearest->value.addr.off == addr->off && nearest->n_lines > 1 )
1036 return NOT_ON_LINENUMBER;
1039 if( (nearest->sourcefile != NULL)
1040 && (addr->off - nearest->value.addr.off < 0x100000) )
1042 low = 0;
1043 high = nearest->n_lines;
1044 while ((high - low) > 1)
1046 mid = (high + low) / 2;
1047 if (addr->off < nearest->linetab[mid].pc_offset.off) high = mid;
1048 else low = mid;
1050 if (addr->off == nearest->linetab[low].pc_offset.off)
1051 return AT_LINENUMBER;
1052 else
1053 return NOT_ON_LINENUMBER;
1056 return FUNC_HAS_NO_LINES;
1059 /***********************************************************************
1060 * DEBUG_GetFuncInfo
1062 * Find the symbol nearest to a given address.
1063 * Returns sourcefile name and line number in a format that the listing
1064 * handler can deal with.
1066 void
1067 DEBUG_GetFuncInfo( struct list_id * ret, const char * filename,
1068 const char * name)
1070 char buffer[256];
1071 char * pnt;
1072 struct name_hash *nh;
1074 for(nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
1076 if( filename != NULL )
1079 if( nh->sourcefile == NULL )
1081 continue;
1084 pnt = strrchr(nh->sourcefile, '/');
1085 if( strcmp(nh->sourcefile, filename) != 0
1086 && (pnt == NULL || strcmp(pnt + 1, filename) != 0) )
1088 continue;
1091 if (!strcmp(nh->name, name)) break;
1094 if (!nh && (name[0] != '_'))
1096 buffer[0] = '_';
1097 strcpy(buffer+1, name);
1098 for(nh = name_hash_table[name_hash(buffer)]; nh; nh = nh->next)
1100 if( filename != NULL )
1102 if( nh->sourcefile == NULL )
1104 continue;
1107 pnt = strrchr(nh->sourcefile, '/');
1108 if( strcmp(nh->sourcefile, filename) != 0
1109 && (pnt == NULL || strcmp(pnt + 1, filename) != 0) )
1111 continue;
1114 if (!strcmp(nh->name, buffer)) break;
1118 if( !nh )
1120 if( filename != NULL )
1122 DEBUG_Printf(DBG_CHN_MESG, "No such function %s in %s\n", name, filename);
1124 else
1126 DEBUG_Printf(DBG_CHN_MESG, "No such function %s\n", name);
1128 ret->sourcefile = NULL;
1129 ret->line = -1;
1130 return;
1133 ret->sourcefile = nh->sourcefile;
1136 * Search for the specific line number. If we don't find it,
1137 * then return FALSE.
1139 if( nh->linetab == NULL )
1141 ret->line = -1;
1143 else
1145 ret->line = nh->linetab[0].line_number;
1149 /***********************************************************************
1150 * DEBUG_GetStackSymbolValue
1152 * Get the address of a named symbol from the current stack frame.
1154 static
1155 BOOL DEBUG_GetStackSymbolValue( const char * name, DBG_VALUE *value )
1157 struct name_hash * curr_func;
1158 unsigned int ebp;
1159 unsigned int eip;
1160 int i;
1162 if( DEBUG_GetCurrentFrame(&curr_func, &eip, &ebp) == FALSE )
1164 return FALSE;
1167 for(i=0; i < curr_func->n_locals; i++ )
1170 * Test the range of validity of the local variable. This
1171 * comes up with RBRAC/LBRAC stabs in particular.
1173 if( (curr_func->local_vars[i].pc_start != 0)
1174 && ((eip - curr_func->value.addr.off)
1175 < curr_func->local_vars[i].pc_start) )
1177 continue;
1180 if( (curr_func->local_vars[i].pc_end != 0)
1181 && ((eip - curr_func->value.addr.off)
1182 > curr_func->local_vars[i].pc_end) )
1184 continue;
1187 if( strcmp(name, curr_func->local_vars[i].name) == 0 )
1190 * OK, we found it. Now figure out what to do with this.
1192 if( curr_func->local_vars[i].regno != 0 )
1195 * Register variable. Point to DEBUG_context field.
1197 assert(curr_func->local_vars[i].regno - 1 < sizeof(reg_ofs)/sizeof(reg_ofs[0]));
1198 value->addr.off = ((DWORD)&DEBUG_context) +
1199 reg_ofs[curr_func->local_vars[i].regno - 1];
1200 value->cookie = DV_HOST;
1202 else
1204 value->addr.off = ebp + curr_func->local_vars[i].offset;
1205 value->cookie = DV_TARGET;
1207 value->addr.seg = 0;
1208 value->type = curr_func->local_vars[i].type;
1210 return TRUE;
1214 return FALSE;
1218 DEBUG_InfoLocals(void)
1220 struct name_hash * curr_func;
1221 unsigned int ebp;
1222 unsigned int eip;
1223 int i;
1224 unsigned int * ptr;
1225 unsigned int val;
1227 if( DEBUG_GetCurrentFrame(&curr_func, &eip, &ebp) == FALSE )
1229 return FALSE;
1232 DEBUG_Printf(DBG_CHN_MESG, "%s:\n", curr_func->name);
1234 for(i=0; i < curr_func->n_locals; i++ )
1237 * Test the range of validity of the local variable. This
1238 * comes up with RBRAC/LBRAC stabs in particular.
1240 if( (curr_func->local_vars[i].pc_start != 0)
1241 && ((eip - curr_func->value.addr.off)
1242 < curr_func->local_vars[i].pc_start) )
1244 continue;
1247 if( (curr_func->local_vars[i].pc_end != 0)
1248 && ((eip - curr_func->value.addr.off)
1249 > curr_func->local_vars[i].pc_end) )
1251 continue;
1254 DEBUG_PrintTypeCast(curr_func->local_vars[i].type);
1256 if( curr_func->local_vars[i].regno != 0 )
1258 ptr = (unsigned int *)(((DWORD)&DEBUG_context)
1259 + reg_ofs[curr_func->local_vars[i].regno - 1]);
1260 DEBUG_Printf(DBG_CHN_MESG, " %s (optimized into register $%s) == 0x%8.8x\n",
1261 curr_func->local_vars[i].name,
1262 reg_name[curr_func->local_vars[i].regno - 1],
1263 *ptr);
1265 else
1267 DEBUG_READ_MEM_VERBOSE((void*)(ebp + curr_func->local_vars[i].offset),
1268 &val, sizeof(val));
1269 DEBUG_Printf(DBG_CHN_MESG, " %s == 0x%8.8x\n",
1270 curr_func->local_vars[i].name, val);
1274 return TRUE;
1278 DEBUG_SetSymbolSize(struct name_hash * sym, unsigned int len)
1280 sym->symbol_size = len;
1282 return TRUE;
1286 DEBUG_SetSymbolBPOff(struct name_hash * sym, unsigned int off)
1288 sym->breakpoint_offset = off;
1290 return TRUE;
1294 DEBUG_GetSymbolAddr(struct name_hash * sym, DBG_ADDR * addr)
1297 *addr = sym->value.addr;
1299 return TRUE;
1302 int DEBUG_SetLocalSymbolType(struct wine_locals * sym, struct datatype * type)
1304 sym->type = type;
1306 return TRUE;