|
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, ®isters, ®isters);
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 |
|
|