[9] | 1 | ! Doshead.s - DOS & BIOS support for boot.c Author: Kees J. Bot
|
---|
| 2 | !
|
---|
| 3 | !
|
---|
| 4 | ! This file contains the startup and low level support for the secondary
|
---|
| 5 | ! boot program. It contains functions for disk, tty and keyboard I/O,
|
---|
| 6 | ! copying memory to arbitrary locations, etc.
|
---|
| 7 | !
|
---|
| 8 | ! This runs under MS-DOS as a .COM file. A .COM file is what Minix calls
|
---|
| 9 | ! a common I&D executable, except that the first 256 bytes contains DOS
|
---|
| 10 | ! thingies.
|
---|
| 11 | !
|
---|
| 12 | .sect .text; .sect .rom; .sect .data; .sect .bss
|
---|
| 13 |
|
---|
| 14 | K_I386 = 0x0001 ! Call Minix in 386 mode
|
---|
| 15 | STACK = 16384 ! Number of bytes for the stack
|
---|
| 16 |
|
---|
| 17 | DS_SELECTOR = 3*8 ! Kernel data selector
|
---|
| 18 | ES_SELECTOR = 4*8 ! Flat 4 Gb
|
---|
| 19 | SS_SELECTOR = 5*8 ! Monitor stack
|
---|
| 20 | CS_SELECTOR = 6*8 ! Kernel code
|
---|
| 21 | MCS_SELECTOR= 7*8 ! Monitor code
|
---|
| 22 |
|
---|
| 23 | ESC = 0x1B ! Escape character
|
---|
| 24 |
|
---|
| 25 | ! Imported variables and functions:
|
---|
| 26 | .extern _caddr, _daddr, _runsize, _edata, _end ! Runtime environment
|
---|
| 27 | .extern _k_flags ! Special kernel flags
|
---|
| 28 | .extern _mem ! Free memory list
|
---|
| 29 | .extern _vdisk ! Name of the virtual disk
|
---|
| 30 |
|
---|
| 31 | .sect .text
|
---|
| 32 |
|
---|
| 33 | .use16 ! Tell 386 assembler we're in 16-bit mode
|
---|
| 34 |
|
---|
| 35 | .define _PSP
|
---|
| 36 | _PSP:
|
---|
| 37 | .space 256 ! Program Segment Prefix
|
---|
| 38 |
|
---|
| 39 | dosboot:
|
---|
| 40 | cld ! C compiler wants UP
|
---|
| 41 | xor ax, ax ! Zero
|
---|
| 42 | mov di, _edata ! Start of bss is at end of data
|
---|
| 43 | mov cx, _end ! End of bss (begin of heap)
|
---|
| 44 | sub cx, di ! Number of bss bytes
|
---|
| 45 | shr cx, 1 ! Number of words
|
---|
| 46 | rep stos ! Clear bss
|
---|
| 47 | cmp sp, _end+STACK
|
---|
| 48 | jb 0f
|
---|
| 49 | mov sp, _end+STACK ! "chmem" to 16 kb
|
---|
| 50 | 0:
|
---|
| 51 |
|
---|
| 52 | ! Are we called with the /U option?
|
---|
| 53 | movb cl, (_PSP+0x80) ! Argument byte count
|
---|
| 54 | xorb ch, ch
|
---|
| 55 | mov bx, _PSP+0x81 ! Argument string
|
---|
| 56 | 0: jcxz notuflag
|
---|
| 57 | cmpb (bx), 0x20 ! Whitespace?
|
---|
| 58 | ja 1f
|
---|
| 59 | inc bx
|
---|
| 60 | dec cx
|
---|
| 61 | jmp 0b
|
---|
| 62 | 1: cmp cx, 2 ! '/U' is two bytes
|
---|
| 63 | jne notuflag
|
---|
| 64 | cmpb (bx), 0x2F ! '/'?
|
---|
| 65 | jne notuflag
|
---|
| 66 | movb al, 1(bx)
|
---|
| 67 | andb al, ~0x20 ! Ignore case
|
---|
| 68 | cmpb al, 0x55 ! 'U'?
|
---|
| 69 | jne notuflag
|
---|
| 70 | jmp keepumb ! Go grab an UMB
|
---|
| 71 | notuflag:
|
---|
| 72 |
|
---|
| 73 | ! Remember the current video mode for restoration on exit.
|
---|
| 74 | movb ah, 0x0F ! Get current video mode
|
---|
| 75 | int 0x10
|
---|
| 76 | andb al, 0x7F ! Mask off bit 7 (no blanking)
|
---|
| 77 | movb (old_vid_mode), al
|
---|
| 78 | movb (cur_vid_mode), al
|
---|
| 79 |
|
---|
| 80 | ! We require at least MS-DOS 3.0.
|
---|
| 81 | mov ax, 0x3000 ! Get DOS version
|
---|
| 82 | int 0x21
|
---|
| 83 | cmpb al, 3 ! DOS 3.0+ ?
|
---|
| 84 | jae dosok
|
---|
| 85 | push tellbaddos
|
---|
| 86 | call _printf
|
---|
| 87 | jmp quit
|
---|
| 88 | .sect .rom
|
---|
| 89 | tellbaddos: .ascii "MS-DOS 3.0 or better required\n\0"
|
---|
| 90 | .sect .text
|
---|
| 91 | dosok:
|
---|
| 92 |
|
---|
| 93 | ! Find out how much "low" memory there is available, where it starts and
|
---|
| 94 | ! where it ends.
|
---|
| 95 | mov di, _mem ! di = memory list
|
---|
| 96 | mov ax, _PSP+0x80 ! From PSP:80 to next PSP is ours
|
---|
| 97 | mov dx, ds
|
---|
| 98 | call seg2abs
|
---|
| 99 | mov (di), ax
|
---|
| 100 | mov 2(di), dx ! mem[0].base = ds * 16 + 0x80
|
---|
| 101 | xor ax, ax
|
---|
| 102 | mov dx, (_PSP+2) ! First in-use segment far above us
|
---|
| 103 | call seg2abs
|
---|
| 104 | sub ax, (di)
|
---|
| 105 | sbb dx, 2(di) ! Minus base gives size
|
---|
| 106 | mov 4(di), ax
|
---|
| 107 | mov 6(di), dx ! mem[1].size = free low memory size
|
---|
| 108 |
|
---|
| 109 | ! Give C code access to the code segment, data segment and the size of this
|
---|
| 110 | ! process.
|
---|
| 111 | xor ax, ax
|
---|
| 112 | mov dx, cs
|
---|
| 113 | call seg2abs
|
---|
| 114 | mov (_caddr+0), ax
|
---|
| 115 | mov (_caddr+2), dx
|
---|
| 116 | xor ax, ax
|
---|
| 117 | mov dx, ds
|
---|
| 118 | call seg2abs
|
---|
| 119 | mov (_daddr+0), ax
|
---|
| 120 | mov (_daddr+2), dx
|
---|
| 121 | mov ax, sp
|
---|
| 122 | mov dx, ss ! End of stack = end of program
|
---|
| 123 | call seg2abs
|
---|
| 124 | sub ax, (_caddr+0)
|
---|
| 125 | sbb dx, (_caddr+2) ! Minus start of our code
|
---|
| 126 | mov (_runsize+0), ax
|
---|
| 127 | mov (_runsize+2), dx ! Is our size
|
---|
| 128 |
|
---|
| 129 | ! Patch the regular _getprocessor library routine to jump to 'getprocessor',
|
---|
| 130 | ! that checks if we happen to be in a V8086 straightjacket by returning '86'.
|
---|
| 131 | cseg movb (_getprocessor+0), 0xE9
|
---|
| 132 | mov ax, getprocessor
|
---|
| 133 | sub ax, _getprocessor+3
|
---|
| 134 | cseg mov (_getprocessor+1), ax
|
---|
| 135 |
|
---|
| 136 | ! Grab the largest chunk of extended memory available.
|
---|
| 137 | call _getprocessor
|
---|
| 138 | cmp ax, 286 ! Only 286s and above have extended memory
|
---|
| 139 | jb no_ext
|
---|
| 140 | mov ax, 0x4300 ! XMS driver check
|
---|
| 141 | int 0x2F
|
---|
| 142 | cmpb al, 0x80 ! XMS driver exists?
|
---|
| 143 | je xmsthere
|
---|
| 144 | get_ext: ! No driver, so can use all ext memory directly
|
---|
| 145 | call _getprocessor
|
---|
| 146 | cmp ax, 486 ! Assume 486s were the first to have >64M
|
---|
| 147 | jb small_ext ! (It helps to be paranoid when using the BIOS)
|
---|
| 148 | big_ext:
|
---|
| 149 | mov ax, 0xE801 ! Code for get memory size for >64M
|
---|
| 150 | int 0x15 ! ax = mem at 1M per 1K, bx = mem at 16M per 64K
|
---|
| 151 | jnc got_ext
|
---|
| 152 | small_ext:
|
---|
| 153 | movb ah, 0x88 ! Code for get extended memory size
|
---|
| 154 | clc ! Carry will stay clear if call exists
|
---|
| 155 | int 0x15 ! Returns size (in K) in ax for AT's
|
---|
| 156 | jc no_ext
|
---|
| 157 | test ax, ax ! An AT with no extended memory?
|
---|
| 158 | jz no_ext
|
---|
| 159 | xor bx, bx ! bx = mem above 16M per 64K = 0
|
---|
| 160 | got_ext:
|
---|
| 161 | mov cx, ax ! cx = copy of ext mem at 1M
|
---|
| 162 | mov 10(di), 0x0010 ! mem[1].base = 0x00100000 (1M)
|
---|
| 163 | mul (c1024)
|
---|
| 164 | mov 12(di), ax ! mem[1].size = "ext mem at 1M" * 1024
|
---|
| 165 | mov 14(di), dx
|
---|
| 166 | test bx, bx
|
---|
| 167 | jz no_ext ! No more ext mem above 16M?
|
---|
| 168 | cmp cx, 15*1024 ! Chunks adjacent? (precisely 15M at 1M?)
|
---|
| 169 | je adj_ext
|
---|
| 170 | mov 18(di), 0x0100 ! mem[2].base = 0x01000000 (16M)
|
---|
| 171 | mov 22(di), bx ! mem[2].size = "ext mem at 16M" * 64K
|
---|
| 172 | jmp no_ext
|
---|
| 173 | adj_ext:
|
---|
| 174 | add 14(di), bx ! Add ext mem above 16M to mem below 16M
|
---|
| 175 | no_ext:
|
---|
| 176 | jmp gotxms
|
---|
| 177 |
|
---|
| 178 | xmsthere:
|
---|
| 179 | mov ax, 0x4310 ! Get XMS driver address
|
---|
| 180 | int 0x2F
|
---|
| 181 | mov (xms_driver+0), bx
|
---|
| 182 | mov (xms_driver+2), es
|
---|
| 183 | push ds
|
---|
| 184 | pop es
|
---|
| 185 | movb ah, 0x08 ! Query free extended memory
|
---|
| 186 | xorb bl, bl
|
---|
| 187 | callf (xms_driver)
|
---|
| 188 | testb bl, bl
|
---|
| 189 | jnz xmserr
|
---|
| 190 | push ax ! ax = size of largest block in kb
|
---|
| 191 | mul (c1024)
|
---|
| 192 | mov 12(di), ax
|
---|
| 193 | mov 14(di), dx ! mem[1].size = ax * 1024
|
---|
| 194 | pop dx ! dx = size of largest block in kb
|
---|
| 195 | movb ah, 0x09 ! Allocate XMS block of size dx
|
---|
| 196 | callf (xms_driver)
|
---|
| 197 | test ax, ax
|
---|
| 198 | jz xmserr
|
---|
| 199 | mov (xms_handle), dx ! Save handle
|
---|
| 200 | movb ah, 0x0C ! Lock XMS block (handle in dx)
|
---|
| 201 | callf (xms_driver)
|
---|
| 202 | test ax, ax
|
---|
| 203 | jz xmserr
|
---|
| 204 | mov 8(di), bx
|
---|
| 205 | mov 10(di), dx ! mem[1].base = Address of locked block
|
---|
| 206 | gotxms:
|
---|
| 207 |
|
---|
| 208 | ! If we're running in a DOS box then they're might be an Upper Memory Block
|
---|
| 209 | ! we can use. Every little bit helps when in real mode.
|
---|
| 210 | mov ax, 20(di)
|
---|
| 211 | or ax, 22(di) ! Can we use mem[2]?
|
---|
| 212 | jnz gotumb
|
---|
| 213 | mov dx, 0xFFFF ! dx = Maximum size, i.e. gimme all
|
---|
| 214 | call getumb ! Get UMB, dx = segment, cx = length
|
---|
| 215 | test cx, cx ! Did we get a block?
|
---|
| 216 | jz gotumb
|
---|
| 217 | xor ax, ax ! dx:ax = memory block
|
---|
| 218 | call seg2abs
|
---|
| 219 | mov 16(di), ax
|
---|
| 220 | mov 18(di), dx ! mem[2].base = memory block base
|
---|
| 221 | mov dx, cx
|
---|
| 222 | xor ax, ax ! dx:ax = length of memory block
|
---|
| 223 | call seg2abs
|
---|
| 224 | mov 20(di), ax
|
---|
| 225 | mov 22(di), dx ! mem[2].size = memory block length
|
---|
| 226 | gotumb:
|
---|
| 227 |
|
---|
| 228 | ! Set up an INT 24 "critical error" handler that returns "fail". This way
|
---|
| 229 | ! Minix won't suffer from "(A)bort, (R)etry, (I)nfluence with a large hammer?".
|
---|
| 230 | mov (0x007C), 0x03B0 ! movb al, 0x03 (fail code)
|
---|
| 231 | movb (0x007E), 0xCF ! iret
|
---|
| 232 | movb ah, 0x25 ! Set interrupt vector
|
---|
| 233 | mov dx, 0x007C ! ds:dx = ds:0x007C = interrupt handler
|
---|
| 234 | int 0x21
|
---|
| 235 |
|
---|
| 236 | ! Time to switch to a higher level language (not much higher)
|
---|
| 237 | call _boot
|
---|
| 238 |
|
---|
| 239 | ! void ..exit(int status)
|
---|
| 240 | ! Exit the monitor by returning to DOS.
|
---|
| 241 | .define _exit, __exit, ___exit ! Make various compilers happy
|
---|
| 242 | _exit:
|
---|
| 243 | __exit:
|
---|
| 244 | ___exit:
|
---|
| 245 | mov dx, (xms_handle)
|
---|
| 246 | cmp dx, -1 ! Is there an ext mem block in use?
|
---|
| 247 | je nohandle
|
---|
| 248 | movb ah, 0x0D ! Unlock extended memory block
|
---|
| 249 | callf (xms_driver)
|
---|
| 250 | mov dx, (xms_handle)
|
---|
| 251 | movb ah, 0x0A ! Free extended memory block
|
---|
| 252 | callf (xms_driver)
|
---|
| 253 | nohandle:
|
---|
| 254 | call restore_video
|
---|
| 255 | pop ax
|
---|
| 256 | pop ax ! Return code in al
|
---|
| 257 | movb ah, 0x4C ! Terminate with return code
|
---|
| 258 | int 0x21
|
---|
| 259 |
|
---|
| 260 | quit: ! exit(1)
|
---|
| 261 | movb al, 1
|
---|
| 262 | push ax
|
---|
| 263 | call _exit
|
---|
| 264 |
|
---|
| 265 | xmserr:
|
---|
| 266 | xorb bh, bh
|
---|
| 267 | push bx
|
---|
| 268 | push tellxmserr
|
---|
| 269 | call _printf
|
---|
| 270 | jmp quit
|
---|
| 271 | .sect .rom
|
---|
| 272 | tellxmserr: .ascii "Extended memory problem, error 0x%02x\n\0"
|
---|
| 273 | .sect .text
|
---|
| 274 |
|
---|
| 275 | ! int getprocessor(void)
|
---|
| 276 | ! Prefix for the regular _getprocessor call that first checks if we're
|
---|
| 277 | ! running in a virtual 8086 box.
|
---|
| 278 | getprocessor:
|
---|
| 279 | push sp ! Is pushed sp equal to sp?
|
---|
| 280 | pop ax
|
---|
| 281 | cmp ax, sp
|
---|
| 282 | jne gettrueproc ! If not then it's a plain 8086 or 80186
|
---|
| 283 | .data1 0x0F,0x01,0xE0 ! Use old 286 SMSW instruction to get the MSW
|
---|
| 284 | testb al, 0x01 ! Protected mode enabled?
|
---|
| 285 | jz gettrueproc ! If not then a 286 or better in real mode
|
---|
| 286 | mov ax, 86 ! Forget fancy tricks, say it's an 8086
|
---|
| 287 | ret
|
---|
| 288 | gettrueproc: ! Get the true processor type
|
---|
| 289 | push bp ! _getprocessor prologue that is patched over.
|
---|
| 290 | mov bp, sp
|
---|
| 291 | jmp _getprocessor+3
|
---|
| 292 |
|
---|
| 293 | ! Try to get an Upper Memory Block under MS-DOS 5+. Try to get one up to size
|
---|
| 294 | ! dx, return segment of UMB found in dx and size in paragraphs in cx.
|
---|
| 295 | getumb:
|
---|
| 296 | xor cx, cx ! Initially nothing found
|
---|
| 297 | mov ax, 0x3000 ! Get DOS version
|
---|
| 298 | int 0x21
|
---|
| 299 | cmpb al, 5 ! MS-DOS 5.0 or better?
|
---|
| 300 | jb retumb
|
---|
| 301 | mov ax, 0x544D ! Get UMB kept by BOOT /U
|
---|
| 302 | int 0x15 ! Returns dx = segment, cx = size
|
---|
| 303 | jc 0f
|
---|
| 304 | cmp ax, 0x4D54 ! Carry clear and ax byte swapped?
|
---|
| 305 | je retumb
|
---|
| 306 | 0: mov ax, 0x5802 ! Get UMB link state
|
---|
| 307 | int 0x21
|
---|
| 308 | xorb ah, ah
|
---|
| 309 | push ax ! Save UMB link state
|
---|
| 310 | mov ax, 0x5803 ! Set UMB link state
|
---|
| 311 | mov bx, 0x0001 ! Add UMBs to DOS memory chain
|
---|
| 312 | int 0x21
|
---|
| 313 | mov ax, 0x5800 ! Get memory allocation strategy
|
---|
| 314 | int 0x21
|
---|
| 315 | push ax ! Save allocation strategy
|
---|
| 316 | mov ax, 0x5801 ! Set memory allocation strategy
|
---|
| 317 | mov bx, 0x0080 ! First fit, try high then low memory
|
---|
| 318 | int 0x21
|
---|
| 319 | movb ah, 0x48 ! Allocate memory
|
---|
| 320 | mov bx, dx ! Number of paragraphs wanted
|
---|
| 321 | int 0x21 ! Fails with bx = size of largest
|
---|
| 322 | jnc 0f ! Succeeds with ax = allocated block
|
---|
| 323 | test bx, bx ! Is there any?
|
---|
| 324 | jz no_umb
|
---|
| 325 | movb ah, 0x48 ! Allocate memory
|
---|
| 326 | int 0x21
|
---|
| 327 | jc no_umb ! Did we get some?
|
---|
| 328 | 0: mov dx, ax ! dx = segment
|
---|
| 329 | mov cx, bx ! cx = size
|
---|
| 330 | no_umb: mov ax, 0x5801 ! Set memory allocation strategy
|
---|
| 331 | pop bx ! bx = saved former strategy
|
---|
| 332 | int 0x21
|
---|
| 333 | mov ax, 0x5803 ! Set UMB link state
|
---|
| 334 | pop bx ! bx = saved former link state
|
---|
| 335 | int 0x21
|
---|
| 336 | retumb: ret
|
---|
| 337 |
|
---|
| 338 | ! 'BOOT /U' instructs this program to grab the biggest available UMB and to
|
---|
| 339 | ! sit on it until the next invocation of BOOT wants it back. These shenanigans
|
---|
| 340 | ! are necessary because Windows 95 keeps all UMBs to itself unless you get hold
|
---|
| 341 | ! of them first.
|
---|
| 342 | umb = 0x80 ! UMB base and size
|
---|
| 343 | old15 = 0x84 ! Old 15 interrupt vector
|
---|
| 344 | new15 = 0x88 ! New 15 interrupt handler
|
---|
| 345 | keepumb:
|
---|
| 346 | mov ax, 0x544D ! "Keep UMB" handler already present?
|
---|
| 347 | int 0x15
|
---|
| 348 | jc 0f
|
---|
| 349 | cmp ax, 0x4D54
|
---|
| 350 | je exitumb ! Already present, so quit
|
---|
| 351 | 0:
|
---|
| 352 | mov si, new15start
|
---|
| 353 | mov di, new15
|
---|
| 354 | mov cx, new15end
|
---|
| 355 | sub cx, si
|
---|
| 356 | rep movsb ! Copy handler into place
|
---|
| 357 | add di, 15
|
---|
| 358 | movb cl, 4
|
---|
| 359 | shr di, cl ! di = first segment above handler
|
---|
| 360 | mov cx, cs
|
---|
| 361 | cmp cx, 0xA000 ! Are we loaded high perchance?
|
---|
| 362 | jb nothigh
|
---|
| 363 | werehigh:
|
---|
| 364 | add cx, di
|
---|
| 365 | mov (umb+0), cx ! Use my own memory as the UMB to keep
|
---|
| 366 | mov ax, (_PSP+2) ! Up to the next in-use segment
|
---|
| 367 | sub ax, dx ! ax = size of my free memory
|
---|
| 368 | cmp ax, 0x1000 ! At least 64K?
|
---|
| 369 | jb exitumb ! Don't bother if less
|
---|
| 370 | mov (umb+2), 0x1000 ! Size of UMB
|
---|
| 371 | add di, 0x1000 ! Keep my code plus 64K when TSR
|
---|
| 372 | jmp hook15
|
---|
| 373 | nothigh:
|
---|
| 374 | mov dx, 0x1000
|
---|
| 375 | call getumb ! Grab an UMB of at most 64K
|
---|
| 376 | cmp cx, 0x1000 ! Did we get 64K?
|
---|
| 377 | jb exitumb ! Otherwise don't bother
|
---|
| 378 | mov (umb+0), dx
|
---|
| 379 | mov (umb+2), cx
|
---|
| 380 | hook15:
|
---|
| 381 | mov ax, 0x3515 ! Get interrupt vector
|
---|
| 382 | int 0x21
|
---|
| 383 | mov (old15+0), bx
|
---|
| 384 | mov (old15+2), es ! Old 15 interrupt
|
---|
| 385 | mov ax, 0x2515 ! Set interrupt vector
|
---|
| 386 | mov dx, new15 ! ds:dx = new 15 handler
|
---|
| 387 | int 0x21
|
---|
| 388 | mov ax, 0x3100 ! Terminate and stay resident
|
---|
| 389 | mov dx, di ! dx = di = paragraphs we keep
|
---|
| 390 | int 0x21
|
---|
| 391 | exitumb:
|
---|
| 392 | mov ax, 0x4C00 ! exit(0)
|
---|
| 393 | int 0x21
|
---|
| 394 |
|
---|
| 395 | new15start: ! New interrupt 15 handler
|
---|
| 396 | pushf
|
---|
| 397 | cmp ax, 0x544D ! Is it my call?
|
---|
| 398 | je my15
|
---|
| 399 | popf
|
---|
| 400 | cseg jmpf (old15) ! No, continue with old 15
|
---|
| 401 | my15: popf
|
---|
| 402 | push bp
|
---|
| 403 | mov bp, sp
|
---|
| 404 | andb 6(bp), ~0x01 ! clear carry, call will succeed
|
---|
| 405 | xchgb al, ah ! ax = 4D54, also means call works
|
---|
| 406 | cseg mov dx, (umb+0) ! dx = base of UMB
|
---|
| 407 | cseg mov cx, (umb+2) ! cx = size of UMB
|
---|
| 408 | pop bp
|
---|
| 409 | iret ! return to caller
|
---|
| 410 | new15end:
|
---|
| 411 |
|
---|
| 412 | ! u32_t mon2abs(void *ptr)
|
---|
| 413 | ! Address in monitor data to absolute address.
|
---|
| 414 | .define _mon2abs
|
---|
| 415 | _mon2abs:
|
---|
| 416 | mov bx, sp
|
---|
| 417 | mov ax, 2(bx) ! ptr
|
---|
| 418 | mov dx, ds ! Monitor data segment
|
---|
| 419 | !jmp seg2abs
|
---|
| 420 |
|
---|
| 421 | seg2abs: ! Translate dx:ax to the 32 bit address dx-ax
|
---|
| 422 | push cx
|
---|
| 423 | movb ch, dh
|
---|
| 424 | movb cl, 4
|
---|
| 425 | shl dx, cl
|
---|
| 426 | shrb ch, cl ! ch-dx = dx << 4
|
---|
| 427 | add ax, dx
|
---|
| 428 | adcb ch, 0 ! ch-ax = ch-dx + ax
|
---|
| 429 | movb dl, ch
|
---|
| 430 | xorb dh, dh ! dx-ax = ch-ax
|
---|
| 431 | pop cx
|
---|
| 432 | ret
|
---|
| 433 |
|
---|
| 434 | abs2seg: ! Translate the 32 bit address dx-ax to dx:ax
|
---|
| 435 | push cx
|
---|
| 436 | movb ch, dl
|
---|
| 437 | mov dx, ax ! ch-dx = dx-ax
|
---|
| 438 | and ax, 0x000F ! Offset in ax
|
---|
| 439 | movb cl, 4
|
---|
| 440 | shr dx, cl
|
---|
| 441 | shlb ch, cl
|
---|
| 442 | orb dh, ch ! dx = ch-dx >> 4
|
---|
| 443 | pop cx
|
---|
| 444 | ret
|
---|
| 445 |
|
---|
| 446 | ! void raw_copy(u32_t dstaddr, u32_t srcaddr, u32_t count)
|
---|
| 447 | ! Copy count bytes from srcaddr to dstaddr. Don't do overlaps.
|
---|
| 448 | ! Also handles copying words to or from extended memory.
|
---|
| 449 | .define _raw_copy
|
---|
| 450 | _raw_copy:
|
---|
| 451 | push bp
|
---|
| 452 | mov bp, sp
|
---|
| 453 | push si
|
---|
| 454 | push di ! Save C variable registers
|
---|
| 455 | copy:
|
---|
| 456 | cmp 14(bp), 0
|
---|
| 457 | jnz bigcopy
|
---|
| 458 | mov cx, 12(bp)
|
---|
| 459 | jcxz copydone ! Count is zero, end copy
|
---|
| 460 | cmp cx, 0xFFF0
|
---|
| 461 | jb smallcopy
|
---|
| 462 | bigcopy:mov cx, 0xFFF0 ! Don't copy more than about 64K at once
|
---|
| 463 | smallcopy:
|
---|
| 464 | push cx ! Save copying count
|
---|
| 465 | mov ax, 4(bp)
|
---|
| 466 | mov dx, 6(bp)
|
---|
| 467 | cmp dx, 0x0010 ! Copy to extended memory?
|
---|
| 468 | jae ext_copy
|
---|
| 469 | cmp 10(bp), 0x0010 ! Copy from extended memory?
|
---|
| 470 | jae ext_copy
|
---|
| 471 | call abs2seg
|
---|
| 472 | mov di, ax
|
---|
| 473 | mov es, dx ! es:di = dstaddr
|
---|
| 474 | mov ax, 8(bp)
|
---|
| 475 | mov dx, 10(bp)
|
---|
| 476 | call abs2seg
|
---|
| 477 | mov si, ax
|
---|
| 478 | mov ds, dx ! ds:si = srcaddr
|
---|
| 479 | shr cx, 1 ! Words to move
|
---|
| 480 | rep movs ! Do the word copy
|
---|
| 481 | adc cx, cx ! One more byte?
|
---|
| 482 | rep movsb ! Do the byte copy
|
---|
| 483 | mov ax, ss ! Restore ds and es from the remaining ss
|
---|
| 484 | mov ds, ax
|
---|
| 485 | mov es, ax
|
---|
| 486 | jmp copyadjust
|
---|
| 487 | ext_copy:
|
---|
| 488 | mov (x_dst_desc+2), ax
|
---|
| 489 | movb (x_dst_desc+4), dl ! Set base of destination segment
|
---|
| 490 | mov ax, 8(bp)
|
---|
| 491 | mov dx, 10(bp)
|
---|
| 492 | mov (x_src_desc+2), ax
|
---|
| 493 | movb (x_src_desc+4), dl ! Set base of source segment
|
---|
| 494 | mov si, x_gdt ! es:si = global descriptor table
|
---|
| 495 | shr cx, 1 ! Words to move
|
---|
| 496 | movb ah, 0x87 ! Code for extended memory move
|
---|
| 497 | int 0x15
|
---|
| 498 | copyadjust:
|
---|
| 499 | pop cx ! Restore count
|
---|
| 500 | add 4(bp), cx
|
---|
| 501 | adc 6(bp), 0 ! srcaddr += copycount
|
---|
| 502 | add 8(bp), cx
|
---|
| 503 | adc 10(bp), 0 ! dstaddr += copycount
|
---|
| 504 | sub 12(bp), cx
|
---|
| 505 | sbb 14(bp), 0 ! count -= copycount
|
---|
| 506 | jmp copy ! and repeat
|
---|
| 507 | copydone:
|
---|
| 508 | pop di
|
---|
| 509 | pop si ! Restore C variable registers
|
---|
| 510 | pop bp
|
---|
| 511 | ret
|
---|
| 512 |
|
---|
| 513 | ! u16_t get_word(u32_t addr);
|
---|
| 514 | ! void put_word(u32_t addr, u16_t word);
|
---|
| 515 | ! Read or write a 16 bits word at an arbitrary location.
|
---|
| 516 | .define _get_word, _put_word
|
---|
| 517 | _get_word:
|
---|
| 518 | mov bx, sp
|
---|
| 519 | call gp_getaddr
|
---|
| 520 | mov ax, (bx) ! Word to get from addr
|
---|
| 521 | jmp gp_ret
|
---|
| 522 | _put_word:
|
---|
| 523 | mov bx, sp
|
---|
| 524 | push 6(bx) ! Word to store at addr
|
---|
| 525 | call gp_getaddr
|
---|
| 526 | pop (bx) ! Store the word
|
---|
| 527 | jmp gp_ret
|
---|
| 528 | gp_getaddr:
|
---|
| 529 | mov ax, 2(bx)
|
---|
| 530 | mov dx, 4(bx)
|
---|
| 531 | call abs2seg
|
---|
| 532 | mov bx, ax
|
---|
| 533 | mov ds, dx ! ds:bx = addr
|
---|
| 534 | ret
|
---|
| 535 | gp_ret:
|
---|
| 536 | push es
|
---|
| 537 | pop ds ! Restore ds
|
---|
| 538 | ret
|
---|
| 539 |
|
---|
| 540 | ! void relocate(void);
|
---|
| 541 | ! After the program has copied itself to a safer place, it needs to change
|
---|
| 542 | ! the segment registers. Caddr has already been set to the new location.
|
---|
| 543 | .define _relocate
|
---|
| 544 | _relocate:
|
---|
| 545 | pop bx ! Return address
|
---|
| 546 | mov ax, (_caddr+0)
|
---|
| 547 | mov dx, (_caddr+2)
|
---|
| 548 | call abs2seg
|
---|
| 549 | mov cx, dx ! cx = new code segment
|
---|
| 550 | mov ax, cs ! Old code segment
|
---|
| 551 | sub ax, cx ! ax = -(new - old) = -Moving offset
|
---|
| 552 | mov dx, ds
|
---|
| 553 | sub dx, ax
|
---|
| 554 | mov ds, dx ! ds += (new - old)
|
---|
| 555 | mov es, dx
|
---|
| 556 | mov ss, dx
|
---|
| 557 | xor ax, ax
|
---|
| 558 | call seg2abs
|
---|
| 559 | mov (_daddr+0), ax
|
---|
| 560 | mov (_daddr+2), dx ! New data address
|
---|
| 561 | push cx ! New text segment
|
---|
| 562 | push bx ! Return offset of this function
|
---|
| 563 | retf ! Relocate
|
---|
| 564 |
|
---|
| 565 | ! void *brk(void *addr)
|
---|
| 566 | ! void *sbrk(size_t incr)
|
---|
| 567 | ! Cannot fail implementations of brk(2) and sbrk(3), so we can use
|
---|
| 568 | ! malloc(3). They reboot on stack collision instead of returning -1.
|
---|
| 569 | .sect .data
|
---|
| 570 | .align 2
|
---|
| 571 | break: .data2 _end ! A fake heap pointer
|
---|
| 572 | .sect .text
|
---|
| 573 | .define _brk, __brk, _sbrk, __sbrk
|
---|
| 574 | _brk:
|
---|
| 575 | __brk: ! __brk is for the standard C compiler
|
---|
| 576 | xor ax, ax
|
---|
| 577 | jmp sbrk ! break= 0; return sbrk(addr);
|
---|
| 578 | _sbrk:
|
---|
| 579 | __sbrk:
|
---|
| 580 | mov ax, (break) ! ax= current break
|
---|
| 581 | sbrk: push ax ! save it as future return value
|
---|
| 582 | mov bx, sp ! Stack is now: (retval, retaddr, incr, ...)
|
---|
| 583 | add ax, 4(bx) ! ax= break + increment
|
---|
| 584 | mov (break), ax ! Set new break
|
---|
| 585 | lea dx, -1024(bx) ! sp minus a bit of breathing space
|
---|
| 586 | cmp dx, ax ! Compare with the new break
|
---|
| 587 | jb heaperr ! Suffocating noises
|
---|
| 588 | pop ax ! Return old break (0 for brk)
|
---|
| 589 | ret
|
---|
| 590 | heaperr:push nomem
|
---|
| 591 | call _printf
|
---|
| 592 | call quit
|
---|
| 593 | .sect .rom
|
---|
| 594 | nomem: .ascii "\nOut of memory\n\0"
|
---|
| 595 | .sect .text
|
---|
| 596 |
|
---|
| 597 | ! int dev_open(void);
|
---|
| 598 | ! Open file 'vdisk' to use as the Minix virtual disk. Store handle in
|
---|
| 599 | ! vfd. Returns 0 for success, otherwise the DOS error code.
|
---|
| 600 | .define _dev_open
|
---|
| 601 | _dev_open:
|
---|
| 602 | call _dev_close ! If already open then first close
|
---|
| 603 | mov dx, (_vdisk) ! ds:dx = Address of file name
|
---|
| 604 | mov ax, 0x3D22 ! Open file read-write & deny write
|
---|
| 605 | int 0x21
|
---|
| 606 | jnc opok ! Open succeeded?
|
---|
| 607 | cmp ax, 5 ! Open failed, "access denied"?
|
---|
| 608 | jne opbad
|
---|
| 609 | mov ax, 0x3D40 ! Open file read-only
|
---|
| 610 | int 0x21
|
---|
| 611 | jc opbad
|
---|
| 612 | opok: mov (vfd), ax ! File handle to open file
|
---|
| 613 | xor ax, ax ! Zero for success
|
---|
| 614 | opbad: ret
|
---|
| 615 |
|
---|
| 616 | ! int dev_close(void);
|
---|
| 617 | ! Close the dos virtual disk.
|
---|
| 618 | .define _dev_close
|
---|
| 619 | _dev_close:
|
---|
| 620 | mov bx, -1
|
---|
| 621 | cmp (vfd), bx ! Already closed?
|
---|
| 622 | je 1f
|
---|
| 623 | movb ah, 0x3E ! Close file
|
---|
| 624 | xchg bx, (vfd) ! bx = vfd; vfd = -1;
|
---|
| 625 | int 0x21
|
---|
| 626 | jc 0f
|
---|
| 627 | 1: xor ax, ax
|
---|
| 628 | 0: ret
|
---|
| 629 |
|
---|
| 630 | ! int dev_boundary(u32_t sector);
|
---|
| 631 | ! Returns false; files have no visible boundaries.
|
---|
| 632 | .define _dev_boundary
|
---|
| 633 | _dev_boundary:
|
---|
| 634 | xor ax, ax
|
---|
| 635 | ret
|
---|
| 636 |
|
---|
| 637 | ! int readsectors(u32_t bufaddr, u32_t sector, u8_t count)
|
---|
| 638 | ! int writesectors(u32_t bufaddr, u32_t sector, u8_t count)
|
---|
| 639 | ! Read/write several sectors from/to the Minix virtual disk. Count
|
---|
| 640 | ! must fit in a byte. The external variable vfd is the file handle.
|
---|
| 641 | ! Returns 0 for success, otherwise the DOS error code.
|
---|
| 642 | !
|
---|
| 643 | .define _readsectors, _writesectors
|
---|
| 644 | _writesectors:
|
---|
| 645 | push bp
|
---|
| 646 | mov bp, sp
|
---|
| 647 | movb 13(bp), 0x40 ! Code for a file write
|
---|
| 648 | jmp rwsec
|
---|
| 649 | _readsectors:
|
---|
| 650 | push bp
|
---|
| 651 | mov bp, sp
|
---|
| 652 | movb 13(bp), 0x3F ! Code for a file read
|
---|
| 653 | rwsec:
|
---|
| 654 | cmp (vfd), -1 ! Currently closed?
|
---|
| 655 | jne 0f
|
---|
| 656 | call _dev_open ! Open file if needed
|
---|
| 657 | test ax, ax
|
---|
| 658 | jnz rwerr
|
---|
| 659 | 0: mov dx, 8(bp)
|
---|
| 660 | mov bx, 10(bp) ! bx-dx = Sector number
|
---|
| 661 | mov cx, 9
|
---|
| 662 | mul512: shl dx, 1
|
---|
| 663 | rcl bx, 1 ! bx-dx *= 512
|
---|
| 664 | loop mul512
|
---|
| 665 | mov cx, bx ! cx-dx = Byte position in file
|
---|
| 666 | mov bx, (vfd) ! bx = File handle
|
---|
| 667 | mov ax, 0x4200 ! Lseek absolute
|
---|
| 668 | int 0x21
|
---|
| 669 | jb rwerr
|
---|
| 670 | mov bx, (vfd) ! bx = File handle
|
---|
| 671 | mov ax, 4(bp)
|
---|
| 672 | mov dx, 6(bp) ! dx-ax = Address to transfer data to/from
|
---|
| 673 | call abs2seg
|
---|
| 674 | mov ds, dx
|
---|
| 675 | mov dx, ax ! ds:dx = Address to transfer data to/from
|
---|
| 676 | xorb cl, cl
|
---|
| 677 | movb ch, 12(bp) ! ch = Number of sectors to transfer
|
---|
| 678 | shl cx, 1 ! cx = Number of bytes to transfer
|
---|
| 679 | push cx ! Save count
|
---|
| 680 | movb ah, 13(bp) ! Read or write
|
---|
| 681 | int 0x21
|
---|
| 682 | pop cx ! Restore count
|
---|
| 683 | push es
|
---|
| 684 | pop ds ! Restore ds
|
---|
| 685 | jb rwerr
|
---|
| 686 | cmp ax, cx ! All bytes transferred?
|
---|
| 687 | je rwall
|
---|
| 688 | mov ax, 0x05 ! The DOS code for "I/O error", but different
|
---|
| 689 | jmp rwerr
|
---|
| 690 | rwall: call wheel ! Display tricks
|
---|
| 691 | xor ax, ax
|
---|
| 692 | rwerr: pop bp
|
---|
| 693 | ret
|
---|
| 694 |
|
---|
| 695 | ! int getch(void);
|
---|
| 696 | ! Read a character from the keyboard, and check for an expired timer.
|
---|
| 697 | ! A carriage return is changed into a linefeed for UNIX compatibility.
|
---|
| 698 | .define _getch
|
---|
| 699 | _getch:
|
---|
| 700 | xor ax, ax
|
---|
| 701 | xchg ax, (unchar) ! Ungotten character?
|
---|
| 702 | test ax, ax
|
---|
| 703 | jnz gotch
|
---|
| 704 | getch: hlt ! Play dead until interrupted (see pause())
|
---|
| 705 | movb ah, 0x01 ! Keyboard status
|
---|
| 706 | int 0x16
|
---|
| 707 | jnz press ! Keypress?
|
---|
| 708 | call _expired ! Timer expired?
|
---|
| 709 | test ax, ax
|
---|
| 710 | jz getch
|
---|
| 711 | mov ax, ESC ! Return ESC
|
---|
| 712 | ret
|
---|
| 713 | press:
|
---|
| 714 | xorb ah, ah ! Read character from keyboard
|
---|
| 715 | int 0x16
|
---|
| 716 | cmpb al, 0x0D ! Carriage return?
|
---|
| 717 | jnz nocr
|
---|
| 718 | movb al, 0x0A ! Change to linefeed
|
---|
| 719 | nocr: cmpb al, ESC ! Escape typed?
|
---|
| 720 | jne noesc
|
---|
| 721 | inc (escape) ! Set flag
|
---|
| 722 | noesc: xorb ah, ah ! ax = al
|
---|
| 723 | gotch: ret
|
---|
| 724 |
|
---|
| 725 | ! int ungetch(void);
|
---|
| 726 | ! Return a character to undo a getch().
|
---|
| 727 | .define _ungetch
|
---|
| 728 | _ungetch:
|
---|
| 729 | mov bx, sp
|
---|
| 730 | mov ax, 2(bx)
|
---|
| 731 | mov (unchar), ax
|
---|
| 732 | ret
|
---|
| 733 |
|
---|
| 734 | ! int escape(void);
|
---|
| 735 | ! True if ESC has been typed.
|
---|
| 736 | .define _escape
|
---|
| 737 | _escape:
|
---|
| 738 | movb ah, 0x01 ! Keyboard status
|
---|
| 739 | int 0x16
|
---|
| 740 | jz escflg ! Keypress?
|
---|
| 741 | cmpb al, ESC ! Escape typed?
|
---|
| 742 | jne escflg
|
---|
| 743 | xorb ah, ah ! Discard the escape
|
---|
| 744 | int 0x16
|
---|
| 745 | inc (escape) ! Set flag
|
---|
| 746 | escflg: xor ax, ax
|
---|
| 747 | xchg ax, (escape) ! Escape typed flag
|
---|
| 748 | ret
|
---|
| 749 |
|
---|
| 750 | ! int putch(int c);
|
---|
| 751 | ! Write a character in teletype mode. The putk synonym is
|
---|
| 752 | ! for the kernel printf function that uses it.
|
---|
| 753 | ! Newlines are automatically preceded by a carriage return.
|
---|
| 754 | !
|
---|
| 755 | .define _putch, _putk
|
---|
| 756 | _putch:
|
---|
| 757 | _putk: mov bx, sp
|
---|
| 758 | movb al, 2(bx) ! al = character to be printed
|
---|
| 759 | testb al, al ! Kernel printf adds a null char to flush queue
|
---|
| 760 | jz nulch
|
---|
| 761 | cmpb al, 0x0A ! al = newline?
|
---|
| 762 | jnz putc
|
---|
| 763 | movb al, 0x20 ! Erase wheel and do a carriage return
|
---|
| 764 | call plotc ! plotc(' ');
|
---|
| 765 | nodirt: movb al, 0x0D
|
---|
| 766 | call putc ! putc('\r')
|
---|
| 767 | movb al, 0x0A ! Restore the '\n' and print it
|
---|
| 768 | putc: movb ah, 0x0E ! Print character in teletype mode
|
---|
| 769 | mov bx, 0x0001 ! Page 0, foreground color
|
---|
| 770 | int 0x10 ! Call BIOS VIDEO_IO
|
---|
| 771 | nulch: ret
|
---|
| 772 |
|
---|
| 773 | ! |/-\|/-\|/-\|/-\|/-\ (playtime)
|
---|
| 774 | wheel: mov bx, (gp)
|
---|
| 775 | movb al, (bx)
|
---|
| 776 | inc bx ! al = *gp++;
|
---|
| 777 | cmp bx, glyphs+4
|
---|
| 778 | jne 0f
|
---|
| 779 | mov bx, glyphs
|
---|
| 780 | 0: mov (gp), bx ! gp= gp == glyphs + 4 ? glyphs : gp;
|
---|
| 781 | !jmp plotc
|
---|
| 782 | plotc: movb ah, 0x0A ! 0x0A = write character at cursor
|
---|
| 783 | mov bx, 0x0001 ! Page 0, foreground color
|
---|
| 784 | mov cx, 0x0001 ! Just one character
|
---|
| 785 | int 0x10
|
---|
| 786 | ret
|
---|
| 787 | .sect .data
|
---|
| 788 | .align 2
|
---|
| 789 | gp: .data2 glyphs
|
---|
| 790 | glyphs: .ascii "|/-\\"
|
---|
| 791 | .sect .text
|
---|
| 792 |
|
---|
| 793 | ! void pause(void);
|
---|
| 794 | ! Wait for an interrupt using the HLT instruction. This either saves
|
---|
| 795 | ! power, or tells an x86 emulator that nothing is happening right now.
|
---|
| 796 | .define _pause
|
---|
| 797 | _pause:
|
---|
| 798 | hlt
|
---|
| 799 | ret
|
---|
| 800 |
|
---|
| 801 | ! void set_mode(unsigned mode);
|
---|
| 802 | ! void clear_screen(void);
|
---|
| 803 | ! Set video mode / clear the screen.
|
---|
| 804 | .define _set_mode, _clear_screen
|
---|
| 805 | _set_mode:
|
---|
| 806 | mov bx, sp
|
---|
| 807 | mov ax, 2(bx) ! Video mode
|
---|
| 808 | cmp ax, (cur_vid_mode)
|
---|
| 809 | je modeok ! Mode already as requested?
|
---|
| 810 | mov (cur_vid_mode), ax
|
---|
| 811 | _clear_screen:
|
---|
| 812 | mov ax, (cur_vid_mode)
|
---|
| 813 | andb ah, 0x7F ! Test bits 8-14, clear bit 15 (8x8 flag)
|
---|
| 814 | jnz xvesa ! VESA extended mode?
|
---|
| 815 | int 0x10 ! Reset video (ah = 0)
|
---|
| 816 | jmp mdset
|
---|
| 817 | xvesa: mov bx, ax ! bx = extended mode
|
---|
| 818 | mov ax, 0x4F02 ! Reset video
|
---|
| 819 | int 0x10
|
---|
| 820 | mdset: testb (cur_vid_mode+1), 0x80
|
---|
| 821 | jz setcur ! 8x8 font requested?
|
---|
| 822 | mov ax, 0x1112 ! Load ROM 8 by 8 double-dot patterns
|
---|
| 823 | xorb bl, bl ! Load block 0
|
---|
| 824 | int 0x10
|
---|
| 825 | setcur: xor dx, dx ! dl = column = 0, dh = row = 0
|
---|
| 826 | xorb bh, bh ! Page 0
|
---|
| 827 | movb ah, 0x02 ! Set cursor position
|
---|
| 828 | int 0x10
|
---|
| 829 | modeok: ret
|
---|
| 830 |
|
---|
| 831 | restore_video: ! To restore the video mode on exit
|
---|
| 832 | movb al, 0x20
|
---|
| 833 | call plotc ! Erase wheel
|
---|
| 834 | push (old_vid_mode)
|
---|
| 835 | call _set_mode
|
---|
| 836 | pop ax
|
---|
| 837 | ret
|
---|
| 838 |
|
---|
| 839 | ! u32_t get_tick(void);
|
---|
| 840 | ! Return the current value of the clock tick counter. This counter
|
---|
| 841 | ! increments 18.2 times per second. Poll it to do delays. Does not
|
---|
| 842 | ! work on the original PC, but works on the PC/XT.
|
---|
| 843 | .define _get_tick
|
---|
| 844 | _get_tick:
|
---|
| 845 | xorb ah, ah ! Code for get tick count
|
---|
| 846 | int 0x1A
|
---|
| 847 | mov ax, dx
|
---|
| 848 | mov dx, cx ! dx:ax = cx:dx = tick count
|
---|
| 849 | ret
|
---|
| 850 |
|
---|
| 851 |
|
---|
| 852 | ! Functions used to obtain info about the hardware. Boot uses this information
|
---|
| 853 | ! itself, but will also pass them on to a pure 386 kernel, because one can't
|
---|
| 854 | ! make BIOS calls from protected mode. The video type could probably be
|
---|
| 855 | ! determined by the kernel too by looking at the hardware, but there is a small
|
---|
| 856 | ! chance on errors that the monitor allows you to correct by setting variables.
|
---|
| 857 |
|
---|
| 858 | .define _get_bus ! returns type of system bus
|
---|
| 859 | .define _get_video ! returns type of display
|
---|
| 860 |
|
---|
| 861 | ! u16_t get_bus(void)
|
---|
| 862 | ! Return type of system bus, in order: XT, AT, MCA.
|
---|
| 863 | _get_bus:
|
---|
| 864 | call gettrueproc
|
---|
| 865 | xor dx, dx ! Assume XT
|
---|
| 866 | cmp ax, 286 ! An AT has at least a 286
|
---|
| 867 | jb got_bus
|
---|
| 868 | inc dx ! Assume AT
|
---|
| 869 | movb ah, 0xC0 ! Code for get configuration
|
---|
| 870 | int 0x15
|
---|
| 871 | jc got_bus ! Carry clear and ah = 00 if supported
|
---|
| 872 | testb ah, ah
|
---|
| 873 | jne got_bus
|
---|
| 874 | eseg movb al, 5(bx) ! Load feature byte #1
|
---|
| 875 | inc dx ! Assume MCA
|
---|
| 876 | testb al, 0x02 ! Test bit 1 - "bus is Micro Channel"
|
---|
| 877 | jnz got_bus
|
---|
| 878 | dec dx ! Assume AT
|
---|
| 879 | testb al, 0x40 ! Test bit 6 - "2nd 8259 installed"
|
---|
| 880 | jnz got_bus
|
---|
| 881 | dec dx ! It is an XT
|
---|
| 882 | got_bus:
|
---|
| 883 | push ds
|
---|
| 884 | pop es ! Restore es
|
---|
| 885 | mov ax, dx ! Return bus code
|
---|
| 886 | mov (bus), ax ! Keep bus code, A20 handler likes to know
|
---|
| 887 | ret
|
---|
| 888 |
|
---|
| 889 | ! u16_t get_video(void)
|
---|
| 890 | ! Return type of display, in order: MDA, CGA, mono EGA, color EGA,
|
---|
| 891 | ! mono VGA, color VGA.
|
---|
| 892 | _get_video:
|
---|
| 893 | mov ax, 0x1A00 ! Function 1A returns display code
|
---|
| 894 | int 0x10 ! al = 1A if supported
|
---|
| 895 | cmpb al, 0x1A
|
---|
| 896 | jnz no_dc ! No display code function supported
|
---|
| 897 |
|
---|
| 898 | mov ax, 2
|
---|
| 899 | cmpb bl, 5 ! Is it a monochrome EGA?
|
---|
| 900 | jz got_video
|
---|
| 901 | inc ax
|
---|
| 902 | cmpb bl, 4 ! Is it a color EGA?
|
---|
| 903 | jz got_video
|
---|
| 904 | inc ax
|
---|
| 905 | cmpb bl, 7 ! Is it a monochrome VGA?
|
---|
| 906 | jz got_video
|
---|
| 907 | inc ax
|
---|
| 908 | cmpb bl, 8 ! Is it a color VGA?
|
---|
| 909 | jz got_video
|
---|
| 910 |
|
---|
| 911 | no_dc: movb ah, 0x12 ! Get information about the EGA
|
---|
| 912 | movb bl, 0x10
|
---|
| 913 | int 0x10
|
---|
| 914 | cmpb bl, 0x10 ! Did it come back as 0x10? (No EGA)
|
---|
| 915 | jz no_ega
|
---|
| 916 |
|
---|
| 917 | mov ax, 2
|
---|
| 918 | cmpb bh, 1 ! Is it monochrome?
|
---|
| 919 | jz got_video
|
---|
| 920 | inc ax
|
---|
| 921 | jmp got_video
|
---|
| 922 |
|
---|
| 923 | no_ega: int 0x11 ! Get bit pattern for equipment
|
---|
| 924 | and ax, 0x30 ! Isolate color/mono field
|
---|
| 925 | sub ax, 0x30
|
---|
| 926 | jz got_video ! Is it an MDA?
|
---|
| 927 | mov ax, 1 ! No it's CGA
|
---|
| 928 |
|
---|
| 929 | got_video:
|
---|
| 930 | ret
|
---|
| 931 |
|
---|
| 932 |
|
---|
| 933 | ! Function to leave the boot monitor and run Minix.
|
---|
| 934 | .define _minix
|
---|
| 935 |
|
---|
| 936 | ! void minix(u32_t koff, u32_t kcs, u32_t kds,
|
---|
| 937 | ! char *bootparams, size_t paramsize, u32_t aout);
|
---|
| 938 | _minix:
|
---|
| 939 | push bp
|
---|
| 940 | mov bp, sp ! Pointer to arguments
|
---|
| 941 |
|
---|
| 942 | mov dx, 0x03F2 ! Floppy motor drive control bits
|
---|
| 943 | movb al, 0x0C ! Bits 4-7 for floppy 0-3 are off
|
---|
| 944 | outb dx ! Kill the motors
|
---|
| 945 | push ds
|
---|
| 946 | xor ax, ax ! Vector & BIOS data segments
|
---|
| 947 | mov ds, ax
|
---|
| 948 | andb (0x043F), 0xF0 ! Clear diskette motor status bits of BIOS
|
---|
| 949 | pop ds
|
---|
| 950 | cli ! No more interruptions
|
---|
| 951 |
|
---|
| 952 | test (_k_flags), K_I386 ! Minix-386?
|
---|
| 953 | jnz minix386
|
---|
| 954 |
|
---|
| 955 | ! Call Minix in real mode.
|
---|
| 956 | minix86:
|
---|
| 957 | push 22(bp) ! Address of a.out headers
|
---|
| 958 | push 20(bp)
|
---|
| 959 |
|
---|
| 960 | push 18(bp) ! # bytes of boot parameters
|
---|
| 961 | push 16(bp) ! Address of boot parameters
|
---|
| 962 |
|
---|
| 963 | mov dx, cs ! Monitor far return address
|
---|
| 964 | mov ax, ret86
|
---|
| 965 | cmp (_mem+14), 0 ! Any extended memory? (mem[1].size > 0 ?)
|
---|
| 966 | jnz 0f
|
---|
| 967 | xor dx, dx ! If no ext mem then monitor not preserved
|
---|
| 968 | xor ax, ax
|
---|
| 969 | 0: push dx ! Push monitor far return address or zero
|
---|
| 970 | push ax
|
---|
| 971 |
|
---|
| 972 | mov ax, 8(bp)
|
---|
| 973 | mov dx, 10(bp)
|
---|
| 974 | call abs2seg
|
---|
| 975 | push dx ! Kernel code segment
|
---|
| 976 | push 4(bp) ! Kernel code offset
|
---|
| 977 | mov ax, 12(bp)
|
---|
| 978 | mov dx, 14(bp)
|
---|
| 979 | call abs2seg
|
---|
| 980 | mov ds, dx ! Kernel data segment
|
---|
| 981 | mov es, dx ! Set es to kernel data too
|
---|
| 982 | retf ! Make a far call to the kernel
|
---|
| 983 |
|
---|
| 984 | ! Call 386 Minix in 386 mode.
|
---|
| 985 | minix386:
|
---|
| 986 | cseg mov (cs_real-2), cs ! Patch CS and DS into the instructions that
|
---|
| 987 | cseg mov (ds_real-2), ds ! reload them when switching back to real mode
|
---|
| 988 | mov eax, cr0
|
---|
| 989 | orb al, 0x01 ! Set PE (protection enable) bit
|
---|
| 990 | o32 mov (msw), eax ! Save as protected mode machine status word
|
---|
| 991 |
|
---|
| 992 | mov dx, ds ! Monitor ds
|
---|
| 993 | mov ax, p_gdt ! dx:ax = Global descriptor table
|
---|
| 994 | call seg2abs
|
---|
| 995 | mov (p_gdt_desc+2), ax
|
---|
| 996 | movb (p_gdt_desc+4), dl ! Set base of global descriptor table
|
---|
| 997 |
|
---|
| 998 | mov ax, 12(bp)
|
---|
| 999 | mov dx, 14(bp) ! Kernel ds (absolute address)
|
---|
| 1000 | mov (p_ds_desc+2), ax
|
---|
| 1001 | movb (p_ds_desc+4), dl ! Set base of kernel data segment
|
---|
| 1002 |
|
---|
| 1003 | mov dx, ss ! Monitor ss
|
---|
| 1004 | xor ax, ax ! dx:ax = Monitor stack segment
|
---|
| 1005 | call seg2abs ! Minix starts with the stack of the monitor
|
---|
| 1006 | mov (p_ss_desc+2), ax
|
---|
| 1007 | movb (p_ss_desc+4), dl
|
---|
| 1008 |
|
---|
| 1009 | mov ax, 8(bp)
|
---|
| 1010 | mov dx, 10(bp) ! Kernel cs (absolute address)
|
---|
| 1011 | mov (p_cs_desc+2), ax
|
---|
| 1012 | movb (p_cs_desc+4), dl
|
---|
| 1013 |
|
---|
| 1014 | mov dx, cs ! Monitor cs
|
---|
| 1015 | xor ax, ax ! dx:ax = Monitor code segment
|
---|
| 1016 | call seg2abs
|
---|
| 1017 | mov (p_mcs_desc+2), ax
|
---|
| 1018 | movb (p_mcs_desc+4), dl
|
---|
| 1019 |
|
---|
| 1020 | push MCS_SELECTOR
|
---|
| 1021 | push int86 ! Far address to INT86 support
|
---|
| 1022 |
|
---|
| 1023 | o32 push 20(bp) ! Address of a.out headers
|
---|
| 1024 |
|
---|
| 1025 | push 0
|
---|
| 1026 | push 18(bp) ! 32 bit size of parameters on stack
|
---|
| 1027 | push 0
|
---|
| 1028 | push 16(bp) ! 32 bit address of parameters (ss relative)
|
---|
| 1029 |
|
---|
| 1030 | push MCS_SELECTOR
|
---|
| 1031 | push ret386 ! Monitor far return address
|
---|
| 1032 |
|
---|
| 1033 | push 0
|
---|
| 1034 | push CS_SELECTOR
|
---|
| 1035 | push 6(bp)
|
---|
| 1036 | push 4(bp) ! 32 bit far address to kernel entry point
|
---|
| 1037 |
|
---|
| 1038 | call real2prot ! Switch to protected mode
|
---|
| 1039 | mov ax, DS_SELECTOR
|
---|
| 1040 | mov ds, ax ! Kernel data
|
---|
| 1041 | mov ax, ES_SELECTOR
|
---|
| 1042 | mov es, ax ! Flat 4 Gb
|
---|
| 1043 | o32 retf ! Make a far call to the kernel
|
---|
| 1044 |
|
---|
| 1045 | ! Minix-86 returns here on a halt or reboot.
|
---|
| 1046 | ret86:
|
---|
| 1047 | mov 8(bp), ax
|
---|
| 1048 | mov 10(bp), dx ! Return value
|
---|
| 1049 | jmp return
|
---|
| 1050 |
|
---|
| 1051 | ! Minix-386 returns here on a halt or reboot.
|
---|
| 1052 | ret386:
|
---|
| 1053 | o32 mov 8(bp), eax ! Return value
|
---|
| 1054 | call prot2real ! Switch to real mode
|
---|
| 1055 |
|
---|
| 1056 | return:
|
---|
| 1057 | mov sp, bp ! Pop parameters
|
---|
| 1058 | sti ! Can take interrupts again
|
---|
| 1059 |
|
---|
| 1060 | call _get_video ! MDA, CGA, EGA, ...
|
---|
| 1061 | movb dh, 24 ! dh = row 24
|
---|
| 1062 | cmp ax, 2 ! At least EGA?
|
---|
| 1063 | jb is25 ! Otherwise 25 rows
|
---|
| 1064 | push ds
|
---|
| 1065 | xor ax, ax ! Vector & BIOS data segments
|
---|
| 1066 | mov ds, ax
|
---|
| 1067 | movb dh, (0x0484) ! Number of rows on display minus one
|
---|
| 1068 | pop ds
|
---|
| 1069 | is25:
|
---|
| 1070 | xorb dl, dl ! dl = column 0
|
---|
| 1071 | xorb bh, bh ! Page 0
|
---|
| 1072 | movb ah, 0x02 ! Set cursor position
|
---|
| 1073 | int 0x10
|
---|
| 1074 |
|
---|
| 1075 | xorb ah, ah ! Whack the disk system, Minix may have messed
|
---|
| 1076 | movb dl, 0x80 ! it up
|
---|
| 1077 | int 0x13
|
---|
| 1078 |
|
---|
| 1079 | call gettrueproc
|
---|
| 1080 | cmp ax, 286
|
---|
| 1081 | jb noclock
|
---|
| 1082 | xorb al, al
|
---|
| 1083 | tryclk: decb al
|
---|
| 1084 | jz noclock
|
---|
| 1085 | movb ah, 0x02 ! Get real-time clock time (from CMOS clock)
|
---|
| 1086 | int 0x1A
|
---|
| 1087 | jc tryclk ! Carry set, not running or being updated
|
---|
| 1088 | movb al, ch ! ch = hour in BCD
|
---|
| 1089 | call bcd ! al = (al >> 4) * 10 + (al & 0x0F)
|
---|
| 1090 | mulb (c60) ! 60 minutes in an hour
|
---|
| 1091 | mov bx, ax ! bx = hour * 60
|
---|
| 1092 | movb al, cl ! cl = minutes in BCD
|
---|
| 1093 | call bcd
|
---|
| 1094 | add bx, ax ! bx = hour * 60 + minutes
|
---|
| 1095 | movb al, dh ! dh = seconds in BCD
|
---|
| 1096 | call bcd
|
---|
| 1097 | xchg ax, bx ! ax = hour * 60 + minutes, bx = seconds
|
---|
| 1098 | mul (c60) ! dx-ax = (hour * 60 + minutes) * 60
|
---|
| 1099 | add bx, ax
|
---|
| 1100 | adc dx, 0 ! dx-bx = seconds since midnight
|
---|
| 1101 | mov ax, dx
|
---|
| 1102 | mul (c19663)
|
---|
| 1103 | xchg ax, bx
|
---|
| 1104 | mul (c19663)
|
---|
| 1105 | add dx, bx ! dx-ax = dx-bx * (0x1800B0 / (2*2*2*2*5))
|
---|
| 1106 | mov cx, ax ! (0x1800B0 = ticks per day of BIOS clock)
|
---|
| 1107 | mov ax, dx
|
---|
| 1108 | xor dx, dx
|
---|
| 1109 | div (c1080)
|
---|
| 1110 | xchg ax, cx
|
---|
| 1111 | div (c1080) ! cx-ax = dx-ax / (24*60*60 / (2*2*2*2*5))
|
---|
| 1112 | mov dx, ax ! cx-dx = ticks since midnight
|
---|
| 1113 | movb ah, 0x01 ! Set system time
|
---|
| 1114 | int 0x1A
|
---|
| 1115 | noclock:
|
---|
| 1116 |
|
---|
| 1117 | mov ax, 8(bp)
|
---|
| 1118 | mov dx, 10(bp) ! dx-ax = return value from the kernel
|
---|
| 1119 | pop bp
|
---|
| 1120 | ret ! Return to monitor as if nothing much happened
|
---|
| 1121 |
|
---|
| 1122 | ! Transform BCD number in al to a regular value in ax.
|
---|
| 1123 | bcd: movb ah, al
|
---|
| 1124 | shrb ah, 4
|
---|
| 1125 | andb al, 0x0F
|
---|
| 1126 | aad ! ax = (al >> 4) * 10 + (al & 0x0F)
|
---|
| 1127 | ret
|
---|
| 1128 |
|
---|
| 1129 | ! Support function for Minix-386 to make an 8086 interrupt call.
|
---|
| 1130 | int86:
|
---|
| 1131 | mov bp, sp
|
---|
| 1132 | call prot2real
|
---|
| 1133 |
|
---|
| 1134 | o32 xor eax, eax
|
---|
| 1135 | mov es, ax ! Vector & BIOS data segments
|
---|
| 1136 | o32 eseg mov (0x046C), eax ! Clear BIOS clock tick counter
|
---|
| 1137 |
|
---|
| 1138 | sti ! Enable interrupts
|
---|
| 1139 |
|
---|
| 1140 | movb al, 0xCD ! INT instruction
|
---|
| 1141 | movb ah, 8(bp) ! Interrupt number?
|
---|
| 1142 | testb ah, ah
|
---|
| 1143 | jnz 0f ! Nonzero if INT, otherwise far call
|
---|
| 1144 | push cs
|
---|
| 1145 | push intret+2 ! Far return address
|
---|
| 1146 | o32 push 12(bp) ! Far driver address
|
---|
| 1147 | mov ax, 0x90CB ! RETF; NOP
|
---|
| 1148 | 0: cseg mov (intret), ax ! Patch 'INT n' or 'RETF; NOP' into code
|
---|
| 1149 |
|
---|
| 1150 | mov ds, 16(bp) ! Load parameters
|
---|
| 1151 | mov es, 18(bp)
|
---|
| 1152 | o32 mov eax, 20(bp)
|
---|
| 1153 | o32 mov ebx, 24(bp)
|
---|
| 1154 | o32 mov ecx, 28(bp)
|
---|
| 1155 | o32 mov edx, 32(bp)
|
---|
| 1156 | o32 mov esi, 36(bp)
|
---|
| 1157 | o32 mov edi, 40(bp)
|
---|
| 1158 | o32 mov ebp, 44(bp)
|
---|
| 1159 |
|
---|
| 1160 | intret: int 0xFF ! Do the interrupt or far call
|
---|
| 1161 |
|
---|
| 1162 | o32 push ebp ! Save results
|
---|
| 1163 | o32 pushf
|
---|
| 1164 | mov bp, sp
|
---|
| 1165 | o32 pop 8+8(bp) ! eflags
|
---|
| 1166 | mov 8+16(bp), ds
|
---|
| 1167 | mov 8+18(bp), es
|
---|
| 1168 | o32 mov 8+20(bp), eax
|
---|
| 1169 | o32 mov 8+24(bp), ebx
|
---|
| 1170 | o32 mov 8+28(bp), ecx
|
---|
| 1171 | o32 mov 8+32(bp), edx
|
---|
| 1172 | o32 mov 8+36(bp), esi
|
---|
| 1173 | o32 mov 8+40(bp), edi
|
---|
| 1174 | o32 pop 8+44(bp) ! ebp
|
---|
| 1175 |
|
---|
| 1176 | cli ! Disable interrupts
|
---|
| 1177 |
|
---|
| 1178 | xor ax, ax
|
---|
| 1179 | mov ds, ax ! Vector & BIOS data segments
|
---|
| 1180 | o32 mov cx, (0x046C) ! Collect lost clock ticks in ecx
|
---|
| 1181 |
|
---|
| 1182 | mov ax, ss
|
---|
| 1183 | mov ds, ax ! Restore monitor ds
|
---|
| 1184 | call real2prot
|
---|
| 1185 | mov ax, DS_SELECTOR ! Kernel data
|
---|
| 1186 | mov ds, ax
|
---|
| 1187 | o32 retf ! Return to the kernel
|
---|
| 1188 |
|
---|
| 1189 | ! Switch from real to protected mode.
|
---|
| 1190 | real2prot:
|
---|
| 1191 | movb ah, 0x02 ! Code for A20 enable
|
---|
| 1192 | call gate_A20
|
---|
| 1193 |
|
---|
| 1194 | lgdt (p_gdt_desc) ! Global descriptor table
|
---|
| 1195 | o32 mov eax, (pdbr) ! Load page directory base register
|
---|
| 1196 | mov cr3, eax
|
---|
| 1197 | mov eax, cr0
|
---|
| 1198 | o32 xchg eax, (msw) ! Exchange real mode msw for protected mode msw
|
---|
| 1199 | mov cr0, eax
|
---|
| 1200 | jmpf MCS_SELECTOR:cs_prot ! Set code segment selector
|
---|
| 1201 | cs_prot:
|
---|
| 1202 | mov ax, SS_SELECTOR ! Set data selectors
|
---|
| 1203 | mov ds, ax
|
---|
| 1204 | mov es, ax
|
---|
| 1205 | mov ss, ax
|
---|
| 1206 | ret
|
---|
| 1207 |
|
---|
| 1208 | ! Switch from protected to real mode.
|
---|
| 1209 | prot2real:
|
---|
| 1210 | lidt (p_idt_desc) ! Real mode interrupt vectors
|
---|
| 1211 | mov eax, cr3
|
---|
| 1212 | o32 mov (pdbr), eax ! Save page directory base register
|
---|
| 1213 | mov eax, cr0
|
---|
| 1214 | o32 xchg eax, (msw) ! Exchange protected mode msw for real mode msw
|
---|
| 1215 | mov cr0, eax
|
---|
| 1216 | jmpf 0xDEAD:cs_real ! Reload cs register
|
---|
| 1217 | cs_real:
|
---|
| 1218 | mov ax, 0xBEEF
|
---|
| 1219 | ds_real:
|
---|
| 1220 | mov ds, ax ! Reload data segment registers
|
---|
| 1221 | mov es, ax
|
---|
| 1222 | mov ss, ax
|
---|
| 1223 |
|
---|
| 1224 | xorb ah, ah ! Code for A20 disable
|
---|
| 1225 | !jmp gate_A20
|
---|
| 1226 |
|
---|
| 1227 | ! Enable (ah = 0x02) or disable (ah = 0x00) the A20 address line.
|
---|
| 1228 | gate_A20:
|
---|
| 1229 | cmp (bus), 2 ! PS/2 bus?
|
---|
| 1230 | je gate_PS_A20
|
---|
| 1231 | call kb_wait
|
---|
| 1232 | movb al, 0xD1 ! Tell keyboard that a command is coming
|
---|
| 1233 | outb 0x64
|
---|
| 1234 | call kb_wait
|
---|
| 1235 | movb al, 0xDD ! 0xDD = A20 disable code if ah = 0x00
|
---|
| 1236 | orb al, ah ! 0xDF = A20 enable code if ah = 0x02
|
---|
| 1237 | outb 0x60
|
---|
| 1238 | call kb_wait
|
---|
| 1239 | movb al, 0xFF ! Pulse output port
|
---|
| 1240 | outb 0x64
|
---|
| 1241 | call kb_wait ! Wait for the A20 line to settle down
|
---|
| 1242 | ret
|
---|
| 1243 | kb_wait:
|
---|
| 1244 | inb 0x64
|
---|
| 1245 | testb al, 0x02 ! Keyboard input buffer full?
|
---|
| 1246 | jnz kb_wait ! If so, wait
|
---|
| 1247 | ret
|
---|
| 1248 |
|
---|
| 1249 | gate_PS_A20: ! The PS/2 can twiddle A20 using port A
|
---|
| 1250 | inb 0x92 ! Read port A
|
---|
| 1251 | andb al, 0xFD
|
---|
| 1252 | orb al, ah ! Set A20 bit to the required state
|
---|
| 1253 | outb 0x92 ! Write port A
|
---|
| 1254 | jmp .+2 ! Small delay
|
---|
| 1255 | A20ok: inb 0x92 ! Check port A
|
---|
| 1256 | andb al, 0x02
|
---|
| 1257 | cmpb al, ah ! A20 line settled down to the new state?
|
---|
| 1258 | jne A20ok ! If not then wait
|
---|
| 1259 | ret
|
---|
| 1260 |
|
---|
| 1261 | ! void int15(bios_env_t *ep)
|
---|
| 1262 | ! Do an "INT 15" call, primarily for APM (Power Management).
|
---|
| 1263 | .define _int15
|
---|
| 1264 | _int15:
|
---|
| 1265 | push si ! Save callee-save register si
|
---|
| 1266 | mov si, sp
|
---|
| 1267 | mov si, 4(si) ! ep
|
---|
| 1268 | mov ax, (si) ! ep->ax
|
---|
| 1269 | mov bx, 2(si) ! ep->bx
|
---|
| 1270 | mov cx, 4(si) ! ep->cx
|
---|
| 1271 | int 0x15 ! INT 0x15 BIOS call
|
---|
| 1272 | pushf ! Save flags
|
---|
| 1273 | mov (si), ax ! ep->ax
|
---|
| 1274 | mov 2(si), bx ! ep->bx
|
---|
| 1275 | mov 4(si), cx ! ep->cx
|
---|
| 1276 | pop 6(si) ! ep->flags
|
---|
| 1277 | pop si ! Restore
|
---|
| 1278 | ret
|
---|
| 1279 |
|
---|
| 1280 | .sect .rom
|
---|
| 1281 | .align 4
|
---|
| 1282 | c60: .data2 60 ! Constants for MUL and DIV
|
---|
| 1283 | c1024: .data2 1024
|
---|
| 1284 | c1080: .data2 1080
|
---|
| 1285 | c19663: .data2 19663
|
---|
| 1286 |
|
---|
| 1287 | .sect .data
|
---|
| 1288 | .align 4
|
---|
| 1289 |
|
---|
| 1290 | ! Global descriptor tables.
|
---|
| 1291 | UNSET = 0 ! Must be computed
|
---|
| 1292 |
|
---|
| 1293 | ! For "Extended Memory Block Move".
|
---|
| 1294 | x_gdt:
|
---|
| 1295 | x_null_desc:
|
---|
| 1296 | ! Null descriptor
|
---|
| 1297 | .data2 0x0000, 0x0000
|
---|
| 1298 | .data1 0x00, 0x00, 0x00, 0x00
|
---|
| 1299 | x_gdt_desc:
|
---|
| 1300 | ! Descriptor for this descriptor table
|
---|
| 1301 | .data2 6*8-1, UNSET
|
---|
| 1302 | .data1 UNSET, 0x00, 0x00, 0x00
|
---|
| 1303 | x_src_desc:
|
---|
| 1304 | ! Source segment descriptor
|
---|
| 1305 | .data2 0xFFFF, UNSET
|
---|
| 1306 | .data1 UNSET, 0x92, 0x00, 0x00
|
---|
| 1307 | x_dst_desc:
|
---|
| 1308 | ! Destination segment descriptor
|
---|
| 1309 | .data2 0xFFFF, UNSET
|
---|
| 1310 | .data1 UNSET, 0x92, 0x00, 0x00
|
---|
| 1311 | x_bios_desc:
|
---|
| 1312 | ! BIOS segment descriptor (scratch for int 0x15)
|
---|
| 1313 | .data2 UNSET, UNSET
|
---|
| 1314 | .data1 UNSET, UNSET, UNSET, UNSET
|
---|
| 1315 | x_ss_desc:
|
---|
| 1316 | ! BIOS stack segment descriptor (scratch for int 0x15)
|
---|
| 1317 | .data2 UNSET, UNSET
|
---|
| 1318 | .data1 UNSET, UNSET, UNSET, UNSET
|
---|
| 1319 |
|
---|
| 1320 | ! Protected mode descriptor table.
|
---|
| 1321 | p_gdt:
|
---|
| 1322 | p_null_desc:
|
---|
| 1323 | ! Null descriptor
|
---|
| 1324 | .data2 0x0000, 0x0000
|
---|
| 1325 | .data1 0x00, 0x00, 0x00, 0x00
|
---|
| 1326 | p_gdt_desc:
|
---|
| 1327 | ! Descriptor for this descriptor table
|
---|
| 1328 | .data2 8*8-1, UNSET
|
---|
| 1329 | .data1 UNSET, 0x00, 0x00, 0x00
|
---|
| 1330 | p_idt_desc:
|
---|
| 1331 | ! Real mode interrupt descriptor table descriptor
|
---|
| 1332 | .data2 0x03FF, 0x0000
|
---|
| 1333 | .data1 0x00, 0x00, 0x00, 0x00
|
---|
| 1334 | p_ds_desc:
|
---|
| 1335 | ! Kernel data segment descriptor (4Gb flat)
|
---|
| 1336 | .data2 0xFFFF, UNSET
|
---|
| 1337 | .data1 UNSET, 0x92, 0xCF, 0x00
|
---|
| 1338 | p_es_desc:
|
---|
| 1339 | ! Physical memory descriptor (4Gb flat)
|
---|
| 1340 | .data2 0xFFFF, 0x0000
|
---|
| 1341 | .data1 0x00, 0x92, 0xCF, 0x00
|
---|
| 1342 | p_ss_desc:
|
---|
| 1343 | ! Monitor data segment descriptor (64Kb flat)
|
---|
| 1344 | .data2 0xFFFF, UNSET
|
---|
| 1345 | .data1 UNSET, 0x92, 0x00, 0x00
|
---|
| 1346 | p_cs_desc:
|
---|
| 1347 | ! Kernel code segment descriptor (4Gb flat)
|
---|
| 1348 | .data2 0xFFFF, UNSET
|
---|
| 1349 | .data1 UNSET, 0x9A, 0xCF, 0x00
|
---|
| 1350 | p_mcs_desc:
|
---|
| 1351 | ! Monitor code segment descriptor (64 kb flat) (unused)
|
---|
| 1352 | .data2 0xFFFF, UNSET
|
---|
| 1353 | .data1 UNSET, 0x9A, 0x00, 0x00
|
---|
| 1354 |
|
---|
| 1355 | xms_handle: .data2 -1 ! Handle of allocated XMS block
|
---|
| 1356 | vfd: .data2 -1 ! Virtual disk file handle
|
---|
| 1357 |
|
---|
| 1358 | .sect .bss
|
---|
| 1359 | .comm xms_driver, 4 ! Vector to XMS driver
|
---|
| 1360 | .comm old_vid_mode, 2 ! Video mode at startup
|
---|
| 1361 | .comm cur_vid_mode, 2 ! Current video mode
|
---|
| 1362 | .comm msw, 4 ! Saved machine status word (cr0)
|
---|
| 1363 | .comm pdbr, 4 ! Saved page directory base register (cr3)
|
---|
| 1364 | .comm escape, 2 ! Escape typed?
|
---|
| 1365 | .comm bus, 2 ! Saved return value of _get_bus
|
---|
| 1366 | .comm unchar, 2 ! Char returned by ungetch(c)
|
---|
| 1367 |
|
---|
| 1368 | !
|
---|
| 1369 | ! $PchId: doshead.ack.s,v 1.7 2002/02/27 19:37:52 philip Exp $
|
---|