Added some useful IDA scripts (NPOTB).
This commit is contained in:
parent
0f6e8dd311
commit
67ae85b840
126
tools/ida_scripts/makesig.idc
Normal file
126
tools/ida_scripts/makesig.idc
Normal file
@ -0,0 +1,126 @@
|
||||
#include <idc.idc>
|
||||
|
||||
/* makesig.idc: IDA script to automatically create and wildcard a function signature.
|
||||
* Copyright 2012, Asher Baker
|
||||
*
|
||||
* This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software.
|
||||
*
|
||||
* Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions:
|
||||
*
|
||||
* 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
|
||||
*
|
||||
* 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
|
||||
*
|
||||
* 3. This notice may not be removed or altered from any source distribution.
|
||||
*/
|
||||
|
||||
static main()
|
||||
{
|
||||
Wait(); // We won't work until autoanalysis is complete
|
||||
|
||||
SetStatus(IDA_STATUS_WORK);
|
||||
|
||||
auto pAddress = ScreenEA();
|
||||
pAddress = GetFunctionAttr(pAddress, FUNCATTR_START);
|
||||
if (pAddress == BADADDR)
|
||||
{
|
||||
Warning("Make sure you are in a function!");
|
||||
SetStatus(IDA_STATUS_READY);
|
||||
return;
|
||||
}
|
||||
|
||||
auto sig;
|
||||
auto pFunctionEnd = GetFunctionAttr(pAddress, FUNCATTR_END);
|
||||
|
||||
while (pAddress != BADADDR)
|
||||
{
|
||||
auto pInfo = DecodeInstruction(pAddress);
|
||||
if (!pInfo)
|
||||
{
|
||||
Warning("Something went terribly wrong D:");
|
||||
SetStatus(IDA_STATUS_READY);
|
||||
return;
|
||||
}
|
||||
|
||||
// isCode(GetFlags(pAddress)) == Opcode
|
||||
// isTail(GetFlags(pAddress)) == Operand
|
||||
// ((GetFlags(pAddress) & MS_CODE) == FF_IMMD) == :iiam:
|
||||
|
||||
auto bDone = 0;
|
||||
|
||||
if (pInfo.n == 1)
|
||||
{
|
||||
if (pInfo.Op0.type == o_near || pInfo.Op0.type == o_far)
|
||||
{
|
||||
sig = sprintf("%s%02X %s", sig, Byte(pAddress), PrintWildcards(GetDTSize(pInfo.Op0.dtyp)));
|
||||
bDone = 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (!bDone) { // unknown, just wildcard addresses
|
||||
auto i;
|
||||
for (i = 0; i < pInfo.size; i++)
|
||||
{
|
||||
auto pLoc = pAddress + i;
|
||||
if (GetFixupTgtType(pLoc) == FIXUP_OFF32)
|
||||
{
|
||||
sig = sprintf("%s%s", sig, PrintWildcards(4));
|
||||
i = i + 3;
|
||||
} else {
|
||||
sig = sprintf("%s%02X ", sig, Byte(pLoc));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (IsGoodSig(sig))
|
||||
break;
|
||||
|
||||
pAddress = NextHead(pAddress, pFunctionEnd);
|
||||
}
|
||||
|
||||
Message("%s\n", sig);
|
||||
|
||||
SetStatus(IDA_STATUS_READY);
|
||||
return;
|
||||
}
|
||||
|
||||
static GetDTSize(dtyp)
|
||||
{
|
||||
if (dtyp == dt_byte)
|
||||
{
|
||||
return 1;
|
||||
} else if (dtyp == dt_word) {
|
||||
return 2;
|
||||
} else if (dtyp == dt_dword) {
|
||||
return 4;
|
||||
} else if (dtyp == dt_float) {
|
||||
return 4;
|
||||
} else if (dtyp == dt_double) {
|
||||
return 8;
|
||||
} else {
|
||||
Warning("Unknown type size (%d)", dtyp);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
static PrintWildcards(count)
|
||||
{
|
||||
auto i, string;
|
||||
for (i = 0; i < count; i++)
|
||||
{
|
||||
string = sprintf("%s? ", string);
|
||||
}
|
||||
return string;
|
||||
}
|
||||
|
||||
static IsGoodSig(sig)
|
||||
{
|
||||
auto count, addr;
|
||||
addr = FindBinary(addr, SEARCH_DOWN|SEARCH_NEXT, sig);
|
||||
while (addr != BADADDR)
|
||||
{
|
||||
count = count + 1;
|
||||
addr = FindBinary(addr, SEARCH_DOWN|SEARCH_NEXT, sig);
|
||||
}
|
||||
return (count == 1);
|
||||
}
|
120
tools/ida_scripts/vtable_dump.py
Normal file
120
tools/ida_scripts/vtable_dump.py
Normal file
@ -0,0 +1,120 @@
|
||||
"""vtable_dump.py: IDAPython script to dump a linux vtable (and a reconstructed windows one) from a binary."""
|
||||
|
||||
"""
|
||||
This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source distribution.
|
||||
"""
|
||||
|
||||
__author__ = "Asher Baker"
|
||||
__copyright__ = "Copyright 2012, Asher Baker"
|
||||
__license__ = "zlib/libpng"
|
||||
|
||||
import re
|
||||
|
||||
def Analyze():
|
||||
SetStatus(IDA_STATUS_WORK)
|
||||
|
||||
if GetLongPrm(INF_COMPILER).id != COMP_GNU:
|
||||
Warning("This script is for binaries compiled with GCC only.")
|
||||
SetStatus(IDA_STATUS_READY)
|
||||
return
|
||||
|
||||
ea = ScreenEA()
|
||||
|
||||
if not isHead(GetFlags(ea)):
|
||||
ea = PrevHead(ea)
|
||||
|
||||
# Param needed to support old IDAPython versions
|
||||
end = NextHead(ea, 4294967295)
|
||||
|
||||
name = Demangle(Name(ea), GetLongPrm(INF_LONG_DN))
|
||||
if ea == BADADDR or name is None or not re.search(r"vf?table(?: |'\{)for", name):
|
||||
Warning("No vtable selected!\nSelect vtable block first.")
|
||||
SetStatus(IDA_STATUS_READY)
|
||||
return
|
||||
|
||||
linux_vtable = []
|
||||
temp_windows_vtable = []
|
||||
|
||||
# Extract vtable
|
||||
while ea < end:
|
||||
offset = Dword(ea)
|
||||
|
||||
# A vtable starts with some metadata, if it's missing...
|
||||
if isCode(GetFlags(offset)):
|
||||
Warning("Something went wrong!")
|
||||
SetStatus(IDA_STATUS_READY)
|
||||
return
|
||||
|
||||
# Skip thisoffs and typeinfo address
|
||||
ea += 8
|
||||
|
||||
while ea < end and isCode(GetFlags(Dword(ea))):
|
||||
name = Demangle(Name(Dword(ea)), GetLongPrm(INF_LONG_DN))
|
||||
|
||||
if offset == 0:
|
||||
linux_vtable.append(name)
|
||||
temp_windows_vtable.append(name)
|
||||
else:
|
||||
# MI entry, strip "`non-virtual thunk to'" and remove from list
|
||||
# But not if it's a dtor... what the hell is this.
|
||||
if (name.find("`non-virtual thunk to'") != -1) and name.find("::~") == -1:
|
||||
name = name[22:]
|
||||
#print "Stripping '%s' from windows vtable." % (name)
|
||||
temp_windows_vtable.remove(name)
|
||||
|
||||
ea += 4
|
||||
|
||||
for i, v in enumerate(temp_windows_vtable):
|
||||
if v.find("::~") != -1:
|
||||
#print "Found destructor at index %d: %s" % (i, v)
|
||||
del temp_windows_vtable[i]
|
||||
break
|
||||
|
||||
windows_vtable = []
|
||||
overload_stack = []
|
||||
prev_function = ""
|
||||
prev_symbol = ""
|
||||
for v in temp_windows_vtable:
|
||||
function = v.split("(", 1)[0]
|
||||
|
||||
if function == prev_function:
|
||||
# If we don't have a stack, we need to push the last function on first
|
||||
if len(overload_stack) == 0:
|
||||
# We will have added this in the previous run, remove it again...
|
||||
windows_vtable.pop()
|
||||
#print "Storing '%s' (!)" % (prev_symbol)
|
||||
overload_stack.append(prev_symbol)
|
||||
#print "Storing '%s'" % (v)
|
||||
overload_stack.append(v)
|
||||
else:
|
||||
# If we've moved onto something new, dump the stack first
|
||||
if len(overload_stack) > 0:
|
||||
#print overload_stack
|
||||
while len(overload_stack) > 0:
|
||||
windows_vtable.append(overload_stack.pop())
|
||||
|
||||
windows_vtable.append(v)
|
||||
|
||||
prev_function = function
|
||||
prev_symbol = v
|
||||
|
||||
print "Lin Win Function"
|
||||
for i, v in enumerate(linux_vtable):
|
||||
winindex = windows_vtable.index(v) if v in windows_vtable else None
|
||||
if winindex is not None:
|
||||
print "%3d %3d %s" % (i, winindex, v)
|
||||
else:
|
||||
print "%3d %s" % (i, v)
|
||||
|
||||
SetStatus(IDA_STATUS_READY)
|
||||
|
||||
if __name__ == '__main__':
|
||||
Analyze()
|
Loading…
Reference in New Issue
Block a user