现在的位置: 首页 > 综合 > 正文

软磁盘感应技术的实现

2013年07月26日 ⁄ 综合 ⁄ 共 2935字 ⁄ 字号 评论关闭

摘要:本文主要介绍C语言高级应用的又一个重要话题:磁盘感应技术的实现。正如HD-COPY软件一样,当你把软盘插入驱动器的时候,它会帮你执行相应的操作,而不需要你再去选择功能。这是怎么实现的呢?程序怎么知道驱动器中是否插有软盘呢?这些都是本文所讨论的内容。

关键词:INT 08HINT 1CHINT 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 13H04H号子功能来检测软盘的001扇区的状态,如果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,&regs,&regs);  /* 调用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 08HINT 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,&regs,&regs);

    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语言实现磁盘感应技术的核心程序,有兴趣的读者可以继续完善上面的程序,使得自己设计的软件有更好的稳定性和实用性。

抱歉!评论已关闭.