This repository has been archived on 2025-02-27. You can view files and clone it, but cannot push or open issues or pull requests.
portland-os/src/kernel/ide.c

191 lines
No EOL
5.3 KiB
C

#include <stdbool.h>
#include <stdint.h>
#include "drive.h"
#include "panic.h"
#include "util.h"
#include "ata.h"
#include "pci.h"
#define MAX_IDE_DRIVES 8
struct ide_drive_info {
uint16_t base_port;
uint16_t alt_port;
bool slave;
};
static struct ide_drive_info ide_drives[MAX_IDE_DRIVES];
static drive_id_t n_ide_drives = 0;
typedef uint16_t spinner_t;
//returns the status after waiting
static uint8_t wait_for_ready(uint16_t base_port) {
for (spinner_t n = -1; n; --n) {
uint8_t s = inb(base_port | ATA_REG_STATUS);
if (s & ATA_STATUS_ERROR)
panic("Error status in IDE driver.");
if (s & ATA_STATUS_DRIVE_READY)
return s;
}
panic("Spun out in IDE driver.");
}
//returns the status after waiting
static uint8_t wait_for_error_or_ready(uint16_t base_port) {
for (spinner_t n = -1; n; --n) {
uint8_t s = inb(base_port | ATA_REG_STATUS);
if (s & (ATA_STATUS_DRIVE_READY | ATA_STATUS_ERROR))
return s;
}
panic("Spun out in IDE driver.");
}
//returns the status after waiting
static uint8_t wait_for_data_ready_not_busy(uint16_t base_port) {
for (spinner_t n = -1; n; --n) {
uint8_t s = inb(base_port | ATA_REG_STATUS);
if (!(s & ATA_STATUS_BUSY) && (s & ATA_STATUS_DATA_READY))
return s;
if (s & ATA_STATUS_ERROR)
panic("Error status in IDE driver.");
}
panic("Spun out in IDE driver.");
}
static uint8_t ide_ata_rs(const struct drive *d, uint32_t start, uint32_t count, void *buffer) {
if (start >= d->n_sectors)
return 0;
if (start + count > d->n_sectors)
count = d->n_sectors - start;
if (!count)
return 0;
if (start & 0xf0000000)
panic("IDE ATA driver does not support reads starting past 256MiB currently.");
if (count & 0xffffff00)
panic("IDE ATA driver does not support reads over 128kiB in length currently.");
uint32_t lba = start & 0x00ffffff;
struct ide_drive_info *info = ide_drives + d->drive_id;
outb(info->base_port | ATA_REG_SELECT, (info->slave ? 0xf0 : 0xe0) | (start >> 24));
//spin?
outb(info->base_port | ATA_REG_COUNT, count);
outb(info->base_port | ATA_REG_LBA0, lba);
outb(info->base_port | ATA_REG_LBA1, lba >> 8);
outb(info->base_port | ATA_REG_LBA2, lba >> 16);
outb(info->base_port | ATA_REG_CMD, ATA_CMD_READ);
uint16_t *buf16 = (uint16_t *)buffer;
for (uint16_t i = 0; i < count; ++i) {
wait_for_data_ready_not_busy(info->base_port);
for (uint16_t j = 0; j < 256; ++j)
*buf16++ = inw(info->base_port | ATA_REG_DATA);
}
return count;
}
static uint8_t ide_ata_ws(const struct drive *d, uint32_t start, uint32_t count, const void *buffer) {
panic("IDE ATA writing not implemented yet");
return 0;
}
static uint8_t ide_atapi_rs(const struct drive *d, uint32_t start, uint32_t count, void *buffer) {
//panic("IDE ATAPI reading not implemented yet");
return 0;
}
static uint8_t ide_atapi_ws(const struct drive *d, uint32_t start, uint32_t count, const void *buffer) {
panic("IDE ATAPI writing not implemented yet");
return 0;
}
struct id_space {
uint8_t skip1[120];
uint32_t max_lba;//120
uint8_t skip2[512 - (120 + 4)];
} __attribute__ ((__packed__ ));
static void test_drive(uint16_t base_port, uint16_t alt_port, bool slave) {
if (n_ide_drives == MAX_IDE_DRIVES)
panic("Maximum number of IDE drives reached.");
struct ide_drive_info *next = ide_drives + n_ide_drives;
struct drive next_d;
next_d.drive_id = n_ide_drives;
next->base_port = base_port;
next->alt_port = alt_port;
next->slave = slave;
outb(base_port | ATA_REG_SELECT, slave ? 0xb0 : 0xa0);
outb(base_port | ATA_REG_COUNT, 0);
outb(base_port | ATA_REG_LBA0, 0);
outb(base_port | ATA_REG_LBA1, 0);
outb(base_port | ATA_REG_LBA2, 0);
outb(base_port | ATA_REG_CMD, ATA_CMD_ID);
if (!inb(base_port | ATA_REG_STATUS))
return;
struct id_space ids;
if (!(wait_for_error_or_ready(base_port) & ATA_STATUS_ERROR)) {
wait_for_ready(base_port);
for (uint16_t *ids_16 = (uint16_t *)&ids; ids_16 < (uint16_t *)&ids + 256; ++ids_16)
*ids_16 = inw(base_port | ATA_REG_DATA);
next_d.drive_type = "IDE PATA";
next_d.read_sectors = &ide_ata_rs;
next_d.write_sectors = &ide_ata_ws;
}
else {
uint16_t code = inb(base_port | 0x4) + (inb(base_port | 0x5) << 8);
if (!code) {
panic("PATA identification aborted.");
}
if (code == 0xeb14) {
next_d.drive_type = "IDE PATAPI";
next_d.read_sectors = &ide_atapi_rs;
next_d.write_sectors = &ide_atapi_ws;
//TODO: issue proper identify command
ids.max_lba = -1;
}
else if (code == 0xc33c) {
next_d.drive_type = "IDE SATA";
//TODO: set up drive hooks, issue proper identify command
return;
}
}
//in the future, maybe add support for 48-bit LBA, and/or CHS addressing
if (!ids.max_lba)
panic("Encountered ATA drive that doesn't support 28-bit LBA");
next_d.n_sectors = ids.max_lba;
commit_drive(next_d);
++n_ide_drives;
}
void init_ide() {
uint16_t check_from = 0;
struct pci_device *device;
//double parentheses to let gcc know this assignment isn't a mistake
while ((device = find_pci_device_from_class_and_subclass(PCI_MASS_STORAGE, PCI_IDE, check_from, &check_from))) {
++check_from;
test_drive(0x01f0, 0x03f6, false);
test_drive(0x01f0, 0x03f6, true);
test_drive(0x0170, 0x0376, false);
test_drive(0x0170, 0x0376, true);
}
}