fd3619828e
This large patch removes the unnecessary bfd parameter from various bfd section macros and functions. The bfd is hardly ever used and if needed for the bfd_set_section_* or bfd_rename_section functions can be found via section->owner except for the com, und, abs, and ind std_section special sections. Those sections shouldn't be modified anyway. The patch also removes various bfd_get_section_<field> macros, replacing their use with bfd_section_<field>, and adds bfd_set_section_lma. I've also fixed a minor bug in gas where compressed section renaming was done directly rather than calling bfd_rename_section. This would have broken bfd_get_section_by_name and similar functions, but that hardly mattered at such a late stage in gas processing. bfd/ * bfd-in.h (bfd_get_section_name, bfd_get_section_vma), (bfd_get_section_lma, bfd_get_section_alignment), (bfd_get_section_size, bfd_get_section_flags), (bfd_get_section_userdata): Delete. (bfd_section_name, bfd_section_size, bfd_section_vma), (bfd_section_lma, bfd_section_alignment): Lose bfd parameter. (bfd_section_flags, bfd_section_userdata): New. (bfd_is_com_section): Rename parameter. * section.c (bfd_set_section_userdata, bfd_set_section_vma), (bfd_set_section_alignment, bfd_set_section_flags, bfd_rename_section), (bfd_set_section_size): Delete bfd parameter, rename section parameter. (bfd_set_section_lma): New. * bfd-in2.h: Regenerate. * mach-o.c (bfd_mach_o_init_section_from_mach_o): Delete bfd param, update callers. * aoutx.h, * bfd.c, * coff-alpha.c, * coff-arm.c, * coff-mips.c, * coff64-rs6000.c, * coffcode.h, * coffgen.c, * cofflink.c, * compress.c, * ecoff.c, * elf-eh-frame.c, * elf-hppa.h, * elf-ifunc.c, * elf-m10200.c, * elf-m10300.c, * elf-properties.c, * elf-s390-common.c, * elf-vxworks.c, * elf.c, * elf32-arc.c, * elf32-arm.c, * elf32-avr.c, * elf32-bfin.c, * elf32-cr16.c, * elf32-cr16c.c, * elf32-cris.c, * elf32-crx.c, * elf32-csky.c, * elf32-d10v.c, * elf32-epiphany.c, * elf32-fr30.c, * elf32-frv.c, * elf32-ft32.c, * elf32-h8300.c, * elf32-hppa.c, * elf32-i386.c, * elf32-ip2k.c, * elf32-iq2000.c, * elf32-lm32.c, * elf32-m32c.c, * elf32-m32r.c, * elf32-m68hc1x.c, * elf32-m68k.c, * elf32-mcore.c, * elf32-mep.c, * elf32-metag.c, * elf32-microblaze.c, * elf32-moxie.c, * elf32-msp430.c, * elf32-mt.c, * elf32-nds32.c, * elf32-nios2.c, * elf32-or1k.c, * elf32-ppc.c, * elf32-pru.c, * elf32-rl78.c, * elf32-rx.c, * elf32-s390.c, * elf32-score.c, * elf32-score7.c, * elf32-sh.c, * elf32-spu.c, * elf32-tic6x.c, * elf32-tilepro.c, * elf32-v850.c, * elf32-vax.c, * elf32-visium.c, * elf32-xstormy16.c, * elf32-xtensa.c, * elf64-alpha.c, * elf64-bpf.c, * elf64-hppa.c, * elf64-ia64-vms.c, * elf64-mmix.c, * elf64-ppc.c, * elf64-s390.c, * elf64-sparc.c, * elf64-x86-64.c, * elflink.c, * elfnn-aarch64.c, * elfnn-ia64.c, * elfnn-riscv.c, * elfxx-aarch64.c, * elfxx-mips.c, * elfxx-sparc.c, * elfxx-tilegx.c, * elfxx-x86.c, * i386msdos.c, * linker.c, * mach-o.c, * mmo.c, * opncls.c, * pdp11.c, * pei-x86_64.c, * peicode.h, * reloc.c, * section.c, * syms.c, * vms-alpha.c, * xcofflink.c: Update throughout for bfd section macro and function changes. binutils/ * addr2line.c, * bucomm.c, * coffgrok.c, * dlltool.c, * nm.c, * objcopy.c, * objdump.c, * od-elf32_avr.c, * od-macho.c, * od-xcoff.c, * prdbg.c, * rdcoff.c, * rddbg.c, * rescoff.c, * resres.c, * size.c, * srconv.c, * strings.c, * windmc.c: Update throughout for bfd section macro and function changes. gas/ * as.c, * as.h, * dw2gencfi.c, * dwarf2dbg.c, * ecoff.c, * read.c, * stabs.c, * subsegs.c, * subsegs.h, * write.c, * config/obj-coff-seh.c, * config/obj-coff.c, * config/obj-ecoff.c, * config/obj-elf.c, * config/obj-macho.c, * config/obj-som.c, * config/tc-aarch64.c, * config/tc-alpha.c, * config/tc-arc.c, * config/tc-arm.c, * config/tc-avr.c, * config/tc-bfin.c, * config/tc-bpf.c, * config/tc-d10v.c, * config/tc-d30v.c, * config/tc-epiphany.c, * config/tc-fr30.c, * config/tc-frv.c, * config/tc-h8300.c, * config/tc-hppa.c, * config/tc-i386.c, * config/tc-ia64.c, * config/tc-ip2k.c, * config/tc-iq2000.c, * config/tc-lm32.c, * config/tc-m32c.c, * config/tc-m32r.c, * config/tc-m68hc11.c, * config/tc-mep.c, * config/tc-microblaze.c, * config/tc-mips.c, * config/tc-mmix.c, * config/tc-mn10200.c, * config/tc-mn10300.c, * config/tc-msp430.c, * config/tc-mt.c, * config/tc-nds32.c, * config/tc-or1k.c, * config/tc-ppc.c, * config/tc-pru.c, * config/tc-rl78.c, * config/tc-rx.c, * config/tc-s12z.c, * config/tc-s390.c, * config/tc-score.c, * config/tc-score7.c, * config/tc-sh.c, * config/tc-sparc.c, * config/tc-spu.c, * config/tc-tic4x.c, * config/tc-tic54x.c, * config/tc-tic6x.c, * config/tc-tilegx.c, * config/tc-tilepro.c, * config/tc-v850.c, * config/tc-visium.c, * config/tc-wasm32.c, * config/tc-xc16x.c, * config/tc-xgate.c, * config/tc-xstormy16.c, * config/tc-xtensa.c, * config/tc-z8k.c: Update throughout for bfd section macro and function changes. * write.c (compress_debug): Use bfd_rename_section. gdb/ * aarch64-linux-tdep.c, * arm-tdep.c, * auto-load.c, * coff-pe-read.c, * coffread.c, * corelow.c, * dbxread.c, * dicos-tdep.c, * dwarf2-frame.c, * dwarf2read.c, * elfread.c, * exec.c, * fbsd-tdep.c, * gcore.c, * gdb_bfd.c, * gdb_bfd.h, * hppa-tdep.c, * i386-cygwin-tdep.c, * i386-fbsd-tdep.c, * i386-linux-tdep.c, * jit.c, * linux-tdep.c, * machoread.c, * maint.c, * mdebugread.c, * minidebug.c, * mips-linux-tdep.c, * mips-sde-tdep.c, * mips-tdep.c, * mipsread.c, * nto-tdep.c, * objfiles.c, * objfiles.h, * osabi.c, * ppc-linux-tdep.c, * ppc64-tdep.c, * record-btrace.c, * record-full.c, * remote.c, * rs6000-aix-tdep.c, * rs6000-tdep.c, * s390-linux-tdep.c, * s390-tdep.c, * solib-aix.c, * solib-dsbt.c, * solib-frv.c, * solib-spu.c, * solib-svr4.c, * solib-target.c, * spu-linux-nat.c, * spu-tdep.c, * symfile-mem.c, * symfile.c, * symmisc.c, * symtab.c, * target.c, * windows-nat.c, * xcoffread.c, * cli/cli-dump.c, * compile/compile-object-load.c, * mi/mi-interp.c: Update throughout for bfd section macro and function changes. * gcore (gcore_create_callback): Use bfd_set_section_lma. * spu-tdep.c (spu_overlay_new_objfile): Likewise. gprof/ * corefile.c, * symtab.c: Update throughout for bfd section macro and function changes. ld/ * ldcref.c, * ldctor.c, * ldelf.c, * ldlang.c, * pe-dll.c, * emultempl/aarch64elf.em, * emultempl/aix.em, * emultempl/armcoff.em, * emultempl/armelf.em, * emultempl/cr16elf.em, * emultempl/cskyelf.em, * emultempl/m68hc1xelf.em, * emultempl/m68kelf.em, * emultempl/mipself.em, * emultempl/mmix-elfnmmo.em, * emultempl/mmo.em, * emultempl/msp430.em, * emultempl/nios2elf.em, * emultempl/pe.em, * emultempl/pep.em, * emultempl/ppc64elf.em, * emultempl/xtensaelf.em: Update throughout for bfd section macro and function changes. libctf/ * ctf-open-bfd.c: Update throughout for bfd section macro changes. opcodes/ * arc-ext.c: Update throughout for bfd section macro changes. sim/ * common/sim-load.c, * common/sim-utils.c, * cris/sim-if.c, * erc32/func.c, * lm32/sim-if.c, * m32c/load.c, * m32c/trace.c, * m68hc11/interp.c, * ppc/hw_htab.c, * ppc/hw_init.c, * rl78/load.c, * rl78/trace.c, * rx/gdb-if.c, * rx/load.c, * rx/trace.c: Update throughout for bfd section macro changes.
405 lines
9.3 KiB
C
405 lines
9.3 KiB
C
/* Miscellaneous simulator utilities.
|
||
Copyright (C) 1997-2019 Free Software Foundation, Inc.
|
||
Contributed by Cygnus Support.
|
||
|
||
This file is part of GDB, the GNU debugger.
|
||
|
||
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/>. */
|
||
|
||
#include "sim-main.h"
|
||
#include "sim-assert.h"
|
||
|
||
#ifdef HAVE_STDLIB_H
|
||
#include <stdlib.h>
|
||
#endif
|
||
|
||
#ifdef HAVE_TIME_H
|
||
#include <time.h>
|
||
#endif
|
||
|
||
#ifdef HAVE_SYS_TIME_H
|
||
#include <sys/time.h> /* needed by sys/resource.h */
|
||
#endif
|
||
|
||
#ifdef HAVE_SYS_RESOURCE_H
|
||
#include <sys/resource.h>
|
||
#endif
|
||
|
||
#ifdef HAVE_STRING_H
|
||
#include <string.h>
|
||
#else
|
||
#ifdef HAVE_STRINGS_H
|
||
#include <strings.h>
|
||
#endif
|
||
#endif
|
||
|
||
#include "libiberty.h"
|
||
#include "bfd.h"
|
||
#include "sim-utils.h"
|
||
|
||
/* Allocate zero filled memory with xcalloc - xcalloc aborts if the
|
||
allocation fails. */
|
||
|
||
void *
|
||
zalloc (unsigned long size)
|
||
{
|
||
return xcalloc (1, size);
|
||
}
|
||
|
||
/* Allocate a sim_state struct. */
|
||
|
||
SIM_DESC
|
||
sim_state_alloc (SIM_OPEN_KIND kind,
|
||
host_callback *callback)
|
||
{
|
||
SIM_DESC sd = ZALLOC (struct sim_state);
|
||
|
||
STATE_MAGIC (sd) = SIM_MAGIC_NUMBER;
|
||
STATE_CALLBACK (sd) = callback;
|
||
STATE_OPEN_KIND (sd) = kind;
|
||
|
||
#if 0
|
||
{
|
||
int cpu_nr;
|
||
|
||
/* Initialize the back link from the cpu struct to the state struct. */
|
||
/* ??? I can envision a design where the state struct contains an array
|
||
of pointers to cpu structs, rather than an array of structs themselves.
|
||
Implementing this is trickier as one may not know what to allocate until
|
||
one has parsed the args. Parsing the args twice wouldn't be unreasonable,
|
||
IMHO. If the state struct ever does contain an array of pointers then we
|
||
can't do this here.
|
||
??? See also sim_post_argv_init*/
|
||
for (cpu_nr = 0; cpu_nr < MAX_NR_PROCESSORS; cpu_nr++)
|
||
{
|
||
CPU_STATE (STATE_CPU (sd, cpu_nr)) = sd;
|
||
CPU_INDEX (STATE_CPU (sd, cpu_nr)) = cpu_nr;
|
||
}
|
||
}
|
||
#endif
|
||
|
||
#ifdef SIM_STATE_INIT
|
||
SIM_STATE_INIT (sd);
|
||
#endif
|
||
|
||
return sd;
|
||
}
|
||
|
||
/* Free a sim_state struct. */
|
||
|
||
void
|
||
sim_state_free (SIM_DESC sd)
|
||
{
|
||
ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER);
|
||
|
||
#ifdef SIM_STATE_FREE
|
||
SIM_STATE_FREE (sd);
|
||
#endif
|
||
|
||
free (sd);
|
||
}
|
||
|
||
/* Return a pointer to the cpu data for CPU_NAME, or NULL if not found. */
|
||
|
||
sim_cpu *
|
||
sim_cpu_lookup (SIM_DESC sd, const char *cpu_name)
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < MAX_NR_PROCESSORS; ++i)
|
||
if (strcmp (cpu_name, CPU_NAME (STATE_CPU (sd, i))) == 0)
|
||
return STATE_CPU (sd, i);
|
||
return NULL;
|
||
}
|
||
|
||
/* Return the prefix to use for a CPU specific message (typically an
|
||
error message). */
|
||
|
||
const char *
|
||
sim_cpu_msg_prefix (sim_cpu *cpu)
|
||
{
|
||
#if MAX_NR_PROCESSORS == 1
|
||
return "";
|
||
#else
|
||
static char *prefix;
|
||
|
||
if (prefix == NULL)
|
||
{
|
||
int maxlen = 0;
|
||
for (i = 0; i < MAX_NR_PROCESSORS; ++i)
|
||
{
|
||
int len = strlen (CPU_NAME (STATE_CPU (sd, i)));
|
||
if (len > maxlen)
|
||
maxlen = len;
|
||
}
|
||
prefix = (char *) xmalloc (maxlen + 5);
|
||
}
|
||
sprintf (prefix, "%s: ", CPU_NAME (cpu));
|
||
return prefix;
|
||
#endif
|
||
}
|
||
|
||
/* Cover fn to sim_io_eprintf. */
|
||
|
||
void
|
||
sim_io_eprintf_cpu (sim_cpu *cpu, const char *fmt, ...)
|
||
{
|
||
SIM_DESC sd = CPU_STATE (cpu);
|
||
va_list ap;
|
||
|
||
va_start (ap, fmt);
|
||
sim_io_eprintf (sd, "%s", sim_cpu_msg_prefix (cpu));
|
||
sim_io_evprintf (sd, fmt, ap);
|
||
va_end (ap);
|
||
}
|
||
|
||
/* Turn VALUE into a string with commas. */
|
||
|
||
char *
|
||
sim_add_commas (char *buf, int sizeof_buf, unsigned long value)
|
||
{
|
||
int comma = 3;
|
||
char *endbuf = buf + sizeof_buf - 1;
|
||
|
||
*--endbuf = '\0';
|
||
do {
|
||
if (comma-- == 0)
|
||
{
|
||
*--endbuf = ',';
|
||
comma = 2;
|
||
}
|
||
|
||
*--endbuf = (value % 10) + '0';
|
||
} while ((value /= 10) != 0);
|
||
|
||
return endbuf;
|
||
}
|
||
|
||
/* Analyze PROG_NAME/PROG_BFD and set these fields in the state struct:
|
||
STATE_ARCHITECTURE, if not set already and can be determined from the bfd
|
||
STATE_PROG_BFD
|
||
STATE_START_ADDR
|
||
STATE_TEXT_SECTION
|
||
STATE_TEXT_START
|
||
STATE_TEXT_END
|
||
|
||
PROG_NAME is the file name of the executable or NULL.
|
||
PROG_BFD is its bfd or NULL.
|
||
|
||
If both PROG_NAME and PROG_BFD are NULL, this function returns immediately.
|
||
If PROG_BFD is not NULL, PROG_NAME is ignored.
|
||
|
||
Implicit inputs: STATE_MY_NAME(sd), STATE_TARGET(sd),
|
||
STATE_ARCHITECTURE(sd).
|
||
|
||
A new bfd is created so the app isn't required to keep its copy of the
|
||
bfd open. */
|
||
|
||
SIM_RC
|
||
sim_analyze_program (SIM_DESC sd, const char *prog_name, bfd *prog_bfd)
|
||
{
|
||
asection *s;
|
||
SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER);
|
||
|
||
if (prog_bfd != NULL)
|
||
{
|
||
if (prog_bfd == STATE_PROG_BFD (sd))
|
||
/* already analyzed */
|
||
return SIM_RC_OK;
|
||
else
|
||
/* duplicate needed, save the name of the file to be re-opened */
|
||
prog_name = bfd_get_filename (prog_bfd);
|
||
}
|
||
|
||
/* do we need to duplicate anything? */
|
||
if (prog_name == NULL)
|
||
return SIM_RC_OK;
|
||
|
||
/* open a new copy of the prog_bfd */
|
||
prog_bfd = bfd_openr (prog_name, STATE_TARGET (sd));
|
||
if (prog_bfd == NULL)
|
||
{
|
||
sim_io_eprintf (sd, "%s: can't open \"%s\": %s\n",
|
||
STATE_MY_NAME (sd),
|
||
prog_name,
|
||
bfd_errmsg (bfd_get_error ()));
|
||
return SIM_RC_FAIL;
|
||
}
|
||
if (!bfd_check_format (prog_bfd, bfd_object))
|
||
{
|
||
sim_io_eprintf (sd, "%s: \"%s\" is not an object file: %s\n",
|
||
STATE_MY_NAME (sd),
|
||
prog_name,
|
||
bfd_errmsg (bfd_get_error ()));
|
||
bfd_close (prog_bfd);
|
||
return SIM_RC_FAIL;
|
||
}
|
||
if (STATE_ARCHITECTURE (sd) != NULL)
|
||
bfd_set_arch_info (prog_bfd, STATE_ARCHITECTURE (sd));
|
||
else
|
||
{
|
||
if (bfd_get_arch (prog_bfd) != bfd_arch_unknown
|
||
&& bfd_get_arch (prog_bfd) != bfd_arch_obscure)
|
||
{
|
||
STATE_ARCHITECTURE (sd) = bfd_get_arch_info (prog_bfd);
|
||
}
|
||
}
|
||
|
||
/* update the sim structure */
|
||
if (STATE_PROG_BFD (sd) != NULL)
|
||
bfd_close (STATE_PROG_BFD (sd));
|
||
STATE_PROG_BFD (sd) = prog_bfd;
|
||
STATE_START_ADDR (sd) = bfd_get_start_address (prog_bfd);
|
||
|
||
for (s = prog_bfd->sections; s; s = s->next)
|
||
if (strcmp (bfd_section_name (s), ".text") == 0)
|
||
{
|
||
STATE_TEXT_SECTION (sd) = s;
|
||
STATE_TEXT_START (sd) = bfd_section_vma (s);
|
||
STATE_TEXT_END (sd) = STATE_TEXT_START (sd) + bfd_section_size (s);
|
||
break;
|
||
}
|
||
|
||
bfd_cache_close (prog_bfd);
|
||
|
||
return SIM_RC_OK;
|
||
}
|
||
|
||
/* Simulator timing support. */
|
||
|
||
/* Called before sim_elapsed_time_since to get a reference point. */
|
||
|
||
SIM_ELAPSED_TIME
|
||
sim_elapsed_time_get (void)
|
||
{
|
||
#ifdef HAVE_GETRUSAGE
|
||
struct rusage mytime;
|
||
if (getrusage (RUSAGE_SELF, &mytime) == 0)
|
||
return 1 + (SIM_ELAPSED_TIME) (((double) mytime.ru_utime.tv_sec * 1000) + (((double) mytime.ru_utime.tv_usec + 500) / 1000));
|
||
return 1;
|
||
#else
|
||
#ifdef HAVE_TIME
|
||
return 1 + (SIM_ELAPSED_TIME) time ((time_t) 0);
|
||
#else
|
||
return 1;
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
/* Return the elapsed time in milliseconds since START.
|
||
The actual time may be cpu usage (preferred) or wall clock. */
|
||
|
||
unsigned long
|
||
sim_elapsed_time_since (SIM_ELAPSED_TIME start)
|
||
{
|
||
#ifdef HAVE_GETRUSAGE
|
||
return sim_elapsed_time_get () - start;
|
||
#else
|
||
#ifdef HAVE_TIME
|
||
return (sim_elapsed_time_get () - start) * 1000;
|
||
#else
|
||
return 0;
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
|
||
|
||
/* do_command but with printf style formatting of the arguments */
|
||
void
|
||
sim_do_commandf (SIM_DESC sd,
|
||
const char *fmt,
|
||
...)
|
||
{
|
||
va_list ap;
|
||
char *buf;
|
||
int ret;
|
||
|
||
va_start (ap, fmt);
|
||
ret = vasprintf (&buf, fmt, ap);
|
||
va_end (ap);
|
||
|
||
if (ret < 0)
|
||
{
|
||
sim_io_eprintf (sd, "%s: asprintf failed for `%s'\n",
|
||
STATE_MY_NAME (sd), fmt);
|
||
return;
|
||
}
|
||
|
||
sim_do_command (sd, buf);
|
||
free (buf);
|
||
}
|
||
|
||
|
||
/* sim-basics.h defines a number of enumerations, convert each of them
|
||
to a string representation */
|
||
const char *
|
||
map_to_str (unsigned map)
|
||
{
|
||
switch (map)
|
||
{
|
||
case read_map: return "read";
|
||
case write_map: return "write";
|
||
case exec_map: return "exec";
|
||
case io_map: return "io";
|
||
default:
|
||
{
|
||
static char str[10];
|
||
sprintf (str, "(%ld)", (long) map);
|
||
return str;
|
||
}
|
||
}
|
||
}
|
||
|
||
const char *
|
||
access_to_str (unsigned access)
|
||
{
|
||
switch (access)
|
||
{
|
||
case access_invalid: return "invalid";
|
||
case access_read: return "read";
|
||
case access_write: return "write";
|
||
case access_exec: return "exec";
|
||
case access_io: return "io";
|
||
case access_read_write: return "read_write";
|
||
case access_read_exec: return "read_exec";
|
||
case access_write_exec: return "write_exec";
|
||
case access_read_write_exec: return "read_write_exec";
|
||
case access_read_io: return "read_io";
|
||
case access_write_io: return "write_io";
|
||
case access_read_write_io: return "read_write_io";
|
||
case access_exec_io: return "exec_io";
|
||
case access_read_exec_io: return "read_exec_io";
|
||
case access_write_exec_io: return "write_exec_io";
|
||
case access_read_write_exec_io: return "read_write_exec_io";
|
||
default:
|
||
{
|
||
static char str[10];
|
||
sprintf (str, "(%ld)", (long) access);
|
||
return str;
|
||
}
|
||
}
|
||
}
|
||
|
||
const char *
|
||
transfer_to_str (unsigned transfer)
|
||
{
|
||
switch (transfer)
|
||
{
|
||
case read_transfer: return "read";
|
||
case write_transfer: return "write";
|
||
default: return "(error)";
|
||
}
|
||
}
|