l.s 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299
  1. #include "mem.h"
  2. #include "/sys/src/boot/pc/x16.h"
  3. #undef DELAY
  4. #define PADDR(a) ((a) & ~KZERO)
  5. #define KADDR(a) (KZERO|(a))
  6. /*
  7. * Some machine instructions not handled by 8[al].
  8. */
  9. #define OP16 BYTE $0x66
  10. #define DELAY BYTE $0xEB; BYTE $0x00 /* JMP .+2 */
  11. #define CPUID BYTE $0x0F; BYTE $0xA2 /* CPUID, argument in AX */
  12. #define WRMSR BYTE $0x0F; BYTE $0x30 /* WRMSR, argument in AX/DX (lo/hi) */
  13. #define RDTSC BYTE $0x0F; BYTE $0x31 /* RDTSC, result in AX/DX (lo/hi) */
  14. #define RDMSR BYTE $0x0F; BYTE $0x32 /* RDMSR, result in AX/DX (lo/hi) */
  15. #define HLT BYTE $0xF4
  16. #define INVLPG BYTE $0x0F; BYTE $0x01; BYTE $0x39 /* INVLPG (%ecx) */
  17. #define WBINVD BYTE $0x0F; BYTE $0x09
  18. /*
  19. * Macros for calculating offsets within the page directory base
  20. * and page tables. Note that these are assembler-specific hence
  21. * the '<<2'.
  22. */
  23. #define PDO(a) (((((a))>>22) & 0x03FF)<<2)
  24. #define PTO(a) (((((a))>>12) & 0x03FF)<<2)
  25. /*
  26. * For backwards compatiblity with 9load - should go away when 9load is changed
  27. * 9load currently sets up the mmu, however the first 16MB of memory is identity
  28. * mapped, so behave as if the mmu was not setup
  29. */
  30. TEXT _startKADDR(SB), $0
  31. MOVL $_startPADDR(SB), AX
  32. ANDL $~KZERO, AX
  33. JMP* AX
  34. /*
  35. * Must be 4-byte aligned.
  36. */
  37. TEXT _multibootheader(SB), $0
  38. LONG $0x1BADB002 /* magic */
  39. LONG $0x00010003 /* flags */
  40. LONG $-(0x1BADB002 + 0x00010003) /* checksum */
  41. LONG $_multibootheader-KZERO(SB) /* header_addr */
  42. LONG $_startKADDR-KZERO(SB) /* load_addr */
  43. LONG $edata-KZERO(SB) /* load_end_addr */
  44. LONG $end-KZERO(SB) /* bss_end_addr */
  45. LONG $_startKADDR-KZERO(SB) /* entry_addr */
  46. LONG $0 /* mode_type */
  47. LONG $0 /* width */
  48. LONG $0 /* height */
  49. LONG $0 /* depth */
  50. /*
  51. * In protected mode with paging turned off and segment registers setup
  52. * to linear map all memory. Entered via a jump to PADDR(entry),
  53. * the physical address of the virtual kernel entry point of KADDR(entry).
  54. * Make the basic page tables for processor 0. Six pages are needed for
  55. * the basic set:
  56. * a page directory;
  57. * page tables for mapping the first 8MB of physical memory to KZERO;
  58. * a page for the GDT;
  59. * virtual and physical pages for mapping the Mach structure.
  60. * The remaining PTEs will be allocated later when memory is sized.
  61. * An identity mmu map is also needed for the switch to virtual mode.
  62. * This identity mapping is removed once the MMU is going and the JMP has
  63. * been made to virtual memory.
  64. */
  65. TEXT _startPADDR(SB), $0
  66. CLI /* make sure interrupts are off */
  67. /* set up the gdt so we have sane plan 9 style gdts. */
  68. MOVL $tgdtptr(SB), AX
  69. ANDL $~KZERO, AX
  70. MOVL (AX), GDTR
  71. MOVW $1, AX
  72. MOVW AX, MSW
  73. /* clear prefetch queue (weird code to avoid optimizations) */
  74. DELAY
  75. /* set segs to something sane (avoid traps later) */
  76. MOVW $(1<<3), AX
  77. MOVW AX, DS
  78. MOVW AX, SS
  79. MOVW AX, ES
  80. MOVW AX, FS
  81. MOVW AX, GS
  82. /* JMP $(2<<3):$mode32bit(SB) /**/
  83. BYTE $0xEA
  84. LONG $mode32bit-KZERO(SB)
  85. WORD $(2<<3)
  86. /*
  87. * gdt to get us to 32-bit/segmented/unpaged mode
  88. */
  89. TEXT tgdt(SB), $0
  90. /* null descriptor */
  91. LONG $0
  92. LONG $0
  93. /* data segment descriptor for 4 gigabytes (PL 0) */
  94. LONG $(0xFFFF)
  95. LONG $(SEGG|SEGB|(0xF<<16)|SEGP|SEGPL(0)|SEGDATA|SEGW)
  96. /* exec segment descriptor for 4 gigabytes (PL 0) */
  97. LONG $(0xFFFF)
  98. LONG $(SEGG|SEGD|(0xF<<16)|SEGP|SEGPL(0)|SEGEXEC|SEGR)
  99. /*
  100. * pointer to initial gdt
  101. * Note the -KZERO which puts the physical address in the gdtptr.
  102. * that's needed as we start executing in physical addresses.
  103. */
  104. TEXT tgdtptr(SB), $0
  105. WORD $(3*8)
  106. LONG $tgdt-KZERO(SB)
  107. TEXT m0rgdtptr(SB), $0
  108. WORD $(NGDT*8-1)
  109. LONG $(CPU0GDT-KZERO)
  110. TEXT m0gdtptr(SB), $0
  111. WORD $(NGDT*8-1)
  112. LONG $CPU0GDT
  113. TEXT m0idtptr(SB), $0
  114. WORD $(256*8-1)
  115. LONG $IDTADDR
  116. TEXT mode32bit(SB), $0
  117. /* At this point, the GDT setup is done. */
  118. MOVL $PADDR(CPU0PDB), DI /* clear 4 pages for the tables etc. */
  119. XORL AX, AX
  120. MOVL $(4*BY2PG), CX
  121. SHRL $2, CX
  122. CLD
  123. REP; STOSL
  124. MOVL $PADDR(CPU0PDB), AX
  125. ADDL $PDO(KZERO), AX /* page directory offset for KZERO */
  126. MOVL $PADDR(CPU0PTE), (AX) /* PTE's for KZERO */
  127. MOVL $(PTEWRITE|PTEVALID), BX /* page permissions */
  128. ORL BX, (AX)
  129. ADDL $4, AX
  130. MOVL $PADDR(CPU0PTE1), (AX) /* PTE's for KZERO+4MB */
  131. MOVL $(PTEWRITE|PTEVALID), BX /* page permissions */
  132. ORL BX, (AX)
  133. MOVL $PADDR(CPU0PTE), AX /* first page of page table */
  134. MOVL $1024, CX /* 1024 pages in 4MB */
  135. _setpte:
  136. MOVL BX, (AX)
  137. ADDL $(1<<PGSHIFT), BX
  138. ADDL $4, AX
  139. LOOP _setpte
  140. MOVL $PADDR(CPU0PTE1), AX /* second page of page table */
  141. MOVL $1024, CX /* 1024 pages in 4MB */
  142. _setpte1:
  143. MOVL BX, (AX)
  144. ADDL $(1<<PGSHIFT), BX
  145. ADDL $4, AX
  146. LOOP _setpte1
  147. MOVL $PADDR(CPU0PTE), AX
  148. ADDL $PTO(MACHADDR), AX /* page table entry offset for MACHADDR */
  149. MOVL $PADDR(CPU0MACH), (AX) /* PTE for Mach */
  150. MOVL $(PTEWRITE|PTEVALID), BX /* page permissions */
  151. ORL BX, (AX)
  152. /*
  153. * Now ready to use the new map. Make sure the processor options are what is wanted.
  154. * It is necessary on some processors to immediately follow mode switching with a JMP instruction
  155. * to clear the prefetch queues.
  156. */
  157. MOVL $PADDR(CPU0PDB), CX /* load address of page directory */
  158. MOVL (PDO(KZERO))(CX), DX /* double-map KZERO at 0 */
  159. MOVL DX, (PDO(0))(CX)
  160. MOVL CX, CR3
  161. DELAY /* JMP .+2 */
  162. MOVL CR0, DX
  163. ORL $0x80010000, DX /* PG|WP */
  164. ANDL $~0x6000000A, DX /* ~(CD|NW|TS|MP) */
  165. MOVL $_startpg(SB), AX /* this is a virtual address */
  166. MOVL DX, CR0 /* turn on paging */
  167. JMP* AX /* jump to the virtual nirvana */
  168. /*
  169. * Basic machine environment set, can clear BSS and create a stack.
  170. * The stack starts at the top of the page containing the Mach structure.
  171. * The x86 architecture forces the use of the same virtual address for
  172. * each processor's Mach structure, so the global Mach pointer 'm' can
  173. * be initialised here.
  174. */
  175. TEXT _startpg(SB), $0
  176. MOVL $0, (PDO(0))(CX) /* undo double-map of KZERO at 0 */
  177. MOVL CX, CR3 /* load and flush the mmu */
  178. _clearbss:
  179. MOVL $edata(SB), DI
  180. XORL AX, AX
  181. MOVL $end(SB), CX
  182. SUBL DI, CX /* end-edata bytes */
  183. SHRL $2, CX /* end-edata doublewords */
  184. CLD
  185. REP; STOSL /* clear BSS */
  186. MOVL $MACHADDR, SP
  187. MOVL SP, m(SB) /* initialise global Mach pointer */
  188. MOVL $0, 0(SP) /* initialise m->machno */
  189. ADDL $(MACHSIZE-4), SP /* initialise stack */
  190. /*
  191. * Need to do one final thing to ensure a clean machine environment,
  192. * clear the EFLAGS register, which can only be done once there is a stack.
  193. */
  194. MOVL $0, AX
  195. PUSHL AX
  196. POPFL
  197. CALL main(SB)
  198. /*
  199. * Park a processor. Should never fall through a return from main to here,
  200. * should only be called by application processors when shutting down.
  201. */
  202. TEXT idle(SB), $0
  203. _idle:
  204. STI
  205. HLT
  206. JMP _idle
  207. /*
  208. * Save registers.
  209. */
  210. TEXT saveregs(SB), $0
  211. /* appease 8l */
  212. SUBL $32, SP
  213. POPL AX
  214. POPL AX
  215. POPL AX
  216. POPL AX
  217. POPL AX
  218. POPL AX
  219. POPL AX
  220. POPL AX
  221. PUSHL AX
  222. PUSHL BX
  223. PUSHL CX
  224. PUSHL DX
  225. PUSHL BP
  226. PUSHL DI
  227. PUSHL SI
  228. PUSHFL
  229. XCHGL 32(SP), AX /* swap return PC and saved flags */
  230. XCHGL 0(SP), AX
  231. XCHGL 32(SP), AX
  232. RET
  233. TEXT restoreregs(SB), $0
  234. /* appease 8l */
  235. PUSHL AX
  236. PUSHL AX
  237. PUSHL AX
  238. PUSHL AX
  239. PUSHL AX
  240. PUSHL AX
  241. PUSHL AX
  242. PUSHL AX
  243. ADDL $32, SP
  244. XCHGL 32(SP), AX /* swap return PC and saved flags */
  245. XCHGL 0(SP), AX
  246. XCHGL 32(SP), AX
  247. POPFL
  248. POPL SI
  249. POPL DI
  250. POPL BP
  251. POPL DX
  252. POPL CX
  253. POPL BX
  254. POPL AX
  255. RET
  256. /*
  257. * Assumed to be in protected mode at time of call.
  258. * Switch to real mode, execute an interrupt, and
  259. * then switch back to protected mode.
  260. *
  261. * Assumes:
  262. *
  263. * - no device interrupts are going to come in
  264. * - 0-16MB is identity mapped in page tables
  265. * - realmode() has copied us down from 0x100000 to 0x8000
  266. * - can use code segment 0x0800 in real mode
  267. * to get at l.s code
  268. * - l.s code is less than 1 page
  269. */
  270. #define RELOC (RMCODE-KTZERO)
  271. TEXT realmodeidtptr(SB), $0
  272. WORD $(4*256-1)
  273. LONG $0
  274. TEXT realmode0(SB), $0
  275. CALL saveregs(SB)
  276. /* switch to low code address */
  277. LEAL physcode-KZERO(SB), AX
  278. JMP *AX
  279. TEXT physcode(SB), $0
  280. /* switch to low stack */
  281. MOVL SP, AX
  282. MOVL $0x7C00, SP
  283. PUSHL AX
  284. /* change gdt to physical pointer */
  285. MOVL m0rgdtptr-KZERO(SB), GDTR
  286. /* load IDT with real-mode version*/
  287. MOVL realmodeidtptr-KZERO(SB), IDTR
  288. /* edit INT $0x00 instruction below */
  289. MOVL $(RMUADDR-KZERO+48), AX /* &rmu.trap */
  290. MOVL (AX), AX
  291. MOVB AX, realmodeintrinst+(-KZERO+1+RELOC)(SB)
  292. /* disable paging */
  293. MOVL CR0, AX
  294. ANDL $0x7FFFFFFF, AX
  295. MOVL AX, CR0
  296. /* JMP .+2 to clear prefetch queue*/
  297. BYTE $0xEB; BYTE $0x00
  298. /* jump to 16-bit code segment */
  299. /* JMPFAR SELECTOR(KESEG16, SELGDT, 0):$again16bit(SB) /**/
  300. BYTE $0xEA
  301. LONG $again16bit-KZERO(SB)
  302. WORD $SELECTOR(KESEG16, SELGDT, 0)
  303. TEXT again16bit(SB), $0
  304. /*
  305. * Now in 16-bit compatibility mode.
  306. * These are 32-bit instructions being interpreted
  307. * as 16-bit instructions. I'm being lazy and
  308. * not using the macros because I know when
  309. * the 16- and 32-bit instructions look the same
  310. * or close enough.
  311. */
  312. /* disable protected mode and jump to real mode cs */
  313. OPSIZE; MOVL CR0, AX
  314. OPSIZE; XORL BX, BX
  315. OPSIZE; INCL BX
  316. OPSIZE; XORL BX, AX
  317. OPSIZE; MOVL AX, CR0
  318. /* JMPFAR 0x0800:now16real */
  319. BYTE $0xEA
  320. WORD $now16real-KZERO(SB)
  321. WORD $0x0800
  322. TEXT now16real(SB), $0
  323. /* copy the registers for the bios call */
  324. LWI(0x0000, rAX)
  325. MOVW AX,SS
  326. LWI(RMUADDR, rBP)
  327. /* offsets are in Ureg */
  328. LXW(44, xBP, rAX)
  329. MOVW AX, DS
  330. LXW(40, xBP, rAX)
  331. MOVW AX, ES
  332. OPSIZE; LXW(0, xBP, rDI)
  333. OPSIZE; LXW(4, xBP, rSI)
  334. OPSIZE; LXW(16, xBP, rBX)
  335. OPSIZE; LXW(20, xBP, rDX)
  336. OPSIZE; LXW(24, xBP, rCX)
  337. OPSIZE; LXW(28, xBP, rAX)
  338. CLC
  339. TEXT realmodeintrinst(SB), $0
  340. INT $0x00
  341. /* save the registers after the call */
  342. LWI(0x7bfc, rSP)
  343. OPSIZE; PUSHFL
  344. OPSIZE; PUSHL AX
  345. LWI(0, rAX)
  346. MOVW AX,SS
  347. LWI(RMUADDR, rBP)
  348. OPSIZE; SXW(rDI, 0, xBP)
  349. OPSIZE; SXW(rSI, 4, xBP)
  350. OPSIZE; SXW(rBX, 16, xBP)
  351. OPSIZE; SXW(rDX, 20, xBP)
  352. OPSIZE; SXW(rCX, 24, xBP)
  353. OPSIZE; POPL AX
  354. OPSIZE; SXW(rAX, 28, xBP)
  355. MOVW DS, AX
  356. OPSIZE; SXW(rAX, 44, xBP)
  357. MOVW ES, AX
  358. OPSIZE; SXW(rAX, 40, xBP)
  359. OPSIZE; POPL AX
  360. OPSIZE; SXW(rAX, 64, xBP) /* flags */
  361. /* re-enter protected mode and jump to 32-bit code */
  362. OPSIZE; MOVL $1, AX
  363. OPSIZE; MOVL AX, CR0
  364. /* JMPFAR SELECTOR(KESEG, SELGDT, 0):$again32bit(SB) /**/
  365. OPSIZE
  366. BYTE $0xEA
  367. LONG $again32bit-KZERO(SB)
  368. WORD $SELECTOR(KESEG, SELGDT, 0)
  369. TEXT again32bit(SB), $0
  370. MOVW $SELECTOR(KDSEG, SELGDT, 0),AX
  371. MOVW AX,DS
  372. MOVW AX,SS
  373. MOVW AX,ES
  374. MOVW AX,FS
  375. MOVW AX,GS
  376. /* enable paging and jump to kzero-address code */
  377. MOVL CR0, AX
  378. ORL $0x80010000, AX /* PG|WP */
  379. MOVL AX, CR0
  380. LEAL again32kzero(SB), AX
  381. JMP* AX
  382. TEXT again32kzero(SB), $0
  383. /* breathe a sigh of relief - back in 32-bit protected mode */
  384. /* switch to old stack */
  385. PUSHL AX /* match popl below for 8l */
  386. MOVL $0x7BFC, SP
  387. POPL SP
  388. /* restore idt */
  389. MOVL m0idtptr(SB),IDTR
  390. /* restore gdt */
  391. MOVL m0gdtptr(SB), GDTR
  392. CALL restoreregs(SB)
  393. RET
  394. /*
  395. * BIOS32.
  396. */
  397. TEXT bios32call(SB), $0
  398. MOVL ci+0(FP), BP
  399. MOVL 0(BP), AX
  400. MOVL 4(BP), BX
  401. MOVL 8(BP), CX
  402. MOVL 12(BP), DX
  403. MOVL 16(BP), SI
  404. MOVL 20(BP), DI
  405. PUSHL BP
  406. MOVL 12(SP), BP /* ptr */
  407. BYTE $0xFF; BYTE $0x5D; BYTE $0x00 /* CALL FAR 0(BP) */
  408. POPL BP
  409. MOVL DI, 20(BP)
  410. MOVL SI, 16(BP)
  411. MOVL DX, 12(BP)
  412. MOVL CX, 8(BP)
  413. MOVL BX, 4(BP)
  414. MOVL AX, 0(BP)
  415. XORL AX, AX
  416. JCC _bios32xxret
  417. INCL AX
  418. _bios32xxret:
  419. RET
  420. /*
  421. * Port I/O.
  422. * in[bsl] input a byte|short|long
  423. * ins[bsl] input a string of bytes|shorts|longs
  424. * out[bsl] output a byte|short|long
  425. * outs[bsl] output a string of bytes|shorts|longs
  426. */
  427. TEXT inb(SB), $0
  428. MOVL port+0(FP), DX
  429. XORL AX, AX
  430. INB
  431. RET
  432. TEXT insb(SB), $0
  433. MOVL port+0(FP), DX
  434. MOVL address+4(FP), DI
  435. MOVL count+8(FP), CX
  436. CLD
  437. REP; INSB
  438. RET
  439. TEXT ins(SB), $0
  440. MOVL port+0(FP), DX
  441. XORL AX, AX
  442. OP16; INL
  443. RET
  444. TEXT inss(SB), $0
  445. MOVL port+0(FP), DX
  446. MOVL address+4(FP), DI
  447. MOVL count+8(FP), CX
  448. CLD
  449. REP; OP16; INSL
  450. RET
  451. TEXT inl(SB), $0
  452. MOVL port+0(FP), DX
  453. INL
  454. RET
  455. TEXT insl(SB), $0
  456. MOVL port+0(FP), DX
  457. MOVL address+4(FP), DI
  458. MOVL count+8(FP), CX
  459. CLD
  460. REP; INSL
  461. RET
  462. TEXT outb(SB), $0
  463. MOVL port+0(FP), DX
  464. MOVL byte+4(FP), AX
  465. OUTB
  466. RET
  467. TEXT outsb(SB), $0
  468. MOVL port+0(FP), DX
  469. MOVL address+4(FP), SI
  470. MOVL count+8(FP), CX
  471. CLD
  472. REP; OUTSB
  473. RET
  474. TEXT outs(SB), $0
  475. MOVL port+0(FP), DX
  476. MOVL short+4(FP), AX
  477. OP16; OUTL
  478. RET
  479. TEXT outss(SB), $0
  480. MOVL port+0(FP), DX
  481. MOVL address+4(FP), SI
  482. MOVL count+8(FP), CX
  483. CLD
  484. REP; OP16; OUTSL
  485. RET
  486. TEXT outl(SB), $0
  487. MOVL port+0(FP), DX
  488. MOVL long+4(FP), AX
  489. OUTL
  490. RET
  491. TEXT outsl(SB), $0
  492. MOVL port+0(FP), DX
  493. MOVL address+4(FP), SI
  494. MOVL count+8(FP), CX
  495. CLD
  496. REP; OUTSL
  497. RET
  498. /*
  499. * Read/write various system registers.
  500. * CR4 and the 'model specific registers' should only be read/written
  501. * after it has been determined the processor supports them
  502. */
  503. TEXT lgdt(SB), $0 /* GDTR - global descriptor table */
  504. MOVL gdtptr+0(FP), AX
  505. MOVL (AX), GDTR
  506. RET
  507. TEXT lidt(SB), $0 /* IDTR - interrupt descriptor table */
  508. MOVL idtptr+0(FP), AX
  509. MOVL (AX), IDTR
  510. RET
  511. TEXT ltr(SB), $0 /* TR - task register */
  512. MOVL tptr+0(FP), AX
  513. MOVW AX, TASK
  514. RET
  515. TEXT getcr0(SB), $0 /* CR0 - processor control */
  516. MOVL CR0, AX
  517. RET
  518. TEXT getcr2(SB), $0 /* CR2 - page fault linear address */
  519. MOVL CR2, AX
  520. RET
  521. TEXT getcr3(SB), $0 /* CR3 - page directory base */
  522. MOVL CR3, AX
  523. RET
  524. TEXT putcr0(SB), $0
  525. MOVL cr0+0(FP), AX
  526. MOVL AX, CR0
  527. RET
  528. TEXT putcr3(SB), $0
  529. MOVL cr3+0(FP), AX
  530. MOVL AX, CR3
  531. RET
  532. TEXT getcr4(SB), $0 /* CR4 - extensions */
  533. MOVL CR4, AX
  534. RET
  535. TEXT putcr4(SB), $0
  536. MOVL cr4+0(FP), AX
  537. MOVL AX, CR4
  538. RET
  539. TEXT invlpg(SB), $0
  540. /* 486+ only */
  541. MOVL va+0(FP), CX
  542. INVLPG
  543. RET
  544. TEXT wbinvd(SB), $0
  545. WBINVD
  546. RET
  547. TEXT _cycles(SB), $0 /* time stamp counter */
  548. RDTSC
  549. MOVL vlong+0(FP), CX /* &vlong */
  550. MOVL AX, 0(CX) /* lo */
  551. MOVL DX, 4(CX) /* hi */
  552. RET
  553. /*
  554. * stub for:
  555. * time stamp counter; low-order 32 bits of 64-bit cycle counter
  556. * Runs at fasthz/4 cycles per second (m->clkin>>3)
  557. */
  558. TEXT lcycles(SB),1,$0
  559. RDTSC
  560. RET
  561. TEXT rdmsr(SB), $0 /* model-specific register */
  562. MOVL index+0(FP), CX
  563. RDMSR
  564. MOVL vlong+4(FP), CX /* &vlong */
  565. MOVL AX, 0(CX) /* lo */
  566. MOVL DX, 4(CX) /* hi */
  567. RET
  568. TEXT wrmsr(SB), $0
  569. MOVL index+0(FP), CX
  570. MOVL lo+4(FP), AX
  571. MOVL hi+8(FP), DX
  572. WRMSR
  573. RET
  574. /*
  575. * Try to determine the CPU type which requires fiddling with EFLAGS.
  576. * If the Id bit can be toggled then the CPUID instruction can be used
  577. * to determine CPU identity and features. First have to check if it's
  578. * a 386 (Ac bit can't be set). If it's not a 386 and the Id bit can't be
  579. * toggled then it's an older 486 of some kind.
  580. *
  581. * cpuid(fun, regs[4]);
  582. */
  583. TEXT cpuid(SB), $0
  584. MOVL $0x240000, AX
  585. PUSHL AX
  586. POPFL /* set Id|Ac */
  587. PUSHFL
  588. POPL BX /* retrieve value */
  589. MOVL $0, AX
  590. PUSHL AX
  591. POPFL /* clear Id|Ac, EFLAGS initialised */
  592. PUSHFL
  593. POPL AX /* retrieve value */
  594. XORL BX, AX
  595. TESTL $0x040000, AX /* Ac */
  596. JZ _cpu386 /* can't set this bit on 386 */
  597. TESTL $0x200000, AX /* Id */
  598. JZ _cpu486 /* can't toggle this bit on some 486 */
  599. /* load registers */
  600. MOVL regs+4(FP), BP
  601. MOVL fn+0(FP), AX /* cpuid function */
  602. MOVL 4(BP), BX
  603. MOVL 8(BP), CX /* typically an index */
  604. MOVL 12(BP), DX
  605. CPUID
  606. JMP _cpuid
  607. _cpu486:
  608. MOVL $0x400, AX
  609. JMP _maybezapax
  610. _cpu386:
  611. MOVL $0x300, AX
  612. _maybezapax:
  613. CMPL fn+0(FP), $1
  614. JE _zaprest
  615. XORL AX, AX
  616. _zaprest:
  617. XORL BX, BX
  618. XORL CX, CX
  619. XORL DX, DX
  620. _cpuid:
  621. MOVL regs+4(FP), BP
  622. MOVL AX, 0(BP)
  623. MOVL BX, 4(BP)
  624. MOVL CX, 8(BP)
  625. MOVL DX, 12(BP)
  626. RET
  627. /*
  628. * Basic timing loop to determine CPU frequency.
  629. */
  630. TEXT aamloop(SB), $0
  631. MOVL count+0(FP), CX
  632. _aamloop:
  633. AAM
  634. LOOP _aamloop
  635. RET
  636. /*
  637. * Floating point.
  638. * Note: the encodings for the FCLEX, FINIT, FSAVE, FSTCW, FSENV and FSTSW
  639. * instructions do NOT have the WAIT prefix byte (i.e. they act like their
  640. * FNxxx variations) so WAIT instructions must be explicitly placed in the
  641. * code as necessary.
  642. */
  643. #define FPOFF(l) ;\
  644. MOVL CR0, AX ;\
  645. ANDL $0xC, AX /* EM, TS */ ;\
  646. CMPL AX, $0x8 ;\
  647. JEQ l ;\
  648. WAIT ;\
  649. l: ;\
  650. MOVL CR0, AX ;\
  651. ANDL $~0x4, AX /* EM=0 */ ;\
  652. ORL $0x28, AX /* NE=1, TS=1 */ ;\
  653. MOVL AX, CR0
  654. #define FPON ;\
  655. MOVL CR0, AX ;\
  656. ANDL $~0xC, AX /* EM=0, TS=0 */ ;\
  657. MOVL AX, CR0
  658. TEXT fpoff(SB), $0 /* disable */
  659. FPOFF(l1)
  660. RET
  661. TEXT fpinit(SB), $0 /* enable and init */
  662. FPON
  663. FINIT
  664. WAIT
  665. /* setfcr(FPPDBL|FPRNR|FPINVAL|FPZDIV|FPOVFL) */
  666. /* note that low 6 bits are masks, not enables, on this chip */
  667. PUSHW $0x0232
  668. FLDCW 0(SP)
  669. POPW AX
  670. WAIT
  671. RET
  672. TEXT fpsave(SB), $0 /* save state and disable */
  673. MOVL p+0(FP), AX
  674. FSAVE 0(AX) /* no WAIT */
  675. FPOFF(l2)
  676. RET
  677. TEXT fprestore(SB), $0 /* enable and restore state */
  678. FPON
  679. MOVL p+0(FP), AX
  680. FRSTOR 0(AX)
  681. WAIT
  682. RET
  683. TEXT fpstatus(SB), $0 /* get floating point status */
  684. FSTSW AX
  685. RET
  686. TEXT fpenv(SB), $0 /* save state without waiting */
  687. MOVL p+0(FP), AX
  688. FSTENV 0(AX)
  689. RET
  690. TEXT fpclear(SB), $0 /* clear pending exceptions */
  691. FPON
  692. FCLEX /* no WAIT */
  693. FPOFF(l3)
  694. RET
  695. /*
  696. */
  697. TEXT splhi(SB), $0
  698. shi:
  699. PUSHFL
  700. POPL AX
  701. TESTL $0x200, AX
  702. JZ alreadyhi
  703. MOVL $(MACHADDR+0x04), CX /* save PC in m->splpc */
  704. MOVL (SP), BX
  705. MOVL BX, (CX)
  706. alreadyhi:
  707. CLI
  708. RET
  709. TEXT spllo(SB), $0
  710. slo:
  711. PUSHFL
  712. POPL AX
  713. TESTL $0x200, AX
  714. JNZ alreadylo
  715. MOVL $(MACHADDR+0x04), CX /* clear m->splpc */
  716. MOVL $0, (CX)
  717. alreadylo:
  718. STI
  719. RET
  720. TEXT splx(SB), $0
  721. MOVL s+0(FP), AX
  722. TESTL $0x200, AX
  723. JNZ slo
  724. JMP shi
  725. TEXT spldone(SB), $0
  726. RET
  727. TEXT islo(SB), $0
  728. PUSHFL
  729. POPL AX
  730. ANDL $0x200, AX /* interrupt enable flag */
  731. RET
  732. /*
  733. * Test-And-Set
  734. */
  735. TEXT tas(SB), $0
  736. MOVL $0xDEADDEAD, AX
  737. MOVL lock+0(FP), BX
  738. XCHGL AX, (BX) /* lock->key */
  739. RET
  740. TEXT _xinc(SB), $0 /* void _xinc(long*); */
  741. MOVL l+0(FP), AX
  742. LOCK; INCL 0(AX)
  743. RET
  744. TEXT _xdec(SB), $0 /* long _xdec(long*); */
  745. MOVL l+0(FP), BX
  746. XORL AX, AX
  747. LOCK; DECL 0(BX)
  748. JLT _xdeclt
  749. JGT _xdecgt
  750. RET
  751. _xdecgt:
  752. INCL AX
  753. RET
  754. _xdeclt:
  755. DECL AX
  756. RET
  757. TEXT mb386(SB), $0
  758. POPL AX /* return PC */
  759. PUSHFL
  760. PUSHL CS
  761. PUSHL AX
  762. IRETL
  763. TEXT mb586(SB), $0
  764. XORL AX, AX
  765. CPUID
  766. RET
  767. TEXT sfence(SB), $0
  768. BYTE $0x0f
  769. BYTE $0xae
  770. BYTE $0xf8
  771. RET
  772. TEXT lfence(SB), $0
  773. BYTE $0x0f
  774. BYTE $0xae
  775. BYTE $0xe8
  776. RET
  777. TEXT mfence(SB), $0
  778. BYTE $0x0f
  779. BYTE $0xae
  780. BYTE $0xf0
  781. RET
  782. TEXT xchgw(SB), $0
  783. MOVL v+4(FP), AX
  784. MOVL p+0(FP), BX
  785. XCHGW AX, (BX)
  786. RET
  787. TEXT cmpswap486(SB), $0
  788. MOVL addr+0(FP), BX
  789. MOVL old+4(FP), AX
  790. MOVL new+8(FP), CX
  791. LOCK
  792. BYTE $0x0F; BYTE $0xB1; BYTE $0x0B /* CMPXCHGL CX, (BX) */
  793. JNZ didnt
  794. MOVL $1, AX
  795. RET
  796. didnt:
  797. XORL AX,AX
  798. RET
  799. TEXT mul64fract(SB), $0
  800. /*
  801. * Multiply two 64-bit number s and keep the middle 64 bits from the 128-bit result
  802. * See ../port/tod.c for motivation.
  803. */
  804. MOVL r+0(FP), CX
  805. XORL BX, BX /* BX = 0 */
  806. MOVL a+8(FP), AX
  807. MULL b+16(FP) /* a1*b1 */
  808. MOVL AX, 4(CX) /* r2 = lo(a1*b1) */
  809. MOVL a+8(FP), AX
  810. MULL b+12(FP) /* a1*b0 */
  811. MOVL AX, 0(CX) /* r1 = lo(a1*b0) */
  812. ADDL DX, 4(CX) /* r2 += hi(a1*b0) */
  813. MOVL a+4(FP), AX
  814. MULL b+16(FP) /* a0*b1 */
  815. ADDL AX, 0(CX) /* r1 += lo(a0*b1) */
  816. ADCL DX, 4(CX) /* r2 += hi(a0*b1) + carry */
  817. MOVL a+4(FP), AX
  818. MULL b+12(FP) /* a0*b0 */
  819. ADDL DX, 0(CX) /* r1 += hi(a0*b0) */
  820. ADCL BX, 4(CX) /* r2 += carry */
  821. RET
  822. /*
  823. * label consists of a stack pointer and a PC
  824. */
  825. TEXT gotolabel(SB), $0
  826. MOVL label+0(FP), AX
  827. MOVL 0(AX), SP /* restore sp */
  828. MOVL 4(AX), AX /* put return pc on the stack */
  829. MOVL AX, 0(SP)
  830. MOVL $1, AX /* return 1 */
  831. RET
  832. TEXT setlabel(SB), $0
  833. MOVL label+0(FP), AX
  834. MOVL SP, 0(AX) /* store sp */
  835. MOVL 0(SP), BX /* store return pc */
  836. MOVL BX, 4(AX)
  837. MOVL $0, AX /* return 0 */
  838. RET
  839. /*
  840. * Attempt at power saving. -rsc
  841. */
  842. TEXT halt(SB), $0
  843. CLI
  844. CMPL nrdy(SB), $0
  845. JEQ _nothingready
  846. STI
  847. RET
  848. _nothingready:
  849. STI
  850. HLT
  851. RET
  852. /*
  853. * Interrupt/exception handling.
  854. * Each entry in the vector table calls either _strayintr or _strayintrx depending
  855. * on whether an error code has been automatically pushed onto the stack
  856. * (_strayintrx) or not, in which case a dummy entry must be pushed before retrieving
  857. * the trap type from the vector table entry and placing it on the stack as part
  858. * of the Ureg structure.
  859. * The size of each entry in the vector table (6 bytes) is known in trapinit().
  860. */
  861. TEXT _strayintr(SB), $0
  862. PUSHL AX /* save AX */
  863. MOVL 4(SP), AX /* return PC from vectortable(SB) */
  864. JMP intrcommon
  865. TEXT _strayintrx(SB), $0
  866. XCHGL AX, (SP) /* swap AX with vectortable CALL PC */
  867. intrcommon:
  868. PUSHL DS /* save DS */
  869. PUSHL $(KDSEL)
  870. POPL DS /* fix up DS */
  871. MOVBLZX (AX), AX /* trap type -> AX */
  872. XCHGL AX, 4(SP) /* exchange trap type with saved AX */
  873. PUSHL ES /* save ES */
  874. PUSHL $(KDSEL)
  875. POPL ES /* fix up ES */
  876. PUSHL FS /* save the rest of the Ureg struct */
  877. PUSHL GS
  878. PUSHAL
  879. PUSHL SP /* Ureg* argument to trap */
  880. CALL trap(SB)
  881. TEXT forkret(SB), $0
  882. POPL AX
  883. POPAL
  884. POPL GS
  885. POPL FS
  886. POPL ES
  887. POPL DS
  888. ADDL $8, SP /* pop error code and trap type */
  889. IRETL
  890. TEXT vectortable(SB), $0
  891. CALL _strayintr(SB); BYTE $0x00 /* divide error */
  892. CALL _strayintr(SB); BYTE $0x01 /* debug exception */
  893. CALL _strayintr(SB); BYTE $0x02 /* NMI interrupt */
  894. CALL _strayintr(SB); BYTE $0x03 /* breakpoint */
  895. CALL _strayintr(SB); BYTE $0x04 /* overflow */
  896. CALL _strayintr(SB); BYTE $0x05 /* bound */
  897. CALL _strayintr(SB); BYTE $0x06 /* invalid opcode */
  898. CALL _strayintr(SB); BYTE $0x07 /* no coprocessor available */
  899. CALL _strayintrx(SB); BYTE $0x08 /* double fault */
  900. CALL _strayintr(SB); BYTE $0x09 /* coprocessor segment overflow */
  901. CALL _strayintrx(SB); BYTE $0x0A /* invalid TSS */
  902. CALL _strayintrx(SB); BYTE $0x0B /* segment not available */
  903. CALL _strayintrx(SB); BYTE $0x0C /* stack exception */
  904. CALL _strayintrx(SB); BYTE $0x0D /* general protection error */
  905. CALL _strayintrx(SB); BYTE $0x0E /* page fault */
  906. CALL _strayintr(SB); BYTE $0x0F /* */
  907. CALL _strayintr(SB); BYTE $0x10 /* coprocessor error */
  908. CALL _strayintrx(SB); BYTE $0x11 /* alignment check */
  909. CALL _strayintr(SB); BYTE $0x12 /* machine check */
  910. CALL _strayintr(SB); BYTE $0x13
  911. CALL _strayintr(SB); BYTE $0x14
  912. CALL _strayintr(SB); BYTE $0x15
  913. CALL _strayintr(SB); BYTE $0x16
  914. CALL _strayintr(SB); BYTE $0x17
  915. CALL _strayintr(SB); BYTE $0x18
  916. CALL _strayintr(SB); BYTE $0x19
  917. CALL _strayintr(SB); BYTE $0x1A
  918. CALL _strayintr(SB); BYTE $0x1B
  919. CALL _strayintr(SB); BYTE $0x1C
  920. CALL _strayintr(SB); BYTE $0x1D
  921. CALL _strayintr(SB); BYTE $0x1E
  922. CALL _strayintr(SB); BYTE $0x1F
  923. CALL _strayintr(SB); BYTE $0x20 /* VectorLAPIC */
  924. CALL _strayintr(SB); BYTE $0x21
  925. CALL _strayintr(SB); BYTE $0x22
  926. CALL _strayintr(SB); BYTE $0x23
  927. CALL _strayintr(SB); BYTE $0x24
  928. CALL _strayintr(SB); BYTE $0x25
  929. CALL _strayintr(SB); BYTE $0x26
  930. CALL _strayintr(SB); BYTE $0x27
  931. CALL _strayintr(SB); BYTE $0x28
  932. CALL _strayintr(SB); BYTE $0x29
  933. CALL _strayintr(SB); BYTE $0x2A
  934. CALL _strayintr(SB); BYTE $0x2B
  935. CALL _strayintr(SB); BYTE $0x2C
  936. CALL _strayintr(SB); BYTE $0x2D
  937. CALL _strayintr(SB); BYTE $0x2E
  938. CALL _strayintr(SB); BYTE $0x2F
  939. CALL _strayintr(SB); BYTE $0x30
  940. CALL _strayintr(SB); BYTE $0x31
  941. CALL _strayintr(SB); BYTE $0x32
  942. CALL _strayintr(SB); BYTE $0x33
  943. CALL _strayintr(SB); BYTE $0x34
  944. CALL _strayintr(SB); BYTE $0x35
  945. CALL _strayintr(SB); BYTE $0x36
  946. CALL _strayintr(SB); BYTE $0x37
  947. CALL _strayintr(SB); BYTE $0x38
  948. CALL _strayintr(SB); BYTE $0x39
  949. CALL _strayintr(SB); BYTE $0x3A
  950. CALL _strayintr(SB); BYTE $0x3B
  951. CALL _strayintr(SB); BYTE $0x3C
  952. CALL _strayintr(SB); BYTE $0x3D
  953. CALL _strayintr(SB); BYTE $0x3E
  954. CALL _strayintr(SB); BYTE $0x3F
  955. CALL _syscallintr(SB); BYTE $0x40 /* VectorSYSCALL */
  956. CALL _strayintr(SB); BYTE $0x41
  957. CALL _strayintr(SB); BYTE $0x42
  958. CALL _strayintr(SB); BYTE $0x43
  959. CALL _strayintr(SB); BYTE $0x44
  960. CALL _strayintr(SB); BYTE $0x45
  961. CALL _strayintr(SB); BYTE $0x46
  962. CALL _strayintr(SB); BYTE $0x47
  963. CALL _strayintr(SB); BYTE $0x48
  964. CALL _strayintr(SB); BYTE $0x49
  965. CALL _strayintr(SB); BYTE $0x4A
  966. CALL _strayintr(SB); BYTE $0x4B
  967. CALL _strayintr(SB); BYTE $0x4C
  968. CALL _strayintr(SB); BYTE $0x4D
  969. CALL _strayintr(SB); BYTE $0x4E
  970. CALL _strayintr(SB); BYTE $0x4F
  971. CALL _strayintr(SB); BYTE $0x50
  972. CALL _strayintr(SB); BYTE $0x51
  973. CALL _strayintr(SB); BYTE $0x52
  974. CALL _strayintr(SB); BYTE $0x53
  975. CALL _strayintr(SB); BYTE $0x54
  976. CALL _strayintr(SB); BYTE $0x55
  977. CALL _strayintr(SB); BYTE $0x56
  978. CALL _strayintr(SB); BYTE $0x57
  979. CALL _strayintr(SB); BYTE $0x58
  980. CALL _strayintr(SB); BYTE $0x59
  981. CALL _strayintr(SB); BYTE $0x5A
  982. CALL _strayintr(SB); BYTE $0x5B
  983. CALL _strayintr(SB); BYTE $0x5C
  984. CALL _strayintr(SB); BYTE $0x5D
  985. CALL _strayintr(SB); BYTE $0x5E
  986. CALL _strayintr(SB); BYTE $0x5F
  987. CALL _strayintr(SB); BYTE $0x60
  988. CALL _strayintr(SB); BYTE $0x61
  989. CALL _strayintr(SB); BYTE $0x62
  990. CALL _strayintr(SB); BYTE $0x63
  991. CALL _strayintr(SB); BYTE $0x64
  992. CALL _strayintr(SB); BYTE $0x65
  993. CALL _strayintr(SB); BYTE $0x66
  994. CALL _strayintr(SB); BYTE $0x67
  995. CALL _strayintr(SB); BYTE $0x68
  996. CALL _strayintr(SB); BYTE $0x69
  997. CALL _strayintr(SB); BYTE $0x6A
  998. CALL _strayintr(SB); BYTE $0x6B
  999. CALL _strayintr(SB); BYTE $0x6C
  1000. CALL _strayintr(SB); BYTE $0x6D
  1001. CALL _strayintr(SB); BYTE $0x6E
  1002. CALL _strayintr(SB); BYTE $0x6F
  1003. CALL _strayintr(SB); BYTE $0x70
  1004. CALL _strayintr(SB); BYTE $0x71
  1005. CALL _strayintr(SB); BYTE $0x72
  1006. CALL _strayintr(SB); BYTE $0x73
  1007. CALL _strayintr(SB); BYTE $0x74
  1008. CALL _strayintr(SB); BYTE $0x75
  1009. CALL _strayintr(SB); BYTE $0x76
  1010. CALL _strayintr(SB); BYTE $0x77
  1011. CALL _strayintr(SB); BYTE $0x78
  1012. CALL _strayintr(SB); BYTE $0x79
  1013. CALL _strayintr(SB); BYTE $0x7A
  1014. CALL _strayintr(SB); BYTE $0x7B
  1015. CALL _strayintr(SB); BYTE $0x7C
  1016. CALL _strayintr(SB); BYTE $0x7D
  1017. CALL _strayintr(SB); BYTE $0x7E
  1018. CALL _strayintr(SB); BYTE $0x7F
  1019. CALL _strayintr(SB); BYTE $0x80 /* Vector[A]PIC */
  1020. CALL _strayintr(SB); BYTE $0x81
  1021. CALL _strayintr(SB); BYTE $0x82
  1022. CALL _strayintr(SB); BYTE $0x83
  1023. CALL _strayintr(SB); BYTE $0x84
  1024. CALL _strayintr(SB); BYTE $0x85
  1025. CALL _strayintr(SB); BYTE $0x86
  1026. CALL _strayintr(SB); BYTE $0x87
  1027. CALL _strayintr(SB); BYTE $0x88
  1028. CALL _strayintr(SB); BYTE $0x89
  1029. CALL _strayintr(SB); BYTE $0x8A
  1030. CALL _strayintr(SB); BYTE $0x8B
  1031. CALL _strayintr(SB); BYTE $0x8C
  1032. CALL _strayintr(SB); BYTE $0x8D
  1033. CALL _strayintr(SB); BYTE $0x8E
  1034. CALL _strayintr(SB); BYTE $0x8F
  1035. CALL _strayintr(SB); BYTE $0x90
  1036. CALL _strayintr(SB); BYTE $0x91
  1037. CALL _strayintr(SB); BYTE $0x92
  1038. CALL _strayintr(SB); BYTE $0x93
  1039. CALL _strayintr(SB); BYTE $0x94
  1040. CALL _strayintr(SB); BYTE $0x95
  1041. CALL _strayintr(SB); BYTE $0x96
  1042. CALL _strayintr(SB); BYTE $0x97
  1043. CALL _strayintr(SB); BYTE $0x98
  1044. CALL _strayintr(SB); BYTE $0x99
  1045. CALL _strayintr(SB); BYTE $0x9A
  1046. CALL _strayintr(SB); BYTE $0x9B
  1047. CALL _strayintr(SB); BYTE $0x9C
  1048. CALL _strayintr(SB); BYTE $0x9D
  1049. CALL _strayintr(SB); BYTE $0x9E
  1050. CALL _strayintr(SB); BYTE $0x9F
  1051. CALL _strayintr(SB); BYTE $0xA0
  1052. CALL _strayintr(SB); BYTE $0xA1
  1053. CALL _strayintr(SB); BYTE $0xA2
  1054. CALL _strayintr(SB); BYTE $0xA3
  1055. CALL _strayintr(SB); BYTE $0xA4
  1056. CALL _strayintr(SB); BYTE $0xA5
  1057. CALL _strayintr(SB); BYTE $0xA6
  1058. CALL _strayintr(SB); BYTE $0xA7
  1059. CALL _strayintr(SB); BYTE $0xA8
  1060. CALL _strayintr(SB); BYTE $0xA9
  1061. CALL _strayintr(SB); BYTE $0xAA
  1062. CALL _strayintr(SB); BYTE $0xAB
  1063. CALL _strayintr(SB); BYTE $0xAC
  1064. CALL _strayintr(SB); BYTE $0xAD
  1065. CALL _strayintr(SB); BYTE $0xAE
  1066. CALL _strayintr(SB); BYTE $0xAF
  1067. CALL _strayintr(SB); BYTE $0xB0
  1068. CALL _strayintr(SB); BYTE $0xB1
  1069. CALL _strayintr(SB); BYTE $0xB2
  1070. CALL _strayintr(SB); BYTE $0xB3
  1071. CALL _strayintr(SB); BYTE $0xB4
  1072. CALL _strayintr(SB); BYTE $0xB5
  1073. CALL _strayintr(SB); BYTE $0xB6
  1074. CALL _strayintr(SB); BYTE $0xB7
  1075. CALL _strayintr(SB); BYTE $0xB8
  1076. CALL _strayintr(SB); BYTE $0xB9
  1077. CALL _strayintr(SB); BYTE $0xBA
  1078. CALL _strayintr(SB); BYTE $0xBB
  1079. CALL _strayintr(SB); BYTE $0xBC
  1080. CALL _strayintr(SB); BYTE $0xBD
  1081. CALL _strayintr(SB); BYTE $0xBE
  1082. CALL _strayintr(SB); BYTE $0xBF
  1083. CALL _strayintr(SB); BYTE $0xC0
  1084. CALL _strayintr(SB); BYTE $0xC1
  1085. CALL _strayintr(SB); BYTE $0xC2
  1086. CALL _strayintr(SB); BYTE $0xC3
  1087. CALL _strayintr(SB); BYTE $0xC4
  1088. CALL _strayintr(SB); BYTE $0xC5
  1089. CALL _strayintr(SB); BYTE $0xC6
  1090. CALL _strayintr(SB); BYTE $0xC7
  1091. CALL _strayintr(SB); BYTE $0xC8
  1092. CALL _strayintr(SB); BYTE $0xC9
  1093. CALL _strayintr(SB); BYTE $0xCA
  1094. CALL _strayintr(SB); BYTE $0xCB
  1095. CALL _strayintr(SB); BYTE $0xCC
  1096. CALL _strayintr(SB); BYTE $0xCD
  1097. CALL _strayintr(SB); BYTE $0xCE
  1098. CALL _strayintr(SB); BYTE $0xCF
  1099. CALL _strayintr(SB); BYTE $0xD0
  1100. CALL _strayintr(SB); BYTE $0xD1
  1101. CALL _strayintr(SB); BYTE $0xD2
  1102. CALL _strayintr(SB); BYTE $0xD3
  1103. CALL _strayintr(SB); BYTE $0xD4
  1104. CALL _strayintr(SB); BYTE $0xD5
  1105. CALL _strayintr(SB); BYTE $0xD6
  1106. CALL _strayintr(SB); BYTE $0xD7
  1107. CALL _strayintr(SB); BYTE $0xD8
  1108. CALL _strayintr(SB); BYTE $0xD9
  1109. CALL _strayintr(SB); BYTE $0xDA
  1110. CALL _strayintr(SB); BYTE $0xDB
  1111. CALL _strayintr(SB); BYTE $0xDC
  1112. CALL _strayintr(SB); BYTE $0xDD
  1113. CALL _strayintr(SB); BYTE $0xDE
  1114. CALL _strayintr(SB); BYTE $0xDF
  1115. CALL _strayintr(SB); BYTE $0xE0
  1116. CALL _strayintr(SB); BYTE $0xE1
  1117. CALL _strayintr(SB); BYTE $0xE2
  1118. CALL _strayintr(SB); BYTE $0xE3
  1119. CALL _strayintr(SB); BYTE $0xE4
  1120. CALL _strayintr(SB); BYTE $0xE5
  1121. CALL _strayintr(SB); BYTE $0xE6
  1122. CALL _strayintr(SB); BYTE $0xE7
  1123. CALL _strayintr(SB); BYTE $0xE8
  1124. CALL _strayintr(SB); BYTE $0xE9
  1125. CALL _strayintr(SB); BYTE $0xEA
  1126. CALL _strayintr(SB); BYTE $0xEB
  1127. CALL _strayintr(SB); BYTE $0xEC
  1128. CALL _strayintr(SB); BYTE $0xED
  1129. CALL _strayintr(SB); BYTE $0xEE
  1130. CALL _strayintr(SB); BYTE $0xEF
  1131. CALL _strayintr(SB); BYTE $0xF0
  1132. CALL _strayintr(SB); BYTE $0xF1
  1133. CALL _strayintr(SB); BYTE $0xF2
  1134. CALL _strayintr(SB); BYTE $0xF3
  1135. CALL _strayintr(SB); BYTE $0xF4
  1136. CALL _strayintr(SB); BYTE $0xF5
  1137. CALL _strayintr(SB); BYTE $0xF6
  1138. CALL _strayintr(SB); BYTE $0xF7
  1139. CALL _strayintr(SB); BYTE $0xF8
  1140. CALL _strayintr(SB); BYTE $0xF9
  1141. CALL _strayintr(SB); BYTE $0xFA
  1142. CALL _strayintr(SB); BYTE $0xFB
  1143. CALL _strayintr(SB); BYTE $0xFC
  1144. CALL _strayintr(SB); BYTE $0xFD
  1145. CALL _strayintr(SB); BYTE $0xFE
  1146. CALL _strayintr(SB); BYTE $0xFF