Question about which tools to use, bugs, the best way to implement a function, etc should go here. Don't forget to see if your question is answered in the wiki first! When in doubt post here.
I am try to get drive detection working. I am using QEMU to run my OS.
Even though I have only set up 1 drive in QEMU my drive detection code always reports that the primary slave exists. I can't figure out what I am doing wrong. Here is my drive detection routine. I call this routine 4 times. Once for each combination of I/O port and master/slave. As I mentioned the primary slave is being reported as present and when I dump the fields from the IDENTIFY buffer it seems like it is reporting the same values as the master. i.e. the max LBA is the same for both drives. It reports 25 for both drives which is the size of my raw disk image in sectors.
int err = 0; // debugging
int timeout = 0;
while (1 && timeout < 10000) {
// ATAPI devices are supposed to set ERR rather than BSY, and then DRQ
// This is also needed to get CYL_LO and CYL_HI to be reads
// NOTE: Some ATAPI devices do not set ERR (for some reason). Therefore we do the port check anyways to determine whether the device is ATAPI
uint8_t status = ide_read(device, ATA_REG_STATUS);
if (status & ATA_SR_ERR) {
// ERR was set
err = 1;
break;
} else if (!(status & ATA_SR_BSY) && (status & ATA_SR_DRQ)) {
// DRQ was set and BSY is cleared, likely an ATA device instead
break;
}
timeout++; // Keep waiting...
}
// Did it timeout?
if (timeout >= 10000) {
LOG_DEVICE(INFO, device, "Timeout while waiting for ATA_CMD_IDENTIFY - assuming dead\n");
return;
}
Adding a timeout should fix your code. QEMU seems to send the right bytes regardless of if there's actually a drive there, but if the command times out and BSY still isn't clear I'd abandon ship.
I'm not an expert on this, so it might be helpful to check what Bochs does to confirm your code. Hope this helps!
After you select drive, you should read status port. If it equals 0xFF, you can be sure that there is no drive and you should not send IDENTIFY command at all. After you send IDENTIFY command, you need to wait for BSY to be clear and DRQ to be set, not only for BSY to be clear as you do in your code.
Also, I would stongly recommend to not use while() cycles without timeout. You should not assume anything about anything in OS development. If for whatever reason your environment will not change bits, your OS will be locked. It is worth few additional lines of code to have protection against such situation.
Klakap wrote: ↑Sun Feb 23, 2025 12:18 pm
After you select drive, you should read status port. If it equals 0xFF, you can be sure that there is no drive and you should not send IDENTIFY command at all. After you send IDENTIFY command, you need to wait for BSY to be clear and DRQ to be set, not only for BSY to be clear as you do in your code.
Tried that and it didn't make any difference. What seems to have fixed the problem is adding a check for a status of zero after the drive select. Now when I do my drive selection I find the primary master. The primary slave fails with a status of zero. The secondary master fails with a status of 0x41 (RDY | ERR) and the secondary slave fails with a status of zero.
Also, I would stongly recommend to not use while() cycles without timeout. You should not assume anything about anything in OS development. If for whatever reason your environment will not change bits, your OS will be locked. It is worth few additional lines of code to have protection against such situation.