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那两个循环地方,不明白是怎么回事?哪位高手知道的,可否告诉我呢?谢谢!~~~作者: netwinxp 时间: 2008-10-13 00:59 只要愿意学,一切都好办多了^_^
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 ]作者: Joyoung 时间: 2008-10-14 16:53
Quote:
Originally posted by netwinxp at 2008-10-13 00:59:
只要愿意学,一切都好办多了^_^
1、peekb(段地址,偏移量)功能是获取段地址:[偏移量]的内容,0040:0000~0040:01FF为BIOS数据区,0040:0075存放的是硬盘数量。你 ...