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/tapes.js ADDED
@@ -0,0 +1,265 @@
1
+ "use strict";
2
+ import * as utils from "./utils.js";
3
+
4
+ function secsToClocks(secs) {
5
+ return (2 * 1000 * 1000 * secs) | 0;
6
+ }
7
+
8
+ function parityOf(curByte) {
9
+ let parity = false;
10
+ while (curByte) {
11
+ parity = !parity;
12
+ curByte >>>= 1;
13
+ }
14
+ return parity;
15
+ }
16
+
17
+ const ParityN = "N".charCodeAt(0);
18
+
19
+ class UefTape {
20
+ constructor(stream) {
21
+ this.stream = stream;
22
+ this.baseFrequency = 1200;
23
+ this.rewind();
24
+
25
+ this.curChunk = this.readChunk();
26
+ }
27
+
28
+ rewind() {
29
+ this.dummyData = [false, false, true, false, true, false, true, false, true, true];
30
+ this.state = -1;
31
+ this.count = 0;
32
+ this.curByte = 0;
33
+ this.numDataBits = 8;
34
+ this.parity = ParityN;
35
+ this.numParityBits = 0;
36
+ this.numStopBits = 1;
37
+ this.carrierBefore = 0;
38
+ this.carrierAfter = 0;
39
+
40
+ this.stream.seek(10);
41
+ const minor = this.stream.readByte();
42
+ const major = this.stream.readByte();
43
+ if (major !== 0x00) throw "Unsupported UEF version " + major + "." + minor;
44
+ }
45
+
46
+ readChunk() {
47
+ const chunkId = this.stream.readInt16();
48
+ const length = this.stream.readInt32();
49
+ return {
50
+ id: chunkId,
51
+ stream: this.stream.substream(length),
52
+ };
53
+ }
54
+
55
+ poll(acia) {
56
+ if (!this.curChunk) return;
57
+ if (this.state === -1) {
58
+ if (this.stream.eof()) {
59
+ this.curChunk = null;
60
+ return;
61
+ }
62
+ this.curChunk = this.readChunk();
63
+ }
64
+
65
+ let gap;
66
+ switch (this.curChunk.id) {
67
+ case 0x0000:
68
+ console.log("Origin: " + this.curChunk.stream.readNulString());
69
+ break;
70
+ case 0x0100:
71
+ acia.setTapeCarrier(false);
72
+ if (this.state === -1) {
73
+ this.state = 0;
74
+ this.curByte = this.curChunk.stream.readByte();
75
+ acia.tone(this.baseFrequency); // Start bit
76
+ } else if (this.state < 9) {
77
+ if (this.state === 0) {
78
+ // Start bit
79
+ acia.tone(this.baseFrequency);
80
+ } else {
81
+ acia.tone(this.curByte & (1 << (this.state - 1)) ? 2 * this.baseFrequency : this.baseFrequency);
82
+ }
83
+ this.state++;
84
+ } else {
85
+ acia.receive(this.curByte);
86
+ acia.tone(2 * this.baseFrequency); // Stop bit
87
+ if (this.curChunk.stream.eof()) {
88
+ this.state = -1;
89
+ } else {
90
+ this.state = 0;
91
+ this.curByte = this.curChunk.stream.readByte();
92
+ }
93
+ }
94
+ return this.cycles(1);
95
+ case 0x0104: // Defined data
96
+ acia.setTapeCarrier(false);
97
+ if (this.state === -1) {
98
+ this.numDataBits = this.curChunk.stream.readByte();
99
+ this.parity = this.curChunk.stream.readByte();
100
+ this.numStopBits = this.curChunk.stream.readByte();
101
+ this.numParityBits = this.parity !== ParityN ? 1 : 0;
102
+ console.log(
103
+ `Defined data with ${this.numDataBits}${String.fromCharCode(this.parity)}${this.numStopBits}`,
104
+ );
105
+ this.state = 0;
106
+ }
107
+ if (this.state === 0) {
108
+ if (this.curChunk.stream.eof()) {
109
+ this.state = -1;
110
+ } else {
111
+ this.curByte = this.curChunk.stream.readByte() & ((1 << this.numDataBits) - 1);
112
+ acia.tone(this.baseFrequency); // Start bit
113
+ this.state++;
114
+ }
115
+ } else if (this.state < 1 + this.numDataBits) {
116
+ acia.tone(this.curByte & (1 << (this.state - 1)) ? 2 * this.baseFrequency : this.baseFrequency);
117
+ this.state++;
118
+ } else if (this.state < 1 + this.numDataBits + this.numParityBits) {
119
+ let bit = parityOf(this.curByte);
120
+ if (this.parity === ParityN) bit = !bit;
121
+ acia.tone(bit ? 2 * this.baseFrequency : this.baseFrequency);
122
+ this.state++;
123
+ } else if (this.state < 1 + this.numDataBits + this.numParityBits + this.numStopBits) {
124
+ acia.tone(2 * this.baseFrequency); // Stop bits
125
+ this.state++;
126
+ } else {
127
+ acia.receive(this.curByte);
128
+ this.state = 0;
129
+ return 0;
130
+ }
131
+ return this.cycles(1);
132
+ case 0x0111: // Carrier tone with dummy data
133
+ if (this.state === -1) {
134
+ this.state = 0;
135
+ this.carrierBefore = this.curChunk.stream.readInt16();
136
+ this.carrierAfter = this.curChunk.stream.readInt16();
137
+ console.log("Carrier with", this.carrierBefore, this.carrierAfter);
138
+ }
139
+ if (this.state === 0) {
140
+ acia.setTapeCarrier(true);
141
+ acia.tone(2 * this.baseFrequency);
142
+ this.carrierBefore--;
143
+ if (this.carrierBefore <= 0) this.state = 1;
144
+ } else if (this.state < 11) {
145
+ acia.setTapeCarrier(false);
146
+ acia.tone(this.dummyData[this.state - 1] ? this.baseFrequency : 2 * this.baseFrequency);
147
+ if (this.state === 10) {
148
+ acia.receive(0xaa);
149
+ }
150
+ this.state++;
151
+ } else {
152
+ acia.setTapeCarrier(true);
153
+ acia.tone(2 * this.baseFrequency);
154
+ this.carrierAfter--;
155
+ if (this.carrierAfter <= 0) this.state = -1;
156
+ }
157
+ return this.cycles(1);
158
+ case 0x0114:
159
+ console.log("Ignoring security cycles");
160
+ break;
161
+ case 0x0115:
162
+ console.log("Ignoring polarity change");
163
+ break;
164
+ case 0x0110: // Carrier tone.
165
+ if (this.state === -1) {
166
+ this.state = 0;
167
+ this.count = this.curChunk.stream.readInt16();
168
+ }
169
+ acia.setTapeCarrier(true);
170
+ acia.tone(2 * this.baseFrequency);
171
+ this.count--;
172
+ if (this.count <= 0) this.state = -1;
173
+ return this.cycles(1);
174
+ case 0x0113:
175
+ this.baseFrequency = this.curChunk.stream.readFloat32();
176
+ console.log("Frequency change ", this.baseFrequency);
177
+ break;
178
+ case 0x0112:
179
+ acia.setTapeCarrier(false);
180
+ gap = 1 / (2 * this.curChunk.stream.readInt16() * this.baseFrequency);
181
+ console.log("Tape gap of " + gap + "s");
182
+ acia.tone(0);
183
+ return secsToClocks(gap);
184
+ case 0x0116:
185
+ acia.setTapeCarrier(false);
186
+ gap = this.curChunk.stream.readFloat32();
187
+ console.log("Tape gap of " + gap + "s");
188
+ acia.tone(0);
189
+ return secsToClocks(gap);
190
+ default:
191
+ console.log("Skipping unknown chunk " + utils.hexword(this.curChunk.id));
192
+ this.curChunk = this.readChunk();
193
+ break;
194
+ }
195
+ return this.cycles(1);
196
+ }
197
+
198
+ cycles(count) {
199
+ return secsToClocks(count / this.baseFrequency);
200
+ }
201
+ }
202
+
203
+ const dividerTable = [1, 16, 64, -1];
204
+
205
+ class TapefileTape {
206
+ constructor(stream) {
207
+ this.count = 0;
208
+ this.stream = stream;
209
+ }
210
+
211
+ rate(acia) {
212
+ let bitsPerByte = 9;
213
+ if (!(acia.cr & 0x80)) {
214
+ bitsPerByte++; // Not totally correct if the AUG is to be believed.
215
+ }
216
+ const divider = dividerTable[acia.cr & 0x03];
217
+ // http://beebwiki.mdfs.net/index.php/Serial_ULA says the serial rate is ignored
218
+ // for cassette mode.
219
+ const cpp = (2 * 1000 * 1000) / (19200 / divider);
220
+ return Math.floor(bitsPerByte * cpp);
221
+ }
222
+
223
+ rewind() {
224
+ this.stream.seek(10);
225
+ }
226
+
227
+ poll(acia) {
228
+ if (this.stream.eof()) return 100000;
229
+ let byte = this.stream.readByte();
230
+ if (byte === 0xff) {
231
+ byte = this.stream.readByte();
232
+ if (byte === 0) {
233
+ acia.setTapeCarrier(false);
234
+ return 0;
235
+ } else if (byte === 0x04) {
236
+ acia.setTapeCarrier(true);
237
+ // Simulate 5 seconds of carrier.
238
+ return 5 * 2 * 1000 * 1000;
239
+ } else if (byte !== 0xff) {
240
+ throw "Got a weird byte in the tape";
241
+ }
242
+ }
243
+ acia.receive(byte);
244
+ return this.rate(acia);
245
+ }
246
+ }
247
+
248
+ export function loadTapeFromData(name, data) {
249
+ const stream = new utils.DataStream(name, data);
250
+ if (stream.readByte(0) === 0xff && stream.readByte(1) === 0x04) {
251
+ console.log("Detected a 'tapefile' tape");
252
+ return new TapefileTape(stream);
253
+ }
254
+ if (stream.readNulString(0) === "UEF File!") {
255
+ console.log("Detected a UEF tape");
256
+ return new UefTape(stream);
257
+ }
258
+ console.log("Unknown tape format");
259
+ return null;
260
+ }
261
+
262
+ export async function loadTape(name) {
263
+ console.log("Loading tape from " + name);
264
+ return loadTapeFromData(name, await utils.loadData(name));
265
+ }
@@ -0,0 +1,348 @@
1
+ "use strict";
2
+ import { makeChars } from "./teletext_data.js";
3
+ import { makeFast32 } from "./utils.js";
4
+
5
+ export class Teletext {
6
+ constructor() {
7
+ this.prevCol = 0;
8
+ this.col = 7;
9
+ this.bg = 0;
10
+ this.sep = false;
11
+ this.dbl = this.oldDbl = this.secondHalfOfDouble = this.wasDbl = false;
12
+ this.gfx = false;
13
+ this.flash = this.flashOn = false;
14
+ this.flashTime = 0;
15
+ this.heldChar = 0;
16
+ this.holdChar = false;
17
+ this.dataQueue = [0, 0, 0, 0];
18
+ this.scanlineCounter = 0;
19
+ this.levelDEW = false;
20
+ this.levelDISPTMG = false;
21
+ this.levelRA0 = false;
22
+
23
+ this.normalGlyphs = makeFast32(new Uint32Array(96 * 20));
24
+ this.graphicsGlyphs = makeFast32(new Uint32Array(96 * 20));
25
+ this.separatedGlyphs = makeFast32(new Uint32Array(96 * 20));
26
+ this.colour = makeFast32(new Uint32Array(256));
27
+
28
+ this.nextGlyphs = this.normalGlyphs;
29
+ this.curGlyphs = this.normalGlyphs;
30
+ this.heldGlyphs = this.normalGlyphs;
31
+
32
+ this.init();
33
+ }
34
+
35
+ init() {
36
+ const charData = makeChars();
37
+
38
+ // Build palette
39
+ const gamma = 1.0 / 2.2;
40
+ for (let i = 0; i < 256; ++i) {
41
+ const alpha = (i & 3) / 3.0;
42
+ const foregroundR = !!(i & 4);
43
+ const foregroundG = !!(i & 8);
44
+ const foregroundB = !!(i & 16);
45
+ const backgroundR = !!(i & 32);
46
+ const backgroundG = !!(i & 64);
47
+ const backgroundB = !!(i & 128);
48
+ // Gamma-corrected blending
49
+ const blendedR = Math.pow(foregroundR * alpha + backgroundR * (1.0 - alpha), gamma) * 240;
50
+ const blendedG = Math.pow(foregroundG * alpha + backgroundG * (1.0 - alpha), gamma) * 240;
51
+ const blendedB = Math.pow(foregroundB * alpha + backgroundB * (1.0 - alpha), gamma) * 240;
52
+ this.colour[i] = blendedR | (blendedG << 8) | (blendedB << 16) | (0xff << 24);
53
+ }
54
+
55
+ function getLoResGlyphRow(c, row) {
56
+ if (row < 0 || row >= 20) {
57
+ return 0;
58
+ } else {
59
+ let index = c * 60 + (row >>> 1) * 6;
60
+ let result = 0;
61
+ for (let x = 0; x < 6; ++x) {
62
+ result |= (charData[index++] * 3) << (x * 2);
63
+ }
64
+ return result;
65
+ }
66
+ }
67
+
68
+ function combineRows(a, b) {
69
+ return a | ((a >>> 1) & b & ~(b >>> 1)) | ((a << 1) & b & ~(b << 1));
70
+ }
71
+
72
+ function makeHiResGlyphs(dest, graphicsGlyphs) {
73
+ let index = 0;
74
+ for (let c = 0; c < 96; ++c) {
75
+ for (let row = 0; row < 20; ++row) {
76
+ let data;
77
+ if (!graphicsGlyphs || !!(c & 32)) {
78
+ data = combineRows(getLoResGlyphRow(c, row), getLoResGlyphRow(c, row + (row & 1 ? 1 : -1)));
79
+ } else {
80
+ data = getLoResGlyphRow(c, row);
81
+ }
82
+ dest[index++] =
83
+ (data & 0x1) * 0x7 +
84
+ (data & 0x2) * 0x14 +
85
+ (data & 0x4) * 0x34 +
86
+ (data & 0x8) * 0xe0 +
87
+ (data & 0x10) * 0x280 +
88
+ (data & 0x20) * 0x680 +
89
+ (data & 0x40) * 0x1c00 +
90
+ (data & 0x80) * 0x5000 +
91
+ (data & 0x100) * 0xd000 +
92
+ (data & 0x200) * 0x38000 +
93
+ (data & 0x400) * 0xa0000 +
94
+ (data & 0x800) * 0x1a0000;
95
+ }
96
+ }
97
+ }
98
+
99
+ makeHiResGlyphs(this.normalGlyphs, false);
100
+
101
+ function setGraphicsBlock(c, x, y, w, h, sep, n) {
102
+ for (let yy = 0; yy < h; ++yy) {
103
+ for (let xx = 0; xx < w; ++xx) {
104
+ charData[c * 60 + (y + yy) * 6 + (x + xx)] = sep && (xx === 0 || yy === h - 1) ? 0 : n;
105
+ }
106
+ }
107
+ }
108
+
109
+ // Build graphics character set
110
+ for (let c = 0; c < 96; ++c) {
111
+ if (!(c & 32)) {
112
+ setGraphicsBlock(c, 0, 0, 3, 3, false, !!(c & 1));
113
+ setGraphicsBlock(c, 3, 0, 3, 3, false, !!(c & 2));
114
+ setGraphicsBlock(c, 0, 3, 3, 4, false, !!(c & 4));
115
+ setGraphicsBlock(c, 3, 3, 3, 4, false, !!(c & 8));
116
+ setGraphicsBlock(c, 0, 7, 3, 3, false, !!(c & 16));
117
+ setGraphicsBlock(c, 3, 7, 3, 3, false, !!(c & 64));
118
+ }
119
+ }
120
+
121
+ makeHiResGlyphs(this.graphicsGlyphs, true);
122
+
123
+ // Build separated graphics character set
124
+ for (let c = 0; c < 96; ++c) {
125
+ if (!(c & 32)) {
126
+ setGraphicsBlock(c, 0, 0, 3, 3, true, !!(c & 1));
127
+ setGraphicsBlock(c, 3, 0, 3, 3, true, !!(c & 2));
128
+ setGraphicsBlock(c, 0, 3, 3, 4, true, !!(c & 4));
129
+ setGraphicsBlock(c, 3, 3, 3, 4, true, !!(c & 8));
130
+ setGraphicsBlock(c, 0, 7, 3, 3, true, !!(c & 16));
131
+ setGraphicsBlock(c, 3, 7, 3, 3, true, !!(c & 64));
132
+ }
133
+ }
134
+
135
+ makeHiResGlyphs(this.separatedGlyphs, true);
136
+ }
137
+
138
+ setNextChars() {
139
+ if (this.gfx) {
140
+ if (this.sep) {
141
+ this.nextGlyphs = this.separatedGlyphs;
142
+ } else {
143
+ this.nextGlyphs = this.graphicsGlyphs;
144
+ }
145
+ } else {
146
+ this.nextGlyphs = this.normalGlyphs;
147
+ }
148
+ }
149
+
150
+ handleControlCode(data) {
151
+ const wasGfx = this.gfx;
152
+ const wasHoldChar = this.holdChar;
153
+
154
+ switch (data) {
155
+ case 1:
156
+ case 2:
157
+ case 3:
158
+ case 4:
159
+ case 5:
160
+ case 6:
161
+ case 7:
162
+ this.gfx = false;
163
+ this.col = data;
164
+ this.setNextChars();
165
+ break;
166
+ case 8:
167
+ this.flash = true;
168
+ break;
169
+ case 9:
170
+ this.flash = false;
171
+ break;
172
+ case 12:
173
+ case 13:
174
+ this.dbl = !!(data & 1);
175
+ if (this.dbl) this.wasDbl = true;
176
+ break;
177
+ case 17:
178
+ case 18:
179
+ case 19:
180
+ case 20:
181
+ case 21:
182
+ case 22:
183
+ case 23:
184
+ this.gfx = true;
185
+ this.col = data & 7;
186
+ this.setNextChars();
187
+ break;
188
+ case 24:
189
+ this.col = this.prevCol = this.bg;
190
+ break;
191
+ case 25:
192
+ this.sep = false;
193
+ this.setNextChars();
194
+ break;
195
+ case 26:
196
+ this.sep = true;
197
+ this.setNextChars();
198
+ break;
199
+ case 28:
200
+ this.bg = 0;
201
+ break;
202
+ case 29:
203
+ this.bg = this.col;
204
+ break;
205
+ case 30:
206
+ this.holdChar = true;
207
+ break;
208
+ case 31:
209
+ this.holdChar = false;
210
+ break;
211
+ }
212
+ if (wasGfx && (wasHoldChar || this.holdChar) && this.dbl === this.oldDbl) {
213
+ data = this.heldChar;
214
+ if (data >= 0x40 && data < 0x60) data = 0x20;
215
+ this.curGlyphs = this.heldGlyphs;
216
+ } else {
217
+ this.heldChar = 0x20;
218
+ data = 0x20;
219
+ }
220
+
221
+ return data;
222
+ }
223
+
224
+ fetchData(data) {
225
+ this.dataQueue.shift();
226
+ this.dataQueue.push(data & 0x7f);
227
+ }
228
+
229
+ setDEW(level) {
230
+ // The SAA5050 input pin "DEW" is connected to the 6845 output pin
231
+ // "VSYNC" and it is used to track frames.
232
+ const oldLevel = this.levelDEW;
233
+ this.levelDEW = level;
234
+
235
+ // Trigger on high -> low. This appears to be what the hardware does.
236
+ // It needs to be this way for the scanline counter to stay in sync
237
+ // if you set R6>R4.
238
+ if (!oldLevel || level) {
239
+ return;
240
+ }
241
+
242
+ this.scanlineCounter = 0;
243
+ this.secondHalfOfDouble = false;
244
+
245
+ // 3:1 flash ratio.
246
+ if (++this.flashTime === 64) this.flashTime = 0;
247
+ // Flashing text starts off in sync with a slow cursor, extinguished
248
+ // together. Multiple MODE changes gradually desynchronise the
249
+ // frame counters.
250
+ // TODO: this point is being reached a MOS-dependent number of times
251
+ // before Video.frameCount rises. The next line achieves initial
252
+ // sync under MOS 1.20 only.
253
+ this.flashOn = this.flashTime < 16;
254
+ }
255
+
256
+ setDISPTMG(level) {
257
+ // The SAA5050 input pin "LOSE" is connected to the 6845 output pin
258
+ // "DISPTMG" and it is used to track scanlines.
259
+ const oldLevel = this.levelDISPTMG;
260
+ this.levelDISPTMG = level;
261
+
262
+ // Trigger on high -> low. This is probably what the hardware does as
263
+ // we need to increment scanline at the end of the scanline, not the
264
+ // beginning.
265
+ if (!oldLevel || level) {
266
+ return;
267
+ }
268
+
269
+ this.col = 7;
270
+ this.bg = 0;
271
+ this.holdChar = false;
272
+ this.heldChar = 0x20;
273
+ this.nextGlyphs = this.heldGlyphs = this.normalGlyphs;
274
+ this.flash = false;
275
+ this.sep = false;
276
+ this.gfx = false;
277
+ this.dbl = false;
278
+
279
+ this.scanlineCounter++;
280
+ // Check for end of character row.
281
+ if (this.scanlineCounter === 10) {
282
+ this.scanlineCounter = 0;
283
+
284
+ if (this.secondHalfOfDouble) {
285
+ this.secondHalfOfDouble = false;
286
+ } else {
287
+ this.secondHalfOfDouble = this.wasDbl;
288
+ }
289
+ }
290
+
291
+ this.wasDbl = false;
292
+ }
293
+
294
+ setRA0(level) {
295
+ // The SAA5050 input pin "CRS" is connected to the 6845 output pin
296
+ // "RA0", via a signal inverter, and it is used to select between a
297
+ // normal scanline and a calculated smoothing scanline.
298
+ this.levelRA0 = level;
299
+ }
300
+
301
+ render(buf, offset) {
302
+ let data = this.dataQueue[0];
303
+
304
+ let scanline = this.scanlineCounter << 1;
305
+ if (this.levelRA0) {
306
+ scanline++;
307
+ }
308
+
309
+ this.oldDbl = this.dbl;
310
+
311
+ this.prevCol = this.col;
312
+ this.curGlyphs = this.nextGlyphs;
313
+
314
+ const prevFlash = this.flash;
315
+ if (data < 0x20) {
316
+ data = this.handleControlCode(data);
317
+ } else if (this.gfx) {
318
+ if (data & 0x20) {
319
+ this.heldChar = data;
320
+ this.heldGlyphs = this.curGlyphs;
321
+ }
322
+ } else {
323
+ this.heldChar = 32;
324
+ }
325
+
326
+ if (this.oldDbl) {
327
+ scanline = scanline >>> 1;
328
+ if (this.secondHalfOfDouble) {
329
+ scanline += 10;
330
+ }
331
+ }
332
+ let chardef = this.curGlyphs[(data - 32) * 20 + scanline];
333
+
334
+ if ((prevFlash && this.flashOn) || (this.secondHalfOfDouble && !this.dbl)) {
335
+ const backgroundColour = this.colour[(this.bg & 7) << 5];
336
+ for (let i = 0; i < 16; ++i) {
337
+ buf[offset++] = backgroundColour;
338
+ }
339
+ } else {
340
+ const paletteIndex = ((this.bg & 7) << 5) | ((this.prevCol & 7) << 2);
341
+
342
+ for (let pixel = 0; pixel < 16; ++pixel) {
343
+ buf[offset + pixel] = this.colour[paletteIndex + (chardef & 3)];
344
+ chardef >>>= 2;
345
+ }
346
+ }
347
+ }
348
+ }