gemba 0.1.1 → 0.2.0

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.
Files changed (285) hide show
  1. checksums.yaml +4 -4
  2. data/THIRD_PARTY_NOTICES +37 -2
  3. data/assets/placeholder_boxart.png +0 -0
  4. data/bin/gemba +2 -2
  5. data/ext/gemba/extconf.rb +23 -1
  6. data/ext/gemba/gemba_ext.c +436 -2
  7. data/ext/gemba/gemba_ext.h +2 -0
  8. data/gemba.gemspec +5 -3
  9. data/lib/gemba/achievements/achievement.rb +23 -0
  10. data/lib/gemba/achievements/backend.rb +186 -0
  11. data/lib/gemba/achievements/cache.rb +70 -0
  12. data/lib/gemba/achievements/credentials_presenter.rb +142 -0
  13. data/lib/gemba/achievements/fake_backend.rb +205 -0
  14. data/lib/gemba/achievements/null_backend.rb +11 -0
  15. data/lib/gemba/achievements/offline_backend.rb +168 -0
  16. data/lib/gemba/achievements/retro_achievements/backend.rb +453 -0
  17. data/lib/gemba/achievements/retro_achievements/cli_sync_requester.rb +64 -0
  18. data/lib/gemba/achievements/retro_achievements/ping_worker.rb +27 -0
  19. data/lib/gemba/achievements.rb +19 -0
  20. data/lib/gemba/achievements_window.rb +556 -0
  21. data/lib/gemba/app_controller.rb +1015 -0
  22. data/lib/gemba/bios.rb +54 -0
  23. data/lib/gemba/boxart_fetcher/libretro_backend.rb +39 -0
  24. data/lib/gemba/boxart_fetcher/null_backend.rb +12 -0
  25. data/lib/gemba/boxart_fetcher.rb +79 -0
  26. data/lib/gemba/bus_emitter.rb +13 -0
  27. data/lib/gemba/child_window.rb +24 -1
  28. data/lib/gemba/cli/commands/config_cmd.rb +83 -0
  29. data/lib/gemba/cli/commands/decode.rb +154 -0
  30. data/lib/gemba/cli/commands/patch.rb +78 -0
  31. data/lib/gemba/cli/commands/play.rb +78 -0
  32. data/lib/gemba/cli/commands/record.rb +114 -0
  33. data/lib/gemba/cli/commands/replay.rb +161 -0
  34. data/lib/gemba/cli/commands/retro_achievements.rb +213 -0
  35. data/lib/gemba/cli/commands/version.rb +22 -0
  36. data/lib/gemba/cli.rb +52 -364
  37. data/lib/gemba/config.rb +134 -1
  38. data/lib/gemba/data/gb_games.json +1 -0
  39. data/lib/gemba/data/gb_md5.json +1 -0
  40. data/lib/gemba/data/gba_games.json +1 -0
  41. data/lib/gemba/data/gba_md5.json +1 -0
  42. data/lib/gemba/data/gbc_games.json +1 -0
  43. data/lib/gemba/data/gbc_md5.json +1 -0
  44. data/lib/gemba/emulator_frame.rb +1060 -0
  45. data/lib/gemba/event_bus.rb +48 -0
  46. data/lib/gemba/frame_stack.rb +60 -0
  47. data/lib/gemba/game_index.rb +84 -0
  48. data/lib/gemba/game_picker_frame.rb +268 -0
  49. data/lib/gemba/gamepad_map.rb +103 -0
  50. data/lib/gemba/headless.rb +6 -5
  51. data/lib/gemba/headless_player.rb +33 -3
  52. data/lib/gemba/help_window.rb +61 -0
  53. data/lib/gemba/hotkey_map.rb +3 -1
  54. data/lib/gemba/input_recorder.rb +107 -0
  55. data/lib/gemba/input_replayer.rb +119 -0
  56. data/lib/gemba/keyboard_map.rb +90 -0
  57. data/lib/gemba/locales/en.yml +97 -5
  58. data/lib/gemba/locales/ja.yml +97 -5
  59. data/lib/gemba/main_window.rb +56 -0
  60. data/lib/gemba/modal_stack.rb +81 -0
  61. data/lib/gemba/patcher_window.rb +223 -0
  62. data/lib/gemba/platform/gb.rb +21 -0
  63. data/lib/gemba/platform/gba.rb +21 -0
  64. data/lib/gemba/platform/gbc.rb +23 -0
  65. data/lib/gemba/platform.rb +20 -0
  66. data/lib/gemba/platform_open.rb +19 -0
  67. data/lib/gemba/recorder.rb +4 -3
  68. data/lib/gemba/replay_player.rb +691 -0
  69. data/lib/gemba/rom_info.rb +57 -0
  70. data/lib/gemba/rom_info_window.rb +16 -3
  71. data/lib/gemba/rom_library.rb +106 -0
  72. data/lib/gemba/rom_overrides.rb +47 -0
  73. data/lib/gemba/rom_patcher/bps.rb +161 -0
  74. data/lib/gemba/rom_patcher/ips.rb +101 -0
  75. data/lib/gemba/rom_patcher/ups.rb +118 -0
  76. data/lib/gemba/rom_patcher.rb +109 -0
  77. data/lib/gemba/{rom_loader.rb → rom_resolver.rb} +7 -6
  78. data/lib/gemba/runtime.rb +59 -26
  79. data/lib/gemba/save_state_manager.rb +4 -7
  80. data/lib/gemba/save_state_picker.rb +17 -4
  81. data/lib/gemba/session_logger.rb +64 -0
  82. data/lib/gemba/settings/audio_tab.rb +77 -0
  83. data/lib/gemba/settings/gamepad_tab.rb +351 -0
  84. data/lib/gemba/settings/hotkeys_tab.rb +259 -0
  85. data/lib/gemba/settings/paths.rb +11 -0
  86. data/lib/gemba/settings/recording_tab.rb +83 -0
  87. data/lib/gemba/settings/save_states_tab.rb +91 -0
  88. data/lib/gemba/settings/system_tab.rb +362 -0
  89. data/lib/gemba/settings/video_tab.rb +318 -0
  90. data/lib/gemba/settings_window.rb +162 -1036
  91. data/lib/gemba/version.rb +1 -1
  92. data/lib/gemba/virtual_keyboard.rb +19 -0
  93. data/lib/gemba.rb +2 -12
  94. data/test/achievements_window/test_bulk_sync.rb +218 -0
  95. data/test/achievements_window/test_bus_events.rb +125 -0
  96. data/test/achievements_window/test_close_confirmation.rb +201 -0
  97. data/test/achievements_window/test_initial_state.rb +164 -0
  98. data/test/achievements_window/test_sorting.rb +227 -0
  99. data/test/achievements_window/test_tree_rendering.rb +133 -0
  100. data/test/fixtures/fake_bios.bin +0 -0
  101. data/test/fixtures/pong.gba +0 -0
  102. data/test/fixtures/test.gb +0 -0
  103. data/test/fixtures/test.gbc +0 -0
  104. data/test/fixtures/test_quicksave.ss +0 -0
  105. data/test/screenshots/no_focus.png +0 -0
  106. data/test/shared/teek_test_worker.rb +17 -1
  107. data/test/shared/tk_test_helper.rb +91 -4
  108. data/test/support/achievements_window_helpers.rb +18 -0
  109. data/test/support/fake_core.rb +25 -0
  110. data/test/support/fake_ra_runtime.rb +74 -0
  111. data/test/support/fake_requester.rb +68 -0
  112. data/test/support/player_helpers.rb +20 -5
  113. data/test/test_achievement.rb +32 -0
  114. data/test/{test_player.rb → test_app_controller.rb} +353 -85
  115. data/test/test_bios.rb +123 -0
  116. data/test/test_boxart_fetcher.rb +150 -0
  117. data/test/test_cli.rb +17 -265
  118. data/test/test_cli_config.rb +64 -0
  119. data/test/test_cli_decode.rb +97 -0
  120. data/test/test_cli_patch.rb +58 -0
  121. data/test/test_cli_play.rb +213 -0
  122. data/test/test_cli_ra.rb +175 -0
  123. data/test/test_cli_record.rb +69 -0
  124. data/test/test_cli_replay.rb +72 -0
  125. data/test/test_cli_sync_requester.rb +152 -0
  126. data/test/test_cli_version.rb +27 -0
  127. data/test/test_config.rb +2 -3
  128. data/test/test_config_ra.rb +69 -0
  129. data/test/test_core.rb +62 -1
  130. data/test/test_credentials_presenter.rb +192 -0
  131. data/test/test_event_bus.rb +100 -0
  132. data/test/test_fake_backend_achievements.rb +130 -0
  133. data/test/test_fake_backend_auth.rb +68 -0
  134. data/test/test_game_index.rb +77 -0
  135. data/test/test_game_picker_frame.rb +310 -0
  136. data/test/test_gamepad_map.rb +1 -3
  137. data/test/test_headless_player.rb +17 -3
  138. data/test/test_help_window.rb +82 -0
  139. data/test/test_hotkey_map.rb +22 -1
  140. data/test/test_input_recorder.rb +179 -0
  141. data/test/test_input_replay_determinism.rb +113 -0
  142. data/test/test_input_replayer.rb +162 -0
  143. data/test/test_keyboard_map.rb +1 -3
  144. data/test/test_libretro_backend.rb +41 -0
  145. data/test/test_locale.rb +1 -1
  146. data/test/test_logging.rb +123 -0
  147. data/test/test_null_backend.rb +42 -0
  148. data/test/test_offline_backend.rb +116 -0
  149. data/test/test_overlay_renderer.rb +1 -1
  150. data/test/test_platform.rb +149 -0
  151. data/test/test_ra_backend.rb +313 -0
  152. data/test/test_ra_backend_unlock_gate.rb +56 -0
  153. data/test/test_recorder.rb +0 -3
  154. data/test/test_replay_player.rb +316 -0
  155. data/test/test_rom_info.rb +149 -0
  156. data/test/test_rom_overrides.rb +86 -0
  157. data/test/test_rom_patcher.rb +382 -0
  158. data/test/{test_rom_loader.rb → test_rom_resolver.rb} +25 -26
  159. data/test/test_save_state_manager.rb +2 -4
  160. data/test/test_settings_audio.rb +107 -0
  161. data/test/test_settings_hotkeys.rb +83 -66
  162. data/test/test_settings_recording.rb +49 -0
  163. data/test/test_settings_save_states.rb +97 -0
  164. data/test/test_settings_system.rb +133 -0
  165. data/test/test_settings_video.rb +450 -0
  166. data/test/test_settings_window.rb +76 -507
  167. data/test/test_tip_service.rb +6 -6
  168. data/test/test_toast_overlay.rb +1 -1
  169. data/test/test_virtual_events.rb +156 -0
  170. data/test/test_virtual_keyboard.rb +1 -1
  171. data/vendor/rcheevos/CHANGELOG.md +495 -0
  172. data/vendor/rcheevos/LICENSE +21 -0
  173. data/vendor/rcheevos/Package.swift +33 -0
  174. data/vendor/rcheevos/README.md +67 -0
  175. data/vendor/rcheevos/include/module.modulemap +70 -0
  176. data/vendor/rcheevos/include/rc_api_editor.h +296 -0
  177. data/vendor/rcheevos/include/rc_api_info.h +280 -0
  178. data/vendor/rcheevos/include/rc_api_request.h +77 -0
  179. data/vendor/rcheevos/include/rc_api_runtime.h +417 -0
  180. data/vendor/rcheevos/include/rc_api_user.h +262 -0
  181. data/vendor/rcheevos/include/rc_client.h +877 -0
  182. data/vendor/rcheevos/include/rc_client_raintegration.h +101 -0
  183. data/vendor/rcheevos/include/rc_consoles.h +138 -0
  184. data/vendor/rcheevos/include/rc_error.h +59 -0
  185. data/vendor/rcheevos/include/rc_export.h +100 -0
  186. data/vendor/rcheevos/include/rc_hash.h +200 -0
  187. data/vendor/rcheevos/include/rc_runtime.h +148 -0
  188. data/vendor/rcheevos/include/rc_runtime_types.h +452 -0
  189. data/vendor/rcheevos/include/rc_util.h +51 -0
  190. data/vendor/rcheevos/include/rcheevos.h +8 -0
  191. data/vendor/rcheevos/src/rapi/rc_api_common.c +1379 -0
  192. data/vendor/rcheevos/src/rapi/rc_api_common.h +88 -0
  193. data/vendor/rcheevos/src/rapi/rc_api_editor.c +625 -0
  194. data/vendor/rcheevos/src/rapi/rc_api_info.c +587 -0
  195. data/vendor/rcheevos/src/rapi/rc_api_runtime.c +901 -0
  196. data/vendor/rcheevos/src/rapi/rc_api_user.c +483 -0
  197. data/vendor/rcheevos/src/rc_client.c +6941 -0
  198. data/vendor/rcheevos/src/rc_client_external.c +281 -0
  199. data/vendor/rcheevos/src/rc_client_external.h +177 -0
  200. data/vendor/rcheevos/src/rc_client_external_versions.h +171 -0
  201. data/vendor/rcheevos/src/rc_client_internal.h +409 -0
  202. data/vendor/rcheevos/src/rc_client_raintegration.c +566 -0
  203. data/vendor/rcheevos/src/rc_client_raintegration_internal.h +61 -0
  204. data/vendor/rcheevos/src/rc_client_types.natvis +396 -0
  205. data/vendor/rcheevos/src/rc_compat.c +251 -0
  206. data/vendor/rcheevos/src/rc_compat.h +121 -0
  207. data/vendor/rcheevos/src/rc_libretro.c +915 -0
  208. data/vendor/rcheevos/src/rc_libretro.h +98 -0
  209. data/vendor/rcheevos/src/rc_util.c +199 -0
  210. data/vendor/rcheevos/src/rc_version.c +11 -0
  211. data/vendor/rcheevos/src/rc_version.h +32 -0
  212. data/vendor/rcheevos/src/rcheevos/alloc.c +312 -0
  213. data/vendor/rcheevos/src/rcheevos/condition.c +754 -0
  214. data/vendor/rcheevos/src/rcheevos/condset.c +777 -0
  215. data/vendor/rcheevos/src/rcheevos/consoleinfo.c +1215 -0
  216. data/vendor/rcheevos/src/rcheevos/format.c +330 -0
  217. data/vendor/rcheevos/src/rcheevos/lboard.c +287 -0
  218. data/vendor/rcheevos/src/rcheevos/memref.c +805 -0
  219. data/vendor/rcheevos/src/rcheevos/operand.c +607 -0
  220. data/vendor/rcheevos/src/rcheevos/rc_internal.h +390 -0
  221. data/vendor/rcheevos/src/rcheevos/rc_runtime_types.natvis +541 -0
  222. data/vendor/rcheevos/src/rcheevos/rc_validate.c +1406 -0
  223. data/vendor/rcheevos/src/rcheevos/rc_validate.h +18 -0
  224. data/vendor/rcheevos/src/rcheevos/richpresence.c +922 -0
  225. data/vendor/rcheevos/src/rcheevos/runtime.c +852 -0
  226. data/vendor/rcheevos/src/rcheevos/runtime_progress.c +1073 -0
  227. data/vendor/rcheevos/src/rcheevos/trigger.c +344 -0
  228. data/vendor/rcheevos/src/rcheevos/value.c +935 -0
  229. data/vendor/rcheevos/src/rhash/aes.c +480 -0
  230. data/vendor/rcheevos/src/rhash/aes.h +49 -0
  231. data/vendor/rcheevos/src/rhash/cdreader.c +838 -0
  232. data/vendor/rcheevos/src/rhash/hash.c +1402 -0
  233. data/vendor/rcheevos/src/rhash/hash_disc.c +1340 -0
  234. data/vendor/rcheevos/src/rhash/hash_encrypted.c +566 -0
  235. data/vendor/rcheevos/src/rhash/hash_rom.c +426 -0
  236. data/vendor/rcheevos/src/rhash/hash_zip.c +460 -0
  237. data/vendor/rcheevos/src/rhash/md5.c +382 -0
  238. data/vendor/rcheevos/src/rhash/md5.h +91 -0
  239. data/vendor/rcheevos/src/rhash/rc_hash_internal.h +116 -0
  240. data/vendor/rcheevos/test/libretro.h +205 -0
  241. data/vendor/rcheevos/test/rapi/test_rc_api_common.c +941 -0
  242. data/vendor/rcheevos/test/rapi/test_rc_api_editor.c +931 -0
  243. data/vendor/rcheevos/test/rapi/test_rc_api_info.c +545 -0
  244. data/vendor/rcheevos/test/rapi/test_rc_api_runtime.c +2213 -0
  245. data/vendor/rcheevos/test/rapi/test_rc_api_user.c +998 -0
  246. data/vendor/rcheevos/test/rcheevos/mock_memory.h +32 -0
  247. data/vendor/rcheevos/test/rcheevos/test_condition.c +570 -0
  248. data/vendor/rcheevos/test/rcheevos/test_condset.c +5170 -0
  249. data/vendor/rcheevos/test/rcheevos/test_consoleinfo.c +203 -0
  250. data/vendor/rcheevos/test/rcheevos/test_format.c +112 -0
  251. data/vendor/rcheevos/test/rcheevos/test_lboard.c +746 -0
  252. data/vendor/rcheevos/test/rcheevos/test_memref.c +520 -0
  253. data/vendor/rcheevos/test/rcheevos/test_operand.c +692 -0
  254. data/vendor/rcheevos/test/rcheevos/test_rc_validate.c +502 -0
  255. data/vendor/rcheevos/test/rcheevos/test_richpresence.c +1564 -0
  256. data/vendor/rcheevos/test/rcheevos/test_runtime.c +1667 -0
  257. data/vendor/rcheevos/test/rcheevos/test_runtime_progress.c +1821 -0
  258. data/vendor/rcheevos/test/rcheevos/test_timing.c +166 -0
  259. data/vendor/rcheevos/test/rcheevos/test_trigger.c +2521 -0
  260. data/vendor/rcheevos/test/rcheevos/test_value.c +870 -0
  261. data/vendor/rcheevos/test/rcheevos-test.sln +46 -0
  262. data/vendor/rcheevos/test/rcheevos-test.vcxproj +239 -0
  263. data/vendor/rcheevos/test/rcheevos-test.vcxproj.filters +335 -0
  264. data/vendor/rcheevos/test/rhash/data.c +657 -0
  265. data/vendor/rcheevos/test/rhash/data.h +32 -0
  266. data/vendor/rcheevos/test/rhash/mock_filereader.c +236 -0
  267. data/vendor/rcheevos/test/rhash/mock_filereader.h +31 -0
  268. data/vendor/rcheevos/test/rhash/test_cdreader.c +920 -0
  269. data/vendor/rcheevos/test/rhash/test_hash.c +310 -0
  270. data/vendor/rcheevos/test/rhash/test_hash_disc.c +1450 -0
  271. data/vendor/rcheevos/test/rhash/test_hash_rom.c +899 -0
  272. data/vendor/rcheevos/test/rhash/test_hash_zip.c +551 -0
  273. data/vendor/rcheevos/test/test.c +113 -0
  274. data/vendor/rcheevos/test/test_framework.h +205 -0
  275. data/vendor/rcheevos/test/test_rc_client.c +10509 -0
  276. data/vendor/rcheevos/test/test_rc_client_external.c +2197 -0
  277. data/vendor/rcheevos/test/test_rc_client_raintegration.c +441 -0
  278. data/vendor/rcheevos/test/test_rc_libretro.c +952 -0
  279. data/vendor/rcheevos/test/test_types.natvis +9 -0
  280. data/vendor/rcheevos/validator/validator.c +658 -0
  281. data/vendor/rcheevos/validator/validator.vcxproj +152 -0
  282. data/vendor/rcheevos/validator/validator.vcxproj.filters +82 -0
  283. metadata +274 -11
  284. data/lib/gemba/input_mappings.rb +0 -214
  285. data/lib/gemba/player.rb +0 -1525
