123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743 |
- /* $OpenBSD: sdmmc_mem.c,v 1.21 2015/04/22 04:02:06 jsg Exp $ */
- /*
- * Copyright (c) 2006 Uwe Stuehler <uwe@openbsd.org>
- *
- * Permission to use, copy, modify, and distribute this software for any
- * purpose with or without fee is hereby granted, provided that the above
- * copyright notice and this permission notice appear in all copies.
- *
- * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
- * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
- * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
- * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
- * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
- * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
- * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- */
- /* Routines for SD/MMC memory cards. */
- #include <sys/param.h>
- #include <sys/device.h>
- #include <sys/kernel.h>
- #include <sys/malloc.h>
- #include <sys/systm.h>
- #include <dev/sdmmc/sdmmcchip.h>
- #include <dev/sdmmc/sdmmcreg.h>
- #include <dev/sdmmc/sdmmcvar.h>
- int sdmmc_decode_csd(struct sdmmc_softc *, sdmmc_response,
- struct sdmmc_function *);
- int sdmmc_decode_cid(struct sdmmc_softc *, sdmmc_response,
- struct sdmmc_function *);
- void sdmmc_print_cid(struct sdmmc_cid *);
- int sdmmc_mem_send_op_cond(struct sdmmc_softc *, u_int32_t, u_int32_t *);
- int sdmmc_mem_set_blocklen(struct sdmmc_softc *, struct sdmmc_function *);
- int sdmmc_mem_send_cxd_data(struct sdmmc_softc *, int, void *, size_t);
- int sdmmc_mem_mmc_switch(struct sdmmc_function *, uint8_t, uint8_t, uint8_t);
- int sdmmc_mem_sd_init(struct sdmmc_softc *, struct sdmmc_function *);
- int sdmmc_mem_mmc_init(struct sdmmc_softc *, struct sdmmc_function *);
- int sdmmc_mem_single_read_block(struct sdmmc_function *, int, u_char *,
- size_t);
- int sdmmc_mem_read_block_subr(struct sdmmc_function *, int, u_char *,
- size_t);
- int sdmmc_mem_single_write_block(struct sdmmc_function *, int, u_char *,
- size_t);
- int sdmmc_mem_write_block_subr(struct sdmmc_function *, int, u_char *,
- size_t);
- #ifdef SDMMC_DEBUG
- #define DPRINTF(s) printf s
- #else
- #define DPRINTF(s) /**/
- #endif
- /*
- * Initialize SD/MMC memory cards and memory in SDIO "combo" cards.
- */
- int
- sdmmc_mem_enable(struct sdmmc_softc *sc)
- {
- u_int32_t host_ocr;
- u_int32_t card_ocr;
- rw_assert_wrlock(&sc->sc_lock);
- /* Set host mode to SD "combo" card or SD memory-only. */
- SET(sc->sc_flags, SMF_SD_MODE|SMF_MEM_MODE);
- /* Reset memory (*must* do that before CMD55 or CMD1). */
- sdmmc_go_idle_state(sc);
- /*
- * Read the SD/MMC memory OCR value by issuing CMD55 followed
- * by ACMD41 to read the OCR value from memory-only SD cards.
- * MMC cards will not respond to CMD55 or ACMD41 and this is
- * how we distinguish them from SD cards.
- */
- mmc_mode:
- if (sdmmc_mem_send_op_cond(sc, 0, &card_ocr) != 0) {
- if (ISSET(sc->sc_flags, SMF_SD_MODE) &&
- !ISSET(sc->sc_flags, SMF_IO_MODE)) {
- /* Not a SD card, switch to MMC mode. */
- CLR(sc->sc_flags, SMF_SD_MODE);
- goto mmc_mode;
- }
- if (!ISSET(sc->sc_flags, SMF_SD_MODE)) {
- DPRINTF(("%s: can't read memory OCR\n",
- DEVNAME(sc)));
- return 1;
- } else {
- /* Not a "combo" card. */
- CLR(sc->sc_flags, SMF_MEM_MODE);
- return 0;
- }
- }
- /* Set the lowest voltage supported by the card and host. */
- host_ocr = sdmmc_chip_host_ocr(sc->sct, sc->sch);
- if (sdmmc_set_bus_power(sc, host_ocr, card_ocr) != 0) {
- DPRINTF(("%s: can't supply voltage requested by card\n",
- DEVNAME(sc)));
- return 1;
- }
- /* Tell the card(s) to enter the idle state (again). */
- sdmmc_go_idle_state(sc);
- host_ocr &= card_ocr; /* only allow the common voltages */
- if (sdmmc_send_if_cond(sc, card_ocr) == 0)
- host_ocr |= SD_OCR_SDHC_CAP;
- /* Send the new OCR value until all cards are ready. */
- if (sdmmc_mem_send_op_cond(sc, host_ocr, NULL) != 0) {
- DPRINTF(("%s: can't send memory OCR\n", DEVNAME(sc)));
- return 1;
- }
- return 0;
- }
- /*
- * Read the CSD and CID from all cards and assign each card a unique
- * relative card address (RCA). CMD2 is ignored by SDIO-only cards.
- */
- void
- sdmmc_mem_scan(struct sdmmc_softc *sc)
- {
- struct sdmmc_command cmd;
- struct sdmmc_function *sf;
- u_int16_t next_rca;
- int error;
- int i;
- rw_assert_wrlock(&sc->sc_lock);
- /*
- * CMD2 is a broadcast command understood by SD cards and MMC
- * cards. All cards begin to respond to the command, but back
- * off if another card drives the CMD line to a different level.
- * Only one card will get its entire response through. That
- * card remains silent once it has been assigned a RCA.
- */
- for (i = 0; i < 100; i++) {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_ALL_SEND_CID;
- cmd.c_flags = SCF_CMD_BCR | SCF_RSP_R2;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error == ETIMEDOUT) {
- /* No more cards there. */
- break;
- } else if (error != 0) {
- DPRINTF(("%s: can't read CID\n", DEVNAME(sc)));
- break;
- }
- /* In MMC mode, find the next available RCA. */
- next_rca = 1;
- if (!ISSET(sc->sc_flags, SMF_SD_MODE))
- SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list)
- next_rca++;
- /* Allocate a sdmmc_function structure. */
- sf = sdmmc_function_alloc(sc);
- sf->rca = next_rca;
- /*
- * Remember the CID returned in the CMD2 response for
- * later decoding.
- */
- bcopy(cmd.c_resp, sf->raw_cid, sizeof sf->raw_cid);
- /*
- * Silence the card by assigning it a unique RCA, or
- * querying it for its RCA in the case of SD.
- */
- if (sdmmc_set_relative_addr(sc, sf) != 0) {
- printf("%s: can't set mem RCA\n", DEVNAME(sc));
- sdmmc_function_free(sf);
- break;
- }
- #if 0
- /* Verify that the RCA has been set by selecting the card. */
- if (sdmmc_select_card(sc, sf) != 0) {
- printf("%s: can't select mem RCA %d\n",
- DEVNAME(sc), sf->rca);
- sdmmc_function_free(sf);
- break;
- }
- /* Deselect. */
- (void)sdmmc_select_card(sc, NULL);
- #endif
- /*
- * If this is a memory-only card, the card responding
- * first becomes an alias for SDIO function 0.
- */
- if (sc->sc_fn0 == NULL)
- sc->sc_fn0 = sf;
- SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list);
- }
- /*
- * All cards are either inactive or awaiting further commands.
- * Read the CSDs and decode the raw CID for each card.
- */
- SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_SEND_CSD;
- cmd.c_arg = MMC_ARG_RCA(sf->rca);
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R2;
- if (sdmmc_mmc_command(sc, &cmd) != 0) {
- SET(sf->flags, SFF_ERROR);
- continue;
- }
- if (sdmmc_decode_csd(sc, cmd.c_resp, sf) != 0 ||
- sdmmc_decode_cid(sc, sf->raw_cid, sf) != 0) {
- SET(sf->flags, SFF_ERROR);
- continue;
- }
- #ifdef SDMMC_DEBUG
- printf("%s: CID: ", DEVNAME(sc));
- sdmmc_print_cid(&sf->cid);
- #endif
- }
- }
- int
- sdmmc_decode_csd(struct sdmmc_softc *sc, sdmmc_response resp,
- struct sdmmc_function *sf)
- {
- struct sdmmc_csd *csd = &sf->csd;
- if (ISSET(sc->sc_flags, SMF_SD_MODE)) {
- /*
- * CSD version 1.0 corresponds to SD system
- * specification version 1.0 - 1.10. (SanDisk, 3.5.3)
- */
- csd->csdver = SD_CSD_CSDVER(resp);
- switch (csd->csdver) {
- case SD_CSD_CSDVER_2_0:
- sf->flags |= SFF_SDHC;
- csd->capacity = SD_CSD_V2_CAPACITY(resp);
- csd->read_bl_len = SD_CSD_V2_BL_LEN;
- break;
- case SD_CSD_CSDVER_1_0:
- csd->capacity = SD_CSD_CAPACITY(resp);
- csd->read_bl_len = SD_CSD_READ_BL_LEN(resp);
- break;
- default:
- printf("%s: unknown SD CSD structure version 0x%x\n",
- DEVNAME(sc), csd->csdver);
- return 1;
- break;
- }
- } else {
- csd->csdver = MMC_CSD_CSDVER(resp);
- if (csd->csdver == MMC_CSD_CSDVER_1_0 ||
- csd->csdver == MMC_CSD_CSDVER_2_0 ||
- csd->csdver == MMC_CSD_CSDVER_EXT_CSD) {
- csd->mmcver = MMC_CSD_MMCVER(resp);
- csd->capacity = MMC_CSD_CAPACITY(resp);
- csd->read_bl_len = MMC_CSD_READ_BL_LEN(resp);
- } else {
- printf("%s: unknown MMC CSD structure version 0x%x\n",
- DEVNAME(sc), csd->csdver);
- return 1;
- }
- }
- csd->sector_size = MIN(1 << csd->read_bl_len,
- sdmmc_chip_host_maxblklen(sc->sct, sc->sch));
- if (csd->sector_size < (1<<csd->read_bl_len))
- csd->capacity *= (1<<csd->read_bl_len) /
- csd->sector_size;
- return 0;
- }
- int
- sdmmc_decode_cid(struct sdmmc_softc *sc, sdmmc_response resp,
- struct sdmmc_function *sf)
- {
- struct sdmmc_cid *cid = &sf->cid;
- if (ISSET(sc->sc_flags, SMF_SD_MODE)) {
- cid->mid = SD_CID_MID(resp);
- cid->oid = SD_CID_OID(resp);
- SD_CID_PNM_CPY(resp, cid->pnm);
- cid->rev = SD_CID_REV(resp);
- cid->psn = SD_CID_PSN(resp);
- cid->mdt = SD_CID_MDT(resp);
- } else {
- switch(sf->csd.mmcver) {
- case MMC_CSD_MMCVER_1_0:
- case MMC_CSD_MMCVER_1_4:
- cid->mid = MMC_CID_MID_V1(resp);
- MMC_CID_PNM_V1_CPY(resp, cid->pnm);
- cid->rev = MMC_CID_REV_V1(resp);
- cid->psn = MMC_CID_PSN_V1(resp);
- cid->mdt = MMC_CID_MDT_V1(resp);
- break;
- case MMC_CSD_MMCVER_2_0:
- case MMC_CSD_MMCVER_3_1:
- case MMC_CSD_MMCVER_4_0:
- cid->mid = MMC_CID_MID_V2(resp);
- cid->oid = MMC_CID_OID_V2(resp);
- MMC_CID_PNM_V2_CPY(resp, cid->pnm);
- cid->psn = MMC_CID_PSN_V2(resp);
- break;
- default:
- printf("%s: unknown MMC version %d\n",
- DEVNAME(sc), sf->csd.mmcver);
- return 1;
- }
- }
- return 0;
- }
- #ifdef SDMMC_DEBUG
- void
- sdmmc_print_cid(struct sdmmc_cid *cid)
- {
- printf("mid=0x%02x oid=0x%04x pnm=\"%s\" rev=0x%02x psn=0x%08x"
- " mdt=%03x\n", cid->mid, cid->oid, cid->pnm, cid->rev, cid->psn,
- cid->mdt);
- }
- #endif
- int
- sdmmc_mem_send_cxd_data(struct sdmmc_softc *sc, int opcode, void *data,
- size_t datalen)
- {
- struct sdmmc_command cmd;
- void *ptr = NULL;
- int error = 0;
- ptr = malloc(datalen, M_DEVBUF, M_NOWAIT | M_ZERO);
- if (ptr == NULL) {
- error = ENOMEM;
- goto out;
- }
- memset(&cmd, 0, sizeof(cmd));
- cmd.c_data = ptr;
- cmd.c_datalen = datalen;
- cmd.c_blklen = datalen;
- cmd.c_opcode = opcode;
- cmd.c_arg = 0;
- cmd.c_flags = SCF_CMD_ADTC | SCF_CMD_READ;
- if (opcode == MMC_SEND_EXT_CSD)
- SET(cmd.c_flags, SCF_RSP_R1);
- else
- SET(cmd.c_flags, SCF_RSP_R2);
- error = sdmmc_mmc_command(sc, &cmd);
- if (error == 0)
- memcpy(data, ptr, datalen);
- out:
- if (ptr != NULL)
- free(ptr, M_DEVBUF, 0);
- return error;
- }
- int
- sdmmc_mem_mmc_switch(struct sdmmc_function *sf, uint8_t set, uint8_t index,
- uint8_t value)
- {
- struct sdmmc_softc *sc = sf->sc;
- struct sdmmc_command cmd;
- memset(&cmd, 0, sizeof(cmd));
- cmd.c_opcode = MMC_SWITCH;
- cmd.c_arg = (MMC_SWITCH_MODE_WRITE_BYTE << 24) |
- (index << 16) | (value << 8) | set;
- cmd.c_flags = SCF_RSP_R1B | SCF_CMD_AC;
- return sdmmc_mmc_command(sc, &cmd);
- }
- /*
- * Initialize a SD/MMC memory card.
- */
- int
- sdmmc_mem_init(struct sdmmc_softc *sc, struct sdmmc_function *sf)
- {
- int error = 0;
- rw_assert_wrlock(&sc->sc_lock);
- if (sdmmc_select_card(sc, sf) != 0 ||
- sdmmc_mem_set_blocklen(sc, sf) != 0)
- error = 1;
- if (ISSET(sc->sc_flags, SMF_SD_MODE))
- error = sdmmc_mem_sd_init(sc, sf);
- else
- error = sdmmc_mem_mmc_init(sc, sf);
- return error;
- }
- int
- sdmmc_mem_sd_init(struct sdmmc_softc *sc, struct sdmmc_function *sf)
- {
- /* XXX */
- return 0;
- }
- int
- sdmmc_mem_mmc_init(struct sdmmc_softc *sc, struct sdmmc_function *sf)
- {
- int error = 0;
- u_int8_t ext_csd[512];
- int speed = 0;
- int hs_timing = 0;
- u_int32_t sectors = 0;
- if (sf->csd.mmcver >= MMC_CSD_MMCVER_4_0) {
- /* read EXT_CSD */
- error = sdmmc_mem_send_cxd_data(sc,
- MMC_SEND_EXT_CSD, ext_csd, sizeof(ext_csd));
- if (error != 0) {
- SET(sf->flags, SFF_ERROR);
- printf("%s: can't read EXT_CSD\n", DEVNAME(sc));
- return error;
- }
- if (ext_csd[EXT_CSD_CARD_TYPE] & EXT_CSD_CARD_TYPE_F_52M) {
- speed = 52000;
- hs_timing = 1;
- } else if (ext_csd[EXT_CSD_CARD_TYPE] & EXT_CSD_CARD_TYPE_F_26M) {
- speed = 26000;
- } else {
- printf("%s: unknown CARD_TYPE 0x%x\n", DEVNAME(sc),
- ext_csd[EXT_CSD_CARD_TYPE]);
- }
- if (!ISSET(sc->sc_caps, SMC_CAPS_MMC_HIGHSPEED))
- hs_timing = 0;
- if (hs_timing) {
- /* switch to high speed timing */
- error = sdmmc_mem_mmc_switch(sf, EXT_CSD_CMD_SET_NORMAL,
- EXT_CSD_HS_TIMING, hs_timing);
- if (error != 0) {
- printf("%s: can't change high speed\n",
- DEVNAME(sc));
- return error;
- }
- }
- error =
- sdmmc_chip_bus_clock(sc->sct, sc->sch, speed);
- if (error != 0) {
- printf("%s: can't change bus clock\n", DEVNAME(sc));
- return error;
- }
- if (hs_timing) {
- /* read EXT_CSD again */
- error = sdmmc_mem_send_cxd_data(sc,
- MMC_SEND_EXT_CSD, ext_csd, sizeof(ext_csd));
- if (error != 0) {
- printf("%s: can't re-read EXT_CSD\n", DEVNAME(sc));
- return error;
- }
- if (ext_csd[EXT_CSD_HS_TIMING] != 1) {
- printf("%s, HS_TIMING set failed\n", DEVNAME(sc));
- return EINVAL;
- }
- }
- sectors = ext_csd[EXT_CSD_SEC_COUNT + 0] << 0 |
- ext_csd[EXT_CSD_SEC_COUNT + 1] << 8 |
- ext_csd[EXT_CSD_SEC_COUNT + 2] << 16 |
- ext_csd[EXT_CSD_SEC_COUNT + 3] << 24;
- if (sectors > (2u * 1024 * 1024 * 1024) / 512) {
- sf->flags |= SFF_SDHC;
- sf->csd.capacity = sectors;
- }
- }
- return error;
- }
- /*
- * Get or set the card's memory OCR value (SD or MMC).
- */
- int
- sdmmc_mem_send_op_cond(struct sdmmc_softc *sc, u_int32_t ocr,
- u_int32_t *ocrp)
- {
- struct sdmmc_command cmd;
- int error;
- int i;
- rw_assert_wrlock(&sc->sc_lock);
- /*
- * 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 identification.
- */
- for (i = 0; i < 100; i++) {
- bzero(&cmd, sizeof cmd);
- cmd.c_arg = ocr;
- cmd.c_flags = SCF_CMD_BCR | SCF_RSP_R3;
- if (ISSET(sc->sc_flags, SMF_SD_MODE)) {
- cmd.c_opcode = SD_APP_OP_COND;
- error = sdmmc_app_command(sc, &cmd);
- } else {
- cmd.c_opcode = MMC_SEND_OP_COND;
- error = sdmmc_mmc_command(sc, &cmd);
- }
- if (error != 0)
- break;
- if (ISSET(MMC_R3(cmd.c_resp), MMC_OCR_MEM_READY) ||
- ocr == 0)
- break;
- error = ETIMEDOUT;
- sdmmc_delay(10000);
- }
- if (error == 0 && ocrp != NULL)
- *ocrp = MMC_R3(cmd.c_resp);
- return error;
- }
- /*
- * Set the read block length appropriately for this card, according to
- * the card CSD register value.
- */
- int
- sdmmc_mem_set_blocklen(struct sdmmc_softc *sc, struct sdmmc_function *sf)
- {
- struct sdmmc_command cmd;
- rw_assert_wrlock(&sc->sc_lock);
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_SET_BLOCKLEN;
- cmd.c_arg = sf->csd.sector_size;
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1;
- DPRINTF(("%s: read_bl_len=%d sector_size=%d\n", DEVNAME(sc),
- 1 << sf->csd.read_bl_len, sf->csd.sector_size));
- return sdmmc_mmc_command(sc, &cmd);
- }
- int
- sdmmc_mem_read_block_subr(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- struct sdmmc_softc *sc = sf->sc;
- struct sdmmc_command cmd;
- int error;
- if ((error = sdmmc_select_card(sc, sf)) != 0)
- goto err;
- bzero(&cmd, sizeof cmd);
- cmd.c_data = data;
- cmd.c_datalen = datalen;
- cmd.c_blklen = sf->csd.sector_size;
- cmd.c_opcode = (datalen / cmd.c_blklen) > 1 ?
- MMC_READ_BLOCK_MULTIPLE : MMC_READ_BLOCK_SINGLE;
- if (sf->flags & SFF_SDHC)
- cmd.c_arg = blkno;
- else
- cmd.c_arg = blkno << 9;
- cmd.c_flags = SCF_CMD_ADTC | SCF_CMD_READ | SCF_RSP_R1;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- goto err;
- if (ISSET(sc->sc_flags, SMF_STOP_AFTER_MULTIPLE) &&
- cmd.c_opcode == MMC_READ_BLOCK_MULTIPLE) {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_STOP_TRANSMISSION;
- cmd.c_arg = MMC_ARG_RCA(sf->rca);
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1B;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- goto err;
- }
- do {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_SEND_STATUS;
- cmd.c_arg = MMC_ARG_RCA(sf->rca);
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- break;
- /* XXX time out */
- } while (!ISSET(MMC_R1(cmd.c_resp), MMC_R1_READY_FOR_DATA));
- err:
- return (error);
- }
- int
- sdmmc_mem_single_read_block(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- int error = 0;
- int i;
- for (i = 0; i < datalen / sf->csd.sector_size; i++) {
- error = sdmmc_mem_read_block_subr(sf, blkno + i, data + i *
- sf->csd.sector_size, sf->csd.sector_size);
- if (error)
- break;
- }
- return (error);
- }
- int
- sdmmc_mem_read_block(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- struct sdmmc_softc *sc = sf->sc;
- int error;
- rw_enter_write(&sc->sc_lock);
- if (ISSET(sc->sc_caps, SMC_CAPS_SINGLE_ONLY)) {
- error = sdmmc_mem_single_read_block(sf, blkno, data, datalen);
- } else {
- error = sdmmc_mem_read_block_subr(sf, blkno, data, datalen);
- }
- rw_exit(&sc->sc_lock);
- return (error);
- }
- int
- sdmmc_mem_write_block_subr(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- struct sdmmc_softc *sc = sf->sc;
- struct sdmmc_command cmd;
- int error;
- if ((error = sdmmc_select_card(sc, sf)) != 0)
- goto err;
- bzero(&cmd, sizeof cmd);
- cmd.c_data = data;
- cmd.c_datalen = datalen;
- cmd.c_blklen = sf->csd.sector_size;
- cmd.c_opcode = (datalen / cmd.c_blklen) > 1 ?
- MMC_WRITE_BLOCK_MULTIPLE : MMC_WRITE_BLOCK_SINGLE;
- if (sf->flags & SFF_SDHC)
- cmd.c_arg = blkno;
- else
- cmd.c_arg = blkno << 9;
- cmd.c_flags = SCF_CMD_ADTC | SCF_RSP_R1;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- goto err;
- if (ISSET(sc->sc_flags, SMF_STOP_AFTER_MULTIPLE) &&
- cmd.c_opcode == MMC_WRITE_BLOCK_MULTIPLE) {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_STOP_TRANSMISSION;
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1B;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- goto err;
- }
- do {
- bzero(&cmd, sizeof cmd);
- cmd.c_opcode = MMC_SEND_STATUS;
- cmd.c_arg = MMC_ARG_RCA(sf->rca);
- cmd.c_flags = SCF_CMD_AC | SCF_RSP_R1;
- error = sdmmc_mmc_command(sc, &cmd);
- if (error != 0)
- break;
- /* XXX time out */
- } while (!ISSET(MMC_R1(cmd.c_resp), MMC_R1_READY_FOR_DATA));
- err:
- return (error);
- }
- int
- sdmmc_mem_single_write_block(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- int error = 0;
- int i;
- for (i = 0; i < datalen / sf->csd.sector_size; i++) {
- error = sdmmc_mem_write_block_subr(sf, blkno + i, data + i *
- sf->csd.sector_size, sf->csd.sector_size);
- if (error)
- break;
- }
- return (error);
- }
- int
- sdmmc_mem_write_block(struct sdmmc_function *sf, int blkno, u_char *data,
- size_t datalen)
- {
- struct sdmmc_softc *sc = sf->sc;
- int error;
- rw_enter_write(&sc->sc_lock);
- if (ISSET(sc->sc_caps, SMC_CAPS_SINGLE_ONLY)) {
- error = sdmmc_mem_single_write_block(sf, blkno, data, datalen);
- } else {
- error = sdmmc_mem_write_block_subr(sf, blkno, data, datalen);
- }
- rw_exit(&sc->sc_lock);
- return (error);
- }
|