watldsym.c

来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 409 行

C
409
字号
/****************************************************************************
*
*                            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:  WHEN YOU FIGURE OUT WHAT THIS FILE DOES, PLEASE
*               DESCRIBE IT HERE!
*
****************************************************************************/


#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
#include "dipwat.h"


extern void             InfoClear( imp_image_handle * );
extern dip_status       MakeGblLst(section_info * );
extern dip_status       InitDemand(imp_image_handle *);
extern unsigned         GblSymSplit( info_block *, section_info * );
extern void             GblSymFini(section_info *);
extern unsigned         ModInfoSplit( info_block *, section_info * );
extern void             ModInfoFini(section_info *);
extern unsigned         AddrInfoSplit( info_block *, section_info * );
extern void             AddrInfoFini(section_info *);
extern dip_status       AdjustMods(section_info *,unsigned long );
extern void             AdjustAddrInit( void );
extern void             AdjustAddrs( section_info * );
extern void             AdjustSyms( section_info * );
extern void             SetModBase( imp_image_handle * );
extern void             ClearTypeCache( imp_image_handle * );



static void FreeInfBlks( info_block *blk )
{
    info_block          *tmp;

    while( blk != NULL ) {
        tmp = blk;
        blk = blk->next;
        DCFree( tmp );
    }
}

/*
 * DIPImpUnloadInfo -- unload the symbolic information for an image
 */

static void UnloadInfo( imp_image_handle *ii )
{
    section_info        *inf;
    unsigned            i;

    inf = ii->sect;
    if( inf != NULL ) {
        ClearTypeCache( ii );
        for( i = ii->num_sects; i > 0; --i, ++inf ) {
            GblSymFini( inf );
            FreeInfBlks( inf->gbl );
            ModInfoFini( inf );
            FreeInfBlks( inf->mod_info );
            AddrInfoFini( inf );
            FreeInfBlks( inf->addr_info );
        }
    }
    DCFree( ii->lang );
}

void DIPENTRY DIPImpUnloadInfo( imp_image_handle *ii )
{
    InfoClear( ii );
    DCClose( ii->sym_file );
    UnloadInfo( ii );
}


/*
 * GetBlockInfo -- get permanent information into memory
 */

static dip_status GetBlockInfo( section_info *new, unsigned long off,
                            unsigned long size, info_block **owner,
                            unsigned (*split)(info_block *, section_info *) )
{
    unsigned            split_size;
    info_block          *curr;
    unsigned            block_size;

    *owner = NULL;
    if( size == 0 ) return( DS_OK );
    for( ;; ) {
        block_size = (size > INFO_MAX_BLOCK) ? INFO_MAX_BLOCK : size;
        curr = DCAlloc( (sizeof( info_block )-1) + block_size );
        if( curr == NULL ) {
            DCStatus( DS_ERR|DS_NO_MEM );
            return( DS_ERR|DS_NO_MEM );
        }
        *owner = curr;
        curr->size = block_size;
        curr->next = NULL;
        curr->link = NULL;
        if( DCSeek( new->ctl->sym_file, off, DIG_ORG) != off ) {
            DCStatus( DS_ERR|DS_INFO_INVALID );
            return( DS_ERR|DS_INFO_INVALID );
        }
        if( DCRead( new->ctl->sym_file, curr->info, block_size )
                                != block_size ) {
            DCStatus( DS_ERR|DS_INFO_INVALID );
            return( DS_ERR|DS_INFO_INVALID );
        }
        if( block_size == size ) return( DS_OK );
        split_size = split( curr, new );
        curr = DCRealloc( curr, (sizeof( info_block )-1) + split_size );
        curr->size = split_size;
        off += split_size;
        size -= split_size;
        owner = &curr->next;
    }
}


/*
 * GetNumSect - find the number of sections for this load
 */

static dip_status GetNumSect( dig_fhandle sym_file, unsigned long curr,
                             unsigned long end, unsigned *count )
{
    section_dbg_header  header;

    *count = 0;
    for( ;; ) {
        if( curr == end ) return( DS_OK );
        if( curr > end ) {
            DCStatus( DS_ERR|DS_INFO_INVALID );
            return( DS_ERR|DS_INFO_INVALID );
        }
        if( DCRead( sym_file, &header, sizeof(header) ) != sizeof(header) ) {
            DCStatus( DS_ERR|DS_INFO_INVALID );
            return( DS_ERR|DS_INFO_INVALID );
        }
        /* if there are no modules in the section, it's a 'placekeeper' section
            for the linker overlay structure -- just ignore it */
        if( header.mod_offset != header.gbl_offset ) {
            if( header.mod_offset >  header.gbl_offset ) {
                DCStatus( DS_ERR|DS_INFO_INVALID );
                return( DS_ERR|DS_INFO_INVALID );
            }
            if( header.gbl_offset > header.addr_offset ) {
                DCStatus( DS_ERR|DS_INFO_INVALID );
                return( DS_ERR|DS_INFO_INVALID );
            }
            if( header.addr_offset >= header.section_size ) {
                DCStatus( DS_ERR|DS_INFO_INVALID );
                return( DS_ERR|DS_INFO_INVALID );
            }
        }
        (*count)++;
        curr = DCSeek( sym_file,
                header.section_size - sizeof(section_dbg_header), DIG_CUR );
    }
}

