mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-06 12:09:26 +08:00
5b9ca4d43b
If we had this in place before, then the regression fixed by the previous commit would have been been visible is all test runs. E.g.: Running src/gdb/testsuite/gdb.threads/multi-create-ns-info-thr.exp ... FAIL: gdb.threads/multi-create-ns-info-thr.exp: continue to breakpoint 6 Debugging manually we'd see this: gdbserver: Cannot get thread handle for LWP 1467: generic error Instead of: gdbserver: PID mismatch! Expected 27472, got 27471 which is misleading - gdbserver didn't 27471, that was stale stack data from previous function invocations. gdb/gdbserver/ChangeLog: 2017-09-29 Pedro Alves <palves@redhat.com> * proc-service.c (ps_pdread): Return PS_ERR if reading memory fails.
166 lines
4.3 KiB
C
166 lines
4.3 KiB
C
/* libthread_db helper functions for the remote server for GDB.
|
|
Copyright (C) 2002-2017 Free Software Foundation, Inc.
|
|
|
|
Contributed by MontaVista Software.
|
|
|
|
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 "server.h"
|
|
|
|
/* This file is currently tied to GNU/Linux. It should scale well to
|
|
another libthread_db implementation, with the approriate gdbserver
|
|
hooks, but for now this means we can use GNU/Linux's target data. */
|
|
|
|
#include "linux-low.h"
|
|
|
|
#include "gdb_proc_service.h"
|
|
|
|
typedef struct ps_prochandle *gdb_ps_prochandle_t;
|
|
typedef void *gdb_ps_read_buf_t;
|
|
typedef const void *gdb_ps_write_buf_t;
|
|
typedef size_t gdb_ps_size_t;
|
|
|
|
#ifdef HAVE_LINUX_REGSETS
|
|
#define HAVE_REGSETS
|
|
#endif
|
|
|
|
#ifdef HAVE_REGSETS
|
|
static struct regset_info *
|
|
gregset_info (void)
|
|
{
|
|
int i = 0;
|
|
const struct regs_info *regs_info = (*the_low_target.regs_info) ();
|
|
struct regsets_info *regsets_info = regs_info->regsets_info;
|
|
|
|
while (regsets_info->regsets[i].size != -1)
|
|
{
|
|
if (regsets_info->regsets[i].type == GENERAL_REGS)
|
|
break;
|
|
i++;
|
|
}
|
|
|
|
return ®sets_info->regsets[i];
|
|
}
|
|
#endif
|
|
|
|
/* Search for the symbol named NAME within the object named OBJ within
|
|
the target process PH. If the symbol is found the address of the
|
|
symbol is stored in SYM_ADDR. */
|
|
|
|
ps_err_e
|
|
ps_pglobal_lookup (gdb_ps_prochandle_t ph, const char *obj,
|
|
const char *name, psaddr_t *sym_addr)
|
|
{
|
|
CORE_ADDR addr;
|
|
|
|
if (thread_db_look_up_one_symbol (name, &addr) == 0)
|
|
return PS_NOSYM;
|
|
|
|
*sym_addr = (psaddr_t) (unsigned long) addr;
|
|
return PS_OK;
|
|
}
|
|
|
|
/* Read SIZE bytes from the target process PH at address ADDR and copy
|
|
them into BUF. */
|
|
|
|
ps_err_e
|
|
ps_pdread (gdb_ps_prochandle_t ph, psaddr_t addr,
|
|
gdb_ps_read_buf_t buf, gdb_ps_size_t size)
|
|
{
|
|
if (read_inferior_memory ((uintptr_t) addr, (gdb_byte *) buf, size) != 0)
|
|
return PS_ERR;
|
|
return PS_OK;
|
|
}
|
|
|
|
/* Write SIZE bytes from BUF into the target process PH at address ADDR. */
|
|
|
|
ps_err_e
|
|
ps_pdwrite (gdb_ps_prochandle_t ph, psaddr_t addr,
|
|
gdb_ps_write_buf_t buf, gdb_ps_size_t size)
|
|
{
|
|
if (write_inferior_memory ((uintptr_t) addr, (const gdb_byte *) buf, size)
|
|
!= 0)
|
|
return PS_ERR;
|
|
return PS_OK;
|
|
}
|
|
|
|
/* Get the general registers of LWP LWPID within the target process PH
|
|
and store them in GREGSET. */
|
|
|
|
ps_err_e
|
|
ps_lgetregs (gdb_ps_prochandle_t ph, lwpid_t lwpid, prgregset_t gregset)
|
|
{
|
|
#ifdef HAVE_REGSETS
|
|
struct lwp_info *lwp;
|
|
struct thread_info *reg_thread, *saved_thread;
|
|
struct regcache *regcache;
|
|
|
|
lwp = find_lwp_pid (pid_to_ptid (lwpid));
|
|
if (lwp == NULL)
|
|
return PS_ERR;
|
|
|
|
reg_thread = get_lwp_thread (lwp);
|
|
saved_thread = current_thread;
|
|
current_thread = reg_thread;
|
|
regcache = get_thread_regcache (current_thread, 1);
|
|
gregset_info ()->fill_function (regcache, gregset);
|
|
|
|
current_thread = saved_thread;
|
|
return PS_OK;
|
|
#else
|
|
return PS_ERR;
|
|
#endif
|
|
}
|
|
|
|
/* Set the general registers of LWP LWPID within the target process PH
|
|
from GREGSET. */
|
|
|
|
ps_err_e
|
|
ps_lsetregs (gdb_ps_prochandle_t ph, lwpid_t lwpid, const prgregset_t gregset)
|
|
{
|
|
/* Unneeded. */
|
|
return PS_ERR;
|
|
}
|
|
|
|
/* Get the floating-point registers of LWP LWPID within the target
|
|
process PH and store them in FPREGSET. */
|
|
|
|
ps_err_e
|
|
ps_lgetfpregs (gdb_ps_prochandle_t ph, lwpid_t lwpid, prfpregset_t *fpregset)
|
|
{
|
|
/* Unneeded. */
|
|
return PS_ERR;
|
|
}
|
|
|
|
/* Set the floating-point registers of LWP LWPID within the target
|
|
process PH from FPREGSET. */
|
|
|
|
ps_err_e
|
|
ps_lsetfpregs (gdb_ps_prochandle_t ph, lwpid_t lwpid, const prfpregset_t *fpregset)
|
|
{
|
|
/* Unneeded. */
|
|
return PS_ERR;
|
|
}
|
|
|
|
/* Return overall process id of the target PH. Special for GNU/Linux
|
|
-- not used on Solaris. */
|
|
|
|
pid_t
|
|
ps_getpid (gdb_ps_prochandle_t ph)
|
|
{
|
|
return pid_of (current_thread);
|
|
}
|