摘要:本文主要介绍C语言高级应用的又一个重要话题:磁盘感应技术的实现。正如HD-COPY软件一样,当你把软盘插入驱动器的时候,它会帮你执行相应的操作,而不需要你再去选择功能。这是怎么实现的呢?程序怎么知道驱动器中是否插有软盘呢?这些都是本文所讨论的内容。
关键词:INT 08H,INT 1CH,INT 13H,中断调用、磁盘感应技术
很多优秀的软件如HD-COPY等,都具有磁盘感应功能,也就是当软件运行的时候,软驱灯会不停地闪烁以检测软盘是否被装入驱动器,如果这个时候用户向驱动器中插入软盘,程序就会执行响应的操作。那么我们能不能在自己的软件中实现磁盘感应技术呢?答案是肯定的,许多技术文献上都是用汇编语言来实现的,在这里我们不作讨论,我们只分析如何用C语言来实现这一功能。
基本的设计思路如下:当程序执行时不断地读取软盘驱动器的状态,如果状态的返回标记为“驱动器非空”,即状态值不是80H(参见上文的附录),则执行预先定义好的函数,否则继续获得磁盘驱动器的状态。
要获得磁盘驱动器的状态,可以使用BIOS提供的INT 13H中断04H号功能,该功能是用来进行磁盘扇区检测(Verify Disk Sector)的。INT 13H子功能04H的输入参数和输出参数如下:
功能号:INT 13H
子功能:AH=04H
输入参数
AH=04H,AL=扇区数,CH=磁道号,CL=扇区号
DH=磁头号,DL=驱动器代码(0=A驱动器,……)
输出参数
成功:CFLAG=0,AH=0
失败:CFLAG=FFH,AH=错误代码(参考上文附录)
至此,我们的设计思路可以描述为:反复调用INT 13H的04H号子功能来检测软盘的0道0头1扇区的状态,如果AH的返回值不是80H则跳出循环执行相应的功能和操作。为了给软件的用户提供一个换盘的时间间隔,我们可以在每次调用04H号功能前加一个空循环,以此来达到延时的目的。下面是一段用C语言编写的实现磁盘感应功能的程序,其中重要的地方都作了注释,供读者参考。
#include <stdio.h>
#include <dos.h>
int verifydrive ()
{
union REGS regs; /* 定义寄存器变量 */
regs.h.ah=0x04; /* 定义04H号功能并设置参数 */
regs.h.al=1;
regs.h.ch=0;
regs.h.cl=1;
regs.h.dh=0;
regs.h.dl=0;
int86(0x13,®s,®s); /* 调用13H号中断 */
return regs.h.ah; /* 返回AH的值,即将软盘驱动器的状态返回 */
}
void myfunction () /* 这个函数是插入磁盘后要执行的功能,用户自己定义 */
{
/* TODO: Add your program here */
}
void main ()
{
unsigned long i;
while (verifydrive()==0x80)
for (i=0;i<200000;i++); /* 延时 */
myfunction ();/* 调用用户函数 */
}
运行上面的程序后,软盘驱动器一直在工作,直到一张软盘被插入。我们是将磁盘驱动器是否就绪作为是否插入磁盘的标记,磁盘是否损坏不是我们所关心的话题,因此如果向驱动器中插入一张零磁道损坏的软盘,程序照样能正常运行。
我们已经初步实现了磁盘的感应技术,那么还有一个问题就是:对于不同的机器运行循环所用的时间是不一样的,那么对于上面的程序,如果在一般的机器上运行,提供给用户的换盘延时就会比较长,如果在高档机上运行上面的程序,那么提供给用户的延时时间将比较短。如果让磁头始终处于读取的状态,这样在换盘时容易损坏磁盘和磁头。下面介绍一种比较通用的方法来解决这个问题。
先简要介绍一下INT 08H和INT 1CH两个中断。INT 08H是一个时钟中断,计算机运行时,这个中断时时刻刻产生,用来完成系统时钟的刷新。它还有一个特性,就是每秒调用INT 1CH中断约18.2次,该功能是通过对8254系统时钟的0通道读写完成的。对于INT 1CH中断,它只有一条指令:IRET,也就是说,INT 1CH中断什么都不做,仅仅返回函数的调用。如此,我们只要在C语言中将一个计数器counter置于INT 1CH中断中,并让它自加,如果counter自加后的值为18,则将counter清零,也就是说,counter的值由0再次变到0所用的时间大约就是1秒钟。
我们可以将counter定义为全局变量,在main()函数中判断它的值,如果为17,则读取一次软盘驱动器的状态,进而实现我们的磁盘感应技术。下面是一段用C语言调用INT 1CH中断实现磁盘感应技术的程序,重要的地方都作了注释,供读者参考。
#include <stdio.h>
#include <dos.h>
#ifdef __cplusplus /* 判断编译器是否为C++编译器,以确定中断函数参数 */
#define ARGUMENT ...
#else
#define ARGUMENT
#endif
int counter; /* 定义全局变量 counter */
void install (void interrupt (*funaddr)(ARGUMENT), int number) /* 安装中断函数 */
{
disable();
setvect(number, funaddr);
enable();
}
void interrupt new1ch(ARGUMENT) /* 新的INT 1CH 函数 */
{
counter++; /* 计数器自加 */
if (counter==18) counter=0; /* 如果值为18则清零 */
}
int verifydrive () /* 读取磁盘状态函数,同上例 */
{
union REGS regs;
regs.h.ah=0x04;
regs.h.al=1;
regs.h.ch=0;
regs.h.cl=1;
regs.h.dh=0;
regs.h.dl=0;
int86(0x13,®s,®s);
return regs.h.ah;
}
void myfunction () /* 这个函数是插入磁盘后要执行的功能,用户自己定义 */
{
/* TODO: Add your program here */
}
void main ()
{
install (new1ch,0x1c); /* 将新的中断服务程序装入 */
while (1) /* 程序循环 */
if (counter==17) /* 如果计数器为17,证明已经经过约1秒钟 */
if (verifydrive()!=0x80) /* 如果软盘被插入 */
{
myfunction(); /* 调用用户函数 */
break;
}
}
以上的程序仅供参考,它是用C语言实现磁盘感应技术的核心程序,有兴趣的读者可以继续完善上面的程序,使得自己设计的软件有更好的稳定性和实用性。