📄 symbol.c
字号:
/* symbol.c - program symbol and line data routines *//* SimpleScalar(TM) Tool Suite * Copyright (C) 1994-2003 by Todd M. Austin, Ph.D. and SimpleScalar, LLC. * All Rights Reserved. * * THIS IS A LEGAL DOCUMENT, BY USING SIMPLESCALAR, * YOU ARE AGREEING TO THESE TERMS AND CONDITIONS. * * No portion of this work may be used by any commercial entity, or for any * commercial purpose, without the prior, written permission of SimpleScalar, * LLC (info@simplescalar.com). Nonprofit and noncommercial use is permitted * as described below. * * 1. SimpleScalar is provided AS IS, with no warranty of any kind, express * or implied. The user of the program accepts full responsibility for the * application of the program and the use of any results. * * 2. Nonprofit and noncommercial use is encouraged. SimpleScalar may be * downloaded, compiled, executed, copied, and modified solely for nonprofit, * educational, noncommercial research, and noncommercial scholarship * purposes provided that this notice in its entirety accompanies all copies. * Copies of the modified software can be delivered to persons who use it * solely for nonprofit, educational, noncommercial research, and * noncommercial scholarship purposes provided that this notice in its * entirety accompanies all copies. * * 3. ALL COMMERCIAL USE, AND ALL USE BY FOR PROFIT ENTITIES, IS EXPRESSLY * PROHIBITED WITHOUT A LICENSE FROM SIMPLESCALAR, LLC (info@simplescalar.com). * * 4. No nonprofit user may place any restrictions on the use of this software, * including as modified by the user, by any other authorized user. * * 5. Noncommercial and nonprofit users may distribute copies of SimpleScalar * in compiled or executable form as set forth in Section 2, provided that * either: (A) it is accompanied by the corresponding machine-readable source * code, or (B) it is accompanied by a written offer, with no time limit, to * give anyone a machine-readable copy of the corresponding source code in * return for reimbursement of the cost of distribution. This written offer * must permit verbatim duplication by anyone, or (C) it is distributed by * someone who received only the executable form, and is accompanied by a * copy of the written offer of source code. * * 6. SimpleScalar was developed by Todd M. Austin, Ph.D. The tool suite is * currently maintained by SimpleScalar LLC (info@simplescalar.com). US Mail: * 2395 Timbercrest Court, Ann Arbor, MI 48105. * * Copyright (C) 1994-2003 by Todd M. Austin, Ph.D. and SimpleScalar, LLC. */#include <stdio.h>#include <stdlib.h>#include "host.h"#include "misc.h"#include "loader.h"#include "symbol.h"#ifdef BFD_LOADER#include <bfd.h>#else /* !BFD_LOADER */#include "target-alpha/ecoff.h"#endif /* BFD_LOADER *//* #define PRINT_SYMS *//* symbol database in no particular order */struct sym_sym_t *sym_db = NULL;/* all symbol sorted by address */int sym_nsyms = 0;struct sym_sym_t **sym_syms = NULL;/* all symbols sorted by name */struct sym_sym_t **sym_syms_by_name = NULL;/* text symbols sorted by address */int sym_ntextsyms = 0;struct sym_sym_t **sym_textsyms = NULL;/* text symbols sorted by name */struct sym_sym_t **sym_textsyms_by_name = NULL;/* data symbols sorted by address */int sym_ndatasyms = 0;struct sym_sym_t **sym_datasyms = NULL;/* data symbols sorted by name */struct sym_sym_t **sym_datasyms_by_name = NULL;/* symbols loaded? */static int syms_loaded = FALSE;#ifdef PRINT_SYMS/* convert BFD symbols flags to a printable string */static char * /* symbol flags string */flags2str(unsigned int flags) /* bfd symbol flags */{ static char buf[256]; char *p; if (!flags) return ""; p = buf; *p = '\0'; if (flags & BSF_LOCAL) { *p++ = 'L'; *p++ = '|'; } if (flags & BSF_GLOBAL) { *p++ = 'G'; *p++ = '|'; } if (flags & BSF_DEBUGGING) { *p++ = 'D'; *p++ = '|'; } if (flags & BSF_FUNCTION) { *p++ = 'F'; *p++ = '|'; } if (flags & BSF_KEEP) { *p++ = 'K'; *p++ = '|'; } if (flags & BSF_KEEP_G) { *p++ = 'k'; *p++ = '|'; } if (flags & BSF_WEAK) { *p++ = 'W'; *p++ = '|'; } if (flags & BSF_SECTION_SYM) { *p++ = 'S'; *p++ = '|'; } if (flags & BSF_OLD_COMMON) { *p++ = 'O'; *p++ = '|'; } if (flags & BSF_NOT_AT_END) { *p++ = 'N'; *p++ = '|'; } if (flags & BSF_CONSTRUCTOR) { *p++ = 'C'; *p++ = '|'; } if (flags & BSF_WARNING) { *p++ = 'w'; *p++ = '|'; } if (flags & BSF_INDIRECT) { *p++ = 'I'; *p++ = '|'; } if (flags & BSF_FILE) { *p++ = 'f'; *p++ = '|'; } if (p == buf) panic("no flags detected"); *--p = '\0'; return buf;}#endif /* PRINT_SYMS *//* qsort helper function */static intacmp(struct sym_sym_t **sym1, struct sym_sym_t **sym2){ return (int)((*sym1)->addr - (*sym2)->addr);}/* qsort helper function */static intncmp(struct sym_sym_t **sym1, struct sym_sym_t **sym2){ return strcmp((*sym1)->name, (*sym2)->name);}#define RELEVANT_SCOPE(SYM) \(/* global symbol */ \ ((SYM)->flags & BSF_GLOBAL) \ || (/* local symbol */ \ (((SYM)->flags & (BSF_LOCAL|BSF_DEBUGGING)) == BSF_LOCAL) \ && (SYM)->name[0] != '$') \ || (/* compiler local */ \ load_locals \ && ((/* basic block idents */ \ ((SYM)->flags&(BSF_LOCAL|BSF_DEBUGGING))==(BSF_LOCAL|BSF_DEBUGGING)\ && (SYM)->name[0] == '$') \ || (/* local constant idents */ \ ((SYM)->flags & (BSF_LOCAL|BSF_DEBUGGING)) == (BSF_LOCAL) \ && (SYM)->name[0] == '$')))) /* load symbols out of FNAME */voidsym_loadsyms(char *fname, /* file name containing symbols */ int load_locals) /* load local symbols */{ int i, debug_cnt;#ifdef BFD_LOADER bfd *abfd; asymbol **syms; int storage, i, nsyms, debug_cnt;#else /* !BFD_LOADER */ int len; FILE *fobj; struct ecoff_filehdr fhdr; struct ecoff_aouthdr ahdr; struct ecoff_symhdr_t symhdr; char *strtab = NULL; struct ecoff_EXTR *extr;#endif /* BFD_LOADER */ if (syms_loaded) { /* symbols are already loaded */ /* FIXME: can't handle symbols from multiple files */ return; }#ifdef BFD_LOADER /* load the program into memory, try both endians */ if (!(abfd = bfd_openr(fname, "ss-coff-big"))) if (!(abfd = bfd_openr(fname, "ss-coff-little"))) fatal("cannot open executable `%s'", fname); /* this call is mainly for its side effect of reading in the sections. we follow the traditional behavior of `strings' in that we don't complain if we don't recognize a file to be an object file. */ if (!bfd_check_format(abfd, bfd_object)) { bfd_close(abfd); fatal("cannot open executable `%s'", fname); } /* sanity check, endian should be the same as loader.c encountered */ if (abfd->xvec->byteorder_big_p != (unsigned)ld_target_big_endian) panic("binary endian changed"); if ((bfd_get_file_flags(abfd) & (HAS_SYMS|HAS_LOCALS))) { /* file has locals, read them in */ storage = bfd_get_symtab_upper_bound(abfd); if (storage <= 0) fatal("HAS_SYMS is set, but `%s' still lacks symbols", fname); syms = (asymbol **)calloc(storage, 1); if (!syms) fatal("out of virtual memory"); nsyms = bfd_canonicalize_symtab (abfd, syms); if (nsyms <= 0) fatal("HAS_SYMS is set, but `%s' still lacks symbols", fname); /* * convert symbols to local format */ /* first count symbols */ sym_ndatasyms = 0; sym_ntextsyms = 0; for (i=0; i < nsyms; i++) { asymbol *sym = syms[i]; /* decode symbol type */ if (/* from the data section */ (!strcmp(sym->section->name, ".rdata") || !strcmp(sym->section->name, ".data") || !strcmp(sym->section->name, ".sdata") || !strcmp(sym->section->name, ".bss") || !strcmp(sym->section->name, ".sbss")) /* from a scope we are interested in */ && RELEVANT_SCOPE(sym)) { /* data segment symbol */ sym_ndatasyms++;#ifdef PRINT_SYMS fprintf(stderr, "+sym: %s sect: %s flags: %s value: 0x%08lx\n", sym->name, sym->section->name, flags2str(sym->flags), sym->value + sym->section->vma);#endif /* PRINT_SYMS */ } else if (/* from the text section */ !strcmp(sym->section->name, ".text") /* from a scope we are interested in */ && RELEVANT_SCOPE(sym)) { /* text segment symbol */ sym_ntextsyms++;#ifdef PRINT_SYMS fprintf(stderr, "+sym: %s sect: %s flags: %s value: 0x%08lx\n", sym->name, sym->section->name, flags2str(sym->flags), sym->value + sym->section->vma);#endif /* PRINT_SYMS */ } else { /* non-segment sections */#ifdef PRINT_SYMS fprintf(stderr, "-sym: %s sect: %s flags: %s value: 0x%08lx\n", sym->name, sym->section->name, flags2str(sym->flags), sym->value + sym->section->vma);#endif /* PRINT_SYMS */ } } sym_nsyms = sym_ntextsyms + sym_ndatasyms; if (sym_nsyms <= 0) fatal("`%s' has no text or data symbols", fname); /* allocate symbol space */ sym_db = (struct sym_sym_t *)calloc(sym_nsyms, sizeof(struct sym_sym_t)); if (!sym_db) fatal("out of virtual memory"); /* convert symbols to internal format */ for (debug_cnt=0, i=0; i < nsyms; i++) { asymbol *sym = syms[i]; /* decode symbol type */ if (/* from the data section */ (!strcmp(sym->section->name, ".rdata") || !strcmp(sym->section->name, ".data") || !strcmp(sym->section->name, ".sdata") || !strcmp(sym->section->name, ".bss") || !strcmp(sym->section->name, ".sbss")) /* from a scope we are interested in */ && RELEVANT_SCOPE(sym)) { /* data segment symbol, insert into symbol database */ sym_db[debug_cnt].name = mystrdup((char *)sym->name); sym_db[debug_cnt].seg = ss_data; sym_db[debug_cnt].initialized = (!strcmp(sym->section->name, ".rdata") || !strcmp(sym->section->name, ".data") || !strcmp(sym->section->name, ".sdata")); sym_db[debug_cnt].pub = (sym->flags & BSF_GLOBAL); sym_db[debug_cnt].local = (sym->name[0] == '$'); sym_db[debug_cnt].addr = sym->value + sym->section->vma; debug_cnt++; } else if (/* from the text section */ !strcmp(sym->section->name, ".text") /* from a scope we are interested in */ && RELEVANT_SCOPE(sym)) { /* text segment symbol, insert into symbol database */ sym_db[debug_cnt].name = mystrdup((char *)sym->name); sym_db[debug_cnt].seg = ss_text; sym_db[debug_cnt].initialized = /* seems reasonable */TRUE; sym_db[debug_cnt].pub = (sym->flags & BSF_GLOBAL); sym_db[debug_cnt].local = (sym->name[0] == '$'); sym_db[debug_cnt].addr = sym->value + sym->section->vma; debug_cnt++; } else { /* non-segment sections */ } } /* sanity check */ if (debug_cnt != sym_nsyms) panic("could not locate all counted symbols"); /* release bfd symbol storage */ free(syms); } /* done with file, close if */ if (!bfd_close(abfd)) fatal("could not close executable `%s'", fname);#else /* !BFD_LOADER */ /* load the program into memory, try both endians */#if defined(_MSC_VER) fobj = fopen(fname, "rb");#else fobj = fopen(fname, "r");#endif if (!fobj) fatal("cannot open executable `%s'", fname); if (fread(&fhdr, sizeof(struct ecoff_filehdr), 1, fobj) < 1) fatal("cannot read header from executable `%s'", fname); /* record endian of target */ if (fhdr.f_magic != ECOFF_ALPHAMAGIC) fatal("bad magic number in executable `%s'", fname);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -