/* * intel/amd ahci sata bootstrap controller * copyright © 2007-13 coraid, inc. */ #include "u.h" #include "lib.h" #include "mem.h" #include "dat.h" #include "fns.h" #include "io.h" #include "error.h" #include "sd.h" #include #include "ahci.h" #define debug ahcidebug #define dprint(...) if(debug) print(__VA_ARGS__); else USED(debug) #define idprint(...) if(prid) print(__VA_ARGS__); else USED(prid) #define aprint(...) if(datapi) print(__VA_ARGS__); else USED(datapi) #define Pciwaddrh(a) 0 #define tnam(c) tname[(c)->type] #define Ticks m->ticks enum { NCtlr = 2, NCtlrdrv= 8, NDrive = NCtlr*NCtlrdrv, Fahdrs = 4, Read = 0, Write, }; /* pci space configuration */ enum { Pmap = 0x90, Ppcs = 0x91, Prev = 0xa8, }; enum { Tesb, Tsb600, Tjmicron, Tahci, }; typedef struct Ctlr Ctlr; typedef struct Drive Drive; static char *tname[] = { "63xxesb", "sb600", "jmicron", "ahci", }; enum { Dnull, Dmissing, Dnew, Dready, Derror, Dreset, Doffline, Dportreset, Dlast, }; static char *diskstates[Dlast] = { "null", "missing", "new", "ready", "error", "reset", "offline", "portreset", }; extern SDifc sdiahciifc; enum { DMautoneg, DMsatai, DMsataii, DMsataiii, DMlast, }; static char *modes[DMlast] = { "auto", "satai", "sataii", "sataiii", }; struct Drive { Lock; Ctlr *ctlr; SDunit *unit; char name[10]; Aport *port; Aportm portm; Aportc portc; /* redundant ptr to port and portm. */ uchar drivechange; uchar state; uvlong sectors; uint secsize; ulong totick; ulong lastseen; uint wait; uchar mode; uchar active; char serial[20+1]; char firmware[8+1]; char model[40+1]; uvlong wwn; ushort info[0x200]; /* * ahci allows non-sequential ports. * to avoid this hassle, we let * driveno ctlr*NCtlrdrv + unit * portno nth available port */ uint driveno; uint portno; }; struct Ctlr { Lock; int type; int enabled; SDev *sdev; Pcidev *pci; uchar *mmio; u32int *lmmio; Ahba *hba; Drive rawdrive[NCtlrdrv]; Drive* drive[NCtlrdrv]; int ndrive; }; static Ctlr iactlr[NCtlr]; static SDev sdevs[NCtlr]; static int niactlr; static int debug; static int prid; static int datapi; #ifdef NO static char stab[] = { [0] 'i', 'm', [8] 't', 'c', 'p', 'e', [16] 'N', 'I', 'W', 'B', 'D', 'C', 'H', 'S', 'T', 'F', 'X' }; static void serrstr(u32int r, char *s, char *e) { int i; e -= 3; for(i = 0; i < nelem(stab) && s < e; i++) if(r & (1<task, p->cmd, p->ci, p->isr); } static void esleep(int ms) { delay(ms); } typedef struct { Aport *p; int i; } Asleep; static int ahciclear(void *v) { Asleep *s; s = v; return (s->p->ci & s->i) == 0; } static void aesleep(Aportm *, Asleep *a, int ms) { ulong start; start = m->ticks; while((a->p->ci & a->i) != 0) if(TK2MS(m->ticks - start) >= ms) break; } static int ahciwait(Aportc *c, int ms) { Aport *p; Asleep as; p = c->p; p->ci = 1; as.p = p; as.i = 1; aesleep(c->m, &as, ms); if((p->task & 1) == 0 && p->ci == 0) return 0; dreg("ahciwait fail/timeout ", c->p); return -1; } static void mkalist(Aportm *m, uint flags, uchar *data, int len) { Actab *t; Alist *l; Aprdt *p; t = m->ctab; l = m->list; l->flags = flags | 0x5; l->len = 0; l->ctab = PCIWADDR(t); l->ctabhi = Pciwaddrh(t); if(data){ l->flags |= 1<<16; p = &t->prdt; p->dba = PCIWADDR(data); p->dbahi = Pciwaddrh(data); p->count = 1<<31 | len - 2 | 1; } } static int setfeatures(Aportc *pc, uchar f, int w) { uchar *c; c = pc->m->ctab->cfis; featfis(pc->m, c, f); mkalist(pc->m, Lwrite, 0, 0); return ahciwait(pc, w); } /* * ata 7, required for sata, requires that all devices "support" * udma mode 5, however sata:pata bridges allow older devices * which may not. the innodisk satadom, for example allows * only udma mode 2. on the assumption that actual udma is * taking place on these bridges, we set the highest udma mode * available, or pio if there is no udma mode available. */ static int settxmode(Aportc *pc, uchar f) { uchar *c; c = pc->m->ctab->cfis; if(txmodefis(pc->m, c, f) == -1) return 0; mkalist(pc->m, Lwrite, 0, 0); return ahciwait(pc, 3*1000); } static void asleep(int ms) { delay(ms); } static int ahciportreset(Aportc *c, uint mode) { u32int *cmd, i; Aport *p; p = c->p; cmd = &p->cmd; *cmd &= ~(Afre|Ast); for(i = 0; i < 500; i += 25){ if((*cmd & Acr) == 0) break; asleep(25); } p->sctl = 3*Aipm | 0*Aspd | Adet; delay(1); p->sctl = 3*Aipm | mode*Aspd; return 0; } static int ahciidentify0(Aportc *pc, void *id) { uchar *c; Actab *t; t = pc->m->ctab; c = t->cfis; memset(id, 0, 0x200); identifyfis(pc->m, c); mkalist(pc->m, 0, id, 0x200); return ahciwait(pc, 3*1000); } static vlong ahciidentify(Aportc *pc, ushort *id, uint *ss, char *d) { vlong s; Aportm *m; m = pc->m; if(ahciidentify0(pc, id) == -1) return -1; s = idfeat(m, id); *ss = idss(m, id); if(s == -1 || (m->feat&Dlba) == 0){ if((m->feat&Dlba) == 0) dprint("%s: no lba support\n", d); return -1; } return s; } static int ahciquiet(Aport *a) { u32int *p, i; p = &a->cmd; *p &= ~Ast; for(i = 0; i < 500; i += 50){ if((*p & Acr) == 0) goto stop; asleep(50); } return -1; stop: if((a->task & (ASdrq|ASbsy)) == 0){ *p |= Ast; return 0; } *p |= Aclo; for(i = 0; i < 500; i += 50){ if((*p & Aclo) == 0) goto stop1; asleep(50); } return -1; stop1: /* extra check */ dprint("ahci: clo clear %ux\n", a->task); if(a->task & ASbsy) return -1; *p |= Afre | Ast; return 0; } static int ahciidle(Aport *port) { u32int *p, i, r; p = &port->cmd; if((*p & Arun) == 0) return 0; *p &= ~Ast; r = 0; for(i = 0; i < 500; i += 25){ if((*p & Acr) == 0) goto stop; asleep(25); } r = -1; stop: if((*p & Afre) == 0) return r; *p &= ~Afre; for(i = 0; i < 500; i += 25){ if((*p & Afre) == 0) return 0; asleep(25); } return -1; } /* * §6.2.2.1 first part; comreset handled by reset disk. * - remainder is handled by configdisk. * - ahcirecover is a quick recovery from a failed command. */ static int ahciswreset(Aportc *pc) { int i; i = ahciidle(pc->p); pc->p->cmd |= Afre; if(i == -1) return -1; if(pc->p->task & (ASdrq|ASbsy)) return -1; return 0; } static int ahcirecover(Aportc *pc) { ahciswreset(pc); pc->p->cmd |= Ast; if(settxmode(pc, pc->m->udma) == -1) return -1; return 0; } static void* malign(int size, int align) { void *v; v = xspanalloc(size, align, 0); memset(v, 0, size); return v; } static void setupfis(Afis *f) { f->base = malign(0x100, 0x100); f->d = f->base + 0; f->p = f->base + 0x20; f->r = f->base + 0x40; f->u = f->base + 0x60; f->devicebits = (u32int*)(f->base + 0x58); } static void ahciwakeup(Aportc *c, uint mode) { ushort s; s = c->p->sstatus; if((s & Isleepy) == 0) return; if((s & Smask) != Spresent){ print("ahci: slumbering drive missing %.3ux\n", s); return; } ahciportreset(c, mode); // iprint("ahci: wake %.3ux -> %.3lux\n", s, c->p->sstatus); } static int ahciconfigdrive(Ahba *h, Aportc *c, int mode) { Aportm *m; Aport *p; p = c->p; m = c->m; if(m->list == 0){ setupfis(&m->fis); m->list = malign(sizeof *m->list, 1024); m->ctab = malign(sizeof *m->ctab, 128); } p->list = PCIWADDR(m->list); p->listhi = Pciwaddrh(m->list); p->fis = PCIWADDR(m->fis.base); p->fishi = Pciwaddrh(m->fis.base); p->cmd |= Afre; if((p->sstatus & Sbist) == 0 && (p->cmd & Apwr) != Apwr) if((p->sstatus & Sphylink) == 0 && h->cap & Hss){ /* staggered spin-up? */ dprint("ahci: spin up ... [%.3ux]\n", p->sstatus); p->cmd |= Apwr; for(int i = 0; i < 1400; i += 50){ if(p->sstatus & (Sphylink | Sbist)) break; asleep(50); } } p->serror = SerrAll; if((p->sstatus & SSmask) == (Isleepy | Spresent)) ahciwakeup(c, mode); /* disable power managment sequence from book. */ p->sctl = 3*Aipm | mode*Aspd | 0*Adet; p->cmd &= ~Aalpe; p->cmd |= Ast; p->ie = IEM; return 0; } static int ahcienable(Ahba *h) { h->ghc |= Hie; return 0; } static int ahcidisable(Ahba *h) { h->ghc &= ~Hie; return 0; } static int countbits(ulong u) { int i, n; n = 0; for(i = 0; i < 32; i++) if(u & (1<hba = (Ahba*)c->mmio; u = h->cap; if((u & Ham) == 0) h->ghc |= Hae; print("ahci%d port %#p: hba sss %ld ncs %ld coal %ld mport %ld " "led %ld clo %ld ems %ld\n", count++, h, (u>>27) & 1, (u>>8) & 0x1f, (u>>7) & 1, u & 0x1f, (u>>25) & 1, (u>>24) & 1, (u>>6) & 1); return countbits(h->pi); } static int ahcihbareset(Ahba *h) { int wait; h->ghc |= Hhr; for(wait = 0; wait < 1000; wait += 100){ if(h->ghc == 0) return 0; delay(100); } return -1; } /* under development */ static int ahcibioshandoff(Ahba *h) { int i, wait; if((h->cap2 & Boh) == 0) return 0; if((h->bios & Bos) == 0) return 0; print("ahcibioshandoff: claim\n"); h->bios |= Oos; wait = 25; for(i = 0; i < wait; i++){ delay(1); if((h->bios & Bos) == 0) break; if(i < 25 && h->bios & Bb){ print("ahcibioshandoff: busy\n"); wait = 2000; } } if(i == wait){ print("ahcibioshandoff: timeout %.1ux\n", h->bios); h->bios = Oos; } return 0; } static char* dnam(Drive *d) { char *s; s = d->name; if(d->unit && d->unit->name) s = d->unit->name; return s; } static int identify(Drive *d) { uchar oserial[21]; ushort *id; vlong osectors, s; SDunit *u; id = d->info; s = ahciidentify(&d->portc, id, &d->secsize, dnam(d)); if(s == -1){ d->state = Derror; return -1; } osectors = d->sectors; memmove(oserial, d->serial, sizeof d->serial); d->sectors = s; idmove(d->serial, id+10, 20); idmove(d->firmware, id+23, 8); idmove(d->model, id+27, 40); d->wwn = idwwn(d->portc.m, id); u = d->unit; memset(u->inquiry, 0, sizeof u->inquiry); u->inquiry[2] = 2; u->inquiry[3] = 2; u->inquiry[4] = sizeof u->inquiry - 4; memmove(u->inquiry+8, d->model, 40); if(osectors != s || memcmp(oserial, d->serial, sizeof oserial)){ d->drivechange = 1; u->sectors = 0; } return 0; } static void clearci(Aport *p) { if(p->cmd & Ast){ p->cmd &= ~Ast; p->cmd |= Ast; } } static int intel(Ctlr *c) { return c->pci->vid == 0x8086; } static int ignoreahdrs(Drive *d) { return d->portm.feat & Datapi && d->ctlr->type == Tsb600; } static void updatedrive(Drive *d) { u32int f, cause, serr, s0, pr, ewake; Aport *p; static u32int last; pr = 1; ewake = 0; f = 0; p = d->port; cause = p->isr; if(d->ctlr->type == Tjmicron) cause &= ~Aifs; serr = p->serror; p->isr = cause; if(p->ci == 0){ f |= Fdone; pr = 0; }else if(cause & Adps) pr = 0; if(cause & Ifatal){ ewake = 1; dprint("%s: fatal\n", dnam(d)); } if(cause & Adhrs){ if(p->task & 33){ if(ignoreahdrs(d) && serr & ErrE) f |= Fahdrs; dprint("%s: Adhrs cause %ux serr %ux task %ux\n", dnam(d), cause, serr, p->task); f |= Ferror; ewake = 1; } pr = 0; } if(p->task & 1 && last != cause) dprint("%s: err ca %ux serr %ux task %ux sstat %.3ux\n", dnam(d), cause, serr, p->task, p->sstatus); if(pr) dprint("%s: upd %ux ta %ux\n", dnam(d), cause, p->task); if(cause & (Aprcs|Aifs)){ s0 = d->state; switch(p->sstatus & Smask){ case Smissing: d->state = Dmissing; break; case Spresent: if((p->sstatus & Imask) == Islumber) d->state = Dnew; else d->state = Derror; break; case Sphylink: /* power mgnt crap for suprise removal */ p->ie |= Aprcs|Apcs; /* is this required? */ d->state = Dreset; break; case Sbist: d->state = Doffline; break; } dprint("%s: %s → %s [Apcrs] %.3ux\n", dnam(d), diskstates[s0], diskstates[d->state], p->sstatus); if(s0 == Dready && d->state != Dready) idprint("%s: pulled\n", dnam(d)); if(d->state != Dready) f |= Ferror; if(d->state != Dready || p->ci) ewake = 1; } p->serror = serr; if(ewake) clearci(p); if(f){ d->portm.flag = f; wakeup(&d->portm); } last = cause; } static void pstatus(Drive *d, ulong s) { /* * bogus code because the first interrupt is currently dropped. * likely my fault. serror is maybe cleared at the wrong time. */ switch(s){ default: print("%s: pstatus: bad status %.3lux\n", dnam(d), s); case Smissing: d->state = Dmissing; break; case Spresent: break; case Sphylink: d->wait = 0; d->state = Dnew; break; case Sbist: d->state = Doffline; break; } } static int configdrive(Drive *d) { if(ahciconfigdrive(d->ctlr->hba, &d->portc, d->mode) == -1) return -1; ilock(d); pstatus(d, d->port->sstatus & Smask); iunlock(d); return 0; } static void resetdisk(Drive *d) { uint state, det, stat; Aport *p; p = d->port; det = p->sctl & 7; stat = p->sstatus & Smask; state = (p->cmd>>28) & 0xf; dprint("%s: resetdisk: icc %ux det %.3ux sdet %.3ux\n", dnam(d), state, det, stat); ilock(d); state = d->state; if(d->state != Dready || d->state != Dnew) d->portm.flag |= Ferror; clearci(p); /* satisfy sleep condition. */ wakeup(&d->portm); d->state = Derror; iunlock(d); if(stat != Sphylink){ ilock(d); d->state = Dportreset; iunlock(d); return; } qlock(&d->portm); if(p->cmd&Ast && ahciswreset(&d->portc) == -1){ ilock(d); d->state = Dportreset; /* get a bigger stick. */ iunlock(d); }else{ ilock(d); d->state = Dmissing; iunlock(d); configdrive(d); } dprint("%s: resetdisk: %s → %s\n", dnam(d), diskstates[state], diskstates[d->state]); qunlock(&d->portm); } enum { Puissig = 0x5a5a, /* 'ZZ' har har */ }; static int newdrive(Drive *d) { char *s; uint sig; Aportc *c; Aportm *m; c = &d->portc; m = &d->portm; qlock(c->m); setfissig(m, c->p->sig); sig = (ushort)c->p->sig; if(sig == Puissig && setfeatures(c, 0x86, 3*1000) == -1){ dprint("%s: puis failure\n", dnam(d)); goto lose; } if(identify(d) == -1){ dprint("%s: identify failure\n", dnam(d)); goto lose; } if(settxmode(c, m->udma) == -1){ dprint("%s: can't set udma mode\n", dnam(d)); goto lose; } if(m->feat & Dpower && setfeatures(c, 0x85, 3*1000) == -1){ m->feat &= ~Dpower; if(ahcirecover(c) == -1) goto lose; } ilock(d); d->state = Dready; iunlock(d); qunlock(c->m); s = ""; if(m->feat & Dllba) s = "L"; idprint("%s: %sLBA %,lld sectors\n", dnam(d), s, d->sectors); idprint(" %s %s %s %s\n", d->model, d->firmware, d->serial, d->drivechange? "[newdrive]": ""); return 0; lose: idprint("%s: can't be initialized\n", dnam(d)); ilock(d); d->state = Dnull; iunlock(d); qunlock(c->m); return -1; } enum { Nms = 256, Mphywait = 2*1024/Nms - 1, Midwait = 16*1024/Nms - 1, Mcomrwait = 64*1024/Nms - 1, }; static void hangck(Drive *d) { if((d->portm.feat & Datapi) == 0 && d->active && d->totick != 0 && (long)TK2MS(Ticks - d->totick) > 0){ dprint("%s: drive hung; resetting [%ux] ci %ux\n", dnam(d), d->port->task, d->port->ci); d->state = Dreset; } } static ushort olds[NCtlr*NCtlrdrv]; static int doportreset(Drive *d) { int i; i = -1; qlock(&d->portm); if(ahciportreset(&d->portc, d->mode) == -1) dprint("ahci: ahciportreset fails\n"); else i = 0; qunlock(&d->portm); dprint("ahci: portreset → %s [task %.4ux ss %.3ux]\n", diskstates[d->state], d->port->task, d->port->sstatus); return i; } /* drive must be locked */ static void statechange(Drive *d) { switch(d->state){ case Dnull: case Doffline: if(d->unit) if(d->unit->sectors != 0){ d->sectors = 0; d->drivechange = 1; } case Dready: d->wait = 0; } } static void checkdrive(Drive *d, int i) { ushort s, sig; ilock(d); s = d->port->sstatus; if(s) d->lastseen = Ticks; if(s != olds[i]){ dprint("%s: status: %.3ux -> %.3ux: %s\n", dnam(d), olds[i], s, diskstates[d->state]); olds[i] = s; d->wait = 0; } hangck(d); switch(d->state){ case Dnull: case Dready: break; case Dmissing: case Dnew: switch(s & (Iactive|Smask)){ case Spresent: ahciwakeup(&d->portc, d->mode); d->portc.p->cmd |= Ast; case Smissing: break; default: dprint("%s: unknown status %.3ux\n", dnam(d), s); /* fall through */ case Iactive: /* active, no device */ if(++d->wait&Mphywait) break; reset: if(++d->mode > DMsataii) d->mode = 0; if(d->mode == DMsatai){ d->state = Dportreset; goto portreset; } dprint("%s: reset; new mode %s\n", dnam(d), modes[d->mode]); iunlock(d); resetdisk(d); ilock(d); break; case Iactive | Sphylink: if(d->unit == nil) break; if((++d->wait&Midwait) == 0){ dprint("%s: slow reset %.3ux task=%ux; %d\n", dnam(d), s, d->port->task, d->wait); goto reset; } s = (uchar)d->port->task; sig = d->port->sig >> 16; if(s == 0x7f || s&ASbsy || (sig != 0xeb14 && (s & ASdrdy) == 0)) break; iunlock(d); newdrive(d); ilock(d); break; } break; case Doffline: if(d->wait++ & Mcomrwait) break; /* fallthrough */ case Derror: case Dreset: dprint("%s: reset [%s]: mode %d; status %.3ux\n", dnam(d), diskstates[d->state], d->mode, s); iunlock(d); resetdisk(d); ilock(d); break; case Dportreset: portreset: if(d->wait++ & 0xff && (s & Iactive) == 0) break; dprint("%s: portreset [%s]: mode %d; status %.3ux\n", dnam(d), diskstates[d->state], d->mode, s); d->portm.flag |= Ferror; clearci(d->port); wakeup(&d->portm); if((s & Smask) == 0){ d->state = Dmissing; break; } iunlock(d); doportreset(d); ilock(d); break; } statechange(d); iunlock(d); } static void iainterrupt(Ureg*, void *a) { int i; u32int cause, m; Ctlr *c; Drive *d; c = a; ilock(c); cause = c->hba->isr; for(i = 0; cause; i++){ m = 1 << i; if((cause & m) == 0) continue; cause &= ~m; d = c->rawdrive + i; ilock(d); if(d->port->isr && c->hba->pi & m) updatedrive(d); c->hba->isr = m; iunlock(d); } iunlock(c); } static int iaverify(SDunit *u) { int i, reset; Ctlr *c; Drive *d; c = u->dev->ctlr; d = c->drive[u->subno]; ilock(c); ilock(d); d->unit = u; iunlock(d); iunlock(c); reset = 0; for(i = 0; i < 20; i++){ checkdrive(d, d->driveno); switch(d->state){ case Dnew: if(d->port->task & 0x80){ d->state = Dportreset; reset = 1; } break; case Dmissing: if(reset || d->port->sstatus & 0x773) break; case Dnull: case Doffline: return 1; case Dready: if(d->portm.feat & Datapi) return scsiverify(d->unit); return 1; } delay(50); } return 1; } static int iaenable(SDev *s) { Ctlr *c; c = s->ctlr; ilock(c); if(!c->enabled){ if(c->ndrive == 0) panic("iaenable: zero s->ctlr->ndrive"); pcisetbme(c->pci); setvec(c->pci->intl+VectorPIC, iainterrupt, c); /* supposed to squelch leftover interrupts here. */ ahcienable(c->hba); c->enabled = 1; } iunlock(c); return 1; } static int iadisable(SDev *s) { Ctlr *c; c = s->ctlr; ilock(c); ahcidisable(c->hba); // intrdisable(c->irq, iainterrupt, c, c->tbdf, name); c->enabled = 0; iunlock(c); return 1; } static int iaonline(SDunit *unit) { int r; Ctlr *c; Drive *d; c = unit->dev->ctlr; d = c->drive[unit->subno]; r = 0; if(d->portm.feat & Datapi && d->drivechange){ r = scsionline(unit); if(r > 0) d->drivechange = 0; return r; } ilock(d); if(d->drivechange){ r = 2; d->drivechange = 0; /* devsd resets this after online is called; why? */ unit->sectors = d->sectors; unit->secsize = d->secsize; }else if(d->state == Dready) r = 1; iunlock(d); return r; } static Alist* ahcibuild(Aportm *m, int rw, void *data, uint n, vlong lba) { uchar *c; uint flags; Alist *l; l = m->list; c = m->ctab->cfis; rwfis(m, c, rw, n, lba); flags = Lpref; if(rw == SDwrite) flags |= Lwrite; mkalist(m, flags, data, 512*n); return l; } static Alist* ahcibuildpkt(Aportm *m, SDreq *r, void *data, int n) { uint flags; uchar *c; Actab *t; Alist *l; l = m->list; t = m->ctab; c = t->cfis; atapirwfis(m, c, r->cmd, r->clen, n); flags = 1<<16 | Lpref | Latapi; if(r->write != 0 && data) flags |= Lwrite; mkalist(m, flags, data, n); return l; } static int waitready(Drive *d) { u32int s; ulong i, δ; for(i = 0; i < 15000; i += 250){ if(d->state == Dreset || d->state == Dportreset || d->state == Dnew) return 1; δ = Ticks - d->lastseen; if(d->state == Dnull || δ > 10*1000) return -1; ilock(d); s = d->port->sstatus; iunlock(d); if((s & Imask) == 0 && δ > 1500) return -1; if(d->state == Dready && (s & Smask) == Sphylink) return 0; esleep(250); } print("%s: not responding; offline\n", dnam(d)); ilock(d); d->state = Doffline; iunlock(d); return -1; } static int io(Drive *d, uint proto, int to, int) { uint task, flag, rv; Aport *p; Asleep as; switch(waitready(d)){ case -1: return SDeio; case 1: return SDretry; } ilock(d); d->portm.flag = 0; iunlock(d); p = d->port; p->ci = 1; as.p = p; as.i = 1; d->totick = 0; if(to > 0) d->totick = Ticks + to | 1; /* fix fencepost */ d->active++; sleep(&d->portm, ahciclear, &as); d->active--; ilock(d); flag = d->portm.flag; task = d->port->task; iunlock(d); rv = SDok; if(proto & Ppkt){ rv = task >> 8 + 4 & 0xf; flag &= ~Fahdrs; flag |= Fdone; }else if(task & (Efatal<<8) || task & (ASbsy|ASdrq) && d->state == Dready){ d->port->ci = 0; ahcirecover(&d->portc); task = d->port->task; flag &= ~Fdone; /* either an error or do-over */ } if(flag == 0){ print("%s: retry\n", dnam(d)); return SDretry; } if(flag & (Fahdrs | Ferror)){ if((task & Eidnf) == 0) print("%s: i/o error %ux\n", dnam(d), task); return SDcheck; } return rv; } static int iariopkt(SDreq *r, Drive *d) { int n, count, try, max; uchar *cmd; cmd = r->cmd; aprint("%s: %.2ux %.2ux %c %d %p\n", dnam(d), cmd[0], cmd[2], "rw"[r->write], r->dlen, r->data); r->rlen = 0; count = r->dlen; max = 65536; for(try = 0; try < 10; try++){ n = count; if(n > max) n = max; qlock(&d->portm); ahcibuildpkt(&d->portm, r, r->data, n); r->status = io(d, Ppkt, 5000, 0); qunlock(&d->portm); switch(r->status){ case SDeio: return SDeio; case SDretry: continue; } // print("%.2ux :: %.2ux :: %.4ux\n", r->cmd[0], r->status, d->port->task); r->rlen = d->portm.list->len; return SDok; } print("%s: bad disk\n", dnam(d)); return r->status = SDcheck; } static long ahcibio(SDunit *u, int lun, int write, void *a, long count, uvlong lba) { int n, rw, try, status, max; uchar *data; Ctlr *c; Drive *d; c = u->dev->ctlr; d = c->drive[u->subno]; if(d->portm.feat & Datapi) return scsibio(u, lun, write, a, count, lba); max = 128; if(d->portm.feat & Dllba){ max = 8192; /* ahci maximum */ if(c->type == Tsb600) max = 255; /* errata */ } rw = write? SDwrite: SDread; data = a; for(try = 0; try < 10; try++){ n = count; if(n > max) n = max; qlock(&d->portm); ahcibuild(&d->portm, rw, data, n, lba); status = io(d, Pdma, 5000, 0); qunlock(&d->portm); switch(status){ case SDeio: return -1; case SDretry: continue; } count -= n; lba += n; data += n * u->secsize; if(count == 0) return data - (uchar*)a; } print("%s: bad disk\n", dnam(d)); return -1; } static int iario(SDreq *r) { Ctlr *c; Drive *d; SDunit *unit; unit = r->unit; c = unit->dev->ctlr; d = c->drive[unit->subno]; if(d->portm.feat & Datapi) return iariopkt(r, d); print("%s: rio on non-atapi device\n", dnam(d)); return SDeio; } /* configure drives 0-5 as ahci sata (c.f. errata) */ static int iaahcimode(Pcidev *p) { uint u; u = pcicfgr16(p, 0x92); dprint("ahci: %ux: iaahcimode %.2ux %.4ux\n", p->tbdf, pcicfgr8(p, 0x91), u); pcicfgw16(p, 0x92, u | 0xf); /* ports 0-15 (sic) */ return 0; } enum{ Ghc = 0x04/4, /* global host control */ Pi = 0x0c/4, /* ports implemented */ Cmddec = 1<<15, /* enable command block decode */ /* Ghc bits */ Ahcien = 1<<31, /* ahci enable */ }; static void iasetupahci(Ctlr *c) { pcicfgw16(c->pci, 0x40, pcicfgr16(c->pci, 0x40) & ~Cmddec); pcicfgw16(c->pci, 0x42, pcicfgr16(c->pci, 0x42) & ~Cmddec); c->lmmio[Ghc] |= Ahcien; c->lmmio[Pi] = (1 << 6) - 1; /* 5 ports (supposedly ro pi reg) */ /* enable ahci mode; from ich9 datasheet */ pcicfgw16(c->pci, 0x90, 1<<6 | 1<<5); } static void sbsetupahci(Pcidev *p) { print("sbsetupahci: tweaking %.4ux ccru %.2ux ccrp %.2ux\n", p->did, p->ccru, p->ccrp); pcicfgw8(p, 0x40, pcicfgr8(p, 0x40) | 1); pcicfgw8(p, PciCCRp, 1); pcicfgw8(p, PciCCRu, 6); pcicfgw8(p, PciCCRp, 1); p->ccru = 6; p->ccrp = 1; } static ushort itab[] = { 0xfffc, 0x2680, Tesb, 0xfffb, 0x27c1, Tahci, /* 82801g[bh]m */ 0xffff, 0x2821, Tahci, /* 82801h[roh] */ 0xfffe, 0x2824, Tahci, /* 82801h[b] */ 0xfeff, 0x2829, Tahci, /* ich8 */ 0xfffe, 0x2922, Tahci, /* ich9 */ 0xffff, 0x3a02, Tahci, /* 82801jd/do */ 0xfefe, 0x3a22, Tahci, /* ich10, pch */ 0xfff7, 0x3b28, Tahci, /* pchm */ 0xfffe, 0x3b22, Tahci, /* pch */ }; static int didtype(Pcidev *p) { int type, i; type = Tahci; switch(p->vid){ default: return -1; case 0x8086: for(i = 0; i < nelem(itab); i += 3) if((p->did & itab[i]) == itab[i+1]) return itab[i+2]; break; case 0x1002: if(p->ccru == 1 || p->ccrp != 1) if(p->did == 0x4380 || p->did == 0x4390) sbsetupahci(p); type = Tsb600; break; case 0x1106: /* * unconfirmed report that the programming * interface is set incorrectly. */ if(p->did == 0x3349) return Tahci; break; case 0x10de: case 0x1039: case 0x1b4b: case 0x11ab: break; case 0x197b: case 0x10b9: type = Tjmicron; break; } if(p->ccrb == Pcibcstore && p->ccru == 6 && p->ccrp == 1) return type; return -1; } static SDev* iapnp(void) { int i, n, nunit, type; ulong io; Ctlr *c; Drive *d; Pcidev *p; SDev *head, *tail, *s; static int done; if(done || getconf("*noahciload") != nil) return nil; done = 1; memset(olds, 0xff, sizeof olds); p = nil; head = tail = nil; loop: while((p = pcimatch(p, 0, 0)) != nil){ if((type = didtype(p)) == -1) continue; if(p->mem[Abar].bar == 0) continue; if(niactlr == NCtlr){ print("iapnp: %s: too many controllers\n", tname[type]); break; } c = iactlr + niactlr; s = sdevs + niactlr; memset(c, 0, sizeof *c); memset(s, 0, sizeof *s); io = p->mem[Abar].bar & ~0xf; io = upamalloc(io, p->mem[Abar].size, 0); if(io == 0){ print("%s: address %#p in use did %.4ux\n", tnam(c), io, p->did); continue; } c->mmio = KADDR(io); c->lmmio = (u32int*)c->mmio; c->pci = p; c->type = type; if(intel(c) && p->did != 0x2681) iasetupahci(c); ahcibioshandoff((Ahba*)c->mmio); // ahcihbareset((Ahba*)c->mmio); nunit = ahciconf(c); if(intel(c) && iaahcimode(p) == -1) continue; if(nunit < 1){ // vunmap(c->mmio, p->mem[Abar].size); continue; } s->ifc = &sdiahciifc; s->ctlr = c; s->nunit = nunit; s->idno = 'E'; c->sdev = s; c->ndrive = nunit; i = (c->hba->cap >> 21) & 1; print("ahci: %s: sata-%s with %d ports\n", tnam(c), "I\0II"+i*2, nunit); /* map the drives -- they don't all need to be enabled. */ memset(c->rawdrive, 0, sizeof c->rawdrive); n = 0; for(i = 0; i < NCtlrdrv; i++){ d = c->rawdrive + i; d->portno = i; d->driveno = -1; d->sectors = 0; d->serial[0] = ' '; d->ctlr = c; if((c->hba->pi & 1<name, sizeof d->name, "iahci%d.%d", niactlr, i); d->port = (Aport*)(c->mmio + 0x80*i + 0x100); d->portc.p = d->port; d->portc.m = &d->portm; d->driveno = n++; c->drive[d->driveno] = d; } for(i = 0; i < n; i++) if(ahciidle(c->drive[i]->port) == -1){ print("%s: port %d wedged; abort\n", tnam(c), i); goto loop; } for(i = 0; i < n; i++){ c->drive[i]->mode = DMautoneg; configdrive(c->drive[i]); } niactlr++; if(head) tail->next = s; else head = s; tail = s; } return head; } static SDev* iaid(SDev* sdev) { int i; Ctlr *c; for(; sdev; sdev = sdev->next){ if(sdev->ifc != &sdiahciifc) continue; c = sdev->ctlr; for(i = 0; i < NCtlr; i++) if(c == iactlr + i) sdev->idno = 'E' + i; } return nil; } SDifc sdiahciifc = { "iahci", iapnp, nil, /* legacy */ iaid, iaenable, iadisable, iaverify, iaonline, iario, /* rio */ nil, nil, ahcibio, };