BBC Operating System OS 1.20 ============================ Commented disassembly by Geoff Cox, originally published on Micronet. Additional comments by J.G.Harston. Source: https://mdfs.net/Docs/Comp/BBC/OS1-20/ Labels inserted from Acorn's source code by Greg Cook, 27 July 2022 Source: https://www.4corn.co.uk/articles/65hostandmos/ Source: https://github.com/stardot/AcornOS120 * = Label defined but not referenced in the MOS source code. ***************** VDU CHARACTER FONT LOOK UP TABLE **************************** These are the default definitions for characters 32-127. The are accessed with OSWORD 10 (read character definition) and reprogrammed with VDU 23 (define character). If the character set is not exploded, then a block of 32 characters is copied to the soft character buffer at &0C00 when a character is defined. C000 DB 00 ;00000000 ........ &20 32 - ' ' C001 DB 00 ;00000000 ........ C002 DB 00 ;00000000 ........ C003 DB 00 ;00000000 ........ C004 DB 00 ;00000000 ........ C005 DB 00 ;00000000 ........ C006 DB 00 ;00000000 ........ C007 DB 00 ;00000000 ........ C008 DB 18 ;00011000 ...**... &21 33 - '!' C009 DB 18 ;00011000 ...**... C00A DB 18 ;00011000 ...**... C00B DB 18 ;00011000 ...**... C00C DB 18 ;00011000 ...**... C00D DB 00 ;00000000 ........ C00E DB 18 ;00011000 ...**... C00F DB 00 ;00000000 ........ C010 DB 6C ;01101100 .**.**.. &22 34 - '"' C011 DB 6C ;01101100 .**.**.. C012 DB 6C ;01101100 .**.**.. C013 DB 00 ;00000000 ........ C014 DB 00 ;00000000 ........ C015 DB 00 ;00000000 ........ C016 DB 00 ;00000000 ........ C017 DB 00 ;00000000 ........ C018 DB 36 ;00110110 ..**.**. &23 35 - '#' C019 DB 36 ;00110110 ..**.**. C01A DB 7F ;01111111 .******* C01B DB 36 ;00110110 ..**.**. C01C DB 7F ;01111111 .******* C01D DB 36 ;00110110 ..**.**. C01E DB 36 ;00110110 ..**.**. C01F DB 00 ;00000000 ........ C020 DB 0C ;00001100 ....**.. &24 36 - '$' C021 DB 3F ;00111111 ..****** C022 DB 68 ;01101000 .**.*... C023 DB 3E ;00111110 ..*****. C024 DB 0B ;00001011 ....*.** C025 DB 7E ;01111110 .******. C026 DB 18 ;00011000 ...**... C027 DB 00 ;00000000 ........ C028 DB 60 ;01100000 .**..... &25 37 - '%' C029 DB 66 ;01100110 .**..**. C02A DB 0C ;00001100 ....**.. C02B DB 18 ;00011000 ...**... C02C DB 30 ;00110000 ..**.... C02D DB 66 ;01100110 .**..**. C02E DB 06 ;00000110 .....**. C02F DB 00 ;00000000 ........ C030 DB 38 ;00111000 ..***... &26 38 - '&' C031 DB 6C ;01101100 .**.**.. C032 DB 6C ;01101100 .**.**.. C033 DB 38 ;00111000 ..***... C034 DB 6D ;01101101 .**.**.* C035 DB 66 ;01100110 .**..**. C036 DB 3B ;00111011 ..***.** C037 DB 00 ;00000000 ........ C038 DB 0C ;00001100 ....**.. &27 39 - ''' C039 DB 18 ;00011000 ...**... C03A DB 30 ;00110000 ..**.... C03B DB 00 ;00000000 ........ C03C DB 00 ;00000000 ........ C03D DB 00 ;00000000 ........ C03E DB 00 ;00000000 ........ C03F DB 00 ;00000000 ........ C040 DB 0C ;00001100 ....**.. &28 40 - '(' C041 DB 18 ;00011000 ...**... C042 DB 30 ;00110000 ..**.... C043 DB 30 ;00110000 ..**.... C044 DB 30 ;00110000 ..**.... C045 DB 18 ;00011000 ...**... C046 DB 0C ;00001100 ....**.. C047 DB 00 ;00000000 ........ C048 DB 30 ;00110000 ..**.... &29 41 - ')' C049 DB 18 ;00011000 ...**... C04A DB 0C ;00001100 ....**.. C04B DB 0C ;00001100 ....**.. C04C DB 0C ;00001100 ....**.. C04D DB 18 ;00011000 ...**... C04E DB 30 ;00110000 ..**.... C04F DB 00 ;00000000 ........ C050 DB 00 ;00000000 ........ &2A 42 - '*' C051 DB 18 ;00011000 ...**... C052 DB 7E ;01111110 .******. C053 DB 3C ;00111100 ..****.. C054 DB 7E ;01111110 .******. C055 DB 18 ;00011000 ...**... C056 DB 00 ;00000000 ........ C057 DB 00 ;00000000 ........ C058 DB 00 ;00000000 ........ &2B 43 - '+' C059 DB 18 ;00011000 ...**... C05A DB 18 ;00011000 ...**... C05B DB 7E ;01111110 .******. C05C DB 18 ;00011000 ...**... C05D DB 18 ;00011000 ...**... C05E DB 00 ;00000000 ........ C05F DB 00 ;00000000 ........ C060 DB 00 ;00000000 ........ &2C 44 - ',' C061 DB 00 ;00000000 ........ C062 DB 00 ;00000000 ........ C063 DB 00 ;00000000 ........ C064 DB 00 ;00000000 ........ C065 DB 18 ;00011000 ...**... C066 DB 18 ;00011000 ...**... C067 DB 30 ;00110000 ..**.... C068 DB 00 ;00000000 ........ &2D 45 - '-' C069 DB 00 ;00000000 ........ C06A DB 00 ;00000000 ........ C06B DB 7E ;01111110 .******. C06C DB 00 ;00000000 ........ C06D DB 00 ;00000000 ........ C06E DB 00 ;00000000 ........ C06F DB 00 ;00000000 ........ C070 DB 00 ;00000000 ........ &2E 46 - '.' C071 DB 00 ;00000000 ........ C072 DB 00 ;00000000 ........ C073 DB 00 ;00000000 ........ C074 DB 00 ;00000000 ........ C075 DB 18 ;00011000 ...**... C076 DB 18 ;00011000 ...**... C077 DB 00 ;00000000 ........ C078 DB 00 ;00000000 ........ &2F 47 - '/' C079 DB 06 ;00000110 .....**. C07A DB 0C ;00001100 ....**.. C07B DB 18 ;00011000 ...**... C07C DB 30 ;00110000 ..**.... C07D DB 60 ;01100000 .**..... C07E DB 00 ;00000000 ........ C07F DB 00 ;00000000 ........ C080 DB 3C ;00111100 ..****.. &30 48 - '0' C081 DB 66 ;01100110 .**..**. C082 DB 6E ;01101110 .**.***. C083 DB 7E ;01111110 .******. C084 DB 76 ;01110110 .***.**. C085 DB 66 ;01100110 .**..**. C086 DB 3C ;00111100 ..****.. C087 DB 00 ;00000000 ........ C088 DB 18 ;00011000 ...**... &31 49 - '1' C089 DB 38 ;00111000 ..***... C08A DB 18 ;00011000 ...**... C08B DB 18 ;00011000 ...**... C08C DB 18 ;00011000 ...**... C08D DB 18 ;00011000 ...**... C08E DB 7E ;01111110 .******. C08F DB 00 ;00000000 ........ C090 DB 3C ;00111100 ..****.. &32 50 - '2' C091 DB 66 ;01100110 .**..**. C092 DB 06 ;00000110 .....**. C093 DB 0C ;00001100 ....**.. C094 DB 18 ;00011000 ...**... C095 DB 30 ;00110000 ..**.... C096 DB 7E ;01111110 .******. C097 DB 00 ;00000000 ........ C098 DB 3C ;00111100 ..****.. &33 51 - '3' C099 DB 66 ;01100110 .**..**. C09A DB 06 ;00000110 .....**. C09B DB 1C ;00011100 ...***.. C09C DB 06 ;00000110 .....**. C09D DB 66 ;01100110 .**..**. C09E DB 3C ;00111100 ..****.. C09F DB 00 ;00000000 ........ C0A0 DB 0C ;00001100 ....**.. &34 52 - '4' C0A1 DB 1C ;00011100 ...***.. C0A2 DB 3C ;00111100 ..****.. C0A3 DB 6C ;01101100 .**.**.. C0A4 DB 7E ;01111110 .******. C0A5 DB 0C ;00001100 ....**.. C0A6 DB 0C ;00001100 ....**.. C0A7 DB 00 ;00000000 ........ C0A8 DB 7E ;01111110 .******. &35 53 - '5' C0A9 DB 60 ;01100000 .**..... C0AA DB 7C ;01111100 .*****.. C0AB DB 06 ;00000110 .....**. C0AC DB 06 ;00000110 .....**. C0AD DB 66 ;01100110 .**..**. C0AE DB 3C ;00111100 ..****.. C0AF DB 00 ;00000000 ........ C0B0 DB 1C ;00011100 ...***.. &36 54 - '6' C0B1 DB 30 ;00110000 ..**.... C0B2 DB 60 ;01100000 .**..... C0B3 DB 7C ;01111100 .*****.. C0B4 DB 66 ;01100110 .**..**. C0B5 DB 66 ;01100110 .**..**. C0B6 DB 3C ;00111100 ..****.. C0B7 DB 00 ;00000000 ........ C0B8 DB 7E ;01111110 .******. &37 55 - '7' C0B9 DB 06 ;00000110 .....**. C0BA DB 0C ;00001100 ....**.. C0BB DB 18 ;00011000 ...**... C0BC DB 30 ;00110000 ..**.... C0BD DB 30 ;00110000 ..**.... C0BE DB 30 ;00110000 ..**.... C0BF DB 00 ;00000000 ........ C0C0 DB 3C ;00111100 ..****.. &38 56 - '8' C0C1 DB 66 ;01100110 .**..**. C0C2 DB 66 ;01100110 .**..**. C0C3 DB 3C ;00111100 ..****.. C0C4 DB 66 ;01100110 .**..**. C0C5 DB 66 ;01100110 .**..**. C0C6 DB 3C ;00111100 ..****.. C0C7 DB 00 ;00000000 ........ C0C8 DB 3C ;00111100 ..****.. &39 57 - '9' C0C9 DB 66 ;01100110 .**..**. C0CA DB 66 ;01100110 .**..**. C0CB DB 3E ;00111110 ..*****. C0CC DB 06 ;00000110 .....**. C0CD DB 0C ;00001100 ....**.. C0CE DB 38 ;00111000 ..***... C0CF DB 00 ;00000000 ........ C0D0 DB 00 ;00000000 ........ &3A 58 - ':' C0D1 DB 00 ;00000000 ........ C0D2 DB 18 ;00011000 ...**... C0D3 DB 18 ;00011000 ...**... C0D4 DB 00 ;00000000 ........ C0D5 DB 18 ;00011000 ...**... C0D6 DB 18 ;00011000 ...**... C0D7 DB 00 ;00000000 ........ C0D8 DB 00 ;00000000 ........ &3B 59 - ';' C0D9 DB 00 ;00000000 ........ C0DA DB 18 ;00011000 ...**... C0DB DB 18 ;00011000 ...**... C0DC DB 00 ;00000000 ........ C0DD DB 18 ;00011000 ...**... C0DE DB 18 ;00011000 ...**... C0DF DB 30 ;00110000 ..**.... C0E0 DB 0C ;00001100 ....**.. &3C 60 - '<' C0E1 DB 18 ;00011000 ...**... C0E2 DB 30 ;00110000 ..**.... C0E3 DB 60 ;01100000 .**..... C0E4 DB 30 ;00110000 ..**.... C0E5 DB 18 ;00011000 ...**... C0E6 DB 0C ;00001100 ....**.. C0E7 DB 00 ;00000000 ........ C0E8 DB 00 ;00000000 ........ &3D 61 - '=' C0E9 DB 00 ;00000000 ........ C0EA DB 7E ;01111110 .******. C0EB DB 00 ;00000000 ........ C0EC DB 7E ;01111110 .******. C0ED DB 00 ;00000000 ........ C0EE DB 00 ;00000000 ........ C0EF DB 00 ;00000000 ........ C0F0 DB 30 ;00110000 ..**.... &3E 62 - '>' C0F1 DB 18 ;00011000 ...**... C0F2 DB 0C ;00001100 ....**.. C0F3 DB 06 ;00000110 .....**. C0F4 DB 0C ;00001100 ....**.. C0F5 DB 18 ;00011000 ...**... C0F6 DB 30 ;00110000 ..**.... C0F7 DB 00 ;00000000 ........ C0F8 DB 3C ;00111100 ..****.. &3F 63 - '?' C0F9 DB 66 ;01100110 .**..**. C0FA DB 0C ;00001100 ....**.. C0FB DB 18 ;00011000 ...**... C0FC DB 18 ;00011000 ...**... C0FD DB 00 ;00000000 ........ C0FE DB 18 ;00011000 ...**... C0FF DB 00 ;00000000 ........ C100 DB 3C ;00111100 ..****.. &40 64 - '@' C101 DB 66 ;01100110 .**..**. C102 DB 6E ;01101110 .**.***. C103 DB 6A ;01101010 .**.*.*. C104 DB 6E ;01101110 .**.***. C105 DB 60 ;01100000 .**..... C106 DB 3C ;00111100 ..****.. C107 DB 00 ;00000000 ........ C108 DB 3C ;00111100 ..****.. &41 65 - 'A' C109 DB 66 ;01100110 .**..**. C10A DB 66 ;01100110 .**..**. C10B DB 7E ;01111110 .******. C10C DB 66 ;01100110 .**..**. C10D DB 66 ;01100110 .**..**. C10E DB 66 ;01100110 .**..**. C10F DB 00 ;00000000 ........ C110 DB 7C ;01111100 .*****.. &42 66 - 'B' C111 DB 66 ;01100110 .**..**. C112 DB 66 ;01100110 .**..**. C113 DB 7C ;01111100 .*****.. C114 DB 66 ;01100110 .**..**. C115 DB 66 ;01100110 .**..**. C116 DB 7C ;01111100 .*****.. C117 DB 00 ;00000000 ........ C118 DB 3C ;00111100 ..****.. &43 67 - 'C' C119 DB 66 ;01100110 .**..**. C11A DB 60 ;01100000 .**..... C11B DB 60 ;01100000 .**..... C11C DB 60 ;01100000 .**..... C11D DB 66 ;01100110 .**..**. C11E DB 3C ;00111100 ..****.. C11F DB 00 ;00000000 ........ C120 DB 78 ;01111000 .****... &44 68 - 'D' C121 DB 6C ;01101100 .**.**.. C122 DB 66 ;01100110 .**..**. C123 DB 66 ;01100110 .**..**. C124 DB 66 ;01100110 .**..**. C125 DB 6C ;01101100 .**.**.. C126 DB 78 ;01111000 .****... C127 DB 00 ;00000000 ........ C128 DB 7E ;01111110 .******. &45 69 - 'E' C129 DB 60 ;01100000 .**..... C12A DB 60 ;01100000 .**..... C12B DB 7C ;01111100 .*****.. C12C DB 60 ;01100000 .**..... C12D DB 60 ;01100000 .**..... C12E DB 7E ;01111110 .******. C12F DB 00 ;00000000 ........ C130 DB 7E ;01111110 .******. &46 70 - 'F' C131 DB 60 ;01100000 .**..... C132 DB 60 ;01100000 .**..... C133 DB 7C ;01111100 .*****.. C134 DB 60 ;01100000 .**..... C135 DB 60 ;01100000 .**..... C136 DB 60 ;01100000 .**..... C137 DB 00 ;00000000 ........ C138 DB 3C ;00111100 ..****.. &47 71 - 'G' C139 DB 66 ;01100110 .**..**. C13A DB 60 ;01100000 .**..... C13B DB 6E ;01101110 .**.***. C13C DB 66 ;01100110 .**..**. C13D DB 66 ;01100110 .**..**. C13E DB 3C ;00111100 ..****.. C13F DB 00 ;00000000 ........ C140 DB 66 ;01100110 .**..**. &48 72 - 'H' C141 DB 66 ;01100110 .**..**. C142 DB 66 ;01100110 .**..**. C143 DB 7E ;01111110 .******. C144 DB 66 ;01100110 .**..**. C145 DB 66 ;01100110 .**..**. C146 DB 66 ;01100110 .**..**. C147 DB 00 ;00000000 ........ C148 DB 7E ;01111110 .******. &49 73 - 'I' C149 DB 18 ;00011000 ...**... C14A DB 18 ;00011000 ...**... C14B DB 18 ;00011000 ...**... C14C DB 18 ;00011000 ...**... C14D DB 18 ;00011000 ...**... C14E DB 7E ;01111110 .******. C14F DB 00 ;00000000 ........ C150 DB 3E ;00111110 ..*****. &4A 74 - 'J' C151 DB 0C ;00001100 ....**.. C152 DB 0C ;00001100 ....**.. C153 DB 0C ;00001100 ....**.. C154 DB 0C ;00001100 ....**.. C155 DB 6C ;01101100 .**.**.. C156 DB 38 ;00111000 ..***... C157 DB 00 ;00000000 ........ C158 DB 66 ;01100110 .**..**. &4B 75 - 'K' C159 DB 6C ;01101100 .**.**.. C15A DB 78 ;01111000 .****... C15B DB 70 ;01110000 .***.... C15C DB 78 ;01111000 .****... C15D DB 6C ;01101100 .**.**.. C15E DB 66 ;01100110 .**..**. C15F DB 00 ;00000000 ........ C160 DB 60 ;01100000 .**..... &4C 76 - 'L' C161 DB 60 ;01100000 .**..... C162 DB 60 ;01100000 .**..... C163 DB 60 ;01100000 .**..... C164 DB 60 ;01100000 .**..... C165 DB 60 ;01100000 .**..... C166 DB 7E ;01111110 .******. C167 DB 00 ;00000000 ........ C168 DB 63 ;01100011 .**...** &4D 77 - 'M' C169 DB 77 ;01110111 .***.*** C16A DB 7F ;01111111 .******* C16B DB 6B ;01101011 .**.*.** C16C DB 6B ;01101011 .**.*.** C16D DB 63 ;01100011 .**...** C16E DB 63 ;01100011 .**...** C16F DB 00 ;00000000 ........ C170 DB 66 ;01100110 .**..**. &4E 78 - 'N' C171 DB 66 ;01100110 .**..**. C172 DB 76 ;01110110 .***.**. C173 DB 7E ;01111110 .******. C174 DB 6E ;01101110 .**.***. C175 DB 66 ;01100110 .**..**. C176 DB 66 ;01100110 .**..**. C177 DB 00 ;00000000 ........ C178 DB 3C ;00111100 ..****.. &4F 79 - 'O' C179 DB 66 ;01100110 .**..**. C17A DB 66 ;01100110 .**..**. C17B DB 66 ;01100110 .**..**. C17C DB 66 ;01100110 .**..**. C17D DB 66 ;01100110 .**..**. C17E DB 3C ;00111100 ..****.. C17F DB 00 ;00000000 ........ C180 DB 7C ;01111100 .*****.. &50 80 - 'P' C181 DB 66 ;01100110 .**..**. C182 DB 66 ;01100110 .**..**. C183 DB 7C ;01111100 .*****.. C184 DB 60 ;01100000 .**..... C185 DB 60 ;01100000 .**..... C186 DB 60 ;01100000 .**..... C187 DB 00 ;00000000 ........ C188 DB 3C ;00111100 ..****.. &51 81 - 'Q' C189 DB 66 ;01100110 .**..**. C18A DB 66 ;01100110 .**..**. C18B DB 66 ;01100110 .**..**. C18C DB 6A ;01101010 .**.*.*. C18D DB 6C ;01101100 .**.**.. C18E DB 36 ;00110110 ..**.**. C18F DB 00 ;00000000 ........ C190 DB 7C ;01111100 .*****.. &52 82 - 'R' C191 DB 66 ;01100110 .**..**. C192 DB 66 ;01100110 .**..**. C193 DB 7C ;01111100 .*****.. C194 DB 6C ;01101100 .**.**.. C195 DB 66 ;01100110 .**..**. C196 DB 66 ;01100110 .**..**. C197 DB 00 ;00000000 ........ C198 DB 3C ;00111100 ..****.. &53 83 - 'S' C199 DB 66 ;01100110 .**..**. C19A DB 60 ;01100000 .**..... C19B DB 3C ;00111100 ..****.. C19C DB 06 ;00000110 .....**. C19D DB 66 ;01100110 .**..**. C19E DB 3C ;00111100 ..****.. C19F DB 00 ;00000000 ........ C1A0 DB 7E ;01111110 .******. &54 84 - 'T' C1A1 DB 18 ;00011000 ...**... C1A2 DB 18 ;00011000 ...**... C1A3 DB 18 ;00011000 ...**... C1A4 DB 18 ;00011000 ...**... C1A5 DB 18 ;00011000 ...**... C1A6 DB 18 ;00011000 ...**... C1A7 DB 00 ;00000000 ........ C1A8 DB 66 ;01100110 .**..**. &55 85 - 'U' C1A9 DB 66 ;01100110 .**..**. C1AA DB 66 ;01100110 .**..**. C1AB DB 66 ;01100110 .**..**. C1AC DB 66 ;01100110 .**..**. C1AD DB 66 ;01100110 .**..**. C1AE DB 3C ;00111100 ..****.. C1AF DB 00 ;00000000 ........ C1B0 DB 66 ;01100110 .**..**. &56 86 - 'V' C1B1 DB 66 ;01100110 .**..**. C1B2 DB 66 ;01100110 .**..**. C1B3 DB 66 ;01100110 .**..**. C1B4 DB 66 ;01100110 .**..**. C1B5 DB 3C ;00111100 ..****.. C1B6 DB 18 ;00011000 ...**... C1B7 DB 00 ;00000000 ........ C1B8 DB 63 ;01100011 .**...** &57 87 - 'W' C1B9 DB 63 ;01100011 .**...** C1BA DB 6B ;01101011 .**.*.** C1BB DB 6B ;01101011 .**.*.** C1BC DB 7F ;01111111 .******* C1BD DB 77 ;01110111 .***.*** C1BE DB 63 ;01100011 .**...** C1BF DB 00 ;00000000 ........ C1C0 DB 66 ;01100110 .**..**. &58 88 - 'X' C1C1 DB 66 ;01100110 .**..**. C1C2 DB 3C ;00111100 ..****.. C1C3 DB 18 ;00011000 ...**... C1C4 DB 3C ;00111100 ..****.. C1C5 DB 66 ;01100110 .**..**. C1C6 DB 66 ;01100110 .**..**. C1C7 DB 00 ;00000000 ........ C1C8 DB 66 ;01100110 .**..**. &59 89 - 'Y' C1C9 DB 66 ;01100110 .**..**. C1CA DB 66 ;01100110 .**..**. C1CB DB 3C ;00111100 ..****.. C1CC DB 18 ;00011000 ...**... C1CD DB 18 ;00011000 ...**... C1CE DB 18 ;00011000 ...**... C1CF DB 00 ;00000000 ........ C1D0 DB 7E ;01111110 .******. &5A 90 - 'Z' C1D1 DB 06 ;00000110 .....**. C1D2 DB 0C ;00001100 ....**.. C1D3 DB 18 ;00011000 ...**... C1D4 DB 30 ;00110000 ..**.... C1D5 DB 60 ;01100000 .**..... C1D6 DB 7E ;01111110 .******. C1D7 DB 00 ;00000000 ........ C1D8 DB 7C ;01111100 .*****.. &5B 91 - '[' C1D9 DB 60 ;01100000 .**..... C1DA DB 60 ;01100000 .**..... C1DB DB 60 ;01100000 .**..... C1DC DB 60 ;01100000 .**..... C1DD DB 60 ;01100000 .**..... C1DE DB 7C ;01111100 .*****.. C1DF DB 00 ;00000000 ........ C1E0 DB 00 ;00000000 ........ &5C 92 - '\' C1E1 DB 60 ;01100000 .**..... C1E2 DB 30 ;00110000 ..**.... C1E3 DB 18 ;00011000 ...**... C1E4 DB 0C ;00001100 ....**.. C1E5 DB 06 ;00000110 .....**. C1E6 DB 00 ;00000000 ........ C1E7 DB 00 ;00000000 ........ C1E8 DB 3E ;00111110 ..*****. &5D 93 - ']' C1E9 DB 06 ;00000110 .....**. C1EA DB 06 ;00000110 .....**. C1EB DB 06 ;00000110 .....**. C1EC DB 06 ;00000110 .....**. C1ED DB 06 ;00000110 .....**. C1EE DB 3E ;00111110 ..*****. C1EF DB 00 ;00000000 ........ C1F0 DB 18 ;00011000 ...**... &5E 94 - '^' C1F1 DB 3C ;00111100 ..****.. C1F2 DB 66 ;01100110 .**..**. C1F3 DB 42 ;01000010 .*....*. C1F4 DB 00 ;00000000 ........ C1F5 DB 00 ;00000000 ........ C1F6 DB 00 ;00000000 ........ C1F7 DB 00 ;00000000 ........ C1F8 DB 00 ;00000000 ........ &5F 95 - '_' C1F9 DB 00 ;00000000 ........ C1FA DB 00 ;00000000 ........ C1FB DB 00 ;00000000 ........ C1FC DB 00 ;00000000 ........ C1FD DB 00 ;00000000 ........ C1FE DB 00 ;00000000 ........ C1FF DB FF ;11111111 ******** C200 DB 1C ;00011100 ...***.. &60 96 - '`' C201 DB 36 ;00110110 ..**.**. C202 DB 30 ;00110000 ..**.... C203 DB 7C ;01111100 .*****.. C204 DB 30 ;00110000 ..**.... C205 DB 30 ;00110000 ..**.... C206 DB 7E ;01111110 .******. C207 DB 00 ;00000000 ........ C208 DB 00 ;00000000 ........ &61 97 - 'a' C209 DB 00 ;00000000 ........ C20A DB 3C ;00111100 ..****.. C20B DB 06 ;00000110 .....**. C20C DB 3E ;00111110 ..*****. C20D DB 66 ;01100110 .**..**. C20E DB 3E ;00111110 ..*****. C20F DB 00 ;00000000 ........ C210 DB 60 ;01100000 .**..... &62 98 - 'b' C211 DB 60 ;01100000 .**..... C212 DB 7C ;01111100 .*****.. C213 DB 66 ;01100110 .**..**. C214 DB 66 ;01100110 .**..**. C215 DB 66 ;01100110 .**..**. C216 DB 7C ;01111100 .*****.. C217 DB 00 ;00000000 ........ C218 DB 00 ;00000000 ........ &63 99 - 'c' C219 DB 00 ;00000000 ........ C21A DB 3C ;00111100 ..****.. C21B DB 66 ;01100110 .**..**. C21C DB 60 ;01100000 .**..... C21D DB 66 ;01100110 .**..**. C21E DB 3C ;00111100 ..****.. C21F DB 00 ;00000000 ........ C220 DB 06 ;00000110 .....**. &64 100 - 'd' C221 DB 06 ;00000110 .....**. C222 DB 3E ;00111110 ..*****. C223 DB 66 ;01100110 .**..**. C224 DB 66 ;01100110 .**..**. C225 DB 66 ;01100110 .**..**. C226 DB 3E ;00111110 ..*****. C227 DB 00 ;00000000 ........ C228 DB 00 ;00000000 ........ &65 101 - 'e' C229 DB 00 ;00000000 ........ C22A DB 3C ;00111100 ..****.. C22B DB 66 ;01100110 .**..**. C22C DB 7E ;01111110 .******. C22D DB 60 ;01100000 .**..... C22E DB 3C ;00111100 ..****.. C22F DB 00 ;00000000 ........ C230 DB 1C ;00011100 ...***.. &66 102 - 'f' C231 DB 30 ;00110000 ..**.... C232 DB 30 ;00110000 ..**.... C233 DB 7C ;01111100 .*****.. C234 DB 30 ;00110000 ..**.... C235 DB 30 ;00110000 ..**.... C236 DB 30 ;00110000 ..**.... C237 DB 00 ;00000000 ........ C238 DB 00 ;00000000 ........ &67 103 - 'g' C239 DB 00 ;00000000 ........ C23A DB 3E ;00111110 ..*****. C23B DB 66 ;01100110 .**..**. C23C DB 66 ;01100110 .**..**. C23D DB 3E ;00111110 ..*****. C23E DB 06 ;00000110 .....**. C23F DB 3C ;00111100 ..****.. C240 DB 60 ;01100000 .**..... &68 104 - 'h' C241 DB 60 ;01100000 .**..... C242 DB 7C ;01111100 .*****.. C243 DB 66 ;01100110 .**..**. C244 DB 66 ;01100110 .**..**. C245 DB 66 ;01100110 .**..**. C246 DB 66 ;01100110 .**..**. C247 DB 00 ;00000000 ........ C248 DB 18 ;00011000 ...**... &69 105 - 'i' C249 DB 00 ;00000000 ........ C24A DB 38 ;00111000 ..***... C24B DB 18 ;00011000 ...**... C24C DB 18 ;00011000 ...**... C24D DB 18 ;00011000 ...**... C24E DB 3C ;00111100 ..****.. C24F DB 00 ;00000000 ........ C250 DB 18 ;00011000 ...**... &6A 106 - 'j' C251 DB 00 ;00000000 ........ C252 DB 38 ;00111000 ..***... C253 DB 18 ;00011000 ...**... C254 DB 18 ;00011000 ...**... C255 DB 18 ;00011000 ...**... C256 DB 18 ;00011000 ...**... C257 DB 70 ;01110000 .***.... C258 DB 60 ;01100000 .**..... &6B 107 - 'k' C259 DB 60 ;01100000 .**..... C25A DB 66 ;01100110 .**..**. C25B DB 6C ;01101100 .**.**.. C25C DB 78 ;01111000 .****... C25D DB 6C ;01101100 .**.**.. C25E DB 66 ;01100110 .**..**. C25F DB 00 ;00000000 ........ C260 DB 38 ;00111000 ..***... &6C 108 - 'l' C261 DB 18 ;00011000 ...**... C262 DB 18 ;00011000 ...**... C263 DB 18 ;00011000 ...**... C264 DB 18 ;00011000 ...**... C265 DB 18 ;00011000 ...**... C266 DB 3C ;00111100 ..****.. C267 DB 00 ;00000000 ........ C268 DB 00 ;00000000 ........ &6D 109 - 'm' C269 DB 00 ;00000000 ........ C26A DB 36 ;00110110 ..**.**. C26B DB 7F ;01111111 .******* C26C DB 6B ;01101011 .**.*.** C26D DB 6B ;01101011 .**.*.** C26E DB 63 ;01100011 .**...** C26F DB 00 ;00000000 ........ C270 DB 00 ;00000000 ........ &6E 110 - 'n' C271 DB 00 ;00000000 ........ C272 DB 7C ;01111100 .*****.. C273 DB 66 ;01100110 .**..**. C274 DB 66 ;01100110 .**..**. C275 DB 66 ;01100110 .**..**. C276 DB 66 ;01100110 .**..**. C277 DB 00 ;00000000 ........ C278 DB 00 ;00000000 ........ &6F 111 - 'o' C279 DB 00 ;00000000 ........ C27A DB 3C ;00111100 ..****.. C27B DB 66 ;01100110 .**..**. C27C DB 66 ;01100110 .**..**. C27D DB 66 ;01100110 .**..**. C27E DB 3C ;00111100 ..****.. C27F DB 00 ;00000000 ........ C280 DB 00 ;00000000 ........ &70 112 - 'p' C281 DB 00 ;00000000 ........ C282 DB 7C ;01111100 .*****.. C283 DB 66 ;01100110 .**..**. C284 DB 66 ;01100110 .**..**. C285 DB 7C ;01111100 .*****.. C286 DB 60 ;01100000 .**..... C287 DB 60 ;01100000 .**..... C288 DB 00 ;00000000 ........ &71 113 - 'q' C289 DB 00 ;00000000 ........ C28A DB 3E ;00111110 ..*****. C28B DB 66 ;01100110 .**..**. C28C DB 66 ;01100110 .**..**. C28D DB 3E ;00111110 ..*****. C28E DB 06 ;00000110 .....**. C28F DB 07 ;00000111 .....*** C290 DB 00 ;00000000 ........ &72 114 - 'r' C291 DB 00 ;00000000 ........ C292 DB 6C ;01101100 .**.**.. C293 DB 76 ;01110110 .***.**. C294 DB 60 ;01100000 .**..... C295 DB 60 ;01100000 .**..... C296 DB 60 ;01100000 .**..... C297 DB 00 ;00000000 ........ C298 DB 00 ;00000000 ........ &73 115 - 's' C299 DB 00 ;00000000 ........ C29A DB 3E ;00111110 ..*****. C29B DB 60 ;01100000 .**..... C29C DB 3C ;00111100 ..****.. C29D DB 06 ;00000110 .....**. C29E DB 7C ;01111100 .*****.. C29F DB 00 ;00000000 ........ C2A0 DB 30 ;00110000 ..**.... &74 116 - 't' C2A1 DB 30 ;00110000 ..**.... C2A2 DB 7C ;01111100 .*****.. C2A3 DB 30 ;00110000 ..**.... C2A4 DB 30 ;00110000 ..**.... C2A5 DB 30 ;00110000 ..**.... C2A6 DB 1C ;00011100 ...***.. C2A7 DB 00 ;00000000 ........ C2A8 DB 00 ;00000000 ........ &75 117 - 'u' C2A9 DB 00 ;00000000 ........ C2AA DB 66 ;01100110 .**..**. C2AB DB 66 ;01100110 .**..**. C2AC DB 66 ;01100110 .**..**. C2AD DB 66 ;01100110 .**..**. C2AE DB 3E ;00111110 ..*****. C2AF DB 00 ;00000000 ........ C2B0 DB 00 ;00000000 ........ &76 118 - 'v' C2B1 DB 00 ;00000000 ........ C2B2 DB 66 ;01100110 .**..**. C2B3 DB 66 ;01100110 .**..**. C2B4 DB 66 ;01100110 .**..**. C2B5 DB 3C ;00111100 ..****.. C2B6 DB 18 ;00011000 ...**... C2B7 DB 00 ;00000000 ........ C2B8 DB 00 ;00000000 ........ &77 119 - 'w' C2B9 DB 00 ;00000000 ........ C2BA DB 63 ;01100011 .**...** C2BB DB 6B ;01101011 .**.*.** C2BC DB 6B ;01101011 .**.*.** C2BD DB 7F ;01111111 .******* C2BE DB 36 ;00110110 ..**.**. C2BF DB 00 ;00000000 ........ C2C0 DB 00 ;00000000 ........ &78 120 - 'x' C2C1 DB 00 ;00000000 ........ C2C2 DB 66 ;01100110 .**..**. C2C3 DB 3C ;00111100 ..****.. C2C4 DB 18 ;00011000 ...**... C2C5 DB 3C ;00111100 ..****.. C2C6 DB 66 ;01100110 .**..**. C2C7 DB 00 ;00000000 ........ C2C8 DB 00 ;00000000 ........ &79 121 - 'y' C2C9 DB 00 ;00000000 ........ C2CA DB 66 ;01100110 .**..**. C2CB DB 66 ;01100110 .**..**. C2CC DB 66 ;01100110 .**..**. C2CD DB 3E ;00111110 ..*****. C2CE DB 06 ;00000110 .....**. C2CF DB 3C ;00111100 ..****.. C2D0 DB 00 ;00000000 ........ &7A 122 - 'z' C2D1 DB 00 ;00000000 ........ C2D2 DB 7E ;01111110 .******. C2D3 DB 0C ;00001100 ....**.. C2D4 DB 18 ;00011000 ...**... C2D5 DB 30 ;00110000 ..**.... C2D6 DB 7E ;01111110 .******. C2D7 DB 00 ;00000000 ........ C2D8 DB 0C ;00001100 ....**.. &7B 123 - '{' C2D9 DB 18 ;00011000 ...**... C2DA DB 18 ;00011000 ...**... C2DB DB 70 ;01110000 .***.... C2DC DB 18 ;00011000 ...**... C2DD DB 18 ;00011000 ...**... C2DE DB 0C ;00001100 ....**.. C2DF DB 00 ;00000000 ........ C2E0 DB 18 ;00011000 ...**... &7C 124 - '|' C2E1 DB 18 ;00011000 ...**... C2E2 DB 18 ;00011000 ...**... C2E3 DB 00 ;00000000 ........ C2E4 DB 18 ;00011000 ...**... C2E5 DB 18 ;00011000 ...**... C2E6 DB 18 ;00011000 ...**... C2E7 DB 00 ;00000000 ........ C2E8 DB 30 ;00110000 ..**.... &7D 125 - '}' C2E9 DB 18 ;00011000 ...**... C2EA DB 18 ;00011000 ...**... C2EB DB 0E ;00001110 ....***. C2EC DB 18 ;00011000 ...**... C2ED DB 18 ;00011000 ...**... C2EE DB 30 ;00110000 ..**.... C2EF DB 00 ;00000000 ........ C2F0 DB 31 ;00110001 ..**...* &7E 126 - '~' C2F1 DB 6B ;01101011 .**.*.** C2F2 DB 46 ;01000110 .*...**. C2F3 DB 00 ;00000000 ........ C2F4 DB 00 ;00000000 ........ C2F5 DB 00 ;00000000 ........ C2F6 DB 00 ;00000000 ........ C2F7 DB 00 ;00000000 ........ C2F8 DB FF ;11111111 ******** &7F 127 - DEL C2F9 DB FF ;11111111 ******** C2FA DB FF ;11111111 ******** C2FB DB FF ;11111111 ******** C2FC DB FF ;11111111 ******** C2FD DB FF ;11111111 ******** C2FE DB FF ;11111111 ******** C2FF DB FF ;11111111 ******** BBC Operation System OS 1.20 Startup Strings and Tables VDUORG CODE VDINIT C300 JMP &CB1D ;Initialise screen with mode in A. MSGBBC MSG C303 DB 13,'BBC Computer ',0 MSG16K C312 DB '16K',7,0 MSG32K C317 DB '32K',7,0 MSG2CR C31C DB 08,0D,0D ;Termination byte in next table ****** 16 COLOUR MODE BYTE MASK LOOK UP TABLE****** BPFTB C31F DB 00 ;00000000 C320 DB 11 ;00010001 C321 DB 22 ;00100010 C322 DB 33 ;00110011 C323 DB 44 ;01000100 C324 DB 55 ;01010101 C325 DB 66 ;01100110 C326 DB 77 ;01110111 C327 DB 88 ;10001000 C328 DB 99 ;10011001 C329 DB AA ;10101010 C32A DB BB ;10111011 C32B DB CC ;11001100 C32C DB DD ;11011101 C32D DB EE ;11101110 C32E DB FF ;11111111 ****** 4 COLOUR MODE BYTE MASK LOOK UP TABLE****** BPSTB C32F DB 00 ;00000000 C330 DB 55 ;01010101 C331 DB AA ;10101010 C332 DB FF ;11111111 ****** VDU ENTRY POINT LO LOOK UP TABLE****** JTBLO C333 DB 11 ;00010001 VDU 0 C334 DB 3B ;00111011 VDU 1 C335 DB 96 ;10010110 VDU 2 C336 DB A1 ;10100001 VDU 3 C337 DB AD ;10101101 VDU 4 C338 DB B9 ;10111001 VDU 5 C339 DB 11 ;00010001 VDU 6 C33A DB 6F ;01101111 VDU 7 C33B DB C5 ;11000101 VDU 8 C33C DB 64 ;01100100 VDU 9 C33D DB F0 ;11110000 VDU 10 C33E DB 5B ;01011011 VDU 11 C33F DB 59 ;01011001 VDU 12 C340 DB AF ;10101111 VDU 13 C341 DB 8D ;10001101 VDU 14 C342 DB A6 ;10100110 VDU 15 C343 DB C0 ;11000000 VDU 16 C344 DB F9 ;11111001 VDU 17 C345 DB FD ;11111101 VDU 18 C346 DB 92 ;10010010 VDU 19 C347 DB 39 ;00111001 VDU 20 C348 DB 9B ;10011011 VDU 21 C349 DB EB ;11101011 VDU 22 C34A DB F1 ;11110001 VDU 23 C34B DB 39 ;00111001 VDU 24 C34C DB 8C ;10001100 VDU 25 C34D DB BD ;10111101 VDU 26 C34E DB 11 ;00010001 VDU 27 C34F DB FA ;11111010 VDU 28 C350 DB A2 ;10100010 VDU 29 C351 DB 79 ;01111001 VDU 30 C352 DB 87 ;10000111 VDU 31 C353 DB AC ;10101100 VDU 127 ****** VDU ENTRY POINT HI PARAMETER LOOK UP TABLE****** ; 1xxxxxxx - no parameters, address high byte ; 0aaapppp - parameter count 16-p, address high byte &C3+a JTBHI C354 DB C5 ;11000101 VDU 0 - &C511, no parameters C355 DB 2F ;00101111 VDU 1 - &C53B, 1 parameter C356 DB C5 ;11000101 VDU 2 - &C596, no parameters C357 DB C5 ;11000101 VDU 3 - &C5A1, no parameters C358 DB C5 ;11000101 VDU 4 - &C5AD, no parameters C359 DB C5 ;11000101 VDU 5 - &C5B9, no parameters C35A DB C5 ;11000101 VDU 6 - &C511, no parameters C35B DB E8 ;11101000 VDU 7 - &E86F, no parameters C35C DB C5 ;11000101 VDU 8 - &C5C5, no parameters C35D DB C6 ;11000110 VDU 9 - &C664, no parameters C35E DB C6 ;11000110 VDU 10 - &C6F0, no parameters C35F DB C6 ;11000110 VDU 11 - &C65B, no parameters C360 DB C7 ;11000111 VDU 12 - &C759, no parameters C361 DB C7 ;11000111 VDU 13 - &C7AF, no parameters C362 DB C5 ;11000101 VDU 14 - &C58D, no parameters C363 DB C5 ;11000101 VDU 15 - &C5A6, no parameters C364 DB C7 ;11000111 VDU 16 - &C7C0, no parameters C365 DB 4F ;01001111 VDU 17 - &C7F9, 1 parameter C366 DB 4E ;01001110 VDU 18 - &C7FD, 2 parameters C367 DB 5B ;01011011 VDU 19 - &C892, 5 parameters C368 DB C8 ;11001000 VDU 20 - &C839, no parameters C369 DB C5 ;11000101 VDU 21 - &C59B, no parameters C36A DB 5F ;01011111 VDU 22 - &C8EB, 1 parameter C36B DB 57 ;01010111 VDU 23 - &C8F1, 9 parameters C36C DB 78 ;01111000 VDU 24 - &CA39, 8 parameters C36D DB 6B ;01101011 VDU 25 - &C9AC, 5 parameters C36E DB C9 ;11001001 VDU 26 - &C9BD, no parameters C36F DB C5 ;11000101 VDU 27 - &C511, no parameters C370 DB 3C ;00111100 VDU 28 - &C6FA, 4 parameters C371 DB 7C ;01111100 VDU 29 - &CAA2, 4 parameters C372 DB C7 ;11000111 VDU 30 - &C779, no parameters C373 DB 4E ;01001110 VDU 31 - &C787, 2 parameters C374 DB CA ;11001010 VDU 127 - &CAAC, no parameters ****** 640 MULTIPLICATION TABLE 40COL, 80COL MODES HIBYTE, LOBYTE ****** HRGTB C375 DW 0000 ; 0*640 = &0000 C377 DW 8002 ; 1*640 = &0280 C379 DW 0005 ; 2*640 = &0500 C37B DW 8007 ; 3*640 = &0780 C37D DW 000A ; 4*640 = &0A00 C37F DW 800C ; 5*640 = &0C80 C381 DW 000F ; 6*640 = &0F00 C383 DW 8011 ; 7*640 = &1180 C385 DW 0014 ; 8*640 = &1400 C387 DW 8016 ; 9*640 = &1680 C389 DW 0019 ; 10*640 = &1900 C38B DW 801B ; 11*640 = &1B80 C38D DW 001E ; 12*640 = &1E00 C38F DW 8020 ; 13*640 = &2080 C391 DW 0023 ; 14*640 = &2300 C393 DW 8025 ; 15*640 = &2580 C395 DW 0028 ; 16*640 = &2800 C397 DW 802A ; 17*640 = &2A80 C399 DW 002D ; 18*640 = &2D00 C39B DW 802F ; 19*640 = &2F80 C39D DW 0032 ; 20*640 = &3200 C39F DW 8034 ; 21*640 = &3480 C3A1 DW 0037 ; 22*640 = &3700 C3A3 DW 8039 ; 23*640 = &3980 C3A5 DW 003C ; 24*640 = &3C00 C3A7 DW 803E ; 25*640 = &3E80 C3A9 DW 0041 ; 26*640 = &4100 C3AB DW 8043 ; 27*640 = &4380 C3AD DW 0046 ; 28*640 = &4600 C3AF DW 8048 ; 29*640 = &4880 C3B1 DW 004B ; 30*640 = &4B00 C3B3 DW 804D ; 31*640 = &4D80 ****** *40 MULTIPLICATION TABLE TELETEXT MODE HIBYTE, LOBYTE ****** TXTTB C3B5 DW 0000 ; 0*40 = &0000 C3B7 DW 2800 ; 1*40 = &0028 C3B9 DW 5000 ; 2*40 = &0050 C3BB DW 7800 ; 3*40 = &0078 C3BD DW A000 ; 4*40 = &00A0 C3BF DW C800 ; 5*40 = &00C8 C3C1 DW F000 ; 6*40 = &00F0 C3C3 DW 1801 ; 7*40 = &0118 C3C5 DW 4001 ; 8*40 = &0140 C3C7 DW 6801 ; 9*40 = &0168 C3C9 DW 9001 ; 10*40 = &0190 C3CB DW B801 ; 11*40 = &01B8 C3CD DW E001 ; 12*40 = &01E0 C3CF DW 0802 ; 13*40 = &0208 C3D1 DW 3002 ; 14*40 = &0230 C3D3 DW 5802 ; 15*40 = &0258 C3D5 DW 8002 ; 16*40 = &0280 C3D7 DW A802 ; 17*40 = &02A8 C3D9 DW D002 ; 18*40 = &02D0 C3DB DW F802 ; 19*40 = &02F8 C3DD DW 2003 ; 20*40 = &0320 C3DF DW 4803 ; 21*40 = &0348 C3E1 DW 7003 ; 22*40 = &0370 C3E3 DW 9803 ; 23*40 = &0398 C3E5 DW C003 ; 24*40 = &03C0 ****** TEXT WINDOW - MAXIMUM BOTTOM ROW LOOK UP TABLE ****** PROWS C3E7 DB 1F ; MODE 0 - 32 ROWS C3E8 DB 1F ; MODE 1 - 32 ROWS C3E9 DB 1F ; MODE 2 - 32 ROWS C3EA DB 18 ; MODE 3 - 25 ROWS C3EB DB 1F ; MODE 4 - 32 ROWS C3EC DB 1F ; MODE 5 - 32 ROWS C3ED DB 18 ; MODE 6 - 25 ROWS C3EE DB 18 ; MODE 7 - 25 ROWS ****** TEXT WINDOW - MAXIMUM RIGHT HAND COLUMN LOOK UP TABLE ****** PCOLS C3EF DB 4F ; MODE 0 - 80 COLUMNS C3F0 DB 27 ; MODE 1 - 40 COLUMNS C3F1 DB 13 ; MODE 2 - 20 COLUMNS C3F2 DB 4F ; MODE 3 - 80 COLUMNS C3F3 DB 27 ; MODE 4 - 40 COLUMNS C3F4 DB 13 ; MODE 5 - 20 COLUMNS C3F5 DB 27 ; MODE 6 - 40 COLUMNS C3F6 DB 27 ; MODE 7 - 40 COLUMNS ************************************************************************* * * * SEVERAL OF THE FOLLOWING TABLES OVERLAP EACH OTHER * * SOME ARE DUAL PURPOSE * * * ************************************************************************* ************** VIDEO ULA CONTROL REGISTER SETTINGS *********************** INDMOD C3F7 DB 9C ; MODE 0 - 10011100 C3F8 DB D8 ; MODE 1 - 11011000 C3F9 DB F4 ; MODE 2 - 11110100 C3FA DB 9C ; MODE 3 - 10011100 C3FB DB 88 ; MODE 4 - 10001000 C3FC DB C4 ; MODE 5 - 11000100 C3FD DB 88 ; MODE 6 - 10001000 C3FE DB 4B ; MODE 7 - 01001011 ******** NUMBER OF BYTES PER CHARACTER FOR EACH DISPLAY MODE ************ BYTCHT C3FF DB 08 ; MODE 0 - 00001000 C400 DB 10 ; MODE 1 - 00010000 C401 DB 20 ; MODE 2 - 00100000 C402 DB 08 ; MODE 3 - 00001000 C403 DB 08 ; MODE 4 - 00001000 C404 DB 10 ; MODE 5 - 00010000 C405 DB 08 ; MODE 6 - 00001000 C406 DB 01 ; MODE 7 - 00000001 ******************* MASK TABLE FOR 2 COLOUR MODES ********************** MASKTB C407 DB AA ;10101010 C408 DB 55 ;01010101 ****************** MASK TABLE FOR 4 COLOUR MODES *********************** C409 DB 88 ;10001000 C40A DB 44 ;01000100 C40B DB 22 ;00100010 C40C DB 11 ;00010001 ********** MASK TABLE FOR 4 COLOUR MODES FONT FLAG MASK TABLE ********** BITTBL C40D DB 80 ;10000000 C40E DB 40 ;01000000 C40F DB 20 ;00100000 C410 DB 10 ;00010000 C411 DB 08 ;00001000 C412 DB 04 ;00000100 C413 DB 02 ;00000010 - NEXT BYTE IN FOLLOWING TABLE ********* NUMBER OF TEXT COLOURS -1 FOR EACH MODE ************************ COLMOD C414 DB 01 ; MODE 0 - 2 COLOURS C415 DB 03 ; MODE 1 - 4 COLOURS C416 DB 0F ; MODE 2 - 16 COLOURS C417 DB 01 ; MODE 3 - 2 COLOURS C418 DB 01 ; MODE 4 - 2 COLOURS C419 DB 03 ; MODE 5 - 4 COLOURS C41A DB 01 ; MODE 6 - 2 COLOURS ; Overlaps with GCOL table TBZGOE C41B DB 00 ; MODE 7 - 1 'COLOUR' GCOL 0 ************** GCOL PLOT OPTIONS PROCESSING LOOK UP TABLE *************** ; &D4 set from &D5 set from ; ORA EOR EOR TBZGOO C41C DB FF ;11111111 - GCOL 0 GCOL 1 TBZGEO C41D DB 00 ;00000000 - GCOL 1 GCOL 0 GCOL 2 C41E DB 00 ;00000000 - GCOL 2 GCOL 1 GCOL 3 C41F DB FF ;11111111 - GCOL 3 GCOL 2 GCOL 4 TBZGEE C420 DB FF ;11111111 - GCOL 4 GCOL 3 GCOL 5 GCOL 0 C421 DB FF ;11111111 - GCOL 5 GCOL 4 GCOL 1 C422 DB FF ;11111111 - GCOL 6 GCOL 5 GCOL 2 C423 DB 00 ;00000000 - GCOL 6 GCOL 3 ********** 2-COLOUR MODES TEXT MASK TABLE WITHIN TABLE ********** FCOLTB C424 DB 00 ;00000000 - COLOUR 0 GCOL 4 C425 DB FF ;11111111 - COLOUR 1 GCOL 5 ; &D4 set to &D5 set to ;GCOL 0 ORA &FF,EOR &00 ORA &00,EOR &FF -> SET point ;GCOL 1 ORA &00,EOR &00 ORA &FF,EOR &FF -> ORA with current point ;GCOL 2 ORA &00,EOR &FF ORA &00,EOR &FF -> AND with current point ;GCOL 3 ORA &FF,EOR &FF ORA &00,EOR &00 -> XOR with current point ;GCOL 4 ORA &FF,EOR &FF ORA &FF,EOR &00 -> Invert point ;GCOL 5 ORA &FF,EOR &FF ORA &FF,EOR &FF -> No change to current point ;GCOL 6 ORA &FF,EOR &00 ORA &FF,EOR &00 -> Clear current point ;GCOL 7 ORA &FF,EOR &00 ORA &FF,EOR &0F -> table overflow *************** 4-COLOUR MODES TEXT MASK TABLE ****************** C426 DB 00 ;00000000 - COLOUR 0 C427 DB 0F ;00001111 - COLOUR 1 C428 DB F0 ;11110000 - COLOUR 2 C429 DB FF ;11111111 - COLOUR 3 ***************16-COLOUR MODES TEXT MASK TABLE ****************** C42A DB 00 ;00000000 - COLOUR 0 C42B DB 03 ;00000011 - COLOUR 1 C42C DB 0C ;00001100 - COLOUR 2 C42D DB 0F ;00001111 - COLOUR 3 C42E DB 30 ;00110000 - COLOUR 4 C42F DB 33 ;00110011 - COLOUR 5 C430 DB 3C ;00111100 - COLOUR 6 C431 DB 3F ;00111111 - COLOUR 7 C432 DB C0 ;11000000 - COLOUR 8 C433 DB C3 ;11000011 - COLOUR 9 C434 DB CC ;11001100 - COLOUR 10 C435 DB CF ;11001111 - COLOUR 11 C436 DB F0 ;11110000 - COLOUR 12 C437 DB F3 ;11110011 - COLOUR 13 C438 DB FC ;11111100 - COLOUR 14 C439 DB FF ;11111111 - COLOUR 15 ********** DISPLAY MODE PIXELS/BYTE-1 TABLE ********************* NPIXTB C43A DB 07 ; MODE 0 - 8 PIXELS/BYTE C43B DB 03 ; MODE 1 - 4 PIXELS/BYTE C43C DB 01 ; MODE 2 - 2 PIXELS/BYTE SCHOFF C43D DB 00 ; MODE 3 - 1 PIXEL/BYTE (NON-GRAPHICS) C43E DB 07 ; MODE 4 - 8 PIXELS/BYTE C43F DB 03 ; MODE 5 - 4 PIXELS/BYTE ********* SCREEN DISPLAY MEMORY TYPE TABLE OVERLAPS ************ MODETB C440 DB 00 ; MODE 6 - 1 PIXEL/BYTE // MODE 0 - TYPE 0 ***** SOUND PITCH OFFSET BY CHANNEL TABLE WITHIN TABLE ********** SOUNDV C441 DB 00 ; MODE 7 - 1 PIXEL/BYTE // MODE 1 - TYPE 0 // CHANNEL 0 C442 DB 00 ; // MODE 2 - TYPE 0 // CHANNEL 1 C443 DB 01 ; // MODE 3 - TYPE 1 // CHANNEL 2 C444 DB 02 ; // MODE 4 - TYPE 2 // CHANNEL 3 **** REST OF DISPLAY MEMORY TYPE TABLE **** C445 DB 02 ; // MODE 5 - TYPE 2 C446 DB 03 ; // MODE 6 - TYPE 3 ***************** VDU SECTION CONTROL NUMBERS *************************** EWTBA C447 DB 04 ;00000100 // MODE 7 - TYPE 4 C448 DB 00 ;00000000 C449 DB 06 ;00000110 C44A DB 02 ;00000010 *********** CRTC SETUP PARAMETERS TABLE 1 WITHIN TABLE ****************** CONE C44B DB 0D ;00001101 C44C DB 05 ;00000101 C44D DB 0D ;00001101 C44E DB 05 ;00000101 *********** CRTC SETUP PARAMETERS TABLE 2 WITHIN TABLE ***************** CZERO C44F DB 04 ;00000100 C450 DB 04 ;00000100 C451 DB 0C ;00001100 C452 DB 0C ;00001100 C453 DB 04 ;00000100 **** VIDEO ULA CONTROL VALUES **** CLSCTA C454 DB 02 ; Type 0 - 00000010 C455 DB 32 ; Type 1 - 00110010 C456 DB 7A ; Type 2 - 01111010 C457 DB 92 ; Type 3 - 10010010 C458 DB E6 ; Type 4 - 11100110 ************** MSB OF MEMORY OCCUPIED BY SCREEN BUFFER ***************** BYTSTA C459 DB 50 ; Type 0, &5000 - 20K C45A DB 40 ; Type 1, &4000 - 16K C45B DB 28 ; Type 2, &2800 - 10K C45C DB 20 ; Type 3, &2000 - 8K C45D DB 04 ; Type 4, &0400 - 1K ************ MSB OF FIRST LOCATION OCCUPIED BY SCREEN BUFFER ************ STSCTB C45E DB 30 ; Type 0, &3000 C45F DB 40 ; Type 1, &4000 C460 DB 58 ; Type 2, &5800 C461 DB 60 ; Type 3, &6000 C462 DB 7C ; Type 4, &7C00 ***************** NUMBER OF BYTES PER ROW ******************************* BYTRTA C463 DB 28 ;00101000 C464 DB 40 ;01000000 C465 DB 80 ;10000000 ******** ROW MULTIPLIACTION TABLE POINTER TO LOOK UP TABLE ************** TBZTBA C466 DB B5 ;10110101 C467 DB 75 ;01110101 C468 DB 75 ;01110101 ********** CRTC CURSOR END REGISTER SETTING LOOK UP TABLE *************** CRTCX C469 DB 0B ; Map 0, 00001011 C46A DB 17 ; Map 1, 00010111 C46B DB 23 ; Map 2, 00100011 C46C DB 2F ; Map 3, 00101111 C46D DB 3B ; Map 4, 00111011 ************* 6845 REGISTERS 0-11 FOR SCREEN TYPE 0 - MODES 0-2 ********* CRTCTB C46E DB 7F ; 0 Horizontal Total =128 C46F DB 50 ; 1 Horizontal Displayed =80 C470 DB 62 ; 2 Horizontal Sync =&62 C471 DB 28 ; 3 HSync Width+VSync =&28 VSync=2, HSync Width=8 C472 DB 26 ; 4 Vertical Total =38 C473 DB 00 ; 5 Vertial Adjust =0 C474 DB 20 ; 6 Vertical Displayed =32 C475 DB 22 ; 7 VSync Position =&22 C476 DB 01 ; 8 Interlace+Cursor =&01 Cursor=0, Display=0, Interlace=Sync C477 DB 07 ; 9 Scan Lines/Character =8 C478 DB 67 ;10 Cursor Start Line =&67 Blink=On, Speed=1/32, Line=7 C479 DB 08 ;11 Cursor End Line =8 ************* 6845 REGISTERS 0-11 FOR SCREEN TYPE 1 - MODE 3 ************ C47A DB 7F ; 0 Horizontal Total =128 C47B DB 50 ; 1 Horizontal Displayed =80 C47C DB 62 ; 2 Horizontal Sync =&62 C47D DB 28 ; 3 HSync Width+VSync =&28 VSync=2, HSync=8 C47E DB 1E ; 4 Vertical Total =30 C47F DB 02 ; 5 Vertical Adjust =2 C480 DB 19 ; 6 Vertical Displayed =25 C481 DB 1B ; 7 VSync Position =&1B C482 DB 01 ; 8 Interlace+Cursor =&01 Cursor=0, Display=0, Interlace=Sync C483 DB 09 ; 9 Scan Lines/Character =10 C484 DB 67 ;10 Cursor Start Line =&67 Blink=On, Speed=1/32, Line=7 C485 DB 09 ;11 Cursor End Line =9 ************ 6845 REGISTERS 0-11 FOR SCREEN TYPE 2 - MODES 4-5 ********** C486 DB 3F ; 0 Horizontal Total =64 C487 DB 28 ; 1 Horizontal Displayed =40 C488 DB 31 ; 2 Horizontal Sync =&31 C489 DB 24 ; 3 HSync Width+VSync =&24 VSync=2, HSync=4 C48A DB 26 ; 4 Vertical Total =38 C48B DB 00 ; 5 Vertical Adjust =0 C48C DB 20 ; 6 Vertical Displayed =32 C48D DB 22 ; 7 VSync Position =&22 C48E DB 01 ; 8 Interlace+Cursor =&01 Cursor=0, Display=0, Interlace=Sync C48F DB 07 ; 9 Scan Lines/Character =8 C490 DB 67 ;10 Cursor Start Line =&67 Blink=On, Speed=1/32, Line=7 C491 DB 08 ;11 Cursor End Line =8 ********** 6845 REGISTERS 0-11 FOR SCREEN TYPE 3 - MODE 6 *************** C492 DB 3F ; 0 Horizontal Total =64 C493 DB 28 ; 1 Horizontal Displayed =40 C494 DB 31 ; 2 Horizontal Sync =&31 C495 DB 24 ; 3 HSync Width+VSync =&24 VSync=2, HSync=4 C496 DB 1E ; 4 Vertical Total =30 C497 DB 02 ; 5 Vertical Adjust =0 C498 DB 19 ; 6 Vertical Displayed =25 C499 DB 1B ; 7 VSync Position =&1B C49A DB 01 ; 8 Interlace+Cursor =&01 Cursor=0, Display=0, Interlace=Sync C49B DB 09 ; 9 Scan Lines/Character =10 C49C DB 67 ;10 Cursor Start Line =&67 Blink=On, Speed=1/32, Line=7 C49D DB 09 ;11 Cursor End Line =9 ********* 6845 REGISTERS 0-11 FOR SCREEN TYPE 4 - MODE 7 **************** C49E DB 3F ; 0 Horizontal Total =64 C49F DB 28 ; 1 Horizontal Displayed =40 C4A0 DB 33 ; 2 Horizontal Sync =&33 Note: &31 is a better value C4A1 DB 24 ; 3 HSync Width+VSync =&24 VSync=2, HSync=4 C4A2 DB 1E ; 4 Vertical Total =30 C4A3 DB 02 ; 5 Vertical Adjust =2 C4A4 DB 19 ; 6 Vertical Displayed =25 C4A5 DB 1B ; 7 VSync Position =&1B C4A6 DB 93 ; 8 Interlace+Cursor =&93 Cursor=2, Display=1, Interlace=Sync+Video C4A7 DB 12 ; 9 Scan Lines/Character =19 C4A8 DB 72 ;10 Cursor Start Line =&72 Blink=On, Speed=1/32, Line=18 C4A9 DB 13 ;11 Cursor End Line =19 ************* VDU ROUTINE VECTOR ADDRESSES ****************************** D1JTB C4AA DW &D386 C4AC DW &D37E ************ VDU ROUTINE BRANCH VECTOR ADDRESS LO *********************** D2JTBL C4AE DB 6A ;01101010 C4AF DB 74 ;01110100 C4B0 DB 42 ;01000010 C4B1 DB 4B ;01001011 ************ VDU ROUTINE BRANCH VECTOR ADDRESS HI *********************** D2JTBH C4B2 DB D3 ;11010011 C4B3 DB D3 ;11010011 C4B4 DB D3 ;11010011 C4B5 DB D3 ;11010011 *********** TELETEXT CHARACTER CONVERSION TABLE ************************ TTXTB C4B6 DB 23 ; '#' -> '_' C4B7 DB 5F ; '_' -> '`' C4B8 DB 60 ; '`' -> '#' C4B9 DB 23 ; '#' *********** SOFT CHARACTER RAM ALLOCATION ***************************** FFSZOT C4BA DB 04 ; &20-&3F - OSHWM+&0400 C4BB DB 05 ; &40-&5F - OSHWM+&0500 C4BC DB 06 ; &60-&7F - OSHWM+&0600 C4BD DB 00 ; &80-&9F - OSHWM+&0000 C4BE DB 01 ; &A0-&BF - OSHWM+&0100 C4BF DB 02 ; &C0-&DF - OSHWM+&0200 ************************************************************************* * * * VDU FUNCTION ADDRESSES * * * ************************************************************************* ; VDU Address Parameters function ; 0 &C511 0 does nothing ; 1 &C53B 1 next character to printer only ; 2 &C596 0 enable printer ; 3 &C5A1 0 disable printer ; 4 &C5AD 0 select text cursor ; 5 &C5B9 0 select graphics cursor ; 6 &C511 0 enable display ; 7 &E86F 0 bell ; 8 &C5C5 0 cursor left ; 9 &C664 0 cursor right ; 10 &C6F0 0 cursor down ; 11 &C65B 0 cursor up ; 12 &C759 0 clear text window ; 13 &C7AF 0 newline ; 14 &C58D 0 select paged mode ; 15 &C5A6 0 cancel paged mode ; 16 &C7C0 0 clear graphics screen ; 17 &C7F9 1 define text colour ; 18 &C7FD 2 define graphics colour ; 19 &C892 5 define logical colour ; 20 &C839 0 restore default colours ; 21 &C59B 0 disable display ; 22 &C8EB 1 select screen MODE ; 23 &C8F1 9 define character ; 24 &CA39 8 define graphics window ; 25 &C98C 5 PLOT ; 26 &C9BD 0 set default windows ; 27 &C511 0 ESCAPE (does nothing) ; 28 &C6FA 4 define text window ; 29 &CAA2 4 define graphics origin ; 30 &C779 0 home cursor ; 31 &C787 2 position text cursor (TAB) ;127 &CAAC 0 delete ************************************************************************* * * * VDU Variables * * * ************************************************************************* ;D0 VDU status ;Bit 0 printer output enabled ; 1 scrolling disabled ; 2 paged scrolling enabled ; 3 software scrolling selected ; 4 not used ; 5 printing at graphics cursor enabled ; 6 cursor editing mode enabled ; 7 screen disabled ;D1 byte mask for current graphics point ;D2/3 text colour bytes to be ORed and EORed into memory ;D4/5 graphics colour bytes to be ORed and EORed into memory ;D6/7 address of top line of current graphics cell ;D8/9 address of top scan line of current text character ;DA/F temporary workspace ;E0/1 CRTC row multiplication table pointer ;246 Character definition explosion switch ;248 current video ULA control regiter setting ;249 current pallette setting ;251 flash counter ;252 mark-space count ;253 space period count ;256 EXEC file handle ;257 SPOOL file handle ;260 Econet OSWRCH interception flag ;269 print line counter ;26A number of items in VDU queque ;27D cursor editing status ;28F start up options (Keyboard links) bits 0-2 default screen Mode 3 reverse SHIFT/BREAK 4-5 disc timing parameters ;290 screen display vertical adjustment ;291 interlace toggle flag ;300/1 graphics window left (internal) ;302/3 graphics window bottom (internal) ;304/5 graphics window right (internal) ;306/7 graphics window top (internal) ;308 text window left ;309 text window bottom ;30A text window right ;30B text window top ;30C/D graphics origin, horizontal (external values) ;30E/F graphics origin, vertical (external values) ;310/1 current graphics cursor, horizontal (external values) ;312/3 current graphics cursor, vertical (external values) ;314/5 last graphics cursor, horizontal (external values) ;316/7 last graphics cursor, vertical (external values) ;318 text column (WINLEFT+POS) ;319 text line (WINRIGHT+VPOS) ;31A graphics scan line expressed as line of character ;31B-323 VDU parameters, last parameter in &323 ;324/5 current graphics cursor, horizontal (internal values) ;316/7 current graphics cursor, vertical (internal values) ;328-349 general workspace ;34A/B text cursor address to CRT controller ;34C/D width of text window in bytes ;34E hi byte of address of screen RAM start ;34F bytes per character ;350/1 address of window area start ;352/3 bytes per character row ;354 high byte of screen RAM size ;355 Mode ;356 memory map type ;357/35A current colours ;35B/C graphics plot mode ;35D/E jump vector ;35F last setting of CRTC Cursor start register ;360 number of logical colours less 1 ;361 pixels per byte (0 in text only modes) ;362/3 colour masks ;364/5 X/Y for text input cursor ;366 output cursor character for MODE 7 ;367 font flag ;368/E font location bytes ;36F-37E Colour palette BBC Operation System OS 1.20 VDU Main Routines ************************************************************************** ************************************************************************** ** ** ** OSWRCH MAIN ROUTINE entry from E0C5 ** ** ** ** output a byte via the VDU stream ** ** ** ************************************************************************** ************************************************************************** ;This routine takes up over 40% of the operating system ROM ;Entry points are variable, as are the results achieved. ;Tracing any particular path is relatively easy but generalising for ;commenting is not. For clarity comments will not be as detailed as ;for later parts of the Operating System. VDU C4C0 LDX &026A ;get number of items in VDU queue C4C3 BNE &C512 ;if parameters needed then C512 C4C5 BIT &D0 ;else check status byte C4C7 BVC &C4D8 ;if cursor editing enabled two cursors exist C4C9 JSR &C568 ;swap values C4CC JSR &CD6A ;then set up write cursor C4CF BMI &C4D8 ;if display disabled C4D8 C4D1 CMP #&0D ;else if character in A=RETURN teminate edit C4D3 BNE &C4D8 ;else C4D8 C4D5 JSR &D918 ;terminate edit NOTCEA C4D8 CMP #&7F ;is character DELETE ? C4DA BEQ &C4ED ;if so C4ED C4DC CMP #&20 ;is it less than space? (i.e. VDU control code) C4DE BCC &C4EF ;if so C4EF C4E0 BIT &D0 ;else check VDU byte ahain C4E2 BMI &C4EA ;if screen disabled C4EA C4E4 JSR &CFB7 ;else display a character C4E7 JSR &C664 ;and cursor right NOTC20 C4EA JMP &C55E ; ********* read link addresses and number of parameters ***************** ISDEL C4ED LDA #&20 ;to replace delete character ********* read link addresses and number of parameters ***************** NOTDEL C4EF TAY ;Y=A C4F0 LDA &C333,Y ;get lo byte of link address C4F3 STA &035D ;store it in jump vector C4F6 LDA &C354,Y ;get hi byte C4F9 BMI &C545 ;if negative (as it will be if a direct address) ;there are no parameters needed ;so C545 C4FB TAX ;else X=A C4FC ORA #&F0 ;set up negated parameter count C4FE STA &026A ;store it as number of items in VDU queue C501 TXA ;get back A C502 LSR ;A=A/16 C503 LSR ; C504 LSR ; C505 LSR ; C506 CLC ;clear carry C507 ADC #&C3 ;add &C3 to get hi byte of link address C509 STA &035E ; C50C BIT &D0 ;check if cursor editing enabled C50E BVS &C52F ;if so re-exchange pointers C510 CLC ;clear carry ACK C511 RTS ;and exit ;return with carry clear indicates that printer action not required. ; ********** parameters are outstanding *********************************** X=&26A = 2 complement of number of parameters X=&FF for 1, FE for 2 etc. QING C512 STA &0224,X ;store parameter in queue C515 INX ;increment X C516 STX &026A ;store it as VDU queue C519 BNE &C532 ;if not 0 C532 as more parameters are needed C51B BIT &D0 ;get VDU status byte C51D BMI &C534 ;if screen disabled C534 C51F BVS &C526 ;else if cursor editing C526 C521 JSR &CCF5 ;execute required function C524 CLC ;clear carry C525 RTS ;and exit ; NOTCEB C526 JSR &C568 ;swap values of cursors C529 JSR &CD6A ;set up write cursor C52C JSR &CCF5 ;execute required function NOTC10 C52F JSR &C565 ;re-exchange pointers CLCEX C532 CLC ;carry clear C533 RTS ;exit ************************************************************************* * * * VDU 1 - SEND NEXT CHARACTER TO PRINTER * * * * 1 parameter required * * * ************************************************************************* ; DISABL C534 LDY &035E ;if upper byte of link address not &C5 C537 CPY #&C5 ;printer is not interested C539 BNE &C532 ;so C532 SOH C53B TAX ;else X=A C53C LDA &D0 ;A=VDU status byte C53E LSR ;get bit 0 into carry C53F BCC &C511 ;if printer not enabled exit C541 TXA ;restore A C542 JMP &E11E ;else send byte in A (next byte) to printer *********** if explicit link address found, no parameters *************** REXEC C545 STA &035E ;upper byte of link address C548 TYA ;restore A C549 CMP #&08 ;is it 7 or less? C54B BCC &C553 ;if so C553 C54D EOR #&FF ;invert it C54F CMP #&F2 ;c is set if A >&0D C551 EOR #&FF ;re invert REXECZ C553 BIT &D0 ;VDU status byte C555 BMI &C580 ;if display disabled C580 C557 PHP ;push processor flags C558 JSR &CCF5 ;execute required function C55B PLP ;get back flags C55C BCC &C561 ;if carry clear (from C54B/F) **************** main exit routine ************************************** CSWDEX C55E LDA &D0 ;VDU status byte C560 LSR ;Carry is set if printer is enabled REXECB C561 BIT &D0 ;VDU status byte C563 BVC &C511 ;if no cursor editing C511 to exit ***************** cursor editing routines ******************************* REXE10 C565 JSR &CD7A ;restore normal write cursor CSWAP C568 PHP ;save flags and C569 PHA ;A C56A LDX #&18 ;X=&18 C56C LDY #&64 ;Y=&64 C56E JSR &CDDE ;exchange &300/1+X with &300/1+Y C571 JSR &CF06 ;set up display address C574 JSR &CA02 ;set cursor position C577 LDA &D0 ;VDU status byte C579 EOR #&02 ;invert bit 1 to allow or bar scrolling C57B STA &D0 ;VDU status byte C57D PLA ;restore flags and A C57E PLP ; C57F RTS ;and exit ; ACKQ C580 EOR #&06 ;if A<>6 C582 BNE &C58C ;return via C58C C584 LDA #&7F ;A=&7F C586 BCC &C5A8 ;and goto C5A8 ALWAYS!! ******************* check text cursor in use *************************** ANOMQ C588 LDA &D0 ;VDU status byte C58A AND #&20 ;set A from bit 5 of status byte VDURTS C58C RTS ;and exit A=0 if text cursor, &20 if graphics ************************************************************************* * * * VDU 14 - SET PAGED MODE * * * ************************************************************************* ; SO C58D LDY #&00 ;Y=0 C58F STY &0269 ;paged mode counter C592 LDA #&04 ;A=04 C594 BNE &C59D ;jump to C59D ************************************************************************* * * * VDU 2 - PRINTER ON (START PRINT JOB) * * * ************************************************************************* STX C596 JSR &E1A2 ;select printer buffer and output character C599 LDA #&94 ;A=&94 ;when inverted at C59B this becomes =&01 ************************************************************************* * * * VDU 21 - DISABLE DISPLAY * * * ************************************************************************* NAK C59B EOR #&95 ;if A=&15 A now =&80: if A=&94 A now =1 STTOR C59D ORA &D0 ;VDU status byte set bit 0 or bit 7 C59F BNE &C5AA ;branch forward to store ************************************************************************* * * * VDU 3 - PRINTER OFF (END PRINT JOB) * * * ************************************************************************* ETX C5A1 JSR &E1A2 ;select printer buffer and output character C5A4 LDA #&0A ;A=10 to clear status bits below... ************************************************************************* * * * VDU 15 - PAGED MODE OFF * * * ************************************************************************* ; A=&F or &A SI C5A6 EOR #&F4 ;convert to &FB or &FE STTAND C5A8 AND &D0 ;VDU status byte clear bit 0 or bit 2 of status STTEXT C5AA STA &D0 ;VDU status byte EOTEXT C5AC RTS ;exit ************************************************************************* * * * VDU 4 - OUTPUT AT TEXT CURSOR * * * ************************************************************************* ; EOT C5AD LDA &0361 ;pixels per byte C5B0 BEQ &C5AC ;if no graphics in current mode C5AC C5B2 JSR &C951 ;set CRT controller for text cursor C5B5 LDA #&DF ;this to clear bit 5 of status byte C5B7 BNE &C5A8 ;via C5A8 exit ************************************************************************* * * * VDU 5 - OUTPUT AT GRAPHICS CURSOR * * * ************************************************************************* ENQ C5B9 LDA &0361 ;pixels per byte C5BC BEQ &C5AC ;if none this is text mode so exit C5BE LDA #&20 ;set up graphics cursor C5C0 JSR &C954 ;via C954 C5C3 BNE &C59D ;set bit 5 via exit C59D ************************************************************************* * * * VDU 8 - CURSOR LEFT * * * ************************************************************************* BS C5C5 JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C5C8 BNE &C61F ;move cursor left 8 pixels if graphics CSL * C5CA DEC &0318 ;else decrement text column C5CD LDX &0318 ;store new text column C5D0 CPX &0308 ;if it is less than text window left C5D3 BMI &C5EE ;do wraparound cursor to rt of screen 1 line up C5D5 LDA &034A ;text cursor 6845 address C5D8 SEC ;subtract C5D9 SBC &034F ;bytes per character C5DC TAX ;put in X C5DD LDA &034B ;get text cursor 6845 address C5E0 SBC #&00 ;subtract 0 C5E2 CMP &034E ;compare with hi byte of screen RAM address C5E5 BCS &C5EA ;if = or greater C5E7 ADC &0354 ;add screen RAM size hi byte to wrap around CSLA C5EA TAY ;Y=A C5EB JMP &C9F6 ;Y hi and X lo byte of cursor position ***************** execute wraparound left-up***************************** CSLC C5EE LDA &030A ;text window right C5F1 STA &0318 ;text column *************** cursor up *********************************************** CSU C5F4 DEC &0269 ;paged mode counter C5F7 BPL &C5FC ;if still greater than 0 skip next instruction C5F9 INC &0269 ;paged mode counter to restore X=0 CSUC C5FC LDX &0319 ;current text line C5FF CPX &030B ;top of text window C602 BEQ &C60A ;if it's at top of window C60A C604 DEC &0319 ;else decrement current text line C607 JMP &C6AF ;and carry on moving cursor ******** cursor at top of window **************************************** CSUA C60A CLC ;clear carry C60B JSR &CD3F ;check for window violatations C60E LDA #&08 ;A=8 to check for software scrolling C610 BIT &D0 ;compare against VDU status byte C612 BNE &C619 ;if not enabled C619 C614 JSR &C994 ;set screen start register and adjust RAM C617 BNE &C61C ;jump C61C CSUB C619 JSR &CDA4 ;soft scroll 1 line CSUD C61C JMP &C6AC ;and exit **********cursor left and down with graphics cursor in use ************** CSAL C61F LDX #&00 ;X=0 to select horizontal parameters ********** cursor down with graphics in use ***************************** ;X=2 for vertical or 0 for horizontal GCSCHD C621 STX &DB ;store X C623 JSR &D10D ;check for window violations C626 LDX &DB ;restore X C628 SEC ;set carry C629 LDA &0324,X ;current graphics cursor X>1=vertical C62C SBC #&08 ;subtract 8 to move back 1 character C62E STA &0324,X ;store in current graphics cursor X>1=verticaal C631 BCS &C636 ;if carry set skip next C633 DEC &0325,X ;current graphics cursor hi -1 GCSHDA C636 LDA &DA ;&DA=0 if no violation else 1 if vert violation ;2 if horizontal violation C638 BNE &C658 ;if violation C658 C63A JSR &D10D ;check for window violations C63D BEQ &C658 ;if none C658 C63F LDX &DB ;else get back X C641 LDA &0304,X ;graphics window rt X=0 top X=2 C644 CPX #&01 ;is X=0 C646 BCS &C64A ;if not C64A C648 SBC #&06 ;else subtract 7 GCSHDC C64A STA &0324,X ;current graphics cursor X>1=vertical C64D LDA &0305,X ;graphics window hi rt X=0 top X=2 C650 SBC #&00 ;subtract carry C652 STA &0325,X ;current graphics cursor X<2=horizontal else vertical C655 TXA ;A=X C656 BEQ &C660 ;cursor up GCSHDB C658 JMP &D1B8 ;set up external coordinates for graphics ************************************************************************* * * * VDU 11 - CURSOR UP * * * ************************************************************************* VT C65B JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C65E BEQ &C5F4 ;if text cursor then C5F4 CSAU C660 LDX #&02 ;else X=2 C662 BNE &C6B6 ;goto C6B6 ************************************************************************* * * * VDU 9 - CURSOR RIGHT * * * ************************************************************************* HT C664 LDA &D0 ;VDU status byte C666 AND #&20 ;check bit 5 C668 BNE &C6B4 ;if set then graphics cursor in use so C6B4 CSR * C66A LDX &0318 ;text column C66D CPX &030A ;text window right C670 BCS &C684 ;if X exceeds window right then C684 C672 INC &0318 ;text column C675 LDA &034A ;text cursor 6845 address C678 ADC &034F ;add bytes per character C67B TAX ;X=A C67C LDA &034B ;text cursor 6845 address C67F ADC #&00 ;add carry if set C681 JMP &C9F6 ;use X and Y to set new cursor address ********: text cursor down and right ************************************* CSRC C684 LDA &0308 ;text window left C687 STA &0318 ;text column ********: text cursor down ************************************* CSD C68A CLC ;clear carry C68B JSR &CAE3 ;check bottom margin, X=line count C68E LDX &0319 ;current text line C691 CPX &0309 ;bottom margin C694 BCS &C69B ;if X=>current bottom margin C69B C696 INC &0319 ;else increment current text line C699 BCC &C6AF ; CSDD C69B JSR &CD3F ;check for window violations C69E LDA #&08 ;check bit 3 C6A0 BIT &D0 ;VDU status byte C6A2 BNE &C6A9 ;if software scrolling enabled C6A9 C6A4 JSR &C9A4 ;perform hardware scroll C6A7 BNE &C6AC ; CSDB C6A9 JSR &CDFF ;execute upward scroll CSDC C6AC JSR &CEAC ;clear a line CSDA C6AF JSR &CF06 ;set up display address C6B2 BCC &C732 ; *********** graphic cursor right **************************************** CSAR C6B4 LDX #&00 ; ************** graphic cursor up (X=2) ********************************** GCSCHU C6B6 STX &DB ;store X C6B8 JSR &D10D ;check for window violations C6BB LDX &DB ;get back X C6BD CLC ;clear carry C6BE LDA &0324,X ;current graphics cursor X>1=vertical C6C1 ADC #&08 ;Add 8 pixels C6C3 STA &0324,X ;current graphics cursor X>1=vertical C6C6 BCC &C6CB ; C6C8 INC &0325,X ;current graphics cursor X<2=horizontal else vertical GCSHUA C6CB LDA &DA ;A=0 no window violations 1 or 2 indicates violation C6CD BNE &C658 ;if outside window C658 C6CF JSR &D10D ;check for window violations C6D2 BEQ &C658 ;if no violations C658 C6D4 LDX &DB ;get back X C6D6 LDA &0300,X ;graphics window X<2 =left else bottom C6D9 CPX #&01 ;If X=0 C6DB BCC &C6DF ;C6DF C6DD ADC #&06 ;else add 7 GCSHUC C6DF STA &0324,X ;current graphics cursor X>1=vertical C6E2 LDA &0301,X ;graphics window hi X<2 =left else bottom C6E5 ADC #&00 ;add anny carry C6E7 STA &0325,X ;current graphics cursor X<2=horizontal else vertical C6EA TXA ;A=X C6EB BEQ &C6F5 ;if X=0 C6F5 cursor down C6ED JMP &D1B8 ;set up external coordinates for graphics ************************************************************************* * * * VDU 10 - CURSOR DOWN * * * ************************************************************************* LF C6F0 JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C6F3 BEQ &C68A ;if text cursor back to C68A CSAD C6F5 LDX #&02 ;else X=2 to indicate vertical movement C6F7 JMP &C621 ;move graphics cursor down ************************************************************************* * * * VDU 28 - DEFINE TEXT WINDOW * * * * 4 parameters * * * ************************************************************************* ;parameters are set up thus ;0320 P1 left margin ;0321 P2 bottom margin ;0322 P3 right margin ;0323 P4 top margin ;Note that last parameter is always in 0323 FS C6FA LDX &0355 ;screen mode C6FD LDA &0321 ;get bottom margin C700 CMP &0323 ;compare with top margin C703 BCC &C758 ;if bottom margin exceeds top return C705 CMP &C3E7,X ;text window bottom margin maximum C708 BEQ &C70C ;if equal then its OK C70A BCS &C758 ;else exit FSC C70C LDA &0322 ;get right margin C70F TAY ;put it in Y C710 CMP C3EF,X ;text window right hand margin maximum C713 BEQ &C717 ;if equal then OK C715 BCS &C758 ;if greater than maximum exit FSB C717 SEC ;set carry to subtract C718 SBC &0320 ;left margin C71B BMI &C758 ;if left greater than right exit C71D TAY ;else A=Y (window width) C71E JSR &CA88 ;calculate number of bytes in a line C721 LDA #&08 ;A=8 to set bit of &D0 C723 JSR &C59D ;indicating that text window is defined C726 LDX #&20 ;point to parameters C728 LDY #&08 ;point to text window margins C72A JSR &D48A ;(&300/3+Y)=(&300/3+X) C72D JSR &CEE8 ;set up screen address C730 BCS &C779 ;home cursor within window FSD C732 JMP &CA02 ;set cursor position ************************************************************************* * * * OSWORD 9 - READ A PIXEL * * =POINT(X,Y) * * * ************************************************************************* ;on entry &EF=A=9 ; &F0=X=low byte of parameter block address ; &F1=Y=high byte of parameter block address ; PARAMETER BLOCK ; 0,1=X coordinate ; 2,3=Y coordinate ;on exit, result in BLOCK+4 ; =&FF if point was off screen or logical colour of point if on screen ; RPXLXY C735 LDY #&03 ;Y=3 to point to hi byte of Y coordinate PRAAA C737 LDA (&F0),Y ;get it C739 STA &0328,Y ;store it C73C DEY ;point to next byte C73D BPL &C737 ;transfer till Y=&FF lo byte of X coordinate in &328 C73F LDA #&28 ; C741 JSR &D839 ;check window boundaries C744 LDY #&04 ;Y=4 C746 BNE &C750 ;jump to C750 ************************************************************************* * * * OSWORD 11 - READ PALLETTE * * * ************************************************************************* ;on entry &EF=A=11 ; &F0=X=low byte of parameter block address ; &F1=Y=high byte of parameter block address ; PARAMETER BLOCK ; 0=logical colour to read ;on exit, result in BLOCK ; 0=logical colour ; 1=physical colour ; 2=red colour component \ ; 3=green colour component } when set using analogue colours ; 4=blue colour component / RPALET C748 AND &0360 ;number of logical colours less 1 C74B TAX ;put it in X C74C LDA &036F,X ;colour pallette PRACA C74F INY ;increment Y to point to byte 1 PRACB C750 STA (&F0),Y ;store data C752 LDA #&00 ;issue 0s C754 CPY #&04 ;to next bytes until Y=4 C756 BNE &C74F ; PRACX C758 RTS ;and exit ************************************************************************* * * * VDU 12 - CLEAR TEXT SCREEN * * CLS * * * ************************************************************************* ; FF C759 JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C75C BNE &C7BD ;if graphics cursor &C7BD C75E LDA &D0 ;VDU status byte C760 AND #&08 ;check if software scrolling (text window set) C762 BNE &C767 ;if so C767 C764 JMP &CBC1 ;initialise screen display and home cursor FFA C767 LDX &030B ;top of text window FFB C76A STX &0319 ;current text line C76D JSR &CEAC ;clear a line C770 LDX &0319 ;current text line C773 CPX &0309 ;bottom margin C776 INX ;X=X+1 C777 BCC &C76A ;if X at compare is less than bottom margin clear next ************************************************************************* * * * VDU 30 - HOME CURSOR * * * ************************************************************************* RS C779 JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C77C BEQ &C781 ;if text cursor C781 C77E JMP &CFA6 ;home graphic cursor if graphic RSA C781 STA &0323 ;store 0 in last two parameters C784 STA &0322 ; ************************************************************************* * * * VDU 31 - POSITION TEXT CURSOR * * TAB(X,Y) * * * * 2 parameters * * * ************************************************************************* ;0322 = supplied X coordinate ;0323 = supplied Y coordinate US C787 JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C78A BNE &C758 ;exit C78C JSR &C7A8 ;exchange text column/line with workspace 0328/9 C78F CLC ;clear carry C790 LDA &0322 ;get X coordinate C793 ADC &0308 ;add to text window left C796 STA &0318 ;store as text column C799 LDA &0323 ;get Y coordinate C79C CLC ; C79D ADC &030B ;add top of text window C7A0 STA &0319 ;current text line C7A3 JSR &CEE8 ;set up screen address C7A6 BCC &C732 ;set cursor position if C=0 (point on screen) USAF C7A8 LDX #&18 ;else point to workspace C7AA LDY #&28 ;and line/column to restore old values C7AC JMP &CDDE ;exchange &300/1+X with &300/1+Y ************************************************************************* * * * VDU 13 - CARRIAGE RETURN * * * ************************************************************************* CR C7AF JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor C7B2 BEQ &C7B7 ;if text C7B7 C7B4 JMP &CFAD ;else set graphics cursor to left hand columm CRA C7B7 JSR &CE6E ;set text column to left hand column C7BA JMP &C6AF ;set up cursor and display address FFANOM C7BD JSR &CFA6 ;home graphic cursor ************************************************************************* * * * VDU 16 - CLEAR GRAPHICS SCREEN * * CLG * * * ************************************************************************* DLE C7C0 LDA &0361 ;pixels per byte C7C3 BEQ &C7F8 ;if 0 current mode has no graphics so exit C7C5 LDX &035A ;Background graphics colour C7C8 LDY &035C ;background graphics plot mode (GCOL n) C7CB JSR &D0B3 ;set graphics byte mask in &D4/5 C7CE LDX #&00 ;graphics window C7D0 LDY #&28 ;workspace C7D2 JSR &D47C ;set(300/7+Y) from (300/7+X) C7D5 SEC ;set carry C7D6 LDA &0306 ;graphics window top low C7D9 SBC &0302 ;graphics window bottom low C7DC TAY ;Y=difference C7DD INY ;increment C7DE STY &0330 ;and store in workspace (this is line count) DLEA C7E1 LDX #&2C ; C7E3 LDY #&28 ; C7E5 JSR &D6A6 ;clear line C7E8 LDA &032E ;decrement window height in pixels C7EB BNE &C7F0 ; C7ED DEC &032F ; DLEB C7F0 DEC &032E ; C7F3 DEC &0330 ;decrement line count C7F6 BNE &C7E1 ;if <>0 then do it again DLEX C7F8 RTS ;exit ************************************************************************* * * * VDU 17 - DEFINE TEXT COLOUR * * COLOUR n * * * * 1 parameter * * * ************************************************************************* ;parameter in &0323 DCA C7F9 LDY #&00 ;Y=0 C7FB BEQ &C7FF ;jump to C7FF ************************************************************************* * * * VDU 18 - DEFINE GRAPHICS COLOUR * * GCOL k,c * * * * 2 parameters * * * ************************************************************************* ;parameters in 323,322 DCB C7FD LDY #&02 ;Y=2 DCABA C7FF LDA &0323 ;get last parameter C802 BPL &C805 ;if +ve it's foreground colour so C805 C804 INY ;else Y=Y+1 DCABB C805 AND &0360 ;number of logical colours less 1 C808 STA &DA ;store it C80A LDA &0360 ;number of logical colours less 1 C80D BEQ &C82B ;if none exit C80F AND #&07 ;else limit to an available colour and clear M C811 CLC ;clear carry C812 ADC &DA ;Add last parameter to get pointer to table C814 TAX ;pointer into X C815 LDA &C423,X ;get colour bitmap from table C818 STA &0357,Y ; colour Y=0=text fgnd, 1=text bkgnd, 2=graphics fg, etc C81B CPY #&02 ;If Y>1 C81D BCS &C82C ;then its graphics so C82C else C81F LDA &0357 ;foreground text colour C822 EOR #&FF ;invert C824 STA &D3 ;text colour byte to be orred or EORed into memory C826 EOR &0358 ;background text colour C829 STA &D2 ;text colour byte to be orred or EORed into memory DCABX C82B RTS ;and exit ; DCBAC C82C LDA &0322 ; Get GCOL action C82F STA &0359,Y ; Store at &35B foreground, &35C background C832 RTS ; exit ************************************************************************* * * * VDU 20 - RESTORE DEFAULT COLOURS * * * ************************************************************************* ; DCTTX C833 LDA #&20 ; SPACE character for teletext background C835 STA &0358 ; Set as background "colour" C838 RTS ; ; DCD C839 LDX #&05 ; X=5 C83B LDA #&00 ; A=0 DCDA C83D STA &0357,X ; Zero all colour settings C840 DEX ; txtfgd, txtbgd, gfxfdg, gfxbgd, gcolgfd, gcolbgd C841 BPL &C83D ; C843 LDX &0360 ; Number of logical colours less 1 C846 BEQ &C833 ; If zero, teletext so to C833 to set ttxt background C848 LDA #&FF ; A=&FF C84A CPX #&0F ; Is it a 16-colour MODE? C84C BNE &C850 ; Not 16 colours, skip past C84E LDA #&3F ; A=&3F for <16 colours, A=&FF for 16 colours DCDB C850 STA &0357 ;foreground text colour C853 STA &0359 ;foreground graphics colour C856 EOR #&FF ;invert A C858 STA &D2 ;text colour byte to be ORed or EORed into memory C85A STA &D3 ;text colour byte to be ORed or EORed into memory C85C STX &031F ;set first parameter of 5 C85F CPX #&03 ;if there are 4 colours C861 BEQ &C874 ;goto C874 C863 BCC &C885 ;if less there are 2 colours goto C885 ;else there are 16 colours C865 STX &0320 ;set second parameter DCDLF C868 JSR &C892 ;do VDU 19 etc C86B DEC &0320 ;decrement first parameter C86E DEC &031F ;and last parameter C871 BPL &C868 ; C873 RTS ; ; ********* 4 colour mode ************************************************* DCDLA C874 LDX #&07 ;X=7 C876 STX &0320 ;set first parameter DCDLH C879 JSR &C892 ;and do VDU 19 C87C LSR &0320 ; C87F DEC &031F ; C882 BPL &C879 ; C884 RTS ;exit ;********* 2 colour mode ************************************************ DCDLI C885 LDX #&07 ;X=7 C887 JSR &C88F ;execute VDU 19 C88A LDX #&00 ;X=0 C88C STX &031F ;store it as DCDLJ C88F STX &0320 ;both parameters ************************************************************************* * * * VDU 19 - DEFINE COLOURS * * [COLOUR l,p] * * [COLOUR l,r,g,b] * * * * 5 parameters * * * ************************************************************************* ;&31F=first parameter logical colour ;&320=second physical colour DCC C892 PHP ;push processor flags C893 SEI ;disable interrupts C894 LDA &031F ;get first parameter and C897 AND &0360 ;number of logical colours less 1 C89A TAX ;to make legal, then X=A C89B LDA &0320 ;A=second parameter DCC1 C89E AND #&0F ;make legal C8A0 STA &036F,X ;colour pallette C8A3 TAY ;Y=A C8A4 LDA &0360 ;number of logical colours less 1 C8A7 STA &FA ;store it C8A9 CMP #&03 ;is it 4 colour mode?? C8AB PHP ;save flags C8AC TXA ;A=X DCCALA C8AD ROR ;rotate A into &FA C8AE ROR &FA ; C8B0 BCS &C8AD ; C8B2 ASL &FA ; C8B4 TYA ;A=Y C8B5 ORA &FA ; C8B7 TAX ; C8B8 LDY #&00 ;Y=0 DCCALB C8BA PLP ;check flags C8BB PHP ; C8BC BNE &C8CC ;if A<>3 earlier C8CC C8BE AND #&60 ;else A=&60 to test bits 5 and 6 C8C0 BEQ &C8CB ;if not set C8CB C8C2 CMP #&60 ;else if both set C8C4 BEQ &C8CB ;C8CB C8C6 TXA ;A=X C8C7 EOR #&60 ;invert C8C9 BNE &C8CC ;and if not 0 C8CC DCCALD C8CB TXA ;X=A DCCALC C8CC JSR &EA11 ;call Osbyte 155 pass data to pallette register C8CF TYA ; C8D0 SEC ; C8D1 ADC &0360 ;number of logical colours less 1 C8D4 TAY ; C8D5 TXA ; C8D6 ADC #&10 ; C8D8 TAX ; C8D9 CPY #&10 ;if Y<16 do it again C8DB BCC &C8BA ; C8DD PLP ;pull flags twice C8DE PLP ; C8DF RTS ;and exit ************************************************************************* * * * OSWORD 12 - WRITE PALLETTE * * * ************************************************************************* ;on entry X=&F0:Y=&F1:YX points to parameter block ;byte 0 = logical colour; byte 1 physical colour; bytes 2-4=0 PRPAL C8E0 PHP ;push flags C8E1 AND &0360 ;and with number of logical colours less 1 C8E4 TAX ;X=A C8E5 INY ;Y=Y+1 C8E6 LDA (&F0),Y ;get physical colour C8E8 JMP &C89E ;do VDU19 with parameters in X and A ************************************************************************* * * * VDU 22 - SELECT MODE * * MODE n * * * * 1 parameter * * * ************************************************************************* ;parameter in &323 SYN C8EB LDA &0323 ;get parameter C8EE JMP &CB33 ;goto CB33 ************************************************************************* * * * VDU 23 - DEFINE CHARACTERS * * * * 9 parameters * * * ************************************************************************* ;parameters are:- ;31B character to define ;31C to 323 definition ETB C8F1 LDA &031B ;get character to define C8F4 CMP #&20 ;is it ' ' C8F6 BCC &C93F ;if less then it is a control instruction, go to C93F C8F8 PHA ;else save parameter C8F9 LSR ;A=A/32 C8FA LSR ; C8FB LSR ; C8FC LSR ; C8FD LSR ; C8FE TAX ;X=A C8FF LDA &C40D,X ;get font flag mask from table (A=&80/2^X) C902 BIT &0367 ;font flag C905 BNE &C927 ;and if A<>0 C927 storage area is established already C907 ORA &0367 ;or with font flag to set bit found to be 0 C90A STA &0367 ;font flag C90D TXA ;get A back C90E AND #&03 ;And 3 to clear all but bits 0 and 1 C910 CLC ;clear carry C911 ADC #&BF ;add &BF (A=&C0,&C1,&C2) to select a character page C913 STA &DF ;store it C915 LDA &0367,X ;get font location byte (normally &0C) C918 STA &DD ;store it C91A LDY #&00 ;Y=0 so (&DE) holds (&C000 -&C2FF) C91C STY &DC ; C91E STY &DE ; ETBC C920 LDA (&DE),Y ;transfer page to storage area C922 STA (&DC),Y ; C924 DEY ; C925 BNE &C920 ; ETBB C927 PLA ;get A back C928 JSR &D03E ;set up character definition pointers C92B LDY #&07 ;Y=7 ETBA C92D LDA &031C,Y ;transfer definition parameters C930 STA (&DE),Y ;to RAM definition C932 DEY ; C933 BPL &C92D ; C935 RTS ;and exit ; C936 PLA ;Pull A ETCX C937 RTS ;and exit ************ VDU EXTENSION ********************************************** JMIV25 C938 LDA &031F ;A=PLOT action number from VDU 25 C93B CLC ;clear carry=PLOT JMIV23 C93C JMP (&0226) ;jump via VDUV vector, A=PLOT number or VDU 23 number ********** VDU control commands ***************************************** ETBD C93F CMP #&01 ;is A=1 C941 BCC &C958 ;if less (0) then set CRT register directly C943 BNE &C93C ;if not 1 jump to VDUV with CS for VDU extension ********** turn cursor on/off ******************************************* C945 JSR &C588 ;A=0 if text cursor, A=&20 if graphics cursor C948 BNE &C937 ;if graphics exit C94A LDA #&20 ;A=&20 - preload to turn cursor off C94C LDY &031C ;Y=second VDU parameter C94F BEQ &C954 ;if 0, jump to C954 to turn cursor off ETCA C951 LDA &035F ;get last setting of CRT controller register ;for cursor on ETCB C954 LDY #&0A ;Y=10 - cursor control register number C956 BNE &C985 ;jump to C985, Y=register, Y=value ********** set CRT controller ******************************************* ETD C958 LDA &031D ;get third parameter - value C95B LDY &031C ;and second parameter - register CRTCP C95E CPY #&07 ;is Y=7 C960 BCC &C985 ;if less C985 C962 BNE &C967 ;else if >7 C967 C964 ADC &0290 ;else ADD screen vertical display adjustment CRTCPC C967 CPY #&08 ;If Y<>8 C969 BNE &C972 ;C972 C96B ORA #&00 ;if bit 7 set C96D BMI &C972 ;C972 C96F EOR &0291 ;else EOR with interlace toggle CRTCPD C972 CPY #&0A ;Y=10?? C974 BNE &C985 ;if not C985 C976 STA &035F ;last setting of CRT controller register C979 TAY ;Y=A C97A LDA &D0 ;VDU status byte C97C AND #&20 ;check bit 5 printing at graphics cursor? C97E PHP ;push flags C97F TYA ;Y=A C980 LDY #&0A ;Y=10 C982 PLP ;pull flags C983 BNE &C98B ;if graphics in use then C98B CRTCPA C985 STY &FE00 ;else set CRTC address register C988 STA &FE01 ;and poke new value to register Y CRTCPB C98B RTS ;exit ************************************************************************* * * * VDU 25 - PLOT * * PLOT k,x,y * * DRAW x,y * * MOVE x,y * * PLOT x,y * * 5 parameters * * * ************************************************************************* ; EM C98C LDX &0361 ;pixels per byte C98F BEQ &C938 ;if no graphics available go to VDU Extension C991 JMP &D060 ;else enter Plot routine at D060 ********** adjust screen RAM addresses ********************************** HSCRDN C994 LDX &0350 ;window area start address lo C997 LDA &0351 ;window area start address hi C99A JSR &CCF8 ;subtract bytes per character row from this C99D BCS &C9B3 ;if no wraparound needed C9B3 C99F ADC &0354 ;screen RAM size hi byte to wrap around C9A2 BCC &C9B3 ; HSCRUP C9A4 LDX &0350 ;window area start address lo C9A7 LDA &0351 ;window area start address hi C9AA JSR &CAD4 ;add bytes per char. row C9AD BPL &C9B3 ; C9AF SEC ;wrap around in other direction C9B0 SBC &0354 ;screen RAM size hi byte NEWTOP C9B3 STA &0351 ;window area start address hi C9B6 STX &0350 ;window area start address lo C9B9 LDY #&0C ;Y=12 C9BB BNE &CA0E ;jump to CA0E ************************************************************************* * * * VDU 26 - SET DEFAULT WINDOWS * * * ************************************************************************* SUB C9BD LDA #&00 ;A=0 C9BF LDX #&2C ;X=&2C SUBA C9C1 STA &0300,X ;clear all windows C9C4 DEX ; C9C5 BPL &C9C1 ;until X=&FF C9C7 LDX &0355 ;screen mode C9CA LDY C3EF,X ;text window right hand margin maximum C9CD STY &030A ;text window right C9D0 JSR &CA88 ;calculate number of bytes in a line C9D3 LDY &C3E7,X ;text window bottom margin maximum C9D6 STY &0309 ;bottom margin C9D9 LDY #&03 ;Y=3 C9DB STY &0323 ;set as last parameter C9DE INY ;increment Y C9DF STY &0321 ;set parameters C9E2 DEC &0322 ; C9E5 DEC &0320 ; C9E8 JSR &CA39 ;and do VDU 24 C9EB LDA #&F7 ; C9ED JSR &C5A8 ;clear bit 3 of &D0 C9F0 LDX &0350 ;window area start address lo C9F3 LDA &0351 ;window area start address hi CSDSPA C9F6 STX &034A ;text cursor 6845 address C9F9 STA &034B ;text cursor 6845 address C9FC BPL &CA02 ;set cursor position C9FE SEC ; C9FF SBC &0354 ;screen RAM size hi byte **************** set cursor position ************************************ CSDSP CA02 STX &D8 ;set &D8/9 from X/A CA04 STA &D9 ; CA06 LDX &034A ;text cursor 6845 address CA09 LDA &034B ;text cursor 6845 address CA0C LDY #&0E ;Y=15 AXCRTA CA0E PHA ;Push A CA0F LDA &0355 ;screen mode CA12 CMP #&07 ;is it mode 7? - should check map type or colours=0 CA14 PLA ;get back A CA15 BCS &CA27 ;if mode 7 selected CA27 CA17 STX &DA ;else store X CA19 LSR ;divide X/A by 8 CA1A ROR &DA ; CA1C LSR ; CA1D ROR &DA ; CA1F LSR ; CA20 ROR &DA ; CA22 LDX &DA ; CA24 JMP &CA2B ;goto CA2B AXCRTB CA27 SBC #&74 ;mode 7 subtract &74 CA29 EOR #&20 ;EOR with &20 AXCRTC CA2B STY &FE00 ;write to CRTC address file register CA2E STA &FE01 ;and to relevant address (register 14) CA31 INY ;Increment Y CA32 STY &FE00 ;write to CRTC address file register CA35 STX &FE01 ;and to relevant address (register 15) CA38 RTS ;and RETURN ************************************************************************* * * * VDU 24 - DEFINE GRAPHICS WINDOW * * * * 8 parameters * * * ************************************************************************* ;&31C/D Left margin ;&31E/F Bottom margin ;&320/1 Right margin ;&322/3 Top margin CAN CA39 JSR &CA81 ;exchange 310/3 with 328/3 CA3C LDX #&1C ; CA3E LDY #&2C ; CA40 JSR &D411 ;calculate width=right - left ; height = top-bottom CA43 ORA &032D ; CA46 BMI &CA81 ;exchange 310/3 with 328/3 and exit CA48 LDX #&20 ;X=&20 CA4A JSR &D149 ;scale pointers to mode CA4D LDX #&1C ;X=&1C CA4F JSR &D149 ;scale pointers to mode CA52 LDA &031F ;check for negative margins CA55 ORA &031D ; CA58 BMI &CA81 ;if found exchange 310/3 with 328/3 and exit CA5A LDA &0323 ; CA5D BNE &CA81 ;exchange 310/3 with 328/3 and exit CA5F LDX &0355 ;screen mode CA62 LDA &0321 ;right margin hi CA65 STA &DA ;store it CA67 LDA &0320 ;right margin lo CA6A LSR &DA ;/2 CA6C ROR ;A=A/2 CA6D LSR &DA ;/2 CA6F BNE &CA81 ;exchange 310/3 with 328/3 CA71 ROR ;A=A/2 CA72 LSR ;A=A/2 CA73 CMP C3EF,X ;text window right hand margin maximum CA76 BEQ &CA7A ;if equal CA7A CA78 BPL &CA81 ;exchange 310/3 with 328/3 CANA CA7A LDY #&00 ;Y=0 CA7C LDX #&1C ;X=&1C CA7E JSR &D47C ;set(300/7+Y) from (300/7+X) ***************** exchange 310/3 with 328/3 ***************************** CANB CA81 LDX #&10 ;X=10 CA83 LDY #&28 ;Y=&28 CA85 JMP &CDE6 ;exchange 300/3+Y and 300/3+X SETPWW CA88 INY ;Y=Y+1 CA89 TYA ;A=Y CA8A LDY #&00 ;Y=0 CA8C STY &034D ;text window width hi (bytes) CA8F STA &034C ;text window width lo (bytes) CA92 LDA &034F ;bytes per character CA95 LSR ;/2 CA96 BEQ &CAA1 ;if 0 exit SETPWA CA98 ASL &034C ;text window width lo (bytes) CA9B ROL &034D ;text window width hi (bytes) CA9E LSR ;/2 CA9F BCC &CA98 ; SETPWC CAA1 RTS ; ************************************************************************* * * * VDU 29 - SET GRAPHICS ORIGIN * * * * 4 parameters * * * ************************************************************************* ; GS CAA2 LDX #&20 ; CAA4 LDY #&0C ; CAA6 JSR &D48A ;(&300/3+Y)=(&300/3+X) CAA9 JMP &D1B8 ;set up external coordinates for graphics ************************************************************************* * * * VDU 127 (&7F) - DELETE (entry 32) * * * ************************************************************************* DEL CAAC JSR &C5C5 ;cursor left CAAF JSR &C588 ;A=0 if text cursor A=&20 if graphics cursor CAB2 BNE &CAC7 ;if graphics then CAC7 CAB4 LDX &0360 ;number of logical colours less 1 CAB7 BEQ &CAC2 ;if colours=0, mode 7 CAC2 CAB9 STA &DE ;else store A (always 0) CABB LDA #&C0 ;A=&C0 CABD STA &DF ;store in &DF (&DE) now points to C300 SPACE pattern CABF JMP &CFBF ;display a space PTTXSP CAC2 LDA #&20 ;A=&20 CAC4 JMP &CFDC ;and return to display a space DELA CAC7 LDA #&7F ;for graphics cursor CAC9 JSR &D03E ;set up character definition pointers CACC LDX &035A ;Background graphics colour CACF LDY #&00 ;Y=0 CAD1 JMP &CF63 ;invert pattern data (to background colour) ***** Add number of bytes in a line to X/A ****************************** ADDROW CAD4 PHA ;store A CAD5 TXA ;A=X CAD6 CLC ;clear carry CAD7 ADC &0352 ;bytes per character row CADA TAX ;X=A CADB PLA ;get back A CADC ADC &0353 ;bytes per character row CADF RTS ;and return ; ********* control scrolling in paged mode ******************************* PAGEF CAE0 JSR &CB14 ; zero paged mode line counter PAGE CAE3 JSR &E9D9 ; osbyte 118 read ctrl/shift keys CAE6 BCC &CAEA ; if carry clear, ctrl not pressed, CAEA CAE8 BMI &CAE0 ; if M set, shift pressed, loop to CAE0 do it again PAGEE CAEA LDA &D0 ;VDU status byte CAEC EOR #&04 ;invert bit 2 paged scrolling CAEE AND #&46 ;and if 2 cursors, paged mode off, or scrolling CAF0 BNE &CB1C ;barred then CB1C to exit CAF2 LDA &0269 ;paged mode counter CAF5 BMI &CB19 ;if negative then exit via CB19 CAF7 LDA &0319 ;current text line CAFA CMP &0309 ;bottom margin CAFD BCC &CB19 ;increment line counter and exit CAFF LSR ;A=A/4 CB00 LSR ; CB01 SEC ;set carry CB02 ADC &0269 ;paged mode counter CB05 ADC &030B ;top of text window CB08 CMP &0309 ;bottom margin CB0B BCC &CB19 ;increment line counter and exit CB0D CLC ; clear carry PAGEC CB0E JSR &E9D9 ; osbyte 118 read ctrl/shift keys, first loop with CC CB11 SEC ; set carry CB12 BPL &CB0E ; if +ve, shift not pressed, loop until shift pressed **************** zero paged mode counter ******************************* PAGED CB14 LDA #&FF ; CB16 STA &0269 ;paged mode counter PAGEA CB19 INC &0269 ;paged mode counter PAGEX CB1C RTS ; ********* intitialise VDU driver with MODE in A ************************* RESET CB1D PHA ; Save MODE in A CB1E LDX #&7F ; Prepare X=&7F for reset loop CB20 LDA #&00 ; A=0 CB22 STA &D0 ; Clear VDU status byte to set default conditions RESETA CB24 STA &02FF,X ; Zero VDU workspace at &300 to &37E CB27 DEX CB28 BNE &CB24 CB2A JSR &CD07 ; Implode character definitions CB2D PLA ; Get initial MODE back to A CB2E LDX #&7F ; X=&7F CB30 STX &0366 ; MODE 7 copy cursor character ********* enter here from VDU 22,n - MODE ******************************* CHMODE CB33 BIT &028E ; Check available RAM CB36 BMI &CB3A ; If 32K, use all MODEs CB38 ORA #&04 ; Only 16K available, force to use MODEs 4-7 CHMOD1 CB3A AND #&07 ; X=A AND 7 ensure legal mode CB3C TAX ; X=mode CB3D STX &0355 ; Set current screen MODE CB40 LDA &C414,X ; Get number of colours -1 for this MODE CB43 STA &0360 ; Set current number of logical colours less 1 CB46 LDA &C3FF,X ; Get bytes/character for this MODE CB49 STA &034F ; Set bytes per character CB4C LDA &C43A,X ; Get pixels/byte for this MODE CB4F STA &0361 ; Set pixels per byte CB52 BNE &CB56 ; If non-zero, skip past CB54 LDA #&07 ; pixels/byte=0, this is MODE 7, prepare A=7 offset into mask table CHMODB CB56 ASL ;A=A*2 CB57 TAY ;Y=A CB58 LDA &C406,Y ;mask table CB5B STA &0363 ;colour mask left CHMODC CB5E ASL ;A=A*2 CB5F BPL &CB5E ;If still +ve CB5E CB61 STA &0362 ;colour mask right CB64 LDY &C440,X ;screen display memory index table CB67 STY &0356 ;memory map type CB6A LDA &C44F,Y ;VDU section control CB6D JSR &E9F8 ;set hardware scrolling to VIA CB70 LDA &C44B,Y ;VDU section control CB73 JSR &E9F8 ;set hardware scrolling to VIA CB76 LDA &C459,Y ;Screen RAM size hi byte table CB79 STA &0354 ;screen RAM size hi byte CB7C LDA &C45E,Y ;screen ram address hi byte CB7F STA &034E ;hi byte of screen RAM address CB82 TYA ;Y=A CB83 ADC #&02 ;Add 2 CB85 EOR #&07 ; CB87 LSR ;/2 CB88 TAX ;X=A CB89 LDA &C466,X ;row multiplication table pointer CB8C STA &E0 ;store it CB8E LDA #&C3 ;A=&C3 CB90 STA &E1 ;store it (&E0) now points to C3B5 or C375 CB92 LDA &C463,X ;get number of bytes per row from table CB95 STA &0352 ;store as bytes per character row CB98 STX &0353 ;bytes per character row CB9B LDA #&43 ;A=&43 CB9D JSR &C5A8 ;A=A AND &D0, clear all except CursorEdit, NoScroll, PrintOn CBA0 LDX &0355 ;get screen mode CBA3 LDA &C3F7,X ;get video ULA control setting CBA6 JSR &EA00 ;set video ULA using osbyte 154 code CBA9 PHP ; Save IRQ state CBAA SEI ; Disable IRQs while changing CRTC registers CBAB LDX &C469,Y ; Get offset to end of register table for this screen map CBAE LDY #&0B ; Y=11, 12 registers to set CRTCLP CBB0 LDA &C46E,X ; Get register data for this register CBB3 JSR &C95E ; Set register Y CBB6 DEX ; Decrement pointer CBB7 DEY ; Decrement register number CBB8 BPL &CBB0 ; Loop until all done CBBA PLP ; Restore IRQs CBBB JSR &C839 ; VDU 20 - set default colours CBBE JSR &C9BD ; VDU 26 - set default windows ; Fast hardware screen clear CLATXT CBC1 LDX #&00 ;X=0 CBC3 LDA &034E ;hi byte of screen RAM address CBC6 STX &0350 ;window area start address lo CBC9 STA &0351 ;window area start address hi CBCC JSR &C9F6 ;use X and Y to set new cursor address CBCF LDY #&0C ;Y=12 CBD1 JSR &CA2B ;set registers 12 and 13 in CRTC CBD4 LDA &0358 ;background text colour CBD7 LDX &0356 ;memory map type CBDA LDY &C454,X ;get section control number CBDD STY &035D ;set it in jump vector lo CBE0 LDY #&CC ;Y=&CC CBE2 STY &035E ;upper byte of link address CBE5 LDX #&00 ;X=0 CBE7 STX &0269 ; Clear paged mode counter CBEA STX &0318 ; Set POS to zero CBED STX &0319 ; Set VPOS to zero CBF0 JMP (&035D) ; Jump via vector set up previously to clear screen memory ************************************************************************* * * * OSWORD 10 - READ CHARACTER DEFINITION * * * ************************************************************************* ;&EF=A:&F0=X:&F1=Y, on entry YX contains character number to be read ;(&DE) points to address ;on exit byte YX+1 to YX+8 contain definition RCHDEF CBF3 JSR &D03E ;set up character definition pointers CBF6 LDY #&00 ;Y=0 PRABA CBF8 LDA (&DE),Y ;get first byte CBFA INY ;Y=Y+1 CBFB STA (&F0),Y ;store it in YX CBFD CPY #&08 ;until Y=8 CBFF BNE &CBF8 ; CC01 RTS ;then exit ************************************************************************* * * * MAIN SCREEN CLEAR ROUTINE * * * ************************************************************************* ;on entry A contains background colour which is set in every byte ;of the screen ************************ Mode 0,1,2 entry point ************************* CLASCA CC02 STA &3000,X ; CC05 STA &3100,X ; CC08 STA &3200,X ; CC0B STA &3300,X ; CC0E STA &3400,X ; CC11 STA &3500,X ; CC14 STA &3600,X ; CC17 STA &3700,X ; CC1A STA &3800,X ; CC1D STA &3900,X ; CC20 STA &3A00,X ; CC23 STA &3B00,X ; CC26 STA &3C00,X ; CC29 STA &3D00,X ; CC2C STA &3E00,X ; CC2F STA &3F00,X ; ************************ Mode 3 entry point ***************************** CLASCB CC32 STA &4000,X ; CC35 STA &4100,X ; CC38 STA &4200,X ; CC3B STA &4300,X ; CC3E STA &4400,X ; CC41 STA &4500,X ; CC44 STA &4600,X ; CC47 STA &4700,X ; CC4A STA &4800,X ; CC4D STA &4900,X ; CC50 STA &4A00,X ; CC53 STA &4B00,X ; CC56 STA &4C00,X ; CC59 STA &4D00,X ; CC5C STA &4E00,X ; CC5F STA &4F00,X ; CC62 STA &5000,X ; CC65 STA &5100,X ; CC68 STA &5200,X ; CC6B STA &5300,X ; CC6E STA &5400,X ; CC71 STA &5500,X ; CC74 STA &5600,X ; CC77 STA &5700,X ; ************************ Mode 4,5 entry point *************************** CLASCC CC7A STA &5800,X ; CC7D STA &5900,X ; CC80 STA &5A00,X ; CC83 STA &5B00,X ; CC86 STA &5C00,X ; CC89 STA &5D00,X ; CC8C STA &5E00,X ; CC8F STA &5F00,X ; ************************ Mode 6 entry point ***************************** CLASCD CC92 STA &6000,X ; CC95 STA &6100,X ; CC98 STA &6200,X ; CC9B STA &6300,X ; CC9E STA &6400,X ; CCA1 STA &6500,X ; CCA4 STA &6600,X ; CCA7 STA &6700,X ; CCAA STA &6800,X ; CCAD STA &6900,X ; CCB0 STA &6A00,X ; CCB3 STA &6B00,X ; CCB6 STA &6C00,X ; CCB9 STA &6D00,X ; CCBC STA &6E00,X ; CCBF STA &6F00,X ; CCC2 STA &7000,X ; CCC5 STA &7100,X ; CCC8 STA &7200,X ; CCCB STA &7300,X ; CCCE STA &7400,X ; CCD1 STA &7500,X ; CCD4 STA &7600,X ; CCD7 STA &7700,X ; CCDA STA &7800,X ; CCDD STA &7900,X ; CCE0 STA &7A00,X ; CCE3 STA &7B00,X ; ************************ Mode 7 entry point ***************************** CLASCE CCE6 STA &7C00,X ; CCE9 STA &7D00,X ; CCEC STA &7E00,X ; CCEF STA &7F00,X ; CCF2 INX ; CCF3 BEQ &CD65 ;exit ****************** execute required function **************************** RUNNER CCF5 JMP (&035D) ;jump vector set up previously ********* subtract bytes per line from X/A ****************************** SUBROW CCF8 PHA ;Push A CCF9 TXA ;A=X CCFA SEC ;set carry for subtraction CCFB SBC &0352 ;bytes per character row CCFE TAX ;restore X CCFF PLA ;and A CD00 SBC &0353 ;bytes per character row CD03 CMP &034E ;hi byte of screen RAM address FONT0 CD06 RTS ;return ************************************************************************* * * * OSBYTE 20 - EXPLODE CHARACTERS * * * ************************************************************************* ; FONT CD07 LDA #&0F ;A=15 CD09 STA &0367 ;font flag indicating that page &0C,&C0-&C2 are ;used for user defined characters CD0C LDA #&0C ;A=&0C CD0E LDY #&06 ;set loop counter FIMPA CD10 STA &0368,Y ;set all font location bytes CD13 DEY ;to page &0C to indicate only page available CD14 BPL &CD10 ;for user character definitions CD16 CPX #&07 ;is X=7 or greater CD18 BCC &CD1C ;if not CD1C CD1A LDX #&06 ;else X=6 FONTOK CD1C STX &0246 ;character definition explosion switch CD1F LDA &0243 ;A=primary OSHWM CD22 LDX #&00 ;X=0 FEXPA CD24 CPX &0246 ;character definition explosion switch CD27 BCS &CD34 ; CD29 LDY &C4BA,X ;get soft character RAM allocation CD2C STA &0368,Y ;font location bytes CD2F ADC #&01 ;Add 1 CD31 INX ;X=X+1 CD32 BNE &CD24 ;if X<>0 then CD24 FEXPA0 CD34 STA &0244 ; Set new value of OSHWM (bottom of user memory) CD37 TAY ; Y=A CD38 BEQ &CD06 ; If OSHWM=&0000, return via CD06 (ERROR? will never happen) CD3A LDX #&11 ; X=&11 CD3C JMP &F168 ; Issue paged ROM service call &11, ; Font implosion/explosion warning ******** move text cursor to next line ********************************** ENSCRL CD3F LDA #&02 ;A=2 to check if scrolling disabled CD41 BIT &D0 ;VDU status byte CD43 BNE &CD47 ;if scrolling is barred CD47 CD45 BVC &CD79 ;if cursor editing mode disabled RETURN ENSCRM CD47 LDA &0309 ;bottom margin CD4A BCC &CD4F ;if carry clear on entry CD4F CD4C LDA &030B ;else if carry set get top of text window ENSCRN CD4F BVS &CD59 ;and if cursor editing enabled CD59 CD51 STA &0319 ;get current text line CD54 PLA ;pull return link from stack CD55 PLA ; CD56 JMP &C6AF ;set up cursor and display address ENSCRO CD59 PHP ;push flags CD5A CMP &0365 ;Y coordinate of text input cursor CD5D BEQ &CD78 ;if A=line count of text input cursor CD78 to exit CD5F PLP ;get back flags CD60 BCC &CD66 ; CD62 DEC &0365 ;Y coordinate of text input cursor ENSCRX CD65 RTS ;exit ; ENSCRQ CD66 INC &0365 ;Y coordinate of text input cursor CD69 RTS ;exit *********************** set up write cursor ******************************** OCSRDL CD6A PHP ;save flags CD6B PHA ;save A CD6C LDY &034F ;bytes per character CD6F DEY ;Y=Y-1 CD70 BNE &CD8F ;if Y=0 Mode 7 is in use CD72 LDA &0338 ;get previously saved character under cursor CD75 STA (&D8),Y ;restore it to character cell of current character OCSREX CD77 PLA ;pull A ENSCRP CD78 PLP ;pull flags ENSCRR CD79 RTS ;and exit ; OCSRWR CD7A PHP ;push flags CD7B PHA ;push A CD7C LDY &034F ;bytes per character CD7F DEY ; CD80 BNE &CD8F ;if not mode 7 CD82 LDA (&D8),Y ;get character from current character cell CD84 STA &0338 ;store it CD87 LDA &0366 ;mode 7 write cursor character CD8A STA (&D8),Y ;store it in current character cell CD8C JMP &CD77 ;and exit OCSRIV CD8F LDA #&FF ;A=&FF =cursor CD91 CPY #&1F ;except in mode 2 (Y=&1F) CD93 BNE &CD97 ;if not CD97 CD95 LDA #&3F ;load cursor byte mask ********** produce white block write cursor ***************************** OCSRIW CD97 STA &DA ;store it OCSRIU CD99 LDA (&D8),Y ;get scan line byte CD9B EOR &DA ;invert it CD9D STA (&D8),Y ;store it on scan line CD9F DEY ;decrement scan line counter CDA0 BPL &CD99 ;do it again CDA2 BMI &CD77 ;then jump to &CD77 SSCRDN CDA4 JSR &CE5B ;exchange line and column cursors with workspace copies CDA7 LDA &0309 ;bottom margin CDAA STA &0319 ;current text line CDAD JSR &CF06 ;set up display address SSCRDA CDB0 JSR &CCF8 ;subtract bytes per character row from this CDB3 BCS &CDB8 ;wraparound if necessary CDB5 ADC &0354 ;screen RAM size hi byte SSCRDB CDB8 STA &DB ;store A CDBA STX &DA ;X CDBC STA &DC ;A again CDBE BCS &CDC6 ;if C set there was no wraparound so CDC6 SSCRDC CDC0 JSR &CE73 ;copy line to new position ;using (&DA) for read ;and (&D8) for write CDC3 JMP &CDCE ; SSCRDE CDC6 JSR &CCF8 ;subtract bytes per character row from X/A CDC9 BCC &CDC0 ;if a result is outside screen RAM CDC0 CDCB JSR &CE38 ;perform a copy SSCRDD CDCE LDA &DC ;set write pointer from read pointer CDD0 LDX &DA ; CDD2 STA &D9 ; CDD4 STX &D8 ; CDD6 DEC &DE ;decrement window height CDD8 BNE &CDB0 ;and if not zero CDB0 SSCREX CDDA LDX #&28 ;point to workspace CDDC LDY #&18 ;point to text column/line SWAP2 CDDE LDA #&02 ;number of bytes to swap CDE0 BNE &CDE8 ;exchange (328/9)+Y with (318/9)+X SWAPB CDE2 LDX #&24 ;point to graphics cursor SWAPC CDE4 LDY #&14 ;point to last graphics cursor ;A=4 to swap X and Y coordinates *************** exchange 300/3+Y with 300/3+X *************************** SWAPF CDE6 LDA #&04 ;A =4 ************** exchange (300/300+A)+Y with (300/300+A)+X ***************** SWAP CDE8 STA &DA ;store it as loop counter SWAPA CDEA LDA &0300,X ;get byte CDED PHA ;store it CDEE LDA &0300,Y ;get byte pointed to by Y CDF1 STA &0300,X ;put it in 300+X CDF4 PLA ;get back A CDF5 STA &0300,Y ;put it in 300+Y CDF8 INX ;increment pointers CDF9 INY ; CDFA DEC &DA ;decrement loop counter CDFC BNE &CDEA ;and if not 0 do it again CDFE RTS ;and exit ******** execute upward scroll ****************************************** ; SSCRUP CDFF JSR &CE5B ;exchange line and column cursors with workspace copies CE02 LDY &030B ;top of text window CE05 STY &0319 ;current text line CE08 JSR &CF06 ;set up display address SSCRUA CE0B JSR &CAD4 ;add bytes per char. row CE0E BPL &CE14 ; CE10 SEC ; CE11 SBC &0354 ;screen RAM size hi byte SSCRUE CE14 STA &DB ;(&DA)=X/A CE16 STX &DA ; CE18 STA &DC ;&DC=A CE1A BCC &CE22 ; SSCRUC CE1C JSR &CE73 ;copy line to new position ;using (&DA) for read ;and (&D8) for write CE1F JMP &CE2A ;exit SSCRUB CE22 JSR &CAD4 ;add bytes per char. row CE25 BMI &CE1C ;if outside screen RAM CE1C CE27 JSR &CE38 ;perform a copy SSCRUD CE2A LDA &DC ; CE2C LDX &DA ; CE2E STA &D9 ; CE30 STX &D8 ; CE32 DEC &DE ;decrement window height CE34 BNE &CE0B ;CE0B if not 0 CE36 BEQ &CDDA ;exchange text column/linelse CDDA *********** copy routines *********************************************** XMIT CE38 LDX &034D ;text window width hi (bytes) CE3B BEQ &CE4D ;if no more than 256 bytes to copy X=0 so CE4D CE3D LDY #&00 ;Y=0 to set loop counter XMITA CE3F LDA (&DA),Y ;copy 256 bytes CE41 STA (&D8),Y ; CE43 INY ; CE44 BNE &CE3F ;Till Y=0 again CE46 INC &D9 ;increment hi bytes CE48 INC &DB ; CE4A DEX ;decrement window width CE4B BNE &CE3F ;if not 0 go back and do loop again XMITB CE4D LDY &034C ;text window width lo (bytes) CE50 BEQ &CE5A ;if Y=0 CE5A XMITC CE52 DEY ;else Y=Y-1 CE53 LDA (&DA),Y ;copy Y bytes CE55 STA (&D8),Y ; CE57 TYA ;A=Y CE58 BNE &CE52 ;if not 0 CE52 XMITD CE5A RTS ;and exit SSCRIN CE5B JSR &CDDA ;exchange text column/line with workspace CE5E SEC ;set carry CE5F LDA &0309 ;bottom margin CE62 SBC &030B ;top of text window CE65 STA &DE ;store it CE67 BNE &CE6E ;set text column to left hand column CE69 PLA ;get back return address CE6A PLA ; CE6B JMP &CDDA ;exchange text column/line with workspace SSCRIM CE6E LDA &0308 ;text window left CE71 BPL &CEE3 ;Jump CEE3 always! SSCRS CE73 LDA &DA ;get back A CE75 PHA ;push A CE76 SEC ;set carry CE77 LDA &030A ;text window right CE7A SBC &0308 ;text window left CE7D STA &DF ; SSCRSA CE7F LDY &034F ;bytes per character to set loop counter CE82 DEY ;copy loop SSCRSB CE83 LDA (&DA),Y ; CE85 STA (&D8),Y ; CE87 DEY ; CE88 BPL &CE83 ; CE8A LDX #&02 ;X=2 SSCRSC CE8C CLC ;clear carry CE8D LDA &D8,X ; CE8F ADC &034F ;bytes per character CE92 STA &D8,X ; CE94 LDA &D9,X ; CE96 ADC #&00 ; CE98 BPL &CE9E ;if this remains in screen RAM OK CE9A SEC ;else wrap around screen CE9B SBC &0354 ;screen RAM size hi byte SSCRSD CE9E STA &D9,X ; CEA0 DEX ;X=X-2 CEA1 DEX ; CEA2 BEQ &CE8C ;if X=0 adjust second set of pointers CEA4 DEC &DF ;decrement window width CEA6 BPL &CE7F ;and if still +ve do it all again CEA8 PLA ;get back A CEA9 STA &DA ;and store it CEAB RTS ;then exit *********** clear a line ************************************************ CLTLIN CEAC LDA &0318 ;text column CEAF PHA ;save it CEB0 JSR &CE6E ;set text column to left hand column CEB3 JSR &CF06 ;set up display address CEB6 SEC ;set carry CEB7 LDA &030A ;text window right CEBA SBC &0308 ;text window left CEBD STA &DC ;as window width CLTLNA CEBF LDA &0358 ;background text colour CEC2 LDY &034F ;bytes per character CLTNB CEC5 DEY ;Y=Y-1 decrementing loop counter CEC6 STA (&D8),Y ;store background colour at this point on screen CEC8 BNE &CEC5 ;if Y<>0 do it again CECA TXA ;else A=X CECB CLC ;clear carry to add CECC ADC &034F ;bytes per character CECF TAX ;X=A restoring it CED0 LDA &D9 ;get hi byte CED2 ADC #&00 ;Add carry if any CED4 BPL &CEDA ;if +ve CeDA CED6 SEC ;else wrap around CED7 SBC &0354 ;screen RAM size hi byte CLNLC CEDA STX &D8 ;restore D8/9 CEDC STA &D9 ; CEDE DEC &DC ;decrement window width CEE0 BPL &CEBF ;ind if not 0 do it all again CEE2 PLA ;get back A CLTLND CEE3 STA &0318 ;restore text column FAIL CEE6 SEC ;set carry CEE7 RTS ;and exit TADDR CEE8 LDX &0318 ;text column CEEB CPX &0308 ;text window left CEEE BMI &CEE6 ;if less than left margin return with carry set CEF0 CPX &030A ;text window right CEF3 BEQ &CEF7 ;if equal to right margin thats OK CEF5 BPL &CEE6 ;if greater than right margin return with carry set TADDRX CEF7 LDX &0319 ;current text line CEFA CPX &030B ;top of text window CEFD BMI &CEE6 ;if less than top margin CEFF CPX &0309 ;bottom margin CF02 BEQ &CF06 ;set up display address CF04 BPL &CEE6 ;or greater than bottom margin return with carry set ************ set up display address ************************************* ;Mode 0, (0319)*640+(0318)* 8 ;Mode 1, (0319)*640+(0318)*16 ;Mode 2, (0319)*640+(0318)*32 ;Mode 3, (0319)*640+(0318)* 8 ;Mode 4, (0319)*320+(0318)* 8 ;Mode 5, (0319)*320+(0318)*16 ;Mode 6, (0319)*320+(0318)* 8 ;Mode 7, (0319)* 40+(0318) ; ;this gives a displacement relative to the screen RAM start address ;which is added to the calculated number and stored in in 34A/B ;if the result is less than &8000, the top of screen RAM it is copied into X/A ;and D8/9. ;if the result is greater than &7FFF the hi byte of screen RAM size is ;subtracted to wraparound the screen. X/A, D8/9 are then set from this TADDRY CF06 LDA &0319 ;current text line CF09 ASL ;A=A*2 CF0A TAY ;Y=A CF0B LDA (&E0),Y ;get CRTC multiplication table pointer CF0D STA &D9 ;&D9=A CF0F INY ;Y=Y+1 CF10 LDA #&02 ;A=2 CF12 AND &0356 ;memory map type CF15 PHP ;save flags CF16 LDA (&E0),Y ;get CRTC multiplication table pointer CF18 PLP ;pull flags CF19 BEQ &CF1E ; CF1B LSR &D9 ;&D9=&D9/2 CF1D ROR ;A=A/2 +(128*carry) TADDRH CF1E ADC &0350 ;window area start address lo CF21 STA &D8 ;store it CF23 LDA &D9 ; CF25 ADC &0351 ;window area start address hi CF28 TAY ; CF29 LDA &0318 ;text column CF2C LDX &034F ;bytes per character CF2F DEX ;X=X-1 CF30 BEQ &CF44 ;if X=0 mode 7 CF44 CF32 CPX #&0F ;is it mode 1 or mode 5? CF34 BEQ &CF39 ;yes CF39 with carry set CF36 BCC &CF3A ;if its less (mode 0,3,4,6) CF3A CF38 ASL ;A=A*16 if entered here (mode 2) TADDRF CF39 ASL ;A=A*8 if entered here TADDRG CF3A ASL ;A=A*4 if entered here CF3B ASL ; CF3C BCC &CF40 ;if carry clear CF3E INY ;Y=Y+2 CF3F INY ; TADDRB CF40 ASL ;A=A*2 CF41 BCC &CF45 ;if carry clear add to &D8 CF43 INY ;if not Y=Y+1 TADDRE CF44 CLC ;clear carry TADDRC CF45 ADC &D8 ;add to &D8 CF47 STA &D8 ;and store it CF49 STA &034A ;text cursor 6845 address CF4C TAX ;X=A CF4D TYA ;A=Y CF4E ADC #&00 ;Add carry if set CF50 STA &034B ;text cursor 6845 address CF53 BPL &CF59 ;if less than &800 goto &CF59 CF55 SEC ;else wrap around CF56 SBC &0354 ;screen RAM size hi byte TADDRD CF59 STA &D9 ;store in high byte CF5B CLC ;clear carry CF5C RTS ;and exit ******** Graphics cursor display routine ******************************** PRANM CF5D LDX &0359 ;foreground graphics colour CF60 LDY &035B ;foreground graphics plot mode (GCOL n) PRANC CF63 JSR &D0B3 ;set graphics byte mask in &D4/5 CF66 JSR &D486 ;copy (324/7) graphics cursor to workspace (328/B) CF69 LDY #&00 ;Y=0 PRANMC CF6B STY &DC ;&DC=Y CF6D LDY &DC ;Y=&DC CF6F LDA (&DE),Y ;get pattern byte CF71 BEQ &CF86 ;if A=0 CF86 CF73 STA &DD ;else &DD=A PRANMA CF75 BPL &CF7A ;and if >0 CF7A CF77 JSR &D0E3 ;else display a pixel PRANMD CF7A INC &0324 ;current horizontal graphics cursor CF7D BNE &CF82 ; CF7F INC &0325 ;current horizontal graphics cursor PRANME CF82 ASL &DD ;&DD=&DD*2 CF84 BNE &CF75 ;and if<>0 CF75 PRANMG CF86 LDX #&28 ;point to workspace CF88 LDY #&24 ;point to horizontal graphics cursor CF8A JSR &D482 ;0300/1+Y=0300/1+X CF8D LDY &0326 ;current vertical graphics cursor CF90 BNE &CF95 ; CF92 DEC &0327 ;current vertical graphics cursor PRANMF CF95 DEC &0326 ;current vertical graphics cursor CF98 LDY &DC ; CF9A INY ; CF9B CPY #&08 ;if Y<8 then do loop again CF9D BNE &CF6B ;else CF9F LDX #&28 ;point to workspace CFA1 LDY #&24 ;point to graphics cursor CFA3 JMP &D48A ;(&300/3+Y)=(&300/3+X) *********** home graphics cursor *************************************** RSAN CFA6 LDX #&06 ;point to graphics window TOP CFA8 LDY #&26 ;point to workspace CFAA JSR &D482 ;0300/1+Y=0300/1+X ************* set graphics cursor to left hand column ******************* CRAN CFAD LDX #&00 ;X=0 point to graphics window left CFAF LDY #&24 ;Y=&24 CFB1 JSR &D482 ;0300/1+Y=0300/1+X CFB4 JMP &D1B8 ;set up external coordinates for graphics ************ Write text character to screen ***************************** VDUPR CFB7 LDX &0360 ; Number of logical colours less 1 CFBA BEQ &CFDC ; Zero, MODE 7, jump to CFDC CFBC JSR &D03E ; Set up character definition pointers VDUPRB CFBF LDX &0360 ; Get number of logical colours less 1 again CFC2 LDA &D0 ; Get VDU status byte CFC4 AND #&20 ; Test bit 5 for printing at graphics cursor CFC6 BNE &CF5D ; If set, jump to CF5D for graphics printing CFC8 LDY #&07 ; Prepare Y=7 for 8 pixel lines CFCA CPX #&03 ; Test X=3 CFCC BEQ &CFEE ; If four-colour mode, jump to CFEE CFCE BCS &D01E ; If >3, jump to D01E to deal with 16 colours *********** Write text in two-colour modes ****************************** PTWO CFD0 LDA (&DE),Y ; Get character pixel line CFD2 ORA &D2 ; Mask if foreground=background CFD4 EOR &D3 ; Toggle for inverted pixels CFD6 STA (&D8),Y ; Write to screen CFD8 DEY ; Y=Y-1 CFD9 BPL &CFD0 ; Loop for all 8 pixel lines CFDB RTS ; And exit *********** Write text in teletext mode ********************************* PTTX CFDC LDY #&02 ; Y=2 to check 3 teletext characters PTTXA CFDE CMP &C4B6,Y ; Compare with teletext conversion table CFE1 BEQ &CFE9 ; If equal jump to CFE9 to convert it CFE3 DEY ; Else Y=Y-1 CFE4 BPL &CFDE ; And loop to check all three characters PTTXB CFE6 STA (&D8,X) ; Write byte to screen, X=0 as number of colours CFE8 RTS ; And exit PTTXC CFE9 LDA &C4B7,Y ; Convert with teletext conversion table CFEC BNE &CFE6 ; And write it to screen *********** Write text in four-colour modes ***************************** PFOUR CFEE LDA (&DE),Y ; Get character pixel line CFF0 PHA ; Save it CFF1 LSR A ; Move high nybble to low nybble CFF2 LSR A ; CFF3 LSR A ; CFF4 LSR A ; CFF5 TAX ; Index into colour table CFF6 LDA &C31F,X ; Get 4-colour pixel mask for this nybble CFF9 ORA &D2 ; Mask where foreground=background CFFB EOR &D3 ; Toggle where pixels inverted CFFD STA (&D8),Y ; Write to screen CFFF TYA ; A=Y D000 CLC ; Clear carry D001 ADC #&08 ; Add 8 to point to next columns of bytes D003 TAY ; Y=A D004 PLA ; Get pixel line back D005 AND #&0F ; Keep bottom nybble D007 TAX ; Index into colour table D008 LDA &C31F,X ; Get 4-colour pixel mask for this nybble D00B ORA &D2 ; Mask where foreground=background D00D EOR &D3 ; Toggle where pixels inverted D00F STA (&D8),Y ; Write to screen D011 TYA ; A=Y D012 SBC #&08 ; A=A-9, move pointer back to previous column and decrement D014 TAY ; Y=A D015 BPL &CFEE ; if +ve do loop again PSXTC D017 RTS ; And exit PSXTB D018 TYA ;Y=Y-&21 D019 SBC #&21 ; D01B BMI &D017 ;IF Y IS negative then RETURN D01D TAY ;else A=Y ******* Write text in 16-colour modes *********************************** PSXTN D01E LDA (&DE),Y ; Get character pixel line D020 STA &DC ; Store it in workspace D022 SEC ;set carry PSXTA D023 LDA #&00 ;A=0 D025 ROL &DC ;carry now occupies bit 0 of DC D027 BEQ &D018 ;when DC=0 again D018 to deal with next pattern byte D029 ROL A ;get bit 7 from &DC into A bit 0 D02A ASL &DC ;rotate again to get second D02C ROL A ;bit into A D02D TAX ;and store result in X D02E LDA &C32F,X ;multiply by &55 using look up table D031 ORA &D2 ;and set colour factors D033 EOR &D3 ; D035 STA (&D8),Y ;and store result D037 CLC ;clear carry D038 TYA ;Y=Y+8 moving screen RAM pointer on 8 bytes D039 ADC #&08 ; D03B TAY ; D03C BCC &D023 ;loop to D023 to deal with next bit pair ************* calculate character pattern address for given code ******** ; On entry, A contains code on entry = 12345678 CHADDR D03E ASL ;23456780 C holds 1 D03F ROL ;34567801 C holds 2 D040 ROL ;45678012 C holds 3 D041 STA &DE ;save this pattern D043 AND #&03 ;00000012 D045 ROL ;00000123 C=0 D046 TAX ;X=A=0 - 7 D047 AND #&03 ;A=00000023 D049 ADC #&BF ;A=&BF,C0 or C1 D04B TAY ;this is used as a pointer D04C LDA &C40D,X ;A=&80/2^X i.e.1,2,4,8,&10,&20,&40, or &80 D04F BIT &0367 ;with font flag D052 BEQ &D057 ;if 0 D057 D054 LDY &0367,X ;else get hi byte from table CHADDS D057 STY &DF ;store Y D059 LDA &DE ;get back pattern D05B AND #&F8 ;convert to 45678000 D05D STA &DE ;and restore it D05F RTS ;exit ************************************************************************* ************************************************************************* ** ** ** ** ** PLOT ROUTINES ENTER HERE ** ** ** ** ** ************************************************************************* ************************************************************************* ;on entry ADDRESS PARAMETER DESCRIPTION ; 031F 1 plot action ; 0320/1 2,3 X coordinate ; 0322/3 4,5 Y coordinate EMA D060 LDX #&20 ; X=>coords in VDU queue D062 JSR &D14D ; Translate coordinates ; Set up plot masks from PLOT action ; Plot action %xxxxxx00 - uses GCOL 5 - do nothing ; %xxxxxx01 - uses current foreground GCOL action ; %xxxxxx10 - uses GCOL 4 - invert ; %xxxxxx11 - uses current background GCOL action ; ; Note that GCOL 5 still manipulates screen memory, with no visible change. D065 LDA &031F ; Get plot action D068 CMP #&04 ; Test for 4 for MOVE absolute, if so jump to D0D9 D06A BEQ &D0D9 ; to quickly finish, without testing for UKVDUV D06C LDY #&05 ; Prepare Y=5 for GCOL 5 - no action D06E AND #&03 ; Mask plot action to get colour mode in bit 0 and 1 D070 BEQ &D080 ; If 0, move without plotting, but still test fr UKVDUV D072 LSR ; Move bit 0 of action into Cy D073 BCS &D078 ; If set, action=%01 or %11, jump to D078 to use colour D075 DEY ; Set Y=4 for GCOL 4 - invert pixel D076 BNE &D080 ; Skip forward with %10 to invert current colour ******** graphics colour wanted ***************************************** EMAF D078 TAX ; X=A, X=0 for foreground colour, X=1 for background colour D079 LDY &035B,X ; Get foreground or background GCOL action D07C LDA &0359,X ; Get foreground or background graphics colour D07F TAX ; X=colour EMAA D080 JSR &D0B3 ; Set up plot masks in D4/5 from Y=GCOL action, X=colour D083 LDA &031F ; Get plot action D086 BMI &D0AB ; If &80-&FF, not implemented, jump to D0AB to pass to UKVDUV D088 ASL A ; Test bit 6 D089 BPL &D0C6 ; Bit 6 is 0, plot action is 0-63, jump to D0C6 to draw line ******** PLOT 64-127 - plot points and areas **************************** D08B AND #&F0 ; Mask out colour and abs/rel modes in bottom nybble D08D ASL A ; Shift action into top three bits D08E BEQ &D0D6 ; %01000xxx, plot action 64-71, jump to D0D6 to plot a point D090 EOR #&40 ; Toggle bit 6 of PLOT action * 4 D092 BEQ &D0A8 ; %01010xxx, plot action 80-87, jump to D0A8 to fill triangle D094 PHA ; Save A D095 JSR &D0DC ;copy 0320/3 to 0324/7 setting XY in current graphics ;coordinates D098 PLA ; Get back A - PLOT action * 4 D099 EOR #&60 ; Toggle bit 6 and bit 5 D09B BEQ &D0AE ; %01001xxx, plot actions 72-78, jump to D0AE for lateral fill D09D CMP #&40 ; D09F BNE &D0AB ; Not %01011xxx, jump to pass to UKVDUV ; Plot actions 88-95, line blanking D0A1 LDA #&02 ; Set A=2 for... D0A3 STA &DC ; Store it in workspace D0A5 JMP &D506 ; And jump to D506 to do line blanking EMAG D0A8 JMP &D5EA ; Jump to fill triangle routine EXGREF D0AB JMP &C938 ; Pass to UKVDUV - VDU extension vector EME D0AE STA &DC ; Store zero in workspace D0B0 JMP &D4BF ; Jump to do lateral fill ********* Set up plot masks ********************************************** ; On entry, Y=GCOL action ; X=colour ; Sets up: ; &D4 = (colour ORA C41C,gcol) EOR C41D,gcol ; &D5 = (colour ORA C41B,gcol) EOR C420,gcol SETZG D0B3 TXA ; A=colour D0B4 ORA &C41C,Y ; ORA colour with GCOL action table byte D0B7 EOR &C41D,Y ; EOR with following byte D0BA STA &D4 ; Store as plot mask D0BC TXA ; A=colour D0BD ORA &C41B,Y ; D0C0 EOR &C420,Y ; D0C3 STA &D5 ; Store as D0C5 RTS ;exit with masks in &D4/5 ************** PLOT 0-63 - draw lines *********************************** ; EMAD D0C6 ASL ;shift left again D0C7 BMI &D0AB ;if -ve options are in range 32-63 not implemented D0C9 ASL ;shift left twice more D0CA ASL ; D0CB BPL &D0D0 ;if still +ve type is 0-7 or 16-23 so D0D0 D0CD JSR &D0EB ;else display start point EMAB D0D0 JSR &D1ED ;perform calculations and draw line D0D3 JMP &D0D9 ;finish ************************************************************************* * * * PLOT 64-71 -= PLOT A SINGLE POINT * * * ************************************************************************* EMAC D0D6 JSR &D0EB ;plot a point EMAEX D0D9 JSR &CDE2 ;swap current and last graphics position EMAEY D0DC LDY #&24 ;Y=&24 EMAEZ D0DE LDX #&20 ;X=&20 D0E0 JMP &D48A ;copy parameters to 324/7 (300/3 +Y) WPLOTA D0E3 LDX #&24 ; X=>current graphics position D0E5 JSR &D85F ; Check window and calculate pixel address D0E8 BEQ &D0F0 ; Inside graphics windows, skip to plot it D0EA RTS ; Outside graphics winsow, exit ; Plot a single point WPLOT D0EB JSR &D85D ; Check window and calculate address for PLOT coord D0EE BNE &D103 ; If outside window, skip to exit PLOT D0F0 LDY &031A ; Get graphics scan line PLOTS D0F3 LDA &D1 ; Get bits to modify for this pixel D0F5 AND &D4 ; D0F7 ORA (&D6),Y ; Get byte from screen D0F9 STA &DA ; Save it for a moment D0FB LDA &D5 ; D0FD AND &D1 ; D0FF EOR &DA ; D101 STA (&D6),Y ; Write it to screen PLOTEX D103 RTS ; And exit ; this is a more simplistic version of the above ; to always set to foreground colour PLOTFF D104 LDA (&D6),Y ; Get byte from screen D106 ORA &D4 ; Mask in bits to set D108 EOR &D5 ; Toggle any bits to invert D10A STA (&D6),Y ; Store back to screen D10C RTS ; And exit ************************** Check window limits ************************* ; WINDCS D10D LDX #&24 ;X=&24 WIND D10F LDY #&00 ;Y=0 D111 STY &DA ;&DA=0 D113 LDY #&02 ;Y=2 D115 JSR &D128 ;check vertical graphics position 326/7 ;bottom and top margins 302/3, 306/7 D118 ASL &DA ;DATA is set in &DA bits 0 and 1 then shift left D11A ASL &DA ;twice to make room for next pass D11C DEX ;X=&22 D11D DEX ; D11E LDY #&00 ;Y=0 D120 JSR &D128 ;left and right margins 300/1, 304/5 ;cursor horizontal position 324/5 D123 INX ;X=X+2 D124 INX ; D125 LDA &DA ;A=&DA D127 RTS ;exit *** cursor and margins check ****************************************** ; WINDA D128 LDA &0302,X ;check for window violation D12B CMP &0300,Y ;300/1 +Y > 302/3+X D12E LDA &0303,X ;then window fault D131 SBC &0301,Y ; D134 BMI &D146 ;so D146 D136 LDA &0304,Y ;check other windows D139 CMP &0302,X ; D13C LDA &0305,Y ; D13F SBC &0303,X ; D142 BPL &D148 ;if no violation exit D144 INC &DA ;else DA=DA+1 WINDAA D146 INC &DA ;DA=DA+1 WINDAB D148 RTS ;and exit DA=0 no problems DA=1 first check 2, 2nd ***********set up and adjust positional data **************************** EIGABS D149 LDA #&FF ;A=&FF D14B BNE &D150 ;then &D150 EIG D14D LDA &031F ;get first parameter in plot EIGC D150 STA &DA ;store in &DA D152 LDY #&02 ;Y=2 D154 JSR &D176 ;set up vertical coordinates/2 D157 JSR &D1AD ;/2 again to convert 1023 to 0-255 for internal use ;this is why minimum vertical plot separation is 4 D15A LDY #&00 ;Y=0 D15C DEX ;X=x-2 D15D DEX ; D15E JSR &D176 ;set up horiz. coordinates/2 this is OK for mode0,4 D161 LDY &0361 ;get number of pixels/byte (-1) D164 CPY #&03 ;if Y=3 (modes 1 and 5) D166 BEQ &D16D ;D16D D168 BCS &D170 ;for modes 0 & 4 this is 7 so D170 D16A JSR &D1AD ;for other modes divide by 2 twice EIGA D16D JSR &D1AD ;divide by 2 EIGB D170 LDA &0356 ;get screen display type D173 BNE &D1AD ;if not 0 (modes 3-7) divide by 2 again D175 RTS ;and exit ;for mode 0, 1 division 1280 becomes 640 = horizontal resolution ;for mode 1, 2 divisions 1280 becomes 320 = horizontal resolution ;for mode 2, 3 divisions 1280 becomes 160 = horizontal resolution ;for mode 4, 2 divisions 1280 becomes 320 = horizontal resolution ;for mode 5, 3 divisions 1280 becomes 160 = horizontal resolution ********** calculate external coordinates in internal format *********** ;on entry X is usually &1E or &20 EIGS D176 CLC ;clear carry D177 LDA &DA ;get &DA D179 AND #&04 ;if bit 2=0 D17B BEQ &D186 ;then D186 to calculate relative coordinates D17D LDA &0302,X ;else get coordinate D180 PHA ; D181 LDA &0303,X ; D184 BCC &D194 ;and goto D194 EIGSA D186 LDA &0302,X ;get coordinate D189 ADC &0310,Y ;add cursor position D18C PHA ;save it D18D LDA &0303,X ; D190 ADC &0311,Y ;add cursor D193 CLC ;clear carry EIGSB D194 STA &0311,Y ;save new cursor D197 ADC &030D,Y ;add graphics origin D19A STA &0303,X ;store it D19D PLA ;get back lo byte D19E STA &0310,Y ;save it in new cursor lo D1A1 CLC ;clear carry D1A2 ADC &030C,Y ;add to graphics orgin D1A5 STA &0302,X ;store it D1A8 BCC &D1AD ;if carry set D1AA INC &0303,X ;increment hi byte as you would expect! DIVAXT D1AD LDA &0303,X ;get hi byte D1B0 ASL ; D1B1 ROR &0303,X ;divide by 2 D1B4 ROR &0302,X ; D1B7 RTS ;and exit ***** calculate external coordinates from internal coordinates************ IEG D1B8 LDY #&10 ;Y=10 D1BA JSR &D488 ;copy 324/7 to 310/3 i.e.current graphics cursor ;position to position in external values D1BD LDX #&02 ;X=2 D1BF LDY #&02 ;Y=2 D1C1 JSR &D1D5 ;multiply 312/3 by 4 and subtract graphics origin D1C4 LDX #&00 ;X=0 D1C6 LDY #&04 ;Y=4 D1C8 LDA &0361 ;get number of pixels/byte IEGA D1CB DEY ;Y=Y-1 D1CC LSR ;divide by 2 D1CD BNE &D1CB ;if result not 0 D1CB D1CF LDA &0356 ;else get screen display type D1D2 BEQ &D1D5 ;and if 0 D1D5 D1D4 INY ; IEGC D1D5 ASL &0310,X ;multiply coordinate by 2 D1D8 ROL &0311,X ; D1DB DEY ;Y-Y-1 D1DC BNE &D1D5 ;and if Y<>0 do it again D1DE SEC ;set carry D1DF JSR &D1E3 ; D1E2 INX ;increment X IEGD D1E3 LDA &0310,X ;get current graphics position in external coordinates D1E6 SBC &030C,X ;subtract origin D1E9 STA &0310,X ;store in graphics position D1EC RTS ;and exit ************* compare X and Y PLOT spans ******************************** LINE D1ED JSR &D40D ;Set X and Y spans in workspace 328/9 32A/B D1F0 LDA &032B ;compare spans D1F3 EOR &0329 ;if result -ve spans are different in sign so D1F6 BMI &D207 ;goto D207 D1F8 LDA &032A ;else A=hi byte of difference in spans D1FB CMP &0328 ; D1FE LDA &032B ; D201 SBC &0329 ; D204 JMP &D214 ;and goto D214 LINSBA D207 LDA &0328 ;A = hi byte of SUM of spans D20A CLC ; D20B ADC &032A ; D20E LDA &0329 ; D211 ADC &032B ; LINSBB D214 ROR ;A=A/2 D215 LDX #&00 ;X=0 D217 EOR &032B ; D21A BPL &D21E ;if positive result D21E D21C LDX #&02 ;else X=2 LINEA D21E STX &DE ;store it D220 LDA &C4AA,X ;set up vector address D223 STA &035D ;in 35D D226 LDA &C4AB,X ; D229 STA &035E ;and 35E D22C LDA &0329,X ;get hi byte of span D22F BPL &D235 ;if +ve D235 D231 LDX #&24 ;X=&24 D233 BNE &D237 ;jump to D237 LINEB D235 LDX #&20 ;X=&20 LINEC D237 STX &DF ;store it D239 LDY #&2C ;Y=&2C D23B JSR &D48A ;get X coordinate data or horizontal coord of ;curent graphics cursor D23E LDA &DF ;get back original X D240 EOR #&04 ;covert &20 to &24 and vice versa D242 STA &DD ; D244 ORA &DE ; D246 TAX ; D247 JSR &D480 ;copy 330/1 to 300/1+X D24A LDA &031F ;get plot type D24D AND #&10 ;check bit 4 D24F ASL ; D250 ASL ; D251 ASL ;move to bit 7 D252 STA &DB ;store it D254 LDX #&2C ;X=&2C D256 JSR &D10F ;check for window violations D259 STA &DC ; D25B BEQ &D263 ;if none then D263 D25D LDA #&40 ;else set bit 6 of &DB D25F ORA &DB ; D261 STA &DB ; LINED D263 LDX &DD ; D265 JSR &D10F ;check window violations again D268 BIT &DC ;if bit 7 of &DC NOT set D26A BEQ &D26D ;D26D D26C RTS ;else exit ; LINEE D26D LDX &DE ;X=&DE D26F BEQ &D273 ;if X=0 D273 D271 LSR ;A=A/2 D272 LSR ;A=A/2 LINEF D273 AND #&02 ;clear all but bit 2 D275 BEQ &D27E ;if bit 2 set D27E D277 TXA ;else A=X D278 ORA #&04 ;A=A or 4 setting bit 3 D27A TAX ;X=A D27B JSR &D480 ;set 300/1+x to 330/1 LINEG D27E JSR &D42C ;more calcualtions D281 LDA &DE ;A=&DE EOR 2 D283 EOR #&02 ; D285 TAX ;X=A D286 TAY ;Y=A D287 LDA &0329 ;compare upper byte of spans D28A EOR &032B ; D28D BPL &D290 ;if signs are the same D290 D28F INX ;else X=X+1 LINEH D290 LDA &C4AE,X ;get vector addresses and store 332/3 D293 STA &0332 ; D296 LDA &C4B2,X ; D299 STA &0333 ; D29C LDA #&7F ;A=&7F D29E STA &0334 ;store it D2A1 BIT &DB ;if bit 6 set D2A3 BVS &D2CE ;the D2CE D2A5 LDA &C447,X ;get VDU section number D2A8 TAX ;X=A D2A9 SEC ;set carry D2AA LDA &0300,X ;subtract coordinates D2AD SBC &032C,Y ; D2B0 STA &DA ; D2B2 LDA &0301,X ; D2B5 SBC &032D,Y ; D2B8 LDY &DA ;Y=hi D2BA TAX ;X=lo=A D2BB BPL &D2C0 ;and if A+Ve D2C0 D2BD JSR &D49B ;negate Y/A LINEH1 D2C0 TAX ;X=A increment Y/A D2C1 INY ;Y=Y+1 D2C2 BNE &D2C5 ; D2C4 INX ;X=X+1 LINEH2 D2C5 TXA ;A=X D2C6 BEQ &D2CA ;if A=0 D2CA D2C8 LDY #&00 ;else Y=0 LINEH3 D2CA STY &DF ;&DF=Y D2CC BEQ &D2D7 ;if 0 then D2D7 LINEI D2CE TXA ;A=X D2CF LSR ;A=A/4 D2D0 ROR ; D2D1 ORA #&02 ;bit 1 set D2D3 EOR &DE ; D2D5 STA &DE ;and store LINEJ1 D2D7 LDX #&2C ;X=&2C D2D9 JSR &D864 ; D2DC LDX &DC ; D2DE BNE &D2E2 ; D2E0 DEC &DD ; LINEN D2E2 DEX ;X=X-1 LINEJ D2E3 LDA &DB ;A=&3B D2E5 BEQ &D306 ;if 0 D306 D2E7 BPL &D2F9 ;else if +ve D2F9 D2E9 BIT &0334 ; D2EC BPL &D2F3 ;if bit 7=0 D2F3 D2EE DEC &0334 ;else decrement D2F1 BNE &D316 ;and if not 0 D316 DOTY D2F3 INC &0334 ; D2F6 ASL ;A=A*2 D2F7 BPL &D306 ;if +ve D306 MPTEST D2F9 STX &DC ; D2FB LDX #&2C ; D2FD JSR &D85F ;calculate screen position D300 LDX &DC ;get back original X D302 ORA #&00 ; D304 BNE &D316 ; DOTYES D306 LDA &D1 ;byte mask for current graphics point D308 AND &D4 ;and with graphics colour byte D30A ORA (&D6),Y ;or with curent graphics cell line D30C STA &DA ;store result D30E LDA &D5 ;same again with next byte (hi??) D310 AND &D1 ; D312 EOR &DA ; D314 STA (&D6),Y ;then store it in current graphics line DOTNO D316 SEC ;set carry D317 LDA &0335 ;A=&335/6-&337/8 D31A SBC &0337 ; D31D STA &0335 ; D320 LDA &0336 ; D323 SBC &0338 ; D326 BCS &D339 ;if carry set D339 D328 STA &DA ; D32A LDA &0335 ; D32D ADC &0339 ; D330 STA &0335 ; D333 LDA &DA ; D335 ADC &033A ; D338 CLC ; LINED1 D339 STA &0336 ; D33C PHP ; D33D BCS &D348 ;if carry clear jump to VDU routine else D348 D33F JMP (&0332) ; ****** vertical scan module 1****************************************** NB D342 DEY ;Y=Y-1 D343 BPL &D348 ;if + D348 D345 JSR &D3D3 ;else d3d3 to advance pointers LINEE1 D348 JMP (&035D) ;and JUMP (&35D) ****** vertical scan module 2****************************************** SB D34B INY ;Y=Y+1 D34C CPY #&08 ;if Y<>8 D34E BNE &D348 ;then D348 D350 CLC ;else clear carry D351 LDA &D6 ;get address of top line of cuirrent graphics cell D353 ADC &0352 ;add number of bytes/character row D356 STA &D6 ;store it D358 LDA &D7 ;do same for hibyte D35A ADC &0353 ; D35D BPL &D363 ;if result -ve then we are above screen RAM D35F SEC ;so D360 SBC &0354 ;subtract screen memory size hi SBDYA D363 STA &D7 ;store it this wraps around point to screen RAM D365 LDY #&00 ;Y=0 D367 JMP (&035D) ; ****** horizontal scan module 1****************************************** EB D36A LSR &D1 ;shift byte mask D36C BCC &D348 ;if carry clear (&D1 was +ve) goto D348 D36E JSR &D3ED ;else reset pointers D371 JMP (&035D) ;and off to do more ****** horizontal scan module 2****************************************** WB D374 ASL &D1 ;shift byte mask D376 BCC &D348 ;if carry clear (&D1 was +ve) goto D348 D378 JSR &D3FD ;else reset pointers D37B JMP (&035D) ;and off to do more NA D37E DEY ;Y=Y-1 D37F BPL &D38D ;if +ve D38D D381 JSR &D3D3 ;advance pointers D384 BNE &D38D ;goto D38D normally EA D386 LSR &D1 ;shift byte mask D388 BCC &D38D ;if carry clear (&D1 was +ve) goto D348 D38A JSR &D3ED ;else reset pointers LINEF1 D38D PLP ;pull flags D38E INX ;X=X+1 D38F BNE &D395 ;if X>0 D395 D391 INC &DD ;else increment &DD D393 BEQ &D39F ;and if not 0 D39F LINEG1 D395 BIT &DB ;else if BIT 6 = 1 D397 BVS &D3A0 ;goto D3A0 D399 BCS &D3D0 ;if BIT 7=1 D3D0 D39B DEC &DF ;else Decrement &DF D39D BNE &D3D0 ;and if not Zero D3D0 LINEX D39F RTS ;else return ; FRIGA D3A0 LDA &DE ;A=&DE D3A2 STX &DC ;&DC=X D3A4 AND #&02 ;clear all but bit 1 D3A6 TAX ;X=A D3A7 BCS &D3C2 ;and if carry set goto D3C2 D3A9 BIT &DE ;if Bit 7 of &DE =1 D3AB BMI &D3B7 ;then D3B7 D3AD INC &032C,X ;else increment D3B0 BNE &D3C2 ;and if not 0 D3C2 D3B2 INC &032D,X ;else increment hi byte D3B5 BCC &D3C2 ;and if carry clear D3C2 FRIGB D3B7 LDA &032C,X ;esle A=32C,X D3BA BNE &D3BF ;and if not 0 D3BF D3BC DEC &032D,X ;decrement hi byte FRIGF D3BF DEC &032C,X ;decrement lo byte FRIGC D3C2 TXA ;A=X D3C3 EOR #&02 ;invert bit 2 D3C5 TAX ;X=A D3C6 INC &032C,X ;Increment 32C/D D3C9 BNE &D3CE ; D3CB INC &032D,X ; FRIGD D3CE LDX &DC ;X=&DC LINEK D3D0 JMP &D2E3 ;jump to D2E3 **********move display point up a line ********************************** NBDY D3D3 SEC ;SET CARRY D3D4 LDA &D6 ;subtract number of bytes/line from address of D3D6 SBC &0352 ;top line of current graphics cell D3D9 STA &D6 ; D3DB LDA &D7 ; D3DD SBC &0353 ; D3E0 CMP &034E ;compare with bottom of screen memory D3E3 BCS &D3E8 ;if outside screen RAM D3E5 ADC &0354 ;add screen memory size to wrap it around NBDYA D3E8 STA &D7 ;store in current address of graphics cell top line D3EA LDY #&07 ;Y=7 D3EC RTS ;and RETURN EBDY D3ED LDA &0362 ;get current left colour mask D3F0 STA &D1 ;store it D3F2 LDA &D6 ;get current top line of graphics cell D3F4 ADC #&07 ;ADD 7 D3F6 STA &D6 ; D3F8 BCC &D3FC ; D3FA INC &D7 ; EBDYA D3FC RTS ;and return WBDY D3FD LDA &0363 ;get right colour mask D400 STA &D1 ;store it D402 LDA &D6 ;A=top line graphics cell low D404 BNE &D408 ;if not 0 D408 D406 DEC &D7 ;else decrement hi byte WBDYA D408 SBC #&08 ;subtract 9 (8 + carry) D40A STA &D6 ;and store in low byte D40C RTS ;return ********:: coordinate subtraction *************************************** LINSA D40D LDY #&28 ;X=&28 D40F LDX #&20 ;Y=&20 TRISA D411 JSR &D418 ; D414 INX ;X=X+2 D415 INX ; D416 INY ;Y=Y+2 D417 INY ; TRISAA D418 SEC ;set carry D419 LDA &0304,X ;subtract coordinates D41C SBC &0300,X ; D41F STA &0300,Y ; D422 LDA &0305,X ; D425 SBC &0301,X ; D428 STA &0301,Y ; D42B RTS ;and return LINSG D42C LDA &DE ;A=&DE D42E BNE &D437 ;if A=0 D437 D430 LDX #&28 ;X=&28 D432 LDY #&2A ;Y=&2A D434 JSR &CDDE ;exchange 300/1+Y with 300/1+X ;IN THIS CASE THE X AND Y SPANS! LINSGB D437 LDX #&28 ;X=&28 D439 LDY #&37 ;Y=&37 D43B JSR &D48A ;copy &300/4+Y to &300/4+X ;transferring X and Y spans in this case D43E SEC ;set carry D43F LDX &DE ;X=&DE D441 LDA &0330 ;subtract 32C/D,X from 330/1 D444 SBC &032C,X ; D447 TAY ;partial answer in Y D448 LDA &0331 ; D44B SBC &032D,X ; D44E BMI &D453 ;if -ve D453 D450 JSR &D49B ;else negate Y/A LINSGA D453 STA &DD ;store A D455 STY &DC ;and Y D457 LDX #&35 ;X=&35 TRISB D459 JSR &D467 ;get coordinates D45C LSR ; D45D STA &0301,X ; D460 TYA ; D461 ROR ; D462 STA &0300,X ; D465 DEX ; D466 DEX ; TRISBA D467 LDY &0304,X ; D46A LDA &0305,X ; D46D BPL &D47B ;if A is +ve RETURN D46F JSR &D49B ;else negate Y/A D472 STA &0305,X ;store back again D475 PHA ; D476 TYA ; D477 STA &0304,X ; D47A PLA ;get back A TRISBB D47B RTS ;and exit ; COPY8 D47C LDA #&08 ;A=8 D47E BNE &D48C ;copy 8 bytes COPY2H D480 LDY #&30 ;Y=&30 COPY2 D482 LDA #&02 ;A=2 D484 BNE &D48C ;copy 2 bytes COPYFA D486 LDY #&28 ;copy 4 bytes from 324/7 to 328/B COPYFB D488 LDX #&24 ; COPYF D48A LDA #&04 ; ***********copy A bytes from 300,X to 300,Y *************************** COPY D48C STA &DA ; COPYA D48E LDA &0300,X ; D491 STA &0300,Y ; D494 INX ; D495 INY ; D496 DEC &DA ; D498 BNE &D48E ; D49A RTS ;and return ************* negation routine ****************************************** NEGAY D49B PHA ;save A D49C TYA ;A=Y D49D EOR #&FF ;invert D49F TAY ;Y=A D4A0 PLA ;get back A D4A1 EOR #&FF ;invert D4A3 INY ;Y=Y+1 D4A4 BNE &D4A9 ;if not 0 exit D4A6 CLC ;else D4A7 ADC #&01 ;add 1 to A NEGAYA D4A9 RTS ;return ; SPF D4AA JSR &D85D ;check window boundaries and set up screen pointer D4AD BNE &D4B7 ;if A<>0 D4B7 D4AF LDA (&D6),Y ;else get byte from current graphics cell D4B1 EOR &035A ;compare with current background colour D4B4 STA &DA ;store it D4B6 RTS ;and RETURN SPFA D4B7 PLA ;get back return link D4B8 PLA ; SPLOTY D4B9 INC &0326 ;increment current graphics cursor vertical lo D4BC JMP &D545 ; OS SERIES IV GEOFF COX ************************************************************************* * * * LATERAL FILL ROUTINE * * * ************************************************************************* SPLOT D4BF JSR &D4AA ;check current screen state D4C2 AND &D1 ;if A and &D1 <> 0 a plotted point has been found D4C4 BNE &D4B9 ;so D4B9 D4C6 LDX #&00 ;X=0 D4C8 JSR &D592 ;update pointers D4CB BEQ &D4FA ;if 0 then D4FA D4CD LDY &031A ;else Y=graphics scan line D4D0 ASL &D1 ; D4D2 BCS &D4D9 ;if carry set D4D9 D4D4 JSR &D574 ;else D574 D4D7 BCC &D4FA ;if carry clear D4FA SPLOTB D4D9 JSR &D3FD ;else D3FD to pick up colour multiplier D4DC LDA (&D6),Y ;get graphics cell line D4DE EOR &035A ;EOR with background colour D4E1 STA &DA ;and store D4E3 BNE &D4F7 ;if not 0 D4F7 D4E5 SEC ;else set carry D4E6 TXA ;A=X D4E7 ADC &0361 ;add pixels/byte D4EA BCC &D4F0 ;and if carry clear D4F0 D4EC INC &DB ;else increment &DB D4EE BPL &D4F7 ;and if +ve D4F7 SPLOTC D4F0 TAX ;else X=A D4F1 JSR &D104 ;display a pixel D4F4 SEC ;set carry D4F5 BCS &D4D9 ;goto D4D9 SPLOTD D4F7 JSR &D574 ; SPLOTE D4FA LDY #&00 ;Y=0 D4FC JSR &D5AC ; D4FF LDY #&20 ; D501 LDX #&24 ; D503 JSR &CDE6 ;exchange 300/3 +Y with 300/3+X SCHEAT D506 JSR &D4AA ;check screen pixel D509 LDX #&04 ;Y=5 D50B JSR &D592 ; D50E TXA ;A=x D50F BNE &D513 ;if A<>0 d513 D511 DEC &DB ;else &DB=&dB-1 SPLOTA D513 DEX ;X=X-1 SPLOTL D514 JSR &D54B ; D517 BCC &D540 ; SPLOTI D519 JSR &D3ED ;update pointers D51C LDA (&D6),Y ;get byte from graphics line D51E EOR &035A ;EOR with background colour D521 STA &DA ;and store it D523 LDA &DC ; D525 BNE &D514 ;If A-0 back to D514 D527 LDA &DA ;else A=&DA D529 BNE &D53D ;if A<>d53D D52B SEC ;else set carry D52C TXA ;A=x D52D ADC &0361 ;Add number of pixels/byte D530 BCC &D536 ;and if carry clear D536 D532 INC &DB ;else inc DB D534 BPL &D53D ;and if +ve D53D SPLOTF D536 TAX ;get back X D537 JSR &D104 ;display a point D53A SEC ;set carry D53B BCS &D519 ;goto D519 SPLOTJ D53D JSR &D54B ; SPLOTK D540 LDY #&04 ; D542 JSR &D5AC ; SPLOTX D545 JSR &D0D9 ; D548 JMP &D1B8 ;scale pointers SPA D54B LDA &D1 ;get byte mask D54D PHA ;save it D54E CLC ;clear carry D54F BCC &D560 ; SPAA D551 PLA ;get back A D552 INX ;X=X+1 D553 BNE &D559 ;if not 0 D559 D555 INC &DB ;else inc &DB D557 BPL &D56F ;if +ve D56F SPAB D559 LSR &D1 ; D55B BCS &D56F ;if Bit 7 D1 set D56F D55D ORA &D1 ;else or withA D55F PHA ;save result SPAF D560 LDA &D1 ;A=&D1 D562 BIT &DA ;test bits 6 and 7 of &DA D564 PHP ;save flags D565 PLA ;get into A D566 EOR &DC ;EOR and DC D568 PHA ;save A D569 PLP ; D56A BEQ &D551 ; D56C PLA ;A=A EOR &D1 (byte mask) D56D EOR &D1 ; SPAE D56F STA &D1 ;store it D571 JMP &D0F0 ;and display a pixel SPB D574 LDA #&00 ;A=0 D576 CLC ;Clear carry D577 BCC &D583 ;goto D583 if carry clear SPBA D579 INX ;X=X+1 D57A BNE &D580 ;If <>0 D580 D57C INC &DB ;else inc &DB D57E BPL &D56F ;and if +ve d56F SPBB D580 ASL ;A=A*2 D581 BCS &D58E ;if C set D58E SPBF D583 ORA &D1 ;else A=A OR (&D1) D585 BIT &DA ;set V and M from &DA b6 b7 D587 BEQ &D579 ; D589 EOR &D1 ;A=A EOR &D1 D58B LSR ;/2 D58C BCC &D56F ;if carry clear D56F SPBC D58E ROR ;*2 D58F SEC ;set carry D590 BCS &D56F ;to D56F SPS D592 LDA &0300,X ;Y/A=(&300/1 +X)-(&320/1) D595 SEC ; D596 SBC &0320 ; D599 TAY ; D59A LDA &0301,X ; D59D SBC &0321 ; D5A0 BMI &D5A5 ;if result -ve D5A5 D5A2 JSR &D49B ;or negate Y/A SPSA D5A5 STA &DB ;store A D5A7 TYA ;A=Y D5A8 TAX ;X=A D5A9 ORA &DB ; D5AB RTS ;exit SPU D5AC STY &DA ;Y=&DA D5AE TXA ;A=X D5AF TAY ;Y=A D5B0 LDA &DB ;A=&DB D5B2 BMI &D5B6 ;if -ve D5B6 D5B4 LDA #&00 ;A=0 SPUA D5B6 LDX &DA ;X=&DA D5B8 BNE &D5BD ;if <>0 D5BD D5BA JSR &D49B ;negate SPUB D5BD PHA ; D5BE CLC ; D5BF TYA ; D5C0 ADC &0300,X ;Y/A+(&300/1 +X)=(&320/1) D5C3 STA &0320 ; D5C6 PLA ; D5C7 ADC &0301,X ; D5CA STA &0321 ; D5CD RTS ;return ************************************************************************* * * * OSWORD 13 - READ LAST TWO GRAPHIC CURSOR POSITIONS * * * ************************************************************************* ; CSREAD D5CE LDA #&03 ;A=3 D5D0 JSR &D5D5 ; D5D3 LDA #&07 ;A=7 SPCA D5D5 PHA ;Save A D5D6 JSR &CDE2 ;exchange last 2 graphics cursor coordinates with ;current coordinates D5D9 JSR &D1B8 ;convert to external coordinates D5DC LDX #&03 ;X=3 D5DE PLA ;save A D5DF TAY ;Y=A SPCB D5E0 LDA &0310,X ;get graphics coordinate D5E3 STA (&F0),Y ;store it in OS buffer D5E5 DEY ;decrement Y and X D5E6 DEX ; D5E7 BPL &D5E0 ;if +ve do it again D5E9 RTS ;then Exit ************************************************************************* * * * PLOT Fill triangle routine * * * ************************************************************************* EMC D5EA LDX #&20 ;X=&20 D5EC LDY #&3E ;Y=&3E D5EE JSR &D47C ;copy 300/7+X to 300/7+Y ;this gets XY data parameters and current graphics ;cursor position D5F1 JSR &D632 ;exchange 320/3 with 324/7 if 316/7=<322/3 D5F4 LDX #&14 ;X=&14 D5F6 LDY #&24 ;Y=&24 D5F8 JSR &D636 ; D5FB JSR &D632 ; D5FE LDX #&20 ; D600 LDY #&2A ; D602 JSR &D411 ;calculate 032A/B-(324/5-320/1) D605 LDA &032B ;and store D608 STA &0332 ;result D60B LDX #&28 ;set pointers D60D JSR &D459 ; D610 LDY #&2E ; D612 JSR &D0DE ;copy 320/3 32/31 D615 JSR &CDE2 ;exchange 314/7 with 324/7 D618 CLC ; D619 JSR &D658 ;execute fill routine D61C JSR &CDE2 ; D61F LDX #&20 ; D621 JSR &CDE4 ; D624 SEC ; D625 JSR &D658 ; D628 LDX #&3E ;;X=&3E D62A LDY #&20 ;;Y=&20 D62C JSR &D47C ;;copy 300/7+X to 300/7+Y D62F JMP &D0D9 ;;this gets XY data parameters and current graphics ;cursor position SORTA D632 LDX #&20 ;X=&20 D634 LDY #&14 ;Y=&14 SORTY D636 LDA &0302,X ; D639 CMP &0302,Y ; D63C LDA &0303,X ; D63F SBC &0303,Y ; D642 BMI &D657 ;if 302/3+Y>302/3+X return D644 JMP &CDE6 ;else swap 302/3+X with 302/3+Y ************************************************************************* * * * OSBYTE 134 - READ CURSOR POSITION * * * ************************************************************************* CSRPOS D647 LDA &0318 ;read current text cursor (X) D64A SEC ;set carry D64B SBC &0308 ;subtract left hand column of current text window D64E TAX ;X=A D64F LDA &0319 ;get current text cursor (Y) D652 SEC ; D653 SBC &030B ;suptract top row of current window D656 TAY ;Y=A TRSRTS D657 RTS ;and exit ;PLOT routines continue ;many of the following routines are just manipulations ;only points of interest will be explained TRSF D658 PHP ;store flags D659 LDX #&20 ;X=&20 D65B LDY #&35 ;Y=&35 D65D JSR &D411 ;335/6=(324/5+X-320/1) D660 LDA &0336 ; D663 STA &033D ; D666 LDX #&33 ; D668 JSR &D459 ;set pointers D66B LDY #&39 ;set 339/C=320/3 D66D JSR &D0DE ; D670 SEC ; D671 LDA &0322 ; D674 SBC &0326 ; D677 STA &031B ; D67A LDA &0323 ; D67D SBC &0327 ; D680 STA &031C ; D683 ORA &031B ;check VDU queque D686 BEQ &D69F ; TRSFB D688 JSR &D6A2 ;display a line D68B LDX #&33 ; D68D JSR &D774 ;update pointers D690 LDX #&28 ; D692 JSR &D774 ;and again! D695 INC &031B ;update VDU queque D698 BNE &D688 ;and if not empty do it again D69A INC &031C ;else increment next byte D69D BNE &D688 ;and do it again TRSFA D69F PLP ;pull flags D6A0 BCC &D657 ;if carry clear exit HPLOT D6A2 LDX #&39 ; D6A4 LDY #&2E ; HPLOTA D6A6 STX &DE ; D6A8 LDA &0300,X ;is 300/1+x<300/1+Y D6AB CMP &0300,Y ; D6AE LDA &0301,X ; D6B1 SBC &0301,Y ; D6B4 BMI &D6BC ;if so D6BC D6B6 TYA ;else A=Y D6B7 LDY &DE ;Y=&DE D6B9 TAX ;X=A D6BA STX &DE ;&DE=X HPLOTB D6BC STY &DF ;&DF=Y D6BE LDA &0300,Y ; D6C1 PHA ; D6C2 LDA &0301,Y ; D6C5 PHA ; D6C6 LDX &DF ; D6C8 JSR &D10F ;check for window violations D6CB BEQ &D6DA ; D6CD CMP #&02 ; D6CF BNE &D70E ; D6D1 LDX #&04 ; D6D3 LDY &DF ; D6D5 JSR &D482 ; D6D8 LDX &DF ; HPLOTC D6DA JSR &D864 ;set a screen address D6DD LDX &DE ;X=&DE D6DF JSR &D10F ;check for window violations D6E2 LSR ;A=A/2 D6E3 BNE &D70E ;if A<>0 then exit D6E5 BCC &D6E9 ;else if C clear D6E9 D6E7 LDX #&00 ; HPLOTD D6E9 LDY &DF ; D6EB SEC ; D6EC LDA &0300,Y ; D6EF SBC &0300,X ; D6F2 STA &DC ; D6F4 LDA &0301,Y ; D6F7 SBC &0301,X ; D6FA STA &DD ; D6FC LDA #&00 ; HPLOTG D6FE ASL ; D6FF ORA &D1 ; D701 LDY &DC ; D703 BNE &D719 ; D705 DEC &DD ; D707 BPL &D719 ; D709 STA &D1 ; D70B JSR &D0F0 ;display a point HPLOTX D70E LDX &DF ;restore X D710 PLA ;and A D711 STA &0301,X ;store it D714 PLA ;get back A D715 STA &0300,X ;and store it D718 RTS ;exit ; HPLOTE D719 DEC &DC ; D71B TAX ; D71C BPL &D6FE ; D71E STA &D1 ; D720 JSR &D0F0 ;display a point D723 LDX &DC ; D725 INX ; D726 BNE &D72A ; D728 INC &DD ; HPLOTL D72A TXA ; D72B PHA ; D72C LSR &DD ; D72E ROR ; D72F LDY &0361 ;number of pixels/byte D732 CPY #&03 ;if 3 mode = goto D73B D734 BEQ &D73B ; D736 BCC &D73E ;else if <3 mode 2 goto D73E D738 LSR &DD ;else rotate bottom bit of &DD D73A ROR ;into Accumulator HPLOTP D73B LSR &DD ;rotate bottom bit of &DD D73D LSR ;into Accumulator HPLOTQ D73E LDY &031A ;Y=line in current graphics cell containing current ;point D741 TAX ;X=A D742 BEQ &D753 ; HPLOTI D744 TYA ;Y=Y-8 D745 SEC ; D746 SBC #&08 ; D748 TAY ; D749 BCS &D74D ; D74B DEC &D7 ;decrement byte of top line off current graphics cell HPLOTH D74D JSR &D104 ;display a point D750 DEX ; D751 BNE &D744 ; HPLOTF D753 PLA ; D754 AND &0361 ;pixels/byte D757 BEQ &D70E ; D759 TAX ; D75A LDA #&00 ;A=0 HPLOTJ D75C ASL ; D75D ORA &0363 ;or with right colour mask D760 DEX ; D761 BNE &D75C ; D763 STA &D1 ;store as byte mask D765 TYA ;Y=Y-8 D766 SEC ; D767 SBC #&08 ; D769 TAY ; D76A BCS &D76E ;if carry clear D76C DEC &D7 ;decrement byte of top line off current graphics cell HPLOTK D76E JSR &D0F3 ;display a point D771 JMP &D70E ;and exit via D70E ULN D774 INC &0308,X ; D777 BNE &D77C ; D779 INC &0309,X ; ULNA D77C SEC ; D77D LDA &0300,X ; D780 SBC &0302,X ; D783 STA &0300,X ; D786 LDA &0301,X ; D789 SBC &0303,X ; D78C STA &0301,X ; D78F BPL &D7C1 ; ULNF D791 LDA &030A,X ; D794 BMI &D7A1 ; D796 INC &0306,X ; D799 BNE &D7AC ; D79B INC &0307,X ; D79E JMP &D7AC ; ULNC D7A1 LDA &0306,X ; D7A4 BNE &D7A9 ; D7A6 DEC &0307,X ; ULNE D7A9 DEC &0306,X ; ULND D7AC CLC ; D7AD LDA &0300,X ; D7B0 ADC &0304,X ; D7B3 STA &0300,X ; D7B6 LDA &0301,X ; D7B9 ADC &0305,X ; D7BC STA &0301,X ; D7BF BMI &D791 ; ULNB D7C1 RTS ; ************************************************************************* * * * OSBYTE 135 - READ CHARACTER AT TEXT CURSOR POSITION * * * ************************************************************************* READCH D7C2 LDY &0360 ;get number of logical colours D7C5 BNE &D7DC ;if Y<>0 mode <>7 so D7DC D7C7 LDA (&D8),Y ;get character in current character cell D7C9 LDY #&02 ;Y=2 RDTTXA D7CB CMP &C4B7,Y ;compare with conversion table D7CE BNE &D7D4 ;if not equal D7D4 D7D0 LDA &C4B6,Y ;else get next lower byte from table D7D3 DEY ;Y=Y-1 RDTTXZ D7D4 DEY ;Y=Y-1 D7D5 BPL &D7CB ;and if +ve do it again READKE D7D7 LDY &0355 ;Y=current screen mode D7DA TAX ;return with character in X, setting EQ/NE D7DB RTS ; ; READCE D7DC JSR &D808 ;set up copy of the pattern bytes at text cursor D7DF LDX #&20 ;X=&20 READKF D7E1 TXA ;A=&20 D7E2 PHA ;Save it D7E3 JSR &D03E ;get pattern address for code in A D7E6 PLA ;get back A D7E7 TAX ;and X READKA D7E8 LDY #&07 ;Y=7 READKB D7EA LDA &0328,Y ;get byte in pattern copy D7ED CMP (&DE),Y ;check against pattern source D7EF BNE &D7F9 ;if not the same D7F9 D7F1 DEY ;else Y=Y-1 D7F2 BPL &D7EA ;and if +ve D7EA D7F4 TXA ;A=X D7F5 CPX #&7F ;is X=&7F (delete) D7F7 BNE &D7D7 ;if not D7D7 READKC D7F9 INX ;else X=X+1 D7FA LDA &DE ;get byte lo address D7FC CLC ;clear carry D7FD ADC #&08 ;add 8 D7FF STA &DE ;store it D801 BNE &D7E8 ;and go back to check next character if <>0 D803 TXA ;A=X D804 BNE &D7E1 ;if <>0 D7E1 D806 BEQ &D7D7 ;else D7D7 ***************** set up pattern copy *********************************** PTCNV D808 LDY #&07 ;Y=7 PTCNVA D80A STY &DA ;&DA=Y D80C LDA #&01 ;A=1 D80E STA &DB ;&DB=A PTCNVB D810 LDA &0362 ;A=left colour mask D813 STA &DC ;store an &DC D815 LDA (&D8),Y ;get a byte from current text character D817 EOR &0358 ;EOR with text background colour D81A CLC ;clear carry PTCNVC D81B BIT &DC ;and check bits of colour mask D81D BEQ &D820 ;if result =0 then D820 D81F SEC ;else set carry PTCNVD D820 ROL &DB ;&DB=&DB+Carry D822 BCS &D82E ;if carry now set (bit 7 DB originally set) D82E D824 LSR &DC ;else &DC=&DC/2 D826 BCC &D81B ;if carry clear D81B D828 TYA ;A=Y D829 ADC #&07 ;ADD ( (7+carry) D82B TAY ;Y=A D82C BCC &D810 ; PTCNVE D82E LDY &DA ;read modified values into Y and A D830 LDA &DB ; D832 STA &0328,Y ;store copy D835 DEY ;and do it again D836 BPL &D80A ;until 8 bytes copied D838 RTS ;exit ********* pixel reading ************************************************* RPXLD D839 PHA ;store A D83A TAX ;X=A D83B JSR &D149 ;set up positional data D83E PLA ;get back A D83F TAX ;X=A D840 JSR &D85F ;set a screen address after checking for window ;violations D843 BNE &D85A ;if A<>0 D85A to exit with A=&FF D845 LDA (&D6),Y ;else get top line of current graphics cell RPXLA D847 ASL ;A=A*2 C=bit 7 D848 ROL &DA ;&DA=&DA+2 +C C=bit 7 &DA D84A ASL &D1 ;byte mask=bM*2 +carry from &DA D84C PHP ;save flags D84D BCS &D851 ;if carry set D851 D84F LSR &DA ;else restore &DA with bit '=0 RPXLB D851 PLP ;pull flags D852 BNE &D847 ;if Z set D847 D854 LDA &DA ;else A=&DA AND number of colours in current mode -1 D856 AND &0360 ; D859 RTS ;then exit ; RPXLC D85A LDA #&FF ;A=&FF RPXLX D85C RTS ;exit ********** check for window violations and set up screen address ********** GADDW D85D LDX #&20 ;X=&20 GADDV D85F JSR &D10F ; D862 BNE &D85C ;if A<>0 there is a window violation so D85C GADDR D864 LDA &0302,X ;else set up graphics scan line variable D867 EOR #&FF ; D869 TAY ; D86A AND #&07 ; D86C STA &031A ;in 31A D86F TYA ;A=Y D870 LSR ;A=A/2 D871 LSR ;A=A/2 D872 LSR ;A=A/2 D873 ASL ;A=A*2 this gives integer value bit 0 =0 D874 TAY ;Y=A D875 LDA (&E0),Y ;get high byte of offset from screen RAM start D877 STA &DA ;store it D879 INY ;Y=Y+1 D87A LDA (&E0),Y ;get lo byte D87C LDY &0356 ;get screen map type D87F BEQ &D884 ;if 0 (modes 0,1,2) goto D884 D881 LSR &DA ;else &DA=&DA/2 D883 ROR ;and A=A/2 +C if set ;so 2 byte offset =offset/2 GADDRD D884 ADC &0350 ;add screen top left hand corner lo D887 STA &D6 ;store it D889 LDA &DA ;get high byte D88B ADC &0351 ;add top left hi D88E STA &D7 ;store it D890 LDA &0301,X ; D893 STA &DA ; D895 LDA &0300,X ; D898 PHA ; D899 AND &0361 ;and then Add pixels per byte-1 D89C ADC &0361 ; D89F TAY ;Y=A D8A0 LDA &C406,Y ;A=&80 /2^Y using look up table D8A3 STA &D1 ;store it D8A5 PLA ;get back A D8A6 LDY &0361 ;Y=&number of pixels/byte D8A9 CPY #&03 ;is Y=3 (modes 1,6) D8AB BEQ &D8B2 ;goto D8B2 D8AD BCS &D8B5 ;if mode =1 or 4 D8B5 D8AF ASL ;A/&DA =A/&DA *2 D8B0 ROL &DA ; GADDRA D8B2 ASL ; D8B3 ROL &DA ; GADDRB D8B5 AND #&F8 ;clear bits 0-2 D8B7 CLC ;clear carry D8B8 ADC &D6 ;add A/&DA to &D6/7 D8BA STA &D6 ; D8BC LDA &DA ; D8BE ADC &D7 ; D8C0 BPL &D8C6 ;if result +ve D8C6 D8C2 SEC ;else set carry D8C3 SBC &0354 ;and subtract screen memory size making it wrap round GADDRC D8C6 STA &D7 ;store it in &D7 D8C8 LDY &031A ;get line in graphics cell containing current graphics COPYZX D8CB LDA #&00 ;point A=0 D8CD RTS ;And exit ; VDUB D8CE PHA ;Push A D8CF LDA #&A0 ;A=&A0 D8D1 LDX &026A ;X=number of items in VDU queue D8D4 BNE &D916 ;if not 0, D916 D8D6 BIT &D0 ;else check VDU status byte D8D8 BNE &D916 ;if either VDU is disabled or plot to graphics ;cursor enabled then D916 D8DA BVS &D8F5 ;if cursor editing enabled D8F5 D8DC LDA &035F ;else get 6845 register start setting D8DF AND #&9F ;clear bits 5 and 6 D8E1 ORA #&40 ;set bit 6 to modify last cursor size setting D8E3 JSR &C954 ;change write cursor format D8E6 LDX #&18 ;X=&18 D8E8 LDY #&64 ;Y=&64 D8EA JSR &D482 ;set text input cursor from text output cursor D8ED JSR &CD7A ;modify character at cursor poistion D8F0 LDA #&02 ;A=2 D8F2 JSR &C59D ;bit 1 of VDU status is set to bar scrolling VDUBA D8F5 LDA #&BF ;A=&BF D8F7 JSR &C5A8 ;bit 6 of VDU status =0 D8FA PLA ;Pull A D8FB AND #&7F ;clear hi bit (7) D8FD JSR &C4C0 ;entire VDU routine !! D900 LDA #&40 ;A=&40 D902 JMP &C59D ;exit COPYCH D905 LDA #&20 ;A=&20 D907 BIT &D0 ;if bit 6 cursor editing is set D909 BVC &D8CB ; D90B BNE &D8CB ;or bit 5 is set exit &D8CB D90D JSR &D7C2 ;read a character from the screen D910 BEQ &D917 ;if A=0 on return exit via D917 D912 PHA ;else store A D913 JSR &C664 ;perform cursor right VDUBEX D916 PLA ;restore A COPYCX D917 RTS ;and exit ; CSEDEX D918 LDA #&BD ;zero bits 2 and 6 of VDU status D91A JSR &C5A8 ; D91D JSR &C951 ;set normal cursor D920 LDA #&0D ;A=&0D D922 RTS ;and return ;this is response of CR as end of edit line ************************************************************************* * * * OSBYTE 132 - READ BOTTOM OF DISPLAY RAM * * * ************************************************************************* TOPCMD D923 LDX &0355 ; Get current screen mode ************************************************************************* * * * OSBYTE 133 - READ LOWEST ADDRESS FOR GIVEN MODE * * * ************************************************************************* TOPMDX D926 TXA ; A=X D927 AND #&07 ; Ensure mode 0-7 D929 TAY ; Pass to Y into index into screen size table D92A LDX &C440,Y ; X=screen size type, 0-4 D92D LDA &C45E,X ; A=high byte of start address for screen type D930 LDX #&00 ; Returned address is &xx00 D932 BIT &028E ; Check available RAM D935 BMI &D93E ; If bit 7 set then 32K RAM, so return address D937 AND #&3F ; 16K RAM, so drop address to bottom 16K D939 CPY #&04 ; Check screen mode D93B BCS &D93E ; If mode 4-7, return the address D93D TXA ; If mode 0-3, return &0000 as not enough memory ; exit PRBB1 D93E TAY ; Pass high byte of address to Y D93F RTS ; and return address in YX ************************************************************************* ************************************************************************* ** ** ** SYSTEM STARTUP ** ** ** ************************************************************************* ************************************************************************* * DEFAULT PAGE &02 SETTINGS (VECTORS, OSBYTE VARIABLES) * RESET CODE ************************************************************************* * * * DEFAULT SYSTEM SETTINGS FOR PAGE &02 * * * ************************************************************************* ------------------------------------------------------------------------- | | | DEFAULT VECTOR TABLE | | | ------------------------------------------------------------------------- IPAGE0* IPAGE2 ENDVDU ORIGIN D940 DW &E310 ; USERV &200 D942 DW &DC54 ; BRKV &202 D944 DW &DC93 ; IRQ1V &204 D946 DW &DE89 ; IRQ2V &206 D948 DW &DF89 ; CLIV &208 D94A DW &E772 ; BYTEV &20A D94C DW &E7EB ; WORDV &20C D94E DW &E0A4 ; WRCHV &20E D950 DW &DEC5 ; RDCHV &210 CFSTAB D952 DW &F27D ; FILEV &212 D954 DW &F18E ; ARGSV &214 D956 DW &F4C9 ; BGETV &216 D958 DW &F529 ; BPUTV &218 D95A DW &FFA6 ; GBPBV &21A D95C DW &F3CA ; FINDV &21C D95E DW &F1B1 ; FSCV &21E D960 DW &FFA6 ; EVNTV &220 D962 DW &FFA6 ; UPTV &222 D964 DW &FFA6 ; NETV &224 D966 DW &FFA6 ; VDUV &226 D968 DW &EF02 ; KEYV &228 D96A DW &E4B3 ; INSBV &22A D96C DW &E464 ; REMVB &22C D96E DW &E1D1 ; CNPV &22E D970 DW &FFA6 ; IND1V &230 D972 DW &FFA6 ; IND2V &232 D974 DW &FFA6 ; IND3V &234 ------------------------------------------------------------------------- | | | DEFAULT MOS VARIABLES SETTINGS | | | ------------------------------------------------------------------------- * Read/Written by OSBYTE &A6 to &FC D976 DW &0190 ; OSBYTE variables base address &236 *FX166/7 ; (address to add to osbyte number) D978 DB &0D9F ; Address of extended vectors &238 *FX168/9 D97A DB &02A1 ; Address of ROM information table &23A *FX170/1 D97C DB &F02B ; Address of key translation table &23C *FX172/3 D97E DB &0300 ; Address of VDU variables &23E *FX174/5 D980 DB &00 ; CFS/Vertical sync Timeout counter &240 *FX176 D981 DB &00 ; Current input buffer number &241 *FX177 D982 DB &FF ; Keyboard interrupt processing flag &242 *FX178 D983 DB &00 ; Primary OSHWM (default PAGE) &243 *FX179 D984 DB &00 ; Current OSHWM (PAGE) &244 *FX180 D985 DB &01 ; RS423 input mode &245 *FX181 D986 DB &00 ; Character explosion state &246 *FX182 D987 DB &00 ; CFS/RFS selection, CFS=0 ROM=2 &247 *FX183 D988 DB &00 ; Video ULA control register copy &248 *FX184 D989 DB &00 ; Palette setting copy &249 *FX185 D98A DB &00 ; ROM number selected at last BRK &24A *FX186 D98B DB &FF ; BASIC ROM number &24B *FX187 D98C DB &04 ; Current ADC channel number &24C *FX188 D98D DB &04 ; Maximum ADC channel number &24D *FX189 D98E DB &00 ; ADC conversion 0/8bit/12bit &24E *FX190 D98F DB &FF ; RS423 busy flag (bit 7=0, busy) &24F *FX191 D990 DB &56 ; ACIA control register copy &250 *FX192 D991 DB &19 ; Flash counter &251 *FX193 D992 DB &19 ; Flash mark period count &252 *FX194 D993 DB &19 ; Flash space period count &253 *FX195 D994 DB &32 ; Keyboard auto-repeat delay &254 *FX196 D995 DB &08 ; Keyboard auto-repeat rate &255 *FX197 D996 DB &00 ; *EXEC file handle &256 *FX198 D997 DB &00 ; *SPOOL file handle &257 *FX199 D998 DB &00 ; Break/Escape handing &258 *FX200 D999 DB &00 ; Econet keyboard disable flag &259 *FX201 D99A DB &20 ; Keyboard status &25A *FX202 ; bit 3=1 shift pressed ; bit 4=0 caps lock ; bit 5=0 shift lock ; bit 6=1 control bit ; bit 7=1 shift enabled D99B DB &09 ; Serial input buffer full threshold &25B *FX203 D99C DB &00 ; Serial input suppression flag &25C *FX204 D99D DB &00 ; Cassette/RS423 flag (0=CFS, &40=RS423) &25D *FX205 D99E DB &00 ; Econet OSBYTE/OSWORD interception flag &25E *FX206 D99F DB &00 ; Econet OSRDCH interception flag &25F *FX207 D9A0 DB &00 ; Econet OSWRCH interception flag &260 *FX208 D9A1 DB &50 ; Speech enable/disable flag (&20/&50) &261 *FX209 D9A2 DB &00 ; Sound output disable flag &262 *FX210 D9A3 DB &03 ; BELL channel number &263 *FX211 D9A4 DB &90 ; BELL amplitude/Envelope number &264 *FX212 D9A5 DB &64 ; BELL frequency &265 *FX213 D9A6 DB &06 ; BELL duration &266 *FX214 D9A7 DB &81 ; Startup message/!BOOT error status &267 *FX215 D9A8 DB &00 ; Length of current soft key string &268 *FX216 D9A9 DB &00 ; Lines printed since last paged halt &269 *FX217 D9AA DB &00 ; 0-(Number of items in VDU queue) &26A *FX218 D9AB DB &09 ; TAB key value &26B *FX219 D9AC DB &1B ; ESCAPE character &26C *FX220 ; The following are input buffer code interpretation variables for ; bytes entered into the input buffer with b7 set (is 128-255). ; The standard keyboard only enters characters &80-&BF with the ; function keys, but other characters can be entered, for instance ; via serial input of via other keyboard input systems. ; 0=ignore key ; 1=expand as soft key ; 2-FF add to base for ASCII code D9AD DB &01 ; C0-&CF &26D *FX221 D9AE DB &D0 ; D0-&DF &26E *FX222 D9AF DB &E0 ; E0-&EF &26F *FX223 D9B0 DB &F0 ; F0-&FF &270 *FX224 D9B1 DB &01 ; 80-&8F function key &271 *FX225 D9B2 DB &80 ; 90-&9F Shift+function key &272 *FX226 D9B3 DB &90 ; A0-&AF Ctrl+function key &273 *FX227 D9B4 DB &00 ; B0-&BF Shift+Ctrl+function key &274 *FX228 D9B5 DB &00 ; ESCAPE key status (0=ESC, 1=ASCII) &275 *FX229 D9B6 DB &00 ; ESCAPE action &276 *FX230 FFBYT D9B7 DB &FF ; USER 6522 Bit IRQ mask &277 *FX231 D9B8 DB &FF ; 6850 ACIA Bit IRQ bit mask &278 *FX232 D9B9 DB &FF ; System 6522 IRQ bit mask &279 *FX233 D9BA DB &00 ; Tube prescence flag &27A *FX234 D9BB DB &00 ; Speech processor prescence flag &27B *FX235 D9BC DB &00 ; Character destination status &27C *FX236 D9BD DB &00 ; Cursor editing status &27D *FX237 ****************** Soft Reset high water mark *************************** D9BE DB &00 ; unused &27E *FX238 D9BF DB &00 ; unused &27F *FX239 D9C0 DB &00 ; Country code &280 *FX240 D9C1 DB &00 ; User flag &281 *FX241 D9C2 DB &64 ; Serial ULA control register copy &282 *FX242 D9C3 DB &05 ; Current system clock state &283 *FX243 D9C4 DB &FF ; Soft key consitancy flag &284 *FX244 D9C5 DB &01 ; Printer destination &285 *FX245 D9C6 DB &0A ; Printer ignore character &286 *FX246 ****************** Hard Reset High water mark *************************** D9C7 DB &00 ; Break Intercept Vector JMP opcode &288 *FX247 D9C8 DB &00 ; Break Intercept Vector address low &288 *FX248 D9C9 DB &00 ; Break Intercept Vector address high &289 *FX249 D9CA DB &00 ; unused (memory used for VDU) &28A *FX250 D9CB DB &00 ; unused (memory used for display) &28B *FX251 D9CC DB &FF ; Current language ROM number &28C *FX252 ****************** Power-On Reset High Water mark *********************** ************************************************************************** ************************************************************************** ** ** ** RESET (BREAK) ENTRY POINT ** ** ** ** Power up Enter with nothing set, 6522 System VIA IER bits ** ** 0 to 6 will be clear ** ** ** ** BREAK IER bits 0 to 6 one or more will be set 6522 IER ** ** not reset by BREAK ** ** ** ************************************************************************** ************************************************************************** MRESET D9CD LDA #&40 ;set NMI first instruction to RTI D9CF STA &0D00 ;NMI ram start D9D2 SEI ;disable interrupts just in case D9D3 CLD ;clear decimal flag D9D4 LDX #&FF ;reset stack D9D6 TXS ;(&1FF) D9D7 LDA &FE4E ;read interupt enable register of the system VIA D9DA ASL ;shift bit 7 into carry D9DB PHA ;save what's left D9DC BEQ &D9E7 ;if Power up A=0 so D9E7 D9DE LDA &0258 ;else if BREAK pressed read BREAK Action flags (set by ;*FX200,n) D9E1 LSR ;divide by 2 D9E2 CMP #&01 ;if (bit 1 not set by *FX200) D9E4 BNE &DA03 ;then &DA03 D9E6 LSR ;divide A by 2 again (will always be A=0 as 1 DIV 2=0) ********** clear memory routine ****************************************** HRDRST D9E7 LDX #&04 ;get page to start clearance from (4) D9E9 STX &01 ;store it in ZP 01 D9EB STA &00 ;store A at 00 D9ED TAY ;and in Y to set loop counter CLEAR D9EE STA (&00),Y ;clear store D9F0 CMP &01 ;until address &01 =0 D9F2 BEQ &D9FD ; D9F4 INY ;increment pointer D9F5 BNE &D9EE ;if not zero loop round again D9F7 INY ;else increment again (Y=1) this avoids overwriting ;RTI instruction at &D00 D9F8 INX ;increment X D9F9 INC &01 ;increment &01 D9FB BPL &D9EE ;loop until A=&80 then exit ;note that RAM addressing for 16K loops around so ;&4000=&00 hence checking &01 for 00. This avoids ;overwriting zero page on BREAK CLEAR0 D9FD STX &028E ;writes marker for available RAM 40 =16K, 80=32K DA00 STX &0284 ;write soft key consistency flag **+********** set up system VIA ***************************************** SFTRST DA03 LDX #&0F ;set PORT B to output on bits 0-3 Input 4-7 DA05 STX &FE42 ; ************************************************************************* * * * set addressable latch IC 32 for peripherals via PORT B * * * * ;bit 3 set sets addressed latch high adds 8 to VIA address * * ;bit 3 reset sets addressed latch low * * * * Peripheral VIA bit 3=0 VIA bit 3=1 * * * * Sound chip Enabled Disabled * * speech chip (RS) Low High * * speech chip (WS) Low High * * Keyboard Auto Scan Disabled Enabled * * C0 address modifier Low High * * C1 address modifier Low High * * Caps lock LED ON OFF * * Shift lock LED ON OFF * * * * C0 & C1 are involved with hardware scroll screen address * ************************************************************************* ;X=&F on entry IALTCH DA08 DEX ;loop start DA09 STX &FE40 ;write latch IC32 DA0C CPX #&09 ;is it 9 DA0E BCS &DA08 ;if so go back and do it again ;X=8 at this point ;Caps lock On, SHIFT lock undetermined ;Keyboard Autoscan on ;sound disabled (may still sound) DA10 INX ;X=9 RDOPT1 DA11 TXA ;A=X DA12 JSR &F02A ;interrogate keyboard DA15 CPX #&80 ;for keyboard links 9-2 and CTRL key (1) DA17 ROR &FC ;rotate MSB into bit 7 of &FC DA19 TAX ;get back value of X for loop DA1A DEX ;decrement it DA1B BNE &DA11 ;and if >0 do loop again ; on exit if Carry set link 3 made ;link 2 = bit 0 of &FC and so on ;if CTRL pressed bit 7 of &FC=1 ;X=0 DA1D STX &028D ;clear last BREAK flag DA20 ROL &FC ;CTRL is now in carry &FC is keyboard links DA22 JSR &EEEB ;set LEDs carry on entry, CTRL=bit 0 of A on exit DA25 ROR ;get carry back into carry flag ****** set up page 2 **************************************************** DA26 LDX #&9C ; DA28 LDY #&8D ; DA2A PLA ;get back A from &D9DB DA2B BEQ &DA36 ;if A=0 power up reset so DA36 with X=&9C Y=&8D DA2D LDY #&7E ;else Y=&7E DA2F BCC &DA42 ;and if not CTRL-BREAK DA42 WARM RESET DA31 LDY #&87 ;else Y=&87 COLD RESET DA33 INC &028D ;&28D=1 POWER DA36 INC &028D ;&28D=&28D+1 DA39 LDA &FC ;get keyboard links set DA3B EOR #&FF ;invert DA3D STA &028F ;and store at &28F DA40 LDX #&90 ;X=&90 **********: set up page 2 ************************************************* ;on entry &28D=0 Warm reset, X=&9C, Y=&7E ;&28D=1 Power up , X=&90, Y=&8D ;&28D=2 Cold reset, X=&9C, Y=&87 SOFT DA42 LDA #&00 ;A=0 IP0SX DA44 CPX #&CE ;zero &200+X to &2CD DA46 BCC &DA4A ; DA48 LDA #&FF ;then set &2CE to &2FF to &FF Z0BYTE DA4A STA &0200,X ; DA4D INX ; DA4E BNE &DA44 ; ;A=&FF X=0 DA50 STA &FE63 ;set port A of user via to all outputs (printer out) DA53 TXA ;A=0 DA54 LDX #&E2 ;X=&E2 IP0SW DA56 STA &00,X ; Clear zeropage &E2 to &FF DA58 INX ; DA59 BNE &DA56 ; X is now &00 IP2SV DA5B LDA &D93F,Y ; Set up default vectors and OSBYTE variables DA5E STA &01FF,Y ; from the top set by Break type DA61 DEY ; DA62 BNE &DA5B ; DA64 LDA #&62 ; Set Last Key Pressed in &ED to &62 - SPACEBAR DA66 STA &ED ; DA68 JSR &FB0A ; Set up ACIA ;X=0 ************** clear interrupt and enable registers of Both VIAs ******** DA6B LDA #&7F ; DA6D INX ; IFRIER DA6E STA &FE4D,X ; DA71 STA &FE6D,X ; DA74 DEX ; DA75 BPL &DA6E ; DA77 CLI ;briefly allow interrupts to clear anything pending DA78 SEI ;disallow again NB, all VIA IRQs are disabled DA79 BIT &FC ;if bit 6=1 then JSR &F055 (normally 0) DA7B BVC &DA80 ;else DA80 DA7D JSR &F055 ;F055 JMP (&FDFE) probably causes a BRK unless ;hardware there redirects it. ; NOTST1 DA80 LDX #&F2 ;enable interrupts 1,4,5,6 of system VIA DA82 STX &FE4E ; ;0 Keyboard enabled as needed ;1 Frame sync pulse ;4 End of A/D conversion ;5 T2 counter (for speech) ;6 T1 counter (10 mSec intervals) ; DA85 LDX #&04 ;set system VIA PCR DA87 STX &FE4C ; ;CA1 to interrupt on negative edge (Frame sync) ;CA2 Handshake output for Keyboard ;CB1 interrupt on negative edge (end of conversion) ;CB2 Negative edge (Light pen strobe) ; DA8A LDA #&60 ;set system VIA ACR DA8C STA &FE4B ; ;disable latching ;disable shift register ;T1 counter continuous interrupts ;T2 counter timed interrupt DA8F LDA #&0E ;set system VIA T1 counter (Low) DA91 STA &FE46 ; ;this becomes effective when T1 hi set DA94 STA &FE6C ;set user VIA PCR ;CA1 interrupt on -ve edge (Printer Acknowledge) ;CA2 High output (printer strobe) ;CB1 Interrupt on -ve edge (user port) ;CB2 Negative edge (user port) DA97 STA &FEC0 ;set up A/D converter ;Bits 0 & 1 determine channel selected ;Bit 3=0 8 bit conversion bit 3=1 12 bit DA9A CMP &FE6C ;read user VIA IER if = &0E then DAA2 chip present DA9D BEQ &DAA2 ;so goto DAA2 DA9F INC &0277 ;else increment user VIA mask to 0 to bar all ;user VIA interrupts B6522E DAA2 LDA #&27 ;set T1 (hi) to &27 this sets T1 to &270E (9998 uS) DAA4 STA &FE47 ;or 10msec, interrupts occur every 10msec therefore DAA7 STA &FE45 ; DAAA JSR &EC60 ;clear the sound channels DAAD LDA &0282 ;read serial ULA control register DAB0 AND #&7F ;zero bit 7 DAB2 JSR &E6A7 ;and set up serial ULA DAB5 LDX &0284 ;get soft key status flag DAB8 BEQ &DABD ;if 0 (keys OK) then DABD DABA JSR &E9C8 ;else reset function keys ************************************************************************* * * * Check sideways ROMs and make ROM list * * * ************************************************************************* ;X=0 CRX SKOK DABD JSR &DC16 ;set up ROM latch and RAM copy to X DAC0 LDX #&03 ;set X to point to offset in table DAC2 LDY &8007 ;get copyright offset from ROM ; DF0C = ")C(",0 CRCPYR DAC5 LDA &8000,Y ;get first byte DAC8 CMP &DF0C,X ;compare it with table byte DACB BNE &DAFB ;if not the same then goto DAFB DACD INY ;point to next byte DACE DEX ;(s) DACF BPL &DAC5 ;and if still +ve go back to check next byte ;this point is reached if 5 bytes indicate valid ;ROM (offset +4 in (C) string) ************************************************************************* * Check first 1K of each ROM against higher priority ROMs to ensure that* * there are no duplicates, if duplicate found ignore lower priority ROM * ************************************************************************* DAD1 LDX &F4 ;get RAM copy of ROM No. in X DAD3 LDY &F4 ;and Y CRY DAD5 INY ;increment Y to check DAD6 CPY #&10 ;if ROM 15 is current ROM DAD8 BCS &DAFF ;if equal or more than 16 goto &DAFF ;to store catalogue byte DADA TYA ;else put Y in A DADB EOR #&FF ;invert it DADD STA &FA ;and store at &FA DADF LDA #&7F ;store &7F at DAE1 STA &FB ;&FB to get address &7FFF-Y CRLOOP DAE3 STY &FE30 ;set new ROM DAE6 LDA (&FA),Y ;Get byte DAE8 STX &FE30 ;switch back to previous ROM DAEB CMP (&FA),Y ;and compare with previous byte called DAED BNE &DAD5 ;if not the same then go back and do it again ;with next rom up DAEF INC &FA ;else increment &FA to point to new location DAF1 BNE &DAE3 ;if &FA<>0 then check next byte DAF3 INC &FB ;else inc &FB DAF5 LDA &FB ;and check that it doesn't exceed DAF7 CMP #&84 ;&84 (1K checked) DAF9 BCC &DAE3 ;then check next byte(s) CRBAD DAFB LDX &F4 ;X=(&F4) DAFD BPL &DB0C ;if +ve then &DB0C CRNX DAFF LDA &8006 ;get ROM type DB02 STA &02A1,X ;store it in ROM list DB05 AND #&8F ;check for BASIC (bit 7 not set) DB07 BNE &DB0C ;if not BASIC the DB0C DB09 STX &024B ;else store X in BASIC ROM number CRNX1 DB0C INX ;increment X to point to next ROM DB0D CPX #&10 ;is it 15 or less DB0F BCC &DABD ;if so goto &DABD for next ROM OS SERIES V GEOFF COX ************************************************************************* * * * Check SPEECH System * * * ************************************************************************* ;X=&10 DB11 BIT &FE40 ;if bit 7 low then we have speech system fitted DB14 BMI &DB27 ;else goto DB27 DB16 DEC &027B ;(027B)=&FF to indicate speech present SPRST1 DB19 LDY #&FF ;Y=&FF DB1B JSR &EE7F ;initialise speech generator DB1E DEX ;via this DB1F BNE &DB19 ;loop ;X=0 DB21 STX &FE48 ;set T2 timer for speech DB24 STX &FE49 ; *********** SCREEN SETUP ********************************************** ;X=0 START1 DB27 LDA &028F ;get back start up options (mode) DB2A JSR &C300 ;then jump to screen initialisation DB2D LDY #&CA ;Y=&CA DB2F JSR &E4F1 ;to enter this in keyboard buffer ;this enables the *KEY 10 facility ********* enter BREAK intercept with Carry Clear ************************ DB32 JSR &EAD9 ;check to see if BOOT address is set up, if so ;JMP to it DB35 JSR &F140 ;set up cassette options DB38 LDA #&81 ;test for tube to FIFO buffer 1 DB3A STA &FEE0 ; DB3D LDA &FEE0 ; DB40 ROR ;put bit 0 into carry DB41 BCC &DB4D ;if no tube then DB4D DB43 LDX #&FF ;else DB45 JSR &F168 ;issue ROM service call &FF ;to initialise TUBE system DB48 BNE &DB4D ;if not 0 on exit (Tube not initialised) DB4D DB4A DEC &027A ;else set tube flag to show it's active NOTUBE DB4D LDY #&0E ;set current value of PAGE DB4F LDX #&01 ;issue claim absolute workspace call DB51 JSR &F168 ;via F168 DB54 LDX #&02 ;send private workspace claim call DB56 JSR &F168 ;via F168 DB59 STY &0243 ;set primary OSHWM DB5C STY &0244 ;set current OSHWM DB5F LDX #&FE ;issue call for Tube to explode character set etc. DB61 LDY &027A ;Y=FF if tube present else Y=0 DB64 JSR &F168 ;and make call via F168 DB67 AND &0267 ;if A=&FE and bit 7 of 0267 is set then continue DB6A BPL &DB87 ;else ignore start up message DB6C LDY #&02 ;output to screen DB6E JSR &DEA9 ;'BBC Computer ' message DB71 LDA &028D ;0=warm reset, anything else continue DB74 BEQ &DB82 ; DB76 LDY #&16 ;by checking length of RAM DB78 BIT &028E ; DB7B BMI &DB7F ;and either DB7D LDY #&11 ; B DB7F JSR &DEA9 ;finishing message with '16K' or '32K' Z DB82 LDY #&1B ;and two newlines DB84 JSR &DEA9 ; *********: enter BREAK INTERCEPT ROUTINE WITH CARRY SET (call 1) ENTER DB87 SEC ; DB88 JSR &EAD9 ; Look for break intercept DB8B JSR &E9D9 ; Read ctrl/shift keys, M=Shift, C=Ctrl, A=%sxxxxxxx DB8E PHP ; Save flags, M=Shift DB8F PLA ; and get back in A DB90 LSR ; Move SHIFT state into bit 3 DB91 LSR ; DB92 LSR ; DB93 LSR ; (This could be done without PHP/PLA as A=%sxxxxxxx) DB94 EOR &028F ; EOR with start-up options which may or may not DB97 AND #&08 ; invert bit 3 DB99 TAY ;Y=A DB9A LDX #&03 ;make fs initialisation call, passing boot option in Y DB9C JSR &F168 ;Eg, RUN, EXEC or LOAD !BOOT file DB9F BEQ &DBBE ;if a ROM accepts this call then DBBE DBA1 TYA ;else put Y in A DBA2 BNE &DBB8 ;if Y<>0 DBB8 DBA4 LDA #&8D ;else set up standard cassette baud rates DBA6 JSR &F135 ;via &F135 DBA9 LDX #&D2 ; DBAB LDY #&EA ; DBAD DEC &0267 ;decrement ignore start up message flag DBB0 JSR OSCLI ;and attempto to execute */!BOOT DBB3 INC &0267 ;restore start up message flag DBB6 BNE &DBBE ;if not zero then DBBE EBRK NOAUTO DBB8 LDA #&00 ; Error from */!BOOT jumps back to here DBBA TAX ; A=0, X=0 DBBB JSR &F137 ; Set tape speed ******** Preserve current language on soft RESET ************************ SLANG DNRFS DBBE LDA &028D ;get last RESET Type DBC1 BNE &DBC8 ;if not soft reset DBC8 DBC3 LDX &028C ;else get current language ROM address DBC6 BPL &DBE6 ;if +ve (language available) then skip search routine ************************************************************************* * * * SEARCH FOR LANGUAGE TO ENTER (Highest priority) * * * ************************************************************************* SLANG1 DBC8 LDX #&0F ;set pointer to highest available rom ILR1 DBCA LDA &02A1,X ;get ROM type from ROM table DBCD ROL ;put high bit into carry, bit 6 into bit 7 DBCE BMI &DBE6 ;if bit 7 set then ROM has a language entry so DBE6 DBD0 DEX ;else search for language until X<0 DBD1 BPL &DBCA ; *************** check if tube present *********************************** DBD3 LDA #&00 ;if bit 7 of tube flag is set BMI succeeds DBD5 BIT &027A ;and TUBE is connected else DBD8 BMI &DC08 ;make error ********* no language error *********************************************** DBDA BRK ; DBDB DB &F9 ;error number DBDC DB 'Language?' ;message DBE5 BRK ; ILR2 DBE6 CLC ; CLC=Entering language from RESET ; ; Note that IRQs are disabled at this point, the language ; ; has to explicitly enable them. ************************************************************************* * * * OSBYTE 142 - ENTER LANGUAGE ROM AT &8000 * * * * X=ROM number, CS if OSBYTE call, CC if entered from RESET * * * ************************************************************************* SELANG DBE7 PHP ; Save flags DBE8 STX &028C ; Set as current language ROM DBEB JSR &DC16 ; Select the ROM DBEE LDA #&80 ; A=128 DBF0 LDY #&08 ; Y=8 DBF2 JSR &DEAB ; Display ROM title at &8009 DBF5 STY &FD ; Save Y on exit, points FAULT to string after title DBF7 JSR OSNEWL ; Two line feeds DBFA JSR OSNEWL ; are output DBFD PLP ; Get flags back DBFE LDA #&01 ; A=1 to indicate language entry DC00 BIT &027A ; Check if tube exists DC03 BMI &DC08 ; Of it is, skip to DC08 to start language via Tube DC05 JMP &8000 ; Else enter language at &8000 ************************************************************************* * * * TUBE FOUND, ENTER TUBE SOFTWARE * * * ************************************************************************* TLANG DC08 JMP &0400 ; Enter Tube code to start language ************************************************************************* * * * OSRDRM entry point * * * * get byte from PHROM or page ROM * * Y=ROM number, address is in &F6/7 * ************************************************************************* RPROM DC0B LDX &F4 ; Get current ROM number into X DC0D STY &F4 ; Store new number in &F4 DC0F STY &FE30 ; Switch in ROM DC12 LDY #&00 ; Y=0 to point to addressed byte DC14 LDA (&F6),Y ; Get byte from address ******** Select Sideways ROM ************************* ;on entry X=ROM number STXROM DC16 STX &F4 ; RAM copy of ROM latch DC18 STX &FE30 ; Write to ROM latch DC1B RTS ; And return ************************************************************************** ************************************************************************** ** ** ** MAIN IRQ Entry point ** ** ** ** ** ************************************************************************** ************************************************************************** ;ON ENTRY STACK contains STATUS REGISTER,PCH,PCL ; IRQ DC1C STA &FC ;save A DC1E PLA ;get back status (flags) DC1F PHA ;and save again DC20 AND #&10 ;check if BRK flag set DC22 BNE &DC27 ;if so goto DC27 DC24 JMP (&0204) ;else JMP (IRQ1V) ************************************************************************* * * * BRK handling routine * * * ************************************************************************* BRK DC27 TXA ;save X on stack DC28 PHA ; DC29 TSX ;get status pointer DC2A LDA &0103,X ;get Program Counter lo DC2D CLD ; DC2E SEC ;set carry DC2F SBC #&01 ;subtract 2 (1+carry) DC31 STA &FD ;and store it in &FD DC33 LDA &0104,X ;get hi byte DC36 SBC #&00 ;subtract 1 if necessary DC38 STA &FE ;and store in &FE DC3A LDA &F4 ;get currently active ROM DC3C STA &024A ;and store it in &24A DC3F STX &F0 ;store stack pointer in &F0 DC41 LDX #&06 ;and issue ROM service call 6 DC43 JSR &F168 ;(User BRK) to ROMs ;at this point &FD/E point to byte after BRK ;ROMS may use BRK for their own purposes DC46 LDX &028C ;get current language DC49 JSR &DC16 ;select ROM DC4C PLA ;get back original value of X DC4D TAX ; DC4E LDA &FC ;get back original value of A DC50 CLI ;allow interrupts DC51 JMP (&0202) ;and JUMP via BRKV (normally into current language) ************************************************************************* * * * DEFAULT BRK HANDLER * * * ************************************************************************* DEFBRK DC54 LDY #&00 ;Y=0 to point to byte after BRK DC56 JSR &DEB1 ;print message DC59 LDA &0267 ;if BIT 0 set and DISC EXEC error DC5C ROR ;occurs STOP DC5D BCS &DC5D ;hang up machine!!!! DC5F JSR OSNEWL ;else print two newlines DC62 JSR OSNEWL ; DC65 JMP &DBB8 ;and set tape speed before entering current ;language ; ACIA IRQ, RxRDY but both Serial and Printer buffers empty ; --------------------------------------------------------- TXDINT DC68 SEC DC69 ROR &024F ; Set b7 of RS423 busy flag DC6C BIT &0250 ;check bit 7 of current ACIA control register DC6F BPL &DC78 ;if interrupts NOT enabled DC78 DC71 JSR &E741 ;else E741 to check if serial buffer full DC74 LDX #&00 ; X=&00 to set RTS low DC76 BCS &DC7A ;if carry set goto DC7A to transfer data RLO DC78 LDX #&40 ; X=&40 to set RTS high RSOK DC7A JMP &E17A ; Jump to set ACIA control register ; Serial IRQ and RxRDY - Get byte and store in serial buffer ; ---------------------------------------------------------- RX68 DC7D LDY &FE09 ; Read data from ACIA DC80 AND #&3A ; Check PE:RO:FE:DCD DC82 BNE &DCB8 ; If any set, jump to generate Serial Error Event ; Serial IRQ and RxRDY, no errors ; ------------------------------- DC84 LDX &025C ; Read RS423 input suppression flag DC87 BNE &DC92 ; If not 0, jump to ignore DC89 INX ; X=1, serial input buffer DC8A JSR &E4F3 ; Put byte in buffer DC8D JSR &E741 ; Check if serial buffer almost full DC90 BCC &DC78 ; If almost full, jump to set RTS high NRSI DC92 RTS ; Return ************************************************************************* * * * Main IRQ Handling routines, default IRQ1V destination * * * ************************************************************************* DEFIRQ DC93 CLD ; Clear decimal flag DC94 LDA &FC ; Get original value of A DC96 PHA ; Save it DC97 TXA ; Save X DC98 PHA ; DC99 TYA ; and Y DC9A PHA ; DC9B LDA #&DE ; Stack return address to &DE82 DC9D PHA DC9E LDA #&81 DCA0 PHA DCA1 CLV ; Clear V flag IRQ68 DCA2 LDA &FE08 ; Read ACIA status register DCA5 BVS &DCA9 ; b6 set, jump with serial parity error DCA7 BPL &DD06 ; b7=0, no ACIA interrupt, jump to check VIAs ; ACIA Interrupt or ACIA Parity Error ; ----------------------------------- RSTIME DCA9 LDX &EA ; Get RS423 timeout counter DCAB DEX ; Decrement it DCAC BMI &DCDE ; If 0 or <0, RS423 owns 6850, jump to DCDE DCAE BVS &DCDD ; If &41..&80, nobody owns 6850, jump to exit DCB0 JMP &F588 ; CFS owns 6850, jump to read ACIA in CFS routines ; ACIA Data Carrier Detect ; ------------------------ MDCD DCB3 LDY &FE09 ; Read ACIA data DCB6 ROL A ; DCB7 ASL A ; Rotate ACIA Status back RX68BD DCB8 TAX ; X=ACIA Status DCB9 TYA ; A=ACIA Data DCBA LDY #&07 ; Y=07 for RS423 Error Event DCBC JMP &E494 ; Jump to issue event ; ACIA IRQ, TxRDY - Send a byte ; ----------------------------- TDRE DCBF LDX #&02 DCC1 JSR &E460 ; Read from Serial output buffer DCC4 BCC &DCD6 ; Buffer is not empty, jump to send byte DCC6 LDA &0285 ; Read printer destination DCC9 CMP #&02 ; Is it serial printer?? DCCB BNE &DC68 ; Serial buffer empty, not Serial printer, jump to ... DC68 DCCD INX ; X=3 for Printer buffer DCCE JSR &E460 ; Read from Printer buffer DCD1 ROR &02D2 ; Copy Byte Fetched/Not fetched into Printer Buffer full flag DCD4 BMI &DC68 ; Printer buffer was empty, so jump to ... DC68 TXCH DCD6 STA &FE09 ; Send byte to ACIA DCD9 LDA #&E7 ; Set timeout counter to &E7 DCDB STA &EA ; Serial owns 6850 for 103 more calls UNK0 RTS68 DCDD RTS ; Exit IRQ ; RS423 owns 6850, PE or RxRDY interupt occured ; --------------------------------------------- ; On entry, A contains ACIA status ; NOTCFS DCDE AND &0278 ; AND with ACIA IRQ mask (normally &FF) DCE1 LSR A ; Move RxRDY into Carry DCE2 BCC &DCEB ; If no RxData, jump to check DCD and TxRDY ; ; Data in RxData, check for errors ; DCE4 BVS &DCEB ; If IRQ=1 (now in b6) RxIRQ must have occured, so jump to DCEB ; ; RxData but no RxIRQ, check that IRQs are actually disabled ; DCE6 LDY &0250 ; Get ACIA control setting DCE9 BMI &DC7D ; If bit 7=1, IRQs enabled so jump to read byte and insert into buffer ; ; DCE9 -> RxData, no RxIRQ, IRQs disabled ; DCE4 -> RxData and RxIRQ ; DCE2 -> No RxData ; ; Check TxRDY and DCD, if neither set, send a Serial Error Event ; -------------------------------------------------------------- NOTRX DCEB LSR A ; Move TxRDY into Carry DCEC ROR A ; Rotate TxRDY into b7 and DCD into Carry DCED BCS &DCB3 ; If Data Carrier Detected, jump to DCB3 DCEF BMI &DCBF ; If TxRDY (now in b7) jump to to DCBF to send a byte DCF1 BVS &DCDD ; b6 should always be zero by now, but if set, then jump to exit ; Issue Unknown Interupt service call ; =================================== UNKIRQ DCF3 LDX #&05 DCF5 JSR &F168 ; Issue service call 5, 'Unknown Interrupt' DCF8 BEQ &DCDD ; If claimed, then jump to exit DCFA PLA ; Otherwise drop return address from stack DCFB PLA ; DCFC PLA ; And restore registers DCFD TAY ; DCFE PLA ; DCFF TAX ; DD00 PLA ; DD01 STA &FC ; Store A in IRQA DD03 JMP (&0206) ; And pass the IRQ in to IRQ2V ************************************************************************* * * * VIA INTERUPTS ROUTINES * * * ************************************************************************* IRQA DD06 LDA &FE4D ; Read System VIA interrupt flag register DD09 BPL &DD47 ; No System VIA interrupt, jump to check User VIA ; System VIA interupt ; DD0B AND &0279 ; Mask with System VIA bit mask DD0E AND &FE4E ; and interrupt enable register DD11 ROR ; Rotate to check for CA1 interupt (frame sync) DD12 ROR ; DD13 BCC &DD69 ; No CA1 (frame sync), jump to check speech ; System VIA CA1 interupt (Frame Sync) ; DD15 DEC &0240 ;decrement vertical sync counter DD18 LDA &EA ;A=RS423 Timeout counter DD1A BPL &DD1E ;if +ve then DD1E DD1C INC &EA ;else increment it NPRST DD1E LDA &0251 ;load flash counter DD21 BEQ &DD3D ;if 0 then system is not in use, ignore it DD23 DEC &0251 ;else decrement counter DD26 BNE &DD3D ;and if not 0 go on past reset routine DD28 LDX &0252 ;else get mark period count in X DD2B LDA &0248 ;current VIDEO ULA control setting in A DD2E LSR ;shift bit 0 into C to check if first colour DD2F BCC &DD34 ;is effective if so C=0 jump to DD34 DD31 LDX &0253 ;else get space period count in X FLASH1 DD34 ROL ;restore bit DD35 EOR #&01 ;and invert it DD37 JSR &EA00 ;then change colour DD3A STX &0251 ;&0251=X resetting the counter IVS1 DD3D LDY #&04 ;Y=4 and call E494 to check and implement vertical DD3F JSR &E494 ;sync event (4) if necessary DD42 LDA #&02 ;A=2 DD44 JMP &DE6E ;clear interrupt 1 and exit ************************************************************************* * * * PRINTER INTERRUPT USER VIA 1 * * * ************************************************************************* IRQB DD47 LDA &FE6D ; Read User VIA interrupt flag register DD4A BPL &DCF3 ; No User VIA interrupt, jump to pass to ROMs ; User VIA interupt ; ----------------- DD4C AND &0277 ;else check for USER IRQ 1 DD4F AND &FE6E ; DD52 ROR ; DD53 ROR ; DD54 BCC &DCF3 ;if bit 1=0 the no interrupt 1 so DCF3 IPRNT * DD56 LDY &0285 ;else get printer type DD59 DEY ;decrement DD5A BNE &DCF3 ;if not parallel then DCF3 DD5C LDA #&02 ;reset interrupt 1 flag DD5E STA &FE6D ; DD61 STA &FE6E ;disable interrupt 1 DD64 LDX #&03 ;and output data to parallel printer DD66 JMP &E13A ; ************************************************************************* * * * SYSTEM INTERRUPT 5 Speech * * * ************************************************************************* ITIME2 DD69 ROL ; Rotate bit 5 into bit 7 DD6A ROL ; DD6B ROL ; DD6C ROL ; DD6D BPL &DDCA ; Not a Timer 2 interrupt, jump to check timers ; System VIA Timer 2 interupt - Speech interupt ; --------------------------------------------- DD6F LDA #&20 ; Prepare to clear VIA interupt DD71 LDX #&00 DD73 STA &FE4D ; Clear VIA interupt DD76 STX &FE49 ; Zero high byte of T2 Timer PRSPRQ DD79 LDX #&08 ; X=8 for Speech buffer DD7B STX &FB ; Prepare to loop up to four times for Speak from RAM SPEREP DD7D JSR &E45B ; Examine Speech buffer DD80 ROR &02D7 ; Shift carry into bit 7 of Buffer Busy flag DD83 BMI &DDC9 ; Buffer empty, so exit, leaving Buffer Not Busy DD85 TAY ; Buffer not empty, A=first byte waiting DD86 BEQ &DD8D ; Waiting byte=&00 (Speak, no reset), skip past DD88 JSR &EE6D ; Read Speech Processor status DD8B BMI &DDC9 ; If bit 7 set, Speech Processor is talking, exit and try later SPOK DD8D JSR &E460 ; Fetch Speech command byte from buffer DD90 STA &F5 ; Store it DD92 JSR &E460 ; Fetch Speech word high byte from buffer DD95 STA &F7 ; Store it DD97 JSR &E460 ; Fetch Speech word low byte from buffer DD9A STA &F6 ; Store it, giving &F6/7=address to be accessed DD9C LDY &F5 ; Y=Speech command byte DD9E BEQ &DDBB ; SOUND &FF00 - Speak from RAM, no reset DDA0 BPL &DDB8 ; SOUND &FF01-&FF7F - Speak from RAM, with reset DDA2 BIT &F5 ; Check bit 6 of Speech command DDA4 BVS &DDAB ; SOUND &FFC0-&FFFF - Speak word number ; SOUND &FF80-&FFBF - Speak from absolute address ; &F5=command &80-&BF (b0-b3=PHROM number), &F6/7=address ; DDA6 JSR &EEBB ; Write address %xxrrrraaaaaaaaaaaaaa to speech processor ; ;\ command b5-b0---^ ^-----address DDA9 BVC &DDB2 ; Skip forward to speak from selected address ; ;\ SOUND &FFC0-&FFFF - Speak word number ; ;\ &F5=command &C0-&FF (b0-b3=PHROM number), &F6/7=word number ; ;\ Word number is 32 to 8191, 32-126 representing ASCII characters ; ;\ So, first word 32 is at PHROM address 64 ; RAB DDAB ASL &F6 ; Multiply address by 2 to index into word table DDAD ROL &F7 ; DDAF JSR &EE3B ; Read address from specified PHROM ; Speak from PHROM address ; By now, the address in the PHROM specified in Command b0-b3 has been set ; to the start of the speech data to be voiced. ; PHROM set from Command b0-b3, address set from absolute or lookup address ; ISPX DDB2 LDY &0261 ; Fetch command code, usually &50=Speak or &00=Nop DDB5 JMP &EE7F ; Jump to send command to speak from current address ; SOUND &FF01-&FF7F - Speak from RAM with reset ; Y=Speech command byte, &F6/7=Speech data ; Use SOUND &FF60 to send Speak External command ; ISP3 DDB8 JSR &EE7F ; Send command to Speech processor to start a new command ; &0x NOP ; &1x Read byte ; &2x NOP ; &3x Read and Branch ; &4a Load address ; &5x Speak ; &6x Speak external ; &7x Reset ; ; SOUND &FF00 - Speak from RAM without reset ; &6/7=Speech data ; ISP2 DDBB LDY &F6 DDBD JSR &EE7F ; Send Speech data low byte DDC0 LDY &F7 DDC2 JSR &EE7F ; Send Speech data high byte DDC5 LSR &FB ; Shift loop counter DDC7 BNE &DD7D ; Loop to send up to four byte-pairs NSPECH DDC9 RTS ************************************************************************* * * * SYSTEM INTERRUPT 6 10mS Clock * * * ************************************************************************* ITIME1 DDCA BCC &DE47 ;bit 6 is in carry so if clear there is no 6 int ;so go on to DE47 DDCC LDA #&40 ;Clear interrupt 6 DDCE STA &FE4D ; ;UPDATE timers routine, There are 2 timer stores &292-6 and &297-B ;these are updated by adding 1 to the current timer and storing the ;result in the other, the direction of transfer being changed each ;time of update. This ensures that at least 1 timer is valid at any call ;as the current timer is only read. Other methods would cause inaccuracies ;if a timer was read whilst being updated. DDD1 LDA &0283 ;get current system clock store pointer (5,or 10) DDD4 TAX ;put A in X DDD5 EOR #&0F ;and invert lo nybble (5 becomes 10 and vv) DDD7 PHA ;store A DDD8 TAY ;put A in Y ;Carry is always set at this point IT1 DDD9 LDA &0291,X ;get timer value DDDC ADC #&00 ;update it DDDE STA &0291,Y ;store result in alternate DDE1 DEX ;decrement X DDE2 BEQ &DDE7 ;if 0 exit DDE4 DEY ;else decrement Y DDE5 BNE &DDD9 ;and go back and do next byte IT2 DDE7 PLA ;get back A DDE8 STA &0283 ;and store back in clock pointer (i.e. inverse previous ;contents) DDEB LDX #&05 ;set loop pointer for countdown timer IVIAA5 DDED INC &029B,X ;increment byte and if DDF0 BNE &DDFA ;not 0 then DDFA DDF2 DEX ;else decrement pointer DDF3 BNE &DDED ;and if not 0 do it again DDF5 LDY #&05 ;process EVENT 5 interval timer DDF7 JSR &E494 ; IINKEY DDFA LDA &02B1 ;get byte of inkey countdown timer DDFD BNE &DE07 ;if not 0 then DE07 DDFF LDA &02B2 ;else get next byte DE02 BEQ &DE0A ;if 0 DE0A DE04 DEC &02B2 ;decrement 2B2 IIN1 DE07 DEC &02B1 ;and 2B1 IIN0 DE0A BIT &02CE ;read bit 7 of envelope processing byte DE0D BPL &DE1A ;if 0 then DE1A DE0F INC &02CE ;else increment to 0 DE12 CLI ;allow interrupts DE13 JSR &EB47 ;and do routine sound processes DE16 SEI ;bar interrupts DE17 DEC &02CE ;DEC envelope processing byte back to 0 SINUSE DE1A BIT &02D7 ;read speech buffer busy flag DE1D BMI &DE2B ;if set speech buffer is empty, skip routine DE1F JSR &EE6D ;update speech system variables DE22 EOR #&A0 ; DE24 CMP #&60 ; DE26 BCC &DE2B ;if result >=&60 DE2B DE28 JSR &DD79 ;else more speech work SPBUSY DE2B BIT &D9B7 ;set V and C DE2E JSR &DCA2 ;check if ACIA needs attention DE31 LDA &EC ;check if key has been pressed DE33 ORA &ED ; DE35 AND &0242 ;(this is 0 if keyboard is to be ignored, else &FF) DE38 BEQ &DE3E ;if 0 ignore keyboard DE3A SEC ;else set carry DE3B JSR &F065 ; Call keyboard handler centisecond interupt ICLK0 DE3E JSR &E19B ; Check for data in user printer channel DE41 BIT &FEC0 ;if ADC bit 6 is set ADC is not busy DE44 BVS &DE4A ;so DE4A DE46 RTS ;else return ************************************************************************* * * * SYSTEM INTERRUPT 4 ADC end of conversion * * * ************************************************************************* IADC DE47 ROL ;put original bit 4 from FE4D into bit 7 of A DE48 BPL &DE72 ;if not set DE72 IADCB DE4A LDX &024C ;else get current ADC channel DE4D BEQ &DE6C ;if 0 DE6C DE4F LDA &FEC2 ;read low data byte DE52 STA &02B5,X ;store it in &2B6,7,8 or 9 DE55 LDA &FEC1 ;get high data byte DE58 STA &02B9,X ;and store it in hi byte DE5B STX &02BE ;store in Analogue system flag marking last channel DE5E LDY #&03 ;handle event 3 conversion complete DE60 JSR &E494 ; DE63 DEX ;decrement X DE64 BNE &DE69 ;if X=0 DE66 LDX &024D ;get highest ADC channel preseny IADC2 DE69 JSR &DE8F ;and start new conversion IADC0 DE6C LDA #&10 ;reset interrupt 4 ; Reset System VIA interupt and return ; ------------------------------------ IVIAA0 DE6E STA &FE4D ; DE71 RTS ;and return ************************************************************************* * * * SYSTEM INTERRUPT 0 Keyboard * * * ************************************************************************* IKEYBD DE72 ROL ;get original bit 0 in bit 7 position DE73 ROL ; DE74 ROL ; DE75 ROL ; DE76 BPL &DE7F ; If bit 7 clear not a keyboard interrupt DE78 JSR &F065 ; Call keyboard handler keypress interupt DE7B LDA #&01 ; A=1 DE7D BNE &DE6E ; and off to reset interrupt and exit JUNKIR DE7F JMP &DCF3 ; Jump off to pass Unknown Interrupt to ROMs ************** exit routine ********************************************* IRQ0 DE82 PLA ;restore registers DE83 TAY ; DE84 PLA ; DE85 TAX ; DE86 PLA ; DE87 STA &FC ;store A ************************************************************************* * * * IRQ2V default entry * * * ************************************************************************* IRQTWO DE89 LDA &FC ;get back original value of A DE8B RTI ;and return to calling routine ************************************************************************* * * * OSBYTE 17 Start conversion * * * ************************************************************************* ADCBEG DE8C STY &02BE ;set last channel to finish conversion ADCBGX DE8F CPX #&05 ;if X<4 then DE91 BCC &DE95 ;DE95 DE93 LDX #&04 ;else X=4 ADCBG1 DE95 STX &024C ;store it as current ADC channel DE98 LDY &024E ;get conversion type DE9B DEY ;decrement DE9C TYA ;A=Y DE9D AND #&08 ;and it with 08 DE9F CLC ;clear carry DEA0 ADC &024C ;add to current ADC DEA3 SBC #&00 ;-1 DEA5 STA &FEC0 ;store to the A/D control panel DEA8 RTS ;and return WRSTR DEA9 LDA #&C3 ;point to start of string at &C300,Y WRSTRA DEAB STA &FE ;store it DEAD LDA #&00 ;point to lo byte DEAF STA &FD ;store it and start loop WRSTR1 DEB1 INY ;print character in string DEB2 LDA (&FD),Y ;pointed to by &FD/E DEB4 JSR OSASCI ;print it expanding Carriage returns DEB7 TAX ;store A in X DEB8 BNE &DEB1 ;and loop again if not =0 DEBA RTS ;else exit *********** OSBYTE 129 TIMED ROUTINE ****************************** ;ON ENTRY TIME IS IN X,Y INKEY DEBB STX &02B1 ;store time in INKEY countdown timer DEBE STY &02B2 ;which is decremented every 10ms DEC1 LDA #&FF ;A=&FF to flag timed wait DEC3 BNE &DEC7 ;goto DEC7 ************************************************************************** ************************************************************************** ** ** ** OSRDCH Default entry point ** ** ** ** RDCHV entry point read a character ** ** ** ************************************************************************** ************************************************************************** RDCH DEC5 LDA #&00 ;A=0 to flag wait forever RDCH5 DEC7 STA &E6 ;store entry value of A DEC9 TXA ;save X and Y DECA PHA ; DECB TYA ; DECC PHA ; DECD LDY &0256 ;get *EXEC file handle DED0 BEQ &DEE6 ;if 0 (not open) then DEE6 DED2 SEC ;set carry DED3 ROR &EB ;set bit 7 of CFS active flag to prevent clashes DED5 JSR OSBGET ;get a byte from the file DED8 PHP ;push processor flags to preserve carry DED9 LSR &EB ;restore &EB DEDB PLP ;get back flags DEDC BCC &DF03 ;and if carry clear, character found so exit via DF03 DEDE LDA #&00 ;else A=00 as EXEC file empty DEE0 STA &0256 ;store it in exec file handle DEE3 JSR OSFIND ;and close file via OSFIND RDCHQ DEE6 BIT &FF ;check ESCAPE flag, if bit 7 set Escape pressed DEE8 BMI &DF00 ;so off to DF00 DEEA LDX &0241 ;else get current input buffer number DEED JSR &E577 ;get a byte from input buffer DEF0 BCC &DF03 ;and exit if character returned DEF2 BIT &E6 ;(E6=0 or FF) DEF4 BVC &DEE6 ;if entry was OSRDCH not timed keypress, so go back and ;do it again i.e. perform GET function DEF6 LDA &02B1 ;else check timers DEF9 ORA &02B2 ; DEFC BNE &DEE6 ;and if not zero go round again DEFE BCS &DF05 ;else exit DEF0 BCC &DF03 DEF2 BIT &E6 DEF4 BVC &DEE6 DEF6 LDA &02B1 DEF9 ORA &02B2 DEFC BNE &DEE6 DEFE BCS &DF05 RDCH1 DF00 SEC DF01 LDA #&1B RDCHN RDCH3 DF03 STA &E6 RDCH2 DF05 PLA DF06 TAY DF07 PLA DF08 TAX DF09 LDA &E6 DF0B RTS **** STRINGS **** COPYR DF0C DB ')C(',0 ; Copyright string match **** COMMMANDS **** ; Command Address Call goes to COMTAB DF10 DB '.',E0,31,05 ; *. &E031, A=5 XY=>string, FSC &05 DF14 DB 'FX',E3,42,FF ; *FX &E342, A=&FF XY=>string, number parameters DF19 DB 'BASIC',E0,18,00 ; *BASIC &E018, A=0 XY=>string DF21 DB 'CAT',E0,31,05 ; *CAT &E031, A=5 XY=>string, FSC &05 DF27 DB 'CODE',E3,48,88 ; *CODE &E348, A=&88 X,Y=parameters, OSBYTE &88 DF2E DB 'EXEC',F6,8D,00 ; *EXEC &F68D, A=0 XY=>string DF35 DB 'HELP',F0,B9,FF ; *HELP &F0B9, A=&FF (&F2),Y=>string DF3C DB 'KEY',E3,27,FF ; *KEY &E327, A=&FF (&F2),Y=>string DF42 DB 'LOAD',E2,3C,00 ; *LOAD &E23C, A=0 XY=>string DF49 DB 'LINE',E6,59,01 ; *LINE &E659, A=1 XY=>string, USERV &01 DF50 DB 'MOTOR',E3,48,89 ; *MOTOR &E348, A=&89 X,Y=parameters, OSBYTE &89 DF58 DB 'OPT',E3,48,8B ; *OPT &E348, A=&8B X,Y=parameters, OSBYTE &8B DF5E DB 'RUN',E0,31,04 ; *RUN &E031, A=4 XY=>string, FSC &04 DF64 DB 'ROM',E3,48,8D ; *ROM &E348, A=&8D X,Y=parameters, OSBYTE &8D DF6A DB 'SAVE',E2,3E,00 ; *SAVE &E23E, A=0 XY=>string DF70 DB 'SPOOL',E2,81,00 ; *SPOOL &E281, A=0 XY=>string DF79 DB 'TAPE',E3,48,8C ; *TAPE &E348, A=&8C X,Y=parameters, OSBYTE &8C DF80 DB 'TV',E3,48,90 ; *TV &E348, A=&90 X,Y=parameters, OSBYTE &90 DF85 DB '',E0,31,03 ; Unmatched &E031, A=3 XY=>string, FSC &03 DF88 DB 00 ; Table end marker ; Command routines are entered with XY=>command tail, A=table parameter, ; &F2/3,&E6=>start of command string ; If table parameter if <&80, F2/3,Y converted to XY before entering ************************************************************************* * CLI - COMMAND LINE INTERPRETER * * * * ENTRY: XY=>Command line * * EXIT: All registers corrupted * * [ A=13 if unterminated string ] * ************************************************************************* ; CLINT DF89 STX &F2 ; Store XY in &F2/3 DF8B STY &F3 DF8D LDA #&08 DF8F JSR &E031 ; Inform filing system CLI being processed DF92 LDY #&00 ; Check the line is correctly terminated CLICHK DF94 LDA (&F2),Y DF96 CMP #&0D ; Loop until CR is found DF98 BEQ &DF9E DF9A INY ; Move to next character DF9B BNE &DF94 ; Loop back if less than 256 bytes long DF9D RTS ; Exit if string > 255 characters ; String is terminated - skip prepended spaces and '*'s CLILOK DF9E LDY #&FF CLIR DFA0 JSR &E039 ; Skip any spaces DFA3 BEQ &E017 ; Exit if at CR DFA5 CMP #&2A ; Is this character '*'? DFA7 BEQ &DFA0 ; Loop back to skip it, and check for spaces again DFA9 JSR &E03A ; Skip any more spaces DFAC BEQ &E017 ; Exit if at CR DFAE CMP #&7C ; Is it '|' - a comment DFB0 BEQ &E017 ; Exit if so DFB2 CMP #&2F ; Is it '/' - pass straight to filing system DFB4 BNE &DFBE ; Jump forward if not DFB6 INY ; Step past the '/' DFB7 JSR &E009 ; Convert &F2/3,Y->XY, ignore returned A DFBA LDA #&02 ; 2=RunSlashCommand DFBC BNE &E031 ; Jump to pass to FSCV ; ; Look command up in command table NOTSL DFBE STY &E6 ; Store offset to start of command DFC0 LDX #&00 DFC2 BEQ &DFD7 ; CLISCN DFC4 EOR &DF10,X DFC7 AND #&DF DFC9 BNE &DFE2 DFCB INY DFCC CLC ; CLISTP DFCD BCS &DFF4 DFCF INX DFD0 LDA (&F2),Y DFD2 JSR &E4E3 DFD5 BCC &DFC4 ; DOT DFD7 LDA &DF10,X DFDA BMI &DFF2 DFDC LDA (&F2),Y DFDE CMP #&2E DFE0 BEQ &DFE6 CLINXC DFE2 CLC DFE3 LDY &E6 DFE5 DEY CLINXS DFE6 INY DFE7 INX CLISKP DFE8 INX DFE9 LDA &DF0E,X DFEC BEQ &E021 DFEE BPL &DFE8 DFF0 BMI &DFCD ; CLIMFN DFF2 INX DFF3 INX ; CLIMCH DFF4 DEX DFF5 DEX DFF6 PHA DFF7 LDA &DF11,X DFFA PHA DFFB JSR &E03A DFFE CLC DFFF PHP E000 JSR &E004 E003 RTI ; Jump to routine CLIXYA E004 LDA &DF12,X ; Get table parameter E007 BMI &E017 ; If >=&80, number follow ; ; else string follows CLIXY E009 TYA ; Pass Y line offset to A for later E00A LDY &DF12,X ; Get looked-up parameter from table ; Convert &F2/3,A to XY, put Y in A CLIXY2 E00D CLC E00E ADC &F2 E010 TAX E011 TYA ; Pass supplied Y into A E012 LDY &F3 E014 BCC &E017 E016 INY ; CLIXY0 CLIXY1 SVACK CLIRTS E017 RTS ; *BASIC ; ====== BASIC E018 LDX &024B ; Get BASIC ROM number E01B BMI &E021 ; If none set, jump to pass command on E01D SEC ; Set Carry = not entering from RESET E01E JMP &DBE7 ; Enter language ROM in X ; Pass command on to other ROMs and to filing system CLIEND NBASIC E021 LDY &E6 ; Restore pointer to start of command E023 LDX #&04 ; 4=UnknownCommand E025 JSR &F168 ; Pass to sideways ROMs E028 BEQ &E017 ; If claimed, exit E02A LDA &E6 ; Restore pointer to start of command E02C JSR &E00D ; Convert &F2/3,A to XY, ignore returned A E02F LDA #&03 ; 3=PassCommandToFilingSystem ; Pass to current filing system JMIFSC E031 JMP (&021E) OPT E034 ASL A EOF E035 AND #&01 E037 BPL &E031 ; Skip spaces SKIPSN E039 INY SKIPSP E03A LDA (&F2),Y E03C CMP #&20 E03E BEQ &E039 SKPSP1 E040 CMP #&0D E042 RTS SKPCM E043 BCC &E03A SKPCOM E045 JSR &E03A E048 CMP #&2C E04A BNE &E040 E04C INY E04D RTS GETNUM E04E JSR &E03A E051 JSR &E07D E054 BCC &E08D GN0 E056 STA &E6 E058 JSR &E07C E05B BCC &E076 E05D TAX E05E LDA &E6 E060 ASL A E061 BCS &E08D E063 ASL A E064 BCS &E08D E066 ADC &E6 E068 BCS &E08D E06A ASL A E06B BCS &E08D E06D STA &E6 E06F TXA E070 ADC &E6 E072 BCS &E08D E074 BCC &E056 GTNS E076 LDX &E6 E078 CMP #&0D E07A SEC E07B RTS TSTNUX E07C INY TSTNUM E07D LDA (&F2),Y E07F CMP #&3A E081 BCS &E08D E083 CMP #&30 E085 BCC &E08D E087 AND #&0F E089 RTS TSTNRX E08A JSR &E045 TSTNR E08D CLC E08E RTS TSTHEX E08F JSR &E07D E092 BCS &E0A2 E094 AND #&DF E096 CMP #&47 E098 BCS &E08A E09A CMP #&41 E09C BCC &E08A E09E PHP E09F SBC #&37 E0A1 PLP TSTH0 E0A2 INY E0A3 RTS ************************************************************************** ************************************************************************** ** ** ** OSWRCH HANDLER ** ** ** ** Output a character to current output streams ** ** ** ************************************************************************** ************************************************************************** WRCH E0A4 PHA ; Save all registers E0A5 TXA E0A6 PHA E0A7 TYA E0A8 PHA E0A9 TSX E0AA LDA &0103,X ; Get A back from stack E0AD PHA ; Save A E0AE BIT &0260 ; Check OSWRCH interception flag E0B1 BPL &E0BB ; Not set, skip interception call E0B3 TAY ; Pass character to Y E0B4 LDA #&04 ; A=4 for OSWRCH call E0B6 JSR &E57E ; Call interception code E0B9 BCS &E10D ; If claimed, jump past to exit WRCNTN E0BB CLC ; Prepare to not send this to printer E0BC LDA #&02 ; Check output destination E0BE BIT &027C ; Is VDU driver disabled? E0C1 BNE &E0C8 ; Yes, skip past VDU driver E0C3 PLA ; Get character back E0C4 PHA ; Resave character E0C5 JSR &C4C0 ; Call VDU driver ; On exit, C=1 if character to be sent to printer WRCHNV E0C8 LDA #&08 ; Check output destination E0CA BIT &027C ; Is printer seperately enabled? E0CD BNE &E0D1 ; Yes, jump to call printer driver E0CF BCC &E0D6 ; Carry clear, don't sent to printer WRCHNC E0D1 PLA ; Get character back E0D2 PHA ; Resave character E0D3 JSR &E114 ; Call printer driver WRCHNP E0D6 LDA &027C ; Check output destination E0D9 ROR A ; Is serial output enabled? E0DA BCC &E0F7 ; No, skip past serial output E0DC LDY &EA ; Get serial timout counter E0DE DEY ; Decrease counter E0DF BPL &E0F7 ; Timed out, skip past serial code E0E1 PLA ; Get character back E0E2 PHA ; Resave character E0E3 PHP ; Save IRQs E0E4 SEI ; Disable IRQs E0E5 LDX #&02 ; X=2 for serial output buffer E0E7 PHA ; Save character E0E8 JSR &E45B ; Examine serial output buffer E0EB BCC &E0F0 ; Buffer not full, jump to send character E0ED JSR &E170 ; Wait for buffer to empty a bit WRCNP1 E0F0 PLA ; Get character back E0F1 LDX #&02 ; X=2 for serial output buffer E0F3 JSR &E1F8 ; Send character to serial output buffer E0F6 PLP ; Restore IRQs WRCHNR E0F7 LDA #&10 ; Check output destination E0F9 BIT &027C ; Is SPOOL output disabled? E0FC BNE &E10D ; Yes, skip past SPOOL output E0FE LDY &0257 ; Get SPOOL handle E101 BEQ &E10D ; If not open, skip past SPOOL output E103 PLA ; Get character back E104 PHA ; Resave character E105 SEC E106 ROR &EB ; Set RFS/CFS's 'spooling' flag E108 JSR &FFD4 ; Write character to SPOOL channel E10B LSR &EB ; Reset RFS/CFS's 'spooling' flag WRCHNS WRCHNN E10D PLA ; Restore all registers E10E PLA E10F TAY E110 PLA E111 TAX E112 PLA E113 RTS ; Exit ************************************************************************* * * * PRINTER DRIVER * * * ************************************************************************* ; Enter here from OSWRCH control routine ; A=character to print MPRINT E114 BIT &027C ; Test bit 6 of OSBYTE 3 OSWRCH destination E117 BVS &E139 ; Set, printer disabled, jump to E139 to exit E119 CMP &0286 ; Compare with printer ignore character E11C BEQ &E139 ; Matches, jump to E139 to ignore and exit ; Enter here from VDU 1,n to send directly to the printer ; A=character to print ; PRINT E11E PHP ; Save flags E11F SEI ; Disable IRQs E120 TAX ; Save character in X E121 LDA #&04 ; A=4 E123 BIT &027C ; Test bit 2 of OSBYTE 3 OSWRCH destination E126 BNE &E138 ; If set, 'printer driver disabled', so jump to E138 to exit E128 TXA ; Get character back to A E129 LDX #&03 ; X=3, printer buffer number E12B JSR &E1F8 ; Attempt to put character into buffer 3 - printer buffer E12E BCS &E138 ; If carry set, not inserted, buffer was full, aborted with Escape E130 BIT &02D2 ; Test buffer 3 busy flag, MI=not busy, PL=busy E133 BPL &E138 ; Buffer busy, jump to E138 to exit E135 JSR &E13A ; Call E13A to send first to printer to start interupts PRINT4 PRINT3 E138 PLP ; Restore IRQs PRINT0 E139 RTS ; And exit ; Fetch a byte from printer buffer and send it to current printer STROBE E13A LDA &0285 ; Get printer destination E13D BEQ &E1AD ; If 0 then sink output, jump to E1AD clear buffer and exit E13F CMP #&01 ; Check if printer 1 E141 BNE &E164 ; Not parallel printer, jump to E164 ;********* parallel printer ******************************************* ; One character is sent when PrinterAck generates an interupt /after/ ; a character has been processed. No test is made for Printer Ready, ; code assumes that PrinterAck IRQ means the printer is ready. ; ; First character sent to an empty printer buffer is sent to printer to ; trigger 'PrinterAck' interupt later to send following characters. ; E143 JSR &E460 ; Fetch a byte from the printer buffer, returns CS=empty E146 ROR &02D2 ; Rotate buffer status into buffer 2 busy flag E149 BMI &E190 ; If buffer empty, set 'not busy', and jump to E190 to exit ; ; Next character to enter buffer will come back here to ; ; start the interupt process again E14B LDY #&82 ; Enable interrupt 1 of the User VIA E14D STY &FE6E ; E150 STA &FE61 ; Write character to centronics port E153 LDA &FE6C ; Pulse CA2 line to generate STROBE signal E156 AND #&F1 ; to advise printer that E158 ORA #&0C ; valid data is E15A STA &FE6C ; waiting E15D ORA #&0E ; E15F STA &FE6C ; E162 BNE &E190 ; Then exit ********* serial printer ********************************************* STROB1 E164 CMP #&02 ;is it Serial printer?? E166 BNE &E191 ;if not E191 E168 LDY &EA ;else is RS423 in use by cassette?? E16A DEY ; E16B BPL &E1AD ;if so E1AD to flush buffer E16D LSR &02D2 ;else clear buffer busy flag RSBUSY E170 LSR &024F ;and RS423 busy flag RSETX E173 JSR &E741 ;count buffer if C is clear on return E176 BCC &E190 ;no room in buffer so exit E178 LDX #&20 ;else RSED E17A LDY #&9F ; ************************************************************************* * * * OSBYTE 156 update ACIA setting and RAM copy * * * ************************************************************************* ;on entry RS423 E17C PHP ;push flags E17D SEI ;bar interrupts E17E TYA ;A=Y E17F STX &FA ;&FA=X E181 AND &0250 ;A=old value AND Y EOR X E184 EOR &FA ; E186 LDX &0250 ;get old value in X STARSC E189 STA &0250 ;put new value in E18C STA &FE08 ;and store to ACIA control register E18F PLP ;get back flags STROB0 RSOVF E190 RTS ;and exit ************ printer is neither serial or parallel so its user type ***** STROB3 E191 CLC ; Clear carry E192 LDA #&01 ; A=1, character sent to printer E194 JSR &E1A2 ; Call NETV and UPTV ; ; External printer driver would normally return CS to ; ; cause characters to continue to be sent. If it returns ; ; CC, it will be called on 100Hz poll and needs to remove ; ; bytes from the printer buffer so that it does not stall ; ; when full. ************************************************************************* * * * OSBYTE 123 Warn printer driver going dormant * * * ************************************************************************* ; On entry from OSBYTE, Carry is set PFREE E197 ROR &02D2 ; Set or clear printer buffer busy flag UPTRTS E19A RTS ; And exit ******** Printer Driver 100Hz poll ************** UPTINT E19B BIT &02D2 ;if bit 7 is set, printer is not being polled E19E BMI &E19A ;so exit E1A0 LDA #&00 ;else A=0 (A already is 0 from caller) MUPT E1A2 LDX #&03 ;X=3 UPTX E1A4 LDY &0285 ;Y=printer destination E1A7 JSR &E57E ;to JMP (NETV) E1AA JMP (&0222) ;jump to PRINT VECTOR for special routines *************** Buffer handling ***************************************** ;X=buffer number ;Buffer number Address Flag Out pointer In pointer ;0=Keyboard 3E0-3FF 2CF 2D8 2E1 ;1=RS423 Input A00-AFF 2D0 2D9 2E2 ;2=RS423 output 900-9BF 2D1 2DA 2E3 ;3=printer 880-8BF 2D2 2DB 2E4 ;4=sound0 840-84F 2D3 2DC 2E5 ;5=sound1 850-85F 2D4 2DD 2E6 ;6=sound2 860-86F 2D5 2DE 2E7 ;7=sound3 870-87F 2D6 2DF 2E8 ;8=speech 8C0-8FF 2D7 2E0 2E9 PURGE STCRS STROB4* E1AD CLC ;clear carry PURGEX E1AE PHA ;save A E1AF PHP ;save flags E1B0 SEI ;set interrupts E1B1 BCS &E1BB ;if carry set on entry then E1BB E1B3 LDA &E9AD,X ;else get byte from baud rate/sound data table E1B6 BPL &E1BB ;if +ve the E1BB E1B8 JSR &ECA2 ;else clear sound data PURNSB E1BB SEC ;set carry E1BC ROR &02CF,X ;rotate buffer flag to show buffer empty E1BF CPX #&02 ;if X>1 then its not an input buffer E1C1 BCS &E1CB ;so E1CB E1C3 LDA #&00 ;else Input buffer so A=0 E1C5 STA &0268 ;store as length of key string E1C8 STA &026A ;and length of VDU queue PURGE3 E1CB JSR &E73B ;then enter via count purge vector any user routines E1CE PLP ;restore flags E1CF PLA ;restore A E1D0 RTS ;and exit ************************************************************************* * * * COUNT PURGE VECTOR DEFAULT ENTRY * * * ************************************************************************* ;on entry if V set clear buffer ; if C set get space left ; else get bytes used CNP E1D1 BVC &E1DA ;if V set then E1DA E1D3 LDA &02D8,X ;else start of buffer=end of buffer E1D6 STA &02E1,X ; E1D9 RTS ;and exit COUNT E1DA PHP ;push flags E1DB SEI ;bar interrupts E1DC PHP ;push flags E1DD SEC ;set carry E1DE LDA &02E1,X ;get end of buffer E1E1 SBC &02D8,X ;subtract start of buffer E1E4 BCS &E1EA ;if carry caused E1EA E1E6 SEC ;set carry E1E7 SBC &E447,X ;subtract buffer start offset (i.e. add buffer length) BUFNX1 E1EA PLP ;pull flags E1EB BCC &E1F3 ;if carry clear E1F3 to exit E1ED CLC ;clear carry E1EE ADC &E447,X ;adc to get bytes used E1F1 EOR #&FF ;and invert to get space left BFCNT0 E1F3 LDY #&00 ;Y=0 E1F5 TAX ;X=A E1F6 PLP ;get back flags E1F7 RTS ;and exit ********** enter byte in buffer, wait and flash lights if full ********** ; Attempts to enter byte into a buffer. ; If the buffer is full, flashes the LEDs and loops until background makes ; space in the buffer, or until Escape aborts the attempt and returns CS. ; ; On entry ; X=buffer number ; A=byte to enter into buffer ; On exit ; CC=Successful ; CS=Aborted with Escape ; WRITE WRITE1 E1F8 SEI ; Disable interrupts E1F9 JSR &E4B0 ; Enter a byte in buffer X E1FC BCC &E20D ; If successful exit E1FE JSR &E9EA ; Switch on both keyboard LEDs E201 PHP ; Save flags (MI=Escape) E202 PHA ; Save byte attempted to insert E203 JSR &EEEB ; Switch off the LEDs E206 PLA ; Get the byte back E207 PLP ; Get the flags back E208 BMI &E20D ; If MI, Escape pressed, so abort and return CS E20A CLI ; Allow interrupts for a few cycles E20B BCS &E1F8 ; Byte didn't enter buffer go and try it again WRITE0 E20D RTS ; Return OS SERIES VI GEOFF COX ************************************************************************* * * * *SAVE/*LOAD SETUP * * * ************************************************************************* **************: clear osfile control block workspace ******************** CLRFB E20E PHA ;push A E20F LDA #&00 ;A=0 E211 STA &02EE,X ;clear osfile control block workspace E214 STA &02EF,X ; E217 STA &02F0,X ; E21A STA &02F1,X ; E21D PLA ;get back A E21E RTS ;and exit *********** shift through osfile control block ************************** ROL4 E21F STY &E6 ;&E6=Y E221 ROL ;A=A*2 E222 ROL ;*4 E223 ROL ;*8 E224 ROL ;*16 E225 LDY #&04 ;Y=4 ROL4A E227 ROL ;A=A*32 E228 ROL &02EE,X ;shift bit 7 of A into shift register E22B ROL &02EF,X ;and E22E ROL &02F0,X ;shift E231 ROL &02F1,X ;along E234 BCS &E267 ;if carry set on exit then register has overflowed ;so bad address error E236 DEY ;decrement Y E237 BNE &E227 ;and if Y>0 then do another shift E239 LDY &E6 ;get back original Y E23B RTS ;and exit ************************************************************************* * * * *LOAD ENTRY * * * ************************************************************************* LOAD E23C LDA #&FF ;signal that load is being performed ************************************************************************* * * * *SAVE ENTRY * * * ************************************************************************* ;on entry A=0 for save &ff for load SAVE E23E STX &F2 ;store address of rest of command line E240 STY &F3 ; E242 STX &02EE ;X and Y are stored in OSfile control block E245 STY &02EF ; E248 PHA ;Push A E249 LDX #&02 ;X=2 E24B JSR &E20E ;clear the shift register E24E LDY #&FF ;Y=255 E250 STY &02F4 ;store im 2F4 E253 INY ;Y=0 E254 JSR &EA1D ;call GSINIT to prepare for reading text line LS2 E257 JSR &EA2F ;read GSTRANS character from command line E25A BCC &E257 ;loop to step past filename E25C PLA ;get back A without stack changes E25D PHA ; E25E BEQ &E2C2 ;IF A=0 (SAVE) E2C2 E260 JSR &E2AD ;set up file block E263 BCS &E2A0 ;if carry set do OSFILE E265 BEQ &E2A5 ;else if A=0 goto OSFILE MBADHX E267 BRK ; E268 DB &FC ; E269 DB 'Bad Address' ;error E274 BRK ; ************************************************************************* * * * OSBYTE 119 ENTRY * * CLOSE SPOOL/ EXEC FILES * * * ************************************************************************* SHUTES E275 LDX #&10 ;X=10 issue *SPOOL/EXEC files warning E277 JSR &F168 ;and issue call E27A BEQ &E29F ;if a rom accepts and issues a 0 then E29F to return E27C JSR &F68B ;else close the current exec file E27F LDA #&00 ;A=0 ************************************************************************** * * * *SPOOL * * * ************************************************************************** SPOOL E281 PHP ;if A=0 file is closed so E282 STY &E6 ;Store Y E284 LDY &0257 ;get file handle E287 STA &0257 ;store A as file handle E28A BEQ &E28F ;if Y<>0 then E28F E28C JSR OSFIND ;else close file via osfind SPOOL2 E28F LDY &E6 ;get back original Y E291 PLP ;pull flags E292 BEQ &E29F ;if A=0 on entry then exit E294 LDA #&80 ;else A=&80 E296 JSR OSFIND ;to open file Y for output E299 TAY ;Y=A E29A BEQ &E310 ;and if this is =0 then E310 BAD COMMAND ERROR E29C STA &0257 ;store file handle SPOOL0 E29F RTS ;and exit LOADX1 E2A0 BNE &E310 ;if NE then BAD COMMAND error E2A2 INC &02F4 ;increment 2F4 to 00 LS0 E2A5 LDX #&EE ;X=&EE E2A7 LDY #&02 ;Y=&02 E2A9 PLA ;get back A E2AA JMP OSFILE ;and JUMP to OSFILE **** check for hex digit ************************************************ LSA E2AD JSR &E03A ;look for end of line E2B0 JSR &E08F ;carry is set if it finds hex digit E2B3 BCC &E2C1 ;so E2C1 exit E2B5 JSR &E20E ;clear shift register ************** shift byte into control block *************************** LSA1 E2B8 JSR &E21F ;shift lower nybble of A into shift register E2BB JSR &E08F ;then check for Hex digit E2BE BCS &E2B8 ;if found then do it again E2C0 SEC ;else set carry LSA0 E2C1 RTS ;and exit **************; set up OSfile control block **************************** SAVEX E2C2 LDX #&0A ;X=0A E2C4 JSR &E2AD ; E2C7 BCC &E310 ;if no hex digit found EXIT via BAD Command error E2C9 CLV ;clear bit 6 ******************Read file length from command line********************* E2CA LDA (&F2),Y ;read next byte from text line E2CC CMP #&2B ;is it '+' E2CE BNE &E2D4 ;if not assume its a last byte address so e2d4 E2D0 BIT &D9B7 ;else set V and M flags E2D3 INY ;increment Y to point to hex group SAVEX3 E2D4 LDX #&0E ;X=E E2D6 JSR &E2AD ; E2D9 BCC &E310 ;if carry clear no hex digit so exit via error E2DB PHP ;save flags E2DC BVC &E2ED ;if V set them E2ED explicit end address found E2DE LDX #&FC ;else X=&FC E2E0 CLC ;clear carry SAVEX5 E2E1 LDA &01FC,X ;and add length data to start address E2E4 ADC &0200,X ; E2E7 STA &0200,X ; E2EA INX ; E2EB BNE &E2E1 ;repeat until X=0 SAVEX4 E2ED LDX #&03 ;X=3 SAVEX1 E2EF LDA &02F8,X ;copy start adddress to load and execution addresses E2F2 STA &02F4,X ; E2F5 STA &02F0,X ; E2F8 DEX ; E2F9 BPL &E2EF ; E2FB PLP ;get back flag E2FC BEQ &E2A5 ;if end of command line reached then E2A5 ; to do osfile E2FE LDX #&06 ;else set up execution address E300 JSR &E2AD ; E303 BCC &E310 ;if error BAD COMMAND E305 BEQ &E2A5 ;and if end of line reached do OSFILE E307 LDX #&02 ;else set up load address E309 JSR &E2AD ; E30C BCC &E310 ;if error BAD command E30E BEQ &E2A5 ;else on end of line do OSFILE ;anything else is an error ******** Bad command error ************************************ MBDCMD BADFX E310 BRK ; E311 DB &FE ;error number E312 DB 'Bad Command' ; BADKEY E31D BRK E31E DB &FB ; E31F DB 'Bad Key' ; E326 BRK ************************************************************************* * * * *KEY ENTRY * * * ************************************************************************* KEY E327 JSR &E04E ;set up key number in A E32A BCC &E31D ;if not valid number give error E32C CPX #&10 ;if key number greater than 15 E32E BCS &E31D ;if greater then give error E330 JSR &E045 ;otherwise skip commas, and check for CR E333 PHP ;save flags for later E334 LDX &0B10 ;get pointer to top of existing key strings E337 TYA ;save Y E338 PHA ;to preserve text pointer E339 JSR &E3D1 ;set up soft key definition E33C PLA ;get back Y E33D TAY ; E33E PLP ;and flags E33F BNE &E377 ;if CR found return else E377 to set up new string E341 RTS ;else return to set null string ************************************************************************* * * * *FX OSBYTE * * * ************************************************************************* A=number FX E342 JSR &E04E ;convert the number to binary E345 BCC &E310 ;if bad number call bad command E347 TXA ;save X ************************************************************************* * * * *CODE *MOTOR *OPT *ROM *TAPE *TV * * * ************************************************************************* ;enter codes *CODE &88 *MOTOR &89 *OPT &8B *TAPE &8C *ROM &8D *TV &90 FXN E348 PHA ;save A E349 LDA #&00 ;clear &E4/E5 E34B STA &E5 ; E34D STA &E4 ; E34F JSR &E043 ;skip commas and check for newline (CR) E352 BEQ &E36C ;if CR found E36C E354 JSR &E04E ;convert character to binary E357 BCC &E310 ;if bad character bad command error E359 STX &E5 ;else save it E35B JSR &E045 ;skip comma and check CR E35E BEQ &E36C ;if CR then E36C E360 JSR &E04E ;get another parameter E363 BCC &E310 ;if bad error E365 STX &E4 ;else store in E4 E367 JSR &E03A ;now we must have a newline E36A BNE &E310 ;if none then output an error FX1 E36C LDY &E4 ;Y=third osbyte parameter E36E LDX &E5 ;X=2nd E370 PLA ;A=first E371 JSR OSBYTE ;call osbyte E374 BVS &E310 ;if V set on return then error E376 RTS ;else RETURN ********* *KEY CONTINUED ************************************************ ;X points to last byte of current key definitions CLI1 E377 SEC ; E378 JSR &EA1E ;GSINIT to scan dull string CLIWP E37B JSR &EA2F ;call GSREAD, carry is set if end of line found E37E BCS &E388 ;E388 to deal with end of line E380 INX ;point to first byte of new key definition E381 BEQ &E31D ;if X=0 buffer WILL overflow so exit with BAD KEY error E383 STA &0B00,X ;store character E386 BCC &E37B ;and loop to get next byte if end of line not found CLIY E388 BNE &E31D ;if Z clear then no matching '"' found or for some ;other reason line doesn't terminate properly E38A PHP ;else if all OK save flags E38B SEI ;bar interrupts E38C JSR &E3D1 ;and move string E38F LDX #&10 ;set loop counter CLIYP E391 CPX &E6 ;if key being defined is found E393 BEQ &E3A3 ;then skip rest of loop E395 LDA &0B00,X ;else get start of string X E398 CMP &0B00,Y ;compare with start of string Y E39B BNE &E3A3 ;if not the same then skip rest of loop E39D LDA &0B10 ;else store top of string definition E3A0 STA &0B00,X ;in designated key pointer CLIYQ E3A3 DEX ;decrement loop pointer X E3A4 BPL &E391 ;and do it all again E3A6 PLP ;get back flags E3A7 RTS ;and exit ***********: set string lengths ***************************************** SKSIZE E3A8 PHP ;push flags E3A9 SEI ;bar interrupts E3AA LDA &0B10 ;get top of currently defined strings E3AD SEC ; E3AE SBC &0B00,Y ;subtract to get the number of bytes in strings ;above end of string Y E3B1 STA &FB ;store this E3B3 TXA ;save X E3B4 PHA ; E3B5 LDX #&10 ;and X=16 SKS1 E3B7 LDA &0B00,X ;get start offset (from B00) of key string X E3BA SEC ; E3BB SBC &0B00,Y ;subtract offset of string we are working on E3BE BCC &E3C8 ;if carry clear (B00+Y>B00+X) or E3C0 BEQ &E3C8 ;result (in A)=0 E3C2 CMP &FB ;or greater or equal to number of bytes above ;string we are working on E3C4 BCS &E3C8 ;then E3C8 E3C6 STA &FB ;else store A in &FB SKS2 E3C8 DEX ;point to next lower key offset E3C9 BPL &E3B7 ;and if 0 or +ve go back and do it again E3CB PLA ;else get back value of X E3CC TAX ; E3CD LDA &FB ;get back latest value of A E3CF PLP ;pull flags E3D0 RTS ;and return ***********: set up soft key definition ********************************* SKGCR E3D1 PHP ;push P E3D2 SEI ;bar interrupts E3D3 TXA ;save X E3D4 PHA ;push A E3D5 LDY &E6 ;get key number E3D7 JSR &E3A8 ;and set up &FB E3DA LDA &0B00,Y ;get start of string E3DD TAY ;put it in Y E3DE CLC ;clear carry E3DF ADC &FB ;add number of bytes above string E3E1 TAX ;put this in X E3E2 STA &FA ;and store it E3E4 LDA &0268 ;check number of bytes left to remove from key buffer ;if not 0 key is being used (definition expanded so ;error. This stops *KEY 1 "*key1 FRED" etc. E3E7 BEQ &E3F6 ;if not in use continue E3E9 BRK ; E3EA DB &FA ;error number E3EB DB 'Key in use' ; E3F5 BRK ; SK3 E3F6 DEC &0284 ;decrement consistence flag to &FF to warn that key ;definitions are being changed E3F9 PLA ;pull A E3FA SEC ; E3FB SBC &FA ;subtract &FA E3FD STA &FA ;and re store it E3FF BEQ &E40D ;if 0 then E40D SKGCS E401 LDA &0B01,X ;else move string E404 STA &0B01,Y ;from X to Y E407 INY ; E408 INX ; E409 DEC &FA ;for length of string E40B BNE &E401 ; SKGCV E40D TYA ;store end of moved string(s) E40E PHA ; E40F LDY &E6 ;get back key number E411 LDX #&10 ;point at top of last string SKGCT E413 LDA &0B00,X ;get this value E416 CMP &0B00,Y ;compare it with start of new or re defined key E419 BCC &E422 ;if less then E422 E41B BEQ &E422 ;if = then E422 E41D SBC &FB ;shift key definitions accordingly E41F STA &0B00,X ; SKGCU E422 DEX ;point to next lowest string def E423 BPL &E413 ;and if =>0 then loop and do it again E425 LDA &0B10 ;else make top of key definitions E428 STA &0B00,Y ;the start of our key def E42B PLA ;get new end of strings E42C STA &0B10 ;and store it E42F TAX ;put A in X E430 INC &0284 ;reset consistency flag E433 PLP ;restore flags E434 RTS ;and exit **************** BUFFER ADDRESS HI LOOK UP TABLE ************************ BFADHI E435 DB &03 ; &300+&E0 => &3E0+32 => keyboard buffer E436 DB &0A ; &A00+&00 => &A00+256 => rs423 input E437 DB &08 ; &8C0+&40 => &900+192 => rs423 output E438 DB &07 ; &7C0+&C0 => &880+64 => printer E439 DB &07 ; &750+&F0 => &840+16 => sound 0 E43A DB &07 ; &760+&F0 => &850+16 => sound 1 E43B DB &07 ; &770+&F0 => &860+16 => sound 2 E43C DB &07 ; &780+&F0 => &870+16 => sound 3 E43D DB &09 ; &900+&C0 => &9C0+64 => speech **************** BUFFER ADDRESS LO LOOK UP TABLE ************************ BFADLO E43E DB &00 E43F DB &00 E440 DB &C0 E441 DB &C0 E442 DB &50 E443 DB &60 E444 DB &70 E445 DB &80 E446 DB &00 **************** BUFFER START ADDRESS OFFSET **************************** BFORG E447 DB &E0 E448 DB &00 E449 DB &40 E44A DB &C0 E44B DB &F0 E44C DB &F0 E44D DB &F0 E44E DB &F0 E44F DB &C0 ******* get nominal buffer addresses in &FA/B ************************** ; ON ENTRY X=buffer number ;Buffer number Address Flag Out pointer In pointer ;0=Keyboard 3E0-3FF 2CF 2D8 2E1 ;1=RS423 Input A00-AFF 2D0 2D9 2E2 ;2=RS423 output 900-9BF 2D1 2DA 2E3 ;3=printer 880-8BF 2D2 2DB 2E4 ;4=sound0 840-84F 2D3 2DC 2E5 ;5=sound1 850-85F 2D4 2DD 2E6 ;6=sound2 860-86F 2D5 2DE 2E7 ;7=sound3 870-87F 2D6 2DF 2E8 ;8=speech 8C0-8FF 2D7 2E0 2E9 ; ; The buffers are arranged so that an offset of &FF is the last entry, and ; incrementing to &00 passes the end of the buffer ; ON ENTRY X=buffer number SETBUF E450 LDA &E43E,X ;get buffer base address lo E453 STA &FA ;store it E455 LDA &E435,X ;get buffer base address hi E458 STA &FB ;store it E45A RTS ;exit ************************************************************************* * * * OSBYTE 152 Examine Buffer status * * * ************************************************************************* ; On entry X = buffer number ; On exit FA/B points to buffer start ; CS = buffer empty ; Y = preserved ; A = corrupted ; CC = buffer not empty ; Y = offset to next character ; A = examined byte ; *BUG* default REMV code branches to the wrong exit point, omitting TAY ; so examined byte is lost to OSBYTE caller ; EXAM E45B BIT &D9B7 ;set V and E45E BVS &E461 ;jump to E461 ************************************************************************* * * * OSBYTE 145 Get byte from Buffer * * * ************************************************************************* ; On entry X = buffer number ; On exit FA/B points to buffer start ; CS = buffer empty ; Y = preserved ; A = corrupted ; CC = buffer not empty ; Y = offset to next character ; A = byte removed from buffer ; REMOV E460 CLV ;clear V EXAM1 E461 JMP (&022C) ;Jump via REMV ************************************************************************* * * * REMV buffer remove vector default entry point * * * ************************************************************************* ; On entry, X=buffer number, VS=examine buffer, VC=remove from buffer ; On exit, CS=buffer empty, CC=buffer not empty, A=byte from buffer ; if byte removed from buffer, Y also equals byte from buffer ; to pass to OSBYTE caller ; REM E464 PHP ;push flags E465 SEI ;bar interrupts E466 LDA &02D8,X ;get output pointer for buffer X E469 CMP &02E1,X ;compare to input pointer E46C BEQ &E4E0 ;if equal buffer is empty so jump to return with SEC E46E TAY ;else A=Y E46F JSR &E450 ;and get buffer pointer into FA/B E472 LDA (&FA),Y ;read byte from buffer E474 BVS &E491 ;if V was set on input, jump to return with CLC ; ;*BUG* Should be BVS LE490 to set Y=byte E476 PHA ;otherwise, update buffer pointer E477 INY ;increment Y E478 TYA ;A=Y E479 BNE &E47E ;pointer<>0, not reached end of buffer E47B LDA &E447,X ;get offset to start of buffer REMOV2 E47E STA &02D8,X ;set update buffer output pointer E481 CPX #&02 ;if buffer is input (0 or 1) E483 BCC &E48F ;then skip checking empty buffer E485 CMP &02E1,X ;compare output pointer with input pointer E488 BNE &E48F ;not the same, so not empty, so go to E48F E48A LDY #&00 ;buffer is empty E48C JSR &E494 ;send Event 0 for Buffer Empty event REMOV4 E48F PLA ;get back the byte from buffer E490 TAY ;put it in Y for OSBYTE caller REMOV3 E491 PLP ;restore IRQs E492 CLC ;clear carry to indicate success E493 RTS ;and exit ************************************************************************** ************************************************************************** ** ** ** CAUSE AN EVENT ** ** ** ************************************************************************** ************************************************************************** \on entry, \ Y=event number, passed in A to EVENTV \ X=parameter 1, passed in X to EVENTV \ A=parameter 2, passed in Y to EVENTV \on exit, \ carry clear=event enabled, A=preserved, X,Y returned \ carry set= event disabled, A,X,Y preserved EVENT E494 PHP ;push flags E495 SEI ;bar interrupts E496 PHA ;push A E497 STA &FA ;&FA=A E499 LDA &02BF,Y ;get enable event flag E49C BEQ &E4DF ;if 0 event is not enabled so exit with SEC E49E TYA ;else A=Y E49F LDY &FA ;Y=A E4A1 JSR &F0A5 ;vector through &220 E4A4 PLA ;get back A E4A5 PLP ;get back flags E4A6 CLC ;clear carry for success E4A7 RTS ;and exit ********* check event 2 character entering buffer *********************** CKESCY E4A8 TYA ;A=Y E4A9 LDY #&02 ;Y=2 E4AB JSR &E494 ;check event E4AE TAY ;Y=A ************************************************************************* * * * OSBYTE 138 Put byte into Buffer * * * ************************************************************************* ;on entry X is buffer number, Y is character to be written INSERT E4AF TYA ;A=Y INSRT E4B0 JMP (&022A) ;jump to INSBV ************************************************************************* * * * INSBV insert character in buffer vector default entry point * * * ************************************************************************* ;on entry X is buffer number, A is character to be written INS E4B3 PHP ;save flags E4B4 SEI ;bar interrupts E4B5 PHA ;save A E4B6 LDY &02E1,X ;get buffer input pointer E4B9 INY ;increment Y E4BA BNE &E4BF ;if Y<>0 then hasn't wrapped past end E4BC LDY &E447,X ;get offset to buffer start INSRT2 E4BF TYA ;put it in A E4C0 CMP &02D8,X ;compare it with output pointer E4C3 BEQ &E4D4 ;if equal buffer is full so jump forwards E4C5 LDY &02E1,X ;get unupdated input pointer back E4C8 STA &02E1,X ;store updated input pointer E4CB JSR &E450 ;and point &FA/B to buffer base E4CE PLA ;get back byte E4CF STA (&FA),Y ;store it in buffer E4D1 PLP ;restore interupts E4D2 CLC ;clear carry for success E4D3 RTS ;and exit INSRT1 E4D4 PLA ;get back byte E4D5 CPX #&02 ;are we are working on an input buffer? E4D7 BCS &E4E0 ;if output buffer, skip buffer full event E4D9 LDY #&01 ;else Y=1 E4DB JSR &E494 ;send Event 1 - Input Buffer Full E4DE PHA ;push A to balance next PLA ***** return with carry set ********************************************* EVENT0 E4DF PLA ;restore A INOBUF REMOV1 E4E0 PLP ;restore flags E4E1 SEC ;set carry E4E2 RTS ;and exit ***************** CODE MODIFIER ROUTINE ********************************* * CHECK FOR ALPHA CHARACTER * ************************************************************************* ;ENTRY character in A ;exit with carry set if non-Alpha character CAPS E4E3 PHA ;Save A E4E4 AND #&DF ;convert lower to upper case E4E6 CMP #&41 ;is it 'A' or greater? E4E8 BCC &E4EE ;if not exit routine with carry set E4EA CMP #&5B ;is it less than 'Z' E4EC BCC &E4EF ;if so exit with carry clear CAPSC1 E4EE SEC ;else clear carry CAPSX E4EF PLA ;get back original value of A E4F0 RTS ;and Return ******* INSERT byte in Keyboard buffer ********************************* RDCHS E4F1 LDX #&00 ;X=0 to indicate keyboard buffer ************************************************************************* * * * OSBYTE 153 Put byte in input Buffer checking for ESCAPE * * * ************************************************************************* ;on entry X = buffer number (either 0 or 1) ;X=1 is RS423 input ;X=0 is Keyboard ;Y is character to be written RNSRT E4F3 TXA ;A=buffer number E4F4 AND &0245 ;and with RS423 mode (0 treat as keyboard ;1 ignore Escapes no events no soft keys) E4F7 BNE &E4AF ;so if RS423 buffer AND RS423 in normal mode (1) E4AF E4F9 TYA ;else Y=A character to write E4FA EOR &026C ;compare with current escape ASCII code (0=match) E4FD ORA &0275 ;or with current ESCAPE status (0=ESC, 1=ASCII) E500 BNE &E4A8 ;if ASCII or no match E4A8 to enter byte in buffer E502 LDA &0258 ;else get ESCAPE/BREAK action byte E505 ROR ;Rotate to get ESCAPE bit into carry E506 TYA ;get character back in A E507 BCS &E513 ;and if escape disabled exit with carry clear E509 LDY #&06 ;else signal EVENT 6 Escape pressed E50B JSR &E494 ; E50E BCC &E513 ;if event handles ESCAPE then exit with carry clear E510 JSR &E674 ;else set ESCAPE flag CLCRTS CKESC1 E513 CLC ;clear carry E514 RTS ;and exit ******** get a byte from keyboard buffer and interpret as necessary ***** ;on entry A=cursor editing status ;1=return &87-&8B, ;2= use cursor keys as soft keys 11-15 ;this area not reached if cursor editing is normal RDCHGX E515 ROR ;get bit 1 into carry E516 PLA ;get keycode back into A E517 BCS &E592 ;if carry is set return keycode ;else cursor keys are 'soft' ; Translate soft keys &80-&FF RDCHGS E519 TYA ;A=Y get back original key code (&80-&FF) E51A PHA ;PUSH A E51B LSR ;get high nybble into lo E51C LSR ; E51D LSR ; E51E LSR ;A=8-&F E51F EOR #&04 ;and invert bit 2 ;&8 becomes &C ;&9 becomes &D ;&A becomes &E ;&B becomes &F ;&C becomes &8 ;&D becomes &9 ;&E becomes &A ;&F becomes &B E521 TAY ;Y=A = 8-F E522 LDA &0265,Y ;read 026D to 0274 code interpretation status ;0=ignore key, 1=expand as 'soft' key ;2-&FF add this to base for ASCII code ;note that provision is made for keypad operation ;as codes &C0-&FF cannot be generated from keyboard ;but are recognised by OS ; E525 CMP #&01 ;is it 01 E527 BEQ &E594 ;if so expand as 'soft' key via E594 E529 PLA ;else get back original byte E52A BCC &E539 ;if above CMP generated Carry then code 0 must have ;been returned so E539 to ignore E52C AND #&0F ;else add ASCII to BASE key number so clear hi nybble E52E CLC ;clear carry E52F ADC &0265,Y ;add ASCII base E532 CLC ;clear carry E533 RTS ;and exit ; *********** ERROR MADE IN USING EDIT FACILITY *************************** RDCHGB E534 JSR &E86F ;produce bell E537 PLA ;get back A, buffer number E538 TAX ;X=buffer number ********get byte from buffer ******************************************** RDCHG1 E539 JSR &E460 ;get byte from buffer X E53C BCS &E593 ;if buffer empty E593 to exit E53E PHA ;else Push byte E53F CPX #&01 ;if not RS423 input buffer, skip past E541 BNE &E549 ; E543 JSR &E173 ; E546 LDX #&01 ;Restore X=1 (RS423 input buffer) E548 SEC ;set carry RDCHL E549 PLA ;get back original byte E54A BCC &E551 ;if carry clear (not RS423 input), skip past E54C LDY &0245 ;check RS423 mode (0 treat as keyboard) E54F BNE &E592 ;if not 0 ignore escapes etc., return with byte RDCHK E551 TAY ;Y=byte from input buffer E552 BPL &E592 ;if code is less that &80 it's simple so return with it E554 AND #&0F ;else clear high nybble E556 CMP #&0B ;if less than 11 then treat as top-bit keypress E558 BCC &E519 ;function key or special keys E55A ADC #&7B ;else add &7C (&7B +C) to convert codes &xB-&xF to &87-&8B E55C PHA ;Push A E55D LDA &027D ;get cursor editing status E560 BNE &E515 ;not cursor editing, treat as top-bit keypress E562 LDA &027C ;else get character destination status ;Bit 0 enables RS423 driver ;Bit 1 disables VDU driver ;Bit 2 disables printer driver ;Bit 3 enables printer independent of CTRL B or CTRL C ;Bit 4 disables spooled output ;Bit 5 not used ;Bit 6 disables printer driver unless VDU 1 precedes character ;Bit 7 not used E565 ROR ;get bit 1 into carry E566 ROR ; E567 PLA ; E568 BCS &E539 ;if carry is set, VDU disabled, loop back for another keypress E56A CMP #&87 ;else is it COPY key E56C BEQ &E5A6 ;jump with COPY key E56E TAY ;else Y=A E56F TXA ;A=X E570 PHA ;Push X E571 TYA ;get back Y E572 JSR &D8CE ;move edit cursor E575 PLA ;restore X E576 TAX ;and drop through to loop back for another keypress RDCHG E577 BIT &025F ;check econet RDCH flag E57A BPL &E581 ;if not set goto E581 E57C LDA #&06 ;else Econet function 6 JMINET E57E JMP (&0224) ;Jump to NETV, Econet vector ********* get byte from key string ************************************** ;on entry 0268 contains key length ;and 02C9 key string pointer to next byte RDCHNM E581 LDA &0268 ;get length of keystring E584 BEQ &E539 ;if 0 E539 get a character from the buffer E586 LDY &02C9 ;get soft key expansion pointer E589 LDA &0B01,Y ;get character from string E58C INC &02C9 ;increment pointer E58F DEC &0268 ;decrement length ************** exit with carry clear ************************************ RDCHGC E592 CLC ; RDCHG0 E593 RTS ;exit ; *** expand soft key strings ********************************************* Y=pointer to sring number GKEY E594 PLA ;restore original code E595 AND #&0F ;blank high nybble to get function key number E597 TAY ;Y=A E598 JSR &E3A8 ;get string length in A E59B STA &0268 ;and store it E59E LDA &0B00,Y ;get start point E5A1 STA &02C9 ;and store it E5A4 BNE &E577 ;if not 0 then get byte via E577 and exit *********** deal with COPY key ****************************************** RDCHGY E5A6 TXA ;A=X E5A7 PHA ;Push A E5A8 JSR &D905 ;read a character from the screen E5AB TAY ;Y=A E5AC BEQ &E534 ;if not valid A=0 so BEEP E5AE PLA ;else restore X E5AF TAX ; E5B0 TYA ;and Y E5B1 CLC ;clear carry E5B2 RTS ;and exit ************************************************************************* * * * OSBYTE DISPATCH TABLE * * * ************************************************************************* BYTET E5B3 DW &E821 ;OSBYTE 0 (&E821) OS Version E5B5 DW &E988 ;OSBYTE 1 (&E988) User flag E5B7 DW &E6D3 ;OSBYTE 2 (&E6D3) Input source E5B9 DW &E997 ;OSBYTE 3 (&E997) Output dests E5BB DW &E997 ;OSBYTE 4 (&E997) Cursor keys state E5BD DW &E976 ;OSBYTE 5 (&E976) Printer destination E5BF DW &E988 ;OSBYTE 6 (&E988) Printer ignore character E5C1 DW &E98B ;OSBYTE 7 (&E68B) Serial input baud E5C3 DW &E689 ;OSBYTE 8 (&E689) Serial output baud E5C5 DW &E6B0 ;OSBYTE 9 (&E6B0) E5C7 DW &E6B2 ;OSBYTE 10 (&E6B2) E5C9 DW &E995 ;OSBYTE 11 (&E995) E5CB DW &E98C ;OSBYTE 12 (&E98C) E5CD DW &E6F9 ;OSBYTE 13 (&E6F9) Disable event E5CF DW &E6FA ;OSBYTE 14 (&E6FA) Enable event E5D1 DW &F0A8 ;OSBYTE 15 (&F0A8) Flush buffer E5D3 DW &E706 ;OSBYTE 16 (&E706) E5D5 DW &DE8C ;OSBYTE 17 (&DE8C) E5D7 DW &E9C8 ;OSBYTE 18 (&E9C8) E5D9 DW &E9B6 ;OSBYTE 19 (&E9B6) E5DB DW &CD07 ;OSBYTE 20 (&CD07) Explode font E5DD DW &F0B4 ;OSBYTE 21 (&F0B4) Flush buffer BYTET2 E5DF DW &E86C ;OSBYTE 117 (&E86C) E5E1 DW &E9D9 ;OSBYTE 118 (&E9D9) E5E3 DW &E275 ;OSBYTE 119 (&E275) E5E5 DW &F045 ;OSBYTE 120 (&F045) E5E7 DW &F0CF ;OSBYTE 121 (&F0CF) E5E9 DW &F0CD ;OSBYTE 122 (&F0CD) E5EB DW &E197 ;OSBYTE 123 (&E197) E5ED DW &E673 ;OSBYTE 124 (&E673) Clear Escape state E5EF DW &E674 ;OSBYTE 125 (&E674) Set Escape state E5F1 DW &E65C ;OSBYTE 126 (&E65C) Acknowledge Escape E5F3 DW &E035 ;OSBYTE 127 (&E035) EOF E5F5 DW &E74F ;OSBYTE 128 (&E74F) ADVAL E5F7 DW &E713 ;OSBYTE 129 (&E713) INKEY E5F9 DW &E729 ;OSBYTE 130 (&E729) E5FB DW &F085 ;OSBYTE 131 (&F085) E5FD DW &D923 ;OSBYTE 132 (&D923) E5FF DW &D926 ;OSBYTE 133 (&D926) E601 DW &D647 ;OSBYTE 134 (&D647) E603 DW &D7C2 ;OSBYTE 135 (&D7C2) E605 DW &E657 ;OSBYTE 136 (&E657) E607 DW &E67F ;OSBYTE 137 (&E67F) E609 DW &E4AF ;OSBYTE 138 (&E4AF) Insert into buffer E60B DW &E034 ;OSBYTE 139 (&E034) E60D DW &F135 ;OSBYTE 140 (&F135) E60F DW &F135 ;OSBYTE 141 (&F135) E611 DW &DBE7 ;OSBYTE 142 (&DBE7) Select language E613 DW &F168 ;OSBYTE 143 (&F168) Service call E615 DW &EAE3 ;OSBYTE 144 (&EAE3) *TV E617 DW &E460 ;OSBYTE 145 (&E460) E619 DW &FFAA ;OSBYTE 146 (&FFAA) E61B DW &EAF4 ;OSBYTE 147 (&EAF4) E61D DW &FFAE ;OSBYTE 148 (&FFAE) E61F DW &EAF9 ;OSBYTE 149 (&EAF9) E621 DW &FFB2 ;OSBYTE 150 (&FFB2) E623 DW &EAFE ;OSBYTE 151 (&EAFE) E625 DW &E45B ;OSBYTE 152 (&E45B) E627 DW &E4F3 ;OSBYTE 153 (&E4F3) E629 DW &E9FF ;OSBYTE 154 (&E9FF) E62B DW &EA10 ;OSBYTE 155 (&EA10) E62D DW &E17C ;OSBYTE 156 (&E17C) E62F DW &FFA7 ;OSBYTE 157 (&FFA7) E631 DW &EE6D ;OSBYTE 158 (&EE6D) E633 DW &EE7F ;OSBYTE 159 (&EE7F) E635 DW &E9C0 ;OSBYTE 160 (&E9C0) Read VDU variable E637 DW &E99C ;OSBYTE 166+ Read/Write OSBYTE variable E639 DW &E659 ;OSWORD &E0+ ************************************************************************* * * * OSWORD DISPATCH TABLE * * * ************************************************************************* WORDT E63B DW &E902 ;OSWORD 0 (&E902) Read input line E63D DW &E8D5 ;OSWORD 1 (&E8D5) Read TIME E63F DW &E8E8 ;OSWORD 2 (&E8E8) Write TIME E641 DW &E8D1 ;OSWORD 3 (&E8D1) Read interval timer E643 DW &E8E4 ;OSWORD 4 (&E8E4) Write interval timer E645 DW &E803 ;OSWORD 5 (&E803) Read I/O memory E647 DW &E80B ;OSWORD 6 (&E80B) Write I/O memory E649 DW &E82D ;OSWORD 7 (&E82D) SOUND E64B DW &E8AE ;OSWORD 8 (&E8AE) ENVELOPE E64D DW &C735 ;OSWORD 9 (&C735) POINT E64F DW &CBF3 ;OSWORD 10 (&CBF3) Read character bitmap E651 DW &C748 ;OSWORD 11 (&C748) Read palette E653 DW &C8E0 ;OSWORD 12 (&C8E0) Write palette E655 DW &C5CE ;OSWORD 13 (&D5CE) Read graphics coordinates ************************************************************************* * * * OSBYTE 136 Execute Code via User Vector * * * * *CODE effectively * * * ************************************************************************* MCODE E657 LDA #00 ;A=0 ************************************************************************* * * * *LINE entry * * * ************************************************************************* JMIUSR E659 JMP (&0200) ;Jump via USERV ************************************************************************* * * * OSBYTE 126 Acknowledge detection of ESCAPE condition * * * ************************************************************************* CESCP E65C LDX #&00 ;X=0 E65E BIT &FF ;if bit 7 not set there is no ESCAPE condition E660 BPL &E673 ;so E673 E662 LDA &0276 ;else get ESCAPE Action, if this is 0 ;Clear ESCAPE ;close EXEC files ;purge all buffers ;reset VDU paging counter E665 BNE &E671 ;else do none of the above E667 CLI ;allow interrupts E668 STA &0269 ;number of lines printed since last halt in paged ;mode = 0 E66B JSR &F68D ;close any open EXEC files E66E JSR &F0AA ;clear all buffers CESCX E671 LDX #&FF ;X=&FF to indicate ESCAPE acknowledged ************************************************************************* * * * OSBYTE 124 Clear ESCAPE condition * * * ************************************************************************* CESCPC E673 CLC ;clear carry ************************************************************************* * * * OSBYTE 125 Set ESCAPE flag * * * ************************************************************************* CESCPS E674 ROR &FF ;clear bit 7 of ESCAPE flag E676 BIT &027A ;read bit 7 of Tube flag E679 BMI &E67C ;if set TUBE exists so E67C E67B RTS ;else RETURN ; TESC E67C JMP &0403 ;Jump to Tube entry point ************************************************************************* * * * OSBYTE 137 Turn on Tape motor * * * ************************************************************************* MOTOR E67F LDA &0282 ;get serial ULA control setting E682 TAY ;Y=A E683 ROL ;rotate left to get bit 7 into carry E684 CPX #&01 ;if X=1 then user wants motor on so CARRY set else ;carry is cleared E686 ROR ;put carry back in control RAM copy E687 BVC &E6A7 ;if bit 6 is clear then cassette is selected ;so write to control register and RAM copy ************************************************************************* * * * OSBYTE 08/07 set serial baud rates * * * ************************************************************************* ; on entry X=baud rate ; A=&07 receive -> &07 ; A=&08 transmit -> &38 ; ; *FX 8 entry point TXBAUD E689 LDA #&38 ; Convert 8 to &38 for *FX8 ; *FX 7 entry point RXBAUD E68B EOR #&3F ;convert to mask, rx 7 to &38, tx 8 to &07 E68D STA &FA ;store result E68F LDY &0282 ;get serial ULA control register setting E692 CPX #&09 ;is it 9 or more? E694 BCS &E6AD ;if so exit, return current ULA setting E696 AND &E9AD,X ;AND with byte from lookup table E699 STA &FB ;store it, %00abc000 for rx, &00000def for tx E69B TYA ;get current serial ULA register in Y into A E69C ORA &FA ;ORA with rx/tx mask, %st111xyz for rx, %stuvw111 for tx E69E EOR &FA ;toggle them to zero, %st000xyz for rx, %stuvw000 for tx E6A0 ORA &FB ;merge in lookup bits, %stabcxyz for rx, %stuvwdef for tx E6A2 ORA #&40 ;set bit 6 for serial, %x1abcxyz for rx, %s1uvwdef for tx ; ;not needed as table has bit 6 set, LE68B could be EOR #&7F E6A4 EOR &025D ;toggle bit 6 via cassette/RS423 flag STASP E6A7 STA &0282 ;store serial ULA flag E6AA STA &FE10 ;and write to control register BAUD1 TYATAX E6AD TYA ;put Y in A to return old contents in X BADEVT E6AE TAX ;return old setting in X E6AF RTS ;and return OS SERIES VII GEOFF COX ************************************************************************* * * * OSBYTE 9 Duration of first colour * * * ************************************************************************* ;on entry Y=0, X=new value FMRK E6B0 INY ;Y is incremented to 1 E6B1 CLC ;clear carry ************************************************************************* * * * OSBYTE 10 Duration of second colour * * * ************************************************************************* ;on entry Y=0 or 1 if from FX 9 call, X=new value FSPC E6B2 LDA &0252,Y ;get mark period count E6B5 PHA ;push it E6B6 TXA ;get new count E6B7 STA &0252,Y ;store it E6BA PLA ;get back original value E6BB TAY ;put it in Y E6BC LDA &0251 ;get value of flash counter E6BF BNE &E6D1 ;if not zero E6D1 E6C1 STX &0251 ;else restore old value E6C4 LDA &0248 ;get current video ULA control register setting E6C7 PHP ;push flags E6C8 ROR ;rotate bit 0 into carry, carry into bit 7 E6C9 PLP ;get back flags E6CA ROL ;rotate back carry into bit 0 E6CB STA &0248 ;store it in RAM copy E6CE STA &FE20 ;and ULA control register FMS1 E6D1 BVC &E6AD ;return old setting in X ************************************************************************* * * * OSBYTE 2 select input stream * * * ************************************************************************* ;on input X contains stream number SELIN E6D3 TXA ;A=X E6D4 AND #&01 ;blank out bits 1 - 7 E6D6 PHA ;push A E6D7 LDA &0250 ;and get current ACIA control setting E6DA ROL ;Bit 7 into carry E6DB CPX #&01 ;if X>=1 then E6DD ROR ;bit 7 of A=1 E6DE CMP &0250 ;compare this with ACIA control setting E6E1 PHP ;push flags E6E2 STA &0250 ;put A into ACIA control setting E6E5 STA &FE08 ;and write to control register E6E8 JSR &E173 ;set up RS423 buffer E6EB PLP ;get flags back E6EC BEQ &E6F1 ;if new setting different from old E6F1 else E6EE BIT &FE09 ;set bit 6 and 7 SELIN1 E6F1 LDX &0241 ;get current input buffer number E6F4 PLA ;get A back E6F5 STA &0241 ;store it E6F8 RTS ;and return ************************************************************************* * * * OSBYTE 13 disable events * * * ************************************************************************* ;X contains event number 0-9 MDSABL E6F9 TYA ;Y=0 A=0 ************************************************************************* * * * OSBYTE 14 enable events * * * ************************************************************************* ;X contains event number 0-9 ENABLE E6FA CPX #&0A ;if X>9 E6FC BCS &E6AE ;goto E6AE for exit E6FE LDY &02BF,X ;else get event enable flag E701 STA &02BF,X ;store new value in flag E704 BVC &E6AD ;and exit via E6AD ************************************************************************* * * * OSBYTE 16 Select A/D channel * * * ************************************************************************* ;X contains channel number or 0 if disable conversion SETADC E706 BEQ &E70B ;if X=0 then E70B E708 JSR &DE8C ;start conversion SETAD1 E70B LDA &024D ;get current maximum ADC channel number E70E STX &024D ;store new value E711 TAX ;put old value in X E712 RTS ;and exit ************************************************************************* * * * OSBYTE 129 Read key within time limit * * * ************************************************************************* ;X and Y contains either time limit in centi seconds Y=&7F max ; or Y=&FF and X=-ve INKEY value INKY E713 TYA ;A=Y E714 BMI &E721 ;if Y=&FF the E721 E716 CLI ;else allow interrupts E717 JSR &DEBB ;and go to timed routine E71A BCS &E71F ;if carry set then E71F E71C TAX ;then X=A E71D LDA #&00 ;A=0 INKY1 E71F TAY ;Y=A E720 RTS ;and return ; ;scan keyboard RDKY E721 TXA ;A=X E722 EOR #&7F ;convert to keyboard scan code, keeping bit 7 E724 TAX ;X=A E725 JSR &F068 ;then scan keyboard E728 ROL ;put bit 7 into carry RMHA BADADC E729 LDX #&FF ;X=&FF E72B LDY #&FF ;Y=&FF E72D BCS &E731 ;if bit 7 of A was set goto E731 (RTS) E72F INX ;else X=0 E730 INY ;and Y=0 RDKY0 E731 RTS ;and exit ********** check occupancy of input or free space of output buffer ******* ;X=buffer number ;Buffer number Address Flag Out pointer In pointer ;0=Keyboard 3E0-3FF 2CF 2D8 2E1 ;1=RS423 Input A00-AFF 2D0 2D9 2E2 ;2=RS423 output 900-9BF 2D1 2DA 2E3 ;3=printer 880-8BF 2D2 2DB 2E4 ;4=sound0 840-84F 2D3 2DC 2E5 ;5=sound1 850-85F 2D4 2DD 2E6 ;6=sound2 860-86F 2D5 2DE 2E7 ;7=sound3 870-87F 2D6 2DF 2E8 ;8=speech 8C0-8FF 2D7 2E0 2E9 BUFN E732 TXA ;buffer number in A E733 EOR #&FF ;invert it E735 TAX ;X=A E736 CPX #&02 ;is X>1 JMICNT E738 CLV ;clear V flag E739 BVC &E73E ;and goto E73E count buffer JMIPUR E73B BIT &D9B7 ;set V JMICNP E73E JMP (&022E) ;CNPV defaults to E1D1 ************* check RS423 input buffer ************************************ CNTRS E741 SEC E742 LDX #&01 ;X=1 to point to buffer E744 JSR &E738 ;and count it E747 CPY #&01 ;if the hi byte of the answer is 1 or more E749 BCS &E74E ;then Return E74B CPX &025B ;else compare with minimum buffer space CNTRS0 E74E RTS ;and exit ************************************************************************* * * * OSBYTE 128 READ ADC CHANNEL * * * ************************************************************************* ;ON Entry: X=0 Exit Y contains number of last channel converted ; X=channel number X,Y contain 16 bit value read from channe ; X<0 Y=&FF X returns information about various buffers ; X=&FF (keyboard) X=number of characters in buffer ; X=&FE (RS423 input) X=number of characters in buffer ; X=&FD (RS423 output) X=number of empty spaces in buffer ; X=&FC (Printer) X=number of empty spaces in buffer ; X=&FB (sound 0) X=number of empty spaces in buffer ; X=&FA (sound 1) X=number of empty spaces in buffer ; X=&F9 (sound 2) X=number of empty spaces in buffer ; X=&F8 (sound 3) X=number of empty spaces in buffer ; X=&F7 (Speech) X=number of empty spaces in buffer RADCX E74F BMI &E732 ;if X is -ve then E732 count spaces E751 BEQ &E75F ;if X=0 then E75F E753 CPX #&05 ;else check for Valid channel E755 BCS &E729 ;if not E729 set X & Y to 0 and exit (*BUG* should pass to ROMs) E757 LDY &02B9,X ;get conversion values for channel of interest Hi & E75A LDA &02B5,X ;lo byte E75D TAX ;X=lo byte E75E RTS ;and exit FIRE E75F LDA &FE40 ;read system VIA port B E762 ROR ;move high nybble to low E763 ROR ; E764 ROR ; E765 ROR ; E766 EOR #&FF ;and invert it E768 AND #&03 ;isolate the FIRE buttons E76A LDY &02BE ;get analogue system flag byte E76D STX &02BE ;store X here E770 TAX ;A=X bits 0 and 1 indicate fire buttons E771 RTS ;and return ************************************************************************** ************************************************************************** ** ** ** OSBYTE DEFAULT ENTRY POINT ** ** ** ** pointed to by default BYTEV ** ** ** ************************************************************************** ************************************************************************** BYTE E772 PHA ;save A E773 PHP ;save Processor flags E774 SEI ;disable interrupts E775 STA &EF ;store A,X,Y in zero page E777 STX &F0 ; E779 STY &F1 ; E77B LDX #&07 ;X=7 to signal osbyte is being attempted E77D CMP #&75 ;if A=0-116 E77F BCC &E7C2 ;then E7C2 E781 CMP #&A1 ;if A<161 E783 BCC &E78E ;then E78E E785 CMP #&A6 ;if A=161-165 E787 BCC &E7C8 ;then EC78 E789 CLC ;clear carry WORDU E78A LDA #&A1 ;A=&A1 E78C ADC #&00 ; ********* process osbyte calls 117 - 160 ***************************** BYTEHI E78E SEC ;set carry E78F SBC #&5F ;convert to &16 to &41 (22-65) BYTE1 E791 ASL ;double it (44-130) E792 SEC ;set carry BYTE2 E793 STY &F1 ;store Y E795 TAY ;Y=A E796 BIT &025E ;read econet intercept flag E799 BPL &E7A2 ;if no econet intercept required E7A2 E79B TXA ;else A=X E79C CLV ;V=0 E79D JSR &E57E ; to JMP via ECONET vector E7A0 BVS &E7BC ;if return with V set E7BC NET1 E7A2 LDA &E5B4,Y ;get address from table E7A5 STA &FB ;store it as hi byte E7A7 LDA &E5B3,Y ;repeat for lo byte E7AA STA &FA ; E7AC LDA &EF ;restore A E7AE LDY &F1 ;Y E7B0 BCS &E7B6 ;if carry is set E7B6 E7B2 LDY #&00 ;else E7B4 LDA (&F0),Y ;get value from address pointed to by &F0/1 (Y,X) OSB E7B6 SEC ;set carry E7B7 LDX &F0 ;restore X E7B9 JSR &F058 ;call &FA/B OSRET E7BC ROR ;C=bit 0 E7BD PLP ;get back flags E7BE ROL ;bit 0=Carry E7BF PLA ;get back A E7C0 CLV ;clear V E7C1 RTS ;and exit *************** Process OSBYTE CALLS BELOW &75 ************************** BYTELO E7C2 LDY #&00 ;Y=0 E7C4 CMP #&16 ;if A<&16 E7C6 BCC &E791 ;goto E791 WORD0 BYTE0 E7C8 PHP ;push flags E7C9 PHP ;push flags EXTSND E7CA PLA ;pull flags E7CB PLA ;pull flags E7CC JSR &F168 ;offer paged ROMs service 7/8 unrecognised osbyte/word E7CF BNE &E7D6 ;if roms don't recognise it then E7D6 E7D1 LDX &F0 ;else restore X E7D3 JMP &E7BC ;and exit BADOSC E7D6 PLP ;else pull flags E7D7 PLA ;and A E7D8 BIT &D9B7 ;set V and C E7DB RTS ;and return MSGON E7DC LDA &EB ;read cassette critical flag bit 7 = busy E7DE BMI &E812 ;if busy then EB12 E7E0 LDA #&08 ;else A=8 to check current Catalogue status E7E2 AND &E2 ;by ANDing with CFS status flag E7E4 BNE &E7EA ;if not set (not in use) then E7EA RTS E7E6 LDA #&88 ;A=%10001000 E7E8 AND &BB ;AND with FS options (short msg bits) MSGONX E7EA RTS ;RETURN ************************************************************************** ************************************************************************** ** ** ** OSWORD DEFAULT ENTRY POINT ** ** ** ** pointed to by default WORDV ** ** ** ************************************************************************** ************************************************************************** WORD E7EB PHA ;Push A E7EC PHP ;Push flags E7ED SEI ;disable interrupts E7EE STA &EF ;store A,X,Y E7F0 STX &F0 ; E7F2 STY &F1 ; E7F4 LDX #&08 ;X=8 E7F6 CMP #&E0 ;if A=>224 E7F8 BCS &E78A ;then E78A with carry set E7FA CMP #&0E ;else if A=>14 E7FC BCS &E7C8 ;else E7C8 with carry set pass to ROMs & exit E7FE ADC #&44 ;add to form pointer to table E800 ASL ;double it E801 BCC &E793 ;goto E793 ALWAYS (carry clear E7F8) ;this reads bytes from table and enters routine ************************************************************************* * * * OSWORD 05 ENTRY POINT * * * * read a byte from I/O memory * * * ************************************************************************* ;block of 4 bytes set at address pointed to by 00F0/1 (Y,X) ;XY +0 ADDRESS of byte ; +4 on exit byte read PEEK E803 JSR &E815 ;set up address of data block E806 LDA (&F9,X) ;get byte E808 STA (&F0),Y ;store it E80A RTS ;exit ************************************************************************* * * * OSWORD 06 ENTRY POINT * * * * write a byte to I/O memory * * * ************************************************************************* ;block of 5 bytes set at address pointed to by 00F0/1 (Y,X) ;XY +0 ADDRESS of byte ; +4 byte to be written POKE E80B JSR &E815 ;set up address E80E LDA (&F0),Y ;get byte E810 STA (&F9,X) ;store it MSGONY E812 LDA #&00 ;a=0 E814 RTS ;exit ******************** set up address ********************************* PEKBYT E815 STA &FA ;&FA=A E817 INY ;Y=1 E818 LDA (&F0),Y ;get byte from block E81A STA &FB ;store it E81C LDY #&04 ;Y=4 RVER E81E LDX #&01 ;X=1 E820 RTS ;and exit ************************************************************************* * * * OSBYTE 00 ENTRY POINT * * * * read OS version number * * * ************************************************************************* VER E821 BNE &E81E ;if A <> 0 then exit with X=OS, else generate error E823 BRK ; E824 DB &F7 ;error number E825 DB 'OS 1.20' ;error message E82C BRK ************************************************************************* ************************************************************************* ** ** ** SOUND SYSTEM ** ** ** ************************************************************************* ************************************************************************* ------------------------------------------------------------------------- | | | OSWORD 07 - Make a sound | | | ------------------------------------------------------------------------- ; On entry, control block pointed to by &F0/1 ; Y=0 on entry ; XY +0 Channel - &hsfc for Hold, Sync, Flush, Channel ; 2 Amplitude/Envelope ; 4 Pitch ; 6 Duration NOTE E82D INY E82E LDA (&F0),Y ; Get channel high byte E830 CMP #&FF E832 BEQ &E88D ; Channel &FFxx, speech command E834 CMP #&20 ; Is channel>=&20 ? E836 LDX #&08 ; Prepare X=8 for unrecognised OSWORD call E838 BCS &E7CA ; Pass to sideways ROMs for channel &2000+ E83A DEY ; Point back to channel low byte E83B JSR &E8C9 ; Get Channel 0-3, and Cy if >=&10 for Flush E83E ORA #&04 ; Convert to buffer number 4-7 E840 TAX E841 BCC &E848 ; If not Flush, skip past E843 JSR &E1AE ; Flush buffer E846 LDY #&01 ; Point back to channel high byte NOTENF E848 JSR &E8C9 ; Get Sync 0-3, and Cy if >=&10 for Hold E84B STA &FA ; Save Sync in &FA E84D PHP ; Stack flags E84E LDY #&06 E850 LDA (&F0),Y ; Get Duration byte E852 PHA ; and stack it E853 LDY #&04 E855 LDA (&F0),Y ; Get pitch byte E857 PHA ; and stack it E858 LDY #&02 ; E85A LDA (&F0),Y ; Get amplitude/envelope byte E85C ROL ; Move Hold into bit 0 E85D SEC ;set carry E85E SBC #&02 ;subract 2 E860 ASL ;multiply by 4 E861 ASL ; E862 ORA &FA ;add S byte (0-3) ; At this point, ; b7, 0=envelope, 1=volume ; b6-3, envelope-1 or volume+15 ; b2, Hold ; b1-0, Sync E864 JSR &E1F8 ; Wait for space and insert into buffer E867 BCC &E887 ; Byte inserted, jump to insert the rest SPESC E869 PLA ; Insert aborted (Escape pressed), drop stacked pitch E86A PLA ; Drop stacked duration E86B PLP ; Restore flags ; And exit ------------------------------------------------------------------------- | | | OSBYTE 117 - Read VDU status | | | ------------------------------------------------------------------------- VDUST E86C LDX &D0 ;get VDU status byte in X E86E RTS ;and return ************* set up sound data for Bell ******************************** BELL E86F PHP ;push P E870 SEI ;bar interrupts E871 LDA &0263 ;get bell channel number in A E874 AND #&07 ; (bits 0-3 only set) E876 ORA #&04 ;set bit 2 E878 TAX ;X=A = bell channel number +4=buffer number E879 LDA &0264 ;get bell amplitude/envelope number E87C JSR &E4B0 ;store it in buffer pointed to by X E87F LDA &0266 ;get bell duration E882 PHA ;save it E883 LDA &0265 ;get bell frequency E886 PHA ;save it ; Insert sound pitch and duration into sound buffer ; NOTE2 E887 SEC ; Set carry E888 ROR &0800,X ; Set bit 7 of channel flags to indicate it's active E88B BMI &E8A4 ; Jump forward to insert pitch and duration ------------------------------------------------------------------------- | | | SOUND &FFxx - Speech System | | | ------------------------------------------------------------------------- ; On entry, control block pointed to by &F0/1 ; Y=1 on entry ; XY +0 Channel - &FFxx - xx=Speech command ; 2 Word number/Address ; 4 Ignored ; 6 Ignored SPEECH E88D PHP ; Save flags E88E INY ; Y=2 E88F LDA (&F0),Y ; Get word number low byte E891 PHA ; and stack it E892 INY ; Y=3 E893 LDA (&F0),Y ; Get word number high byte E895 PHA ; and stack it E896 LDY #&00 ; Y=0 E898 LDA (&F0),Y ; Get speech command E89A LDX #&08 ; X=8 for Speech buffer E89C JSR &E1F8 ; Insert speech command into speech buffer E89F BCS &E869 ; Buffer insert aborted (Escape pressed), drop stack and abandon E8A1 ROR &02D7 ; Clear bit 7 of speech buffer busy flag ; Insert two bytes into buffer ; NOTE3 E8A4 PLA ; Get word number high byte or pitch back E8A5 JSR &E4B0 ; Insert into speech buffer or sound buffer E8A8 PLA ; Get word number low byte or duration back E8A9 JSR &E4B0 ; Insert into speech buffer or sound buffer E8AC PLP ; Restore flags E8AD RTS ; and return ************************************************************************* * * * OSWORD 08 - Define Envelope * * * ************************************************************************* ; On entry, control block pointed to by &F0/1 ; Y=0 on entry ; A=envelope number from (&F0),0 ;XY +0 Envelope number, also in A ; 1 bits 0-6 length of each step in centi-secsonds bit 7=0 auto repeat ; 2 change of Pitch per step (-128-+127) in section 1 ; 3 change of Pitch per step (-128-+127) in section 2 ; 4 change of Pitch per step (-128-+127) in section 3 ; 5 number of steps in section 1 (0-255) ; 6 number of steps in section 2 (0-255) ; 7 number of steps in section 3 (0-255) ; 8 change of amplitude per step during attack phase (-127 to +127) ; 9 change of amplitude per step during decay phase (-127 to +127) ; 10 change of amplitude per step during sustain phase (-127 to +127) ; 11 change of amplitude per step during release phase (-127 to +127) ; 12 target level at end of attack phase (0-126) ; 13 target level at end of decay phase (0-126) NVLP E8AE SBC #&01 ;set up appropriate displacement to storage area E8B0 ASL ;A=(A-1)*16 or 15 E8B1 ASL ; E8B2 ASL ; E8B3 ASL ; E8B4 ORA #&0F ; E8B6 TAX ;X=A E8B7 LDA #&00 ;A=0 E8B9 LDY #&10 ;Y=&10 NVLP1 E8BB CPY #&0E ;is Y>=14?? E8BD BCS &E8C1 ;yes then E8C1 E8BF LDA (&F0),Y ;else get byte from parameter block NVLP2 E8C1 STA &08C0,X ;and store it in appropriate area E8C4 DEX ;decrement X E8C5 DEY ;Decrement Y E8C6 BNE &E8BB ;if not 0 then do it again E8C8 RTS ;else exit ;note that envelope number is NOT transferred ; NOTEX E8C9 LDA (&F0),Y ;get byte E8CB CMP #&10 ;is it greater than 15, if so set carry E8CD AND #&03 ;AND 3 to keep channel in bits 0-1 E8CF INY ;increment Y, step to next parameter byte E8D0 RTS ;and exit ************************************************************************* * * * OSWORD 03 ENTRY POINT * * * * read interval timer * * * ************************************************************************* ;F0/1 points to block to store data RITM E8D1 LDX #&0F ;X=&F displacement from clock to timer E8D3 BNE &E8D8 ;jump to E8D8 ************************************************************************* * * * OSWORD 01 ENTRY POINT * * * * read system clock * * * ************************************************************************* ;F0/1 points to block to store data RATM E8D5 LDX &0283 ;X=current system clock store pointer RTM E8D8 LDY #&04 ;Y=4 RTM1 E8DA LDA &028D,X ;read byte E8DD STA (&F0),Y ;store it in parameter block E8DF INX ;X=x+1 E8E0 DEY ;Y=Y-1 E8E1 BPL &E8DA ;if Y>0 then do it again WTM0 E8E3 RTS ;else exit ************************************************************************* * * * OSWORD 04 ENTRY POINT * * * * write interval timer * * * ************************************************************************* F0/1 points to block to store data WITM E8E4 LDA #&0F ;offset between clock and timer E8E6 BNE &E8EE ;jump to E8EE ALWAYS!! ************************************************************************* * * * OSWORD 02 ENTRY POINT * * * * write system clock * * * ************************************************************************* F0/1 points to block to store data WATM E8E8 LDA &0283 ;get current clock store pointer E8EB EOR #&0F ;and invert to get inactive timer E8ED CLC ;clear carry WTM E8EE PHA ;store A E8EF TAX ;X=A E8F0 LDY #&04 ;Y=4 WTM1 E8F2 LDA (&F0),Y ;and transfer all 5 bytes E8F4 STA &028D,X ;to the clock or timer E8F7 INX ; E8F8 DEY ; E8F9 BPL &E8F2 ;if Y>0 then E8F2 E8FB PLA ;get back stack E8FC BCS &E8E3 ;if set (write to timer) E8E3 exit E8FE STA &0283 ;write back current clock store E901 RTS ;and exit ************************************************************************* * * * OSWORD 00 ENTRY POINT * * * * read line from current input to memory * * * ************************************************************************* ;F0/1 points to parameter block ; +0/1 Buffer address for input ; +2 Maximum line length ; +3 minimum acceptable ASCII value ; +4 maximum acceptable ASCII value RDLN E902 LDY #&04 ;Y=4 BUFFPR E904 LDA (&F0),Y ;transfer bytes 4,3,2 to 2B3-2B5 E906 STA &02B1,Y ; E909 DEY ;decrement Y E90A CPY #&02 ;until Y=1 E90C BCS &E904 ; E90E LDA (&F0),Y ;get address of input buffer E910 STA &E9 ;store it in &E9 as temporary buffer E912 DEY ;decrement Y E913 STY &0269 ;Y=0 store in print line counter for paged mode E916 LDA (&F0),Y ;get lo byte of address E918 STA &E8 ;and store in &E8 E91A CLI ;allow interrupts E91B BCC &E924 ;Jump to E924 BUFEXH E91D LDA #&07 ;A=7 BUFIGN E91F DEY ;decrement Y BUFECH E920 INY ;increment Y BFWRCH E921 JSR OSWRCH ;and call OSWRCH BUFFIN E924 JSR OSRDCH ;else read character from input stream E927 BCS &E972 ;if carry set then illegal character or other error ;exit via E972 E929 TAX ;X=A E92A LDA &027C ;A=&27C get character destination status E92D ROR ;put VDU driver bit in carry E92E ROR ;if this is 1 VDU driver is disabled E92F TXA ;X=A E930 BCS &E937 ;if Carry set E937 E932 LDX &026A ;get number of items in VDU queque E935 BNE &E921 ;if not 0 output character and loop round again BUFNQ E937 CMP #&7F ;if character is not delete E939 BNE &E942 ;goto E942 E93B CPY #&00 ;else is Y=0 E93D BEQ &E924 ;and goto E924 E93F DEY ;decrement Y E940 BCS &E921 ;and if carry set E921 to output it BUFNXA E942 CMP #&15 ;is it delete line &21 E944 BNE &E953 ;if not E953 E946 TYA ;else Y=A, if its 0 we are still reading first ;character E947 BEQ &E924 ;so E924 E949 LDA #&7F ;else output DELETES BUFCNL E94B JSR OSWRCH ;until Y=0 E94E DEY ; E94F BNE &E94B ; E951 BEQ &E924 ;then read character again BUFNXB E953 STA (&E8),Y ;store character in designated buffer E955 CMP #&0D ;is it CR? E957 BEQ &E96C ;if so E96C E959 CPY &02B3 ;else check the line length E95C BCS &E91D ;if = or greater loop to ring bell E95E CMP &02B4 ;check minimum character E961 BCC &E91F ;if less than minimum backspace E963 CMP &02B5 ;check maximum character E966 BEQ &E920 ;if equal E920 E968 BCC &E920 ;or less E920 E96A BCS &E91F ;then JUMP E91F BUFEND E96C JSR OSNEWL ;output CR/LF E96F JSR &E57E ;call Econet vector with A=13 BUFESC E972 LDA &FF ;A=ESCAPE FLAG E974 ROL ;put bit 7 into carry E975 RTS ;and exit routine ************************************************************************* * * * OSBYTE 05 ENTRY POINT * * * * SELECT PRINTER TYPE * * * ************************************************************************* SPRINT E976 CLI ;allow interrupts briefly E977 SEI ;bar interrupts E978 BIT &FF ;check if ESCAPE is pending E97A BMI &E9AC ;if it is E9AC E97C BIT &02D2 ;else check bit 7 buffer 3 (printer) E97F BPL &E976 ;if not empty bit 7=0 E976 E981 JSR &E1A4 ;check for user defined routine E984 LDY #&00 ;Y=0 E986 STY &F1 ;F1=0 ************************************************************************* * * * OSBYTE 01 ENTRY POINT * * * * READ/WRITE USER FLAG (&281) * * * * AND * * * * OSBYTE 06 ENTRY POINT * * * * SET PRINTER IGNORE CHARACTER * * * ************************************************************************* ; A contains osbyte number V2B156 E988 ORA #&F0 ;A=osbyte +&F0 E98A BNE &E99A ;JUMP to E99A ************************************************************************* * * * OSBYTE 0C ENTRY POINT * * * * SET KEYBOARD AUTOREPEAT RATE * * * ************************************************************************* ARRPT E98C BNE &E995 ;if &F0<>0 goto E995 E98E LDX #&32 ;if X=0 in original call then X=32 E990 STX &0254 ;to set keyboard autorepeat delay ram copy E993 LDX #&08 ;X=8 ************************************************************************* * * * OSBYTE 0B ENTRY POINT * * * * SET KEYBOARD AUTOREPEAT DELAY * * * ************************************************************************* ARDLY ARRPT1 E995 ADC #&CF ;A=A+&D0 (carry set) ************************************************************************* * * * OSBYTE 03 ENTRY POINT * * * * SELECT OUTPUT STREAM * * * * AND * * * * * * OSBYTE 04 ENTRY POINT * * * * ENABLE/DISABLE CURSOR EDITING * * * ************************************************************************* V2B34 E997 CLC ;clear carry E998 ADC #&E9 ;A=A+&E9 SETVX E99A STX &F0 ;store X ************************************************************************* * * * OSBYTE A6-FF ENTRY POINT * * * * READ/ WRITE SYSTEM VARIABLE OSBYTE NO. +&190 * * * ************************************************************************* SETV E99C TAY ;Y=A E99D LDA &0190,Y ;i.e. A=&190 +osbyte call number E9A0 TAX ;preserve this E9A1 AND &F1 ;new value = OLD value AND Y EOR X! E9A3 EOR &F0 ; E9A5 STA &0190,Y ;store it E9A8 LDA &0191,Y ;get value of next byte into A E9AB TAY ;Y=A SPR0 E9AC RTS ;and exit ******* SERIAL BAUD RATE LOOK UP TABLE ************************************ \ b0-b2=tx baud setting, b3-b5=rx baud setting, b6='serial, not CFS' \ Also 'is a SOUND buffer' table, b7=buffer is a SOUND buffer BUFTYP BAUDS E9AD DB &64 ; % 01100100 0 = 9600 buffer 0 = not SOUND E9AE DB &7F ; % 01111111 1 = 75 buffer 1 = not SOUND E9AF DB &5B ; % 01011011 2 = 150 buffer 2 = not SOUND E9B0 DB &6D ; % 01101101 3 = 300 buffer 3 = not SOUND E9B1 DB &C9 ; % 11001001 4 = 1200 buffer 4 = SOUND E9B2 DB &F6 ; % 11110110 5 = 2400 buffer 5 = SOUND E9B3 DB &D2 ; % 11010010 6 = 4800 buffer 6 = SOUND E9B4 DB &E4 ; % 11100100 7 = 9600 buffer 7 = SOUND E9B5 DB &40 ; % 01000000 8 = 19200 buffer 8 = not SOUND ************************************************************************* * * * OSBYTE &13 ENTRY POINT * * * * Wait for VSync * * * ************************************************************************* WFVS E9B6 LDA &0240 ;read vertical sync counter WFVS1 E9B9 CLI ;allow interrupts briefly E9BA SEI ;bar interrupts E9BB CMP &0240 ;has it changed? E9BE BEQ &E9B9 ;no then E9B9 ; falls through and reads VDU variable X ************************************************************************* * * * OSBYTE 160 ENTRY POINT * * * * READ VDU VARIABLE * * * ************************************************************************* ;X contains the variable number ; 0 =lefthand column in pixels current graphics window ; 2 =Bottom row in pixels current graphics window ; 4 =Right hand column in pixels current graphics window ; 6 =Top row in pixels current graphics window ; 8 =lefthand column in absolute characters current text window ; 9 =Bottom row in absolute characters current text window ;10 =Right hand column in absolute characters current text window ;11 =Top row in absolute characters current text window ;12-15 =current graphics origin in external coordinates ;16-19 =current graphics cursor in external coordina4es ;20-23 =old graphics cursor in internal coordinates ;24 =current text cursor in X and Y VDUVAR E9C0 LDY &0301,X ;get VDU variable hi E9C3 LDA &0300,X ;low E9C6 TAX ;X=low byte E9C7 RTS ;and exit ************************************************************************* * * * OSBYTE 18 ENTRY POINT * * * * RESET SOFT KEYS * * * ************************************************************************* RSOFT E9C8 LDA #&10 ;set consistency flag E9CA STA &0284 ; E9CD LDX #&00 ;X=0 STTA E9CF STA &0B00,X ;and wipe clean E9D2 INX ;soft key buffer E9D3 BNE &E9CF ;until X=0 again E9D5 STX &0284 ;zero consistency flag E9D8 RTS ;and exit ************************************************************************* * * * OSBYTE &76 (118) SET LEDs to Keyboard Status * * * ************************************************************************* ;osbyte entry with carry set ;called from &CB0E, &CAE3, &DB8B MOSTST E9D9 PHP ;save flags E9DA SEI ;disable interupts E9DB LDA #&40 ;switch on CAPS and SHIFT lock lights E9DD JSR &E9EA ;via subroutine E9E0 BMI &E9E7 ;if ESCAPE exists (M set) E9E7 E9E2 CLC ;else clear V and C E9E3 CLV ;before calling main keyboard routine to E9E4 JSR &F068 ;switch on lights and read Ctrl and Shift to A=%csxxxxxx SHIFTX E9E7 PLP ;restore IRQs E9E8 ROL ;rotate bit 7 (Ctrl) into Carry, and bit 6 (Shift) into Minus E9E9 RTS ;return to calling routine, A=%sxxxxxxx, C=Ctrl, M=Shift ; ***************** Turn on keyboard lights and Test Escape flag ************ ;called from &E1FE, &E9DD ; LEDSON E9EA BCC &E9F5 ;if carry clear E9EC LDY #&07 ;switch on shift lock light E9EE STY &FE40 ; E9F1 DEY ;Y=6 E9F2 STY &FE40 ;switch on Caps lock light LEDS0 E9F5 BIT &FF ;set minus flag if bit 7 of &00FF is set indicating E9F7 RTS ;that ESCAPE condition exists, then return ; ****************** Write A to SYSTEM VIA register B ************************* ;called from &CB6D, &CB73 CREG E9F8 PHP ;push flags E9F9 SEI ;disable interupts E9FA STA &FE40 ;write register B from Accumulator E9FD PLP ;get back flags E9FE RTS ;and exit ************************************************************************* * * * OSBYTE 154 (&9A) SET VIDEO ULA * * * ************************************************************************* VP0 E9FF TXA ;osbyte entry, X transferred to A thence to *******Set Video ULA control register **entry from VDU routines ************** ;called from &CBA6, &DD37 MVPRC0 EA00 PHP ;save flags EA01 SEI ;disable interupts EA02 STA &0248 ;save RAM copy of new parameter EA05 STA &FE20 ;write to control register EA08 LDA &0253 ;read space count EA0B STA &0251 ;set flash counter to this value EA0E PLP ;get back status EA0F RTS ;and return ************************************************************************* * * * OSBYTE &9B (155) write to palette register * * * ************************************************************************* ;entry X contains value to write VP1 EA10 TXA ;A=X MVPRC1 EA11 EOR #&07 ;convert to palette format EA13 PHP ; EA14 SEI ;prevent interrupts EA15 STA &0249 ;store as current palette setting EA18 STA &FE21 ;store actual colour in register EA1B PLP ;get back flags EA1C RTS ;and exit ************************************************************************* * GSINIT string initialisation * * F2/3 points to string offset by Y * * * * ON EXIT * * Z flag set indicates null string, * * Y points to first non-blank character * * A contains first non-blank character * ************************************************************************* GSINTC EA1D CLC ;clear carry GSINTS EA1E ROR &E4 ;Rotate moves carry to &E4 EA20 JSR &E03A ;skip spaces and get character EA23 INY ;increment Y to point to next character EA24 CMP #&22 ;check to see if it's '"' EA26 BEQ &EA2A ;if so EA2A (carry set) EA28 DEY ;decrement Y EA29 CLC ;clear carry CLIOQ EA2A ROR &E4 ;move bit 7 to bit 6 and put carry in bit 7 EA2C CMP #&0D ;check to see if it's CR to set Z EA2E RTS ;and return ************************************************************************* * GSREAD string read routine * * F2/3 points to string offset by Y * * * ************************************************************************* ; GSREAD EA2F LDA #&00 ;A=0 GSRA EA31 STA &E5 ;store A EA33 LDA (&F2),Y ;read first character EA35 CMP #&0D ;is it CR EA37 BNE &EA3F ;if not goto EA3F EA39 BIT &E4 ;if bit 7=1 no 2nd '"' found EA3B BMI &EA8F ;goto EA8F - error 'Bad string' EA3D BPL &EA5A ;if not EA5A KEYNCR EA3F CMP #&20 ;is it a control character? EA41 BCC &EA8F ;if so, go to EA8F - error 'Bad string' EA43 BNE &EA4B ;if it's not a space EA4B EA45 BIT &E4 ;is bit 7 of &E4 =1 EA47 BMI &EA89 ;if so goto EA89 EA49 BVC &EA5A ;if bit 6 = 0 EA5A KEYNSP EA4B CMP #&22 ;is it '"' EA4D BNE &EA5F ;if not EA5F EA4F BIT &E4 ;if so and Bit 7 of &E4 =0 (no previous ") EA51 BPL &EA89 ;then EA89 EA53 INY ;else point at next character EA54 LDA (&F2),Y ;get it EA56 CMP #&22 ;is it '"' EA58 BEQ &EA89 ;if so then EA89 GSR1 EA5A JSR &E03A ;skip spaces and read a byte from text EA5D SEC ;and return with carry set EA5E RTS ;and (&F2),Y=>any following text ; KEYNDQ EA5F CMP #&7C ;is it '|' EA61 BNE &EA89 ;if not EA89 EA63 INY ;if so increase Y to point to next character EA64 LDA (&F2),Y ;get it EA66 CMP #&7C ;and compare it with '|' again EA68 BEQ &EA89 ;if its '|' then EA89 EA6A CMP #&22 ;else is it '"' EA6C BEQ &EA89 ;if so then EA89 EA6E CMP #&21 ;is it ! EA70 BNE &EA77 ;if not then EA77 EA72 INY ;increment Y again EA73 LDA #&80 ;set bit 7 EA75 BNE &EA31 ;loop back to EA31 to set bit 7 in next CHR KEYNPL EA77 CMP #&20 ;is it a space EA79 BCC &EA8F ;if less than EA8F Bad String Error EA7B CMP #&3F ;is it '?' EA7D BEQ &EA87 ;if so EA87 EA7F JSR &EABF ;else modify code as if CTRL had been pressed EA82 BIT &D9B7 ;if bit 6 set EA85 BVS &EA8A ;then EA8A KEYDEL EA87 LDA #&7F ;else set bits 0 to 6 in A GSR0 EA89 CLV ;clear V GSRX EA8A INY ;increment Y EA8B ORA &E5 ;add in any bit 7 from |! prefix EA8D CLC ;clear carry EA8E RTS ;Return ; BADSTR EA8F BRK ; EA90 DB &FD ;error number EA93 DB 'Bad String' ; message EA9B BRK ; ************ Modify code as if SHIFT pressed ***************************** SHIFT EA9C CMP #&30 ;if A='0' skip routine EA9E BEQ &EABE ; EAA0 CMP #&40 ;if A='@' skip routine EAA2 BEQ &EABE ; EAA4 BCC &EAB8 ;if A<'@' then EAB8 EAA6 CMP #&7F ;else is it DELETE EAA8 BEQ &EABE ;if so skip routine EAAA BCS &EABC ;if greater than &7F then toggle bit 4 CTRLSK EAAC EOR #&30 ;reverse bits 4 and 5 EAAE CMP #&6F ;is it &6F (previously ' _' (&5F)) EAB0 BEQ &EAB6 ;goto EAB6 EAB2 CMP #&50 ;is it &50 (previously '`' (&60)) EAB4 BNE &EAB8 ;if not EAB8 SHIFT2 EAB6 EOR #&1F ;else continue to convert ` _ SHIFT1 EAB8 CMP #&21 ;compare &21 '!' EABA BCC &EABE ;if less than return SHFLIP EABC EOR #&10 ;else finish conversion by toggling bit 4 SHIFT0 EABE RTS ;exit ; ;ASCII codes &00 &20 no change ;21-3F have bit 4 reverses (31-3F) ;41-5E A-Z have bit 5 reversed a-z ;5F & 60 are reversed ;61-7E bit 5 reversed a-z becomes A-Z ;DELETE unchanged ;&80+ has bit 4 changed ************** Implement CTRL codes ************************************* CTRLCH EABF CMP #&7F ;is it DEL EAC1 BEQ &EAD1 ;if so ignore routine EAC3 BCS &EAAC ;if greater than &7F go to EAAC EAC5 CMP #&60 ;if A<>'`' EAC7 BNE &EACB ;goto EACB EAC9 LDA #&5F ;if A=&60, A=&5F CTRL1 EACB CMP #&40 ;if A<&40 EACD BCC &EAD1 ;goto EAD1 and return unchanged EACF AND #&1F ;else zero bits 5 to 7 CTRL4 CTRL0 EAD1 RTS ;return ; SLBOOT EAD2 DB '/!BOOT',&0D OS SERIES 8 GEOFF COX ************************************************************************* * * * OSBYTE &F7 (247) INTERCEPT BREAK * * * ************************************************************************* BREAK EAD9 LDA &0287 ;get BREAK vector code EADC EOR #&4C ;produces 0 if JMP not in &287 EADE BNE &EAF3 ;if not goto EAF3 EAE0 JMP &0287 ;else jump to user BREAK code ************************************************************************* * * * OSBYTE &90 (144) *TV * * * ************************************************************************* ; on entry, X=vertical display offset ; Y=interlace ; on exit, XY=previous settings ; TV EAE3 LDA &0290 ;VDU vertical adjustment EAE6 STX &0290 ;store new value EAE9 TAX ;put old value in X EAEA TYA ;put interlace flag in A EAEB AND #&01 ;maximum value =1 EAED LDY &0291 ;get old value into Y EAF0 STA &0291 ;put new value into A BREAK0 EAF3 RTS ;and Exit ************************************************************************* * * * OSBYTE &93 (147) WRITE TO FRED * * * ************************************************************************* ;X is offset within page ;Y is byte to write WFRED EAF4 TYA ; EAF5 STA &FC00,X ; EAF8 RTS ; ************************************************************************* * * * OSBYTE &95 (149) WRITE TO JIM * * * ************************************************************************* ;X is offset within page ;Y is byte to write ; WJIM EAF9 TYA ; EAFA STA &FD00,X ; EAFD RTS ; ************************************************************************* * * * OSBYTE &97 (151) WRITE TO SHEILA * * * ************************************************************************* ;X is offset within page ;Y is byte to write ; WSHEIL EAFE TYA ; EAFF STA &FE00,X ; EB02 RTS ; ****************** Silence a sound channel ******************************* ;X=channel number MINATN EB03 LDA #&04 ;mark end of release phase EB05 STA &0808,X ;to channel X EB08 LDA #&C0 ;load code for zero volume ****** if sound not disabled set sound generator volume ****************** UPATN EB0A STA &0804,X ;store A to give basic sound level of Zero EB0D LDY &0262 ;get sound output/enable flag EB10 BEQ &EB14 ;if sound enabled goto EB14 EB12 LDA #&C0 ;else load zero sound code NOISY EB14 SEC ;set carry EB15 SBC #&40 ;subtract &40 EB17 LSR ;divide by 8 EB18 LSR ;to get into bits 0 - 3 EB19 LSR ; EB1A EOR #&0F ;invert bits 0-3 EB1C ORA &EB3C,X ;get channel number into top nybble EB1F ORA #&10 ; SEND SNDOUT* EB21 PHP ; SENDX EB22 SEI ;disable interrupts EB23 LDY #&FF ;System VIA port A all outputs EB25 STY &FE43 ;set EB28 STA &FE4F ;output A on port A EB2B INY ;Y=0 EB2C STY &FE40 ;enable sound chip EB2F LDY #&02 ;set and SWAIT EB31 DEY ;execute short delay EB32 BNE &EB31 ; EB34 LDY #&08 ;then disable sound chip again EB36 STY &FE40 ; EB39 LDY #&04 ;set delay SWAT2 EB3B DEY ;and loop delay CHAN EB3C BNE &EB3B ; EB3E PLP ;get back flags EB3F RTS ;and exit *******: Sound parameters look up table ********************************** RCHAN EB40 DB &E0 ;channel 0 register address EB41 DB &C0 ;channel 1 register address EB42 DB &A0 ;channel 2 register address EB43 DB &80 ;channel 3 register address HENDS EB44 JMP &EC59 ;just to allow relative branches in early part ;of sound interrupt routine ************************************************************************* * * * PROCESS SOUND INTERRUPT * * * ************************************************************************* SNDIRQ EB47 LDA #&00 ; EB49 STA &083B ;zero number of channels on hold for sync EB4C LDA &0838 ;get number of channels required for sync EB4F BNE &EB57 ;if this <>0 then EB57 EB51 INC &083B ;else number of chanels on hold for sync =1 EB54 DEC &0838 ;number of channels required for sync =255 MAN2 EB57 LDX #&08 ;set loop counter MNLOP EB59 DEX ;loop EB5A LDA &0800,X ;get value of &800 +offset (sound queue occupancy) EB5D BEQ &EB44 ;if 0 goto EC59 no sound this channel EB5F LDA &02CF,X ;else get buffer busy flag EB62 BMI &EB69 ;if negative (buffer empty) goto EB69 EB64 LDA &0818,X ;else if duration count not zer0 EB67 BNE &EB6C ;goto EB6C MAN8 EB69 JSR &EC6B ;check and pick up new sound if required MAN10 EB6C LDA &0818,X ;if duration count 0 EB6F BEQ &EB84 ;goto EB84 EB71 CMP #&FF ;else if it is &FF (infinite duration) EB73 BEQ &EB87 ;go onto EB87 EB75 DEC &081C,X ;decrement 10 mS count EB78 BNE &EB87 ;and if 0 EB7A LDA #&05 ;reset to 5 EB7C STA &081C,X ;to give 50 mSec delay EB7F DEC &0818,X ;and decrement main counter EB82 BNE &EB87 ;if not zero then EB87 MAN13 EB84 JSR &EC6B ;else check and get nw sound MAN12 EB87 LDA &0824,X ;if step progress counter is 0 no envelope involved EB8A BEQ &EB91 ;so jump to EB91 EB8C DEC &0824,X ;else decrement it EB8F BNE &EB44 ;and if not zero go on to EC59 MAN11 EB91 LDY &0820,X ;get envelope data offset from (8C0) EB94 CPY #&FF ;if 255 no envelope set so EB96 BEQ &EB44 ;goto EC59 EB98 LDA &08C0,Y ;else get get step length EB9B AND #&7F ;zero repeat bit EB9D STA &0824,X ;and store it EBA0 LDA &0808,X ;get phase counter EBA3 CMP #&04 ;if release phase completed EBA5 BEQ &EC07 ;goto EC07 EBA7 LDA &0808,X ;else start new step by getting phase EBAA CLC ; EBAB ADC &0820,X ;add it to interval multiplier EBAE TAY ;transfer to Y EBAF LDA &08CB,Y ;and get target value base for envelope EBB2 SEC ; EBB3 SBC #&3F ; EBB5 STA &083A ;store modified number as current target amplitude EBB8 LDA &08C7,Y ;get byte from envelope store EBBB STA &0839 ;store as current amplitude step EBBE LDA &0804,X ;get base volumelevel EBC1 PHA ;save it EBC2 CLC ;clear carry EBC3 ADC &0839 ;add to current amplitude step EBC6 BVC &EBCF ;if no overflow EBC8 ROL ;double it Carry = bit 7 EBC9 LDA #&3F ;if bit =1 A=&3F EBCB BCS &EBCF ;into &EBCF EBCD EOR #&FF ;else toggle bits (A=&C0) ;at this point the BASIC volume commands are converted ; &C0 (0) to &38 (-15) 3 times, In fact last 3 bits ;are ignored so &3F represents -15 OK8 EBCF STA &0804,X ;store in current volume EBD2 ROL ;multiply by 2 EBD3 EOR &0804,X ;if bits 6 and 7 are equal EBD6 BPL &EBE1 ;goto &EBE1 EBD8 LDA #&3F ;if carry clear A=&3F (maximum) EBDA BCC &EBDE ;or EBDC EOR #&FF ;&C0 minimum LMATN EBDE STA &0804,X ;and this is stored in current volume ATOK EBE1 DEC &0839 ;decrement amplitude change per step EBE4 LDA &0804,X ;get volume again EBE7 SEC ;set carry EBE8 SBC &083A ;subtract target value EBEB EOR &0839 ;negative value undicates correct trend EBEE BMI &EBF9 ;so jump to next part EBF0 LDA &083A ;else enter new phase EBF3 STA &0804,X ; EBF6 INC &0808,X ; SMPER EBF9 PLA ;get the old volume level EBFA EOR &0804,X ;and compare with the old EBFD AND #&F8 ; EBFF BEQ &EC07 ;if they are the same goto EC07 EC01 LDA &0804,X ;else set new level EC04 JSR &EB0A ;via EB0A NNEWA ENDEV EC07 LDA &0810,X ;get absolute pitch value EC0A CMP #&03 ;if it =3 EC0C BEQ &EC59 ;skip rest of loop as all sections are finished EC0E LDA &0814,X ;else if 814,X is not 0 current section is not ;complete EC11 BNE &EC3D ;so EC3D EC13 INC &0810,X ;else implement a section change EC16 LDA &0810,X ;check if its complete EC19 CMP #&03 ;if not EC1B BNE &EC2D ;goto EC2D EC1D LDY &0820,X ;else set A from EC20 LDA &08C0,Y ;&820 and &8C0 (first envelope byte) EC23 BMI &EC59 ;if negative there is no repeat EC25 LDA #&00 ;else restart section sequence EC27 STA &0830,X ; EC2A STA &0810,X ; NTLF EC2D LDA &0810,X ;get number of steps in new section EC30 CLC ; EC31 ADC &0820,X ; EC34 TAY ; EC35 LDA &08C4,Y ; EC38 STA &0814,X ;set in 814+X EC3B BEQ &EC59 ;and if 0 then EC59 GTPRD EC3D DEC &0814,X ;decrement EC40 LDA &0820,X ;and pick up rate of pitch change EC43 CLC ; EC44 ADC &0810,X ; EC47 TAY ; EC48 LDA &08C1,Y ; EC4B CLC ; EC4C ADC &0830,X ;add to rate of differential pitch change EC4F STA &0830,X ;and save it EC52 CLC ; EC53 ADC &080C,X ;ad to base pitch EC56 JSR &ED01 ;and set new pitch ENDSW EC59 CPX #&04 ;if X=4 (last channel) EC5B BEQ &EC6A ;goto EC6A (RTS) EC5D JMP &EB59 ;else do loop again SNDRST EC60 LDX #&08 ;X=7 again RLOP EC62 DEX ;loop EC63 JSR &ECA2 ;clear channel EC66 CPX #&04 ;if not 4 EC68 BNE &EC62 ;do it again EDUP EC6A RTS ;and return ; NXTNT EC6B LDA &0808,X ;check for last channel EC6E CMP #&04 ;is it 4 (release complete) EC70 BEQ &EC77 ;if so EC77 EC72 LDA #&03 ;else mark release in progress EC74 STA &0808,X ;and store it ELDDN EC77 LDA &02CF,X ;is buffer not empty EC7A BEQ &EC90 ;if so EC90 EC7C LDA #&00 ;else mark buffer not empty EC7E STA &02CF,X ;an store it EC81 LDY #&04 ;loop counter SWTLOP EC83 STA &082B,Y ;zero sync bytes EC86 DEY ; EC87 BNE &EC83 ;until Y=0 EC89 STA &0818,X ;zero duration count EC8C DEY ;and set sync count to EC8D STY &0838 ;&FF NOFL EC90 LDA &0828,X ;get synchronising flag EC93 BEQ &ECDB ;if its 0 then ECDB EC95 LDA &083B ;else get number of channels on hold EC98 BEQ &ECD0 ;if 0 then ECD0 EC9A LDA #&00 ;else EC9C STA &0828,X ;zero note length interval NNWAT EC9F JMP &ED98 ;and goto ED98 SCRST ECA2 JSR &EB03 ;silence the channel ECA5 TYA ;Y=0 A=Y ECA6 STA &0818,X ;zero main count ECA9 STA &02CF,X ;mark buffer not empty ECAC STA &0800,X ;mark channel dormant ECAF LDY #&03 ;loop counter CWP ECB1 STA &082C,Y ;zero sync flags ECB4 DEY ; ECB5 BPL &ECB1 ; ECB7 STY &0838 ;number of channels to &FF ECBA BMI &ED06 ;jump to ED06 ALWAYS NGONG ECBC PHP ;save flags ECBD SEI ;and disable interrupts ECBE LDA &0808,X ;check for end of release ECC1 CMP #&04 ; ECC3 BNE &ECCF ;and if not found ECCF ECC5 JSR &E45B ;elseexamine buffer ECC8 BCC &ECCF ;if not empty ECCF ECCA LDA #&00 ;else mark channel dormant ECCC STA &0800,X ; NOGO2 ECCF PLP ;get back flags NOTGO ECD0 LDY &0820,X ;if no envelope 820=&FF ECD3 CPY #&FF ; ECD5 BNE &ECDA ;then terminate sound ECD7 JSR &EB03 ;via EB03 PRTS PROG ECDA RTS ;else return ************ Synchronise sound routines ********************************** NOTWT ECDB JSR &E45B ;examine buffer if empty carry set ECDE BCS &ECBC ; ECE0 AND #&03 ;else examine next word if>3 or 0 ECE2 BEQ &EC9F ;goto ED98 (via EC9F) ECE4 LDA &0838 ;else get synchronising count ECE7 BEQ &ECFE ;in 0 (complete) goto ECFE ECE9 INC &0828,X ;else set sync flag ECEC BIT &0838 ;if 0838 is +ve S has already been set so ECEF BPL &ECFB ;jump to ECFB ECF1 JSR &E45B ;else get first byte ECF4 AND #&03 ;mask bits 0,1 ECF6 STA &0838 ;and store result ECF9 BPL &ECFE ;Jump to ECFE (ALWAYS!!) SMCRD ECFB DEC &0838 ;decrement 0838 SRTS LNOTGO ECFE JMP &ECD0 ;and silence the channel if envelope not in use ************ Pitch setting *********************************************** CNVRT ED01 CMP &082C,X ;If A=&82C,X then pitch is unchanged ED04 BEQ &ECDA ;then exit via ECDA FCNVRT ED06 STA &082C,X ;store new pitch ED09 CPX #&04 ;if X<>4 then not noise so ED0B BNE &ED16 ;jump to ED16 *********** Noise setting ************************************************ ED0D AND #&0F ;convert to chip format ED0F ORA &EB3C,X ; ED12 PHP ;save flags ED13 JMP &ED95 ;and pass to chip control routine at EB22 via ED95 TONE ED16 PHA ; ED17 AND #&03 ; ED19 STA &083C ;lose eigth tone surplus ED1C LDA #&00 ; ED1E STA &083D ; ED21 PLA ;get back A ED22 LSR ;divide by 12 ED23 LSR ; OCTL ED24 CMP #&0C ; ED26 BCC &ED2F ; ED28 INC &083D ;store result ED2B SBC #&0C ;with remainder in A ED2D BNE &ED24 ; ;at this point 83D defines the Octave ;A the semitone within the octave ENDOL ED2F TAY ;Y=A ED30 LDA &083D ;get octave number into A ED33 PHA ;push it ED34 LDA &EDFB,Y ;get byte from look up table ED37 STA &083D ;store it ED3A LDA &EE07,Y ;get byte from second table ED3D PHA ;push it ED3E AND #&03 ;keep two LS bits only ED40 STA &083E ;save them ED43 PLA ;pull second table byte ED44 LSR ;push hi nybble into lo nybble ED45 LSR ; ED46 LSR ; ED47 LSR ; ED48 STA &083F ;store it ED4B LDA &083D ;get back octave number ED4E LDY &083C ;adjust for surplus eighth tones ED51 BEQ &ED5F ; INTRPL ED53 SEC ; ED54 SBC &083F ; ED57 BCS &ED5C ; ED59 DEC &083E ; INTRPO ED5C DEY ; ED5D BNE &ED53 ; INTRPE ED5F STA &083D ; ED62 PLA ; ED63 TAY ; ED64 BEQ &ED6F ; SSHIFT ED66 LSR &083E ; ED69 ROR &083D ; ED6C DEY ; ED6D BNE &ED66 ; ODONE ED6F LDA &083D ; ED72 CLC ; ED73 ADC &C43D,X ; ED76 STA &083D ; ED79 BCC &ED7E ; ED7B INC &083E ; NOFOV ED7E AND #&0F ; ED80 ORA &EB3C,X ; ED83 PHP ;push P ED84 SEI ;bar interrupts ED85 JSR &EB21 ;set up chip access 1 ED88 LDA &083D ; ED8B LSR &083E ; ED8E ROR ; ED8F LSR &083E ; ED92 ROR ; ED93 LSR ; ED94 LSR ; CDONE ED95 JMP &EB22 ;set up chip access 2 and return **************** Pick up and interpret sound buffer data ***************** STRNT ED98 PHP ;push flags ED99 SEI ;disable interrupts ED9A JSR &E460 ;read a byte from buffer ED9D PHA ;push A ED9E AND #&04 ;isolate H bit EDA0 BEQ &EDB7 ;if 0 then EDB7 EDA2 PLA ;get back A EDA3 LDY &0820,X ;get envelope setting for this channel EDA6 CPY #&FF ;check for &FF=no envelope EDA8 BNE &EDAD ;envelope is in use, keep playing it EDAA JSR &EB03 ;no envelope, so call EB03 to silence channel SCON1 EDAD JSR &E460 ;clear buffer of redundant data, get ignored pitch EDB0 JSR &E460 ;get ignored duration EDB3 PLP ;get flags back EDB4 JMP &EDF7 ;set main duration count using last byte from buffer NOCONT EDB7 PLA ;get back A EDB8 AND #&F8 ;zero bits 0-2, Hold and Sync bits EDBA ASL ;put bit 7 volume/envelope flag into carry EDBB BCC &EDC8 ;if zero (envelope) jump to EDC8 EDBD EOR #&FF ;invert A EDBF LSR ;shift right EDC0 SEC ; EDC1 SBC #&40 ;subtract &40 EDC3 JSR &EB0A ;and set volume EDC6 LDA #&FF ;A=&FF PENV EDC8 STA &0820,X ;get envelope no.-1 *16 into A EDCB LDA #&05 ;set duration sub-counter EDCD STA &081C,X ; EDD0 LDA #&01 ;set phase counter EDD2 STA &0824,X ; EDD5 LDA #&00 ;set step counter EDD7 STA &0814,X ; EDDA STA &0808,X ;and envelope phase EDDD STA &0830,X ;and pitch differential EDE0 LDA #&FF ; EDE2 STA &0810,X ;set step count EDE5 JSR &E460 ;get pitch from buffer EDE8 STA &080C,X ;set it EDEB JSR &E460 ;get duration from buffer EDEE PLP ; EDEF PHA ;save duration EDF0 LDA &080C,X ;get back pitch value EDF3 JSR &ED01 ;and set it EDF6 PLA ;get back duration DODUR EDF7 STA &0818,X ;set it EDFA RTS ;and return ********************* Pitch lookup table 1***************************** NTABL EDFB DB &F0 EDFC DB &B7 EDFD DB &82 EDFE DB &4F EDFF DB &20 EE00 DB &F3 EE01 DB &C8 EE02 DB &A0 EE03 DB &7B EE04 DB &57 EE05 DB &35 EE06 DB &16 ********************* Pitch lookup table 2 ***************************** NTABH EE07 DB &E7 EE08 DB &D7 EE09 DB &CB EE0A DB &C3 EE0B DB &B7 EE0C DB &AA EE0D DB &A2 EE0E DB &9A EE0F DB &92 EE10 DB &8A EE11 DB &82 EE12 DB &7A *********: set current filing system ROM/PHROM ************************** SRINIT EE13 LDA #&EF ;start with ROM &F0-1 EE15 STA &F5 ;store it EE17 RTS ;return ********** Initialise ROMFS or PHROM ROM *************************************** SRNEXT SPCPYF EE18 LDX #&0D ;X=13 EE1A INC &F5 ;increment current ROM number EE1C LDY &F5 ;get ROM EE1E BPL &EE59 ;if +ve it's a sideways ROM else it's a PHROM EE20 LDX #&00 ;PHROM EE22 STX &F7 ;set address pointer in PHROM EE24 INX ; EE25 STX &F6 ;to 0001 EE27 JSR &EEBB ;pass info to speech processor EE2A LDX #&03 ;X=3 SPCPY EE2C JSR &EE62 ;check for speech processor and output until ;it reports, read byte from ROM EE2F CMP &DF0C,X ;if A <> DF0C+X then EE18 (DF0C = (C)) EE32 BNE &EE18 ; EE34 DEX ;else decrement X EE35 BPL &EE2C ;and do it again EE37 LDA #&3E ; EE39 STA &F6 ;set lo byte address SPLIND EE3B JSR &EEBB ;select address and PROM from &F5/6/7 EE3E LDX #&FF ;read to bytes, to &F6 and &F7 RABLP EE40 JSR &EE62 ;read byte from this address from PHROM EE43 LDY #&08 ;8 bits to rotate RABLO EE45 ASL ;reverse the byte into &F6 on first pass EE46 ROR &F7,X ;and then into &F7 on the second pass EE48 DEY ; EE49 BNE &EE45 ;loop for 8 bits EE4B INX ; EE4C BEQ &EE40 ;loop for two bytes EE4E CLC ; EE4F BCC &EEBB ;jump to set PHROM address to this address we've just read ************ ROM SERVICE ************************************************ RDSPCH EE51 LDX #&0E ; EE53 LDY &F5 ;if Y is negative (PHROM) EE55 BMI &EE62 ;GOTO EE62 EE57 LDY #&FF ;else Y=255 ; BUG! If ROMFS 15 (ROM 0) has been scanned, &F5 incremented to &10 all the ; way to &EF and back through PHROMs and back to ROMFS 0 (ROM 15) and loops ; through ROMFS ROMs again. ; Should do CPY #16, BCS return SPSV EE59 PHP ;push flags EE5A JSR &F168 ;offer paged rom service EE5D PLP ;pull processor flags EE5E CMP #&01 ;if A>0 set carry EE60 TYA ;A=Y EE61 RTS ;return ********* PHROM SERVICE ************************************************* ; RDBYTE EE62 PHP ;push processor flags EE63 SEI ;disable interrupts EE64 LDY #&10 ;Y=&10 - Read Byte EE66 JSR &EE7F ;Send command to Speech Processor EE69 LDY #&00 ;Y=0 EE6B BEQ &EE84 ;Jump to EE84 to read resultant byte from Speech ************************************************************************* * * * OSBYTE 158 read from speech processor * * * ************************************************************************* RSPECH EE6D LDY #&00 ;Y=0 to set speech proc to read EE6F BEQ &EE82 ;jump to EE82 always ;write A to speech processor as two nybbles SPL EE71 PHA ;push A EE72 JSR &EE7A ;to write to speech processor EE75 PLA ;get back A EE76 ROR ;bring upper nybble to lower nybble EE77 ROR ;by rotate right EE78 ROR ;4 times EE79 ROR ; SPL1 EE7A AND #&0F ;Y=lo nybble A +&40 EE7C ORA #&40 ; EE7E TAY ;forming command for speech processor ************************************************************************* * * * OSBYTE 159 Write to speech processor * * * ************************************************************************* ; on entry data or command in Y WSPECH EE7F TYA ;transfer command to A EE80 LDY #&01 ;to set speech proc to write ;if Y=0 read speech processor ;if Y=1 write speech processor SENDSP EE82 PHP ;push flags EE83 SEI ;disable interrupts SENDSX EE84 BIT &027B ;test for presence of speech processor EE87 BPL &EEAA ;if not there goto EEAA EE89 PHA ;else push A EE8A LDA &F075,Y ; EE8D STA &FE43 ;set DDRA of system VIA to give input (Y=0) or output (Y=1) EE90 PLA ;get command back in A EE91 STA &FE4F ;and send to speech chip, write to PortA EE94 LDA &F077,Y ; Get A=&01 for read, A=&02 for write EE97 STA &FE40 ; Write to PortB, A=&01 set VP_READ, A=&02 set VP_WRITE SPWAIT EE9A BIT &FE40 ; Test System VIA PortA, bit 7 is VP_RDY EE9D BMI &EE9A ; Loop until Speech Processor reports ready EE9F LDA &FE4F ; Read Speech Processor data from Port A EEA2 PHA ; Save it - will be returned data for read, data written for write EEA3 LDA &F079,Y ; Get A=&09 for read, A=&0A for write EEA6 STA &FE40 ; Write to PortB, A=&09 clear VP_READ, A=&0A clear VP_WRITE EEA9 PLA ; Get data byte back SNDSX0 EEAA PLP ; Restore IRQs EEAB TAY ; Transfer A to Y to set flags EEAC RTS ; And return ; SRSKIP EEAD LDA &03CB ;set rom displacement pointer EEB0 STA &F6 ;in &F6 EEB2 LDA &03CC ; EEB5 STA &F7 ;And &F7 EEB7 LDA &F5 ;if F5 is +ve ROM is selected so EEB9 BPL &EED9 ;goto EED9 SPLOAD EEBB PHP ;else push processor EEBC SEI ;disable interrupts EEBD LDA &F6 ;get lo displacement EEBF JSR &EE71 ;pass two nybbles to speech proc. EEC2 LDA &F5 ;&FA=&F5 = ROM number or SOUND command &80-&FF EEC4 STA &FA ; EEC6 LDA &F7 ;get hi displacement value EEC8 ROL ;replace two most significant bits of A EEC9 ROL ;by 2 LSBs of &FA EECA LSR &FA ; EECC ROR ; EECD LSR &FA ; EECF ROR ;A= now %rraaaaaa EED0 JSR &EE71 ;pass two nybbles to speech processor EED3 LDA &FA ;FA has now been divided by 4 so pass EED5 JSR &EE7A ;lower nybble to speech proc., passing %xxrr EED8 PLP ;get back flags SRSK0 EED9 RTS ;and Return \************************************************************************* \* * \* MAIN KEYBOARD HANDLER * \* ===================== * \* * \************************************************************************* \ Keyboard input housekeeping \ Jumped to from &F00C, is here to drop through to LED setting \ ------------------------------------------------------------ KBDAW EEDA LDX #&FF ; EEDC LDA &EC ;get value of most recently pressed key EEDE ORA &ED ;OR it with previous key to check for presses EEE0 BNE &EEE8 ;if A=0 no keys pressed so off you go EEE2 LDA #&81 ;else enable keybd interupt only by writing bit 7 EEE4 STA &FE4E ;and bit 0 of system VIA interupt register EEE7 INX ;set X=0 KBDISX EEE8 STX &0242 ;reset keyboard semaphore **********: Turn on Keyboard indicators ******************************* KBDIS EEEB PHP ;save flags EEEC LDA &025A ;read keyboard status; ;Bit 7 =1 shift enabled ;Bit 6 =1 control pressed ;bit 5 =0 shift lock ;Bit 4 =0 Caps lock ;Bit 3 =1 shift pressed EEEF LSR ;shift Caps bit into bit 3 EEF0 AND #&18 ;mask out all but 4 and 3 EEF2 ORA #&06 ;returns &06 if caps lock OFF, &0E if ON EEF4 STA &FE40 ;turn on or off caps light if required EEF7 LSR ;bring shift bit into bit 3 EEF8 ORA #&07 ; EEFA STA &FE40 ;turn on or off shift lock light EEFD JSR &F12E ;set keyboard counter EF00 PLA ;pop flags into A register, setting M from saved M EF01 RTS ;return, A.b7=Control, A.b6=Shift, MI=Control, VS=Shift ************************************************************************* * * * MAIN KEYBOARD HANDLER - ENTRY FROM KEYV * * ======================================= * * * * ENTRY CONDITIONS * * ================ * * VS=Keyboard interupt * * CC=Keypress interupt * * CS=Centisecond keypress poll * * * * VC=Scan keyboard for keypresses * * CC=Test SHIFT and CTRL keys * * CS=Scan full keyboard as with OSBYTE &79/negative INKEY * * * ************************************************************************* DEFKEY EF02 BVC &EF0E ; If V is clear jump to keyscan routines EF04 LDA #&01 ; Disable keyboard interrupts EF06 STA &FE4E ; by writing to VIA interrupt settings EF09 BCS &EF13 ; Jump with timer interupt EF0B JMP &F00F ; Otherwise, jump with keypress interupt KEYV0 EF0E BCC &EF16 ; Jump to test SHIFT and CTRL EF10 JMP &F0D1 ; Otherwise, jump to scan whole keyboard \************************************************************************* \* Keyboard Timer interrupt entry * \* Enters with VS, CS * \************************************************************************* KYBD EF13 INC &0242 ; Increment keyboard semaphore (to 0) ; ; Continue through testing state of SHIFT and CTRL keys ************************************************************************* * Keyboard Test Shift and Control Keys entry * * Enters with VC, CC * * Exits with MI=Control, VS=Shift, A.b7=Control, A.b6=Shift * ************************************************************************* TSHIFT EF16 LDA &025A ; Get current keyboard status ;Bit 7 =1 Shift enabled ;Bit 6 =1 Control pressed ;bit 5 =0 Shift Lock ;Bit 4 =0 Caps Lock ;Bit 3 =1 Shift pressed EF19 AND #&B7 ; Clear bits 3 and 6, Shift & Ctrl off EF1B LDX #&00 ; X=&00 to test for Shift key EF1D JSR &F02A ; Interrogate keyboard, does INKEY-(0+1), ie INKEY-1 ; On return X=&80 if key is pressed EF20 STX &FA ; Save SHIFT keypress state EF22 CLV ; Clear V = SHIFT not pressed EF23 BPL &EF2A ; If SHIFT not pressed, if no key press (X=0) then EF2A else EF25 BIT &D9B7 ; Set V = SHIFT is pressed EF28 ORA #&08 ; Set bit 3 to indicate Shift was pressed KYBD1 EF2A INX ; X=&01 or &81 to test for Control key EF2B JSR &F02A ; Interrogate keyboard, does INKEY-(1+1), ie INKEY-2 ; ; On return X=&81 if key is pressed, M set EF2E BCC &EEEB ; If Carry Clear, was entered to test Shift & Control, ; jump to update keyboard lights and return EF30 BPL &EF34 ; If CTRL not pressed, skip past EF32 ORA #&40 ; Set bit 6 to indicate Control pressed KYBD2 EF34 STA &025A ; Update keyboard status byte EF37 LDX &EC ; Get previous keypress EF39 BEQ &EF4D ;if no key previously pressed then EF4D EF3B JSR &F02A ;else check to see if key still pressed EF3E BMI &EF50 ;if so enter repeat routine at EF50 EF40 CPX &EC ;else compare X with last key pressed (set flags) KBDW EF42 STX &EC ;store X in last key pressed EF44 BNE &EF4D ;if different from previous (Z clear) then EF4D EF46 LDX #&00 ;else zero EF48 STX &EC ;last key pressed FCSP EF4A JSR &F01F ;and reset repeat system KBDWU EF4D JMP &EFE9 ; ********** REPEAT ACTION ************************************************* KBDX EF50 CPX &EC ;if X<>than last key pressed EF52 BNE &EF42 ;then back to EF42 EF54 LDA &E7 ;else get value of AUTO REPEAT COUNTDOWN TIMER EF56 BEQ &EF7B ;if 0 goto EF7B EF58 DEC &E7 ;else decrement EF5A BNE &EF7B ;and if not 0 goto EF7B ;this means that either the repeat system is dormant ;or it is not at the end of its count EF5C LDA &02CA ;next value for countdown timer EF5F STA &E7 ;store it EF61 LDA &0255 ;get auto repeat rate from 0255 EF64 STA &02CA ;store it as next value for Countdown timer EF67 LDA &025A ;get keyboard status EF6A LDX &EC ;get last key pressed EF6C CPX #&D0 ;if not SHIFT LOCK key (&D0) goto EF6E BNE &EF7E ;EF7E EF70 ORA #&90 ;sets shift enabled, & no caps lock all else preserved EF72 EOR #&A0 ;reverses shift lock disables Caps lock and Shift enab KEYBD3 EF74 STA &025A ;reset keyboard status EF77 LDA #&00 ;and set timer EF79 STA &E7 ;to 0 KBDXS EF7B JMP &EFE9 ; KCPLK EF7E CPX #&C0 ;if not CAPS LOCK EF80 BNE &EF91 ;goto EF91 EF82 ORA #&A0 ;sets shift enabled and disables SHIFT LOCK EF84 BIT &FA ;if bit 7 not set by (EF20) shift NOT pressed EF86 BPL &EF8C ;goto EF8C EF88 ORA #&10 ;else set CAPS LOCK not enabled EF8A EOR #&80 ;reverse SHIFT enabled KCPLK2 EF8C EOR #&90 ;reverse both SHIFT enabled and CAPs Lock EF8E JMP &EF74 ;reset keyboard status and set timer \*********** Convert keypress to ASCII and insert in keyboard buffer **** ;on entry X=key pressed internal number KCPLK1 EF91 LDA &EFAB,X ;get code from lookup table EF94 BNE &EF99 ;if not zero goto EF99 else TAB pressed EF96 LDA &026B ;get TAB character NTAB EF99 LDX &025A ;get keyboard status EF9C STX &FA ;store it in &FA EF9E ROL &FA ;rotate to get CTRL pressed into bit 7 EFA0 BPL &EFA9 ;if CTRL NOT pressed EFA9 EFA2 LDX &ED ;get no. of previously pressed key FCSP1 EFA4 BNE &EF4A ;if not 0 goto EF4A to reset repeat system etc. EFA6 JSR &EABF ;else perform code changes for CTRL KCONV6 EFA9 ROL &FA ;move shift lock into bit 7 EFAB BMI &EFB5 ;if not effective goto EFB5 else EFAD JSR &EA9C ;make code changes for SHIFT EFB0 ROL &FA ;move CAPS LOCK into bit 7 EFB2 JMP &EFC1 ;and Jump to EFC1 KCONV3 EFB5 ROL &FA ;move CAPS LOCK into bit 7 EFB7 BMI &EFC6 ;if not effective goto EFC6 EFB9 JSR &E4E3 ;else make changes for CAPS LOCK on, return with ;C clear for Alphabetic codes EFBC BCS &EFC6 ;if carry set goto EFC6 else make changes for EFBE JSR &EA9C ;SHIFT as above KCONV4 EFC1 LDX &025A ;if shift enabled bit is clear EFC4 BPL &EFD1 ;goto EFD1 KCONV2 EFC6 ROL &FA ;else get shift bit into 7 EFC8 BPL &EFD1 ;if not set goto EFD1 EFCA LDX &ED ;get previous key press EFCC BNE &EFA4 ;if not 0 reset repeat system etc. via EFA4 EFCE JSR &EA9C ;else make code changes for SHIFT KCONV1 EFD1 CMP &026C ;if A<>ESCAPE code EFD4 BNE &EFDD ;goto EFDD EFD6 LDX &0275 ;get Escape key status EFD9 BNE &EFDD ;if ESCAPE returns ASCII code goto EFDD EFDB STX &E7 ;store in Auto repeat countdown timer KESC1 EFDD TAY ; EFDE JSR &F129 ;disable keyboard EFE1 LDA &0259 ;read Keyboard disable flag used by Econet EFE4 BNE &EFE9 ;if keyboard locked goto EFE9 EFE6 JSR &E4F1 ;put character in input buffer KBDY EFE9 LDX &ED ;get previous keypress EFEB BEQ &EFF8 ;if none, go to EFF8 EFED JSR &F02A ;examine to see if key still pressed EFF0 STX &ED ;store result EFF2 BMI &EFF8 ;if pressed goto EFF8 EFF4 LDX #&00 ;else zero X EFF6 STX &ED ;and &ED KBDA EFF8 LDX &ED ;get &ED EFFA BNE &F012 ;if not 0 goto F012 EFFC LDY #&EC ;get first keypress into Y EFFE JSR &F0CC ;scan keyboard from &10 (osbyte 122) F001 BMI &F00C ;if exit is negative goto F00C F003 LDA &EC ;else make last key the F005 STA &ED ;first key pressed i.e. rollover KBDAR F007 STX &EC ;save X into &EC F009 JSR &F01F ;set keyboard repeat delay KBDAS F00C JMP &EEDA ;go back to EEDA ************************************************************************* * Keyboard Key pressed interrupt * * Enters with VS, CC * ************************************************************************* ;enters with X=key KBDAT F00F JSR &F02A ; Stop autoscan by testing for a random keypress KBDAU F012 LDA &EC ; Get previous key press F014 BNE &F00C ; If key currently pressed, back to housekeeping routine F016 LDY #&ED ; Point Y to last keypress location F018 JSR &F0CC ; Scan keyboard for a pressed key F01B BMI &F00C ; If negative, no key found, exit F01D BPL &F007 ; Positive, key press found, store it and prepare autorepeat **************** Set Autorepeat countdown timer ************************** AR F01F LDX #&01 ;set timer to 1 F021 STX &E7 ; F023 LDX &0254 ;get next timer value F026 STX &02CA ;and store it F029 RTS ; *************** Interrogate Keyboard routine *********************** ; KC F02A LDY #&03 ;stop Auto scan F02C STY &FE40 ;by writing to system VIA F02F LDY #&7F ;set bits 0 to 6 of port A to input on bit 7 ;output on bits 0 to 6 F031 STY &FE43 ; F034 STX &FE4F ;write X to Port A system VIA F037 LDX &FE4F ;read back &80 if key pressed (M set) F03A RTS ;and return ************************************************************************* * * * KEY TRANSLATION TABLES * * * * 7 BLOCKS interspersed with unrelated code * ************************************************************************* key data block 1 KTBL0 F03B DB 71,33,34,35,84,38,87,2D,5E,8C ; q ,3 ,4 ,5 ,f4,8 ,f7,- ,^ ,rt ************************************************************************* * * * OSBYTE 120 Write KEY pressed Data * * * ************************************************************************* OLDKEY F045 STY &EC ;store Y as latest key pressed F047 STX &ED ;store X as previous key pressed F049 RTS ;and exit key data block 2 F04B DB 80,77,65,74,37,69,39,30,5F,8E ; f0,w ,e ,t ,7 ,i ,9 ,0 ,_ ,lft JMITST F055 JMP (&FDFE) ;Jim paged entry vector JMI F058 JMP (&FA) ; key data block 3 F05B DB 31,32,64,72,36,75,6F,70,5B,8F ; 1 ,2 ,d ,r ,6 ,u ,o ,p ,[ ,dn ************************************************************************* * * * Jump to keyboard handler interupt routine * * * ************************************************************************* JMIKYV F065 BIT &D9B7 ; Set V to signal interupt call JMIKEY F068 JMP (&0228) ; Jump via KEYV to keyboard handler key data block 4 F06B DB 01,61,78,66,79,6A,6B,40,3A,0D ; CL,a ,x ,f ,y ,j ,k ,@ ,: ,RETN CL=CAPS LOCK speech routine data SENDD F075 DB 00,FF,01,02,09,0A key data block 5 F07B DB 02,73,63,67,68,6E,6C,3B,5D,7F ; SL,s ,c ,g ,h ,n ,l ,; ,] ,DEL SL=SHIFT LOCK ************************************************************************* * * * OSBYTE 131 READ OSHWM (PAGE in BASIC) * * * ************************************************************************* RMLA F085 LDY &0244 ;read current OSHWM F088 LDX #&00 ;address is &xx00 F08A RTS ; key data block 6 F08B DB 00 ,7A,20 ,76,62,6D,2C,2E,2F,8B ; TAB,Z ,SPACE,V ,b ,m ,, ,. ,/ ,copy ***** set input buffer number and flush it ***************************** JFLUSH F095 LDX &0241 ;get current input buffer FLUSH2 F098 JMP &E1AD ;flush it key data block 7 F09B DB 1B,81,82,83,85,86,88,89,5C,8D ; ESC,f1,f2,f3,f5,f6,f8,f9,\ ,rgt ENDKT * JMIEVT F0A5 JMP (&0220) ;go to eventV handling routine ************************************************************************* * * * OSBYTE 15 FLUSH SELECTED BUFFER CLASS * * * ************************************************************************* ;flush selected buffer ;X=0 flush all buffers ;X>1 flush input buffer FLUSHC F0A8 BNE &F095 ;if X<>1 flush input buffer only FLSHAL F0AA LDX #&08 ;else load highest buffer number (8) FLUSH3 F0AC CLI ;allow interrupts F0AD SEI ;briefly! F0AE JSR &F0B4 ;flush buffer F0B1 DEX ;decrement X to point at next buffer F0B2 BPL &F0AC ;if X>=0 flush next buffer ;at this point X=255 ************************************************************************* * * * OSBYTE 21 FLUSH SPECIFIC BUFFER * * * ************************************************************************* ;on entry X=buffer number FLUSHS F0B4 CPX #&09 ;is X<9? F0B6 BCC &F098 ;if so flush buffer or else F0B8 RTS ;exit ************************************************************************* * * * Issue *HELP to ROMS * * * ************************************************************************* HELP F0B9 LDX #&09 ; F0BB JSR &F168 ;send service call 9 to ROMs F0BE JSR &FA4A ;print inline message, corrupting &B8/9, assumes Y<>0 F0C1 DB &0D ;carriage return F0C2 DS 'OS 1.20' ;help message F0C9 DB &0D ;carriage return F0CA BRK ; F0CB RTS ; ************************************************************************* * * * OSBYTE 122 KEYBOARD SCAN FROM &10 (16) * * * ************************************************************************* ISCAN F0CC CLC ; Clear carry to bypass KEYV and use supplied Y as we ; ; are calling direct from keyboard handler itself XSCANA F0CD LDX #&10 ; Set X to scan from &10, skipping modifier keys ************************************************************************* * * * OSBYTE 121 KEYBOARD SCAN FROM VALUE IN X * * * ************************************************************************* XSCANB F0CF BCS &F068 ; If carry set from OSBYTE entry, vector via KEYV ; And thence back to: ************************************************************************* * Scan Keyboard C=1, V=0 entry via KEYV * * X=&00-&7F scan range, X=&80-&FF scan for single keypress * ************************************************************************* SKFMA F0D1 TXA ;if X is +ve goto F0D9 F0D2 BPL &F0D9 ;&00-&7F, scan for single keypress (INKEY-129 to -256) F0D4 JSR &F02A ;else interrogate keyboard, scan for single keypress F0D7 BCS &F12E ;if carry set from entry, jump to F12E to exit SKFMA2 F0D9 PHP ;push flags F0DA BCC &F0DE ;if carry clear, Y already set, skip to LF0DE F0DC LDY #&EE ;set Y so next operation saves to &02CD SKFMA1 F0DE STA &01DF,Y ;Store keypress number in &02CB, &02CC or &02CD F0E1 LDX #&09 ;Start with column 9 in X ISCAN1 F0E3 JSR &F129 ;select auto scan F0E6 LDA #&7F ;set port A for input on bit 7 others outputs F0E8 STA &FE43 ; F0EB LDA #&03 ;stop auto scan F0ED STA &FE40 ; F0F0 LDA #&0F ;select non-existent keyboard column F (only 0-9 have keys) F0F2 STA &FE4F ; F0F5 LDA #&01 ;cancel keyboard interrupt F0F7 STA &FE4D ; F0FA STX &FE4F ;select column X (9 max) F0FD BIT &FE4D ;if bit 1 =0 there is no keyboard interrupt so F100 BEQ &F123 ;goto F123 F102 TXA ;else put column address in A ISCANC F103 CMP &01DF,Y ;compare with 1DF+Y F106 BCC &F11E ;if less then F11E F108 STA &FE4F ;else select column again F10B BIT &FE4F ;and if bit 7 is 0 F10E BPL &F11E ;then F11E ISCANB* F110 PLP ;else push and pull flags F111 PHP ; F112 BCS &F127 ;and if carry set goto F127 F114 PHA ;else Push A F115 EOR &0000,Y ;EOR with EC,ED, or EE depending on Y value F118 ASL ;shift left F119 CMP #&01 ;set carry if = or greater than number holds EC,ED,EE F11B PLA ;get back A F11C BCS &F127 ;if carry set F127 ISCANE F11E CLC ;else clear carry F11F ADC #&10 ;add 16 F121 BPL &F103 ;and do it again if 0==3 C set X=3 Z set F13D JMP &F14B ; ******** set cassette options ******************************************* ;called after BREAK etc ;lower nybble sets sequential access ;upper sets load and save options ;0000 Ignore errors, no messages ;0001 Abort if error, no messages ;0010 Retry after error, no messages ;1000 Ignore error short messages ;1001 Abort if error short messages ;1010 Retry after error short messages ;1100 Ignore error long messages ;1101 Abort if error long messages ;1110 Retry after error long messages INIT F140 PHP ;save flags F141 LDA #&A1 ;set sequential access abort if error, no messages F143 STA &E3 ;set load/save retry if error, short messages F145 LDA #&19 ;set interblock gap F147 STA &03D1 ;and store it F14A PLP ;get back flags SOFTPT F14B PHP ;push flags F14C LDA #&06 ;'filing system taking over' call to FSCV F14E JSR &E031 ;call OSFSC F151 LDX #&06 ; F153 PLP ;get back flags F154 BEQ &F157 ;if Z set earlier F156 DEX ;do not decrement X INITON F157 STX &C6 ;set current baud rate X=5 300 baud X=6 1200 baud ********* CFS/RFS claim FILEV,ARGSV,BGETV,BPUTV,GBPBV,FINDV,FSCV ********** ********** set to F27D, F18E, F4C9, F529, FFA6, F3CA, F1B1 **************** F159 LDX #&0E ;CLAIM FILING SYSTEM VECTORS INITLP F15B LDA &D951,X ; F15E STA &0211,X ; F161 DEX ; F162 BNE &F15B ; F164 STX &C2 ;&C2=&00 set progress flag = idle F166 LDX #&0F ;set X to make ROM service call &0F, vectors changed ************************************************************************* * * * OSBYTE 143 * * Pass service commands to sideways ROMs * * * ************************************************************************* ; On entry X = service call number ; Y = any additional parameter ; On exit X=0 if claimed, or preserved if unclaimed ; Y = any returned parameter ; When called internally, EQ set if call claimed SVOP F168 LDA &F4 ; Get current ROM number F16A PHA ; Save it F16B TXA ; Pass service call number to A F16C LDX #&0F ; Start at ROM 15 ; Loop through all 16 ROMs SVOP1 F16E INC &02A1,X ; Read bit 7 on ROM type table F171 DEC &02A1,X ; F174 BPL &F183 ; If not set (+ve result), step to next ROM down F176 STX &F4 ; Otherwise, select this ROM, &F4 RAM copy F178 STX &FE30 ; Page in selected ROM F17B JSR &8003 ; Call the ROM's service entry ; X and P do not need to be preserved by the ROM F17E TAX ; On exit pass A to X to chech if claimed F17F BEQ &F186 ; If 0, service call claimed, reselect ROM and exit F181 LDX &F4 ; Otherwise, get current ROM back SVOP2 F183 DEX ; Step to next ROM down F184 BPL &F16E ; Loop until done ROM 0 SVOP3 F186 PLA ; Get back original ROM number F187 STA &F4 ; Set ROM number RAM copy F189 STA &FE30 ; Page in the original ROM F18C TXA ; Pass X back to A to set zero flag F18D RTS ; And return ************************************************************************* * * * CFS OSARGS entry point * * * ************************************************************************* ARGGO F18E ORA #&00 ;is A=00 F190 BNE &F1A2 ;if not return F192 CPY #&00 ;is Y=0 F194 BNE &F1A2 ;if not return F196 LDA &C6 ;else get current baud rate and zero bit 2 F198 AND #&FB ;C6=5 becomes 1, 6 becomes 2 F19A ORA &0247 ;if cassette selected A=0 else A=2 F19D ASL ;multiply by 2 F19E ORA &0247 ;Or it again F1A1 LSR ;divide by 2 INTRTS F1A2 RTS ;return cassette =0 ************************************************************************* * * * CFS/RFS FSC DISPATCH TABLE * * * ************************************************************************* JMPTAB F1A3 DB 4C,F5 ; 0 *OPT (F54C) F1A5 DB 1D,F6 ; 1 check EOF (F61D) F1A7 DB 04,F3 ; 2 */ (F304) F1A9 DB 0F,E3 ; 3 unknown command (E30F) F1AB DB 04,F3 ; 4 *RUN (F304) F1AD DB 2A,F3 ; 5 *CAT (F32A) F1AF DB 74,E2 ; 6 New filing system (E274) ************************************************************************* * Filing System control entry OSFSC * * Entry via 021E FSCV * * A = FSC function * ************************************************************************* ;on entry A is reason code ;A=0 A *OPT command has been used X & Y are the 2 parameters ;A=1 EOF is being checked, on entry X=File handle on Exit X<>&00 = EOF exists, X=&00 = not at EOF ;A=2 A */ command has been used *RUN the file ;A=3 An unrecognised OS command has ben used X,Y point at command ;A=4 A *RUN command has been used X,Y point at filename ;A=5 A *CAT cammand has been issued X,Y point to rest of command ;A=6 New filing system about to take over, close SPOOL & EXEC files ;A=7 Return in X and Y lowest and highest file handle used ;A=8 OS about to process *command FSFUNC F1B1 CMP #&07 ;if A>6 F1B3 BCS &F1A2 ;goto F1A2 (RTS) F1B5 STX &BC ;else save X F1B7 ASL ;A=A*2 F1B8 TAX ;X=A to get offset F1B9 LDA &F1A4,X ;get hi byte of address-1 F1BC PHA ;push it F1BD LDA &F1A3,X ;get lo byte of address-1 F1C0 PHA ;push it F1C1 LDX &BC ;get back X F1C3 RTS ;this now jumps to the address got from the table +1 ************************************************************************* * * * LOAD FILE * * * ************************************************************************* LOADGO F1C4 PHP ;save flags on stack, allows sibroutines to turn off IRQs F1C5 PHA ;save load/run action F1C6 JSR &FB27 ; Set &BB to load/save options, set &C7=6 for load/save block gap ;claim serial system for cassette F1C9 LDA &03C2 ;execution address LO F1CC PHA ;save A on stack F1CD JSR &F631 ;search for file LOADL6* F1D0 PLA ;get back A F1D1 BEQ &F1ED ;if A=0, RUN, use file's own load address, go to F1ED F1D3 LDX #&03 ;else X=3 F1D5 LDA #&FF ;A=&FF LOADL1 F1D7 PHA ;save A on stack F1D8 LDA &03BE,X ;get load address F1DB STA &B0,X ;store it as current load address F1DD PLA ;get back A F1DE AND &B0,X ;test for &FFFFFFFF load address F1E0 DEX ;X=X-1 F1E1 BPL &F1D7 ;until all 4 bytes copied F1E3 CMP #&FF ;if all bytes contain don't contain &FF F1E5 BNE &F1ED ;continue F1E7 JSR &FAE8 ;else sound bell, reset ACIA & motor off F1EA JMP &E267 ;'Bad Address' error LOADLB F1ED LDA &03CA ; Get block flag F1F0 LSR ; Set carry from Locked flag in bit 0 F1F1 PLA ; Get action back F1F2 BEQ &F202 ; If A=0, *RUN, jump to F202 to test Locked flag F1F4 BCC &F209 ; A<>0, *LOAD, if unlocked, jump to load *************** LOCKED FILE ROUTINE ************************************* PROTER F1F6 JSR &FAF2 ; Release Tube and reset serial system F1F9 BRK ; F1FA DB &E5 ;error number F1FC 'Locked' ; F201 BRK ; LOADLC F202 BCC &F209 ; Carry clear, unlocked, jump to F209 to load file F204 LDA #&03 ; else A=3 F206 STA &0258 ; store to cause ESCAPE disable and memory clear on break LOADLA F209 LDA #&30 ; F20B AND &BB ; Get current error OPTions F20D BEQ &F213 ; If all errors ignored, skip checking checksum F20F LDA &C1 ; Else get checksum result F211 BNE &F21D ; If checksum not ok, jump to F21D to... LOADL8 F213 TYA ; A=Y F214 PHA ; Save Y on stack F215 JSR &FBBB ; Claim Tube, start Host->Tube transfer F218 PLA ; F219 TAY ; Get Y back F21A JSR &F7D5 ; Set block offset=0, status='load data block' LOADL7 F21D JSR &F9B4 ; Load data block F220 BNE &F255 ; If not found return to search F222 JSR &FB69 ; Increment current block number F225 BIT &03CA ; Test block flag F228 BMI &F232 ; If bit 7=1 then this is last block so jump to F232 F22A JSR &F96A ; Else increment current load address F22D JSR &F77B ; Read next block header F230 BNE &F209 ; And loop back to F209 ******** store data in OSFILE parameter block *************************** \ BUG! &C8/9 only set when called from OSFILE, will not be set by *RUN \ BUG! &CD/E only updated when *OPT 1,2 has been used LOADEX F232 LDY #&0A ; Y=>length in control block F234 LDA &CC ; File length counter low F236 STA (&C8),Y ; Store in OSFILE parameter block F238 INY ; Y=Y+1 F239 LDA &CD ; File length counter high F23B STA (&C8),Y ; Store in OSFILE parameter block F23D LDA #&00 ; Always return 16-bit length ; &CE/&CF is unused, could just use 4-byte loop F23F INY ; Y=Y+1 F240 STA (&C8),Y ; Store in OSFILE parameter block F242 INY ; Y=Y+1 F243 STA (&C8),Y ; Store in OSFILE parameter block F245 PLP ; Get flags back FINOP F246 JSR &FAE8 ; Ring bell, reset ACIA & motor NOOLN1 F249 BIT &BA ; Test if error reported, set in current block flag F24B BMI &F254 ; If bit 7 set, already reported an error, skip NOOLIN F24D PHP ; Save flags on stack F24E JSR &FA46 ; Print message following call (in this case NEWLINE!) F251 DB &0D,&00 ; could be replaced by single JSR OSNEWL call F253 PLP ; Restore flags FINOPX F254 RTS ; And return ************RETRY AFTER A FAILURE ROUTINE ******************************* LODRTR F255 JSR &F637 ;search for a specified block F258 BNE &F209 ; loop back to F209 *********** Read Filename using General String Readerr ****************** ;filename pointed to by X and Y MOVFT GETFLN F25A STX &F2 ;OS filename/command line pointer lo F25C STY &F3 ;OS filename/command line pointer hi F25E LDY #&00 ;Y=0 F260 JSR &EA1D ;initialise GS string reading F263 LDX #&00 ;X=0 MOVFTL F265 JSR &EA2F ;call GSREAD F268 BCS &F277 ;if end of character string F277 F26A BEQ &F274 ;if CHR$0 jump to error as &00 used as a terminator F26C STA &03D2,X ;store character in CFS filename area F26F INX ;increment character count F270 CPX #&0B ;if X<>11 F272 BNE &F265 ;loop for up to ten characters STRERR STRNG0 F274 JMP &EA8F ;else Bad String error ************* terminate Filename **************************************** MOVFTX F277 LDA #&00 ;terminate filename with 0 F279 STA &03D2,X ; F27C RTS ;return ************************************************************************* * * * OSFILE ENTRY * * * * on entry A determines action * * A=0 save block of memory as a file * * A=1 write catalogue info for existing file * * A=2 write load address only for existing file * * A=3 write execution address only for existing file * * A=4 write attributes only for existing file * * A=5 Read catalogue info, return file type in A * * A=6 Delete named file * * A=&FF load the named file if lo byte of Exec address=0 use * * address in parameter block else files own load address * * X,Y point to parameter block * * bytes 0,1 filename address, 2-5 load,6-9 exec,A-D length or * * start of data for save, 0E End address /attributes * ************************************************************************* ;parameter block located by XY ;0/1 Address of Filename terminated by &0D ;2/4 Load Address of File ;6/9 Execution Address of File ;A/D Start address of data for write operations or length of file ; for read operations ;E/11 End address of Data; i.e. byte AFTER last byte to be written ; or file attributes ; ;On Entry action is determined by value in A ; ;A=0 Save section of memory as named file, write catalogue information ;A=1 Write catalogue information for named file ;A=2 Write the LOAD address (only) for the named File ;A=3 Write the EXECUTION address (only) for the named File ;A=4 Write the ATTRIBUTES for the named File ;A=5 Read the named files catalogue information Place file type in A ;A=6 Delete the named file ;A=&FF Load the named file and read its catalogue information FILEGO F27D PHA ;save A on stack F27E STX &C8 ;osfile block pointer lo F280 STY &C9 ;osfile block pointer hi F282 LDY #&00 ;Y=0 F284 LDA (&C8),Y ;OSFILE parameter block F286 TAX ;X=A F287 INY ;Y=Y+1 F288 LDA (&C8),Y ;OSFILE parameter block F28A TAY ;Y=A F28B JSR &F25A ;get filename from BUFFER F28E LDY #&02 ;Y=2 FILELP F290 LDA (&C8),Y ;copy parameters to Cassette block at 3BE/C5 F292 STA &03BC,Y ;from LOAD and EXEC address F295 STA &00AE,Y ;make second copy at B0-B8 F298 INY ;Y=Y+1 F299 CPY #&0A ;until Y=10 F29B BNE &F290 ; F29D PLA ;get back A F29E BEQ &F2A7 ;if A=0, jump to do SAVE F2A0 CMP #&FF ;else if A<>&FF F2A2 BNE &F254 ;RETURN as cassette has no other options F2A4 JMP &F1C4 ;if A=&FF, jump to do LOAD ************** Save a file ********************************************** FILEL2 F2A7 STA &03C6 ; Zero block number low F2AA STA &03C7 ; Zero block number high FILEL3 F2AD LDA (&C8),Y ; Copy start/end from control block F2AF STA &00A6,Y ; Store to zero page copy (&B0 to &B7) F2B2 INY ; Data start and data end address F2B3 CPY #&12 ; Loop until Y=18 F2B5 BNE &F2AD ; F2B7 TXA ; A=X F2B8 BEQ &F274 ; If X=0 no filename found so jump to 'Bad string' error F2BA JSR &FB27 ; Set load/save cassette options into &BB, set C7=6 ; Claim serial system for cassette F2BD JSR &F934 ; Prompt to start recording, generate error if ROMFS F2C0 LDA #&00 ; A=0 F2C2 JSR &FBBD ; Claim Tube for Tube->Host transfers F2C5 JSR &FBE2 ; Set up CFS for write operation ; Loop to save blocks SAVELP F2C8 SEC ; Set carry flag F2C9 LDX #&FD ; X=&FD SAVEL1 F2CB LDA &FFB7,X ; Calculate 24-bit end-start F2CE SBC &FFB3,X ; F2D1 STA &02CB,X ; Store remaining length in block header F2D4 INX ; Overlaps into block flag to allow &10000 F2D5 BNE &F2CB ; F2D7 TAY ; Check length b16-b23 F2D8 BNE &F2E8 ; Jump if length>&FFFF F2DA CPX &03C8 ; Check length b0-b7, CC if len<>&xx00, CS if len=&xx00 F2DD LDA #&01 ; A=1 F2DF SBC &03C9 ; Subtract block length high, A=number of blocks to save F2E2 BCC &F2E8 ; If carry clear, 256 or more bytes, jump to save F2E4 LDX #&80 ; If length<257, X=&80, b7=last block F2E6 BNE &F2F0 ; Jump F2F0 SAVEL2 F2E8 LDA #&01 ; Set block length=256 F2EA STA &03C9 ; block length high F2ED STX &03C8 ; block length SAVEL3 F2F0 STX &03CA ; block flag, flags=&00 not last block, or &80 last block F2F3 JSR &F7EC ; Write block to Tape F2F6 BMI &F341 ; If negative, all done, return F2F8 JSR &F96A ; Increment current start address F2FB INC &03C6 ; Increment block number low F2FE BNE &F2C8 ; If not 0, loop back to save next block F300 INC &03C7 ; Increment block number high F303 BNE &F2C8 ; And loop back ************************************************************************* * * * *RUN ENTRY * * * ************************************************************************* RUN F305 JSR &F25A ;get filename from BUFFER F308 LDX #&FF ;X=&FF F30A STX &03C2 ;execution address F30D JSR &F1C4 ;load file F310 BIT &027A ; &FF if tube present F313 BPL &F31F ; No Tube present, enter at I/O address F315 LDA &03C4 ; Get execution address b16-b23 F318 AND &03C5 ; AND with execution address b24-b31 F31B CMP #&FF ; If exec is not &FFFFxxxx, jump to I/O address F31D BNE &F322 ; If exec<>&FFFFxxxx, call Tube NOTUB1 F31F JMP (&03C2) ; Call file in I/O via exec address RUNL1 F322 LDX #&C2 ; Point XY to exec address at &03C2 F324 LDY #&03 ; F326 LDA #&04 ; Tube call 4=Execute code F328 JMP &FBC7 ; And issue to Tube to run file ************************************************************************* * * * *CAT ENTRY * * * ************************************************************************* ;CASSETTE OPTIONS in &E2 ;bit 0 input file open ;bit 1 output file open ;bit 2,4,5 not used ;bit 3 current CATalogue status ;bit 6 EOF reached ;bit 7 EOF warning given CAT F32B LDA #&08 ;A=8 F32D JSR &F344 ;set status bits from A F330 JSR &FB27 ;Set cassette options into (BB),set C7=6 ;claim serial system for cassette F333 LDA #&00 ;A=0 F335 JSR &F348 ;search for a header F338 JSR &FAFC ;perform read CATOFF F33B LDA #&F7 ;A=&F7 ANDSTA F33D AND &E2 ;clear bit 3 of CFS status bit, not cataloging ANDSTB F33F STA &E2 ; SAVFIN F341 RTS ;return ORLST F342 LDA #&40 ;set bit 6 of E2 cassette options, EOF met ORSTAT F344 ORA &E2 ; F346 BNE &F33F ;and Jump F33F ********** search routine *********************************************** LOOK F348 PHA ;save A on stack, A=0 for CAT, A<>0 for LOAD F349 LDA &0247 ;filing system flag 0=CFS 2=RFS F34C BEQ &F359 ;if CFS, jump to read header F34E JSR &EE13 ;set current ROM/PHROM to &EF F351 JSR &EE18 ;get byte from ROM/PHROM, CC=byte returned, CS=no response F354 BCC &F359 ;if carry clear, jump to read header F356 CLV ;clear overflow flag F357 BVC &F39A ;jump to return with VC, CS *********** CFS/RFS read header ******************************************** LOOKLR F359 JSR &F77B ;read block header F35C LDA &03C6 ;block number F35F STA &B4 ;current block no. lo F361 LDA &03C7 ;block number hi F364 STA &B5 ;current block no. hi F366 LDX #&FF ;X=&FF F368 STX &03DF ;set copy of last read block flag to &FF F36B INX ;X=X+1 F36C STX &BA ;set current block flag to &00 F36E BEQ &F376 ;jump to F376 LOOKL1 F370 JSR &FB69 ;inc. current block no. F373 JSR &F77B ;read block header LOOKL5 F376 LDA &0247 ;get filing system flag 0=CFS 2=RFS F379 BEQ &F37D ;if CFS F37D F37B BVC &F39A ;if V clear F39A LOOKL7 F37D PLA ;get back A F37E PHA ;save A on stack F37F BEQ &F3AE ;if A=0, CAT, skip checking filename match F381 JSR &FA72 ;else check filename header block matches searched filename F384 BNE &F39C ;filename doesn't match, jump to keep searching F386 LDA #&30 ;else A=&30 to read bits 4/5 of current OPTions F388 AND &BB ;current error OPTions F38A BEQ &F39A ;if ignore all errors, don't check block number F38C LDA &03C6 ;block number F38F CMP &B6 ;compare with expected block no. lo F391 BNE &F39C ;wrong block F393 LDA &03C7 ;loaded block number hi F396 CMP &B7 ;compare with expected block no. hi F398 BNE &F39C ;wrong block LOOKEX F39A PLA ;get back A F39B RTS ;matching file header found, return ; LOOKL2 F39C LDA &0247 ;filing system flag 0=CFS 2=RFS F39F BEQ &F3AE ;if tape, jump to go back to searching LOOKL8 F3A1 JSR &EEAD ;ROMFS, use ROM displacement address LOOKLA F3A4 LDA #&FF ;set block number to &FFFF F3A6 STA &03C6 ;block number lo F3A9 STA &03C7 ;block number hi F3AC BNE &F370 ;jump F370 LOOKL4 F3AE BVC &F3B5 ;if overflow clear F3B5 F3B0 LDA #&FF ;A=&FF F3B2 JSR &F7D7 ;set flags LOOKL3 F3B5 LDX #&00 ;X=0 F3B7 JSR &F9D9 ;report 'DATA?' F3BA LDA &0247 ;filing system flag 0=CFS 2=RFS F3BD BEQ &F3C3 ; F3BF BIT &BB ;current OPTions F3C1 BVC &F3A1 ;long messages not required if BIT 6 =0 LOOKL9 F3C3 BIT &03CA ;block flag F3C6 BMI &F3A4 ;if -ve F3A4 F3C8 BPL &F370 ;else loop back and do it again ************************************************************************* * * * OSFIND ENTRY * * file handling * * * ************************************************************************* ;on entry A determines Action Y may contain file handle or ;X/Y point to filename terminated by &0D in memory ;A=0 closes file in channel Y if Y=0 closes all files ;A=&40 open a file for input (reading) X/Y points to filename ;A=&80 open a file for output (writing) X/Y points to filename ;A=&C0 open a file for input and output (random access) ;ON EXIT Y=0 if no file found else Y=channel number in use for file ;save A X and Y FIND F3CA STA &BC ;file status or temporary store F3CC TXA ;A=X F3CD PHA ;save X on stack F3CE TYA ;A=Y F3CF PHA ;save Y on stack F3D0 LDA &BC ;file status or temporary store F3D2 BNE &F3F2 ;if A is non zero open a file via F3F2 ************ close a file *********************************************** F3D4 TYA ;A=Y F3D5 BNE &F3E3 ;if A<> 0 close specified file else close them all F3D7 JSR &E275 ;close spool/exec files via OSBYTE 77 F3DA JSR &F478 ;tidy up SHUTR F3DD LSR &E2 ;CFS status byte is shifted left and right to clear F3DF ASL &E2 ;bit 0 F3E1 BCC &F3EF ;and if carry clear no input file was open so F3EF SHUTL1 F3E3 LSR ;A contains file handle so shift bit 0 into carry F3E4 BCS &F3DD ;if carry set close input file F3E6 LSR ;else shift bit 1 into carry F3E7 BCS &F3EC ;if carry set close output file F3E9 JMP &FBB1 ;else report 'Channel Error' as CFS can only support ;1 input and 1 output file SHUTL2 F3EC JSR &F478 ;tidy up JFINDE F3EF JMP &F471 ;and exit ************ OPEN A FILE ************************************************ FINDON F3F2 JSR &F25A ;get filename from BUFFER F3F5 BIT &BC ;file status or temporary store F3F7 BVC &F436 ;check A at input if bit 6 not set its an output file ********* Input files +************************************************** \ OSFIND %x1xxxxxx - &40 and &C0 F3F9 LDA #&00 ;else its an input file F3FB STA &039E ;BGET buffer offset for next byte F3FE STA &03DD ;Expected BGET file block number lo F401 STA &03DE ;expected BGET file block number hi F404 LDA #&3E ;A=&3E F406 JSR &F33D ;CFS status =CFS status AND A F409 JSR &FB1A ;claim serial system and set OPTions F40C PHP ;save flags on stack F40D JSR &F631 ;search for file F410 JSR &F6B4 ;check protection bit of block status and respond F413 PLP ;get back flags F414 LDX #&FF ;X=&FF increment to 0 on next instruction FINDL5 F416 INX ;X=X+1 F417 LDA &03B2,X ;get file name and F41A STA &03A7,X ;store as BGET filename F41D BNE &F416 ;until end of filename F41F LDA #&01 ;A=1 to show file open F421 JSR &F344 ;set status bits from A F424 LDA &02EA ;CFS currently resident file block length lo F427 ORA &02EB ;CFS currently resident file block length hi F42A BNE &F42F ;if block length is 0 F42C JSR &F342 ;set CFS status bit 3 (EOF reached) ;else FINDL6 F42F LDA #&01 ;A=1 F431 ORA &0247 ;filing system flag 0=CFS 2=RFS F434 BNE &F46F ;and exit after restoring registers ******************* open an output file*********************************** \ OSFIND %x0xxxxxx - &80 FINDO F436 TXA ;A=X F437 BNE &F43C ;if X=0 then zero length filename so F439 JMP &EA8F ;Bad String error FINABC F43C LDX #&FF ;X=&FF FINDL2 F43E INX ;X=X+1 ;copy sought filename to header block F43F LDA &03D2,X ;sought filename F442 STA &0380,X ;BPUT file header block F445 BNE &F43E ;until A=0 end of filename F447 LDA #&FF ;A=&FF F449 LDX #&08 ;X=8 FINDL4 F44B STA &038B,X ;set 38C-93 to &FF F44E DEX ;set load/exec to &FFFFFFFF F44F BNE &F44B ; F451 TXA ;A=X=0 F452 LDX #&14 ;X=14 FINDL3 F454 STA &0380,X ;BPUT file header block F457 INX ;X=X+1 F458 CPX #&1E ;this zeros 394-D F45A BNE &F454 ; F45C ROL &0397 ; F45F JSR &FB27 ;Set cassette options into &BB, set &C7=6 ;claim serial system for cassette F462 JSR &F934 ;prompt to start recording, generate error if ROMFS F465 JSR &FAF2 ;claim Tube and reset serial system F468 LDA #&02 ;A=2 F46A JSR &F344 ;set status bits from A F46D LDA #&02 ;A=2 OPENEX F46F STA &BC ;file status or temporary store GBEXX FINDEX F471 PLA ;get back A F472 TAY ;Y=A F473 PLA ;get back A F474 TAX ;X=A F475 LDA &BC ;file status or temporary store FINDXX F477 RTS ;return ; SHUTW F478 LDA #&02 ;A=2 clearing all but bit 1 of status byte F47A AND &E2 ;CFS status byte, with output file open F47C BEQ &F477 ;if file not open then exit F47E LDA #&00 ;else A=0 F480 STA &0397 ;set block length to current value of BPUT offset F483 LDA #&80 ;A=&80 F485 LDX &039D ;get BPUT buffer offset F488 STX &0396 ;set block length to current value of BPUT offset F48B STA &0398 ;mark current block as last F48E JSR &F496 ;save block to tape F491 LDA #&FD ;A=&FD F493 JMP &F33D ;CFS status =CFS status AND A *********** SAVE BLOCK TO TAPE ******************************************** WRTBFR F496 JSR &FB1A ;claim serial system and set OPTions F499 LDX #&11 ;X=11 WRTBL3 F49B LDA &038C,X ;copy header block from 38C-39D F49E STA &03BE,X ;to 3BE/DF F4A1 DEX ;X=X-1 F4A2 BPL &F49B ; ;X=&FF F4A4 STX &B2 ;set start address to &FFFFxxxx F4A6 STX &B3 ;to point to cassette buffers in I/O memory F4A8 INX ;X=X+1, (X=0) F4A9 STX &B0 ;current load address low byte set to &00 F4AB LDA #&09 ;A=9 to set current load address at &900 F4AD STA &B1 ;start address set to &FFFF0900 F4AF LDX #&7F ;X=&7F F4B1 JSR &FB81 ;copy from 301/C+X to 3D2/C sought filename F4B4 STA &03DF ;copy of last read block flag F4B7 JSR &FB8E ;switch Motor On F4BA JSR &FBE2 ;set up CFS for write operation F4BD JSR &F7EC ;write block to Tape F4C0 INC &0394 ;block number lo F4C3 BNE &F4C8 ; F4C5 INC &0395 ;block number hi WRTBL4 F4C8 RTS ;return ************************************************************************* * * * * * OSBGET get a byte from a file * * * * * ************************************************************************* ;on ENTRY Y contains channel number ;on EXIT X and Y are preserved C=0 indicates valid character ; A contains character (or error) A=&FE End Of File ;push X and Y GETBYT F4C9 TXA ;A=X F4CA PHA ;save A on stack F4CB TYA ;A=Y F4CC PHA ;save A on stack F4CD LDA #&01 ;A=1 F4CF JSR &FB9C ;check conditions for OSBGET are OK F4D2 LDA &E2 ;CFS status byte F4D4 ASL ;shift bit 7 into carry (EOF warning given) F4D5 BCS &F523 ;if carry set F523 F4D7 ASL ;shift bit 6 into carry F4D8 BCC &F4E3 ;if clear EOF not reached F4E3 F4DA LDA #&80 ;else A=&80 setting bit 7 of status byte EOF warning F4DC JSR &F344 ;set status bits from A F4DF LDA #&FE ;A=&FE F4E1 BCS &F51B ;if carry set F51B GBLF F4E3 LDX &039E ;BGET buffer offset for next byte F4E6 INX ;X=X+1 F4E7 CPX &02EA ;CFS currently resident file block length lo F4EA BNE &F516 ;read a byte ;else F4EC BIT &02EC ;block flag of currently resident block F4EF BMI &F513 ;if bit 7=1 this is the last block so F513 else F4F1 LDA &02ED ;last character of currently resident block F4F4 PHA ;save A on stack F4F5 JSR &FB1A ;claim serial system and set OPTions F4F8 PHP ;save flags on stack F4F9 JSR &F6AC ;read in a new block F4FC PLP ;get back flags F4FD PLA ;get back A F4FE STA &BC ;file status or temporary store F500 CLC ;clear carry flag F501 BIT &02EC ;block flag of currently resident block F504 BPL &F51D ;if not last block (bit 7=0) F506 LDA &02EA ;CFS currently resident file block length lo F509 ORA &02EB ;CFS currently resident file block length hi F50C BNE &F51D ;if block size not 0 F51D else F50E JSR &F342 ;set CFS status bit 6 (EOF reached) F511 BNE &F51D ;goto F51D GBLH F513 JSR &F342 ;set CFS status bit 6 (EOF reached) GBLDE F516 DEX ;X=X-1 F517 CLC ;clear carry flag F518 LDA &0A00,X ;read byte from cassette buffer GBEXY F51B STA &BC ;file status or temporary store GBEXIT F51D INC &039E ;BGET buffer offset for next byte F520 JMP &F471 ;exit via F471 GBEOF F523 BRK ; F524 DB &DF ;error number F525 DB 'EOF' ; F528 BRK ; ************************************************************************* * * * * * OSBPUT WRITE A BYTE TO FILE * * * * * ************************************************************************* ;ON ENTRY Y contains channel number A contains byte to be written PUTBYT F529 STA &C4 ;store A in temorary store F52B TXA ;and stack X and Y F52C PHA ;save on stack F52D TYA ;A=Y F52E PHA ;save on stack F52F LDA #&02 ;A=2 F531 JSR &FB9C ;check conditions necessary for OSBPUT are OK F534 LDX &039D ;BPUT buffer offset for next byte F537 LDA &C4 ;get back original value of A F539 STA &0900,X ;Cassette buffer F53C INX ;X=X+1 F53D BNE &F545 ;if not 0 F545, otherwise buffer is full so F53F JSR &F496 ;save block to tape F542 JSR &FAF2 ;enable second processor and reset serial system PBEXIT F545 INC &039D ;BPUT buffer offset for next byte F548 LDA &C4 ;get back A F54A JMP &F46F ;and exit ************************************************************************* * * * * * OSBYTE 139 Select file options * * * * * ************************************************************************* ;ON ENTRY Y contains option value X contains option No. see *OPT X,Y ;this applies largely to CFS LOAD SAVE CAT and RUN ;X=1 is message switch ; Y=0 no messages ; Y=1 short messages ; Y=2 gives detailed information on load and execution addresses ;X=2 is error handling ; Y=0 ignore errors ; Y=1 prompt for a retry ; Y=2 abort operation ;X=3 is interblock gap for BPUT# and PRINT# ; Y=1-127 set gap in 1/10ths Second ; Y=0 or Y>127 use default values CFSOPT F54D TXA ; A=X F54E BEQ &F57E ; *OPT 0 - set default options F550 CPX #&03 ; if X=3 F552 BEQ &F573 ; F573 to set interblock gap F554 CPY #&03 ; else if Y>2 then BAD COMMAND error F556 BCS &F55E ; F558 DEX ; X=X-1 F559 BEQ &F561 ; i.e. if X=1 F561 message control F55B DEX ; X=X-1 F55C BEQ &F568 ; i.e. if X=2 F568 error response OPTERR F55E JMP &E310 ; else E310 to issue Bad Command error *********** message control ********************************************* MSGS F561 LDA #&33 ; keep lower two bits of each nybble F563 INY ; Y=Y+1 F564 INY ; Y=Y+1 F565 INY ; Y=Y+1 F566 BNE &F56A ; go to F56A *********** error response ********************************************* ERRS F568 LDA #&CC ; keep top two bits of each nybble XSETFL F56A INY ; Y=Y+1 F56B AND &E3 ; clear lower two bits of each nybble SETFL1 F56D ORA &F581,Y ; ORA with table value F570 STA &E3 ; store it in &E3 F572 RTS ; return ;setting of &E3 ; ;lower nybble sets BGET/BPUT options ;upper nybble sets LOAD/SAVE options ;0000 Ignore errors, no messages ;0001 Abort if error, no messages ;0010 Retry after error, no messages ;1000 Ignore error short messages ;1001 Abort if error short messages ;1010 Retry after error short messages ;1100 Ignore error long messages ;1101 Abort if error long messages ;1110 Retry after error long messages ***********set interblock gap ******************************************* SETGAP F573 TYA ; A=Y F574 BMI &F578 ; If Y>127 use default values F576 BNE &F57A ; If Y<>0 don't use default value SETGP2 F578 LDA #&19 ; Else A=&19 default value SETGP1 F57A STA &03D1 ; Sequential block gap F57D RTS ; Return DEF F57E TAY ;Y=A F57F BEQ &F56D ;jump to F56D *********** DEFAULT OPT VALUES TABLE ************************************ OPTTAB F581 DB &A1 ; %1010 0001 - default ; ; +------- load/save=short messages, retry errors ; ; +-- bget/bput=no messages, abort errors F582 DB &00 ; %0000 0000 - *OPT 2,0 - ignore errors F583 DB &22 ; %0010 0010 - *OPT 2,1 - retry errors F584 DB &11 ; %0001 0001 - *OPT 2,2 - abort errors F585 DB &00 ; %0000 0000 - *OPT 1,0 - no messages F586 DB &88 ; %1000 1000 - *OPT 1,1 - short messages F587 DB &CC ; %1100 1100 - *OPT 1,2 - long messages IRUPT F588 DEC &C0 ;filing system buffer flag F58A LDA &0247 ;filing system flag 0=CFS 2=RFS F58D BEQ &F596 ;if CFS F596 F58F JSR &EE51 ;read RFS data rom or Phrom F592 TAY ;Y=A F593 CLC ;clear carry flag F594 BCC &F5B0 ;jump to F5B0 NOTSPK F596 LDA &FE08 ;ACIA status register F599 PHA ;save A on stack F59A AND #&02 ;clear all but bits 0,1 A=(0-3) F59C BEQ &F5A9 ;if 0 F5A9 transmit data register full or RDR empty F59E LDY &CA ; F5A0 BEQ &F5A9 ; F5A2 PLA ;get back A F5A3 LDA &BD ;character temporary storage F5A5 STA &FE09 ;ACIA transmit data register F5A8 RTS ;return ; ITXON F5A9 LDY &FE09 ;read ACIA recieve data register F5AC PLA ;get back A F5AD LSR ;bit 2 to carry (data carrier detect) F5AE LSR ; F5AF LSR ; DOIRPT F5B0 LDX &C2 ;progress flag F5B2 BEQ &F61D ;if &C2=0 exit F5B4 DEX ;X=X-1 F5B5 BNE &F5BD ;if &C2>1 then F5BD F5B7 BCC &F61D ;else if carrier tone from cassette detected exit F5B9 LDY #&02 ;Y=2 F5BB BNE &F61B ; FNDHDR F5BD DEX ;X=X-1 F5BE BNE &F5D3 ;if &C2>2 F5C0 BCS &F61D ;if carrier tone from cassette not detected exit F5C2 TYA ;A=byte read F5C3 JSR &FB78 ;set (BE/C0) to 0 F5C6 LDY #&03 ;Y=3, sync byte found F5C8 CMP #&2A ;is A= to synchronising byte &2A? F5CA BEQ &F61B ;if so F61B F5CC JSR &FB50 ;control cassette system F5CF LDY #&01 ;Y=1, waiting for carrier tone F5D1 BNE &F61B ;goto F61B RDCHAR F5D3 DEX ;X=X-1 F5D4 BNE &F5E2 ;if &C2>3 F5D6 BCS &F5DC ; F5D8 STY &BD ;get character read into Y F5DA BEQ &F61D ;if 0 exit via F61D GOTDCD F5DC LDA #&80 ;else A=&80 F5DE STA &C0 ;filing system buffer flag F5E0 BNE &F61D ;and exit RDBLK F5E2 DEX ;X=X-1 F5E3 BNE &F60E ;if &C2>4 F60E F5E5 BCS &F616 ;if carry set F616 F5E7 TYA ;A=Y F5E8 JSR &F7B0 ;perform CRC F5EB LDY &BC ;get offset into current block F5ED INC &BC ;increment offset into current block F5EF BIT &BD ;if bit 7 set this is the last byte read F5F1 BMI &F600 ;so F600 F5F3 JSR &FBD3 ;check if second processor file test tube prescence F5F6 BEQ &F5FD ;if return with A=0 F5FD F5F8 STX &FEE5 ;Tube FIFO3 F5FB BNE &F600 ; RDBLKB F5FD TXA ;A=X restore value F5FE STA (&B0),Y ;store to current load address RDBLKA F600 INY ;Y=Y+1 F601 CPY &03C8 ;block length F604 BNE &F61D ;exit with NE F606 LDA #&01 ;A=1, two bytes of CRC to fetch F608 STA &BC ;store for next call F60A LDY #&05 ;Y=5 F60C BNE &F61B ;exit, setting progress=&05, looking for CRC ; Progress=&05, fetching CRC SUMIN F60E TYA ;A=Y F60F JSR &F7B0 ;perform CRC F612 DEC &BC ;decrement number of CRC bytes to fetch F614 BPL &F61D ;more to fetch, exit IFIN F616 JSR &FB46 ;reset ACIA F619 LDY #&00 ;Y=0, progress=idle NEWINT F61B STY &C2 ;set progress flag IEX F61D RTS ;return ************************************************************************* * * * FSCV &01 - check for end of file * * * ************************************************************************* ; CFSEOF F61E PHA ;save A on stack F61F TYA ;A=Y F620 PHA ;save Y on stack F621 TXA ;A=X to put X into Y F622 TAY ;Y=A F623 LDA #&03 ;A=3 F625 JSR &FB9C ;confirm file is open F628 LDA &E2 ;CFS status byte F62A AND #&40 ; F62C TAX ;X=A F62D PLA ;get back A F62E TAY ;Y=A F62F PLA ;get back A F630 RTS ;return ; Search for file SRCH0 F631 LDA #&00 ;A=0 F633 STA &B4 ;current block no. lo F635 STA &B5 ;current block no. hi SEARCH F637 LDA &B4 ;current block no. lo F639 PHA ;save current block lo on stack F63A STA &B6 ;set expected block to current block F63C LDA &B5 ;current block no. hi F63E PHA ;save current block hi F63F STA &B7 ;set expected block to current block F641 JSR &FA46 ;print inline message F644 DB 'Searching'; F64C DB &0D ;newline F64E BRK ; F64F LDA #&FF ; A=&FF for loading (A=0 for CAT) F651 JSR &F348 ; Search for header F654 PLA ; F655 STA &B5 ; Restore current block hi F657 PLA ; F658 STA &B4 ; Restore current block lo F65A LDA &B6 ; Check expected block number F65C ORA &B7 ; F65E BNE &F66D ; If expected block<>0, jump to return F660 STA &B4 ; Set current block back to zero F662 STA &B5 ; F664 LDA &C1 ; Get checksum result F666 BNE &F66D ; Bad checksum, skip setting current filename F668 LDX #&B1 ; X+1=>filename in block header F66A JSR &FB81 ; Copy filename from block header to current filename SRCHXX F66D LDA &0247 ; Filing system flag 0=CFS 2=RFS F670 BEQ &F685 ; If cassette, exit with Y=&FF at F685 F672 BVS &F685 ; If ROMFS and not end-of-ROM, exit FNF F674 BRK ; ROMFS and no more ROM, give error F675 DB &D6 ; Error number F676 DB 'File Not found' F684 BRK ; SRCHYY F685 LDY #&FF ; Y=&FF F687 STY &03DF ; Copy of last read block flag F68A RTS ; Return ******** CLOSE EXEC FILE ********************************************** ECLOSE F68B LDA #&00 ;A=0 ************************************************************************* * * * *EXEC * * * ************************************************************************* EXEC F68D PHP ;save flags on stack F68E STY &E6 ;&E6=Y F690 LDY &0256 ;EXEC file handle F693 STA &0256 ;EXEC file handle F696 BEQ &F69B ;if not 0 close file via OSFIND F698 JSR OSFIND ; EXEC2 F69B LDY &E6 ;else Y= original Y F69D PLP ;get back flags F69E BEQ &F6AB ;if A=0 on entry exit else F6A0 LDA #&40 ;A=&40 F6A2 JSR OSFIND ;to open an input file F6A5 TAY ;Y=A F6A6 BEQ &F674 ;If Y=0 'File not found' else store F6A8 STA &0256 ;EXEC file handle EXEC0 F6AB RTS ;return ******* read a block ************************************************* RDBFER F6AC LDX #&A6 ;X=&A6 F6AE JSR &FB81 ;copy from 301/C+X to 3D2/C sought filename F6B1 JSR &F77B ;read block header RDBFRS F6B4 LDA &03CA ;block flag F6B7 LSR ;A=A/2 bit 0 into carry to check for locked file F6B8 BCC &F6BD ;if not set then skip next instruction F6BA JMP &F1F6 ;'locked' file routine RDBFLL F6BD LDA &03DD ;Expected BGET file block number lo F6C0 STA &B4 ; current block no. lo F6C2 LDA &03DE ;expected BGET file block number hi F6C5 STA &B5 ;current block no. hi F6C7 LDA #&00 ;A=0 F6C9 STA &B0 ;current load address F6CB LDA #&0A ;A=&A setting current load address to the CFS/RFS F6CD STA &B1 ;current load address buffer at &A00 F6CF LDA #&FF ;A=&FF to set other 2 bytes F6D1 STA &B2 ;current load address high word F6D3 STA &B3 ;current load address high word F6D5 JSR &F7D5 ;reset flags F6D8 JSR &F9B4 ;load file from tape F6DB BNE &F702 ;if return non zero F702 else F6DD LDA &0AFF ;get last character from input buffer F6E0 STA &02ED ;last character currently resident block F6E3 JSR &FB69 ;inc. current block no. F6E6 STX &03DD ;expected BGET file block number lo F6E9 STY &03DE ;expected BGET file block number hi F6EC LDX #&02 ;X=2 RDBFL1 F6EE LDA &03C8,X ;read bytes from block flag/block length F6F1 STA &02EA,X ;store into current values of above F6F4 DEX ;X=X-1 F6F5 BPL &F6EE ;until X=-1 (&FF) F6F7 BIT &02EC ;block flag of currently resident block F6FA BPL &F6FF ; F6FC JSR &F249 ;print newline if needed RDBFL8 F6FF JMP &FAF2 ;enable second processor and reset serial system RDBERR F702 JSR &F637 ;search for a specified block F705 BNE &F6B4 ;if NE check for locked condition else RDHDLC F707 CMP #&2A ;is it Synchronising byte &2A? F709 BEQ &F742 ;if so F742 F70B CMP #&23 ;else is it &23 (header substitute in ROM files) F70D BNE &F71E ;if not BAD ROM error F70F INC &03C6 ;block number F712 BNE &F717 ; F714 INC &03C7 ;block number hi RDHDX2 F717 LDX #&FF ;X=&FF F719 BIT &D9B7 ;to set V & M F71C BNE &F773 ;and jump (ALWAYS!!) to F773 ROMER F71E LDA #&F7 ;clear bit 3 of RFS status (current CAT status) F720 JSR &F33D ;RFS status =RFS status AND A F723 BRK ;and cause error F724 DB &D7 ;error number F725 DB 'Bad Rom' F72C BRK ; **********: pick up a header ******************************************** RDHDL2 F72D LDY &FF ;get ESCAPE flag F72F JSR &FB90 ;switch Motor on F732 LDA #&01 ;A=1 F734 STA &C2 ;progress flag F736 JSR &FB50 ;control serial system GETHLP F739 JSR &F995 ;confirm ESC not set and CFS not executing F73C LDA #&03 ;A=3 F73E CMP &C2 ;progress flag F740 BNE &F739 ;back until &C2=3 RDHDLA F742 LDY #&00 ;Y=0 F744 JSR &FB7C ;zero checksum bytes RDHDL1 F747 JSR &F797 ;get character from file and do CRC F74A BVC &F766 ;if V clear on exit F766 F74C STA &03B2,Y ;else store F74F BEQ &F757 ;or if A=0 F757 F751 INY ;Y=Y+1 F752 CPY #&0B ;if Y<>&B F754 BNE &F747 ;go back for next character F756 DEY ;Y=Y-1 RDHDL3 F757 LDX #&0C ;X=12 RDHDL4 F759 JSR &F797 ;get character from file and do CRC F75C BVC &F766 ;if V clear on exit F766 F75E STA &03B2,X ;else store byte F761 INX ;X=X+1 F762 CPX #&1F ;if X<>31 F764 BNE &F759 ;goto F759 RDHDEX F766 TYA ;A=Y F767 TAX ;X=A F768 LDA #&00 ;A=0 F76A STA &03B2,Y ;store it F76D LDA &BE ;CRC workspace F76F ORA &BF ;CRC workspace F771 STA &C1 ;Checksum result RDHDX1 F773 JSR &FB78 ;set (BE/C0) to 0 F776 STY &C2 ;progress flag F778 TXA ;A=X F779 BNE &F7D4 ; RDHEAD F77B LDA &0247 ;filing system flag 0=CFS 2=RFS F77E BEQ &F72D ;if cassette F72D RDHDLD F780 JSR &EE51 ;read RFS data rom or Phrom F783 CMP #&2B ;is it ROM file terminator? F785 BNE &F707 ;if not F707 ********* terminator found ********************************************** F787 LDA #&08 ;A=8 isolating bit 3 CAT status F789 AND &E2 ;CFS status byte F78B BEQ &F790 ;if clera skip next instruction F78D JSR &F24D ;print CR if CFS not operational RDHDL5 F790 JSR &EE18 ;get byte from data Rom F793 BCC &F780 ;if carry set F780 F795 CLV ;clear overflow flag F796 RTS ;return **************** get character from file and do CRC ******************* ; CASIN F797 LDA &0247 ;filing system flag 0=CFS 2=RFS F79A BEQ &F7AD ;if cassette F7AD F79C TXA ;A=X to save X and Y F79D PHA ;save X on stack F79E TYA ;A=Y F79F PHA ;save Y on stack F7A0 JSR &EE51 ;read RFS data rom or Phrom F7A3 STA &BD ;put it in temporary storage F7A5 LDA #&FF ;A=&FF F7A7 STA &C0 ;filing system buffer flag F7A9 PLA ;get back Y F7AA TAY ;Y=A F7AB PLA ;get back X F7AC TAX ;X=A CASIN1 F7AD JSR &F884 ;check for Escape and loop till bit 7 of FS buffer ;flag=1 ************************** perform CRC ********************************** ADDCRC F7B0 PHP ;save flags on stack F7B1 PHA ;save A on stack F7B2 SEC ;set carry flag F7B3 ROR &CB ;CRC Bit counter F7B5 EOR &BF ;CRC workspace F7B7 STA &BF ;CRC workspace CRC1 F7B9 LDA &BF ;CRC workspace F7BB ROL ;A=A*2 C=bit 7 F7BC BCC &F7CA ; F7BE ROR ;A=A/2 F7BF EOR #&08 ; F7C1 STA &BF ;CRC workspace F7C3 LDA &BE ;CRC workspace F7C5 EOR #&10 ; F7C7 STA &BE ;CRC workspace F7C9 SEC ;set carry flag CRC2 F7CA ROL &BE ;CRC workspace F7CC ROL &BF ;CRC workspace F7CE LSR &CB ;CRC Bit counter F7D0 BNE &F7B9 ; F7D2 PLA ;get back A F7D3 PLP ;get back flags RDHDXX F7D4 RTS ;return READBK RDBLOK F7D5 LDA #&00 ;A=0 CHKBIN F7D7 STA &BD ;&BD=character temporary storage buffer=0 F7D9 LDX #&00 ;X=0 F7DB STX &BC ;file status or temporary store F7DD BVC &F7E9 ; F7DF LDA &03C8 ;block length F7E2 ORA &03C9 ;block length hi F7E5 BEQ &F7E9 ;if 0 F7E9 F7E7 LDX #&04 ;else X=4 RDBKEM F7E9 STX &C2 ;filename length/progress flag F7EB RTS ;return *************** SAVE A BLOCK ******************************************** WBLOCK WRBLOK F7EC PHP ;save flags on stack F7ED LDX #&03 ;X=3 F7EF LDA #&00 ;A=0 SETSPR F7F1 STA &03CB,X ;clear 03CB/E (RFS EOF+1?) F7F4 DEX ;X=X-1 F7F5 BPL &F7F1 ; F7F7 LDA &03C6 ;block number F7FA ORA &03C7 ;block number hi F7FD BNE &F804 ;if block =0 F804 else F7FF JSR &F892 ;generate a 5 second delay F802 BEQ &F807 ;goto F807 WRBKL5 F804 JSR &F896 ;generate delay set by interblock gap WRBKL4 F807 LDA #&2A ;A=&2A F809 STA &BD ;store it in temporary file F80B JSR &FB78 ;set (BE/C0) to 0 F80E JSR &FB4A ;set ACIA control register F811 JSR &F884 ;check for Escape and loop till bit 7 of FS buffer ;flag=1 F814 DEY ;Y=Y-1 WRBKL1 F815 INY ;Y=Y+1 F816 LDA &03D2,Y ;move sought filename F819 STA &03B2,Y ;into filename block F81C JSR &F875 ;transfer byte to CFS and do CRC F81F BNE &F815 ;if filename not complet then do it again ******: deal with rest of header **************************************** F821 LDX #&0C ;X=12 WRBKL2 F823 LDA &03B2,X ;get filename byte F826 JSR &F875 ;transfer byte to CFS and do CRC F829 INX ;X=X+1 F82A CPX #&1D ;until X=29 F82C BNE &F823 ; F82E JSR &F87B ;save checksum to TAPE reset buffer flag F831 LDA &03C8 ;block length F834 ORA &03C9 ;block length hi F837 BEQ &F855 ;if 0 F855 F839 LDY #&00 ;else Y=0 F83B JSR &FB7C ;zero checksum bytes WRBKL3 F83E LDA (&B0),Y ;get a data byte F840 JSR &FBD3 ;check if second processor file test tube prescence F843 BEQ &F848 ;if not F848 else F845 LDX &FEE5 ;Tube FIFO3 WRBKL6 F848 TXA ;A=X F849 JSR &F875 ;transfer byte to CFS and do CRC F84C INY ;Y=Y+1 F84D CPY &03C8 ;block length F850 BNE &F83E ; F852 JSR &F87B ;save checksum to TAPE reset buffer flag WRBKEM F855 JSR &F884 ;check for Escape and loop till bit 7 of FS buffer ;flag=1 F858 JSR &F884 ;check for Escape and loop till bit 7 of FS buffer ;flag=1 F85B JSR &FB46 ;reset ACIA F85E LDA #&01 ;A=1 F860 JSR &F898 ;generate 0.1 * A second delay F863 PLP ;get back flags F864 JSR &F8B9 ;update block flag, PRINT filename (& address if reqd) F867 BIT &03CA ;block flag F86A BPL &F874 ;is this last block (bit 7 set)? F86C PHP ;save flags on stack F86D JSR &F892 ;generate a 5 second delay F870 JSR &F246 ;sound bell and abort F873 PLP ;get back flags WRBKEX F874 RTS ;return ****************** transfer byte to CFS and do CRC ********************** ; PUTCHK F875 JSR &F882 ;save byte to buffer, transfer to CFS & reset flag F878 JMP &F7B0 ;perform CRC ***************** save checksum to TAPE reset buffer flag **************** PUTCRC F87B LDA &BF ;CRC workspace F87D JSR &F882 ;save byte to buffer, transfer to CFS & reset flag F880 LDA &BE ;CRC workspace ************** save byte to buffer, transfer to CFS & reset flag ******** PUTCAS F882 STA &BD ;store A in temporary buffer ***** check for Escape and loop untill bit 7 of FS buffer flag=1 *********** IWAIT F884 JSR &F995 ;confirm ESC not set and CFS not executing F887 BIT &C0 ;filing system buffer flag F889 BPL &F884 ;loop until bit 7 of &C0 is set F88B LDA #&00 ;A=0 F88D STA &C0 ;filing system buffer flag F88F LDA &BD ;get temporary store byte F891 RTS ;return ****************** generate a 5 second delay *************************** TAIL LEADER F892 LDA #&32 ;A=50 F894 BNE &F898 ;generate delay 100ms *A (5 seconds) *************** generate delay set by interblock gap ******************** PUTGAP F896 LDA &C7 ;get current interblock flag *************** generate delay ****************************************** PAUSE F898 LDX #&05 ;X=5 PSLOP1 F89A STA &0240 ;CFS timeout counter PSLOOP F89D JSR &F995 ;confirm ESC not set and CFS not executing F8A0 BIT &0240 ;CFS timeout counter (decremented each 20ms) F8A3 BPL &F89D ;if +ve F89D F8A5 DEX ;X=X-1 F8A6 BNE &F89A ; F8A8 RTS ;return ************: generate screen reports *********************************** PRTHDR F8A9 LDA &03C6 ;block number F8AC ORA &03C7 ;block number hi F8AF BEQ &F8B6 ;if 0 F8B6 F8B1 BIT &03DF ;copy of last read block flag F8B4 BPL &F8B9 ;update block flag, PRINT filename (& address if reqd) PRTHL8 PRTHDL F8B6 JSR &F249 ;print newline if needed ************** update block flag, PRINT filename (& address if reqd) **** SAVHDR PRTHL3 F8B9 LDY #&00 ;Y=0 F8BB STY &BA ;current block flag F8BD LDA &03CA ;block flag F8C0 STA &03DF ;copy of last read block flag F8C3 JSR &E7DC ;check if free to print message F8C6 BEQ &F933 ;if A=0 on return Cassette system is busy F8C8 LDA #&0D ;else A=&0D :carriage return F8CA JSR OSWRCH ;print it (note no linefeed as it's via OSWRCH) PRTHL1 F8CD LDA &03B2,Y ;get byte from filename F8D0 BEQ &F8E2 ;if 0 filename is ended F8D2 CMP #&20 ;if ? F988 CMP #&3A ;if A< ASC(':') F98A BCC &F98E ;goto F98E F98C ADC #&06 ;else add 7 to convert : ; < = > ? to A B C D E F HXDSKP F98E JMP OSWRCH ;print character and return ******************** print a space ************************************* PRTSP F991 LDA #&20 ;A=' ' F993 BNE &F98E ;goto F98E to print it ******************** confirm CFS not operating, nor ESCAPE flag set ***** ESCAPE F995 PHP ;save flags on stack F996 BIT &EB ;CFS Active flag F998 BMI &F99E ; F99A BIT &FF ;if ESCAPE condition F99C BMI &F9A0 ;goto F9A0 ESCLL0 F99E PLP ;get back flags F99F RTS ;return ESCERR F9A0 JSR &F33B ;close input file F9A3 JSR &FAF2 ;enable second processor and reset serial system F9A6 LDA #&7E ;A=&7E (126) Acknowledge ESCAPE F9A8 JSR OSBYTE ;OSBYTE Call F9AB BRK ; F9AC DB &11 ;error 17 F9AD DB 'Escape' ; F9B3 BRK ; OS SERIES 10 LAST PART GEOFF COX ****************************** LOAD ************************************* LODHDR F9B4 TYA ;A=Y F9B5 BEQ &F9C4 ; F9B7 JSR &FA46 ; print message following call F9BA DB &0D ; F9BB DB 'Loading'; F9C2 DB &0D ; F9C3 BRK ; LODHD1 F9C4 STA &BA ;current block flag F9C6 LDX #&FF ;X=&FF F9C8 LDA &C1 ;Checksum result F9CA BNE &F9D9 ;if not 0 F9D9 F9CC JSR &FA72 ;else check filename header block matches searched ;filename if this returns NE then no match F9CF PHP ;save flags on stack F9D0 LDX #&FF ;X=&FF F9D2 LDY #&99 ;Y=&99 F9D4 LDA #&FA ;A=&FA this set Y/A to point to 'File?' FA99 F9D6 PLP ;get back flags F9D7 BNE &F9F5 ;report a query unexpected file name CATERR HDRCHK F9D9 LDY #&8E ;making Y/A point to 'Data' FA8E for CRC error F9DB LDA &C1 ;Checksum result F9DD BEQ &F9E3 ;if 0 F9E3 F9DF LDA #&FA ;A=&FA F9E1 BNE &F9F5 ;jump to F9F5 CHKBKN F9E3 LDA &03C6 ;block number F9E6 CMP &B4 ;current block no. lo F9E8 BNE &F9F1 ;if not eual F9F1 F9EA LDA &03C7 ;block number hi F9ED CMP &B5 ;current block no. hi F9EF BEQ &FA04 ;if equal FA04 BLKERR F9F1 LDY #&A4 ;Y=&A4 F9F3 LDA #&FA ;A=&FA point to 'Block?' error unexpected block no. ;at this point an error HAS occurred TAPER0 F9F5 PHA ;save A on stack F9F6 TYA ;A=Y F9F7 PHA ;save Y on stack F9F8 TXA ;A=X F9F9 PHA ;save X on stack F9FA JSR &F8B6 ;print CR if indicated by current block flag F9FD PLA ;get back A F9FE TAX ;X=A F9FF PLA ;get back A FA00 TAY ;Y=A FA01 PLA ;get back A FA02 BNE &FA18 ;jump to FA18 CHEKSM FA04 TXA ;A=X FA05 PHA ;save A on stack FA06 JSR &F8A9 ;report FA09 JSR &FAD6 ;check loading progress, read another byte FA0C PLA ;get back A FA0D TAX ;X=A FA0E LDA &BE ;CRC workspace FA10 ORA &BF ;CRC workspace FA12 BEQ &FA8D ; FA14 LDY #&8E ;Y=&8E FA16 LDA #&FA ;A=&FA FA8E points to 'Data?' TAPERR FA18 DEC &BA ;current block flag FA1A PHA ;save A on stack FA1B BIT &EB ;CFS Active flag FA1D BMI &FA2C ;if active FA2C FA1F TXA ;A=X FA20 AND &0247 ;filing system flag 0=CFS 2=RFS FA23 BNE &FA2C ; FA25 TXA ;A=X FA26 AND #&11 ; FA28 AND &BB ;current OPTions FA2A BEQ &FA3C ;ignore errors ERRABT FA2C PLA ;get back A FA2D STA &B9 ;store A on &B9 FA2F STY &B8 ;store Y on &B8 FA31 JSR &F68B ;do *EXEC 0 to tidy up FA34 LSR &EB ;halve CFS Active flag to clear bit 7 FA36 JSR &FAE8 ;bell, reset ACIA & motor FA39 JMP (&00B8) ;display selected error report PRTERR FA3C PLA ;get back A FA3D INY ;Y=Y+1 FA3E BNE &FA43 ; FA40 CLC ;clear carry flag FA41 ADC #&01 ;Add 1 PRTEON FA43 PHA ;save A on stack FA44 TYA ;A=Y FA45 PHA ;save Y on stack MSGOUT FA46 JSR &E7DC ;check if free to print message FA49 TAY ;Y=A STROUT FA4A PLA ;get back A FA4B STA &B8 ;&B8=8 FA4D PLA ;get back A FA4E STA &B9 ;&B9=A FA50 TYA ;A=Y FA51 PHP ;save flags on stack VSTRLP FA52 INC &B8 ; FA54 BNE &FA58 ; FA56 INC &B9 ; VSTRL1 FA58 LDY #&00 ;Y=0 FA5A LDA (&B8),Y ;get byte FA5C BEQ &FA68 ;if 0 Fa68 FA5E PLP ;get back flags FA5F PHP ;save flags on stack FA60 BEQ &FA52 ;if 0 FA52 to get next character FA62 JSR OSASCI ;else print FA65 JMP &FA52 ;and do it again VSTREX FA68 PLP ;get back flags FA69 INC &B8 ;increment pointers FA6B BNE &FA6F ; FA6D INC &B9 ; VSTREZ FA6F JMP (&00B8) ;and print error message so no error condition ;occcurs ************ compare filenames ****************************************** CMPFT FA72 LDX #&FF ;X=&FF inx will mean X=0 CMPFT2 FA74 INX ;X=X+1 CMPFT1* FA75 LDA &03D2,X ;sought filename byte FA78 BNE &FA81 ;if not 0 FA81 FA7A TXA ;else A=X FA7B BEQ &FA80 ;if X=0 A=0 exit FA7D LDA &03B2,X ;else A=filename byte CMPFT0 FA80 RTS ;return ; CMPFT3 FA81 JSR &E4E3 ;set carry if byte in A is not upper case Alpha FA84 EOR &03B2,X ;compare with filename FA87 BCS &FA8B ;if carry set FA8B FA89 AND #&DF ;else convert to upper case NALPHA FA8B BEQ &FA74 ;and if A=0 filename characters match so do it again ERREXX FA8D RTS ;return ; DATAER FA8E BRK ; FA8F DB &D8 ;error number FA90 DB 'Data' ; FA96 BRK ; FA97 BNE &FAAE ; FILEER FA99 BRK ; FA9A DB &DB ;error number FA9B DB 'File?' ; FAA1 BRK ; FAA2 BNE &FAAE ; BLOKER FAA4 BRK ; FAA5 DB &DA ;error number FAA6 DB 'Block?' FAAD BRK ; REWIND FAAE LDA &BA ;current block flag FAB0 BEQ &FAD3 ;if 0 FAD3 else FAB2 TXA ;A=X FAB3 BEQ &FAD3 ;If X=0 FAD3 FAB5 LDA #&22 ;A=&22 FAB7 BIT &BB ;current OPTions checking bits 1 and 5 FAB9 BEQ &FAD3 ;if neither set no retry so FAD3 else FABB JSR &FB46 ;reset ACIA FABE TAY ;Y=A FABF JSR &FA4A ;print following message FAC2 DB &0D ;Carriage RETURN FAC3 DB &07 ;BEEP FAC4 DB 'Rewind Tape' ; FACF DW &0D0D ;two more newlines FAD1 BRK ; CSON1 FAD2 RTS ;return ERROEX FAD3 JSR &F24D ;print CR if CFS not operational CHEKLP FAD6 LDA &C2 ;filename length/progress flag FAD8 BEQ &FAD2 ;if 0 return else FADA JSR &F995 ;confirm ESC not set and CFS not executing FADD LDA &0247 ;filing system flag 0=CFS 2=RFS FAE0 BEQ &FAD6 ;if CFS FAD6 FAE2 JSR &F588 ;else set up ACIA etc FAE5 JMP &FAD6 ;and loop back again ********** sound bell, reset ACIA, motor off **************************** BEEPOF FAE8 JSR &E7DC ;check if free to print message FAEB BEQ &FAF2 ;enable second processor and reset serial system FAED LDA #&07 ;beep FAEF JSR OSWRCH ; CASMOF FAF2 LDA #&80 ; FAF4 JSR &FBBD ;enable 2nd proc. if present and set up osfile block FAF7 LDX #&00 ; FAF9 JSR &FB95 ;switch on motor RELEAS FAFC PHP ;save flags on stack FAFD SEI ;prevent IRQ interrupts FAFE LDA &0282 ;get serial ULA control register setting FB01 STA &FE10 ;write to serial ULA control register setting FB04 LDA #&00 ;A=0 FB06 STA &EA ;store A RS423 timeout counter FB08 BEQ &FB0B ;jump FB0B MC6850 FB0A PHP ;save flags on stacksave flags RELX FB0B JSR &FB46 ;release ACIA (by &FE08=3) FB0E LDA &0250 ;get last setting of ACIA FB11 JMP &E189 ;set ACIA and &250 from A before exit RSWT1 FB14 PLP ;get back flags FB15 BIT &FF ;if bit 7of ESCAPE flag not set FB17 BPL &FB31 ;then FB31 FB19 RTS ;else return as unserviced ESCAPE is pending ************************************************************************* * * * Claim serial system for sequential Access * * * ************************************************************************* SETSDF FB1A LDA &E3 ;get cassette filing system options byte ;high nybble used for LOAD & SAVE operations ;low nybble used for sequential access ;0000 Ignore errors, no messages ;0001 Abort if error, no messages ;0010 Retry after error, no messages ;1000 Ignore error short messages ;1001 Abort if error short messages ;1010 Retry after error short messages ;1100 Ignore error long messages ;1101 Abort if error long messages ;1110 Retry after error long messages FB1C ASL ;move low nybble into high nybble FB1D ASL ; FB1E ASL ; FB1F ASL ; FB20 STA &BB ;current OPTions save into &BB FB22 LDA &03D1 ;get sequential block gap FB25 BNE &FB2F ;goto to &FB2F ************************************************************************* * * * claim serial system for cassette etc. * * * ************************************************************************* SETDEF FB27 LDA &E3 ;get cassette filing system options byte ;high nybble used for LOAD & SAVE operations ;low nybble used for sequential access ;0000 Ignore errors, no messages ;0001 Abort if error, no messages ;0010 Retry after error, no messages ;1000 Ignore error short messages ;1001 Abort if error short messages ;1010 Retry after error short messages ;1100 Ignore error long messages ;1101 Abort if error long messages ;1110 Retry after error long messages FB29 AND #&F0 ;clear low nybble FB2B STA &BB ;as current OPTions FB2D LDA #&06 ;set current interblock gap GAPOUT FB2F STA &C7 ;to 6 CLAIM FB31 CLI ;allow interrupts FB32 PHP ;save flags on stack FB33 SEI ;prevent interrupts FB34 BIT &024F ;check if RS423 is busy FB37 BPL &FB14 ;if not FB14 FB39 LDA &EA ;see if RS423 has timed out FB3B BMI &FB14 ;if not FB14 FB3D LDA #&01 ;else load RS423 timeout counter with FB3F STA &EA ;1 to indicate that cassette has 6850 FB41 JSR &FB46 ;reset ACIA with &FE80=3 FB44 PLP ;get back flags FB45 RTS ;return CRESET FB46 LDA #&03 ;A=3 FB48 BNE &FB65 ;and exit after resetting ACIA ********************** set ACIA control register ********************** TXINIT FB4A LDA #&30 ;set current ACIA control register FB4C STA &CA ;to &30 FB4E BNE &FB63 ;and goto FB63 ;if bit 7=0 motor off 1=motor on ***************** control cassette system ******************************* RXINIT FB50 LDA #&05 ;set &FE10 to 5 FB52 STA &FE10 ;setting a transmit baud rate of 300,motor off FB55 LDX #&FF ; RXINL1 FB57 DEX ;delay loop FB58 BNE &FB57 ; FB5A STX &CA ;&CA=0 FB5C LDA #&85 ;Turn motor on and keep baud rate at 300 recieve FB5E STA &FE10 ;19200 transmit FB61 LDA #&D0 ;A=&D0 CHPINT FB63 ORA &C6 ; SETCHP FB65 STA &FE08 ;set up ACIA control register FB68 RTS ;returnand return NXTBLK FB69 LDX &03C6 ;block number FB6C LDY &03C7 ;block number hi FB6F INX ;X=X+1 FB70 STX &B4 ;current block no. lo FB72 BNE &FB75 ; FB74 INY ;Y=Y+1 NXTBLX FB75 STY &B5 ;current block no. hi FB77 RTS ;return ; SETJOB FB78 LDY #&00 ; FB7A STY &C0 ;filing system buffer flag *****************set (zero) checksum bytes ****************************** SETJB1 FB7C STY &BE ;CRC workspace FB7E STY &BF ;CRC workspace FB80 RTS ;return *********** copy sought filename routine ******************************** MOVFN FB81 LDY #&FF ;Y=&FF MOVFNL FB83 INY ;Y=Y+1 FB84 INX ;X=X+1 FB85 LDA &0300,X ; FB88 STA &03D2,Y ;sought filename FB8B BNE &FB83 ;until end of filename (0) FB8D RTS ;return ; MOTONW FB8E LDY #&00 ;Y=0 ********************** switch Motor on ********************************** MOTON FB90 CLI ;allow IRQ interrupts FB91 LDX #&01 ;X=1 FB93 STY &C3 ;store Y as current file handle ********************: control motor ************************************ CMOTOR FB95 LDA #&89 ;do osbyte 137 FB97 LDY &C3 ;get back file handle (preserved thru osbyte) FB99 JMP OSBYTE ;turn on motor ****************** confirm file is open ******************************** CHKHND FB9C STA &BC ;file status or temporary store FB9E TYA ;A=Y FB9F EOR &0247 ;filing system flag 0=CFS 2=RFS FBA2 TAY ;Y=A FBA3 LDA &E2 ;CFS status byte FBA5 AND &BC ;file status or temporary store FBA7 LSR ;A=A/2 FBA8 DEY ;Y=Y-1 FBA9 BEQ &FBAF ; FBAB LSR ;A=A/2 FBAC DEY ;Y=Y-1 FBAD BNE &FBB1 ; CHKHN1 FBAF BCS &FBFE ; HNDERR FBB1 BRK ; FBB2 DB &DE ;error number FBB3 DB 'Channel' ; FBBA BRK ; ************* read from second processor ******************************** TWINIT FBBB LDA #&01 ;A=1 TBINIT FBBD JSR &FBD3 ;check if second processor file test tube prescence FBC0 BEQ &FBFE ;if not exit FBC2 TXA ;A=X FBC3 LDX #&B0 ;current load address FBC5 LDY #&00 ;Y=00 TBINT1 FBC7 PHA ;save A on stack FBC8 LDA #&C0 ;filing system buffer flag TUBELP FBCA JSR &0406 ;and out to TUBE FBCD BCC &FBCA ; FBCF PLA ;get back A FBD0 JMP &0406 ; *************** check if second processor file test tube prescence ****** TUBCHK FBD3 TAX ;X=A FBD4 LDA &B2 ;current load address high word FBD6 AND &B3 ;current load address high word FBD8 CMP #&FF ; FBDA BEQ &FBE1 ;if &FF then its for base processor FBDC LDA &027A ;&FF if tube present FBDF AND #&80 ;to set bit 7 alone TUBCHX FBE1 RTS ;return ******** control ACIA and Motor ***************************************** BUGFIX FBE2 LDA #&85 ;A=&85 FBE4 STA &FE10 ;write to serial ULA control register setting FBE7 JSR &FB46 ;reset ACIA FBEA LDA #&10 ;A=16 FBEC JSR &FB63 ;set ACIA to CFS baud rate BUGLP FBEF JSR &F995 ;confirm ESC not set and CFS not executing FBF2 LDA &FE08 ;read ACIA status register FBF5 AND #&02 ;clear all but bit 1 FBF7 BEQ &FBEF ;if clear FBEF FBF9 LDA #&AA ;else A=&AA FBFB STA &FE09 ;transmit data register CHKHXX TINIT0 FBFE RTS ;return END * FBFF BRK ; ************** FRED 1MHz Bus memory-mapped I/O ************************** FRED FC00 ;test hardware FC10-13 ;teletext FC14-1F ;Prestel FC20-27 ;IEEE interface FC30 ; FC40-47 ;winchester disc interface FC50 ; FC60 ; FC70 ; FC80 ; FC90 ; FCA0 ; FCB0 ; FCC0 ; FCD0 ; FCE0 ; FCF0 ; MEG1P * FCFF ;paging register for JIM expansion memory ************** JIM 1MHz Bus memory-expansion page *********************** JIM FD00-FF ; TSTSFT FDFE ;Ecosoak Vector ************** SHEILA MOS memory-mapped I/O *************************** ;DEVICE WRITE READ SHEIL CRTCA FE00 ;6845 CRTC address register CRTCD FE01 ;6845 CRTC data register FE02 ;Border colour border colour FE03 ; FE04 ; FE05 ; FE06 ; FE07 ; RSTAT STATUS ACIA RSCTL FE08 ;6850 ACIA control register status register PORT RSDATA FE09 ;6850 ACIA transmit data recieve data FE0A ; FE0B ; FE0C ; FE0D ; FE0E ; FE0F ; SERPRC FE10 ;SERIAL ULA control register FE11 ; FE12 ; FE13 ; FE14 ; FE15 ; FE16 ; FE17 ; FE18 ;68B54 ADLC Disable interrupts Econet station ID FE19 ; FE1A ; FE1B ; FE1C ; FE1D ; FE1E ; FE1F ; STNID * VPROC0 FE20 ;Video ULA control register VPROC1 FE21 ;Video ULA palette register palette register FE22 ; FE23 ; FE24 ; FE25 ; FE26 ; FE27 ; FE28 ; FE29 ; FE2A ; FE2B ; FE2C ; FE2D ; FE2E ; FE2F ; ROM FE30 ;ROM latch paged ROM ID write only FE31 ;ALTAIR RAM protect FE32 ; FE33 ; FE34 ;Shadow RAM B+ only note different OS FE35 ; FE36 ; FE37 ; FE38 ; FE39 ; FE3A ; FE3B ; FE3C ; FE3D ; FE3E ; FE3F ; PBQ VIAA FE40 ;MOS 6522 VIA Output Register B Input Register B PAQ * FE41 ;MOS 6522 VIA Output Register A Input Register A DDRBQ FE42 ;MOS 6522 VIA data direction register B DDRAQ FE43 ;MOS 6522 VIA data direction register A TQCLQ * FE44 ;MOS 6522 VIA T1C-L latches T1 low Order counter TQCHQ FE45 ;MOS 6522 VIA T1C-H counter TQLLQ FE46 ;MOS 6522 VIA T1L-L low order latches TQLHQ FE47 ;MOS 6522 VIA T1L-H high order latches TRCLQ FE48 ;MOS 6522 VIA T2C-L latches T2C-L lo order counter TRCHQ FE49 ;MOS 6522 VIA T2C-H T2 high order counter SRQ * FE4A ;MOS 6522 VIA shift register ACRQ FE4B ;MOS 6522 VIA auxilliary control register ACR PCRQ FE4C ;MOS 6522 VIA Peripheral control register PCR IFRQ FE4D ;MOS 6522 VIA Interrupt flag register IFR IERQ FE4E ;MOS 6522 VIA Interrupt enable register IER PAPQ FE4F ;MOS 6522 VIA ORB/IRB but no handshake FE50 ; FE51 ; FE52 ; FE53 ; FE54 ; FE55 ; FE56 ; FE57 ; FE58 ; FE59 ; FE5A ; FE5B ; FE5C ; FE5D ; FE5E ; FE5F ; PBR * VIAB FE60 ;USER 6522 VIA Output Register B Input Register B PAR FE61 ;USER 6522 VIA Output Register A Input Register A DDRBR * FE62 ;USER 6522 VIA data direction register B DDRAR FE63 ;USER 6522 VIA data direction register A TQCLR * FE64 ;USER 6522 VIA T1C-L latches T1 low Order counter TQCHR * FE65 ;USER 6522 VIA T1C-H counter TQLLR * FE66 ;USER 6522 VIA T1L-L low order latches TQLHR * FE67 ;USER 6522 VIA T1L-H high order latches TRCLR * FE68 ;USER 6522 VIA T2C-L latches T2C-L lo order counter TRCHR * FE69 ;USER 6522 VIA T2C-H T2 high order counter SRR * FE6A ;USER 6522 VIA shift register ACRR * FE6B ;USER 6522 VIA auxilliary control register ACR PCRR FE6C ;USER 6522 VIA Peripheral control register PCR IFRR FE6D ;USER 6522 VIA Interrupt flag register IFR IERR FE6E ;USER 6522 VIA Interrupt enable register IER PAPR * FE6F ;USER 6522 VIA ORB/IRB but no handshake FE70 ; FE71 ; FE72 ; FE73 ; FE74 ; FE75 ; FE76 ; FE77 ; FE78 ; FE79 ; FE7A ; FE7B ; FE7C ; FE7D ; FE7E ; FE7F ; FE80 ;8271 FDC command register status register FE81 ;8271 FDC parameter register result register FE82 ;8271 FDC reset register FE83 ;8271 FDC illegal illegal FE84 ;8271 FDC data data FE85 ; FE86 ; FE87 ; FE88 ; FE89 ; FE8A ; FE8B ; FE8C ; FE8D ; FE8E ; FE8F ; FE90 ; FE91 ; FE92 ; FE93 ; FE94 ; FE95 ; FE96 ; FE97 ; FE98 ; FE99 ; FE9A ; FE9B ; FE9C ; FE9D ; FE9E ; FE9F ; FEA0 ;68B54 ADLC control register 1 status register 1 FEA1 ;68B54 ADLC control register 2/3 status register 2/3 FEA2 ;68B54 ADLC Tx FIFO (frame continue) Rx FIFO FEA3 ;68B54 ADLC Tx FIFO (frame terminate) Rx FIFO FEA4 ; FEA5 ; FEA6 ; FEA7 ; FEA8 ; FEA9 ; FEAA ; FEAB ; FEAC ; FEAD ; FEAE ; FEAF ; FEB0 ; FEB1 ; FEB2 ; FEB3 ; FEB4 ; FEB5 ; FEB6 ; FEB7 ; FEB8 ; FEB9 ; FEBA ; FEBB ; FEBC ; FEBD ; FEBE ; FEBF ; ADCCTL ADCONV FEC0 ;7002 ADC data latch A/D start status ADCHI FEC1 ;7002 ADC hi data byte ADCLO FEC2 ;7002 ADC lo data byte FEC3 ; FEC4 ; FEC5 ; FEC6 ; FEC7 ; FEC8 ; FEC9 ; FECA ; FECB ; FECC ; FECD ; FECE ; FECF ; FED0 ; FED1 ; FED2 ; FED3 ; FED4 ; FED5 ; FED6 ; FED7 ; FED8 ; FED9 ; FEDA ; FEDB ; FEDC ; FEDD ; FEDE ; FEDF ; TREG0 FEE0 ;TUBE FIFO1 status register FEE1 ;TUBE FIFO1 FEE2 ;TUBE FIFO2 status register FEE3 ;TUBE FIFO2 FEE4 ;TUBE FIFO3 status register TREG3 FEE5 ;TUBE FIFO3 FEE6 ;TUBE FIFO4 status register FEE7 ;TUBE FIFO4 FEE8 ; FEE9 ; FEEA ; FEEB ; FEEC ; FEED ; FEEE ; FEEF ; FEF0 ; FEF1 ; FEF2 ; FEF3 ; FEF4 ; FEF5 ; FEF6 ; FEF7 ; FEF8 ; FEF9 ; FEFA ; FEFB ; FEFC ; FEFD ; FEFE ; FEFF ; ********** EXTENDED VECTOR ENTRY POINTS********************************** ;vectors are pointed to &F000 +vector No. vectors may then be directed thru ;a three byte vector table whose XY address is given by osbyte A8, X=0, Y=&FF ;this is set up as lo-hi byte in ROM and ROM number ROMUSR* FF00 JSR &FF51 ;XUSERV ROMBRK* FF03 JSR &FF51 ;XBRKV ROMIRA* FF06 JSR &FF51 ;XIRQ1V ROMIRB* FF09 JSR &FF51 ;XIRQ2V ROMCOM* FF0C JSR &FF51 ;XCLIV ROMBYT* FF0F JSR &FF51 ;XBYTEV ROMWRD* FF12 JSR &FF51 ;XWORDV ROMWRC* FF15 JSR &FF51 ;XWRCHV ROMRDC* FF18 JSR &FF51 ;XRDCHV ROMFIL* FF1B JSR &FF51 ;XFILEV ROMARG* FF1E JSR &FF51 ;XARGSV ROMBGT* FF21 JSR &FF51 ;XBGETV ROMBPT* FF24 JSR &FF51 ;XBPUTV ROMGPB* FF27 JSR &FF51 ;XGBPBV ROMFND* FF2A JSR &FF51 ;XFINDV ROMFSC* FF2D JSR &FF51 ;XFSCV ROMEVT* FF30 JSR &FF51 ;XEVENTV ROMUPT* FF33 JSR &FF51 ;XUPTV ROMNET* FF36 JSR &FF51 ;XNETV ROMVDU* FF39 JSR &FF51 ;XVDUV ROMKEY* FF3C JSR &FF51 ;XKEYV ROMINS* FF3F JSR &FF51 ;XINSV ROMREM* FF42 JSR &FF51 ;XREMV ROMCNP* FF45 JSR &FF51 ;XCNPV FF48 JSR &FF51 ;XIND1V FF4B JSR &FF51 ;XIND2V FF4E JSR &FF51 ;XIND3V ;at this point the stack will hold 4 bytes (at least) ;S 0,1 extended vector address ;S 2,3 address of calling routine ;A,X,Y,P will be as at entry OSROM FF51 PHA ;save A on stack FF52 PHA ;save A on stack FF53 PHA ;save A on stack FF54 PHA ;save A on stack FF55 PHA ;save A on stack FF56 PHP ;save flags on stack FF57 PHA ;save A on stack FF58 TXA ;A=X FF59 PHA ;save X on stack FF5A TYA ;A=Y FF5B PHA ;save Y on stack FF5C TSX ;get stack pointer into X (&F2 or less) FF5D LDA #&FF ;A=&FF FF5F STA &0108,X ;A FF62 LDA #&88 ; FF64 STA &0107,X ; FF67 LDY &010A,X ;this is VECTOR number*3+2!! FF6A LDA &0D9D,Y ;lo byte of action address FF6D STA &0105,X ;store it on stack FF70 LDA &0D9E,Y ;get hi byte FF73 STA &0106,X ;store it on stack ;at this point stack has YXAP and action address ;followed by return address and 5 more bytes FF76 LDA &F4 ; FF78 STA &0109,X ;store original ROM number below this FF7B LDA &0D9F,Y ;get new ROM number FF7E STA &F4 ;store it as ram copy FF80 STA &FE30 ;and switch to that ROM FF83 PLA ;get back A FF84 TAY ;Y=A FF85 PLA ;get back A FF86 TAX ;X=A FF87 PLA ;get back A FF88 RTI ;get back flags and jump to ROM vectored entry ;leaving return address and 5 more bytes on stack ************ return address from ROM indirection ************************ ;at this point stack comprises original ROM number,return from JSR &FF51, ;return from original call the return from FF51 is garbage so; OSROM2 FF89 PHP ;save flags on stack FF8A PHA ;save A on stack FF8B TXA ;A=X FF8C PHA ;save X on stack FF8D TSX ; (&F7 or less) FF8E LDA &0102,X ;STORE A AND P OVER FF91 STA &0105,X ;return address from (JSR &FF51) FF94 LDA &0103,X ;hiding garbage by duplicating A and X just saved FF97 STA &0106,X ; ;now we have ;flags, ;A, ;X, ;ROM number, ;A, ;flags, ;and original return address on stack ;so FF9A PLA ;get back X FF9B TAX ;X=A FF9C PLA ;get back A lose next two bytes FF9D PLA ;get back A lose FF9E PLA ;get back A ROM number FF9F STA &F4 ;store it FFA1 STA &FE30 ;and set it FFA4 PLA ;get back A FFA5 PLP ;get back flags DEFNET DEFUPT DEFEVT RTS FFA6 RTS ;return and exit pulling original return address ;from stack ;FFA6 is also default input for CFS OSBPGB, VDUV, IND1V,IND2V,IND3V ;as these functions are not implemented by the OS but may be used ;by software or other filing systems or ROMs ************************************************************************* * * * OSBYTE &9D FAST BPUT * * * ************************************************************************* FXBPUT FFA7 TXA ;A=X FFA8 BCS &FFD4 ;carry always set, jump to BPUT ************************************************************************* * * * OSBYTE &92 READ A BYTE FROM FRED * * * ************************************************************************* ; RFRED FFAA LDY &FC00,X ;read a byte from FRED area FFAD RTS ;return ************************************************************************* * * * OSBYTE &94 READ A BYTE FROM JIM * * * ************************************************************************* ; ; RJIM FFAE LDY &FD00,X ;read a byte from JIM area FFB1 RTS ;return ************************************************************************* * * * OSBYTE &96 READ A BYTE FROM SHEILA * * * ************************************************************************* ; ; RSHEIL FFB2 LDY &FE00,X ;read a byte from SHEILA memory mapped I/O area FFB5 RTS ;return *********** DEFAULT VECTOR TABLE **************************************** FFB6 DB 36 ;length of look up table in bytes FFB7 DB 40 ;low byte of address of this table FFB8 DB D9 ;high byte of address of this table ************************************************************************** ************************************************************************** ** ** ** OPERATING SYSTEM FUNCTION CALLS ** ** ** ************************************************************************** ************************************************************************** FFB9 JMP &DC0B ;OSRDRM get a byte from sideways ROM FFBC JMP &C4C0 ;VDUCHR VDU character output FFBF JMP &E494 ;OSEVEN generate an EVENT FFC2 JMP &EA1E ;GSINIT initialise OS string FFC5 JMP &EA2F ;GSREAD read character from input stream FFC8 JMP &DEC5 ;NVRDCH non vectored OSRDCH FFCB JMP &E0A4 ;NVWRCH non vectored OSWRCH OSFIND FFCE JMP (&021C) ;OSFIND open or close a file OSBGPB* FFD1 JMP (&021A) ;OSGBPB transfer block to or from a file OSBPUT FFD4 JMP (&0218) ;OSBPUT save a byte to file OSBGET FFD7 JMP (&0216) ;OSBGET get a byte from file OSARGS* FFDA JMP (&0214) ;OSARGS read or write file arguments OSFILE FFDD JMP (&0212) ;OSFILE read or write a file OSRDCH FFE0 JMP (&0210) ;OSRDCH get a byte from current input stream OSASCI FFE3 CMP #&0D ;OSASCI output a byte to VDU stream expanding FFE5 BNE &FFEE ; carriage returns (&0D) to LF/CR (&0A,&0D) OSNEWL FFE7 LDA #&0A ;OSNEWL output a CR/LF to VDU stream FFE9 JSR OSWRCH ;Outputs A followed by CR to VDU stream FFEC LDA #&0D ;OSWRCR output a CR to VDU stream OSWRCH FFEE JMP (&020E) ;OSWRCH output a character to the VDU stream OSWORD* FFF1 JMP (&020C) ;OSWORD perform operation using parameter table OSBYTE FFF4 JMP (&020A) ;OSBYTE perform operation with single bytes OSCLI FFF7 JMP (&0208) ;OSCLI pass string to command line interpreter ************************************************************************* * * * 6502 Vectors * * * ************************************************************************* FFFA DW &0D00 ;NMI address FFFC DW &D9CD ;RESET address FFFE DW &DC1C ;IRQ address That's it the end of the series and the end of Micronet. See you on the new system or in the paper mags. Geoff KTBL F02B POUNDM* F053 SENDS F077 SENDF F079