ref: 457ba087b13a881b013940f6e54c09566ec10ed1
dir: /sys/src/9/port/sdmmc.c/
/* * mmc / sd memory card * * Copyright © 2012 Richard Miller <r.miller@acm.org> * */ #include "u.h" #include "../port/lib.h" #include "../port/error.h" #include "mem.h" #include "dat.h" #include "fns.h" #include "io.h" #include "../port/sd.h" /* Commands */ SDiocmd GO_IDLE_STATE = { 0, 0, 0, 0, "GO_IDLE_STATE" }; SDiocmd SEND_OP_COND = { 1, 3, 0, 0, "SEND_OP_COND" }; SDiocmd ALL_SEND_CID = { 2, 2, 0, 0, "ALL_SEND_CID" }; SDiocmd SET_RELATIVE_ADDR = { 3, 1, 0, 0, "SET_RELATIVE_ADDR" }; SDiocmd SEND_RELATIVE_ADDR = { 3, 6, 0, 0, "SEND_RELATIVE_ADDR" }; SDiocmd SWITCH = { 6, 1, 1, 0, "SWITCH" }; SDiocmd SWITCH_FUNC = { 6, 1, 0, 1, "SWITCH_FUNC" }; SDiocmd SELECT_CARD = { 7, 1, 1, 0, "SELECT_CARD" }; SDiocmd SEND_EXT_CSD = { 8, 1, 0, 1, "SEND_EXT_CSD" }; SDiocmd SD_SEND_IF_COND = { 8, 1, 0, 0, "SD_SEND_IF_COND" }; SDiocmd SEND_CSD = { 9, 2, 0, 0, "SEND_CSD" }; SDiocmd STOP_TRANSMISSION = {12, 1, 1, 0, "STOP_TRANSMISSION" }; SDiocmd SEND_STATUS = {13, 1, 0, 0, "SEND_STATUS" }; SDiocmd SET_BLOCKLEN = {16, 1, 0, 0, "SET_BLOCKLEN" }; SDiocmd READ_SINGLE_BLOCK = {17, 1, 0, 1, "READ_SINGLE_BLOCK" }; SDiocmd READ_MULTIPLE_BLOCK = {18, 1, 0, 3, "READ_MULTIPLE_BLOCK" }; SDiocmd WRITE_SINGLE_BLOCK = {24, 1, 0, 2, "WRITE_SINGLE_BLOCK" }; SDiocmd WRITE_MULTIPLE_BLOCK = {25, 1, 0, 4, "WRITE_MULTIPLE_BLOCK" }; /* prefix for following app-specific commands */ SDiocmd APP_CMD = {55, 1, 0, 0, "APP_CMD" }; SDiocmd SD_SET_BUS_WIDTH = { 6, 1, 0, 0, "SD_SET_BUS_WIDTH" }; SDiocmd SD_SEND_OP_COND = {41, 3, 0, 0, "SD_SEND_OP_COND" }; /* Command arguments */ enum { /* SD_SEND_IF_COND */ Voltage = 1<<8, /* Host supplied voltage range 2.7-3.6 volts*/ Checkpattern = 0x42, /* SELECT_CARD */ Rcashift = 16, /* SD_SET_BUS_WIDTH */ Width1 = 0<<0, Width4 = 2<<0, /* SWITCH_FUNC */ Dfltspeed = 0<<0, Hispeed = 1<<0, Checkfunc = 0x00FFFFF0, Setfunc = 0x80FFFFF0, Funcbytes = 64, /* SWITCH */ MMCSetSDTiming = 0x3B90000, MMCSetHSTiming = 0x3B90100, MMCSetBusWidth1 = 0x3B70000, MMCSetBusWidth4 = 0x3B70100, MMCSetBusWidth8 = 0x3B70200, /* OCR (operating conditions register) */ Powerup = 1<<31, Hcs = 1<<30, /* Host capacity support */ Ccs = 1<<30, /* Card capacity status */ V3_3 = 3<<20, /* 3.2-3.4 volts */ }; enum { Multiblock = 1, Inittimeout = 15, Initfreq = 400000, /* initialisation frequency for MMC */ SDfreq = 25000000, /* standard SD frequency */ SDfreqhs = 50000000, /* highspeed frequency */ }; typedef struct Card Card; struct Card { SDev *dev; SDio *io; int ismmc; int specver; int buswidth; int busspeed; /* SD card registers */ u32int rca; u32int ocr; u32int cid[4]; u32int csd[4]; u8int ext_csd[512]; int retry; }; extern SDifc sdmmcifc; static SDio *sdio[8]; static int nsdio, isdio; void addmmcio(SDio *io) { assert(io != nil); assert(isdio == 0); if(nsdio >= nelem(sdio)){ print("addmmcio: out of slots for %s\n", io->name); return; } sdio[nsdio++] = io; } static SDev* init1(void) { SDev *sdev; Card *card; SDio *io; int more; if(isdio >= nsdio) return nil; if((sdev = malloc(sizeof(SDev))) == nil) return nil; if((card = malloc(sizeof(Card))) == nil){ free(sdev); return nil; } if((io = malloc(sizeof(SDio))) == nil){ free(card); free(sdev); return nil; } Next: memmove(io, sdio[isdio++], sizeof(SDio)); if(io->init != nil){ more = (*io->init)(io); if(more < 0){ if(isdio < nsdio) goto Next; free(io); free(card); free(sdev); return nil; } if(more > 0) isdio--; /* try again */ } sdev->idno = 'M'; sdev->ifc = &sdmmcifc; sdev->nunit = 1; sdev->ctlr = card; card->dev = sdev; card->io = io; return sdev; } static SDev* mmcpnp(void) { SDev *list = nil, **link = &list; while((*link = init1()) != nil) link = &(*link)->next; return list; } static void readextcsd(Card *card) { SDio *io = card->io; u32int r[4]; uchar *buf; buf = sdmalloc(512); if(waserror()){ sdfree(buf); nexterror(); } (*io->iosetup)(io, 0, buf, 512, 1); (*io->cmd)(io, &SEND_EXT_CSD, 0, r); (*io->io)(io, 0, buf, 512); memmove(card->ext_csd, buf, sizeof card->ext_csd); sdfree(buf); poperror(); } static uint rbits(u32int *p, uint start, uint len) { uint w, off, v; w = start / 32; off = start % 32; if(off == 0) v = p[w]; else v = p[w] >> off | p[w+1] << (32-off); if(len < 32) return v & ((1<<len) - 1); else return v; } static uvlong rbytes(uchar *p, uint start, uint len) { uvlong v = 0; uint i; p += start; for(i = 0; i < len; i++) v |= (uvlong)p[i] << 8*i; return v; } static void identify(SDunit *unit) { Card *card = unit->dev->ctlr; uint csize, mult; uvlong capacity; #define CSD(end, start) rbits(card->csd, start, (end)-(start)+1) mult = CSD(49, 47); csize = CSD(73, 62); unit->sectors = (csize+1) * (1<<(mult+2)); unit->secsize = 1 << CSD(83, 80); card->specver = 0; if(card->ismmc){ switch(CSD(125, 122)){ case 0: default: card->specver = 120; /* 1.2 */ break; case 1: card->specver = 140; /* 1.4 */ break; case 2: card->specver = 122; /* 1.22 */ break; case 3: card->specver = 300; /* 3.0 */ break; case 4: card->specver = 400; /* 4.0 */ break; } switch(CSD(127, 126)){ case 2: /* MMC CSD Version 1.2 */ case 3: /* MMC Version coded in EXT_CSD */ if(card->specver < 400) break; readextcsd(card); #define EXT_CSD(end, start) rbytes(card->ext_csd, start, (end)-(start)+1) switch((uchar)EXT_CSD(192, 192)){ case 8: card->specver = 510; /* 5.1 */ break; case 7: card->specver = 500; /* 5.0 */ break; case 6: card->specver = 450; /* 4.5/4.51 */ break; case 5: card->specver = 441; /* 4.41 */ break; case 3: card->specver = 430; /* 4.3 */ break; case 2: card->specver = 420; /* 4.2 */ break; case 1: card->specver = 410; /* 4.1 */ break; case 0: card->specver = 400; /* 4.0 */ break; } } if(card->specver >= 420) { capacity = EXT_CSD(215, 212) * 512ULL; if(capacity > 0x80000000ULL) unit->sectors = capacity / unit->secsize; } } else { switch(CSD(127, 126)){ case 0: /* SD Version 1.0 */ card->specver = 100; break; case 1: /* SD Version 2.0 */ card->specver = 200; csize = CSD(69, 48); capacity = (csize+1) * 0x80000ULL; unit->sectors = capacity / unit->secsize; break; } } if(unit->secsize == 1024){ unit->sectors <<= 1; unit->secsize = 512; } } static int mmcverify(SDunit *unit) { Card *card = unit->dev->ctlr; SDio *io = card->io; int n; n = (*io->inquiry)(io, (char*)&unit->inquiry[8], sizeof(unit->inquiry)-8); if(n < 0) return 0; unit->inquiry[0] = 0x00; /* direct access (disk) */ unit->inquiry[1] = 0x80; /* removable */ unit->inquiry[4] = sizeof(unit->inquiry)-4; return 1; } static int mmcenable(SDev* dev) { Card *card = dev->ctlr; SDio *io = card->io; (*io->enable)(io); return 1; } static void mmcswitch(Card *card, u32int arg) { SDio *io = card->io; u32int r[4]; int i; (*io->cmd)(io, &SWITCH, arg, r); for(i=0; i<10;i++){ tsleep(&up->sleep, return0, nil, 100); (*io->cmd)(io, &SEND_STATUS, card->rca<<Rcashift, r); if(r[0] & (1<<7)) error(Eio); if(r[0] & (1<<8)) return; } error(Eio); } static void sdswitchfunc(Card *card, int arg) { SDio *io = card->io; u32int r[4]; uchar *buf; int n; n = Funcbytes; buf = sdmalloc(n); if(waserror()){ sdfree(buf); nexterror(); } (*io->iosetup)(io, 0, buf, n, 1); (*io->cmd)(io, &SWITCH_FUNC, arg, r); if((arg & 0xFFFFFFF0) == Setfunc){ card->busspeed = (arg & Hispeed) != 0? SDfreqhs: SDfreq; tsleep(&up->sleep, return0, nil, 10); (*io->bus)(io, 0, card->busspeed); tsleep(&up->sleep, return0, nil, 10); } (*io->io)(io, 0, buf, n); sdfree(buf); poperror(); } static int cardinit(Card *card) { SDio *io = card->io; u32int r[4], ocr; int i; card->buswidth = 1; card->busspeed = Initfreq; (*io->bus)(io, card->buswidth, card->busspeed); tsleep(&up->sleep, return0, nil, 10); (*io->cmd)(io, &GO_IDLE_STATE, 0, r); /* card type unknown */ card->ismmc = -1; if(!waserror()){ /* try SD card first */ ocr = V3_3; if(!waserror()){ (*io->cmd)(io, &SD_SEND_IF_COND, Voltage|Checkpattern, r); if(r[0] == (Voltage|Checkpattern)){ /* SD 2.0 or above */ ocr |= Hcs; card->ismmc = 0; /* this is SD card */ } poperror(); } for(i = 0; i < Inittimeout; i++){ tsleep(&up->sleep, return0, nil, 100); (*io->cmd)(io, &APP_CMD, 0, r); (*io->cmd)(io, &SD_SEND_OP_COND, ocr, r); if(r[0] & Powerup) break; } card->ismmc = 0; /* this is SD card */ poperror(); if(i == Inittimeout) return 2; } else if(card->ismmc) { /* try MMC if not ruled out */ (*io->cmd)(io, &GO_IDLE_STATE, 0, r); ocr = Hcs|V3_3; for(i = 0; i < Inittimeout; i++){ tsleep(&up->sleep, return0, nil, 100); (*io->cmd)(io, &SEND_OP_COND, ocr, r); if(r[0] & Powerup) break; } card->ismmc = 1; /* this is MMC */ if(i == Inittimeout) return 2; } card->ocr = r[0]; (*io->cmd)(io, &ALL_SEND_CID, 0, r); memmove(card->cid, r, sizeof card->cid); if(card->ismmc){ card->rca = 0; (*io->cmd)(io, &SET_RELATIVE_ADDR, card->rca<<16, r); } else { (*io->cmd)(io, &SEND_RELATIVE_ADDR, 0, r); card->rca = r[0]>>16; } (*io->cmd)(io, &SEND_CSD, card->rca<<Rcashift, r); memmove(card->csd, r, sizeof card->csd); return 1; } static void retryproc(void *arg) { Card *card = arg; int i = 0; while(waserror()) ; if(i++ < card->retry) cardinit(card); USED(i); card->retry = 0; pexit("", 1); } static int mmconline(SDunit *unit) { Card *card = unit->dev->ctlr; SDio *io = card->io; u32int r[4]; assert(unit->subno == 0); if(card->retry) return 0; if(waserror()){ unit->sectors = 0; if(card->retry++ == 0) kproc(unit->name, retryproc, card); return 0; } if(unit->sectors != 0){ (*io->cmd)(io, &SEND_STATUS, card->rca<<Rcashift, r); poperror(); return 1; } if(cardinit(card) != 1){ poperror(); return 2; } (*io->cmd)(io, &SELECT_CARD, card->rca<<Rcashift, r); tsleep(&up->sleep, return0, nil, 10); (*io->bus)(io, 0, card->busspeed = SDfreq); tsleep(&up->sleep, return0, nil, 10); identify(unit); (*io->cmd)(io, &SET_BLOCKLEN, unit->secsize, r); if(card->ismmc && card->specver >= 400){ if(!waserror()){ mmcswitch(card, MMCSetHSTiming); (*io->bus)(io, 0, card->busspeed = SDfreqhs); tsleep(&up->sleep, return0, nil, 10); readextcsd(card); poperror(); } else { mmcswitch(card, MMCSetSDTiming); (*io->bus)(io, 0, card->busspeed = SDfreq); tsleep(&up->sleep, return0, nil, 10); readextcsd(card); } if(!waserror()){ mmcswitch(card, MMCSetBusWidth8); (*io->bus)(io, card->buswidth = 8, 0); readextcsd(card); poperror(); } else if(!waserror()){ mmcswitch(card, MMCSetBusWidth4); (*io->bus)(io, card->buswidth = 4, 0); readextcsd(card); poperror(); } else { mmcswitch(card, MMCSetBusWidth1); (*io->bus)(io, card->buswidth = 1, 0); readextcsd(card); } } else if(!card->ismmc) { (*io->cmd)(io, &APP_CMD, card->rca<<Rcashift, r); (*io->cmd)(io, &SD_SET_BUS_WIDTH, Width4, r); (*io->bus)(io, card->buswidth = 4, 0); if(!waserror()){ sdswitchfunc(card, Hispeed|Setfunc); poperror(); } } poperror(); return 1; } static int mmcrctl(SDunit *unit, char *p, int l) { Card *card = unit->dev->ctlr; char *s = p, *e = s + l; int i; assert(unit->subno == 0); if(unit->sectors == 0){ mmconline(unit); if(unit->sectors == 0) return 0; } p = seprint(p, e, "version %s %d.%2.2d\n", card->ismmc? "MMC": "SD", card->specver/100, card->specver%100); p = seprint(p, e, "rca %4.4ux ocr %8.8ux", card->rca, card->ocr); p = seprint(p, e, "\ncid "); for(i = nelem(card->cid)-1; i >= 0; i--) p = seprint(p, e, "%8.8ux", card->cid[i]); p = seprint(p, e, "\ncsd "); for(i = nelem(card->csd)-1; i >= 0; i--) p = seprint(p, e, "%8.8ux", card->csd[i]); p = seprint(p, e, "\ngeometry %llud %ld\n", unit->sectors, unit->secsize); return p - s; } static long mmcbio(SDunit *unit, int lun, int write, void *data, long nb, uvlong bno) { Card *card = unit->dev->ctlr; SDio *io = card->io; int len, tries; u32int r[4]; uchar *buf; ulong b; USED(lun); assert(unit->subno == 0); if(unit->sectors == 0) error(Echange); buf = data; len = unit->secsize; if(Multiblock && (!write || !io->nomultiwrite)){ b = bno; tries = 0; while(waserror()) if(++tries == 3) nexterror(); (*io->iosetup)(io, write, buf, len, nb); if(waserror()){ (*io->cmd)(io, &STOP_TRANSMISSION, 0, r); nexterror(); } (*io->cmd)(io, write? &WRITE_MULTIPLE_BLOCK: &READ_MULTIPLE_BLOCK, card->ocr & Ccs? b: b * len, r); (*io->io)(io, write, buf, nb * len); poperror(); (*io->cmd)(io, &STOP_TRANSMISSION, 0, r); poperror(); b += nb; }else{ for(b = bno; b < bno + nb; b++){ (*io->iosetup)(io, write, buf, len, 1); (*io->cmd)(io, write? &WRITE_SINGLE_BLOCK: &READ_SINGLE_BLOCK, card->ocr & Ccs? b: b * len, r); (*io->io)(io, write, buf, len); buf += len; } } return (b - bno) * len; } static int mmcrio(SDreq *r) { int i, rw, count; uvlong lba; if((i = sdfakescsi(r)) != SDnostatus) return r->status = i; if((i = sdfakescsirw(r, &lba, &count, &rw)) != SDnostatus) return i; r->rlen = mmcbio(r->unit, r->lun, rw == SDwrite, r->data, count, lba); return r->status = SDok; } SDifc sdmmcifc = { .name = "mmc", .pnp = mmcpnp, .enable = mmcenable, .verify = mmcverify, .online = mmconline, .rctl = mmcrctl, .bio = mmcbio, .rio = mmcrio, };