dbgmodcodeview.cpp revision 44f4668fd6b017b6d71e83a53d771d67f5e00ab7
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/* $Id$ */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** @file
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * IPRT - Debug Module Reader For Microsoft CodeView and COFF.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Based on the following documentation (plus guess work and googling):
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
e64031e20c39650a7bc902a3e1aba613b9415deevboxsync * - "Tools Interface Standard (TIS) Formats Specification for Windows",
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * dated February 1993, version 1.0.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * - "Visual C++ 5.0 Symbolic Debug Information Specification" chapter of
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * SPECS.CHM from MSDN Library October 2001.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * - "High Level Languages Debug Table Documentation", aka HLLDBG.HTML, aka
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * IBMHLL.HTML, last changed 1996-07-08.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/*
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Copyright (C) 2013 Oracle Corporation
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * This file is part of VirtualBox Open Source Edition (OSE), as
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * available from http://www.virtualbox.org. This file is free software;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * you can redistribute it and/or modify it under the terms of the GNU
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * General Public License (GPL) as published by the Free Software
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Foundation, in version 2 as it comes in the "COPYING" file of the
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
b0db50948c349fa76655abf252f7946b515e8204vboxsync *
2f0d866e126dd288169fed591c259c1c6b4016e5vboxsync * The contents of this file may alternatively be used under the terms
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * of the Common Development and Distribution License Version 1.0
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * VirtualBox OSE distribution, in which case the provisions of the
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * CDDL are applicable instead of those of the GPL.
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync *
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * You may elect to license modified versions of this file under the
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * terms and conditions of either the GPL or the CDDL or both.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync/*******************************************************************************
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync* Header Files *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync*******************************************************************************/
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define LOG_GROUP RTLOGGROUP_DBG
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync#include <iprt/dbg.h>
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync#include "internal/iprt.h"
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync#include <iprt/alloca.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/asm.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/assert.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/err.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/file.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/log.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/mem.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/param.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/path.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/string.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include <iprt/strcache.h>
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include "internal/dbgmod.h"
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include "internal/ldrPE.h"
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#include "internal/magics.h"
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/*******************************************************************************
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync* Structures and Typedefs *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync*******************************************************************************/
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * CodeView Header. There are two of this, base header at the start of the debug
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * information and a trailing header at the end.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef struct RTCVHDR
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync{
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The magic ('NBxx'), see RTCVHDR_MAGIC_XXX. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t u32Magic;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Base header: Subsection directory offset relative to this header (start).
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Trailing header: Offset of the base header relative to the end of the file.
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync *
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * Called lfoBase, lfaBase, lfoDirectory, lfoDir and probably other things in
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * the various specs/docs available. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t off;
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync} RTCVHDR;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Pointer to a CodeView header. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef RTCVHDR *PRTCVHDR;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** @name CodeView magic values (RTCVHDR::u32Magic).
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * @{ */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** CodeView from Visual C++ 5.0. Specified in the 2001 MSDN specs.chm file. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync#define RTCVHDR_MAGIC_NB11 RT_MAKE_U32_FROM_U8('N', 'B', '1', '1')
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** External PDB reference (often referred to as PDB 2.0). */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB10 RT_MAKE_U32_FROM_U8('N', 'B', '1', '0')
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** CodeView v4.10, packed. Specified in the TIS document. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB09 RT_MAKE_U32_FROM_U8('N', 'B', '0', '9')
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** CodeView v4.00 thru v4.05. Specified in the TIS document? */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB08 RT_MAKE_U32_FROM_U8('N', 'B', '0', '8')
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync/** Quick C for Windows 1.0 debug info. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync#define RTCVHDR_MAGIC_NB07 RT_MAKE_U32_FROM_U8('N', 'B', '0', '7')
0c4004948fca34f2db87e7b38013137e9472c306vboxsync/** Emitted by ILINK indicating incremental link. Comparable to NB05? */
0c4004948fca34f2db87e7b38013137e9472c306vboxsync#define RTCVHDR_MAGIC_NB06 RT_MAKE_U32_FROM_U8('N', 'B', '0', '6')
0c4004948fca34f2db87e7b38013137e9472c306vboxsync/** Emitted by LINK version 5.20 and later before packing. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB05 RT_MAKE_U32_FROM_U8('N', 'B', '0', '5')
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Emitted by IBM ILINK for HLL (similar to NB02 in many ways). */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB04 RT_MAKE_U32_FROM_U8('N', 'B', '0', '4')
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Emitted by LINK version 5.10 (or similar OMF linkers), as shipped with
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync * Microsoft C v6.0 for example. More or less entirely 16-bit. */
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync#define RTCVHDR_MAGIC_NB02 RT_MAKE_U32_FROM_U8('N', 'B', '0', '2')
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync/* No idea what NB03 might have been. */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync/** AIX debugger format according to "IBM OS/2 16/32-bit Object Module Format
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * (OMF) and Linear eXecutable Module Format (LX)" revision 10 (LXOMF.PDF). */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB01 RT_MAKE_U32_FROM_U8('N', 'B', '0', '1')
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Ancient CodeView format according to LXOMF.PDF. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVHDR_MAGIC_NB00 RT_MAKE_U32_FROM_U8('N', 'B', '0', '0')
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** @} */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync/** @name CV directory headers.
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync * @{ */
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync/**
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync * Really old CV directory header used with NB00 and NB02.
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync *
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync * Uses 16-bit directory entires (RTCVDIRENT16).
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync */
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsynctypedef struct RTCVDIRHDR16
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync{
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** The number of directory entries. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t cEntries;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync} RTCVDIRHDR16;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a old CV directory header. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVDIRHDR16 *PRTCVDIRHDR16;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Simple 32-bit CV directory base header, used by NB04 (aka IBM HLL).
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef struct RTCVDIRHDR32
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The number of bytes of this header structure. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t cbHdr;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The number of bytes per entry. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t cbEntry;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The number of directory entries. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint32_t cEntries;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync} RTCVDIRHDR32;
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync/** Pointer to a 32-bit CV directory header. */
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsynctypedef RTCVDIRHDR32 *PRTCVDIRHDR32;
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync/**
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync * Extended 32-bit CV directory header as specified in the TIS doc.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * The two extra fields seems to never have been assigned any official purpose.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef struct RTCVDIRHDR32EX
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** This starts the same way as the NB04 header. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync RTCVDIRHDR32 Core;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Tentatively decleared as the offset to the next directory generated by
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * the incremental linker. Haven't seen this used yet. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t offNextDir;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Flags, non defined apparently, so MBZ. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t fFlags;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync} RTCVDIRHDR32EX;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to an extended 32-bit CV directory header. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVDIRHDR32EX *PRTCVDIRHDR32EX;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** @} */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync/**
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * 16-bit CV directory entry used with NB00 and NB02.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef struct RTCVDIRENT16
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Subsection type (RTCVSST). */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t uSubSectType;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Which module (1-based, 0xffff is special). */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t iMod;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The lowe offset of this subsection relative to the base CV header. */
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync uint16_t offLow;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The high part of the subsection offset. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t offHigh;
93f91841f87620d1cb6d0238b3d0d5e52cd3b9a4vboxsync /** The size of the subsection. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t cb;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync} RTCVDIRENT16;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsyncAssertCompileSize(RTCVDIRENT16, 10);
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a 16-bit CV directory entry. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVDIRENT16 *PRTCVDIRENT16;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/**
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * 32-bit CV directory entry used starting with NB04.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef struct RTCVDIRENT32
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** Subsection type (RTCVSST). */
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync uint16_t uSubSectType;
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** Which module (1-based, 0xffff is special). */
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync uint16_t iMod;
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** The offset of this subsection relative to the base CV header. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t off;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The size of the subsection. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t cb;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync} RTCVDIRENT32;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsyncAssertCompileSize(RTCVDIRENT32, 12);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Pointer to a 32-bit CV directory entry. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVDIRENT32 *PRTCVDIRENT32;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Pointer to a const 32-bit CV directory entry. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVDIRENT32 const *PCRTCVDIRENT32;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * CodeView subsection types.
f687f34bd232be13744edbc0cc5155fa5d4540edvboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef enum RTCVSST
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** @name NB00, NB02 and NB04 subsection types.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * The actual format of each subsection varies between NB04 and the others,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * and it may further vary in NB04 depending on the module type.
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync * @{ */
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync kCvSst_OldModule = 0x101,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSst_OldPublic,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSst_OldTypes,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSst_OldSymbols,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSst_OldSrcLines,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSst_OldLibraries,
ead016c68c61b5f2e1fe4d237054eebea9327d4bvboxsync kCvSst_OldImports,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSst_OldCompacted,
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync kCvSst_OldSrcLnSeg = 0x109,
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync kCvSst_OldSrcLines3 = 0x10b,
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** @} */
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync /** @name NB09, NB11 (and possibly NB05, NB06, NB07, and NB08) subsection types.
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync * @{ */
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_Module = 0x120,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_Types,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_Public,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_PublicSym,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_Symbols,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_AlignSym,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_SrcLnSeg,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_SrcModule,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_Libraries,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_GlobalSym,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_GlobalPub,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_GlobalTypes,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_MPC,
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync kCvSst_SegMap,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_SegName,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_PreComp,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_PreCompMap,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_OffsetMap16,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_OffsetMap32,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_FileIndex = 0x133,
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync kCvSst_StaticSym
64f58e4154eaa20c47782b429eeaff09070369bfvboxsync /** @} */
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsync} RTCVSST;
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync/** Pointer to a CV subsection type value. */
7ccfefe49db4cd93c3701d7b60873ebf404b5b87vboxsynctypedef RTCVSST *PRTCVSST;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Pointer to a const CV subsection type value. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVSST const *PCRTCVSST;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/**
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * CV4 module segment info.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef struct RTCVMODSEGINFO32
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The segment number. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t iSeg;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Explicit padding. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t u16Padding;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** Offset into the segment. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint32_t off;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The size of the contribution. */
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync uint32_t cb;
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync} RTCVMODSEGINFO32;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVMODSEGINFO32 *PRTCVMODSEGINFO32;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef RTCVMODSEGINFO32 const *PCRTCVMODSEGINFO32;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * CV4 segment map header.
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsynctypedef struct RTCVSEGMAPHDR
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync{
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync /** Number of segments descriptors in the table. */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync uint16_t cSegs;
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync /** Number of logical segment descriptors. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t cLogSegs;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync} RTCVSEGMAPHDR;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a CV4 segment map header. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVSEGMAPHDR *PRTCVSEGMAPHDR;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** Pointer to a const CV4 segment map header. */
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsynctypedef RTCVSEGMAPHDR const *PCRTCVSEGMAPHDR;
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync/**
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync * CV4 Segment map descriptor entry.
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef struct RTCVSEGMAPDESC
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Segment flags. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t fFlags;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The overlay number. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t iOverlay;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** Group index into this segment descriptor array. 0 if not relevant.
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * The group descriptors are found in the second half of the table. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t iGroup;
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync /** Complicated. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t iFrame;
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync /** Offset (byte) into the kCvSst_SegName table of the segment name, or
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * 0xffff. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t offSegName;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Offset (byte) into the kCvSst_SegName table of the class name, or 0xffff. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t offClassName;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Offset into the physical segment. */
a9f41cb889f53e8407561a6155052c441eb0fc5fvboxsync uint32_t off;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Size of segment. */
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync uint32_t cb;
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync} RTCVSEGMAPDESC;
0dd6dfbebcda0af90da4413aaea5f3b9d1817556vboxsync/** Pointer to a segment map descriptor entry. */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsynctypedef RTCVSEGMAPDESC *PRTCVSEGMAPDESC;
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync/** Pointer to a const segment map descriptor entry. */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsynctypedef RTCVSEGMAPDESC const *PCRTCVSEGMAPDESC;
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync/** @name RTCVSEGMAPDESC_F_XXX - RTCVSEGMAPDESC::fFlags values.
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync * @{ */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync#define RTCVSEGMAPDESC_F_READ UINT16_C(0x0001)
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync#define RTCVSEGMAPDESC_F_WRITE UINT16_C(0x0002)
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync#define RTCVSEGMAPDESC_F_EXECUTE UINT16_C(0x0004)
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync#define RTCVSEGMAPDESC_F_32BIT UINT16_C(0x0008)
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync#define RTCVSEGMAPDESC_F_SEL UINT16_C(0x0100)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync#define RTCVSEGMAPDESC_F_ABS UINT16_C(0x0200)
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync#define RTCVSEGMAPDESC_F_GROUP UINT16_C(0x1000)
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync#define RTCVSEGMAPDESC_F_RESERVED UINT16_C(0xecf0)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** @} */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync/**
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync * CV4 segment map subsection.
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsynctypedef struct RTCVSEGMAP
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync{
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** The header. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync RTCVSEGMAPHDR Hdr;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** Descriptor array. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync RTCVSEGMAPDESC aDescs[1];
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync} RTCVSEGMAP;
c2f2661efd8da5281e2a3af6ddd10e737d333909vboxsync/** Pointer to a segment map subsection. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsynctypedef RTCVSEGMAP *PRTCVSEGMAP;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync/** Pointer to a const segment map subsection. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsynctypedef RTCVSEGMAP const *PCRTCVSEGMAP;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync/**
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync * Global symbol table header, used by kCvSst_GlobalSym and kCvSst_GlobalPub.
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsynctypedef struct RTCVGLOBALSYMTABHDR
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync{
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** The symbol hash function. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync uint16_t uSymHash;
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** The address hash function. */
4569bf0ad094b40d2e177299a00d37e94d28616cvboxsync uint16_t uAddrHash;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** The amount of symbol information following immediately after the header. */
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync uint32_t cbSymbols;
fc78e01f665145ab3641c5f8095e9ae984ddcb84vboxsync /** The amount of symbol hash tables following the symbols. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t cbSymHash;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The amount of address hash tables following the symbol hash tables. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint32_t cbAddrHash;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync} RTCVGLOBALSYMTABHDR;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a global symbol table header. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTCVGLOBALSYMTABHDR *PRTCVGLOBALSYMTABHDR;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a const global symbol table header. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef RTCVGLOBALSYMTABHDR const *PCRTCVGLOBALSYMTABHDR;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef enum RTCVSYMTYPE
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync{
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync /** @name Symbols that doesn't change with compilation model or target machine.
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync * @{ */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Compile = 0x0001,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSymType_Register,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Constant,
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync kCvSymType_UDT,
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync kCvSymType_SSearch,
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync kCvSymType_End,
96a7e06717e2d7398642eadb5ebab1bf13fbe2dbvboxsync kCvSymType_Skip,
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync kCvSymType_CVReserve,
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync kCvSymType_ObjName,
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync kCvSymType_EndArg,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_CobolUDT,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_ManyReg,
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_Return,
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync kCvSymType_EntryThis,
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** @} */
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync
ad27e1d5e48ca41245120c331cc88b50464813cevboxsync /** @name Symbols with 16:16 addresses.
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync * @{ */
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_BpRel16 = 0x0100,
0c4004948fca34f2db87e7b38013137e9472c306vboxsync kCvSymType_LData16,
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_GData16,
0c4004948fca34f2db87e7b38013137e9472c306vboxsync kCvSymType_Pub16,
0c4004948fca34f2db87e7b38013137e9472c306vboxsync kCvSymType_LProc16,
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_GProc16,
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_Thunk16,
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync kCvSymType_BLock16,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_With16,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Label16,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_CExModel16,
ad27e1d5e48ca41245120c331cc88b50464813cevboxsync kCvSymType_VftPath16,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_RegRel16,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @} */
0c4004948fca34f2db87e7b38013137e9472c306vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @name Symbols with 16:32 addresses.
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * @{ */
0c4004948fca34f2db87e7b38013137e9472c306vboxsync kCvSymType_BpRel32 = 0x0200,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_LData32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_GData32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Pub32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_LProc32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_GProc32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Thunk32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Block32,
0c4004948fca34f2db87e7b38013137e9472c306vboxsync kCvSymType_With32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Label32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_CExModel32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_VftPath32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_RegRel32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_LThread32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_GThread32,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @} */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @name Symbols for MIPS.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @{ */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_LProcMips = 0x0300,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_GProcMips,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @} */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @name Symbols for Microsoft CodeView.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @{ */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_ProcRef,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_DataRef,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync kCvSymType_Align
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @} */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync} RTCVSYMTYPE;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsynctypedef RTCVSYMTYPE *PRTCVSYMTYPE;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsynctypedef RTCVSYMTYPE const *PCRTCVSYMTYPE;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync/** The $$SYMBOL table signature for CV4. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync#define RTCVSYMBOLS_SIGNATURE_CV4 UINT32_C(0x00000001)
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync/**
8a339f91959bb7a3315b51a23461b68c7b0cb50evboxsync * File type.
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsynctypedef enum RTCVFILETYPE
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync{
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync RTCVFILETYPE_INVALID = 0,
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** Executable image. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync RTCVFILETYPE_IMAGE,
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** A DBG-file with a IMAGE_SEPARATE_DEBUG_HEADER. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync RTCVFILETYPE_DBG,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** A PDB file. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTCVFILETYPE_PDB,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** Some other kind of file with CV at the end. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTCVFILETYPE_OTHER_AT_END,
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** The end of the valid values. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTCVFILETYPE_END,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** Type blowup. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTCVFILETYPE_32BIT_HACK = 0x7fffffff
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync} RTCVFILETYPE;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync/**
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync * CodeView debug info reader instance.
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync */
0c4004948fca34f2db87e7b38013137e9472c306vboxsynctypedef struct RTDBGMODCV
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync{
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync /** Using a container for managing the debug info. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync RTDBGMOD hCnt;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** @name Codeview details
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync * @{ */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync /** The code view magic (used as format indicator). */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t u32CvMagic;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The offset of the CV debug info in the file. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t offBase;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The size of the CV debug info. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t cbDbgInfo;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** The offset of the subsection directory (relative to offBase). */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint32_t offDir;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** @} */
b0db50948c349fa76655abf252f7946b515e8204vboxsync
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** @name COFF details.
b0db50948c349fa76655abf252f7946b515e8204vboxsync * @{ */
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** Offset of the COFF header. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint32_t offCoffDbgInfo;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** The size of the COFF debug info. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint32_t cbCoffDbgInfo;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** The COFF debug info header. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync IMAGE_COFF_SYMBOLS_HEADER CoffHdr;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** @} */
b0db50948c349fa76655abf252f7946b515e8204vboxsync
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** The file type. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync RTCVFILETYPE enmType;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** The file handle (if external). */
b0db50948c349fa76655abf252f7946b515e8204vboxsync RTFILE hFile;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** Pointer to the module (no reference retained). */
b0db50948c349fa76655abf252f7946b515e8204vboxsync PRTDBGMODINT pMod;
b0db50948c349fa76655abf252f7946b515e8204vboxsync
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** The image size, if we know it. This is 0 if we don't know it. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint32_t cbImage;
b0db50948c349fa76655abf252f7946b515e8204vboxsync
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** Indicates that we've loaded segments intot he container already. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync bool fHaveLoadedSegments;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** @name Codeview Parsing state.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @{ */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /** Number of directory entries. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint32_t cDirEnts;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The directory (converted to 32-bit). */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync PRTCVDIRENT32 paDirEnts;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** Current debugging style when parsing modules. */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint16_t uCurStyle;
b0db50948c349fa76655abf252f7946b515e8204vboxsync /** Current debugging style version (HLL only). */
b0db50948c349fa76655abf252f7946b515e8204vboxsync uint16_t uCurStyleVer;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The segment map (if present). */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync PRTCVSEGMAP pSegMap;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** Segment names. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync char *pszzSegNames;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** The size of the segment names. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint32_t cbSegNames;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** @} */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync} RTDBGMODCV;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/** Pointer to a codeview debug info reader instance. */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsynctypedef RTDBGMODCV *PRTDBGMODCV;
faee255cc48bfbf17cb9f72fca70c8b9d3020ec4vboxsync/** Pointer to a const codeview debug info reader instance. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsynctypedef RTDBGMODCV *PCRTDBGMODCV;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Subsection callback.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * @returns IPRT status code.
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * @param pThis The CodeView debug info reader instance.
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * @param pvSubSect Pointer to the subsection data.
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync * @param cbSubSect The size of the subsection data.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param pDirEnt The directory entry.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsynctypedef DECLCALLBACK(int) FNDBGMODCVSUBSECTCALLBACK(PRTDBGMODCV pThis, void const *pvSubSect, size_t cbSubSect,
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync PCRTCVDIRENT32 pDirEnt);
a39ea3668b7019c23a68936259545f9b71bce1aavboxsync/** Pointer to a subsection callback. */
0db6a029780d9f9b347500e117320a8d5661efe5vboxsynctypedef FNDBGMODCVSUBSECTCALLBACK *PFNDBGMODCVSUBSECTCALLBACK;
da3503c04ce76e653401396fe2795a9bc2427a1dvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ee4d840f54fd2dcea8a73b1b86d5ec0db370b05dvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/*******************************************************************************
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync* Defined Constants And Macros *
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync*******************************************************************************/
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync/** Light weight assert + return w/ fixed status code. */
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync#define RTDBGMODCV_CHECK_RET_BF(a_Expr, a_LogArgs) \
0c4004948fca34f2db87e7b38013137e9472c306vboxsync do { \
0c4004948fca34f2db87e7b38013137e9472c306vboxsync if (!(a_Expr)) \
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync { \
0c4004948fca34f2db87e7b38013137e9472c306vboxsync Log(("RTDbgCv: Check failed on line %d: " #a_Expr "\n", __LINE__)); \
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync Log(a_LogArgs); \
0c4004948fca34f2db87e7b38013137e9472c306vboxsync return VERR_CV_BAD_FORMAT; \
0c4004948fca34f2db87e7b38013137e9472c306vboxsync } \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync } while (0)
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync/** Light weight assert + return w/ fixed status code. */
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync#define RTDBGMODCV_CHECK_NOMSG_RET_BF(a_Expr) \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync do { \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync if (!(a_Expr)) \
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync { \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync Log(("RTDbgCv: Check failed on line %d: " #a_Expr "\n", __LINE__)); \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync return VERR_CV_BAD_FORMAT; \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync } \
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync } while (0)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
0c4004948fca34f2db87e7b38013137e9472c306vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync/**
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * Reads CodeView information.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync *
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * @returns IPRT status.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * @param pThis The CodeView reader instance.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * @param off The offset to start reading at, relative to the
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * CodeView base header.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * @param pvBuf The buffer to read into.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * @param cb How many bytes to read.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync */
71f6a34b72f9cc873da208630959de49df1a28a5vboxsyncstatic int rtDbgModCvReadAt(PRTDBGMODCV pThis, uint32_t off, void *pvBuf, size_t cb)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync{
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync int rc;
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync if (pThis->hFile == NIL_RTFILE)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync rc = pThis->pMod->pImgVt->pfnReadAt(pThis->pMod, UINT32_MAX, off + pThis->offBase, pvBuf, cb);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync else
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync rc = RTFileReadAt(pThis->hFile, off + pThis->offBase, pvBuf, cb, NULL);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync return rc;
0c4004948fca34f2db87e7b38013137e9472c306vboxsync}
0c4004948fca34f2db87e7b38013137e9472c306vboxsync
0c4004948fca34f2db87e7b38013137e9472c306vboxsync
0c4004948fca34f2db87e7b38013137e9472c306vboxsync/**
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * Reads CodeView information into an allocated buffer.
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync *
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * @returns IPRT status.
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * @param pThis The CodeView reader instance.
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync * @param off The offset to start reading at, relative to the
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * CodeView base header.
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * @param ppvBuf Where to return the allocated buffer on success.
0c4004948fca34f2db87e7b38013137e9472c306vboxsync * @param cb How many bytes to read.
0c4004948fca34f2db87e7b38013137e9472c306vboxsync */
71f6a34b72f9cc873da208630959de49df1a28a5vboxsyncstatic int rtDbgModCvReadAtAlloc(PRTDBGMODCV pThis, uint32_t off, void **ppvBuf, size_t cb)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync{
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync int rc;
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync void *pvBuf = *ppvBuf = RTMemAlloc(cb);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync if (pvBuf)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync {
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync if (pThis->hFile == NIL_RTFILE)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync rc = pThis->pMod->pImgVt->pfnReadAt(pThis->pMod, UINT32_MAX, off + pThis->offBase, pvBuf, cb);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync else
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync rc = RTFileReadAt(pThis->hFile, off + pThis->offBase, pvBuf, cb, NULL);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync if (RT_SUCCESS(rc))
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync return VINF_SUCCESS;
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync RTMemFree(pvBuf);
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync *ppvBuf = NULL;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync }
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync else
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync rc = VERR_NO_MEMORY;
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync return rc;
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync}
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync
0c4004948fca34f2db87e7b38013137e9472c306vboxsync/**
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync * Gets a name string for a subsection type.
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync *
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync * @returns Section name (read only).
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync * @param uSubSectType The subsection type.
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync */
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsyncstatic const char *rtDbgModCvGetSubSectionName(uint16_t uSubSectType)
0c4004948fca34f2db87e7b38013137e9472c306vboxsync{
0c4004948fca34f2db87e7b38013137e9472c306vboxsync switch (uSubSectType)
0c4004948fca34f2db87e7b38013137e9472c306vboxsync {
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldModule: return "sstOldModule";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldPublic: return "sstOldPublic";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldTypes: return "sstOldTypes";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldSymbols: return "sstOldSymbols";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldSrcLines: return "sstOldSrcLines";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldLibraries: return "sstOldLibraries";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldImports: return "sstOldImports";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync case kCvSst_OldCompacted: return "sstOldCompacted";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_OldSrcLnSeg: return "sstOldSrcLnSeg";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_OldSrcLines3: return "sstOldSrcLines3";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_Module: return "sstModule";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_Types: return "sstTypes";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_Public: return "sstPublic";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_PublicSym: return "sstPublicSym";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_Symbols: return "sstSymbols";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_AlignSym: return "sstAlignSym";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_SrcLnSeg: return "sstSrcLnSeg";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_SrcModule: return "sstSrcModule";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_Libraries: return "sstLibraries";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_GlobalSym: return "sstGlobalSym";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_GlobalPub: return "sstGlobalPub";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_GlobalTypes: return "sstGlobalTypes";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_MPC: return "sstMPC";
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync case kCvSst_SegMap: return "sstSegMap";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_SegName: return "sstSegName";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_PreComp: return "sstPreComp";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_PreCompMap: return "sstPreCompMap";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_OffsetMap16: return "sstOffsetMap16";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_OffsetMap32: return "sstOffsetMap32";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_FileIndex: return "sstFileIndex";
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync case kCvSst_StaticSym: return "sstStaticSym";
0c4004948fca34f2db87e7b38013137e9472c306vboxsync }
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync static char s_sz[32];
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync RTStrPrintf(s_sz, sizeof(s_sz), "Unknown%#x", uSubSectType);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return s_sz;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync}
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/**
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Adds a symbol to the container.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @returns IPRT status code
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param pThis The CodeView debug info reader instance.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param iSeg Segment number.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param off Offset into the segment
e74eef731a813e4e06680c587a6759b9974b29c9vboxsync * @param pchName The symbol name (not necessarily terminated).
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param cchName The symbol name length.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param fFlags Flags reserved for future exploits, MBZ.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
8cd2f2e64725096acb682f34a5568b7fb816eda7vboxsyncstatic int rtDbgModCvAddSymbol(PRTDBGMODCV pThis, uint32_t iSeg, uint64_t off, const char *pchName,
8cd2f2e64725096acb682f34a5568b7fb816eda7vboxsync uint8_t cchName, uint32_t fFlags)
8cd2f2e64725096acb682f34a5568b7fb816eda7vboxsync{
e74eef731a813e4e06680c587a6759b9974b29c9vboxsync const char *pszName = RTStrCacheEnterN(g_hDbgModStrCache, pchName, cchName);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync if (!pszName)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return VERR_NO_STR_MEMORY;
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync#if 1
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync if (iSeg == 0)
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync iSeg = RTDBGSEGIDX_ABS;
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync else if (pThis->pSegMap)
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync {
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync if ( iSeg > pThis->pSegMap->Hdr.cSegs
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync || iSeg == 0
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync || off > pThis->pSegMap->aDescs[iSeg - 1].cb)
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync {
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync Log(("Invalid segment index/offset %#06x:%08x for symbol %.*s\n", iSeg, off, cchName, pchName));
26947320577c481b4afefdb0afbb855181e5b2e8vboxsync return VERR_CV_BAD_FORMAT;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync }
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync off += pThis->pSegMap->aDescs[iSeg - 1].off;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync if (pThis->pSegMap->aDescs[iSeg - 1].fFlags & RTCVSEGMAPDESC_F_ABS)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync iSeg = RTDBGSEGIDX_ABS;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync else
cba6719bd64ec749967bbe931230452664109857vboxsync iSeg = pThis->pSegMap->aDescs[iSeg - 1].iGroup;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync }
c28fa006ba669ad8f26ae31d00a338379c04ea1bvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync int rc = RTDbgModSymbolAdd(pThis->hCnt, pszName, iSeg, off, 0, 0 /*fFlags*/, NULL);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync Log(("Symbol: %04x:%08x %.*s [%Rrc]\n", iSeg, off, cchName, pchName, rc));
c28fa006ba669ad8f26ae31d00a338379c04ea1bvboxsync if (rc == VERR_DBG_ADDRESS_CONFLICT || rc == VERR_DBG_DUPLICATE_SYMBOL)
e74eef731a813e4e06680c587a6759b9974b29c9vboxsync rc = VINF_SUCCESS;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTStrCacheRelease(g_hDbgModStrCache, pszName);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return rc;
a8139954a84d6e9090dd3a8371aa788351d45bc3vboxsync#else
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync Log(("Symbol: %04x:%08x %.*s\n", iSeg, off, cchName, pchName));
a8139954a84d6e9090dd3a8371aa788351d45bc3vboxsync return VINF_SUCCESS;
a8139954a84d6e9090dd3a8371aa788351d45bc3vboxsync#endif
a8139954a84d6e9090dd3a8371aa788351d45bc3vboxsync}
a8139954a84d6e9090dd3a8371aa788351d45bc3vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync/**
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync * Parses a CV4 symbol table, adding symbols to the container.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync *
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @returns IPRT status code
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param pThis The CodeView debug info reader instance.
cba6719bd64ec749967bbe931230452664109857vboxsync * @param pbSymTab The symbol table.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param cbSymTab The size of the symbol table.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * @param fFlags Flags reserved for future exploits, MBZ.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsyncstatic int rtDbgModCvSsProcessV4SymTab(PRTDBGMODCV pThis, void const *pvSymTab, size_t cbSymTab, uint32_t fFlags)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync{
da3503c04ce76e653401396fe2795a9bc2427a1dvboxsync int rc = VINF_SUCCESS;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTCPTRUNION uCursor;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uCursor.pv = pvSymTab;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync while (cbSymTab > 0 && RT_SUCCESS(rc))
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync {
da3503c04ce76e653401396fe2795a9bc2427a1dvboxsync uint8_t const * const pbRecStart = uCursor.pu8;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t cbRec = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync if (cbRec >= 2)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync {
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t uSymType = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync Log3((" %p: uSymType=%#06x LB %#x\n", pbRecStart - (uint8_t *)pvSymTab, uSymType, cbRec));
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_RET_BF(cbRec >= 2 && cbRec <= cbSymTab, ("cbRec=%#x cbSymTab=%#x\n", cbRec, cbSymTab));
0db6a029780d9f9b347500e117320a8d5661efe5vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync switch (uSymType)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync {
da3503c04ce76e653401396fe2795a9bc2427a1dvboxsync case kCvSymType_LData16:
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync case kCvSymType_GData16:
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync case kCvSymType_Pub16:
0db6a029780d9f9b347500e117320a8d5661efe5vboxsync {
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbRec > 2 + 2+2+2+1);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t off = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t iSeg = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /*uint16_t iType =*/ *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint8_t cchName = *uCursor.pu8++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cchName > 0);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbRec >= 2 + 2+2+2+1 + cchName);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync rc = rtDbgModCvAddSymbol(pThis, iSeg, off, uCursor.pch, cchName, 0);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync break;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync }
da3503c04ce76e653401396fe2795a9bc2427a1dvboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync case kCvSymType_LData32:
cba6719bd64ec749967bbe931230452664109857vboxsync case kCvSymType_GData32:
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync case kCvSymType_Pub32:
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync {
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbRec > 2 + 4+2+2+1);
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync uint32_t off = *uCursor.pu32++;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint16_t iSeg = *uCursor.pu16++;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /*uint16_t iType =*/ *uCursor.pu16++;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync uint8_t cchName = *uCursor.pu8++;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cchName > 0);
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbRec >= 2 + 4+2+2+1 + cchName);
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync rc = rtDbgModCvAddSymbol(pThis, iSeg, off, uCursor.pch, cchName, 0);
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync break;
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync }
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync /** @todo add GProc and LProc so we can gather sizes as well as just symbols. */
ea779b55cc87f3e3fadddca4672c6697c82606edvboxsync }
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync }
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /*else: shorter records can be used for alignment, I guess. */
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync /* next */
9b789c281103a2489742bf32f6ab500e38b2ecd5vboxsync uCursor.pu8 = pbRecStart + cbRec + 2;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync cbSymTab -= cbRec + 2;
c91345f92b829d3fba05ce7a97206d83c5183ce0vboxsync }
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return rc;
c91345f92b829d3fba05ce7a97206d83c5183ce0vboxsync}
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync/** @callback_method_impl{FNDBGMODCVSUBSECTCALLBACK,
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync * Parses kCvSst_GlobalPub, kCvSst_GlobalSym and kCvSst_StaticSym subsections,
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync * adding symbols it finds to the container.} */
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsyncstatic DECLCALLBACK(int)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsyncrtDbgModCvSs_GlobalPub_GlobalSym_StaticSym(PRTDBGMODCV pThis, void const *pvSubSect, size_t cbSubSect, PCRTCVDIRENT32 pDirEnt)
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync{
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync PCRTCVGLOBALSYMTABHDR pHdr = (PCRTCVGLOBALSYMTABHDR)pvSubSect;
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync /*
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync * Quick data validation.
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync */
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync Log2(("RTDbgModCv: %s: uSymHash=%#x uAddrHash=%#x cbSymbols=%#x cbSymHash=%#x cbAddrHash=%#x\n",
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync rtDbgModCvGetSubSectionName(pDirEnt->uSubSectType), pHdr->uSymHash,
b0db50948c349fa76655abf252f7946b515e8204vboxsync pHdr->uAddrHash, pHdr->cbSymbols, pHdr->cbSymHash, pHdr->cbAddrHash));
b0db50948c349fa76655abf252f7946b515e8204vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbSubSect >= sizeof(RTCVGLOBALSYMTABHDR));
71f6a34b72f9cc873da208630959de49df1a28a5vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF((uint64_t)pHdr->cbSymbols + pHdr->cbSymHash + pHdr->cbAddrHash <= cbSubSect - sizeof(*pHdr));
b0db50948c349fa76655abf252f7946b515e8204vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(pHdr->uSymHash < 0x20);
b0db50948c349fa76655abf252f7946b515e8204vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(pHdr->uAddrHash < 0x20);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync if (!pHdr->cbSymbols)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return VINF_SUCCESS;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync /*
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Parse the symbols.
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return rtDbgModCvSsProcessV4SymTab(pThis, pHdr + 1, pHdr->cbSymbols, 0);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync}
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** @callback_method_impl{FNDBGMODCVSUBSECTCALLBACK,
5b465a7c1237993faf8bb50120d247f3f0319adavboxsync * Parses kCvSst_Module subsection, storing the debugging style in pThis.} */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsyncstatic DECLCALLBACK(int)
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsyncrtDbgModCvSs_Module(PRTDBGMODCV pThis, void const *pvSubSect, size_t cbSubSect, PCRTCVDIRENT32 pDirEnt)
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsync{
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsync RTCPTRUNION uCursor;
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsync uCursor.pv = pvSubSect;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbSubSect >= 2 + 2 + 2 + 2 + 0 + 1);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t iOverlay = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t iLib = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint16_t cSegs = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync pThis->uCurStyle = *uCursor.pu16++;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync if (pThis->uCurStyle == 0)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync pThis->uCurStyle = RT_MAKE_U16('C', 'V');
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync pThis->uCurStyleVer = 0;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync uint8_t cchName = uCursor.pu8[cSegs * 12];
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbSubSect >= 2 + 2 + 2 + 2 + cSegs * 12U + 1 + cchName);
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync const char *pchName = (const char *)&uCursor.pu8[cSegs * 12 + 1];
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync Log2(("RTDbgModCv: Module: iOverlay=%#x iLib=%#x cSegs=%#x Style=%c%c (%#x) %.*s\n", iOverlay, iLib, cSegs,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RT_BYTE1(pThis->uCurStyle), RT_BYTE2(pThis->uCurStyle), pThis->uCurStyle, cchName, pchName));
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(pThis->uCurStyle == RT_MAKE_U16('C', 'V'));
ad77e3ec3cde24263bc7537575f5cae442bee3b1vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync PCRTCVMODSEGINFO32 paSegs = (PCRTCVMODSEGINFO32)uCursor.pv;
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync for (uint16_t iSeg = 0; iSeg < cSegs; iSeg++)
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync Log2((" #%02u: %04x:%08x LB %08x\n", iSeg, paSegs[iSeg].iSeg, paSegs[iSeg].off, paSegs[iSeg].cb));
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync return VINF_SUCCESS;
d4e9ccea0ea1ed303b5708ff94f6c202755f0dc6vboxsync}
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync/** @callback_method_impl{FNDBGMODCVSUBSECTCALLBACK,
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync * Parses kCvSst_Symbols, kCvSst_PublicSym and kCvSst_AlignSym subsections,
cba6719bd64ec749967bbe931230452664109857vboxsync * adding symbols it finds to the container.} */
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsyncstatic DECLCALLBACK(int)
cba6719bd64ec749967bbe931230452664109857vboxsyncrtDbgModCvSs_Symbols_PublicSym_AlignSym(PRTDBGMODCV pThis, void const *pvSubSect, size_t cbSubSect, PCRTCVDIRENT32 pDirEnt)
d7125f3a1b435761c393f9ec406e85a73ae2a3e7vboxsync{
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(pThis->uCurStyle == RT_MAKE_U16('C', 'V'));
8f0fc87a72dee210b62acc9dd859a4bebf8bfb33vboxsync RTDBGMODCV_CHECK_NOMSG_RET_BF(cbSubSect >= 8);
cba6719bd64ec749967bbe931230452664109857vboxsync
cba6719bd64ec749967bbe931230452664109857vboxsync uint32_t u32Signature = *(uint32_t const *)pvSubSect;
cba6719bd64ec749967bbe931230452664109857vboxsync RTDBGMODCV_CHECK_RET_BF(u32Signature == RTCVSYMBOLS_SIGNATURE_CV4,
cba6719bd64ec749967bbe931230452664109857vboxsync ("%#x, expected %#x\n", u32Signature, RTCVSYMBOLS_SIGNATURE_CV4));
9dca051a5f8ff457ef1692990f6ecfa280daf265vboxsync
44a2ecaf2d0fc196ab76cab13b3f909299e386d1vboxsync return rtDbgModCvSsProcessV4SymTab(pThis, (uint8_t const *)pvSubSect + 4, cbSubSect - 4, 0);
}
static int rtDbgModCvLoadSegmentMap(PRTDBGMODCV pThis)
{
/*
* Search for the segment map and segment names. They will be at the end of the directory.
*/
uint32_t iSegMap = UINT32_MAX;
uint32_t iSegNames = UINT32_MAX;
uint32_t i = pThis->cDirEnts;
while (i-- > 0)
{
if ( pThis->paDirEnts[i].iMod != 0xffff
&& pThis->paDirEnts[i].iMod != 0x0000)
break;
if (pThis->paDirEnts[i].uSubSectType == kCvSst_SegMap)
iSegMap = i;
else if (pThis->paDirEnts[i].uSubSectType == kCvSst_SegName)
iSegNames = i;
}
if (iSegMap == UINT32_MAX)
{
Log(("RTDbgModCv: No segment map present, using segment indexes as is then...\n"));
return VINF_SUCCESS;
}
RTDBGMODCV_CHECK_RET_BF(pThis->paDirEnts[iSegMap].cb >= sizeof(RTCVSEGMAPHDR),
("Bad sstSegMap entry: cb=%#x\n", pThis->paDirEnts[iSegMap].cb));
RTDBGMODCV_CHECK_NOMSG_RET_BF(iSegNames == UINT32_MAX || pThis->paDirEnts[iSegNames].cb > 0);
/*
* Read them into memory.
*/
int rc = rtDbgModCvReadAtAlloc(pThis, pThis->paDirEnts[iSegMap].off, (void **)&pThis->pSegMap,
pThis->paDirEnts[iSegMap].cb);
if (iSegNames != UINT32_MAX && RT_SUCCESS(rc))
{
pThis->cbSegNames = pThis->paDirEnts[iSegNames].cb;
rc = rtDbgModCvReadAtAlloc(pThis, pThis->paDirEnts[iSegNames].off, (void **)&pThis->pszzSegNames,
pThis->paDirEnts[iSegNames].cb);
}
if (RT_FAILURE(rc))
return rc;
RTDBGMODCV_CHECK_NOMSG_RET_BF(!pThis->pszzSegNames || !pThis->pszzSegNames[pThis->cbSegNames - 1]); /* must be terminated */
/* Use local pointers to avoid lots of indirection and typing. */
PCRTCVSEGMAPHDR pHdr = &pThis->pSegMap->Hdr;
PRTCVSEGMAPDESC paDescs = &pThis->pSegMap->aDescs[0];
/*
* If there are only logical segments, assume a direct mapping.
* PE images, like the NT4 kernel, does it like this.
*/
bool const fNoGroups = pHdr->cSegs == pHdr->cLogSegs;
/*
* Validate and display it all.
*/
Log2(("RTDbgModCv: SegMap: cSegs=%#x cLogSegs=%#x (cbSegNames=%#x)\n", pHdr->cSegs, pHdr->cLogSegs, pThis->cbSegNames));
RTDBGMODCV_CHECK_RET_BF(pThis->paDirEnts[iSegMap].cb >= sizeof(*pHdr) + pHdr->cSegs * sizeof(paDescs[0]),
("SegMap is out of bounds: cbSubSect=%#x cSegs=%#x\n", pThis->paDirEnts[iSegMap].cb, pHdr->cSegs));
RTDBGMODCV_CHECK_NOMSG_RET_BF(pHdr->cSegs >= pHdr->cLogSegs);
Log2(("Logical segment descriptors: %u\n", pHdr->cLogSegs));
for (i = 0; i < pHdr->cSegs; i++)
{
if (i == pHdr->cLogSegs)
Log2(("Group/Physical descriptors: %u\n", pHdr->cSegs - pHdr->cLogSegs));
uint16_t idx = i < pHdr->cLogSegs ? i : i - pHdr->cLogSegs;
char szFlags[16];
memset(szFlags, '-', sizeof(szFlags));
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_READ)
szFlags[0] = 'R';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_WRITE)
szFlags[1] = 'W';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_EXECUTE)
szFlags[2] = 'X';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_32BIT)
szFlags[3] = '3', szFlags[4] = '2';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_SEL)
szFlags[5] = 'S';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_ABS)
szFlags[6] = 'A';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_GROUP)
szFlags[7] = 'G';
szFlags[8] = '\0';
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_RESERVED)
szFlags[8] = '!', szFlags[9] = '\0';
Log2((" #%02u: %#010x LB %#010x flags=%#06x ovl=%#06x group=%#06x frame=%#06x iSegName=%#06x iClassName=%#06x %s\n",
idx, paDescs[i].off, paDescs[i].cb, paDescs[i].fFlags, paDescs[i].iOverlay, paDescs[i].iGroup,
paDescs[i].iFrame, paDescs[i].offSegName, paDescs[i].offClassName, szFlags));
RTDBGMODCV_CHECK_NOMSG_RET_BF(paDescs[i].offSegName == UINT16_MAX || paDescs[i].offSegName < pThis->cbSegNames);
RTDBGMODCV_CHECK_NOMSG_RET_BF(paDescs[i].offClassName == UINT16_MAX || paDescs[i].offClassName < pThis->cbSegNames);
const char *pszName = paDescs[i].offSegName != UINT16_MAX
? pThis->pszzSegNames + paDescs[i].offSegName
: NULL;
const char *pszClass = paDescs[i].offClassName != UINT16_MAX
? pThis->pszzSegNames + paDescs[i].offClassName
: NULL;
if (pszName || pszClass)
Log2((" pszName=%s pszClass=%s\n", pszName, pszClass));
/* Validate the group link. */
RTDBGMODCV_CHECK_NOMSG_RET_BF(paDescs[i].iGroup == 0 || !(paDescs[i].fFlags & RTCVSEGMAPDESC_F_GROUP));
RTDBGMODCV_CHECK_NOMSG_RET_BF( paDescs[i].iGroup == 0
|| ( paDescs[i].iGroup >= pHdr->cLogSegs
&& paDescs[i].iGroup < pHdr->cSegs));
RTDBGMODCV_CHECK_NOMSG_RET_BF( paDescs[i].iGroup == 0
|| (paDescs[paDescs[i].iGroup].fFlags & RTCVSEGMAPDESC_F_GROUP));
RTDBGMODCV_CHECK_NOMSG_RET_BF(!(paDescs[i].fFlags & RTCVSEGMAPDESC_F_GROUP) || paDescs[i].off == 0); /* assumed below */
if (fNoGroups)
{
RTDBGMODCV_CHECK_NOMSG_RET_BF(paDescs[i].iGroup == 0);
RTDBGMODCV_CHECK_NOMSG_RET_BF(paDescs[i].off == 0);
}
}
/*
* Modify the groups index to be the loader segment index instead, also
* add the segments to the container if we haven't done that already.
*/
/* Guess work: Group can be implicit if used. Observed Visual C++ v1.5,
omitting the CODE group. */
const char *pszGroup0 = NULL;
uint64_t cbGroup0 = 0;
if (!fNoGroups)
{
for (i = 0; i < pHdr->cSegs; i++)
if ( !(paDescs[i].fFlags & (RTCVSEGMAPDESC_F_GROUP | RTCVSEGMAPDESC_F_ABS))
&& paDescs[i].iGroup == 0)
{
if (pszGroup0 == NULL && paDescs[i].offClassName != UINT16_MAX)
pszGroup0 = pThis->pszzSegNames + paDescs[i].offClassName;
uint64_t offEnd = (uint64_t)paDescs[i].off + paDescs[i].cb;
if (offEnd > cbGroup0)
cbGroup0 = offEnd;
}
}
/* Add the segments.
Note! The RVAs derived from this exercise are all wrong. :-/
Note! We don't have an image loader, so we cannot add any fake sections. */
/** @todo Try see if we can figure something out from the frame value later. */
if (!pThis->fHaveLoadedSegments)
{
Assert(!pThis->pMod->pImgVt); Assert(pThis->enmType != RTCVFILETYPE_DBG);
uint16_t iSeg = 0;
uint64_t uRva = 0;
if (cbGroup0 && !fNoGroups)
{
rc = RTDbgModSegmentAdd(pThis->hCnt, 0, cbGroup0, pszGroup0 ? pszGroup0 : "Seg00", 0 /*fFlags*/, NULL);
uRva += cbGroup0;
iSeg++;
}
for (i = 0; RT_SUCCESS(rc) && i < pHdr->cSegs; i++)
if ((paDescs[i].fFlags & RTCVSEGMAPDESC_F_GROUP) || fNoGroups)
{
char szName[16];
char *pszName = szName;
if (paDescs[i].offSegName != UINT16_MAX)
pszName = pThis->pszzSegNames + paDescs[i].offSegName;
else
RTStrPrintf(szName, sizeof(szName), "Seg%02u", iSeg);
rc = RTDbgModSegmentAdd(pThis->hCnt, uRva, paDescs[i].cb, pszName, 0 /*fFlags*/, NULL);
uRva += paDescs[i].cb;
iSeg++;
}
if (RT_FAILURE(rc))
{
Log(("RTDbgModCv: %Rrc while adding segments from SegMap\n", rc));
return rc;
}
pThis->fHaveLoadedSegments = true;
}
/* The PE image has an extra section/segment for the headers, the others doesn't. */
RTLDRFMT enmImgFmt = RTLDRFMT_INVALID;
if (pThis->pMod->pImgVt)
enmImgFmt = pThis->pMod->pImgVt->pfnGetFormat(pThis->pMod);
/* Pass one: Fixate the group segment indexes. */
uint16_t iSeg0 = enmImgFmt == RTLDRFMT_PE || pThis->enmType == RTCVFILETYPE_DBG ? 1 : 0;
uint16_t iSeg = iSeg0 + cbGroup0 > 0;
for (i = 0; i < pHdr->cSegs; i++)
if (paDescs[i].fFlags & RTCVSEGMAPDESC_F_ABS)
paDescs[i].iGroup = (uint16_t)RTDBGSEGIDX_ABS;
else if ((paDescs[i].fFlags & RTCVSEGMAPDESC_F_GROUP) || fNoGroups)
paDescs[i].iGroup = iSeg++;
/* Pass two: Resolve group references in to segment indexes. */
Log2(("Mapped segments (both kinds):\n"));
for (i = 0; i < pHdr->cSegs; i++)
{
if (!fNoGroups && !(paDescs[i].fFlags & (RTCVSEGMAPDESC_F_GROUP | RTCVSEGMAPDESC_F_ABS)))
paDescs[i].iGroup = paDescs[i].iGroup == 0 ? iSeg0 : paDescs[paDescs[i].iGroup].iGroup;
Log2((" #%02u: %#010x LB %#010x -> %#06x (flags=%#06x ovl=%#06x frame=%#06x)\n",
i, paDescs[i].off, paDescs[i].cb, paDescs[i].iGroup,
paDescs[i].fFlags, paDescs[i].iOverlay, paDescs[i].iFrame));
}
return VINF_SUCCESS;
}
/**
* Loads the directory into memory (RTDBGMODCV::paDirEnts and
* RTDBGMODCV::cDirEnts).
*
* Converting old format version into the newer format to simplifying the code
* using the directory.
*
*
* @returns IPRT status code. (May leave with paDirEnts allocated on failure.)
* @param pThis The CV reader instance.
*/
static int rtDbgModCvLoadDirectory(PRTDBGMODCV pThis)
{
/*
* Read in the CV directory.
*/
int rc;
if ( pThis->u32CvMagic == RTCVHDR_MAGIC_NB00
|| pThis->u32CvMagic == RTCVHDR_MAGIC_NB02)
{
/*
* 16-bit type.
*/
RTCVDIRHDR16 DirHdr;
rc = rtDbgModCvReadAt(pThis, pThis->offDir, &DirHdr, sizeof(DirHdr));
if (RT_SUCCESS(rc))
{
if (DirHdr.cEntries > 2 && DirHdr.cEntries < _64K - 32U)
{
pThis->cDirEnts = DirHdr.cEntries;
pThis->paDirEnts = (PRTCVDIRENT32)RTMemAlloc(DirHdr.cEntries * sizeof(pThis->paDirEnts[0]));
if (pThis->paDirEnts)
{
rc = rtDbgModCvReadAt(pThis, pThis->offDir + sizeof(DirHdr),
pThis->paDirEnts, DirHdr.cEntries * sizeof(RTCVDIRENT16));
if (RT_SUCCESS(rc))
{
/* Convert the entries (from the end). */
uint32_t cLeft = DirHdr.cEntries;
RTCVDIRENT32 volatile *pDst = pThis->paDirEnts + cLeft;
RTCVDIRENT16 volatile *pSrc = (RTCVDIRENT16 volatile *)pThis->paDirEnts + cLeft;
while (cLeft--)
{
pDst--;
pSrc--;
pDst->cb = pSrc->cb;
pDst->off = RT_MAKE_U32(pSrc->offLow, pSrc->offHigh);
pDst->iMod = pSrc->iMod;
pDst->uSubSectType = pSrc->uSubSectType;
}
}
}
else
rc = VERR_NO_MEMORY;
}
else
{
Log(("Old CV directory count is out of considered valid range: %#x\n", DirHdr.cEntries));
rc = VERR_CV_BAD_FORMAT;
}
}
}
else
{
/*
* 32-bit type (reading too much for NB04 is no problem).
*/
RTCVDIRHDR32EX DirHdr;
rc = rtDbgModCvReadAt(pThis, pThis->offDir, &DirHdr, sizeof(DirHdr));
if (RT_SUCCESS(rc))
{
if ( DirHdr.Core.cbHdr != sizeof(DirHdr.Core)
&& DirHdr.Core.cbHdr != sizeof(DirHdr))
{
Log(("Unexpected CV directory size: %#x\n", DirHdr.Core.cbHdr));
rc = VERR_CV_BAD_FORMAT;
}
if ( DirHdr.Core.cbHdr == sizeof(DirHdr)
&& ( DirHdr.offNextDir != 0
|| DirHdr.fFlags != 0) )
{
Log(("Extended CV directory headers fields are not zero: fFlags=%#x offNextDir=%#x\n",
DirHdr.fFlags, DirHdr.offNextDir));
rc = VERR_CV_BAD_FORMAT;
}
if (DirHdr.Core.cbEntry != sizeof(RTCVDIRENT32))
{
Log(("Unexpected CV directory entry size: %#x (expected %#x)\n", DirHdr.Core.cbEntry, sizeof(RTCVDIRENT32)));
rc = VERR_CV_BAD_FORMAT;
}
if (DirHdr.Core.cEntries < 2 || DirHdr.Core.cEntries >= _512K)
{
Log(("CV directory count is out of considered valid range: %#x\n", DirHdr.Core.cEntries));
rc = VERR_CV_BAD_FORMAT;
}
if (RT_SUCCESS(rc))
{
pThis->cDirEnts = DirHdr.Core.cEntries;
pThis->paDirEnts = (PRTCVDIRENT32)RTMemAlloc(DirHdr.Core.cEntries * sizeof(pThis->paDirEnts[0]));
if (pThis->paDirEnts)
rc = rtDbgModCvReadAt(pThis, pThis->offDir + DirHdr.Core.cbHdr,
pThis->paDirEnts, DirHdr.Core.cEntries * sizeof(RTCVDIRENT32));
else
rc = VERR_NO_MEMORY;
}
}
}
/*
* Validate the information in the directory a little.
*/
if (RT_SUCCESS(rc))
{
uint16_t iMod = 0;
uint32_t const cbDbgInfo = pThis->cbDbgInfo;
uint32_t const cDirEnts = pThis->cDirEnts;
Log2(("RTDbgModCv: %u (%#x) directory entries:\n", cDirEnts, cDirEnts));
for (uint32_t i = 0; i < cDirEnts; i++)
{
PCRTCVDIRENT32 pDirEnt = &pThis->paDirEnts[i];
Log2((" #%04u mod=%#06x sst=%#06x at %#010x LB %#07x %s\n",
i, pDirEnt->iMod, pDirEnt->uSubSectType, pDirEnt->off, pDirEnt->cb,
rtDbgModCvGetSubSectionName(pDirEnt->uSubSectType)));
if ( pDirEnt->off >= cbDbgInfo
|| pDirEnt->cb >= cbDbgInfo
|| pDirEnt->off + pDirEnt->cb > cbDbgInfo)
{
Log(("CV directory entry #%u is out of bounds: %#x LB %#x, max %#x\n", i, pDirEnt->off, pDirEnt->cb, cbDbgInfo));
rc = VERR_CV_BAD_FORMAT;
}
if ( pDirEnt->iMod == 0
&& pThis->u32CvMagic != RTCVHDR_MAGIC_NB04
&& pThis->u32CvMagic != RTCVHDR_MAGIC_NB02
&& pThis->u32CvMagic != RTCVHDR_MAGIC_NB00)
{
Log(("CV directory entry #%u uses module index 0 (uSubSectType=%#x)\n", i, pDirEnt->iMod, pDirEnt->uSubSectType));
rc = VERR_CV_BAD_FORMAT;
}
if ( pDirEnt->iMod < iMod
&& ( pDirEnt->iMod != 0
|| ( pThis->u32CvMagic != RTCVHDR_MAGIC_NB00 /* May be first, maybe last. */
&& pThis->u32CvMagic != RTCVHDR_MAGIC_NB02
&& pThis->u32CvMagic != RTCVHDR_MAGIC_NB04) ) )
{
Log(("CV directory entry #%u is out of module order, this mod %u, prev mod %#u\n", i, pDirEnt->iMod, iMod));
rc = VERR_CV_BAD_FORMAT;
}
if (pDirEnt->iMod != iMod)
{
iMod = pDirEnt->iMod;
if ( iMod != 0
&& iMod != 0xffff
&& pDirEnt->uSubSectType != kCvSst_Module
&& pDirEnt->uSubSectType != kCvSst_OldModule)
{
Log(("CV directory entry #%u: expected module subsection first, found %s (%#x)\n",
i, rtDbgModCvGetSubSectionName(pDirEnt->uSubSectType), pDirEnt->uSubSectType));
rc = VERR_CV_BAD_FORMAT;
}
}
}
}
return rc;
}
static int rtDbgModCvLoadCodeViewInfo(PRTDBGMODCV pThis)
{
/*
* Load the directory, the segment map (if any) and then scan for segments
* if necessary.
*/
int rc = rtDbgModCvLoadDirectory(pThis);
if (RT_SUCCESS(rc))
rc = rtDbgModCvLoadSegmentMap(pThis);
if (RT_SUCCESS(rc) && !pThis->fHaveLoadedSegments)
{
rc = VERR_CV_TODO; /** @todo Scan anything containing address, in particular sstSegMap and sstModule,
* and reconstruct the segments from that information. */
pThis->cbImage = 0x1000;
rc = VINF_SUCCESS;
}
/*
* Process the directory.
*/
for (uint32_t i = 0; RT_SUCCESS(rc) && i < pThis->cDirEnts; i++)
{
PCRTCVDIRENT32 pDirEnt = &pThis->paDirEnts[i];
Log3(("Processing subsection %#u %s\n", i, rtDbgModCvGetSubSectionName(pDirEnt->uSubSectType)));
PFNDBGMODCVSUBSECTCALLBACK pfnCallback = NULL;
switch (pDirEnt->uSubSectType)
{
case kCvSst_GlobalPub:
case kCvSst_GlobalSym:
case kCvSst_StaticSym:
pfnCallback = rtDbgModCvSs_GlobalPub_GlobalSym_StaticSym;
break;
case kCvSst_Module:
pfnCallback = rtDbgModCvSs_Module;
break;
case kCvSst_PublicSym:
case kCvSst_Symbols:
case kCvSst_AlignSym:
pfnCallback = rtDbgModCvSs_Symbols_PublicSym_AlignSym;
break;
case kCvSst_OldModule:
case kCvSst_OldPublic:
case kCvSst_OldTypes:
case kCvSst_OldSymbols:
case kCvSst_OldSrcLines:
case kCvSst_OldLibraries:
case kCvSst_OldImports:
case kCvSst_OldCompacted:
case kCvSst_OldSrcLnSeg:
case kCvSst_OldSrcLines3:
case kCvSst_Types:
case kCvSst_Public:
case kCvSst_SrcLnSeg:
case kCvSst_SrcModule:
case kCvSst_Libraries:
case kCvSst_GlobalTypes:
case kCvSst_MPC:
case kCvSst_PreComp:
case kCvSst_PreCompMap:
case kCvSst_OffsetMap16:
case kCvSst_OffsetMap32:
case kCvSst_FileIndex:
default:
/** @todo implement more. */
break;
/* Skip because we've already processed them: */
case kCvSst_SegMap:
case kCvSst_SegName:
pfnCallback = NULL;
break;
}
if (pfnCallback)
{
void *pvSubSect;
rc = rtDbgModCvReadAtAlloc(pThis, pDirEnt->off, &pvSubSect, pDirEnt->cb);
if (RT_SUCCESS(rc))
{
rc = pfnCallback(pThis, pvSubSect, pDirEnt->cb, pDirEnt);
RTMemFree(pvSubSect);
}
}
}
return rc;
}
/*
*
* COFF Debug Info Parsing.
* COFF Debug Info Parsing.
* COFF Debug Info Parsing.
*
*/
static const char *rtDbgModCvGetCoffStorageClassName(uint8_t bStorageClass)
{
switch (bStorageClass)
{
case IMAGE_SYM_CLASS_END_OF_FUNCTION: return "END_OF_FUNCTION";
case IMAGE_SYM_CLASS_NULL: return "NULL";
case IMAGE_SYM_CLASS_AUTOMATIC: return "AUTOMATIC";
case IMAGE_SYM_CLASS_EXTERNAL: return "EXTERNAL";
case IMAGE_SYM_CLASS_STATIC: return "STATIC";
case IMAGE_SYM_CLASS_REGISTER: return "REGISTER";
case IMAGE_SYM_CLASS_EXTERNAL_DEF: return "EXTERNAL_DEF";
case IMAGE_SYM_CLASS_LABEL: return "LABEL";
case IMAGE_SYM_CLASS_UNDEFINED_LABEL: return "UNDEFINED_LABEL";
case IMAGE_SYM_CLASS_MEMBER_OF_STRUCT: return "MEMBER_OF_STRUCT";
case IMAGE_SYM_CLASS_ARGUMENT: return "ARGUMENT";
case IMAGE_SYM_CLASS_STRUCT_TAG: return "STRUCT_TAG";
case IMAGE_SYM_CLASS_MEMBER_OF_UNION: return "MEMBER_OF_UNION";
case IMAGE_SYM_CLASS_UNION_TAG: return "UNION_TAG";
case IMAGE_SYM_CLASS_TYPE_DEFINITION: return "TYPE_DEFINITION";
case IMAGE_SYM_CLASS_UNDEFINED_STATIC: return "UNDEFINED_STATIC";
case IMAGE_SYM_CLASS_ENUM_TAG: return "ENUM_TAG";
case IMAGE_SYM_CLASS_MEMBER_OF_ENUM: return "MEMBER_OF_ENUM";
case IMAGE_SYM_CLASS_REGISTER_PARAM: return "REGISTER_PARAM";
case IMAGE_SYM_CLASS_BIT_FIELD: return "BIT_FIELD";
case IMAGE_SYM_CLASS_FAR_EXTERNAL: return "FAR_EXTERNAL";
case IMAGE_SYM_CLASS_BLOCK: return "BLOCK";
case IMAGE_SYM_CLASS_FUNCTION: return "FUNCTION";
case IMAGE_SYM_CLASS_END_OF_STRUCT: return "END_OF_STRUCT";
case IMAGE_SYM_CLASS_FILE: return "FILE";
case IMAGE_SYM_CLASS_SECTION: return "SECTION";
case IMAGE_SYM_CLASS_WEAK_EXTERNAL: return "WEAK_EXTERNAL";
case IMAGE_SYM_CLASS_CLR_TOKEN: return "CLR_TOKEN";
}
static char s_szName[32];
RTStrPrintf(s_szName, sizeof(s_szName), "Unknown%#04x", bStorageClass);
return s_szName;
}
/**
* Adds a chunk of COFF line numbers.
*
* @param pThis The COFF/CodeView reader instance.
* @param pszFile The source file name.
* @param iSection The section number.
* @param paLines Pointer to the first line number table entry.
* @param cLines The number of line number table entries to add.
*/
static void rtDbgModCvAddCoffLineNumbers(PRTDBGMODCV pThis, const char *pszFile, uint32_t iSection,
PCIMAGE_LINENUMBER paLines, uint32_t cLines)
{
Log4(("Adding %u line numbers in section #%u for %s\n", cLines, iSection, pszFile));
PCIMAGE_LINENUMBER pCur = paLines;
while (cLines-- > 0)
{
if (pCur->Linenumber)
{
int rc = RTDbgModLineAdd(pThis->hCnt, pszFile, pCur->Linenumber, RTDBGSEGIDX_RVA, pCur->Type.VirtualAddress, NULL);
Log4((" %#010x: %u [%Rrc]\n", pCur->Type.VirtualAddress, pCur->Linenumber, rc));
}
pCur++;
}
}
/**
* Adds a COFF symbol.
*
* @returns IPRT status (ignored)
* @param pThis The COFF/CodeView reader instance.
* @param idxSeg IPRT RVA or ABS segment index indicator.
* @param uValue The symbol value.
* @param pszName The symbol name.
*/
static int rtDbgModCvAddCoffSymbol(PRTDBGMODCV pThis, uint32_t idxSeg, uint32_t uValue, const char *pszName)
{
int rc = RTDbgModSymbolAdd(pThis->hCnt, pszName, idxSeg, uValue, 0, 0 /*fFlags*/, NULL);
Log(("Symbol: %s:%08x %s [%Rrc]\n", idxSeg == RTDBGSEGIDX_RVA ? "rva" : "abs", uValue, pszName, rc));
if (rc == VERR_DBG_ADDRESS_CONFLICT || rc == VERR_DBG_DUPLICATE_SYMBOL)
rc = VINF_SUCCESS;
return rc;
}
/**
* Processes the COFF symbol table.
*
* @returns IPRT status code
* @param pThis The COFF/CodeView reader instance.
* @param paSymbols Pointer to the symbol table.
* @param cSymbols The number of entries in the symbol table.
* @param paLines Pointer to the line number table.
* @param cLines The number of entires in the line number table.
* @param pszzStrTab Pointer to the string table.
* @param cbStrTab Size of the string table.
*/
static int rtDbgModCvProcessCoffSymbolTable(PRTDBGMODCV pThis,
PCIMAGE_SYMBOL paSymbols, uint32_t cSymbols,
PCIMAGE_LINENUMBER paLines, uint32_t cLines,
const char *pszzStrTab, uint32_t cbStrTab)
{
Log3(("Processing COFF symbol table with %#x symbols\n", cSymbols));
/*
* Making some bold assumption that the line numbers for the section in
* the file are allocated sequentially, we do multiple passes until we've
* gathered them all.
*/
int rc = VINF_SUCCESS;
uint32_t cSections = 1;
uint32_t iLineSect = 1;
uint32_t iLine = 0;
do
{
/*
* Process the symbols.
*/
char szShort[9];
char szFile[RTPATH_MAX];
uint32_t iSymbol = 0;
szFile[0] = '\0';
szShort[8] = '\0'; /* avoid having to terminate it all the time. */
while (iSymbol < cSymbols && RT_SUCCESS(rc))
{
/* Copy the symbol in and hope it works around the misalignment
issues everywhere. */
IMAGE_SYMBOL Sym;
memcpy(&Sym, &paSymbols[iSymbol], sizeof(Sym));
RTDBGMODCV_CHECK_NOMSG_RET_BF(Sym.NumberOfAuxSymbols < cSymbols);
/* Calc a zero terminated symbol name. */
const char *pszName;
if (Sym.N.Name.Short)
pszName = (const char *)memcpy(szShort, &Sym.N, 8);
else
{
RTDBGMODCV_CHECK_NOMSG_RET_BF(Sym.N.Name.Long < cbStrTab);
pszName = pszzStrTab + Sym.N.Name.Long;
}
/* Only log stuff and count sections the in the first pass.*/
if (iLineSect == 1)
{
Log3(("%04x: s=%#06x v=%#010x t=%#06x a=%#04x c=%#04x (%s) name='%s'\n",
iSymbol, Sym.SectionNumber, Sym.Value, Sym.Type, Sym.NumberOfAuxSymbols,
Sym.StorageClass, rtDbgModCvGetCoffStorageClassName(Sym.StorageClass), pszName));
if ((int16_t)cSections <= Sym.SectionNumber && Sym.SectionNumber > 0)
cSections = Sym.SectionNumber + 1;
}
/*
* Use storage class to pick what we need (which isn't much because,
* MS only provides a very restricted set of symbols).
*/
IMAGE_AUX_SYMBOL Aux;
switch (Sym.StorageClass)
{
case IMAGE_SYM_CLASS_NULL:
/* a NOP */
break;
case IMAGE_SYM_CLASS_FILE:
{
/* Change the current file name (for line numbers). Pretend
ANSI and ISO-8859-1 are similar enough for out purposes... */
RTDBGMODCV_CHECK_NOMSG_RET_BF(Sym.NumberOfAuxSymbols > 0);
const char *pszFile = (const char *)&paSymbols[iSymbol + 1];
char *pszDst = szFile;
rc = RTLatin1ToUtf8Ex(pszFile, Sym.NumberOfAuxSymbols * sizeof(IMAGE_SYMBOL), &pszDst, sizeof(szFile), NULL);
if (RT_FAILURE(rc))
Log(("Error converting COFF filename: %Rrc\n", rc));
else if (iLineSect == 1)
Log3((" filename='%s'\n", szFile));
break;
}
case IMAGE_SYM_CLASS_STATIC:
if ( Sym.NumberOfAuxSymbols == 1
&& ( iLineSect == 1
|| Sym.SectionNumber == (int32_t)iLineSect) )
{
memcpy(&Aux, &paSymbols[iSymbol + 1], sizeof(Aux));
if (iLineSect == 1)
Log3((" section: cb=%#010x #relocs=%#06x #lines=%#06x csum=%#x num=%#x sel=%x rvd=%u\n",
Aux.Section.Length, Aux.Section.NumberOfRelocations,
Aux.Section.NumberOfLinenumbers,
Aux.Section.CheckSum,
RT_MAKE_U32(Aux.Section.Number, Aux.Section.HighNumber),
Aux.Section.Selection,
Aux.Section.bReserved));
if ( Sym.SectionNumber == (int32_t)iLineSect
&& Aux.Section.NumberOfLinenumbers > 0)
{
uint32_t cLinesToAdd = RT_MIN(Aux.Section.NumberOfLinenumbers, cLines - iLine);
if (iLine < cLines && szFile[0])
rtDbgModCvAddCoffLineNumbers(pThis, szFile, iLineSect, &paLines[iLine], cLinesToAdd);
iLine += cLinesToAdd;
}
}
/* Not so sure about the quality here, but might be useful. */
else if ( iLineSect == 1
&& Sym.NumberOfAuxSymbols == 0
&& Sym.SectionNumber != IMAGE_SYM_UNDEFINED
&& Sym.SectionNumber != IMAGE_SYM_ABSOLUTE
&& Sym.SectionNumber != IMAGE_SYM_DEBUG
&& Sym.Value > 0
&& *pszName)
rtDbgModCvAddCoffSymbol(pThis, RTDBGSEGIDX_RVA, Sym.Value, pszName);
break;
case IMAGE_SYM_CLASS_EXTERNAL:
/* Add functions (first pass only). */
if ( iLineSect == 1
&& (ISFCN(Sym.Type) || Sym.Type == 0)
&& Sym.NumberOfAuxSymbols == 0
&& *pszName )
{
if (Sym.SectionNumber == IMAGE_SYM_ABSOLUTE)
rtDbgModCvAddCoffSymbol(pThis, RTDBGSEGIDX_ABS, Sym.Value, pszName);
else if ( Sym.SectionNumber != IMAGE_SYM_UNDEFINED
&& Sym.SectionNumber != IMAGE_SYM_DEBUG)
rtDbgModCvAddCoffSymbol(pThis, RTDBGSEGIDX_RVA, Sym.Value, pszName);
}
break;
case IMAGE_SYM_CLASS_FUNCTION:
/* Not sure this is really used. */
break;
case IMAGE_SYM_CLASS_END_OF_FUNCTION:
case IMAGE_SYM_CLASS_AUTOMATIC:
case IMAGE_SYM_CLASS_REGISTER:
case IMAGE_SYM_CLASS_EXTERNAL_DEF:
case IMAGE_SYM_CLASS_LABEL:
case IMAGE_SYM_CLASS_UNDEFINED_LABEL:
case IMAGE_SYM_CLASS_MEMBER_OF_STRUCT:
case IMAGE_SYM_CLASS_ARGUMENT:
case IMAGE_SYM_CLASS_STRUCT_TAG:
case IMAGE_SYM_CLASS_MEMBER_OF_UNION:
case IMAGE_SYM_CLASS_UNION_TAG:
case IMAGE_SYM_CLASS_TYPE_DEFINITION:
case IMAGE_SYM_CLASS_UNDEFINED_STATIC:
case IMAGE_SYM_CLASS_ENUM_TAG:
case IMAGE_SYM_CLASS_MEMBER_OF_ENUM:
case IMAGE_SYM_CLASS_REGISTER_PARAM:
case IMAGE_SYM_CLASS_BIT_FIELD:
case IMAGE_SYM_CLASS_FAR_EXTERNAL:
case IMAGE_SYM_CLASS_BLOCK:
case IMAGE_SYM_CLASS_END_OF_STRUCT:
case IMAGE_SYM_CLASS_SECTION:
case IMAGE_SYM_CLASS_WEAK_EXTERNAL:
case IMAGE_SYM_CLASS_CLR_TOKEN:
/* Not used by MS, I think. */
break;
default:
Log(("RTDbgCv: Unexpected COFF storage class %#x (%u)\n", Sym.StorageClass, Sym.StorageClass));
break;
}
/* next symbol */
iSymbol += 1 + Sym.NumberOfAuxSymbols;
}
/* Next section with line numbers. */
iLineSect++;
} while (iLine < cLines && iLineSect < cSections && RT_SUCCESS(rc));
return rc;
}
/**
* Loads COFF debug information into the container.
*
* @returns IPRT status code.
* @param pThis The COFF/CodeView debug reader instance.
*/
static int rtDbgModCvLoadCoffInfo(PRTDBGMODCV pThis)
{
/*
* Read the whole section into memory.
* Note! Cannot use rtDbgModCvReadAt or rtDbgModCvReadAtAlloc here.
*/
int rc;
uint8_t *pbDbgSect = (uint8_t *)RTMemAlloc(pThis->cbCoffDbgInfo);
if (pbDbgSect)
{
if (pThis->hFile == NIL_RTFILE)
rc = pThis->pMod->pImgVt->pfnReadAt(pThis->pMod, UINT32_MAX, pThis->offCoffDbgInfo, pbDbgSect, pThis->cbCoffDbgInfo);
else
rc = RTFileReadAt(pThis->hFile, pThis->offCoffDbgInfo, pbDbgSect, pThis->cbCoffDbgInfo, NULL);
if (RT_SUCCESS(rc))
{
/* The string table follows after the symbol table. */
const char *pszzStrTab = (const char *)( pbDbgSect
+ pThis->CoffHdr.LvaToFirstSymbol
+ pThis->CoffHdr.NumberOfSymbols * sizeof(IMAGE_SYMBOL));
uint32_t cbStrTab = (uint32_t)((uintptr_t)(pbDbgSect + pThis->cbCoffDbgInfo) - (uintptr_t)pszzStrTab);
/** @todo The symbol table starts with a size. Read it and checking. Also verify
* that the symtab ends with a terminator character. */
rc = rtDbgModCvProcessCoffSymbolTable(pThis,
(PCIMAGE_SYMBOL)(pbDbgSect + pThis->CoffHdr.LvaToFirstSymbol),
pThis->CoffHdr.NumberOfSymbols,
(PCIMAGE_LINENUMBER)(pbDbgSect + pThis->CoffHdr.LvaToFirstLinenumber),
pThis->CoffHdr.NumberOfLinenumbers,
pszzStrTab, cbStrTab);
}
RTMemFree(pbDbgSect);
}
else
rc = VERR_NO_MEMORY;
return rc;
}
/*
*
* CodeView Debug module implementation.
* CodeView Debug module implementation.
* CodeView Debug module implementation.
*
*/
/** @interface_method_impl{RTDBGMODVTDBG,pfnLineByAddr} */
static DECLCALLBACK(int) rtDbgModCv_LineByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off,
PRTINTPTR poffDisp, PRTDBGLINE pLineInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModLineByAddr(pThis->hCnt, iSeg, off, poffDisp, pLineInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnLineByOrdinal} */
static DECLCALLBACK(int) rtDbgModCv_LineByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGLINE pLineInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModLineByOrdinal(pThis->hCnt, iOrdinal, pLineInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnLineCount} */
static DECLCALLBACK(uint32_t) rtDbgModCv_LineCount(PRTDBGMODINT pMod)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModLineCount(pThis->hCnt);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnLineAdd} */
static DECLCALLBACK(int) rtDbgModCv_LineAdd(PRTDBGMODINT pMod, const char *pszFile, size_t cchFile, uint32_t uLineNo,
uint32_t iSeg, RTUINTPTR off, uint32_t *piOrdinal)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
Assert(!pszFile[cchFile]); NOREF(cchFile);
return RTDbgModLineAdd(pThis->hCnt, pszFile, uLineNo, iSeg, off, piOrdinal);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByAddr} */
static DECLCALLBACK(int) rtDbgModCv_SymbolByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off, uint32_t fFlags,
PRTINTPTR poffDisp, PRTDBGSYMBOL pSymInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModSymbolByAddr(pThis->hCnt, iSeg, off, fFlags, poffDisp, pSymInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByName} */
static DECLCALLBACK(int) rtDbgModCv_SymbolByName(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol,
PRTDBGSYMBOL pSymInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
Assert(!pszSymbol[cchSymbol]);
return RTDbgModSymbolByName(pThis->hCnt, pszSymbol/*, cchSymbol*/, pSymInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolByOrdinal} */
static DECLCALLBACK(int) rtDbgModCv_SymbolByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGSYMBOL pSymInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModSymbolByOrdinal(pThis->hCnt, iOrdinal, pSymInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolCount} */
static DECLCALLBACK(uint32_t) rtDbgModCv_SymbolCount(PRTDBGMODINT pMod)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModSymbolCount(pThis->hCnt);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSymbolAdd} */
static DECLCALLBACK(int) rtDbgModCv_SymbolAdd(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol,
RTDBGSEGIDX iSeg, RTUINTPTR off, RTUINTPTR cb, uint32_t fFlags,
uint32_t *piOrdinal)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
Assert(!pszSymbol[cchSymbol]); NOREF(cchSymbol);
return RTDbgModSymbolAdd(pThis->hCnt, pszSymbol, iSeg, off, cb, fFlags, piOrdinal);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentByIndex} */
static DECLCALLBACK(int) rtDbgModCv_SegmentByIndex(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, PRTDBGSEGMENT pSegInfo)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModSegmentByIndex(pThis->hCnt, iSeg, pSegInfo);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentCount} */
static DECLCALLBACK(RTDBGSEGIDX) rtDbgModCv_SegmentCount(PRTDBGMODINT pMod)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModSegmentCount(pThis->hCnt);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnSegmentAdd} */
static DECLCALLBACK(int) rtDbgModCv_SegmentAdd(PRTDBGMODINT pMod, RTUINTPTR uRva, RTUINTPTR cb, const char *pszName, size_t cchName,
uint32_t fFlags, PRTDBGSEGIDX piSeg)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
Assert(!pszName[cchName]); NOREF(cchName);
return RTDbgModSegmentAdd(pThis->hCnt, uRva, cb, pszName, fFlags, piSeg);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnImageSize} */
static DECLCALLBACK(RTUINTPTR) rtDbgModCv_ImageSize(PRTDBGMODINT pMod)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
if (pThis->cbImage)
return pThis->cbImage;
return RTDbgModImageSize(pThis->hCnt);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnRvaToSegOff} */
static DECLCALLBACK(RTDBGSEGIDX) rtDbgModCv_RvaToSegOff(PRTDBGMODINT pMod, RTUINTPTR uRva, PRTUINTPTR poffSeg)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
return RTDbgModRvaToSegOff(pThis->hCnt, uRva, poffSeg);
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnClose} */
static DECLCALLBACK(int) rtDbgModCv_Close(PRTDBGMODINT pMod)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
RTDbgModRelease(pThis->hCnt);
if (pThis->hFile != NIL_RTFILE)
RTFileClose(pThis->hFile);
RTMemFree(pThis->paDirEnts);
RTMemFree(pThis);
pMod->pvDbgPriv = NULL; /* for internal use */
return VINF_SUCCESS;
}
/*
*
* Probing code used by rtDbgModCv_TryOpen.
* Probing code used by rtDbgModCv_TryOpen.
*
*/
/** @callback_method_impl{FNRTLDRENUMSEGS,
* Used to add segments from the image.} */
static DECLCALLBACK(int) rtDbgModCvAddSegmentsCallback(RTLDRMOD hLdrMod, PCRTLDRSEG pSeg, void *pvUser)
{
PRTDBGMODCV pThis = (PRTDBGMODCV)pvUser;
Log(("Segment %s: LinkAddress=%#llx RVA=%#llx cb=%#llx\n",
pSeg->pszName, (uint64_t)pSeg->LinkAddress, (uint64_t)pSeg->RVA, pSeg->cb));
NOREF(hLdrMod);
/* If the segment doesn't have a mapping, just add a dummy so the indexing
works out correctly (same as for the image). */
if (pSeg->RVA == NIL_RTLDRADDR)
return RTDbgModSegmentAdd(pThis->hCnt, 0, 0, pSeg->pszName, 0 /*fFlags*/, NULL);
RTLDRADDR cb = RT_MAX(pSeg->cb, pSeg->cbMapped);
return RTDbgModSegmentAdd(pThis->hCnt, pSeg->RVA, cb, pSeg->pszName, 0 /*fFlags*/, NULL);
}
/**
* Copies the sections over from the DBG file.
*
* Called if we don't have an associated executable image.
*
* @returns IPRT status code.
* @param pThis The CV module instance.
* @param pDbgHdr The DBG file header.
* @param pszFilename The filename (for logging).
*/
static int rtDbgModCvAddSegmentsFromDbg(PRTDBGMODCV pThis, PCIMAGE_SEPARATE_DEBUG_HEADER pDbgHdr, const char *pszFilename)
{
/*
* Validate the header fields a little.
*/
if ( pDbgHdr->NumberOfSections < 1
|| pDbgHdr->NumberOfSections > 4096)
{
Log(("RTDbgModCv: Bad NumberOfSections: %d\n", pDbgHdr->NumberOfSections));
return VERR_CV_BAD_FORMAT;
}
if (!RT_IS_POWER_OF_TWO(pDbgHdr->SectionAlignment))
{
Log(("RTDbgModCv: Bad SectionAlignment: %#x\n", pDbgHdr->SectionAlignment));
return VERR_CV_BAD_FORMAT;
}
/*
* Read the section table.
*/
size_t cbShs = pDbgHdr->NumberOfSections * sizeof(IMAGE_SECTION_HEADER);
PIMAGE_SECTION_HEADER paShs = (PIMAGE_SECTION_HEADER)RTMemAlloc(cbShs);
if (!paShs)
return VERR_NO_MEMORY;
int rc = RTFileReadAt(pThis->hFile, sizeof(*pDbgHdr), paShs, cbShs, NULL);
if (RT_SUCCESS(rc))
{
/*
* Do some basic validation.
*/
uint32_t cbHeaders = 0;
uint32_t uRvaPrev = 0;
for (uint32_t i = 0; i < pDbgHdr->NumberOfSections; i++)
{
Log3(("RTDbgModCv: Section #%02u %#010x LB %#010x %.*s\n",
i, paShs[i].VirtualAddress, paShs[i].Misc.VirtualSize, sizeof(paShs[i].Name), paShs[i].Name));
if (paShs[i].Characteristics & IMAGE_SCN_TYPE_NOLOAD)
continue;
if (paShs[i].VirtualAddress < uRvaPrev)
{
Log(("RTDbgModCv: %s: Overlap or soring error, VirtualAddress=%#x uRvaPrev=%#x - section #%d '%.*s'!!!\n",
pszFilename, paShs[i].VirtualAddress, uRvaPrev, i, sizeof(paShs[i].Name), paShs[i].Name));
rc = VERR_CV_BAD_FORMAT;
}
else if ( paShs[i].VirtualAddress > pDbgHdr->SizeOfImage
|| paShs[i].Misc.VirtualSize > pDbgHdr->SizeOfImage
|| paShs[i].VirtualAddress + paShs[i].Misc.VirtualSize > pDbgHdr->SizeOfImage)
{
Log(("RTDbgModCv: %s: VirtualAddress=%#x VirtualSize=%#x (total %x) - beyond image size (%#x) - section #%d '%.*s'!!!\n",
pszFilename, paShs[i].VirtualAddress, paShs[i].Misc.VirtualSize,
paShs[i].VirtualAddress + paShs[i].Misc.VirtualSize,
pThis->cbImage, i, sizeof(paShs[i].Name), paShs[i].Name));
rc = VERR_CV_BAD_FORMAT;
}
else if (paShs[i].VirtualAddress & (pDbgHdr->SectionAlignment - 1))
{
Log(("RTDbgModCv: %s: VirtualAddress=%#x misaligned (%#x) - section #%d '%.*s'!!!\n",
pszFilename, paShs[i].VirtualAddress, pDbgHdr->SectionAlignment, i, sizeof(paShs[i].Name), paShs[i].Name));
rc = VERR_CV_BAD_FORMAT;
}
else
{
if (uRvaPrev == 0)
cbHeaders = paShs[i].VirtualAddress;
uRvaPrev = paShs[i].VirtualAddress + paShs[i].Misc.VirtualSize;
continue;
}
}
if (RT_SUCCESS(rc) && uRvaPrev == 0)
{
Log(("RTDbgModCv: %s: No loadable sections.\n", pszFilename));
rc = VERR_CV_BAD_FORMAT;
}
if (RT_SUCCESS(rc) && cbHeaders == 0)
{
Log(("RTDbgModCv: %s: No space for PE headers.\n", pszFilename));
rc = VERR_CV_BAD_FORMAT;
}
if (RT_SUCCESS(rc))
{
/*
* Add sections.
*/
rc = RTDbgModSegmentAdd(pThis->hCnt, 0, cbHeaders, "NtHdrs", 0 /*fFlags*/, NULL);
for (uint32_t i = 0; RT_SUCCESS(rc) && i < pDbgHdr->NumberOfSections; i++)
{
char szName[sizeof(paShs[i].Name) + 1];
memcpy(szName, paShs[i].Name, sizeof(paShs[i].Name));
szName[sizeof(szName) - 1] = '\0';
if (paShs[i].Characteristics & IMAGE_SCN_TYPE_NOLOAD)
rc = RTDbgModSegmentAdd(pThis->hCnt, 0, 0, szName, 0 /*fFlags*/, NULL);
else
rc = RTDbgModSegmentAdd(pThis->hCnt, paShs[i].VirtualAddress, paShs[i].Misc.VirtualSize, szName,
0 /*fFlags*/, NULL);
}
if (RT_SUCCESS(rc))
pThis->fHaveLoadedSegments = true;
}
}
RTMemFree(paShs);
return rc;
}
/**
* Instantiates the CV/COFF reader.
*
* @returns IPRT status code
* @param pDbgMod The debug module instance.
* @param enmFileType The type of input file.
* @param hFile The file handle, NIL_RTFILE of image.
* @param ppThis Where to return the reader instance.
*/
static int rtDbgModCvCreateInstance(PRTDBGMODINT pDbgMod, RTCVFILETYPE enmFileType, RTFILE hFile, PRTDBGMODCV *ppThis)
{
/*
* Do we already have an instance? Happens if we find multiple debug
* formats we support.
*/
PRTDBGMODCV pThis = (PRTDBGMODCV)pDbgMod->pvDbgPriv;
if (pThis)
{
Assert(pThis->enmType == enmFileType);
Assert(pThis->hFile == hFile);
Assert(pThis->pMod == pDbgMod);
*ppThis = pThis;
return VINF_SUCCESS;
}
/*
* Create a new instance.
*/
pThis = (PRTDBGMODCV)RTMemAllocZ(sizeof(RTDBGMODCV));
if (!pThis)
return VERR_NO_MEMORY;
int rc = RTDbgModCreate(&pThis->hCnt, pDbgMod->pszName, 0 /*cbSeg*/, 0 /*fFlags*/);
if (RT_SUCCESS(rc))
{
pDbgMod->pvDbgPriv = pThis;
pThis->enmType = enmFileType;
pThis->hFile = hFile;
pThis->pMod = pDbgMod;
*ppThis = pThis;
return VINF_SUCCESS;
}
RTMemFree(pThis);
return rc;
}
/**
* Common part of the COFF probing.
*
* @returns status code.
* @param pDbgMod The debug module instance. On success pvDbgPriv
* will point to a valid RTDBGMODCV.
* @param hFile The file with debug info in it.
* @param off The offset where to expect CV debug info.
* @param cb The number of bytes of debug info.
* @param enmArch The desired image architecture.
* @param pszFilename The path to the file (for logging).
*/
static int rtDbgModCvProbeCoff(PRTDBGMODINT pDbgMod, RTCVFILETYPE enmFileType, RTFILE hFile,
uint32_t off, uint32_t cb, const char *pszFilename)
{
/*
* Check that there is sufficient data for a header, then read it.
*/
if (cb < sizeof(IMAGE_COFF_SYMBOLS_HEADER))
{
Log(("RTDbgModCv: Not enough room for COFF header.\n"));
return VERR_BAD_EXE_FORMAT;
}
if (cb >= UINT32_C(128) * _1M)
{
Log(("RTDbgModCv: COFF debug information is to large (%'u bytes), max is 128MB\n", cb));
return VERR_BAD_EXE_FORMAT;
}
int rc;
IMAGE_COFF_SYMBOLS_HEADER Hdr;
if (hFile == NIL_RTFILE)
rc = pDbgMod->pImgVt->pfnReadAt(pDbgMod, UINT32_MAX, off, &Hdr, sizeof(Hdr));
else
rc = RTFileReadAt(hFile, off, &Hdr, sizeof(Hdr), NULL);
if (RT_FAILURE(rc))
{
Log(("RTDbgModCv: Error reading COFF header: %Rrc\n", rc));
return rc;
}
Log2(("RTDbgModCv: Found COFF debug info header at %#x (LB %#x) in %s\n", off, cb, pszFilename));
Log2((" NumberOfSymbols = %#010x\n", Hdr.NumberOfSymbols));
Log2((" LvaToFirstSymbol = %#010x\n", Hdr.LvaToFirstSymbol));
Log2((" NumberOfLinenumbers = %#010x\n", Hdr.NumberOfLinenumbers));
Log2((" LvaToFirstLinenumber = %#010x\n", Hdr.LvaToFirstLinenumber));
Log2((" RvaToFirstByteOfCode = %#010x\n", Hdr.RvaToFirstByteOfCode));
Log2((" RvaToLastByteOfCode = %#010x\n", Hdr.RvaToLastByteOfCode));
Log2((" RvaToFirstByteOfData = %#010x\n", Hdr.RvaToFirstByteOfData));
Log2((" RvaToLastByteOfData = %#010x\n", Hdr.RvaToLastByteOfData));
/*
* Validate the COFF header.
*/
if ( (uint64_t)Hdr.LvaToFirstSymbol + (uint64_t)Hdr.NumberOfSymbols * sizeof(IMAGE_SYMBOL) > cb
|| (Hdr.LvaToFirstSymbol < sizeof(Hdr) && Hdr.NumberOfSymbols > 0))
{
Log(("RTDbgModCv: Bad COFF symbol count or/and offset: LvaToFirstSymbol=%#x, NumberOfSymbols=%#x cbCoff=%#x\n",
Hdr.LvaToFirstSymbol, Hdr.NumberOfSymbols, cb));
return VERR_BAD_EXE_FORMAT;
}
if ( (uint64_t)Hdr.LvaToFirstLinenumber + (uint64_t)Hdr.NumberOfLinenumbers * sizeof(IMAGE_LINENUMBER) > cb
|| (Hdr.LvaToFirstLinenumber < sizeof(Hdr) && Hdr.NumberOfLinenumbers > 0))
{
Log(("RTDbgModCv: Bad COFF symbol count or/and offset: LvaToFirstSymbol=%#x, NumberOfSymbols=%#x cbCoff=%#x\n",
Hdr.LvaToFirstSymbol, Hdr.NumberOfSymbols, cb));
return VERR_BAD_EXE_FORMAT;
}
if (Hdr.NumberOfSymbols < 2)
{
Log(("RTDbgModCv: The COFF symbol table is too short to be of any worth... (%u syms)\n", Hdr.NumberOfSymbols));
return VERR_NO_DATA;
}
/*
* What we care about looks fine, use it.
*/
PRTDBGMODCV pThis;
rc = rtDbgModCvCreateInstance(pDbgMod, enmFileType, hFile, &pThis);
if (RT_SUCCESS(rc))
{
pThis->offCoffDbgInfo = off;
pThis->cbCoffDbgInfo = cb;
pThis->CoffHdr = Hdr;
}
return rc;
}
/**
* Common part of the CodeView probing.
*
* @returns status code.
* @param pDbgMod The debug module instance. On success pvDbgPriv
* will point to a valid RTDBGMODCV.
* @param pCvHdr The CodeView base header.
* @param enmFileType The kind of file this is we're probing.
* @param hFile The file with debug info in it.
* @param off The offset where to expect CV debug info.
* @param cb The number of bytes of debug info.
* @param pszFilename The path to the file (for logging).
*/
static int rtDbgModCvProbeCommon(PRTDBGMODINT pDbgMod, PRTCVHDR pCvHdr, RTCVFILETYPE enmFileType, RTFILE hFile,
uint32_t off, uint32_t cb, RTLDRARCH enmArch, const char *pszFilename)
{
int rc = VERR_DBG_NO_MATCHING_INTERPRETER;
/* Is a codeview format we (wish to) support? */
if ( pCvHdr->u32Magic == RTCVHDR_MAGIC_NB11
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB09
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB08
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB05
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB04
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB02
|| pCvHdr->u32Magic == RTCVHDR_MAGIC_NB00
)
{
/* We're assuming it's a base header, so the offset must be within
the area defined by the debug info we got from the loader. */
if (pCvHdr->off < cb && pCvHdr->off >= sizeof(*pCvHdr))
{
Log(("RTDbgModCv: Found %c%c%c%c at %#RTfoff - size %#x, directory at %#x. file type %d\n",
RT_BYTE1(pCvHdr->u32Magic), RT_BYTE2(pCvHdr->u32Magic), RT_BYTE3(pCvHdr->u32Magic), RT_BYTE4(pCvHdr->u32Magic),
off, cb, pCvHdr->off, enmFileType));
/*
* Create a module instance, if not already done.
*/
PRTDBGMODCV pThis;
rc = rtDbgModCvCreateInstance(pDbgMod, enmFileType, hFile, &pThis);
if (RT_SUCCESS(rc))
{
pThis->u32CvMagic = pCvHdr->u32Magic;
pThis->offBase = off;
pThis->cbDbgInfo = cb;
pThis->offDir = pCvHdr->off;
return VINF_SUCCESS;
}
}
}
return rc;
}
/** @callback_method_impl{FNRTLDRENUMDBG} */
static DECLCALLBACK(int) rtDbgModCvEnumCallback(RTLDRMOD hLdrMod, PCRTLDRDBGINFO pDbgInfo, void *pvUser)
{
PRTDBGMODINT pDbgMod = (PRTDBGMODINT)pvUser;
Assert(!pDbgMod->pvDbgPriv);
/* Skip external files, RTDbgMod will deal with those
via RTDBGMODINT::pszDbgFile. */
if (pDbgInfo->pszExtFile)
return VINF_SUCCESS;
/* We only handle the codeview sections. */
if (pDbgInfo->enmType == RTLDRDBGINFOTYPE_CODEVIEW)
{
/* Read the specified header and check if we like it. */
RTCVHDR CvHdr;
int rc = pDbgMod->pImgVt->pfnReadAt(pDbgMod, pDbgInfo->iDbgInfo, pDbgInfo->offFile, &CvHdr, sizeof(CvHdr));
if (RT_SUCCESS(rc))
rc = rtDbgModCvProbeCommon(pDbgMod, &CvHdr, RTCVFILETYPE_IMAGE, NIL_RTFILE, pDbgInfo->offFile, pDbgInfo->cb,
pDbgMod->pImgVt->pfnGetArch(pDbgMod), pDbgMod->pszImgFile);
}
else if (pDbgInfo->enmType == RTLDRDBGINFOTYPE_COFF)
{
/* Join paths with the DBG code. */
rtDbgModCvProbeCoff(pDbgMod, RTCVFILETYPE_IMAGE, NIL_RTFILE, pDbgInfo->offFile, pDbgInfo->cb, pDbgMod->pszImgFile);
}
return VINF_SUCCESS;
}
/**
* Part two of the external file probing.
*
* @returns status code.
* @param pDbgMod The debug module instance. On success pvDbgPriv
* will point to a valid RTDBGMODCV.
* @param enmFileType The kind of file this is we're probing.
* @param hFile The file with debug info in it.
* @param off The offset where to expect CV debug info.
* @param cb The number of bytes of debug info.
* @param enmArch The desired image architecture.
* @param pszFilename The path to the file (for logging).
*/
static int rtDbgModCvProbeFile2(PRTDBGMODINT pThis, RTCVFILETYPE enmFileType, RTFILE hFile, uint32_t off, uint32_t cb,
RTLDRARCH enmArch, const char *pszFilename)
{
RTCVHDR CvHdr;
int rc = RTFileReadAt(hFile, off, &CvHdr, sizeof(CvHdr), NULL);
if (RT_SUCCESS(rc))
rc = rtDbgModCvProbeCommon(pThis, &CvHdr, enmFileType, hFile, off, cb, enmArch, pszFilename);
return rc;
}
/**
* Probes an external file for CodeView information.
*
* @returns status code.
* @param pDbgMod The debug module instance. On success pvDbgPriv
* will point to a valid RTDBGMODCV.
* @param pszFilename The path to the file to probe.
* @param enmArch The desired image architecture.
*/
static int rtDbgModCvProbeFile(PRTDBGMODINT pDbgMod, const char *pszFilename, RTLDRARCH enmArch)
{
RTFILE hFile;
int rc = RTFileOpen(&hFile, pszFilename, RTFILE_O_READ | RTFILE_O_DENY_WRITE | RTFILE_O_OPEN);
if (RT_FAILURE(rc))
return rc;
/*
* Check for .DBG file
*/
IMAGE_SEPARATE_DEBUG_HEADER DbgHdr;
rc = RTFileReadAt(hFile, 0, &DbgHdr, sizeof(DbgHdr), NULL);
if ( RT_SUCCESS(rc)
&& DbgHdr.Signature == IMAGE_SEPARATE_DEBUG_SIGNATURE)
{
Log2(("RTDbgModCv: Found separate debug header in %s:\n", pszFilename));
Log2((" Flags = %#x\n", DbgHdr.Flags));
Log2((" Machine = %#x\n", DbgHdr.Machine));
Log2((" Characteristics = %#x\n", DbgHdr.Characteristics));
Log2((" TimeDateStamp = %#x\n", DbgHdr.TimeDateStamp));
Log2((" CheckSum = %#x\n", DbgHdr.CheckSum));
Log2((" ImageBase = %#x\n", DbgHdr.ImageBase));
Log2((" SizeOfImage = %#x\n", DbgHdr.SizeOfImage));
Log2((" NumberOfSections = %#x\n", DbgHdr.NumberOfSections));
Log2((" ExportedNamesSize = %#x\n", DbgHdr.ExportedNamesSize));
Log2((" DebugDirectorySize = %#x\n", DbgHdr.DebugDirectorySize));
Log2((" SectionAlignment = %#x\n", DbgHdr.SectionAlignment));
/*
* Match up the architecture if specified.
*/
switch (enmArch)
{
case RTLDRARCH_X86_32:
if (DbgHdr.Machine != IMAGE_FILE_MACHINE_I386)
rc = VERR_LDR_ARCH_MISMATCH;
break;
case RTLDRARCH_AMD64:
if (DbgHdr.Machine != IMAGE_FILE_MACHINE_AMD64)
rc = VERR_LDR_ARCH_MISMATCH;
break;
default:
case RTLDRARCH_HOST:
AssertFailed();
case RTLDRARCH_WHATEVER:
break;
}
if (RT_FAILURE(rc))
{
RTFileClose(hFile);
return rc;
}
/*
* Probe for readable debug info in the debug directory.
*/
uint32_t offDbgDir = sizeof(DbgHdr)
+ DbgHdr.NumberOfSections * sizeof(IMAGE_SECTION_HEADER)
+ DbgHdr.ExportedNamesSize;
uint32_t cEntries = DbgHdr.DebugDirectorySize / sizeof(IMAGE_DEBUG_DIRECTORY);
for (uint32_t i = 0; i < cEntries; i++, offDbgDir += sizeof(IMAGE_DEBUG_DIRECTORY))
{
IMAGE_DEBUG_DIRECTORY DbgDir;
rc = RTFileReadAt(hFile, offDbgDir, &DbgDir, sizeof(DbgDir), NULL);
if (RT_FAILURE(rc))
break;
if (DbgDir.Type == IMAGE_DEBUG_TYPE_CODEVIEW)
rc = rtDbgModCvProbeFile2(pDbgMod, RTCVFILETYPE_DBG, hFile,
DbgDir.PointerToRawData, DbgDir.SizeOfData,
enmArch, pszFilename);
else if (DbgDir.Type == IMAGE_DEBUG_TYPE_COFF)
rc = rtDbgModCvProbeCoff(pDbgMod, RTCVFILETYPE_DBG, hFile,
DbgDir.PointerToRawData, DbgDir.SizeOfData, pszFilename);
}
/*
* If we get down here with an instance, it prooves that we've found
* something, regardless of any errors. Add the sections and such.
*/
PRTDBGMODCV pThis = (PRTDBGMODCV)pDbgMod->pvDbgPriv;
if (pThis)
{
pThis->cbImage = DbgHdr.SizeOfImage;
if (pDbgMod->pImgVt)
rc = VINF_SUCCESS;
else
{
rc = rtDbgModCvAddSegmentsFromDbg(pThis, &DbgHdr, pszFilename);
if (RT_FAILURE(rc))
rtDbgModCv_Close(pDbgMod);
}
return rc;
}
/* Failed to find CV or smth, look at the end of the file just to be sure... */
}
/*
* Look for CV tail header.
*/
uint64_t cbFile;
rc = RTFileSeek(hFile, -(RTFOFF)sizeof(RTCVHDR), RTFILE_SEEK_END, &cbFile);
if (RT_SUCCESS(rc))
{
cbFile += sizeof(RTCVHDR);
RTCVHDR CvHdr;
rc = RTFileRead(hFile, &CvHdr, sizeof(CvHdr), NULL);
if (RT_SUCCESS(rc))
rc = rtDbgModCvProbeFile2(pDbgMod, RTCVFILETYPE_OTHER_AT_END, hFile,
cbFile - CvHdr.off, CvHdr.off, enmArch, pszFilename);
}
if (RT_FAILURE(rc))
RTFileClose(hFile);
return rc;
}
/** @interface_method_impl{RTDBGMODVTDBG,pfnTryOpen} */
static DECLCALLBACK(int) rtDbgModCv_TryOpen(PRTDBGMODINT pMod, RTLDRARCH enmArch)
{
/*
* Look for debug info.
*/
int rc = VERR_DBG_NO_MATCHING_INTERPRETER;
if (pMod->pszDbgFile)
rc = rtDbgModCvProbeFile(pMod, pMod->pszDbgFile, enmArch);
if (!pMod->pvDbgPriv && pMod->pImgVt)
{
int rc2 = pMod->pImgVt->pfnEnumDbgInfo(pMod, rtDbgModCvEnumCallback, pMod);
if (RT_FAILURE(rc2))
rc = rc2;
if (!pMod->pvDbgPriv)
{
/* Try the executable in case it has a NBxx tail header. */
rc2 = rtDbgModCvProbeFile(pMod, pMod->pszImgFile, enmArch);
if (RT_FAILURE(rc2) && (RT_SUCCESS(rc) || VERR_DBG_NO_MATCHING_INTERPRETER))
rc = rc2;
}
}
PRTDBGMODCV pThis = (PRTDBGMODCV)pMod->pvDbgPriv;
if (!pThis)
return RT_SUCCESS_NP(rc) ? VERR_DBG_NO_MATCHING_INTERPRETER : rc;
Assert(pThis->offBase || pThis->offCoffDbgInfo);
/*
* Load the debug info.
*/
if (pMod->pImgVt)
{
rc = pMod->pImgVt->pfnEnumSegments(pMod, rtDbgModCvAddSegmentsCallback, pThis);
pThis->fHaveLoadedSegments = true;
}
if (RT_SUCCESS(rc) && pThis->offBase)
rc = rtDbgModCvLoadCodeViewInfo(pThis);
if (RT_SUCCESS(rc) && pThis->offCoffDbgInfo)
rc = rtDbgModCvLoadCoffInfo(pThis);
if (RT_SUCCESS(rc))
{
Log(("RTDbgCv: Successfully loaded debug info\n"));
return VINF_SUCCESS;
}
Log(("RTDbgCv: Debug info load error %Rrc\n", rc));
rtDbgModCv_Close(pMod);
return rc;
}
/** Virtual function table for the CodeView debug info reader. */
DECL_HIDDEN_CONST(RTDBGMODVTDBG) const g_rtDbgModVtDbgCodeView =
{
/*.u32Magic = */ RTDBGMODVTDBG_MAGIC,
/*.fSupports = */ RT_DBGTYPE_CODEVIEW,
/*.pszName = */ "codeview",
/*.pfnTryOpen = */ rtDbgModCv_TryOpen,
/*.pfnClose = */ rtDbgModCv_Close,
/*.pfnRvaToSegOff = */ rtDbgModCv_RvaToSegOff,
/*.pfnImageSize = */ rtDbgModCv_ImageSize,
/*.pfnSegmentAdd = */ rtDbgModCv_SegmentAdd,
/*.pfnSegmentCount = */ rtDbgModCv_SegmentCount,
/*.pfnSegmentByIndex = */ rtDbgModCv_SegmentByIndex,
/*.pfnSymbolAdd = */ rtDbgModCv_SymbolAdd,
/*.pfnSymbolCount = */ rtDbgModCv_SymbolCount,
/*.pfnSymbolByOrdinal = */ rtDbgModCv_SymbolByOrdinal,
/*.pfnSymbolByName = */ rtDbgModCv_SymbolByName,
/*.pfnSymbolByAddr = */ rtDbgModCv_SymbolByAddr,
/*.pfnLineAdd = */ rtDbgModCv_LineAdd,
/*.pfnLineCount = */ rtDbgModCv_LineCount,
/*.pfnLineByOrdinal = */ rtDbgModCv_LineByOrdinal,
/*.pfnLineByAddr = */ rtDbgModCv_LineByAddr,
/*.u32EndMagic = */ RTDBGMODVTDBG_MAGIC
};