@@ -0,0 +1,1340 @@
1
+ #include "rc_hash.h"
2
+
3
+ #include "rc_hash_internal.h"
4
+
5
+ #include "../rc_compat.h"
6
+
7
+ #include <ctype.h>
8
+
9
+ /* ===================================================== */
10
+
11
+ static struct rc_hash_cdreader g_cdreader_funcs;
12
+ static struct rc_hash_cdreader* g_cdreader = NULL;
13
+
14
+ void rc_hash_reset_iterator_disc(rc_hash_iterator_t* iterator)
15
+ {
16
+ if (g_cdreader)
17
+ memcpy(&iterator->callbacks.cdreader, g_cdreader, sizeof(*g_cdreader));
18
+ else
19
+ rc_hash_get_default_cdreader(&iterator->callbacks.cdreader);
20
+ }
21
+
22
+ void rc_hash_init_custom_cdreader(struct rc_hash_cdreader* reader)
23
+ {
24
+ if (reader) {
25
+ memcpy(&g_cdreader_funcs, reader, sizeof(g_cdreader_funcs));
26
+ g_cdreader = &g_cdreader_funcs;
27
+ }
28
+ else {
29
+ g_cdreader = NULL;
30
+ }
31
+ }
32
+
33
+ static void* rc_cd_open_track(const rc_hash_iterator_t* iterator, uint32_t track)
34
+ {
35
+ if (iterator->callbacks.cdreader.open_track_iterator)
36
+ return iterator->callbacks.cdreader.open_track_iterator(iterator->path, track, iterator);
37
+
38
+ if (iterator->callbacks.cdreader.open_track)
39
+ return iterator->callbacks.cdreader.open_track(iterator->path, track);
40
+
41
+ if (g_cdreader && g_cdreader->open_track)
42
+ return g_cdreader->open_track(iterator->path, track);
43
+
44
+ rc_hash_iterator_error(iterator, "no hook registered for cdreader_open_track");
45
+ return NULL;
46
+ }
47
+
48
+ static size_t rc_cd_read_sector(const rc_hash_iterator_t* iterator, void* track_handle, uint32_t sector, void* buffer, size_t requested_bytes)
49
+ {
50
+ if (iterator->callbacks.cdreader.read_sector)
51
+ return iterator->callbacks.cdreader.read_sector(track_handle, sector, buffer, requested_bytes);
52
+
53
+ if (g_cdreader && g_cdreader->read_sector)
54
+ return g_cdreader->read_sector(track_handle, sector, buffer, requested_bytes);
55
+
56
+ rc_hash_iterator_error(iterator, "no hook registered for cdreader_read_sector");
57
+ return 0;
58
+ }
59
+
60
+ static uint32_t rc_cd_first_track_sector(const rc_hash_iterator_t* iterator, void* track_handle)
61
+ {
62
+ if (iterator->callbacks.cdreader.first_track_sector)
63
+ return iterator->callbacks.cdreader.first_track_sector(track_handle);
64
+
65
+ if (g_cdreader && g_cdreader->first_track_sector)
66
+ return g_cdreader->first_track_sector(track_handle);
67
+
68
+ rc_hash_iterator_error(iterator, "no hook registered for cdreader_first_track_sector");
69
+ return 0;
70
+ }
71
+
72
+ static void rc_cd_close_track(const rc_hash_iterator_t* iterator, void* track_handle)
73
+ {
74
+ if (iterator->callbacks.cdreader.close_track)
75
+ iterator->callbacks.cdreader.close_track(track_handle);
76
+ else if (g_cdreader && g_cdreader->close_track)
77
+ g_cdreader->close_track(track_handle);
78
+ else
79
+ rc_hash_iterator_error(iterator, "no hook registered for cdreader_close_track");
80
+ }
81
+
82
+ static uint32_t rc_cd_find_file_sector(const rc_hash_iterator_t* iterator, void* track_handle, const char* path, uint32_t* size)
83
+ {
84
+ uint8_t buffer[2048], *tmp;
85
+ int sector;
86
+ uint32_t num_sectors = 0;
87
+ size_t filename_length;
88
+ const char* slash;
89
+
90
+ if (!track_handle)
91
+ return 0;
92
+
93
+ /* we start at the root. don't need to explicitly find it */
94
+ if (*path == '\\')
95
+ ++path;
96
+
97
+ filename_length = strlen(path);
98
+ slash = strrchr(path, '\\');
99
+ if (slash) {
100
+ /* find the directory record for the first part of the path */
101
+ memcpy(buffer, path, slash - path);
102
+ buffer[slash - path] = '\0';
103
+
104
+ sector = rc_cd_find_file_sector(iterator, track_handle, (const char *)buffer, NULL);
105
+ if (!sector)
106
+ return 0;
107
+
108
+ ++slash;
109
+ filename_length -= (slash - path);
110
+ path = slash;
111
+ }
112
+ else {
113
+ uint32_t logical_block_size;
114
+
115
+ /* find the cd information */
116
+ if (!rc_cd_read_sector(iterator, track_handle, rc_cd_first_track_sector(iterator, track_handle) + 16, buffer, 256))
117
+ return 0;
118
+
119
+ /* the directory_record starts at 156, the sector containing the table of contents is 2 bytes into that.
120
+ * https://www.cdroller.com/htm/readdata.html
121
+ */
122
+ sector = buffer[156 + 2] | (buffer[156 + 3] << 8) | (buffer[156 + 4] << 16);
123
+
124
+ /* if the table of contents spans more than one sector, it's length of section will exceed it's logical block size */
125
+ logical_block_size = (buffer[128] | (buffer[128 + 1] << 8)); /* logical block size */
126
+ if (logical_block_size == 0) {
127
+ num_sectors = 1;
128
+ } else {
129
+ num_sectors = (buffer[156 + 10] | (buffer[156 + 11] << 8) | (buffer[156 + 12] << 16) | (buffer[156 + 13] << 24)); /* length of section */
130
+ num_sectors /= logical_block_size;
131
+ }
132
+ }
133
+
134
+ /* fetch and process the directory record */
135
+ if (!rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer)))
136
+ return 0;
137
+
138
+ tmp = buffer;
139
+ do {
140
+ if (tmp >= buffer + sizeof(buffer) || !*tmp) {
141
+ /* end of this path table block. if the path table spans multiple sectors, keep scanning */
142
+ if (num_sectors > 1) {
143
+ --num_sectors;
144
+ if (rc_cd_read_sector(iterator, track_handle, ++sector, buffer, sizeof(buffer))) {
145
+ tmp = buffer;
146
+ continue;
147
+ }
148
+ }
149
+ break;
150
+ }
151
+
152
+ /* filename is 33 bytes into the record and the format is "FILENAME;version" or "DIRECTORY" */
153
+ if ((tmp[32] == filename_length || tmp[33 + filename_length] == ';') &&
154
+ strncasecmp((const char*)(tmp + 33), path, filename_length) == 0) {
155
+ sector = tmp[2] | (tmp[3] << 8) | (tmp[4] << 16);
156
+
157
+ rc_hash_iterator_verbose_formatted(iterator, "Found %s at sector %d", path, sector);
158
+
159
+ if (size)
160
+ *size = tmp[10] | (tmp[11] << 8) | (tmp[12] << 16) | (tmp[13] << 24);
161
+
162
+ return sector;
163
+ }
164
+
165
+ /* the first byte of the record is the length of the record */
166
+ tmp += *tmp;
167
+ } while (1);
168
+
169
+ return 0;
170
+ }
171
+
172
+ /* ===================================================== */
173
+
174
+ static int rc_hash_cd_file(md5_state_t* md5, const rc_hash_iterator_t* iterator, void* track_handle, uint32_t sector, const char* name, uint32_t size, const char* description)
175
+ {
176
+ uint8_t buffer[2048];
177
+ size_t num_read;
178
+
179
+ if ((num_read = rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer))) < sizeof(buffer))
180
+ return rc_hash_iterator_error_formatted(iterator, "Could not read %s", description);
181
+
182
+ if (size > MAX_BUFFER_SIZE)
183
+ size = MAX_BUFFER_SIZE;
184
+
185
+ if (name)
186
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %s title (%u bytes) and contents (%u bytes) ", name, (unsigned)strlen(name), size);
187
+ else
188
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %s contents (%u bytes @ sector %u)", description, size, sector);
189
+
190
+ if (size < (unsigned)num_read) /* we read a whole sector - only hash the part containing file data */
191
+ num_read = (size_t)size;
192
+
193
+ do {
194
+ md5_append(md5, buffer, (int)num_read);
195
+
196
+ if (size <= (unsigned)num_read)
197
+ break;
198
+ size -= (unsigned)num_read;
199
+
200
+ ++sector;
201
+ if (size >= sizeof(buffer))
202
+ num_read = rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
203
+ else
204
+ num_read = rc_cd_read_sector(iterator, track_handle, sector, buffer, size);
205
+ } while (num_read > 0);
206
+
207
+ return 1;
208
+ }
209
+
210
+ /* ===================================================== */
211
+
212
+ int rc_hash_3do(char hash[33], const rc_hash_iterator_t* iterator)
213
+ {
214
+ uint8_t buffer[2048];
215
+ const uint8_t operafs_identifier[7] = { 0x01, 0x5A, 0x5A, 0x5A, 0x5A, 0x5A, 0x01 };
216
+ void* track_handle;
217
+ md5_state_t md5;
218
+ int sector;
219
+ int block_size, block_location;
220
+ int offset, stop;
221
+ size_t size = 0;
222
+
223
+ track_handle = rc_cd_open_track(iterator, 1);
224
+ if (!track_handle)
225
+ return rc_hash_iterator_error(iterator, "Could not open track");
226
+
227
+ /* the Opera filesystem stores the volume information in the first 132 bytes of sector 0
228
+ * https://github.com/barbeque/3dodump/blob/master/OperaFS-Format.md
229
+ */
230
+ rc_cd_read_sector(iterator, track_handle, 0, buffer, 132);
231
+
232
+ if (memcmp(buffer, operafs_identifier, sizeof(operafs_identifier)) == 0) {
233
+ rc_hash_iterator_verbose_formatted(iterator, "Found 3DO CD, title=%.32s", &buffer[0x28]);
234
+
235
+ /* include the volume header in the hash */
236
+ md5_init(&md5);
237
+ md5_append(&md5, buffer, 132);
238
+
239
+ /* the block size is at offset 0x4C (assume 0x4C is always 0) */
240
+ block_size = buffer[0x4D] * 65536 + buffer[0x4E] * 256 + buffer[0x4F];
241
+
242
+ /* the root directory block location is at offset 0x64 (and duplicated several
243
+ * times, but we just look at the primary record) (assume 0x64 is always 0)*/
244
+ block_location = buffer[0x65] * 65536 + buffer[0x66] * 256 + buffer[0x67];
245
+
246
+ /* multiply the block index by the block size to get the real address */
247
+ block_location *= block_size;
248
+
249
+ /* convert that to a sector and read it */
250
+ sector = block_location / 2048;
251
+
252
+ do {
253
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
254
+
255
+ /* offset to start of entries is at offset 0x10 (assume 0x10 and 0x11 are always 0) */
256
+ offset = buffer[0x12] * 256 + buffer[0x13];
257
+
258
+ /* offset to end of entries is at offset 0x0C (assume 0x0C is always 0) */
259
+ stop = buffer[0x0D] * 65536 + buffer[0x0E] * 256 + buffer[0x0F];
260
+
261
+ while (offset < stop) {
262
+ if (buffer[offset + 0x03] == 0x02) { /* file */
263
+ if (strcasecmp((const char*)&buffer[offset + 0x20], "LaunchMe") == 0) {
264
+ /* the block size is at offset 0x0C (assume 0x0C is always 0) */
265
+ block_size = buffer[offset + 0x0D] * 65536 + buffer[offset + 0x0E] * 256 + buffer[offset + 0x0F];
266
+
267
+ /* the block location is at offset 0x44 (assume 0x44 is always 0) */
268
+ block_location = buffer[offset + 0x45] * 65536 + buffer[offset + 0x46] * 256 + buffer[offset + 0x47];
269
+ block_location *= block_size;
270
+
271
+ /* the file size is at offset 0x10 (assume 0x10 is always 0) */
272
+ size = (size_t)buffer[offset + 0x11] * 65536 + buffer[offset + 0x12] * 256 + buffer[offset + 0x13];
273
+
274
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing header (%u bytes) and %.32s (%u bytes) ", 132, &buffer[offset + 0x20], (unsigned)size);
275
+
276
+ break;
277
+ }
278
+ }
279
+
280
+ /* the number of extra copies of the file is at offset 0x40 (assume 0x40-0x42 are always 0) */
281
+ offset += 0x48 + buffer[offset + 0x43] * 4;
282
+ }
283
+
284
+ if (size != 0)
285
+ break;
286
+
287
+ /* did not find the file, see if the directory listing is continued in another sector */
288
+ offset = buffer[0x02] * 256 + buffer[0x03];
289
+
290
+ /* no more sectors to search*/
291
+ if (offset == 0xFFFF)
292
+ break;
293
+
294
+ /* get next sector */
295
+ offset *= block_size;
296
+ sector = (block_location + offset) / 2048;
297
+ } while (1);
298
+
299
+ if (size == 0) {
300
+ rc_cd_close_track(iterator, track_handle);
301
+ return rc_hash_iterator_error(iterator, "Could not find LaunchMe");
302
+ }
303
+
304
+ sector = block_location / 2048;
305
+
306
+ while (size > 2048) {
307
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
308
+ md5_append(&md5, buffer, sizeof(buffer));
309
+
310
+ ++sector;
311
+ size -= 2048;
312
+ }
313
+
314
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, size);
315
+ md5_append(&md5, buffer, (int)size);
316
+ }
317
+ else {
318
+ rc_cd_close_track(iterator, track_handle);
319
+ return rc_hash_iterator_error(iterator, "Not a 3DO CD");
320
+ }
321
+
322
+ rc_cd_close_track(iterator, track_handle);
323
+
324
+ return rc_hash_finalize(iterator, &md5, hash);
325
+ }
326
+
327
+ int rc_hash_dreamcast(char hash[33], const rc_hash_iterator_t* iterator)
328
+ {
329
+ uint8_t buffer[256] = "";
330
+ void* track_handle;
331
+ char exe_file[32] = "";
332
+ uint32_t size;
333
+ uint32_t sector;
334
+ int result = 0;
335
+ md5_state_t md5;
336
+ int i = 0;
337
+
338
+ /* track 03 is the data track that contains the TOC and IP.BIN */
339
+ track_handle = rc_cd_open_track(iterator, 3);
340
+ if (track_handle) {
341
+ /* first 256 bytes from first sector should have IP.BIN structure that stores game meta information
342
+ * https://mc.pp.se/dc/ip.bin.html */
343
+ rc_cd_read_sector(iterator, track_handle, rc_cd_first_track_sector(iterator, track_handle), buffer, sizeof(buffer));
344
+ }
345
+
346
+ if (memcmp(&buffer[0], "SEGA SEGAKATANA ", 16) != 0) {
347
+ if (track_handle)
348
+ rc_cd_close_track(iterator, track_handle);
349
+
350
+ /* not a gd-rom dreamcast file. check for mil-cd by looking for the marker in the first data track */
351
+ track_handle = rc_cd_open_track(iterator, RC_HASH_CDTRACK_FIRST_DATA);
352
+ if (!track_handle)
353
+ return rc_hash_iterator_error(iterator, "Could not open track");
354
+
355
+ rc_cd_read_sector(iterator, track_handle, rc_cd_first_track_sector(iterator, track_handle), buffer, sizeof(buffer));
356
+ if (memcmp(&buffer[0], "SEGA SEGAKATANA ", 16) != 0) {
357
+ /* did not find marker on track 3 or first data track */
358
+ rc_cd_close_track(iterator, track_handle);
359
+ return rc_hash_iterator_error(iterator, "Not a Dreamcast CD");
360
+ }
361
+ }
362
+
363
+ /* start the hash with the game meta information */
364
+ md5_init(&md5);
365
+ md5_append(&md5, (md5_byte_t*)buffer, 256);
366
+
367
+ if (iterator->callbacks.verbose_message) {
368
+ uint8_t* ptr = &buffer[0xFF];
369
+ while (ptr > &buffer[0x80] && ptr[-1] == ' ')
370
+ --ptr;
371
+ *ptr = '\0';
372
+
373
+ rc_hash_iterator_verbose_formatted(iterator, "Found Dreamcast CD: %.128s (%.16s)", (const char*)&buffer[0x80], (const char*)&buffer[0x40]);
374
+ }
375
+
376
+ /* the boot filename is 96 bytes into the meta information (https://mc.pp.se/dc/ip0000.bin.html) */
377
+ /* remove whitespace from bootfile */
378
+ i = 0;
379
+ while (!isspace((unsigned char)buffer[96 + i]) && i < 16)
380
+ ++i;
381
+
382
+ /* sometimes boot file isn't present on meta information.
383
+ * nothing can be done, as even the core doesn't run the game in this case. */
384
+ if (i == 0) {
385
+ rc_cd_close_track(iterator, track_handle);
386
+ return rc_hash_iterator_error(iterator, "Boot executable not specified on IP.BIN");
387
+ }
388
+
389
+ memcpy(exe_file, &buffer[96], i);
390
+ exe_file[i] = '\0';
391
+
392
+ sector = rc_cd_find_file_sector(iterator, track_handle, exe_file, &size);
393
+ if (sector == 0) {
394
+ rc_cd_close_track(iterator, track_handle);
395
+ return rc_hash_iterator_error(iterator, "Could not locate boot executable");
396
+ }
397
+
398
+ if (rc_cd_read_sector(iterator, track_handle, sector, buffer, 1)) {
399
+ /* the boot executable is in the primary data track */
400
+ }
401
+ else {
402
+ rc_cd_close_track(iterator, track_handle);
403
+
404
+ /* the boot executable is normally in the last track */
405
+ track_handle = rc_cd_open_track(iterator, RC_HASH_CDTRACK_LAST);
406
+ }
407
+
408
+ result = rc_hash_cd_file(&md5, iterator, track_handle, sector, NULL, size, "boot executable");
409
+ rc_cd_close_track(iterator, track_handle);
410
+
411
+ rc_hash_finalize(iterator, &md5, hash);
412
+ return result;
413
+ }
414
+
415
+ static int rc_hash_nintendo_disc_partition(md5_state_t* md5, const rc_hash_iterator_t* iterator,
416
+ void* file_handle, const uint32_t part_offset, uint8_t wii_shift)
417
+ {
418
+ const uint32_t BASE_HEADER_SIZE = 0x2440;
419
+ const uint32_t MAX_HEADER_SIZE = 1024 * 1024;
420
+
421
+ uint32_t apploader_header_size, apploader_body_size, apploader_trailer_size, header_size;
422
+
423
+ uint8_t quad_buffer[4];
424
+ uint8_t addr_buffer[0xD8];
425
+ uint8_t* buffer;
426
+
427
+ uint64_t dol_offset;
428
+ uint64_t dol_offsets[18];
429
+ uint64_t dol_sizes[18];
430
+
431
+ uint8_t ix;
432
+ uint64_t remaining_size;
433
+ const uint32_t MAX_CHUNK_SIZE = 1024 * 1024;
434
+
435
+ /* GetApploaderSize */
436
+ rc_file_seek(iterator, file_handle, part_offset + BASE_HEADER_SIZE + 0x14, SEEK_SET);
437
+ apploader_header_size = 0x20;
438
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
439
+ apploader_body_size =
440
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
441
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
442
+ apploader_trailer_size =
443
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
444
+ header_size = BASE_HEADER_SIZE + apploader_header_size + apploader_body_size + apploader_trailer_size;
445
+ if (header_size > MAX_HEADER_SIZE) header_size = MAX_HEADER_SIZE;
446
+
447
+ /* Hash headers */
448
+ buffer = (uint8_t*)malloc(header_size);
449
+ if (!buffer) {
450
+ rc_file_close(iterator, file_handle);
451
+ return rc_hash_iterator_error(iterator, "Could not allocate temporary buffer");
452
+ }
453
+ rc_file_seek(iterator, file_handle, part_offset, SEEK_SET);
454
+ rc_file_read(iterator, file_handle, buffer, header_size);
455
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte partition header", header_size);
456
+ md5_append(md5, buffer, header_size);
457
+
458
+ /* GetBootDOLOffset
459
+ * Base header size is guaranteed larger than 0x423 therefore buffer contains dol_offset right now
460
+ */
461
+ dol_offset = (((uint64_t)buffer[0x420] << 24) |
462
+ ((uint64_t)buffer[0x421] << 16) |
463
+ ((uint64_t)buffer[0x422] << 8) |
464
+ (uint64_t)buffer[0x423]) << wii_shift;
465
+ free(buffer);
466
+
467
+ /* Find offsets and sizes for the 7 main.dol code segments and 11 main.dol data segments */
468
+ rc_file_seek(iterator, file_handle, part_offset + dol_offset, SEEK_SET);
469
+ rc_file_read(iterator, file_handle, addr_buffer, 0xD8);
470
+ for (ix = 0; ix < 18; ix++) {
471
+ dol_offsets[ix] = (((uint64_t)addr_buffer[0x0 + ix * 4] << 24) |
472
+ ((uint64_t)addr_buffer[0x1 + ix * 4] << 16) |
473
+ ((uint64_t)addr_buffer[0x2 + ix * 4] << 8) |
474
+ (uint64_t)addr_buffer[0x3 + ix * 4]) << wii_shift;
475
+ dol_sizes[ix] = (((uint64_t)addr_buffer[0x90 + ix * 4] << 24) |
476
+ ((uint64_t)addr_buffer[0x91 + ix * 4] << 16) |
477
+ ((uint64_t)addr_buffer[0x92 + ix * 4] << 8) |
478
+ (uint64_t)addr_buffer[0x93 + ix * 4]) << wii_shift;
479
+ }
480
+
481
+ /* Iterate through the 18 main.dol segments and hash each */
482
+ buffer = (uint8_t*)malloc(MAX_CHUNK_SIZE);
483
+ if (!buffer) {
484
+ rc_file_close(iterator, file_handle);
485
+ return rc_hash_iterator_error(iterator, "Could not allocate temporary buffer");
486
+ }
487
+
488
+ for (ix = 0; ix < 18; ix++) {
489
+ if (dol_sizes[ix] == 0)
490
+ continue;
491
+
492
+ rc_file_seek(iterator, file_handle, part_offset + dol_offsets[ix], SEEK_SET);
493
+ if (ix < 7)
494
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte main.dol code segment %u", dol_sizes[ix], ix);
495
+ else
496
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte main.dol data segment %u", dol_sizes[ix], ix - 7);
497
+
498
+ remaining_size = dol_sizes[ix];
499
+ while (remaining_size > MAX_CHUNK_SIZE) {
500
+ rc_file_read(iterator, file_handle, buffer, MAX_CHUNK_SIZE);
501
+ md5_append(md5, buffer, MAX_CHUNK_SIZE);
502
+ remaining_size -= MAX_CHUNK_SIZE;
503
+ }
504
+ rc_file_read(iterator, file_handle, buffer, (int32_t)remaining_size);
505
+ md5_append(md5, buffer, (int32_t)remaining_size);
506
+ }
507
+
508
+ free(buffer);
509
+ return 1;
510
+ }
511
+
512
+ int rc_hash_gamecube(char hash[33], const rc_hash_iterator_t* iterator)
513
+ {
514
+ md5_state_t md5;
515
+ void* file_handle;
516
+
517
+ uint8_t quad_buffer[4];
518
+ uint8_t success;
519
+
520
+ file_handle = rc_file_open(iterator, iterator->path);
521
+ if (!file_handle)
522
+ return rc_hash_iterator_error(iterator, "Could not open file");
523
+
524
+ md5_init(&md5);
525
+ /* Check Magic Word */
526
+ rc_file_seek(iterator, file_handle, 0x1c, SEEK_SET);
527
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
528
+ if (quad_buffer[0] == 0xC2 && quad_buffer[1] == 0x33 && quad_buffer[2] == 0x9F && quad_buffer[3] == 0x3D)
529
+ success = rc_hash_nintendo_disc_partition(&md5, iterator, file_handle, 0, 0);
530
+ else
531
+ success = rc_hash_iterator_error(iterator, "Not a Gamecube disc");
532
+
533
+ /* Finalize */
534
+ rc_file_close(iterator, file_handle);
535
+
536
+ if (success)
537
+ return rc_hash_finalize(iterator, &md5, hash);
538
+
539
+ return 0;
540
+ }
541
+
542
+ /* helper variable only used for testing */
543
+ const char* _rc_hash_jaguar_cd_homebrew_hash = NULL;
544
+
545
+ int rc_hash_jaguar_cd(char hash[33], const rc_hash_iterator_t* iterator)
546
+ {
547
+ uint8_t buffer[2352];
548
+ void* track_handle;
549
+ md5_state_t md5;
550
+ int byteswapped = 0;
551
+ uint32_t size = 0;
552
+ uint32_t offset = 0;
553
+ uint32_t sector = 0;
554
+ uint32_t remaining;
555
+ uint32_t i;
556
+
557
+ /* Jaguar CD header is in the first sector of the first data track OF THE SECOND SESSION.
558
+ * The first track must be an audio track, but may be a warning message or actual game audio */
559
+ track_handle = rc_cd_open_track(iterator, RC_HASH_CDTRACK_FIRST_OF_SECOND_SESSION);
560
+ if (!track_handle)
561
+ return rc_hash_iterator_error(iterator, "Could not open track");
562
+
563
+ /* The header is an unspecified distance into the first sector, but usually two bytes in.
564
+ * It consists of 64 bytes of "TAIR" or "ATRI" repeating, depending on whether or not the data
565
+ * is byteswapped. Then another 32 byte that reads "ATARI APPROVED DATA HEADER ATRI "
566
+ * (possibly byteswapped). Then a big-endian 32-bit value for the address where the boot code
567
+ * should be loaded, and a second big-endian 32-bit value for the size of the boot code. */
568
+ sector = rc_cd_first_track_sector(iterator, track_handle);
569
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
570
+
571
+ for (i = 64; i < sizeof(buffer) - 32 - 4 * 3; i++) {
572
+ if (memcmp(&buffer[i], "TARA IPARPVODED TA AEHDAREA RT I", 32) == 0) {
573
+ byteswapped = 1;
574
+ offset = i + 32 + 4;
575
+ size = (buffer[offset] << 16) | (buffer[offset + 1] << 24) | (buffer[offset + 2]) | (buffer[offset + 3] << 8);
576
+ break;
577
+ }
578
+ else if (memcmp(&buffer[i], "ATARI APPROVED DATA HEADER ATRI ", 32) == 0) {
579
+ byteswapped = 0;
580
+ offset = i + 32 + 4;
581
+ size = (buffer[offset] << 24) | (buffer[offset + 1] << 16) | (buffer[offset + 2] << 8) | (buffer[offset + 3]);
582
+ break;
583
+ }
584
+ }
585
+
586
+ if (size == 0) { /* did not see ATARI APPROVED DATA HEADER */
587
+ rc_cd_close_track(iterator, track_handle);
588
+ return rc_hash_iterator_error(iterator, "Not a Jaguar CD");
589
+ }
590
+
591
+ i = 0; /* only loop once */
592
+ do {
593
+ md5_init(&md5);
594
+
595
+ offset += 4;
596
+
597
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing boot executable (%u bytes starting at %u bytes into sector %u)", size, offset, sector);
598
+
599
+ if (size > MAX_BUFFER_SIZE)
600
+ size = MAX_BUFFER_SIZE;
601
+
602
+ do {
603
+ if (byteswapped)
604
+ rc_hash_byteswap16(buffer, &buffer[sizeof(buffer)]);
605
+
606
+ remaining = sizeof(buffer) - offset;
607
+ if (remaining >= size) {
608
+ md5_append(&md5, &buffer[offset], size);
609
+ size = 0;
610
+ break;
611
+ }
612
+
613
+ md5_append(&md5, &buffer[offset], remaining);
614
+ size -= remaining;
615
+ offset = 0;
616
+ } while (rc_cd_read_sector(iterator, track_handle, ++sector, buffer, sizeof(buffer)) == sizeof(buffer));
617
+
618
+ rc_cd_close_track(iterator, track_handle);
619
+
620
+ if (size > 0)
621
+ return rc_hash_iterator_error(iterator, "Not enough data");
622
+
623
+ rc_hash_finalize(iterator, &md5, hash);
624
+
625
+ /* homebrew games all seem to have the same boot executable and store the actual game code in track 2.
626
+ * if we generated something other than the homebrew hash, return it. assume all homebrews are byteswapped. */
627
+ if (strcmp(hash, "254487b59ab21bc005338e85cbf9fd2f") != 0 || !byteswapped) {
628
+ if (_rc_hash_jaguar_cd_homebrew_hash == NULL || strcmp(hash, _rc_hash_jaguar_cd_homebrew_hash) != 0)
629
+ return 1;
630
+ }
631
+
632
+ /* if we've already been through the loop a second time, just return the hash */
633
+ if (i == 1)
634
+ return 1;
635
+ ++i;
636
+
637
+ rc_hash_iterator_verbose_formatted(iterator, "Potential homebrew at sector %u, checking for KART data in track 2", sector);
638
+
639
+ track_handle = rc_cd_open_track(iterator, 2);
640
+ if (!track_handle)
641
+ return rc_hash_iterator_error(iterator, "Could not open track");
642
+
643
+ /* track 2 of the homebrew code has the 64 bytes or ATRI followed by 32 bytes of "ATARI APPROVED DATA HEADER ATRI!",
644
+ * then 64 bytes of KART repeating. */
645
+ sector = rc_cd_first_track_sector(iterator, track_handle);
646
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
647
+ if (memcmp(&buffer[0x5E], "RT!IRTKA", 8) != 0)
648
+ return rc_hash_iterator_error(iterator, "Homebrew executable not found in track 2");
649
+
650
+ /* found KART data*/
651
+ rc_hash_iterator_verbose(iterator, "Found KART data in track 2");
652
+
653
+ offset = 0xA6;
654
+ size = (buffer[offset] << 16) | (buffer[offset + 1] << 24) | (buffer[offset + 2]) | (buffer[offset + 3] << 8);
655
+ } while (1);
656
+ }
657
+
658
+ int rc_hash_neogeo_cd(char hash[33], const rc_hash_iterator_t* iterator)
659
+ {
660
+ char buffer[1024], *ptr;
661
+ void* track_handle;
662
+ uint32_t sector;
663
+ uint32_t size;
664
+ md5_state_t md5;
665
+
666
+ track_handle = rc_cd_open_track(iterator, 1);
667
+ if (!track_handle)
668
+ return rc_hash_iterator_error(iterator, "Could not open track");
669
+
670
+ /* https://wiki.neogeodev.org/index.php?title=IPL_file, https://wiki.neogeodev.org/index.php?title=PRG_file
671
+ * IPL file specifies data to be loaded before the game starts. PRG files are the executable code
672
+ */
673
+ sector = rc_cd_find_file_sector(iterator, track_handle, "IPL.TXT", &size);
674
+ if (!sector) {
675
+ rc_cd_close_track(iterator, track_handle);
676
+ return rc_hash_iterator_error(iterator, "Not a NeoGeo CD game disc");
677
+ }
678
+
679
+ if (rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer)) == 0) {
680
+ rc_cd_close_track(iterator, track_handle);
681
+ return 0;
682
+ }
683
+
684
+ md5_init(&md5);
685
+
686
+ buffer[sizeof(buffer) - 1] = '\0';
687
+ ptr = &buffer[0];
688
+ do {
689
+ char* start = ptr;
690
+ while (*ptr && *ptr != '.')
691
+ ++ptr;
692
+
693
+ if (strncasecmp(ptr, ".PRG", 4) == 0) {
694
+ ptr += 4;
695
+ *ptr++ = '\0';
696
+
697
+ sector = rc_cd_find_file_sector(iterator, track_handle, start, &size);
698
+ if (!sector || !rc_hash_cd_file(&md5, iterator, track_handle, sector, NULL, size, start)) {
699
+ rc_cd_close_track(iterator, track_handle);
700
+ return rc_hash_iterator_error_formatted(iterator, "Could not read %.16s", start);
701
+ }
702
+ }
703
+
704
+ while (*ptr && *ptr != '\n')
705
+ ++ptr;
706
+ if (*ptr != '\n')
707
+ break;
708
+ ++ptr;
709
+ } while (*ptr != '\0' && *ptr != '\x1a');
710
+
711
+ rc_cd_close_track(iterator, track_handle);
712
+ return rc_hash_finalize(iterator, &md5, hash);
713
+ }
714
+
715
+ static int rc_hash_pce_track(char hash[33], void* track_handle, const rc_hash_iterator_t* iterator)
716
+ {
717
+ uint8_t buffer[2048];
718
+ md5_state_t md5;
719
+ uint32_t sector, num_sectors;
720
+ uint32_t size;
721
+
722
+ /* the PC-Engine uses the second sector to specify boot information and program name.
723
+ * the string "PC Engine CD-ROM SYSTEM" should exist at 32 bytes into the sector
724
+ * http://shu.sheldows.com/shu/download/pcedocs/pce_cdrom.html
725
+ */
726
+ if (rc_cd_read_sector(iterator, track_handle, rc_cd_first_track_sector(iterator, track_handle) + 1, buffer, 128) < 128)
727
+ return rc_hash_iterator_error(iterator, "Not a PC Engine CD");
728
+
729
+ /* normal PC Engine CD will have a header block in sector 1 */
730
+ if (memcmp("PC Engine CD-ROM SYSTEM", &buffer[32], 23) == 0) {
731
+ /* the title of the disc is the last 22 bytes of the header */
732
+ md5_init(&md5);
733
+ md5_append(&md5, &buffer[106], 22);
734
+
735
+ buffer[128] = '\0';
736
+ rc_hash_iterator_verbose_formatted(iterator, "Found PC Engine CD, title=%.22s", &buffer[106]);
737
+
738
+ /* the first three bytes specify the sector of the program data, and the fourth byte
739
+ * is the number of sectors.
740
+ */
741
+ sector = (buffer[0] << 16) + (buffer[1] << 8) + buffer[2];
742
+ num_sectors = buffer[3];
743
+
744
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %d sectors starting at sector %d", num_sectors, sector);
745
+
746
+ sector += rc_cd_first_track_sector(iterator, track_handle);
747
+ while (num_sectors > 0) {
748
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
749
+ md5_append(&md5, buffer, sizeof(buffer));
750
+
751
+ ++sector;
752
+ --num_sectors;
753
+ }
754
+ }
755
+ /* GameExpress CDs use a standard Joliet filesystem - locate and hash the BOOT.BIN */
756
+ else if ((sector = rc_cd_find_file_sector(iterator, track_handle, "BOOT.BIN", &size)) != 0 && size < MAX_BUFFER_SIZE) {
757
+ md5_init(&md5);
758
+ while (size > sizeof(buffer)) {
759
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
760
+ md5_append(&md5, buffer, sizeof(buffer));
761
+
762
+ ++sector;
763
+ size -= sizeof(buffer);
764
+ }
765
+
766
+ if (size > 0) {
767
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, size);
768
+ md5_append(&md5, buffer, size);
769
+ }
770
+ }
771
+ else {
772
+ return rc_hash_iterator_error(iterator, "Not a PC Engine CD");
773
+ }
774
+
775
+ return rc_hash_finalize(iterator, &md5, hash);
776
+ }
777
+
778
+ int rc_hash_pce_cd(char hash[33], const rc_hash_iterator_t* iterator)
779
+ {
780
+ int result;
781
+ void* track_handle = rc_cd_open_track(iterator, RC_HASH_CDTRACK_FIRST_DATA);
782
+ if (!track_handle)
783
+ return rc_hash_iterator_error(iterator, "Could not open track");
784
+
785
+ result = rc_hash_pce_track(hash, track_handle, iterator);
786
+
787
+ rc_cd_close_track(iterator, track_handle);
788
+
789
+ return result;
790
+ }
791
+
792
+ int rc_hash_pcfx_cd(char hash[33], const rc_hash_iterator_t* iterator)
793
+ {
794
+ uint8_t buffer[2048];
795
+ void* track_handle;
796
+ md5_state_t md5;
797
+ int sector, num_sectors;
798
+
799
+ /* PC-FX executable can be in any track. Assume it's in the largest data track and check there first */
800
+ track_handle = rc_cd_open_track(iterator, RC_HASH_CDTRACK_LARGEST);
801
+ if (!track_handle)
802
+ return rc_hash_iterator_error(iterator, "Could not open track");
803
+
804
+ /* PC-FX CD will have a header marker in sector 0 */
805
+ sector = rc_cd_first_track_sector(iterator, track_handle);
806
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, 32);
807
+ if (memcmp("PC-FX:Hu_CD-ROM", &buffer[0], 15) != 0) {
808
+ rc_cd_close_track(iterator, track_handle);
809
+
810
+ /* not found in the largest data track, check track 2 */
811
+ track_handle = rc_cd_open_track(iterator, 2);
812
+ if (!track_handle)
813
+ return rc_hash_iterator_error(iterator, "Could not open track");
814
+
815
+ sector = rc_cd_first_track_sector(iterator, track_handle);
816
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, 32);
817
+ }
818
+
819
+ if (memcmp("PC-FX:Hu_CD-ROM", &buffer[0], 15) == 0) {
820
+ /* PC-FX boot header fills the first two sectors of the disc
821
+ * https://bitbucket.org/trap15/pcfxtools/src/master/pcfx-cdlink.c
822
+ * the important stuff is the first 128 bytes of the second sector (title being the first 32) */
823
+ rc_cd_read_sector(iterator, track_handle, sector + 1, buffer, 128);
824
+
825
+ md5_init(&md5);
826
+ md5_append(&md5, buffer, 128);
827
+
828
+ rc_hash_iterator_verbose_formatted(iterator, "Found PC-FX CD, title=%.32s", &buffer[0]);
829
+
830
+ /* the program sector is in bytes 33-36 (assume byte 36 is 0) */
831
+ sector = (buffer[34] << 16) + (buffer[33] << 8) + buffer[32];
832
+
833
+ /* the number of sectors the program occupies is in bytes 37-40 (assume byte 40 is 0) */
834
+ num_sectors = (buffer[38] << 16) + (buffer[37] << 8) + buffer[36];
835
+
836
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %d sectors starting at sector %d", num_sectors, sector);
837
+
838
+ sector += rc_cd_first_track_sector(iterator, track_handle);
839
+ while (num_sectors > 0) {
840
+ rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer));
841
+ md5_append(&md5, buffer, sizeof(buffer));
842
+
843
+ ++sector;
844
+ --num_sectors;
845
+ }
846
+ }
847
+ else {
848
+ int result = 0;
849
+ rc_cd_read_sector(iterator, track_handle, sector + 1, buffer, 128);
850
+
851
+ /* some PC-FX CDs still identify as PCE CDs */
852
+ if (memcmp("PC Engine CD-ROM SYSTEM", &buffer[32], 23) == 0)
853
+ result = rc_hash_pce_track(hash, track_handle, iterator);
854
+
855
+ rc_cd_close_track(iterator, track_handle);
856
+ if (result)
857
+ return result;
858
+
859
+ return rc_hash_iterator_error(iterator, "Not a PC-FX CD");
860
+ }
861
+
862
+ rc_cd_close_track(iterator, track_handle);
863
+
864
+ return rc_hash_finalize(iterator, &md5, hash);
865
+ }
866
+
867
+ static int rc_hash_find_playstation_executable(const rc_hash_iterator_t* iterator, void* track_handle,
868
+ const char* boot_key, const char* cdrom_prefix,
869
+ char exe_name[], uint32_t exe_name_size, uint32_t* exe_size)
870
+ {
871
+ uint8_t buffer[2048];
872
+ uint32_t size;
873
+ char* ptr;
874
+ char* start;
875
+ const size_t boot_key_len = strlen(boot_key);
876
+ const size_t cdrom_prefix_len = strlen(cdrom_prefix);
877
+ int sector;
878
+
879
+ sector = rc_cd_find_file_sector(iterator, track_handle, "SYSTEM.CNF", NULL);
880
+ if (!sector)
881
+ return 0;
882
+
883
+ size = (uint32_t)rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer) - 1);
884
+ buffer[size] = '\0';
885
+
886
+ sector = 0;
887
+ for (ptr = (char*)buffer; *ptr; ++ptr) {
888
+ if (strncmp(ptr, boot_key, boot_key_len) == 0) {
889
+ ptr += boot_key_len;
890
+ while (isspace((unsigned char)*ptr))
891
+ ++ptr;
892
+
893
+ if (*ptr == '=') {
894
+ ++ptr;
895
+ while (isspace((unsigned char)*ptr))
896
+ ++ptr;
897
+
898
+ if (strncmp(ptr, cdrom_prefix, cdrom_prefix_len) == 0)
899
+ ptr += cdrom_prefix_len;
900
+ while (*ptr == '\\')
901
+ ++ptr;
902
+
903
+ start = ptr;
904
+ while (!isspace((unsigned char)*ptr) && *ptr != ';')
905
+ ++ptr;
906
+
907
+ size = (uint32_t)(ptr - start);
908
+ if (size >= exe_name_size)
909
+ size = exe_name_size - 1;
910
+
911
+ memcpy(exe_name, start, size);
912
+ exe_name[size] = '\0';
913
+
914
+ rc_hash_iterator_verbose_formatted(iterator, "Looking for boot executable: %s", exe_name);
915
+
916
+ sector = rc_cd_find_file_sector(iterator, track_handle, exe_name, exe_size);
917
+ break;
918
+ }
919
+ }
920
+
921
+ /* advance to end of line */
922
+ while (*ptr && *ptr != '\n')
923
+ ++ptr;
924
+ }
925
+
926
+ return sector;
927
+ }
928
+
929
+ int rc_hash_psx(char hash[33], const rc_hash_iterator_t* iterator)
930
+ {
931
+ uint8_t buffer[32];
932
+ char exe_name[64] = "";
933
+ void* track_handle;
934
+ uint32_t sector;
935
+ uint32_t size;
936
+ int result = 0;
937
+ md5_state_t md5;
938
+
939
+ track_handle = rc_cd_open_track(iterator, 1);
940
+ if (!track_handle)
941
+ return rc_hash_iterator_error(iterator, "Could not open track");
942
+
943
+ sector = rc_hash_find_playstation_executable(iterator, track_handle, "BOOT", "cdrom:", exe_name, sizeof(exe_name), &size);
944
+ if (!sector) {
945
+ sector = rc_cd_find_file_sector(iterator, track_handle, "PSX.EXE", &size);
946
+ if (sector)
947
+ memcpy(exe_name, "PSX.EXE", 8);
948
+ }
949
+
950
+ if (!sector) {
951
+ rc_hash_iterator_error(iterator, "Could not locate primary executable");
952
+ }
953
+ else if (rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer)) < sizeof(buffer)) {
954
+ rc_hash_iterator_error(iterator, "Could not read primary executable");
955
+ }
956
+ else {
957
+ if (memcmp(buffer, "PS-X EXE", 7) != 0) {
958
+ rc_hash_iterator_verbose_formatted(iterator, "%s did not contain PS-X EXE marker", exe_name);
959
+ }
960
+ else {
961
+ /* the PS-X EXE header specifies the executable size as a 4-byte value 28 bytes into the header, which doesn't
962
+ * include the header itself. We want to include the header in the hash, so append another 2048 to that value.
963
+ */
964
+ size = (((uint8_t)buffer[31] << 24) | ((uint8_t)buffer[30] << 16) | ((uint8_t)buffer[29] << 8) | (uint8_t)buffer[28]) + 2048;
965
+ }
966
+
967
+ /* there's a few games that use a singular engine and only differ via their data files. luckily, they have unique
968
+ * serial numbers, and use the serial number as the boot file in the standard way. include the boot file name in the hash.
969
+ */
970
+ md5_init(&md5);
971
+ md5_append(&md5, (md5_byte_t*)exe_name, (int)strlen(exe_name));
972
+
973
+ result = rc_hash_cd_file(&md5, iterator, track_handle, sector, exe_name, size, "primary executable");
974
+ rc_hash_finalize(iterator, &md5, hash);
975
+ }
976
+
977
+ rc_cd_close_track(iterator, track_handle);
978
+
979
+ return result;
980
+ }
981
+
982
+ int rc_hash_ps2(char hash[33], const rc_hash_iterator_t* iterator)
983
+ {
984
+ uint8_t buffer[4];
985
+ char exe_name[64] = "";
986
+ void* track_handle;
987
+ uint32_t sector;
988
+ uint32_t size;
989
+ int result = 0;
990
+ md5_state_t md5;
991
+
992
+ track_handle = rc_cd_open_track(iterator, 1);
993
+ if (!track_handle)
994
+ return rc_hash_iterator_error(iterator, "Could not open track");
995
+
996
+ sector = rc_hash_find_playstation_executable(iterator, track_handle, "BOOT2", "cdrom0:", exe_name, sizeof(exe_name), &size);
997
+ if (!sector) {
998
+ rc_hash_iterator_error(iterator, "Could not locate primary executable");
999
+ }
1000
+ else if (rc_cd_read_sector(iterator, track_handle, sector, buffer, sizeof(buffer)) < sizeof(buffer)) {
1001
+ rc_hash_iterator_error(iterator, "Could not read primary executable");
1002
+ }
1003
+ else {
1004
+ if (memcmp(buffer, "\x7f\x45\x4c\x46", 4) != 0)
1005
+ rc_hash_iterator_verbose_formatted(iterator, "%s did not contain ELF marker", exe_name);
1006
+
1007
+ /* there's a few games that use a singular engine and only differ via their data files. luckily, they have unique
1008
+ * serial numbers, and use the serial number as the boot file in the standard way. include the boot file name in the hash.
1009
+ */
1010
+ md5_init(&md5);
1011
+ md5_append(&md5, (md5_byte_t*)exe_name, (int)strlen(exe_name));
1012
+
1013
+ result = rc_hash_cd_file(&md5, iterator, track_handle, sector, exe_name, size, "primary executable");
1014
+ rc_hash_finalize(iterator, &md5, hash);
1015
+ }
1016
+
1017
+ rc_cd_close_track(iterator, track_handle);
1018
+
1019
+ return result;
1020
+ }
1021
+
1022
+ int rc_hash_psp(char hash[33], const rc_hash_iterator_t* iterator)
1023
+ {
1024
+ void* track_handle;
1025
+ uint32_t sector;
1026
+ uint32_t size;
1027
+ md5_state_t md5;
1028
+
1029
+ /* https://www.psdevwiki.com/psp/PBP
1030
+ * A PBP file is an archive containing the PARAM.SFO, primary executable, and a bunch of metadata.
1031
+ * While we could extract the PARAM.SFO and primary executable to mimic the normal PSP hashing logic,
1032
+ * it's easier to just hash the entire file. This also helps alleviate issues where the primary
1033
+ * executable is just a game engine and the only differentiating data would be the metadata. */
1034
+ if (rc_path_compare_extension(iterator->path, "pbp"))
1035
+ return rc_hash_whole_file(hash, iterator);
1036
+
1037
+ track_handle = rc_cd_open_track(iterator, 1);
1038
+ if (!track_handle)
1039
+ return rc_hash_iterator_error(iterator, "Could not open track");
1040
+
1041
+ /* http://www.romhacking.net/forum/index.php?topic=30899.0
1042
+ * PSP_GAME/PARAM.SFO contains key/value pairs identifying the game for the system (i.e. serial number,
1043
+ * name, version). PSP_GAME/SYSDIR/EBOOT.BIN is the encrypted primary executable.
1044
+ */
1045
+ sector = rc_cd_find_file_sector(iterator, track_handle, "PSP_GAME\\PARAM.SFO", &size);
1046
+ if (!sector) {
1047
+ rc_cd_close_track(iterator, track_handle);
1048
+ return rc_hash_iterator_error(iterator, "Not a PSP game disc");
1049
+ }
1050
+
1051
+ md5_init(&md5);
1052
+ if (!rc_hash_cd_file(&md5, iterator, track_handle, sector, NULL, size, "PSP_GAME\\PARAM.SFO")) {
1053
+ rc_cd_close_track(iterator, track_handle);
1054
+ return 0;
1055
+ }
1056
+
1057
+ sector = rc_cd_find_file_sector(iterator, track_handle, "PSP_GAME\\SYSDIR\\EBOOT.BIN", &size);
1058
+ if (!sector) {
1059
+ rc_cd_close_track(iterator, track_handle);
1060
+ return rc_hash_iterator_error(iterator, "Could not find primary executable");
1061
+ }
1062
+
1063
+ if (!rc_hash_cd_file(&md5, iterator, track_handle, sector, NULL, size, "PSP_GAME\\SYSDIR\\EBOOT.BIN")) {
1064
+ rc_cd_close_track(iterator, track_handle);
1065
+ return 0;
1066
+ }
1067
+
1068
+ rc_cd_close_track(iterator, track_handle);
1069
+ return rc_hash_finalize(iterator, &md5, hash);
1070
+ }
1071
+
1072
+ int rc_hash_sega_cd(char hash[33], const rc_hash_iterator_t* iterator)
1073
+ {
1074
+ uint8_t buffer[512];
1075
+ void* track_handle;
1076
+
1077
+ track_handle = rc_cd_open_track(iterator, 1);
1078
+ if (!track_handle)
1079
+ return rc_hash_iterator_error(iterator, "Could not open track");
1080
+
1081
+ /* the first 512 bytes of sector 0 are a volume header and ROM header that uniquely identify the game.
1082
+ * After that is an arbitrary amount of code that ensures the game is being run in the correct region.
1083
+ * Then more arbitrary code follows that actually starts the boot process. Somewhere in there, the
1084
+ * primary executable is loaded. In many cases, a single game will have multiple executables, so even
1085
+ * if we could determine the primary one, it's just the tip of the iceberg. As such, we've decided that
1086
+ * hashing the volume and ROM headers is sufficient for identifying the game, and we'll have to trust
1087
+ * that our players aren't modifying anything else on the disc.
1088
+ */
1089
+ rc_cd_read_sector(iterator, track_handle, 0, buffer, sizeof(buffer));
1090
+ rc_cd_close_track(iterator, track_handle);
1091
+
1092
+ if (memcmp(buffer, "SEGADISCSYSTEM ", 16) != 0 && /* Sega CD */
1093
+ memcmp(buffer, "SEGA SEGASATURN ", 16) != 0) { /* Sega Saturn */
1094
+ return rc_hash_iterator_error(iterator, "Not a Sega CD");
1095
+ }
1096
+
1097
+ return rc_hash_buffer(hash, buffer, sizeof(buffer), iterator);
1098
+ }
1099
+
1100
+ static int rc_hash_wii_disc(md5_state_t* md5, const rc_hash_iterator_t* iterator, void* file_handle)
1101
+ {
1102
+ const uint32_t MAIN_HEADER_SIZE = 0x80;
1103
+ const uint64_t REGION_CODE_ADDRESS = 0x4E000;
1104
+ const uint32_t CLUSTER_SIZE = 0x7C00;
1105
+ const uint32_t MAX_CLUSTER_COUNT = 1024;
1106
+
1107
+ uint32_t partition_info_table[8];
1108
+ uint32_t total_partition_count = 0;
1109
+ uint32_t* partition_table;
1110
+ uint64_t tmd_offset;
1111
+ uint32_t tmd_size;
1112
+ uint64_t part_offset;
1113
+ uint64_t part_size;
1114
+ uint32_t cluster_count;
1115
+
1116
+ uint8_t quad_buffer[4];
1117
+ uint8_t* buffer;
1118
+
1119
+ uint32_t ix, jx, kx;
1120
+ uint8_t encrypted;
1121
+
1122
+ /* Check encryption byte - if 0x61 is 0, disc is encrypted */
1123
+ rc_file_seek(iterator, file_handle, 0x61, SEEK_SET);
1124
+ rc_file_read(iterator, file_handle, quad_buffer, 1);
1125
+ encrypted = (quad_buffer[0] == 0);
1126
+
1127
+ /* Hash main headers */
1128
+ buffer = (uint8_t*)malloc(CLUSTER_SIZE);
1129
+ if (!buffer) {
1130
+ rc_file_close(iterator, file_handle);
1131
+ return rc_hash_iterator_error(iterator, "Could not allocate temporary buffer");
1132
+ }
1133
+
1134
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte main header for [%c%c%c%c%c%c]",
1135
+ MAIN_HEADER_SIZE, buffer[0], buffer[1], buffer[2], buffer[3], buffer[4], buffer[5]);
1136
+ rc_file_seek(iterator, file_handle, 0, SEEK_SET);
1137
+ rc_file_read(iterator, file_handle, buffer, MAIN_HEADER_SIZE);
1138
+ md5_append(md5, buffer, MAIN_HEADER_SIZE);
1139
+
1140
+ /* Hash region code */
1141
+ rc_file_seek(iterator, file_handle, REGION_CODE_ADDRESS, SEEK_SET);
1142
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1143
+ md5_append(md5, quad_buffer, 4);
1144
+
1145
+ /* Scan partition table */
1146
+ rc_file_seek(iterator, file_handle, 0x40000, SEEK_SET);
1147
+ for (ix = 0; ix < 8; ix++) {
1148
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1149
+ partition_info_table[ix] =
1150
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1151
+ if (ix % 2 == 0)
1152
+ total_partition_count += partition_info_table[ix];
1153
+ }
1154
+
1155
+ if (total_partition_count == 0) {
1156
+ free(buffer);
1157
+ rc_file_close(iterator, file_handle);
1158
+ return rc_hash_iterator_error(iterator, "No partitions found");
1159
+ }
1160
+
1161
+ partition_table = (uint32_t*)malloc(total_partition_count * 4 * 2);
1162
+ kx = 0;
1163
+ for (jx = 0; jx < 8; jx += 2) {
1164
+ rc_file_seek(iterator, file_handle, ((uint64_t)partition_info_table[jx + 1]) << 2, SEEK_SET);
1165
+ for (ix = 0; ix < partition_info_table[jx]; ix++) {
1166
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1167
+ partition_table[kx++] =
1168
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1169
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1170
+ partition_table[kx++] =
1171
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1172
+ }
1173
+ }
1174
+
1175
+ /* Read each partition */
1176
+ for (jx = 0; jx < total_partition_count * 2; jx += 2) {
1177
+ /* Don't hash Update partition*/
1178
+ if (partition_table[jx + 1] == 1)
1179
+ continue;
1180
+
1181
+ /* Hash title metadata */
1182
+ rc_file_seek(iterator, file_handle, ((uint64_t)partition_table[jx] << 2) + 0x2A4, SEEK_SET);
1183
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1184
+ tmd_size =
1185
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1186
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1187
+ tmd_offset =
1188
+ ((uint64_t)((quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3])) << 2;
1189
+
1190
+ if (tmd_size > CLUSTER_SIZE)
1191
+ tmd_size = CLUSTER_SIZE;
1192
+
1193
+ rc_file_seek(iterator, file_handle, ((uint64_t)partition_table[jx] << 2) + tmd_offset, SEEK_SET);
1194
+ rc_file_read(iterator, file_handle, buffer, tmd_size);
1195
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte title metadata (partition type %u)",
1196
+ tmd_size, partition_table[jx + 1]);
1197
+ md5_append(md5, buffer, tmd_size);
1198
+
1199
+ /* Hash partition */
1200
+ rc_file_seek(iterator, file_handle, ((uint64_t)partition_table[jx] << 2) + 0x2B8, SEEK_SET);
1201
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1202
+ part_offset =
1203
+ ((uint64_t)((quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3])) << 2;
1204
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1205
+ part_size =
1206
+ ((uint64_t)((quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3])) << 2;
1207
+
1208
+ if (encrypted) {
1209
+ cluster_count = (part_size / 0x8000 > MAX_CLUSTER_COUNT) ? MAX_CLUSTER_COUNT : (uint32_t)(part_size / 0x8000);
1210
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u encrypted clusters (%u bytes)",
1211
+ cluster_count, cluster_count * CLUSTER_SIZE);
1212
+ for (ix = 0; ix < cluster_count; ix++) {
1213
+ rc_file_seek(iterator, file_handle, part_offset + (ix * 0x8000) + 0x400, SEEK_SET);
1214
+ rc_file_read(iterator, file_handle, buffer, CLUSTER_SIZE);
1215
+ md5_append(md5, buffer, CLUSTER_SIZE);
1216
+ }
1217
+ }
1218
+ else { /* Decrypted */
1219
+ if (rc_hash_nintendo_disc_partition(md5, iterator, file_handle, (uint32_t)part_offset, 2) == 0) {
1220
+ free(partition_table);
1221
+ free(buffer);
1222
+ return rc_hash_iterator_error(iterator, "Failed to hash Wii partition");
1223
+ }
1224
+ }
1225
+ }
1226
+ free(partition_table);
1227
+ free(buffer);
1228
+ return 1;
1229
+ }
1230
+
1231
+ static int rc_hash_wiiware(md5_state_t* md5, const rc_hash_iterator_t* iterator, void* file_handle)
1232
+ {
1233
+ uint32_t cert_chain_size, ticket_size, tmd_size;
1234
+ uint32_t tmd_start_addr, content_count, content_addr, content_size, buffer_size;
1235
+ uint32_t ix;
1236
+
1237
+ uint8_t quad_buffer[4];
1238
+ uint8_t* buffer;
1239
+
1240
+ rc_file_seek(iterator, file_handle, 0x08, SEEK_SET);
1241
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1242
+ cert_chain_size =
1243
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1244
+ /* Each content is individually aligned to a 0x40-byte boundary. */
1245
+ cert_chain_size = (cert_chain_size + 0x3F) & ~0x3F;
1246
+ rc_file_seek(iterator, file_handle, 0x10, SEEK_SET);
1247
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1248
+ ticket_size =
1249
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1250
+ ticket_size = (ticket_size + 0x3F) & ~0x3F;
1251
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1252
+ tmd_size =
1253
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1254
+ tmd_size = (tmd_size + 0x3F) & ~0x3F;
1255
+ if (tmd_size > MAX_BUFFER_SIZE)
1256
+ tmd_size = MAX_BUFFER_SIZE;
1257
+
1258
+ tmd_start_addr = 0x40 + cert_chain_size + ticket_size;
1259
+
1260
+ /* Hash TMD */
1261
+ buffer = (uint8_t*)malloc(tmd_size);
1262
+ rc_file_seek(iterator, file_handle, tmd_start_addr, SEEK_SET);
1263
+ rc_file_read(iterator, file_handle, buffer, tmd_size);
1264
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u byte TMD", tmd_size);
1265
+ md5_append(md5, buffer, tmd_size);
1266
+ free(buffer);
1267
+
1268
+ /* Get count of content sections */
1269
+ rc_file_seek(iterator, file_handle, (uint64_t)tmd_start_addr + 0x1de, SEEK_SET);
1270
+ rc_file_read(iterator, file_handle, quad_buffer, 2);
1271
+ content_count = (quad_buffer[0] << 8) | quad_buffer[1];
1272
+ rc_hash_iterator_verbose_formatted(iterator, "Hashing %u content sections", content_count);
1273
+ content_addr = tmd_start_addr + tmd_size;
1274
+ for (ix = 0; ix < content_count; ix++) {
1275
+ /* Get content section size */
1276
+ rc_file_seek(iterator, file_handle, (uint64_t)tmd_start_addr + 0x1e4 + 8 + ix * 0x24, SEEK_SET);
1277
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1278
+ if (quad_buffer[0] == 0x00 && quad_buffer[1] == 0x00 && quad_buffer[2] == 0x00 && quad_buffer[3] == 0x00) {
1279
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1280
+ content_size =
1281
+ (quad_buffer[0] << 24) | (quad_buffer[1] << 16) | (quad_buffer[2] << 8) | quad_buffer[3];
1282
+ /* Padding between content should be ignored. But because the content data is encrypted,
1283
+ the size to hash for each content should be rounded up to the size of an AES block (16 bytes). */
1284
+ content_size = (content_size + 0x0F) & ~0x0F;
1285
+ }
1286
+ else {
1287
+ /* size > 4GB, just assume MAX_BUFFER_SIZE */
1288
+ content_size = MAX_BUFFER_SIZE;
1289
+ }
1290
+ buffer_size = (content_size > MAX_BUFFER_SIZE) ? MAX_BUFFER_SIZE : content_size;
1291
+
1292
+ /* Hash content */
1293
+ buffer = (uint8_t*)malloc(buffer_size);
1294
+ rc_file_seek(iterator, file_handle, content_addr, SEEK_SET);
1295
+ rc_file_read(iterator, file_handle, buffer, buffer_size);
1296
+ md5_append(md5, buffer, buffer_size);
1297
+ content_addr += content_size;
1298
+ content_addr = (content_addr + 0x3F) & ~0x3F;
1299
+ free(buffer);
1300
+ }
1301
+
1302
+ return 1;
1303
+ }
1304
+
1305
+ int rc_hash_wii(char hash[33], const rc_hash_iterator_t* iterator)
1306
+ {
1307
+ md5_state_t md5;
1308
+ void* file_handle;
1309
+
1310
+ uint8_t quad_buffer[4];
1311
+ uint8_t success;
1312
+
1313
+ file_handle = rc_file_open(iterator, iterator->path);
1314
+ if (!file_handle)
1315
+ return rc_hash_iterator_error(iterator, "Could not open file");
1316
+
1317
+ md5_init(&md5);
1318
+ /* Check Magic Words */
1319
+ rc_file_seek(iterator, file_handle, 0x18, SEEK_SET);
1320
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1321
+ if (quad_buffer[0] == 0x5D && quad_buffer[1] == 0x1C && quad_buffer[2] == 0x9E && quad_buffer[3] == 0xA3) {
1322
+ success = rc_hash_wii_disc(&md5, iterator, file_handle);
1323
+ }
1324
+ else {
1325
+ rc_file_seek(iterator, file_handle, 0x04, SEEK_SET);
1326
+ rc_file_read(iterator, file_handle, quad_buffer, 4);
1327
+ if (quad_buffer[0] == 'I' && quad_buffer[1] == 's' && quad_buffer[2] == 0x00 && quad_buffer[3] == 0x00)
1328
+ success = rc_hash_wiiware(&md5, iterator, file_handle);
1329
+ else
1330
+ success = rc_hash_iterator_error(iterator, "Not a supported Wii file");
1331
+ }
1332
+
1333
+ /* Finalize */
1334
+ rc_file_close(iterator, file_handle);
1335
+
1336
+ if (success)
1337
+ return rc_hash_finalize(iterator, &md5, hash);
1338
+
1339
+ return 0;
1340
+ }