BIOS Interrupt to Read Sector, Get Disk Status, Find Errors (37345)
The information in this article applies to:
- Microsoft Visual Basic for MS-DOS
- Microsoft QuickBASIC 4.0
- Microsoft QuickBASIC 4.0b
- Microsoft QuickBASIC 4.5
- Microsoft BASIC Compiler for MS-DOS and OS/2 6.0
- Microsoft BASIC Compiler for MS-DOS and OS/2 6.0b
- Microsoft Basic Professional Development System (PDS) for MS-DOS and MS OS/2 7.0
- Microsoft Basic Professional Development System (PDS) for MS-DOS and MS OS/2 7.1
This article was previously published under Q37345 SUMMARY
The ROM BIOS Interrupt 19 decimal (13 hex), with function 2, returns
the status of a specified disk if it is unable to read the specified
sector. This interrupt can be used to determine if the drive door is
open, if the disk is formatted, if the disk is write-protected, and
many other conditions. The code generated for this is smaller than a
similar program using ON ERROR GOTO statements to trap disk errors.
Note that ROM BIOS and MS-DOS Interrupts are not allowed in MS OS/2
protected mode.
MORE INFORMATION
ROM BIOS Interrupt call 19 provides a set of functions to access the
disk driver. Interrupt 19 with function 2 reads one or more sectors
from disk into memory. Interrupt 19 with function 0 (reset disk system)
should be called after a failed floppy disk Read, Write, Verify, or
Format request before retrying the operation. For more information
about this and other interrupts for IBM ROM BIOS and MS-DOS, see
"Advanced MS-DOS Programming, Second Edition," by Ray Duncan
(Microsoft Press, 1988).
Unlike other MS-DOS interrupt functions that access the disk drive using
CALL INTERRUPT or CALL INTERRUPTX in Microsoft QuickBasic for MS-DOS,
versions 4.0, 4.0b, or 4.5; or Microsoft Basic Compiler for MS-DOS or
MS OS/2 (real mode only), versions 6.0 or 6.0b; Interrupt 19 with function
2 and 0 does NOT cause a system hang if a critical error occurs (such as
if a drive door is open or a disk is not formatted).
Note: The CALL INTERRUPT or CALL INTERRUPTX statement hanging upon
critical disk errors is a known problem with the INTERRUPT and
INTERRUPTX routines in Microsoft QuickBasic for MS-DOS, versions 4.0,
4.0b, and 4.5; and Microsoft Basic Compiler for MS-DOS and MS OS/2
(real mode only), versions 6.0 and 6.0b. This problem was corrected in
Microsoft Basic PDS for MS-DOS and MS OS/2 (real mode only),
version 7.0. For information on how to modify the INTERRUPT and
INTERRUPTX routines so that critical errors will not hang the machine
in Microsoft QuickBasic for MS-DOS, versions 4.0, 4.0b, and 4.5; and
Microsoft Basic Compiler for MS-DOS and MS OS/2 (real mode only),
versions 6.0 and 6.0b; query the Knowledge Base on
the following keyword:
Listed below is a code example for calling ROM BIOS Interrupt call 19
decimal (13h) with functions 0 and 2. In order to run this program in
the QB.EXE environment for Microsoft QuickBasic (or QBX.EXE for
Microsoft Basic PDS), you must invoke QB /L QB.QLB (or QBX /L QBX.QLB
for Microsoft Basic PDS). To link this program into an .EXE program, you
must link with QB.LIB (or QBX.LIB for Basic PDS) (which is done
automatically when you choose the Make EXE File command from the File
menu in the environment).
Example
' To try this example in VBDOS.EXE:
' 1. From the File menu, choose New Project.
' 2. Copy the code example to the Code window.
' 3. Press F5 to run the program.
'
' To run this program in the encironment, you must invoke the
' environment with ther /L switch to load the default Quick library:
' VBDOS.EXE /L for Visual Basic 1.0 for MS-DOS
'
DECLARE SUB ResetDrive (drive%)
DECLARE SUB ReadSector (drive%, sector%)
' Use the following include file for Visual Basic 1.0 for MS-DOS:
REM $INCLUDE: 'VBDOS.BI'
' Use the following include file for QuickBasic for MS-DOS:
REM $INCLUDE: 'QB.BI'
' Use the following include file for Basic PDS for MS-DOS:
REM $INCLUDE: 'QBX.BI'
DIM reg AS RegTypeX, regS AS RegType
DIM databuffer%(5120) ' Room for 10 sectors of data.
sector% = 1 ' Sector number 1-9.
drive% = 0 ' Drive A.
FOR i = 1 TO 3
CALL ReadSector(drive%, sector%)
CALL ResetDrive(drive%)
PRINT reg.flags, (reg.flags AND 1)
IF (reg.flags AND 1) = 1 THEN
errornum = reg.ax AND &HFF
ELSE
errornum = 0
END IF
NEXT
PRINT errornum
END
SUB ReadSector (drive%, sector%)
SHARED databuffer%(), reg AS RegTypeX
intnum% = 19 ' Interrupt number 19 decimal (13 hex).
numsectors% = 1 ' # read 1,8, or 9.
track% = 0 ' track# 0-39.
head% = 0 ' side 0 or 1.
reg.ds = -1
reg.es = VARSEG(databuffer%(0))
reg.bx = VARPTR(databuffer%(0))
reg.cx = 256 * track% + sector%
reg.dx = 256 * head% + drive%
' AH loaded with function 2; AL loaded with number of sectors.
reg.ax = 2 * 256 + numsectors% ' AX register contains AH and AL
CALL interruptx(intnum%, reg, reg)
END SUB
SUB ResetDrive (drive%)
DIM reg AS RegTypeX
reg.ax = 0 ' function number 0
reg.dx = drive%
CALL interruptx(&H13, reg, reg)
END SUB
Modification Type: | Minor | Last Reviewed: | 8/16/2005 |
---|
Keywords: | KB37345 |
---|
|