|
@@ -2,6 +2,7 @@
|
|
|
|
|
|
import configparser
|
|
|
import contextlib
|
|
|
+import dataclasses
|
|
|
import enum
|
|
|
import functools
|
|
|
import gzip
|
|
@@ -123,6 +124,42 @@ class NumericOutputFormat(enum.StrEnum):
|
|
|
raise NotImplementedError(f"Missing numeric output for {self}")
|
|
|
|
|
|
|
|
|
+@dataclasses.dataclass
|
|
|
+class BinaryPosition:
|
|
|
+ """
|
|
|
+ Represents a file-offset pair.
|
|
|
+
|
|
|
+ This is designed to mimic SourceMod's gamedata address API; a value is produced after
|
|
|
+ chaining read operations.
|
|
|
+ """
|
|
|
+
|
|
|
+ position: int
|
|
|
+ bin: BaseBinary
|
|
|
+
|
|
|
+ def read(self, offset: int = 0) -> "BinaryPosition":
|
|
|
+ s = struct.Struct("<I")
|
|
|
+ next_position, *_ = s.unpack(self.bin.read(self.position + offset, s.size))
|
|
|
+ return BinaryPosition(next_position, self.bin)
|
|
|
+
|
|
|
+ def value(self, struct_str: str, offset: int = 0) -> typing.Any:
|
|
|
+ s = struct.Struct(struct_str)
|
|
|
+ result, *_ = s.unpack(self.bin.read(self.position + offset, s.size))
|
|
|
+ return result
|
|
|
+
|
|
|
+ def string_value(
|
|
|
+ self, encoding: str = "utf-8", errors: str = "strict", offset: int = 0
|
|
|
+ ) -> str:
|
|
|
+ """
|
|
|
+ Returns the zero-terminated string at the given position.
|
|
|
+ """
|
|
|
+ with self.bin.mmap() as mm:
|
|
|
+ start = self.position + offset
|
|
|
+ end = mm.find(b"\x00", start)
|
|
|
+ if end == -1:
|
|
|
+ raise ValueError(f"Could not find null terminator after position {start:02x}")
|
|
|
+ return mm[start:end].decode(encoding, errors)
|
|
|
+
|
|
|
+
|
|
|
class BaseEntry(msgspec.Struct, kw_only=True):
|
|
|
# the partial path pointing to a binary
|
|
|
target: pathlib.Path
|
|
@@ -227,6 +264,8 @@ class ByteSigEntry(LocationEntry, tag="bytesig", kw_only=True):
|
|
|
# most bytesigs are expected to be unique; escape hatch for those that are just typecasted
|
|
|
allow_multiple: bool = False
|
|
|
|
|
|
+ assert_stmt: Code | None = msgspec.field(default=None, name="assert")
|
|
|
+
|
|
|
def process(self, bin: PlatformBinary) -> ResultValues:
|
|
|
outputs = {
|
|
|
KEY_AS_IS: self.contents.gameconf_str,
|
|
@@ -246,6 +285,11 @@ class ByteSigEntry(LocationEntry, tag="bytesig", kw_only=True):
|
|
|
raise AssertionError(
|
|
|
f"Assertion failed: {self.contents.display_str} != {actual_disp}"
|
|
|
)
|
|
|
+
|
|
|
+ bp = BinaryPosition(address, bin)
|
|
|
+ if self.assert_stmt and not self.assert_stmt.eval(addr=bp, **eval_functions):
|
|
|
+ raise AssertionError(f"'{self.assert_stmt}' failed")
|
|
|
+
|
|
|
return outputs
|
|
|
|
|
|
with bin.mmap() as memory:
|