mirror of https://github.com/microsoft/MS-DOS.git
587 lines
22 KiB
NASM
587 lines
22 KiB
NASM
|
TITLE MS-DOS SYS Program
|
|||
|
; SYS - Copies system programs IBMBIO.COM/IO.SYS and IBMDOS.COM/MSDOS.SYS
|
|||
|
; 1.6 05/21/82 Added rev number message
|
|||
|
; 1.61 06/04/82 Allow SYS to blank disk TimP at SCP
|
|||
|
; 1.70 06/30/82 NON contiguous DOS allowed on 2.00 IBM. Allows SYS to
|
|||
|
; 1.0 1.1 disks.
|
|||
|
; 1.71 07/02/82 Put in CHDIRs to make sure everything done in root dir.
|
|||
|
; 1.80 04/26/83 MZ make sys work in small machines; use full 2.0 system
|
|||
|
; calls
|
|||
|
; 1.81 07/22/83 ARR Added check in IBM version for valid FAT ID on
|
|||
|
; destination because of IBM problem with SYSing to
|
|||
|
; unformatted disks which are really formatted.
|
|||
|
; Prints NoDest message for ridic IBM reasons, should
|
|||
|
; have a better message.
|
|||
|
|
|||
|
FALSE EQU 0
|
|||
|
TRUE EQU NOT FALSE
|
|||
|
|
|||
|
IBMJAPVER EQU FALSE
|
|||
|
IBMVER EQU FALSE
|
|||
|
MSVER EQU TRUE
|
|||
|
|
|||
|
.xlist
|
|||
|
.xcref
|
|||
|
INCLUDE DOSSYM.ASM
|
|||
|
.cref
|
|||
|
.list
|
|||
|
|
|||
|
|
|||
|
DOSVER_LOW EQU 0136H ; Lowest acceptable DOS version number
|
|||
|
DOSVER_HIGH EQU 020BH ; Highest acceptable DOS version
|
|||
|
|
|||
|
CODE SEGMENT WORD PUBLIC
|
|||
|
CODE ENDS
|
|||
|
|
|||
|
CONST SEGMENT BYTE PUBLIC
|
|||
|
CONST ENDS
|
|||
|
|
|||
|
DATA SEGMENT BYTE PUBLIC
|
|||
|
DATA ENDS
|
|||
|
|
|||
|
DG GROUP CODE,DATA,CONST
|
|||
|
|
|||
|
DATA SEGMENT PUBLIC BYTE
|
|||
|
|
|||
|
EXTRN BADDRV:BYTE, BADDRVLen:WORD
|
|||
|
EXTRN BADPARM:BYTE, BADPARMLen:WORD
|
|||
|
EXTRN GETSYS:BYTE, GETSYSLen:WORD
|
|||
|
EXTRN SYSDRV:BYTE
|
|||
|
EXTRN NODEST:BYTE, NODESTLen:WORD
|
|||
|
EXTRN BADSIZ:BYTE, BADSIZLen:WORD
|
|||
|
EXTRN DONE:BYTE, DONELen:WORD
|
|||
|
EXTRN BADVER:BYTE
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
EXTRN BADDISK:BYTE, BADDISKLen:WORD
|
|||
|
ENDIF
|
|||
|
|
|||
|
DEFALT DB 0
|
|||
|
IF MSVER
|
|||
|
BIOSName DB "A:\IO.SYS",0
|
|||
|
DOSName DB "A:\MSDOS.SYS",0
|
|||
|
ENDIF
|
|||
|
IF IBMVER OR IBMJAPVER
|
|||
|
BIOSName DB "A:\IBMBIO.COM",0
|
|||
|
DOSName DB "A:\IBMDOS.COM",0
|
|||
|
ENDIF
|
|||
|
|
|||
|
BIOSInFH DW ? ; file handle of source BIOS
|
|||
|
BIOSLenLow DW 2 DUP (?) ; 32-bit length of BIOS
|
|||
|
BIOSLenHigh DW 2 DUP (?) ; 32-bit length of BIOS
|
|||
|
BIOSTime DW 2 DUP (?) ; place to store time of BIOS write
|
|||
|
BIOSOutFH DW ? ; fh of BIOS destination
|
|||
|
|
|||
|
DOSInFH DW ? ; file handle of source DOS
|
|||
|
DOSLenLow DW 2 DUP (?) ; 32-bit length of DOS
|
|||
|
DOSLenHigh DW 2 DUP (?) ; 32-bit length of DOS
|
|||
|
DOSTime DW 2 DUP (?) ; place to store time of DOS write
|
|||
|
DOSOutFH DW ? ; fh of DOS destination
|
|||
|
|
|||
|
AllName DB "A:\*.*",0
|
|||
|
|
|||
|
cbBuf DW ? ; number of bytes in buffer
|
|||
|
pDOS DW ? ; offset of beginning of DOS in buffer
|
|||
|
pDOSEnd DW ? ; offset of end of DOS in buffer
|
|||
|
|
|||
|
IF IBMVER OR IBMJAPVER
|
|||
|
BOOT DW 256 DUP (0)
|
|||
|
IF IBMJAPVER
|
|||
|
LLISTBUF DW 256 DUP (0)
|
|||
|
ENDIF
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
RELOC DW 1 DUP(?)
|
|||
|
STARTSECTOR DW 1 DUP(?)
|
|||
|
ENDIF
|
|||
|
|
|||
|
BUF LABEL BYTE ; beginning of area for file reads
|
|||
|
|
|||
|
DATA ENDS
|
|||
|
|
|||
|
CODE SEGMENT PUBLIC
|
|||
|
|
|||
|
ASSUME CS:DG,DS:DG,ES:DG,SS:DG
|
|||
|
|
|||
|
ORG 100H
|
|||
|
|
|||
|
Start:
|
|||
|
JMP SHORT CheckVersion
|
|||
|
|
|||
|
IF IBMVER
|
|||
|
DW OFFSET DG:BOOT
|
|||
|
ENDIF
|
|||
|
HEADER DB "Vers 1.81"
|
|||
|
CheckVersion:
|
|||
|
PUSH AX ; save drive letter validity
|
|||
|
MOV AH,GET_VERSION
|
|||
|
INT 21H ; get dos version
|
|||
|
XCHG AH,AL ; Turn it around to AH.AL
|
|||
|
CMP AX,DOSVER_LOW ; is it too low?
|
|||
|
JB GOTBADDOS ; yes, error
|
|||
|
CMP AX,DOSVER_HIGH ; too high?
|
|||
|
JBE OKDOS ; yes, go check drive letter
|
|||
|
GOTBADDOS:
|
|||
|
MOV DX,OFFSET DG:BADVER ; message to dump
|
|||
|
MOV AH,STD_CON_STRING_OUTPUT ; standard output device
|
|||
|
INT 21H
|
|||
|
INT 20H ; old style exit for compatability
|
|||
|
|
|||
|
OKDOS: POP AX ; get drive validity
|
|||
|
JMP SHORT SYS ; go process
|
|||
|
|
|||
|
ERR0: MOV DX,OFFSET DG:BADPARM ; no drive letter
|
|||
|
MOV CX,BadParmLen
|
|||
|
JMP DisplayError
|
|||
|
|
|||
|
ERR1: MOV DX,OFFSET DG:BADDRV ; drive letter invalid
|
|||
|
MOV CX,BadDrvLen
|
|||
|
JMP DisplayError
|
|||
|
|
|||
|
ERR2: MOV AL,DEFALT ; get default drive number
|
|||
|
ADD AL,'A'-1 ; turn into letter
|
|||
|
MOV SYSDRV,AL ; place into middle of message
|
|||
|
MOV DX,OFFSET DG:GETSYS
|
|||
|
MOV CX,GETSYSLen ; length for output
|
|||
|
MOV BX,stderr ; use stderr
|
|||
|
MOV AH,Write ; Ask for system disk
|
|||
|
INT 21H
|
|||
|
CALL GetKeystroke ; wait for him to type simething
|
|||
|
XOR AL,AL ; valid drive spec now...
|
|||
|
SYS:
|
|||
|
CMP DS:(BYTE PTR 5DH)," " ; Was file specified?
|
|||
|
JNZ ERR0 ; yes, no files are allowed -> error
|
|||
|
CMP AL,-1 ; Invalid drive spec?
|
|||
|
JZ ERR1 ; yes, must have valid drive -> error
|
|||
|
CMP DS:(BYTE PTR 5CH),0 ; No drive specified?
|
|||
|
JZ ERR1 ; yes, cannot sys to default drive error
|
|||
|
MOV AH,GET_DEFAULT_DRIVE ; Get default drive
|
|||
|
INT 21H
|
|||
|
INC AL ; turn from phys drive to logical drive
|
|||
|
MOV DEFALT,AL ; save it for possible printing
|
|||
|
CMP DS:(BYTE PTR 5CH),AL ; did he specify the default drive?
|
|||
|
JZ ERR1 ; yes, default drive not allowed
|
|||
|
|
|||
|
IF IBMVER ; Check for "valid" destination
|
|||
|
PUSH AX
|
|||
|
MOV AL,BYTE PTR DS:[5Ch]
|
|||
|
DEC AL
|
|||
|
MOV BX,OFFSET DG:BUF ; Temp space
|
|||
|
MOV DX,1 ; Sector 1 (first sec of FAT)
|
|||
|
MOV CX,DX ; One sector
|
|||
|
INT 25H ; Read Fat sector
|
|||
|
POP AX ; Flags
|
|||
|
POP AX ; Real AX
|
|||
|
JC OKFAT ; Don't error here, let a CREATE or
|
|||
|
; some other call to the dest
|
|||
|
; generate a more useful INT 24H
|
|||
|
; error
|
|||
|
CMP BYTE PTR [BUF],0F8H
|
|||
|
JAE OKFAT
|
|||
|
JMP ERR3
|
|||
|
OKFAT:
|
|||
|
ENDIF
|
|||
|
|
|||
|
ADD AL,'A'-1 ; turn into letter
|
|||
|
MOV BIOSName,AL ; twiddle source name
|
|||
|
MOV DOSName,AL ; twiddle source name
|
|||
|
CLD
|
|||
|
MOV DX,OFFSET DG:BIOSName ; source name
|
|||
|
MOV DI,OFFSET DG:BIOSInFH ; pointer to block of data
|
|||
|
CALL OpenFile
|
|||
|
JC Err2 ; not found, go and try again
|
|||
|
MOV DX,OFFSET DG:DOSName ; source of DOS
|
|||
|
MOV DI,OFFSET DG:DOSInFH ; pointer to block of data
|
|||
|
CALL OpenFile ; Look for DOS
|
|||
|
JC ERR2 ; not there, go ask for a system disk
|
|||
|
MOV CX,SP ; get lowest available spot
|
|||
|
SUB CX,0200h+(OFFSET DG:BUF); leave room for all sorts of things
|
|||
|
MOV cbBuf,CX ; store length away
|
|||
|
CALL FillMem ; load up memory with files
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
CALL READ_BOOT ; need to copy boot sector too
|
|||
|
ENDIF
|
|||
|
|
|||
|
MOV AL,DS:(BYTE PTR 5CH) ; get drive of destination
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
CALL CHECK_TRAN ; check for bootable device
|
|||
|
JZ DOSWRT ; ok to boot
|
|||
|
MOV DX,OFFSET DG:BADDISK ; incorrect format to boot
|
|||
|
MOV CX,BadDiskLen
|
|||
|
JMP DisplayError ; go error and quit
|
|||
|
DOSWRT:
|
|||
|
ENDIF
|
|||
|
|
|||
|
ADD AL,'A'-1 ; convert to letter
|
|||
|
MOV BIOSName,AL ; point names at destination drive
|
|||
|
MOV DOSName,AL
|
|||
|
MOV AllName,AL ; look for any files
|
|||
|
|
|||
|
MOV AH,Find_First ; look for files
|
|||
|
MOV DX,OFFSET DG:AllName ; path of where to look
|
|||
|
MOV CX,Attr_Hidden+Attr_System ; attributes to find
|
|||
|
INT 21H
|
|||
|
JC PutSys ; no files - go and copy
|
|||
|
|
|||
|
IF MSVER
|
|||
|
MOV DL,DS:(BYTE PTR 5CH) ; get drive number
|
|||
|
MOV AH,GET_DRIVE_FREESPACE ; get free space available
|
|||
|
INT 21H
|
|||
|
MUL CX ; Compute size of cluster (secsiz*secperclus)
|
|||
|
XCHG CX,AX ; move it to correct spot
|
|||
|
MOV DX,OFFSET DG:BIOSName ; who to open
|
|||
|
MOV AX,BIOSLenLow+2 ; get low part of size
|
|||
|
MOV BX,BIOSLenHigh+2 ; get high size
|
|||
|
CALL CHKLEN ; open and snoop size
|
|||
|
JNZ ERR4 ; Must fit exact so MSDOS is in right place
|
|||
|
MOV DX,OFFSET DG:DOSName ; other guy to open
|
|||
|
MOV AX,DOSLenLow+2 ; get low part of size
|
|||
|
MOV BX,DOSLenHigh+2 ; get high size
|
|||
|
CALL CHKLEN ; open and snoop second size
|
|||
|
JA ERR4 ; Must be enough (or too much) space
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF IBMVER OR IBMJAPVER
|
|||
|
MOV DX,OFFSET DG:BIOSName ; open BIOS
|
|||
|
MOV CX,7 ; attributes
|
|||
|
MOV AH,Find_First
|
|||
|
INT 21H
|
|||
|
JNC FindDos
|
|||
|
Err3J: JMP Err3 ; not found, go and complain
|
|||
|
FindDos:
|
|||
|
MOV DX,OFFSET DG:DOSName ; open DOS
|
|||
|
MOV AH,Find_First
|
|||
|
INT 21H
|
|||
|
JC Err3J ; Not found, go complain
|
|||
|
ENDIF
|
|||
|
|
|||
|
PUTSYS:
|
|||
|
MOV DX,OFFSET DG:BIOSName ; who to change mode
|
|||
|
MOV CX,0 ; undo attributes
|
|||
|
MOV AX,(ChMod SHL 8) + 1 ; set the attributes
|
|||
|
INT 21h
|
|||
|
MOV DX,OFFSET DG:DOSName ; who to change mode
|
|||
|
MOV CX,0 ; undo attributes
|
|||
|
MOV AX,(ChMod SHL 8) + 1 ; set the attributes
|
|||
|
INT 21h
|
|||
|
MOV DX,OFFSET DG:BIOSName ; destination of BIOS
|
|||
|
MOV CX,7 ; fancy attributes
|
|||
|
MOV AH,Creat ; make a new one
|
|||
|
INT 21h
|
|||
|
MOV BIOSOutFH,AX ; save handle
|
|||
|
MOV DX,OFFSET DG:DOSName ; destination of DOS
|
|||
|
MOV AH,Creat ; make a new one
|
|||
|
INT 21h
|
|||
|
MOV DOSOutFH,AX ; save handle
|
|||
|
Copy:
|
|||
|
CALL DumpMem ; flush out memory
|
|||
|
MOV AX,DOSLenHigh ; more DOS?
|
|||
|
OR AX,DOSLenLow ; more low dos
|
|||
|
OR AX,BIOSLenHigh ; more high BIOS
|
|||
|
OR AX,BIOSLenLow ; more low BIOS
|
|||
|
JZ AllDone ; nope, all done
|
|||
|
CALL FillMem ; reload world
|
|||
|
JMP Copy
|
|||
|
ERR4:
|
|||
|
MOV DX,OFFSET DG:BADSIZ
|
|||
|
MOV CX,BadSizLen
|
|||
|
JMP DisplayError
|
|||
|
AllDone:
|
|||
|
MOV CX,BIOSTime ; get time and date
|
|||
|
MOV DX,BIOSTime+2
|
|||
|
MOV BX,BIOSOutFH ; where to stuff the time
|
|||
|
MOV AX,(File_Times SHL 8) + 1
|
|||
|
INT 21h
|
|||
|
MOV AH,Close
|
|||
|
INT 21h
|
|||
|
|
|||
|
MOV CX,DOSTime ; get time and date
|
|||
|
MOV DX,DOSTime+2
|
|||
|
MOV BX,DOSOutFH ; where to stuff the time
|
|||
|
MOV AX,(File_Times SHL 8) + 1
|
|||
|
INT 21h
|
|||
|
MOV AH,Close
|
|||
|
INT 21h
|
|||
|
|
|||
|
IF IBMVER OR IBMJAPVER
|
|||
|
CALL PUTBOOT ; copy the boot sector also
|
|||
|
ENDIF
|
|||
|
|
|||
|
MOV DX,OFFSET DG:DONE ; all finished message
|
|||
|
MOV CX,DoneLen
|
|||
|
XOR AL,AL ; ok error code
|
|||
|
SERROR:
|
|||
|
PUSH AX
|
|||
|
MOV BX,stderr
|
|||
|
MOV AH,Write ; convenient place to display message
|
|||
|
INT 21H
|
|||
|
POP AX
|
|||
|
ErrorExit:
|
|||
|
MOV AH,EXIT ; bye and return error code
|
|||
|
INT 21h
|
|||
|
|
|||
|
DisplayError:
|
|||
|
MOV AL,1
|
|||
|
JMP SERROR
|
|||
|
FillMem:
|
|||
|
MOV CX,cbBuf ; get length of buffer
|
|||
|
MOV BX,BIOSInFH ; get bios source handle
|
|||
|
MOV DX,OFFSET DG:BUF ; point to beginning of buffer
|
|||
|
PUSH CX ; save away total length
|
|||
|
CMP BIOSLenHigh,0 ; > 64K to read?
|
|||
|
JA UseCX ; use CX
|
|||
|
CMP BIOSLenLow,CX ; more left to read?
|
|||
|
JA UseCX ; use CX
|
|||
|
MOV CX,BIOSLenLow ; move new
|
|||
|
UseCX:
|
|||
|
MOV AH,Read
|
|||
|
INT 21h ; read in what we can
|
|||
|
ADD DX,AX ; update pointer for DOS Read
|
|||
|
MOV pDOS,DX ; point to beginning of DOS
|
|||
|
SUB BIOSLenLow,AX ; decrement remaining
|
|||
|
SBB BIOSLenHigh,0 ; do 32 bit
|
|||
|
POP CX ; get original length
|
|||
|
SUB CX,AX ; this much is left
|
|||
|
|
|||
|
MOV BX,DOSInFH ; get bios source handle
|
|||
|
CMP DOSLenHigh,0 ; > 64K to read?
|
|||
|
JA UseCXDOS ; use CX
|
|||
|
CMP DOSLenLow,CX ; more left to read?
|
|||
|
JA UseCXDOS ; use CX
|
|||
|
MOV CX,DOSLenLow ; move new
|
|||
|
UseCXDOS:
|
|||
|
MOV AH,Read
|
|||
|
INT 21h ; read in what we can
|
|||
|
ADD DX,AX ; update pointer for DOS Read
|
|||
|
MOV pDOSEnd,DX ; point to End of dos DOS
|
|||
|
SUB DOSLenLow,AX ; decrement remaining
|
|||
|
SBB DOSLenHigh,0 ; do 32 bit arithmetic
|
|||
|
return
|
|||
|
|
|||
|
OpenFile:
|
|||
|
MOV AX,(OPEN SHL 8) + 0 ; open for reading only
|
|||
|
INT 21H ; Look for BIOS
|
|||
|
retc ; not found, go and try again
|
|||
|
STOSW ; stash away handle
|
|||
|
MOV BX,AX ; get ready for seeks
|
|||
|
MOV AX,(LSeek SHL 8) + 2 ; seek relative to eof
|
|||
|
XOR CX,CX ; zero offset
|
|||
|
XOR DX,DX ; zero offset
|
|||
|
INT 21h ; get offsets
|
|||
|
STOSW ; save low part of size
|
|||
|
STOSW ; save low part of size
|
|||
|
MOV AX,DX
|
|||
|
STOSW ; save high part of size
|
|||
|
STOSW ; save high part of size
|
|||
|
XOR DX,DX ; zero offset
|
|||
|
MOV AX,(LSeek SHL 8) + 0 ; seek relative to beginning
|
|||
|
INT 21h
|
|||
|
MOV AX,(File_Times SHL 8) + 0
|
|||
|
INT 21h ; get last write times
|
|||
|
MOV AX,CX
|
|||
|
STOSW ; save time
|
|||
|
MOV AX,DX
|
|||
|
STOSW ; save date
|
|||
|
return
|
|||
|
|
|||
|
ERR3:
|
|||
|
MOV DX,OFFSET DG:NODEST
|
|||
|
MOV CX,NoDestLen
|
|||
|
JMP DisplayError
|
|||
|
|
|||
|
DumpMem:
|
|||
|
MOV DX,OFFSET DG:BUF ; get offset of bios start
|
|||
|
MOV CX,pDOS ; beginning of next guy
|
|||
|
SUB CX,DX ; difference is length
|
|||
|
JZ DumpDos ; no bios to move
|
|||
|
MOV BX,BIOSOutFH ; where to output
|
|||
|
MOV AH,Write
|
|||
|
INT 21h ; wham
|
|||
|
DumpDos:
|
|||
|
MOV DX,pDOS ; beginning of dos
|
|||
|
MOV CX,pDOSEnd ; end of dos
|
|||
|
SUB CX,DX ; difference is length
|
|||
|
retz ; if zero no write
|
|||
|
MOV BX,DOSOutFH ; where to output
|
|||
|
MOV AH,Write
|
|||
|
INT 21h ; wham
|
|||
|
ret
|
|||
|
|
|||
|
IF MSVER
|
|||
|
CHKLEN:
|
|||
|
; CX has size of cluster, DX has pointer to file name
|
|||
|
; Returns with flags set on (size of file) - (size of hole)
|
|||
|
PUSH AX ; old size low
|
|||
|
PUSH BX ; old size high
|
|||
|
PUSH CX ; old cluster size
|
|||
|
MOV AH,Find_First
|
|||
|
MOV CX,7 ; attributes to search for
|
|||
|
INT 21H
|
|||
|
JC ERR3 ; cannot find file, error
|
|||
|
POP CX ; get cluster size back
|
|||
|
MOV DX,DS:[80h+find_buf_size_h] ; get destination size high
|
|||
|
MOV AX,DS:[80h+find_buf_size_l] ; get size low
|
|||
|
ADD AX,CX ; add cluster size
|
|||
|
ADC DX,0 ; 32 bit add
|
|||
|
SUB AX,1 ; adding CLUSSIZE-1
|
|||
|
SBB DX,0 ; 32 bit dec
|
|||
|
DIV CX ; compute new cluster size
|
|||
|
POP DX ; get old high
|
|||
|
POP BX ; get old low
|
|||
|
PUSH AX ; save away dividend
|
|||
|
MOV AX,BX ; put into correct register
|
|||
|
ADD AX,CX ; do the same as above (+CLUSSIZE-1)/CLUSSIZE
|
|||
|
ADC DX,0 ; 32 bit add
|
|||
|
SUB AX,1 ; adding CLUSSIZE-1
|
|||
|
SBB DX,0 ; 32 bit dec
|
|||
|
DIV CX ; compute old cluster size
|
|||
|
POP DX ; get new size
|
|||
|
CMP AX,DX ; is old >= new?
|
|||
|
return
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
PUTBOOT:
|
|||
|
CALL READ_LLIST ; Get the list sector and set new boot sector
|
|||
|
MOV AL,DS:(BYTE PTR 5CH)
|
|||
|
DEC AL ; A=0
|
|||
|
MOV CX,1
|
|||
|
XOR DX,DX
|
|||
|
MOV BX,OFFSET DG:BOOT
|
|||
|
INT 26H ; Write out new boot sector
|
|||
|
POPF
|
|||
|
CALL WRITE_LLIST ; Make and write out new list sector
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF IBMVER
|
|||
|
PUTBOOT:
|
|||
|
MOV AH,GET_DPB
|
|||
|
MOV DL,BYTE PTR DS:[5Ch] ; Target drive
|
|||
|
INT 21H
|
|||
|
ASSUME DS:NOTHING
|
|||
|
MOV AL,[BX+16H] ; Media byte
|
|||
|
PUSH CS
|
|||
|
POP DS
|
|||
|
ASSUME DS:DG
|
|||
|
CMP AL,0FEH
|
|||
|
JB RET1
|
|||
|
TEST AL,1
|
|||
|
JZ GOTBOOT
|
|||
|
MOV BX,OFFSET DG:BOOT
|
|||
|
MOV WORD PTR [BX+17],112 ; Set number of dir entries
|
|||
|
MOV WORD PTR [BX+19],2*8*40 ; Set number of sectors
|
|||
|
INC BYTE PTR [BX+21] ; Media = ff
|
|||
|
INC WORD PTR [BX+26] ; Number of heads = 2
|
|||
|
|
|||
|
GOTBOOT:
|
|||
|
MOV AL,BYTE PTR DS:[5Ch]
|
|||
|
DEC AL
|
|||
|
MOV BX,OFFSET DG:BOOT ; Boot sector
|
|||
|
XOR DX,DX ; Sector 0
|
|||
|
MOV CX,DX
|
|||
|
INC CX ; One sector
|
|||
|
INT 26H ; Write out 8 sector boot sector
|
|||
|
POP AX ; Flags
|
|||
|
RET1: RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
IF IBMJAPVER
|
|||
|
READ_BOOT:
|
|||
|
MOV AL,[DEFALT]
|
|||
|
DEC AL ; A=0
|
|||
|
MOV CX,1
|
|||
|
XOR DX,DX
|
|||
|
MOV BX,OFFSET DG:BOOT
|
|||
|
INT 25H
|
|||
|
POPF
|
|||
|
MOV AX,[BOOT+108H] ; Get old first sector of data
|
|||
|
MOV [RELOC],AX
|
|||
|
RET
|
|||
|
|
|||
|
READ_LLIST:
|
|||
|
MOV AL,DS:(BYTE PTR 5CH)
|
|||
|
DEC AL ; A=0
|
|||
|
MOV CX,1
|
|||
|
MOV DX,[STARTSECTOR]
|
|||
|
MOV BX,OFFSET DG:LLISTBUF
|
|||
|
INT 25H
|
|||
|
POPF
|
|||
|
RET
|
|||
|
|
|||
|
WRITE_LLIST:
|
|||
|
MOV AX,[STARTSECTOR]
|
|||
|
MOV DX,AX
|
|||
|
SUB AX,[RELOC] ; True reloc factor
|
|||
|
MOV CL,BYTE PTR [LLISTBUF+0CH] ; Number of entries needing reloc
|
|||
|
XOR CH,CH
|
|||
|
JCXZ NO_RELOCS
|
|||
|
MOV BX,OFFSET DG:LLISTBUF + 10H
|
|||
|
RELLOOP:
|
|||
|
ADD WORD PTR [BX+2],AX
|
|||
|
ADD BX,10H
|
|||
|
LOOP RELLOOP
|
|||
|
NO_RELOCS:
|
|||
|
MOV AL,DS:(BYTE PTR 5CH)
|
|||
|
DEC AL ; A=0
|
|||
|
MOV CX,1
|
|||
|
MOV BX,OFFSET DG:LLISTBUF
|
|||
|
INT 26H
|
|||
|
POPF
|
|||
|
RET
|
|||
|
|
|||
|
CHECK_TRAN:
|
|||
|
; All registers preserved. Returns zero if SYS OK, NZ if SYS FAIL
|
|||
|
; AL is drive (1=A,...) AL=0 is not valid
|
|||
|
|
|||
|
PUSH BX
|
|||
|
PUSH AX
|
|||
|
PUSH DS
|
|||
|
MOV DL,AL
|
|||
|
MOV AH,GET_DPB
|
|||
|
INT 21H
|
|||
|
MOV AX,[BX.dpb_first_sector] ; Get new first sector of data
|
|||
|
MOV BH,[BX.dpb_media]
|
|||
|
POP DS
|
|||
|
MOV [STARTSECTOR],AX
|
|||
|
MOV [BOOT+108H],AX ; Set new start of data in boot
|
|||
|
POP AX
|
|||
|
PUSH AX
|
|||
|
MOV BL,AL
|
|||
|
INT 11H ; IBM EQUIP CALL
|
|||
|
ROL AL,1
|
|||
|
ROL AL,1
|
|||
|
AND AL,3
|
|||
|
JNZ NOT_SINGLE
|
|||
|
INC AL
|
|||
|
NOT_SINGLE:
|
|||
|
INC AL ; AL is now MAX floppy #
|
|||
|
CMP BL,AL
|
|||
|
POP AX
|
|||
|
JBE CHECK_FLOP ; Is a floppy
|
|||
|
XOR BL,BL ; Is Hard file
|
|||
|
POP BX
|
|||
|
RET
|
|||
|
|
|||
|
CHECK_FLOP:
|
|||
|
CMP BH,0FBH ; Only floppy that boots
|
|||
|
POP BX
|
|||
|
RET
|
|||
|
ENDIF
|
|||
|
|
|||
|
GetKeystroke:
|
|||
|
MOV AX,(Std_CON_Input_Flush SHL 8) + Std_CON_Input_No_Echo
|
|||
|
INT 21H
|
|||
|
MOV AX,(Std_CON_Input_Flush SHL 8) + 0
|
|||
|
INT 21H
|
|||
|
|
|||
|
return
|
|||
|
|
|||
|
CODE ENDS
|
|||
|
END START
|
|||
|
|