中国DOS联盟论坛

中国DOS联盟

-- 联合DOS 推动DOS 发展DOS --

联盟域名:www.cn-dos.net  论坛域名:www.cn-dos.net/forum
DOS,代表着自由开放与发展,我们努力起来,学习FreeDOS和Linux的自由开放与GNU精神,共同创造和发展美好的自由与GNU GPL世界吧!

游客:  注册 | 登录 | 命令行 | 会员 | 搜索 | 上传 | 帮助 »
中国DOS联盟论坛 » DOS开发编程 & 发展交流 (开发室) » DOS下用ATA命令如何读取硬盘信息呢
作者:
标题: DOS下用ATA命令如何读取硬盘信息呢 上一主题 | 下一主题
Joyoung
初级用户





积分 48
发帖 18
注册 2008-9-30
状态 离线
『楼 主』:  DOS下用ATA命令如何读取硬盘信息呢

请问高手:
在DOS下编程如何实现用ATA命令来获取硬盘信息(如:容量大小,厂商,FW版本...)呢?
我是个新手,你们可否贴些源码例子出来,或者最好把源码例子发到我邮箱里,小弟感激不尽呀
邮箱:ayouth2004@21cn.com

2008-10-5 20:58
查看资料  发送邮件  发短消息 网志   编辑帖子  回复  引用回复
Joyoung
初级用户





积分 48
发帖 18
注册 2008-9-30
状态 离线
『第 2 楼』:  

以下代码是从网上找到的,但有几个地方(红色)不是很清楚,请大家指点:
#include <stdlib.h>
#include <dos.h>
#include <stdio.h>
#include <conio.h>
#include <bios.h>

char *GetAscii(unsigned int inData[], int offStart, int offEnd);

int main(void)
{
   unsigned int   diskData[256];    /* Disk data                  */
   unsigned int   offset;           /* Disk data offset           */
   int            loop;
   int            numDrv;           /* Number of IDE hard drives  */
   union REGS     registers;
   unsigned int   biosCyl[2];       /* Cylinders, Heads, Sectors */
   unsigned int   biosHead[2];
   unsigned int   biosSec [2];

   numDrv = peekb(0x40, 0x75); /*0x40和0x75是从哪得到的?在哪里定义的?有什么标准文件说明吗?*/   
   for (loop = 0; loop < numDrv; loop++)
   {
      while (inp(0x01f7) != 0x50); /*0x50是从哪得到的?在哪里定义的?有什么标准文件说明吗?*/      
      outp(0x01f6, (loop == 0 ? 0xa0 : 0xb0));  /*0xa0和0xb0是从哪得到的?在哪里定义的?有什么标准文件说明吗?*/
      outp(0x01f7, 0xec);                       /* Get drive info data      */

      while (inp(0x1f7) != 0x58);   /*0x58是从哪得到的?在哪里定义的?有什么标准文件说明吗?*/      
      for (offset = 0; offset != 256; offset++) /* Read "sector"            */
         diskData[offset] = inpw(0x1f0);

      /* Get BIOS drive info */
      registers.h.ah = 0x08;        /* Get drive info                          */
      registers.h.dl = 0x80 + loop; /* Drive is 80H for Disk 0, 81H for Disk 1 */
      int86(0x13, &registers, &registers);
      if (!registers.x.cflag)       /* All OK if carry not set */
      {
         biosHead[loop] = registers.h.dh + 1;      /* Heads are from 0 */
         biosSec[loop]  = registers.h.cl & 0x3f;   /* sec is bits 5 - 0 */

         /* +1 because starts from 0 and +1 for FDISK leaving one out */
         biosCyl[loop]  = ((registers.h.cl & 0xc0) << 2) + registers.h.ch + 2;
      } /* end of if */

      printf("DRIVE %d:\n", loop);
      printf("Model Number______________________: %s\n", GetAscii(diskData, 27, 46));
      printf("Serial Number_____________________: %s\n", GetAscii(diskData, 10, 19));
      printf("Controller Revision Number________: %s\n\n", GetAscii(diskData, 23, 26));
      printf("Able to do Double Word Transfer___: %6s\n", (diskData[48] == 0 ? "No" : "Yes"));
      printf("Controller type___________________:   %04X\n", diskData[20]);
      printf("Controller buffer size (bytes)____: %6u\n", diskData[21] * 512);
      printf("Number of ECC bytes transferred___: %6u\n", diskData[22]);
      printf("Number of sectors per interrupt___: %6u\n\n", diskData[47]);
      printf("Hard Disk Reports\n");
      printf("Number of Cylinders (Fixed)_______: %6u\n", diskData[1]);
      printf("Number of Heads___________________: %6u\n", diskData[3]);
      printf("Number of Sectors per Track_______: %6u\n\n", diskData[6]);
      printf("BIOS Reports\n");
      printf("Number of Cylinders_______________: %6u\n", biosCyl[loop]);
      printf("Number of Heads___________________: %6u\n", biosHead[loop]);
      printf("Number of Sectors per Track_______: %6u\n\n", biosSec[loop]);

      printf("Press any key to continue...\n\n");
      getch();
   } /* end of for */

   return 0;
} /* main() */

