Browse Source

Simplify vtable candidate lookup

The candidate system wasn't an iterable for a long while, so it's
about time to cull that logic.
nosoop 10 months ago
parent
commit
5dec19b39d
2 changed files with 7 additions and 13 deletions
  1. 4 8
      src/smgdc/angr/vtable_disamb.py
  2. 3 5
      src/smgdc/vtable.py

+ 4 - 8
src/smgdc/angr/vtable_disamb.py

@@ -110,7 +110,7 @@ class VtableDisambiguator(angr.Analysis):
             raise ValueError("unknown typeinfo class", typeinfo_class_sym)
 
     def resolve_ambiguous_vfn(
-        self, vtidx: int, ambig_fnsyms: set[Symbol], related_vtsyms: Iterable[Symbol]
+        self, vtidx: int, ambig_fnsyms: set[Symbol], related_vtsyms: set[Symbol]
     ) -> Symbol | None:
         """
         Resolves an ambiguous virtual function by attempting to match against symbols in related
@@ -148,11 +148,9 @@ class VtableDisambiguator(angr.Analysis):
                 if dh.get_dtor_type(demangler.parse(sym.name)) == dtype:
                     return sym
 
-        # prevent iterable from being consumed in the first iteration
-        rel_vtsyms = set(related_vtsyms)
         for fnsym in ambig_fnsyms - non_class_syms:
             funcsig = dh.extract_method_signature(demangler.parse(fnsym.name))
-            for svt in rel_vtsyms:
+            for svt in related_vtsyms:
                 # only check functions at the same index
                 subfn_addr = self.loader.fast_memory_load_pointer(
                     svt.rebased_addr + 0x4 * (2 + vtidx)
@@ -175,14 +173,12 @@ class VtableDisambiguator(angr.Analysis):
                     return fnsym
         return None
 
-    def get_possible_vtable_set_candidates(
-        self, vtsym: Symbol, vtidx: int
-    ) -> Iterable[set[Symbol]]:
+    def get_possible_vtable_set_candidates(self, vtsym: Symbol, vtidx: int) -> set[Symbol]:
         # 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 find the base implementor then return
         # it along with its descendants.
         psym = self.get_parent_vt_implementing(vtsym, vtidx)
-        yield {psym} | self.subclass_map[psym]
+        return {psym} | self.subclass_map[psym]
 
     def get_parent_vt_implementing(self, vtsym: Symbol, vtidx: int) -> Symbol:
         # walk up parents until we reach the first one to implement this method

+ 3 - 5
src/smgdc/vtable.py

@@ -148,11 +148,9 @@ def get_vtables_from_address(bin: "LinuxBinary", vt: Symbol) -> list[VTable]:
                     continue
 
                 # function in vtable is referenced by multiple names; perform disambiguation
-                matched_overload = None
-                for related in vtda.get_possible_vtable_set_candidates(vt, n):
-                    matched_overload = vtda.resolve_ambiguous_vfn(n, fnsyms, related)
-                    if matched_overload:
-                        break
+                matched_overload = vtda.resolve_ambiguous_vfn(
+                    n, fnsyms, vtda.get_possible_vtable_set_candidates(vt, n)
+                )
 
                 # it's possible that the other function(s) is/are resolveable.