1
0
mirror of https://github.com/cc65/cc65.git synced 2024-12-25 17:29:50 +00:00

Replaced the old dbgtest program by a debug info shell.

git-svn-id: svn://svn.cc65.org/cc65/trunk@5188 b7a2c559-68d2-44c3-8de9-860c34a00d81
This commit is contained in:
uz 2011-08-16 22:37:01 +00:00
parent 7d7667f782
commit 771695577d
5 changed files with 455 additions and 330 deletions

View File

@ -1,6 +1,6 @@
/*****************************************************************************/
/* */
/* dbginfo.h */
/* dbginfo.c */
/* */
/* cc65 debug info handling */
/* */
@ -340,6 +340,7 @@ struct SymInfo {
unsigned Id; /* Parent symbol if any */
SymInfo* Info; /* Pointer to parent symbol if any */
} Parent;
Collection* ImportList; /* List of imports if this is an export */
char Name[1]; /* Name of symbol */
};
@ -983,27 +984,6 @@ static void DBGPRINT(const char* format, ...) {}
/*****************************************************************************/
/* Id lists */
/*****************************************************************************/
static void init_cc65_idlist (cc65_idlist* L, unsigned Count)
/* Initialize an idlist with the given count. The count field in the list
* will be set on return and memory for the list is allocated.
*/
{
L->count = Count;
if (Count == 0) {
L->ids = 0;
} else {
L->ids = xmalloc (Count * sizeof (L->ids[0]));
}
}
/*****************************************************************************/
/* File info */
/*****************************************************************************/
@ -1187,18 +1167,11 @@ static cc65_lineinfo* new_cc65_lineinfo (unsigned Count)
static void CopyLineInfo (cc65_linedata* D, const LineInfo* L)
/* Copy data from a LineInfo struct to a cc65_linedata struct */
{
unsigned I;
D->line_id = L->Id;
D->source_id = L->File.Info->Id;
D->source_line = L->Line;
D->line_type = L->Type;
D->count = L->Count;
init_cc65_idlist (&D->span_list, CollCount (&L->SpanInfoList));
for (I = 0; I < CollCount (&L->SpanInfoList); ++I) {
const SpanInfo* S = CollConstAt (&L->SpanInfoList, I);
D->span_list.ids[I] = S->Id;
}
}
@ -1535,7 +1508,8 @@ static SymInfo* NewSymInfo (const StrBuf* Name)
/* Allocate memory */
SymInfo* S = xmalloc (sizeof (SymInfo) + SB_GetLen (Name));
/* Initialize the name */
/* Initialize it as necessary */
S->ImportList = 0;
memcpy (S->Name, SB_GetConstBuf (Name), SB_GetLen (Name) + 1);
/* Return it */
@ -1547,6 +1521,7 @@ static SymInfo* NewSymInfo (const StrBuf* Name)
static void FreeSymInfo (SymInfo* S)
/* Free a SymInfo struct */
{
CollFree (S->ImportList);
xfree (S);
}
@ -3459,7 +3434,7 @@ static void ParseSym (InputData* D)
unsigned ScopeId = CC65_INV_ID;
unsigned SegId = CC65_INV_ID;
cc65_size Size = 0;
cc65_symbol_type Type = CC65_SYM_EQUATE;
cc65_symbol_type Type = CC65_SYM_EQUATE;
long Value = 0;
SymInfo* S;
@ -4015,7 +3990,7 @@ static void ProcessFileInfo (InputData* D)
static void ProcessLineInfo (InputData* D)
/* Postprocess line infos */
{
unsigned I;
unsigned I, J;
/* Get pointers to the collections */
Collection* LineInfos = &D->Info->LineInfoById;
@ -4023,7 +3998,8 @@ static void ProcessLineInfo (InputData* D)
/* Walk over the line infos and replace the id numbers of file and segment
* with pointers to the actual structs. Add the line info to each file
* where it is defined.
* where it is defined. Resolve the spans and add backpointers to the
* spans.
*/
for (I = 0; I < CollCount (LineInfos); ++I) {
@ -4044,8 +4020,32 @@ static void ProcessLineInfo (InputData* D)
CollAppend (&L->File.Info->LineInfoByLine, L);
}
/* Next one */
++I;
/* Resolve the spans ids */
for (J = 0; J < CollCount (&L->SpanInfoList); ++J) {
/* Get the id of this span */
unsigned SpanId = CollIdAt (&L->SpanInfoList, J);
if (SpanId >= CollCount (&D->Info->SpanInfoById)) {
ParseError (D,
CC65_ERROR,
"Invalid span id %u for line with id %u",
SpanId, L->Id);
CollReplace (&L->SpanInfoList, 0, J);
} else {
/* Get a pointer to the span */
SpanInfo* SP = CollAt (&D->Info->SpanInfoById, SpanId);
/* Replace the id by the pointer */
CollReplace (&L->SpanInfoList, SP, J);
/* Insert a backpointer into the span */
if (SP->LineInfoList == 0) {
SP->LineInfoList = CollNew ();
}
CollAppend (SP->LineInfoList, L);
}
}
}
/* Walk over all files and sort the line infos for each file so we can
@ -4295,6 +4295,12 @@ static void ProcessSymInfo (InputData* D)
S->Exp.Info = 0;
} else {
S->Exp.Info = CollAt (&D->Info->SymInfoById, S->Exp.Id);
/* Add a backpointer, so the export knows its imports */
if (S->Exp.Info->ImportList == 0) {
S->Exp.Info->ImportList = CollNew ();
}
CollAppend (S->Exp.Info->ImportList, S);
}
/* Resolve segment */
@ -4898,6 +4904,84 @@ const cc65_spaninfo* cc65_spaninfo_byaddr (cc65_dbginfo Handle, unsigned long Ad
const cc65_spaninfo* cc65_spaninfo_byline (cc65_dbginfo Handle, unsigned LineId)
/* Return span information for the given source line. The function returns NULL
* if the line id is invalid, otherwise the spans for this line (possibly zero).
*/
{
DbgInfo* Info;
LineInfo* L;
cc65_spaninfo* D;
unsigned I;
/* Check the parameter */
assert (Handle != 0);
/* The handle is actually a pointer to a debug info struct */
Info = (DbgInfo*) Handle;
/* Check if the line id is valid */
if (LineId >= CollCount (&Info->LineInfoById)) {
return 0;
}
/* Get the line with this id */
L = CollAt (&Info->LineInfoById, LineId);
/* Allocate memory for the data structure returned to the caller */
D = new_cc65_spaninfo (CollCount (&L->SpanInfoList));
/* Fill in the data */
for (I = 0; I < CollCount (&L->SpanInfoList); ++I) {
/* Copy the data */
CopySpanInfo (D->data + I, CollConstAt (&L->SpanInfoList, I));
}
/* Return the result */
return D;
}
const cc65_spaninfo* cc65_spaninfo_byscope (cc65_dbginfo Handle, unsigned ScopeId)
/* Return span information for the given scope. The function returns NULL if
* the scope id is invalid, otherwise the spans for this scope (possibly zero).
*/
{
DbgInfo* Info;
ScopeInfo* S;
cc65_spaninfo* D;
unsigned I;
/* Check the parameter */
assert (Handle != 0);
/* The handle is actually a pointer to a debug info struct */
Info = (DbgInfo*) Handle;
/* Check if the scope id is valid */
if (ScopeId >= CollCount (&Info->ScopeInfoById)) {
return 0;
}
/* Get the scope with this id */
S = CollAt (&Info->ScopeInfoById, ScopeId);
/* Allocate memory for the data structure returned to the caller */
D = new_cc65_spaninfo (CollCount (&S->SpanInfoList));
/* Fill in the data */
for (I = 0; I < CollCount (&S->SpanInfoList); ++I) {
/* Copy the data */
CopySpanInfo (D->data + I, CollConstAt (&S->SpanInfoList, I));
}
/* Return the result */
return D;
}
void cc65_free_spaninfo (cc65_dbginfo Handle, const cc65_spaninfo* Info)
/* Free a span info record */
{
@ -5249,8 +5333,8 @@ const cc65_symbolinfo* cc65_symbol_inrange (cc65_dbginfo Handle, cc65_addr Start
break;
}
/* Ignore non-labels */
if (Item->Type != CC65_SYM_LABEL) {
/* Ignore non-labels and imports */
if (Item->Type != CC65_SYM_LABEL || Item->Exp.Info != 0) {
continue;
}

View File

@ -61,13 +61,6 @@ typedef unsigned cc65_size; /* Used to store (65xx) sizes */
/* A value that is used to mark invalid ids */
#define CC65_INV_ID (~0U)
/* A structure that is used to store a list of ids */
typedef struct cc65_idlist cc65_idlist;
struct cc65_idlist {
unsigned count; /* Number of elements */
unsigned* ids; /* List of ids, number is dynamic */
};
/*****************************************************************************/
@ -93,7 +86,7 @@ struct cc65_parseerror {
};
/* Function that is called in case of parse errors */
typedef void (*cc65_errorfunc) (const struct cc65_parseerror*);
typedef void (*cc65_errorfunc) (const cc65_parseerror*);
/* Pointer to an opaque data structure containing information from the debug
* info file. Actually a handle to the data in the file.
@ -172,7 +165,6 @@ struct cc65_linedata {
cc65_line source_line; /* Line number */
cc65_line_type line_type; /* Type of line */
unsigned count; /* Nesting counter for macros */
cc65_idlist span_list; /* List of spans for this line */
};
typedef struct cc65_lineinfo cc65_lineinfo;
@ -273,6 +265,18 @@ const cc65_spaninfo* cc65_spaninfo_byaddr (cc65_dbginfo handle,
* if no spans were found for this address.
*/
const cc65_spaninfo* cc65_spaninfo_byline (cc65_dbginfo handle,
unsigned line_id);
/* Return span information for the given source line. The function returns NULL
* if the line id is invalid, otherwise the spans for this line (possibly zero).
*/
const cc65_spaninfo* cc65_spaninfo_byscope (cc65_dbginfo handle,
unsigned scope_id);
/* Return span information for the given scope. The function returns NULL if
* the scope id is invalid, otherwise the spans for this scope (possibly zero).
*/
void cc65_free_spaninfo (cc65_dbginfo handle, const cc65_spaninfo* info);
/* Free a span info record */
@ -381,7 +385,7 @@ typedef enum {
} cc65_symbol_type;
/* Notes:
* - If the symbol is segment relative, the segment id gives segment
* - If the symbol is segment relative, the segment id gives segment
* information, otherwise it contains CC65_INV_ID.
* - If export_id is valid (not CC65_INV_ID), the symbol is an import and
* export_id allows to retrieve the corresponding export. The fields

311
src/dbginfo/dbgsh.c Normal file
View File

@ -0,0 +1,311 @@
/*****************************************************************************/
/* */
/* dbgsh.c */
/* */
/* debug info test shell */
/* */
/* */
/* */
/* (C) 2011, Ullrich von Bassewitz */
/* Roemerstrasse 52 */
/* D-70794 Filderstadt */
/* EMail: uz@cc65.org */
/* */
/* */
/* This software is provided 'as-is', without any expressed or implied */
/* warranty. In no event will the authors be held liable for any damages */
/* arising from the use of this software. */
/* */
/* Permission is granted to anyone to use this software for any purpose, */
/* including commercial applications, and to alter it and redistribute it */
/* freely, subject to the following restrictions: */
/* */
/* 1. The origin of this software must not be misrepresented; you must not */
/* claim that you wrote the original software. If you use this software */
/* in a product, an acknowledgment in the product documentation would be */
/* appreciated but is not required. */
/* 2. Altered source versions must be plainly marked as such, and must not */
/* be misrepresented as being the original software. */
/* 3. This notice may not be removed or altered from any source */
/* distribution. */
/* */
/*****************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include <ctype.h>
#include <limits.h>
#include <assert.h>
#include <errno.h>
/* common */
#include "attrib.h"
#include "chartype.h"
#include "coll.h"
/* dbginfo */
#include "dbginfo.h"
/*****************************************************************************/
/* Data */
/*****************************************************************************/
/* Terminate flag - end program when set to true */
static int Terminate = 0;
/* The debug file data */
static cc65_dbginfo Info = 0;
/* Error and warning counters */
static unsigned FileErrors = 0;
static unsigned FileWarnings = 0;
/* Structure that contains a command description */
typedef struct CmdEntry CmdEntry;
struct CmdEntry {
char Cmd[12];
int ArgCount;
void (*Func) (Collection*);
};
/*****************************************************************************/
/* Debug file handling */
/*****************************************************************************/
static void CloseFile (void)
/* Close the debug info file */
{
if (Info) {
cc65_free_dbginfo (Info);
Info = 0;
}
}
static int FileIsOpen (void)
/* Return true if the file is open and has loaded without errors: If not,
* print an error message and return false.
*/
{
/* File open? */
if (Info == 0) {
printf ("No debug info file\n");
return 0;
}
/* Errors on load? */
if (FileErrors > 0) {
printf ("File had load errors!\n");
return 0;
}
/* Warnings on load? */
if (FileWarnings > 0) {
printf ("Beware - file had load warnings!\n");
}
/* Ok */
return 1;
}
/*****************************************************************************/
/* Helpers */
/*****************************************************************************/
static void FileError (const cc65_parseerror* Info)
/* Callback function - is called in case of errors */
{
/* Output a message */
printf ("%s:%s(%lu): %s\n",
Info->type? "Error" : "Warning",
Info->name,
(unsigned long) Info->line,
Info->errormsg);
/* Bump the counters */
switch (Info->type) {
case CC65_WARNING: ++FileWarnings; break;
default: ++FileErrors; break;
}
}
/*****************************************************************************/
/* Code */
/*****************************************************************************/
static void CmdOpen (Collection* Args)
/* Open a debug info file */
{
/* Argument is file name */
if (CollCount (Args) != 2) {
printf ("Command requires exactly one argument\n");
return;
}
/* Close an open file */
CloseFile ();
/* Clear the counters */
FileErrors = 0;
FileWarnings = 0;
/* Open the debug info file */
Info = cc65_read_dbginfo (CollAt (Args, 1), FileError);
}
static void CmdClose (Collection* Args attribute ((unused)))
/* Close a debug info file */
{
CloseFile ();
}
static void CmdQuit (Collection* Args attribute ((unused)))
/* Terminate the application */
{
Terminate = 1;
}
static int Parse (char* CmdLine, Collection* Args)
/* Parse the command line and store the arguments in Args. Return true if ok,
* false on error.
*/
{
char* End;
/* Clear the collection */
CollDeleteAll (Args);
/* Parse the command line */
while (1) {
/* Break out on end of line */
if (*CmdLine == '\0') {
break;
}
/* Search for start of next command */
if (IsSpace (*CmdLine)) {
++CmdLine;
continue;
}
/* Allow double quotes to terminate a command */
if (*CmdLine == '\"' || *CmdLine == '\'') {
char Term = *CmdLine++;
End = CmdLine;
while (*End != Term) {
if (*End == '\0') {
fputs ("Unterminated argument\n", stdout);
return 0;
}
++End;
}
*End++ = '\0';
} else {
End = CmdLine;
while (!IsSpace (*End)) {
if (*End == '\0') {
fputs ("Unterminated argument\n", stdout);
return 0;
}
++End;
}
*End++ = '\0';
}
CollAppend (Args, CmdLine);
CmdLine = End;
}
/* Ok */
return 1;
}
static const CmdEntry* FindCmd (const char* Cmd)
/* Search for a command */
{
static const CmdEntry CmdTab[] = {
{ "close", 0, CmdClose },
{ "open", 1, CmdOpen },
{ "quit", 0, CmdQuit },
};
unsigned I;
for (I = 0; I < sizeof (CmdTab) / sizeof (CmdTab[0]); ++I) {
if (strcmp (Cmd, CmdTab[I].Cmd) == 0) {
return CmdTab + I;
}
}
return 0;
}
int main (void)
/* Main program */
{
char Input[256];
Collection Args = STATIC_COLLECTION_INITIALIZER;
const char* Cmd;
const CmdEntry* E;
while (!Terminate) {
/* Output a prompt, then read the input */
fputs ("dbgsh> ", stdout);
fflush (stdout);
if (fgets (Input, sizeof (Input), stdin) == 0) {
fputs ("(EOF)\n", stdout);
break;
}
/* Parse the command line */
if (Parse (Input, &Args) == 0 || CollCount (&Args) == 0) {
continue;
}
/* Search for the command, then execute it */
Cmd = CollAt (&Args, 0);
E = FindCmd (Cmd);
if (E == 0) {
printf ("No such command: %s\n", Cmd);
} else {
E->Func (&Args);
}
}
/* Free arguments */
DoneCollection (&Args);
return 0;
}

View File

@ -1,278 +0,0 @@
/*****************************************************************************/
/* */
/* dbgtest.c */
/* */
/* Test file for the cc65 dbginfo module */
/* */
/* */
/* */
/* (C) 2010-2011, Ullrich von Bassewitz */
/* Roemerstrasse 52 */
/* D-70794 Filderstadt */
/* EMail: uz@cc65.org */
/* */
/* */
/* This software is provided 'as-is', without any expressed or implied */
/* warranty. In no event will the authors be held liable for any damages */
/* arising from the use of this software. */
/* */
/* Permission is granted to anyone to use this software for any purpose, */
/* including commercial applications, and to alter it and redistribute it */
/* freely, subject to the following restrictions: */
/* */
/* 1. The origin of this software must not be misrepresented; you must not */
/* claim that you wrote the original software. If you use this software */
/* in a product, an acknowledgment in the product documentation would be */
/* appreciated but is not required. */
/* 2. Altered source versions must be plainly marked as such, and must not */
/* be misrepresented as being the original software. */
/* 3. This notice may not be removed or altered from any source */
/* distribution. */
/* */
/*****************************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include "dbginfo.h"
static cc65_dbginfo Info;
static void ErrorFunc (const struct cc65_parseerror* E)
/* Handle errors or warnings that occur while parsing a debug file */
{
fprintf (stderr,
"%s:%s(%lu): %s\n",
E->type? "Error" : "Warning",
E->name,
(unsigned long) E->line,
E->errormsg);
}
static void Usage (void)
/* Print usage information and exit */
{
printf ("Usage: dbgtest debug-file\n");
exit (1);
}
static void PrintSourceData (const cc65_sourcedata* D)
/* Print the data for one source file */
{
printf (" %s\n", D->source_name);
}
static void PrintSegmentData (const cc65_segmentdata* D)
/* Print the data for one segment */
{
printf (" %-20s $%06lX $%04lX",
D->segment_name,
(unsigned long) D->segment_start,
(unsigned long) D->segment_size);
if (D->output_name) {
printf (" %-20s $%06lX", D->output_name, D->output_offs);
}
putchar ('\n');
}
static void PrintLineData (const cc65_linedata* D)
/* Print the data for one source line */
{
printf (" file %u", D->source_id);
switch (D->line_type) {
case CC65_LINE_ASM:
printf (": Assembler source");
break;
case CC65_LINE_EXT:
printf (": Externally supplied");
break;
case CC65_LINE_MACRO:
printf (": Macro expansion (%u)", D->count);
break;
default:
printf (": Unknown type %u (%u)", D->line_type, D->count);
break;
}
putchar ('\n');
}
static void PrintSymbolData (const cc65_symboldata* D)
/* Print the data for one symbol */
{
char Segment[256] = { 0 }; /* Needs dynamic alloc ### */
if (D->segment_id != CC65_INV_ID) {
const cc65_segmentinfo* I = cc65_segmentinfo_byid (Info, D->segment_id);
if (I && I->count == 1) {
sprintf (Segment, "segment=%s,", I->data[0].segment_name);
cc65_free_segmentinfo (Info, I);
}
}
printf (" %-20s = %04lX (%ssize=%u)\n",
D->symbol_name,
D->symbol_value,
Segment,
D->symbol_size);
}
static void PrintSourceInfo (const cc65_sourceinfo* Sources)
/* Output the list of source files */
{
unsigned I;
if (Sources) {
for (I = 0; I < Sources->count; ++I) {
PrintSourceData (Sources->data + I);
}
}
}
static void PrintSegmentInfo (const cc65_segmentinfo* Segments)
/* Output the list of segments */
{
unsigned I;
if (Segments) {
for (I = 0; I < Segments->count; ++I) {
PrintSegmentData (Segments->data + I);
}
}
}
static void PrintLineInfo (const cc65_lineinfo* Info)
/* Print a list of line infos */
{
unsigned I;
if (Info) {
for (I = 0; I < Info->count; ++I) {
PrintLineData (Info->data + I);
}
}
}
static void PrintSymbolInfo (const cc65_symbolinfo* Symbols)
/* Print a list of symbol infos */
{
unsigned I;
if (Symbols) {
for (I = 0; I < Symbols->count; ++I) {
PrintSymbolData (Symbols->data + I);
}
}
}
int main (int argc, char** argv)
{
const char* Input;
const cc65_sourceinfo* Sources;
const cc65_segmentinfo* Segments;
const cc65_lineinfo* Lines;
const cc65_symbolinfo* Symbols;
cc65_addr Addr;
/* Input file is argument */
if (argc != 2) {
Usage ();
}
Input = argv[1];
/* Read the file */
Info = cc65_read_dbginfo (Input, ErrorFunc);
if (Info == 0) {
fprintf (stderr, "Error reading input file - aborting\n");
return 1;
}
printf ("Input file \"%s\" successfully read\n", Input);
/* Output a list of files */
printf ("List of source files:\n");
Sources = cc65_get_sourcelist (Info);
PrintSourceInfo (Sources);
cc65_free_sourceinfo (Info, Sources);
/* Output a list of segments */
printf ("Segments processed when linking:\n");
Segments = cc65_get_segmentlist (Info);
PrintSegmentInfo (Segments);
cc65_free_segmentinfo (Info, Segments);
#if 0
/* Check one line */
printf ("Requesting line info for crt0.s(59):\n");
Lines = cc65_lineinfo_byname (Info, "crt0.s", 59);
if (Lines == 0) {
printf (" Not found\n");
} else {
PrintLineInfo (Lines);
cc65_free_lineinfo (Info, Lines);
}
/* Output debug information for all addresses in the complete 6502 address
* space. This is also sort of a benchmark for the search algorithms.
*/
printf ("Line info:\n");
for (Addr = 0; Addr < 0x10000; ++Addr) {
Lines = cc65_lineinfo_byaddr (Info, Addr);
if (Lines) {
printf (" $%04lX:\n", (unsigned long) Addr);
PrintLineInfo (Lines);
cc65_free_lineinfo (Info, Lines);
}
}
#endif
/* Check for address of main */
printf ("Requesting address of _main:\n");
Symbols = cc65_symbol_byname (Info, "_main");
if (Symbols == 0) {
printf (" Not found\n");
Addr = 0x800;
} else {
PrintSymbolInfo (Symbols);
Addr = Symbols->data[0].symbol_value;
cc65_free_symbolinfo (Info, Symbols);
}
/* Print symbols for the next $100 bytes starting from main (or 0x800) */
printf ("Requesting labels for $%04lX-$%04lX:\n",
(unsigned long) Addr, (unsigned long) Addr + 0xFF);
Symbols = cc65_symbol_inrange (Info, Addr, Addr + 0xFF);
if (Symbols == 0) {
printf (" None found\n");
} else {
PrintSymbolInfo (Symbols);
cc65_free_symbolinfo (Info, Symbols);
}
/* Free the debug info */
cc65_free_dbginfo (Info);
return 0;
}

View File

@ -8,11 +8,14 @@
# ------------------------------------------------------------------------------
# The executable to build
EXE = dbgtest
EXE = dbgsh
# Library dir
COMMON = ../common
#
CC = gcc
CFLAGS = -g -Wall -W
CFLAGS = -g -Wall -W -I$(COMMON)
EBIND = emxbind
LDFLAGS =
@ -20,8 +23,9 @@ LDFLAGS =
# Object files to link
OBJS = dbginfo.o \
dbgtest.o
dbgsh.o
LIBS = $(COMMON)/common.a
# ------------------------------------------------------------------------------
# Makefile targets
@ -36,8 +40,8 @@ all: depend
@$(MAKE) -f make/gcc.mak all
endif
$(EXE): $(OBJS)
$(CC) $(OBJS) $(LDFLAGS) -o $@
$(EXE): $(OBJS) $(LIBS)
$(CC) $(LDFLAGS) $(OBJS) $(LIBS) -o $@
@if [ $(OS2_SHELL) ] ; then $(EBIND) $(EXE) ; fi
clean: