875 lines
25 KiB
C
875 lines
25 KiB
C
/* Target-dependent code for GNU/Linux, architecture independent.
|
|
|
|
Copyright (C) 2009-2012 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/>. */
|
|
|
|
#include "defs.h"
|
|
#include "gdbtypes.h"
|
|
#include "linux-tdep.h"
|
|
#include "auxv.h"
|
|
#include "target.h"
|
|
#include "gdbthread.h"
|
|
#include "gdbcore.h"
|
|
#include "regcache.h"
|
|
#include "regset.h"
|
|
#include "elf/common.h"
|
|
#include "elf-bfd.h" /* for elfcore_write_* */
|
|
#include "inferior.h"
|
|
#include "cli/cli-utils.h"
|
|
|
|
#include <ctype.h>
|
|
|
|
static struct gdbarch_data *linux_gdbarch_data_handle;
|
|
|
|
struct linux_gdbarch_data
|
|
{
|
|
struct type *siginfo_type;
|
|
};
|
|
|
|
static void *
|
|
init_linux_gdbarch_data (struct gdbarch *gdbarch)
|
|
{
|
|
return GDBARCH_OBSTACK_ZALLOC (gdbarch, struct linux_gdbarch_data);
|
|
}
|
|
|
|
static struct linux_gdbarch_data *
|
|
get_linux_gdbarch_data (struct gdbarch *gdbarch)
|
|
{
|
|
return gdbarch_data (gdbarch, linux_gdbarch_data_handle);
|
|
}
|
|
|
|
/* This function is suitable for architectures that don't
|
|
extend/override the standard siginfo structure. */
|
|
|
|
struct type *
|
|
linux_get_siginfo_type (struct gdbarch *gdbarch)
|
|
{
|
|
struct linux_gdbarch_data *linux_gdbarch_data;
|
|
struct type *int_type, *uint_type, *long_type, *void_ptr_type;
|
|
struct type *uid_type, *pid_type;
|
|
struct type *sigval_type, *clock_type;
|
|
struct type *siginfo_type, *sifields_type;
|
|
struct type *type;
|
|
|
|
linux_gdbarch_data = get_linux_gdbarch_data (gdbarch);
|
|
if (linux_gdbarch_data->siginfo_type != NULL)
|
|
return linux_gdbarch_data->siginfo_type;
|
|
|
|
int_type = arch_integer_type (gdbarch, gdbarch_int_bit (gdbarch),
|
|
0, "int");
|
|
uint_type = arch_integer_type (gdbarch, gdbarch_int_bit (gdbarch),
|
|
1, "unsigned int");
|
|
long_type = arch_integer_type (gdbarch, gdbarch_long_bit (gdbarch),
|
|
0, "long");
|
|
void_ptr_type = lookup_pointer_type (builtin_type (gdbarch)->builtin_void);
|
|
|
|
/* sival_t */
|
|
sigval_type = arch_composite_type (gdbarch, NULL, TYPE_CODE_UNION);
|
|
TYPE_NAME (sigval_type) = xstrdup ("sigval_t");
|
|
append_composite_type_field (sigval_type, "sival_int", int_type);
|
|
append_composite_type_field (sigval_type, "sival_ptr", void_ptr_type);
|
|
|
|
/* __pid_t */
|
|
pid_type = arch_type (gdbarch, TYPE_CODE_TYPEDEF,
|
|
TYPE_LENGTH (int_type), "__pid_t");
|
|
TYPE_TARGET_TYPE (pid_type) = int_type;
|
|
TYPE_TARGET_STUB (pid_type) = 1;
|
|
|
|
/* __uid_t */
|
|
uid_type = arch_type (gdbarch, TYPE_CODE_TYPEDEF,
|
|
TYPE_LENGTH (uint_type), "__uid_t");
|
|
TYPE_TARGET_TYPE (uid_type) = uint_type;
|
|
TYPE_TARGET_STUB (uid_type) = 1;
|
|
|
|
/* __clock_t */
|
|
clock_type = arch_type (gdbarch, TYPE_CODE_TYPEDEF,
|
|
TYPE_LENGTH (long_type), "__clock_t");
|
|
TYPE_TARGET_TYPE (clock_type) = long_type;
|
|
TYPE_TARGET_STUB (clock_type) = 1;
|
|
|
|
/* _sifields */
|
|
sifields_type = arch_composite_type (gdbarch, NULL, TYPE_CODE_UNION);
|
|
|
|
{
|
|
const int si_max_size = 128;
|
|
int si_pad_size;
|
|
int size_of_int = gdbarch_int_bit (gdbarch) / HOST_CHAR_BIT;
|
|
|
|
/* _pad */
|
|
if (gdbarch_ptr_bit (gdbarch) == 64)
|
|
si_pad_size = (si_max_size / size_of_int) - 4;
|
|
else
|
|
si_pad_size = (si_max_size / size_of_int) - 3;
|
|
append_composite_type_field (sifields_type, "_pad",
|
|
init_vector_type (int_type, si_pad_size));
|
|
}
|
|
|
|
/* _kill */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_pid", pid_type);
|
|
append_composite_type_field (type, "si_uid", uid_type);
|
|
append_composite_type_field (sifields_type, "_kill", type);
|
|
|
|
/* _timer */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_tid", int_type);
|
|
append_composite_type_field (type, "si_overrun", int_type);
|
|
append_composite_type_field (type, "si_sigval", sigval_type);
|
|
append_composite_type_field (sifields_type, "_timer", type);
|
|
|
|
/* _rt */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_pid", pid_type);
|
|
append_composite_type_field (type, "si_uid", uid_type);
|
|
append_composite_type_field (type, "si_sigval", sigval_type);
|
|
append_composite_type_field (sifields_type, "_rt", type);
|
|
|
|
/* _sigchld */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_pid", pid_type);
|
|
append_composite_type_field (type, "si_uid", uid_type);
|
|
append_composite_type_field (type, "si_status", int_type);
|
|
append_composite_type_field (type, "si_utime", clock_type);
|
|
append_composite_type_field (type, "si_stime", clock_type);
|
|
append_composite_type_field (sifields_type, "_sigchld", type);
|
|
|
|
/* _sigfault */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_addr", void_ptr_type);
|
|
append_composite_type_field (sifields_type, "_sigfault", type);
|
|
|
|
/* _sigpoll */
|
|
type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
append_composite_type_field (type, "si_band", long_type);
|
|
append_composite_type_field (type, "si_fd", int_type);
|
|
append_composite_type_field (sifields_type, "_sigpoll", type);
|
|
|
|
/* struct siginfo */
|
|
siginfo_type = arch_composite_type (gdbarch, NULL, TYPE_CODE_STRUCT);
|
|
TYPE_NAME (siginfo_type) = xstrdup ("siginfo");
|
|
append_composite_type_field (siginfo_type, "si_signo", int_type);
|
|
append_composite_type_field (siginfo_type, "si_errno", int_type);
|
|
append_composite_type_field (siginfo_type, "si_code", int_type);
|
|
append_composite_type_field_aligned (siginfo_type,
|
|
"_sifields", sifields_type,
|
|
TYPE_LENGTH (long_type));
|
|
|
|
linux_gdbarch_data->siginfo_type = siginfo_type;
|
|
|
|
return siginfo_type;
|
|
}
|
|
|
|
static int
|
|
linux_has_shared_address_space (struct gdbarch *gdbarch)
|
|
{
|
|
/* Determine whether we are running on uClinux or normal Linux
|
|
kernel. */
|
|
CORE_ADDR dummy;
|
|
int target_is_uclinux;
|
|
|
|
target_is_uclinux
|
|
= (target_auxv_search (¤t_target, AT_NULL, &dummy) > 0
|
|
&& target_auxv_search (¤t_target, AT_PAGESZ, &dummy) == 0);
|
|
|
|
return target_is_uclinux;
|
|
}
|
|
|
|
/* This is how we want PTIDs from core files to be printed. */
|
|
|
|
static char *
|
|
linux_core_pid_to_str (struct gdbarch *gdbarch, ptid_t ptid)
|
|
{
|
|
static char buf[80];
|
|
|
|
if (ptid_get_lwp (ptid) != 0)
|
|
{
|
|
snprintf (buf, sizeof (buf), "LWP %ld", ptid_get_lwp (ptid));
|
|
return buf;
|
|
}
|
|
|
|
return normal_pid_to_str (ptid);
|
|
}
|
|
|
|
/* Service function for corefiles and info proc. */
|
|
|
|
static void
|
|
read_mapping (const char *line,
|
|
ULONGEST *addr, ULONGEST *endaddr,
|
|
const char **permissions, size_t *permissions_len,
|
|
ULONGEST *offset,
|
|
const char **device, size_t *device_len,
|
|
ULONGEST *inode,
|
|
const char **filename)
|
|
{
|
|
const char *p = line;
|
|
|
|
*addr = strtoulst (p, &p, 16);
|
|
if (*p == '-')
|
|
p++;
|
|
*endaddr = strtoulst (p, &p, 16);
|
|
|
|
while (*p && isspace (*p))
|
|
p++;
|
|
*permissions = p;
|
|
while (*p && !isspace (*p))
|
|
p++;
|
|
*permissions_len = p - *permissions;
|
|
|
|
*offset = strtoulst (p, &p, 16);
|
|
|
|
while (*p && isspace (*p))
|
|
p++;
|
|
*device = p;
|
|
while (*p && !isspace (*p))
|
|
p++;
|
|
*device_len = p - *device;
|
|
|
|
*inode = strtoulst (p, &p, 10);
|
|
|
|
while (*p && isspace (*p))
|
|
p++;
|
|
*filename = p;
|
|
}
|
|
|
|
/* Implement the "info proc" command. */
|
|
|
|
static void
|
|
linux_info_proc (struct gdbarch *gdbarch, char *args,
|
|
enum info_proc_what what)
|
|
{
|
|
/* A long is used for pid instead of an int to avoid a loss of precision
|
|
compiler warning from the output of strtoul. */
|
|
long pid;
|
|
int cmdline_f = (what == IP_MINIMAL || what == IP_CMDLINE || what == IP_ALL);
|
|
int cwd_f = (what == IP_MINIMAL || what == IP_CWD || what == IP_ALL);
|
|
int exe_f = (what == IP_MINIMAL || what == IP_EXE || what == IP_ALL);
|
|
int mappings_f = (what == IP_MAPPINGS || what == IP_ALL);
|
|
int status_f = (what == IP_STATUS || what == IP_ALL);
|
|
int stat_f = (what == IP_STAT || what == IP_ALL);
|
|
char filename[100];
|
|
gdb_byte *data;
|
|
int target_errno;
|
|
|
|
if (args && isdigit (args[0]))
|
|
pid = strtoul (args, &args, 10);
|
|
else
|
|
{
|
|
if (!target_has_execution)
|
|
error (_("No current process: you must name one."));
|
|
if (current_inferior ()->fake_pid_p)
|
|
error (_("Can't determine the current process's PID: you must name one."));
|
|
|
|
pid = current_inferior ()->pid;
|
|
}
|
|
|
|
args = skip_spaces (args);
|
|
if (args && args[0])
|
|
error (_("Too many parameters: %s"), args);
|
|
|
|
printf_filtered (_("process %ld\n"), pid);
|
|
if (cmdline_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/cmdline", pid);
|
|
data = target_fileio_read_stralloc (filename);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
printf_filtered ("cmdline = '%s'\n", data);
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to open /proc file '%s'"), filename);
|
|
}
|
|
if (cwd_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/cwd", pid);
|
|
data = target_fileio_readlink (filename, &target_errno);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
printf_filtered ("cwd = '%s'\n", data);
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to read link '%s'"), filename);
|
|
}
|
|
if (exe_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/exe", pid);
|
|
data = target_fileio_readlink (filename, &target_errno);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
printf_filtered ("exe = '%s'\n", data);
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to read link '%s'"), filename);
|
|
}
|
|
if (mappings_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/maps", pid);
|
|
data = target_fileio_read_stralloc (filename);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
char *line;
|
|
|
|
printf_filtered (_("Mapped address spaces:\n\n"));
|
|
if (gdbarch_addr_bit (gdbarch) == 32)
|
|
{
|
|
printf_filtered ("\t%10s %10s %10s %10s %s\n",
|
|
"Start Addr",
|
|
" End Addr",
|
|
" Size", " Offset", "objfile");
|
|
}
|
|
else
|
|
{
|
|
printf_filtered (" %18s %18s %10s %10s %s\n",
|
|
"Start Addr",
|
|
" End Addr",
|
|
" Size", " Offset", "objfile");
|
|
}
|
|
|
|
for (line = strtok (data, "\n"); line; line = strtok (NULL, "\n"))
|
|
{
|
|
ULONGEST addr, endaddr, offset, inode;
|
|
const char *permissions, *device, *filename;
|
|
size_t permissions_len, device_len;
|
|
|
|
read_mapping (line, &addr, &endaddr,
|
|
&permissions, &permissions_len,
|
|
&offset, &device, &device_len,
|
|
&inode, &filename);
|
|
|
|
if (gdbarch_addr_bit (gdbarch) == 32)
|
|
{
|
|
printf_filtered ("\t%10s %10s %10s %10s %s\n",
|
|
paddress (gdbarch, addr),
|
|
paddress (gdbarch, endaddr),
|
|
hex_string (endaddr - addr),
|
|
hex_string (offset),
|
|
*filename? filename : "");
|
|
}
|
|
else
|
|
{
|
|
printf_filtered (" %18s %18s %10s %10s %s\n",
|
|
paddress (gdbarch, addr),
|
|
paddress (gdbarch, endaddr),
|
|
hex_string (endaddr - addr),
|
|
hex_string (offset),
|
|
*filename? filename : "");
|
|
}
|
|
}
|
|
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to open /proc file '%s'"), filename);
|
|
}
|
|
if (status_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/status", pid);
|
|
data = target_fileio_read_stralloc (filename);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
puts_filtered (data);
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to open /proc file '%s'"), filename);
|
|
}
|
|
if (stat_f)
|
|
{
|
|
xsnprintf (filename, sizeof filename, "/proc/%ld/stat", pid);
|
|
data = target_fileio_read_stralloc (filename);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
const char *p = data;
|
|
|
|
printf_filtered (_("Process: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
|
|
while (*p && isspace (*p))
|
|
p++;
|
|
if (*p == '(')
|
|
{
|
|
const char *ep = strchr (p, ')');
|
|
if (ep != NULL)
|
|
{
|
|
printf_filtered ("Exec file: %.*s\n",
|
|
(int) (ep - p - 1), p + 1);
|
|
p = ep + 1;
|
|
}
|
|
}
|
|
|
|
while (*p && isspace (*p))
|
|
p++;
|
|
if (*p)
|
|
printf_filtered (_("State: %c\n"), *p++);
|
|
|
|
if (*p)
|
|
printf_filtered (_("Parent process: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Process group: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Session id: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("TTY: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("TTY owner process group: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
|
|
if (*p)
|
|
printf_filtered (_("Flags: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Minor faults (no memory page): %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Minor faults, children: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Major faults (memory page faults): %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Major faults, children: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("utime: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("stime: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("utime, children: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("stime, children: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("jiffies remaining in current "
|
|
"time slice: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("'nice' value: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("jiffies until next timeout: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("jiffies until next SIGALRM: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("start time (jiffies since "
|
|
"system boot): %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Virtual memory size: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Resident set size: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("rlim: %s\n"),
|
|
pulongest (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Start of text: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("End of text: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Start of stack: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
#if 0 /* Don't know how architecture-dependent the rest is...
|
|
Anyway the signal bitmap info is available from "status". */
|
|
if (*p)
|
|
printf_filtered (_("Kernel stack pointer: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Kernel instr pointer: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Pending signals bitmap: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Blocked signals bitmap: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Ignored signals bitmap: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("Catched signals bitmap: %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
if (*p)
|
|
printf_filtered (_("wchan (system call): %s\n"),
|
|
hex_string (strtoulst (p, &p, 10)));
|
|
#endif
|
|
do_cleanups (cleanup);
|
|
}
|
|
else
|
|
warning (_("unable to open /proc file '%s'"), filename);
|
|
}
|
|
}
|
|
|
|
/* List memory regions in the inferior for a corefile. */
|
|
|
|
static int
|
|
linux_find_memory_regions (struct gdbarch *gdbarch,
|
|
find_memory_region_ftype func, void *obfd)
|
|
{
|
|
char filename[100];
|
|
gdb_byte *data;
|
|
|
|
/* We need to know the real target PID to access /proc. */
|
|
if (current_inferior ()->fake_pid_p)
|
|
return 1;
|
|
|
|
xsnprintf (filename, sizeof filename,
|
|
"/proc/%d/maps", current_inferior ()->pid);
|
|
data = target_fileio_read_stralloc (filename);
|
|
if (data)
|
|
{
|
|
struct cleanup *cleanup = make_cleanup (xfree, data);
|
|
char *line;
|
|
|
|
for (line = strtok (data, "\n"); line; line = strtok (NULL, "\n"))
|
|
{
|
|
ULONGEST addr, endaddr, offset, inode;
|
|
const char *permissions, *device, *filename;
|
|
size_t permissions_len, device_len;
|
|
int read, write, exec;
|
|
|
|
read_mapping (line, &addr, &endaddr, &permissions, &permissions_len,
|
|
&offset, &device, &device_len, &inode, &filename);
|
|
|
|
/* Decode permissions. */
|
|
read = (memchr (permissions, 'r', permissions_len) != 0);
|
|
write = (memchr (permissions, 'w', permissions_len) != 0);
|
|
exec = (memchr (permissions, 'x', permissions_len) != 0);
|
|
|
|
/* Invoke the callback function to create the corefile segment. */
|
|
func (addr, endaddr - addr, read, write, exec, obfd);
|
|
}
|
|
|
|
do_cleanups (cleanup);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Determine which signal stopped execution. */
|
|
|
|
static int
|
|
find_signalled_thread (struct thread_info *info, void *data)
|
|
{
|
|
if (info->suspend.stop_signal != GDB_SIGNAL_0
|
|
&& ptid_get_pid (info->ptid) == ptid_get_pid (inferior_ptid))
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static enum gdb_signal
|
|
find_stop_signal (void)
|
|
{
|
|
struct thread_info *info =
|
|
iterate_over_threads (find_signalled_thread, NULL);
|
|
|
|
if (info)
|
|
return info->suspend.stop_signal;
|
|
else
|
|
return GDB_SIGNAL_0;
|
|
}
|
|
|
|
/* Generate corefile notes for SPU contexts. */
|
|
|
|
static char *
|
|
linux_spu_make_corefile_notes (bfd *obfd, char *note_data, int *note_size)
|
|
{
|
|
static const char *spu_files[] =
|
|
{
|
|
"object-id",
|
|
"mem",
|
|
"regs",
|
|
"fpcr",
|
|
"lslr",
|
|
"decr",
|
|
"decr_status",
|
|
"signal1",
|
|
"signal1_type",
|
|
"signal2",
|
|
"signal2_type",
|
|
"event_mask",
|
|
"event_status",
|
|
"mbox_info",
|
|
"ibox_info",
|
|
"wbox_info",
|
|
"dma_info",
|
|
"proxydma_info",
|
|
};
|
|
|
|
enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch);
|
|
gdb_byte *spu_ids;
|
|
LONGEST i, j, size;
|
|
|
|
/* Determine list of SPU ids. */
|
|
size = target_read_alloc (¤t_target, TARGET_OBJECT_SPU,
|
|
NULL, &spu_ids);
|
|
|
|
/* Generate corefile notes for each SPU file. */
|
|
for (i = 0; i < size; i += 4)
|
|
{
|
|
int fd = extract_unsigned_integer (spu_ids + i, 4, byte_order);
|
|
|
|
for (j = 0; j < sizeof (spu_files) / sizeof (spu_files[0]); j++)
|
|
{
|
|
char annex[32], note_name[32];
|
|
gdb_byte *spu_data;
|
|
LONGEST spu_len;
|
|
|
|
xsnprintf (annex, sizeof annex, "%d/%s", fd, spu_files[j]);
|
|
spu_len = target_read_alloc (¤t_target, TARGET_OBJECT_SPU,
|
|
annex, &spu_data);
|
|
if (spu_len > 0)
|
|
{
|
|
xsnprintf (note_name, sizeof note_name, "SPU/%s", annex);
|
|
note_data = elfcore_write_note (obfd, note_data, note_size,
|
|
note_name, NT_SPU,
|
|
spu_data, spu_len);
|
|
xfree (spu_data);
|
|
|
|
if (!note_data)
|
|
{
|
|
xfree (spu_ids);
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (size > 0)
|
|
xfree (spu_ids);
|
|
|
|
return note_data;
|
|
}
|
|
|
|
/* Records the thread's register state for the corefile note
|
|
section. */
|
|
|
|
static char *
|
|
linux_collect_thread_registers (const struct regcache *regcache,
|
|
ptid_t ptid, bfd *obfd,
|
|
char *note_data, int *note_size,
|
|
enum gdb_signal stop_signal)
|
|
{
|
|
struct gdbarch *gdbarch = get_regcache_arch (regcache);
|
|
struct core_regset_section *sect_list;
|
|
unsigned long lwp;
|
|
|
|
sect_list = gdbarch_core_regset_sections (gdbarch);
|
|
gdb_assert (sect_list);
|
|
|
|
/* For remote targets the LWP may not be available, so use the TID. */
|
|
lwp = ptid_get_lwp (ptid);
|
|
if (!lwp)
|
|
lwp = ptid_get_tid (ptid);
|
|
|
|
while (sect_list->sect_name != NULL)
|
|
{
|
|
const struct regset *regset;
|
|
char *buf;
|
|
|
|
regset = gdbarch_regset_from_core_section (gdbarch,
|
|
sect_list->sect_name,
|
|
sect_list->size);
|
|
gdb_assert (regset && regset->collect_regset);
|
|
|
|
buf = xmalloc (sect_list->size);
|
|
regset->collect_regset (regset, regcache, -1, buf, sect_list->size);
|
|
|
|
/* PRSTATUS still needs to be treated specially. */
|
|
if (strcmp (sect_list->sect_name, ".reg") == 0)
|
|
note_data = (char *) elfcore_write_prstatus
|
|
(obfd, note_data, note_size, lwp,
|
|
gdb_signal_to_host (stop_signal), buf);
|
|
else
|
|
note_data = (char *) elfcore_write_register_note
|
|
(obfd, note_data, note_size,
|
|
sect_list->sect_name, buf, sect_list->size);
|
|
xfree (buf);
|
|
sect_list++;
|
|
|
|
if (!note_data)
|
|
return NULL;
|
|
}
|
|
|
|
return note_data;
|
|
}
|
|
|
|
struct linux_corefile_thread_data
|
|
{
|
|
struct gdbarch *gdbarch;
|
|
int pid;
|
|
bfd *obfd;
|
|
char *note_data;
|
|
int *note_size;
|
|
int num_notes;
|
|
enum gdb_signal stop_signal;
|
|
linux_collect_thread_registers_ftype collect;
|
|
};
|
|
|
|
/* Called by gdbthread.c once per thread. Records the thread's
|
|
register state for the corefile note section. */
|
|
|
|
static int
|
|
linux_corefile_thread_callback (struct thread_info *info, void *data)
|
|
{
|
|
struct linux_corefile_thread_data *args = data;
|
|
|
|
if (ptid_get_pid (info->ptid) == args->pid)
|
|
{
|
|
struct cleanup *old_chain;
|
|
struct regcache *regcache;
|
|
regcache = get_thread_arch_regcache (info->ptid, args->gdbarch);
|
|
|
|
old_chain = save_inferior_ptid ();
|
|
inferior_ptid = info->ptid;
|
|
target_fetch_registers (regcache, -1);
|
|
do_cleanups (old_chain);
|
|
|
|
args->note_data = args->collect (regcache, info->ptid, args->obfd,
|
|
args->note_data, args->note_size,
|
|
args->stop_signal);
|
|
args->num_notes++;
|
|
}
|
|
|
|
return !args->note_data;
|
|
}
|
|
|
|
/* Fills the "to_make_corefile_note" target vector. Builds the note
|
|
section for a corefile, and returns it in a malloc buffer. */
|
|
|
|
char *
|
|
linux_make_corefile_notes (struct gdbarch *gdbarch, bfd *obfd, int *note_size,
|
|
linux_collect_thread_registers_ftype collect)
|
|
{
|
|
struct linux_corefile_thread_data thread_args;
|
|
char *note_data = NULL;
|
|
gdb_byte *auxv;
|
|
int auxv_len;
|
|
|
|
/* Process information. */
|
|
if (get_exec_file (0))
|
|
{
|
|
const char *fname = lbasename (get_exec_file (0));
|
|
char *psargs = xstrdup (fname);
|
|
|
|
if (get_inferior_args ())
|
|
psargs = reconcat (psargs, psargs, " ", get_inferior_args (),
|
|
(char *) NULL);
|
|
|
|
note_data = elfcore_write_prpsinfo (obfd, note_data, note_size,
|
|
fname, psargs);
|
|
xfree (psargs);
|
|
|
|
if (!note_data)
|
|
return NULL;
|
|
}
|
|
|
|
/* Thread register information. */
|
|
thread_args.gdbarch = gdbarch;
|
|
thread_args.pid = ptid_get_pid (inferior_ptid);
|
|
thread_args.obfd = obfd;
|
|
thread_args.note_data = note_data;
|
|
thread_args.note_size = note_size;
|
|
thread_args.num_notes = 0;
|
|
thread_args.stop_signal = find_stop_signal ();
|
|
thread_args.collect = collect;
|
|
iterate_over_threads (linux_corefile_thread_callback, &thread_args);
|
|
note_data = thread_args.note_data;
|
|
if (!note_data)
|
|
return NULL;
|
|
|
|
/* Auxillary vector. */
|
|
auxv_len = target_read_alloc (¤t_target, TARGET_OBJECT_AUXV,
|
|
NULL, &auxv);
|
|
if (auxv_len > 0)
|
|
{
|
|
note_data = elfcore_write_note (obfd, note_data, note_size,
|
|
"CORE", NT_AUXV, auxv, auxv_len);
|
|
xfree (auxv);
|
|
|
|
if (!note_data)
|
|
return NULL;
|
|
}
|
|
|
|
/* SPU information. */
|
|
note_data = linux_spu_make_corefile_notes (obfd, note_data, note_size);
|
|
if (!note_data)
|
|
return NULL;
|
|
|
|
make_cleanup (xfree, note_data);
|
|
return note_data;
|
|
}
|
|
|
|
static char *
|
|
linux_make_corefile_notes_1 (struct gdbarch *gdbarch, bfd *obfd, int *note_size)
|
|
{
|
|
/* FIXME: uweigand/2011-10-06: Once all GNU/Linux architectures have been
|
|
converted to gdbarch_core_regset_sections, we no longer need to fall back
|
|
to the target method at this point. */
|
|
|
|
if (!gdbarch_core_regset_sections (gdbarch))
|
|
return target_make_corefile_notes (obfd, note_size);
|
|
else
|
|
return linux_make_corefile_notes (gdbarch, obfd, note_size,
|
|
linux_collect_thread_registers);
|
|
}
|
|
|
|
/* To be called from the various GDB_OSABI_LINUX handlers for the
|
|
various GNU/Linux architectures and machine types. */
|
|
|
|
void
|
|
linux_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
|
|
{
|
|
set_gdbarch_core_pid_to_str (gdbarch, linux_core_pid_to_str);
|
|
set_gdbarch_info_proc (gdbarch, linux_info_proc);
|
|
set_gdbarch_find_memory_regions (gdbarch, linux_find_memory_regions);
|
|
set_gdbarch_make_corefile_notes (gdbarch, linux_make_corefile_notes_1);
|
|
set_gdbarch_has_shared_address_space (gdbarch,
|
|
linux_has_shared_address_space);
|
|
}
|
|
|
|
/* Provide a prototype to silence -Wmissing-prototypes. */
|
|
extern initialize_file_ftype _initialize_linux_tdep;
|
|
|
|
void
|
|
_initialize_linux_tdep (void)
|
|
{
|
|
linux_gdbarch_data_handle =
|
|
gdbarch_data_register_post_init (init_linux_gdbarch_data);
|
|
}
|