mirror of
https://github.com/darlinghq/darling-gdb.git
synced 2025-01-22 17:16:29 +00:00
199b2450f6
Change all references to stdout/stderr to gdb_stdout/gdb_stderr. Replace all calls to stdio output functions with calls to corresponding _unfiltered functions (`fprintf_unfiltered') Replaced calls to fopen for output to gdb_fopen. Added sufficient goo to utils.c and defs.h to make the above work. The net effect is that stdio output functions are only directly used in utils.c. Elsewhere, the _unfiltered and _filtered functions and GDB_FILE type are used. In the near future, GDB_FILE will stop being equivalant to FILE. The semantics of some commands has changed in a very subtle way: called in the right context, they may cause new occurences of prompt_for_continue() behavior. The testsuite doesn't notice anything like this, though. Please respect this change by not reintroducing stdio output dependencies in the main body of gdb code. All output from commands should go to a GDB_FILE. Target-specific code can still use stdio directly to communicate with targets.
1718 lines
38 KiB
C
1718 lines
38 KiB
C
/* Energize (formerly known as Cadillac) interface routines.
|
||
Copyright 1991, 1992 Free Software Foundation, Inc.
|
||
|
||
This file is part of GDB.
|
||
|
||
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 of the License, 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., 675 Mass Ave, Cambridge, MA 02139, USA. */
|
||
|
||
#include "defs.h"
|
||
#include "symtab.h"
|
||
#include "inferior.h"
|
||
#include "command.h"
|
||
#include "bfd.h"
|
||
#include "symfile.h"
|
||
#include "objfiles.h"
|
||
#include "target.h"
|
||
#include <sys/types.h>
|
||
#include <sys/time.h>
|
||
#include <sys/param.h>
|
||
#include "energize/connection.h"
|
||
#include "energize/genericreq.h"
|
||
#include "energize/debuggerreq.h"
|
||
#include "energize/debuggerconn.h"
|
||
#include "energize/ttyconn.h"
|
||
#include <varargs.h>
|
||
#include <sys/stat.h>
|
||
#ifdef USG
|
||
#include <sys/file.h>
|
||
#endif
|
||
#include <fcntl.h>
|
||
#include <sys/filio.h>
|
||
#include <setjmp.h>
|
||
#include <signal.h>
|
||
#include <sys/errno.h>
|
||
#include <termios.h>
|
||
#include <string.h>
|
||
#ifdef NCR486
|
||
#include <sys/stropts.h>
|
||
#endif
|
||
|
||
/* Non-zero means that we're doing the energize interface. */
|
||
int energize = 0;
|
||
|
||
/* Non-zero means we are reloading breakpoints, etc from the
|
||
Energize kernel, and we should suppress various messages */
|
||
static int energize_reloading = 0;
|
||
|
||
/* Connection block for debugger<=>kernel communications. */
|
||
static Connection *conn = 0;
|
||
|
||
/* fd for our socket to the kernel. */
|
||
static int kerfd;
|
||
|
||
/* The kernel's ID for this instance of the program. */
|
||
static int program_id;
|
||
|
||
static int instance_id;
|
||
|
||
/* The fd for the pty associated with the inferior. */
|
||
static int inferior_pty = -1;
|
||
static int inferior_tty = -1;
|
||
|
||
static int has_run = 0;
|
||
|
||
extern int pgrp_inferior;
|
||
|
||
extern char *source_path;
|
||
|
||
/* The name of the executable file */
|
||
static char *exec_file;
|
||
|
||
/* Tell energize_command_line_input() where to get its text from */
|
||
static int doing_breakcommands_message = 0;
|
||
|
||
/* Stash command text here */
|
||
static char *command_line_text = 0;
|
||
static int command_line_length = 0;
|
||
|
||
/* Flags returned by wait_for_events() */
|
||
#define KERNEL_EVENT 1
|
||
#define PTY_EVENT 2
|
||
|
||
static void execute_command_1();
|
||
|
||
|
||
/* This routine redirects the output of fputs_filtered to the kernel so that
|
||
the user can see what's going on in his debugger window. */
|
||
|
||
void
|
||
energize_fputs(ptr)
|
||
const char *ptr;
|
||
{
|
||
if (conn)
|
||
CVWriteTranscriptInfo (conn, instance_id, (char *)ptr);
|
||
else
|
||
fputs (ptr, stdout);
|
||
}
|
||
|
||
void
|
||
energize_query(query, args)
|
||
char *query;
|
||
va_list args;
|
||
{
|
||
char buf[100];
|
||
|
||
if (!energize)
|
||
return;
|
||
|
||
vsprintf(buf, query, args);
|
||
|
||
CVWriteQueryInfo(conn,
|
||
instance_id,
|
||
CQueryConfirm,
|
||
qno_unknown,
|
||
buf,
|
||
""); /* transcript */
|
||
}
|
||
|
||
void
|
||
energize_acknowledge_query(ack)
|
||
char *ack;
|
||
{
|
||
CVWriteQueryInfo(conn,
|
||
instance_id,
|
||
CQueryAcknowleged,
|
||
0,
|
||
ack,
|
||
""); /* transcript */
|
||
}
|
||
|
||
/* Copy all data from the pty to the kernel. */
|
||
|
||
static void
|
||
pty_to_kernel()
|
||
{
|
||
CTtyRequest *req;
|
||
char buf[1024];
|
||
int cc;
|
||
|
||
while (1)
|
||
{
|
||
cc = read(inferior_pty, buf, sizeof(buf));
|
||
|
||
if (cc == 0
|
||
|| (cc < 0
|
||
&& (errno == EWOULDBLOCK
|
||
|| errno == EAGAIN)))
|
||
break;
|
||
|
||
if (cc < 0)
|
||
{
|
||
close(inferior_pty);
|
||
inferior_pty = -1;
|
||
perror("pty_to_kernel: pty read error");
|
||
break;
|
||
}
|
||
|
||
req = CWriteTtyRequest(conn, TextIORType);
|
||
CWriteVstringLen(conn, buf, cc);
|
||
CWriteLength(conn);
|
||
}
|
||
CWriteRequestBuffer(conn);
|
||
}
|
||
|
||
/* Copy data from the kernel to the pty. */
|
||
|
||
static void
|
||
kernel_to_pty(data, len)
|
||
char *data;
|
||
int len;
|
||
{
|
||
int cc;
|
||
|
||
cc = write(inferior_pty, data, len);
|
||
|
||
if (cc != len)
|
||
{
|
||
if (cc < 0)
|
||
{
|
||
close(inferior_pty);
|
||
inferior_pty = -1;
|
||
perror("kernel_to_pty: pty write error");
|
||
return;
|
||
}
|
||
printf("Couldn't write all the data to the pty, wanted %d, got %d\n",
|
||
len, cc);
|
||
}
|
||
}
|
||
|
||
static char *
|
||
full_filename(symtab)
|
||
struct symtab *symtab;
|
||
{
|
||
int pathlen;
|
||
char *filename;
|
||
|
||
if (!symtab)
|
||
return NULL;
|
||
|
||
if (symtab->fullname)
|
||
return savestring(symtab->fullname, strlen(symtab->fullname));
|
||
|
||
if (symtab->filename[0] == '/')
|
||
return savestring(symtab->filename, strlen(symtab->filename));
|
||
|
||
if (symtab->dirname)
|
||
pathlen = strlen(symtab->dirname);
|
||
else
|
||
pathlen = 0;
|
||
if (symtab->filename)
|
||
pathlen += strlen(symtab->filename);
|
||
|
||
filename = xmalloc(pathlen+1);
|
||
|
||
if (symtab->dirname)
|
||
strcpy(filename, symtab->dirname);
|
||
else
|
||
*filename = '\000';
|
||
if (symtab->filename)
|
||
strcat(filename, symtab->filename);
|
||
|
||
return filename;
|
||
}
|
||
|
||
/* Tell the energize kernel how high the stack is so that frame numbers (which
|
||
are relative to the current stack height make sense.
|
||
|
||
Calculate the number of frames on the stack, and the number of subroutine
|
||
invocations that haven't changed since the last call to this routine. The
|
||
second number is calculated by comparing the PCs of the current stack frames
|
||
to the PCs of the previous set of stack frames. The screw here is that a
|
||
subroutine may call several different procedures, which means that the PC
|
||
in its frame changes, even though you are still in the same subroutine. We
|
||
resolve this by converting the frames PC into the PC at the start of the
|
||
function (for efficiency, this is done only if the simple comparison test
|
||
fails). */
|
||
|
||
struct pclist
|
||
{
|
||
CORE_ADDR pc;
|
||
struct pclist *next;
|
||
};
|
||
|
||
/* Non-zero means that Energize kernel already knows how high the stack is. */
|
||
static int stack_info_valid = 0;
|
||
|
||
static void
|
||
send_stack_info()
|
||
{
|
||
struct pclist *pclist = 0, *pli, *opli;
|
||
static struct pclist *old_pclist;
|
||
FRAME frame;
|
||
int height, similar;
|
||
|
||
if (stack_info_valid)
|
||
return;
|
||
|
||
height = 0;
|
||
similar = 0;
|
||
|
||
/* First, calculate the stack height, and build the new pclist */
|
||
|
||
for (frame = get_current_frame();
|
||
frame != 0;
|
||
frame = get_prev_frame(frame))
|
||
{
|
||
(height)++;
|
||
pli = (struct pclist *)xmalloc(sizeof(struct pclist));
|
||
|
||
pli->pc = frame->pc;
|
||
pli->next = pclist;
|
||
pclist = pli;
|
||
}
|
||
|
||
/* Now, figure out how much of the stack hasn't changed */
|
||
|
||
for (pli = pclist, opli = old_pclist;
|
||
pli != 0 && opli != 0;
|
||
pli = pli->next, opli = opli->next, (similar)++)
|
||
{
|
||
if ((pli->pc != opli->pc)
|
||
&& (get_pc_function_start(pli->pc)
|
||
!= get_pc_function_start(opli->pc)))
|
||
break;
|
||
}
|
||
|
||
/* Free up all elements of the old pclist */
|
||
|
||
opli = old_pclist;
|
||
|
||
while (opli)
|
||
{
|
||
pli = opli->next;
|
||
free (opli);
|
||
opli = pli;
|
||
}
|
||
|
||
old_pclist = pclist; /* Install the new pclist */
|
||
|
||
CVWriteStackSizeInfo(conn,
|
||
instance_id,
|
||
height, /* Frame depth */
|
||
CInnerFrameIs0,
|
||
similar, /* Frame diff */
|
||
"" /* Transcript */
|
||
);
|
||
|
||
stack_info_valid = 1;
|
||
}
|
||
|
||
/* Tell the Energize server about the file and line # that corresponds to pc,
|
||
and which stack frame level that pc corresponds to. */
|
||
|
||
static void
|
||
send_location(pc, frame_level)
|
||
CORE_ADDR pc;
|
||
int frame_level;
|
||
{
|
||
char *funcname, *filename;
|
||
struct symtab_and_line sal;
|
||
struct symbol *symbol;
|
||
|
||
sal = find_pc_line(pc, 0);
|
||
symbol = find_pc_function(pc);
|
||
|
||
funcname = symbol ? symbol->name : "";
|
||
filename = full_filename(sal.symtab);
|
||
|
||
send_stack_info();
|
||
|
||
CVWriteStackFrameInfo(conn,
|
||
instance_id,
|
||
sal.line,
|
||
CFileLinePos,
|
||
frame_level,
|
||
funcname,
|
||
filename,
|
||
"" /* XXX ? transcript */
|
||
);
|
||
if (filename)
|
||
free(filename);
|
||
}
|
||
|
||
/* Tell the kernel where we are in the program, and what the stack looks like.
|
||
*/
|
||
|
||
static void
|
||
send_status()
|
||
{
|
||
char *funcname;
|
||
struct symbol *symbol;
|
||
static int sent_prog_inst = 0;
|
||
|
||
symbol = find_pc_function(stop_pc);
|
||
funcname = symbol ? symbol->name : "";
|
||
|
||
if (!has_run)
|
||
return;
|
||
|
||
if (inferior_pid == 0) /* target has died */
|
||
{
|
||
CVWriteProgramTerminatedInfo(conn,
|
||
instance_id,
|
||
""
|
||
);
|
||
return;
|
||
}
|
||
|
||
if (!sent_prog_inst)
|
||
{
|
||
sent_prog_inst = 1;
|
||
CVWriteProgramInstanceInfo(conn,
|
||
program_id,
|
||
instance_id,
|
||
"", /* hostname */
|
||
"", /* arglist */
|
||
""
|
||
);
|
||
}
|
||
|
||
send_location(stop_pc,
|
||
selected_frame_level); /* Had better be 0! */
|
||
|
||
CVWriteProgramStoppedInfo(conn,
|
||
instance_id,
|
||
0, /* XXX - breakpoint # or signal # */
|
||
CDebuggerCommand,
|
||
funcname,
|
||
"" /* XXX ? transcript */
|
||
);
|
||
|
||
}
|
||
|
||
/* Call this to output annotated function names. Names will be demangled if
|
||
necessary. arg_mode contains flags that are passed on to cplus_demangle. */
|
||
|
||
void
|
||
energize_annotate_function(funcname, arg_mode, level)
|
||
char *funcname;
|
||
int arg_mode;
|
||
int level;
|
||
{
|
||
extern int demangle;
|
||
char *demangled_name = NULL;
|
||
|
||
if (funcname == NULL)
|
||
return;
|
||
|
||
if (demangle)
|
||
{
|
||
demangled_name = cplus_demangle(funcname, arg_mode);
|
||
|
||
if (demangled_name)
|
||
{
|
||
funcname = demangled_name;
|
||
printf_filtered("'");
|
||
}
|
||
}
|
||
|
||
send_stack_info();
|
||
|
||
if (level < 0) level = 0;
|
||
|
||
CVWriteBackTraceEntryInfo(conn,
|
||
instance_id,
|
||
level, /* frameNo */
|
||
funcname);
|
||
|
||
if (demangled_name)
|
||
{
|
||
free(demangled_name);
|
||
printf_filtered("'");
|
||
}
|
||
}
|
||
|
||
/* Call this just prior to printing out the name & value of a variable. This
|
||
tells the kernel where to annotate the output. */
|
||
|
||
/* The args are:
|
||
expression - A text handle on what GDB can use to reference this value.
|
||
This can be a symbol name, or a convenience var, etc...
|
||
symbol - Used to determine the scope of the data. May be NULL.
|
||
type - Determines if we have a pointer ref, and the print name of the type.
|
||
Used in ShowValue message.
|
||
valaddr - The address in target memory of the data.
|
||
field - The field name of the struct or union element being referenced.
|
||
*/
|
||
|
||
static char cum_expr[200]; /* Cumulative expression */
|
||
static char *expr_stack[100] = {cum_expr}; /* Pointers to end of expressions */
|
||
static char **last_expr = expr_stack; /* Current expr stack pointer */
|
||
|
||
void
|
||
energize_start_variable_annotation(expression, symbol, type, valaddr, field)
|
||
char *expression;
|
||
struct symbol *symbol;
|
||
struct type *type;
|
||
CORE_ADDR valaddr;
|
||
char *field;
|
||
{
|
||
int ref_type;
|
||
int stor_cl;
|
||
enum type_code type_code;
|
||
enum address_class sym_class;
|
||
char *type_cast;
|
||
|
||
if (!energize)
|
||
return;
|
||
|
||
send_stack_info();
|
||
|
||
strcpy(*last_expr++, expression);
|
||
*last_expr = *(last_expr-1) + strlen(expression);
|
||
|
||
switch (TYPE_CODE(type))
|
||
{
|
||
case TYPE_CODE_ARRAY:
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_FLT:
|
||
ref_type = CValueValueRef;
|
||
break;
|
||
case TYPE_CODE_PTR:
|
||
ref_type = CValuePointerRef;
|
||
break;
|
||
default:
|
||
ref_type = CValueUndefRef;
|
||
break;
|
||
}
|
||
|
||
/* Make sure that pointer points at something we understand */
|
||
|
||
if (ref_type == CValuePointerRef)
|
||
switch (TYPE_CODE(TYPE_TARGET_TYPE(type)))
|
||
{
|
||
case TYPE_CODE_PTR:
|
||
case TYPE_CODE_ARRAY:
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
case TYPE_CODE_ENUM:
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_FLT:
|
||
break;
|
||
default:
|
||
ref_type = CValueUndefRef;
|
||
break;
|
||
}
|
||
|
||
if (symbol)
|
||
{
|
||
sym_class = SYMBOL_CLASS(symbol);
|
||
|
||
switch (sym_class)
|
||
{
|
||
case LOC_CONST:
|
||
case LOC_CONST_BYTES:
|
||
stor_cl = CValueStorStaticConst;
|
||
break;
|
||
case LOC_STATIC:
|
||
stor_cl = CValueStorStaticVar;
|
||
break;
|
||
case LOC_REGISTER:
|
||
case LOC_REGPARM:
|
||
stor_cl = CValueStorRegister;
|
||
break;
|
||
case LOC_ARG:
|
||
case LOC_REF_ARG:
|
||
case LOC_LOCAL:
|
||
case LOC_LOCAL_ARG:
|
||
stor_cl = CValueStorLocalVar;
|
||
break;
|
||
default:
|
||
stor_cl = CValueStorUndef;
|
||
break;
|
||
}
|
||
}
|
||
else
|
||
stor_cl = CValueStorUndef;
|
||
|
||
type_cast = TYPE_NAME(type);
|
||
|
||
CVWriteValueBeginInfo(conn,
|
||
instance_id,
|
||
valaddr,
|
||
ref_type,
|
||
stor_cl,
|
||
0, /* XXX - frameno */
|
||
cum_expr,
|
||
field,
|
||
type_cast,
|
||
""); /* transcript */
|
||
}
|
||
|
||
void
|
||
energize_end_variable_annotation()
|
||
{
|
||
if (!energize)
|
||
return;
|
||
|
||
last_expr--; /* Pop the expr stack */
|
||
**last_expr = '\000'; /* Cut off the last part of the expr */
|
||
|
||
CVWriteValueEndInfo(conn,
|
||
instance_id,
|
||
""); /* transcript */
|
||
}
|
||
|
||
/* Tell the kernel that the target is now running. */
|
||
|
||
static void
|
||
go_busy()
|
||
{
|
||
CVWriteProgramBusyInfo(conn,
|
||
instance_id,
|
||
""); /* XXX ? transcript */
|
||
CWriteRequestBuffer(conn); /* Must take place synchronusly! */
|
||
stack_info_valid = 0;
|
||
}
|
||
|
||
|
||
void
|
||
energize_symbol_file(objfile)
|
||
struct objfile *objfile;
|
||
{
|
||
if (!energize)
|
||
return;
|
||
|
||
CVWriteSymbolTableInfo(conn,
|
||
objfile->name,
|
||
""); /* Transcript */
|
||
}
|
||
|
||
/* execute_command_1(echo, queue, cmd, args) - echo - non-zero means echo the
|
||
command. queue - non-zero means don't execute it now, just save it away for
|
||
later. cmd - string containing printf control sequences. args - list of
|
||
arguments needed by those control sequences.
|
||
*/
|
||
|
||
/* Linked list of queued up commands */
|
||
static struct command_line *queued_commands = 0;
|
||
static struct command_line *last_queued_command = 0;
|
||
|
||
/* Call this routine to save a command for later. The command string is
|
||
copied into freshly malloc'ed memory. */
|
||
|
||
static void
|
||
queue_command(cmd)
|
||
char *cmd;
|
||
{
|
||
char *buf;
|
||
struct command_line *cl;
|
||
unsigned long s;
|
||
|
||
s = (strlen(cmd) + 1) + 7 & ~(unsigned long)7;
|
||
|
||
buf = (char *)xmalloc(s + sizeof(struct command_line));
|
||
cl = (struct command_line *)(buf + s);
|
||
cl->next = 0;
|
||
cl->line = buf;
|
||
|
||
strncpy(cl->line, cmd, s);
|
||
|
||
if (queued_commands)
|
||
last_queued_command->next = cl;
|
||
else
|
||
queued_commands = cl;
|
||
|
||
last_queued_command = cl;
|
||
}
|
||
|
||
/* Call this procedure to take a command off of the command queue. It returns
|
||
a pointer to a buf which the caller is responsible for freeing. NULL is
|
||
returned if there are no commands queued. */
|
||
|
||
static char *
|
||
dequeue_command()
|
||
{
|
||
struct command_line *cl;
|
||
char *cmd;
|
||
|
||
cl = queued_commands;
|
||
|
||
if (!cl)
|
||
return NULL;
|
||
|
||
queued_commands = cl->next;
|
||
|
||
return cl->line;
|
||
}
|
||
|
||
static void
|
||
execute_command_1(va_alist)
|
||
va_dcl
|
||
{
|
||
char buf[100]; /* XXX - make buf dynamic! */
|
||
|
||
int echo;
|
||
int queue;
|
||
char *cmd;
|
||
va_list args;
|
||
|
||
va_start(args);
|
||
echo = va_arg(args, int);
|
||
|
||
queue = va_arg(args, int);
|
||
cmd = va_arg(args, char *);
|
||
|
||
vsprintf(buf, cmd, args);
|
||
|
||
if (queue)
|
||
queue_command(buf);
|
||
else
|
||
{
|
||
if (echo)
|
||
printf_filtered("%s\n", buf);
|
||
execute_command(buf, 1);
|
||
}
|
||
|
||
va_end(args);
|
||
}
|
||
|
||
#ifdef KERNEL_RECORD
|
||
GDB_FILE *kerout;
|
||
|
||
static int
|
||
kernel_record(fd, ptr, num)
|
||
int fd, num;
|
||
char *ptr;
|
||
|
||
{
|
||
fwrite(ptr, num, 1, kerout);
|
||
fflush(kerout);
|
||
return write(fd, ptr, num);
|
||
}
|
||
#endif
|
||
|
||
void
|
||
energize_condition_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
if (energize)
|
||
CVWriteBreakConditionInfo(conn,
|
||
instance_id,
|
||
b->number,
|
||
b->cond_string ? b->cond_string : "",
|
||
"" /* transcript */
|
||
);
|
||
}
|
||
|
||
void
|
||
energize_commands_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
struct command_line *l;
|
||
|
||
if (!energize)
|
||
return;
|
||
|
||
CVWriteBreakCommandBegInfo(conn,
|
||
instance_id,
|
||
b->number,
|
||
""); /* transcript */
|
||
|
||
for (l = b->commands; l; l = l->next)
|
||
CVWriteBreakCommandEntryInfo(conn,
|
||
instance_id,
|
||
l->line,
|
||
""); /* transcript */
|
||
|
||
CVWriteBreakCommandEndInfo(conn,
|
||
instance_id,
|
||
""); /* transcript */
|
||
}
|
||
|
||
static void
|
||
breakpoint_notify(b, action)
|
||
struct breakpoint *b;
|
||
int action;
|
||
{
|
||
struct symbol *sym;
|
||
char *funcname = "";
|
||
char *filename;
|
||
char *included_in_filename = "";
|
||
|
||
if (!energize
|
||
|| energize_reloading) /* Don't notify energize about breakpoint changes, as it's about to send us
|
||
a new bunch. */
|
||
return;
|
||
|
||
if (b->type != bp_breakpoint)
|
||
return;
|
||
|
||
filename = full_filename(b->symtab);
|
||
|
||
sym = find_pc_function(b->address);
|
||
if (sym)
|
||
funcname = SYMBOL_NAME(sym);
|
||
|
||
CVWriteBreakpointInfo (conn,
|
||
instance_id,
|
||
b->number,
|
||
b->line_number,
|
||
CFileLinePos,
|
||
CBreakOnInstrAccess,
|
||
action,
|
||
b->ignore_count,
|
||
funcname,
|
||
filename ? filename : "",
|
||
"", /* included_in_filename */
|
||
"" /* transcript */
|
||
);
|
||
|
||
if (b->commands)
|
||
energize_commands_breakpoint(b);
|
||
|
||
energize_condition_breakpoint(b);
|
||
|
||
if (filename)
|
||
free(filename);
|
||
}
|
||
|
||
void
|
||
energize_create_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
breakpoint_notify(b, CEnableBreakpoint);
|
||
}
|
||
|
||
void
|
||
energize_delete_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
breakpoint_notify(b, CDeleteBreakpoint);
|
||
}
|
||
|
||
void
|
||
energize_enable_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
breakpoint_notify(b, CEnableBreakpoint);
|
||
}
|
||
|
||
void
|
||
energize_disable_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
breakpoint_notify(b, CDisableBreakpoint);
|
||
}
|
||
|
||
void
|
||
energize_ignore_breakpoint(b)
|
||
struct breakpoint *b;
|
||
{
|
||
breakpoint_notify(b, CBreakAttrUnchanged);
|
||
}
|
||
|
||
/* Open up a pty and its associated tty. Return the fd of the tty. */
|
||
|
||
#ifndef NCR486
|
||
static void
|
||
getpty()
|
||
{
|
||
int n, ptyfd, ttyfd;
|
||
static char dev[30];
|
||
struct stat statbuf;
|
||
struct termios termios;
|
||
|
||
#define HIGHPTY (('z' - 'p') * 16 - 1)
|
||
|
||
for (n = 0; n <= HIGHPTY; n++)
|
||
{
|
||
sprintf(dev, "/dev/pty%c%x", n/16 + 'p', n%16);
|
||
if (stat(dev, &statbuf))
|
||
break;
|
||
ptyfd = open(dev, O_RDWR);
|
||
if (ptyfd < 0)
|
||
continue;
|
||
sprintf(dev, "/dev/tty%c%x", n/16 + 'p', n%16);
|
||
ttyfd = open(dev, O_RDWR);
|
||
if (ttyfd < 0)
|
||
{
|
||
close(ptyfd);
|
||
continue;
|
||
}
|
||
|
||
/* Setup pty for non-blocking I/O. Also make it give us a SIGIO when
|
||
there's data available. */
|
||
|
||
n = fcntl(ptyfd, F_GETFL, 0);
|
||
fcntl(ptyfd, F_SETFL, n|FNDELAY|FASYNC);
|
||
fcntl(ptyfd, F_SETOWN, getpid());
|
||
|
||
tcgetattr(ttyfd, &termios);
|
||
termios.c_oflag &= ~OPOST; /* No post-processing */
|
||
tcsetattr(ttyfd, TCSANOW, &termios);
|
||
|
||
inferior_pty = ptyfd;
|
||
inferior_tty = ttyfd;
|
||
return;
|
||
}
|
||
|
||
error ("getpty: can't get a pty\n");
|
||
}
|
||
#endif
|
||
/* Alternate getpty for NCRs */
|
||
|
||
#ifdef NCR486 /* LTL */
|
||
#define MAX_PTM_TRY 16
|
||
#define MAX_GRANTPT_TRY 4
|
||
static void
|
||
getpty()
|
||
{
|
||
char *slavename;
|
||
extern char *ptsname();
|
||
int j, i;
|
||
int n, mfd, sfd;
|
||
struct stat statbuf;
|
||
struct termios termios;
|
||
|
||
mfd = open("/dev/ptmx", O_RDWR); /* get the master */
|
||
if (mfd < 0)
|
||
error ("getpty: can't locate master\n");
|
||
|
||
if (grantpt(mfd) < 0) /* get a slave */
|
||
error ("getpty: can't acquire slave");
|
||
|
||
unlockpt(mfd);
|
||
|
||
slavename = ptsname(mfd); /* get the slave device name */
|
||
if (!slavename)
|
||
error ("getpty: can't get a pts\n");
|
||
|
||
/* Drop controlling tty, become pgrp master */
|
||
|
||
if (setpgid(0, getppid()) == -1)
|
||
perror("setpgid() failed: ");
|
||
|
||
if (setsid() == -1)
|
||
perror("setsid() failed: ");
|
||
|
||
sfd = open(slavename, O_RDWR);
|
||
if (sfd < 0)
|
||
{
|
||
close(mfd);
|
||
error ("getpty: can't open slave\n");
|
||
}
|
||
|
||
|
||
if (ioctl(sfd, I_PUSH, "ptem")) perror ("getpty: ioctl I_PUSH fails");
|
||
if (ioctl(sfd, I_PUSH, "ldterm")) perror ("getpty: ioctl I_PUSH fails");
|
||
|
||
/* setup mty for non-blocking I/O. */
|
||
|
||
n = fcntl(mfd, F_GETFL);
|
||
if (n < 0)
|
||
perror ("getpty: fcntl F_GETFL failed");
|
||
|
||
if (fcntl(mfd, F_SETFL, n|O_NDELAY) <0)
|
||
perror("getpty: fcntl F_SETFL failed");
|
||
|
||
/* set up for async i/o - V.4 will send SIGPOLL when data available */
|
||
|
||
if (ioctl (mfd, I_SETSIG, S_INPUT|S_RDNORM) < 0)
|
||
perror ("getpty: ioctl I_SETSIG failed");
|
||
|
||
if (tcgetattr(sfd, &termios))
|
||
perror("getpty: tcgetattr fails");
|
||
termios.c_oflag &= ~OPOST; /* no post-processing */
|
||
if (tcsetattr(sfd, TCSANOW, &termios))
|
||
perror("getpty: tcsetattr fails");
|
||
|
||
inferior_pty=mfd;
|
||
inferior_tty=sfd;
|
||
|
||
return;
|
||
}
|
||
|
||
#endif /* NCR486 */
|
||
|
||
/* Examine a protocol packet from the driver. */
|
||
|
||
static void
|
||
kernel_dispatch(queue)
|
||
int queue; /* Non-zero means we should just queue up
|
||
commands. */
|
||
{
|
||
register CHeader *head;
|
||
|
||
head = (CHeader *)CPeekNextRequest (conn);
|
||
if (head == NULL)
|
||
{
|
||
fprintf (stderr, "EOF on kernel read!\n");
|
||
exit (1);
|
||
}
|
||
|
||
if (head->reqType < LastTtyRequestRType)
|
||
{
|
||
CTtyRequest* req = CReadTtyRequest (conn);
|
||
switch (req->head.reqType)
|
||
{
|
||
case AcceptConnectionRType:
|
||
/* Tell the rest of the world that energize is now set up */
|
||
CSkipRequest (conn);
|
||
break;
|
||
|
||
case RefuseConnectionRType:
|
||
fprintf (stderr, "Debugger connection refused\n");
|
||
exit (1);
|
||
|
||
case KillProgramRType:
|
||
exit (0);
|
||
|
||
case TextIORType:
|
||
{
|
||
char *p;
|
||
ReqLen len;
|
||
|
||
p = CGetVstring(conn, &len);
|
||
kernel_to_pty(p, len);
|
||
}
|
||
break;
|
||
default:
|
||
fprintf(stderr, "Unknown Tty request type = %d\n",
|
||
req->head.reqType);
|
||
break;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
CVDebuggerRequest *req = CVReadDebuggerRequest (conn);
|
||
if (!req)
|
||
{
|
||
fprintf (stderr, "CVReadDebuggerRequest returned NULL, type = %d\n",
|
||
head->reqType);
|
||
exit(1);
|
||
}
|
||
|
||
switch (req->head.request->reqType)
|
||
{
|
||
case OpenProgramInstanceRType:
|
||
{
|
||
char *arglist, buf[100]; /* XXX - Make buf dynamic! */
|
||
int arglen;
|
||
/* XXX - should notice when program_id changes */
|
||
arglist = req->openProgramInstance.progArglist.text;
|
||
arglen = req->openProgramInstance.progArglist.byteLen;
|
||
|
||
execute_command_1(1, queue, "break main");
|
||
execute_command_1(1, queue, "enable delete $bpnum");
|
||
if (arglist)
|
||
{
|
||
execute_command_1(1, queue, "set args %.*s", arglen, arglist);
|
||
}
|
||
execute_command_1(1, queue, "run");
|
||
}
|
||
break;
|
||
case SearchPathRType:
|
||
directory_command(req->searchPath.searchPath.text, 0);
|
||
break;
|
||
case QuitDebuggerRType:
|
||
execute_command_1(1, queue, "quit");
|
||
break;
|
||
case RunRType:
|
||
if (req->run.request->useArglist == CNewArglist)
|
||
{
|
||
execute_command_1(1, queue, "set args %.*s",
|
||
req->run.progArglist.byteLen,
|
||
req->run.progArglist.text);
|
||
}
|
||
execute_command_1(1, queue, "run");
|
||
break;
|
||
case ContinueRType:
|
||
execute_command_1(1, queue, "continue");
|
||
break;
|
||
case StepRType:
|
||
execute_command_1(1, queue, "step %d", req->step.request->stepCount);
|
||
break;
|
||
case NextRType:
|
||
execute_command_1(1, queue, "next %d", req->next.request->nextCount);
|
||
break;
|
||
case ChangeStackFrameRType:
|
||
switch (req->changeStackFrame.request->frameMovement)
|
||
{
|
||
case CToCurrentStackFrame:
|
||
execute_command_1(1, queue, "frame %d",
|
||
req->changeStackFrame.request->frameNo);
|
||
break;
|
||
case CToInnerStackFrame:
|
||
execute_command_1(1, queue, "down %d",
|
||
req->changeStackFrame.request->frameNo);
|
||
break;
|
||
case CToOuterStackFrame:
|
||
execute_command_1(1, queue, "up %d",
|
||
req->changeStackFrame.request->frameNo);
|
||
break;
|
||
case CToAbsoluteStackFrame:
|
||
execute_command_1(1, queue, "frame %d",
|
||
req->changeStackFrame.request->frameNo);
|
||
break;
|
||
}
|
||
break;
|
||
case BackTraceRType:
|
||
/* XXX - deal with limit??? */
|
||
execute_command_1(1, queue, "backtrace");
|
||
break;
|
||
case FinishRType:
|
||
execute_command_1(1, queue, "finish");
|
||
break;
|
||
case TerminateProgramRType:
|
||
execute_command_1(1, queue, "kill");
|
||
break;
|
||
case NewBreakpointRType:
|
||
{
|
||
char *tail;
|
||
int skipped;
|
||
|
||
tail = strrchr(req->newBreakpoint.fileName.text, '/');
|
||
if (!tail)
|
||
tail = req->newBreakpoint.fileName.text;
|
||
else
|
||
tail++;
|
||
skipped = tail - req->newBreakpoint.fileName.text;
|
||
execute_command_1(1, queue, "break %.*s:%d",
|
||
req->newBreakpoint.fileName.byteLen - skipped,
|
||
tail,
|
||
req->newBreakpoint.request->fileLinePos);
|
||
}
|
||
break;
|
||
case StopRType:
|
||
kill(-pgrp_inferior, SIGINT);
|
||
break;
|
||
case UserInputRType:
|
||
{
|
||
char *text;
|
||
long len;
|
||
|
||
/* XXX - should really break command up into seperate lines
|
||
and spoon-feed it to execute_command */
|
||
|
||
text = req->userInput.userInput.text;
|
||
len = req->userInput.userInput.byteLen;
|
||
|
||
if (text[len-1] == '\n') text[len-1] = '\000';
|
||
|
||
while (*text == ' ' || *text == '\t') text++;
|
||
|
||
if (STREQ(text, "]*[")) /* XXX - What does this mean??? */
|
||
break;
|
||
|
||
if (*text != '\000')
|
||
execute_command_1(0, queue, "%s", text);
|
||
else
|
||
print_prompt(); /* User just typed a blank line */
|
||
}
|
||
break;
|
||
case QueryResponseRType:
|
||
{
|
||
char *resp;
|
||
|
||
if (req->queryResponse.request->response)
|
||
resp = "y";
|
||
else
|
||
resp = "n";
|
||
execute_command_1(1, 1, resp);
|
||
printf_filtered("%s\n", resp);
|
||
}
|
||
break;
|
||
case ChangeBreakpointRType:
|
||
switch (req->changeBreakpoint.request->breakpointAttr)
|
||
{
|
||
case CBreakAttrUnchanged:
|
||
execute_command_1(1, queue, "ignore %d %d",
|
||
req->changeBreakpoint.request->breakpointId,
|
||
req->changeBreakpoint.request->ignoreCount);
|
||
break;
|
||
case CEnableBreakpoint:
|
||
execute_command_1(1, queue, "enable %d",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
break;
|
||
case CDisableBreakpoint:
|
||
execute_command_1(1, queue, "disable %d",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
break;
|
||
case CDeleteBreakpoint:
|
||
execute_command_1(1, queue, "delete %d",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
break;
|
||
case CEnableDisableBreakpoint:
|
||
execute_command_1(1, queue, "enable once %d",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
break;
|
||
case CEnableDeleteBreakpoint:
|
||
execute_command_1(1, queue, "enable delete %d",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
break;
|
||
default:
|
||
printf_filtered("ChangeBreakpointRType: unknown breakpointAttr\n");
|
||
printf_filtered(" breakpointAttr = %d\n",
|
||
req->changeBreakpoint.request->breakpointAttr);
|
||
printf_filtered(" breakpointId = %d\n",
|
||
req->changeBreakpoint.request->breakpointId);
|
||
printf_filtered(" breakpointType = %d\n",
|
||
req->changeBreakpoint.request->breakpointType);
|
||
printf_filtered(" ignoreCount = %d\n",
|
||
req->changeBreakpoint.request->ignoreCount);
|
||
break;
|
||
}
|
||
break;
|
||
case BreakConditionRType:
|
||
execute_command_1(1, queue, "condition %d %.*s",
|
||
req->breakCondition.request->breakpointId,
|
||
req->breakCondition.condition.byteLen,
|
||
req->breakCondition.condition.text);
|
||
break;
|
||
case BreakCommandsRType:
|
||
/* Put pointers to where energize_command_line_input() can find
|
||
them. */
|
||
doing_breakcommands_message = 1;
|
||
command_line_length = req->breakCommands.commands.byteLen;
|
||
command_line_text = req->breakCommands.commands.text;
|
||
execute_command_1(1, queue, "commands %d",
|
||
req->breakCommands.request->breakpointId);
|
||
command_line_text = (char *)NULL;
|
||
command_line_length = 0;
|
||
doing_breakcommands_message = 0;
|
||
break;
|
||
case ShowValueRType:
|
||
{
|
||
char expr[100], *p = expr;
|
||
|
||
expr[0] = 0;
|
||
|
||
if (req->showValue.request->ref_type == CValuePointerRef)
|
||
strcat(expr, "* ");
|
||
|
||
if (req->showValue.type_cast.byteLen)
|
||
{
|
||
strcat(expr, "(");
|
||
strncat(expr, req->showValue.type_cast.text,
|
||
req->showValue.type_cast.byteLen);
|
||
strcat(expr, ") ");
|
||
}
|
||
|
||
if (req->showValue.field.byteLen)
|
||
strcat(expr, "(");
|
||
|
||
strncat(expr, req->showValue.expression.text,
|
||
req->showValue.expression.byteLen);
|
||
|
||
if (req->showValue.field.byteLen)
|
||
{
|
||
strcat(expr, ")");
|
||
|
||
strncat(expr, req->showValue.field.text,
|
||
req->showValue.field.byteLen);
|
||
}
|
||
|
||
execute_command_1(1, queue, "print %s", expr);
|
||
}
|
||
break;
|
||
case SetValueRType:
|
||
{
|
||
char expr[100], *p = expr;
|
||
|
||
expr[0] = 0;
|
||
|
||
if (req->setValue.request->ref_type == CValuePointerRef)
|
||
strcat(expr, "* ");
|
||
|
||
#if 0
|
||
if (req->setValue.type_cast.byteLen)
|
||
{
|
||
strcat(expr, "(");
|
||
strncat(expr, req->setValue.type_cast.text,
|
||
req->setValue.type_cast.byteLen);
|
||
strcat(expr, ") ");
|
||
}
|
||
#endif
|
||
if (req->setValue.field.byteLen)
|
||
strcat(expr, "(");
|
||
|
||
strncat(expr, req->setValue.expression.text,
|
||
req->setValue.expression.byteLen);
|
||
|
||
if (req->setValue.field.byteLen)
|
||
{
|
||
strcat(expr, ")");
|
||
|
||
strncat(expr, req->setValue.field.text,
|
||
req->setValue.field.byteLen);
|
||
}
|
||
|
||
execute_command_1(1, queue, "print %s = (%s) %s", expr,
|
||
req->setValue.type_cast.text,
|
||
req->setValue.value.text);
|
||
}
|
||
break;
|
||
case DynamicLoadRType:
|
||
{
|
||
char *filename;
|
||
|
||
filename = req->dynamicLoad.filenames.byteLen ?
|
||
req->dynamicLoad.filenames.text : exec_file;
|
||
|
||
switch (req->dynamicLoad.request->action)
|
||
{
|
||
case CDynamicLoadUpdateSymtab:
|
||
energize_reloading = 1;
|
||
execute_command_1(1, queue, "set confirm no");
|
||
execute_command_1(1, queue, "delete");
|
||
/* execute_command_1(1, queue, "set $bpnum=1");*/ /* Use this to reset breakpoint #s */
|
||
execute_command_1(1, queue, "exec-file %s", filename);
|
||
execute_command_1(1, queue, "symbol-file %s", filename);
|
||
execute_command_1(1, queue, "set confirm yes");
|
||
energize_reloading = 0;
|
||
break;
|
||
case CDynamicLoadRestoreStart:
|
||
break;
|
||
case CDynamicLoadRestoreEnd: /* Not used anymore??? */
|
||
printf_filtered("\n[Target has changed, automatic restoration of state has been done.]\n");
|
||
print_prompt();
|
||
break;
|
||
default:
|
||
printf_filtered("DynamicLoadRType: unknown action=%d, filename=%s\n",
|
||
req->dynamicLoad.request->action,
|
||
req->dynamicLoad.filenames.text);
|
||
break;
|
||
}
|
||
}
|
||
break;
|
||
default:
|
||
fprintf(stderr, "Unknown Debugger request type = %d\n",
|
||
req->head.request->reqType);
|
||
break;
|
||
}
|
||
free (req); /* Should probably call CVFreeDebuggerRequest() here, but
|
||
can't do so if interrupt level has mucked with req->
|
||
request. CVFreeDebuggerRequest() only ends up calling
|
||
free() anyway! */
|
||
}
|
||
}
|
||
|
||
/* Return a bitmask indicating if the kernel or the pty did something
|
||
interesting. Set poll to non-zero if you don't want to wait. */
|
||
|
||
static int
|
||
wait_for_events(poll)
|
||
int poll;
|
||
{
|
||
fd_set readfds;
|
||
int numfds;
|
||
int eventmask = 0;
|
||
static struct timeval tv = {0};
|
||
|
||
/* Output all pending requests. */
|
||
CWriteRequestBuffer(conn);
|
||
|
||
FD_ZERO(&readfds);
|
||
|
||
/* Wait till there's some activity from the kernel or the pty. */
|
||
do
|
||
{
|
||
FD_SET(kerfd, &readfds);
|
||
|
||
FD_SET(inferior_pty, &readfds);
|
||
|
||
if (poll)
|
||
numfds = select(sizeof(readfds)*8, &readfds, 0, 0, &tv);
|
||
else
|
||
numfds = select(sizeof(readfds)*8, &readfds, 0, 0, 0);
|
||
}
|
||
while (numfds <= 0 && !poll);
|
||
|
||
if (numfds == 0)
|
||
return 0;
|
||
|
||
if (FD_ISSET(inferior_pty, &readfds))
|
||
eventmask |= PTY_EVENT;
|
||
|
||
if (FD_ISSET(kerfd, &readfds))
|
||
eventmask |= KERNEL_EVENT;
|
||
|
||
return eventmask;
|
||
}
|
||
|
||
/* This is called from read_command_lines() to provide the text for breakpoint
|
||
commands, which is supplied in a BreakCommands message. Each call to this
|
||
routine supplies a single line of text, with the newline removed. */
|
||
|
||
/* This routine may be invoked in two different contexts. In the first, it
|
||
is being called as a result of the BreakCommands message. In this case,
|
||
all of the command text is immediately available. In the second case, it is
|
||
called as a result of the user typing the 'command' command. The command
|
||
text then needs to be glommed out of UserInput messages (and possibly other
|
||
messages as well). The most 'straighforward' way of doing this is to
|
||
basically simulate the main loop, but just accumulate the command text
|
||
instead of sending it to execute_command(). */
|
||
|
||
char *
|
||
energize_command_line_input(prompt, repeat)
|
||
char *prompt;
|
||
int repeat;
|
||
{
|
||
char *p;
|
||
|
||
if (!energize)
|
||
return command_line_input(prompt, repeat);
|
||
|
||
if (doing_breakcommands_message)
|
||
{
|
||
if (command_line_length <= 0)
|
||
return (char *)NULL;
|
||
|
||
p = command_line_text;
|
||
|
||
while (command_line_length-- > 0)
|
||
{
|
||
if (*command_line_text == '\n')
|
||
{
|
||
*command_line_text = '\000';
|
||
command_line_text++;
|
||
break;
|
||
}
|
||
command_line_text++;
|
||
}
|
||
|
||
printf_filtered("%s\n", p);
|
||
return p;
|
||
}
|
||
else
|
||
{
|
||
/* We come here when the user has typed the 'command' or 'define' command
|
||
to the GDB window. We are basically deep inside of the 'command'
|
||
command processing routine right now, and will be called to get a new
|
||
line of input. We expect that kernel_dispatch will queue up only one
|
||
command at a time. */
|
||
|
||
int eventmask;
|
||
static char buf[100];
|
||
|
||
eventmask = wait_for_events(0);
|
||
|
||
if (eventmask & PTY_EVENT)
|
||
pty_to_kernel();
|
||
|
||
if (eventmask & KERNEL_EVENT)
|
||
kernel_dispatch(1); /* Queue up commands */
|
||
|
||
/* Note that command has been echoed by the time we get here */
|
||
|
||
p = dequeue_command();
|
||
|
||
if (p)
|
||
{
|
||
strncpy(buf, p, sizeof(buf));
|
||
free(p);
|
||
return buf;
|
||
}
|
||
else
|
||
return NULL;
|
||
}
|
||
}
|
||
|
||
/* Establish contact with the kernel. */
|
||
|
||
void
|
||
energize_initialize(energize_id, execarg)
|
||
char *energize_id;
|
||
char *execarg;
|
||
{
|
||
CTtyRequest *req;
|
||
char *ctmp;
|
||
extern long strtol(char *str, char **ptr, int base);
|
||
char pathname[MAXPATHLEN];
|
||
int n;
|
||
|
||
if (!energize_id)
|
||
return;
|
||
|
||
if (!execarg) execarg = "";
|
||
|
||
exec_file = strdup(execarg); /* Save for later */
|
||
|
||
printf("\ngdb-debugger pid=%d\n", getpid()); /* XXX - debugging only */
|
||
|
||
/* First establish the connection with the kernel. */
|
||
|
||
kerfd = COpenClientSocket(NULL);
|
||
if (kerfd < 0) {
|
||
printf("COpenClientSocket() failed\n");
|
||
exit(1);
|
||
}
|
||
|
||
/* Setup for I/O interrupts when appropriate. */
|
||
|
||
signal(SIGIO, SIG_IGN);
|
||
|
||
#ifdef NCR486
|
||
if (ioctl (kerfd, I_SETSIG, S_INPUT|S_RDNORM) < 0)
|
||
perror ("getpty: ioctl I_SETSIG failed");
|
||
#else
|
||
n = fcntl(kerfd, F_GETFL, 0);
|
||
fcntl(kerfd, F_SETFL, n|FASYNC);
|
||
fcntl(kerfd, F_SETOWN, getpid());
|
||
#endif
|
||
|
||
/* Setup connection buffering. */
|
||
|
||
CSetSocketBufferSize (kerfd, 12000);
|
||
|
||
/* Generate a new connection control block. */
|
||
|
||
conn = NewConnection (0, kerfd, kerfd);
|
||
if (!conn) {
|
||
printf("NewConnection() failed\n");
|
||
exit(1);
|
||
}
|
||
|
||
#ifdef KERNEL_RECORD
|
||
kerout = fopen("kernel.output", "+w");
|
||
|
||
CReadWriteHooks(conn, conn->previewMethod, conn->readMethod, kernel_record);
|
||
#endif
|
||
|
||
/* Tell the kernel that we are the "debugger". */
|
||
|
||
req = CWriteTtyRequest (conn, QueryConnectionRType);
|
||
req->generic.queryconnection.major = 0;
|
||
req->generic.queryconnection.minor = 0;
|
||
req->generic.queryconnection.cadillacId1=strtol(energize_id, &ctmp, 16);
|
||
req->generic.queryconnection.cadillacId2 = strtol(++ctmp, NULL, 16);
|
||
req->generic.queryconnection.nProtocols = 1;
|
||
CWriteProtocol (conn, 0, 0, "debugger");
|
||
CWriteLength (conn);
|
||
|
||
/* Tell the kernel that we are actually running. */
|
||
|
||
/* KROCK ALERT!!! The kernel doesn't really care about the arguments to
|
||
the program at all! It only cares that argument 7 be the name of the
|
||
target program. So, we just fill in the rest of the slots with
|
||
padding. I hope the kernel never cares about this! */
|
||
|
||
req = CWriteTtyRequest (conn, RunningProgramRType);
|
||
req->runningprogram.argc = 8;
|
||
getcwd (pathname, MAXPATHLEN);
|
||
CWriteVstring0 (conn, pathname);
|
||
|
||
CWriteVstring0 (conn, "0");
|
||
CWriteVstring0 (conn, "1");
|
||
CWriteVstring0 (conn, "2");
|
||
CWriteVstring0 (conn, "3");
|
||
CWriteVstring0 (conn, "4");
|
||
CWriteVstring0 (conn, "5");
|
||
CWriteVstring0 (conn, "6");
|
||
CWriteVstring0 (conn, execarg);
|
||
CWriteLength (conn);
|
||
|
||
/* Tell the kernel our PID and all that */
|
||
|
||
program_id = 1;
|
||
CVWriteDebugProgramInfo(conn,
|
||
getpid(),
|
||
program_id,
|
||
execarg,
|
||
"");
|
||
|
||
/* Tell the rest of the world that Energize is now set up. */
|
||
energize = 1;
|
||
|
||
getpty(); /* Setup the pty */
|
||
|
||
/* Attach all GDB I/O to the pty */
|
||
|
||
dup2(inferior_tty, 0);
|
||
dup2(inferior_tty, 1);
|
||
dup2(inferior_tty, 2);
|
||
}
|
||
|
||
/* This is called from execute_command, and provides a wrapper around
|
||
various command routines in a place where both protocol messages and
|
||
user input both flow through.
|
||
*/
|
||
|
||
void
|
||
energize_call_command(cmdblk, arg, from_tty)
|
||
struct cmd_list_element *cmdblk;
|
||
char *arg;
|
||
int from_tty;
|
||
{
|
||
if (!energize)
|
||
{
|
||
(*cmdblk->function.cfunc) (arg, from_tty);
|
||
return;
|
||
}
|
||
|
||
if (cmdblk->class == class_run)
|
||
{
|
||
go_busy();
|
||
has_run = 1;
|
||
(*cmdblk->function.cfunc)(arg, from_tty);
|
||
send_status();
|
||
}
|
||
else
|
||
(*cmdblk->function.cfunc)(arg, from_tty);
|
||
|
||
if (STREQ(cmdblk->name, "up")
|
||
|| STREQ(cmdblk->name, "down")
|
||
|| STREQ(cmdblk->name, "frame"))
|
||
send_location(get_frame_info(selected_frame)->pc,
|
||
selected_frame_level);
|
||
print_prompt();
|
||
}
|
||
|
||
void
|
||
energize_new_process()
|
||
{
|
||
instance_id = inferior_pid;
|
||
}
|
||
|
||
static void
|
||
iosig(signo)
|
||
int signo;
|
||
{
|
||
while (1)
|
||
{
|
||
int eventmask;
|
||
|
||
eventmask = wait_for_events(1);
|
||
|
||
if (eventmask == 0)
|
||
return;
|
||
|
||
if (eventmask & PTY_EVENT)
|
||
pty_to_kernel();
|
||
|
||
if (eventmask & KERNEL_EVENT)
|
||
kernel_dispatch(1);
|
||
}
|
||
}
|
||
|
||
int
|
||
energize_wait(status)
|
||
int *status;
|
||
{
|
||
int pid;
|
||
struct sigaction action;
|
||
static sigset_t nullsigmask = {0};
|
||
|
||
if (!energize)
|
||
return target_wait(status);
|
||
|
||
#ifdef NCR486
|
||
action.sa_handler = iosig;
|
||
action.sa_mask = nullsigmask;
|
||
action.sa_flags = SA_RESTART;
|
||
sigaction(SIGIO, &action, NULL);
|
||
#else
|
||
signal(SIGIO, iosig);
|
||
#endif
|
||
|
||
pid = target_wait(status);
|
||
|
||
signal(SIGIO, SIG_IGN);
|
||
return pid;
|
||
}
|
||
|
||
int
|
||
energize_shell_wait(status)
|
||
int *status;
|
||
{
|
||
int pid;
|
||
struct sigaction action;
|
||
static sigset_t nullsigmask = {0};
|
||
|
||
if (!energize)
|
||
return wait(status);
|
||
|
||
#ifdef NCR486
|
||
action.sa_handler = iosig;
|
||
action.sa_mask = nullsigmask;
|
||
action.sa_flags = SA_RESTART;
|
||
sigaction(SIGIO, &action, NULL);
|
||
#else
|
||
signal(SIGIO, iosig);
|
||
#endif
|
||
|
||
pid = wait(status);
|
||
|
||
signal(SIGIO, SIG_IGN);
|
||
return pid;
|
||
}
|
||
|
||
static void
|
||
null_routine(arg)
|
||
int arg;
|
||
{
|
||
}
|
||
|
||
/* All requests from the Energize kernel eventually end up here. */
|
||
|
||
void
|
||
energize_main_loop()
|
||
{
|
||
CTtyRequest *req;
|
||
struct cleanup *old_chain;
|
||
|
||
doing_breakcommands_message = 0;
|
||
|
||
/* We will come thru here any time there is an error, so send status if
|
||
necessary. */
|
||
|
||
send_status();
|
||
|
||
print_prompt();
|
||
|
||
/* The actual event loop! */
|
||
|
||
while (1)
|
||
{
|
||
int eventmask;
|
||
char *cmd;
|
||
|
||
old_chain = make_cleanup(null_routine, 0);
|
||
|
||
/* First, empty out the command queue, then check for new requests. */
|
||
|
||
while (cmd = dequeue_command())
|
||
{
|
||
execute_command_1(1, 0, cmd);
|
||
free(cmd);
|
||
}
|
||
|
||
eventmask = wait_for_events(0);
|
||
|
||
if (eventmask & PTY_EVENT)
|
||
pty_to_kernel();
|
||
|
||
if (eventmask & KERNEL_EVENT)
|
||
kernel_dispatch(0);
|
||
|
||
bpstat_do_actions(&stop_bpstat);
|
||
do_cleanups(old_chain);
|
||
}
|
||
}
|