Skip to content

Commit

Permalink
Start implementing RiscyROP gadget chaining
Browse files Browse the repository at this point in the history
  • Loading branch information
bkrl committed Dec 19, 2024
1 parent f008b8a commit ba92dc9
Showing 1 changed file with 66 additions and 0 deletions.
66 changes: 66 additions & 0 deletions angrop/chain_builder/reg_setter.py
Original file line number Diff line number Diff line change
@@ -1,13 +1,15 @@
import heapq
import logging
from collections import defaultdict
from typing import Iterator

import claripy
from angr.errors import SimUnsatError

from .builder import Builder
from .. import rop_utils
from ..rop_chain import RopChain
from ..rop_gadget import RopGadget
from ..errors import RopException

l = logging.getLogger("angrop.chain_builder.reg_setter")
Expand Down Expand Up @@ -508,3 +510,67 @@ def _check_if_sufficient_partial_control(self, gadget, reg, value):
return False
return True
return False

def _backwards_recursive_search(
self,
gadgets: list[RopGadget],
current_chain: list[RopGadget],
registers: set[str],
preserve_regs: set[str],
modifiable_memory_range: tuple[int, int] | None
) -> Iterator[list[RopGadget]]:
"""Recursively build ROP chains starting from the end using the RiscyROP algorithm."""
# Base case.
if not registers:
yield current_chain
return

for gadget in gadgets:
if not gadget.changed_regs.isdisjoint(preserve_regs):
continue
remaining_regs = self._get_remaining_regs(gadget, registers)
if remaining_regs is None:
continue
current_chain.append(gadget)
yield from self._backwards_recursive_search(gadgets, current_chain, remaining_regs, preserve_regs, modifiable_memory_range)
current_chain.pop()

def _get_remaining_regs(self, gadget: RopGadget, registers: set[str]) -> set[str] | None:
"""
Get the registers that still need to be controlled after prepending a gadget.
Returns None if this gadget cannot be used.
"""
# Check if the gadget sets any registers that we need.
if gadget.popped_regs.isdisjoint(registers) and not any(
reg_move.to_reg in registers and reg_move.bits == self.project.arch.bits
for reg_move in gadget.reg_moves
):
return None

remaining_regs = set()

for reg in registers:
if reg in gadget.popped_regs:
continue
new_reg = reg
for reg_move in gadget.reg_moves:
if reg_move.to_reg == reg:
if reg_move.bits != self.project.arch.bits:
# Register is only partially overwritten.
return None
new_reg = reg_move.from_reg
break
if new_reg in remaining_regs:
# Conflict, can't put two different values in the same register.
return None
remaining_regs.add(new_reg)

if gadget.transit_type == 'jmp_reg':
# I don't know what's the difference between these two so just error if they're different.
assert gadget.jump_reg == gadget.pc_reg
if gadget.jump_reg in remaining_regs:
return None
remaining_regs.add(gadget.jump_reg)

return remaining_regs

0 comments on commit ba92dc9

Please sign in to comment.