⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 core_nto.c

📁 开放源码的编译器open watcom 1.6.0版的源代码
💻 C
字号:
/****************************************************************************
*
*                            Open Watcom Project
*
*    Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
*  ========================================================================
*
*    This file contains Original Code and/or Modifications of Original
*    Code as defined in and that are subject to the Sybase Open Watcom
*    Public License version 1.0 (the 'License'). You may not use this file
*    except in compliance with the License. BY USING THIS FILE YOU AGREE TO
*    ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
*    provided with the Original Code and Modifications, and is also
*    available at www.sybase.com/developer/opensource.
*
*    The Original Code and all software distributed under the License are
*    distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
*    EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
*    ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
*    MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
*    NON-INFRINGEMENT. Please see the License for the specific language
*    governing rights and limitations under the License.
*
*  ========================================================================
*
* Description:  QNX Neutrino ELF core file driver.
*
****************************************************************************/


#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "exeelf.h"
#include "mad.h"
#include "madregs.h"
#include "elfcore.h"
#include "core_nto.h"
#include "core_fbsd.h"


typedef struct {
    int         fd;
    bool        swap;
    Elf32_Ehdr  *e_hdr;
    Elf32_Phdr  *p_hdr;
} ctx_neutrino;


/* Initializes the platform-specific driver.
 *
 * Parameters:
 *  fd      - core file descriptor
 *  ehdr    - core file ELF header
 *  phdr    - core file ELF program header
 *
 * Returns:
 *  pointer to driver context or NULL if core file not recognized
 */
static void *neutrino_init( int fd, Elf32_Ehdr *ehdr, Elf32_Phdr *phdr )
{
    ctx_neutrino    *ctx = NULL;
    Elf_Note        note;
    char            *note_name;
    bool            swap = FALSE;

    if( ehdr->e_ident[EI_DATA] != NATIVE_ELF_ORDER ) {
        swap = TRUE;
    }

    /* See if the core file looks like something we understand */
    note.n_type = QNT_CORE_INFO;
    note_name = find_note( fd, ehdr, phdr, swap, &note );
    if( note_name ) {
        if( !strcmp( note_name, QNX_NOTE_NAME ) ) {
            free( note_name );
            ctx = malloc( sizeof( ctx_neutrino ) );
            if( ctx ) {
                ctx->fd    = fd;
                ctx->e_hdr = ehdr;
                ctx->p_hdr = phdr;
                ctx->swap  = swap;
            }
        }
    }
    return( ctx );
}


/* Attempts to determine the name of the executable from the core
 * file. The executable is likely to be required for determining
 * code segment data.
 *
 * Parameters:
 *  ctx     - driver context
 *  name    - pointer to output buffer
 *  len     - size of provided buffer
 *
 * Returns:
 *  length of name string (sans terminating null) or zero in case of failure
 */
static size_t neutrino_name( void *_ctx, char *name, size_t len )
{
    ctx_neutrino    *ctx = _ctx;
    Elf_Note        note;
    prpsinfo_t      info;
    char            *note_name;

    note.n_type = NT_PRPSINFO;
    *name = '\0';
    note_name = find_note( ctx->fd, ctx->e_hdr, ctx->p_hdr, ctx->swap, &note );
    if( note_name ) {
        /* should we check the name string? */
        free( note_name );
        if( read( ctx->fd, &info, sizeof( info ) ) == sizeof( info ) ) {
            strlcpy( name, info.pr_fname, len );
        }
    }
    return( strlen( name ) );
}


/* Fill in MAD configuration data
 *
 * Parameters:
 *  ctx     - driver context
 *  mad     - MAD architecture
 *  os      - operating system
 *  cpu     - CPU model
 *  fpu     - FPU model
 *
 * Returns:
 *  zero if configuration could not be determined, non-zero otherwise
 */
static int neutrino_qcfg( void *_ctx, int *mad, int *os, int *cpu, int *fpu )
{
    ctx_neutrino    *ctx = _ctx;
    long            rc = FALSE;

    // TODO: add support for non-x86 architectures
    if( ctx->e_hdr->e_machine == EM_386 ) {
        /* There doesn't appear to be a way to determine CPU type etc.
         * from FreeBSD core files. Just say it's a 486 and be done with it.
         */
        *os  = OS_NEUTRINO;
        *mad = MAD_X86;
        *cpu = X86_486;
        *fpu = X86_487;
        rc = TRUE;
    }
    return( rc );
}


