jsbeeb 1.1.1

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 (498) hide show
  1. package/.editorconfig +15 -0
  2. package/.git-blame-ignore-revs +3 -0
  3. package/.github/copilot-instructions.md +94 -0
  4. package/.github/workflows/claude-issue-triage.yml +105 -0
  5. package/.github/workflows/claude.yml +63 -0
  6. package/.github/workflows/release-please.yml +75 -0
  7. package/.github/workflows/test-and-deploy.yml +86 -0
  8. package/.gitmodules +6 -0
  9. package/.husky/pre-commit +1 -0
  10. package/.idea/codeStyleSettings.xml +9 -0
  11. package/.idea/codeStyles/Project.xml +62 -0
  12. package/.idea/codeStyles/codeStyleConfig.xml +5 -0
  13. package/.idea/compiler.xml +22 -0
  14. package/.idea/copyright/profiles_settings.xml +3 -0
  15. package/.idea/encodings.xml +6 -0
  16. package/.idea/inspectionProfiles/Project_Default.xml +7 -0
  17. package/.idea/jsLibraryMappings.xml +6 -0
  18. package/.idea/jsLinters/jshint.xml +85 -0
  19. package/.idea/jsLinters/jslint.xml +15 -0
  20. package/.idea/jsbeeb.iml +11 -0
  21. package/.idea/misc.xml +6 -0
  22. package/.idea/modules.xml +8 -0
  23. package/.idea/prettier.xml +7 -0
  24. package/.idea/runConfigurations/Debug.xml +5 -0
  25. package/.idea/scopes/scope_settings.xml +5 -0
  26. package/.idea/vcs.xml +8 -0
  27. package/.prettierignore +4 -0
  28. package/.prettierrc.json +1 -0
  29. package/.release-please-manifest.json +3 -0
  30. package/.vscode/launch.json +14 -0
  31. package/.vscode/settings.json +6 -0
  32. package/CHANGELOG.md +32 -0
  33. package/CLAUDE.md +136 -0
  34. package/COPYING +674 -0
  35. package/Dockerfile +22 -0
  36. package/Makefile +30 -0
  37. package/README.md +259 -0
  38. package/docker/nginx-default.conf +10 -0
  39. package/docs/pal-comb-filter-research.md +129 -0
  40. package/docs/pal-simulation-design.md +368 -0
  41. package/eslint.config.js +35 -0
  42. package/index.html +954 -0
  43. package/jsconfig.json +10 -0
  44. package/package.json +102 -0
  45. package/public/discs/README.Irq-Timing +3 -0
  46. package/public/discs/README.bcdtest +5 -0
  47. package/public/discs/README.elite +6 -0
  48. package/public/discs/README.eng_test +3 -0
  49. package/public/discs/README.protection +7 -0
  50. package/public/favicon.ico +0 -0
  51. package/public/images/botbar.png +0 -0
  52. package/public/images/cub-monitor.png +0 -0
  53. package/public/images/jsbeeb-example.png +0 -0
  54. package/public/images/placeholder.png +0 -0
  55. package/public/images/red-off-16.png +0 -0
  56. package/public/images/red-on-16.png +0 -0
  57. package/public/images/sb/CD-left.jpg +0 -0
  58. package/public/images/sb/CD-right.jpg +0 -0
  59. package/public/images/sidebar.png +0 -0
  60. package/public/images/tv.png +0 -0
  61. package/public/images/yellow-off-16.png +0 -0
  62. package/public/images/yellow-on-16.png +0 -0
  63. package/public/jsbeeb-icon.png +0 -0
  64. package/public/robots.txt +3 -0
  65. package/public/roms/ADFS1-53.rom +0 -0
  66. package/public/roms/BASIC.ROM +0 -0
  67. package/public/roms/README +4 -0
  68. package/public/roms/a01/BASIC1.rom +0 -0
  69. package/public/roms/ample.rom +0 -0
  70. package/public/roms/ats-3.0.rom +0 -0
  71. package/public/roms/b/DFS-0.9.rom +0 -0
  72. package/public/roms/b/DFS-1.2.rom +0 -0
  73. package/public/roms/b1770/dfs1770.rom +0 -0
  74. package/public/roms/b1770/zADFS.ROM +0 -0
  75. package/public/roms/bp/dfs.rom +0 -0
  76. package/public/roms/bp/zADFS.ROM +0 -0
  77. package/public/roms/bpos.rom +0 -0
  78. package/public/roms/compact/adfs210.rom +0 -0
  79. package/public/roms/compact/basic48.rom +0 -0
  80. package/public/roms/compact/basic486.rom +0 -0
  81. package/public/roms/compact/os51.rom +0 -0
  82. package/public/roms/compact/utils.rom +0 -0
  83. package/public/roms/deos.rom +0 -0
  84. package/public/roms/master/anfs-4.25.rom +0 -0
  85. package/public/roms/master/mos.txt +68819 -0
  86. package/public/roms/master/mos3.20 +0 -0
  87. package/public/roms/os.rom +0 -0
  88. package/public/roms/os01.rom +0 -0
  89. package/public/roms/tube/6502Tube.rom +0 -0
  90. package/public/roms/tube/ARMeval_100.rom +0 -0
  91. package/public/roms/tube/BIOS.ROM +0 -0
  92. package/public/roms/tube/ReCo6502ROM_816 +0 -0
  93. package/public/roms/tube/Z80_120.rom +0 -0
  94. package/public/roms/us/USBASIC.rom +0 -0
  95. package/public/roms/us/USDNFS.rom +0 -0
  96. package/public/roms/usmos.rom +0 -0
  97. package/public/sounds/disc525/motor.wav +0 -0
  98. package/public/sounds/disc525/motoroff.wav +0 -0
  99. package/public/sounds/disc525/motoron.wav +0 -0
  100. package/public/sounds/disc525/seek.wav +0 -0
  101. package/public/sounds/disc525/seek2.wav +0 -0
  102. package/public/sounds/disc525/seek3.wav +0 -0
  103. package/public/sounds/disc525/step.wav +0 -0
  104. package/public/teletext/txt0.dat +0 -0
  105. package/public/teletext/txt1.dat +0 -0
  106. package/public/teletext/txt2.dat +0 -0
  107. package/public/teletext/txt3.dat +0 -0
  108. package/release-please-config.json +13 -0
  109. package/run-container.sh +92 -0
  110. package/src/6502.js +1347 -0
  111. package/src/6502.opcodes.js +1411 -0
  112. package/src/acia.js +261 -0
  113. package/src/adc.js +149 -0
  114. package/src/analogue-source.js +21 -0
  115. package/src/app/app.js +175 -0
  116. package/src/app/electron.js +20 -0
  117. package/src/app/preload.js +8 -0
  118. package/src/app-bench.js +33 -0
  119. package/src/basic/multiline-tetris +9 -0
  120. package/src/basic-tokenise.js +104 -0
  121. package/src/canvas.js +177 -0
  122. package/src/cmos.js +141 -0
  123. package/src/config.js +165 -0
  124. package/src/ddnoise.js +138 -0
  125. package/src/disc-drive.js +371 -0
  126. package/src/disc-hfe.js +396 -0
  127. package/src/disc.js +997 -0
  128. package/src/econet/L3FS.dat +0 -0
  129. package/src/econet/scsi.dat +0 -0
  130. package/src/econet.js +714 -0
  131. package/src/fake6502.js +32 -0
  132. package/src/fdc.js +248 -0
  133. package/src/filestore.js +666 -0
  134. package/src/gamepad-source.js +59 -0
  135. package/src/gamepads.js +268 -0
  136. package/src/google-drive.js +160 -0
  137. package/src/intel-fdc.js +1717 -0
  138. package/src/jsbeeb.css +363 -0
  139. package/src/keyboard.js +411 -0
  140. package/src/lib/README +1 -0
  141. package/src/lib/webgl-debug.js +911 -0
  142. package/src/main.js +1759 -0
  143. package/src/microphone-input.js +149 -0
  144. package/src/models.js +200 -0
  145. package/src/mouse-joystick-source.js +107 -0
  146. package/src/music5000-worklet.js +43 -0
  147. package/src/music5000.js +207 -0
  148. package/src/scheduler.js +148 -0
  149. package/src/serial.js +31 -0
  150. package/src/soundchip.js +314 -0
  151. package/src/sth.js +59 -0
  152. package/src/tapes.js +265 -0
  153. package/src/teletext.js +348 -0
  154. package/src/teletext_adaptor.js +172 -0
  155. package/src/teletext_data.js +1064 -0
  156. package/src/touchscreen.js +86 -0
  157. package/src/tube.js +349 -0
  158. package/src/url-params.js +256 -0
  159. package/src/utils.js +1090 -0
  160. package/src/via.js +702 -0
  161. package/src/video-filters/pal-composite.js +94 -0
  162. package/src/video-filters/passthrough-filter.js +70 -0
  163. package/src/video-filters/shaders/pal-composite.frag.glsl +147 -0
  164. package/src/video-filters/shaders/pal-composite.vert.glsl +8 -0
  165. package/src/video-filters/shaders/passthrough.frag.glsl +6 -0
  166. package/src/video-filters/shaders/passthrough.vert.glsl +7 -0
  167. package/src/video.js +794 -0
  168. package/src/wd-fdc.js +1344 -0
  169. package/src/web/audio-handler.js +146 -0
  170. package/src/web/audio-renderer.js +115 -0
  171. package/src/web/debug.js +529 -0
  172. package/tests/integration/RmwX.asm +47 -0
  173. package/tests/integration/TestInstructionsSource +27 -0
  174. package/tests/integration/TestTimingsResults +27 -0
  175. package/tests/integration/TestTimingsSource +61 -0
  176. package/tests/integration/bcd.js +23 -0
  177. package/tests/integration/dormann.js +101 -0
  178. package/tests/integration/dp111timing.js +42 -0
  179. package/tests/integration/ensure-submodules.js +25 -0
  180. package/tests/integration/nops.bas +119 -0
  181. package/tests/integration/nops.js +24 -0
  182. package/tests/integration/protection.js +26 -0
  183. package/tests/integration/rmw.js +69 -0
  184. package/tests/integration/teletext/expected_bug_469.png +0 -0
  185. package/tests/integration/teletext/expected_flash_0.png +0 -0
  186. package/tests/integration/teletext/expected_flash_1.png +0 -0
  187. package/tests/integration/teletext/expected_hoglet_held_char.png +0 -0
  188. package/tests/integration/teletext/expected_reveal_flash_0.png +0 -0
  189. package/tests/integration/teletext/expected_reveal_flash_1.png +0 -0
  190. package/tests/integration/teletext.js +126 -0
  191. package/tests/integration/timings.js +56 -0
  192. package/tests/integration/via.js +1125 -0
  193. package/tests/suite/README.md +7 -0
  194. package/tests/suite/Test Suite 2.15.txt +373 -0
  195. package/tests/suite/bin/ start +0 -0
  196. package/tests/suite/bin/adca +0 -0
  197. package/tests/suite/bin/adcax +0 -0
  198. package/tests/suite/bin/adcay +0 -0
  199. package/tests/suite/bin/adcb +0 -0
  200. package/tests/suite/bin/adcix +0 -0
  201. package/tests/suite/bin/adciy +0 -0
  202. package/tests/suite/bin/adcz +0 -0
  203. package/tests/suite/bin/adczx +0 -0
  204. package/tests/suite/bin/alrb +0 -0
  205. package/tests/suite/bin/ancb +0 -0
  206. package/tests/suite/bin/anda +0 -0
  207. package/tests/suite/bin/andax +0 -0
  208. package/tests/suite/bin/anday +0 -0
  209. package/tests/suite/bin/andb +0 -0
  210. package/tests/suite/bin/andix +0 -0
  211. package/tests/suite/bin/andiy +0 -0
  212. package/tests/suite/bin/andz +0 -0
  213. package/tests/suite/bin/andzx +0 -0
  214. package/tests/suite/bin/aneb +0 -0
  215. package/tests/suite/bin/arrb +0 -0
  216. package/tests/suite/bin/asla +0 -0
  217. package/tests/suite/bin/aslax +0 -0
  218. package/tests/suite/bin/asln +0 -0
  219. package/tests/suite/bin/aslz +0 -0
  220. package/tests/suite/bin/aslzx +0 -0
  221. package/tests/suite/bin/asoa +0 -0
  222. package/tests/suite/bin/asoax +0 -0
  223. package/tests/suite/bin/asoay +0 -0
  224. package/tests/suite/bin/asoix +0 -0
  225. package/tests/suite/bin/asoiy +0 -0
  226. package/tests/suite/bin/asoz +0 -0
  227. package/tests/suite/bin/asozx +0 -0
  228. package/tests/suite/bin/axsa +0 -0
  229. package/tests/suite/bin/axsix +0 -0
  230. package/tests/suite/bin/axsz +0 -0
  231. package/tests/suite/bin/axszy +0 -0
  232. package/tests/suite/bin/bccr +0 -0
  233. package/tests/suite/bin/bcsr +0 -0
  234. package/tests/suite/bin/beqr +0 -0
  235. package/tests/suite/bin/bita +0 -0
  236. package/tests/suite/bin/bitz +0 -0
  237. package/tests/suite/bin/bmir +0 -0
  238. package/tests/suite/bin/bner +0 -0
  239. package/tests/suite/bin/bplr +0 -0
  240. package/tests/suite/bin/branchwrap +0 -0
  241. package/tests/suite/bin/brkn +0 -0
  242. package/tests/suite/bin/bvcr +0 -0
  243. package/tests/suite/bin/bvsr +0 -0
  244. package/tests/suite/bin/cia1pb6 +0 -0
  245. package/tests/suite/bin/cia1pb7 +0 -0
  246. package/tests/suite/bin/cia1ta +0 -0
  247. package/tests/suite/bin/cia1tab +0 -0
  248. package/tests/suite/bin/cia1tb +0 -0
  249. package/tests/suite/bin/cia1tb123 +0 -0
  250. package/tests/suite/bin/cia2pb6 +0 -0
  251. package/tests/suite/bin/cia2pb7 +0 -0
  252. package/tests/suite/bin/cia2ta +0 -0
  253. package/tests/suite/bin/cia2tb +0 -0
  254. package/tests/suite/bin/cia2tb123 +0 -0
  255. package/tests/suite/bin/clcn +0 -0
  256. package/tests/suite/bin/cldn +0 -0
  257. package/tests/suite/bin/clin +0 -0
  258. package/tests/suite/bin/clvn +0 -0
  259. package/tests/suite/bin/cmpa +0 -0
  260. package/tests/suite/bin/cmpax +0 -0
  261. package/tests/suite/bin/cmpay +0 -0
  262. package/tests/suite/bin/cmpb +0 -0
  263. package/tests/suite/bin/cmpix +0 -0
  264. package/tests/suite/bin/cmpiy +0 -0
  265. package/tests/suite/bin/cmpz +0 -0
  266. package/tests/suite/bin/cmpzx +0 -0
  267. package/tests/suite/bin/cntdef +0 -0
  268. package/tests/suite/bin/cnto2 +0 -0
  269. package/tests/suite/bin/cpuport +0 -0
  270. package/tests/suite/bin/cputiming +0 -0
  271. package/tests/suite/bin/cpxa +0 -0
  272. package/tests/suite/bin/cpxb +0 -0
  273. package/tests/suite/bin/cpxz +0 -0
  274. package/tests/suite/bin/cpya +0 -0
  275. package/tests/suite/bin/cpyb +0 -0
  276. package/tests/suite/bin/cpyz +0 -0
  277. package/tests/suite/bin/dcma +0 -0
  278. package/tests/suite/bin/dcmax +0 -0
  279. package/tests/suite/bin/dcmay +0 -0
  280. package/tests/suite/bin/dcmix +0 -0
  281. package/tests/suite/bin/dcmiy +0 -0
  282. package/tests/suite/bin/dcmz +0 -0
  283. package/tests/suite/bin/dcmzx +0 -0
  284. package/tests/suite/bin/deca +0 -0
  285. package/tests/suite/bin/decax +0 -0
  286. package/tests/suite/bin/decz +0 -0
  287. package/tests/suite/bin/deczx +0 -0
  288. package/tests/suite/bin/dexn +0 -0
  289. package/tests/suite/bin/deyn +0 -0
  290. package/tests/suite/bin/eora +0 -0
  291. package/tests/suite/bin/eorax +0 -0
  292. package/tests/suite/bin/eoray +0 -0
  293. package/tests/suite/bin/eorb +0 -0
  294. package/tests/suite/bin/eorix +0 -0
  295. package/tests/suite/bin/eoriy +0 -0
  296. package/tests/suite/bin/eorz +0 -0
  297. package/tests/suite/bin/eorzx +0 -0
  298. package/tests/suite/bin/finish +0 -0
  299. package/tests/suite/bin/flipos +0 -0
  300. package/tests/suite/bin/icr01 +0 -0
  301. package/tests/suite/bin/imr +0 -0
  302. package/tests/suite/bin/inca +0 -0
  303. package/tests/suite/bin/incax +0 -0
  304. package/tests/suite/bin/incz +0 -0
  305. package/tests/suite/bin/inczx +0 -0
  306. package/tests/suite/bin/insa +0 -0
  307. package/tests/suite/bin/insax +0 -0
  308. package/tests/suite/bin/insay +0 -0
  309. package/tests/suite/bin/insix +0 -0
  310. package/tests/suite/bin/insiy +0 -0
  311. package/tests/suite/bin/insz +0 -0
  312. package/tests/suite/bin/inszx +0 -0
  313. package/tests/suite/bin/inxn +0 -0
  314. package/tests/suite/bin/inyn +0 -0
  315. package/tests/suite/bin/irq +0 -0
  316. package/tests/suite/bin/jmpi +0 -0
  317. package/tests/suite/bin/jmpw +0 -0
  318. package/tests/suite/bin/jsrw +0 -0
  319. package/tests/suite/bin/lasay +0 -0
  320. package/tests/suite/bin/laxa +0 -0
  321. package/tests/suite/bin/laxay +0 -0
  322. package/tests/suite/bin/laxix +0 -0
  323. package/tests/suite/bin/laxiy +0 -0
  324. package/tests/suite/bin/laxz +0 -0
  325. package/tests/suite/bin/laxzy +0 -0
  326. package/tests/suite/bin/ldaa +0 -0
  327. package/tests/suite/bin/ldaax +0 -0
  328. package/tests/suite/bin/ldaay +0 -0
  329. package/tests/suite/bin/ldab +0 -0
  330. package/tests/suite/bin/ldaix +0 -0
  331. package/tests/suite/bin/ldaiy +0 -0
  332. package/tests/suite/bin/ldaz +0 -0
  333. package/tests/suite/bin/ldazx +0 -0
  334. package/tests/suite/bin/ldxa +0 -0
  335. package/tests/suite/bin/ldxay +0 -0
  336. package/tests/suite/bin/ldxb +0 -0
  337. package/tests/suite/bin/ldxz +0 -0
  338. package/tests/suite/bin/ldxzy +0 -0
  339. package/tests/suite/bin/ldya +0 -0
  340. package/tests/suite/bin/ldyax +0 -0
  341. package/tests/suite/bin/ldyb +0 -0
  342. package/tests/suite/bin/ldyz +0 -0
  343. package/tests/suite/bin/ldyzx +0 -0
  344. package/tests/suite/bin/loadth +0 -0
  345. package/tests/suite/bin/lsea +0 -0
  346. package/tests/suite/bin/lseax +0 -0
  347. package/tests/suite/bin/lseay +0 -0
  348. package/tests/suite/bin/lseix +0 -0
  349. package/tests/suite/bin/lseiy +0 -0
  350. package/tests/suite/bin/lsez +0 -0
  351. package/tests/suite/bin/lsezx +0 -0
  352. package/tests/suite/bin/lsra +0 -0
  353. package/tests/suite/bin/lsrax +0 -0
  354. package/tests/suite/bin/lsrn +0 -0
  355. package/tests/suite/bin/lsrz +0 -0
  356. package/tests/suite/bin/lsrzx +0 -0
  357. package/tests/suite/bin/lxab +0 -0
  358. package/tests/suite/bin/mmu +0 -0
  359. package/tests/suite/bin/mmufetch +0 -0
  360. package/tests/suite/bin/nmi +0 -0
  361. package/tests/suite/bin/nopa +0 -0
  362. package/tests/suite/bin/nopax +0 -0
  363. package/tests/suite/bin/nopb +0 -0
  364. package/tests/suite/bin/nopn +0 -0
  365. package/tests/suite/bin/nopz +0 -0
  366. package/tests/suite/bin/nopzx +0 -0
  367. package/tests/suite/bin/oneshot +0 -0
  368. package/tests/suite/bin/oraa +0 -0
  369. package/tests/suite/bin/oraax +0 -0
  370. package/tests/suite/bin/oraay +0 -0
  371. package/tests/suite/bin/orab +0 -0
  372. package/tests/suite/bin/oraix +0 -0
  373. package/tests/suite/bin/oraiy +0 -0
  374. package/tests/suite/bin/oraz +0 -0
  375. package/tests/suite/bin/orazx +0 -0
  376. package/tests/suite/bin/phan +0 -0
  377. package/tests/suite/bin/phpn +0 -0
  378. package/tests/suite/bin/plan +0 -0
  379. package/tests/suite/bin/plpn +0 -0
  380. package/tests/suite/bin/rlaa +0 -0
  381. package/tests/suite/bin/rlaax +0 -0
  382. package/tests/suite/bin/rlaay +0 -0
  383. package/tests/suite/bin/rlaix +0 -0
  384. package/tests/suite/bin/rlaiy +0 -0
  385. package/tests/suite/bin/rlaz +0 -0
  386. package/tests/suite/bin/rlazx +0 -0
  387. package/tests/suite/bin/rola +0 -0
  388. package/tests/suite/bin/rolax +0 -0
  389. package/tests/suite/bin/roln +0 -0
  390. package/tests/suite/bin/rolz +0 -0
  391. package/tests/suite/bin/rolzx +0 -0
  392. package/tests/suite/bin/rora +0 -0
  393. package/tests/suite/bin/rorax +0 -0
  394. package/tests/suite/bin/rorn +0 -0
  395. package/tests/suite/bin/rorz +0 -0
  396. package/tests/suite/bin/rorzx +0 -0
  397. package/tests/suite/bin/rraa +0 -0
  398. package/tests/suite/bin/rraax +0 -0
  399. package/tests/suite/bin/rraay +0 -0
  400. package/tests/suite/bin/rraix +0 -0
  401. package/tests/suite/bin/rraiy +0 -0
  402. package/tests/suite/bin/rraz +0 -0
  403. package/tests/suite/bin/rrazx +0 -0
  404. package/tests/suite/bin/rtin +0 -0
  405. package/tests/suite/bin/rtsn +0 -0
  406. package/tests/suite/bin/sbca +0 -0
  407. package/tests/suite/bin/sbcax +0 -0
  408. package/tests/suite/bin/sbcay +0 -0
  409. package/tests/suite/bin/sbcb +0 -0
  410. package/tests/suite/bin/sbcb(eb) +0 -0
  411. package/tests/suite/bin/sbcix +0 -0
  412. package/tests/suite/bin/sbciy +0 -0
  413. package/tests/suite/bin/sbcz +0 -0
  414. package/tests/suite/bin/sbczx +0 -0
  415. package/tests/suite/bin/sbxb +0 -0
  416. package/tests/suite/bin/secn +0 -0
  417. package/tests/suite/bin/sedn +0 -0
  418. package/tests/suite/bin/sein +0 -0
  419. package/tests/suite/bin/shaay +0 -0
  420. package/tests/suite/bin/shaiy +0 -0
  421. package/tests/suite/bin/shsay +0 -0
  422. package/tests/suite/bin/shxay +0 -0
  423. package/tests/suite/bin/shyax +0 -0
  424. package/tests/suite/bin/staa +0 -0
  425. package/tests/suite/bin/staax +0 -0
  426. package/tests/suite/bin/staay +0 -0
  427. package/tests/suite/bin/staix +0 -0
  428. package/tests/suite/bin/staiy +0 -0
  429. package/tests/suite/bin/staz +0 -0
  430. package/tests/suite/bin/stazx +0 -0
  431. package/tests/suite/bin/stxa +0 -0
  432. package/tests/suite/bin/stxz +0 -0
  433. package/tests/suite/bin/stxzy +0 -0
  434. package/tests/suite/bin/stya +0 -0
  435. package/tests/suite/bin/styz +0 -0
  436. package/tests/suite/bin/styzx +0 -0
  437. package/tests/suite/bin/taxn +0 -0
  438. package/tests/suite/bin/tayn +0 -0
  439. package/tests/suite/bin/trap1 +0 -0
  440. package/tests/suite/bin/trap10 +0 -0
  441. package/tests/suite/bin/trap11 +0 -0
  442. package/tests/suite/bin/trap12 +0 -0
  443. package/tests/suite/bin/trap13 +0 -0
  444. package/tests/suite/bin/trap14 +0 -0
  445. package/tests/suite/bin/trap15 +0 -0
  446. package/tests/suite/bin/trap16 +0 -0
  447. package/tests/suite/bin/trap17 +0 -0
  448. package/tests/suite/bin/trap2 +0 -0
  449. package/tests/suite/bin/trap3 +0 -0
  450. package/tests/suite/bin/trap4 +0 -0
  451. package/tests/suite/bin/trap5 +0 -0
  452. package/tests/suite/bin/trap6 +0 -0
  453. package/tests/suite/bin/trap7 +0 -0
  454. package/tests/suite/bin/trap8 +0 -0
  455. package/tests/suite/bin/trap9 +0 -0
  456. package/tests/suite/bin/tsxn +0 -0
  457. package/tests/suite/bin/txan +0 -0
  458. package/tests/suite/bin/txsn +0 -0
  459. package/tests/suite/bin/tyan +0 -0
  460. package/tests/suite/cbm-hackers-post.html +178 -0
  461. package/tests/suite/cbm-hackers-post.md +78 -0
  462. package/tests/test-machine.js +288 -0
  463. package/tests/test-suite.js +147 -0
  464. package/tests/test.css +7 -0
  465. package/tests/unit/gzip/test-1 +0 -0
  466. package/tests/unit/gzip/test-1.gz +0 -0
  467. package/tests/unit/gzip/test-2 +0 -0
  468. package/tests/unit/gzip/test-2.gz +0 -0
  469. package/tests/unit/gzip/test-3 +0 -0
  470. package/tests/unit/gzip/test-3.gz +0 -0
  471. package/tests/unit/gzip/test-4 +0 -0
  472. package/tests/unit/gzip/test-4.gz +0 -0
  473. package/tests/unit/test-adc.js +307 -0
  474. package/tests/unit/test-bcd.js +30 -0
  475. package/tests/unit/test-cmos.js +266 -0
  476. package/tests/unit/test-disc-drive.js +85 -0
  477. package/tests/unit/test-disc-hfe.js +347 -0
  478. package/tests/unit/test-disc.js +232 -0
  479. package/tests/unit/test-fifo.js +35 -0
  480. package/tests/unit/test-gamepad-source.js +67 -0
  481. package/tests/unit/test-gzip.js +22 -0
  482. package/tests/unit/test-intel-fdc.js +93 -0
  483. package/tests/unit/test-keyboard.js +410 -0
  484. package/tests/unit/test-mouse-joystick-source.js +128 -0
  485. package/tests/unit/test-scheduler.js +190 -0
  486. package/tests/unit/test-serial.js +154 -0
  487. package/tests/unit/test-teletext-adaptor.js +359 -0
  488. package/tests/unit/test-tokenise.js +65 -0
  489. package/tests/unit/test-url-params.js +398 -0
  490. package/tests/unit/test-utils.js +276 -0
  491. package/tests/unit/test-video.js +498 -0
  492. package/tests/unit/test-zip.js +56 -0
  493. package/tests/unit/zip/test-mixed.zip +0 -0
  494. package/tests/unit/zip/test-rom.zip +0 -0
  495. package/tests/unit/zip/test-ssd.zip +0 -0
  496. package/tools/fir-generator.js +80 -0
  497. package/tools/vite-plugin-fir-shader.js +131 -0
  498. package/vite.config.js +34 -0
