xref: /dflybsd-src/contrib/gdb-7/gdb/bsd-kvm.c (revision c50c785cb49e9377ca78104c5540c7b33f768771)
15796c8dcSSimon Schubert /* BSD Kernel Data Access Library (libkvm) interface.
25796c8dcSSimon Schubert 
3*c50c785cSJohn Marino    Copyright (C) 2004, 2005, 2007, 2008, 2009, 2010, 2011
4cf7f2e2dSJohn Marino    Free Software Foundation, Inc.
55796c8dcSSimon Schubert 
65796c8dcSSimon Schubert    This file is part of GDB.
75796c8dcSSimon Schubert 
85796c8dcSSimon Schubert    This program is free software; you can redistribute it and/or modify
95796c8dcSSimon Schubert    it under the terms of the GNU General Public License as published by
105796c8dcSSimon Schubert    the Free Software Foundation; either version 3 of the License, or
115796c8dcSSimon Schubert    (at your option) any later version.
125796c8dcSSimon Schubert 
135796c8dcSSimon Schubert    This program is distributed in the hope that it will be useful,
145796c8dcSSimon Schubert    but WITHOUT ANY WARRANTY; without even the implied warranty of
155796c8dcSSimon Schubert    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
165796c8dcSSimon Schubert    GNU General Public License for more details.
175796c8dcSSimon Schubert 
185796c8dcSSimon Schubert    You should have received a copy of the GNU General Public License
195796c8dcSSimon Schubert    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
205796c8dcSSimon Schubert 
215796c8dcSSimon Schubert #include "defs.h"
225796c8dcSSimon Schubert #include "cli/cli-cmds.h"
235796c8dcSSimon Schubert #include "command.h"
245796c8dcSSimon Schubert #include "frame.h"
255796c8dcSSimon Schubert #include "regcache.h"
265796c8dcSSimon Schubert #include "target.h"
275796c8dcSSimon Schubert #include "value.h"
285796c8dcSSimon Schubert #include "gdbcore.h"		/* for get_exec_file */
295796c8dcSSimon Schubert #include "gdbthread.h"
305796c8dcSSimon Schubert 
315796c8dcSSimon Schubert #include "gdb_assert.h"
325796c8dcSSimon Schubert #include <fcntl.h>
335796c8dcSSimon Schubert #include <kvm.h>
345796c8dcSSimon Schubert #ifdef HAVE_NLIST_H
355796c8dcSSimon Schubert #include <nlist.h>
365796c8dcSSimon Schubert #endif
375796c8dcSSimon Schubert #include <paths.h>
385796c8dcSSimon Schubert #include "readline/readline.h"
395796c8dcSSimon Schubert #include <sys/param.h>
405796c8dcSSimon Schubert #include <sys/proc.h>
415796c8dcSSimon Schubert #include <sys/user.h>
425796c8dcSSimon Schubert 
435796c8dcSSimon Schubert #include "bsd-kvm.h"
445796c8dcSSimon Schubert 
455796c8dcSSimon Schubert /* Kernel memory device file.  */
465796c8dcSSimon Schubert static const char *bsd_kvm_corefile;
475796c8dcSSimon Schubert 
485796c8dcSSimon Schubert /* Kernel memory interface descriptor.  */
495796c8dcSSimon Schubert static kvm_t *core_kd;
505796c8dcSSimon Schubert 
515796c8dcSSimon Schubert /* Address of process control block.  */
525796c8dcSSimon Schubert static struct pcb *bsd_kvm_paddr;
535796c8dcSSimon Schubert 
545796c8dcSSimon Schubert /* Pointer to architecture-specific function that reconstructs the
555796c8dcSSimon Schubert    register state from PCB and supplies it to REGCACHE.  */
565796c8dcSSimon Schubert static int (*bsd_kvm_supply_pcb)(struct regcache *regcache, struct pcb *pcb);
575796c8dcSSimon Schubert 
585796c8dcSSimon Schubert /* Target ops for libkvm interface.  */
595796c8dcSSimon Schubert static struct target_ops bsd_kvm_ops;
605796c8dcSSimon Schubert 
615796c8dcSSimon Schubert /* This is the ptid we use while we're connected to kvm.  The kvm
625796c8dcSSimon Schubert    target currently doesn't export any view of the running processes,
635796c8dcSSimon Schubert    so this represents the kernel task.  */
645796c8dcSSimon Schubert static ptid_t bsd_kvm_ptid;
655796c8dcSSimon Schubert 
665796c8dcSSimon Schubert static void
675796c8dcSSimon Schubert bsd_kvm_open (char *filename, int from_tty)
685796c8dcSSimon Schubert {
695796c8dcSSimon Schubert   char errbuf[_POSIX2_LINE_MAX];
705796c8dcSSimon Schubert   char *execfile = NULL;
715796c8dcSSimon Schubert   kvm_t *temp_kd;
725796c8dcSSimon Schubert 
735796c8dcSSimon Schubert   target_preopen (from_tty);
745796c8dcSSimon Schubert 
755796c8dcSSimon Schubert   if (filename)
765796c8dcSSimon Schubert     {
775796c8dcSSimon Schubert       char *temp;
785796c8dcSSimon Schubert 
795796c8dcSSimon Schubert       filename = tilde_expand (filename);
805796c8dcSSimon Schubert       if (filename[0] != '/')
815796c8dcSSimon Schubert 	{
825796c8dcSSimon Schubert 	  temp = concat (current_directory, "/", filename, (char *)NULL);
835796c8dcSSimon Schubert 	  xfree (filename);
845796c8dcSSimon Schubert 	  filename = temp;
855796c8dcSSimon Schubert 	}
865796c8dcSSimon Schubert     }
875796c8dcSSimon Schubert 
885796c8dcSSimon Schubert   execfile = get_exec_file (0);
895796c8dcSSimon Schubert   temp_kd = kvm_openfiles (execfile, filename, NULL,
905796c8dcSSimon Schubert 			   write_files ? O_RDWR : O_RDONLY, errbuf);
915796c8dcSSimon Schubert   if (temp_kd == NULL)
925796c8dcSSimon Schubert     error (("%s"), errbuf);
935796c8dcSSimon Schubert 
945796c8dcSSimon Schubert   bsd_kvm_corefile = filename;
955796c8dcSSimon Schubert   unpush_target (&bsd_kvm_ops);
965796c8dcSSimon Schubert   core_kd = temp_kd;
975796c8dcSSimon Schubert   push_target (&bsd_kvm_ops);
985796c8dcSSimon Schubert 
995796c8dcSSimon Schubert   add_thread_silent (bsd_kvm_ptid);
1005796c8dcSSimon Schubert   inferior_ptid = bsd_kvm_ptid;
1015796c8dcSSimon Schubert 
1025796c8dcSSimon Schubert   target_fetch_registers (get_current_regcache (), -1);
1035796c8dcSSimon Schubert 
1045796c8dcSSimon Schubert   reinit_frame_cache ();
1055796c8dcSSimon Schubert   print_stack_frame (get_selected_frame (NULL), -1, 1);
1065796c8dcSSimon Schubert }
1075796c8dcSSimon Schubert 
1085796c8dcSSimon Schubert static void
1095796c8dcSSimon Schubert bsd_kvm_close (int quitting)
1105796c8dcSSimon Schubert {
1115796c8dcSSimon Schubert   if (core_kd)
1125796c8dcSSimon Schubert     {
1135796c8dcSSimon Schubert       if (kvm_close (core_kd) == -1)
1145796c8dcSSimon Schubert 	warning (("%s"), kvm_geterr(core_kd));
1155796c8dcSSimon Schubert       core_kd = NULL;
1165796c8dcSSimon Schubert     }
1175796c8dcSSimon Schubert 
1185796c8dcSSimon Schubert   inferior_ptid = null_ptid;
1195796c8dcSSimon Schubert   delete_thread_silent (bsd_kvm_ptid);
1205796c8dcSSimon Schubert }
1215796c8dcSSimon Schubert 
1225796c8dcSSimon Schubert static LONGEST
1235796c8dcSSimon Schubert bsd_kvm_xfer_memory (CORE_ADDR addr, ULONGEST len,
1245796c8dcSSimon Schubert 		     gdb_byte *readbuf, const gdb_byte *writebuf)
1255796c8dcSSimon Schubert {
1265796c8dcSSimon Schubert   ssize_t nbytes = len;
1275796c8dcSSimon Schubert 
1285796c8dcSSimon Schubert   if (readbuf)
1295796c8dcSSimon Schubert     nbytes = kvm_read (core_kd, addr, readbuf, nbytes);
1305796c8dcSSimon Schubert   if (writebuf && nbytes > 0)
1315796c8dcSSimon Schubert     nbytes = kvm_write (core_kd, addr, writebuf, nbytes);
1325796c8dcSSimon Schubert   return nbytes;
1335796c8dcSSimon Schubert }
1345796c8dcSSimon Schubert 
1355796c8dcSSimon Schubert static LONGEST
1365796c8dcSSimon Schubert bsd_kvm_xfer_partial (struct target_ops *ops, enum target_object object,
1375796c8dcSSimon Schubert 		      const char *annex, gdb_byte *readbuf,
1385796c8dcSSimon Schubert 		      const gdb_byte *writebuf,
1395796c8dcSSimon Schubert 		      ULONGEST offset, LONGEST len)
1405796c8dcSSimon Schubert {
1415796c8dcSSimon Schubert   switch (object)
1425796c8dcSSimon Schubert     {
1435796c8dcSSimon Schubert     case TARGET_OBJECT_MEMORY:
1445796c8dcSSimon Schubert       return bsd_kvm_xfer_memory (offset, len, readbuf, writebuf);
1455796c8dcSSimon Schubert 
1465796c8dcSSimon Schubert     default:
1475796c8dcSSimon Schubert       return -1;
1485796c8dcSSimon Schubert     }
1495796c8dcSSimon Schubert }
1505796c8dcSSimon Schubert 
1515796c8dcSSimon Schubert static void
1525796c8dcSSimon Schubert bsd_kvm_files_info (struct target_ops *ops)
1535796c8dcSSimon Schubert {
1545796c8dcSSimon Schubert   if (bsd_kvm_corefile && strcmp (bsd_kvm_corefile, _PATH_MEM) != 0)
1555796c8dcSSimon Schubert     printf_filtered (_("\tUsing the kernel crash dump %s.\n"),
1565796c8dcSSimon Schubert 		     bsd_kvm_corefile);
1575796c8dcSSimon Schubert   else
1585796c8dcSSimon Schubert     printf_filtered (_("\tUsing the currently running kernel.\n"));
1595796c8dcSSimon Schubert }
1605796c8dcSSimon Schubert 
1615796c8dcSSimon Schubert /* Fetch process control block at address PADDR.  */
1625796c8dcSSimon Schubert 
1635796c8dcSSimon Schubert static int
1645796c8dcSSimon Schubert bsd_kvm_fetch_pcb (struct regcache *regcache, struct pcb *paddr)
1655796c8dcSSimon Schubert {
1665796c8dcSSimon Schubert   struct pcb pcb;
1675796c8dcSSimon Schubert 
1685796c8dcSSimon Schubert   if (kvm_read (core_kd, (unsigned long) paddr, &pcb, sizeof pcb) == -1)
1695796c8dcSSimon Schubert     error (("%s"), kvm_geterr (core_kd));
1705796c8dcSSimon Schubert 
1715796c8dcSSimon Schubert   gdb_assert (bsd_kvm_supply_pcb);
1725796c8dcSSimon Schubert   return bsd_kvm_supply_pcb (regcache, &pcb);
1735796c8dcSSimon Schubert }
1745796c8dcSSimon Schubert 
1755796c8dcSSimon Schubert static void
1765796c8dcSSimon Schubert bsd_kvm_fetch_registers (struct target_ops *ops,
1775796c8dcSSimon Schubert 			 struct regcache *regcache, int regnum)
1785796c8dcSSimon Schubert {
1795796c8dcSSimon Schubert   struct nlist nl[2];
1805796c8dcSSimon Schubert 
1815796c8dcSSimon Schubert   if (bsd_kvm_paddr)
1825796c8dcSSimon Schubert     {
1835796c8dcSSimon Schubert       bsd_kvm_fetch_pcb (regcache, bsd_kvm_paddr);
1845796c8dcSSimon Schubert       return;
1855796c8dcSSimon Schubert     }
1865796c8dcSSimon Schubert 
1875796c8dcSSimon Schubert   /* On dumping core, BSD kernels store the faulting context (PCB)
1885796c8dcSSimon Schubert      in the variable "dumppcb".  */
1895796c8dcSSimon Schubert   memset (nl, 0, sizeof nl);
1905796c8dcSSimon Schubert   nl[0].n_name = "_dumppcb";
1915796c8dcSSimon Schubert 
1925796c8dcSSimon Schubert   if (kvm_nlist (core_kd, nl) == -1)
1935796c8dcSSimon Schubert     error (("%s"), kvm_geterr (core_kd));
1945796c8dcSSimon Schubert 
1955796c8dcSSimon Schubert   if (nl[0].n_value != 0)
1965796c8dcSSimon Schubert     {
1975796c8dcSSimon Schubert       /* Found dumppcb.  If it contains a valid context, return
1985796c8dcSSimon Schubert 	 immediately.  */
1995796c8dcSSimon Schubert       if (bsd_kvm_fetch_pcb (regcache, (struct pcb *) nl[0].n_value))
2005796c8dcSSimon Schubert 	return;
2015796c8dcSSimon Schubert     }
2025796c8dcSSimon Schubert 
2035796c8dcSSimon Schubert   /* Traditional BSD kernels have a process proc0 that should always
2045796c8dcSSimon Schubert      be present.  The address of proc0's PCB is stored in the variable
2055796c8dcSSimon Schubert      "proc0paddr".  */
2065796c8dcSSimon Schubert 
2075796c8dcSSimon Schubert   memset (nl, 0, sizeof nl);
2085796c8dcSSimon Schubert   nl[0].n_name = "_proc0paddr";
2095796c8dcSSimon Schubert 
2105796c8dcSSimon Schubert   if (kvm_nlist (core_kd, nl) == -1)
2115796c8dcSSimon Schubert     error (("%s"), kvm_geterr (core_kd));
2125796c8dcSSimon Schubert 
2135796c8dcSSimon Schubert   if (nl[0].n_value != 0)
2145796c8dcSSimon Schubert     {
2155796c8dcSSimon Schubert       struct pcb *paddr;
2165796c8dcSSimon Schubert 
2175796c8dcSSimon Schubert       /* Found proc0paddr.  */
2185796c8dcSSimon Schubert       if (kvm_read (core_kd, nl[0].n_value, &paddr, sizeof paddr) == -1)
2195796c8dcSSimon Schubert 	error (("%s"), kvm_geterr (core_kd));
2205796c8dcSSimon Schubert 
2215796c8dcSSimon Schubert       bsd_kvm_fetch_pcb (regcache, paddr);
2225796c8dcSSimon Schubert       return;
2235796c8dcSSimon Schubert     }
2245796c8dcSSimon Schubert 
2255796c8dcSSimon Schubert #ifdef HAVE_STRUCT_THREAD_TD_PCB
2265796c8dcSSimon Schubert   /* In FreeBSD kernels for 5.0-RELEASE and later, the PCB no longer
2275796c8dcSSimon Schubert      lives in `struct proc' but in `struct thread'.  The `struct
2285796c8dcSSimon Schubert      thread' for the initial thread for proc0 can be found in the
2295796c8dcSSimon Schubert      variable "thread0".  */
2305796c8dcSSimon Schubert 
2315796c8dcSSimon Schubert   memset (nl, 0, sizeof nl);
2325796c8dcSSimon Schubert   nl[0].n_name = "_thread0";
2335796c8dcSSimon Schubert 
2345796c8dcSSimon Schubert   if (kvm_nlist (core_kd, nl) == -1)
2355796c8dcSSimon Schubert     error (("%s"), kvm_geterr (core_kd));
2365796c8dcSSimon Schubert 
2375796c8dcSSimon Schubert   if (nl[0].n_value != 0)
2385796c8dcSSimon Schubert     {
2395796c8dcSSimon Schubert       struct pcb *paddr;
2405796c8dcSSimon Schubert 
2415796c8dcSSimon Schubert       /* Found thread0.  */
2425796c8dcSSimon Schubert       nl[0].n_value += offsetof (struct thread, td_pcb);
2435796c8dcSSimon Schubert       if (kvm_read (core_kd, nl[0].n_value, &paddr, sizeof paddr) == -1)
2445796c8dcSSimon Schubert 	error (("%s"), kvm_geterr (core_kd));
2455796c8dcSSimon Schubert 
2465796c8dcSSimon Schubert       bsd_kvm_fetch_pcb (regcache, paddr);
2475796c8dcSSimon Schubert       return;
2485796c8dcSSimon Schubert     }
2495796c8dcSSimon Schubert #endif
2505796c8dcSSimon Schubert 
251*c50c785cSJohn Marino   /* i18n: PCB == "Process Control Block".  */
2525796c8dcSSimon Schubert   error (_("Cannot find a valid PCB"));
2535796c8dcSSimon Schubert }
2545796c8dcSSimon Schubert 
2555796c8dcSSimon Schubert 
2565796c8dcSSimon Schubert /* Kernel memory interface commands.  */
2575796c8dcSSimon Schubert struct cmd_list_element *bsd_kvm_cmdlist;
2585796c8dcSSimon Schubert 
2595796c8dcSSimon Schubert static void
2605796c8dcSSimon Schubert bsd_kvm_cmd (char *arg, int fromtty)
2615796c8dcSSimon Schubert {
2625796c8dcSSimon Schubert   /* ??? Should this become an alias for "target kvm"?  */
2635796c8dcSSimon Schubert }
2645796c8dcSSimon Schubert 
2655796c8dcSSimon Schubert #ifndef HAVE_STRUCT_THREAD_TD_PCB
2665796c8dcSSimon Schubert 
2675796c8dcSSimon Schubert static void
2685796c8dcSSimon Schubert bsd_kvm_proc_cmd (char *arg, int fromtty)
2695796c8dcSSimon Schubert {
2705796c8dcSSimon Schubert   CORE_ADDR addr;
2715796c8dcSSimon Schubert 
2725796c8dcSSimon Schubert   if (arg == NULL)
2735796c8dcSSimon Schubert     error_no_arg (_("proc address"));
2745796c8dcSSimon Schubert 
2755796c8dcSSimon Schubert   if (core_kd == NULL)
2765796c8dcSSimon Schubert     error (_("No kernel memory image."));
2775796c8dcSSimon Schubert 
2785796c8dcSSimon Schubert   addr = parse_and_eval_address (arg);
2795796c8dcSSimon Schubert #ifdef HAVE_STRUCT_LWP
2805796c8dcSSimon Schubert   addr += offsetof (struct lwp, l_addr);
2815796c8dcSSimon Schubert #else
2825796c8dcSSimon Schubert   addr += offsetof (struct proc, p_addr);
2835796c8dcSSimon Schubert #endif
2845796c8dcSSimon Schubert 
2855796c8dcSSimon Schubert   if (kvm_read (core_kd, addr, &bsd_kvm_paddr, sizeof bsd_kvm_paddr) == -1)
2865796c8dcSSimon Schubert     error (("%s"), kvm_geterr (core_kd));
2875796c8dcSSimon Schubert 
2885796c8dcSSimon Schubert   target_fetch_registers (get_current_regcache (), -1);
2895796c8dcSSimon Schubert 
2905796c8dcSSimon Schubert   reinit_frame_cache ();
2915796c8dcSSimon Schubert   print_stack_frame (get_selected_frame (NULL), -1, 1);
2925796c8dcSSimon Schubert }
2935796c8dcSSimon Schubert 
2945796c8dcSSimon Schubert #endif
2955796c8dcSSimon Schubert 
2965796c8dcSSimon Schubert static void
2975796c8dcSSimon Schubert bsd_kvm_pcb_cmd (char *arg, int fromtty)
2985796c8dcSSimon Schubert {
2995796c8dcSSimon Schubert   if (arg == NULL)
300*c50c785cSJohn Marino     /* i18n: PCB == "Process Control Block".  */
3015796c8dcSSimon Schubert     error_no_arg (_("pcb address"));
3025796c8dcSSimon Schubert 
3035796c8dcSSimon Schubert   if (core_kd == NULL)
3045796c8dcSSimon Schubert     error (_("No kernel memory image."));
3055796c8dcSSimon Schubert 
3065796c8dcSSimon Schubert   bsd_kvm_paddr = (struct pcb *)(u_long) parse_and_eval_address (arg);
3075796c8dcSSimon Schubert 
3085796c8dcSSimon Schubert   target_fetch_registers (get_current_regcache (), -1);
3095796c8dcSSimon Schubert 
3105796c8dcSSimon Schubert   reinit_frame_cache ();
3115796c8dcSSimon Schubert   print_stack_frame (get_selected_frame (NULL), -1, 1);
3125796c8dcSSimon Schubert }
3135796c8dcSSimon Schubert 
3145796c8dcSSimon Schubert static int
3155796c8dcSSimon Schubert bsd_kvm_thread_alive (struct target_ops *ops,
3165796c8dcSSimon Schubert 		      ptid_t ptid)
3175796c8dcSSimon Schubert {
3185796c8dcSSimon Schubert   return 1;
3195796c8dcSSimon Schubert }
3205796c8dcSSimon Schubert 
3215796c8dcSSimon Schubert static char *
3225796c8dcSSimon Schubert bsd_kvm_pid_to_str (struct target_ops *ops, ptid_t ptid)
3235796c8dcSSimon Schubert {
3245796c8dcSSimon Schubert   static char buf[64];
3255796c8dcSSimon Schubert   xsnprintf (buf, sizeof buf, "<kvm>");
3265796c8dcSSimon Schubert   return buf;
3275796c8dcSSimon Schubert }
3285796c8dcSSimon Schubert 
3295796c8dcSSimon Schubert static int
3305796c8dcSSimon Schubert bsd_kvm_return_one (struct target_ops *ops)
3315796c8dcSSimon Schubert {
3325796c8dcSSimon Schubert   return 1;
3335796c8dcSSimon Schubert }
3345796c8dcSSimon Schubert 
3355796c8dcSSimon Schubert /* Add the libkvm interface to the list of all possible targets and
3365796c8dcSSimon Schubert    register CUPPLY_PCB as the architecture-specific process control
3375796c8dcSSimon Schubert    block interpreter.  */
3385796c8dcSSimon Schubert 
3395796c8dcSSimon Schubert void
3405796c8dcSSimon Schubert bsd_kvm_add_target (int (*supply_pcb)(struct regcache *, struct pcb *))
3415796c8dcSSimon Schubert {
3425796c8dcSSimon Schubert   gdb_assert (bsd_kvm_supply_pcb == NULL);
3435796c8dcSSimon Schubert   bsd_kvm_supply_pcb = supply_pcb;
3445796c8dcSSimon Schubert 
3455796c8dcSSimon Schubert   bsd_kvm_ops.to_shortname = "kvm";
3465796c8dcSSimon Schubert   bsd_kvm_ops.to_longname = _("Kernel memory interface");
3475796c8dcSSimon Schubert   bsd_kvm_ops.to_doc = _("Use a kernel virtual memory image as a target.\n\
3485796c8dcSSimon Schubert Optionally specify the filename of a core dump.");
3495796c8dcSSimon Schubert   bsd_kvm_ops.to_open = bsd_kvm_open;
3505796c8dcSSimon Schubert   bsd_kvm_ops.to_close = bsd_kvm_close;
3515796c8dcSSimon Schubert   bsd_kvm_ops.to_fetch_registers = bsd_kvm_fetch_registers;
3525796c8dcSSimon Schubert   bsd_kvm_ops.to_xfer_partial = bsd_kvm_xfer_partial;
3535796c8dcSSimon Schubert   bsd_kvm_ops.to_files_info = bsd_kvm_files_info;
3545796c8dcSSimon Schubert   bsd_kvm_ops.to_thread_alive = bsd_kvm_thread_alive;
3555796c8dcSSimon Schubert   bsd_kvm_ops.to_pid_to_str = bsd_kvm_pid_to_str;
3565796c8dcSSimon Schubert   bsd_kvm_ops.to_stratum = process_stratum;
3575796c8dcSSimon Schubert   bsd_kvm_ops.to_has_memory = bsd_kvm_return_one;
3585796c8dcSSimon Schubert   bsd_kvm_ops.to_has_stack = bsd_kvm_return_one;
3595796c8dcSSimon Schubert   bsd_kvm_ops.to_has_registers = bsd_kvm_return_one;
3605796c8dcSSimon Schubert   bsd_kvm_ops.to_magic = OPS_MAGIC;
3615796c8dcSSimon Schubert 
3625796c8dcSSimon Schubert   add_target (&bsd_kvm_ops);
3635796c8dcSSimon Schubert 
3645796c8dcSSimon Schubert   add_prefix_cmd ("kvm", class_obscure, bsd_kvm_cmd, _("\
3655796c8dcSSimon Schubert Generic command for manipulating the kernel memory interface."),
3665796c8dcSSimon Schubert 		  &bsd_kvm_cmdlist, "kvm ", 0, &cmdlist);
3675796c8dcSSimon Schubert 
3685796c8dcSSimon Schubert #ifndef HAVE_STRUCT_THREAD_TD_PCB
3695796c8dcSSimon Schubert   add_cmd ("proc", class_obscure, bsd_kvm_proc_cmd,
3705796c8dcSSimon Schubert 	   _("Set current context from proc address"), &bsd_kvm_cmdlist);
3715796c8dcSSimon Schubert #endif
3725796c8dcSSimon Schubert   add_cmd ("pcb", class_obscure, bsd_kvm_pcb_cmd,
373*c50c785cSJohn Marino 	   /* i18n: PCB == "Process Control Block".  */
3745796c8dcSSimon Schubert 	   _("Set current context from pcb address"), &bsd_kvm_cmdlist);
3755796c8dcSSimon Schubert 
3765796c8dcSSimon Schubert   /* Some notes on the ptid usage on this target.
3775796c8dcSSimon Schubert 
3785796c8dcSSimon Schubert      The pid field represents the kvm inferior instance.  Currently,
3795796c8dcSSimon Schubert      we don't support multiple kvm inferiors, but we start at 1
3805796c8dcSSimon Schubert      anyway.  The lwp field is set to != 0, in case the core wants to
3815796c8dcSSimon Schubert      refer to the whole kvm inferior with ptid(1,0,0).
3825796c8dcSSimon Schubert 
3835796c8dcSSimon Schubert      If kvm is made to export running processes as gdb threads,
3845796c8dcSSimon Schubert      the following form can be used:
3855796c8dcSSimon Schubert      ptid (1, 1, 0) -> kvm inferior 1, in kernel
3865796c8dcSSimon Schubert      ptid (1, 1, 1) -> kvm inferior 1, process 1
3875796c8dcSSimon Schubert      ptid (1, 1, 2) -> kvm inferior 1, process 2
388*c50c785cSJohn Marino      ptid (1, 1, n) -> kvm inferior 1, process n  */
389*c50c785cSJohn Marino 
3905796c8dcSSimon Schubert   bsd_kvm_ptid = ptid_build (1, 1, 0);
3915796c8dcSSimon Schubert }
392