Internal PostScript printer driver improvements.
[wine/testsucceed.git] / debugger / hash.c
blobc0e139181599b1c9660f39528ff52345b7823e00
1 /*
2 * File hash.c - generate hash tables for Wine debugger symbols
4 * Copyright (C) 1993, Eric Youngdale.
5 */
8 #include <stdlib.h>
9 #include <stdio.h>
10 #include <string.h>
11 #include <limits.h>
12 #include <sys/types.h>
13 #include <neexe.h>
14 #include "module.h"
15 #include "process.h"
16 #include "selectors.h"
17 #include "debugger.h"
18 #include "toolhelp.h"
19 #include "xmalloc.h"
21 #define NR_NAME_HASH 16384
22 #ifndef PATH_MAX
23 #define PATH_MAX _MAX_PATH
24 #endif
26 static char * reg_name[] =
28 "eax", "ecx", "edx", "ebx", "esp", "ebp", "esi", "edi"
31 static unsigned reg_ofs[] =
33 FIELD_OFFSET(CONTEXT, Eax), FIELD_OFFSET(CONTEXT, Ecx),
34 FIELD_OFFSET(CONTEXT, Edx), FIELD_OFFSET(CONTEXT, Ebx),
35 FIELD_OFFSET(CONTEXT, Esp), FIELD_OFFSET(CONTEXT, Ebp),
36 FIELD_OFFSET(CONTEXT, Esi), FIELD_OFFSET(CONTEXT, Edi)
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_ADDR addr;
55 unsigned short flags;
56 unsigned short breakpoint_offset;
57 unsigned int symbol_size;
61 static BOOL32 DEBUG_GetStackSymbolValue( const char * name, DBG_ADDR *addr );
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)->addr.seg > (*name2)->addr.seg )
108 return 1;
111 if( (*name1)->addr.seg < (*name2)->addr.seg )
113 return -1;
116 if( (*name1)->addr.off > (*name2)->addr.off )
118 return 1;
121 if( (*name1)->addr.off < (*name2)->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()
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 nsym++;
150 sorttab_nsym = nsym;
151 if( nsym == 0 )
153 return;
156 addr_sorttab = (struct name_hash **) xrealloc(addr_sorttab,
157 nsym * sizeof(struct name_hash *));
159 nsym = 0;
160 for(i=0; i<NR_NAME_HASH; i++)
162 for (nh = name_hash_table[i]; nh; nh = nh->next)
164 addr_sorttab[nsym++] = nh;
168 qsort(addr_sorttab, nsym,
169 sizeof(struct name_hash *), DEBUG_cmp_sym);
170 sortlist_valid = TRUE;
174 /***********************************************************************
175 * DEBUG_AddSymbol
177 * Add a symbol to the table.
179 struct name_hash *
180 DEBUG_AddSymbol( const char * name, const DBG_ADDR *addr, const char * source,
181 int flags)
183 struct name_hash * new;
184 struct name_hash *nh;
185 static char prev_source[PATH_MAX] = {'\0', };
186 static char * prev_duped_source = NULL;
187 char * c;
188 int hash;
190 hash = name_hash(name);
191 for (nh = name_hash_table[hash]; nh; nh = nh->next)
193 if( ((nh->flags & SYM_INVALID) != 0) && strcmp(name, nh->name) == 0 )
195 nh->addr.off = addr->off;
196 nh->addr.seg = addr->seg;
197 if( nh->addr.type == NULL && addr->type != NULL )
199 nh->addr.type = addr->type;
201 nh->flags &= ~SYM_INVALID;
202 return nh;
204 if (nh->addr.seg == addr->seg &&
205 nh->addr.off == addr->off &&
206 strcmp(name, nh->name) == 0 )
208 return nh;
213 * First see if we already have an entry for this symbol. If so
214 * return it, so we don't end up with duplicates.
217 new = (struct name_hash *) xmalloc(sizeof(struct name_hash));
218 new->addr = *addr;
219 new->name = xstrdup(name);
221 if( source != NULL )
224 * This is an enhancement to reduce memory consumption. The idea
225 * is that we duplicate a given string only once. This is a big
226 * win if there are lots of symbols defined in a given source file.
228 if( strcmp(source, prev_source) == 0 )
230 new->sourcefile = prev_duped_source;
232 else
234 strcpy(prev_source, source);
235 prev_duped_source = new->sourcefile = xstrdup(source);
238 else
240 new->sourcefile = NULL;
243 new->n_lines = 0;
244 new->lines_alloc = 0;
245 new->linetab = NULL;
247 new->n_locals = 0;
248 new->locals_alloc = 0;
249 new->local_vars = NULL;
251 new->flags = flags;
252 new->next = NULL;
254 /* Now insert into the hash table */
255 new->next = name_hash_table[hash];
256 name_hash_table[hash] = new;
259 * Check some heuristics based upon the file name to see whether
260 * we want to step through this guy or not. These are machine generated
261 * assembly files that are used to translate between the MS way of
262 * calling things and the GCC way of calling things. In general we
263 * always want to step through.
265 if( source != NULL )
267 c = strrchr(source, '.');
268 if( c != NULL && strcmp(c, ".s") == 0 )
270 c = strrchr(source, '/');
271 if( c != NULL )
273 c++;
274 if( (strcmp(c, "callfrom16.s") == 0)
275 || (strcmp(c, "callto16.s") == 0)
276 || (strcmp(c, "call32.s") == 0) )
278 new->flags |= SYM_TRAMPOLINE;
284 sortlist_valid = FALSE;
285 return new;
288 BOOL32 DEBUG_Normalize(struct name_hash * nh )
292 * We aren't adding any more locals or linenumbers to this function.
293 * Free any spare memory that we might have allocated.
295 if( nh == NULL )
297 return TRUE;
300 if( nh->n_locals != nh->locals_alloc )
302 nh->locals_alloc = nh->n_locals;
303 nh->local_vars = xrealloc(nh->local_vars,
304 nh->locals_alloc * sizeof(WineLocals));
307 if( nh->n_lines != nh->lines_alloc )
309 nh->lines_alloc = nh->n_lines;
310 nh->linetab = xrealloc(nh->linetab,
311 nh->lines_alloc * sizeof(WineLineNo));
314 return TRUE;
317 /***********************************************************************
318 * DEBUG_GetSymbolValue
320 * Get the address of a named symbol.
322 BOOL32 DEBUG_GetSymbolValue( const char * name, const int lineno,
323 DBG_ADDR *addr, int bp_flag )
325 char buffer[256];
326 struct name_hash *nh;
328 for(nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
330 if( (nh->flags & SYM_INVALID) != 0 )
332 continue;
335 if (!strcmp(nh->name, name)) break;
338 if (!nh && (name[0] != '_'))
340 buffer[0] = '_';
341 strcpy(buffer+1, name);
342 for(nh = name_hash_table[name_hash(buffer)]; nh; nh = nh->next)
344 if( (nh->flags & SYM_INVALID) != 0 )
346 continue;
348 if (!strcmp(nh->name, buffer)) break;
353 * If we don't have anything here, then try and see if this
354 * is a local symbol to the current stack frame. No matter
355 * what, we have nothing more to do, so we let that function
356 * decide what we ultimately return.
358 if (!nh)
360 return DEBUG_GetStackSymbolValue(name, addr);
363 return DEBUG_GetLineNumberAddr( nh, lineno, addr, bp_flag );
366 /***********************************************************************
367 * DEBUG_GetLineNumberAddr
369 * Get the address of a named symbol.
371 BOOL32 DEBUG_GetLineNumberAddr( struct name_hash * nh, const int lineno,
372 DBG_ADDR *addr, int bp_flag )
374 int i;
376 if( lineno == -1 )
378 *addr = nh->addr;
379 if( bp_flag )
381 addr->off += nh->breakpoint_offset;
384 else
387 * Search for the specific line number. If we don't find it,
388 * then return FALSE.
390 if( nh->linetab == NULL )
392 return FALSE;
395 for(i=0; i < nh->n_lines; i++ )
397 if( nh->linetab[i].line_number == lineno )
399 *addr = nh->linetab[i].pc_offset;
400 return TRUE;
405 * This specific line number not found.
407 return FALSE;
410 return TRUE;
414 /***********************************************************************
415 * DEBUG_SetSymbolValue
417 * Set the address of a named symbol.
419 BOOL32 DEBUG_SetSymbolValue( const char * name, const DBG_ADDR *addr )
421 char buffer[256];
422 struct name_hash *nh;
424 for(nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
425 if (!strcmp(nh->name, name)) break;
427 if (!nh && (name[0] != '_'))
429 buffer[0] = '_';
430 strcpy(buffer+1, name);
431 for(nh = name_hash_table[name_hash(buffer)]; nh; nh = nh->next)
432 if (!strcmp(nh->name, buffer)) break;
435 if (!nh) return FALSE;
436 nh->addr = *addr;
437 nh->flags &= SYM_INVALID;
438 DBG_FIX_ADDR_SEG( &nh->addr, DS_reg(&DEBUG_context) );
439 return TRUE;
443 /***********************************************************************
444 * DEBUG_FindNearestSymbol
446 * Find the symbol nearest to a given address.
447 * If ebp is specified as non-zero, it means we should dump the argument
448 * list into the string we return as well.
450 const char * DEBUG_FindNearestSymbol( const DBG_ADDR *addr, int flag,
451 struct name_hash ** rtn,
452 unsigned int ebp,
453 struct list_id * source)
455 static char name_buffer[MAX_PATH + 256];
456 static char arglist[1024];
457 static char argtmp[256];
458 struct name_hash * nearest = NULL;
459 int mid, high, low;
460 unsigned int * ptr;
461 int lineno;
462 char * lineinfo, *sourcefile;
463 int i;
464 char linebuff[16];
466 if( rtn != NULL )
468 *rtn = NULL;
471 if( source != NULL )
473 source->sourcefile = NULL;
474 source->line = -1;
477 if( sortlist_valid == FALSE )
479 DEBUG_ResortSymbols();
482 if( sortlist_valid == FALSE )
484 return NULL;
488 * FIXME - use the binary search that we added to
489 * the function DEBUG_CheckLinenoStatus. Better yet, we should
490 * probably keep some notion of the current function so we don't
491 * have to search every time.
494 * Binary search to find closest symbol.
496 low = 0;
497 high = sorttab_nsym;
498 if( addr_sorttab[0]->addr.seg > addr->seg
499 || ( addr_sorttab[0]->addr.seg == addr->seg
500 && addr_sorttab[0]->addr.off > addr->off) )
502 nearest = NULL;
504 else if( addr_sorttab[high - 1]->addr.seg < addr->seg
505 || ( addr_sorttab[high - 1]->addr.seg == addr->seg
506 && addr_sorttab[high - 1]->addr.off < addr->off) )
508 nearest = addr_sorttab[high - 1];
510 else
512 while(1==1)
514 mid = (high + low)/2;
515 if( mid == low )
518 * See if there are any other entries that might also
519 * have the same address, and would also have a line
520 * number table.
522 if( mid > 0 && addr_sorttab[mid]->linetab == NULL )
524 if( (addr_sorttab[mid - 1]->addr.seg ==
525 addr_sorttab[mid]->addr.seg)
526 && (addr_sorttab[mid - 1]->addr.off ==
527 addr_sorttab[mid]->addr.off)
528 && (addr_sorttab[mid - 1]->linetab != NULL) )
530 mid--;
534 if( (mid < sorttab_nsym - 1)
535 && (addr_sorttab[mid]->linetab == NULL) )
537 if( (addr_sorttab[mid + 1]->addr.seg ==
538 addr_sorttab[mid]->addr.seg)
539 && (addr_sorttab[mid + 1]->addr.off ==
540 addr_sorttab[mid]->addr.off)
541 && (addr_sorttab[mid + 1]->linetab != NULL) )
543 mid++;
546 nearest = addr_sorttab[mid];
547 #if 0
548 fprintf(stderr, "Found %x:%x when looking for %x:%x %x %s\n",
549 addr_sorttab[mid ]->addr.seg,
550 addr_sorttab[mid ]->addr.off,
551 addr->seg, addr->off,
552 addr_sorttab[mid ]->linetab,
553 addr_sorttab[mid ]->name);
554 #endif
555 break;
557 if( (addr_sorttab[mid]->addr.seg < addr->seg)
558 || ( addr_sorttab[mid]->addr.seg == addr->seg
559 && addr_sorttab[mid]->addr.off <= addr->off) )
561 low = mid;
563 else
565 high = mid;
570 if (!nearest) return NULL;
572 if( rtn != NULL )
574 *rtn = nearest;
578 * Fill in the relevant bits to the structure so that we can
579 * locate the source and line for this bit of code.
581 if( source != NULL )
583 source->sourcefile = nearest->sourcefile;
584 if( nearest->linetab == NULL )
586 source->line = -1;
588 else
590 source->line = nearest->linetab[0].line_number;
594 lineinfo = "";
595 lineno = -1;
598 * Prepare to display the argument list. If ebp is specified, it is
599 * the framepointer for the function in question. If not specified,
600 * we don't want the arglist.
602 memset(arglist, '\0', sizeof(arglist));
603 if( ebp != 0 )
605 for(i=0; i < nearest->n_locals; i++ )
608 * If this is a register (offset == 0) or a local
609 * variable, we don't want to know about it.
611 if( nearest->local_vars[i].offset <= 0 )
613 continue;
616 ptr = (unsigned int *) (ebp + nearest->local_vars[i].offset);
617 if( arglist[0] == '\0' )
619 arglist[0] = '(';
621 else
623 strcat(arglist, ", ");
626 sprintf(argtmp, "%s=0x%x", nearest->local_vars[i].name,
627 *ptr);
628 strcat(arglist, argtmp);
630 if( arglist[0] == '(' )
632 strcat(arglist, ")");
636 if( (nearest->sourcefile != NULL) && (flag == TRUE)
637 && (addr->off - nearest->addr.off < 0x100000) )
641 * Try and find the nearest line number to the current offset.
643 if( nearest->linetab != NULL )
645 low = 0;
646 high = nearest->n_lines;
647 while ((high - low) > 1)
649 mid = (high + low) / 2;
650 if (addr->off < nearest->linetab[mid].pc_offset.off)
651 high = mid;
652 else
653 low = mid;
655 lineno = nearest->linetab[low].line_number;
658 if( lineno != -1 )
660 sprintf(linebuff, ":%d", lineno);
661 lineinfo = linebuff;
662 if( source != NULL )
664 source->line = lineno;
668 /* Remove the path from the file name */
669 sourcefile = strrchr( nearest->sourcefile, '/' );
670 if (!sourcefile) sourcefile = nearest->sourcefile;
671 else sourcefile++;
673 if (addr->off == nearest->addr.off)
674 sprintf( name_buffer, "%s%s [%s%s]", nearest->name,
675 arglist, sourcefile, lineinfo);
676 else
677 sprintf( name_buffer, "%s+0x%lx%s [%s%s]", nearest->name,
678 addr->off - nearest->addr.off,
679 arglist, sourcefile, lineinfo );
681 else
683 if (addr->off == nearest->addr.off)
684 sprintf( name_buffer, "%s%s", nearest->name, arglist);
685 else {
686 if (addr->seg && (nearest->addr.seg!=addr->seg))
687 return NULL;
688 else
689 sprintf( name_buffer, "%s+0x%lx%s", nearest->name,
690 addr->off - nearest->addr.off, arglist);
693 return name_buffer;
697 /***********************************************************************
698 * DEBUG_ReadSymbolTable
700 * Read a symbol file into the hash table.
702 void DEBUG_ReadSymbolTable( const char * filename )
704 FILE * symbolfile;
705 DBG_ADDR addr = { 0, 0 };
706 int nargs;
707 char type;
708 char * cpnt;
709 char buffer[256];
710 char name[256];
712 if (!(symbolfile = fopen(filename, "r")))
714 fprintf( stderr, "Unable to open symbol table %s\n", filename );
715 return;
718 fprintf( stderr, "Reading symbols from file %s\n", filename );
720 while (1)
722 fgets( buffer, sizeof(buffer), symbolfile );
723 if (feof(symbolfile)) break;
725 /* Strip any text after a # sign (i.e. comments) */
726 cpnt = buffer;
727 while (*cpnt)
728 if(*cpnt++ == '#') { *cpnt = 0; break; }
730 /* Quietly ignore any lines that have just whitespace */
731 cpnt = buffer;
732 while(*cpnt)
734 if(*cpnt != ' ' && *cpnt != '\t') break;
735 cpnt++;
737 if (!(*cpnt) || *cpnt == '\n') continue;
739 nargs = sscanf(buffer, "%lx %c %s", &addr.off, &type, name);
740 DEBUG_AddSymbol( name, &addr, NULL, SYM_WINE );
742 fclose(symbolfile);
746 /***********************************************************************
747 * DEBUG_LoadEntryPoints16
749 * Load the entry points of a Win16 module into the hash table.
751 static void DEBUG_LoadEntryPoints16( HMODULE16 hModule, NE_MODULE *pModule,
752 const char *name )
754 DBG_ADDR addr;
755 char buffer[256];
756 FARPROC16 address;
758 /* First search the resident names */
760 unsigned char *cpnt = (unsigned char *)pModule + pModule->name_table;
761 while (*cpnt)
763 cpnt += *cpnt + 1 + sizeof(WORD);
764 sprintf( buffer, "%s.%.*s", name, *cpnt, cpnt + 1 );
765 if ((address = NE_GetEntryPoint(hModule, *(WORD *)(cpnt + *cpnt + 1))))
767 addr.seg = HIWORD(address);
768 addr.off = LOWORD(address);
769 addr.type = NULL;
770 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
774 /* Now search the non-resident names table */
776 if (!pModule->nrname_handle) return; /* No non-resident table */
777 cpnt = (char *)GlobalLock16( pModule->nrname_handle );
778 while (*cpnt)
780 cpnt += *cpnt + 1 + sizeof(WORD);
781 sprintf( buffer, "%s.%.*s", name, *cpnt, cpnt + 1 );
782 if ((address = NE_GetEntryPoint(hModule, *(WORD *)(cpnt + *cpnt + 1))))
784 addr.seg = HIWORD(address);
785 addr.off = LOWORD(address);
786 addr.type = NULL;
787 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
793 /***********************************************************************
794 * DEBUG_LoadEntryPoints32
796 * Load the entry points of a Win32 module into the hash table.
798 static void DEBUG_LoadEntryPoints32( HMODULE32 hModule, const char *name )
800 #define RVA(x) (hModule+(DWORD)(x))
802 DBG_ADDR addr;
803 char buffer[256];
804 int i, j;
805 IMAGE_SECTION_HEADER *pe_seg;
806 IMAGE_EXPORT_DIRECTORY *exports;
807 IMAGE_DATA_DIRECTORY *dir;
808 WORD *ordinals;
809 void **functions;
810 const char **names;
812 addr.seg = 0;
813 addr.type = NULL;
815 /* Add start of DLL */
817 addr.off = hModule;
818 DEBUG_AddSymbol( name, &addr, NULL, SYM_WIN32 | SYM_FUNC );
820 /* Add entry point */
822 sprintf( buffer, "%s.EntryPoint", name );
823 addr.off = (DWORD)RVA_PTR( hModule, OptionalHeader.AddressOfEntryPoint );
824 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
826 /* Add start of sections */
828 pe_seg = PE_SECTIONS(hModule);
829 for (i = 0; i < PE_HEADER(hModule)->FileHeader.NumberOfSections; i++)
831 sprintf( buffer, "%s.%s", name, pe_seg->Name );
832 addr.off = RVA(pe_seg->VirtualAddress );
833 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
834 pe_seg++;
837 /* Add exported functions */
839 dir = &PE_HEADER(hModule)->OptionalHeader.
840 DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT];
841 if (dir->Size)
843 exports = (IMAGE_EXPORT_DIRECTORY *)RVA( dir->VirtualAddress );
844 ordinals = (WORD *)RVA( exports->AddressOfNameOrdinals );
845 names = (const char **)RVA( exports->AddressOfNames );
846 functions = (void **)RVA( exports->AddressOfFunctions );
848 for (i = 0; i < exports->NumberOfNames; i++)
850 if (!names[i]) continue;
851 sprintf( buffer, "%s.%s", name, (char *)RVA(names[i]) );
852 addr.off = RVA( functions[ordinals[i]] );
853 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
856 for (i = 0; i < exports->NumberOfFunctions; i++)
858 if (!functions[i]) continue;
859 /* Check if we already added it with a name */
860 for (j = 0; j < exports->NumberOfNames; j++)
861 if ((ordinals[j] == i) && names[j]) break;
862 if (j < exports->NumberOfNames) continue;
863 sprintf( buffer, "%s.%ld", name, i + exports->Base );
864 addr.off = (DWORD)RVA( functions[i] );
865 DEBUG_AddSymbol( buffer, &addr, NULL, SYM_WIN32 | SYM_FUNC );
869 dir = &PE_HEADER(hModule)->OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_DEBUG];
870 if (dir->Size)
871 DEBUG_RegisterDebugInfo(hModule, name, dir->VirtualAddress, dir->Size);
872 #undef RVA
876 /***********************************************************************
877 * DEBUG_LoadEntryPoints
879 * Load the entry points of all the modules into the hash table.
881 void DEBUG_LoadEntryPoints(void)
883 MODULEENTRY entry;
884 NE_MODULE *pModule;
885 BOOL32 ok;
886 WINE_MODREF *wm;
887 int rowcount = 3;
889 fprintf( stderr, " " );
890 for (ok = ModuleFirst(&entry); ok; ok = ModuleNext(&entry))
892 if (!(pModule = NE_GetPtr( entry.hModule ))) continue;
893 if (!(pModule->flags & NE_FFLAGS_WIN32)) /* NE module */
895 if ((rowcount + strlen(entry.szModule)) > 76)
897 fprintf( stderr,"\n ");
898 rowcount = 3;
900 fprintf( stderr, " %s", entry.szModule );
901 rowcount += strlen(entry.szModule) + 1;
903 DEBUG_LoadEntryPoints16( entry.hModule, pModule, entry.szModule );
906 for (wm=PROCESS_Current()->modref_list;wm;wm=wm->next)
908 if ((rowcount + strlen(wm->modname)) > 76)
910 fprintf( stderr,"\n ");
911 rowcount = 3;
913 fprintf( stderr, " %s", wm->modname );
914 rowcount += strlen(wm->modname) + 1;
915 DEBUG_LoadEntryPoints32( wm->module, wm->modname );
917 fprintf( stderr, "\n" );
921 void
922 DEBUG_AddLineNumber( struct name_hash * func, int line_num,
923 unsigned long offset )
925 if( func == NULL )
927 return;
930 if( func->n_lines + 1 >= func->lines_alloc )
932 func->lines_alloc += 64;
933 func->linetab = xrealloc(func->linetab,
934 func->lines_alloc * sizeof(WineLineNo));
937 func->linetab[func->n_lines].line_number = line_num;
938 func->linetab[func->n_lines].pc_offset.seg = func->addr.seg;
939 func->linetab[func->n_lines].pc_offset.off = func->addr.off + offset;
940 func->linetab[func->n_lines].pc_offset.type = NULL;
941 func->n_lines++;
945 struct wine_locals *
946 DEBUG_AddLocal( struct name_hash * func, int regno,
947 int offset,
948 int pc_start,
949 int pc_end,
950 char * name)
952 if( func == NULL )
954 return NULL;
957 if( func->n_locals + 1 >= func->locals_alloc )
959 func->locals_alloc += 32;
960 func->local_vars = xrealloc(func->local_vars,
961 func->locals_alloc * sizeof(WineLocals));
964 func->local_vars[func->n_locals].regno = regno;
965 func->local_vars[func->n_locals].offset = offset;
966 func->local_vars[func->n_locals].pc_start = pc_start;
967 func->local_vars[func->n_locals].pc_end = pc_end;
968 func->local_vars[func->n_locals].name = xstrdup(name);
969 func->local_vars[func->n_locals].type = NULL;
970 func->n_locals++;
972 return &func->local_vars[func->n_locals - 1];
975 void
976 DEBUG_DumpHashInfo()
978 int i;
979 int depth;
980 struct name_hash *nh;
983 * Utility function to dump stats about the hash table.
985 for(i=0; i<NR_NAME_HASH; i++)
987 depth = 0;
988 for (nh = name_hash_table[i]; nh; nh = nh->next)
990 depth++;
992 fprintf(stderr, "Bucket %d: %d\n", i, depth);
996 /***********************************************************************
997 * DEBUG_CheckLinenoStatus
999 * Find the symbol nearest to a given address.
1000 * If ebp is specified as non-zero, it means we should dump the argument
1001 * list into the string we return as well.
1003 int DEBUG_CheckLinenoStatus( const DBG_ADDR *addr)
1005 struct name_hash * nearest = NULL;
1006 int mid, high, low;
1008 if( sortlist_valid == FALSE )
1010 DEBUG_ResortSymbols();
1014 * Binary search to find closest symbol.
1016 low = 0;
1017 high = sorttab_nsym;
1018 if( addr_sorttab[0]->addr.seg > addr->seg
1019 || ( addr_sorttab[0]->addr.seg == addr->seg
1020 && addr_sorttab[0]->addr.off > addr->off) )
1022 nearest = NULL;
1024 else if( addr_sorttab[high - 1]->addr.seg < addr->seg
1025 || ( addr_sorttab[high - 1]->addr.seg == addr->seg
1026 && addr_sorttab[high - 1]->addr.off < addr->off) )
1028 nearest = addr_sorttab[high - 1];
1030 else
1032 while(1==1)
1034 mid = (high + low)/2;
1035 if( mid == low )
1038 * See if there are any other entries that might also
1039 * have the same address, and would also have a line
1040 * number table.
1042 if( mid > 0 && addr_sorttab[mid]->linetab == NULL )
1044 if( (addr_sorttab[mid - 1]->addr.seg ==
1045 addr_sorttab[mid]->addr.seg)
1046 && (addr_sorttab[mid - 1]->addr.off ==
1047 addr_sorttab[mid]->addr.off)
1048 && (addr_sorttab[mid - 1]->linetab != NULL) )
1050 mid--;
1054 if( (mid < sorttab_nsym - 1)
1055 && (addr_sorttab[mid]->linetab == NULL) )
1057 if( (addr_sorttab[mid + 1]->addr.seg ==
1058 addr_sorttab[mid]->addr.seg)
1059 && (addr_sorttab[mid + 1]->addr.off ==
1060 addr_sorttab[mid]->addr.off)
1061 && (addr_sorttab[mid + 1]->linetab != NULL) )
1063 mid++;
1066 nearest = addr_sorttab[mid];
1067 #if 0
1068 fprintf(stderr, "Found %x:%x when looking for %x:%x %x %s\n",
1069 addr_sorttab[mid ]->addr.seg,
1070 addr_sorttab[mid ]->addr.off,
1071 addr->seg, addr->off,
1072 addr_sorttab[mid ]->linetab,
1073 addr_sorttab[mid ]->name);
1074 #endif
1075 break;
1077 if( (addr_sorttab[mid]->addr.seg < addr->seg)
1078 || ( addr_sorttab[mid]->addr.seg == addr->seg
1079 && addr_sorttab[mid]->addr.off <= addr->off) )
1081 low = mid;
1083 else
1085 high = mid;
1090 if (!nearest) return FUNC_HAS_NO_LINES;
1092 if( nearest->flags & SYM_STEP_THROUGH )
1095 * This will cause us to keep single stepping until
1096 * we get to the other side somewhere.
1098 return NOT_ON_LINENUMBER;
1101 if( (nearest->flags & SYM_TRAMPOLINE) )
1104 * This will cause us to keep single stepping until
1105 * we get to the other side somewhere.
1107 return FUNC_IS_TRAMPOLINE;
1110 if( nearest->linetab == NULL )
1112 return FUNC_HAS_NO_LINES;
1117 * We never want to stop on the first instruction of a function
1118 * even if it has it's own linenumber. Let the thing keep running
1119 * until it gets past the function prologue. We only do this if there
1120 * is more than one line number for the function, of course.
1122 if( nearest->addr.off == addr->off && nearest->n_lines > 1 )
1124 return NOT_ON_LINENUMBER;
1127 if( (nearest->sourcefile != NULL)
1128 && (addr->off - nearest->addr.off < 0x100000) )
1130 low = 0;
1131 high = nearest->n_lines;
1132 while ((high - low) > 1)
1134 mid = (high + low) / 2;
1135 if (addr->off < nearest->linetab[mid].pc_offset.off) high = mid;
1136 else low = mid;
1138 if (addr->off == nearest->linetab[low].pc_offset.off)
1139 return AT_LINENUMBER;
1140 else
1141 return NOT_ON_LINENUMBER;
1144 return FUNC_HAS_NO_LINES;
1147 /***********************************************************************
1148 * DEBUG_GetFuncInfo
1150 * Find the symbol nearest to a given address.
1151 * Returns sourcefile name and line number in a format that the listing
1152 * handler can deal with.
1154 void
1155 DEBUG_GetFuncInfo( struct list_id * ret, const char * filename,
1156 const char * name)
1158 char buffer[256];
1159 char * pnt;
1160 struct name_hash *nh;
1162 for(nh = name_hash_table[name_hash(name)]; nh; nh = nh->next)
1164 if( filename != NULL )
1167 if( nh->sourcefile == NULL )
1169 continue;
1172 pnt = strrchr(nh->sourcefile, '/');
1173 if( strcmp(nh->sourcefile, filename) != 0
1174 && (pnt == NULL || strcmp(pnt + 1, filename) != 0) )
1176 continue;
1179 if (!strcmp(nh->name, name)) break;
1182 if (!nh && (name[0] != '_'))
1184 buffer[0] = '_';
1185 strcpy(buffer+1, name);
1186 for(nh = name_hash_table[name_hash(buffer)]; nh; nh = nh->next)
1188 if( filename != NULL )
1190 if( nh->sourcefile == NULL )
1192 continue;
1195 pnt = strrchr(nh->sourcefile, '/');
1196 if( strcmp(nh->sourcefile, filename) != 0
1197 && (pnt == NULL || strcmp(pnt + 1, filename) != 0) )
1199 continue;
1202 if (!strcmp(nh->name, buffer)) break;
1206 if( !nh )
1208 if( filename != NULL )
1210 fprintf(stderr, "No such function %s in %s\n", name, filename);
1212 else
1214 fprintf(stderr, "No such function %s\n", name);
1216 ret->sourcefile = NULL;
1217 ret->line = -1;
1218 return;
1221 ret->sourcefile = nh->sourcefile;
1224 * Search for the specific line number. If we don't find it,
1225 * then return FALSE.
1227 if( nh->linetab == NULL )
1229 ret->line = -1;
1231 else
1233 ret->line = nh->linetab[0].line_number;
1237 /***********************************************************************
1238 * DEBUG_GetStackSymbolValue
1240 * Get the address of a named symbol from the current stack frame.
1242 static
1243 BOOL32 DEBUG_GetStackSymbolValue( const char * name, DBG_ADDR *addr )
1245 struct name_hash * curr_func;
1246 unsigned int ebp;
1247 unsigned int eip;
1248 int i;
1250 if( DEBUG_GetCurrentFrame(&curr_func, &eip, &ebp) == FALSE )
1252 return FALSE;
1255 for(i=0; i < curr_func->n_locals; i++ )
1258 * Test the range of validity of the local variable. This
1259 * comes up with RBRAC/LBRAC stabs in particular.
1261 if( (curr_func->local_vars[i].pc_start != 0)
1262 && ((eip - curr_func->addr.off)
1263 < curr_func->local_vars[i].pc_start) )
1265 continue;
1268 if( (curr_func->local_vars[i].pc_end != 0)
1269 && ((eip - curr_func->addr.off)
1270 > curr_func->local_vars[i].pc_end) )
1272 continue;
1275 if( strcmp(name, curr_func->local_vars[i].name) == 0 )
1278 * OK, we found it. Now figure out what to do with this.
1280 /* FIXME: what if regno == 0 ($eax) */
1281 if( curr_func->local_vars[i].regno != 0 )
1284 * Register variable. Point to DEBUG_context field.
1286 addr->seg = 0;
1287 addr->off = ((DWORD)&DEBUG_context) + reg_ofs[curr_func->local_vars[i].regno];
1288 addr->type = curr_func->local_vars[i].type;
1290 return TRUE;
1293 addr->seg = 0;
1294 addr->off = ebp + curr_func->local_vars[i].offset;
1295 addr->type = curr_func->local_vars[i].type;
1297 return TRUE;
1300 return FALSE;
1304 DEBUG_InfoLocals()
1306 struct name_hash * curr_func;
1307 unsigned int ebp;
1308 unsigned int eip;
1309 int i;
1310 unsigned int * ptr;
1311 int rtn = FALSE;
1313 if( DEBUG_GetCurrentFrame(&curr_func, &eip, &ebp) == FALSE )
1315 return FALSE;
1318 for(i=0; i < curr_func->n_locals; i++ )
1321 * Test the range of validity of the local variable. This
1322 * comes up with RBRAC/LBRAC stabs in particular.
1324 if( (curr_func->local_vars[i].pc_start != 0)
1325 && ((eip - curr_func->addr.off)
1326 < curr_func->local_vars[i].pc_start) )
1328 continue;
1331 if( (curr_func->local_vars[i].pc_end != 0)
1332 && ((eip - curr_func->addr.off)
1333 > curr_func->local_vars[i].pc_end) )
1335 continue;
1338 if( curr_func->local_vars[i].offset == 0 )
1340 ptr = (unsigned int *) (((DWORD)&DEBUG_context)
1341 + reg_ofs[curr_func->local_vars[i].regno]);
1342 fprintf(stderr, "%s:%s (optimized into register $%s) == 0x%8.8x\n",
1343 curr_func->name, curr_func->local_vars[i].name,
1344 reg_name[curr_func->local_vars[i].regno],
1345 *ptr);
1347 else
1349 ptr = (unsigned int *) (ebp + curr_func->local_vars[i].offset);
1350 fprintf(stderr, "%s:%s == 0x%8.8x\n",
1351 curr_func->name, curr_func->local_vars[i].name,
1352 *ptr);
1356 rtn = TRUE;
1358 return (rtn);
1362 DEBUG_SetSymbolSize(struct name_hash * sym, unsigned int len)
1364 sym->symbol_size = len;
1366 return TRUE;
1370 DEBUG_SetSymbolBPOff(struct name_hash * sym, unsigned int off)
1372 sym->breakpoint_offset = off;
1374 return TRUE;
1378 DEBUG_GetSymbolAddr(struct name_hash * sym, DBG_ADDR * addr)
1381 *addr = sym->addr;
1383 return TRUE;
1386 int DEBUG_SetLocalSymbolType(struct wine_locals * sym, struct datatype * type)
1388 sym->type = type;
1390 return TRUE;