summaryrefslogtreecommitdiff
path: root/sys/dev/sdmmc/sdmmc_io.c
diff options
context:
space:
mode:
authorUwe Stuehler <uwe@cvs.openbsd.org>2006-06-01 21:53:42 +0000
committerUwe Stuehler <uwe@cvs.openbsd.org>2006-06-01 21:53:42 +0000
commitb89f9d40f1fe1476b0c5078e2a2e451157066db2 (patch)
tree9e87d5289bdf55af01bdec6676ba02079a9c38af /sys/dev/sdmmc/sdmmc_io.c
parent6b37ef9f877f52e97b18ea962dbb1c3aa7458dfc (diff)
SDIO card identification
Diffstat (limited to 'sys/dev/sdmmc/sdmmc_io.c')
-rw-r--r--sys/dev/sdmmc/sdmmc_io.c333
1 files changed, 321 insertions, 12 deletions
diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c
index f739b210721..f811eb0e7e3 100644
--- a/sys/dev/sdmmc/sdmmc_io.c
+++ b/sys/dev/sdmmc/sdmmc_io.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: sdmmc_io.c,v 1.2 2006/05/28 18:45:23 uwe Exp $ */
+/* $OpenBSD: sdmmc_io.c,v 1.3 2006/06/01 21:53:41 uwe Exp $ */
/*
* Copyright (c) 2006 Uwe Stuehler <uwe@openbsd.org>
@@ -21,16 +21,36 @@
#include <sys/param.h>
#include <sys/systm.h>
+#include <dev/sdmmc/sdmmc_ioreg.h>
#include <dev/sdmmc/sdmmcchip.h>
#include <dev/sdmmc/sdmmcreg.h>
#include <dev/sdmmc/sdmmcvar.h>
+int sdmmc_submatch(struct device *, void *, void *);
+int sdmmc_print(void *, const char *);
+int sdmmc_io_rw_direct(struct sdmmc_softc *, struct sdmmc_function *,
+ int, u_char *, int);
+int sdmmc_io_rw_extended(struct sdmmc_softc *, struct sdmmc_function *,
+ int, u_char *, int, int);
+int sdmmc_io_write(struct sdmmc_softc *, struct sdmmc_function *,
+ int, u_char);
+int sdmmc_io_xchg(struct sdmmc_softc *, struct sdmmc_function *,
+ int, u_char *);
+void sdmmc_io_reset(struct sdmmc_softc *);
+int sdmmc_io_send_op_cond(struct sdmmc_softc *, u_int32_t, u_int32_t *);
+
#ifdef SDMMC_DEBUG
#define DPRINTF(s) printf s
#else
#define DPRINTF(s) /**/
#endif
+#ifdef SDMMC_DEBUG
+int sdmmc_verbose = 1;
+#else
+int sdmmc_verbose = 0;
+#endif
+
/*
* Initialize SD I/O card functions (before memory cards). The host
* system and controller must support card interrupts in order to use
@@ -45,7 +65,7 @@ sdmmc_io_enable(struct sdmmc_softc *sc)
/* Set host mode to SD "combo" card. */
SET(sc->sc_flags, SMF_SD_MODE|SMF_IO_MODE|SMF_MEM_MODE);
- /* Reset I/O functions (*must* do that before CMD5). */
+ /* Reset I/O functions. */
sdmmc_io_reset(sc);
/*
@@ -60,11 +80,20 @@ sdmmc_io_enable(struct sdmmc_softc *sc)
return 0;
}
- if (SD_IO_OCR_NF(card_ocr) == 0) {
- /* No I/O functions. */
+ /* Parse the additional bits in the I/O OCR value. */
+ if (!ISSET(card_ocr, SD_IO_OCR_MEM_PRESENT)) {
+ /* SDIO card without memory (not a "combo card"). */
+ DPRINTF(("%s: no memory present\n", SDMMCDEVNAME(sc)));
+ CLR(sc->sc_flags, SMF_MEM_MODE);
+ }
+ sc->sc_function_count = SD_IO_OCR_NUM_FUNCTIONS(card_ocr);
+ if (sc->sc_function_count == 0) {
+ /* Useless SDIO card without any I/O functions. */
DPRINTF(("%s: no I/O functions\n", SDMMCDEVNAME(sc)));
+ CLR(sc->sc_flags, SMF_IO_MODE);
return 0;
}
+ card_ocr &= SD_IO_OCR_MASK;
/* Set the lowest voltage supported by the card and host. */
host_ocr = sdmmc_chip_host_ocr(sc->sct, sc->sch);
@@ -86,18 +115,300 @@ sdmmc_io_enable(struct sdmmc_softc *sc)
}
/*
- * Send the "I/O RESET" command.
+ * Allocate sdmmc_function structures for SD card I/O function
+ * (including function 0).
*/
void
-sdmmc_io_reset(struct sdmmc_softc *sc)
+sdmmc_io_scan(struct sdmmc_softc *sc)
+{
+ struct sdmmc_function *sf0, *sf;
+ int i;
+
+ sf0 = sdmmc_function_alloc(sc);
+ sf0->number = 0;
+ if (sdmmc_set_relative_addr(sc, sf0) != 0) {
+ printf("%s: can't set I/O RCA\n", SDMMCDEVNAME(sc));
+ SET(sf0->flags, SFF_ERROR);
+ return;
+ }
+ sc->sc_fn0 = sf0;
+ SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf0, sf_list);
+
+ /* Verify that the RCA has been set by selecting the card. */
+ if (sdmmc_select_card(sc, sf0) != 0) {
+ printf("%s: can't select I/O RCA %d\n", SDMMCDEVNAME(sc),
+ sf0->rca);
+ SET(sf0->flags, SFF_ERROR);
+ return;
+ }
+
+ for (i = 1; i <= sc->sc_function_count; i++) {
+ sf = sdmmc_function_alloc(sc);
+ sf->number = i;
+ sf->rca = sf0->rca;
+
+ SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list);
+ }
+}
+
+/*
+ * Initialize SDIO card functions.
+ */
+int
+sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf)
+{
+ if (sf->number == 0) {
+ (void)sdmmc_io_write(sc, sf, SD_IO_CCCR_BUS_WIDTH,
+ CCCR_BUS_WIDTH_1);
+
+ if (sdmmc_read_cis(sf, &sf->cis) != 0) {
+ printf("%s: can't read CIS\n");
+ SET(sf->flags, SFF_ERROR);
+ return 1;
+ }
+
+ sdmmc_check_cis_quirks(sf);
+
+ if (sdmmc_verbose)
+ sdmmc_print_cis(sf);
+ }
+ return 0;
+}
+
+void
+sdmmc_io_function_enable(struct sdmmc_function *sf)
+{
+ struct sdmmc_function *sf0 = sf->sc->sc_fn0;
+ u_int8_t rv;
+
+ rv = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_ENABLE);
+ rv |= (1<<sf->number);
+ sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_ENABLE, rv);
+}
+
+void
+sdmmc_io_function_disable(struct sdmmc_function *sf)
+{
+ struct sdmmc_function *sf0 = sf->sc->sc_fn0;
+ u_int8_t rv;
+
+ rv = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_ENABLE);
+ rv &= ~(1<<sf->number);
+ sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_ENABLE, rv);
+}
+
+void
+sdmmc_io_attach(struct sdmmc_softc *sc)
+{
+ struct sdmmc_function *sf;
+ struct sdmmc_attach_args saa;
+
+ SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) {
+ bzero(&saa, sizeof saa);
+ saa.sf = sf;
+
+ sf->child = config_found_sm(&sc->sc_dev, &saa, sdmmc_print,
+ sdmmc_submatch);
+ }
+}
+
+int
+sdmmc_submatch(struct device *parent, void *match, void *aux)
+{
+ struct cfdata *cf = match;
+
+ /* Skip the scsibus, it is configured directly. */
+ if (strcmp(cf->cf_driver->cd_name, "scsibus") == 0)
+ return 0;
+
+ return cf->cf_attach->ca_match(parent, cf, aux);
+}
+
+int
+sdmmc_print(void *aux, const char *pnp)
+{
+ struct sdmmc_attach_args *sa = aux;
+ struct sdmmc_function *sf = sa->sf;
+ struct sdmmc_cis *cis = &sf->sc->sc_fn0->cis;
+ int i;
+
+ if (pnp) {
+ if (sf->number == 0)
+ return QUIET;
+
+ for (i = 0; i < 4 && cis->cis1_info[i]; i++)
+ printf("%s%s", i ? ", " : "\"", cis->cis1_info[i]);
+ if (i != 0)
+ printf("\"");
+
+ if (cis->manufacturer != SDMMC_VENDOR_INVALID &&
+ cis->product != SDMMC_PRODUCT_INVALID) {
+ printf("%s(", i ? " " : "");
+ if (cis->manufacturer != SDMMC_VENDOR_INVALID)
+ printf("manufacturer 0x%x%s",
+ cis->manufacturer,
+ cis->product == SDMMC_PRODUCT_INVALID ?
+ "" : ", ");
+ if (cis->product != SDMMC_PRODUCT_INVALID)
+ printf("product 0x%x", cis->product);
+ printf(")");
+ }
+ printf("%sat %s", i ? " " : "", pnp);
+ }
+ printf(" function %d", sf->number);
+ return UNCONF;
+}
+
+void
+sdmmc_io_detach(struct sdmmc_softc *sc)
+{
+ struct sdmmc_function *sf;
+
+ SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) {
+ if (sf->child != NULL) {
+ config_detach(sf->child, DETACH_FORCE);
+ sf->child = NULL;
+ }
+ }
+}
+
+int
+sdmmc_io_rw_direct(struct sdmmc_softc *sc, struct sdmmc_function *sf,
+ int reg, u_char *datap, int arg)
{
struct sdmmc_command cmd;
+ int error;
+
+ /* Make sure the card is selected. */
+ if ((error = sdmmc_select_card(sc, sf)) != 0)
+ return error;
+
+ arg |= ((sf == NULL ? 0 : sf->number) & SD_ARG_CMD52_FUNC_MASK) <<
+ SD_ARG_CMD52_FUNC_SHIFT;
+ arg |= (reg & SD_ARG_CMD52_REG_MASK) <<
+ SD_ARG_CMD52_REG_SHIFT;
+ arg |= (*datap & SD_ARG_CMD52_DATA_MASK) <<
+ SD_ARG_CMD52_DATA_SHIFT;
bzero(&cmd, sizeof cmd);
- cmd.c_opcode = SD_IO_RESET;
- cmd.c_flags = SCF_CMD_BC | SCF_RSP_R0;
+ cmd.c_opcode = SD_IO_RW_DIRECT;
+ cmd.c_arg = arg;
+ cmd.c_flags = SCF_CMD_BC/* XXX */ | SCF_RSP_R5;
- (void)sdmmc_mmc_command(sc, &cmd);
+ error = sdmmc_mmc_command(sc, &cmd);
+ *datap = SD_R5_DATA(cmd.c_resp);
+ return error;
+}
+
+int
+sdmmc_io_rw_extended(struct sdmmc_softc *sc, struct sdmmc_function *sf,
+ int reg, u_char *datap, int datalen, int arg)
+{
+ struct sdmmc_command cmd;
+ int error;
+
+ /* Make sure the card is selected. */
+ if ((error = sdmmc_select_card(sc, sf)) != 0)
+ return error;
+
+ arg |= ((sf == NULL ? 0 : sf->number) & SD_ARG_CMD53_FUNC_MASK) <<
+ SD_ARG_CMD53_FUNC_SHIFT;
+ arg |= SD_ARG_CMD53_INCREMENT;
+ arg |= (reg & SD_ARG_CMD53_REG_MASK) <<
+ SD_ARG_CMD53_REG_SHIFT;
+ arg |= (datalen & SD_ARG_CMD53_LENGTH_MASK) <<
+ SD_ARG_CMD53_LENGTH_SHIFT;
+
+ bzero(&cmd, sizeof cmd);
+ cmd.c_opcode = SD_IO_RW_EXTENDED;
+ cmd.c_arg = arg;
+ cmd.c_flags = SCF_CMD_BC/* XXX */ | SCF_RSP_R5;
+ cmd.c_data = datap;
+ cmd.c_datalen = datalen;
+ cmd.c_blklen = datalen;
+ if (!ISSET(arg, SD_ARG_CMD53_WRITE))
+ cmd.c_flags |= SCF_CMD_READ;
+
+ return sdmmc_mmc_command(sc, &cmd);
+}
+
+u_int8_t
+sdmmc_io_read_1(struct sdmmc_function *sf, int reg)
+{
+ u_int8_t data = 0;
+
+ (void)sdmmc_io_rw_direct(sf->sc, sf, reg, (u_char *)&data,
+ SD_ARG_CMD52_READ);
+ return data;
+}
+
+void
+sdmmc_io_write_1(struct sdmmc_function *sf, int reg, u_int8_t data)
+{
+ (void)sdmmc_io_rw_direct(sf->sc, sf, reg, (u_char *)&data,
+ SD_ARG_CMD52_WRITE);
+}
+
+u_int16_t
+sdmmc_io_read_2(struct sdmmc_function *sf, int reg)
+{
+ u_int16_t data = 0;
+
+ (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 2,
+ SD_ARG_CMD53_READ);
+ return data;
+}
+
+void
+sdmmc_io_write_2(struct sdmmc_function *sf, int reg, u_int16_t data)
+{
+ (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 2,
+ SD_ARG_CMD53_WRITE);
+}
+
+u_int32_t
+sdmmc_io_read_4(struct sdmmc_function *sf, int reg)
+{
+ u_int32_t data = 0;
+
+ (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 4,
+ SD_ARG_CMD53_READ);
+ return data;
+}
+
+void
+sdmmc_io_write_4(struct sdmmc_function *sf, int reg, u_int32_t data)
+{
+ (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 4,
+ SD_ARG_CMD53_WRITE);
+}
+
+int
+sdmmc_io_write(struct sdmmc_softc *sc, struct sdmmc_function *sf,
+ int reg, u_char data)
+{
+ return sdmmc_io_rw_direct(sc, sf, reg, &data,
+ SD_ARG_CMD53_WRITE);
+}
+
+int
+sdmmc_io_xchg(struct sdmmc_softc *sc, struct sdmmc_function *sf,
+ int reg, u_char *datap)
+{
+ return sdmmc_io_rw_direct(sc, sf, reg, datap,
+ SD_ARG_CMD52_WRITE|SD_ARG_CMD52_EXCHANGE);
+}
+
+/*
+ * Reset the I/O functions of the card.
+ */
+void
+sdmmc_io_reset(struct sdmmc_softc *sc)
+{
+#if 0 /* XXX command fails */
+ (void)sdmmc_io_write(sc, NULL, SD_IO_REG_CCCR_CTL, CCCR_CTL_RES);
+ sdmmc_delay(100000);
+#endif
}
/*
@@ -113,7 +424,7 @@ sdmmc_io_send_op_cond(struct sdmmc_softc *sc, u_int32_t ocr, u_int32_t *ocrp)
/*
* If we change the OCR value, retry the command until the OCR
* we receive in response has the "CARD BUSY" bit set, meaning
- * that all cards are ready for card identification.
+ * that all cards are ready for identification.
*/
for (i = 0; i < 100; i++) {
bzero(&cmd, sizeof cmd);
@@ -130,8 +441,6 @@ sdmmc_io_send_op_cond(struct sdmmc_softc *sc, u_int32_t ocr, u_int32_t *ocrp)
error = ETIMEDOUT;
sdmmc_delay(10000);
}
- if (error == 0)
- printf("ocr: %x\n", MMC_R4(cmd.c_resp));
if (error == 0 && ocrp != NULL)
*ocrp = MMC_R4(cmd.c_resp);
return error;