2010-04-29 09:00:34 -06:00
|
|
|
/*
|
2011-04-11 14:13:06 -06:00
|
|
|
* Simple, pretty item dump example.
|
2010-04-29 09:00:34 -06:00
|
|
|
*/
|
2010-11-03 03:45:29 -06:00
|
|
|
|
2011-04-11 14:13:06 -06:00
|
|
|
#include <cstdio>
|
2009-11-17 08:38:33 -07:00
|
|
|
#include <iostream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <sstream>
|
|
|
|
#include <climits>
|
|
|
|
#include <vector>
|
2011-04-11 14:13:06 -06:00
|
|
|
#include <cstring>
|
2009-11-17 08:38:33 -07:00
|
|
|
using namespace std;
|
2011-04-13 17:42:03 -06:00
|
|
|
#define DFHACK_WANT_MISCUTILS
|
2010-05-25 22:48:23 -06:00
|
|
|
#include <DFHack.h>
|
2010-06-04 16:02:02 -06:00
|
|
|
#include <dfhack/DFVector.h>
|
2009-11-17 08:38:33 -07:00
|
|
|
|
2011-05-12 10:30:42 -06:00
|
|
|
int main (int argc, char *argv[])
|
2009-11-17 08:38:33 -07:00
|
|
|
{
|
2011-05-12 10:30:42 -06:00
|
|
|
bool print_refs = false;
|
|
|
|
bool print_hex = false;
|
|
|
|
bool print_acc = false;
|
|
|
|
|
|
|
|
for(int i = 1; i < argc; i++)
|
|
|
|
{
|
|
|
|
char *arg = argv[i];
|
|
|
|
if (arg[0] != '-')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (; *arg; arg++) {
|
|
|
|
switch (arg[0]) {
|
|
|
|
case 'r': print_refs = true; break;
|
|
|
|
case 'x': print_hex = true; break;
|
|
|
|
case 'a': print_acc = true; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-01 18:38:18 -06:00
|
|
|
DFHack::Process * p;
|
2010-05-23 15:06:10 -06:00
|
|
|
DFHack::ContextManager DFMgr("Memory.xml");
|
|
|
|
DFHack::Context * DF;
|
2010-05-01 18:38:18 -06:00
|
|
|
try
|
|
|
|
{
|
2010-05-23 15:06:10 -06:00
|
|
|
DF = DFMgr.getSingleContext();
|
|
|
|
DF->Attach();
|
2010-05-01 18:38:18 -06:00
|
|
|
}
|
|
|
|
catch (exception& e)
|
|
|
|
{
|
|
|
|
cerr << e.what() << endl;
|
2010-04-29 09:00:34 -06:00
|
|
|
#ifndef LINUX_BUILD
|
2010-05-01 18:38:18 -06:00
|
|
|
cin.ignore();
|
2010-04-29 09:00:34 -06:00
|
|
|
#endif
|
2010-05-01 18:38:18 -06:00
|
|
|
return 1;
|
|
|
|
}
|
2011-04-11 14:13:06 -06:00
|
|
|
DFHack::Materials * Materials = DF->getMaterials();
|
|
|
|
Materials->ReadAllMaterials();
|
2011-04-13 17:42:03 -06:00
|
|
|
|
|
|
|
DFHack::Gui * Gui = DF->getGui();
|
2011-04-11 14:13:06 -06:00
|
|
|
|
|
|
|
DFHack::Items * Items = DF->getItems();
|
|
|
|
Items->Start();
|
2010-05-01 18:38:18 -06:00
|
|
|
|
2010-08-20 06:10:05 -06:00
|
|
|
DFHack::VersionInfo * mem = DF->getMemoryInfo();
|
2010-05-23 15:06:10 -06:00
|
|
|
p = DF->getProcess();
|
2011-04-13 17:42:03 -06:00
|
|
|
int32_t x,y,z;
|
|
|
|
Gui->getCursorCoords(x,y,z);
|
2011-05-12 10:30:42 -06:00
|
|
|
|
|
|
|
std::vector<uint32_t> p_items;
|
|
|
|
Items->readItemVector(p_items);
|
|
|
|
uint32_t size = p_items.size();
|
|
|
|
|
2011-04-11 14:13:06 -06:00
|
|
|
// FIXME: tools should never be exposed to DFHack internals!
|
2010-10-19 04:07:07 -06:00
|
|
|
DFHack::OffsetGroup* itemGroup = mem->getGroup("Items");
|
2011-05-12 10:30:42 -06:00
|
|
|
uint32_t ref_vector = itemGroup->getOffset("item_ref_vector");
|
2010-05-01 18:38:18 -06:00
|
|
|
|
2011-05-23 13:30:51 -06:00
|
|
|
for(size_t i = 0; i < size; i++)
|
2011-04-11 14:13:06 -06:00
|
|
|
{
|
|
|
|
DFHack::dfh_item itm;
|
|
|
|
memset(&itm, 0, sizeof(DFHack::dfh_item));
|
|
|
|
Items->readItem(p_items[i],itm);
|
2011-05-12 10:30:42 -06:00
|
|
|
|
|
|
|
if (x != -30000
|
|
|
|
&& !(itm.base.x == x && itm.base.y == y && itm.base.z == z
|
|
|
|
&& itm.base.flags.on_ground
|
|
|
|
&& !itm.base.flags.in_chest
|
|
|
|
&& !itm.base.flags.in_inventory
|
|
|
|
&& !itm.base.flags.in_building))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
printf(
|
|
|
|
"%5d: %08x %6d %08x (%d,%d,%d) #%08x [%d] *%d %s - %s\n",
|
|
|
|
i, itm.origin, itm.id, itm.base.flags.whole,
|
|
|
|
itm.base.x, itm.base.y, itm.base.z,
|
|
|
|
itm.base.vtable,
|
|
|
|
itm.wear_level,
|
|
|
|
itm.quantity,
|
|
|
|
Items->getItemClass(itm.matdesc.itemType).c_str(),
|
|
|
|
Items->getItemDescription(itm, Materials).c_str()
|
|
|
|
);
|
|
|
|
|
|
|
|
if (print_hex)
|
|
|
|
hexdump(DF,p_items[i],0x300);
|
|
|
|
|
|
|
|
if (print_acc)
|
|
|
|
cout << Items->dumpAccessors(itm) << endl;
|
|
|
|
|
|
|
|
if (print_refs) {
|
|
|
|
DFHack::DfVector<uint32_t> p_refs(p, itm.origin + ref_vector);
|
2011-05-23 13:30:51 -06:00
|
|
|
for (size_t j = 0; j < p_refs.size(); j++) {
|
2011-05-12 10:30:42 -06:00
|
|
|
uint32_t vptr = p->readDWord(p_refs[j]);
|
|
|
|
uint32_t val = p->readDWord(p_refs[j]+4);
|
|
|
|
printf("\t-> %d \t%s\n", int(val), p->readClassName(vptr).c_str());
|
2011-04-13 17:42:03 -06:00
|
|
|
}
|
|
|
|
}
|
2011-04-11 14:13:06 -06:00
|
|
|
}
|
|
|
|
/*
|
2010-05-01 18:38:18 -06:00
|
|
|
printf("type\tvtable\tname\tquality\tdecorate\n");
|
|
|
|
for (i=0;i<size;i++)
|
|
|
|
{
|
2010-10-19 04:07:07 -06:00
|
|
|
uint32_t curItem = p_items[i];
|
2010-05-01 18:38:18 -06:00
|
|
|
uint32_t vtable = p->readDWord(p_items[i]);
|
|
|
|
uint32_t func0 = p->readDWord(vtable);
|
|
|
|
uint64_t funct0 = p->readQuad(func0);
|
|
|
|
uint32_t func1 = p->readDWord(vtable+0x238);
|
|
|
|
uint64_t funct1 = p->readQuad(func1);
|
|
|
|
uint32_t func2 = p->readDWord(vtable+0x288);
|
|
|
|
uint64_t funct2 = p->readQuad(func2);
|
|
|
|
uint32_t type = (funct0>>8)&0xffff;
|
|
|
|
|
|
|
|
uint32_t funcB = p->readDWord(vtable + 4);
|
|
|
|
uint32_t funcC = p->readDWord(vtable + 8);
|
|
|
|
uint32_t funcD = p->readDWord(vtable + 12);
|
|
|
|
uint64_t funcBt = p->readQuad(funcB);
|
|
|
|
uint64_t funcCt = p->readQuad(funcC);
|
|
|
|
uint64_t funcDt = p->readQuad(funcD);
|
|
|
|
int16_t typeB = -1;
|
|
|
|
int16_t typeC = -1;
|
|
|
|
int32_t typeD = -1;
|
|
|
|
uint32_t quality = 0;
|
|
|
|
bool hasDecorations;
|
|
|
|
string desc = p->readClassName(vtable);
|
2011-04-11 14:13:06 -06:00
|
|
|
DFHack::dfh_item itm;
|
2010-05-02 03:27:16 -06:00
|
|
|
|
2011-04-11 14:13:06 -06:00
|
|
|
Items->readItem(p_items[i], itm);
|
2010-05-01 18:38:18 -06:00
|
|
|
|
|
|
|
if ( (funct0&0xFFFFFFFFFF000000LL) != 0xCCCCC30000000000LL )
|
|
|
|
{
|
|
|
|
if (funct0 == 0xCCCCCCCCCCC3C033LL)
|
|
|
|
type = 0;
|
|
|
|
else
|
|
|
|
printf("bad type function address=%p", (void*)func0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (funct1 == 0xC300000092818B66LL)
|
|
|
|
quality = p->readWord(p_items[i]+0x92);
|
2010-10-19 04:07:07 -06:00
|
|
|
if (funct1 == 0xC300000082818B66LL)
|
|
|
|
quality = p->readWord(p_items[i]+0x82);
|
2010-05-01 18:38:18 -06:00
|
|
|
else if (funct1 == 0xCCCCCCCCCCC3C033LL)
|
|
|
|
quality = 0;
|
|
|
|
else
|
|
|
|
printf("bad quality function address=%p\n", (void*) func1);
|
|
|
|
|
|
|
|
if (funct2 == 0xCCCCCCCCCCC3C032LL)
|
|
|
|
hasDecorations = false;
|
|
|
|
else if (funct2 == 0xCCCCCCCCCCC301B0LL)
|
|
|
|
hasDecorations = true;
|
|
|
|
else
|
|
|
|
printf("bad decoration function address=%p\n", (void*) func2);
|
|
|
|
|
|
|
|
if (funcBt == 0xCCCCCCCCC3FFC883LL)
|
|
|
|
typeB = -1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint64_t funcBtEnd = p->readQuad(funcB+8);
|
|
|
|
if (
|
|
|
|
((funcBt&0xFFFFFFFF0000FFFFLL) == 0x8B6600000000818BLL) &&
|
|
|
|
((funcBtEnd&0xFFFFFFFFFFFF00FFLL) == 0xCCCCCCCCCCC30040LL) )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcBt>>16) & 0xffff;
|
|
|
|
uint32_t off2 = (funcBtEnd>>8) & 0xff;
|
|
|
|
uint32_t pt1 = p->readDWord(p_items[i] + off1);
|
|
|
|
typeB = p->readWord(pt1 + off2);
|
|
|
|
}
|
|
|
|
else if ((funcBt&0xFFFFFF0000FFFFFFLL)==0xC300000000818B66LL)
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcBt>>24) & 0xffff;
|
|
|
|
typeB = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
2010-10-19 04:07:07 -06:00
|
|
|
else if ( (funcBt&0x000000FF00FFFFFFLL) == 0x000000C300418B66LL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcBt>>24) & 0xff;
|
|
|
|
typeB = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
2010-05-01 18:38:18 -06:00
|
|
|
else
|
|
|
|
printf("bad typeB func @%p\n", (void*) funcB);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (funcCt == 0xCCCCCCCCC3FFC883LL)
|
|
|
|
typeC = -1;
|
|
|
|
else if ( (funcCt&0xFFFFFF0000FFFFFFLL) == 0xC300000000818B66LL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcCt>>24)&0xffff;
|
|
|
|
typeC = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
2010-10-19 04:07:07 -06:00
|
|
|
else if ( (funcCt&0x000000FF00FFFFFFLL) == 0x000000C300418B66LL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcCt>>24) & 0xff;
|
|
|
|
typeC = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
|
|
|
else if ( (funcCt&0x00000000FF00FFFFLL) == 0x00000000C300418BLL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcCt>>16) & 0xff;
|
|
|
|
typeC = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
2010-05-01 18:38:18 -06:00
|
|
|
else
|
|
|
|
printf("bad typeC func @%p\n", (void*) funcC);
|
|
|
|
|
|
|
|
if (funcDt == 0xCCCCCCCCC3FFC883LL)
|
|
|
|
typeD = -1;
|
|
|
|
else if ( (funcDt&0xFFFFFFFF0000FFFFLL) == 0xCCC300000000818BLL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcDt>>16) & 0xffff;
|
2010-10-19 04:07:07 -06:00
|
|
|
typeD = p->readWord(p_items[i] + off1);
|
2010-05-01 18:38:18 -06:00
|
|
|
}
|
|
|
|
else if ( (funcDt&0xFFFFFF0000FFFFFFLL) == 0xC30000000081BF0FLL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcDt>>24) & 0xffff;
|
2010-10-19 04:07:07 -06:00
|
|
|
typeD = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
|
|
|
else if ( (funcDt&0x000000FF00FFFFFFLL) == 0x000000C30041BF0FLL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcDt>>24) & 0xff;
|
|
|
|
typeD = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
|
|
|
else if ( (funcDt&0x000000FF00FFFFFFLL) == 0x000000C300418B66LL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcDt>>24) & 0xff;
|
|
|
|
typeD = p->readWord(p_items[i] + off1);
|
|
|
|
}
|
|
|
|
else if ( (funcDt&0x00000000FF00FFFFLL) == 0x00000000C300418BLL )
|
|
|
|
{
|
|
|
|
uint32_t off1 = (funcDt>>16) & 0xff;
|
|
|
|
typeD = p->readDWord(p_items[i] + off1);
|
2010-05-01 18:38:18 -06:00
|
|
|
}
|
|
|
|
else
|
|
|
|
printf("bad typeD func @%p\n", (void*) funcD);
|
|
|
|
|
|
|
|
// printf("%p\t%.16LX\t", (void*) func2, funct2);
|
2010-05-02 04:50:51 -06:00
|
|
|
printf("%d\t%p\t%s\t%d\t[%d,%d,%d]", type, (void*)vtable, desc.c_str(), quality,
|
|
|
|
typeB, typeC, typeD);
|
|
|
|
printf("\t%s", Items->getItemDescription(p_items[i], Materials).c_str());
|
2010-05-01 18:38:18 -06:00
|
|
|
// printf("\t%p\t%.16LX", (void *) funcD, funcDt);
|
2010-05-02 04:50:51 -06:00
|
|
|
if( (type!=itm.matdesc.itemType) || (typeB!=itm.matdesc.subType) || (typeC!=itm.matdesc.subIndex) || (typeD!=itm.matdesc.index) || (quality!=itm.quality) )
|
2010-05-02 03:27:16 -06:00
|
|
|
printf("\tbad[%d,%d,%d,%d]", itm.matdesc.itemType, itm.matdesc.subType, itm.matdesc.subIndex, itm.matdesc.index);
|
2010-05-01 18:38:18 -06:00
|
|
|
if (hasDecorations)
|
|
|
|
{
|
|
|
|
bool sep = false;
|
|
|
|
printf("\tdeco=[");
|
2010-10-19 04:07:07 -06:00
|
|
|
uint32_t decStart = p->readDWord(p_items[i] + 0x90); // 0xAC pre .13
|
|
|
|
uint32_t decEnd = p->readDWord(p_items[i] + 0x94); // 0xB0 pre .13
|
2010-05-01 18:38:18 -06:00
|
|
|
if (decStart != decEnd)
|
|
|
|
{
|
|
|
|
for (j=decStart;j<decEnd;j+=4)
|
|
|
|
{
|
|
|
|
uint32_t decoration = p->readDWord(j);
|
|
|
|
uint32_t dvtable = p->readDWord(decoration);
|
|
|
|
string ddesc = p->readClassName(dvtable);
|
|
|
|
uint32_t dtypefunc = p->readDWord(dvtable + 20);
|
|
|
|
uint64_t dtypefunct = p->readQuad(dtypefunc);
|
|
|
|
uint32_t dtypeC = p->readWord(decoration + 0x4);
|
|
|
|
uint32_t dtypeD = p->readDWord(decoration + 0x8);
|
|
|
|
uint32_t dtype = 0;
|
|
|
|
uint32_t dqual = p->readWord(decoration + 20);
|
|
|
|
if ( (dtypefunct&0xFFFFFFFFFFFF00FFLL) == 0xCCCCC300000000B8LL)
|
|
|
|
dtype = (dtypefunct>>8)&0xfffffff;
|
2010-05-04 08:46:15 -06:00
|
|
|
else if ( dtypefunct == 0xCCCCCCCCCCC3C033LL)
|
|
|
|
dtype = 0;
|
|
|
|
else
|
2010-05-01 18:38:18 -06:00
|
|
|
printf("bad decoration type function, address=%p\n", (void*) dtypefunc);
|
|
|
|
if (sep)
|
|
|
|
printf(",");
|
2010-05-02 04:50:51 -06:00
|
|
|
//printf("%s[t=%d,q=%d,%s{%d,%d}]", ddesc.c_str(), dtype, dqual, getMatDesc(-1, dtypeC, dtypeD).c_str(), dtypeC, dtypeD);
|
2010-05-01 18:38:18 -06:00
|
|
|
sep = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("]");
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
2011-04-11 14:13:06 -06:00
|
|
|
*/
|
2010-04-29 09:00:34 -06:00
|
|
|
#ifndef LINUX_BUILD
|
2010-05-01 18:38:18 -06:00
|
|
|
cout << "Done. Press any key to continue" << endl;
|
|
|
|
cin.ignore();
|
2010-04-29 09:00:34 -06:00
|
|
|
#endif
|
2010-05-01 18:38:18 -06:00
|
|
|
return 0;
|
2009-11-17 08:38:33 -07:00
|
|
|
}
|