/*
 * This function assumes that section 0 is the lowest numbered section and
 * that section numbers are contiguous
 */

static dip_status ProcSectionInfo( imp_image_handle *ctl, unsigned long pos )
{
    section_dbg_header  header;
    section_info        *new;
    dip_status          status;

    DCRead( ctl->sym_file, &header, sizeof( header ) );
    new = ctl->sect + header.section_id;
    new->sect_id = header.section_id;
    new->ctl = ctl;
    new->mod_info = NULL;
    new->addr_info = NULL;
    new->gbl = NULL;
    new->dmnd_link = NULL;
    /* if there are no modules in the section, it's a 'placekeeper' section
        for the linker overlay structure -- just ignore it */
    if( header.mod_offset != header.gbl_offset ) {
        status = GetBlockInfo( new, header.mod_offset + pos,
                            header.gbl_offset - header.mod_offset,
                            &new->mod_info, &ModInfoSplit );
        if( status != DS_OK ) return( status );
        status = GetBlockInfo( new, header.gbl_offset + pos,
                            header.addr_offset - header.gbl_offset,
                            &new->gbl, &GblSymSplit );
        if( status != DS_OK ) return( status );
        status = GetBlockInfo( new, header.addr_offset+pos,
                            header.section_size - header.addr_offset,
                            &new->addr_info, &AddrInfoSplit );
        if( status != DS_OK ) return( status );
        status = MakeGblLst( new );
        if( status != DS_OK ) return( status );
        status = AdjustMods( new, pos );
        if( status != DS_OK ) return( status );
    }
    ctl->num_sects++;
    pos += header.section_size;
    if( DCSeek( ctl->sym_file, pos, DIG_ORG ) != pos ) {
        DCStatus( DS_ERR|DS_INFO_INVALID );
        return( DS_ERR|DS_INFO_INVALID );
    }
    return( DS_OK );
}


