1 /* Startup code for gdbtk.
2 Copyright 1994, 1995, 1996, 1997, 1998 Free Software Foundation, Inc.
4 Written by Stu Grossman <grossman@cygnus.com> of Cygnus Support.
6 This file is part of GDB.
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
31 #include "tracepoint.h"
48 /* start-sanitize-ide */
52 /* end-sanitize-ide */
55 #ifdef ANSI_PROTOTYPES
65 #include <sys/ioctl.h>
66 #include "gdb_string.h"
76 extern int (*ui_load_progress_hook
) PARAMS ((char *, unsigned long));
77 extern void (*pre_add_symbol_hook
) PARAMS ((char *));
78 extern void (*post_add_symbol_hook
) PARAMS ((void));
79 extern void (*selected_frame_level_changed_hook
) PARAMS ((int));
81 extern void (*ui_loop_hook
) PARAMS ((int));
84 static void gdbtk_create_tracepoint
PARAMS ((struct tracepoint
*));
85 static void gdbtk_delete_tracepoint
PARAMS ((struct tracepoint
*));
86 static void gdbtk_modify_tracepoint
PARAMS ((struct tracepoint
*));
87 static void gdbtk_trace_find
PARAMS ((char *arg
, int from_tty
));
88 static void gdbtk_trace_start_stop
PARAMS ((int, int));
89 static void gdbtk_create_breakpoint
PARAMS ((struct breakpoint
*));
90 static void gdbtk_delete_breakpoint
PARAMS ((struct breakpoint
*));
91 static void gdbtk_modify_breakpoint
PARAMS ((struct breakpoint
*));
92 static void gdbtk_file_changed
PARAMS ((char *));
93 static void gdbtk_exec_file_display
PARAMS ((char *));
94 static void tk_command_loop
PARAMS ((void));
95 static void gdbtk_call_command
PARAMS ((struct cmd_list_element
*, char *, int));
96 static int gdbtk_wait
PARAMS ((int, struct target_waitstatus
*));
97 void x_event
PARAMS ((int));
98 static int gdbtk_query
PARAMS ((const char *, va_list));
99 static void gdbtk_warning
PARAMS ((const char *, va_list));
100 void gdbtk_ignorable_warning
PARAMS ((const char *));
101 static char* gdbtk_readline
PARAMS ((char *));
103 #ifdef ANSI_PROTOTYPES
104 gdbtk_readline_begin (char *format
, ...);
106 gdbtk_readline_begin ();
108 static void gdbtk_readline_end
PARAMS ((void));
109 static void gdbtk_flush
PARAMS ((GDB_FILE
*));
110 static void gdbtk_pre_add_symbol
PARAMS ((char *));
111 static void gdbtk_print_frame_info
PARAMS ((struct symtab
*, int, int, int));
112 static void gdbtk_post_add_symbol
PARAMS ((void));
113 static void gdbtk_register_changed
PARAMS ((int regno
));
114 static void gdbtk_memory_changed
PARAMS ((CORE_ADDR addr
, int len
));
115 static void tracepoint_notify
PARAMS ((struct tracepoint
*, const char *));
116 static void gdbtk_selected_frame_changed
PARAMS ((int));
117 static void gdbtk_context_change
PARAMS ((int));
118 static void gdbtk_error_begin
PARAMS ((void));
121 * gdbtk_fputs can't be static, because we need to call it in gdbtk.c.
122 * See note there for details.
125 void gdbtk_fputs
PARAMS ((const char *, GDB_FILE
*));
126 int gdbtk_load_hash
PARAMS ((char *, unsigned long));
127 static void breakpoint_notify
PARAMS ((struct breakpoint
*, const char *));
130 * gdbtk_add_hooks - add all the hooks to gdb. This will get called by the
131 * startup code to fill in the hooks needed by core gdb.
135 gdbtk_add_hooks(void)
137 command_loop_hook
= tk_command_loop
;
138 call_command_hook
= gdbtk_call_command
;
139 readline_begin_hook
= gdbtk_readline_begin
;
140 readline_hook
= gdbtk_readline
;
141 readline_end_hook
= gdbtk_readline_end
;
143 print_frame_info_listing_hook
= gdbtk_print_frame_info
;
144 query_hook
= gdbtk_query
;
145 warning_hook
= gdbtk_warning
;
146 flush_hook
= gdbtk_flush
;
148 create_breakpoint_hook
= gdbtk_create_breakpoint
;
149 delete_breakpoint_hook
= gdbtk_delete_breakpoint
;
150 modify_breakpoint_hook
= gdbtk_modify_breakpoint
;
152 interactive_hook
= gdbtk_interactive
;
153 target_wait_hook
= gdbtk_wait
;
154 ui_load_progress_hook
= gdbtk_load_hash
;
157 ui_loop_hook
= x_event
;
159 pre_add_symbol_hook
= gdbtk_pre_add_symbol
;
160 post_add_symbol_hook
= gdbtk_post_add_symbol
;
161 file_changed_hook
= gdbtk_file_changed
;
162 exec_file_display_hook
= gdbtk_exec_file_display
;
164 create_tracepoint_hook
= gdbtk_create_tracepoint
;
165 delete_tracepoint_hook
= gdbtk_delete_tracepoint
;
166 modify_tracepoint_hook
= gdbtk_modify_tracepoint
;
167 trace_find_hook
= gdbtk_trace_find
;
168 trace_start_stop_hook
= gdbtk_trace_start_stop
;
170 register_changed_hook
= gdbtk_register_changed
;
171 memory_changed_hook
= gdbtk_memory_changed
;
172 selected_frame_level_changed_hook
= gdbtk_selected_frame_changed
;
173 context_hook
= gdbtk_context_change
;
175 error_begin_hook
= gdbtk_error_begin
;
178 /* These control where to put the gdb output which is created by
179 {f}printf_{un}filtered and friends. gdbtk_fputs and gdbtk_flush are the
180 lowest level of these routines and capture all output from the rest of GDB.
182 The reason to use the result_ptr rather than the gdbtk_interp's result
183 directly is so that a call_wrapper invoked function can preserve its result
184 across calls into Tcl which might be made in the course of the function's
187 * result_ptr->obj_ptr is where to accumulate the result.
188 * GDBTK_TO_RESULT flag means the output goes to the gdbtk_tcl_fputs proc
189 instead of to the result_ptr.
190 * GDBTK_MAKES_LIST flag means add to the result as a list element.
194 gdbtk_result
*result_ptr
= NULL
;
197 /* This allows you to Tcl_Eval a tcl command which takes
198 a command word, and then a single argument. */
200 int gdbtk_two_elem_cmd (cmd_name
, argv1
)
205 int result
, flags_ptr
, arg_len
, cmd_len
;
207 arg_len
= Tcl_ScanElement (argv1
, &flags_ptr
);
208 cmd_len
= strlen (cmd_name
);
209 command
= malloc(arg_len
+ cmd_len
+ 2);
210 strcpy (command
, cmd_name
);
211 strcat (command
, " ");
213 Tcl_ConvertElement (argv1
, command
+ cmd_len
+ 1, flags_ptr
);
215 result
= Tcl_Eval (gdbtk_interp
, command
);
226 /* Force immediate screen update */
228 Tcl_VarEval (gdbtk_interp
, "gdbtk_tcl_flush", NULL
);
232 /* This handles all the output from gdb. All the gdb printf_xxx functions
233 * eventually end up here. The output is either passed to the result_ptr
234 * where it will go to the result of some gdbtk command, or passed to the
235 * Tcl proc gdbtk_tcl_fputs (where it is usually just dumped to the console
240 * 1) result_ptr == NULL - This happens when some output comes from gdb which
241 * is not generated by a command in gdbtk-cmds, usually startup stuff.
242 * In this case we just route the data to gdbtk_tcl_fputs.
243 * 2) The GDBTK_TO_RESULT flag is set - The result is supposed to go to Tcl.
244 * We place the data into the result_ptr, either as a string,
245 * or a list, depending whether the GDBTK_MAKES_LIST bit is set.
246 * 3) The GDBTK_TO_RESULT flag is unset - We route the data to gdbtk_tcl_fputs
247 * UNLESS it was coming to gdb_stderr. Then we place it in the result_ptr
248 * anyway, so it can be dealt with.
253 gdbtk_fputs (ptr
, stream
)
259 if (result_ptr
!= NULL
)
261 if (result_ptr
->flags
& GDBTK_TO_RESULT
)
263 if (result_ptr
->flags
& GDBTK_MAKES_LIST
)
264 Tcl_ListObjAppendElement(NULL
, result_ptr
->obj_ptr
,
265 Tcl_NewStringObj((char *) ptr
, -1));
267 Tcl_AppendToObj (result_ptr
->obj_ptr
, (char *) ptr
, -1);
269 else if (stream
== gdb_stderr
|| result_ptr
->flags
& GDBTK_ERROR_ONLY
)
271 if (result_ptr
->flags
& GDBTK_ERROR_STARTED
)
272 Tcl_AppendToObj (result_ptr
->obj_ptr
, (char *) ptr
, -1);
275 Tcl_SetStringObj (result_ptr
->obj_ptr
, (char *) ptr
, -1);
276 result_ptr
->flags
|= GDBTK_ERROR_STARTED
;
281 gdbtk_two_elem_cmd ("gdbtk_tcl_fputs", (char *) ptr
);
282 if (result_ptr
->flags
& GDBTK_MAKES_LIST
)
283 gdbtk_two_elem_cmd ("gdbtk_tcl_fputs", " ");
288 gdbtk_two_elem_cmd ("gdbtk_tcl_fputs", (char *) ptr
);
295 * This routes all warnings to the Tcl function "gdbtk_tcl_warning".
299 gdbtk_warning (warning
, args
)
305 vsprintf (buf
, warning
, args
);
306 gdbtk_two_elem_cmd ("gdbtk_tcl_warning", buf
);
311 * This routes all ignorable warnings to the Tcl function
312 * "gdbtk_tcl_ignorable_warning".
316 gdbtk_ignorable_warning (warning
)
320 sprintf (buf
, warning
);
321 gdbtk_two_elem_cmd ("gdbtk_tcl_ignorable_warning", buf
);
325 gdbtk_register_changed(regno
)
328 Tcl_Eval (gdbtk_interp
, "gdbtk_register_changed");
332 gdbtk_memory_changed(addr
, len
)
336 Tcl_Eval (gdbtk_interp
, "gdbtk_memory_changed");
340 /* This function is called instead of gdb's internal command loop. This is the
341 last chance to do anything before entering the main Tk event loop.
342 At the end of the command, we enter the main loop. */
347 extern FILE *instream
;
349 /* We no longer want to use stdin as the command input stream */
352 if (Tcl_Eval (gdbtk_interp
, "gdbtk_tcl_preloop") != TCL_OK
)
356 /* Force errorInfo to be set up propertly. */
357 Tcl_AddErrorInfo (gdbtk_interp
, "");
359 msg
= Tcl_GetVar (gdbtk_interp
, "errorInfo", TCL_GLOBAL_ONLY
);
361 MessageBox (NULL
, msg
, NULL
, MB_OK
| MB_ICONERROR
| MB_TASKMODAL
);
363 fputs_unfiltered (msg
, gdb_stderr
);
374 /* Come here when there is activity on the X file descriptor. */
380 static int in_x_event
= 0;
381 static Tcl_Obj
*varname
= NULL
;
382 if (in_x_event
|| in_fputs
)
392 /* Process pending events */
393 while (Tcl_DoOneEvent (TCL_DONT_WAIT
|TCL_ALL_EVENTS
) != 0)
396 if (load_in_progress
)
401 Tcl_Obj
*varnamestrobj
= Tcl_NewStringObj("download_cancel_ok",-1);
402 varname
= Tcl_ObjGetVar2(gdbtk_interp
,varnamestrobj
,NULL
,TCL_GLOBAL_ONLY
);
404 if ((Tcl_GetIntFromObj(gdbtk_interp
,varname
,&val
) == TCL_OK
) && val
)
420 #ifdef ANSI_PROTOTYPES
421 gdbtk_readline_begin (char *format
, ...)
423 gdbtk_readline_begin (va_alist
)
430 #ifdef ANSI_PROTOTYPES
431 va_start (args
, format
);
435 format
= va_arg (args
, char *);
438 vsprintf (buf
, format
, args
);
439 gdbtk_two_elem_cmd ("gdbtk_tcl_readline_begin", buf
);
444 gdbtk_readline (prompt
)
453 result
= gdbtk_two_elem_cmd ("gdbtk_tcl_readline", prompt
);
455 if (result
== TCL_OK
)
457 return (strdup (gdbtk_interp
-> result
));
461 gdbtk_fputs (gdbtk_interp
-> result
, gdb_stdout
);
462 gdbtk_fputs ("\n", gdb_stdout
);
468 gdbtk_readline_end ()
470 Tcl_Eval (gdbtk_interp
, "gdbtk_tcl_readline_end");
474 gdbtk_call_command (cmdblk
, arg
, from_tty
)
475 struct cmd_list_element
*cmdblk
;
480 if (cmdblk
->class == class_run
|| cmdblk
->class == class_trace
)
485 Tcl_Eval (gdbtk_interp
, "gdbtk_tcl_busy");
486 (*cmdblk
->function
.cfunc
)(arg
, from_tty
);
489 Tcl_Eval (gdbtk_interp
, "gdbtk_tcl_idle");
492 (*cmdblk
->function
.cfunc
)(arg
, from_tty
);
495 /* The next three functions use breakpoint_notify to allow the GUI
496 * to handle creating, deleting and modifying breakpoints. These three
497 * functions are put into the appropriate gdb hooks in gdbtk_init.
501 gdbtk_create_breakpoint(b
)
502 struct breakpoint
*b
;
504 breakpoint_notify (b
, "create");
508 gdbtk_delete_breakpoint(b
)
509 struct breakpoint
*b
;
511 breakpoint_notify (b
, "delete");
515 gdbtk_modify_breakpoint(b
)
516 struct breakpoint
*b
;
518 breakpoint_notify (b
, "modify");
521 /* This is the generic function for handling changes in
522 * a breakpoint. It routes the information to the Tcl
523 * command "gdbtk_tcl_breakpoint" in the form:
524 * gdbtk_tcl_breakpoint action b_number b_address b_line b_file
525 * On error, the error string is written to gdb_stdout.
529 breakpoint_notify(b
, action
)
530 struct breakpoint
*b
;
535 struct symtab_and_line sal
;
538 if (b
->type
!= bp_breakpoint
)
541 /* We ensure that ACTION contains no special Tcl characters, so we
543 sal
= find_pc_line (b
->address
, 0);
544 filename
= symtab_to_filename (sal
.symtab
);
545 if (filename
== NULL
)
548 sprintf (buf
, "gdbtk_tcl_breakpoint %s %d 0x%lx %d {%s} {%s} %d %d",
549 action
, b
->number
, (long)b
->address
, b
->line_number
, filename
,
550 bpdisp
[b
->disposition
], b
->enable
, b
->thread
);
552 v
= Tcl_Eval (gdbtk_interp
, buf
);
556 gdbtk_fputs (Tcl_GetStringResult (gdbtk_interp
), gdb_stdout
);
557 gdbtk_fputs ("\n", gdb_stdout
);
562 gdbtk_load_hash (section
, num
)
567 sprintf (buf
, "download_hash %s %ld", section
, num
);
568 Tcl_Eval (gdbtk_interp
, buf
);
569 return atoi (gdbtk_interp
->result
);
573 /* This hook is called whenever we are ready to load a symbol file so that
574 the UI can notify the user... */
576 gdbtk_pre_add_symbol (name
)
580 gdbtk_two_elem_cmd("gdbtk_tcl_pre_add_symbol", name
);
584 /* This hook is called whenever we finish loading a symbol file. */
586 gdbtk_post_add_symbol ()
588 Tcl_Eval (gdbtk_interp
, "gdbtk_tcl_post_add_symbol");
591 /* This hook function is called whenever we want to wait for the
595 gdbtk_wait (pid
, ourstatus
)
597 struct target_waitstatus
*ourstatus
;
599 /* Don't run the timer on various targets... */
600 if (!STREQ (target_shortname
, "ice"))
601 gdbtk_start_timer ();
602 pid
= target_wait (pid
, ourstatus
);
608 * This handles all queries from gdb.
609 * The first argument is a printf style format statement, the rest are its
610 * arguments. The resultant formatted string is passed to the Tcl function
612 * It returns the users response to the query, as well as putting the value
613 * in the result field of the Tcl interpreter.
617 gdbtk_query (query
, args
)
624 vsprintf (buf
, query
, args
);
625 gdbtk_two_elem_cmd ("gdbtk_tcl_query", buf
);
627 val
= atol (gdbtk_interp
->result
);
633 gdbtk_print_frame_info (s
, line
, stopline
, noerror
)
639 current_source_symtab
= s
;
640 current_source_line
= line
;
644 gdbtk_create_tracepoint (tp
)
645 struct tracepoint
*tp
;
647 tracepoint_notify (tp
, "create");
651 gdbtk_delete_tracepoint (tp
)
652 struct tracepoint
*tp
;
654 tracepoint_notify (tp
, "delete");
658 gdbtk_modify_tracepoint (tp
)
659 struct tracepoint
*tp
;
661 tracepoint_notify (tp
, "modify");
665 tracepoint_notify(tp
, action
)
666 struct tracepoint
*tp
;
671 struct symtab_and_line sal
;
674 /* We ensure that ACTION contains no special Tcl characters, so we
676 sal
= find_pc_line (tp
->address
, 0);
678 filename
= symtab_to_filename (sal
.symtab
);
679 if (filename
== NULL
)
681 sprintf (buf
, "gdbtk_tcl_tracepoint %s %d 0x%lx %d {%s} %d", action
, tp
->number
,
682 (long)tp
->address
, sal
.line
, filename
, tp
->pass_count
);
684 v
= Tcl_Eval (gdbtk_interp
, buf
);
688 gdbtk_fputs (gdbtk_interp
->result
, gdb_stdout
);
689 gdbtk_fputs ("\n", gdb_stdout
);
696 * This is run by the trace_find_command. arg is the argument that was passed
697 * to that command, from_tty is 1 if the command was run from a tty, 0 if it
698 * was run from a script. It runs gdbtk_tcl_tfind_hook passing on these two
704 gdbtk_trace_find (arg
, from_tty
)
711 Tcl_GlobalEval (gdbtk_interp
, "debug {*** In gdbtk_trace_find, from_tty is true}");
712 cmdObj
= Tcl_NewListObj (0, NULL
);
713 Tcl_ListObjAppendElement (gdbtk_interp
, cmdObj
,
714 Tcl_NewStringObj ("gdbtk_tcl_trace_find_hook", -1));
715 Tcl_ListObjAppendElement (gdbtk_interp
, cmdObj
, Tcl_NewStringObj (arg
, -1));
716 Tcl_ListObjAppendElement (gdbtk_interp
, cmdObj
, Tcl_NewIntObj(from_tty
));
717 Tcl_GlobalEvalObj (gdbtk_interp
, cmdObj
);
719 Tcl_GlobalEval (gdbtk_interp
, "debug {*** In gdbtk_trace_find, from_tty is false}");
724 * gdbtk_trace_start_stop
726 * This is run by the trace_start_command and trace_stop_command.
727 * The START variable determines which, 1 meaning trace_start was run,
728 * 0 meaning trace_stop was run.
733 gdbtk_trace_start_stop (start
, from_tty
)
740 Tcl_GlobalEval (gdbtk_interp
, "debug {*** In gdbtk_trace_start, from_tty is true}");
741 cmdObj
= Tcl_NewListObj (0, NULL
);
743 Tcl_ListObjAppendElement (gdbtk_interp
, cmdObj
,
744 Tcl_NewStringObj ("gdbtk_tcl_tstart", -1));
746 Tcl_ListObjAppendElement (gdbtk_interp
, cmdObj
,
747 Tcl_NewStringObj ("gdbtk_tcl_tstop", -1));
748 Tcl_GlobalEvalObj (gdbtk_interp
, cmdObj
);
750 Tcl_GlobalEval (gdbtk_interp
, "debug {*** In gdbtk_trace_startd, from_tty is false}");
756 gdbtk_selected_frame_changed (level
)
759 Tcl_UpdateLinkedVar (gdbtk_interp
, "gdb_selected_frame_level");
762 /* Called when the current thread changes. */
763 /* gdb_context is linked to the tcl variable "gdb_context_id" */
765 gdbtk_context_change (num
)
771 /* Called from file_command */
773 gdbtk_file_changed (filename
)
776 gdbtk_two_elem_cmd ("gdbtk_tcl_file_changed", filename
);
779 /* Called from exec_file_command */
781 gdbtk_exec_file_display (filename
)
784 gdbtk_two_elem_cmd ("gdbtk_tcl_exec_file_display", filename
);
787 /* Called from error_begin, this hook is used to warn the gui
788 about multi-line error messages */
792 if (result_ptr
!= NULL
)
793 result_ptr
->flags
|= GDBTK_ERROR_ONLY
;
796 /* Local variables: */
797 /* change-log-default-name: "ChangeLog-gdbtk" */