aboutsummaryrefslogtreecommitdiff
path: root/src/kernel/fs/ata.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/kernel/fs/ata.c')
-rw-r--r--src/kernel/fs/ata.c332
1 files changed, 80 insertions, 252 deletions
diff --git a/src/kernel/fs/ata.c b/src/kernel/fs/ata.c
index b637dbd..d5c758c 100644
--- a/src/kernel/fs/ata.c
+++ b/src/kernel/fs/ata.c
@@ -1,292 +1,120 @@
+#include <stdint.h>
+#include <stddef.h>
+#include <stdbool.h>
+#include <kernel/fs/ata.h>
#include <kernel/system.h>
#include <kernel/io/io.h>
-#include <kernel/fs/ata.h>
-#include <kernel/lib/lib.h>
-#include <kernel/lib/stdlib.h>
-#include <kernel/lib/stdio.h>
-#include <kernel/memory/alloc.h>
-#include <kernel/pci/pci.h>
-#include <kernel/interrupts/interrupts.h>
-uint32_t ata_device = 0x00000000;
+static uint16_t sel_base_port = 0;
+static uint8_t sel_master_or_slave = 0;
-ata_dev_t primary_master = { .slave = 0 };
-ata_dev_t primary_slave = { .slave = 1 };
-ata_dev_t secondary_master = { .slave = 0 };
-ata_dev_t secondary_slave = { .slave = 1 };
-
-void io_wait(ata_dev_t *dev)
-{
- inb(dev->alt_status);
- inb(dev->alt_status);
- inb(dev->alt_status);
- inb(dev->alt_status);
-}
+static uint32_t max_sector;
-void software_reset(ata_dev_t *dev)
+static uint8_t read_stat(uint16_t base)
{
- outb(dev->control, CONTROL_SOFTWARE_RESET);
- io_wait(dev);
- outb(dev->control, CONTROL_ZERO);
-}
+ inb(base + COM_STAT);
+ inb(base + COM_STAT);
+ inb(base + COM_STAT);
+ inb(base + COM_STAT);
-void ata_handler(struct regs *r)
-{
- inb(primary_master.status);
- inb(primary_master.BMR_STATUS);
- outb(primary_master.BMR_COMMAND, BMR_COMMAND_DMA_STOP);
- //irq_ack(14);
+ return inb(base + COM_STAT);
}
-void ata_open(vfs_node_t *node, uint32_t flags)
+static void check_drive(uint16_t base, uint8_t master_or_slave)
{
- return;
-}
+ if (sel_base_port != 0)
+ return;
-void ata_close(vfs_node_t *node)
-{
- return;
-}
+ outb(base + DRIVE_SELECT, master_or_slave);
-uint32_t ata_read(vfs_node_t *node, uint32_t offset, uint32_t size, char *buf)
-{
- uint32_t start = offset / SECTOR_SIZE;
- uint32_t start_offset = offset % SECTOR_SIZE;
+ outb(base + SECTOR_COUNT, 0);
+ outb(base + LBA_LOW, 0);
+ outb(base + LBA_MID, 0);
+ outb(base + LBA_HIGH, 0);
- uint32_t end = (offset + size - 1) / SECTOR_SIZE;
- uint32_t end_offset = (offset + size - 1) % SECTOR_SIZE;
+ outb(base + COM_STAT, IDENTIFY);
+ uint8_t stat = read_stat(base);
- char *buf_curr = buf;
- uint32_t counter = start;
- uint32_t read_size;
- uint32_t off, total = 0;
-
- while (counter <= end) {
- off = 0;
- read_size = SECTOR_SIZE;
+ if (stat == 0)
+ return;
- char *ret = ata_read_sector((ata_dev_t *)node->device, counter);
+ while ((stat & BSY) != 0)
+ stat = read_stat(base);
- if (counter == start) {
- off = start_offset;
- read_size = SECTOR_SIZE - off;
- }
- if (counter == end)
- read_size = end_offset - off + 1;
+ while ((stat & DRQ) == 0 && (stat & ERR) == 0)
+ stat = read_stat(base);
- memcpy(buf_curr, ret + off, read_size);
- buf_curr = buf_curr + read_size;
- total = total + read_size;
- counter++;
- }
- return total;
-}
-
-uint32_t ata_write(vfs_node_t *node, uint32_t offset, uint32_t size, char *buf)
-{
- uint32_t start = offset / SECTOR_SIZE;
- uint32_t start_offset = offset % SECTOR_SIZE;
+ if ((stat & ERR) != 0)
+ return;
- uint32_t end = (offset + size - 1) / SECTOR_SIZE;
- uint32_t end_offset = (offset + size - 1) % SECTOR_SIZE;
+ uint16_t drive_data[256];
+ for (size_t i = 0; i < 256; i++)
+ drive_data[i] = inw(base + DATA);
- char *buf_curr = buf;
- uint32_t counter = start;
- uint32_t write_size;
- uint32_t off, total = 0;
+ max_sector = drive_data[MAX_28LBA_SECTORS] | drive_data[MAX_28LBA_SECTORS + 1] << 16;
- while (counter <= end) {
- off = 0;
- write_size = SECTOR_SIZE;
- char *ret = ata_read_sector((ata_dev_t *)node->device, counter);
- if (counter == start) {
- off = start_offset;
- write_size = SECTOR_SIZE - off;
- }
- if (counter == end) {
- write_size = end_offset - off + 1;
- }
- memcpy(ret + off, buf_curr, write_size);
- ata_write_sector((ata_dev_t *)node->device, counter, ret);
- buf_curr = buf_curr + write_size;
- total = total + write_size;
- counter++;
- }
- return total;
+ sel_base_port = base;
+ sel_master_or_slave = master_or_slave;
}
-void ata_write_sector(ata_dev_t *dev, uint32_t lba, char *buf)
+void ata_init()
{
- memcpy(dev->mem_buffer, buf, SECTOR_SIZE);
-
- outb(dev->BMR_COMMAND, 0);
- outl(dev->BMR_prdt, (uint32_t)dev->prdt_phys);
- outb(dev->drive, 0xe0 | dev->slave << 4 | (lba & 0x0f000000) >> 24);
- outb(dev->sector_count, 1);
- outb(dev->lba_lo, lba & 0x000000ff);
- outb(dev->lba_mid, (lba & 0x0000ff00) >> 8);
- outb(dev->lba_high, (lba & 0x00ff0000) >> 16);
-
- outb(dev->command, 0xCA);
-
- outb(dev->BMR_COMMAND, 0x1);
-
- while (1) {
- int status = inb(dev->BMR_STATUS);
- int dstatus = inb(dev->status);
- if (!(status & 0x04)) {
- continue;
- }
- if (!(dstatus & 0x80)) {
- break;
- }
+ uint8_t pri_status = inb(PRIMARY_BASE + COM_STAT);
+ uint8_t sec_status = inb(SECONDARY_BASE + COM_STAT);
+ bool primary_floating = false;
+ bool secondary_floating = false;
+ if (pri_status == 0xFF)
+ primary_floating = true;
+ if (sec_status == 0xFF)
+ secondary_floating = true;
+
+ if (primary_floating && secondary_floating) {
+ log("No drives attached! What's going on?");
+ return;
}
-}
-
-char *ata_read_sector(ata_dev_t *dev, uint32_t lba)
-{
- char *buf = kmalloc(SECTOR_SIZE);
- outb(dev->BMR_COMMAND, 0);
- outl(dev->BMR_prdt, (uint32_t)dev->prdt_phys);
- outb(dev->drive, 0xe0 | dev->slave << 4 | (lba & 0x0f000000) >> 24);
- outb(dev->sector_count, 1);
- outb(dev->lba_lo, lba & 0x000000ff);
- outb(dev->lba_mid, (lba & 0x0000ff00) >> 8);
- outb(dev->lba_high, (lba & 0x00ff0000) >> 16);
-
- outb(dev->command, 0xC8);
-
- outb(dev->BMR_COMMAND, 0x8 | 0x1);
-
- while (1) {
- int status = inb(dev->BMR_STATUS);
- int dstatus = inb(dev->status);
- if (!(status & 0x04)) {
- continue;
- }
- if (!(dstatus & 0x80)) {
- break;
- }
+ check_drive(PRIMARY_BASE, SEL_MASTER);
+ check_drive(PRIMARY_BASE, SEL_SLAVE);
+ check_drive(SECONDARY_BASE, SEL_MASTER);
+ check_drive(SECONDARY_BASE, SEL_SLAVE);
+
+ if (sel_base_port == 0)
+ log("No drives attached! What's going on?");
+ else {
+ log("Found a drive!\nSelected drive is the %s on the %s bus",
+ sel_master_or_slave == SEL_MASTER ? "master" : "slave",
+ sel_base_port == PRIMARY_BASE ? "primary" : "secondary");
+ log("Max LBA value is %d", max_sector);
}
-
- memcpy(buf, dev->mem_buffer, SECTOR_SIZE);
- return buf;
}
-vfs_node_t *create_ata_device(ata_dev_t *dev)
+static void poll()
{
- vfs_node_t *t = kcalloc(sizeof(vfs_node_t), 1);
- strcpy(t->name, "ata device ");
- t->name[strlen(t->name)] = dev->mountpoint[strlen(dev->mountpoint) - 1];
- t->device = dev;
- t->flags = FS_BLOCKDEVICE;
- t->read = ata_read;
- t->write = ata_write;
- t->open = ata_open;
- t->close = ata_close;
- return t;
-}
-
-void ata_device_init(ata_dev_t *dev, int primary)
-{
- dev->prdt = (void *)kmalloc(sizeof(prdt_t));
- memset(dev->prdt, 0, sizeof(prdt_t));
- dev->prdt_phys = (uint8_t *)paging_get_phys((uint32_t)dev->prdt);
- dev->mem_buffer = (void *)kmalloc(4096);
- memset(dev->mem_buffer, 0, 4096);
-
- dev->prdt[0].buffer_phys = (uint32_t)paging_get_phys((uint32_t)dev->mem_buffer);
- dev->prdt[0].transfer_size = SECTOR_SIZE;
- dev->prdt[0].mark_end = MARK_END;
+ uint8_t stat;
- uint16_t base_addr = primary ? (0x1F0) : (0x170);
- uint16_t alt_status = primary ? (0x3F6) : (0x376);
-
- dev->data = base_addr;
- dev->error = base_addr + 1;
- dev->sector_count = base_addr + 2;
- dev->lba_lo = base_addr + 3;
- dev->lba_mid = base_addr + 4;
- dev->lba_high = base_addr + 5;
- dev->drive = base_addr + 6;
- dev->command = base_addr + 7;
- dev->alt_status = alt_status;
-
- dev->bar4 = pci_read_field(ata_device, PCI_BAR4, 4);
- if (dev->bar4 & 0x1) {
- dev->bar4 = dev->bar4 & 0xfffffffc;
- }
- dev->BMR_COMMAND = dev->bar4;
- dev->BMR_STATUS = dev->bar4 + 2;
- dev->BMR_prdt = dev->bar4 + 4;
-
- memset(dev->mountpoint, 0, 32);
- strcpy(dev->mountpoint, "/dev/hd");
- dev->mountpoint[strlen(dev->mountpoint)] = 'a' + (((!primary) << 1) | dev->slave);
+ do
+ stat = read_stat(sel_base_port);
+ while ((stat & BSY) != 0);
}
-void ata_device_detect(ata_dev_t *dev, int primary)
+void read_abs_sectors(uint32_t lba, uint8_t sector_count, uint16_t buf[])
{
- ata_device_init(dev, primary);
+ assert(lba >> LBA_BITS == 0);
- software_reset(dev);
- io_wait(dev);
- outb(dev->drive, (0xA + dev->slave) << 4);
- outb(dev->sector_count, 0);
- outb(dev->lba_lo, 0);
- outb(dev->lba_mid, 0);
- outb(dev->lba_high, 0);
+ outb(sel_base_port + DRIVE_SELECT, (lba >> (LBA_BITS - 4)) | sel_master_or_slave | 1 << 6);
+ outb(sel_base_port + SECTOR_COUNT, sector_count);
- outb(dev->command, COMMAND_IDENTIFY);
- if (!inb(dev->status)) {
- warn("Device does not exist: %s", dev->mountpoint);
- return;
- }
+ outb(sel_base_port + LBA_LOW, lba & 0xFF);
+ outb(sel_base_port + LBA_MID, (lba >> 8) & 0xFF);
+ outb(sel_base_port + LBA_HIGH, (lba >> 16) & 0xFF);
- uint8_t lba_lo = inb(dev->lba_lo);
- uint8_t lba_hi = inb(dev->lba_high);
- if (lba_lo != 0 || lba_hi != 0) {
- warn("Device is not ata-compatible: %s", dev->mountpoint);
- return;
- }
- uint8_t drq = 0, err = 0;
- while (!drq && !err) {
- drq = inb(dev->status) & STATUS_DRQ;
- err = inb(dev->status) & STATUS_ERR;
- }
- if (err) {
- warn("Error while polling: %s", dev->mountpoint);
- return;
- }
+ outb(sel_base_port + COM_STAT, READ_SECTORS);
- for (int i = 0; i < 256; i++)
- inw(dev->data);
+ size_t i = 0;
+ for (; sector_count > 0; sector_count--) {
+ poll();
- uint32_t pci_command_reg = pci_read_field(ata_device, PCI_COMMAND, 2);
- if (!(pci_command_reg & (1 << 2))) {
- pci_command_reg |= (1 << 2);
- pci_write_field(ata_device, PCI_COMMAND, pci_command_reg);
+ asm("rep insw" ::"c"(SECTOR_SIZE / 2), "d"(sel_base_port + DATA), "D"(buf + i));
+ i += SECTOR_SIZE / 2;
}
-
- log("Detected drive: %s", dev->mountpoint);
- vfs_mount(dev->mountpoint, create_ata_device(dev));
-}
-
-void ata_find(uint32_t device, uint16_t vendor_id, uint16_t device_id, void *extra)
-{
- if ((vendor_id == ATA_VENDOR_ID) && (device_id == ATA_DEVICE_ID))
- *((uint32_t *)extra) = device;
-}
-
-void ata_init()
-{
- pci_scan(&ata_find, -1, &ata_device);
-
- irq_install_handler(14, ata_handler);
-
- ata_device_detect(&primary_master, 1);
- ata_device_detect(&primary_slave, 1);
- // ata_device_detect(&secondary_master, 0);
- // ata_device_detect(&secondary_slave, 0);
} \ No newline at end of file