forked from Bananymous/banan-os
Kernel: Implement more error handling in IDE controller
This commit is contained in:
parent
ff2486f58c
commit
4be726b130
|
@ -11,6 +11,8 @@
|
||||||
namespace Kernel
|
namespace Kernel
|
||||||
{
|
{
|
||||||
|
|
||||||
|
static constexpr uint64_t s_ata_timeout_ms = 100;
|
||||||
|
|
||||||
BAN::ErrorOr<BAN::RefPtr<ATABus>> ATABus::create(uint16_t base, uint16_t ctrl, uint8_t irq)
|
BAN::ErrorOr<BAN::RefPtr<ATABus>> ATABus::create(uint16_t base, uint16_t ctrl, uint8_t irq)
|
||||||
{
|
{
|
||||||
auto* bus_ptr = new ATABus(base, ctrl);
|
auto* bus_ptr = new ATABus(base, ctrl);
|
||||||
|
@ -73,16 +75,37 @@ namespace Kernel
|
||||||
select_delay();
|
select_delay();
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool identify_all_ones(BAN::Span<const uint16_t> identify_data)
|
static bool identify_all_same(BAN::Span<const uint16_t> identify_data)
|
||||||
{
|
{
|
||||||
for (size_t i = 0; i < 256; i++)
|
uint16_t value = identify_data[0];
|
||||||
if (identify_data[i] != 0xFFFF)
|
for (size_t i = 1; i < 256; i++)
|
||||||
|
if (identify_data[i] != value)
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<ATABus::DeviceType> ATABus::identify(bool secondary, BAN::Span<uint16_t> buffer)
|
BAN::ErrorOr<ATABus::DeviceType> ATABus::identify(bool secondary, BAN::Span<uint16_t> buffer)
|
||||||
{
|
{
|
||||||
|
// Try to detect whether port contains device
|
||||||
|
uint8_t status = io_read(ATA_PORT_STATUS);
|
||||||
|
if (status & ATA_STATUS_BSY)
|
||||||
|
{
|
||||||
|
uint64_t timeout = SystemTimer::get().ms_since_boot() + s_ata_timeout_ms;
|
||||||
|
while ((status = io_read(ATA_PORT_STATUS)) & ATA_STATUS_BSY)
|
||||||
|
{
|
||||||
|
if (SystemTimer::get().ms_since_boot() >= timeout)
|
||||||
|
{
|
||||||
|
dprintln("BSY flag clear timeout, assuming no drive on port");
|
||||||
|
return BAN::Error::from_errno(ETIMEDOUT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (__builtin_popcount(status) >= 4)
|
||||||
|
{
|
||||||
|
dprintln("STATUS contains garbage, assuming no drive on port");
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
select_device(secondary);
|
select_device(secondary);
|
||||||
|
|
||||||
// Disable interrupts
|
// Disable interrupts
|
||||||
|
@ -125,7 +148,7 @@ namespace Kernel
|
||||||
ASSERT(buffer.size() >= 256);
|
ASSERT(buffer.size() >= 256);
|
||||||
read_buffer(ATA_PORT_DATA, buffer.data(), 256);
|
read_buffer(ATA_PORT_DATA, buffer.data(), 256);
|
||||||
|
|
||||||
if (identify_all_ones(buffer))
|
if (identify_all_same(buffer))
|
||||||
return BAN::Error::from_errno(ENODEV);
|
return BAN::Error::from_errno(ENODEV);
|
||||||
|
|
||||||
return type;
|
return type;
|
||||||
|
@ -189,12 +212,17 @@ namespace Kernel
|
||||||
for (uint32_t i = 0; i < 4; i++)
|
for (uint32_t i = 0; i < 4; i++)
|
||||||
io_read(ATA_PORT_ALT_STATUS);
|
io_read(ATA_PORT_ALT_STATUS);
|
||||||
|
|
||||||
uint8_t status = ATA_STATUS_BSY;
|
uint64_t timeout = SystemTimer::get().ms_since_boot() + s_ata_timeout_ms;
|
||||||
while (status & ATA_STATUS_BSY)
|
|
||||||
status = io_read(ATA_PORT_STATUS);
|
uint8_t status;
|
||||||
|
while ((status = io_read(ATA_PORT_STATUS)) & ATA_STATUS_BSY)
|
||||||
|
if (SystemTimer::get().ms_since_boot() >= timeout)
|
||||||
|
return BAN::Error::from_errno(ETIMEDOUT);
|
||||||
|
|
||||||
while (wait_drq && !(status & ATA_STATUS_DRQ))
|
while (wait_drq && !(status & ATA_STATUS_DRQ))
|
||||||
{
|
{
|
||||||
|
if (SystemTimer::get().ms_since_boot() >= timeout)
|
||||||
|
return BAN::Error::from_errno(ETIMEDOUT);
|
||||||
if (status & ATA_STATUS_ERR)
|
if (status & ATA_STATUS_ERR)
|
||||||
return error();
|
return error();
|
||||||
if (status & ATA_STATUS_DF)
|
if (status & ATA_STATUS_DF)
|
||||||
|
|
Loading…
Reference in New Issue