splat64 0.37.0__py3-none-any.whl → 0.37.2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
splat/__init__.py CHANGED
@@ -1,7 +1,7 @@
1
1
  __package_name__ = __name__
2
2
 
3
3
  # Should be synced with pyproject.toml
4
- __version__ = "0.37.0"
4
+ __version__ = "0.37.2"
5
5
  __author__ = "ethteck"
6
6
 
7
7
  from . import util as util
splat/scripts/split.py CHANGED
@@ -6,6 +6,8 @@ import importlib
6
6
  from typing import Any, Dict, List, Optional, Set, Tuple, Union
7
7
  from pathlib import Path
8
8
 
9
+ from collections import defaultdict, deque
10
+
9
11
  from .. import __package_name__, __version__
10
12
  from ..disassembler import disassembler_instance
11
13
  from ..util import cache_handler, progress_bar, vram_classes, statistics, file_presets
@@ -222,6 +224,47 @@ def calc_segment_dependences(
222
224
  return vram_class_to_follows_segments
223
225
 
224
226
 
227
+ def sort_segments_by_vram_class_dependency(
228
+ all_segments: List[Segment],
229
+ ) -> List[Segment]:
230
+ # map all "_VRAM_END" strings to segments
231
+ end_sym_to_seg: Dict[str, Segment] = {}
232
+ for seg in all_segments:
233
+ end_sym_to_seg[get_segment_vram_end_symbol_name(seg)] = seg
234
+
235
+ # build dependency graph: A -> B means "A must come before B"
236
+ graph: Dict[Segment, List[Segment]] = defaultdict(list)
237
+ indeg: Dict[Segment, int] = {seg: 0 for seg in all_segments}
238
+
239
+ for seg in all_segments:
240
+ sym = seg.vram_symbol
241
+ if sym is None:
242
+ continue
243
+ dep = end_sym_to_seg.get(sym)
244
+ if dep is None or dep is seg:
245
+ continue
246
+ graph[dep].append(seg)
247
+ indeg[seg] += 1
248
+
249
+ # stable topo sort with queue seeded in original order
250
+ q = deque([seg for seg in all_segments if indeg[seg] == 0])
251
+ out: List[Segment] = []
252
+
253
+ while q:
254
+ n = q.popleft()
255
+ out.append(n)
256
+ for m in graph.get(n, []):
257
+ indeg[m] -= 1
258
+ if indeg[m] == 0:
259
+ q.append(m)
260
+
261
+ assert len(out) == len(all_segments), (
262
+ "Encountered cyclic dependency when reordering segments by vram class."
263
+ )
264
+
265
+ return out
266
+
267
+
225
268
  def read_target_binary() -> bytes:
226
269
  rom_bytes = options.opts.target_path.read_bytes()
227
270
 
@@ -317,6 +360,9 @@ def do_split(
317
360
 
318
361
 
319
362
  def write_linker_script(all_segments: List[Segment]) -> LinkerWriter:
363
+ if options.opts.ld_sort_segments_by_vram_class_dependency:
364
+ all_segments = sort_segments_by_vram_class_dependency(all_segments)
365
+
320
366
  vram_class_dependencies = calc_segment_dependences(all_segments)
321
367
  vram_classes_to_search = set(vram_class_dependencies.keys())
322
368
 
@@ -51,7 +51,7 @@ class CommonSegC(CommonSegCodeSubsegment):
51
51
 
52
52
  @staticmethod
53
53
  def get_funcs_defined_in_c(c_file: Path) -> Set[str]:
54
- with open(c_file, "r") as f:
54
+ with open(c_file, "r", encoding="utf-8") as f:
55
55
  text = CommonSegC.strip_c_comments(f.read())
56
56
 
57
57
  return set(m.group(1) for m in C_FUNC_RE.finditer(text))
@@ -105,7 +105,7 @@ class CommonSegC(CommonSegCodeSubsegment):
105
105
 
106
106
  @staticmethod
107
107
  def get_global_asm_funcs(c_file: Path) -> Set[str]:
108
- with c_file.open() as f:
108
+ with c_file.open(encoding="utf-8") as f:
109
109
  text = CommonSegC.strip_c_comments(f.read())
110
110
  if options.opts.compiler == IDO:
111
111
  return set(m.group(2) for m in C_GLOBAL_ASM_IDO_RE.finditer(text))
@@ -114,7 +114,7 @@ class CommonSegC(CommonSegCodeSubsegment):
114
114
 
115
115
  @staticmethod
116
116
  def get_global_asm_rodata_syms(c_file: Path) -> Set[str]:
117
- with c_file.open() as f:
117
+ with c_file.open(encoding="utf-8") as f:
118
118
  text = CommonSegC.strip_c_comments(f.read())
119
119
  if options.opts.compiler == IDO:
120
120
  return set(m.group(2) for m in C_GLOBAL_ASM_IDO_RE.finditer(text))
@@ -152,6 +152,8 @@ class CommonSegC(CommonSegCodeSubsegment):
152
152
 
153
153
  def split(self, rom_bytes: bytes):
154
154
  if self.is_auto_segment:
155
+ if options.opts.make_full_disasm_for_code:
156
+ self.split_as_asmtu_file(self.asm_out_path())
155
157
  return
156
158
 
157
159
  if self.rom_start != self.rom_end:
@@ -252,22 +252,21 @@ class CommonSegCodeSubsegment(Segment):
252
252
  f.write(self.spim_section.disassemble())
253
253
 
254
254
  # Same as above but write all sections from siblings
255
- def split_as_asmtu_file(self, out_path: Optional[Path]):
256
- if self.spim_section is None:
257
- return
258
-
259
- if not out_path:
260
- return
261
-
255
+ def split_as_asmtu_file(self, out_path: Path):
262
256
  out_path.parent.mkdir(parents=True, exist_ok=True)
263
257
 
264
258
  self.print_file_boundaries()
265
259
 
266
260
  with open(out_path, "w", newline="\n") as f:
267
- # Write `.text` contents
268
- for line in self.get_asm_file_header():
269
- f.write(line + "\n")
270
- f.write(self.spim_section.disassemble())
261
+ # self.spim_section would be None if the current section was
262
+ # declared `auto` in the yaml.
263
+ # This can be useful when there are TUs with only data/rodata/bss/etc
264
+ # sections but not text section.
265
+ if self.spim_section is not None:
266
+ # Write `.text` contents
267
+ for line in self.get_asm_file_header():
268
+ f.write(line + "\n")
269
+ f.write(self.spim_section.disassemble())
271
270
 
272
271
  # Disassemble the siblings to this file by respecting the `section_order`
273
272
  for sect in self.section_order:
@@ -281,25 +280,31 @@ class CommonSegCodeSubsegment(Segment):
281
280
  if (
282
281
  isinstance(sibling, CommonSegCodeSubsegment)
283
282
  and sibling.spim_section is not None
284
- and not sibling.should_self_split()
283
+ and (
284
+ not sibling.should_self_split()
285
+ or options.opts.make_full_disasm_for_code
286
+ )
285
287
  ):
286
288
  f.write("\n")
287
289
  f.write(f"{sibling.get_section_asm_line()}\n\n")
288
290
  f.write(sibling.spim_section.disassemble())
289
291
 
290
- # Another loop to check anything that somehow may not be present on the `section_order`
292
+ # Another loop to check anything that somehow may not be present in the `section_order`
291
293
  for sect, sibling in self.siblings.items():
292
294
  if sect == self.get_linker_section_linksection():
293
295
  continue
294
296
 
295
297
  if sect in self.section_order:
296
- # Already handled on the above loop
298
+ # Already handled in the above loop
297
299
  continue
298
300
 
299
301
  if (
300
302
  isinstance(sibling, CommonSegCodeSubsegment)
301
303
  and sibling.spim_section is not None
302
- and not sibling.should_self_split()
304
+ and (
305
+ not sibling.should_self_split()
306
+ or options.opts.make_full_disasm_for_code
307
+ )
303
308
  ):
304
309
  f.write("\n")
305
310
  f.write(f"{sibling.get_section_asm_line()}\n\n")
@@ -42,9 +42,6 @@ def write_include_asm_h():
42
42
  return
43
43
 
44
44
  directory = options.opts.generated_asm_macros_directory.as_posix()
45
- print()
46
- print(directory)
47
- print()
48
45
 
49
46
  if options.opts.include_asm_macro_style == "maspsx_hack":
50
47
  include_asm_macro = """\
splat/util/n64/rominfo.py CHANGED
@@ -162,8 +162,7 @@ class N64EntrypointInfo:
162
162
  register_values[insn.rs.value] + insn.getProcessedImmediate()
163
163
  )
164
164
  completed_pair[insn.rt.value] = True
165
- if not insn.isUnsigned():
166
- lo_assignments[insn.rt.value] = current_rom
165
+ lo_assignments[insn.rt.value] = current_rom
167
166
  elif insn.doesStore():
168
167
  if insn.rt == rabbitizer.RegGprO32.zero:
169
168
  # Try to detect the zero-ing bss algorithm
splat/util/options.py CHANGED
@@ -127,6 +127,8 @@ class SplatOpts:
127
127
  # `vram_symbol` / `follows_classes` (vram_class options) to calculate vram addresses in the linker script.
128
128
  # If disabled, this uses the plain integer values for vram addresses defined in the yaml.
129
129
  ld_use_symbolic_vram_addresses: bool
130
+ # Ensures segments of vram classes with dependencies (`vram_symbol` / `follows_classes`) are written to the linker script AFTER the segments they depend on.
131
+ ld_sort_segments_by_vram_class_dependency: bool
130
132
  # Change linker script generation to allow partially linking segments. Requires both `ld_partial_scripts_path` and `ld_partial_build_segments_path` to be set.
131
133
  ld_partial_linking: bool
132
134
  # Folder were each intermediary linker script will be written to.
@@ -512,6 +514,9 @@ def _parse_yaml(
512
514
  ld_sections_allowlist=p.parse_opt("ld_sections_allowlist", list, []),
513
515
  ld_sections_denylist=p.parse_opt("ld_sections_denylist", list, []),
514
516
  ld_wildcard_sections=p.parse_opt("ld_wildcard_sections", bool, False),
517
+ ld_sort_segments_by_vram_class_dependency=p.parse_opt(
518
+ "ld_sort_segments_by_vram_class_dependency", bool, False
519
+ ),
515
520
  ld_use_symbolic_vram_addresses=p.parse_opt(
516
521
  "ld_use_symbolic_vram_addresses", bool, True
517
522
  ),
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: splat64
3
- Version: 0.37.0
3
+ Version: 0.37.2
4
4
  Summary: A binary splitting tool to assist with decompilation and modding projects
5
5
  Project-URL: Repository, https://github.com/ethteck/splat
6
6
  Project-URL: Issues, https://github.com/ethteck/splat/issues
@@ -76,7 +76,7 @@ The brackets corresponds to the optional dependencies to install while installin
76
76
  If you use a `requirements.txt` file in your repository, then you can add this library with the following line:
77
77
 
78
78
  ```txt
79
- splat64[mips]>=0.37.0,<1.0.0
79
+ splat64[mips]>=0.37.2,<1.0.0
80
80
  ```
81
81
 
82
82
  ### Optional dependencies
@@ -1,4 +1,4 @@
1
- splat/__init__.py,sha256=rvIlPAMo8x-ro2Ehl1n7-IVrTyHavDimSnE80nIzgno,291
1
+ splat/__init__.py,sha256=-m9-FH6WSFV9bHUlbGLqooheqgbSGqaA_-Y0hfB1p9Y,291
2
2
  splat/__main__.py,sha256=T333dHDgr-2HYYhtARnYMEjdECnYiNIKfcXDD99o22A,732
3
3
  splat/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
4
4
  splat/disassembler/__init__.py,sha256=LXuCElHxOvbb9xuSSeswT8W_v78chwtqjKq_9nxFVzQ,167
@@ -15,7 +15,7 @@ splat/platforms/psx.py,sha256=YxQERdOBr4p3ab9Wk80FNhVYi-uvmh7p_jeykSFp23M,40
15
15
  splat/scripts/__init__.py,sha256=OY0nHg6a7JB437Sb96qLbZ_7ByVsal1gStj35wJAI_Y,101
16
16
  splat/scripts/capy.py,sha256=svbOfLO34-QN3xLiBy9vk2RGs_To8TWMWEKBw6yx2xQ,3674
17
17
  splat/scripts/create_config.py,sha256=Xt0BgS9ql5GfLrNKWQRMuKWea0sXUygRV1_wCYGrrPE,17949
18
- splat/scripts/split.py,sha256=rstKTy1osrD4Cqp07NGNVpLeBVQSNbeeTYC-AORrSJs,21937
18
+ splat/scripts/split.py,sha256=mOqaaXWX47EQpZiAmfMFLFpEBN5-KsZ_Uk2sl1SNh7k,23334
19
19
  splat/segtypes/__init__.py,sha256=-upUw_4JGQtvyp6IfTMzOq_CK3xvVaT_0K0_EipHyOo,208
20
20
  splat/segtypes/linker_entry.py,sha256=WHFdVFC_NW6_ub-LJbfwze-FZGCTRtjEZHz_bhIkX-M,25160
21
21
  splat/segtypes/segment.py,sha256=IcmF7FxDrJ1ZP5AKFO-skNrGfYhGIYwawyepKzeTF3A,30619
@@ -24,9 +24,9 @@ splat/segtypes/common/asm.py,sha256=gpR4uZ57l4Q9G0XTOeO1JIk9P5d8xXBxRfkrPIN-M1A,
24
24
  splat/segtypes/common/asmtu.py,sha256=C52kKh-8YeDHu0EucEfQ-tQMtDgfKfwAJ6wwiW6nOBU,354
25
25
  splat/segtypes/common/bin.py,sha256=qZeuvHNSrRuxdk-8V0xW0tcFhtzPk6doyf43E_t7D9Q,1250
26
26
  splat/segtypes/common/bss.py,sha256=cX0RO7LRcBRO6t463AUSSV8sGdlxZTcd5_McZjjqfFo,3107
27
- splat/segtypes/common/c.py,sha256=RkevmqKXK9RKQ_yPClFFur1Uq4jGGdVtL1BnScbHhok,21327
27
+ splat/segtypes/common/c.py,sha256=uKc8V4VSz2bRkt2g7CVbKibRdp2R1Ro9UAxBVLyB8M8,21494
28
28
  splat/segtypes/common/code.py,sha256=FfBuy6x8fQXoJYX2Ul80uOs-CzlVslUaveD-hBmvKXM,10693
29
- splat/segtypes/common/codesubsegment.py,sha256=XpPzNwGSNL4HE7Qgi7-tSVXlYtfdXug9PrzQIJu3yDI,10720
29
+ splat/segtypes/common/codesubsegment.py,sha256=a9AGNfWrQRfMVKP84_a_2BBfQSwOVviAltp8ZjOivT0,11140
30
30
  splat/segtypes/common/cpp.py,sha256=p-duowCt4czrmaWgj1LuFw5LGfdyB_uaeLTKVmH80Cg,180
31
31
  splat/segtypes/common/data.py,sha256=l3CCa95vHH8XwiksamaxuZuSF_rOGqcozGlZj0bt9Yw,5712
32
32
  splat/segtypes/common/databin.py,sha256=nDI1ciAv0QLt2xDYjEtGJHSoONZzJZKyST-BuSKML2Q,1169
@@ -80,9 +80,9 @@ splat/util/cache_handler.py,sha256=BrTWo8U4bj8TBcZfIRwiAYmogc3YUlBz9P-34Y0aIrg,1
80
80
  splat/util/color.py,sha256=FSmy0dAQJ9FzRBc99Yt4kBEyB62MC_YiVkqoWgPMsRU,371
81
81
  splat/util/compiler.py,sha256=n5hlxazUy8rq-t7DADBbC0n6ZGuLDzp3ma3883oHQ3E,1848
82
82
  splat/util/conf.py,sha256=0Qcv5d5XAvdVzqF4i4L_lJuC5hI0PEufeob58YaN-xs,3230
83
- splat/util/file_presets.py,sha256=BJh56jCpJCFUqEYTA6xiAnJlyFInVzqtRGUY6eGqz-w,19030
83
+ splat/util/file_presets.py,sha256=4UsDMqQy5ZACSafjCba1pwwTQKK_2leOycwTFkNG4Ac,18985
84
84
  splat/util/log.py,sha256=aJA1rg8IirJu1wGzjNuATHvepYvD3k5CtEyMasyJWxI,1193
85
- splat/util/options.py,sha256=Lq1noBa-2f_j7lSvFfOyq4b2rFhBuCvB-M_s1Czu1Pg,31931
85
+ splat/util/options.py,sha256=wPW2T1jZ3m6fAgVnVfsnvAH0ZK4pF_-mlVP967RdtNU,32287
86
86
  splat/util/palettes.py,sha256=d3KoZnwt-zunI9eNwb3txysXg4DY3xnF0O5aQRxM4so,2920
87
87
  splat/util/progress_bar.py,sha256=41VehpIFK1cphENaXV_Aq6_3mFx25eQ8V7Z51SKFPeM,166
88
88
  splat/util/relocs.py,sha256=j8fzM9u0ZQwZa1b1Dd26ho9GwIooBXt8mE0jAN7wqV0,4153
@@ -92,13 +92,13 @@ splat/util/utils.py,sha256=kAC0DAjaGa0Kq2k86RfQtFkheV87XuyeOgZIVIk3CHE,208
92
92
  splat/util/vram_classes.py,sha256=UH4rkugEwoec_-alJ4smNOcnU51G-V5OG7Pfsj47eaM,3491
93
93
  splat/util/n64/__init__.py,sha256=hsBkPh6NUz-bW7HVspcNZ0mCxBhdfcPC07_7gbga95o,84
94
94
  splat/util/n64/find_code_length.py,sha256=uUoPoUORAjsAvH8oGqwnGvw6j8I_NnSrZtA-x9h9h7E,1414
95
- splat/util/n64/rominfo.py,sha256=Ms9P-MuuOSyEF4QNB3wBuC-U9OdrfByqR0VPAHqTBQI,16976
95
+ splat/util/n64/rominfo.py,sha256=o3c3m8GG8iCu8ruohfrYRmlTjXo1W8gZJHEVHcH_QGc,16926
96
96
  splat/util/ps2/__init__.py,sha256=hAU12HtQLOc_sxc8_rzxOERBT5wjg3DhQ8XRN9hBXfA,39
97
97
  splat/util/ps2/ps2elfinfo.py,sha256=v6nTPabzXpG1N-MhhkbVeLoGjgsjBEpHL1DoPv26-Hs,7982
98
98
  splat/util/psx/__init__.py,sha256=kCCaR-KB1mNlIcXB4OuuSQ2zVLbWg_SnIZIUeyjeBBI,39
99
99
  splat/util/psx/psxexeinfo.py,sha256=Oxd5Lt8HTNHuUzw3cVujdyJRqEG-yo0XT78wEISUXms,5705
100
- splat64-0.37.0.dist-info/METADATA,sha256=Q-9gJotMBsqiMMtThLZaPEgL5g8uzaw9zm_QJd033po,3879
101
- splat64-0.37.0.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
102
- splat64-0.37.0.dist-info/entry_points.txt,sha256=O7Xy-qNOHcI87-OQrWJ-OhRDws74SuwVb_4rtnp0eLo,52
103
- splat64-0.37.0.dist-info/licenses/LICENSE,sha256=97VMVzjG8yQvsf8NG2M9IFSbh7R8cifJnc6QK1cZqj8,1070
104
- splat64-0.37.0.dist-info/RECORD,,
100
+ splat64-0.37.2.dist-info/METADATA,sha256=BEiQ_5Eq0N_JpZfq2YdJ8cv7XjYfhxzqfOb4n9SCjIY,3879
101
+ splat64-0.37.2.dist-info/WHEEL,sha256=WLgqFyCfm_KASv4WHyYy0P3pM_m7J5L9k2skdKLirC8,87
102
+ splat64-0.37.2.dist-info/entry_points.txt,sha256=O7Xy-qNOHcI87-OQrWJ-OhRDws74SuwVb_4rtnp0eLo,52
103
+ splat64-0.37.2.dist-info/licenses/LICENSE,sha256=97VMVzjG8yQvsf8NG2M9IFSbh7R8cifJnc6QK1cZqj8,1070
104
+ splat64-0.37.2.dist-info/RECORD,,