mirror of
https://github.com/darlinghq/darling-gdb.git
synced 2024-12-11 22:24:42 +00:00
197e01b6dc
* arm-tdep.c: * ia64-tdep.c: * i386-tdep.c: * hpread.c: * hppa-tdep.c: * hppa-hpux-tdep.c: * gnu-nat.c: * gdbtypes.c: * gdbarch.h: * gdbarch.c: * eval.c: * dwarf2read.c: * dbxread.c: * copying: * symfile.c: * stabsread.c: * sh64-tdep.c: * sh-tdep.c: * s390-tdep.c: * rs6000-tdep.c: * remote.c: * remote-mips.c: * mips-tdep.c: * mdebugread.c: * linux-nat.c: * infrun.c: * xcoffread.c: * win32-nat.c: * valops.c: * utils.c: * tracepoint.c: * target.c: * symtab.c: * c-exp.y: * ada-valprint.c: * ada-typeprint.c: * ada-lex.l: * ada-lang.h: * ada-lang.c: * ada-exp.y: * alphafbsd-tdep.c: * alphabsd-tdep.h: * alphabsd-tdep.c: * alphabsd-nat.c: * alpha-tdep.h: * alpha-tdep.c: * alpha-osf1-tdep.c: * alpha-nat.c: * alpha-mdebug-tdep.c: * alpha-linux-tdep.c: * alpha-linux-nat.c: * aix-thread.c: * abug-rom.c: * arch-utils.c: * annotate.h: * annotate.c: * amd64obsd-tdep.c: * amd64obsd-nat.c: * amd64nbsd-tdep.c: * amd64nbsd-nat.c: * amd64fbsd-tdep.c: * amd64fbsd-nat.c: * amd64bsd-nat.c: * amd64-tdep.h: * amd64-tdep.c: * amd64-sol2-tdep.c: * amd64-nat.h: * amd64-nat.c: * amd64-linux-tdep.c: * amd64-linux-nat.c: * alphanbsd-tdep.c: * block.h: * block.c: * bfd-target.h: * bfd-target.c: * bcache.h: * bcache.c: * ax.h: * ax-general.c: * ax-gdb.h: * ax-gdb.c: * avr-tdep.c: * auxv.h: * auxv.c: * armnbsd-tdep.c: * armnbsd-nat.c: * arm-tdep.h: * arm-linux-nat.c: * arch-utils.h: * charset.c: * call-cmds.h: * c-valprint.c: * c-typeprint.c: * c-lang.h: * c-lang.c: * buildsym.h: * buildsym.c: * bsd-uthread.h: * bsd-uthread.c: * bsd-kvm.h: * bsd-kvm.c: * breakpoint.h: * core-regset.c: * core-aout.c: * completer.h: * completer.c: * complaints.h: * complaints.c: * command.h: * coffread.c: * coff-solib.h: * coff-solib.c: * coff-pe-read.h: * coff-pe-read.c: * cli-out.h: * cli-out.c: * charset.h: * dink32-rom.c: * dictionary.h: * dictionary.c: * demangle.c: * defs.h: * dcache.h: * dcache.c: * d10v-tdep.c: * cpu32bug-rom.c: * cp-valprint.c: * cp-support.h: * cp-support.c: * cp-namespace.c: * cp-abi.h: * cp-abi.c: * corelow.c: * corefile.c: * environ.c: * elfread.c: * dwarfread.c: * dwarf2loc.c: * dwarf2expr.h: * dwarf2expr.c: * dwarf2-frame.h: * dwarf2-frame.c: * dve3900-rom.c: * dummy-frame.h: * dummy-frame.c: * dsrec.c: * doublest.h: * doublest.c: * disasm.h: * disasm.c: * fork-child.c: * findvar.c: * fbsd-nat.h: * fbsd-nat.c: * f-valprint.c: * f-typeprint.c: * f-lang.h: * f-lang.c: * expression.h: * expprint.c: * exec.h: * exec.c: * exceptions.h: * exceptions.c: * event-top.h: * event-top.c: * event-loop.h: * event-loop.c: * gdb.c: * gdb-stabs.h: * gdb-events.h: * gdb-events.c: * gcore.c: * frv-tdep.h: * frv-tdep.c: * frv-linux-tdep.c: * frame.h: * frame.c: * frame-unwind.h: * frame-unwind.c: * frame-base.h: * frame-base.c: * gdb_vfork.h: * gdb_thread_db.h: * gdb_string.h: * gdb_stat.h: * gdb_regex.h: * gdb_ptrace.h: * gdb_proc_service.h: * gdb_obstack.h: * gdb_locale.h: * gdb_dirent.h: * gdb_curses.h: * gdb_assert.h: * gdbarch.sh: * gdb.h: * hpux-thread.c: * hppabsd-nat.c: * hppa-tdep.h: * hpacc-abi.c: * h8300-tdep.c: * gregset.h: * go32-nat.c: * gnu-v3-abi.c: * gnu-v2-abi.h: * gnu-v2-abi.c: * gnu-nat.h: * glibc-tdep.c: * gdbtypes.h: * gdbcore.h: * gdbcmd.h: * i386nbsd-tdep.c: * i386nbsd-nat.c: * i386gnu-tdep.c: * i386gnu-nat.c: * i386fbsd-tdep.c: * i386fbsd-nat.c: * i386bsd-tdep.c: * i386bsd-nat.h: * i386bsd-nat.c: * i386-tdep.h: * i386-sol2-nat.c: * i386-nto-tdep.c: * i386-nat.c: * i386-linux-tdep.h: * i386-linux-tdep.c: * i386-linux-nat.c: * i386-cygwin-tdep.c: * inf-ttrace.c: * inf-ptrace.h: * inf-ptrace.c: * inf-loop.h: * inf-loop.c: * inf-child.h: * inf-child.c: * ia64-tdep.h: * ia64-linux-nat.c: * i387-tdep.h: * i387-tdep.c: * i386v4-nat.c: * i386v-nat.c: * i386obsd-tdep.c: * i386obsd-nat.c: * kod.c: * jv-valprint.c: * jv-typeprint.c: * jv-lang.h: * jv-lang.c: * irix5-nat.c: * iq2000-tdep.c: * interps.h: * interps.c: * inftarg.c: * inflow.h: * inflow.c: * inferior.h: * infcmd.c: * infcall.h: * infcall.c: * inf-ttrace.h: * m32r-tdep.h: * m32r-tdep.c: * m32r-rom.c: * m32r-linux-tdep.c: * m32r-linux-nat.c: * m2-valprint.c: * m2-typeprint.c: * m2-lang.h: * m2-lang.c: * lynx-nat.c: * linux-thread-db.c: * linux-nat.h: * linespec.c: * libunwind-frame.h: * libunwind-frame.c: * language.h: * language.c: * macroexp.c: * macrocmd.c: * m88kbsd-nat.c: * m88k-tdep.h: * m88k-tdep.c: * m68klinux-tdep.c: * m68klinux-nat.c: * m68kbsd-tdep.c: * m68kbsd-nat.c: * m68k-tdep.h: * m68k-tdep.c: * mips-linux-nat.c: * mips-irix-tdep.c: * minsyms.c: * memattr.h: * memattr.c: * mem-break.c: * mdebugread.h: * main.h: * main.c: * macrotab.h: * macrotab.c: * macroscope.h: * macroscope.c: * macroexp.h: * nbsd-tdep.c: * mt-tdep.c: * monitor.h: * monitor.c: * mn10300-tdep.h: * mn10300-tdep.c: * mn10300-linux-tdep.c: * mipsv4-nat.c: * mipsread.c: * mipsnbsd-tdep.h: * mipsnbsd-tdep.c: * mipsnbsd-nat.c: * mips64obsd-tdep.c: * mips64obsd-nat.c: * mips-tdep.h: * mips-mdebug-tdep.c: * mips-linux-tdep.c: * osabi.h: * osabi.c: * ocd.h: * ocd.c: * observer.c: * objfiles.h: * objfiles.c: * objc-lang.h: * objc-lang.c: * objc-exp.y: * nto-tdep.h: * nto-tdep.c: * nto-procfs.c: * nlmread.c: * nbsd-tdep.h: * ppcobsd-tdep.c: * ppcobsd-nat.c: * ppcnbsd-tdep.h: * ppcnbsd-tdep.c: * ppcnbsd-nat.c: * ppcbug-rom.c: * ppc-tdep.h: * ppc-sysv-tdep.c: * ppc-linux-tdep.c: * ppc-linux-nat.c: * ppc-bdm.c: * parser-defs.h: * parse.c: * p-valprint.c: * p-typeprint.c: * p-lang.h: * p-lang.c: * remote-fileio.h: * remote-fileio.c: * remote-est.c: * remote-e7000.c: * regset.h: * regset.c: * reggroups.h: * reggroups.c: * regcache.h: * regcache.c: * proc-why.c: * proc-service.c: * proc-events.c: * printcmd.c: * ppcobsd-tdep.h: * sentinel-frame.h: * sentinel-frame.c: * scm-valprint.c: * scm-tags.h: * scm-lang.h: * scm-lang.c: * scm-exp.c: * s390-tdep.h: * rom68k-rom.c: * remote.h: * remote-utils.c: * remote-st.c: * remote-sim.c: * remote-sds.c: * remote-rdp.c: * remote-rdi.c: * remote-hms.c: * sim-regno.h: * shnbsd-tdep.h: * shnbsd-tdep.c: * shnbsd-nat.c: * sh-tdep.h: * serial.h: * serial.c: * ser-unix.h: * ser-unix.c: * ser-tcp.c: * ser-pipe.c: * ser-go32.c: * ser-e7kpc.c: * ser-base.h: * ser-base.c: * solib.c: * solib-svr4.h: * solib-svr4.c: * solib-sunos.c: * solib-som.h: * solib-som.c: * solib-pa64.h: * solib-pa64.c: * solib-osf.c: * solib-null.c: * solib-legacy.c: * solib-irix.c: * solib-frv.c: * solib-aix5.c: * sol-thread.c: * sparc64-linux-tdep.c: * sparc64-linux-nat.c: * sparc-tdep.h: * sparc-tdep.c: * sparc-sol2-tdep.c: * sparc-sol2-nat.c: * sparc-nat.h: * sparc-nat.c: * sparc-linux-tdep.c: * sparc-linux-nat.c: * source.h: * source.c: * somread.c: * solist.h: * solib.h: * std-regs.c: * stack.h: * stack.c: * stabsread.h: * sparcobsd-tdep.c: * sparcnbsd-tdep.c: * sparcnbsd-nat.c: * sparc64obsd-tdep.c: * sparc64nbsd-tdep.c: * sparc64nbsd-nat.c: * sparc64fbsd-tdep.c: * sparc64fbsd-nat.c: * sparc64-tdep.h: * sparc64-tdep.c: * sparc64-sol2-tdep.c: * sparc64-nat.c: * ui-file.c: * typeprint.h: * typeprint.c: * tramp-frame.h: * tramp-frame.c: * trad-frame.h: * trad-frame.c: * tracepoint.h: * top.c: * tobs.inc: * thread.c: * terminal.h: * target.h: * symfile.h: * stop-gdb.c: * vaxbsd-nat.c: * vax-tdep.h: * vax-tdep.c: * vax-nat.c: * varobj.h: * varobj.c: * value.h: * value.c: * valprint.h: * valprint.c: * v850-tdep.c: * uw-thread.c: * user-regs.c: * ui-out.h: * ui-out.c: * ui-file.h: * xcoffsolib.h: * xcoffsolib.c: * wrapper.c: * wince.c: * wince-stub.h: * wince-stub.c: * vaxobsd-tdep.c: * vaxnbsd-tdep.c: * gdb_gcore.sh: * copying.c: * configure.ac: * aclocal.m4: * acinclude.m4: * reply_mig_hack.awk: * observer.sh: * gdb_mbuild.sh: * arm-linux-tdep.c: * blockframe.c: * dbug-rom.c: * environ.h: * dwarf2loc.h: * gdb-events.sh: * glibc-tdep.h: * gdb_wait.h: * gdbthread.h: * i386-sol2-tdep.c: * hppabsd-tdep.c: * hppa-linux-nat.c: * hppa-hpux-nat.c: * ia64-linux-tdep.c: * infptrace.c: * linespec.h: * maint.c: * mips-mdebug-tdep.h: * remote-m32r-sdi.c: * s390-nat.c: * rs6000-nat.c: * remote-utils.h: * sh3-rom.c: * sh-linux-tdep.c: * top.h: * symtab.h: * symmisc.c: * symfile-mem.c: * srec.h: * user-regs.h: * version.h: * valarith.c: * xstormy16-tdep.c: * wrapper.h: * Makefile.in: * f-exp.y: * cris-tdep.c: * cp-name-parser.y: * procfs.c: * proc-utils.h: * proc-flags.c: * proc-api.c: * p-exp.y: * m68hc11-tdep.c: * m2-exp.y: * kod.h: * kod-cisco.c: * jv-exp.y: * hppa-linux-tdep.c: Add (c) after Copyright. Update the FSF address.
920 lines
32 KiB
C
920 lines
32 KiB
C
/* Perform an inferior function call, for GDB, the GNU debugger.
|
||
|
||
Copyright (C) 1986, 1987, 1988, 1989, 1990, 1991, 1992, 1993, 1994,
|
||
1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005
|
||
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., 51 Franklin Street, Fifth Floor,
|
||
Boston, MA 02110-1301, USA. */
|
||
|
||
#include "defs.h"
|
||
#include "breakpoint.h"
|
||
#include "target.h"
|
||
#include "regcache.h"
|
||
#include "inferior.h"
|
||
#include "gdb_assert.h"
|
||
#include "block.h"
|
||
#include "gdbcore.h"
|
||
#include "language.h"
|
||
#include "objfiles.h"
|
||
#include "gdbcmd.h"
|
||
#include "command.h"
|
||
#include "gdb_string.h"
|
||
#include "infcall.h"
|
||
#include "dummy-frame.h"
|
||
|
||
/* NOTE: cagney/2003-04-16: What's the future of this code?
|
||
|
||
GDB needs an asynchronous expression evaluator, that means an
|
||
asynchronous inferior function call implementation, and that in
|
||
turn means restructuring the code so that it is event driven. */
|
||
|
||
/* How you should pass arguments to a function depends on whether it
|
||
was defined in K&R style or prototype style. If you define a
|
||
function using the K&R syntax that takes a `float' argument, then
|
||
callers must pass that argument as a `double'. If you define the
|
||
function using the prototype syntax, then you must pass the
|
||
argument as a `float', with no promotion.
|
||
|
||
Unfortunately, on certain older platforms, the debug info doesn't
|
||
indicate reliably how each function was defined. A function type's
|
||
TYPE_FLAG_PROTOTYPED flag may be clear, even if the function was
|
||
defined in prototype style. When calling a function whose
|
||
TYPE_FLAG_PROTOTYPED flag is clear, GDB consults this flag to
|
||
decide what to do.
|
||
|
||
For modern targets, it is proper to assume that, if the prototype
|
||
flag is clear, that can be trusted: `float' arguments should be
|
||
promoted to `double'. For some older targets, if the prototype
|
||
flag is clear, that doesn't tell us anything. The default is to
|
||
trust the debug information; the user can override this behavior
|
||
with "set coerce-float-to-double 0". */
|
||
|
||
static int coerce_float_to_double_p = 1;
|
||
static void
|
||
show_coerce_float_to_double_p (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("\
|
||
Coercion of floats to doubles when calling functions is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
/* This boolean tells what gdb should do if a signal is received while
|
||
in a function called from gdb (call dummy). If set, gdb unwinds
|
||
the stack and restore the context to what as it was before the
|
||
call.
|
||
|
||
The default is to stop in the frame where the signal was received. */
|
||
|
||
int unwind_on_signal_p = 0;
|
||
static void
|
||
show_unwind_on_signal_p (struct ui_file *file, int from_tty,
|
||
struct cmd_list_element *c, const char *value)
|
||
{
|
||
fprintf_filtered (file, _("\
|
||
Unwinding of stack if a signal is received while in a call dummy is %s.\n"),
|
||
value);
|
||
}
|
||
|
||
|
||
/* Perform the standard coercions that are specified
|
||
for arguments to be passed to C functions.
|
||
|
||
If PARAM_TYPE is non-NULL, it is the expected parameter type.
|
||
IS_PROTOTYPED is non-zero if the function declaration is prototyped. */
|
||
|
||
static struct value *
|
||
value_arg_coerce (struct value *arg, struct type *param_type,
|
||
int is_prototyped)
|
||
{
|
||
struct type *arg_type = check_typedef (value_type (arg));
|
||
struct type *type
|
||
= param_type ? check_typedef (param_type) : arg_type;
|
||
|
||
switch (TYPE_CODE (type))
|
||
{
|
||
case TYPE_CODE_REF:
|
||
if (TYPE_CODE (arg_type) != TYPE_CODE_REF
|
||
&& TYPE_CODE (arg_type) != TYPE_CODE_PTR)
|
||
{
|
||
arg = value_addr (arg);
|
||
deprecated_set_value_type (arg, param_type);
|
||
return arg;
|
||
}
|
||
break;
|
||
case TYPE_CODE_INT:
|
||
case TYPE_CODE_CHAR:
|
||
case TYPE_CODE_BOOL:
|
||
case TYPE_CODE_ENUM:
|
||
/* If we don't have a prototype, coerce to integer type if necessary. */
|
||
if (!is_prototyped)
|
||
{
|
||
if (TYPE_LENGTH (type) < TYPE_LENGTH (builtin_type_int))
|
||
type = builtin_type_int;
|
||
}
|
||
/* Currently all target ABIs require at least the width of an integer
|
||
type for an argument. We may have to conditionalize the following
|
||
type coercion for future targets. */
|
||
if (TYPE_LENGTH (type) < TYPE_LENGTH (builtin_type_int))
|
||
type = builtin_type_int;
|
||
break;
|
||
case TYPE_CODE_FLT:
|
||
if (!is_prototyped && coerce_float_to_double_p)
|
||
{
|
||
if (TYPE_LENGTH (type) < TYPE_LENGTH (builtin_type_double))
|
||
type = builtin_type_double;
|
||
else if (TYPE_LENGTH (type) > TYPE_LENGTH (builtin_type_double))
|
||
type = builtin_type_long_double;
|
||
}
|
||
break;
|
||
case TYPE_CODE_FUNC:
|
||
type = lookup_pointer_type (type);
|
||
break;
|
||
case TYPE_CODE_ARRAY:
|
||
/* Arrays are coerced to pointers to their first element, unless
|
||
they are vectors, in which case we want to leave them alone,
|
||
because they are passed by value. */
|
||
if (current_language->c_style_arrays)
|
||
if (!TYPE_VECTOR (type))
|
||
type = lookup_pointer_type (TYPE_TARGET_TYPE (type));
|
||
break;
|
||
case TYPE_CODE_UNDEF:
|
||
case TYPE_CODE_PTR:
|
||
case TYPE_CODE_STRUCT:
|
||
case TYPE_CODE_UNION:
|
||
case TYPE_CODE_VOID:
|
||
case TYPE_CODE_SET:
|
||
case TYPE_CODE_RANGE:
|
||
case TYPE_CODE_STRING:
|
||
case TYPE_CODE_BITSTRING:
|
||
case TYPE_CODE_ERROR:
|
||
case TYPE_CODE_MEMBER:
|
||
case TYPE_CODE_METHOD:
|
||
case TYPE_CODE_COMPLEX:
|
||
default:
|
||
break;
|
||
}
|
||
|
||
return value_cast (type, arg);
|
||
}
|
||
|
||
/* Determine a function's address and its return type from its value.
|
||
Calls error() if the function is not valid for calling. */
|
||
|
||
CORE_ADDR
|
||
find_function_addr (struct value *function, struct type **retval_type)
|
||
{
|
||
struct type *ftype = check_typedef (value_type (function));
|
||
enum type_code code = TYPE_CODE (ftype);
|
||
struct type *value_type;
|
||
CORE_ADDR funaddr;
|
||
|
||
/* If it's a member function, just look at the function
|
||
part of it. */
|
||
|
||
/* Determine address to call. */
|
||
if (code == TYPE_CODE_FUNC || code == TYPE_CODE_METHOD)
|
||
{
|
||
funaddr = VALUE_ADDRESS (function);
|
||
value_type = TYPE_TARGET_TYPE (ftype);
|
||
}
|
||
else if (code == TYPE_CODE_PTR)
|
||
{
|
||
funaddr = value_as_address (function);
|
||
ftype = check_typedef (TYPE_TARGET_TYPE (ftype));
|
||
if (TYPE_CODE (ftype) == TYPE_CODE_FUNC
|
||
|| TYPE_CODE (ftype) == TYPE_CODE_METHOD)
|
||
{
|
||
funaddr = gdbarch_convert_from_func_ptr_addr (current_gdbarch,
|
||
funaddr,
|
||
¤t_target);
|
||
value_type = TYPE_TARGET_TYPE (ftype);
|
||
}
|
||
else
|
||
value_type = builtin_type_int;
|
||
}
|
||
else if (code == TYPE_CODE_INT)
|
||
{
|
||
/* Handle the case of functions lacking debugging info.
|
||
Their values are characters since their addresses are char */
|
||
if (TYPE_LENGTH (ftype) == 1)
|
||
funaddr = value_as_address (value_addr (function));
|
||
else
|
||
/* Handle integer used as address of a function. */
|
||
funaddr = (CORE_ADDR) value_as_long (function);
|
||
|
||
value_type = builtin_type_int;
|
||
}
|
||
else
|
||
error (_("Invalid data type for function to be called."));
|
||
|
||
if (retval_type != NULL)
|
||
*retval_type = value_type;
|
||
return funaddr + DEPRECATED_FUNCTION_START_OFFSET;
|
||
}
|
||
|
||
/* Call breakpoint_auto_delete on the current contents of the bpstat
|
||
pointed to by arg (which is really a bpstat *). */
|
||
|
||
static void
|
||
breakpoint_auto_delete_contents (void *arg)
|
||
{
|
||
breakpoint_auto_delete (*(bpstat *) arg);
|
||
}
|
||
|
||
static CORE_ADDR
|
||
generic_push_dummy_code (struct gdbarch *gdbarch,
|
||
CORE_ADDR sp, CORE_ADDR funaddr, int using_gcc,
|
||
struct value **args, int nargs,
|
||
struct type *value_type,
|
||
CORE_ADDR *real_pc, CORE_ADDR *bp_addr)
|
||
{
|
||
/* Something here to findout the size of a breakpoint and then
|
||
allocate space for it on the stack. */
|
||
int bplen;
|
||
/* This code assumes frame align. */
|
||
gdb_assert (gdbarch_frame_align_p (gdbarch));
|
||
/* Force the stack's alignment. The intent is to ensure that the SP
|
||
is aligned to at least a breakpoint instruction's boundary. */
|
||
sp = gdbarch_frame_align (gdbarch, sp);
|
||
/* Allocate space for, and then position the breakpoint on the
|
||
stack. */
|
||
if (gdbarch_inner_than (gdbarch, 1, 2))
|
||
{
|
||
CORE_ADDR bppc = sp;
|
||
gdbarch_breakpoint_from_pc (gdbarch, &bppc, &bplen);
|
||
sp = gdbarch_frame_align (gdbarch, sp - bplen);
|
||
(*bp_addr) = sp;
|
||
/* Should the breakpoint size/location be re-computed here? */
|
||
}
|
||
else
|
||
{
|
||
(*bp_addr) = sp;
|
||
gdbarch_breakpoint_from_pc (gdbarch, bp_addr, &bplen);
|
||
sp = gdbarch_frame_align (gdbarch, sp + bplen);
|
||
}
|
||
/* Inferior resumes at the function entry point. */
|
||
(*real_pc) = funaddr;
|
||
return sp;
|
||
}
|
||
|
||
/* For CALL_DUMMY_ON_STACK, push a breakpoint sequence that the called
|
||
function returns to. */
|
||
|
||
static CORE_ADDR
|
||
push_dummy_code (struct gdbarch *gdbarch,
|
||
CORE_ADDR sp, CORE_ADDR funaddr, int using_gcc,
|
||
struct value **args, int nargs,
|
||
struct type *value_type,
|
||
CORE_ADDR *real_pc, CORE_ADDR *bp_addr)
|
||
{
|
||
if (gdbarch_push_dummy_code_p (gdbarch))
|
||
return gdbarch_push_dummy_code (gdbarch, sp, funaddr, using_gcc,
|
||
args, nargs, value_type, real_pc, bp_addr);
|
||
else
|
||
return generic_push_dummy_code (gdbarch, sp, funaddr, using_gcc,
|
||
args, nargs, value_type, real_pc, bp_addr);
|
||
}
|
||
|
||
/* All this stuff with a dummy frame may seem unnecessarily complicated
|
||
(why not just save registers in GDB?). The purpose of pushing a dummy
|
||
frame which looks just like a real frame is so that if you call a
|
||
function and then hit a breakpoint (get a signal, etc), "backtrace"
|
||
will look right. Whether the backtrace needs to actually show the
|
||
stack at the time the inferior function was called is debatable, but
|
||
it certainly needs to not display garbage. So if you are contemplating
|
||
making dummy frames be different from normal frames, consider that. */
|
||
|
||
/* Perform a function call in the inferior.
|
||
ARGS is a vector of values of arguments (NARGS of them).
|
||
FUNCTION is a value, the function to be called.
|
||
Returns a value representing what the function returned.
|
||
May fail to return, if a breakpoint or signal is hit
|
||
during the execution of the function.
|
||
|
||
ARGS is modified to contain coerced values. */
|
||
|
||
struct value *
|
||
call_function_by_hand (struct value *function, int nargs, struct value **args)
|
||
{
|
||
CORE_ADDR sp;
|
||
CORE_ADDR dummy_addr;
|
||
struct type *values_type;
|
||
unsigned char struct_return;
|
||
CORE_ADDR struct_addr = 0;
|
||
struct regcache *retbuf;
|
||
struct cleanup *retbuf_cleanup;
|
||
struct inferior_status *inf_status;
|
||
struct cleanup *inf_status_cleanup;
|
||
CORE_ADDR funaddr;
|
||
int using_gcc; /* Set to version of gcc in use, or zero if not gcc */
|
||
CORE_ADDR real_pc;
|
||
struct type *ftype = check_typedef (value_type (function));
|
||
CORE_ADDR bp_addr;
|
||
struct regcache *caller_regcache;
|
||
struct cleanup *caller_regcache_cleanup;
|
||
struct frame_id dummy_id;
|
||
|
||
if (!target_has_execution)
|
||
noprocess ();
|
||
|
||
if (!gdbarch_push_dummy_call_p (current_gdbarch))
|
||
error (_("This target does not support function calls"));
|
||
|
||
/* Create a cleanup chain that contains the retbuf (buffer
|
||
containing the register values). This chain is create BEFORE the
|
||
inf_status chain so that the inferior status can cleaned up
|
||
(restored or discarded) without having the retbuf freed. */
|
||
retbuf = regcache_xmalloc (current_gdbarch);
|
||
retbuf_cleanup = make_cleanup_regcache_xfree (retbuf);
|
||
|
||
/* A cleanup for the inferior status. Create this AFTER the retbuf
|
||
so that this can be discarded or applied without interfering with
|
||
the regbuf. */
|
||
inf_status = save_inferior_status (1);
|
||
inf_status_cleanup = make_cleanup_restore_inferior_status (inf_status);
|
||
|
||
/* Save the caller's registers so that they can be restored once the
|
||
callee returns. To allow nested calls the registers are (further
|
||
down) pushed onto a dummy frame stack. Include a cleanup (which
|
||
is tossed once the regcache has been pushed). */
|
||
caller_regcache = frame_save_as_regcache (get_current_frame ());
|
||
caller_regcache_cleanup = make_cleanup_regcache_xfree (caller_regcache);
|
||
|
||
/* Ensure that the initial SP is correctly aligned. */
|
||
{
|
||
CORE_ADDR old_sp = read_sp ();
|
||
if (gdbarch_frame_align_p (current_gdbarch))
|
||
{
|
||
sp = gdbarch_frame_align (current_gdbarch, old_sp);
|
||
/* NOTE: cagney/2003-08-13: Skip the "red zone". For some
|
||
ABIs, a function can use memory beyond the inner most stack
|
||
address. AMD64 called that region the "red zone". Skip at
|
||
least the "red zone" size before allocating any space on
|
||
the stack. */
|
||
if (INNER_THAN (1, 2))
|
||
sp -= gdbarch_frame_red_zone_size (current_gdbarch);
|
||
else
|
||
sp += gdbarch_frame_red_zone_size (current_gdbarch);
|
||
/* Still aligned? */
|
||
gdb_assert (sp == gdbarch_frame_align (current_gdbarch, sp));
|
||
/* NOTE: cagney/2002-09-18:
|
||
|
||
On a RISC architecture, a void parameterless generic dummy
|
||
frame (i.e., no parameters, no result) typically does not
|
||
need to push anything the stack and hence can leave SP and
|
||
FP. Similarly, a frameless (possibly leaf) function does
|
||
not push anything on the stack and, hence, that too can
|
||
leave FP and SP unchanged. As a consequence, a sequence of
|
||
void parameterless generic dummy frame calls to frameless
|
||
functions will create a sequence of effectively identical
|
||
frames (SP, FP and TOS and PC the same). This, not
|
||
suprisingly, results in what appears to be a stack in an
|
||
infinite loop --- when GDB tries to find a generic dummy
|
||
frame on the internal dummy frame stack, it will always
|
||
find the first one.
|
||
|
||
To avoid this problem, the code below always grows the
|
||
stack. That way, two dummy frames can never be identical.
|
||
It does burn a few bytes of stack but that is a small price
|
||
to pay :-). */
|
||
if (sp == old_sp)
|
||
{
|
||
if (INNER_THAN (1, 2))
|
||
/* Stack grows down. */
|
||
sp = gdbarch_frame_align (current_gdbarch, old_sp - 1);
|
||
else
|
||
/* Stack grows up. */
|
||
sp = gdbarch_frame_align (current_gdbarch, old_sp + 1);
|
||
}
|
||
gdb_assert ((INNER_THAN (1, 2) && sp <= old_sp)
|
||
|| (INNER_THAN (2, 1) && sp >= old_sp));
|
||
}
|
||
else
|
||
/* FIXME: cagney/2002-09-18: Hey, you loose!
|
||
|
||
Who knows how badly aligned the SP is!
|
||
|
||
If the generic dummy frame ends up empty (because nothing is
|
||
pushed) GDB won't be able to correctly perform back traces.
|
||
If a target is having trouble with backtraces, first thing to
|
||
do is add FRAME_ALIGN() to the architecture vector. If that
|
||
fails, try unwind_dummy_id().
|
||
|
||
If the ABI specifies a "Red Zone" (see the doco) the code
|
||
below will quietly trash it. */
|
||
sp = old_sp;
|
||
}
|
||
|
||
funaddr = find_function_addr (function, &values_type);
|
||
CHECK_TYPEDEF (values_type);
|
||
|
||
{
|
||
struct block *b = block_for_pc (funaddr);
|
||
/* If compiled without -g, assume GCC 2. */
|
||
using_gcc = (b == NULL ? 2 : BLOCK_GCC_COMPILED (b));
|
||
}
|
||
|
||
/* Are we returning a value using a structure return or a normal
|
||
value return? */
|
||
|
||
struct_return = using_struct_return (values_type, using_gcc);
|
||
|
||
/* Determine the location of the breakpoint (and possibly other
|
||
stuff) that the called function will return to. The SPARC, for a
|
||
function returning a structure or union, needs to make space for
|
||
not just the breakpoint but also an extra word containing the
|
||
size (?) of the structure being passed. */
|
||
|
||
/* The actual breakpoint (at BP_ADDR) is inserted separatly so there
|
||
is no need to write that out. */
|
||
|
||
switch (CALL_DUMMY_LOCATION)
|
||
{
|
||
case ON_STACK:
|
||
/* "dummy_addr" is here just to keep old targets happy. New
|
||
targets return that same information via "sp" and "bp_addr". */
|
||
if (INNER_THAN (1, 2))
|
||
{
|
||
sp = push_dummy_code (current_gdbarch, sp, funaddr,
|
||
using_gcc, args, nargs, values_type,
|
||
&real_pc, &bp_addr);
|
||
dummy_addr = sp;
|
||
}
|
||
else
|
||
{
|
||
dummy_addr = sp;
|
||
sp = push_dummy_code (current_gdbarch, sp, funaddr,
|
||
using_gcc, args, nargs, values_type,
|
||
&real_pc, &bp_addr);
|
||
}
|
||
break;
|
||
case AT_ENTRY_POINT:
|
||
real_pc = funaddr;
|
||
dummy_addr = entry_point_address ();
|
||
/* Make certain that the address points at real code, and not a
|
||
function descriptor. */
|
||
dummy_addr = gdbarch_convert_from_func_ptr_addr (current_gdbarch,
|
||
dummy_addr,
|
||
¤t_target);
|
||
/* A call dummy always consists of just a single breakpoint, so
|
||
it's address is the same as the address of the dummy. */
|
||
bp_addr = dummy_addr;
|
||
break;
|
||
case AT_SYMBOL:
|
||
/* Some executables define a symbol __CALL_DUMMY_ADDRESS whose
|
||
address is the location where the breakpoint should be
|
||
placed. Once all targets are using the overhauled frame code
|
||
this can be deleted - ON_STACK is a better option. */
|
||
{
|
||
struct minimal_symbol *sym;
|
||
|
||
sym = lookup_minimal_symbol ("__CALL_DUMMY_ADDRESS", NULL, NULL);
|
||
real_pc = funaddr;
|
||
if (sym)
|
||
dummy_addr = SYMBOL_VALUE_ADDRESS (sym);
|
||
else
|
||
dummy_addr = entry_point_address ();
|
||
/* Make certain that the address points at real code, and not
|
||
a function descriptor. */
|
||
dummy_addr = gdbarch_convert_from_func_ptr_addr (current_gdbarch,
|
||
dummy_addr,
|
||
¤t_target);
|
||
/* A call dummy always consists of just a single breakpoint,
|
||
so it's address is the same as the address of the dummy. */
|
||
bp_addr = dummy_addr;
|
||
break;
|
||
}
|
||
default:
|
||
internal_error (__FILE__, __LINE__, _("bad switch"));
|
||
}
|
||
|
||
if (nargs < TYPE_NFIELDS (ftype))
|
||
error (_("too few arguments in function call"));
|
||
|
||
{
|
||
int i;
|
||
for (i = nargs - 1; i >= 0; i--)
|
||
{
|
||
int prototyped;
|
||
struct type *param_type;
|
||
|
||
/* FIXME drow/2002-05-31: Should just always mark methods as
|
||
prototyped. Can we respect TYPE_VARARGS? Probably not. */
|
||
if (TYPE_CODE (ftype) == TYPE_CODE_METHOD)
|
||
prototyped = 1;
|
||
else if (i < TYPE_NFIELDS (ftype))
|
||
prototyped = TYPE_PROTOTYPED (ftype);
|
||
else
|
||
prototyped = 0;
|
||
|
||
if (i < TYPE_NFIELDS (ftype))
|
||
param_type = TYPE_FIELD_TYPE (ftype, i);
|
||
else
|
||
param_type = NULL;
|
||
|
||
args[i] = value_arg_coerce (args[i], param_type, prototyped);
|
||
|
||
/* elz: this code is to handle the case in which the function
|
||
to be called has a pointer to function as parameter and the
|
||
corresponding actual argument is the address of a function
|
||
and not a pointer to function variable. In aCC compiled
|
||
code, the calls through pointers to functions (in the body
|
||
of the function called by hand) are made via
|
||
$$dyncall_external which requires some registers setting,
|
||
this is taken care of if we call via a function pointer
|
||
variable, but not via a function address. In cc this is
|
||
not a problem. */
|
||
|
||
if (using_gcc == 0)
|
||
{
|
||
if (param_type != NULL && TYPE_CODE (ftype) != TYPE_CODE_METHOD)
|
||
{
|
||
/* if this parameter is a pointer to function. */
|
||
if (TYPE_CODE (param_type) == TYPE_CODE_PTR)
|
||
if (TYPE_CODE (TYPE_TARGET_TYPE (param_type)) == TYPE_CODE_FUNC)
|
||
/* elz: FIXME here should go the test about the
|
||
compiler used to compile the target. We want to
|
||
issue the error message only if the compiler
|
||
used was HP's aCC. If we used HP's cc, then
|
||
there is no problem and no need to return at
|
||
this point. */
|
||
/* Go see if the actual parameter is a variable of
|
||
type pointer to function or just a function. */
|
||
if (VALUE_LVAL (args[i]) == not_lval)
|
||
{
|
||
char *arg_name;
|
||
/* NOTE: cagney/2005-01-02: THIS IS BOGUS. */
|
||
if (find_pc_partial_function ((CORE_ADDR) value_contents (args[i])[0], &arg_name, NULL, NULL))
|
||
error (_("\
|
||
You cannot use function <%s> as argument. \n\
|
||
You must use a pointer to function type variable. Command ignored."), arg_name);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (DEPRECATED_REG_STRUCT_HAS_ADDR_P ())
|
||
{
|
||
int i;
|
||
/* This is a machine like the sparc, where we may need to pass a
|
||
pointer to the structure, not the structure itself. */
|
||
for (i = nargs - 1; i >= 0; i--)
|
||
{
|
||
struct type *arg_type = check_typedef (value_type (args[i]));
|
||
if ((TYPE_CODE (arg_type) == TYPE_CODE_STRUCT
|
||
|| TYPE_CODE (arg_type) == TYPE_CODE_UNION
|
||
|| TYPE_CODE (arg_type) == TYPE_CODE_ARRAY
|
||
|| TYPE_CODE (arg_type) == TYPE_CODE_STRING
|
||
|| TYPE_CODE (arg_type) == TYPE_CODE_BITSTRING
|
||
|| TYPE_CODE (arg_type) == TYPE_CODE_SET
|
||
|| (TYPE_CODE (arg_type) == TYPE_CODE_FLT
|
||
&& TYPE_LENGTH (arg_type) > 8)
|
||
)
|
||
&& DEPRECATED_REG_STRUCT_HAS_ADDR (using_gcc, arg_type))
|
||
{
|
||
CORE_ADDR addr;
|
||
int len; /* = TYPE_LENGTH (arg_type); */
|
||
int aligned_len;
|
||
arg_type = check_typedef (value_enclosing_type (args[i]));
|
||
len = TYPE_LENGTH (arg_type);
|
||
|
||
aligned_len = len;
|
||
if (INNER_THAN (1, 2))
|
||
{
|
||
/* stack grows downward */
|
||
sp -= aligned_len;
|
||
/* ... so the address of the thing we push is the
|
||
stack pointer after we push it. */
|
||
addr = sp;
|
||
}
|
||
else
|
||
{
|
||
/* The stack grows up, so the address of the thing
|
||
we push is the stack pointer before we push it. */
|
||
addr = sp;
|
||
sp += aligned_len;
|
||
}
|
||
/* Push the structure. */
|
||
write_memory (addr, value_contents_all (args[i]), len);
|
||
/* The value we're going to pass is the address of the
|
||
thing we just pushed. */
|
||
/*args[i] = value_from_longest (lookup_pointer_type (values_type),
|
||
(LONGEST) addr); */
|
||
args[i] = value_from_pointer (lookup_pointer_type (arg_type),
|
||
addr);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
/* Reserve space for the return structure to be written on the
|
||
stack, if necessary. Make certain that the value is correctly
|
||
aligned. */
|
||
|
||
if (struct_return)
|
||
{
|
||
int len = TYPE_LENGTH (values_type);
|
||
if (INNER_THAN (1, 2))
|
||
{
|
||
/* Stack grows downward. Align STRUCT_ADDR and SP after
|
||
making space for the return value. */
|
||
sp -= len;
|
||
if (gdbarch_frame_align_p (current_gdbarch))
|
||
sp = gdbarch_frame_align (current_gdbarch, sp);
|
||
struct_addr = sp;
|
||
}
|
||
else
|
||
{
|
||
/* Stack grows upward. Align the frame, allocate space, and
|
||
then again, re-align the frame??? */
|
||
if (gdbarch_frame_align_p (current_gdbarch))
|
||
sp = gdbarch_frame_align (current_gdbarch, sp);
|
||
struct_addr = sp;
|
||
sp += len;
|
||
if (gdbarch_frame_align_p (current_gdbarch))
|
||
sp = gdbarch_frame_align (current_gdbarch, sp);
|
||
}
|
||
}
|
||
|
||
/* Create the dummy stack frame. Pass in the call dummy address as,
|
||
presumably, the ABI code knows where, in the call dummy, the
|
||
return address should be pointed. */
|
||
sp = gdbarch_push_dummy_call (current_gdbarch, function, current_regcache,
|
||
bp_addr, nargs, args, sp, struct_return,
|
||
struct_addr);
|
||
|
||
/* Set up a frame ID for the dummy frame so we can pass it to
|
||
set_momentary_breakpoint. We need to give the breakpoint a frame
|
||
ID so that the breakpoint code can correctly re-identify the
|
||
dummy breakpoint. */
|
||
/* Sanity. The exact same SP value is returned by PUSH_DUMMY_CALL,
|
||
saved as the dummy-frame TOS, and used by unwind_dummy_id to form
|
||
the frame ID's stack address. */
|
||
dummy_id = frame_id_build (sp, bp_addr);
|
||
|
||
/* Create a momentary breakpoint at the return address of the
|
||
inferior. That way it breaks when it returns. */
|
||
|
||
{
|
||
struct breakpoint *bpt;
|
||
struct symtab_and_line sal;
|
||
init_sal (&sal); /* initialize to zeroes */
|
||
sal.pc = bp_addr;
|
||
sal.section = find_pc_overlay (sal.pc);
|
||
/* Sanity. The exact same SP value is returned by
|
||
PUSH_DUMMY_CALL, saved as the dummy-frame TOS, and used by
|
||
unwind_dummy_id to form the frame ID's stack address. */
|
||
bpt = set_momentary_breakpoint (sal, dummy_id, bp_call_dummy);
|
||
bpt->disposition = disp_del;
|
||
}
|
||
|
||
/* Everything's ready, push all the info needed to restore the
|
||
caller (and identify the dummy-frame) onto the dummy-frame
|
||
stack. */
|
||
dummy_frame_push (caller_regcache, &dummy_id);
|
||
discard_cleanups (caller_regcache_cleanup);
|
||
|
||
/* - SNIP - SNIP - SNIP - SNIP - SNIP - SNIP - SNIP - SNIP - SNIP -
|
||
If you're looking to implement asynchronous dummy-frames, then
|
||
just below is the place to chop this function in two.. */
|
||
|
||
/* Now proceed, having reached the desired place. */
|
||
clear_proceed_status ();
|
||
|
||
/* Execute a "stack dummy", a piece of code stored in the stack by
|
||
the debugger to be executed in the inferior.
|
||
|
||
The dummy's frame is automatically popped whenever that break is
|
||
hit. If that is the first time the program stops,
|
||
call_function_by_hand returns to its caller with that frame
|
||
already gone and sets RC to 0.
|
||
|
||
Otherwise, set RC to a non-zero value. If the called function
|
||
receives a random signal, we do not allow the user to continue
|
||
executing it as this may not work. The dummy frame is poped and
|
||
we return 1. If we hit a breakpoint, we leave the frame in place
|
||
and return 2 (the frame will eventually be popped when we do hit
|
||
the dummy end breakpoint). */
|
||
|
||
{
|
||
struct cleanup *old_cleanups = make_cleanup (null_cleanup, 0);
|
||
int saved_async = 0;
|
||
|
||
/* If all error()s out of proceed ended up calling normal_stop
|
||
(and perhaps they should; it already does in the special case
|
||
of error out of resume()), then we wouldn't need this. */
|
||
make_cleanup (breakpoint_auto_delete_contents, &stop_bpstat);
|
||
|
||
disable_watchpoints_before_interactive_call_start ();
|
||
proceed_to_finish = 1; /* We want stop_registers, please... */
|
||
|
||
if (target_can_async_p ())
|
||
saved_async = target_async_mask (0);
|
||
|
||
proceed (real_pc, TARGET_SIGNAL_0, 0);
|
||
|
||
if (saved_async)
|
||
target_async_mask (saved_async);
|
||
|
||
enable_watchpoints_after_interactive_call_stop ();
|
||
|
||
discard_cleanups (old_cleanups);
|
||
}
|
||
|
||
if (stopped_by_random_signal || !stop_stack_dummy)
|
||
{
|
||
/* Find the name of the function we're about to complain about. */
|
||
const char *name = NULL;
|
||
{
|
||
struct symbol *symbol = find_pc_function (funaddr);
|
||
if (symbol)
|
||
name = SYMBOL_PRINT_NAME (symbol);
|
||
else
|
||
{
|
||
/* Try the minimal symbols. */
|
||
struct minimal_symbol *msymbol = lookup_minimal_symbol_by_pc (funaddr);
|
||
if (msymbol)
|
||
name = SYMBOL_PRINT_NAME (msymbol);
|
||
}
|
||
if (name == NULL)
|
||
{
|
||
/* Can't use a cleanup here. It is discarded, instead use
|
||
an alloca. */
|
||
char *tmp = xstrprintf ("at %s", hex_string (funaddr));
|
||
char *a = alloca (strlen (tmp) + 1);
|
||
strcpy (a, tmp);
|
||
xfree (tmp);
|
||
name = a;
|
||
}
|
||
}
|
||
if (stopped_by_random_signal)
|
||
{
|
||
/* We stopped inside the FUNCTION because of a random
|
||
signal. Further execution of the FUNCTION is not
|
||
allowed. */
|
||
|
||
if (unwind_on_signal_p)
|
||
{
|
||
/* The user wants the context restored. */
|
||
|
||
/* We must get back to the frame we were before the
|
||
dummy call. */
|
||
frame_pop (get_current_frame ());
|
||
|
||
/* FIXME: Insert a bunch of wrap_here; name can be very
|
||
long if it's a C++ name with arguments and stuff. */
|
||
error (_("\
|
||
The program being debugged was signaled while in a function called from GDB.\n\
|
||
GDB has restored the context to what it was before the call.\n\
|
||
To change this behavior use \"set unwindonsignal off\"\n\
|
||
Evaluation of the expression containing the function (%s) will be abandoned."),
|
||
name);
|
||
}
|
||
else
|
||
{
|
||
/* The user wants to stay in the frame where we stopped
|
||
(default).*/
|
||
/* If we restored the inferior status (via the cleanup),
|
||
we would print a spurious error message (Unable to
|
||
restore previously selected frame), would write the
|
||
registers from the inf_status (which is wrong), and
|
||
would do other wrong things. */
|
||
discard_cleanups (inf_status_cleanup);
|
||
discard_inferior_status (inf_status);
|
||
/* FIXME: Insert a bunch of wrap_here; name can be very
|
||
long if it's a C++ name with arguments and stuff. */
|
||
error (_("\
|
||
The program being debugged was signaled while in a function called from GDB.\n\
|
||
GDB remains in the frame where the signal was received.\n\
|
||
To change this behavior use \"set unwindonsignal on\"\n\
|
||
Evaluation of the expression containing the function (%s) will be abandoned."),
|
||
name);
|
||
}
|
||
}
|
||
|
||
if (!stop_stack_dummy)
|
||
{
|
||
/* We hit a breakpoint inside the FUNCTION. */
|
||
/* If we restored the inferior status (via the cleanup), we
|
||
would print a spurious error message (Unable to restore
|
||
previously selected frame), would write the registers
|
||
from the inf_status (which is wrong), and would do other
|
||
wrong things. */
|
||
discard_cleanups (inf_status_cleanup);
|
||
discard_inferior_status (inf_status);
|
||
/* The following error message used to say "The expression
|
||
which contained the function call has been discarded."
|
||
It is a hard concept to explain in a few words. Ideally,
|
||
GDB would be able to resume evaluation of the expression
|
||
when the function finally is done executing. Perhaps
|
||
someday this will be implemented (it would not be easy). */
|
||
/* FIXME: Insert a bunch of wrap_here; name can be very long if it's
|
||
a C++ name with arguments and stuff. */
|
||
error (_("\
|
||
The program being debugged stopped while in a function called from GDB.\n\
|
||
When the function (%s) is done executing, GDB will silently\n\
|
||
stop (instead of continuing to evaluate the expression containing\n\
|
||
the function call)."), name);
|
||
}
|
||
|
||
/* The above code errors out, so ... */
|
||
internal_error (__FILE__, __LINE__, _("... should not be here"));
|
||
}
|
||
|
||
/* If we get here the called FUNCTION run to completion. */
|
||
|
||
/* On normal return, the stack dummy has been popped already. */
|
||
regcache_cpy_no_passthrough (retbuf, stop_registers);
|
||
|
||
/* Restore the inferior status, via its cleanup. At this stage,
|
||
leave the RETBUF alone. */
|
||
do_cleanups (inf_status_cleanup);
|
||
|
||
/* Figure out the value returned by the function. */
|
||
{
|
||
struct value *retval = NULL;
|
||
|
||
if (TYPE_CODE (values_type) == TYPE_CODE_VOID)
|
||
{
|
||
/* If the function returns void, don't bother fetching the
|
||
return value. */
|
||
retval = allocate_value (values_type);
|
||
}
|
||
else
|
||
{
|
||
struct gdbarch *arch = current_gdbarch;
|
||
|
||
switch (gdbarch_return_value (arch, values_type, NULL, NULL, NULL))
|
||
{
|
||
case RETURN_VALUE_REGISTER_CONVENTION:
|
||
case RETURN_VALUE_ABI_RETURNS_ADDRESS:
|
||
case RETURN_VALUE_ABI_PRESERVES_ADDRESS:
|
||
retval = allocate_value (values_type);
|
||
gdbarch_return_value (current_gdbarch, values_type, retbuf,
|
||
value_contents_raw (retval), NULL);
|
||
break;
|
||
case RETURN_VALUE_STRUCT_CONVENTION:
|
||
retval = value_at (values_type, struct_addr);
|
||
break;
|
||
}
|
||
}
|
||
|
||
do_cleanups (retbuf_cleanup);
|
||
|
||
gdb_assert(retval);
|
||
return retval;
|
||
}
|
||
}
|
||
|
||
|
||
/* Provide a prototype to silence -Wmissing-prototypes. */
|
||
void _initialize_infcall (void);
|
||
|
||
void
|
||
_initialize_infcall (void)
|
||
{
|
||
add_setshow_boolean_cmd ("coerce-float-to-double", class_obscure,
|
||
&coerce_float_to_double_p, _("\
|
||
Set coercion of floats to doubles when calling functions."), _("\
|
||
Show coercion of floats to doubles when calling functions"), _("\
|
||
Variables of type float should generally be converted to doubles before\n\
|
||
calling an unprototyped function, and left alone when calling a prototyped\n\
|
||
function. However, some older debug info formats do not provide enough\n\
|
||
information to determine that a function is prototyped. If this flag is\n\
|
||
set, GDB will perform the conversion for a function it considers\n\
|
||
unprototyped.\n\
|
||
The default is to perform the conversion.\n"),
|
||
NULL,
|
||
show_coerce_float_to_double_p,
|
||
&setlist, &showlist);
|
||
|
||
add_setshow_boolean_cmd ("unwindonsignal", no_class,
|
||
&unwind_on_signal_p, _("\
|
||
Set unwinding of stack if a signal is received while in a call dummy."), _("\
|
||
Show unwinding of stack if a signal is received while in a call dummy."), _("\
|
||
The unwindonsignal lets the user determine what gdb should do if a signal\n\
|
||
is received while in a function called from gdb (call dummy). If set, gdb\n\
|
||
unwinds the stack and restore the context to what as it was before the call.\n\
|
||
The default is to stop in the frame where the signal was received."),
|
||
NULL,
|
||
show_unwind_on_signal_p,
|
||
&setlist, &showlist);
|
||
}
|