中国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 楼』:  

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
查看资料  发短消息  网志   编辑帖子  回复  引用回复

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


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



论坛跳转: