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.
- package/.editorconfig +15 -0
- package/.git-blame-ignore-revs +3 -0
- package/.github/copilot-instructions.md +94 -0
- package/.github/workflows/claude-issue-triage.yml +105 -0
- package/.github/workflows/claude.yml +63 -0
- package/.github/workflows/release-please.yml +75 -0
- package/.github/workflows/test-and-deploy.yml +86 -0
- package/.gitmodules +6 -0
- package/.husky/pre-commit +1 -0
- package/.idea/codeStyleSettings.xml +9 -0
- package/.idea/codeStyles/Project.xml +62 -0
- package/.idea/codeStyles/codeStyleConfig.xml +5 -0
- package/.idea/compiler.xml +22 -0
- package/.idea/copyright/profiles_settings.xml +3 -0
- package/.idea/encodings.xml +6 -0
- package/.idea/inspectionProfiles/Project_Default.xml +7 -0
- package/.idea/jsLibraryMappings.xml +6 -0
- package/.idea/jsLinters/jshint.xml +85 -0
- package/.idea/jsLinters/jslint.xml +15 -0
- package/.idea/jsbeeb.iml +11 -0
- package/.idea/misc.xml +6 -0
- package/.idea/modules.xml +8 -0
- package/.idea/prettier.xml +7 -0
- package/.idea/runConfigurations/Debug.xml +5 -0
- package/.idea/scopes/scope_settings.xml +5 -0
- package/.idea/vcs.xml +8 -0
- package/.prettierignore +4 -0
- package/.prettierrc.json +1 -0
- package/.release-please-manifest.json +3 -0
- package/.vscode/launch.json +14 -0
- package/.vscode/settings.json +6 -0
- package/CHANGELOG.md +32 -0
- package/CLAUDE.md +136 -0
- package/COPYING +674 -0
- package/Dockerfile +22 -0
- package/Makefile +30 -0
- package/README.md +259 -0
- package/docker/nginx-default.conf +10 -0
- package/docs/pal-comb-filter-research.md +129 -0
- package/docs/pal-simulation-design.md +368 -0
- package/eslint.config.js +35 -0
- package/index.html +954 -0
- package/jsconfig.json +10 -0
- package/package.json +102 -0
- package/public/discs/README.Irq-Timing +3 -0
- package/public/discs/README.bcdtest +5 -0
- package/public/discs/README.elite +6 -0
- package/public/discs/README.eng_test +3 -0
- package/public/discs/README.protection +7 -0
- package/public/favicon.ico +0 -0
- package/public/images/botbar.png +0 -0
- package/public/images/cub-monitor.png +0 -0
- package/public/images/jsbeeb-example.png +0 -0
- package/public/images/placeholder.png +0 -0
- package/public/images/red-off-16.png +0 -0
- package/public/images/red-on-16.png +0 -0
- package/public/images/sb/CD-left.jpg +0 -0
- package/public/images/sb/CD-right.jpg +0 -0
- package/public/images/sidebar.png +0 -0
- package/public/images/tv.png +0 -0
- package/public/images/yellow-off-16.png +0 -0
- package/public/images/yellow-on-16.png +0 -0
- package/public/jsbeeb-icon.png +0 -0
- package/public/robots.txt +3 -0
- package/public/roms/ADFS1-53.rom +0 -0
- package/public/roms/BASIC.ROM +0 -0
- package/public/roms/README +4 -0
- package/public/roms/a01/BASIC1.rom +0 -0
- package/public/roms/ample.rom +0 -0
- package/public/roms/ats-3.0.rom +0 -0
- package/public/roms/b/DFS-0.9.rom +0 -0
- package/public/roms/b/DFS-1.2.rom +0 -0
- package/public/roms/b1770/dfs1770.rom +0 -0
- package/public/roms/b1770/zADFS.ROM +0 -0
- package/public/roms/bp/dfs.rom +0 -0
- package/public/roms/bp/zADFS.ROM +0 -0
- package/public/roms/bpos.rom +0 -0
- package/public/roms/compact/adfs210.rom +0 -0
- package/public/roms/compact/basic48.rom +0 -0
- package/public/roms/compact/basic486.rom +0 -0
- package/public/roms/compact/os51.rom +0 -0
- package/public/roms/compact/utils.rom +0 -0
- package/public/roms/deos.rom +0 -0
- package/public/roms/master/anfs-4.25.rom +0 -0
- package/public/roms/master/mos.txt +68819 -0
- package/public/roms/master/mos3.20 +0 -0
- package/public/roms/os.rom +0 -0
- package/public/roms/os01.rom +0 -0
- package/public/roms/tube/6502Tube.rom +0 -0
- package/public/roms/tube/ARMeval_100.rom +0 -0
- package/public/roms/tube/BIOS.ROM +0 -0
- package/public/roms/tube/ReCo6502ROM_816 +0 -0
- package/public/roms/tube/Z80_120.rom +0 -0
- package/public/roms/us/USBASIC.rom +0 -0
- package/public/roms/us/USDNFS.rom +0 -0
- package/public/roms/usmos.rom +0 -0
- package/public/sounds/disc525/motor.wav +0 -0
- package/public/sounds/disc525/motoroff.wav +0 -0
- package/public/sounds/disc525/motoron.wav +0 -0
- package/public/sounds/disc525/seek.wav +0 -0
- package/public/sounds/disc525/seek2.wav +0 -0
- package/public/sounds/disc525/seek3.wav +0 -0
- package/public/sounds/disc525/step.wav +0 -0
- package/public/teletext/txt0.dat +0 -0
- package/public/teletext/txt1.dat +0 -0
- package/public/teletext/txt2.dat +0 -0
- package/public/teletext/txt3.dat +0 -0
- package/release-please-config.json +13 -0
- package/run-container.sh +92 -0
- package/src/6502.js +1347 -0
- package/src/6502.opcodes.js +1411 -0
- package/src/acia.js +261 -0
- package/src/adc.js +149 -0
- package/src/analogue-source.js +21 -0
- package/src/app/app.js +175 -0
- package/src/app/electron.js +20 -0
- package/src/app/preload.js +8 -0
- package/src/app-bench.js +33 -0
- package/src/basic/multiline-tetris +9 -0
- package/src/basic-tokenise.js +104 -0
- package/src/canvas.js +177 -0
- package/src/cmos.js +141 -0
- package/src/config.js +165 -0
- package/src/ddnoise.js +138 -0
- package/src/disc-drive.js +371 -0
- package/src/disc-hfe.js +396 -0
- package/src/disc.js +997 -0
- package/src/econet/L3FS.dat +0 -0
- package/src/econet/scsi.dat +0 -0
- package/src/econet.js +714 -0
- package/src/fake6502.js +32 -0
- package/src/fdc.js +248 -0
- package/src/filestore.js +666 -0
- package/src/gamepad-source.js +59 -0
- package/src/gamepads.js +268 -0
- package/src/google-drive.js +160 -0
- package/src/intel-fdc.js +1717 -0
- package/src/jsbeeb.css +363 -0
- package/src/keyboard.js +411 -0
- package/src/lib/README +1 -0
- package/src/lib/webgl-debug.js +911 -0
- package/src/main.js +1759 -0
- package/src/microphone-input.js +149 -0
- package/src/models.js +200 -0
- package/src/mouse-joystick-source.js +107 -0
- package/src/music5000-worklet.js +43 -0
- package/src/music5000.js +207 -0
- package/src/scheduler.js +148 -0
- package/src/serial.js +31 -0
- package/src/soundchip.js +314 -0
- package/src/sth.js +59 -0
- package/src/tapes.js +265 -0
- package/src/teletext.js +348 -0
- package/src/teletext_adaptor.js +172 -0
- package/src/teletext_data.js +1064 -0
- package/src/touchscreen.js +86 -0
- package/src/tube.js +349 -0
- package/src/url-params.js +256 -0
- package/src/utils.js +1090 -0
- package/src/via.js +702 -0
- package/src/video-filters/pal-composite.js +94 -0
- package/src/video-filters/passthrough-filter.js +70 -0
- package/src/video-filters/shaders/pal-composite.frag.glsl +147 -0
- package/src/video-filters/shaders/pal-composite.vert.glsl +8 -0
- package/src/video-filters/shaders/passthrough.frag.glsl +6 -0
- package/src/video-filters/shaders/passthrough.vert.glsl +7 -0
- package/src/video.js +794 -0
- package/src/wd-fdc.js +1344 -0
- package/src/web/audio-handler.js +146 -0
- package/src/web/audio-renderer.js +115 -0
- package/src/web/debug.js +529 -0
- package/tests/integration/RmwX.asm +47 -0
- package/tests/integration/TestInstructionsSource +27 -0
- package/tests/integration/TestTimingsResults +27 -0
- package/tests/integration/TestTimingsSource +61 -0
- package/tests/integration/bcd.js +23 -0
- package/tests/integration/dormann.js +101 -0
- package/tests/integration/dp111timing.js +42 -0
- package/tests/integration/ensure-submodules.js +25 -0
- package/tests/integration/nops.bas +119 -0
- package/tests/integration/nops.js +24 -0
- package/tests/integration/protection.js +26 -0
- package/tests/integration/rmw.js +69 -0
- package/tests/integration/teletext/expected_bug_469.png +0 -0
- package/tests/integration/teletext/expected_flash_0.png +0 -0
- package/tests/integration/teletext/expected_flash_1.png +0 -0
- package/tests/integration/teletext/expected_hoglet_held_char.png +0 -0
- package/tests/integration/teletext/expected_reveal_flash_0.png +0 -0
- package/tests/integration/teletext/expected_reveal_flash_1.png +0 -0
- package/tests/integration/teletext.js +126 -0
- package/tests/integration/timings.js +56 -0
- package/tests/integration/via.js +1125 -0
- package/tests/suite/README.md +7 -0
- package/tests/suite/Test Suite 2.15.txt +373 -0
- package/tests/suite/bin/ start +0 -0
- package/tests/suite/bin/adca +0 -0
- package/tests/suite/bin/adcax +0 -0
- package/tests/suite/bin/adcay +0 -0
- package/tests/suite/bin/adcb +0 -0
- package/tests/suite/bin/adcix +0 -0
- package/tests/suite/bin/adciy +0 -0
- package/tests/suite/bin/adcz +0 -0
- package/tests/suite/bin/adczx +0 -0
- package/tests/suite/bin/alrb +0 -0
- package/tests/suite/bin/ancb +0 -0
- package/tests/suite/bin/anda +0 -0
- package/tests/suite/bin/andax +0 -0
- package/tests/suite/bin/anday +0 -0
- package/tests/suite/bin/andb +0 -0
- package/tests/suite/bin/andix +0 -0
- package/tests/suite/bin/andiy +0 -0
- package/tests/suite/bin/andz +0 -0
- package/tests/suite/bin/andzx +0 -0
- package/tests/suite/bin/aneb +0 -0
- package/tests/suite/bin/arrb +0 -0
- package/tests/suite/bin/asla +0 -0
- package/tests/suite/bin/aslax +0 -0
- package/tests/suite/bin/asln +0 -0
- package/tests/suite/bin/aslz +0 -0
- package/tests/suite/bin/aslzx +0 -0
- package/tests/suite/bin/asoa +0 -0
- package/tests/suite/bin/asoax +0 -0
- package/tests/suite/bin/asoay +0 -0
- package/tests/suite/bin/asoix +0 -0
- package/tests/suite/bin/asoiy +0 -0
- package/tests/suite/bin/asoz +0 -0
- package/tests/suite/bin/asozx +0 -0
- package/tests/suite/bin/axsa +0 -0
- package/tests/suite/bin/axsix +0 -0
- package/tests/suite/bin/axsz +0 -0
- package/tests/suite/bin/axszy +0 -0
- package/tests/suite/bin/bccr +0 -0
- package/tests/suite/bin/bcsr +0 -0
- package/tests/suite/bin/beqr +0 -0
- package/tests/suite/bin/bita +0 -0
- package/tests/suite/bin/bitz +0 -0
- package/tests/suite/bin/bmir +0 -0
- package/tests/suite/bin/bner +0 -0
- package/tests/suite/bin/bplr +0 -0
- package/tests/suite/bin/branchwrap +0 -0
- package/tests/suite/bin/brkn +0 -0
- package/tests/suite/bin/bvcr +0 -0
- package/tests/suite/bin/bvsr +0 -0
- package/tests/suite/bin/cia1pb6 +0 -0
- package/tests/suite/bin/cia1pb7 +0 -0
- package/tests/suite/bin/cia1ta +0 -0
- package/tests/suite/bin/cia1tab +0 -0
- package/tests/suite/bin/cia1tb +0 -0
- package/tests/suite/bin/cia1tb123 +0 -0
- package/tests/suite/bin/cia2pb6 +0 -0
- package/tests/suite/bin/cia2pb7 +0 -0
- package/tests/suite/bin/cia2ta +0 -0
- package/tests/suite/bin/cia2tb +0 -0
- package/tests/suite/bin/cia2tb123 +0 -0
- package/tests/suite/bin/clcn +0 -0
- package/tests/suite/bin/cldn +0 -0
- package/tests/suite/bin/clin +0 -0
- package/tests/suite/bin/clvn +0 -0
- package/tests/suite/bin/cmpa +0 -0
- package/tests/suite/bin/cmpax +0 -0
- package/tests/suite/bin/cmpay +0 -0
- package/tests/suite/bin/cmpb +0 -0
- package/tests/suite/bin/cmpix +0 -0
- package/tests/suite/bin/cmpiy +0 -0
- package/tests/suite/bin/cmpz +0 -0
- package/tests/suite/bin/cmpzx +0 -0
- package/tests/suite/bin/cntdef +0 -0
- package/tests/suite/bin/cnto2 +0 -0
- package/tests/suite/bin/cpuport +0 -0
- package/tests/suite/bin/cputiming +0 -0
- package/tests/suite/bin/cpxa +0 -0
- package/tests/suite/bin/cpxb +0 -0
- package/tests/suite/bin/cpxz +0 -0
- package/tests/suite/bin/cpya +0 -0
- package/tests/suite/bin/cpyb +0 -0
- package/tests/suite/bin/cpyz +0 -0
- package/tests/suite/bin/dcma +0 -0
- package/tests/suite/bin/dcmax +0 -0
- package/tests/suite/bin/dcmay +0 -0
- package/tests/suite/bin/dcmix +0 -0
- package/tests/suite/bin/dcmiy +0 -0
- package/tests/suite/bin/dcmz +0 -0
- package/tests/suite/bin/dcmzx +0 -0
- package/tests/suite/bin/deca +0 -0
- package/tests/suite/bin/decax +0 -0
- package/tests/suite/bin/decz +0 -0
- package/tests/suite/bin/deczx +0 -0
- package/tests/suite/bin/dexn +0 -0
- package/tests/suite/bin/deyn +0 -0
- package/tests/suite/bin/eora +0 -0
- package/tests/suite/bin/eorax +0 -0
- package/tests/suite/bin/eoray +0 -0
- package/tests/suite/bin/eorb +0 -0
- package/tests/suite/bin/eorix +0 -0
- package/tests/suite/bin/eoriy +0 -0
- package/tests/suite/bin/eorz +0 -0
- package/tests/suite/bin/eorzx +0 -0
- package/tests/suite/bin/finish +0 -0
- package/tests/suite/bin/flipos +0 -0
- package/tests/suite/bin/icr01 +0 -0
- package/tests/suite/bin/imr +0 -0
- package/tests/suite/bin/inca +0 -0
- package/tests/suite/bin/incax +0 -0
- package/tests/suite/bin/incz +0 -0
- package/tests/suite/bin/inczx +0 -0
- package/tests/suite/bin/insa +0 -0
- package/tests/suite/bin/insax +0 -0
- package/tests/suite/bin/insay +0 -0
- package/tests/suite/bin/insix +0 -0
- package/tests/suite/bin/insiy +0 -0
- package/tests/suite/bin/insz +0 -0
- package/tests/suite/bin/inszx +0 -0
- package/tests/suite/bin/inxn +0 -0
- package/tests/suite/bin/inyn +0 -0
- package/tests/suite/bin/irq +0 -0
- package/tests/suite/bin/jmpi +0 -0
- package/tests/suite/bin/jmpw +0 -0
- package/tests/suite/bin/jsrw +0 -0
- package/tests/suite/bin/lasay +0 -0
- package/tests/suite/bin/laxa +0 -0
- package/tests/suite/bin/laxay +0 -0
- package/tests/suite/bin/laxix +0 -0
- package/tests/suite/bin/laxiy +0 -0
- package/tests/suite/bin/laxz +0 -0
- package/tests/suite/bin/laxzy +0 -0
- package/tests/suite/bin/ldaa +0 -0
- package/tests/suite/bin/ldaax +0 -0
- package/tests/suite/bin/ldaay +0 -0
- package/tests/suite/bin/ldab +0 -0
- package/tests/suite/bin/ldaix +0 -0
- package/tests/suite/bin/ldaiy +0 -0
- package/tests/suite/bin/ldaz +0 -0
- package/tests/suite/bin/ldazx +0 -0
- package/tests/suite/bin/ldxa +0 -0
- package/tests/suite/bin/ldxay +0 -0
- package/tests/suite/bin/ldxb +0 -0
- package/tests/suite/bin/ldxz +0 -0
- package/tests/suite/bin/ldxzy +0 -0
- package/tests/suite/bin/ldya +0 -0
- package/tests/suite/bin/ldyax +0 -0
- package/tests/suite/bin/ldyb +0 -0
- package/tests/suite/bin/ldyz +0 -0
- package/tests/suite/bin/ldyzx +0 -0
- package/tests/suite/bin/loadth +0 -0
- package/tests/suite/bin/lsea +0 -0
- package/tests/suite/bin/lseax +0 -0
- package/tests/suite/bin/lseay +0 -0
- package/tests/suite/bin/lseix +0 -0
- package/tests/suite/bin/lseiy +0 -0
- package/tests/suite/bin/lsez +0 -0
- package/tests/suite/bin/lsezx +0 -0
- package/tests/suite/bin/lsra +0 -0
- package/tests/suite/bin/lsrax +0 -0
- package/tests/suite/bin/lsrn +0 -0
- package/tests/suite/bin/lsrz +0 -0
- package/tests/suite/bin/lsrzx +0 -0
- package/tests/suite/bin/lxab +0 -0
- package/tests/suite/bin/mmu +0 -0
- package/tests/suite/bin/mmufetch +0 -0
- package/tests/suite/bin/nmi +0 -0
- package/tests/suite/bin/nopa +0 -0
- package/tests/suite/bin/nopax +0 -0
- package/tests/suite/bin/nopb +0 -0
- package/tests/suite/bin/nopn +0 -0
- package/tests/suite/bin/nopz +0 -0
- package/tests/suite/bin/nopzx +0 -0
- package/tests/suite/bin/oneshot +0 -0
- package/tests/suite/bin/oraa +0 -0
- package/tests/suite/bin/oraax +0 -0
- package/tests/suite/bin/oraay +0 -0
- package/tests/suite/bin/orab +0 -0
- package/tests/suite/bin/oraix +0 -0
- package/tests/suite/bin/oraiy +0 -0
- package/tests/suite/bin/oraz +0 -0
- package/tests/suite/bin/orazx +0 -0
- package/tests/suite/bin/phan +0 -0
- package/tests/suite/bin/phpn +0 -0
- package/tests/suite/bin/plan +0 -0
- package/tests/suite/bin/plpn +0 -0
- package/tests/suite/bin/rlaa +0 -0
- package/tests/suite/bin/rlaax +0 -0
- package/tests/suite/bin/rlaay +0 -0
- package/tests/suite/bin/rlaix +0 -0
- package/tests/suite/bin/rlaiy +0 -0
- package/tests/suite/bin/rlaz +0 -0
- package/tests/suite/bin/rlazx +0 -0
- package/tests/suite/bin/rola +0 -0
- package/tests/suite/bin/rolax +0 -0
- package/tests/suite/bin/roln +0 -0
- package/tests/suite/bin/rolz +0 -0
- package/tests/suite/bin/rolzx +0 -0
- package/tests/suite/bin/rora +0 -0
- package/tests/suite/bin/rorax +0 -0
- package/tests/suite/bin/rorn +0 -0
- package/tests/suite/bin/rorz +0 -0
- package/tests/suite/bin/rorzx +0 -0
- package/tests/suite/bin/rraa +0 -0
- package/tests/suite/bin/rraax +0 -0
- package/tests/suite/bin/rraay +0 -0
- package/tests/suite/bin/rraix +0 -0
- package/tests/suite/bin/rraiy +0 -0
- package/tests/suite/bin/rraz +0 -0
- package/tests/suite/bin/rrazx +0 -0
- package/tests/suite/bin/rtin +0 -0
- package/tests/suite/bin/rtsn +0 -0
- package/tests/suite/bin/sbca +0 -0
- package/tests/suite/bin/sbcax +0 -0
- package/tests/suite/bin/sbcay +0 -0
- package/tests/suite/bin/sbcb +0 -0
- package/tests/suite/bin/sbcb(eb) +0 -0
- package/tests/suite/bin/sbcix +0 -0
- package/tests/suite/bin/sbciy +0 -0
- package/tests/suite/bin/sbcz +0 -0
- package/tests/suite/bin/sbczx +0 -0
- package/tests/suite/bin/sbxb +0 -0
- package/tests/suite/bin/secn +0 -0
- package/tests/suite/bin/sedn +0 -0
- package/tests/suite/bin/sein +0 -0
- package/tests/suite/bin/shaay +0 -0
- package/tests/suite/bin/shaiy +0 -0
- package/tests/suite/bin/shsay +0 -0
- package/tests/suite/bin/shxay +0 -0
- package/tests/suite/bin/shyax +0 -0
- package/tests/suite/bin/staa +0 -0
- package/tests/suite/bin/staax +0 -0
- package/tests/suite/bin/staay +0 -0
- package/tests/suite/bin/staix +0 -0
- package/tests/suite/bin/staiy +0 -0
- package/tests/suite/bin/staz +0 -0
- package/tests/suite/bin/stazx +0 -0
- package/tests/suite/bin/stxa +0 -0
- package/tests/suite/bin/stxz +0 -0
- package/tests/suite/bin/stxzy +0 -0
- package/tests/suite/bin/stya +0 -0
- package/tests/suite/bin/styz +0 -0
- package/tests/suite/bin/styzx +0 -0
- package/tests/suite/bin/taxn +0 -0
- package/tests/suite/bin/tayn +0 -0
- package/tests/suite/bin/trap1 +0 -0
- package/tests/suite/bin/trap10 +0 -0
- package/tests/suite/bin/trap11 +0 -0
- package/tests/suite/bin/trap12 +0 -0
- package/tests/suite/bin/trap13 +0 -0
- package/tests/suite/bin/trap14 +0 -0
- package/tests/suite/bin/trap15 +0 -0
- package/tests/suite/bin/trap16 +0 -0
- package/tests/suite/bin/trap17 +0 -0
- package/tests/suite/bin/trap2 +0 -0
- package/tests/suite/bin/trap3 +0 -0
- package/tests/suite/bin/trap4 +0 -0
- package/tests/suite/bin/trap5 +0 -0
- package/tests/suite/bin/trap6 +0 -0
- package/tests/suite/bin/trap7 +0 -0
- package/tests/suite/bin/trap8 +0 -0
- package/tests/suite/bin/trap9 +0 -0
- package/tests/suite/bin/tsxn +0 -0
- package/tests/suite/bin/txan +0 -0
- package/tests/suite/bin/txsn +0 -0
- package/tests/suite/bin/tyan +0 -0
- package/tests/suite/cbm-hackers-post.html +178 -0
- package/tests/suite/cbm-hackers-post.md +78 -0
- package/tests/test-machine.js +288 -0
- package/tests/test-suite.js +147 -0
- package/tests/test.css +7 -0
- package/tests/unit/gzip/test-1 +0 -0
- package/tests/unit/gzip/test-1.gz +0 -0
- package/tests/unit/gzip/test-2 +0 -0
- package/tests/unit/gzip/test-2.gz +0 -0
- package/tests/unit/gzip/test-3 +0 -0
- package/tests/unit/gzip/test-3.gz +0 -0
- package/tests/unit/gzip/test-4 +0 -0
- package/tests/unit/gzip/test-4.gz +0 -0
- package/tests/unit/test-adc.js +307 -0
- package/tests/unit/test-bcd.js +30 -0
- package/tests/unit/test-cmos.js +266 -0
- package/tests/unit/test-disc-drive.js +85 -0
- package/tests/unit/test-disc-hfe.js +347 -0
- package/tests/unit/test-disc.js +232 -0
- package/tests/unit/test-fifo.js +35 -0
- package/tests/unit/test-gamepad-source.js +67 -0
- package/tests/unit/test-gzip.js +22 -0
- package/tests/unit/test-intel-fdc.js +93 -0
- package/tests/unit/test-keyboard.js +410 -0
- package/tests/unit/test-mouse-joystick-source.js +128 -0
- package/tests/unit/test-scheduler.js +190 -0
- package/tests/unit/test-serial.js +154 -0
- package/tests/unit/test-teletext-adaptor.js +359 -0
- package/tests/unit/test-tokenise.js +65 -0
- package/tests/unit/test-url-params.js +398 -0
- package/tests/unit/test-utils.js +276 -0
- package/tests/unit/test-video.js +498 -0
- package/tests/unit/test-zip.js +56 -0
- package/tests/unit/zip/test-mixed.zip +0 -0
- package/tests/unit/zip/test-rom.zip +0 -0
- package/tests/unit/zip/test-ssd.zip +0 -0
- package/tools/fir-generator.js +80 -0
- package/tools/vite-plugin-fir-shader.js +131 -0
- 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
|
+
}
|
package/src/teletext.js
ADDED
|
@@ -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
|
+
}
|