ultimatepp/uppsrc/ide/Debuggers/Sym.cpp
mdelfede 263ff5f895 changed svn layout
git-svn-id: svn://ultimatepp.org/upp/trunk@281 f0d560ea-af0d-0410-9eb7-867de7ffcac7
2008-06-07 22:31:27 +00:00

427 lines
11 KiB
C++

#include "Debuggers.h"
#ifdef COMPILER_MSC
#define LLOG(x) // LOG(x)
#ifdef _DEBUG
char * SymTagAsString( DWORD symTag )
{
switch( symTag )
{
#define SYMTAG(t) case SymTag##t: return #t;
SYMTAG(Function)
SYMTAG(Data)
SYMTAG(PublicSymbol)
SYMTAG(UDT)
SYMTAG(Enum)
SYMTAG(Typedef)
SYMTAG(PointerType)
SYMTAG(ArrayType)
SYMTAG(BaseType)
SYMTAG(VTableShape)
SYMTAG(VTable)
#undef SYMTAG
default: return "???";
}
}
char * BaseTypeAsString( DWORD baseType )
{
switch ( baseType )
{
case btNoType: return "btNoType";
case btVoid: return "btVoid";
case btChar: return "btChar";
case btWChar: return "btWChar";
case btInt: return "btInt";
case btUInt: return "btUInt";
case btFloat: return "btFloat";
case btBCD: return "btBCD";
case btBool: return "btBool";
case btLong: return "btLong";
case btULong: return "btULong";
case btCurrency: return "btCurrency";
case btDate: return "btDate";
case btVariant: return "btVariant";
case btComplex: return "btComplex";
case btBit: return "btBit";
case btBSTR: return "btBSTR";
case btHresult: return "btHresult";
default: return "???";
}
}
#endif
adr_t Pdb::GetAddress(FilePos p)
{
LONG dummy;
IMAGEHLP_LINE ln;
ln.SizeOfStruct = sizeof(ln);
char h[MAX_PATH];
strcpy(h, p.path);
if(SymGetLineFromName(hProcess, NULL, h, p.line + 1, &dummy, &ln)) {
LLOG("GetAddress " << p.path << "(" << p.line << "): " << FormatIntHex(ln.Address));
return ln.Address;
}
LLOG("GetAddress " << p.path << "(" << p.line << "): ??");
return NULL;
}
Pdb::FilePos Pdb::GetFilePos(adr_t address)
{
FilePos fp;
DWORD dummy;
IMAGEHLP_LINE ln;
ln.SizeOfStruct = sizeof(ln);
fp.address = address;
if(SymGetLineFromAddr(hProcess, address, &dummy, &ln) && FileExists(ln.FileName)) {
fp.line = ln.LineNumber - 1;
fp.path = ln.FileName;
fp.address = ln.Address;
}
LLOG("GetFilePos(" << FormatIntHex(address) << "): " << fp.path << ": " << fp.line);
return fp;
}
#define MAX_SYM_NAME 1024
Pdb::FnInfo Pdb::GetFnInfo(adr_t address)
{
DWORD64 h;
ULONG64 buffer[(sizeof(SYMBOL_INFO) +
MAX_SYM_NAME +
sizeof(ULONG64) - 1) /
sizeof(ULONG64)];
SYMBOL_INFO *f = (SYMBOL_INFO*)buffer;
f->SizeOfStruct = sizeof(SYMBOL_INFO);
f->MaxNameLen = MAX_SYM_NAME;
FnInfo fn;
if(SymFromAddr(hProcess, address, &h, f)) {
LLOG("GetFnInfo " << f->Name
<< ", type index: " << f->TypeIndex
<< ", Flags: " << FormatIntHex(f->Flags)
<< ", Address: " << FormatIntHex((dword)f->Address)
<< ", Size: " << FormatIntHex((dword)f->Size)
<< ", Tag: " << SymTagAsString(f->Tag));
fn.name = f->Name;
fn.address = (dword)f->Address;
fn.size = f->Size;
fn.pdbtype = f->TypeIndex;
}
return fn;
}
/*
Pdb::FnInfo Pdb::GetFnInfo(String name)
{
ULONG64 b[(sizeof(SYMBOL_INFO) + MAX_SYM_NAME + sizeof(ULONG64) - 1) / sizeof(ULONG64)];
SYMBOL_INFO *f = (SYMBOL_INFO*)b;
f->SizeOfStruct = sizeof(SYMBOL_INFO);
f->MaxNameLen = MAX_SYM_NAME;
FnInfo fn;
if(SymFromName(hProcess, const_cast<char *>(~name), f)) {
LLOG("GetFnInfo " << f->Name
<< ", type index: " << f->TypeIndex
<< ", Flags: " << FormatIntHex(f->Flags)
<< ", Address: " << FormatIntHex((dword)f->Address)
<< ", Size: " << FormatIntHex((dword)f->Size)
<< ", Tag: " << SymTagAsString(f->Tag));
fn.name = f->Name;
fn.address = (dword)f->Address;
fn.size = f->Size;
fn.pdbtype = f->TypeIndex;
}
return fn;
}
*/
void Pdb::TypeVal(Pdb::Val& v, int typeId, adr_t modbase)
{
adr_t tag;
for(;;) {
tag = GetSymInfo(modbase, typeId, TI_GET_SYMTAG);
if(tag == SymTagPointerType) {
v.ref++;
const Type& tptr = GetTypeId(modbase, typeId);
}
else
if(tag == SymTagArrayType)
v.array = true;
else
break;
typeId = GetSymInfo(modbase, typeId, TI_GET_TYPE);
}
v.type = UNKNOWN;
if(tag == SymTagUDT)
v.type = GetTypeIndex(modbase, typeId);
else {
ULONG64 sz = 0;
SymGetTypeInfo(hProcess, modbase, typeId, TI_GET_LENGTH, &sz);
dword size = (dword)sz;
if(tag == SymTagEnum)
v.type = size == 8 ? SINT8 : size == 4 ? SINT4 : size == 2 ? SINT2 : SINT1;
else {
switch(GetSymInfo(modbase, typeId, TI_GET_BASETYPE)) {
case btBool:
v.type = BOOL1;
break;
case btChar:
case btWChar:
case btInt:
case btLong:
v.type = size == 8 ? SINT8 : size == 4 ? SINT4 : size == 2 ? SINT2 : SINT1;
break;
case btUInt:
case btULong:
v.type = size == 8 ? UINT8 : size == 4 ? UINT4 : size == 2 ? UINT2 : UINT1;
break;
case btFloat:
v.type = size == 8 ? DBL : FLT;
break;
}
}
}
}
struct Pdb::LocalsCtx {
adr_t ebp;
VectorMap<String, Pdb::Val> param;
VectorMap<String, Pdb::Val> local;
Pdb *pdb;
};
BOOL CALLBACK Pdb::EnumLocals(PSYMBOL_INFO pSym, ULONG SymbolSize, PVOID UserContext)
{
LocalsCtx& c = *(LocalsCtx *)UserContext;
if(pSym->Tag == SymTagFunction)
return TRUE;
if(pSym->Flags & IMAGEHLP_SYMBOL_INFO_REGISTER)
return TRUE;
Val& v = (pSym->Flags & IMAGEHLP_SYMBOL_INFO_PARAMETER ? c.param : c.local).GetAdd(pSym->Name);
v.address = (adr_t)pSym->Address;
if(pSym->Flags & IMAGEHLP_SYMBOL_INFO_REGRELATIVE)
v.address += c.ebp;
c.pdb->TypeVal(v, pSym->TypeIndex, (adr_t)pSym->ModBase);
return TRUE;
}
void Pdb::GetLocals(adr_t eip, adr_t ebp, VectorMap<String, Pdb::Val>& param,
VectorMap<String, Pdb::Val>& local)
{
static IMAGEHLP_STACK_FRAME f;
f.InstructionOffset = eip;
SymSetContext(hProcess, &f, 0);
LocalsCtx c;
c.ebp = ebp;
c.pdb = this;
SymEnumSymbols(hProcess, 0, 0, &EnumLocals, &c);
param = c.param;
local = c.local;
}
Pdb::Val Pdb::GetGlobal(const char *fn, const String& name)
{
return Val();
ULONG64 b[(sizeof(SYMBOL_INFO) + MAX_SYM_NAME + sizeof(ULONG64) - 1) / sizeof(ULONG64)];
PSYMBOL_INFO pSymbol = (PSYMBOL_INFO)b;
pSymbol->SizeOfStruct = sizeof(SYMBOL_INFO);
pSymbol->MaxNameLen = MAX_SYM_NAME;
SymSetOptions(SYMOPT_LOAD_LINES|SYMOPT_UNDNAME
|SYMOPT_NO_UNQUALIFIED_LOADS|SYMOPT_NO_PUBLICS);
Vector<String> n = Split(fn, ':');
if(noglobal.GetCount() > 5000)
noglobal.Clear();
String nm;
for(;;) {
if(n.GetCount())
n.Drop();
nm = n.GetCount() ? Join(n, "::") + "::" + name : name;
int q = global.Find(nm);
if(q >= 0) {
SymSetOptions(SYMOPT_LOAD_LINES|SYMOPT_UNDNAME|SYMOPT_NO_UNQUALIFIED_LOADS);
return global[q];
}
if(noglobal.Find(nm) < 0) {
Buffer<char> h(nm.GetLength() + 1);
strcpy(h, nm);
if(SymFromName(hProcess, h, pSymbol))
break;
}
noglobal.FindAdd(nm);
if(n.GetCount() == 0) {
SymSetOptions(SYMOPT_LOAD_LINES|SYMOPT_UNDNAME|SYMOPT_NO_UNQUALIFIED_LOADS);
return Val();
}
}
SymSetOptions(SYMOPT_LOAD_LINES|SYMOPT_UNDNAME|SYMOPT_NO_UNQUALIFIED_LOADS);
if(pSymbol->Flags & (IMAGEHLP_SYMBOL_INFO_PARAMETER|IMAGEHLP_SYMBOL_INFO_LOCAL) ||
pSymbol->Tag != SymTagData) {
noglobal.FindAdd(nm);
return Val();
}
Val v;
v.address = (adr_t)pSymbol->Address;
TypeVal(v, pSymbol->TypeIndex, (adr_t)pSymbol->ModBase);
global.Add(nm, v);
return v;
}
String Pdb::GetSymName(adr_t modbase, dword typeindex)
{
WCHAR *pwszTypeName;
if(SymGetTypeInfo(hProcess, modbase, typeindex, TI_GET_SYMNAME, &pwszTypeName)) {
WString w((const wchar *)pwszTypeName);
LocalFree(pwszTypeName);
return w.ToString();
}
return Null;
}
dword Pdb::GetSymInfo(adr_t modbase, dword typeindex, IMAGEHLP_SYMBOL_TYPE_INFO info)
{
dword dw = 0;
SymGetTypeInfo(hProcess, modbase, typeindex, info, &dw);
return dw;
}
int Pdb::GetTypeIndex(adr_t modbase, dword typeindex)
{
int q = type.Find(typeindex);
if(q < 0) {
q = type.GetCount();
type.Add(typeindex).modbase = modbase;
}
return q;
}
const Pdb::Type& Pdb::GetType(int ti)
{
Type& t = type[ti];
int typeindex = type.GetKey(ti);
if(t.size < 0) {
t.name = GetSymName(t.modbase, typeindex);
ULONG64 sz = 0;
SymGetTypeInfo(hProcess, t.modbase, typeindex, TI_GET_LENGTH, &sz);
t.size = (dword)sz;
dword count = GetSymInfo(t.modbase, typeindex, TI_GET_CHILDRENCOUNT);
if(count) {
Buffer<byte> b(sizeof(TI_FINDCHILDREN_PARAMS) + sizeof(ULONG) * count);
TI_FINDCHILDREN_PARAMS *children = (TI_FINDCHILDREN_PARAMS *) ~b;
children->Count = count;
children->Start = 0;
if(SymGetTypeInfo(hProcess, t.modbase, typeindex, TI_FINDCHILDREN, children)) {
for(dword i = 0; i < count; i++) {
dword ch = children->ChildId[i];
dword tag = GetSymInfo(t.modbase, ch, TI_GET_SYMTAG);
dword kind = GetSymInfo(t.modbase, ch, TI_GET_DATAKIND);
if(tag == SymTagData) {
String name = GetSymName(t.modbase, ch);
if(kind == DataIsMember) {
Val& v = t.member.Add(name);
TypeVal(v, GetSymInfo(t.modbase, ch, TI_GET_TYPEID), t.modbase);
v.address = GetSymInfo(t.modbase, ch, TI_GET_OFFSET);
ULONG64 bitcnt = 0;
SymGetTypeInfo(hProcess, t.modbase, ch, TI_GET_LENGTH, &bitcnt);
if(bitcnt) {
v.bitcnt = (byte)bitcnt;
v.bitpos = (byte)GetSymInfo(t.modbase, ch, TI_GET_BITPOSITION);
}
}
if(kind == DataIsStaticMember || kind == DataIsGlobal) {
Val& v = t.static_member.Add(name);
TypeVal(v, GetSymInfo(t.modbase, ch, TI_GET_TYPEID), t.modbase);
ULONG64 adr = 0;
SymGetTypeInfo(hProcess, t.modbase, ch, TI_GET_ADDRESS, &adr);
v.address = (adr_t)adr;
}
}
else
if(tag == SymTagBaseClass) {
DUMP(kind);
Val& v = t.base.Add();
TypeVal(v, GetSymInfo(t.modbase, ch, TI_GET_TYPEID), t.modbase);
v.address = GetSymInfo(t.modbase, ch, TI_GET_OFFSET);
}
else
if(tag == SymTagVTable) {
t.vtbl_offset = GetSymInfo(t.modbase, ch, TI_GET_OFFSET);
dword typeId = GetSymInfo(t.modbase, ch, TI_GET_TYPEID);
while(GetSymInfo(t.modbase, typeId, TI_GET_SYMTAG) == SymTagPointerType)
typeId = GetSymInfo(t.modbase, typeId, TI_GET_TYPE);
if((t.vtbl_typeindex = type.Find(typeId)) < 0) {
t.vtbl_typeindex = type.GetCount();
Type& vt = type.Add(typeId);
vt.modbase = t.modbase;
vt.size = 0;
vt.vtbl_typeindex = -2;
}
}
}
}
}
}
return t;
}
#ifdef _DEBUG
String Pdb::TypeAsString(int ti, bool deep)
{
String r;
#define sTYPE(x) case x: return #x;
switch(ti) {
sTYPE(BOOL1)
sTYPE(UINT1)
sTYPE(SINT1)
sTYPE(UINT2)
sTYPE(SINT2)
sTYPE(UINT4)
sTYPE(SINT4)
sTYPE(UINT8)
sTYPE(SINT8)
sTYPE(FLT)
sTYPE(DBL)
sTYPE(UNKNOWN)
}
if(ti < 0)
return r;
const Type& t = GetType(ti);
r << t.name << "(sizeof = " << t.size << ") ";
if(!deep)
return r;
if(t.member.GetCount()) {
r << "{ ";
for(int i = 0; i < t.member.GetCount(); i++) {
if(i)
r << ", ";
r << t.member.GetKey(i) << " +" << t.member[i].address;
if(t.member[i].ref)
r << TypeAsString(t.member[i].type, false) << String('*', t.member[i].ref);
else
r << ": " << TypeAsString(t.member[i].type);
}
r << " }";
}
if(t.base.GetCount()) {
r << " BASE: ";
for(int i = 0; i < t.base.GetCount(); i++) {
if(i)
r << ", ";
r << " +" << t.base[i].address;
r << TypeAsString(t.base[i].type);
}
}
return r;
}
#endif
#endif