mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-18 12:24:38 +08:00
53f1f3d4aa
This commit contains QUIT processing updates for GDB's Guile support. As with the Python updates, we don't want to permit this code to swallow the exception, gdb_exception_forced_quit, which is associated with GDB receiving a SIGTERM. I've adopted the same solution that I used for Python; whereever a gdb_exception is caught in try/catch code in the Guile extension language support, a catch for gdb_exception_forced_quit has been added; this catch block will simply call quit_force(), which will cause the necessary cleanups to occur followed by GDB exiting. Bug: https://sourceware.org/bugzilla/show_bug.cgi?id=26761 Tested-by: Tom de Vries <tdevries@suse.de> Approved-By: Pedro Alves <pedro@palves.net>
726 lines
21 KiB
C++
726 lines
21 KiB
C++
/* Internal header for GDB/Scheme code.
|
||
|
||
Copyright (C) 2014-2023 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 3 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, see <http://www.gnu.org/licenses/>. */
|
||
|
||
#ifndef GUILE_GUILE_INTERNAL_H
|
||
#define GUILE_GUILE_INTERNAL_H
|
||
|
||
/* See README file in this directory for implementation notes, coding
|
||
conventions, et.al. */
|
||
|
||
|
||
#include "hashtab.h"
|
||
#include "extension-priv.h"
|
||
#include "symtab.h"
|
||
#include "libguile.h"
|
||
#include "objfiles.h"
|
||
#include "top.h" /* For quit_force(). */
|
||
|
||
struct block;
|
||
struct frame_info;
|
||
struct objfile;
|
||
struct symbol;
|
||
|
||
/* A function to pass to the safe-call routines to ignore things like
|
||
memory errors. */
|
||
typedef int excp_matcher_func (SCM key);
|
||
|
||
/* Scheme variables to define during initialization. */
|
||
|
||
struct scheme_variable
|
||
{
|
||
const char *name;
|
||
SCM value;
|
||
const char *doc_string;
|
||
};
|
||
|
||
/* End of scheme_variable table mark. */
|
||
|
||
#define END_VARIABLES { NULL, SCM_BOOL_F, NULL }
|
||
|
||
/* Although scm_t_subr is meant to hold a function pointer, at least
|
||
in some versions of guile, it is actually a typedef to "void *".
|
||
That means that in C++, an explicit cast is necessary to convert
|
||
function pointer to scm_t_subr. But a cast also makes it possible
|
||
to pass function pointers with the wrong type by mistake. So
|
||
instead of adding such casts throughout, we use 'as_a_scm_t_subr'
|
||
to do the conversion, which (only) has overloads for function
|
||
pointer types that are valid.
|
||
|
||
See https://lists.gnu.org/archive/html/guile-devel/2013-03/msg00001.html.
|
||
*/
|
||
|
||
static inline scm_t_subr
|
||
as_a_scm_t_subr (SCM (*func) (void))
|
||
{
|
||
return (scm_t_subr) func;
|
||
}
|
||
|
||
static inline scm_t_subr
|
||
as_a_scm_t_subr (SCM (*func) (SCM))
|
||
{
|
||
return (scm_t_subr) func;
|
||
}
|
||
|
||
static inline scm_t_subr
|
||
as_a_scm_t_subr (SCM (*func) (SCM, SCM))
|
||
{
|
||
return (scm_t_subr) func;
|
||
}
|
||
|
||
static inline scm_t_subr
|
||
as_a_scm_t_subr (SCM (*func) (SCM, SCM, SCM))
|
||
{
|
||
return (scm_t_subr) func;
|
||
}
|
||
|
||
/* Scheme functions to define during initialization. */
|
||
|
||
struct scheme_function
|
||
{
|
||
const char *name;
|
||
int required;
|
||
int optional;
|
||
int rest;
|
||
scm_t_subr func;
|
||
const char *doc_string;
|
||
};
|
||
|
||
/* End of scheme_function table mark. */
|
||
|
||
#define END_FUNCTIONS { NULL, 0, 0, 0, NULL, NULL }
|
||
|
||
/* Useful for defining a set of constants. */
|
||
|
||
struct scheme_integer_constant
|
||
{
|
||
const char *name;
|
||
int value;
|
||
};
|
||
|
||
#define END_INTEGER_CONSTANTS { NULL, 0 }
|
||
|
||
/* Pass this instead of 0 to routines like SCM_ASSERT to indicate the value
|
||
is not a function argument. */
|
||
#define GDBSCM_ARG_NONE 0
|
||
|
||
/* Ensure new code doesn't accidentally try to use this. */
|
||
#undef scm_make_smob_type
|
||
#define scm_make_smob_type USE_gdbscm_make_smob_type_INSTEAD
|
||
|
||
/* They brought over () == #f from lisp.
|
||
Let's avoid that for now. */
|
||
#undef scm_is_bool
|
||
#undef scm_is_false
|
||
#undef scm_is_true
|
||
#define scm_is_bool USE_gdbscm_is_bool_INSTEAD
|
||
#define scm_is_false USE_gdbscm_is_false_INSTEAD
|
||
#define scm_is_true USE_gdbscm_is_true_INSTEAD
|
||
#define gdbscm_is_bool(scm) \
|
||
(scm_is_eq ((scm), SCM_BOOL_F) || scm_is_eq ((scm), SCM_BOOL_T))
|
||
#define gdbscm_is_false(scm) scm_is_eq ((scm), SCM_BOOL_F)
|
||
#define gdbscm_is_true(scm) (!gdbscm_is_false (scm))
|
||
|
||
#ifndef HAVE_SCM_NEW_SMOB
|
||
|
||
/* Guile <= 2.0.5 did not provide this function, so provide it here. */
|
||
|
||
static inline SCM
|
||
scm_new_smob (scm_t_bits tc, scm_t_bits data)
|
||
{
|
||
SCM_RETURN_NEWSMOB (tc, data);
|
||
}
|
||
|
||
#endif
|
||
|
||
/* Function name that is passed around in case an error needs to be reported.
|
||
__func is in C99, but we provide a wrapper "just in case",
|
||
and because FUNC_NAME is the canonical value used in guile sources.
|
||
IWBN to use the Scheme version of the name (e.g. foo-bar vs foo_bar),
|
||
but let's KISS for now. */
|
||
#define FUNC_NAME __func__
|
||
|
||
extern const char gdbscm_module_name[];
|
||
extern const char gdbscm_init_module_name[];
|
||
|
||
extern int gdb_scheme_initialized;
|
||
|
||
extern int gdbscm_guile_major_version;
|
||
extern int gdbscm_guile_minor_version;
|
||
extern int gdbscm_guile_micro_version;
|
||
|
||
extern const char gdbscm_print_excp_none[];
|
||
extern const char gdbscm_print_excp_full[];
|
||
extern const char gdbscm_print_excp_message[];
|
||
extern const char *gdbscm_print_excp;
|
||
|
||
extern SCM gdbscm_documentation_symbol;
|
||
extern SCM gdbscm_invalid_object_error_symbol;
|
||
|
||
extern SCM gdbscm_map_string;
|
||
extern SCM gdbscm_array_string;
|
||
extern SCM gdbscm_string_string;
|
||
|
||
/* scm-utils.c */
|
||
|
||
extern void gdbscm_define_variables (const scheme_variable *, int is_public);
|
||
|
||
extern void gdbscm_define_functions (const scheme_function *, int is_public);
|
||
|
||
extern void gdbscm_define_integer_constants (const scheme_integer_constant *,
|
||
int is_public);
|
||
|
||
extern void gdbscm_printf (SCM port, const char *format, ...)
|
||
ATTRIBUTE_PRINTF (2, 3);
|
||
|
||
extern void gdbscm_debug_display (SCM obj);
|
||
|
||
extern void gdbscm_debug_write (SCM obj);
|
||
|
||
extern void gdbscm_parse_function_args (const char *function_name,
|
||
int beginning_arg_pos,
|
||
const SCM *keywords,
|
||
const char *format, ...);
|
||
|
||
extern SCM gdbscm_scm_from_longest (LONGEST l);
|
||
|
||
extern LONGEST gdbscm_scm_to_longest (SCM l);
|
||
|
||
extern SCM gdbscm_scm_from_ulongest (ULONGEST l);
|
||
|
||
extern ULONGEST gdbscm_scm_to_ulongest (SCM u);
|
||
|
||
extern void gdbscm_dynwind_xfree (void *ptr);
|
||
|
||
extern int gdbscm_is_procedure (SCM proc);
|
||
|
||
extern char *gdbscm_gc_xstrdup (const char *);
|
||
|
||
extern const char * const *gdbscm_gc_dup_argv (char **argv);
|
||
|
||
extern int gdbscm_guile_version_is_at_least (int major, int minor, int micro);
|
||
|
||
/* GDB smobs, from scm-gsmob.c */
|
||
|
||
/* All gdb smobs must contain one of the following as the first member:
|
||
gdb_smob, chained_gdb_smob, or eqable_gdb_smob.
|
||
|
||
Chained GDB smobs should have chained_gdb_smob as their first member. The
|
||
next,prev members of chained_gdb_smob allow for chaining gsmobs together so
|
||
that, for example, when an objfile is deleted we can clean up all smobs that
|
||
reference it.
|
||
|
||
Eq-able GDB smobs should have eqable_gdb_smob as their first member. The
|
||
containing_scm member of eqable_gdb_smob allows for returning the same gsmob
|
||
instead of creating a new one, allowing them to be eq?-able.
|
||
|
||
All other smobs should have gdb_smob as their first member.
|
||
FIXME: dje/2014-05-26: gdb_smob was useful during early development as a
|
||
"baseclass" for all gdb smobs. If it's still unused by gdb 8.0 delete it.
|
||
|
||
IMPORTANT: chained_gdb_smob and eqable_gdb-smob are "subclasses" of
|
||
gdb_smob. The layout of chained_gdb_smob,eqable_gdb_smob must match
|
||
gdb_smob as if it is a subclass. To that end we use macro GDB_SMOB_HEAD
|
||
to ensure this. */
|
||
|
||
#define GDB_SMOB_HEAD \
|
||
int empty_base_class;
|
||
|
||
struct gdb_smob
|
||
{
|
||
GDB_SMOB_HEAD
|
||
};
|
||
|
||
struct chained_gdb_smob
|
||
{
|
||
GDB_SMOB_HEAD
|
||
|
||
chained_gdb_smob *prev;
|
||
chained_gdb_smob *next;
|
||
};
|
||
|
||
struct eqable_gdb_smob
|
||
{
|
||
GDB_SMOB_HEAD
|
||
|
||
/* The object we are contained in.
|
||
This can be used for several purposes.
|
||
This is used by the eq? machinery: We need to be able to see if we have
|
||
already created an object for a symbol, and if so use that SCM.
|
||
This may also be used to protect the smob from GC if there is
|
||
a reference to this smob from outside of GC space (i.e., from gdb).
|
||
This can also be used in place of chained_gdb_smob where we need to
|
||
keep track of objfile referencing objects. When the objfile is deleted
|
||
we need to invalidate the objects: we can do that using the same hashtab
|
||
used to record the smob for eq-ability. */
|
||
SCM containing_scm;
|
||
};
|
||
|
||
#undef GDB_SMOB_HEAD
|
||
|
||
struct objfile;
|
||
|
||
/* A predicate that returns non-zero if an object is a particular kind
|
||
of gsmob. */
|
||
typedef int (gsmob_pred_func) (SCM);
|
||
|
||
extern scm_t_bits gdbscm_make_smob_type (const char *name, size_t size);
|
||
|
||
extern void gdbscm_init_gsmob (gdb_smob *base);
|
||
|
||
extern void gdbscm_init_chained_gsmob (chained_gdb_smob *base);
|
||
|
||
extern void gdbscm_init_eqable_gsmob (eqable_gdb_smob *base,
|
||
SCM containing_scm);
|
||
|
||
extern htab_t gdbscm_create_eqable_gsmob_ptr_map (htab_hash hash_fn,
|
||
htab_eq eq_fn);
|
||
|
||
extern eqable_gdb_smob **gdbscm_find_eqable_gsmob_ptr_slot
|
||
(htab_t htab, eqable_gdb_smob *base);
|
||
|
||
extern void gdbscm_fill_eqable_gsmob_ptr_slot (eqable_gdb_smob **slot,
|
||
eqable_gdb_smob *base);
|
||
|
||
extern void gdbscm_clear_eqable_gsmob_ptr_slot (htab_t htab,
|
||
eqable_gdb_smob *base);
|
||
|
||
/* Exceptions and calling out to Guile. */
|
||
|
||
/* scm-exception.c */
|
||
|
||
extern SCM gdbscm_make_exception (SCM tag, SCM args);
|
||
|
||
extern int gdbscm_is_exception (SCM scm);
|
||
|
||
extern SCM gdbscm_exception_key (SCM excp);
|
||
|
||
extern SCM gdbscm_exception_args (SCM excp);
|
||
|
||
extern SCM gdbscm_make_exception_with_stack (SCM key, SCM args, SCM stack);
|
||
|
||
extern SCM gdbscm_make_error_scm (SCM key, SCM subr, SCM message,
|
||
SCM args, SCM data);
|
||
|
||
extern SCM gdbscm_make_error (SCM key, const char *subr, const char *message,
|
||
SCM args, SCM data);
|
||
|
||
extern SCM gdbscm_make_type_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *expected_type);
|
||
|
||
extern SCM gdbscm_make_invalid_object_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error);
|
||
|
||
extern void gdbscm_invalid_object_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error)
|
||
ATTRIBUTE_NORETURN;
|
||
|
||
extern SCM gdbscm_make_out_of_range_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error);
|
||
|
||
extern void gdbscm_out_of_range_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error)
|
||
ATTRIBUTE_NORETURN;
|
||
|
||
extern SCM gdbscm_make_misc_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error);
|
||
|
||
extern void gdbscm_misc_error (const char *subr, int arg_pos,
|
||
SCM bad_value, const char *error)
|
||
ATTRIBUTE_NORETURN;
|
||
|
||
extern void gdbscm_throw (SCM exception) ATTRIBUTE_NORETURN;
|
||
|
||
struct gdbscm_gdb_exception;
|
||
extern SCM gdbscm_scm_from_gdb_exception
|
||
(const gdbscm_gdb_exception &exception);
|
||
|
||
extern void gdbscm_throw_gdb_exception (gdbscm_gdb_exception exception)
|
||
ATTRIBUTE_NORETURN;
|
||
|
||
extern void gdbscm_print_exception_with_stack (SCM port, SCM stack,
|
||
SCM key, SCM args);
|
||
|
||
extern void gdbscm_print_gdb_exception (SCM port, SCM exception);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_exception_message_to_string
|
||
(SCM exception);
|
||
|
||
extern excp_matcher_func gdbscm_memory_error_p;
|
||
|
||
extern excp_matcher_func gdbscm_user_error_p;
|
||
|
||
extern SCM gdbscm_make_memory_error (const char *subr, const char *msg,
|
||
SCM args);
|
||
|
||
extern void gdbscm_memory_error (const char *subr, const char *msg, SCM args)
|
||
ATTRIBUTE_NORETURN;
|
||
|
||
/* scm-safe-call.c */
|
||
|
||
extern const char *gdbscm_with_guile (const char *(*func) (void *), void *data);
|
||
|
||
extern SCM gdbscm_call_guile (SCM (*func) (void *), void *data,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_call_0 (SCM proc, excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_call_1 (SCM proc, SCM arg0,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_call_2 (SCM proc, SCM arg0, SCM arg1,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_call_3 (SCM proc, SCM arg0, SCM arg1, SCM arg2,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_call_4 (SCM proc, SCM arg0, SCM arg1, SCM arg2,
|
||
SCM arg3,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_safe_apply_1 (SCM proc, SCM arg0, SCM args,
|
||
excp_matcher_func *ok_excps);
|
||
|
||
extern SCM gdbscm_unsafe_call_1 (SCM proc, SCM arg0);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_safe_eval_string
|
||
(const char *string, int display_result);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_safe_source_script
|
||
(const char *filename);
|
||
|
||
extern void gdbscm_enter_repl (void);
|
||
|
||
/* Interface to various GDB objects, in alphabetical order. */
|
||
|
||
/* scm-arch.c */
|
||
|
||
struct arch_smob;
|
||
|
||
extern struct gdbarch *arscm_get_gdbarch (arch_smob *a_smob);
|
||
|
||
extern arch_smob *arscm_get_arch_smob_arg_unsafe (SCM arch_scm, int arg_pos,
|
||
const char *func_name);
|
||
|
||
extern SCM arscm_scm_from_arch (struct gdbarch *gdbarch);
|
||
|
||
/* scm-block.c */
|
||
|
||
extern SCM bkscm_scm_from_block (const struct block *block,
|
||
struct objfile *objfile);
|
||
|
||
extern const struct block *bkscm_scm_to_block
|
||
(SCM block_scm, int arg_pos, const char *func_name, SCM *excp);
|
||
|
||
/* scm-cmd.c */
|
||
|
||
extern char *gdbscm_parse_command_name (const char *name,
|
||
const char *func_name, int arg_pos,
|
||
struct cmd_list_element ***base_list,
|
||
struct cmd_list_element **start_list);
|
||
|
||
extern int gdbscm_valid_command_class_p (int command_class);
|
||
|
||
extern char *gdbscm_canonicalize_command_name (const char *name,
|
||
int want_trailing_space);
|
||
|
||
/* scm-frame.c */
|
||
|
||
struct frame_smob;
|
||
|
||
extern int frscm_is_frame (SCM scm);
|
||
|
||
extern frame_smob *frscm_get_frame_smob_arg_unsafe (SCM frame_scm, int arg_pos,
|
||
const char *func_name);
|
||
|
||
extern struct frame_info_ptr frscm_frame_smob_to_frame (frame_smob *);
|
||
|
||
/* scm-iterator.c */
|
||
|
||
struct iterator_smob;
|
||
|
||
extern SCM itscm_iterator_smob_object (iterator_smob *i_smob);
|
||
|
||
extern SCM itscm_iterator_smob_progress (iterator_smob *i_smob);
|
||
|
||
extern void itscm_set_iterator_smob_progress_x (iterator_smob *i_smob,
|
||
SCM progress);
|
||
|
||
extern const char *itscm_iterator_smob_name (void);
|
||
|
||
extern SCM gdbscm_make_iterator (SCM object, SCM progress, SCM next);
|
||
|
||
extern int itscm_is_iterator (SCM scm);
|
||
|
||
extern SCM gdbscm_end_of_iteration (void);
|
||
|
||
extern int itscm_is_end_of_iteration (SCM obj);
|
||
|
||
extern SCM itscm_safe_call_next_x (SCM iter, excp_matcher_func *ok_excps);
|
||
|
||
extern SCM itscm_get_iterator_arg_unsafe (SCM self, int arg_pos,
|
||
const char *func_name);
|
||
|
||
/* scm-lazy-string.c */
|
||
|
||
extern int lsscm_is_lazy_string (SCM scm);
|
||
|
||
extern SCM lsscm_make_lazy_string (CORE_ADDR address, int length,
|
||
const char *encoding, struct type *type);
|
||
|
||
extern struct value *lsscm_safe_lazy_string_to_value (SCM string,
|
||
int arg_pos,
|
||
const char *func_name,
|
||
SCM *except_scmp);
|
||
|
||
extern void lsscm_val_print_lazy_string
|
||
(SCM string, struct ui_file *stream,
|
||
const struct value_print_options *options);
|
||
|
||
/* scm-objfile.c */
|
||
|
||
struct objfile_smob;
|
||
|
||
extern SCM ofscm_objfile_smob_pretty_printers (objfile_smob *o_smob);
|
||
|
||
extern objfile_smob *ofscm_objfile_smob_from_objfile (struct objfile *objfile);
|
||
|
||
extern SCM ofscm_scm_from_objfile (struct objfile *objfile);
|
||
|
||
/* scm-progspace.c */
|
||
|
||
struct pspace_smob;
|
||
|
||
extern SCM psscm_pspace_smob_pretty_printers (const pspace_smob *);
|
||
|
||
extern pspace_smob *psscm_pspace_smob_from_pspace (struct program_space *);
|
||
|
||
extern SCM psscm_scm_from_pspace (struct program_space *);
|
||
|
||
/* scm-string.c */
|
||
|
||
extern int gdbscm_scm_string_to_int (SCM string);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_scm_to_c_string (SCM string);
|
||
|
||
extern SCM gdbscm_scm_from_c_string (const char *string);
|
||
|
||
extern SCM gdbscm_scm_from_printf (const char *format, ...)
|
||
ATTRIBUTE_PRINTF (1, 2);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_scm_to_string
|
||
(SCM string, size_t *lenp, const char *charset, int strict, SCM *except_scmp);
|
||
|
||
extern SCM gdbscm_scm_from_string (const char *string, size_t len,
|
||
const char *charset, int strict);
|
||
|
||
extern gdb::unique_xmalloc_ptr<char> gdbscm_scm_to_host_string
|
||
(SCM string, size_t *lenp, SCM *except);
|
||
|
||
extern SCM gdbscm_scm_from_host_string (const char *string, size_t len);
|
||
|
||
/* scm-symbol.c */
|
||
|
||
extern int syscm_is_symbol (SCM scm);
|
||
|
||
extern SCM syscm_scm_from_symbol (struct symbol *symbol);
|
||
|
||
extern struct symbol *syscm_get_valid_symbol_arg_unsafe
|
||
(SCM self, int arg_pos, const char *func_name);
|
||
|
||
/* scm-symtab.c */
|
||
|
||
extern SCM stscm_scm_from_symtab (struct symtab *symtab);
|
||
|
||
extern SCM stscm_scm_from_sal (struct symtab_and_line sal);
|
||
|
||
/* scm-type.c */
|
||
|
||
struct type_smob;
|
||
|
||
extern int tyscm_is_type (SCM scm);
|
||
|
||
extern SCM tyscm_scm_from_type (struct type *type);
|
||
|
||
extern type_smob *tyscm_get_type_smob_arg_unsafe (SCM type_scm, int arg_pos,
|
||
const char *func_name);
|
||
|
||
extern struct type *tyscm_scm_to_type (SCM t_scm);
|
||
|
||
extern struct type *tyscm_type_smob_type (type_smob *t_smob);
|
||
|
||
extern SCM tyscm_scm_from_field (SCM type_scm, int field_num);
|
||
|
||
/* scm-value.c */
|
||
|
||
extern struct value *vlscm_scm_to_value (SCM scm);
|
||
|
||
extern int vlscm_is_value (SCM scm);
|
||
|
||
extern SCM vlscm_scm_from_value (struct value *value);
|
||
extern SCM vlscm_scm_from_value_no_release (struct value *value);
|
||
|
||
extern struct value *vlscm_convert_typed_value_from_scheme
|
||
(const char *func_name, int obj_arg_pos, SCM obj,
|
||
int type_arg_pos, SCM type_scm, struct type *type, SCM *except_scmp,
|
||
struct gdbarch *gdbarch, const struct language_defn *language);
|
||
|
||
extern struct value *vlscm_convert_value_from_scheme
|
||
(const char *func_name, int obj_arg_pos, SCM obj, SCM *except_scmp,
|
||
struct gdbarch *gdbarch, const struct language_defn *language);
|
||
|
||
/* stript_lang methods */
|
||
|
||
extern objfile_script_sourcer_func gdbscm_source_objfile_script;
|
||
extern objfile_script_executor_func gdbscm_execute_objfile_script;
|
||
|
||
/* Return true if auto-loading Guile scripts is enabled.
|
||
This is the extension_language_script_ops.auto_load_enabled "method". */
|
||
|
||
extern bool gdbscm_auto_load_enabled (const struct extension_language_defn *);
|
||
|
||
extern void gdbscm_preserve_values
|
||
(const struct extension_language_defn *,
|
||
struct objfile *, htab_t copied_types);
|
||
|
||
extern enum ext_lang_rc gdbscm_apply_val_pretty_printer
|
||
(const struct extension_language_defn *,
|
||
struct value *val,
|
||
struct ui_file *stream, int recurse,
|
||
const struct value_print_options *options,
|
||
const struct language_defn *language);
|
||
|
||
extern int gdbscm_breakpoint_has_cond (const struct extension_language_defn *,
|
||
struct breakpoint *b);
|
||
|
||
extern enum ext_lang_bp_stop gdbscm_breakpoint_cond_says_stop
|
||
(const struct extension_language_defn *, struct breakpoint *b);
|
||
|
||
/* Initializers for each piece of Scheme support, in alphabetical order. */
|
||
|
||
extern void gdbscm_initialize_arches (void);
|
||
extern void gdbscm_initialize_auto_load (void);
|
||
extern void gdbscm_initialize_blocks (void);
|
||
extern void gdbscm_initialize_breakpoints (void);
|
||
extern void gdbscm_initialize_commands (void);
|
||
extern void gdbscm_initialize_disasm (void);
|
||
extern void gdbscm_initialize_exceptions (void);
|
||
extern void gdbscm_initialize_frames (void);
|
||
extern void gdbscm_initialize_iterators (void);
|
||
extern void gdbscm_initialize_lazy_strings (void);
|
||
extern void gdbscm_initialize_math (void);
|
||
extern void gdbscm_initialize_objfiles (void);
|
||
extern void gdbscm_initialize_pretty_printers (void);
|
||
extern void gdbscm_initialize_parameters (void);
|
||
extern void gdbscm_initialize_ports (void);
|
||
extern void gdbscm_initialize_pspaces (void);
|
||
extern void gdbscm_initialize_smobs (void);
|
||
extern void gdbscm_initialize_strings (void);
|
||
extern void gdbscm_initialize_symbols (void);
|
||
extern void gdbscm_initialize_symtabs (void);
|
||
extern void gdbscm_initialize_types (void);
|
||
extern void gdbscm_initialize_values (void);
|
||
|
||
|
||
/* A complication with the Guile code is that we have two types of
|
||
exceptions to consider. GDB/C++ exceptions, and Guile/SJLJ
|
||
exceptions. Code that is facing the Guile interpreter must not
|
||
throw GDB exceptions, instead Scheme exceptions must be thrown.
|
||
Also, because Guile exceptions are SJLJ based, Guile-facing code
|
||
must not use local objects with dtors, unless wrapped in a scope
|
||
with a TRY/CATCH, because the dtors won't otherwise be run when a
|
||
Guile exceptions is thrown. */
|
||
|
||
/* This is a destructor-less clone of gdb_exception. */
|
||
|
||
struct gdbscm_gdb_exception
|
||
{
|
||
enum return_reason reason;
|
||
enum errors error;
|
||
/* The message is xmalloc'd. */
|
||
char *message;
|
||
};
|
||
|
||
/* Return a gdbscm_gdb_exception representing EXC. */
|
||
|
||
inline gdbscm_gdb_exception
|
||
unpack (const gdb_exception &exc)
|
||
{
|
||
gdbscm_gdb_exception result;
|
||
result.reason = exc.reason;
|
||
result.error = exc.error;
|
||
if (exc.message == nullptr)
|
||
result.message = nullptr;
|
||
else
|
||
result.message = xstrdup (exc.message->c_str ());
|
||
/* The message should be NULL iff the reason is zero. */
|
||
gdb_assert ((result.reason == 0) == (result.message == nullptr));
|
||
return result;
|
||
}
|
||
|
||
/* Use this after a TRY/CATCH to throw the appropriate Scheme
|
||
exception if a GDB error occurred. */
|
||
|
||
#define GDBSCM_HANDLE_GDB_EXCEPTION(exception) \
|
||
do { \
|
||
if (exception.reason < 0) \
|
||
{ \
|
||
gdbscm_throw_gdb_exception (exception); \
|
||
/*NOTREACHED */ \
|
||
} \
|
||
} while (0)
|
||
|
||
/* Use this to wrap a callable to throw the appropriate Scheme
|
||
exception if the callable throws a GDB error. ARGS are forwarded
|
||
to FUNC. Returns the result of FUNC, unless FUNC returns a Scheme
|
||
exception, in which case that exception is thrown. Note that while
|
||
the callable is free to use objects of types with destructors,
|
||
because GDB errors are C++ exceptions, the caller of gdbscm_wrap
|
||
must not use such objects, because their destructors would not be
|
||
called when a Scheme exception is thrown. */
|
||
|
||
template<typename Function, typename... Args>
|
||
SCM
|
||
gdbscm_wrap (Function &&func, Args &&... args)
|
||
{
|
||
SCM result = SCM_BOOL_F;
|
||
gdbscm_gdb_exception exc {};
|
||
|
||
try
|
||
{
|
||
result = func (std::forward<Args> (args)...);
|
||
}
|
||
catch (const gdb_exception_forced_quit &e)
|
||
{
|
||
quit_force (NULL, 0);
|
||
}
|
||
catch (const gdb_exception &except)
|
||
{
|
||
exc = unpack (except);
|
||
}
|
||
|
||
GDBSCM_HANDLE_GDB_EXCEPTION (exc);
|
||
|
||
if (gdbscm_is_exception (result))
|
||
gdbscm_throw (result);
|
||
|
||
return result;
|
||
}
|
||
|
||
#endif /* GUILE_GUILE_INTERNAL_H */
|