|
@@ -2,7 +2,8 @@
|
|
|
|
|
|
import collections
|
|
import collections
|
|
import itertools
|
|
import itertools
|
|
-from typing import Iterable
|
|
|
|
|
|
+import operator
|
|
|
|
+from typing import Iterable, NamedTuple
|
|
|
|
|
|
import itanium_demangler as demangler
|
|
import itanium_demangler as demangler
|
|
from cle.backends.symbol import Symbol
|
|
from cle.backends.symbol import Symbol
|
|
@@ -165,6 +166,9 @@ class VtableDisambiguator(angr.Analysis):
|
|
# It's possible that this class inherited a base version of a method that is specialized
|
|
# It's possible that this class inherited a base version of a method that is specialized
|
|
# in a different part of the class hierarchy, so yield lists of subclasses of parents
|
|
# in a different part of the class hierarchy, so yield lists of subclasses of parents
|
|
# that include the vtable index too.
|
|
# that include the vtable index too.
|
|
|
|
+ #
|
|
|
|
+ # FIXME: This can iterate over unrelated classes; properly filter this down to the
|
|
|
|
+ # subclasses of the base that implemented this.
|
|
max_vtsize = 4 * (vtidx + 2)
|
|
max_vtsize = 4 * (vtidx + 2)
|
|
for parent_vtsym in sorted(
|
|
for parent_vtsym in sorted(
|
|
filter(lambda vt: vt.size > max_vtsize, self.superclass_map[vtsym]),
|
|
filter(lambda vt: vt.size > max_vtsize, self.superclass_map[vtsym]),
|
|
@@ -172,5 +176,36 @@ class VtableDisambiguator(angr.Analysis):
|
|
):
|
|
):
|
|
yield self.subclass_map[parent_vtsym]
|
|
yield self.subclass_map[parent_vtsym]
|
|
|
|
|
|
|
|
+ def get_vfptrs_from_table(self, vtsym: Symbol) -> list[list[int]]:
|
|
|
|
+ # returns a list of addresses for each vtable present on the class
|
|
|
|
+ class VTableFunction(NamedTuple):
|
|
|
|
+ tblidx: int
|
|
|
|
+ address: int
|
|
|
|
+
|
|
|
|
+ table_index = 0
|
|
|
|
+ function_list: list[VTableFunction] = []
|
|
|
|
+ vtable_range = enumerate(
|
|
|
|
+ range(vtsym.rebased_addr + 0x4 * 2, vtsym.rebased_addr + vtsym.size, 4)
|
|
|
|
+ )
|
|
|
|
+ for n, addr in vtable_range:
|
|
|
|
+ # get symbols that map to that address
|
|
|
|
+ deref = self.loader.fast_memory_load_pointer(addr)
|
|
|
|
+ fnsyms = set(self.syms_by_addr.get(deref) or set()) if deref else set()
|
|
|
|
+ if not fnsyms:
|
|
|
|
+ # vtable boundary; consume typeinfo so it doesn't get added to the list
|
|
|
|
+ # NOTE: we don't actually care if the indices skip; all that matters is that the
|
|
|
|
+ # functions are grouped correctly
|
|
|
|
+ table_index += 1
|
|
|
|
+ next(vtable_range)
|
|
|
|
+ continue
|
|
|
|
+ function_list.append(VTableFunction(table_index, deref))
|
|
|
|
+
|
|
|
|
+ return [
|
|
|
|
+ list(vfn.address for vfn in vtbl)
|
|
|
|
+ for tblidx, vtbl in itertools.groupby(
|
|
|
|
+ function_list, key=operator.attrgetter("tblidx")
|
|
|
|
+ )
|
|
|
|
+ ]
|
|
|
|
+
|
|
|
|
|
|
angr.analyses.AnalysesHub.register_default("VtableDisambiguator", VtableDisambiguator)
|
|
angr.analyses.AnalysesHub.register_default("VtableDisambiguator", VtableDisambiguator)
|