aa7c744796
* common/linux-ptrace.c: Include gdb_assert.h. <__i386__> (linux_ptrace_test_ret_to_nx_instr): New declaration. <__i386__>: Include sys/reg.h, sys/mman.h, signal.h, sys/wait.h and stdint.h. (linux_ptrace_test_ret_to_nx, linux_ptrace_init_warnings): New functions. * common/linux-ptrace.h (linux_ptrace_init_warnings): New declarations. * linux-nat.c (linux_child_post_attach) (linux_child_post_startup_inferior): Call linux_ptrace_init_warnings. gdb/gdbserver/ * gdbserver/linux-low.c (initialize_low): Call linux_ptrace_init_warnings.
173 lines
4.6 KiB
C
173 lines
4.6 KiB
C
/* Linux-specific ptrace manipulation routines.
|
|
Copyright (C) 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/>. */
|
|
|
|
#ifdef GDBSERVER
|
|
#include "server.h"
|
|
#else
|
|
#include "defs.h"
|
|
#include "gdb_string.h"
|
|
#endif
|
|
|
|
#include "linux-ptrace.h"
|
|
#include "linux-procfs.h"
|
|
#include "buffer.h"
|
|
#include "gdb_assert.h"
|
|
|
|
/* Find all possible reasons we could fail to attach PID and append these
|
|
newline terminated reason strings to initialized BUFFER. '\0' termination
|
|
of BUFFER must be done by the caller. */
|
|
|
|
void
|
|
linux_ptrace_attach_warnings (pid_t pid, struct buffer *buffer)
|
|
{
|
|
pid_t tracerpid;
|
|
|
|
tracerpid = linux_proc_get_tracerpid (pid);
|
|
if (tracerpid > 0)
|
|
buffer_xml_printf (buffer, _("warning: process %d is already traced "
|
|
"by process %d\n"),
|
|
(int) pid, (int) tracerpid);
|
|
|
|
if (linux_proc_pid_is_zombie (pid))
|
|
buffer_xml_printf (buffer, _("warning: process %d is a zombie "
|
|
"- the process has already terminated\n"),
|
|
(int) pid);
|
|
}
|
|
|
|
#ifdef __i386__
|
|
|
|
/* Address of the 'ret' instruction in asm code block below. */
|
|
extern void (linux_ptrace_test_ret_to_nx_instr) (void);
|
|
|
|
#include <sys/reg.h>
|
|
#include <sys/mman.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <stdint.h>
|
|
|
|
#endif /* __i386__ */
|
|
|
|
/* Test broken off-trunk Linux kernel patchset for NX support on i386. It was
|
|
removed in Fedora kernel 88fa1f0332d188795ed73d7ac2b1564e11a0b4cd. */
|
|
|
|
static void
|
|
linux_ptrace_test_ret_to_nx (void)
|
|
{
|
|
#ifdef __i386__
|
|
pid_t child, got_pid;
|
|
gdb_byte *return_address, *pc;
|
|
long l;
|
|
int status;
|
|
|
|
return_address = mmap (NULL, 2, PROT_READ | PROT_WRITE,
|
|
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
|
|
if (return_address == MAP_FAILED)
|
|
{
|
|
warning (_("linux_ptrace_test_ret_to_nx: Cannot mmap: %s"),
|
|
strerror (errno));
|
|
return;
|
|
}
|
|
|
|
/* Put there 'int3'. */
|
|
*return_address = 0xcc;
|
|
|
|
child = fork ();
|
|
switch (child)
|
|
{
|
|
case -1:
|
|
warning (_("linux_ptrace_test_ret_to_nx: Cannot fork: %s"),
|
|
strerror (errno));
|
|
return;
|
|
|
|
case 0:
|
|
l = ptrace (PTRACE_TRACEME, 0, NULL, NULL);
|
|
if (l != 0)
|
|
warning (_("linux_ptrace_test_ret_to_nx: Cannot PTRACE_TRACEME: %s"),
|
|
strerror (errno));
|
|
else
|
|
{
|
|
asm volatile ("pushl %0;"
|
|
".globl linux_ptrace_test_ret_to_nx_instr;"
|
|
"linux_ptrace_test_ret_to_nx_instr:"
|
|
"ret"
|
|
: : "r" (return_address) : "%esp", "memory");
|
|
gdb_assert_not_reached ("asm block did not terminate");
|
|
}
|
|
|
|
_exit (1);
|
|
}
|
|
|
|
got_pid = waitpid (child, &status, 0);
|
|
gdb_assert (got_pid == child);
|
|
gdb_assert (WIFSTOPPED (status));
|
|
|
|
/* We may get SIGSEGV due to missing PROT_EXEC of the return_address. */
|
|
gdb_assert (WSTOPSIG (status) == SIGTRAP || WSTOPSIG (status) == SIGSEGV);
|
|
|
|
errno = 0;
|
|
l = ptrace (PTRACE_PEEKUSER, child, (void *) (uintptr_t) (EIP * 4), NULL);
|
|
gdb_assert (errno == 0);
|
|
pc = (void *) (uintptr_t) l;
|
|
|
|
if (ptrace (PTRACE_KILL, child, NULL, NULL) != 0)
|
|
warning (_("linux_ptrace_test_ret_to_nx: Cannot PTRACE_KILL: %s"),
|
|
strerror (errno));
|
|
else
|
|
{
|
|
int kill_status;
|
|
|
|
got_pid = waitpid (child, &kill_status, 0);
|
|
gdb_assert (got_pid == child);
|
|
gdb_assert (WIFSIGNALED (kill_status));
|
|
}
|
|
|
|
/* + 1 is there as x86* stops after the 'int3' instruction. */
|
|
if (WSTOPSIG (status) == SIGTRAP && pc == return_address + 1)
|
|
{
|
|
/* PASS */
|
|
return;
|
|
}
|
|
|
|
/* We may get SIGSEGV due to missing PROT_EXEC of the RETURN_ADDRESS page. */
|
|
if (WSTOPSIG (status) == SIGSEGV && pc == return_address)
|
|
{
|
|
/* PASS */
|
|
return;
|
|
}
|
|
|
|
gdb_assert ((void (*) (void)) pc == &linux_ptrace_test_ret_to_nx_instr);
|
|
|
|
warning (_("Cannot call inferior functions, you have broken "
|
|
"Linux kernel i386 NX (non-executable pages) support!"));
|
|
#endif /* __i386__ */
|
|
}
|
|
|
|
/* Display possible problems on this system. Display them only once per GDB
|
|
execution. */
|
|
|
|
void
|
|
linux_ptrace_init_warnings (void)
|
|
{
|
|
static int warned = 0;
|
|
|
|
if (warned)
|
|
return;
|
|
warned = 1;
|
|
|
|
linux_ptrace_test_ret_to_nx ();
|
|
}
|