package/src/via.js ADDED
@@ -0,0 +1,702 @@
1
+ "use strict";
2
+ import * as utils from "./utils.js";
3
+
4
+ const ORB = 0x0,
5
+ ORA = 0x1,
6
+ DDRB = 0x2,
7
+ DDRA = 0x3,
8
+ T1CL = 0x4,
9
+ T1CH = 0x5,
10
+ T1LL = 0x6,
11
+ T1LH = 0x7,
12
+ T2CL = 0x8,
13
+ T2CH = 0x9,
14
+ SR = 0xa,
15
+ ACR = 0xb,
16
+ PCR = 0xc,
17
+ IFR = 0xd,
18
+ IER = 0xe,
19
+ ORAnh = 0xf,
20
+ TIMER1INT = 0x40,
21
+ TIMER2INT = 0x20,
22
+ INT_CA1 = 0x02,
23
+ INT_CA2 = 0x01,
24
+ INT_CB1 = 0x10,
25
+ INT_CB2 = 0x08;
26
+
27
+ class Via {
28
+ constructor(cpu, scheduler, irq) {
29
+ this.cpu = cpu;
30
+ this.irq = irq;
31
+ this.scheduler = scheduler;
32
+
33
+ this.ora = 0;
34
+ this.orb = 0;
35
+ this.ira = 0;
36
+ this.irb = 0;
37
+ this.ddra = 0;
38
+ this.ddrb = 0;
39
+ this.sr = 0;
40
+ this.t1l = 0;
41
+ this.t2l = 0;
42
+ this.t1c = 0;
43
+ this.t2c = 0;
44
+ this.acr = 0;
45
+ this.pcr = 0;
46
+ this.ifr = 0;
47
+ this.ier = 0;
48
+ this.t1hit = false;
49
+ this.t2hit = false;
50
+ this.portapins = 0;
51
+ this.portbpins = 0;
52
+ this.ca1 = false;
53
+ this.ca2 = false;
54
+ this.cb1 = false;
55
+ this.cb2 = false;
56
+ this.ca2changecallback = null;
57
+ this.cb2changecallback = null;
58
+ this.justhit = 0;
59
+ this.t1_pb7 = 0;
60
+
61
+ this.task = this.scheduler.newTask(() => this._onTimeout());
62
+ this.lastPolltime = 0;
63
+ }
64
+
65
+ reset() {
66
+ // http://archive.6502.org/datasheets/mos_6522_preliminary_nov_1977.pdf
67
+ // "Reset sets all registers to zero except t1 t2 and sr"
68
+ this.ora = this.orb = 0x00;
69
+ this.ddra = this.ddrb = 0x00;
70
+ this.ifr = this.ier = 0x00;
71
+ this.t1c = this.t1l = this.t2c = this.t2l = 0x1fffe;
72
+ this.t1hit = this.t2hit = true;
73
+ this.acr = this.pcr = 0;
74
+ this.t1_pb7 = 1;
75
+ this.updateNextTime();
76
+ }
77
+
78
+ updateNextTime() {
79
+ let nextTimer = this.t1c;
80
+ if (!(this.acr & 0x20)) nextTimer = Math.min(this.t2c, nextTimer);
81
+ this.task.reschedule(Math.max(1, nextTimer));
82
+ }
83
+
84
+ _onTimeout() {
85
+ this._catchUp();
86
+ }
87
+
88
+ _catchUp() {
89
+ const cycles = this.scheduler.epoch - this.lastPolltime;
90
+ if (cycles) this._polltime(cycles);
91
+ this.lastPolltime = this.scheduler.epoch;
92
+ this.updateNextTime();
93
+ }
94
+
95
+ _polltime(cycles) {
96
+ cycles |= 0;
97
+ this.justhit = 0;
98
+ const newT1c = this.t1c - cycles;
99
+ if (newT1c < -2) this.t1c = this._handleT1c(newT1c);
100
+ else this.t1c = newT1c;
101
+
102
+ if (!(this.acr & 0x20)) {
103
+ const newT2c = this.t2c - cycles;
104
+ if (newT2c < -2) this.t2c = this._handleT2c(newT2c);
105
+ else this.t2c = newT2c;
106
+ }
107
+ }
108
+
109
+ _handleT1c(newT1c) {
110
+ if (newT1c < -2 && this.t1c > -3) {
111
+ if (!this.t1hit) {
112
+ this.ifr |= TIMER1INT;
113
+ this.updateIFR();
114
+ if (newT1c === -3) this.justhit |= 1;
115
+ this.t1_pb7 = !this.t1_pb7;
116
+ }
117
+ if (!(this.acr & 0x40)) this.t1hit = true;
118
+ }
119
+ while (newT1c < -3) newT1c += this.t1l + 4;
120
+ return newT1c;
121
+ }
122
+
123
+ _handleT2c(newT2c) {
124
+ if (!this.t2hit) {
125
+ this.ifr |= TIMER2INT;
126
+ this.updateIFR();
127
+ if (newT2c === -3) this.justhit |= 2;
128
+ this.t2hit = true;
129
+ }
130
+ newT2c += 0x20000;
131
+ return newT2c;
132
+ }
133
+
134
+ updateIFR() {
135
+ if (this.ifr & this.ier & 0x7f) {
136
+ this.ifr |= 0x80;
137
+ this.cpu.interrupt |= this.irq;
138
+ } else {
139
+ this.ifr &= ~0x80;
140
+ this.cpu.interrupt &= ~this.irq;
141
+ }
142
+ }
143
+
144
+ write(addr, val) {
145
+ this._catchUp();
146
+ let mode;
147
+ val |= 0;
148
+ switch (addr & 0xf) {
149
+ case ORA:
150
+ this.ifr &= ~INT_CA1;
151
+ if ((this.pcr & 0x0a) !== 0x02) {
152
+ // b-em: Not independent interrupt for CA2
153
+ this.ifr &= ~INT_CA2;
154
+ }
155
+ this.updateIFR();
156
+
157
+ mode = this.pcr & 0x0e;
158
+ if (mode === 8) {
159
+ // Handshake mode
160
+ this.setca2(false);
161
+ } else if (mode === 0x0a) {
162
+ // Pulse mode
163
+ this.setca2(false);
164
+ this.setca2(true);
165
+ }
166
+ /* falls through */
167
+ case ORAnh:
168
+ this.ora = val;
169
+ this.recalculatePortAPins();
170
+ break;
171
+
172
+ case ORB:
173
+ this.ifr &= ~INT_CB1;
174
+ if ((this.pcr & 0xa0) !== 0x20) {
175
+ // b-em: Not independent interrupt for CB2
176
+ this.ifr &= ~INT_CB2;
177
+ }
178
+ this.updateIFR();
179
+
180
+ this.orb = val;
181
+ this.recalculatePortBPins();
182
+
183
+ mode = (this.pcr & 0xe0) >>> 4;
184
+ if (mode === 8) {
185
+ // Handshake mode
186
+ this.setcb2(false);
187
+ } else if (mode === 0x0a) {
188
+ // Pulse mode
189
+ this.setcb2(false);
190
+ this.setcb2(true);
191
+ }
192
+ break;
193
+
194
+ case DDRA:
195
+ this.ddra = val;
196
+ this.recalculatePortAPins();
197
+ break;
198
+
199
+ case DDRB:
200
+ this.ddrb = val;
201
+ this.recalculatePortBPins();
202
+ break;
203
+
204
+ case ACR:
205
+ this.acr = val;
206
+ if (this.justhit & 1 && !(val & 0x40)) this.t1hit = true;
207
+ break;
208
+
209
+ case PCR:
210
+ this.pcr = val;
211
+ if ((val & 0xe) === 0xc) this.setca2(false);
212
+ else if (val & 0x08) this.setca2(true);
213
+ if ((val & 0xe0) === 0xc0) this.setcb2(false);
214
+ else if (val & 0x80) this.setcb2(true);
215
+ break;
216
+
217
+ case SR:
218
+ this.sr = val;
219
+ break;
220
+
221
+ case T1LL:
222
+ case T1CL:
223
+ this.t1l &= 0x1fe00;
224
+ this.t1l |= val << 1;
225
+ break;
226
+
227
+ case T1LH:
228
+ this.t1l &= 0x1fe;
229
+ this.t1l |= val << 9;
230
+ if (!(this.justhit & 1)) {
231
+ this.ifr &= ~TIMER1INT;
232
+ this.updateIFR();
233
+ }
234
+ break;
235
+
236
+ case T1CH:
237
+ this.t1l &= 0x1fe;
238
+ this.t1l |= val << 9;
239
+ this.t1c = this.t1l + 1;
240
+ this.t1hit = false;
241
+ if (!(this.justhit & 1)) {
242
+ this.ifr &= ~TIMER1INT;
243
+ this.updateIFR();
244
+ }
245
+ this.t1_pb7 = 0;
246
+ this.updateNextTime();
247
+ break;
248
+
249
+ case T2CL:
250
+ this.t2l &= 0x1fe00;
251
+ this.t2l |= val << 1;
252
+ break;
253
+
254
+ case T2CH:
255
+ this.t2l &= 0x1fe;
256
+ this.t2l |= val << 9;
257
+ this.t2c = this.t2l + 1;
258
+ if (this.acr & 0x20) this.t2c -= 2;
259
+ if (!(this.justhit & 2)) {
260
+ this.ifr &= ~TIMER2INT;
261
+ this.updateIFR();
262
+ }
263
+ this.t2hit = false;
264
+ this.updateNextTime();
265
+ break;
266
+
267
+ case IER:
268
+ if (val & 0x80) this.ier |= val & 0x7f;
269
+ else this.ier &= ~(val & 0x7f);
270
+ this.updateIFR();
271
+ break;
272
+
273
+ case IFR:
274
+ this.ifr &= ~(val & 0x7f);
275
+ if (this.justhit & 1) this.ifr |= TIMER1INT;
276
+ if (this.justhit & 2) this.ifr |= TIMER2INT;
277
+ this.updateIFR();
278
+ break;
279
+ }
280
+ }
281
+
282
+ read(addr) {
283
+ this._catchUp();
284
+ switch (addr & 0xf) {
285
+ case ORA:
286
+ this.ifr &= ~INT_CA1;
287
+ if ((this.pcr & 0xa) !== 0x2) this.ifr &= ~INT_CA2;
288
+ this.updateIFR();
289
+ /* falls through */
290
+ case ORAnh:
291
+ // Reading ORA reads pin levels regardless of DDRA.
292
+ // Of the various 6522 datasheets, this one is clear:
293
+ // http://archive.6502.org/datasheets/wdc_w65c22s_mar_2004.pdf
294
+ if (this.acr & 1) {
295
+ return this.ira;
296
+ }
297
+ this.recalculatePortAPins();
298
+ return this.portapins;
299
+
300
+ case ORB: {
301
+ this.ifr &= ~INT_CB1;
302
+ if ((this.pcr & 0xa0) !== 0x20) this.ifr &= ~INT_CB2;
303
+ this.updateIFR();
304
+
305
+ this.recalculatePortBPins();
306
+ let temp = this.orb & this.ddrb;
307
+ if (this.acr & 2) temp |= this.irb & ~this.ddrb;
308
+ else temp |= this.portbpins & ~this.ddrb;
309
+ // If PB7 is active, it is mixed in regardless of
310
+ // whether bit 7 is an input or output.
311
+ if (this.acr & 0x80) {
312
+ temp &= 0x7f;
313
+ temp |= this.t1_pb7 << 7;
314
+ }
315
+
316
+ return temp;
317
+ }
318
+ case DDRA:
319
+ return this.ddra;
320
+ case DDRB:
321
+ return this.ddrb;
322
+ case T1LL:
323
+ return ((this.t1l & 0x1fe) >>> 1) & 0xff;
324
+ case T1LH:
325
+ return (this.t1l >>> 9) & 0xff;
326
+
327
+ case T1CL:
328
+ if (!(this.justhit & 1)) {
329
+ this.ifr &= ~TIMER1INT;
330
+ this.updateIFR();
331
+ }
332
+ return ((this.t1c + 1) >>> 1) & 0xff;
333
+
334
+ case T1CH:
335
+ return ((this.t1c + 1) >>> 9) & 0xff;
336
+
337
+ case T2CL:
338
+ if (!(this.justhit & 2)) {
339
+ this.ifr &= ~TIMER2INT;
340
+ this.updateIFR();
341
+ }
342
+ return ((this.t2c + 1) >>> 1) & 0xff;
343
+
344
+ case T2CH:
345
+ return ((this.t2c + 1) >>> 9) & 0xff;
346
+
347
+ case SR:
348
+ return this.sr;
349
+ case ACR:
350
+ return this.acr;
351
+ case PCR:
352
+ return this.pcr;
353
+ case IER:
354
+ return this.ier | 0x80;
355
+ case IFR:
356
+ return this.ifr;
357
+ default:
358
+ throw "Unknown VIA read";
359
+ }
360
+ }
361
+
362
+ // May be overridden in subclasses
363
+ drivePortA() {}
364
+
365
+ portAUpdated() {}
366
+
367
+ drivePortB() {}
368
+
369
+ portBUpdated() {}
370
+
371
+ rawPortB() {
372
+ return 0xff;
373
+ }
374
+
375
+ recalculatePortAPins() {
376
+ this.portapins = this.ora & this.ddra;
377
+ this.portapins |= ~this.ddra & 0xff;
378
+ this.drivePortA();
379
+ this.portAUpdated();
380
+ }
381
+
382
+ recalculatePortBPins() {
383
+ const prevPb6 = !!(this.portbpins & 0x40);
384
+ this.portbpins = (this.orb & this.ddrb) | (~this.ddrb & this.rawPortB());
385
+
386
+ this.drivePortB();
387
+ if (prevPb6 && !(this.portbpins & 0x40)) {
388
+ // If we see a high to low transition on pb6, and we are in timer2 pulse counting mode, count a pulse.
389
+ if (this.acr & 0x20) {
390
+ this.t2c -= 2;
391
+ // Not clear what happens here. Docs say:
392
+ // "When the T2 counter reaches a count of zero, IFR5 is set and the counter continues to decrement with
393
+ // each pulse on PB6. To enable IFR5 for subsequent countdowns, it is necessary to reload high order T2
394
+ // counter."
395
+ if (this.t2c < 0) {
396
+ this.t2c = 0xffff;
397
+ this.ifr |= TIMER2INT;
398
+ this.updateIFR();
399
+ }
400
+ }
401
+ }
402
+ this.portBUpdated();
403
+ }
404
+
405
+ setca1(level) {
406
+ if (level === this.ca1) return;
407
+ const pcrSet = !!(this.pcr & 1);
408
+ if (pcrSet === level) {
409
+ if (this.acr & 1) this.ira = this.portapins;
410
+ this.ifr |= INT_CA1;
411
+ this.updateIFR();
412
+ if ((this.pcr & 0xc) === 0x8) {
413
+ // handshaking
414
+ this.setca2(true);
415
+ }
416
+ }
417
+ this.ca1 = level;
418
+ }
419
+
420
+ setca2(level) {
421
+ if (level === this.ca2) return;
422
+ this.ca2 = level;
423
+ const output = !!(this.pcr & 0x08);
424
+ if (this.ca2changecallback) this.ca2changecallback(level, output);
425
+ if (output) return;
426
+ const pcrSet = !!(this.pcr & 4);
427
+ if (pcrSet === level) {
428
+ this.ifr |= INT_CA2;
429
+ this.updateIFR();
430
+ }
431
+ }
432
+
433
+ setcb1(level) {
434
+ if (level === this.cb1) return;
435
+ const pcrSet = !!(this.pcr & 0x10);
436
+ if (pcrSet === level) {
437
+ if (this.acr & 2) this.irb = this.portbpins;
438
+ this.ifr |= INT_CB1;
439
+ this.updateIFR();
440
+ if ((this.pcr & 0xc0) === 0x80) {
441
+ // handshaking
442
+ this.setcb2(true);
443
+ }
444
+ }
445
+ this.cb1 = level;
446
+ }
447
+
448
+ setcb2(level) {
449
+ if (level === this.cb2) return;
450
+ this.cb2 = level;
451
+ const output = !!(this.pcr & 0x80);
452
+ if (this.cb2changecallback) this.cb2changecallback(level, output);
453
+ if (output) return;
454
+ const pcrSet = !!(this.pcr & 0x40);
455
+ if (pcrSet === level) {
456
+ this.ifr |= INT_CB2;
457
+ this.updateIFR();
458
+ }
459
+ }
460
+ }
461
+
462
+ export class SysVia extends Via {
463
+ constructor(cpu, scheduler, video, soundChip, cmos, isMaster, initialLayout, getGamepads) {
464
+ super(cpu, scheduler, 0x01);
465
+
466
+ this.IC32 = 0;
467
+ this.capsLockLight = false;
468
+ this.shiftLockLight = false;
469
+ this.keys = [];
470
+ for (let i = 0; i < 16; ++i) {
471
+ this.keys[i] = new Uint8Array(16);
472
+ }
473
+ // Mouse joystick button state
474
+ this.mouseButton1 = false;
475
+ this.mouseButton2 = false;
476
+ this.keyboardEnabled = true;
477
+ this.setKeyLayout(initialLayout);
478
+ this.video = video;
479
+ this.soundChip = soundChip;
480
+ this.cmos = cmos;
481
+ this.isMaster = isMaster;
482
+ this.getGamepadsFunc = getGamepads;
483
+
484
+ this.reset();
485
+ }
486
+
487
+ setKeyLayout(map) {
488
+ this.keycodeToRowCol = utils.getKeyMap(map);
489
+ }
490
+
491
+ setVBlankInt(level) {
492
+ this.setca1(level);
493
+ }
494
+
495
+ clearKeys() {
496
+ for (let i = 0; i < this.keys.length; ++i) {
497
+ for (let j = 0; j < this.keys[i].length; ++j) {
498
+ this.keys[i][j] = false;
499
+ }
500
+ }
501
+ this.updateKeys();
502
+ }
503
+
504
+ disableKeyboard() {
505
+ this.keyboardEnabled = false;
506
+ this.clearKeys();
507
+ }
508
+
509
+ enableKeyboard() {
510
+ this.keyboardEnabled = true;
511
+ this.clearKeys();
512
+ }
513
+
514
+ set(key, val, shiftDown) {
515
+ if (!this.keyboardEnabled) return;
516
+ const colrow = this.keycodeToRowCol[!!shiftDown][key];
517
+ if (!colrow) return;
518
+ this.keys[colrow[0]][colrow[1]] = val;
519
+ this.updateKeys();
520
+ }
521
+
522
+ keyDown(key, shiftDown) {
523
+ this.set(key, 1, shiftDown);
524
+ }
525
+
526
+ keyUp(key) {
527
+ // set up for both keymaps
528
+ // (with and without shift)
529
+ this.set(key, 0, true);
530
+ this.set(key, 0, false);
531
+ }
532
+
533
+ keyDownRaw(colrow) {
534
+ this.keys[colrow[0]][colrow[1]] = 1;
535
+ this.updateKeys();
536
+ }
537
+
538
+ keyUpRaw(colrow) {
539
+ this.keys[colrow[0]][colrow[1]] = 0;
540
+ this.updateKeys();
541
+ }
542
+
543
+ keyToggleRaw(colrow) {
544
+ this.keys[colrow[0]][colrow[1]] = 1 - this.keys[colrow[0]][colrow[1]];
545
+ this.updateKeys();
546
+ }
547
+
548
+ hasAnyKeyDown() {
549
+ // 10 for BBC, 13 for Master 128
550
+ const numCols = 13;
551
+ for (let i = 0; i < numCols; ++i) {
552
+ for (let j = 0; j < 8; ++j) {
553
+ if (this.keys[i][j]) {
554
+ return true;
555
+ }
556
+ }
557
+ }
558
+ return false;
559
+ }
560
+
561
+ updateKeys() {
562
+ // 10 for BBC, 13 for Master 128
563
+ const numCols = 13;
564
+ if (this.IC32 & 8) {
565
+ for (let i = 0; i < numCols; ++i) {
566
+ for (let j = 1; j < 8; ++j) {
567
+ if (this.keys[i][j]) {
568
+ this.setca2(true);
569
+ return;
570
+ }
571
+ }
572
+ }
573
+ } else {
574
+ // Keyboard sets bit 7 to 0 or 1, and testing shows it always
575
+ // "wins" vs. CMOS.
576
+ // At 0 also wins against an output pin.
577
+
578
+ const portapins = this.portapins;
579
+ const keyrow = (portapins >>> 4) & 7;
580
+ const keycol = portapins & 0xf;
581
+ if (!this.keys[keycol][keyrow]) {
582
+ this.portapins &= 0x7f;
583
+ } else if (!(this.ddra & 0x80)) {
584
+ this.portapins |= 0x80;
585
+ }
586
+
587
+ if (keycol < numCols) {
588
+ for (let j = 1; j < 8; ++j) {
589
+ if (this.keys[keycol][j]) {
590
+ this.setca2(true);
591
+ return;
592
+ }
593
+ }
594
+ }
595
+ }
596
+ this.setca2(false);
597
+ }
598
+
599
+ portAUpdated() {
600
+ this.updateKeys();
601
+ this.soundChip.updateSlowDataBus(this.portapins, !(this.IC32 & 1));
602
+ }
603
+
604
+ rawPortB() {
605
+ let result = 0xff;
606
+ // AUG p418
607
+ // ### PB4 and PB5 inputs
608
+ // These are the inputs from the joystick FIRE buttons. They are active low.
609
+ const buttons = this.getJoysticks();
610
+ if (buttons.button1) result &= ~(1 << 4); // Clear PB4 if button1 pressed
611
+ if (buttons.button2) result &= ~(1 << 5); // Clear PB5 if button2 pressed
612
+ return result;
613
+ }
614
+
615
+ portBUpdated() {
616
+ const portbpins = this.portbpins;
617
+ if (portbpins & 8) this.IC32 |= 1 << (portbpins & 7);
618
+ else this.IC32 &= ~(1 << (portbpins & 7));
619
+
620
+ this.capsLockLight = !(this.IC32 & 0x40);
621
+ this.shiftLockLight = !(this.IC32 & 0x80);
622
+
623
+ this.video.setScreenHwScroll((this.IC32 >> 4) & 3);
624
+
625
+ if (this.isMaster) this.cmos.writeControl(portbpins, this.portapins, this.IC32);
626
+
627
+ // Updating IC32 may have enabled peripherals attached to port A.
628
+ this.recalculatePortAPins();
629
+ }
630
+
631
+ drivePortA() {
632
+ // For experiments where we tested these behaviors, see:
633
+ // https://stardot.org.uk/forums/viewtopic.php?f=4&t=17597
634
+ // If either keyboard or CMOS pulls a given pin low, it "wins"
635
+ // vs. via output.
636
+ let busval = 0xff;
637
+ if (this.isMaster) busval &= this.cmos.read(this.IC32);
638
+ this.portapins &= busval;
639
+ this.updateKeys();
640
+ }
641
+
642
+ drivePortB() {
643
+ // Nothing driving here.
644
+ // Note that if speech were fitted, it drives bit 7 low.
645
+ }
646
+
647
+ getGamepads() {
648
+ if (this.getGamepadsFunc) return this.getGamepadsFunc();
649
+ return null;
650
+ }
651
+
652
+ /**
653
+ * Set joystick button state (for mouse joystick)
654
+ * @param {number} buttonNumber - Button number (0 for button1, 1 for button2)
655
+ * @param {boolean} pressed - Whether the button is pressed
656
+ */
657
+ setJoystickButton(buttonNumber, pressed) {
658
+ if (buttonNumber === 0) {
659
+ this.mouseButton1 = pressed;
660
+ } else if (buttonNumber === 1) {
661
+ this.mouseButton2 = pressed;
662
+ }
663
+ // Trigger port B recalculation to update button state
664
+ this.recalculatePortBPins();
665
+ }
666
+
667
+ getJoysticks() {
668
+ let button1 = this.mouseButton1; // Start with mouse button state
669
+ let button2 = this.mouseButton2;
670
+
671
+ const pads = this.getGamepads();
672
+ if (pads && pads[0]) {
673
+ const pad = pads[0];
674
+ const pad2 = pads[1];
675
+
676
+ // Combine gamepad and mouse button states (OR logic)
677
+ button1 = button1 || pad.buttons[10].pressed;
678
+ // if two gamepads, use button from 2nd
679
+ // otherwise use 2nd button from first
680
+ button2 = button2 || (pad2 ? pad2.buttons[10].pressed : pad.buttons[11].pressed);
681
+ }
682
+
683
+ return { button1: button1, button2: button2 };
684
+ }
685
+ }
686
+
687
+ export class UserVia extends Via {
688
+ constructor(cpu, scheduler, isMaster, userPortPeripheral) {
689
+ super(cpu, scheduler, 0x02);
690
+ this.isMaster = isMaster;
691
+ this.userPortPeripheral = userPortPeripheral;
692
+ this.reset();
693
+ }
694
+
695
+ portBUpdated() {
696
+ this.userPortPeripheral.write(this.portbpins);
697
+ }
698
+
699
+ drivePortB() {
700
+ this.portbpins &= this.userPortPeripheral.read();
701
+ }
702
+ }