static dip_status DoPermInfo( imp_image_handle *ii )
{
    master_dbg_header   header;
    dip_status           ret;
    unsigned long       end;
    unsigned long       curr;
    unsigned            num_segs;
    unsigned            num_sects;
    byte                v2;
    byte                *new;

    end = DCSeek( ii->sym_file, -(int)sizeof( header ), DIG_END );
    if( DCRead( ii->sym_file, &header, sizeof(header) ) != sizeof(header) ) {
        return( DS_FAIL );
    }
    while( header.signature == FOX_SIGNATURE1
            || header.signature == FOX_SIGNATURE2
            || header.signature == WAT_RES_SIG ) {
        if( header.debug_size > end ) {
            DCStatus( DS_ERR|DS_INFO_INVALID );
            return( DS_ERR|DS_INFO_INVALID );
        }
        end = DCSeek( ii->sym_file, end - header.debug_size, DIG_ORG );
        DCRead( ii->sym_file, &header, sizeof(header) );
    }
    if( header.signature != VALID_SIGNATURE ) return( DS_FAIL );
    switch( header.exe_major_ver ) {
    case EXE_MAJOR_VERSION:
        v2 = 0;
        break;
    case OLD_EXE_MAJOR_VERSION:
        v2 = 1;
        break;
    default:
        DCStatus( DS_ERR|DS_INFO_BAD_VERSION );
        return( DS_ERR|DS_INFO_BAD_VERSION );
    }
    if( header.exe_minor_ver > EXE_MINOR_VERSION ) {
        DCStatus( DS_ERR|DS_INFO_BAD_VERSION );
        return( DS_ERR|DS_INFO_BAD_VERSION );
    }
    if( header.obj_major_ver != OBJ_MAJOR_VERSION ) {
        DCStatus( DS_ERR|DS_INFO_BAD_VERSION );
        return( DS_ERR|DS_INFO_BAD_VERSION );
    }
    if( header.obj_minor_ver > OBJ_MINOR_VERSION ) {
        DCStatus( DS_ERR|DS_INFO_BAD_VERSION );
        return( DS_ERR|DS_INFO_BAD_VERSION );
    }
    if( ( end + sizeof( header ) ) < header.debug_size ) {
        DCStatus( DS_ERR|DS_INFO_INVALID );
        return( DS_ERR|DS_INFO_INVALID );
    }
    num_segs = header.segment_size / sizeof( addr_seg );
    curr = DCSeek( ii->sym_file, -header.debug_size + header.lang_size
                        + header.segment_size, DIG_CUR );
    ret = GetNumSect( ii->sym_file, curr, end, &num_sects );
    if( ret != DS_OK ) return( ret );
    new = DCAlloc( header.lang_size
                + num_segs * ( sizeof( addr_seg ) + sizeof( addr_ptr ) )
                + num_sects * sizeof( section_info ) );
    if( new == NULL ) {
        DCStatus( DS_ERR|DS_NO_MEM );
        return( DS_ERR|DS_NO_MEM );
    }
    ii->v2 = v2;
    ii->lang = new;
    ii->num_segs = num_segs;
    ii->map_segs = (void *)&new[ header.lang_size ];
    ii->real_segs = (void *)(ii->map_segs + num_segs);
    ii->sect = (void *)(ii->real_segs + num_segs);
    ii->num_sects = 0;
    DCSeek( ii->sym_file, curr - header.lang_size - header.segment_size,
                DIG_ORG );
    if( DCRead( ii->sym_file, ii->lang, header.lang_size ) != header.lang_size ) {
        DCStatus( DS_ERR|DS_INFO_INVALID );
        return( DS_ERR|DS_INFO_INVALID );
    }
    if( DCRead( ii->sym_file, ii->map_segs, header.segment_size ) != header.segment_size ) {
        DCStatus( DS_ERR|DS_INFO_INVALID );
        return( DS_ERR|DS_INFO_INVALID );
    }
    while( ii->num_sects < num_sects ) {
        curr = DCSeek( ii->sym_file, 0L, DIG_CUR );
        ret = ProcSectionInfo( ii, curr );
        if( ret != DS_OK ) return( ret );
    }
    SetModBase( ii );
    return( InitDemand( ii ) );
}

/*
 * DIPImpLoadInfo -- process symbol table info on end of .exe file
 */
dip_status DIPENTRY DIPImpLoadInfo( dig_fhandle file, imp_image_handle *ii )
{
    dip_status          ret;

    if( file == DIG_NIL_HANDLE ) {
        DCStatus( DS_ERR|DS_FOPEN_FAILED );
        return( DS_ERR|DS_FOPEN_FAILED );
    }
    ii->sym_file = file;
    ii->sect = NULL;
    ii->lang = NULL;
    ret = DoPermInfo( ii );
    if( ret != DS_OK ) UnloadInfo( ii );
    return( ret );
}


/*
 * InfoRead -- read demand information from disk
 */

dip_status InfoRead( section_info *inf, unsigned long offset, unsigned size,
                void *buff )
{
    dig_fhandle  sym_file;

    sym_file = inf->ctl->sym_file;
    if( DCSeek( sym_file, offset, DIG_ORG ) != offset ) {
        DCStatus( DS_ERR|DS_FSEEK_FAILED );
        return( DS_ERR|DS_FSEEK_FAILED );
    }
    if( DCRead( sym_file, buff, size ) != size ) {
        DCStatus( DS_ERR|DS_FREAD_FAILED );
        return( DS_ERR|DS_FREAD_FAILED );
    }
    return( DS_OK );
}



/*
 * DIPImpMapInfo -- change all map addresses into real addresses
 */


void DIPENTRY DIPImpMapInfo( imp_image_handle *ii, void *d )
{
    unsigned        i;

    for( i = 0; i < ii->num_segs; ++i ) {
        ii->real_segs[ i ].offset = 0;
        ii->real_segs[ i ].segment = ii->map_segs[ i ];
        DCMapAddr( &ii->real_segs[ i ], d );
    }
    AdjustAddrInit();
    for( i = 0; i < ii->num_sects; ++i ) {
        AdjustAddrs( &ii->sect[i] );
        AdjustSyms(  &ii->sect[i] );
    }
}


/*
 * AddressMap - take a map address and turn it into a real address
 */

void AddressMap( imp_image_handle *ii, addr_ptr *addr )
{
    unsigned            i;

    /* could probably binary search this */
    for( i = 0; i < ii->num_segs; ++i ) {
        if( addr->segment == ii->map_segs[ i ] ) {
            addr->segment = ii->real_segs[ i ].segment;
            addr->offset += ii->real_segs[ i ].offset;
            return;
        }
    }
}

⌨️ 快捷键说明

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