📄 sdata.c
字号:
#include "u.h"#include "lib.h"#include "mem.h"#include "dat.h"#include "fns.h"#include "io.h"#include "ureg.h"#include "error.h"#include "sd.h"extern SDifc sdataifc;enum { DbgCONFIG = 0x01, /* detected drive config info */ DbgIDENTIFY = 0x02, /* detected drive identify info */ DbgSTATE = 0x04, /* dump state on panic */ DbgPROBE = 0x08, /* trace device probing */};#define DEBUG 0/* (DbgSTATE|DbgCONFIG) */enum { /* I/O ports */ Data = 0, Error = 1, /* (read) */ Features = 1, /* (write) */ Count = 2, /* sector count */ Ir = 2, /* interrupt reason (PACKET) */ Sector = 3, /* sector number, LBA<7-0> */ Cyllo = 4, /* cylinder low, LBA<15-8> */ Bytelo = 4, /* byte count low (PACKET) */ Cylhi = 5, /* cylinder high, LBA<23-16> */ Bytehi = 5, /* byte count hi (PACKET) */ Dh = 6, /* Device/Head, LBA<32-14> */ Status = 7, /* (read) */ Command = 7, /* (write) */ As = 2, /* Alternate Status (read) */ Dc = 2, /* Device Control (write) */};enum { /* Error */ Med = 0x01, /* Media error */ Ili = 0x01, /* command set specific (PACKET) */ Nm = 0x02, /* No Media */ Eom = 0x02, /* command set specific (PACKET) */ Abrt = 0x04, /* Aborted command */ Mcr = 0x08, /* Media Change Request */ Idnf = 0x10, /* no user-accessible address */ Mc = 0x20, /* Media Change */ Unc = 0x40, /* Uncorrectable data error */ Wp = 0x40, /* Write Protect */ Icrc = 0x80, /* Interface CRC error */};enum { /* Features */ Dma = 0x01, /* data transfer via DMA (PACKET) */ Ovl = 0x02, /* command overlapped (PACKET) */};enum { /* Interrupt Reason */ Cd = 0x01, /* Command/Data */ Io = 0x02, /* I/O direction */ Rel = 0x04, /* Bus Release */};enum { /* Device/Head */ Dev0 = 0xA0, /* Master */ Dev1 = 0xB0, /* Slave */ Lba = 0x40, /* LBA mode */};enum { /* Status, Alternate Status */ Err = 0x01, /* Error */ Chk = 0x01, /* Check error (PACKET) */ Drq = 0x08, /* Data Request */ Dsc = 0x10, /* Device Seek Complete */ Serv = 0x10, /* Service */ Df = 0x20, /* Device Fault */ Dmrd = 0x20, /* DMA ready (PACKET) */ Drdy = 0x40, /* Device Ready */ Bsy = 0x80, /* Busy */};enum { /* Command */ Cnop = 0x00, /* NOP */ Cdr = 0x08, /* Device Reset */ Crs = 0x20, /* Read Sectors */ Cws = 0x30, /* Write Sectors */ Cedd = 0x90, /* Execute Device Diagnostics */ Cpkt = 0xA0, /* Packet */ Cidpkt = 0xA1, /* Identify Packet Device */ Crsm = 0xC4, /* Read Multiple */ Cwsm = 0xC5, /* Write Multiple */ Crdq = 0xC7, /* Read DMA queued */ Crd = 0xC8, /* Read DMA */ Cwd = 0xCA, /* Write DMA */ Cwdq = 0xCC, /* Write DMA queued */ Cid = 0xEC, /* Identify Device */ Csf = 0xEF, /* Set Features */};enum { /* Device Control */ Nien = 0x02, /* (not) Interrupt Enable */ Srst = 0x04, /* Software Reset */};enum { /* PCI Configuration Registers */ Bmiba = 0x20, /* Bus Master Interface Base Address */ Idetim = 0x40, /* IE Timing */ Sidetim = 0x44, /* Slave IE Timing */ Udmactl = 0x48, /* Ultra DMA/33 Control */ Udmatim = 0x4A, /* Ultra DMA/33 Timing */};enum { /* Bus Master IDE I/O Ports */ Bmicx = 0, /* Command */ Bmisx = 2, /* Status */ Bmidtpx = 4, /* Descriptor Table Pointer */};enum { /* Bmicx */ Ssbm = 0x01, /* Start/Stop Bus Master */ Rwcon = 0x08, /* Read/Write Control */};enum { /* Bmisx */ Bmidea = 0x01, /* Bus Master IDE Active */ Idedmae = 0x02, /* IDE DMA Error (R/WC) */ Ideints = 0x04, /* IDE Interrupt Status (R/WC) */ Dma0cap = 0x20, /* Drive 0 DMA Capable */ Dma1cap = 0x40, /* Drive 0 DMA Capable */};enum { /* Physical Region Descriptor */ PrdEOT = 0x80000000, /* Bus Master IDE Active */};enum { /* offsets into the identify info. */ Iconfig = 0, /* general configuration */ Ilcyl = 1, /* logical cylinders */ Ilhead = 3, /* logical heads */ Ilsec = 6, /* logical sectors per logical track */ Iserial = 10, /* serial number */ Ifirmware = 23, /* firmware revision */ Imodel = 27, /* model number */ Imaxrwm = 47, /* max. read/write multiple sectors */ Icapabilities = 49, /* capabilities */ Istandby = 50, /* device specific standby timer */ Ipiomode = 51, /* PIO data transfer mode number */ Ivalid = 53, Iccyl = 54, /* cylinders if (valid&0x01) */ Ichead = 55, /* heads if (valid&0x01) */ Icsec = 56, /* sectors if (valid&0x01) */ Iccap = 57, /* capacity if (valid&0x01) */ Irwm = 59, /* read/write multiple */ Ilba0 = 60, /* LBA size */ Ilba1 = 61, /* LBA size */ Imwdma = 63, /* multiword DMA mode */ Iapiomode = 64, /* advanced PIO modes supported */ Iminmwdma = 65, /* min. multiword DMA cycle time */ Irecmwdma = 66, /* rec. multiword DMA cycle time */ Iminpio = 67, /* min. PIO cycle w/o flow control */ Iminiordy = 68, /* min. PIO cycle with IORDY */ Ipcktbr = 71, /* time from PACKET to bus release */ Iserbsy = 72, /* time from SERVICE to !Bsy */ Iqdepth = 75, /* max. queue depth */ Imajor = 80, /* major version number */ Iminor = 81, /* minor version number */ Icmdset0 = 82, /* command sets supported */ Icmdset1 = 83, /* command sets supported */ Icmdset2 = 84, /* command sets supported extension */ Icmdset3 = 85, /* command sets enabled */ Icmdset4 = 86, /* command sets enabled */ Icmdset5 = 87, /* command sets enabled extension */ Iudma = 88, /* ultra DMA mode */ Ierase = 89, /* time for security erase */ Ieerase = 90, /* time for enhanced security erase */ Ipower = 91, /* current advanced power management */ Irmsn = 127, /* removable status notification */ Istatus = 128, /* security status */};typedef struct Ctlr Ctlr;typedef struct Drive Drive;typedef struct Prd { ulong pa; /* Physical Base Address */ int count;} Prd;enum { Nprd = SDmaxio/(64*1024)+2,};typedef struct Ctlr { int cmdport; int ctlport; int irq;// int bmiba; /* bus master interface base address */ Pcidev* pcidev; void (*ienable)(Ctlr*); Drive* drive[2];// Prd* prdt; /* physical region descriptor table */// QLock; /* current command */ Drive* curdrive;// Rendez; int done; Lock; /* register access */} Ctlr;typedef struct Drive { Ctlr* ctlr; int dev; ushort info[256]; int c; /* cylinder */ int h; /* head */ int s; /* sector */ int sectors; /* total */ int secsize; /* sector size */ int block; /* R/W multiple size */ int pior; /* PIO read command */ int piow; /* PIO write command */ int dma; /* DMA R/W possible */ int pkt; /* PACKET device, length of pktcmd */ uchar sense[18]; uchar inquiry[48];// QLock; /* drive access */ int command; /* current command */ int write; uchar* data; int dlen; uchar* limit; int count; /* sectors */ int status; int error; uchar pktcmd[16]; int pktdma;} Drive;static voidpc87415ienable(Ctlr* ctlr){ Pcidev *p; int x; p = ctlr->pcidev; if(p == nil) return; x = pcicfgr32(p, 0x40); if(ctlr->cmdport == p->mem[0].bar) x &= ~0x00000100; else x &= ~0x00000200; pcicfgw32(p, 0x40, x);}intatadebug(int cmdport, int ctlport, char* fmt, ...){ int i, n; va_list arg; char buf[PRINTSIZE]; if(!DEBUG){ USED(cmdport, ctlport, fmt); return 0; } va_start(arg, fmt); n = doprint(buf, buf+sizeof(buf), fmt, arg) - buf; va_end(arg); if(cmdport){ if(buf[n-1] == '\n') n--; n += snprint(buf+n, PRINTSIZE-n, " ataregs 0x%uX:", cmdport); for(i = 1; i < 7; i++) n += snprint(buf+n, PRINTSIZE-n, " 0x%2.2uX", inb(cmdport+i)); if(ctlport) n += snprint(buf+n, PRINTSIZE-n, " 0x%2.2uX", inb(ctlport+As)); n += snprint(buf+n, PRINTSIZE-n, "\n"); } putstrn(buf, n); return n;}static intataready(int cmdport, int ctlport, int dev, int reset, int ready, int micro){ int as; atadebug(cmdport, ctlport, "ataready: dev %uX reset %uX ready %uX", dev, reset, ready); for(;;){ /* * Wait for the controller to become not busy and * possibly for a status bit to become true (usually * Drdy). Must change to the appropriate device * register set before testing for ready. * Always run through the loop at least once so it * can be used as a test for !Bsy. */ as = inb(ctlport+As); if(dev && !(as & (Bsy|Drq))){ outb(cmdport+Dh, dev); dev = 0; continue; } else if(!(as & reset)){ if(ready == 0 || (as & ready)){ atadebug(0, 0, "ataready: %d 0x%2.2uX\n", micro, as); return as; } } if(micro-- <= 0){ atadebug(0, 0, "ataready: %d 0x%2.2uX\n", micro, as); break; } microdelay(1); } atadebug(cmdport, ctlport, "ataready: timeout"); return -1;}static intatacsfenabled(Drive* drive, vlong csf){ int cmdset, i, x; for(i = 0; i < 3; i++){ x = (csf>>(16*i)) & 0xFFFF; if(x == 0) continue; cmdset = drive->info[Icmdset3+i]; if(cmdset == 0 || cmdset == 0xFFFF) return 0; return cmdset & x; } return 0;}static intatasetrwmode(Drive* drive, int cmdport, int ctlport, int dev){ int as, block; drive->block = drive->secsize; drive->pior = Crs; drive->piow = Cws; if((block = drive->info[Imaxrwm] & 0xFF) == 0) return 0; /* * Prior to ATA-4 there was no way to determine the * current block count (now in Irwm). * Sometimes drives come up with the current count set * to 0, so always set a suitable value. */ if(ataready(cmdport, ctlport, dev, Bsy|Drq, Drdy, 100*1000) < 0) return 0; outb(cmdport+Sector, block); outb(cmdport+Command, Csf); microdelay(1); as = ataready(cmdport, ctlport, dev, Bsy, Drdy|Df|Err, 1000); if(as < 0 || (as & (Df|Err))) return -1; drive->info[Irwm] = 0x100|block; drive->block = block*drive->secsize; drive->pior = Crsm; drive->piow = Cwsm; return block;}static Drive*ataidentify(int cmdport, int ctlport, int dev){ Drive *drive; uchar buf[512], *p; int command, i, as; ushort *sp; atadebug(0, 0, "identify: port 0x%uX dev 0x%2.2uX\n", cmdport, dev); command = Cidpkt;retry: as = ataready(cmdport, ctlport, dev, Bsy|Drq, 0, 100*1000); if(as < 0) return nil; outb(cmdport+Command, command); microdelay(1); as = ataready(cmdport, ctlport, dev, Bsy, Drq|Err, 100*1000); if(as < 0) return nil; if(as & Err){ if(command == Cid) return nil; command = Cid; goto retry; } memset(buf, 0, sizeof(buf)); inss(cmdport+Data, buf, 256); inb(cmdport+Status); if(DEBUG & DbgIDENTIFY){ int i; ushort *sp; sp = (ushort*)buf; for(i = 0; i < 256; i++){ if(i && (i%16) == 0) print("\n"); print("%4.4uX ", *sp); sp++; } print("\n"); } if((drive = malloc(sizeof(Drive))) == nil) return nil; drive->dev = dev; memmove(drive->info, buf, sizeof(drive->info)); drive->sense[0] = 0x70; drive->sense[7] = sizeof(drive->sense)-7; drive->inquiry[2] = 2; drive->inquiry[3] = 2; drive->inquiry[4] = sizeof(drive->inquiry)-4; p = &drive->inquiry[8]; sp = &drive->info[Imodel]; for(i = 0; i < 20; i++){ *p++ = *sp>>8; *p++ = *sp++; } drive->secsize = 512; if((drive->info[Iconfig] & 0xC000) == 0x8000){ if(drive->info[Iconfig] & 0x01) drive->pkt = 16; else drive->pkt = 12; } else{ if(drive->info[Ivalid] & 0x0001){
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -