;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; RDOS operating system ; Copyright (C) 1988-2000, Leif Ekblad ; ; This program is free software; you can redistribute it and/or modify ; it under the terms of the GNU General Public License as published by ; the Free Software Foundation; either version 2 of the License, or ; (at your option) any later version. The only exception to this rule ; is for commercial usage in embedded systems. For information on ; usage in commercial embedded systems, contact embedded@rdos.net ; ; This program is distributed in the hope that it will be useful, ; but WITHOUT ANY WARRANTY; without even the implied warranty of ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ; GNU General Public License for more details. ; ; You should have received a copy of the GNU General Public License ; along with this program; if not, write to the Free Software ; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ; ; The author of this program may be contacted at leif@rdos.net ; ; FLOPPY.ASM ; Floppy disk driver ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; NAME floppy GateSize = 16 INCLUDE ..\driver.def INCLUDE ..\user.def INCLUDE ..\os.def INCLUDE ..\user.inc INCLUDE ..\os.inc INCLUDE ..\drive.inc INCLUDE ..\os\port.def boot_struc STRUC boot_jmp DB ?,?,? boot_name DB 8 DUP(?) boot_bytes_per_sector DW ? boot_sectors_per_cluster DB ? boot_resv_sectors DW ? boot_fats DB ? boot_root_dirs DW ? boot_sectors DW ? boot_media DB ? boot_fat_sectors DW ? boot_sectors_per_cyl DW ? boot_heads DW ? boot_hidden_sectors DD ? boot_big_sectors DD ? boot_drive_nr DB ?,? boot_signature DB ? boot_serial DD ? boot_volume DB 11 DUP(?) boot_fs DB 8 DUP(?) boot_struc ENDS disc_struc STRUC boot_sect DB 512 DUP(?) disc_section section_typ <> disc_fs_handle DW ? disc_sel DW ? disc_thread DW ? disc_sub_unit DB ? disc_nr DB ? disc_struc ENDS MotorOnWait EQU 500 Spec1 EQU 0DFh Spec2 EQU 2 floppy_data SEGMENT AT 0 FloppyThread DW ? DriveControl DB ? IntFlag DB ? TimeoutCount DB ? Gap DB 4 DUP(?) Tracks DB 4 DUP(?) MotorCount DB 4 DUP(?) BootValid DB 4 DUP(?) DiscArr DW 2 DUP(?) CmdCode DB ? DriveHead DB ? cTrack DB ? cDriveHead DB ? cSector DB ? cBytesPerSector DB ? cTracks DB ? cGap DB ? cDataLen DB ? st0 DB ? st1 DB ? st2 DB ? sTrack DB ? sHead DB ? sSector DB ? sBytesPerSector DB ? FloppySection section_typ <> floppy_data ENDS ;;;;;;;;; INTERNAL PROCEDURES ;;;;;;;;;;; code SEGMENT byte public 'CODE' assume cs:code .386c PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FLOPPY_INT ; ; DESCRIPTION: Floppy disk interrupt ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; floppy_int Proc far mov al,ds:IntFlag or al,al jnz floppy_int_done inc al mov ds:IntFlag,al mov bx,ds:FloppyThread Signal floppy_int_done: ret floppy_int Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CommandPhase ; ; DESCRIPTION: Execute command phase ; ; PARAMETERS: CX Number of bytes to output ; ; RETURNS: NC Performed ok ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CommandPhase Proc near push ax push bx push dx push si ClearSignal mov ds:IntFlag,0 mov si,2000 mov bx,OFFSET CmdCode CommandPhaseLoop: push cx xor cx,cx CommandPhaseReqLoop: mov dx,3F4h in al,dx test al,80h jnz CommandPhaseReqSet mov ax,50 WaitMicroSec sub si,1 jnc CommandPhaseReqLoop pop cx jmp CommandPhaseDone CommandPhaseReqSet: pop cx test al,40h stc jnz CommandPhaseDone ; mov dx,3F5h mov al,[bx] inc bx out dx,al ; output data byte loop CommandPhaseLoop mov ax,30 WaitMicroSec clc CommandPhaseDone: pop si pop dx pop bx pop ax ret CommandPhase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ExecutePhase ; ; DESCRIPTION: Wait for execute phase to be ready ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ExecutePhase Proc near push es push ax push dx push di mov dx,ds mov es,dx mov dx,3F4h in al,dx test al,10h clc jz ExecutePhaseDone test al,80h clc jnz ExecutePhaseDone mov ds:TimeoutCount,20 WaitForSignal mov al,ds:IntFlag or al,al clc jnz ExecutePhaseDone stc ExecutePhaseDone: sti pop di pop dx pop ax pop es ret ExecutePhase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ResultPhase ; ; DESCRIPTION: Execute result phase ; ; RETURNS: NC Performed ok ; CX Number of bytes in result ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ResultPhase Proc near push ax push bx push dx push si mov si,200 xor cx,cx mov bx,OFFSET st0 ResultPhaseLoop: mov dx,3F4h in al,dx test al,10h clc jz ResultPhaseDone test al,80h jnz ResultPhaseReqSet mov ax,50 WaitMicroSec sub si,1 jnc ResultPhaseLoop jmp ResultPhaseDone ResultPhaseReqSet: test al,40h stc jz ResultPhaseDone inc cx mov dx,3F5h in al,dx mov [bx],al inc bx jmp ResultPhaseLoop ResultPhaseDone: pop si pop dx pop bx pop ax ret ResultPhase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RecoverPhase ; ; DESCRIPTION: Set controller in idle state ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RecoverPhase Proc near push ax push dx push si mov si,200 RecoverPhaseLoop: mov dx,3F4h in al,dx test al,10h jz RecoverPhaseDone test al,80h jnz RecoverPhaseReqSet mov ax,50 WaitMicroSec sub si,1 jnc RecoverPhaseLoop jmp RecoverPhaseDone RecoverPhaseReqSet: test al,40h jnz RecoverPhaseRead RecoverPhaseWrite: mov dx,3F5h mov al,0FFh out dx,al jmp RecoverPhaseLoop RecoverPhaseRead: mov dx,3F5h in al,dx jmp RecoverPhaseLoop RecoverPhaseDone: pop si pop dx pop ax ret RecoverPhase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DecodeStatus ; ; DESCRIPTION: Decode status of previous operation ; ; RETURNS: NC Success ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; DecodeStatus Proc near push ax mov al,ds:st0 test al,0C0h clc jz DecodeStatusDone stc DecodeStatusDone: pop ax ret DecodeStatus Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: Command ; ; DESCRIPTION: Execute command ; ; PARAMETERS: CX Number of bytes to output ; ; RETURNS: NC Performed ok ; CX Number of bytes in answer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; Command Proc near push ax GetThread mov ds:FloppyThread,ax pop ax ; call CommandPhase jc CommandDone call ExecutePhase jc CommandDone call ResultPhase jc CommandDone or cx,cx clc je CommandDone call DecodeStatus CommandDone: mov ds:FloppyThread,0 ret Command Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SenseIntCmd ; ; DESCRIPTION: Sense interrupt command ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SenseIntCmd Proc near push ax push cx mov ax,10 WaitMilliSec mov ds:CmdCode,8 mov cx,1 call CommandPhase jc SenseIntDone call ResultPhase SenseIntDone: pop cx pop ax ret SenseIntCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ExecuteSensePhase ; ; DESCRIPTION: Execute phase using sense int command ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ExecuteSensePhase Proc near push es push ax push di mov ax,ds mov es,ax mov ds:TimeoutCount,20 WaitForSignal mov al,ds:IntFlag or al,al stc jz ExecuteSenseDone ExecuteSenseCmd: sti call SenseIntCmd ExecuteSenseDone: pop di pop ax pop es ret ExecuteSensePhase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SenseDriveStatusCmd ; ; DESCRIPTION: Sense drive status command ; ; PARAMETERS: AL Drive # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SenseDriveStatusCmd Proc near push bx push cx mov bx,OFFSET CmdCode mov byte ptr [bx],4 mov [bx+1],al mov cx,2 call CommandPhase jc SenseDriveStatusDone call ResultPhase SenseDriveStatusDone: pop cx pop bx ret SenseDriveStatusCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SpecifyCmd ; ; DESCRIPTION: Specify command ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SpecifyCmd Proc near push bx push cx mov bx,OFFSET CmdCode mov byte ptr [bx],3 mov byte ptr [bx+1],Spec1 mov byte ptr [bx+2],Spec2 mov cx,3 call Command pop cx pop bx ret SpecifyCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: RecalibrateCmd ; ; DESCRIPTION: Recalibrate command ; ; PARAMETERS: AL Drive # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; RecalibrateCmd Proc near push cx mov ds:CmdCode,7 mov ds:DriveHead,al mov cx,2 call Command jc RecalibrateDone call SenseIntCmd RecalibrateDone: pop cx ret RecalibrateCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadIdCmd ; ; DESCRIPTION: Read drive ID command ; ; PARAMETERS: AL Drive # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ReadIdCmd Proc near push cx mov ds:CmdCode,4Ah mov ds:DriveHead,al mov cx,2 call Command mov cl,ds:sBytesPerSector mov ds:cBytesPerSector,cl pop cx ret ReadIdCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ResetController ; ; DESCRIPTION: ResetController ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ResetController Proc near push es push ax push cx push dx push di mov ax,ds mov es,ax mov cx,100 mov ds:cTrack,-1 GetThread mov ds:FloppyThread,ax ResetControllerLoop: ClearSignal mov ds:IntFlag,0 mov al,8 mov dx,3F2h out dx,al mov ax,250 WaitMicroSec ; mov al,0Ch mov dx,3F2h out dx,al mov ds:DriveControl,al ; mov ds:TimeoutCount,20 WaitForSignal mov al,ds:IntFlag or al,al jz ResetControllerFailed ResetControllerStart: mov al,0Ch out dx,al mov ds:DriveControl,al ResetControllerSense: call SenseIntCmd jc ResetControllerFailed call SpecifyCmd jnc ResetControllerDone ResetControllerFailed: mov ax,250 WaitMicroSec call RecoverPhase call SenseIntCmd mov ax,10 WaitMilliSec loop ResetControllerLoop stc ResetControllerDone: mov ds:FloppyThread,0 pop di pop dx pop cx pop ax pop es ret ResetController Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: TestDrive ; ; DESCRIPTION: Test drive with a specified data rate ; ; PARAMETERS: AL Drive # ; AH Data rate ; ; RETURNS: NC Success ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; TestDrive Proc near push ax push cx push dx ; xchg al,ah mov dx,3F7h out dx,al xchg al,ah ; call RecalibrateCmd jnc TestDriveStart call RecalibrateCmd jc TestDriveDone TestDriveStart: mov cx,3 TestDriveLoop: call ReadIdCmd jnc TestDriveDone loop TestDriveLoop stc TestDriveDone: pop dx pop cx pop ax ret TestDrive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InitDrive ; ; DESCRIPTION: Init drive and setup data rate ; ; PARAMETERS: AL Drive # ; ; RETURNS: NC Success ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InitDrive Proc near push ax push bx movzx bx,al ; mov ah,0 call TestDrive mov al,7 mov ah,18 jnc InitDriveOk ; mov ah,1 call TestDrive mov al,4 mov ah,42 jnc InitDriveOk ; mov ah,2 call TestDrive mov al,7 mov ah,35 jnc InitDriveOk ; mov ah,3 call TestDrive mov al,7 mov ah,27 jnc InitDriveOk xor al,al int 3 stc InitDriveOk: mov ds:[bx].Gap,al mov ds:[bx].Tracks,ah pop bx pop ax ret InitDrive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SelectDrive ; ; DESCRIPTION: Set current drive ; ; PARAMETERS: AL Drive # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SelectDrive Proc near push ax push bx push cx push dx movzx bx,al mov cl,al mov ah,10h rol ah,cl cli mov al,ds:DriveControl test al,ah jnz SelectDriveMotorOn ; sti call ResetController jc SelectDriveDone ; cli mov ds:[bx].MotorCount,255 mov al,ds:DriveControl and al,0F0h or al,ah or al,ds:DriveControl mov ds:DriveControl,al mov dx,3F2h out dx,al sti mov ax,MotorOnWait WaitMilliSec mov al,ds:DriveControl jmp SelectDriveInit SelectDriveMotorOn: cli mov al,ds:DriveControl mov ah,al and ah,3 cmp ah,cl clc je SelectDriveDone SelectDriveInit: and al,NOT 3 or al,cl mov dx,3F2h out dx,al mov ds:DriveControl,al sti mov ax,100 WaitMicroSec mov al,ds:[bx].Gap or al,al mov al,bl jnz SelectDriveRecal call InitDrive jc SelectDriveDone SelectDriveRecal: mov ax,100 WaitMilliSec mov al,bl call RecalibrateCmd jnc SelectDriveDone call RecalibrateCmd SelectDriveDone: sti mov ds:[bx].MotorCount,25 pop dx pop cx pop bx pop ax ret SelectDrive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetupDMA ; ; DESCRIPTION: Setup DMA controller ; ; PARAMETERS: AL Mode ; 42h = Verify ; 46h = Read ; 4Ah = Write ; EBX Linear Address (must be within same page) ; CX Number of bytes to transfer ; ; RETURNS: NC Success ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SetupDMA Proc near push ax push edx ; mov edx,ebx push es push eax mov ax,flat_sel mov es,ax ; GetPhysicalPage and ax,0F000h or eax,eax jz setup_dma_alloc ; cmp eax,1000000h jc setup_dma_inrange ; int 3 FreePhysical setup_dma_alloc: AllocateDmaPhysical or al,3 SetPhysicalPage and ax,0F000h cmp eax,1000000h jc setup_dma_inrange ; int 3 setup_dma_inrange: and dx,0FFFh or ax,dx mov edx,eax pop eax pop es cli out 0Ch,al jcxz short $+2 jcxz short $+2 out 0Bh,al jcxz short $+2 jcxz short $+2 dec cx mov al,cl out 5,al jcxz short $+2 jcxz short $+2 mov al,ch out 5,al jcxz short $+2 jcxz short $+2 inc cx mov al,dl out 4,al jcxz short $+2 jcxz short $+2 mov al,dh out 4,al jcxz short $+2 jcxz short $+2 shr edx,16 mov al,dl out 81h,al jcxz short $+2 jcxz short $+2 mov al,2 out 0Ah,al sti pop edx pop ax ret SetupDMA Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SeekCmd ; ; DESCRIPTION: Seek to track ; ; PARAMETERS: AL Drive # ; AH Cylinder # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SeekCmd Proc near push ax GetThread mov ds:FloppyThread,ax pop ax push cx mov ds:CmdCode,0Fh mov ds:DriveHead,al mov ds:cTrack,ah mov cx,3 call CommandPhase jc SeekDone call ExecuteSensePhase SeekDone: mov ds:FloppyThread,0 pop cx ret SeekCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadCmd ; ; DESCRIPTION: Read sectors ; ; PARAMETERS: AL Drive # ; DH Head # ; DL Sector ; AH Cylinder ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ReadCmd Proc near push bx push cx movzx bx,al mov ds:CmdCode,0E6h mov ds:cTrack,ah mov ds:cDriveHead,dh mov ds:cSector,dl mov cl,dh shl cl,2 or cl,al mov ds:DriveHead,cl mov cl,ds:[bx].Gap mov ds:cGap,cl mov cl,ds:[bx].Tracks mov ds:cTracks,cl mov ds:cDataLen,0FFh mov cx,9 call Command pop cx pop bx ret ReadCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WriteCmd ; ; DESCRIPTION: Write sectors ; ; PARAMETERS: AL Drive # ; DH Head # ; DL Sector ; AH Cylinder # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; WriteCmd Proc near push bx push cx movzx bx,al mov ds:CmdCode,0C5h mov ds:cTrack,ah mov ds:cDriveHead,dh mov ds:cSector,dl mov cl,dh shl cl,2 or cl,al mov ds:DriveHead,cl mov cl,ds:[bx].Gap mov ds:cGap,cl mov cl,ds:[bx].Tracks mov ds:cTracks,cl mov ds:cDataLen,0FFh mov cx,9 call Command pop cx pop bx ret WriteCmd Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadDrive ; ; DESCRIPTION: Read sectors ; ; PARAMETERS: AL Drive # ; CX Number of bytes ; EBX Logical address of buffer ; AH Cylinder # ; DL Sector ; DH Head ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ReadDrive Proc near push si mov si,3 call SelectDrive jc ReadDriveRetry cmp ah,ds:cTrack je ReadDriveSetup ReadDriveLoop: call SeekCmd pushf call check_media jnc ReadDriveMediaOk ; popf jmp ReadDriveRetry ReadDriveMediaOk: popf jc ReadDriveRetry ReadDriveSetup: push ax mov al,46h call SetupDMA pop ax call ReadCmd jnc ReadDriveDone ReadDriveRetry: mov ds:cTrack,-1 push ax push dx cli mov al,0Ch mov ds:DriveControl,al mov dx,3F2h out dx,al sti pop dx pop ax sub si,1 jc ReadDriveDone ; call check_media jc ReadDriveRetry ; push ax mov ax,250 WaitMilliSec pop ax jmp ReadDriveLoop ReadDriveDone: pop si ret ReadDrive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: WriteDrive ; ; DESCRIPTION: Write sectors ; ; PARAMETERS: AL Drive # ; CX Number of bytes ; EBX Logical address of buffer ; AH Cylinder # ; DL Sector ; DH Head ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; WriteDrive Proc near push si mov si,3 call SelectDrive jc WriteDriveRetry cmp ah,ds:cTrack je WriteDriveSetup WriteDriveLoop: call SeekCmd pushf call check_media jnc WriteDriveMediaOk ; popf jmp WriteDriveRetry WriteDriveMediaOk: popf jc WriteDriveRetry WriteDriveSetup: push ax mov al,4Ah call SetupDMA pop ax call WriteCmd jnc WriteDriveDone WriteDriveRetry: mov ds:cTrack,-1 push ax push dx cli mov al,0Ch mov ds:DriveControl,al mov dx,3F2h out dx,al sti pop dx pop ax sub si,1 jc WriteDriveDone ; call check_media jc WriteDriveRetry ; push ax mov ax,250 WaitMilliSec pop ax jmp WriteDriveLoop WriteDriveDone: pop si ret WriteDrive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ReadBootSector ; ; DESCRIPTION: Get boot-record and drive parameters ; ; PARAMETERS: AL Sub-unit # ; ES:EDI Buffer ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ReadBootSector Proc near push ds push ax push ebx push ecx push edx push esi push edi ; push ax mov ax,floppy_data_sel mov ds,ax mov eax,1000h AllocateBigLinear pop ax ; push edx mov ebx,edx call SelectDrive jnc read_boot_sector_read read_boot_sector_retry: push ax mov al,0Ch mov ds:DriveControl,al mov dx,3F2h out dx,al pop ax ; push ax mov ax,250 WaitMilliSec pop ax ; call SelectDrive jc read_boot_sector_done read_boot_sector_read: mov ah,1 call SeekCmd mov ds:cTrack,-1 jc read_boot_sector_retry ; mov ah,0 call SeekCmd mov ds:cTrack,-1 jc read_boot_sector_retry ; mov dh,0 mov dl,1 mov cx,200h push ax mov al,46h call SetupDMA pop ax ; call ReadCmd jc read_boot_sector_retry ; mov cx,flat_sel mov ds,cx mov esi,ebx mov ecx,128 rep movs dword ptr es:[edi],[esi] mov cx,floppy_data_sel mov ds,cx movzx bx,al mov es:boot_drive_nr,al mov cl,ds:cBytesPerSector mov ax,80h shl ax,cl cmp ax,es:boot_bytes_per_sector stc jne read_boot_sector_done ; mov ax,es:boot_sectors_per_cyl mov ds:[bx].Tracks,al clc read_boot_sector_done: pop edx pushf mov ecx,1000h FreeLinear popf mov ds:[bx].BootValid,1 ; pop edi pop esi pop edx pop ecx pop ebx pop ax pop ds ret ReadBootSector Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InstallMain ; ; DESCRIPTION: Install main drive ; ; PARAMETERS: AL Sub-unit # ; ES Boot sector ; FS Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; fat12 DB 'FAT12 ' InstallMain Proc near push es push ax push bx push cx push edx push di ; mov cl,es:boot_media cmp cl,0F0h je install_main_fat12 mov di,OFFSET boot_fs jmp install_main_fs install_main_fat12: mov cx,cs mov es,cx mov di,OFFSET fat12 install_main_fs: InstallFileSystem ; pop di pop edx pop cx pop bx pop ax pop es ret InstallMain Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DRIVE_ASSIGN1 ; ; DESCRIPTION: Assign disc drives, pass 1 ; ; PARAMETERS: BX Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; drive_assign1 Proc far mov es,bx mov al,es:disc_sub_unit mov ah,es:disc_nr xor edx,edx mov ecx,-1 OpenDrive DemandLoadFileSystem ret drive_assign1 Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DRIVE_ASSIGN2 ; ; DESCRIPTION: Assign disc drives, pass 2 ; ; PARAMETERS: BX Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; drive_assign2 Proc far ret drive_assign2 Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DEMAND_MOUNT ; ; DESCRIPTION: Mount disc drive on demand ; ; PARAMETERS: BX Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; demand_mount Proc far push ds push es pushad ; mov es,bx mov ax,floppy_data_sel mov ds,ax mov al,es:disc_sub_unit mov edi,OFFSET boot_sect EnterSection ds:FloppySection call ReadBootSector LeaveSection ds:FloppySection jc drive_assign_done1 ; call InstallMain pushf mov ax,es:boot_sectors_per_cyl mul es:boot_heads mov cx,es:boot_bytes_per_sector mov edx,80 mov si,es:boot_sectors_per_cyl mov di,es:boot_heads mov bx,es:disc_sel SetDiscParam ; mov bx,es:disc_thread Signal popf jc drive_assign_done1 ; mov bx,es:disc_sel StartDisc drive_assign_done1: popad pop es pop ds ret demand_mount Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ERASE ; ; DESCRIPTION: Erase sectors ; ; PARAMETERS: BX Disc handle ; EDX Start sector ; ECX Number of sectors ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; erase Proc far stc ret erase Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: check_media ; ; DESCRIPTION: Check for media change ; ; PARAMETERS: FS Disc selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; check_media Proc near push ds push eax push bx push dx ; mov bx,fs mov ds,bx EnterSection ds:disc_section ; mov al,fs:boot_drive_nr mov bx,floppy_data_sel mov ds,bx movzx bx,al mov bl,ds:[bx].Gap or bl,bl stc jz check_media_done ; call SelectDrive jc check_media_do ; push ax mov dx,3F7h in al,dx shl al,1 pop ax jnc check_media_done check_media_do: push es push ecx push esi push edi ; push eax mov eax,200h AllocateSmallGlobalMem pop eax check_media_retry: xor edi,edi call ReadBootSector ; mov esi,OFFSET boot_sect mov ecx,80h repe cmps dword ptr fs:[esi],es:[edi] clc jz check_media_free ; mov bx,fs:disc_sel StopDisc push ax mov ax,250 WaitMilliSec pop ax jmp check_media_retry check_media_free: pushf FreeMem popf pop edi pop esi pop ecx pop es check_media_done: mov bx,fs mov ds,bx LeaveSection ds:disc_section ; pop dx pop bx pop eax pop ds ret check_media Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: check_media_proc ; ; DESCRIPTION: Check for media change ; ; PARAMETERS: BX Handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; check_media_proc Proc far push ds push fs push ax ; mov ax,floppy_data_sel mov ds,ax mov fs,bx EnterSection ds:FloppySection call check_media LeaveSection ds:FloppySection ; pop ax pop fs pop ds ret check_media_proc Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: FLOPPY_SUPER ; ; DESCRIPTION: Supervisor thread ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; floppy_super_name DB 'Floppy Supervisor',0 floppy_super: mov ax,floppy_data_sel mov ds,ax floppy_super_loop: xor bx,bx mov cx,4 mov ah,NOT 10h floppy_super_motor_loop: cli mov al,ds:[bx].MotorCount or al,al jz floppy_super_motor_next sub al,1 mov ds:[bx].MotorCount,al sti jnz floppy_super_motor_next cli mov al,ds:DriveControl and al,ah mov dx,3F2h mov al,0 out dx,al mov ds:DriveControl,al floppy_super_motor_next: sti inc bx shl ah,1 loop floppy_super_motor_loop ; mov bx,ds:FloppyThread or bx,bx jz floppy_super_wait ; cli mov al,ds:IntFlag or al,al jnz floppy_super_signal ; mov al,ds:TimeoutCount sub al,1 mov ds:TimeoutCount,al jnz floppy_super_wait floppy_super_signal: Signal floppy_super_wait: sti mov ax,100 WaitMilliSec jmp floppy_super_loop ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: read_drive ; ; DESCRIPTION: Read drive ; ; PARAMETERS: FS Disc selector ; EDI Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; read_drive Proc near push cx mov cx,3 read_drive_loop: push eax push cx push edx push edi ; movzx eax,es:[edi].dh_sector movzx edx,es:[edi].dh_unit mov edi,es:[edi].dh_data ; div byte ptr fs:boot_sectors_per_cyl mov dh,al xchg ah,dl inc dl mov al,fs:boot_drive_nr ; mov cx,200h mov ebx,edi call ReadDrive ; pop edi pop edx pop cx pop eax jnc read_drive_ok ; call check_media jc read_drive_done ; loop read_drive_loop ; mov es:[edi].dh_state,STATE_BAD stc jmp read_drive_done read_drive_ok: mov es:[edi].dh_state,STATE_USED read_drive_done: pop cx ret read_drive Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: write_drive ; ; DESCRIPTION: Perform a write request ; ; PARAMETERS: DS Disc selector ; EDI Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; write_drive Proc near mov es:[edi].dh_state,STATE_USED push cx ; or dx,dx jnz write_drive_do ; or ax,ax jnz write_drive_do ; int 3 stc jmp write_drive_done write_drive_do: mov cx,3 write_drive_loop: push eax push cx push edx push edi ; movzx eax,es:[edi].dh_sector movzx edx,es:[edi].dh_unit mov edi,es:[edi].dh_data ; div byte ptr fs:boot_sectors_per_cyl mov dh,al xchg ah,dl inc dl mov al,fs:boot_drive_nr ; mov cx,200h mov ebx,edi call WriteDrive ; pop edi pop edx pop cx pop eax jnc write_drive_done ; call check_media jc write_drive_done ; loop write_drive_loop ; stc write_drive_done: pop cx ret write_drive Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: perform_one ; ; DESCRIPTION: Perform one request ; ; PARAMETERS: FS Disc selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; perform_one Proc near perform_one_loop: GetDiscRequest jc perform_one_done ; mov al,fs:disc_sub_unit movzx bx,al mov bl,ds:[bx].BootValid or bl,bl jnz perform_mounted ; push es push edi mov bx,fs mov es,bx mov al,es:disc_sub_unit mov edi,OFFSET boot_sect call ReadBootSector pop edi pop es perform_mounted: mov al,es:[edi].dh_state cmp al,STATE_EMPTY je perform_one_read ; cmp al,STATE_DIRTY je perform_one_write ; cmp al,STATE_SEQ jne perform_one_done perform_one_write: call check_media jc perform_one_fail ; call write_drive jc perform_one_fail jmp perform_one_completed perform_one_read: call check_media jc perform_one_fail ; call read_drive jc perform_one_fail perform_one_completed: mov bx,fs:disc_sel DiscRequestCompleted jmp perform_one_loop perform_one_fail: int 3 mov bx,fs:disc_sel DiscRequestCompleted ; FlushDisc perform_one_done: ret perform_one Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DISCBUF_THREAD ; ; DESCRIPTION: Thread to handle disc buffer queue ; ; PARAMETERS: FS Disc handle ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; discbuf_thread: GetThread mov fs:disc_thread,ax mov bx,fs:disc_sel ; mov ax,floppy_data_sel mov ds,ax mov ax,flat_sel mov es,ax ; WaitForSignal discbuf_thread_loop: WaitForDiscRequest EnterSection ds:FloppySection call perform_one LeaveSection ds:FloppySection jmp discbuf_thread_loop ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INSTALL_UNIT ; ; DESCRIPTION: Install a disk unit ; ; PARAMETERS: AL UNIT # ; DI NAME ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; install_unit Proc near movzx si,al add si,si add si,OFFSET DiscArr push ax mov eax,SIZE disc_struc AllocateSmallGlobalMem ; push ax push di xor di,di mov cx,128 mov eax,0FFFFFFFFh rep stosd pop di pop ax ; mov bx,es mov fs,bx pop ax mov fs:disc_sub_unit,al InitSection fs:disc_section ; mov ecx,200h mov bx,fs InstallDisc mov fs:disc_sel,bx mov fs:disc_nr,al mov ds:[si],fs ; push di mov ax,cs mov es,ax mov di,OFFSET check_media_proc RegisterDiscChange pop di ; mov ax,cs mov ds,ax mov es,ax mov si,OFFSET discbuf_thread mov ax,2 mov cx,100h CreateThread ret install_unit Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: disc_assign ; ; DESCRIPTION: Assign discs ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; floppy0 DB 'Floppy Drive 0',0 floppy1 DB 'Floppy Drive 1',0 disc_assign Proc far push ds push es pusha ; mov ax,cs mov ds,ax mov es,ax mov si,OFFSET floppy_super mov di,OFFSET floppy_super_name mov ax,4 mov cx,1024 CreateThread ; in al,INT0_MASK and al,NOT 40h out INT0_MASK,al ; mov ax,floppy_data_sel mov ds,ax call ResetController ; mov al,0 mov di,OFFSET floppy0 call install_unit mov al,1 mov di,OFFSET floppy1 ; call install_unit ; popa pop es pop ds ret disc_assign Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: GetFloppyDisc ; ; description: Get disc # for a physical disc unit ; ; PARAMETERS: BL Floppy disc # ; ; RETURNS: AL disc # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; get_floppy_disc_name DB 'Get Floppy Disc',0 get_floppy_disc Proc far push ds push bx ; mov ax,floppy_data_sel mov ds,ax cmp bl,2 jae get_floppy_fail ; movzx bx,bl add bx,bx mov bx,ds:[bx].DiscArr or bx,bx jz get_floppy_fail ; mov ds,bx mov al,ds:disc_nr clc jmp get_floppy_done get_floppy_fail: stc get_floppy_done: pop bx pop ds retf32 get_floppy_disc Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: INIT ; ; DESCRIPTION: Init device ; ; PARAMETERS: ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; disc_ctrl: dct00 DW OFFSET disc_assign, floppy_code_sel dct01 DW OFFSET drive_assign1, floppy_code_sel dct02 DW OFFSET drive_assign2, floppy_code_sel dct03 DW OFFSET demand_mount, floppy_code_sel dct04 DW OFFSET erase, floppy_code_sel init PROC far push ds push es pusha mov bx,floppy_code_sel InitDevice ; mov al,0 AllocateFixedDrive mov al,1 AllocateFixedDrive ; mov ax,start_floppy_nr IsValidOsGate jc open_floppy_started ; StartFloppy open_floppy_started: mov dx,3F4h in al,dx cmp al,-1 je init_floppy_done ; mov eax,SIZE floppy_data mov bx,floppy_data_sel AllocateFixedSystemMem xor di,di mov cx,ax xor al,al rep stosb ; mov ax,cs mov ds,ax mov es,ax ; mov si,OFFSET get_floppy_disc mov di,OFFSET get_floppy_disc_name xor dx,dx mov ax,get_floppy_disc_nr RegisterBimodalUserGate ; mov di,OFFSET disc_ctrl HookInitDisc ; mov al,6 mov bx,floppy_data_sel mov ds,bx mov bx,cs mov es,bx mov di,OFFSET floppy_int RequestPrivateIrqHandler init_floppy_done: popa pop es pop ds ret init ENDP code ENDS END init