ftp.nice.ch/Attic/openStep/developer/bundles/GDBbundle.1.0.s.tgz#/GDBbundle-1.0.s/debug/gdb/gdb/next/NeXT-core.m

This is NeXT-core.m in view mode; [Download] [Up]

#import "gdb.h"
#import "SegmentManagerThreads.h"
#import "threads.h"
#import "TellDisplayer.h"

int current_core_thread = 0;
static struct target_ops core_ops;

extern void child_attach (char *name, int from_tty);
extern void child_create_inferior (char *exec, char *args, char **env);
extern RegionManager *regionManager;
extern void core_get_registers(int regnum);
extern CORE_ADDR
getThreadPCAndNameFromState(REGS_STRUCT *regsState, char **pName);
extern char *
getThreadNameFromState(USER_REG_STRUCT *userRegState, int n);

static void
core_cleanup()
{
  if (regionManager) {
    [regionManager free];
    regionManager = nil;
  }
}

void
core_open (filename, from_tty)
     char *filename;
     int from_tty;
{
  char *temp;
  struct cleanup *old_chain;
  SegmentManager *newCM;

  if (!filename) {	/* CAUTION: using ANSI string merging */
    error(regionManager
	  ? "No core file specified."
	    "(Use `detach' to stop debugging a core file.)"
	  : "No core file specified.");
  }

  filename = tilde_expand (filename);
  if (filename[0] != '/') {
    temp = concat (current_directory, "/", filename, NULL);
    free (filename);
    filename = temp;
  }

  old_chain = make_cleanup (free, filename);
  newCM = [SegmentManager newCore: filename];
  if (!newCM)
    error("\"%s\" is not a core file.", filename);

  if (![newCM validate])
    error("\"%s\" is an invalid core file.", filename);


  discard_cleanups(old_chain);
  core_cleanup();
  regionManager = (RegionManager *)newCM;
  old_chain = make_cleanup(core_cleanup, NULL);

  push_target(&core_ops);
  core_get_registers(-1);
  set_current_frame(create_new_frame(read_register(FP_REGNUM), read_pc()));
  select_frame(get_current_frame(), 0);
  print_sel_frame(0);
  tell_displayer_state_changed(DBG_STATE_INFERIOR_STOPPED);
  discard_cleanups(old_chain);
}

void
core_detach(quitting)
{
  core_cleanup();
}

void
core_file_command(filename, from_tty)
     char *filename;
     int from_tty;
{
  if (!filename)
    core_cleanup();
  else
    core_open(filename, from_tty);
}

int
have_core_file_p ()
{
  return regionManager != NULL;
}

static void
core_files_info()
{
}

static int
core_xfer_memory (memaddr, myaddr, len, write)
     CORE_ADDR memaddr;
     char *myaddr;
     int len;
     int write;
{
  if (write)
    error("Can't write to core files.");
  return [regionManager getDataAt: (void *)memaddr for: len into: myaddr];
}

void
validate_files()
{
}

void
core_thread_list()
{
  int numThreads, i;
  ThreadInfo *tInfos, *tInfo;
  SegmentManager *coreManager = (SegmentManager *)regionManager;
  char *name, *fName;
  CORE_ADDR pc;

  numThreads = [coreManager numThreads];
  if (numThreads == 1)
    printf("There is 1 thread.\n");
  else
    printf("There are %d threads.\n", numThreads);
  if (numThreads == 0)
    return;		/* MVS: nothing more to do (no threads to list).  */
  tInfos = [coreManager threadInfos];	/* Note: tInfos must be freed below */
  printf("Thread PC         Name            Function\n");         
  for (i = 0; i < numThreads; i++) {
    name = getThreadNameFromState(THREADINFO_NAME_STATE(tInfos[i]), i);
    pc = getThreadPCAndNameFromState(THREADINFO_PC_STATE(tInfos[i]), &fName);
    printf ("%-6d 0x%-8x %-15s %-40s\n", i, pc, name, fName);
  }
  free(tInfos);
}

void
core_thread_select(args, from_tty)
     char *args;
     int from_tty;
{
  int numThreads, numThread;
  SegmentManager *coreManager = (SegmentManager *)regionManager;

  if (!args)
    error_no_arg ("thread index to select");
  numThreads = [coreManager numThreads];
  numThread = atoi(args);
  if ((numThreads < numThreads) || (numThreads <= numThread))
    error("Invalid thread.\n");
  else
    current_core_thread = numThread;
}

static struct target_ops core_ops = {
    "core",			/* to_shortname */
    "Local core dump file",		/* to_longname */
    "Use a core file as a target.\n"
    "Specify the filename of the core file.",	/* to_doc */
    core_open,			/* to_open */
    core_detach,		/* to_close */
    0,	/* to_attach */
    0, 				/* to_detach */
    0,				/* to_resume */
    0,				/* to_wait */
    core_get_registers,				/* to_fetch_registers */
    0,				/* to_store_registers */
    0,				/* to_prepare_to_store */
    core_xfer_memory,		/* to_xfer_memory */
    core_files_info,				/* to_files_info */
    0,				/* to_insert_breakpoint */
    0,				/* to_remove_breakpoint */
    0,				/* to_terminal_init */
    0, 				/* to_terminal_inferior */
    0,				/* to_terminal_ours_for_output */
    0,				/* to_terminal_ours */
    0,				/* to_terminal_info */
    0,				/* to_kill */
    0,				/* to_load */
    0,				/* to_lookup_symbol */
    find_default_create_inferior,	/* to_create_inferior */
    0,				/* to_mourn_inferior */
    0,				/* to_can_run */
    0, 				/* to_notice_signals */
    0,   			/* to_stop */
    core_stratum,		/* to_stratum */
    0,				/* DONT_USE (formerly to_next) */
    0,				/* to_has_all_memory */
    1,				/* to_has_memory */
    1,				/* to_has_stack */
    1,				/* to_has_registers */
    0,				/* to_has_execution */
    0,				/* sections */
    0,				/* sections_end */
    OPS_MAGIC			/* to_magic */
};


void (*exec_file_display_hook) () = NULL;

/* The current default bfd target.  Points to storage allocated for
   gnutarget_string.  */
char *gnutarget;

/* Same thing, except it is "auto" not NULL for the default case.  */
static char *gnutarget_string;

static void set_gnutarget_command
  PARAMS ((char *, int, struct cmd_list_element *));

static void
set_gnutarget_command (ignore, from_tty, c)
     char *ignore;
     int from_tty;
     struct cmd_list_element *c;
{
  if (STREQ (gnutarget_string, "auto"))
    gnutarget = NULL;
  else
    gnutarget = gnutarget_string;
}

/* Set the gnutarget.  */
void
set_gnutarget (newtarget)
     char *newtarget;
{
  if (gnutarget_string != NULL)
    free (gnutarget_string);
  gnutarget_string = savestring (newtarget, strlen (newtarget));
  set_gnutarget_command (NULL, 0, NULL);
}

void
_initialize_core()
{
  add_com ("core-file", class_files, core_file_command,
	   "Use FILE as core dump for examining memory and registers.\n\
No arg means have no core file.  This command has been superseded by the\n\
`target core' and `detach' commands.");
  add_target (&core_ops);
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.