diff options
author | dzwdz | 2022-12-26 23:22:03 +0100 |
---|---|---|
committer | dzwdz | 2022-12-26 23:22:03 +0100 |
commit | 7ea0ddc733285cfa25d6535954de4784b054902a (patch) | |
tree | 0ae76ba3dd7628e5304153808962190d37559388 /src/kernel/arch/amd64/ata.c | |
parent | 2041919a2bb6cd8f4ef8ef6565962cfa82bb89f7 (diff) |
amd64/ata: write support
Diffstat (limited to 'src/kernel/arch/amd64/ata.c')
-rw-r--r-- | src/kernel/arch/amd64/ata.c | 49 |
1 files changed, 49 insertions, 0 deletions
diff --git a/src/kernel/arch/amd64/ata.c b/src/kernel/arch/amd64/ata.c index fa55335..4c1f2e2 100644 --- a/src/kernel/arch/amd64/ata.c +++ b/src/kernel/arch/amd64/ata.c @@ -1,7 +1,9 @@ +/* http://hddguru.com/documentation/2006.01.27-ATA-ATAPI-7/ */ #include <kernel/arch/amd64/ata.h> #include <kernel/arch/amd64/port_io.h> #include <kernel/panic.h> #include <kernel/util.h> +#include <shared/mem.h> #include <stdbool.h> static struct { @@ -171,3 +173,50 @@ int ata_read(int drive, void *buf, size_t len, size_t off) { } return len; } + +static void ata_rawwrite(int drive, const void *buf, uint32_t lba, uint32_t cnt) { + int iobase = ata_iobase(drive); + ata_driveselect(drive, lba); + port_out8(iobase + FEAT, 0); + port_out8(iobase + SCNT, cnt); + port_out8(iobase + LBAlo, lba); + port_out8(iobase + LBAmid, lba >> 8); + port_out8(iobase + LBAhi, lba >> 16); + port_out8(iobase + CMD, 0x30); /* WRITE SECTORS */ + + for (uint32_t i = 0; i < cnt; i++) { + ata_poll(drive, -1); + for (int j = 0; j < 256; j++) { + port_out16(iobase, ((uint16_t*)buf)[i * 256 + j]); + } + } +} + +int ata_write(int drive, const void *buf, size_t len, size_t off) { + char sec[512]; + size_t clen; + if (ata_drives[drive].type != DEV_PATA) { + panic_unimplemented(); + } + if (off & 511) { + clen = min(len, 512 - off % 512); + ata_read(drive, sec, 512, off / 512); + memcpy(sec + off % 512, buf, clen); + ata_rawwrite(drive, sec, off / 512, 1); + off += clen, + buf += clen; + len -= clen; + } + if (512 <= len) { + ata_rawwrite(drive, buf, off / 512, len / 512); + } + if (len & 511) { + buf += len & ~511; + off += len & ~511; + len = len & 511; + ata_read(drive, sec, 512, off / 512); + memcpy(sec, buf, len); + ata_rawwrite(drive, sec, off / 512, 1); + } + return 0; +} |