gambas-source-code/main/lib/debug/debug.c
Benoît Minisini d44be704d3 Sending a void line to the debugger repeats the last command only if the debugger is run from the command line.
[GB.DEBUG]
* NEW: If the 'GB_DEBUG_DEBUG' environment variable is '1', then a message is printed each time the debugger fifo fails and must be reopened.
* NEW: Sending a void line to the debugger repeats the last command only if the debugger is run from the command line.
2023-03-18 01:34:22 +01:00

1532 lines
28 KiB
C

/***************************************************************************
debug.c
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
***************************************************************************/
#define __DEBUG_C
// Do not include gbx_debug.h
#define __GBX_DEBUG_H
#include "gb_common.h"
#include "gambas.h"
#include <unistd.h>
#include <signal.h>
#include <fcntl.h>
#include "gb_error.h"
#include "gbx_type.h"
#include "gb_limit.h"
#include "gbx_stack.h"
#include "gbx_class.h"
#include "gbx_exec.h"
#include "gbx_local.h"
#include "gbx_object.h"
#include "gbx_eval.h"
#include "print.h"
#include "debug.h"
//#define DEBUG_ME
typedef
struct {
int id;
FUNCTION *func;
PCODE *addr;
CLASS *class;
ushort line;
VALUE *bp;
FUNCTION *fp;
}
DEBUG_BREAK;
typedef
struct {
int id;
EXPRESSION *expr;
VALUE value;
unsigned changed : 1;
}
DEBUG_WATCH;
typedef
struct {
char *pattern;
DEBUG_TYPE type;
void (*func)(char *);
bool loop;
}
DEBUG_COMMAND;
DEBUG_INFO DEBUG_info = { 0 };
GB_DEBUG_INTERFACE *DEBUG_interface;
char DEBUG_buffer[DEBUG_BUFFER_MAX + 1];
char *DEBUG_fifo = NULL;
static DEBUG_BREAK *_breakpoints;
static char *_error = NULL;
static DEBUG_WATCH *_watches;
static EVAL_INTERFACE EVAL;
static int _fdr;
static int _fdw;
static FILE *_out;
static FILE *_in = NULL;
static bool _fifo;
static bool _debug = FALSE;
#define EXEC_current (*(STACK_CONTEXT *)GB_DEBUG.GetExec())
#ifdef DEBUG_ME
#define WARNING(_msg, ...) fprintf(stderr, "W\t" _msg "\n", ##__VA_ARGS__)
#define INFO(_msg, ...) fprintf(stderr, "I\t" _msg "\n", ##__VA_ARGS__)
#else
#define WARNING(_msg, ...) fprintf(_out, "W\t" _msg "\n", ##__VA_ARGS__)
#define INFO(_msg, ...) fprintf(_out, "I\t" _msg "\n", ##__VA_ARGS__)
#endif
static void init_eval_interface()
{
static bool init = FALSE;
if (!init)
{
GB.GetInterface("gb.eval", EVAL_INTERFACE_VERSION, &EVAL);
init = TRUE;
}
}
void DEBUG_break_on_next_line(void)
{
DEBUG_info.stop = TRUE;
DEBUG_info.leave = FALSE;
DEBUG_info.fp = NULL;
DEBUG_info.bp = NULL;
DEBUG_info.pp = NULL;
}
static void signal_user(int sig)
{
signal(SIGUSR2, signal_user);
#ifdef DEBUG_ME
fprintf(stderr, "Got SIGUSR2\n");
#endif
/*CAPP_got_signal();*/
DEBUG_break_on_next_line();
}
bool DEBUG_calc_line_from_position(CLASS *class, FUNCTION *func, PCODE *addr, ushort *line)
{
int lo, hi;
int mid;
ushort pos = addr - func->code;
ushort *post;
if (func->debug)
{
post = func->debug->pos;
lo = 0;
hi = func->debug->nline - 1;
while (lo < hi)
{
mid = (lo + hi) >> 1;
if (pos < post[mid])
{
hi = mid;
}
else if (pos >= post[mid + 1])
{
lo = mid + 1;
}
else
{
*line = mid + func->debug->line;
return FALSE;
}
}
}
return TRUE;
}
static bool calc_position_from_line(CLASS *class, ushort line, FUNCTION **function, PCODE **addr)
{
int i;
ushort pos, pos_after;
FUNCTION *func = NULL;
FUNC_DEBUG *debug = NULL;
// Look at the first functions last, because global variables can be declared and
// initialized everywhere in the file, and initialization function is the first one.
for(i = class->load->n_func - 1; i >= 0; i--)
{
func = &class->load->func[i];
debug = func->debug;
//fprintf(stderr, "calc_position_from_line: %s (%d -> %d) / %d\n", debug->name, debug->line, debug->line + debug->nline - 1, line);
if (debug && line >= debug->line && line < (debug->line + debug->nline))
break;
}
if (i < 0)
return TRUE;
//fprintf(stderr, "calc_position_from_line: %s OK\n", debug->name);
line -= debug->line;
for(;;)
{
pos = debug->pos[line];
pos_after = debug->pos[line + 1];
if (pos != pos_after)
break;
line++;
if (line >= debug->nline)
return TRUE;
}
*function = func;
*addr = &func->code[pos];
/*printf("%s.%d -> %04X\n", class->name, line + debug->line, **addr);*/
return FALSE;
}
DEBUG_INFO *DEBUG_init(GB_DEBUG_INTERFACE *debug, bool fifo, const char *fifo_name)
{
char *env;
char path[DEBUG_FIFO_PATH_MAX];
DEBUG_interface = debug;
_fifo = fifo;
if (_fifo)
{
DEBUG_fifo = GB.NewZeroString(fifo_name);
snprintf(path, sizeof(path), "%sin", fifo_name);
for(;;)
{
_fdw = open(path, O_WRONLY | O_CLOEXEC);
if (_fdw >= 0)
break;
if (errno != EAGAIN && errno != EINTR)
{
fprintf(stderr, "gb.debug: unable to open input fifo: %s: %s\n", strerror(errno), path);
return NULL;
}
}
//fprintf(stderr, "gb.debug: open input fifo: %d\n", _fdw);
_out = fdopen(_fdw, "w");
if (!_out)
{
fprintf(stderr, "gb.debug: unable to create stream on input fifo: %s: %s\n", strerror(errno), path);
return NULL;
}
}
else
{
_out = stdout;
}
//ARRAY_create(&_breakpoints);
GB.NewArray(&_breakpoints, sizeof(DEBUG_BREAK), 16);
GB.NewArray(&_watches, sizeof(DEBUG_WATCH), 0);
signal(SIGUSR2, signal_user);
signal(SIGPIPE, SIG_IGN);
setlinebuf(_out);
env = getenv("GB_DEBUG_DEBUG");
if (env && env[0] == '1' && env[1] == 0)
_debug = TRUE;
return &DEBUG_info;
}
void DEBUG_exit(void)
{
int i;
GB.FreeArray(&_breakpoints);
if (_watches)
{
for (i = 0; i < GB.Count(_watches); i++)
EVAL.Free(POINTER(&_watches[i].expr));
GB.FreeArray(&_watches);
}
GB.FreeString(&DEBUG_fifo);
GB.FreeString(&_error);
// Don't do it, it blocks!
/*if (EXEC_fifo)
{
fclose(_in);
fclose(_out);
}
*/
}
static int find_free_breakpoint(void)
{
int i;
char used[MAX_BREAKPOINT];
memset(used, FALSE, MAX_BREAKPOINT);
for (i = 0; i < ARRAY_count(_breakpoints); i++)
used[_breakpoints[i].id - 1] = TRUE;
for (i = 0; i < MAX_BREAKPOINT; i++)
if (!used[i])
return (i + 1);
return 0;
}
static bool init_breakpoint(DEBUG_BREAK *brk)
{
PCODE *addr = NULL;
FUNCTION *func;
//fprintf(stderr, "init_breakpoint: id = %d\n", brk->id);
if (brk->addr || !CLASS_is_loaded(brk->class))
{
WARNING("breakpoint is pending");
return TRUE;
}
if (CLASS_is_native(brk->class) || !brk->class->debug)
{
WARNING("Cannot set breakpoint: no debugging information");
return TRUE;
}
if (calc_position_from_line(brk->class, brk->line, &func, &addr))
{
WARNING("Cannot set breakpoint: cannot calculate position");
//fprintf(_out, "Cannot calc position from line number\n");
return TRUE;
}
if (!PCODE_is_breakpoint(*addr))
{
//WARNING("Cannot set breakpoint: Not a line beginning: %04d: %04X", addr - func->code, *addr);
WARNING("Cannot set breakpoint: Not a line beginning");
//fprintf(_out, "Not a line beginning ?\n");
return TRUE;
}
if (*addr & 0xFF)
{
WARNING("breakpoint already set");
//fprintf(_out, "_breakpoints already set\n");
return FALSE;
}
brk->addr = addr;
*addr = PCODE_BREAKPOINT(brk->id);
//fprintf(stderr, "init_breakpoint: OK\n");
#ifdef DEBUG_ME
fprintf(stderr, "init_breakpoint: %s.%d\n", brk->class->name, brk->line);
#endif
INFO("breakpoint set: %s.%d", brk->class->name, brk->line);
return FALSE;
}
static bool set_breakpoint(CLASS *class, ushort line)
{
DEBUG_BREAK *brk;
int id;
if (GB.Count(_breakpoints) >= MAX_BREAKPOINT)
{
WARNING("Too many breakpoints");
return TRUE;
}
id = find_free_breakpoint();
if (id == 0)
{
WARNING("Cannot create breakpoint");
return TRUE;
}
brk = (DEBUG_BREAK *)GB.Add(&_breakpoints);
brk->id = id;
brk->addr = NULL;
brk->class = class;
brk->line = line;
#ifdef DEBUG_ME
fprintf(stderr, "set_breakpoint: %s.%d\n", class->name, line);
#endif
init_breakpoint(brk);
return FALSE;
}
static bool unset_breakpoint(CLASS *class, ushort line)
{
int i;
DEBUG_BREAK *brk;
for (i = 0; i < GB.Count(_breakpoints); i++)
{
brk = &_breakpoints[i];
if (brk->class == class && brk->line == line)
{
if (brk->addr)
*(brk->addr) = PCODE_BREAKPOINT(0);
GB.Remove(&_breakpoints, i, 1);
#ifdef DEBUG_ME
fprintf(stderr, "unset_breakpoint: %s.%d\n", class->name, line);
#endif
INFO("breakpoint removed");
return FALSE;
}
}
WARNING("Unknown breakpoint");
return TRUE;
}
static void unset_all_breakpoints()
{
int i;
DEBUG_BREAK *brk;
for (i = 0; i < GB.Count(_breakpoints); i++)
{
brk = &_breakpoints[i];
if (brk->addr)
*(brk->addr) = PCODE_BREAKPOINT(0);
}
GB.Remove(&_breakpoints, 0, GB.Count(_breakpoints));
}
void DEBUG_init_breakpoints(CLASS *class)
{
int i;
DEBUG_BREAK *brk;
#ifdef DEBUG_ME
fprintf(stderr, "DEBUG_init_breakpoints: %p %s\n", class, class->name);
#endif
for (i = 0; i < GB.Count(_breakpoints); i++)
{
brk = &_breakpoints[i];
if (brk->class == class)
{
//fprintf(stderr, "DEBUG_init_breakpoints: %s\n", class->name);
init_breakpoint(brk);
}
}
}
static void print_local()
{
int i;
FUNCTION *fp;
LOCAL_SYMBOL *lp;
fp = DEBUG_info.fp;
if (!fp || !fp->debug)
return;
for (i = 0; i < fp->debug->n_local; i++)
{
lp = &fp->debug->local[i];
fprintf(_out, "%.*s ", lp->sym.len, lp->sym.name);
}
}
static void print_symbol(GLOBAL_SYMBOL *gp, bool is_static, bool is_public)
{
if (CTYPE_get_kind(gp->ctype) != TK_VARIABLE && CTYPE_get_kind(gp->ctype) != TK_CONST)
return;
if (CTYPE_is_static(gp->ctype) && !is_static)
return;
if (!CTYPE_is_static(gp->ctype) && is_static)
return;
if (CTYPE_is_public(gp->ctype) && !is_public)
return;
if (!CTYPE_is_public(gp->ctype) && is_public)
return;
fprintf(_out, "%.*s ", gp->sym.len, gp->sym.name);
}
static void print_object()
{
int i;
GLOBAL_SYMBOL *gp;
CLASS *cp = DEBUG_info.cp;
void *op = DEBUG_info.op;
if (!cp || !cp->load)
return;
fprintf(_out, "S: ");
for (i = 0; i < cp->load->n_global; i++)
{
gp = &cp->load->global[i];
print_symbol(gp, TRUE, TRUE);
}
fprintf(_out, "s: ");
for (i = 0; i < cp->load->n_global; i++)
{
gp = &cp->load->global[i];
print_symbol(gp, TRUE, FALSE);
}
if (op)
{
fprintf(_out, "D: ");
for (i = 0; i < cp->load->n_global; i++)
{
gp = &cp->load->global[i];
print_symbol(gp, FALSE, TRUE);
}
fprintf(_out, "d: ");
for (i = 0; i < cp->load->n_global; i++)
{
gp = &cp->load->global[i];
print_symbol(gp, FALSE, FALSE);
}
}
}
static void command_quit(char *cmd)
{
exit(1);
}
static void command_hold(char *cmd)
{
GB_DEBUG.DebugHold();
}
static void command_go(char *cmd)
{
GB.Component.Signal(GB_SIGNAL_DEBUG_CONTINUE, 0);
DEBUG_info.stop = FALSE;
DEBUG_info.leave = FALSE;
DEBUG_info.fp = NULL;
DEBUG_info.bp = NULL;
DEBUG_info.pp = NULL;
}
static void command_step(char *cmd)
{
GB.Component.Signal(GB_SIGNAL_DEBUG_FORWARD, 0);
DEBUG_break_on_next_line();
}
static void command_next(char *cmd)
{
GB.Component.Signal(GB_SIGNAL_DEBUG_FORWARD, 0);
DEBUG_info.stop = TRUE;
DEBUG_info.leave = FALSE;
DEBUG_info.fp = FP;
DEBUG_info.bp = BP;
DEBUG_info.pp = PP;
}
static void command_from(char *cmd)
{
STACK_CONTEXT *sc = GB_DEBUG.GetStack(0); //STACK_get_current();
if (sc && sc->pc)
{
GB.Component.Signal(GB_SIGNAL_DEBUG_FORWARD, 0);
DEBUG_info.stop = TRUE;
DEBUG_info.leave = FALSE;
DEBUG_info.fp = sc->fp;
DEBUG_info.bp = sc->bp;
DEBUG_info.pp = sc->pp;
}
else
{
GB.Component.Signal(GB_SIGNAL_DEBUG_FORWARD, 0);
DEBUG_info.stop = TRUE;
DEBUG_info.leave = TRUE;
DEBUG_info.fp = FP;
DEBUG_info.bp = BP;
DEBUG_info.pp = PP;
}
}
static void command_breakpoint(char *cmd)
{
char *comp = NULL;
char class_name[256];
ushort line;
bool unset = *cmd == '-';
char *p;
cmd++;
if (unset && cmd[0] == '*' && cmd[1] == 0)
{
unset_all_breakpoints();
return;
}
if (*cmd == '[')
{
p = index(cmd, ']');
if (p && p[1] == '.')
{
comp = &cmd[1];
*p = 0;
cmd = p + 2;
if (comp[0] == '$' && !comp[1])
comp = NULL;
}
}
if (sscanf(cmd, "%256[^.].%hu", class_name, &line) != 2)
WARNING("Cannot %s breakpoint: syntax error", unset ? "remove" : "add");
else if (unset)
unset_breakpoint((CLASS *)GB_DEBUG.FindClass(comp, class_name), line);
else
set_breakpoint((CLASS *)GB_DEBUG.FindClass(comp, class_name), line);
}
void DEBUG_backtrace(FILE *out)
{
int i, n;
STACK_CONTEXT *context;
fprintf(out, "%s", DEBUG_get_current_position());
//for (i = 0; i < (STACK_frame_count - 1); i++)
n = 0;
for (i = 0;; i++)
{
context = GB_DEBUG.GetStack(i); //&STACK_frame[i];
if (!context)
break;
n += fprintf(out, " %s", DEBUG_get_position(context->cp, context->fp, context->pc));
/*if (context->pc)
{
line = 0;
if (DEBUG_calc_line_from_position(context->cp, context->fp, context->pc, &line))
n += fprintf(out, " %s.?.?", context->cp->name);
else
n += fprintf(out, " %s.%s.%d", context->cp->name, context->fp->debug->name, line);
}
else if (context->cp)
n += fprintf(out, " ?");*/
if (n >= (DEBUG_OUTPUT_MAX_SIZE / 2))
{
fprintf(out, " ...");
break;
}
}
}
static void debug_info(bool frame)
{
const char *p;
char c;
int i;
DEBUG_WATCH *watch;
fprintf(_out, "%c[%d]\t", frame ? '@' : '*', getpid());
if (_error)
{
p = _error;
for(;;)
{
c = *p++;
if (!c)
break;
if (c == '\n' || c == '\r' || c == '\t')
c = ' ';
fputc(c, _out);
}
}
fprintf(_out, "\t");
DEBUG_backtrace(_out);
fprintf(_out, "\t");
print_local();
fprintf(_out, "\t");
print_object();
fprintf(_out, "\t");
for (i = 0; i < GB.Count(_watches); i++)
{
watch = &_watches[i];
if (watch->changed)
fprintf(_out, "%d ", watch->id);
}
fputc('\n', _out);
}
static void set_info(STACK_CONTEXT *context)
{
if (context)
{
DEBUG_info.bp = context->bp;
DEBUG_info.pp = context->pp;
DEBUG_info.fp = context->fp;
DEBUG_info.op = context->op;
DEBUG_info.cp = context->cp;
}
else
{
DEBUG_info.bp = BP;
DEBUG_info.pp = PP;
DEBUG_info.fp = FP;
DEBUG_info.op = OP;
DEBUG_info.cp = CP;
}
}
static void command_frame(char *cmd)
{
int i;
int frame;
STACK_CONTEXT *context = NULL;
if (cmd)
{
frame = atoi(&cmd[1]);
//fprintf(_out, "switching to frame %d\n", frame);
if (frame > 0)
{
for (i = 0;; i++)
{
context = GB_DEBUG.GetStack(i);
if (!context)
break;
if (!context->pc && !context->cp)
continue;
frame--;
if (!frame)
break;
}
}
}
set_info(context);
debug_info(cmd != NULL);
}
static void command_eval(char *cmd)
{
EXPRESSION *expr;
ERROR_INFO save_error = { 0 };
ERROR_INFO save_last = { 0 };
DEBUG_INFO save_debug;
VALUE *val;
VALUE result;
int start, len;
FILE *out;
const char *name;
int ret;
init_eval_interface();
//out = *cmd == '!' ? stdout : _out;
out = _out;
len = strlen(cmd);
for (start = 0; start < len; start++)
{
if (cmd[start] == '\t')
break;
//if (*cmd != '!')
fputc(cmd[start], _out);
}
if (start >= len)
return;
//if (*cmd != '!')
fprintf(_out, "\t");
GB_DEBUG.SaveError(&save_error, &save_last);
save_debug = DEBUG_info;
start++;
EVAL.New(POINTER(&expr), &cmd[start], len - start);
if (EVAL.Compile(expr, *cmd == '='))
{
//if (*cmd != '!')
fprintf(_out, "!");
fputs(expr->error, out);
goto __END;
}
GB_DEBUG.EnterEval();
val = (VALUE *)EVAL.Run(expr, GB_DEBUG.GetValue);
GB_DEBUG.LeaveEval();
if (!val)
goto __ERROR;
result = *val;
GB.BorrowValue((GB_VALUE *)&result);
switch(*cmd)
{
case '?':
PRINT_value(out, val, TRUE);
break;
case '!':
PRINT_value(out, val, FALSE);
break;
case '#':
PRINT_object(out, val);
break;
case '=':
if (!EVAL.GetAssignmentSymbol(expr, &name, &len))
{
ret = GB_DEBUG.SetValue(name, len, val);
if (ret == GB_DEBUG_SET_ERROR)
{
GB.ReleaseValue((GB_VALUE *)&result);
goto __ERROR;
}
else if (ret == GB_DEBUG_SET_READ_ONLY)
{
GB.ReleaseValue((GB_VALUE *)&result);
fprintf(out, "!%.*s is read-only", len, name);
goto __END;
}
}
fprintf(out, "OK");
break;
}
GB.ReleaseValue((GB_VALUE *)&result);
goto __END;
__ERROR:
fprintf(out, "!");
fputs(GB_DEBUG.GetErrorMessage(), out);
__END:
EVAL.Free(POINTER(&expr));
DEBUG_info = save_debug; //.cp = NULL;
GB_DEBUG.RestoreError(&save_error, &save_last);
fputc('\n', out);
fflush(out);
}
static int find_watch(int id)
{
int i;
DEBUG_WATCH *watch;
for (i = 0; i < GB.Count(_watches); i++)
{
watch = &_watches[i];
if (watch->id == id)
return i;
}
return -1;
}
static int add_watch(int id, EXPRESSION *expr, VALUE *value)
{
DEBUG_WATCH *watch = (DEBUG_WATCH *)GB.Add(&_watches);
watch->id = id;
watch->expr = expr;
if (value)
watch->value = *value;
return GB.Count(_watches) - 1;
}
static void remove_watch(int index)
{
DEBUG_WATCH *watch = &_watches[index];
EVAL.Free(POINTER(&watch->expr));
GB.Remove(&_watches, index, 1);
}
static void command_watch(char *cmd)
{
EXPRESSION *expr;
int start, len;
int id;
int index;
VALUE *value;
ERROR_INFO save_error = { 0 };
ERROR_INFO save_last = { 0 };
DEBUG_INFO save_debug;
init_eval_interface();
len = strlen(cmd);
for (start = 0; start < len; start++)
{
if (cmd[start] == '\t')
break;
}
cmd[start] = 0;
id = atoi(&cmd[1]);
if (id == 0)
return;
expr = NULL;
value = NULL;
if (len > start)
{
GB_DEBUG.SaveError(&save_error, &save_last);
save_debug = DEBUG_info;
start++;
EVAL.New(POINTER(&expr), &cmd[start], len - start);
if (EVAL.Compile(expr, FALSE))
{
fputs(expr->error, _out);
EVAL.Free(POINTER(&expr));
}
else
{
GB_DEBUG.EnterEval();
value = (VALUE *)EVAL.Run(expr, GB_DEBUG.GetValue);
GB_DEBUG.LeaveEval();
}
DEBUG_info = save_debug;
GB_DEBUG.RestoreError(&save_error, &save_last);
if (!expr)
return;
}
index = find_watch(id);
if (index >= 0)
remove_watch(index);
if (expr)
add_watch(id, expr, value);
DEBUG_info.watch = GB.Count(_watches) > 0;
}
static void command_symbol(char *cmd)
{
int start, len;
ERROR_INFO save_error = { 0 };
ERROR_INFO save_last = { 0 };
DEBUG_INFO save_debug;
GB_DEBUG.SaveError(&save_error, &save_last);
save_debug = DEBUG_info;
len = strlen(cmd);
for (start = 0; start < len; start++)
{
if (cmd[start] == '\t')
break;
fputc(cmd[start], _out);
}
if (start >= len)
return;
fprintf(_out, "\t");
/*DEBUG_info.bp = BP;
DEBUG_info.fp = FP;
DEBUG_info.op = OP;
DEBUG_info.cp = CP;*/
start++;
PRINT_symbol(_out, &cmd[start], len - start);
fputc('\n', _out);
fflush(_out);
DEBUG_info = save_debug;
GB_DEBUG.RestoreError(&save_error, &save_last);
}
static void command_option(char *cmd)
{
bool on;
if (!cmd[1] || !cmd[2])
return;
on = cmd[2] == '+';
switch(cmd[1])
{
case 'b':
GB_DEBUG.BreakOnError(on);
break;
case 'g':
GB_DEBUG.DebugInside(on);
break;
default:
;
}
}
void DEBUG_breakpoint(int id)
{
DEBUG_main(FALSE);
}
const char *DEBUG_get_position(CLASS *cp, FUNCTION *fp, PCODE *pc)
{
const char *comp_name;
const char *class_name;
const char *func_name;
ushort line = 0;
if (!cp)
return "?";
class_name = cp->name;
while (*class_name == '^')
class_name++;
if (cp->component)
comp_name = cp->component->name;
else
comp_name = "$";
if (fp && fp->debug)
{
func_name = fp->debug->name;
if (pc)
DEBUG_calc_line_from_position(cp, fp, pc, &line);
}
else
func_name = "?";
snprintf(DEBUG_buffer, sizeof(DEBUG_buffer), "[%s].%s.%s.%d", comp_name, class_name, func_name, line);
return DEBUG_buffer;
}
const char *DEBUG_get_profile_position(CLASS *cp, FUNCTION *fp, PCODE *pc)
{
static uint prof_index = 0;
const char *name;
const char *func;
char buffer[16], buffer2[16];
func = "?";
if (cp)
{
if (cp->load && cp->load->prof)
{
if (cp->load->prof[0] == 0)
{
prof_index++;
cp->load->prof[0] = prof_index;
name = cp->name;
}
else
{
sprintf(buffer, "%u", cp->load->prof[0]);
name = buffer;
}
if (fp && fp->debug)
{
int i = fp->debug->index + 1;
if (cp->load->prof[i] == 0)
{
prof_index++;
cp->load->prof[i] = prof_index;
func = fp->debug->name;
}
else
{
sprintf(buffer2, "%u", cp->load->prof[i]);
func = buffer2;
}
}
else
func = "?";
}
else
name = cp->name;
}
else
name = "?";
if (pc)
{
ushort line = 0;
if (fp != NULL && fp->debug)
DEBUG_calc_line_from_position(cp, fp, pc, &line);
snprintf(DEBUG_buffer, sizeof(DEBUG_buffer), "%.64s.%.64s.%d", name, func, line);
}
else
{
snprintf(DEBUG_buffer, sizeof(DEBUG_buffer), "%.64s.%.64s", name, func);
}
return DEBUG_buffer;
}
const char *DEBUG_get_current_position(void)
{
return DEBUG_get_position(CP, FP, PC);
}
void DEBUG_where(void)
{
fprintf(_out ? _out : stderr, "%s: ", DEBUG_get_current_position());
}
void DEBUG_welcome(void)
{
if (!_fifo)
fprintf(_out, DEBUG_WELCOME);
}
static bool compare_values(VALUE *a, VALUE *b)
{
void *jump[] = {
&&__VOID, &&__BOOLEAN, &&__BYTE, &&__SHORT, &&__INTEGER, &&__LONG, &&__SINGLE, &&__FLOAT,
&&__DATE, &&__STRING, &&__STRING, &&__POINTER, &&__VARIANT, &&__FUNCTION, &&__CLASS, &&__NULL
};
void *vjump[] = {
&&__VOID, &&__VBOOLEAN, &&__VBYTE, &&__VSHORT, &&__VINTEGER, &&__VLONG, &&__VSINGLE, &&__VFLOAT,
&&__VDATE, &&__VSTRING, &&__VSTRING, &&__VPOINTER, &&__VOID, &&__VOID, &&__VOID, &&__VOID
};
if (a->type != b->type)
return TRUE;
if (TYPE_is_object(a->type))
goto __OBJECT;
else
goto *jump[a->type];
__VOID:
__NULL:
return FALSE;
__BOOLEAN:
__BYTE:
__SHORT:
__INTEGER:
return a->_integer.value != b->_integer.value;
__LONG:
__DATE:
return a->_long.value != b->_long.value;
__SINGLE:
return a->_single.value != b->_single.value;
__FLOAT:
return a->_float.value != b->_float.value;
__STRING:
return a->_string.addr != b->_string.addr || a->_string.start != b->_string.start || a->_string.len != b->_string.len;
__POINTER:
__OBJECT:
__CLASS:
return a->_pointer.value != b->_pointer.value;
__FUNCTION:
return a->_function.class != b->_function.class || a->_function.object != b->_function.object || a->_function.index != b->_function.index;
__VARIANT:
if (a->_variant.vtype != b->_variant.vtype)
return TRUE;
if (TYPE_is_object(a->_variant.vtype))
goto __VOBJECT;
else
goto *vjump[a->_variant.vtype];
__VBOOLEAN: return a->_variant.value._boolean != b->_variant.value._boolean;
__VBYTE: return a->_variant.value._byte != b->_variant.value._byte;
__VSHORT: return a->_variant.value._short != b->_variant.value._short;
__VINTEGER: return a->_variant.value._integer != b->_variant.value._integer;
__VDATE: __VLONG: return a->_variant.value._long != b->_variant.value._long;
__VSINGLE: return a->_variant.value._single!= b->_variant.value._single;
__VFLOAT: return a->_variant.value._float != b->_variant.value._float;
__VPOINTER: __VOBJECT: __VSTRING: return a->_variant.value._string != b->_variant.value._string;
}
bool DEBUG_check_watches(void)
{
int i;
DEBUG_WATCH *watch;
VALUE *value;
bool changed = FALSE;
ERROR_INFO save_error = { 0 };
ERROR_INFO save_last = { 0 };
DEBUG_INFO save_debug;
set_info(NULL);
GB_DEBUG.SaveError(&save_error, &save_last);
save_debug = DEBUG_info;
for (i = 0; i < GB.Count(_watches); i++)
{
watch = &_watches[i];
watch->changed = FALSE;
GB_DEBUG.EnterEval();
value = (VALUE *)EVAL.Run(watch->expr, GB_DEBUG.GetValue);
GB_DEBUG.LeaveEval();
if (!value)
continue;
if (compare_values(&watch->value, value))
{
if (watch->value.type != T_VOID)
changed = TRUE;
watch->value = *value;
watch->changed = TRUE;
}
}
DEBUG_info = save_debug;
GB_DEBUG.RestoreError(&save_error, &save_last);
if (!changed)
return FALSE;
DEBUG_main(FALSE);
return TRUE;
}
static void open_read_fifo()
{
char path[DEBUG_FIFO_PATH_MAX];
if (_fifo)
{
//fprintf(stderr, "open_read_fifo\n");
snprintf(path, sizeof(path), "%sout", DEBUG_fifo);
for(;;)
{
_fdr = open(path, O_RDONLY | O_CLOEXEC);
if (_fdr >= 0)
break;
if (errno != EAGAIN && errno != EINTR)
{
fprintf(stderr, "gb.debug: unable to open output fifo: %s: %s\n", strerror(errno), path);
return;
}
usleep(20000);
}
//fprintf(stderr, "gb.debug: open read fifo: %d\n", _fdr);
_in = fdopen(_fdr, "r");
if (!_in)
{
fprintf(stderr, "gb.debug: unable to open stream on output fifo: %s: %s\n", strerror(errno), path);
return;
}
setlinebuf(_in);
}
else
{
_in = stdin;
}
}
void DEBUG_main(bool error)
{
static DEBUG_TYPE last_command = TC_NONE;
static DEBUG_COMMAND Command[] =
{
// "p" and "i" are reserved for remote debugging.
{ "q", TC_NONE, command_quit, FALSE },
{ "n", TC_NEXT, command_next, FALSE },
{ "s", TC_STEP, command_step, FALSE },
{ "f", TC_FROM, command_from, FALSE },
{ "g", TC_GO, command_go, FALSE },
{ "+", TC_NONE, command_breakpoint, TRUE },
{ "-", TC_NONE, command_breakpoint, TRUE },
{ "&", TC_NONE, command_symbol, TRUE },
{ "?", TC_NONE, command_eval, TRUE },
{ "!", TC_NONE, command_eval, TRUE },
{ "#", TC_NONE, command_eval, TRUE },
{ "=", TC_NONE, command_eval, TRUE },
{ "@", TC_NONE, command_frame, TRUE },
{ "o", TC_NONE, command_option, TRUE },
{ "w", TC_NONE, command_watch, TRUE },
{ "h", TC_NONE, command_hold, TRUE },
{ NULL }
};
static bool first = TRUE;
char *cmd = NULL;
char cmdbuf[256];
int len;
DEBUG_COMMAND *tc = NULL;
int save_errno = errno;
int try = 0;
GB.FreeString(&_error);
if (error)
_error = GB.NewZeroString(GB_DEBUG.GetErrorMessage());
fflush(_out);
#ifdef DEBUG_ME
fprintf(stderr, "DEBUG_main {\n");
#endif
if (_fifo)
{
fprintf(_out, first ? "!!\n" : "!\n");
fflush(_out);
first = FALSE;
}
command_frame(NULL);
if (!_in)
open_read_fifo();
do
{
/*if (CP == NULL)
printf("[]:");
else
printf("[%s%s]:", DEBUG_get_current_position(), _error ? "*" : "");*/
GB.Component.Signal(GB_SIGNAL_DEBUG_BREAK, 0);
if (!_fifo)
{
fprintf(_out, "> ");
fflush(_out);
}
GB.FreeString(&cmd);
__TRY_AGAIN:
for(;;)
{
*cmdbuf = 0;
errno = 0;
if (fgets(cmdbuf, sizeof(cmdbuf), _in) == NULL && errno != EINTR)
break;
if (!*cmdbuf)
continue;
cmd = GB.AddString(cmd, cmdbuf, 0);
if (cmd[GB.StringLength(cmd) - 1] == '\n')
break;
}
len = GB.StringLength(cmd);
// A null string command means an I/O error or an end-of-file
// We try to reopen the fifo.
if (len == 0)
{
if (errno)
fprintf(stderr, "gb.debug: warning: unable to read debugger input: %s\n", strerror(errno));
else
{
if (_debug)
fprintf(stderr, "gb.debug: input has been closed, reopen it.\n");
usleep(1000);
}
fclose(_in);
open_read_fifo();
try++;
if (try < 3)
goto __TRY_AGAIN;
_exit(1);
}
if (len > 0 && cmd[len - 1] == '\n')
{
len--;
cmd[len] = 0;
}
#ifdef DEBUG_ME
fprintf(stderr, "--> %s\n", cmd);
#endif
if (len == 0)
{
if (last_command == TC_NONE || _fifo)
continue;
for (tc = Command; tc->pattern; tc++)
{
if (tc->type == last_command)
{
(*tc->func)(cmd);
break;
}
}
}
else
{
for (tc = Command; tc->pattern; tc++)
{
if (strncasecmp(tc->pattern, cmd, strlen(tc->pattern)) == 0)
{
if (tc->type != TC_NONE)
last_command = tc->type;
(*tc->func)(cmd);
break;
}
}
}
if (tc->pattern == NULL)
WARNING("Unknown command: %s", cmd);
fflush(_out);
}
while (last_command == TC_NONE || tc->pattern == NULL || tc->loop);
GB.FreeString(&cmd);
errno = save_errno;
#ifdef DEBUG_ME
fprintf(stderr, "} DEBUG_main\n");
#endif
}