/* ----------------------------------------------------------------------------- * $Id: Printer.c,v 1.51 2002/02/04 20:26:25 sof Exp $ * * (c) The GHC Team, 1994-2000. * * Heap printer * * ---------------------------------------------------------------------------*/ #include "PosixSource.h" #include "Rts.h" #include "Printer.h" #ifdef DEBUG #include "RtsUtils.h" #include "RtsFlags.h" #include "MBlock.h" #include "Storage.h" #include "Bytecodes.h" /* for InstrPtr */ #include "Disassembler.h" #include "Printer.h" #if defined(GRAN) || defined(PAR) // HWL: explicit fixed header size to make debugging easier int fixed_hs = FIXED_HS, itbl_sz = sizeofW(StgInfoTable), uf_sz=sizeofW(StgUpdateFrame), sf_sz=sizeofW(StgSeqFrame); #endif /* -------------------------------------------------------------------------- * local function decls * ------------------------------------------------------------------------*/ static void printStdObject( StgClosure *obj, char* tag ); static void reset_table ( int size ); static void prepare_table ( void ); static void insert ( unsigned value, const char *name ); #if 0 /* unused but might be useful sometime */ static rtsBool lookup_name ( char *name, unsigned *result ); static void enZcode ( char *in, char *out ); #endif static char unZcode ( char ch ); const char * lookupGHCName ( void *addr ); static void printZcoded ( const char *raw ); /* -------------------------------------------------------------------------- * Printer * ------------------------------------------------------------------------*/ void printPtr( StgPtr p ) { const char *raw; raw = lookupGHCName(p); if (raw != NULL) { printZcoded(raw); } else { fprintf(stdout, "%p", p); } } void printObj( StgClosure *obj ) { fprintf(stdout,"Object "); printPtr((StgPtr)obj); fprintf(stdout," = "); printClosure(obj); } static inline void printStdObjHdr( StgClosure *obj, char* tag ) { fprintf(stdout,"%s(",tag); printPtr((StgPtr)obj->header.info); #ifdef PROFILING fprintf(stdout,", %s", obj->header.prof.ccs->cc->label); #endif } static void printStdObject( StgClosure *obj, char* tag ) { StgWord i, j; const StgInfoTable* info; printStdObjHdr( obj, tag ); info = get_itbl(obj); for (i = 0; i < info->layout.payload.ptrs; ++i) { fprintf(stdout,", "); printPtr((StgPtr)obj->payload[i]); } for (j = 0; j < info->layout.payload.nptrs; ++j) { fprintf(stdout,", %pd#",obj->payload[i+j]); } fprintf(stdout,")\n"); } void printClosure( StgClosure *obj ) { StgInfoTable *info; info = get_itbl(obj); switch ( info->type ) { case INVALID_OBJECT: barf("Invalid object"); case BCO: disassemble( (StgBCO*)obj ); break; case MUT_VAR: { StgMutVar* mv = (StgMutVar*)obj; fprintf(stdout,"MUT_VAR(var=%p, link=%p)\n", mv->var, mv->mut_link); break; } case AP_UPD: { StgAP_UPD* ap = stgCast(StgAP_UPD*,obj); StgWord i; fprintf(stdout,"AP_UPD("); printPtr((StgPtr)ap->fun); for (i = 0; i < ap->n_args; ++i) { fprintf(stdout,", "); printPtr((P_)ap->payload[i]); } fprintf(stdout,")\n"); break; } case PAP: { StgPAP* pap = stgCast(StgPAP*,obj); StgWord i; fprintf(stdout,"PAP("); printPtr((StgPtr)pap->fun); for (i = 0; i < pap->n_args; ++i) { fprintf(stdout,", "); printPtr((StgPtr)pap->payload[i]); } fprintf(stdout,")\n"); break; } case FOREIGN: fprintf(stdout,"FOREIGN("); printPtr((StgPtr)( ((StgForeignObj*)obj)->data )); fprintf(stdout,")\n"); break; case IND: fprintf(stdout,"IND("); printPtr((StgPtr)stgCast(StgInd*,obj)->indirectee); fprintf(stdout,")\n"); break; case IND_STATIC: fprintf(stdout,"IND_STATIC("); printPtr((StgPtr)stgCast(StgInd*,obj)->indirectee); fprintf(stdout,")\n"); break; case IND_OLDGEN: fprintf(stdout,"IND_OLDGEN("); printPtr((StgPtr)stgCast(StgInd*,obj)->indirectee); fprintf(stdout,")\n"); break; case CAF_BLACKHOLE: fprintf(stdout,"CAF_BH("); printPtr((StgPtr)stgCast(StgBlockingQueue*,obj)->blocking_queue); fprintf(stdout,")\n"); break; case SE_BLACKHOLE: fprintf(stdout,"SE_BH\n"); break; case SE_CAF_BLACKHOLE: fprintf(stdout,"SE_CAF_BH\n"); break; case BLACKHOLE: fprintf(stdout,"BH\n"); break; case BLACKHOLE_BQ: fprintf(stdout,"BQ("); printPtr((StgPtr)stgCast(StgBlockingQueue*,obj)->blocking_queue); fprintf(stdout,")\n"); break; case TSO: fprintf(stdout,"TSO("); fprintf(stdout,"%d (%p)",((StgTSO*)obj)->id, (StgTSO*)obj); fprintf(stdout,")\n"); break; #if defined(PAR) case BLOCKED_FETCH: fprintf(stdout,"BLOCKED_FETCH("); printGA(&(stgCast(StgBlockedFetch*,obj)->ga)); printPtr((StgPtr)(stgCast(StgBlockedFetch*,obj)->node)); fprintf(stdout,")\n"); break; case FETCH_ME: fprintf(stdout,"FETCH_ME("); printGA((globalAddr *)stgCast(StgFetchMe*,obj)->ga); fprintf(stdout,")\n"); break; #ifdef DIST case REMOTE_REF: fprintf(stdout,"REMOTE_REF("); printGA((globalAddr *)stgCast(StgFetchMe*,obj)->ga); fprintf(stdout,")\n"); break; #endif case FETCH_ME_BQ: fprintf(stdout,"FETCH_ME_BQ("); // printGA((globalAddr *)stgCast(StgFetchMe*,obj)->ga); printPtr((StgPtr)stgCast(StgFetchMeBlockingQueue*,obj)->blocking_queue); fprintf(stdout,")\n"); break; #endif #if defined(GRAN) || defined(PAR) case RBH: fprintf(stdout,"RBH("); printPtr((StgPtr)stgCast(StgRBH*,obj)->blocking_queue); fprintf(stdout,")\n"); break; #endif case CONSTR: case CONSTR_1_0: case CONSTR_0_1: case CONSTR_1_1: case CONSTR_0_2: case CONSTR_2_0: case CONSTR_INTLIKE: case CONSTR_CHARLIKE: case CONSTR_STATIC: case CONSTR_NOCAF_STATIC: { /* We can't use printStdObject because we want to print the * tag as well. */ StgWord i, j; #ifdef PROFILING fprintf(stdout,"%s(", info->prof.closure_desc); fprintf(stdout,"%s", obj->header.prof.ccs->cc->label); #else fprintf(stdout,"CONSTR("); printPtr((StgPtr)obj->header.info); fprintf(stdout,"(tag=%d)",info->srt_len); #endif for (i = 0; i < info->layout.payload.ptrs; ++i) { fprintf(stdout,", "); printPtr((StgPtr)obj->payload[i]); } for (j = 0; j < info->layout.payload.nptrs; ++j) { fprintf(stdout,", %p#", obj->payload[i+j]); } fprintf(stdout,")\n"); break; } #ifdef XMLAMBDA /* rows are mutarrays in xmlambda, maybe we should make a new type: ROW */ case MUT_ARR_PTRS_FROZEN: { StgWord i; StgMutArrPtrs* p = stgCast(StgMutArrPtrs*,obj); fprintf(stdout,"Row<%i>(",p->ptrs); for (i = 0; i < p->ptrs; ++i) { if (i > 0) fprintf(stdout,", "); printPtr((StgPtr)(p->payload[i])); } fprintf(stdout,")\n"); break; } #endif case FUN: case FUN_1_0: case FUN_0_1: case FUN_1_1: case FUN_0_2: case FUN_2_0: case FUN_STATIC: printStdObject(obj,"FUN"); break; case THUNK: case THUNK_1_0: case THUNK_0_1: case THUNK_1_1: case THUNK_0_2: case THUNK_2_0: case THUNK_STATIC: /* ToDo: will this work for THUNK_STATIC too? */ #ifdef PROFILING printStdObject(obj,info->prof.closure_desc); #else printStdObject(obj,"THUNK"); #endif break; case THUNK_SELECTOR: printStdObjHdr(obj, "THUNK_SELECTOR"); fprintf(stdout, ", %p)\n", ((StgSelector *)obj)->selectee); break; case ARR_WORDS: { StgWord i; fprintf(stdout,"ARR_WORDS(\""); /* ToDo: we can't safely assume that this is a string! for (i = 0; arrWordsGetChar(obj,i); ++i) { putchar(arrWordsGetChar(obj,i)); } */ for (i=0; i<((StgArrWords *)obj)->words; i++) fprintf(stdout, "%u", ((StgArrWords *)obj)->payload[i]); fprintf(stdout,"\")\n"); break; } case UPDATE_FRAME: { StgUpdateFrame* u = stgCast(StgUpdateFrame*,obj); fprintf(stdout,"UpdateFrame("); printPtr((StgPtr)GET_INFO(u)); fprintf(stdout,","); printPtr((StgPtr)u->updatee); fprintf(stdout,","); printPtr((StgPtr)u->link); fprintf(stdout,")\n"); break; } case CATCH_FRAME: { StgCatchFrame* u = stgCast(StgCatchFrame*,obj); fprintf(stdout,"CatchFrame("); printPtr((StgPtr)GET_INFO(u)); fprintf(stdout,","); printPtr((StgPtr)u->handler); fprintf(stdout,","); printPtr((StgPtr)u->link); fprintf(stdout,")\n"); break; } case SEQ_FRAME: { StgSeqFrame* u = stgCast(StgSeqFrame*,obj); fprintf(stdout,"SeqFrame("); printPtr((StgPtr)GET_INFO(u)); fprintf(stdout,","); printPtr((StgPtr)u->link); fprintf(stdout,")\n"); break; } case STOP_FRAME: { StgStopFrame* u = stgCast(StgStopFrame*,obj); fprintf(stdout,"StopFrame("); printPtr((StgPtr)GET_INFO(u)); fprintf(stdout,")\n"); break; } default: //barf("printClosure %d",get_itbl(obj)->type); fprintf(stdout, "*** printClosure: unknown type %d ****\n", get_itbl(obj)->type ); barf("printClosure %d",get_itbl(obj)->type); return; } } /* void printGraph( StgClosure *obj ) { printClosure(obj); } */ StgPtr printStackObj( StgPtr sp ) { /*fprintf(stdout,"Stack[%d] = ", &stgStack[STACK_SIZE] - sp); */ if (IS_ARG_TAG(*sp)) { nat i; StgWord tag = *sp++; fprintf(stdout,"Tagged{"); for (i = 0; i < tag; i++) { fprintf(stdout,"0x%x#", (unsigned)(*sp++)); if (i < tag-1) fprintf(stdout, ", "); } fprintf(stdout, "}\n"); } else { StgClosure* c = (StgClosure*)(*sp); printPtr((StgPtr)*sp); if (c == (StgClosure*)&stg_ctoi_ret_R1p_info) { fprintf(stdout, "\t\t\tstg_ctoi_ret_R1p_info\n" ); } else if (c == (StgClosure*)&stg_ctoi_ret_R1n_info) { fprintf(stdout, "\t\t\tstg_ctoi_ret_R1n_info\n" ); } else if (c == (StgClosure*)&stg_ctoi_ret_F1_info) { fprintf(stdout, "\t\t\tstg_ctoi_ret_F1_info\n" ); } else if (c == (StgClosure*)&stg_ctoi_ret_D1_info) { fprintf(stdout, "\t\t\tstg_ctoi_ret_D1_info\n" ); } else if (c == (StgClosure*)&stg_ctoi_ret_V_info) { fprintf(stdout, "\t\t\tstg_ctoi_ret_V_info\n" ); } else if (get_itbl(c)->type == BCO) { fprintf(stdout, "\t\t\t"); fprintf(stdout, "BCO(...)\n"); } else { fprintf(stdout, "\t\t\t"); printClosure ( (StgClosure*)(*sp)); } sp += 1; } return sp; } void printStackChunk( StgPtr sp, StgPtr spBottom ) { StgWord bitmap; const StgInfoTable *info; ASSERT(sp <= spBottom); while (sp < spBottom) { if (!IS_ARG_TAG(*sp) && LOOKS_LIKE_GHC_INFO(*sp)) { info = get_itbl((StgClosure *)sp); switch (info->type) { case UPDATE_FRAME: printObj( stgCast(StgClosure*,sp) ); sp += sizeofW(StgUpdateFrame); continue; case SEQ_FRAME: printObj( stgCast(StgClosure*,sp) ); sp += sizeofW(StgSeqFrame); continue; case CATCH_FRAME: printObj( stgCast(StgClosure*,sp) ); sp += sizeofW(StgCatchFrame); continue; case STOP_FRAME: /* not quite: ASSERT(stgCast(StgPtr,su) == spBottom); */ printObj( stgCast(StgClosure*,sp) ); continue; case RET_DYN: fprintf(stdout, "RET_DYN (%p)\n", sp); bitmap = *++sp; ++sp; fprintf(stdout, "Bitmap: 0x%x\n", bitmap); goto small_bitmap; case RET_SMALL: case RET_VEC_SMALL: fprintf(stdout, "RET_SMALL (%p)\n", sp); bitmap = info->layout.bitmap; sp++; small_bitmap: while (bitmap != 0) { fprintf(stdout," stk[%ld] (%p) = ", spBottom-sp, sp); if ((bitmap & 1) == 0) { printPtr((P_)*sp); fprintf(stdout,"\n"); } else { fprintf(stdout,"Word# %ld\n", *sp); } sp++; bitmap = bitmap >> 1; } continue; case RET_BIG: case RET_VEC_BIG: barf("todo"); default: break; } } fprintf(stdout,"Stack[%ld] (%p) = ", spBottom-sp, sp); sp = printStackObj(sp); } } void printStack( StgPtr sp, StgPtr spBottom, StgUpdateFrame* su ) { /* check everything down to the first update frame */ printStackChunk( sp, stgCast(StgPtr,su) ); while ( stgCast(StgPtr,su) < spBottom) { sp = stgCast(StgPtr,su); switch (get_itbl(su)->type) { case UPDATE_FRAME: printObj( stgCast(StgClosure*,su) ); sp += sizeofW(StgUpdateFrame); su = su->link; break; case SEQ_FRAME: printObj( stgCast(StgClosure*,su) ); sp += sizeofW(StgSeqFrame); su = stgCast(StgSeqFrame*,su)->link; break; case CATCH_FRAME: printObj( stgCast(StgClosure*,su) ); sp += sizeofW(StgCatchFrame); su = stgCast(StgCatchFrame*,su)->link; break; case STOP_FRAME: /* not quite: ASSERT(stgCast(StgPtr,su) == spBottom); */ printObj( stgCast(StgClosure*,su) ); return; default: barf("printStack: weird record found on update frame list."); } printStackChunk( sp, stgCast(StgPtr,su) ); } ASSERT(stgCast(StgPtr,su) == spBottom); } void printTSO( StgTSO *tso ) { printStack( tso->sp, tso->stack+tso->stack_size,tso->su); /* printStackChunk( tso->sp, tso->stack+tso->stack_size); */ } /* ----------------------------------------------------------------------------- Closure types NOTE: must be kept in sync with the closure types in includes/ClosureTypes.h -------------------------------------------------------------------------- */ static char *closure_type_names[] = { "INVALID_OBJECT", /* 0 */ "CONSTR", /* 1 */ "CONSTR_1_0", /* 2 */ "CONSTR_0_1", /* 3 */ "CONSTR_2_0", /* 4 */ "CONSTR_1_1", /* 5 */ "CONSTR_0_2", /* 6 */ "CONSTR_INTLIKE", /* 7 */ "CONSTR_CHARLIKE", /* 8 */ "CONSTR_STATIC", /* 9 */ "CONSTR_NOCAF_STATIC", /* 10 */ "FUN", /* 11 */ "FUN_1_0", /* 12 */ "FUN_0_1", /* 13 */ "FUN_2_0", /* 14 */ "FUN_1_1", /* 15 */ "FUN_0_2", /* 16 */ "FUN_STATIC", /* 17 */ "THUNK", /* 18 */ "THUNK_1_0", /* 19 */ "THUNK_0_1", /* 20 */ "THUNK_2_0", /* 21 */ "THUNK_1_1", /* 22 */ "THUNK_0_2", /* 23 */ "THUNK_STATIC", /* 24 */ "THUNK_SELECTOR", /* 25 */ "BCO", /* 26 */ "AP_UPD", /* 27 */ "PAP", /* 28 */ "IND", /* 29 */ "IND_OLDGEN", /* 30 */ "IND_PERM", /* 31 */ "IND_OLDGEN_PERM", /* 32 */ "IND_STATIC", /* 33 */ "CAF_BLACKHOLE", /* 36 */ "RET_BCO", /* 37 */ "RET_SMALL", /* 38 */ "RET_VEC_SMALL", /* 39 */ "RET_BIG", /* 40 */ "RET_VEC_BIG", /* 41 */ "RET_DYN", /* 42 */ "UPDATE_FRAME", /* 43 */ "CATCH_FRAME", /* 44 */ "STOP_FRAME", /* 45 */ "SEQ_FRAME", /* 46 */ "BLACKHOLE", /* 47 */ "BLACKHOLE_BQ", /* 48 */ "SE_BLACKHOLE", /* 49 */ "SE_CAF_BLACKHOLE", /* 50 */ "MVAR", /* 51 */ "ARR_WORDS", /* 52 */ "MUT_ARR_PTRS", /* 53 */ "MUT_ARR_PTRS_FROZEN", /* 54 */ "MUT_VAR", /* 55 */ "WEAK", /* 56 */ "FOREIGN", /* 57 */ "STABLE_NAME", /* 58 */ "TSO", /* 59 */ "BLOCKED_FETCH", /* 60 */ "FETCH_ME", /* 61 */ "FETCH_ME_BQ", /* 62 */ "RBH", /* 63 */ "EVACUATED", /* 64 */ "REMOTE_REF", /* 65 */ "N_CLOSURE_TYPES" /* 66 */ }; char * info_type(StgClosure *closure){ return closure_type_names[get_itbl(closure)->type]; } char * info_type_by_ip(StgInfoTable *ip){ return closure_type_names[ip->type]; } void info_hdr_type(StgClosure *closure, char *res){ strcpy(res,closure_type_names[get_itbl(closure)->type]); } /* -------------------------------------------------------------------------- * Address printing code * * Uses symbol table in (unstripped executable) * ------------------------------------------------------------------------*/ /* -------------------------------------------------------------------------- * Simple lookup table * * Current implementation is pretty dumb! * ------------------------------------------------------------------------*/ struct entry { nat value; const char *name; }; static nat max_table_size; static nat table_size; static struct entry* table; static void reset_table( int size ) { max_table_size = size; table_size = 0; table = (struct entry *) malloc(size * sizeof(struct entry)); } static void prepare_table( void ) { /* Could sort it... */ } static void insert( unsigned value, const char *name ) { if ( table_size >= max_table_size ) { barf( "Symbol table overflow\n" ); } table[table_size].value = value; table[table_size].name = name; table_size = table_size + 1; } #if 0 static rtsBool lookup_name( char *name, unsigned *result ) { int i; for( i = 0; i < table_size && strcmp(name,table[i].name) != 0; ++i ) { } if (i < table_size) { *result = table[i].value; return rtsTrue; } else { return rtsFalse; } } #endif /* Code from somewhere inside GHC (circa 1994) * * Z-escapes: * "std"++xs -> "Zstd"++xs * char_to_c 'Z' = "ZZ" * char_to_c '&' = "Za" * char_to_c '|' = "Zb" * char_to_c ':' = "Zc" * char_to_c '/' = "Zd" * char_to_c '=' = "Ze" * char_to_c '>' = "Zg" * char_to_c '#' = "Zh" * char_to_c '<' = "Zl" * char_to_c '-' = "Zm" * char_to_c '!' = "Zn" * char_to_c '.' = "Zo" * char_to_c '+' = "Zp" * char_to_c '\'' = "Zq" * char_to_c '*' = "Zt" * char_to_c '_' = "Zu" * char_to_c c = "Z" ++ show (ord c) */ static char unZcode( char ch ) { switch (ch) { case 'a' : return ('&'); case 'b' : return ('|'); case 'c' : return (':'); case 'd' : return ('/'); case 'e' : return ('='); case 'g' : return ('>'); case 'h' : return ('#'); case 'l' : return ('<'); case 'm' : return ('-'); case 'n' : return ('!'); case 'o' : return ('.'); case 'p' : return ('+'); case 'q' : return ('\''); case 't' : return ('*'); case 'u' : return ('_'); case 'Z' : case '\0' : return ('Z'); default : return (ch); } } #if 0 /* Precondition: out big enough to handle output (about twice length of in) */ static void enZcode( char *in, char *out ) { int i, j; j = 0; out[ j++ ] = '_'; for( i = 0; in[i] != '\0'; ++i ) { switch (in[i]) { case 'Z' : out[j++] = 'Z'; out[j++] = 'Z'; break; case '&' : out[j++] = 'Z'; out[j++] = 'a'; break; case '|' : out[j++] = 'Z'; out[j++] = 'b'; break; case ':' : out[j++] = 'Z'; out[j++] = 'c'; break; case '/' : out[j++] = 'Z'; out[j++] = 'd'; break; case '=' : out[j++] = 'Z'; out[j++] = 'e'; break; case '>' : out[j++] = 'Z'; out[j++] = 'g'; break; case '#' : out[j++] = 'Z'; out[j++] = 'h'; break; case '<' : out[j++] = 'Z'; out[j++] = 'l'; break; case '-' : out[j++] = 'Z'; out[j++] = 'm'; break; case '!' : out[j++] = 'Z'; out[j++] = 'n'; break; case '.' : out[j++] = 'Z'; out[j++] = 'o'; break; case '+' : out[j++] = 'Z'; out[j++] = 'p'; break; case '\'' : out[j++] = 'Z'; out[j++] = 'q'; break; case '*' : out[j++] = 'Z'; out[j++] = 't'; break; case '_' : out[j++] = 'Z'; out[j++] = 'u'; break; default : out[j++] = in[i]; break; } } out[j] = '\0'; } #endif const char *lookupGHCName( void *addr ) { nat i; for( i = 0; i < table_size && table[i].value != (unsigned) addr; ++i ) { } if (i < table_size) { return table[i].name; } else { return NULL; } } static void printZcoded( const char *raw ) { nat j = 0; while ( raw[j] != '\0' ) { if (raw[j] == 'Z') { fputc(unZcode(raw[j+1]),stdout); j = j + 2; } else { fputc(raw[j],stdout); j = j + 1; } } } /* -------------------------------------------------------------------------- * Symbol table loading * ------------------------------------------------------------------------*/ /* Causing linking trouble on Win32 plats, so I'm disabling this for now. */ #ifdef USING_LIBBFD #include /* Fairly ad-hoc piece of code that seems to filter out a lot of * rubbish like the obj-splitting symbols */ static rtsBool isReal( flagword flags, const char *name ) { #if 0 /* ToDo: make this work on BFD */ int tp = type & N_TYPE; if (tp == N_TEXT || tp == N_DATA) { return (name[0] == '_' && name[1] != '_'); } else { return rtsFalse; } #else (void)flags; /* keep gcc -Wall happy */ if (*name == '\0' || (name[0] == 'g' && name[1] == 'c' && name[2] == 'c') || (name[0] == 'c' && name[1] == 'c' && name[2] == '.')) { return rtsFalse; } return rtsTrue; #endif } extern void DEBUG_LoadSymbols( char *name ) { bfd* abfd; char **matching; bfd_init(); abfd = bfd_openr(name, "default"); if (abfd == NULL) { barf("can't open executable %s to get symbol table", name); } if (!bfd_check_format_matches (abfd, bfd_object, &matching)) { barf("mismatch"); } { long storage_needed; asymbol **symbol_table; long number_of_symbols; long num_real_syms = 0; long i; storage_needed = bfd_get_symtab_upper_bound (abfd); if (storage_needed < 0) { barf("can't read symbol table"); } #if 0 if (storage_needed == 0) { belch("no storage needed"); } #endif symbol_table = (asymbol **) stgMallocBytes(storage_needed,"DEBUG_LoadSymbols"); number_of_symbols = bfd_canonicalize_symtab (abfd, symbol_table); if (number_of_symbols < 0) { barf("can't canonicalise symbol table"); } for( i = 0; i != number_of_symbols; ++i ) { symbol_info info; bfd_get_symbol_info(abfd,symbol_table[i],&info); /*fprintf(stdout,"\t%c\t0x%x \t%s\n",info.type,(nat)info.value,info.name); */ if (isReal(info.type, info.name)) { num_real_syms += 1; } } IF_DEBUG(evaluator, fprintf(stdout,"Loaded %ld symbols. Of which %ld are real symbols\n", number_of_symbols, num_real_syms) ); reset_table( num_real_syms ); for( i = 0; i != number_of_symbols; ++i ) { symbol_info info; bfd_get_symbol_info(abfd,symbol_table[i],&info); if (isReal(info.type, info.name)) { insert( info.value, info.name ); } } free(symbol_table); } prepare_table(); } #else /* HAVE_BFD_H */ extern void DEBUG_LoadSymbols( char *name STG_UNUSED ) { /* nothing, yet */ } #endif /* HAVE_BFD_H */ #include "StoragePriv.h" void findPtr(P_ p, int); /* keep gcc -Wall happy */ void findPtr(P_ p, int follow) { nat s, g; P_ q, r; bdescr *bd; const int arr_size = 1024; StgPtr arr[arr_size]; int i = 0; for (g = 0; g < RtsFlags.GcFlags.generations; g++) { for (s = 0; s < generations[g].n_steps; s++) { if (RtsFlags.GcFlags.generations == 1) { bd = generations[g].steps[s].to_blocks; } else { bd = generations[g].steps[s].blocks; } for (; bd; bd = bd->link) { for (q = bd->start; q < bd->free; q++) { if (*q == (W_)p) { if (i < arr_size) { r = q; while (!LOOKS_LIKE_GHC_INFO(*r) || *r == NULL) { r--; } fprintf(stdout, "%p = ", r); printClosure((StgClosure *)r); arr[i++] = r; } else { return; } } } } } } if (follow && i == 1) { fprintf(stdout, "-->\n"); findPtr(arr[0], 1); } } #else /* DEBUG */ void printPtr( StgPtr p ) { fprintf(stdout, "ptr 0x%p (enable -DDEBUG for more info) " , p ); } void printObj( StgClosure *obj ) { fprintf(stdout, "obj 0x%p (enable -DDEBUG for more info) " , obj ); } #endif /* DEBUG */