char *GetAscii(unsigned int inData[], int offStart, int offEnd)
{
   static char retVal[255];
   int         loop, loop1;

   for (loop = offStart, loop1 = 0; loop <= offEnd; loop++)
   {
      retVal[loop1++] = (char )(inData[loop] / 256);  /* Get High byte */
      retVal[loop1++] = (char )(inData[loop] % 256);  /* Get Low byte  */
   } /* end of for */

   retVal[loop1] = '\0';    /* Make sure it ends in a NULL character */

   return retVal;
} /* GetAscii() */

我运行了以上代码,停在0X50和0X58那两个循环地方,不明白是怎么回事?哪位高手知道的,可否告诉我呢?谢谢!~~~

2008-10-10 17:12
查看资料  发送邮件  发短消息 网志   编辑帖子  回复  引用回复
netwinxp
高级用户





积分 741
发帖 366
注册 2007-7-25
状态 离线
『第 3 楼』:  

只要愿意学,一切都好办多了^_^
1、peekb(段地址,偏移量)功能是获取段地址:[偏移量]的内容,0040:0000~0040:01FF为BIOS数据区,0040:0075存放的是硬盘数量。你可以参考"BIOS DATA AREA"
2、3、4、在ATA命令中有定义,你可以参考"AT Attachment with Packet Interface"
50H、58H在错误标志、状态标志里面有描述。50H表示有硬盘,58H表示数据已准备好。
1F6H在ATA命令包中存放DEV段和部分LBA地址的bit24~27(这4位在你的这个程序中不用考虑),A0H=Master,B0H=Slave(SATA只有单硬盘,不要用这个);1F0H:数据口、1F7H:状态口/命令口。
ECH在ATA命令中为"IDENTIFY DEVICE",也就是读取硬盘信息。

PS:ATA命令是由硬盘完成的,磁盘控制器只不过是提供通道而已;1F?是主通道(PATA硬盘控制器)或者Port1(SATA硬盘控制器)的默认I/O口,实际有些并不一定是这个值,需要通过PCICFG读取0101(PATA/SATA IDE MODE)、0106(AHCI)、0104(RAID)这些类型的设备,并从其256字节PCI配置空间得到相应的I/O基址为准。
C语言俺很久没用了,死循环的话你检查一下循环语句。

[ Last edited by netwinxp on 2008-10-13 at 01:12 ]

2008-10-13 00:59
查看资料  发短消息 网志   编辑帖子  回复  引用回复
Joyoung
初级用户





积分 48
发帖 18
注册 2008-9-30
状态 离线
『第 4 楼』:  



  Quote:
Originally posted by netwinxp at 2008-10-13 00:59:
只要愿意学,一切都好办多了^_^
1、peekb(段地址,偏移量)功能是获取段地址:[偏移量]的内容,0040:0000~0040:01FF为BIOS数据区,0040:0075存放的是硬盘数量。你 ...

netwinxp兄:
   你好,请问你看的AT Attachment with Packet Interface(ATAPI)是哪个版本的?我只有ATAPI-6,这个里我没有找到0X50H和0X58H的相关说明呀,你可否发你的那份ATAPI文档到我邮箱呢?谢谢!~~~~
    程序的死循环也就是在这两个地方里,会不会是这两个数字已经改为其他的了呢?
ayouth2004@21cn.com

2008-10-14 16:53
查看资料  发送邮件  发短消息 网志   编辑帖子  回复  引用回复
netwinxp
高级用户





积分 741
发帖 366
注册 2007-7-25
状态 离线
『第 5 楼』:  

ATA-6里面也有,它就是Status register的内容,不可能会变化的。
Status Register:
bits7:BSY(Busy)。
bits6:DRDY(Device Ready)。
bits5:DF(Device Fault)。
bits4:与命令有关,一般和DSC(Device Seek Complete)同。
bits3:DRQ(Data Request)。
bits2:保留。
bits1:保留。
bits0:ERR(Error)。

于是50H不就是设备准备好,58H不就是数据准备好。
死循环是原来程序的问题,如果遇到主通道的主或从设备不存在,不就死在那了,另外从通道的I/O口是17?关1F?什么事。
原来程序存在严重的考虑不周,如果你要测试的话,建议只保留主通道主设备上的一个硬盘,其他的都不接。

原来的while(XXX!=0x50);遇到一直不等必然就在那里死循环(网上好几个其他类似的也存在这个问题),比较正常的做法是延时若干时间,然后判断DRDY位为1的话就执行后面的语句,否则返回上一级。
同样道理0x58也容易死循环,改成延时一段时间然后判断DRDY和DRQ两个位,如果不都为1,则返回上一级。

[ Last edited by netwinxp on 2008-10-15 at 10:38 ]

2008-10-15 10:24
查看资料  发短消息 网志   编辑帖子  回复  引用回复

请注意:您目前尚未注册或登录,请您注册登录以使用论坛的各项功能,例如发表和回复帖子等。


可打印版本 | 推荐给朋友 | 订阅主题 | 收藏主题



论坛跳转: