mirror of
https://github.com/isledecomp/isle.git
synced 2025-10-24 00:44:21 +00:00
Add datacmp to CI (#746)
This commit is contained in:
7
.github/workflows/build.yml
vendored
7
.github/workflows/build.yml
vendored
@@ -156,6 +156,13 @@ jobs:
|
|||||||
python3 tools/vtable/vtable.py legobin/ISLE.EXE build/ISLE.EXE build/ISLE.PDB .
|
python3 tools/vtable/vtable.py legobin/ISLE.EXE build/ISLE.EXE build/ISLE.PDB .
|
||||||
python3 tools/vtable/vtable.py legobin/LEGO1.DLL build/LEGO1.DLL build/LEGO1.PDB .
|
python3 tools/vtable/vtable.py legobin/LEGO1.DLL build/LEGO1.DLL build/LEGO1.PDB .
|
||||||
|
|
||||||
|
- name: Check Variables
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
python3 tools/datacmp.py legobin/CONFIG.EXE build/CONFIG.EXE build/CONFIG.PDB .
|
||||||
|
python3 tools/datacmp.py legobin/ISLE.EXE build/ISLE.EXE build/ISLE.PDB .
|
||||||
|
python3 tools/datacmp.py legobin/LEGO1.DLL build/LEGO1.DLL build/LEGO1.PDB .
|
||||||
|
|
||||||
- name: Upload Artifact
|
- name: Upload Artifact
|
||||||
uses: actions/upload-artifact@master
|
uses: actions/upload-artifact@master
|
||||||
with:
|
with:
|
||||||
|
@@ -297,7 +297,7 @@
|
|||||||
// GLOBAL: ISLE 0x4105b0
|
// GLOBAL: ISLE 0x4105b0
|
||||||
// __shi_TaskRecord
|
// __shi_TaskRecord
|
||||||
|
|
||||||
// GLOBAL: ISLE 0x4125f8
|
// ~GLOBAL: ISLE 0x4125f8
|
||||||
// ?_pnhHeap@@3P6AHI@ZA
|
// ?_pnhHeap@@3P6AHI@ZA
|
||||||
|
|
||||||
// GLOBAL: ISLE 0x412830
|
// GLOBAL: ISLE 0x412830
|
||||||
|
@@ -47,6 +47,13 @@ def parse_args() -> argparse.Namespace:
|
|||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--no-color", "-n", action="store_true", help="Do not color the output"
|
"--no-color", "-n", action="store_true", help="Do not color the output"
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
"--all",
|
||||||
|
"-a",
|
||||||
|
dest="show_all",
|
||||||
|
action="store_true",
|
||||||
|
help="Only show variables with a problem",
|
||||||
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--print-rec-addr",
|
"--print-rec-addr",
|
||||||
action="store_true",
|
action="store_true",
|
||||||
@@ -236,7 +243,7 @@ def do_the_comparison(args: argparse.Namespace) -> Iterable[ComparisonItem]:
|
|||||||
|
|
||||||
# If we are here, we can do the type-aware comparison.
|
# If we are here, we can do the type-aware comparison.
|
||||||
compared = []
|
compared = []
|
||||||
compare_items = mini_cvdump.types.get_scalars(type_name)
|
compare_items = mini_cvdump.types.get_scalars_gapless(type_name)
|
||||||
format_str = mini_cvdump.types.get_format_string(type_name)
|
format_str = mini_cvdump.types.get_format_string(type_name)
|
||||||
|
|
||||||
orig_data = unpack(format_str, orig_raw)
|
orig_data = unpack(format_str, orig_raw)
|
||||||
@@ -308,8 +315,15 @@ def main():
|
|||||||
)
|
)
|
||||||
return f"{match_color}{result.name}{colorama.Style.RESET_ALL}"
|
return f"{match_color}{result.name}{colorama.Style.RESET_ALL}"
|
||||||
|
|
||||||
|
var_count = 0
|
||||||
|
problems = 0
|
||||||
|
|
||||||
for item in do_the_comparison(args):
|
for item in do_the_comparison(args):
|
||||||
if not args.verbose and item.result == CompareResult.MATCH:
|
var_count += 1
|
||||||
|
if item.result in (CompareResult.DIFF, CompareResult.ERROR):
|
||||||
|
problems += 1
|
||||||
|
|
||||||
|
if not args.show_all and item.result == CompareResult.MATCH:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
address_display = (
|
address_display = (
|
||||||
@@ -334,8 +348,14 @@ def main():
|
|||||||
f" {c.offset:5} {value_get(c.name, '(value)'):30} {value_a} : {value_b}"
|
f" {c.offset:5} {value_get(c.name, '(value)'):30} {value_a} : {value_b}"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if args.verbose:
|
||||||
print()
|
print()
|
||||||
|
|
||||||
|
print(
|
||||||
|
f"{os.path.basename(args.original)} - Variables: {var_count}. Issues: {problems}"
|
||||||
|
)
|
||||||
|
return 0 if problems == 0 else 1
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
raise SystemExit(main())
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import re
|
import re
|
||||||
from typing import Dict, Iterator, List, NamedTuple, Optional
|
from typing import Dict, List, NamedTuple, Optional
|
||||||
|
|
||||||
|
|
||||||
class CvdumpTypeError(Exception):
|
class CvdumpTypeError(Exception):
|
||||||
@@ -109,38 +109,10 @@ def scalar_type_format_char(type_name: str) -> str:
|
|||||||
return char if scalar_type_signed(type_name) else char.upper()
|
return char if scalar_type_signed(type_name) else char.upper()
|
||||||
|
|
||||||
|
|
||||||
def member_string_iter(
|
def member_list_to_struct_string(members: List[ScalarType]) -> str:
|
||||||
members: List[ScalarType], size: Optional[int] = None
|
"""Create a string for use with struct.unpack"""
|
||||||
) -> Iterator[str]:
|
|
||||||
if len(members) == 0:
|
|
||||||
yield "x" * (size or 0)
|
|
||||||
|
|
||||||
last_offset = 0
|
format_string = "".join(m.format_char for m in members)
|
||||||
last_size = 0
|
|
||||||
for m in members:
|
|
||||||
padding = m.offset - last_offset - last_size
|
|
||||||
if padding > 0:
|
|
||||||
yield "x" * padding
|
|
||||||
|
|
||||||
yield m.format_char
|
|
||||||
last_offset = m.offset
|
|
||||||
last_size = m.size
|
|
||||||
|
|
||||||
if size is not None:
|
|
||||||
padding = size - (last_offset + last_size)
|
|
||||||
if padding > 0:
|
|
||||||
yield "x" * padding
|
|
||||||
|
|
||||||
|
|
||||||
def member_list_to_struct_string(
|
|
||||||
members: List[ScalarType], size: Optional[int] = None
|
|
||||||
) -> str:
|
|
||||||
"""Create a string for use with struct.unpack
|
|
||||||
Will pad to `size` bytes if present."""
|
|
||||||
if len(members) == 0:
|
|
||||||
return "x" * (size or 0)
|
|
||||||
|
|
||||||
format_string = "".join(list(member_string_iter(members, size)))
|
|
||||||
if len(format_string) > 0:
|
if len(format_string) > 0:
|
||||||
return "<" + format_string
|
return "<" + format_string
|
||||||
|
|
||||||
@@ -372,11 +344,43 @@ class CvdumpTypesParser:
|
|||||||
for cm in self.get_scalars(m.type)
|
for cm in self.get_scalars(m.type)
|
||||||
]
|
]
|
||||||
|
|
||||||
def get_format_string(self, type_key: str) -> str:
|
def get_scalars_gapless(self, type_key: str) -> List[ScalarType]:
|
||||||
|
"""Reduce the given type to a list of scalars so we can
|
||||||
|
compare each component value."""
|
||||||
|
|
||||||
obj = self.get(type_key)
|
obj = self.get(type_key)
|
||||||
members = self.get_scalars(type_key)
|
total_size = obj.size
|
||||||
# We need both to pad the data to size
|
|
||||||
return member_list_to_struct_string(members, obj.size)
|
scalars = self.get_scalars(type_key)
|
||||||
|
|
||||||
|
output = []
|
||||||
|
last_extent = total_size
|
||||||
|
|
||||||
|
# Walk the scalar list in reverse; we assume a gap could not
|
||||||
|
# come at the start of the struct.
|
||||||
|
for scalar in scalars[::-1]:
|
||||||
|
this_extent = scalar.offset + scalar_type_size(scalar.type)
|
||||||
|
size_diff = last_extent - this_extent
|
||||||
|
# We need to add the gap fillers in reverse here
|
||||||
|
for i in range(size_diff - 1, -1, -1):
|
||||||
|
# Push to front
|
||||||
|
output.insert(
|
||||||
|
0,
|
||||||
|
ScalarType(
|
||||||
|
offset=this_extent + i,
|
||||||
|
name="(padding)",
|
||||||
|
type="T_UCHAR",
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
output.insert(0, scalar)
|
||||||
|
last_extent = scalar.offset
|
||||||
|
|
||||||
|
return output
|
||||||
|
|
||||||
|
def get_format_string(self, type_key: str) -> str:
|
||||||
|
members = self.get_scalars_gapless(type_key)
|
||||||
|
return member_list_to_struct_string(members)
|
||||||
|
|
||||||
def read_line(self, line: str):
|
def read_line(self, line: str):
|
||||||
if (match := self.INDEX_RE.match(line)) is not None:
|
if (match := self.INDEX_RE.match(line)) is not None:
|
||||||
|
@@ -313,14 +313,27 @@ def test_struct(parser):
|
|||||||
|
|
||||||
|
|
||||||
def test_struct_padding(parser):
|
def test_struct_padding(parser):
|
||||||
"""Struct format string should insert padding characters 'x'
|
"""For data comparison purposes, make sure we have no gaps in the
|
||||||
where a value is padded to alignment size (probably 4 bytes)"""
|
list of scalar types. Any gap is filled by an unsigned char."""
|
||||||
|
|
||||||
|
# MxString, padded to 16 bytes. 4 actual members. 2 bytes of padding.
|
||||||
|
assert len(parser.get_scalars("0x4db6")) == 4
|
||||||
|
assert len(parser.get_scalars_gapless("0x4db6")) == 6
|
||||||
|
|
||||||
|
# MxVariable, with two MxStrings (and a vtable)
|
||||||
|
# Fill in the middle gap and the outer gap.
|
||||||
|
assert len(parser.get_scalars("0x22d5")) == 9
|
||||||
|
assert len(parser.get_scalars_gapless("0x22d5")) == 13
|
||||||
|
|
||||||
|
|
||||||
|
def test_struct_format_string(parser):
|
||||||
|
"""Generate the struct.unpack format string using the
|
||||||
|
list of scalars with padding filled in."""
|
||||||
# MxString, padded to 16 bytes.
|
# MxString, padded to 16 bytes.
|
||||||
assert parser.get_format_string("0x4db6") == "<LLLHxx"
|
assert parser.get_format_string("0x4db6") == "<LLLHBB"
|
||||||
|
|
||||||
# MxVariable, with two MxString members.
|
# MxVariable, with two MxString members.
|
||||||
assert parser.get_format_string("0x22d5") == "<LLLLHxxLLLHxx"
|
assert parser.get_format_string("0x22d5") == "<LLLLHBBLLLHBB"
|
||||||
|
|
||||||
|
|
||||||
def test_array(parser):
|
def test_array(parser):
|
||||||
|
Reference in New Issue
Block a user