Blame sysdeps/mach/hurd/ptrace.c

Packit 6c4009
/* Process tracing interface `ptrace' for GNU Hurd.
Packit 6c4009
   Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <sys/ptrace.h>
Packit 6c4009
#include <sys/types.h>
Packit 6c4009
#include <stdarg.h>
Packit 6c4009
#include <hurd.h>
Packit 6c4009
#include <hurd/signal.h>
Packit 6c4009
#include <hurd/msg.h>
Packit 6c4009
#include <thread_state.h>
Packit 6c4009
Packit 6c4009
/* Perform process tracing functions.  REQUEST is one of the values
Packit 6c4009
   in <sys/ptrace.h>, and determines the action to be taken.
Packit 6c4009
   For all requests except PTRACE_TRACEME, PID specifies the process to be
Packit 6c4009
   traced.
Packit 6c4009
Packit 6c4009
   PID and the other arguments described above for the various requests should
Packit 6c4009
   appear (those that are used for the particular request) as:
Packit 6c4009
     pid_t PID, void *ADDR, int DATA, void *ADDR2
Packit 6c4009
   after PID.  */
Packit 6c4009
int
Packit 6c4009
ptrace (enum __ptrace_request request, ... )
Packit 6c4009
{
Packit 6c4009
  pid_t pid;
Packit 6c4009
  void *addr, *addr2;
Packit 6c4009
  natural_t data;
Packit 6c4009
  va_list ap;
Packit 6c4009
Packit 6c4009
  /* Read data from PID's address space, from ADDR for DATA bytes.  */
Packit 6c4009
  error_t read_data (task_t task, vm_address_t *ourpage, vm_size_t *size)
Packit 6c4009
    {
Packit 6c4009
      /* Read the pages containing the addressed range.  */
Packit 6c4009
      error_t err;
Packit 6c4009
      *size = round_page (addr + data) - trunc_page (addr);
Packit 6c4009
      err = __vm_read (task, trunc_page (addr), *size, ourpage, size);
Packit 6c4009
      return err;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Fetch the thread port for PID's user thread.  */
Packit 6c4009
  error_t fetch_user_thread (task_t task, thread_t *thread)
Packit 6c4009
    {
Packit 6c4009
      thread_t threadbuf[3], *threads = threadbuf;
Packit 6c4009
      mach_msg_type_number_t nthreads = 3, i;
Packit 6c4009
      error_t err = __task_threads (task, &threads, &nthreads);
Packit 6c4009
      if (err)
Packit 6c4009
	return err;
Packit 6c4009
      if (nthreads == 0)
Packit 6c4009
	return EINVAL;
Packit 6c4009
      *thread = threads[0];	/* Assume user thread is first.  */
Packit 6c4009
      for (i = 1; i < nthreads; ++i)
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), threads[i]);
Packit 6c4009
      if (threads != threadbuf)
Packit 6c4009
	__vm_deallocate (__mach_task_self (),
Packit 6c4009
			 (vm_address_t) threads, nthreads * sizeof threads[0]);
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Fetch a thread state structure from PID and store it at ADDR.  */
Packit 6c4009
  int get_regs (int flavor, mach_msg_type_number_t count)
Packit 6c4009
    {
Packit 6c4009
      error_t err;
Packit 6c4009
      task_t task = __pid2task (pid);
Packit 6c4009
      thread_t thread;
Packit 6c4009
      if (task == MACH_PORT_NULL)
Packit 6c4009
	return -1;
Packit 6c4009
      err = fetch_user_thread (task, &thread);
Packit 6c4009
      __mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
      if (!err)
Packit 6c4009
	err = __thread_get_state (thread, flavor, addr, &count);
Packit 6c4009
      __mach_port_deallocate (__mach_task_self (), thread);
Packit 6c4009
      return err ? __hurd_fail (err) : 0;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
Packit 6c4009
  switch (request)
Packit 6c4009
    {
Packit 6c4009
    case PTRACE_TRACEME:
Packit 6c4009
      /* Make this process be traced.  */
Packit 6c4009
      __sigfillset (&_hurdsig_traced);
Packit 6c4009
      __USEPORT (PROC, __proc_mark_traced (port));
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case PTRACE_CONT:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      data = va_arg (ap, int);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	/* Send a DATA signal to PID, telling it to take the signal
Packit 6c4009
	   normally even if it's traced.  */
Packit 6c4009
	error_t err;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	if (data == SIGKILL)
Packit 6c4009
	  err = __task_terminate (task);
Packit 6c4009
	else
Packit 6c4009
	  {
Packit 6c4009
	    if (addr != (void *) 1)
Packit 6c4009
	      {
Packit 6c4009
		/* Move the user thread's PC to ADDR.  */
Packit 6c4009
		thread_t thread;
Packit 6c4009
		err = fetch_user_thread (task, &thread);
Packit 6c4009
		if (!err)
Packit 6c4009
		  {
Packit 6c4009
		    struct machine_thread_state state;
Packit 6c4009
		    mach_msg_type_number_t count = MACHINE_THREAD_STATE_COUNT;
Packit 6c4009
		    err = __thread_get_state (thread,
Packit 6c4009
					      MACHINE_THREAD_STATE_FLAVOR,
Packit 6c4009
					      (natural_t *) &state, &count);
Packit 6c4009
		    if (!err)
Packit 6c4009
		      {
Packit 6c4009
			MACHINE_THREAD_STATE_SET_PC (&state, addr);
Packit 6c4009
			err = __thread_set_state (thread,
Packit 6c4009
						  MACHINE_THREAD_STATE_FLAVOR,
Packit 6c4009
						  (natural_t *) &state, count);
Packit 6c4009
		      }
Packit 6c4009
Packit 6c4009
		  }
Packit 6c4009
		__mach_port_deallocate (__mach_task_self (), thread);
Packit 6c4009
	      }
Packit 6c4009
	    else
Packit 6c4009
	      err = 0;
Packit 6c4009
Packit 6c4009
	    if (! err)
Packit 6c4009
	      /* Tell the process to take the signal (or just resume if 0).  */
Packit 6c4009
	      err = HURD_MSGPORT_RPC
Packit 6c4009
		(__USEPORT (PROC, __proc_getmsgport (port, pid, &msgport)),
Packit 6c4009
		 0, 0, __msg_sig_post_untraced (msgport, data, 0, task));
Packit 6c4009
	  }
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	return err ? __hurd_fail (err) : 0;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    case PTRACE_KILL:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      /* SIGKILL always just terminates the task,
Packit 6c4009
	 so normal kill is just the same when traced.  */
Packit 6c4009
      return __kill (pid, SIGKILL);
Packit 6c4009
Packit 6c4009
    case PTRACE_SINGLESTEP:
Packit 6c4009
      /* This is a machine-dependent kernel RPC on
Packit 6c4009
	 machines that support it.  Punt.  */
Packit 6c4009
      return __hurd_fail (EOPNOTSUPP);
Packit 6c4009
Packit 6c4009
    case PTRACE_ATTACH:
Packit 6c4009
    case PTRACE_DETACH:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	/* Tell PID to set or clear its trace bit.  */
Packit 6c4009
	error_t err;
Packit 6c4009
	mach_port_t msgport;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	err = __USEPORT (PROC, __proc_getmsgport (port, pid, &msgport));
Packit 6c4009
	if (! err)
Packit 6c4009
	  {
Packit 6c4009
	    err = __msg_set_init_int (msgport, task, INIT_TRACEMASK,
Packit 6c4009
				      request == PTRACE_DETACH ? 0 :
Packit 6c4009
				      ~(sigset_t) 0);
Packit 6c4009
	    if (! err)
Packit 6c4009
	      {
Packit 6c4009
		if (request == PTRACE_ATTACH)
Packit 6c4009
		  /* Now stop the process.  */
Packit 6c4009
		  err = __msg_sig_post (msgport, SIGSTOP, 0, task);
Packit 6c4009
		else
Packit 6c4009
		  /* Resume the process from tracing stop.  */
Packit 6c4009
		  err = __msg_sig_post_untraced (msgport, 0, 0, task);
Packit 6c4009
	      }
Packit 6c4009
	    __mach_port_deallocate (__mach_task_self (), msgport);
Packit 6c4009
	  }
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	return err ? __hurd_fail (err) : 0;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    case PTRACE_PEEKTEXT:
Packit 6c4009
    case PTRACE_PEEKDATA:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	/* Read the page (or two pages, if the word lies on a boundary)
Packit 6c4009
	   containing the addressed word.  */
Packit 6c4009
	error_t err;
Packit 6c4009
	vm_address_t ourpage;
Packit 6c4009
	vm_size_t size;
Packit 6c4009
	natural_t word;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	data = sizeof word;
Packit 6c4009
	ourpage = 0;
Packit 6c4009
	size = 0;
Packit 6c4009
	err = read_data (task, &ourpage, &size);
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	if (err)
Packit 6c4009
	  return __hurd_fail (err);
Packit 6c4009
	word = *(natural_t *) ((vm_address_t) addr - trunc_page (addr)
Packit 6c4009
			       + ourpage);
Packit 6c4009
	__vm_deallocate (__mach_task_self (), ourpage, size);
Packit 6c4009
	return word;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    case PTRACE_PEEKUSER:
Packit 6c4009
    case PTRACE_POKEUSER:
Packit 6c4009
      /* U area, what's that?  */
Packit 6c4009
      return __hurd_fail (EOPNOTSUPP);
Packit 6c4009
Packit 6c4009
    case PTRACE_GETREGS:
Packit 6c4009
    case PTRACE_SETREGS:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      return get_regs (MACHINE_THREAD_STATE_FLAVOR,
Packit 6c4009
		       MACHINE_THREAD_STATE_COUNT);
Packit 6c4009
Packit 6c4009
    case PTRACE_GETFPREGS:
Packit 6c4009
    case PTRACE_SETFPREGS:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
#ifdef MACHINE_THREAD_FLOAT_STATE_FLAVOR
Packit 6c4009
      return get_regs (MACHINE_THREAD_FLOAT_STATE_FLAVOR,
Packit 6c4009
		       MACHINE_THREAD_FLOAT_STATE_COUNT);
Packit 6c4009
#else
Packit 6c4009
      return __hurd_fail (EOPNOTSUPP);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
    case PTRACE_GETFPAREGS:
Packit 6c4009
    case PTRACE_SETFPAREGS:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
#ifdef MACHINE_THREAD_FPA_STATE_FLAVOR
Packit 6c4009
      return get_regs (MACHINE_THREAD_FPA_STATE_FLAVOR,
Packit 6c4009
		       MACHINE_THREAD_FPA_STATE_COUNT);
Packit 6c4009
#else
Packit 6c4009
      return __hurd_fail (EOPNOTSUPP);
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
    case PTRACE_POKETEXT:
Packit 6c4009
    case PTRACE_POKEDATA:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      data = va_arg (ap, int);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	/* Read the page (or two pages, if the word lies on a boundary)
Packit 6c4009
	   containing the addressed word.  */
Packit 6c4009
	error_t err;
Packit 6c4009
	vm_address_t ourpage;
Packit 6c4009
	vm_size_t size;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	data = sizeof (natural_t);
Packit 6c4009
	ourpage = 0;
Packit 6c4009
	size = 0;
Packit 6c4009
	err = read_data (task, &ourpage, &size);
Packit 6c4009
Packit 6c4009
	if (!err)
Packit 6c4009
	  {
Packit 6c4009
	    /* Now modify the specified word and write the page back.  */
Packit 6c4009
	    *(natural_t *) ((vm_address_t) addr - trunc_page (addr)
Packit 6c4009
			    + ourpage) = data;
Packit 6c4009
	    err = __vm_write (task, trunc_page (addr), ourpage, size);
Packit 6c4009
	    __vm_deallocate (__mach_task_self (), ourpage, size);
Packit 6c4009
	  }
Packit 6c4009
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	return err ? __hurd_fail (err) : 0;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    case PTRACE_READDATA:
Packit 6c4009
    case PTRACE_READTEXT:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      data = va_arg (ap, int);
Packit 6c4009
      addr2 = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	error_t err;
Packit 6c4009
	vm_address_t ourpage;
Packit 6c4009
	vm_size_t size;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	if (((vm_address_t) addr2 + data) % __vm_page_size == 0)
Packit 6c4009
	  {
Packit 6c4009
	    /* Perhaps we can write directly to the user's buffer.  */
Packit 6c4009
	    ourpage = (vm_address_t) addr2;
Packit 6c4009
	    size = data;
Packit 6c4009
	  }
Packit 6c4009
	else
Packit 6c4009
	  {
Packit 6c4009
	    ourpage = 0;
Packit 6c4009
	    size = 0;
Packit 6c4009
	  }
Packit 6c4009
	err = read_data (task, &ourpage, &size);
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	if (!err && ourpage != (vm_address_t) addr2)
Packit 6c4009
	  {
Packit 6c4009
	    memcpy (addr2, (void *) ourpage, data);
Packit 6c4009
	    __vm_deallocate (__mach_task_self (), ourpage, size);
Packit 6c4009
	  }
Packit 6c4009
	return err ? __hurd_fail (err) : 0;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    case PTRACE_WRITEDATA:
Packit 6c4009
    case PTRACE_WRITETEXT:
Packit 6c4009
      va_start (ap, request);
Packit 6c4009
      pid = va_arg (ap, pid_t);
Packit 6c4009
      addr = va_arg (ap, void *);
Packit 6c4009
      data = va_arg (ap, int);
Packit 6c4009
      addr2 = va_arg (ap, void *);
Packit 6c4009
      va_end (ap);
Packit 6c4009
      {
Packit 6c4009
	error_t err;
Packit 6c4009
	vm_address_t ourpage;
Packit 6c4009
	vm_size_t size;
Packit 6c4009
	task_t task = __pid2task (pid);
Packit 6c4009
	if (task == MACH_PORT_NULL)
Packit 6c4009
	  return -1;
Packit 6c4009
	if ((vm_address_t) addr % __vm_page_size == 0 &&
Packit 6c4009
	    (vm_address_t) data % __vm_page_size == 0)
Packit 6c4009
	  {
Packit 6c4009
	    /* Writing whole pages; can go directly from the user's buffer.  */
Packit 6c4009
	    ourpage = (vm_address_t) addr2;
Packit 6c4009
	    size = data;
Packit 6c4009
	    err = 0;
Packit 6c4009
	  }
Packit 6c4009
	else
Packit 6c4009
	  {
Packit 6c4009
	    /* Read the task's pages and modify our own copy.  */
Packit 6c4009
	    ourpage = 0;
Packit 6c4009
	    size = 0;
Packit 6c4009
	    err = read_data (task, &ourpage, &size);
Packit 6c4009
	    if (!err)
Packit 6c4009
	      memcpy ((void *) ((vm_address_t) addr - trunc_page (addr)
Packit 6c4009
				+ ourpage),
Packit 6c4009
		      addr2,
Packit 6c4009
		      data);
Packit 6c4009
	  }
Packit 6c4009
	if (!err)
Packit 6c4009
	  /* Write back the modified pages.  */
Packit 6c4009
	  err = __vm_write (task, trunc_page (addr), ourpage, size);
Packit 6c4009
	__mach_port_deallocate (__mach_task_self (), task);
Packit 6c4009
	return err ? __hurd_fail (err) : 0;
Packit 6c4009
      }
Packit 6c4009
Packit 6c4009
    default:
Packit 6c4009
      errno = EINVAL;
Packit 6c4009
      return -1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}