/* Determine process id of the dead process
 *
 * Parameters:
 *  ctx     - driver context
 *
 * Returns:
 *  process id of dead process or 1 if not available
 */
static long neutrino_qpid( void *_ctx )
{
    ctx_neutrino    *ctx = _ctx;
    Elf_Note        note;
    prstatus_t      status;
    char            *note_name;
    long            pid = 1;

    note.n_type = NT_PRSTATUS;
    note_name = find_note( ctx->fd, ctx->e_hdr, ctx->p_hdr, ctx->swap, &note );
    if( note_name ) {
        /* should we check the name string? */
        free( note_name );
        if( read( ctx->fd, &status, sizeof( status ) ) == sizeof( status ) ) {
            pid = status.pr_pid;
        }
    }
    return( pid );
}


/* Determine signal that killed process
 *
 * Parameters:
 *  ctx     - driver context
 *
 * Returns:
 *  signal which killed dead process or zero if not available
 */
static long neutrino_qsig( void *_ctx )
{
    ctx_neutrino    *ctx = _ctx;
    Elf_Note        note;
    prstatus_t      status;
    char            *note_name;
    long            sig = 0;

    note.n_type = NT_PRSTATUS;
    note_name = find_note( ctx->fd, ctx->e_hdr, ctx->p_hdr, ctx->swap, &note );
    if( note_name ) {
        /* should we check the name string? */
        free( note_name );
        if( read( ctx->fd, &status, sizeof( status ) ) == sizeof( status ) ) {
            sig = status.pr_cursig;
        }
    }
    return( sig );
}


/* Fills in the MAD register structure with CPU register
 * data read from core file.
 *
 * Parameters:
 *  ctx     - driver context
 *  r       - MAD register structure
 *  tid     - thread to return CPU registers for
 *
 * Returns:
 *  size of CPU registers or zero in case of failure
 */
static size_t neutrino_regs( void *_ctx, mad_registers *r, int tid )
{
    ctx_neutrino        *ctx = _ctx;
    Elf_Note            note;
    X86_CPU_REGISTERS   regs;
    char                *note_name;

    note.n_type = QNT_CORE_GREG;
    note_name = find_note( ctx->fd, ctx->e_hdr, ctx->p_hdr, ctx->swap, &note );
    if( note_name ) {
        free( note_name );
        /* Read the program status (note descriptor data) */
        if( read( ctx->fd, &regs, sizeof( regs ) ) == sizeof( regs ) ) {
            r->x86.cpu.eax = regs.eax;
            r->x86.cpu.ebx = regs.ebx;
            r->x86.cpu.ecx = regs.ecx;
            r->x86.cpu.edx = regs.edx;
            r->x86.cpu.esi = regs.esi;
            r->x86.cpu.edi = regs.edi;
            r->x86.cpu.ebp = regs.ebp;
            r->x86.cpu.esp = regs.esp;
            r->x86.cpu.eip = regs.eip;
            r->x86.cpu.efl = regs.efl;
            r->x86.cpu.cs  = regs.cs;
            r->x86.cpu.ss  = regs.ss;
            r->x86.cpu.es  = regs.ss;
            r->x86.cpu.ds  = regs.ss;
            r->x86.cpu.fs  = regs.ss;
            r->x86.cpu.gs  = regs.ss;
            return( sizeof( struct x86_mad_registers ) );
        }
    }
    return( 0 );
}


/* Fills in the MAD register structure with FPU register
 * data read from core file.
 *
 * Parameters:
 *  ctx     - driver context
 *  r       - MAD register structure
 *  tid     - thread to return FPU registers for
 *
 * Returns:
 *  size of FPU registers or zero in case of failure
 */
static size_t neutrino_freg( void *ctx, mad_registers *r, int tid )
{
    return( 0 );
}


/* Frees any resources allocated by driver and generally undoes
 * whatever the init function did.
 */
static void neutrino_done( void *ctx )
{
    free( ctx );
}

plat_drv_t  Drv_Neutrino = {
    neutrino_init,
    neutrino_name,
    neutrino_qcfg,
    neutrino_qpid,
    neutrino_qsig,
    neutrino_regs,
    neutrino_freg,
    neutrino_done
};

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -