Date: Wed, 27 Sep 2006 00:12:58 GMT From: Warner Losh <imp@FreeBSD.org> To: Perforce Change Reviews <perforce@freebsd.org> Subject: PERFORCE change 106752 for review Message-ID: <200609270012.k8R0Cwsj001551@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=106752 Change 106752 by imp@imp_lighthouse on 2006/09/27 00:12:25 queue mmc commands to mmc bridge parts. now, to make the bridge part grok it... Affected files ... .. //depot/projects/arm/src/sys/dev/mmc/mmc.c#5 edit .. //depot/projects/arm/src/sys/dev/mmc/mmcbr_if.m#4 edit .. //depot/projects/arm/src/sys/dev/mmc/mmcreg.h#6 edit Differences ... ==== //depot/projects/arm/src/sys/dev/mmc/mmc.c#5 (text+ko) ==== @@ -62,6 +62,12 @@ static void mmc_delayed_attach(void *); +static void +mmc_ms_delay(int ms) +{ + DELAY(1000 * ms); /* XXX BAD */ +} + static int mmc_probe(device_t dev) { @@ -132,9 +138,61 @@ } static void +mmc_wakeup(struct mmc_request *req) +{ + wakeup(req); +} + +static int +mmc_wait_for_req(struct mmc_softc *sc, struct mmc_request *req) +{ + int err; + + req->done = mmc_wakeup; + MMCBR_REQUEST(device_get_parent(sc->dev), sc->dev, req); + MMC_LOCK(sc); + do { + err = msleep(req, &sc->sc_mtx, PZERO | PCATCH, "mmcreq", + hz / 10); + } while (!(req->flags & MMC_REQ_DONE) && err == 0); + MMC_UNLOCK(sc); + return (err); +} + +static int +mmc_wait_for_cmd(struct mmc_softc *sc, struct mmc_command *cmd, int retries) +{ + struct mmc_request mreq; + + memset(&mreq, 0, sizeof(mreq)); + memset(cmd->resp, 0, sizeof(cmd->resp)); + cmd->retries = retries; + cmd->data = NULL; + mreq.cmd = cmd; + mmc_wait_for_req(sc, &mreq); + return (cmd->error); +} + +static void mmc_idle_cards(struct mmc_softc *sc) { - // XXX write me + device_t dev; + struct mmc_command cmd; + + dev = sc->dev; + mmcbr_set_chip_select(dev, cs_high); + mmcbr_update_ios(dev); + mmc_ms_delay(1); + + cmd.opcode = MMC_GO_IDLE_STATE; + cmd.arg = 0; + cmd.flags = MMC_RSP_NONE | MMC_CMD_BC; + mmc_wait_for_cmd(sc, &cmd, 0); + mmc_ms_delay(1); + + mmcbr_set_chip_select(dev, cs_dontcare); + mmcbr_update_ios(dev); + mmc_ms_delay(1); } static int @@ -164,14 +222,12 @@ mmcbr_set_power_mode(dev, power_up); mmcbr_set_clock(dev, 0); mmcbr_update_ios(dev); - - // delay 1ms ? + mmc_ms_delay(1); mmcbr_set_clock(dev, mmcbr_get_f_min(sc->dev)); mmcbr_set_power_mode(dev, power_on); mmcbr_update_ios(dev); - - // delay 2ms ? + mmc_ms_delay(2); } static void ==== //depot/projects/arm/src/sys/dev/mmc/mmcbr_if.m#4 (text+ko) ==== ==== //depot/projects/arm/src/sys/dev/mmc/mmcreg.h#6 (text+ko) ==== @@ -25,7 +25,7 @@ */ #ifndef DEV_MMC_MMCREG_H -#define DEV_MMC_MMCREG_H +#define DEV_MMC_MMCREG_H /* * This file contains the register definitions for the mmc and sd busses. @@ -40,36 +40,44 @@ uint32_t arg; uint32_t resp[4]; uint32_t flags; /* Expected responses */ -#define MMC_RSP_NONE (0UL << 0) -#define MMC_RSP_SHORT (1UL << 0) -#define MMC_RSP_LONG (2UL << 0) -#define MMC_RSP_MASK (3UL << 0) -#define MMC_RSP_CRC (1UL << 3) /* Expect valid crc */ -#define MMC_RSP_BUSY (1UL << 4) /* Card may send busy */ +#define MMC_RSP_PRESENT (1ul << 0) /* Response */ +#define MMC_RSP_136 (1ul << 1) /* 136 bit response */ +#define MMC_RSP_CRC (1ul << 2) /* Expect valid crc */ +#define MMC_RSP_BUSY (1ul << 3) /* Card may send busy */ +#define MMC_RSP_OPCODE (1ul << 4) /* Response include opcode */ +#define MMC_CMD_AC (0ul << 5) /* AC? */ +#define MMC_CMD_ADTC (1ul << 5) /* ADTC? */ +#define MMC_CMD_BC (2ul << 5) /* BC? */ +#define MMC_CMD_BCR (3ul << 5) /* BCR? */ +#define MMC_RSP_MASK 0x1ful +#define MMC_CMD_MASK (3ul << 5) /* Possible response types defined in the standard: */ -#define MMC_RSP_R1 (MMC_RSP_SHORT | MMC_RSP_CRC) -#define MMC_RSP_R1B (MMC_RSP_SHORT | MMC_RSP_CRC | MMC_RSP_BUSY) -#define MMC_RSP_R2 (MMC_RSP_LONG | MMC_RSP_CRC) -#define MMC_RSP_R3 (MMC_RSP_SHORT) -#define MMC_RSP_R6 (MMC_RSP_SHORT | MMC_RSP_CRC) - uint32_t retires; +#define MMC_RSP_NONE (0) +#define MMC_RSP_R1 (MMC_RPS_PRESENT | MMC_RSP_CRC | MMC_RSP_OPCODE) +#define MMC_RSP_R1B (MMC_RSP_PRESENT | MMC_RSP_CRC | MMC_RSP_OPCODE | MMC_RSP_BUSY) +#define MMC_RSP_R2 (MMC_RSP_PRESENT | MMC_RSP_136 | MMC_RSP_CRC) +#define MMC_RSP_R3 (MMC_RSP_PRESENT) +#define MMC_RSP_R6 (MMC_RSP_PRESENT | MMC_RSP_CRC) +/* R7 -- new in sd 2.0 */ +#define MMC_RSP(x) ((x) & MMC_RSP_MASK) + uint32_t retries; uint32_t error; -#define MMC_ERR_NONE 0 -#define MMC_ERR_TIMEOUT 1 -#define MMC_ERR_BADCRC 2 -#define MMC_ERR_FIFO 3 -#define MMC_ERR_FAILED 4 -#define MMC_ERR_INVALID 5 +#define MMC_ERR_NONE 0 +#define MMC_ERR_TIMEOUT 1 +#define MMC_ERR_BADCRC 2 +#define MMC_ERR_FIFO 3 +#define MMC_ERR_FAILED 4 +#define MMC_ERR_INVALID 5 struct mmc_data *data; /* Data segment with cmd */ struct mmc_request *mrq; /* backpointer to request */ }; struct mmc_data { uint32_t flags; -#define MMC_DATA_WRITE (1UL << 8) -#define MMC_DATA_READ (1UL << 9) -#define MMC_DATA_STREAM (1UL << 10) +#define MMC_DATA_WRITE (1UL << 8) +#define MMC_DATA_READ (1UL << 9) +#define MMC_DATA_STREAM (1UL << 10) struct mmc_request *mrq; }; @@ -79,17 +87,122 @@ struct mmc_command *stop; void (*done)(struct mmc_request *); /* Completion function */ void *done_data; /* requestor set data */ + uint32_t flags; +#define MMC_REQ_DONE 1 }; +/* Command definitions */ + +/* Class 0 and 1: Basic commands & read stream commands */ +#define MMC_GO_IDLE_STATE 0 +#define MMC_SEND_OP_COND 1 +#define MMC_ALL_SEND_CID 2 +#define MMC_SET_RELATIVE_ADDR 3 +#define MMC_SET_DSR 4 + /* reserved: 5 */ +#define MMC_SELECT_CARD 7 +#define MMC_DESELECT_CARD 7 +#define MMC_SEND_IF_COND 8 +#define MMC_SEND_CSD 9 +#define MMC_SEND_CID 10 +#define MMC_READ_DAT_UNTIL_STOP 11 +#define MMC_STOP_TRANSMISSION 12 +#define MMC_SEND_STATUS 13 + /* reserved: 14 */ +#define MMC_GO_INACTIVE_STATE 15 + +/* Class 2: Block oriented read commands */ +#define MMC_SET_BLOCKLEN 16 +#define MMC_READ_SINGLE_BLOCK 17 +#define MMC_READ_MULTIPLE_BLOCK 18 + /* reserved: 19 */ + +/* Class 3: Stream write commands */ +#define MMC_WRITE_DAT_UNTIL_STOP 20 + /* reserved: 21 */ + /* reserved: 22 */ + +/* Class 4: Block oriented write commands */ +#define MMC_SET_BLOCK_COUNT 23 +#define MMC_WRITE_BLOCK 24 +#define MMC_WRITE_MULTIPLE_BLOCK 25 +#define MMC_PROGARM_CID 26 +#define MMC_PROGRAM_CSD 27 + +/* Class 6: Block oriented write protection commands */ +#define MMC_SET_WRITE_PROT 28 +#define MMC_CLR_WRITE_PROT 29 +#define MMC_SEND_WRITE_PROT 30 + /* reserved: 31 */ + +/* Class 5: Erase commands */ +#define SD_ERASE_WR_BLK_START 32 +#define SD_ERASE_WR_BLK_END 33 + /* 34 -- reserved old command */ +#define MMC_ERASE_GROUP_START 35 +#define MMC_ERASE_GROUP_END 36 + /* 37 -- reserved old command */ +#define MMC_ERASE 38 + +/* Class 9: I/O mode commands */ +#define MMC_FAST_IO 39 +#define MMC_GO_IRQ_STATE 40 + /* reserved: 41 */ + +/* Class 7: Lock card */ +#define MMC_LOCK_UNLOCK 42 + /* reserved: 43 */ + /* reserved: 44 */ + /* reserved: 45 */ + /* reserved: 46 */ + /* reserved: 47 */ + /* reserved: 48 */ + /* reserved: 49 */ + /* reserved: 50 */ + /* reserved: 51 */ + /* reserved: 54 */ + +/* Class 8: Application specific commands */ +#define MMC_APP_CMD 55 +#define MMC_GEN_CMD 56 + /* reserved: 57 */ + /* reserved: 58 */ + /* reserved: 59 */ + /* reserved for mfg: 60 */ + /* reserved for mfg: 61 */ + /* reserved for mfg: 62 */ + /* reserved for mfg: 63 */ + +/* Class 9: I/O cards (sd) */ +#define SD_IO_RW_DIRECT 52 +#define SD_IO_RW_EXTENDED 53 + +/* Class 10: Switch function commands */ +#define SD_SWITCH_FUNC 6 + /* reserved: 34 */ + /* reserved: 35 */ + /* reserved: 36 */ + /* reserved: 37 */ + /* reserved: 50 */ + /* reserved: 57 */ + + +/* Application specific commands for SD */ +#define ACMD_SET_BUS_WIDTH 6 +#define ACMD_SD_STATUS 13 +#define ACMD_SEND_NUM_WR_BLOCKS 22 +#define ACMD_SET_WR_BLK_ERASE_COUNT 23 +#define ACMD_SD_SEND_OP_COND 41 +#define ACMD_SET_CLR_CARD_DETECT 42 +#define ACMD_SEND_SCR 51 + /* OCR bits */ -#define MMC_OCR_145_150 (1U << 0) /* Vdd voltage 1.45 ~ 1.50 */ -#define MMC_OCR_150_155 (1U << 1) /* Vdd voltage 1.50 ~ 1.50 */ -#define MMC_OCR_155_160 (1U << 2) /* Vdd voltage 1.55 ~ 1.60 */ -#define MMC_OCR_160_165 (1U << 3) /* Vdd voltage 1.60 ~ 1.65 */ -#define MMC_OCR_165_170 (1U << 4) /* Vdd voltage 1.65 ~ 1.70 */ -#define MMC_OCR_170_180 (1U << 5) /* Vdd voltage 1.70 ~ 1.80 */ -#define MMC_OCR_180_190 (1U << 6) /* Vdd voltage 1.80 ~ 1.90 */ -#define MMC_OCR_190_200 (1U << 7) /* Vdd voltage 1.90 ~ 2.00 */ + +/* in SD 2.0 spec, bits 8-14 are now marked reserved */ +/* Low voltage in SD2.0 spec is bit 7, TBD voltage */ +/* Low voltage in MC 3.31 spec is bit 7, 1.65-1.95V */ +/* Linux has defines for lower bits down to 0, for reasons unknown */ +#define MMC_OCR_LOW_VOLTAGE (1u << 7) /* Low Voltage Range -- tbd */ #define MMC_OCR_200_210 (1U << 8) /* Vdd voltage 2.00 ~ 2.10 */ #define MMC_OCR_210_220 (1U << 9) /* Vdd voltage 2.10 ~ 2.20 */ #define MMC_OCR_220_230 (1U << 10) /* Vdd voltage 2.20 ~ 2.30 */ @@ -106,6 +219,7 @@ #define MMC_OCR_330_340 (1U << 21) /* Vdd voltage 3.30 ~ 3.40 */ #define MMC_OCR_340_350 (1U << 22) /* Vdd voltage 3.40 ~ 3.50 */ #define MMC_OCR_350_360 (1U << 23) /* Vdd voltage 3.50 ~ 3.60 */ -#define MMC_OCR_CARD_BUSY (1U << 31) /* Card Power up status */ +#define MMC_OCR_CCS (1u << 30) /* Card Capacity status (SD vs SDHC) */ +#define MMC_OCR_CARD_BUSY (1U << 31) /* Card Power up status */ #endif /* DEV_MMCREG_H */
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200609270012.k8R0Cwsj001551>