diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 3127012022a2..1e6eb0b50d53 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -84,15 +84,19 @@ static void sdmmc_dump_command(struct sdmmc_softc *, struct sdmmc_command *); static int sdmmc_match(device_t, cfdata_t, void *); static void sdmmc_attach(device_t, device_t, void *); static int sdmmc_detach(device_t, int); +static int sdmmc_rescan(device_t, const char *, const int *); +static void sdmmc_childdet(device_t, device_t); -CFATTACH_DECL_NEW(sdmmc, sizeof(struct sdmmc_softc), - sdmmc_match, sdmmc_attach, sdmmc_detach, NULL); +CFATTACH_DECL2_NEW(sdmmc, sizeof(struct sdmmc_softc), + sdmmc_match, sdmmc_attach, sdmmc_detach, /*activate*/NULL, sdmmc_rescan, + sdmmc_childdet); static void sdmmc_doattach(device_t); static void sdmmc_task_thread(void *); -static void sdmmc_discover_task(void *); +static void sdmmc_discover_thread(void *); static void sdmmc_polling_card(void *); static void sdmmc_card_attach(struct sdmmc_softc *); +static void sdmmc_do_rescan(struct sdmmc_softc *); static void sdmmc_card_detach(struct sdmmc_softc *, int); static int sdmmc_print(void *, const char *); static int sdmmc_enable(struct sdmmc_softc *); @@ -146,12 +150,12 @@ sdmmc_attach(device_t parent, device_t self, void *aux) TAILQ_INIT(&sc->sc_tskq); TAILQ_INIT(&sc->sc_intrq); - sdmmc_init_task(&sc->sc_discover_task, sdmmc_discover_task, sc); sdmmc_init_task(&sc->sc_intr_task, sdmmc_intr_task, sc); mutex_init(&sc->sc_mtx, MUTEX_DEFAULT, IPL_NONE); mutex_init(&sc->sc_tskq_mtx, MUTEX_DEFAULT, IPL_SDMMC); - mutex_init(&sc->sc_discover_task_mtx, MUTEX_DEFAULT, IPL_SDMMC); + mutex_init(&sc->sc_attach_mtx, MUTEX_DEFAULT, IPL_NONE); + mutex_init(&sc->sc_intr_mtx, MUTEX_DEFAULT, IPL_SDMMC); cv_init(&sc->sc_tskq_cv, "mmctaskq"); evcnt_attach_dynamic(&sc->sc_ev_xfer, EVCNT_TYPE_MISC, NULL, @@ -177,6 +181,10 @@ sdmmc_attach(device_t parent, device_t self, void *aux) evcnt_attach_dynamic(&sc->sc_ev_xfer_error, EVCNT_TYPE_MISC, &sc->sc_ev_xfer, device_xname(self), "xfer error"); + mutex_enter(&sc->sc_intr_mtx); + sc->sc_intr_enabled = true; + mutex_exit(&sc->sc_intr_mtx); + if (ISSET(sc->sc_caps, SMC_CAPS_POLL_CARD_DET)) { callout_init(&sc->sc_card_detect_ch, 0); callout_reset(&sc->sc_card_detect_ch, hz, @@ -203,34 +211,74 @@ sdmmc_detach(device_t self, int flags) struct sdmmc_softc *sc = device_private(self); int error, i; + /* + * Suspend any concurrent attach/detach via card discovery, and + * try to detach children. + * + * If we succeed at detaching children, prevent further + * discovery from happening -- when sc->sc_detaching is set, + * discovery will treat the card as having been yanked. + */ + mutex_enter(&sc->sc_attach_mtx); + KASSERTMSG(!sc->sc_detaching, "%s", SDMMCDEVNAME(sc)); + error = config_detach_children(self, flags); + if (error) { + mutex_exit(&sc->sc_attach_mtx); + return error; + } + sc->sc_detaching = true; + mutex_exit(&sc->sc_attach_mtx); + + /* + * Prevent sdmmc_card_intr from scheduling new interrupt tasks. + * After this point, there may be interrupt tasks scheduled, + * but no new ones can be scheduled. + * + * XXX Might be worthwhile to use pserialize here rather than a + * mutex to reduce overhead in the interrupt handler. + */ + mutex_enter(&sc->sc_intr_mtx); + sc->sc_intr_enabled = false; + mutex_exit(&sc->sc_intr_mtx); + + /* + * Prevent periodic polling to schedule discovery. After this + * point, there may be discovery still scheduled, but it can't + * be newly scheduled again. + */ + if (ISSET(sc->sc_caps, SMC_CAPS_POLL_CARD_DET)) { + callout_halt(&sc->sc_card_detect_ch, NULL); + callout_destroy(&sc->sc_card_detect_ch); + } + + /* + * Wake the discovery thread and wait for it to to finish. + */ + sdmmc_needs_discover(self); + (void)kthread_join(sc->sc_discover_lwp); + + /* + * Wait for queued tasks to complete and treat any remaining + * card as forcibly detached. + */ mutex_enter(&sc->sc_tskq_mtx); - sc->sc_dying = 1; + sc->sc_tskq_dying = 1; cv_signal(&sc->sc_tskq_cv); - while (sc->sc_tskq_lwp != NULL) - cv_wait(&sc->sc_tskq_cv, &sc->sc_tskq_mtx); mutex_exit(&sc->sc_tskq_mtx); + (void)kthread_join(sc->sc_tskq_lwp); pmf_device_deregister(self); - error = config_detach_children(self, flags); - if (error) - return error; - if (ISSET(sc->sc_caps, SMC_CAPS_DMA)) { bus_dmamap_unload(sc->sc_dmat, sc->sc_dmap); bus_dmamap_destroy(sc->sc_dmat, sc->sc_dmap); } - if (ISSET(sc->sc_caps, SMC_CAPS_POLL_CARD_DET)) { - callout_halt(&sc->sc_card_detect_ch, NULL); - callout_destroy(&sc->sc_card_detect_ch); - } - sdmmc_del_task(sc, &sc->sc_intr_task, NULL); - sdmmc_del_task(sc, &sc->sc_discover_task, NULL); cv_destroy(&sc->sc_tskq_cv); - mutex_destroy(&sc->sc_discover_task_mtx); + mutex_destroy(&sc->sc_intr_mtx); + mutex_destroy(&sc->sc_attach_mtx); mutex_destroy(&sc->sc_tskq_mtx); mutex_destroy(&sc->sc_mtx); @@ -243,13 +291,54 @@ sdmmc_detach(device_t self, int flags) return 0; } +static int +sdmmc_rescan(device_t self, const char *iattr, const int *locators) +{ + struct sdmmc_softc *sc = device_private(self); + + mutex_enter(&sc->sc_attach_mtx); + mutex_enter(&sc->sc_mtx); + sdmmc_do_rescan(sc); + mutex_exit(&sc->sc_mtx); + mutex_exit(&sc->sc_attach_mtx); + + return 0; +} + +static void +sdmmc_childdet(device_t dev, device_t child) +{ + struct sdmmc_softc *sc = device_private(dev); + struct sdmmc_function *sf; + + mutex_enter(&sc->sc_mtx); + KASSERTMSG(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED), + "%s is not attached but a child detached: %s", + device_xname(dev), device_xname(child)); + SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { + if (sf->child == child) { + sf->child = NULL; + break; + } + } + KASSERTMSG(sf != NULL, "no such child of %s: %s", + device_xname(dev), device_xname(child)); + mutex_exit(&sc->sc_mtx); +} + static void sdmmc_doattach(device_t dev) { struct sdmmc_softc *sc = device_private(dev); if (kthread_create(PRI_SOFTBIO, 0, NULL, - sdmmc_task_thread, sc, &sc->sc_tskq_lwp, "%s", device_xname(dev))) { + sdmmc_discover_thread, sc, &sc->sc_discover_lwp, "%s discover", + device_xname(dev))) { + aprint_error_dev(dev, "couldn't create discovery thread\n"); + } + if (kthread_create(PRI_SOFTBIO, 0, NULL, + sdmmc_task_thread, sc, &sc->sc_tskq_lwp, "%s task", + device_xname(dev))) { aprint_error_dev(dev, "couldn't create task thread\n"); } } @@ -327,9 +416,6 @@ sdmmc_task_thread(void *arg) struct sdmmc_softc *sc = (struct sdmmc_softc *)arg; struct sdmmc_task *task; - sdmmc_discover_task(sc); - config_pending_decr(sc->sc_dev); - mutex_enter(&sc->sc_tskq_mtx); for (;;) { task = TAILQ_FIRST(&sc->sc_tskq); @@ -343,24 +429,11 @@ sdmmc_task_thread(void *arg) cv_broadcast(&sc->sc_tskq_cv); } else { /* Check for the exit condition. */ - if (sc->sc_dying) + if (sc->sc_tskq_dying) break; cv_wait(&sc->sc_tskq_cv, &sc->sc_tskq_mtx); } } - /* time to die. */ - sc->sc_dying = 0; - if (ISSET(sc->sc_flags, SMF_CARD_PRESENT)) { - /* - * sdmmc_card_detach() may issue commands, - * so temporarily drop the interrupt-blocking lock. - */ - mutex_exit(&sc->sc_tskq_mtx); - sdmmc_card_detach(sc, DETACH_FORCE); - mutex_enter(&sc->sc_tskq_mtx); - } - sc->sc_tskq_lwp = NULL; - cv_broadcast(&sc->sc_tskq_cv); mutex_exit(&sc->sc_tskq_mtx); kthread_exit(0); } @@ -370,55 +443,83 @@ sdmmc_needs_discover(device_t dev) { struct sdmmc_softc *sc = device_private(dev); - if (!ISSET(sc->sc_flags, SMF_INITED)) - return; - - sdmmc_add_task(sc, &sc->sc_discover_task); + mutex_enter(&sc->sc_intr_mtx); + sc->sc_discover_needed = true; + cv_broadcast(&sc->sc_discover_cv); + mutex_exit(&sc->sc_intr_mtx); } -static void -sdmmc_discover_task(void *arg) +static bool +sdmmc_discover(struct sdmmc_softc *sc) { - struct sdmmc_softc *sc = (struct sdmmc_softc *)arg; int card_detect, card_present; + bool ok = true; + + mutex_enter(&sc->sc_attach_mtx); - mutex_enter(&sc->sc_discover_task_mtx); - card_detect = sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); + ok = !sc->sc_detaching; + card_detect = sc->sc_detaching ? 0 : + sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); + + mutex_enter(&sc->sc_mtx); card_present = ISSET(sc->sc_flags, SMF_CARD_PRESENT); if (card_detect) SET(sc->sc_flags, SMF_CARD_PRESENT); else CLR(sc->sc_flags, SMF_CARD_PRESENT); - mutex_exit(&sc->sc_discover_task_mtx); if (card_detect) { if (!card_present) { sdmmc_card_attach(sc); - mutex_enter(&sc->sc_discover_task_mtx); if (!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)) CLR(sc->sc_flags, SMF_CARD_PRESENT); - mutex_exit(&sc->sc_discover_task_mtx); } } else { - if (card_present) + if (card_present) { sdmmc_card_detach(sc, DETACH_FORCE); + } + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); } + mutex_exit(&sc->sc_mtx); + + mutex_exit(&sc->sc_attach_mtx); + + return ok; } static void -sdmmc_polling_card(void *arg) +sdmmc_discover_thread(void *arg) { - struct sdmmc_softc *sc = (struct sdmmc_softc *)arg; - int card_detect, card_present; + struct sdmmc_softc *sc = arg; - mutex_enter(&sc->sc_discover_task_mtx); - card_detect = sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); - card_present = ISSET(sc->sc_flags, SMF_CARD_PRESENT); - mutex_exit(&sc->sc_discover_task_mtx); + /* + * Always make one discovery attempt immediately on boot, to + * detect disks, before we let autoconf proceed to mountroot. + */ + (void)sdmmc_discover(sc); + config_pending_decr(sc->sc_dev); - if (card_detect != card_present) - sdmmc_needs_discover(sc->sc_dev); + /* + * Wait until we're notified discovery is needed, and do a + * round of discovery. Repeat until sdmmc_detach tells us + * we're done; then exit the kthread. + */ + do { + mutex_enter(&sc->sc_intr_mtx); + while (!sc->sc_discover_needed) + cv_wait(&sc->sc_discover_cv, &sc->sc_intr_mtx); + sc->sc_discover_needed = false; + mutex_exit(&sc->sc_intr_mtx); + } while (sdmmc_discover(sc)); + kthread_exit(0); +} + +static void +sdmmc_polling_card(void *arg) +{ + struct sdmmc_softc *sc = arg; + sdmmc_needs_discover(sc->sc_dev); callout_schedule(&sc->sc_card_detect_ch, hz); } @@ -428,12 +529,15 @@ sdmmc_polling_card(void *arg) static void sdmmc_card_attach(struct sdmmc_softc *sc) { - struct sdmmc_function *sf; - struct sdmmc_attach_args saa; int error; DPRINTF(1,("%s: attach card\n", DEVNAME(sc))); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + KASSERT(ISSET(sc->sc_flags, SMF_CARD_PRESENT)); + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); + CLR(sc->sc_flags, SMF_CARD_ATTACHED); sdmmc_chip_hw_reset(sc->sc_sct, sc->sc_sch); @@ -468,7 +572,41 @@ sdmmc_card_attach(struct sdmmc_softc *sc) goto err; } + /* + * Card is attached. We can now attach child drivers. + */ + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); + SET(sc->sc_flags, SMF_CARD_ATTACHED); + sdmmc_do_rescan(sc); + + KASSERT(ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); + return; + +err: + sdmmc_card_detach(sc, DETACH_FORCE); + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); +} + +/* + * sdmmc_do_rescan(sc) + * + * Card is attached. Rescan for child drivers to attach. + */ +static void +sdmmc_do_rescan(struct sdmmc_softc *sc) +{ + struct sdmmc_function *sf; + + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + KASSERT(ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); + SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { + struct sdmmc_attach_args saa; + device_t child; + + if (sf->child != NULL) /* child already attached? */ + continue; if (ISSET(sc->sc_flags, SMF_IO_MODE) && sf->number < 1) continue; @@ -478,15 +616,23 @@ sdmmc_card_attach(struct sdmmc_softc *sc) saa.interface = sf->interface; saa.sf = sf; - sf->child = - config_found(sc->sc_dev, &saa, sdmmc_print, CFARGS_NONE); + /* + * Search for a child to attach, with sc_mtx unlocked + * so that it can do sdmmc operations. + */ + mutex_exit(&sc->sc_mtx); + child = config_found_acquire(sc->sc_dev, &saa, sdmmc_print, + CFARGS_NONE); + mutex_enter(&sc->sc_mtx); + + KASSERTMSG(sf->child == NULL, + "%s interface %d already has child: %p", + SDMMCDEVNAME(sc), sf->interface, sf->child); + sf->child = child; + + device_release(child); + child = NULL; /* paranoia */ } - - SET(sc->sc_flags, SMF_CARD_ATTACHED); - return; - -err: - sdmmc_card_detach(sc, DETACH_FORCE); } /* @@ -497,19 +643,39 @@ static void sdmmc_card_detach(struct sdmmc_softc *sc, int flags) { struct sdmmc_function *sf, *sfnext; + int error __diagused; DPRINTF(1,("%s: detach card\n", DEVNAME(sc))); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + + /* + * If the card had finished attaching, forcibly detach any + * children whose software drivers might have attached. Make + * sure to release sc_mtx while we detach children so that the + * child drivers' detach routines can try sdmmc operations. + * + * We assume all children are attached via sf->child for some + * sf on the function list. If sdmmc were ever to sprout other + * children not corresponding to functions of an attached card, + * we would need to go back to forcibly detaching just each + * individual sf->child. + */ if (ISSET(sc->sc_flags, SMF_CARD_ATTACHED)) { - SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { - if (sf->child != NULL) { - config_detach(sf->child, DETACH_FORCE); - sf->child = NULL; - } - } + mutex_exit(&sc->sc_mtx); + error = config_detach_children(sc->sc_dev, DETACH_FORCE); + mutex_enter(&sc->sc_mtx); + KASSERTMSG(error, "%s failed to forcibly detach children: %d", + SDMMCDEVNAME(sc), error); KASSERT(TAILQ_EMPTY(&sc->sc_intrq)); + /* + * All child drivers are detached and no more can be + * attached as long as we hold sc_attach_mtx. We can + * now start detaching the card itself. + */ CLR(sc->sc_flags, SMF_CARD_ATTACHED); } @@ -517,8 +683,7 @@ sdmmc_card_detach(struct sdmmc_softc *sc, int flags) sdmmc_disable(sc); /* Free all sdmmc_function structures. */ - for (sf = SIMPLEQ_FIRST(&sc->sf_head); sf != NULL; sf = sfnext) { - sfnext = SIMPLEQ_NEXT(sf, sf_list); + SIMPLEQ_FOREACH_SAFE(sf, &sc->sf_head, sf_list, sfnext) { sdmmc_function_free(sf); } SIMPLEQ_INIT(&sc->sf_head); @@ -581,6 +746,9 @@ sdmmc_enable(struct sdmmc_softc *sc) { int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + /* * Calculate the equivalent of the card OCR from the host * capabilities and select the maximum supported bus voltage. @@ -633,6 +801,10 @@ out: static void sdmmc_disable(struct sdmmc_softc *sc) { + + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + /* XXX complete commands if card is still present. */ if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { @@ -757,6 +929,29 @@ sdmmc_function_free(struct sdmmc_function *sf) free(sf, M_DEVBUF); } +/* + * sdmmc_function_add(sc, sf) + * + * Publish a function sf, previously allocated with + * sdmmc_function_alloc, in the function list of the sdmmc sc. + * + * Caller must hold sc_attach_mtx -- this is allowed only while + * detecting a newly attached card; once this is done, the + * function list is stable until the card is detached. + */ +void +sdmmc_function_add(struct sdmmc_softc *sc, struct sdmmc_function *sf) +{ + + KASSERT(sc == sf->sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + KASSERT(ISSET(sc->sc_flags, SMF_CARD_PRESENT)); + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); + + SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list); +} + /* * Scan for I/O functions and memory cards on the bus, allocating a * sdmmc_function structure for each. @@ -765,6 +960,9 @@ static int sdmmc_scan(struct sdmmc_softc *sc) { + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { /* Scan for I/O functions. */ if (ISSET(sc->sc_flags, SMF_IO_MODE)) @@ -792,6 +990,9 @@ sdmmc_init(struct sdmmc_softc *sc) { struct sdmmc_function *sf; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + /* Initialize all identified card functions. */ SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { @@ -844,7 +1045,7 @@ sdmmc_app_command(struct sdmmc_softc *sc, struct sdmmc_function *sf, struct sdmm DPRINTF(1,("sdmmc_app_command: start\n")); - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); memset(&acmd, 0, sizeof(acmd)); acmd.c_opcode = MMC_APP_CMD; @@ -878,7 +1079,7 @@ sdmmc_mmc_command(struct sdmmc_softc *sc, struct sdmmc_command *cmd) DPRINTF(1,("sdmmc_mmc_command: cmd=%d, arg=%#x, flags=%#x\n", cmd->c_opcode, cmd->c_arg, cmd->c_flags)); - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); #if defined(DIAGNOSTIC) || defined(SDMMC_DEBUG) if (cmd->c_data && !ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { @@ -916,7 +1117,7 @@ sdmmc_stop_transmission(struct sdmmc_softc *sc) DPRINTF(1,("sdmmc_stop_transmission\n")); - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); memset(&cmd, 0, sizeof(cmd)); cmd.c_opcode = MMC_STOP_TRANSMISSION; @@ -935,7 +1136,7 @@ sdmmc_go_idle_state(struct sdmmc_softc *sc) DPRINTF(1,("sdmmc_go_idle_state\n")); - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); memset(&cmd, 0, sizeof(cmd)); cmd.c_opcode = MMC_GO_IDLE_STATE; @@ -953,7 +1154,7 @@ sdmmc_set_relative_addr(struct sdmmc_softc *sc, struct sdmmc_function *sf) struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); if (ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { device_printf(sc->sc_dev, @@ -986,7 +1187,7 @@ sdmmc_select_card(struct sdmmc_softc *sc, struct sdmmc_function *sf) struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); if (ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { device_printf(sc->sc_dev, diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c index 23cdd8aec984..25922e47470a 100644 --- a/sys/dev/sdmmc/sdmmc_io.c +++ b/sys/dev/sdmmc/sdmmc_io.c @@ -51,6 +51,8 @@ struct sdmmc_intr_handler { TAILQ_ENTRY(sdmmc_intr_handler) entry; }; +static uint8_t sdmmc_io_read_1_locked(struct sdmmc_function *, int); +static void sdmmc_io_write_1_locked(struct sdmmc_function *, int, uint8_t); static int sdmmc_io_rw_direct(struct sdmmc_softc *, struct sdmmc_function *, int, u_char *, int, bool); static int sdmmc_io_rw_extended(struct sdmmc_softc *, @@ -75,7 +77,8 @@ sdmmc_io_enable(struct sdmmc_softc *sc) uint32_t card_ocr; int error; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); /* Set host mode to SD "combo" card. */ SET(sc->sc_flags, SMF_SD_MODE|SMF_IO_MODE|SMF_MEM_MODE); @@ -130,8 +133,6 @@ sdmmc_io_enable(struct sdmmc_softc *sc) } out: - SDMMC_UNLOCK(sc); - return error; } @@ -146,18 +147,19 @@ sdmmc_io_scan(struct sdmmc_softc *sc) int error; int i; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); sf0 = sdmmc_function_alloc(sc); sf0->number = 0; error = sdmmc_set_relative_addr(sc, sf0); if (error) { aprint_error_dev(sc->sc_dev, "couldn't set I/O RCA\n"); - SET(sf0->flags, SFF_ERROR); + sdmmc_function_free(sf0); goto out; } sc->sc_fn0 = sf0; - SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf0, sf_list); + sdmmc_function_add(sc, sf0); /* Go to Data Transfer Mode, if possible. */ sdmmc_chip_bus_rod(sc->sc_sct, sc->sc_sch, 0); @@ -175,11 +177,11 @@ sdmmc_io_scan(struct sdmmc_softc *sc) sf = sdmmc_function_alloc(sc); sf->number = i; sf->rca = sf0->rca; - SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list); + sdmmc_function_add(sc, sf); } out: - SDMMC_UNLOCK(sc); + return; } /* @@ -192,14 +194,15 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) int error = 0; uint8_t reg; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); sf->blklen = sdmmc_chip_host_maxblklen(sc->sc_sct, sc->sc_sch); if (sf->number == 0) { - reg = sdmmc_io_read_1(sf, SD_IO_CCCR_CAPABILITY); + reg = sdmmc_io_read_1_locked(sf, SD_IO_CCCR_CAPABILITY); if (!(reg & CCCR_CAPS_LSC) || (reg & CCCR_CAPS_4BLS)) { - sdmmc_io_write_1(sf, SD_IO_CCCR_BUS_WIDTH, + sdmmc_io_write_1_locked(sf, SD_IO_CCCR_BUS_WIDTH, CCCR_BUS_WIDTH_4); sf->width = 4; error = sdmmc_chip_bus_width(sc->sc_sct, sc->sc_sch, @@ -223,10 +226,11 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) sdmmc_print_cis(sf); #endif - reg = sdmmc_io_read_1(sf, SD_IO_CCCR_HIGH_SPEED); + reg = sdmmc_io_read_1_locked(sf, SD_IO_CCCR_HIGH_SPEED); if (reg & CCCR_HIGH_SPEED_SHS) { reg |= CCCR_HIGH_SPEED_EHS; - sdmmc_io_write_1(sf, SD_IO_CCCR_HIGH_SPEED, reg); + sdmmc_io_write_1_locked(sf, SD_IO_CCCR_HIGH_SPEED, + reg); sf->csd.tran_speed = 50000; /* 50MHz */ /* Wait 400KHz x 8 clock */ @@ -250,11 +254,14 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) } else { - reg = sdmmc_io_read_1(sf0, SD_IO_FBR(sf->number) + 0x000); + reg = sdmmc_io_read_1_locked(sf0, + SD_IO_FBR(sf->number) + 0x000); sf->interface = FBR_STD_FUNC_IF_CODE(reg); - if (sf->interface == 0x0f) + if (sf->interface == 0x0f) { sf->interface = - sdmmc_io_read_1(sf0, SD_IO_FBR(sf->number) + 0x001); + sdmmc_io_read_1_locked(sf0, + SD_IO_FBR(sf->number) + 0x001); + } error = sdmmc_read_cis(sf, &sf->cis); if (error) { aprint_error_dev(sc->sc_dev, "couldn't read CIS\n"); @@ -271,8 +278,6 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) } out: - SDMMC_UNLOCK(sc); - return error; } @@ -289,9 +294,7 @@ sdmmc_io_function_ready(struct sdmmc_function *sf) if (sf->number == 0) return 1; /* FN0 is always ready */ - SDMMC_LOCK(sc); reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_IOREADY); - SDMMC_UNLOCK(sc); return (reg & (1 << sf->number)) != 0; } @@ -306,11 +309,9 @@ sdmmc_io_function_enable(struct sdmmc_function *sf) if (sf->number == 0) return 0; /* FN0 is always enabled */ - SDMMC_LOCK(sc); reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_ENABLE); SET(reg, (1U << sf->number)); sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_ENABLE, reg); - SDMMC_UNLOCK(sc); retry = 5; while (!sdmmc_io_function_ready(sf) && retry-- > 0) @@ -332,11 +333,9 @@ sdmmc_io_function_disable(struct sdmmc_function *sf) if (sf->number == 0) return; /* FN0 is always enabled */ - SDMMC_LOCK(sc); reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_ENABLE); CLR(reg, (1U << sf->number)); sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_ENABLE, reg); - SDMMC_UNLOCK(sc); } static int @@ -346,7 +345,7 @@ sdmmc_io_rw_direct(struct sdmmc_softc *sc, struct sdmmc_function *sf, struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); /* Make sure the card is selected. */ error = sdmmc_select_card(sc, sf); @@ -394,7 +393,7 @@ sdmmc_io_rw_extended(struct sdmmc_softc *sc, struct sdmmc_function *sf, struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); #if 0 /* Make sure the card is selected. */ @@ -433,37 +432,58 @@ sdmmc_io_rw_extended(struct sdmmc_softc *sc, struct sdmmc_function *sf, return error; } -uint8_t -sdmmc_io_read_1(struct sdmmc_function *sf, int reg) +static uint8_t +sdmmc_io_read_1_locked(struct sdmmc_function *sf, int reg) { uint8_t data = 0; - /* Don't lock */ + KASSERT(mutex_owned(&sf->sc->sc_mtx)); (void)sdmmc_io_rw_direct(sf->sc, sf, reg, (u_char *)&data, SD_ARG_CMD52_READ, false); return data; } -void -sdmmc_io_write_1(struct sdmmc_function *sf, int reg, uint8_t data) +uint8_t +sdmmc_io_read_1(struct sdmmc_function *sf, int reg) { + uint8_t data; - /* Don't lock */ + mutex_enter(&sf->sc->sc_mtx); + data = sdmmc_io_read_1_locked(sf, reg); + mutex_exit(&sf->sc->sc_mtx); + + return data; +} + +static void +sdmmc_io_write_1_locked(struct sdmmc_function *sf, int reg, uint8_t data) +{ + + KASSERT(mutex_owned(&sf->sc->sc_mtx)); (void)sdmmc_io_rw_direct(sf->sc, sf, reg, (u_char *)&data, SD_ARG_CMD52_WRITE, false); } +void +sdmmc_io_write_1(struct sdmmc_function *sf, int reg, uint8_t data) +{ + + mutex_enter(&sf->sc->sc_mtx); + sdmmc_io_write_1_locked(sf, reg, data); + mutex_exit(&sf->sc->sc_mtx); +} + uint16_t sdmmc_io_read_2(struct sdmmc_function *sf, int reg) { uint16_t data = 0; - /* Don't lock */ - + mutex_enter(&sf->sc->sc_mtx); (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 2, SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT); + mutex_exit(&sf->sc->sc_mtx); return data; } @@ -471,10 +491,10 @@ void sdmmc_io_write_2(struct sdmmc_function *sf, int reg, uint16_t data) { - /* Don't lock */ - + mutex_enter(&sf->sc->sc_mtx); (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 2, SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT); + mutex_exit(&sf->sc->sc_mtx); } uint32_t @@ -482,10 +502,10 @@ sdmmc_io_read_4(struct sdmmc_function *sf, int reg) { uint32_t data = 0; - /* Don't lock */ - + mutex_enter(&sf->sc->sc_mtx); (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 4, SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT); + mutex_exit(&sf->sc->sc_mtx); return data; } @@ -493,10 +513,10 @@ void sdmmc_io_write_4(struct sdmmc_function *sf, int reg, uint32_t data) { - /* Don't lock */ - + mutex_enter(&sf->sc->sc_mtx); (void)sdmmc_io_rw_extended(sf->sc, sf, reg, (u_char *)&data, 4, SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT); + mutex_exit(&sf->sc->sc_mtx); } @@ -506,7 +526,7 @@ sdmmc_io_read_multi_1(struct sdmmc_function *sf, int reg, u_char *data, { int blocks, bytes, error = 0; - /* Don't lock */ + mutex_enter(&sf->sc->sc_mtx); while (datalen >= sf->blklen) { //blocks = imin(datalen / sf->blklen, @@ -525,6 +545,7 @@ sdmmc_io_read_multi_1(struct sdmmc_function *sf, int reg, u_char *data, error = sdmmc_io_rw_extended(sf->sc, sf, reg, data, datalen, SD_ARG_CMD53_READ); error: + mutex_exit(&sf->sc->sc_mtx); return error; } @@ -534,7 +555,7 @@ sdmmc_io_write_multi_1(struct sdmmc_function *sf, int reg, u_char *data, { int blocks, bytes, error = 0; - /* Don't lock */ + mutex_enter(&sf->sc->sc_mtx); while (datalen >= sf->blklen) { //blocks = imin(datalen / sf->blklen, @@ -553,6 +574,7 @@ sdmmc_io_write_multi_1(struct sdmmc_function *sf, int reg, u_char *data, error = sdmmc_io_rw_extended(sf->sc, sf, reg, data, datalen, SD_ARG_CMD53_WRITE); error: + mutex_exit(&sf->sc->sc_mtx); return error; } @@ -563,7 +585,7 @@ sdmmc_io_read_region_1(struct sdmmc_function *sf, int reg, u_char *data, { int blocks, bytes, error = 0; - /* Don't lock */ + mutex_enter(&sf->sc->sc_mtx); while (datalen >= sf->blklen) { //blocks = imin(datalen / sf->blklen, @@ -583,6 +605,7 @@ sdmmc_io_read_region_1(struct sdmmc_function *sf, int reg, u_char *data, error = sdmmc_io_rw_extended(sf->sc, sf, reg, data, datalen, SD_ARG_CMD53_READ | SD_ARG_CMD53_INCREMENT); error: + mutex_exit(&sf->sc->sc_mtx); return error; } @@ -592,7 +615,7 @@ sdmmc_io_write_region_1(struct sdmmc_function *sf, int reg, u_char *data, { int blocks, bytes, error = 0; - /* Don't lock */ + mutex_enter(&sf->sc->sc_mtx); while (datalen >= sf->blklen) { //blocks = imin(datalen / sf->blklen, @@ -612,6 +635,7 @@ sdmmc_io_write_region_1(struct sdmmc_function *sf, int reg, u_char *data, error = sdmmc_io_rw_extended(sf->sc, sf, reg, data, datalen, SD_ARG_CMD53_WRITE | SD_ARG_CMD53_INCREMENT); error: + mutex_exit(&sf->sc->sc_mtx); return error; } @@ -635,9 +659,14 @@ int sdmmc_io_function_abort(struct sdmmc_function *sf) { u_char data = CCCR_CTL_AS(sf->number); + int error; - return sdmmc_io_rw_direct(sf->sc, NULL, SD_IO_CCCR_CTL, &data, + mutex_enter(&sf->sc->sc_mtx); + error = sdmmc_io_rw_direct(sf->sc, NULL, SD_IO_CCCR_CTL, &data, SD_ARG_CMD52_WRITE, true); + mutex_exit(&sf->sc->sc_mtx); + + return error; } /* @@ -648,9 +677,12 @@ sdmmc_io_reset(struct sdmmc_softc *sc) { u_char data = CCCR_CTL_RES; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (sdmmc_io_rw_direct(sc, NULL, SD_IO_CCCR_CTL, &data, SD_ARG_CMD52_WRITE, true) == 0) - sdmmc_pause(100000, NULL); /* XXX SDMMC_LOCK */ + sdmmc_pause(100000, NULL); } /* @@ -665,7 +697,7 @@ sdmmc_io_send_op_cond(struct sdmmc_softc *sc, u_int32_t ocr, u_int32_t *ocrp) DPRINTF(("sdmmc_io_send_op_cond: ocr = %#x\n", ocr)); - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); /* * If we change the OCR value, retry the command until the OCR @@ -706,11 +738,11 @@ sdmmc_intr_enable(struct sdmmc_function *sf) struct sdmmc_function *sf0 = sc->sc_fn0; uint8_t reg; - SDMMC_LOCK(sc); - reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_INTEN); + mutex_enter(&sc->sc_mtx); + reg = sdmmc_io_read_1_locked(sf0, SD_IO_CCCR_FN_INTEN); reg |= 1 << sf->number; - sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_INTEN, reg); - SDMMC_UNLOCK(sc); + sdmmc_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); + mutex_exit(&sc->sc_mtx); } void @@ -720,11 +752,11 @@ sdmmc_intr_disable(struct sdmmc_function *sf) struct sdmmc_function *sf0 = sc->sc_fn0; uint8_t reg; - SDMMC_LOCK(sc); - reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_INTEN); + mutex_enter(&sc->sc_mtx); + reg = sdmmc_io_read_1_locked(sf0, SD_IO_CCCR_FN_INTEN); reg &= ~(1 << sf->number); - sdmmc_io_write_1(sf0, SD_IO_CCCR_FN_INTEN, reg); - SDMMC_UNLOCK(sc); + sdmmc_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); + mutex_exit(&sc->sc_mtx); } /* @@ -804,7 +836,10 @@ sdmmc_card_intr(device_t dev) if (sc->sc_sct->card_enable_intr == NULL) return; - sdmmc_add_task(sc, &sc->sc_intr_task); + mutex_enter(&sc->sc_intr_mtx); + if (sc->sc_intr_enabled) + sdmmc_add_task(sc, &sc->sc_intr_task); + mutex_exit(&sc->sc_intr_mtx); } void @@ -818,9 +853,8 @@ sdmmc_intr_task(void *arg) /* XXX examine return value and do evcount stuff*/ (void)(*ih->ih_fun)(ih->ih_arg); } - mutex_exit(&sc->sc_mtx); - sdmmc_chip_card_intr_ack(sc->sc_sct, sc->sc_sch); + mutex_exit(&sc->sc_mtx); } int @@ -831,22 +865,22 @@ sdmmc_io_set_blocklen(struct sdmmc_function *sf, struct sdmmc_function *sf0 = sc->sc_fn0; int error = EINVAL; - SDMMC_LOCK(sc); + mutex_enter(&sc->sc_mtx); if (blklen <= 0 || blklen > sdmmc_chip_host_maxblklen(sc->sc_sct, sc->sc_sch)) goto err; - sdmmc_io_write_1(sf0, SD_IO_FBR(sf->number) + + sdmmc_io_write_1_locked(sf0, SD_IO_FBR(sf->number) + SD_IO_FBR_BLOCKLEN, blklen & 0xff); - sdmmc_io_write_1(sf0, SD_IO_FBR(sf->number) + + sdmmc_io_write_1_locked(sf0, SD_IO_FBR(sf->number) + SD_IO_FBR_BLOCKLEN + 1, (blklen >> 8) & 0xff); sf->blklen = blklen; error = 0; err: - SDMMC_UNLOCK(sc); + mutex_enter(&sc->sc_mtx); return error; } diff --git a/sys/dev/sdmmc/sdmmc_mem.c b/sys/dev/sdmmc/sdmmc_mem.c index d58b27111387..7841c2488f24 100644 --- a/sys/dev/sdmmc/sdmmc_mem.c +++ b/sys/dev/sdmmc/sdmmc_mem.c @@ -142,7 +142,8 @@ sdmmc_mem_enable(struct sdmmc_softc *sc) uint32_t ocr = 0; int error; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); /* Set host mode to SD "combo" card or SD memory-only. */ CLR(sc->sc_flags, SMF_UHS_MODE); @@ -265,8 +266,6 @@ mmc_mode: } out: - SDMMC_UNLOCK(sc); - return error; } @@ -275,6 +274,9 @@ sdmmc_mem_signal_voltage(struct sdmmc_softc *sc, int signal_voltage) { int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + /* * Stop the clock */ @@ -325,7 +327,8 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) int error; int retry; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); /* * CMD2 is a broadcast command understood by SD cards and MMC @@ -383,7 +386,7 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) if (sc->sc_fn0 == NULL) sc->sc_fn0 = sf; - SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list); + sdmmc_function_add(sc, sf); /* only one function in SPI mode */ if (ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) @@ -416,8 +419,6 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) sdmmc_print_cid(&sf->cid); #endif } - - SDMMC_UNLOCK(sc); } int @@ -442,6 +443,9 @@ sdmmc_decode_csd(struct sdmmc_softc *sc, sdmmc_response resp, struct sdmmc_csd *csd = &sf->csd; int e, m; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (ISSET(sc->sc_flags, SMF_SD_MODE)) { /* * CSD version 1.0 corresponds to SD system @@ -510,6 +514,9 @@ sdmmc_decode_cid(struct sdmmc_softc *sc, sdmmc_response resp, { struct sdmmc_cid *cid = &sf->cid; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (ISSET(sc->sc_flags, SMF_SD_MODE)) { cid->mid = SD_CID_MID(resp); cid->oid = SD_CID_OID(resp); @@ -577,7 +584,8 @@ sdmmc_mem_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) { int error = 0; - SDMMC_LOCK(sc); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { error = sdmmc_select_card(sc, sf); @@ -598,8 +606,6 @@ sdmmc_mem_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) SET(sf->flags, SFF_ERROR); out: - SDMMC_UNLOCK(sc); - return error; } @@ -613,7 +619,7 @@ sdmmc_mem_send_op_cond(struct sdmmc_softc *sc, uint32_t ocr, uint32_t *ocrp) int error; int retry; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); DPRINTF(("%s: sdmmc_mem_send_op_cond: ocr=%#x\n", SDMMCDEVNAME(sc), ocr)); @@ -671,7 +677,7 @@ sdmmc_mem_send_if_cond(struct sdmmc_softc *sc, uint32_t ocr, uint32_t *ocrp) struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); memset(&cmd, 0, sizeof(cmd)); cmd.c_arg = ocr; @@ -702,7 +708,7 @@ sdmmc_mem_set_blocklen(struct sdmmc_softc *sc, struct sdmmc_function *sf, struct sdmmc_command cmd; int error; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_mtx)); memset(&cmd, 0, sizeof(cmd)); cmd.c_opcode = MMC_SET_BLOCKLEN; @@ -760,6 +766,9 @@ sdmmc_mem_execute_tuning(struct sdmmc_softc *sc, struct sdmmc_function *sf) { int timing = -1; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (ISSET(sc->sc_flags, SMF_SD_MODE)) { if (!ISSET(sc->sc_flags, SMF_UHS_MODE)) return 0; @@ -797,6 +806,9 @@ sdmmc_mem_sd_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) sdmmc_bitfield512_t status; bool ddr = false; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + /* change bus clock */ bus_clock = uimin(sc->sc_busclk, sf->csd.tran_speed); error = sdmmc_chip_bus_clock(sc->sc_sct, sc->sc_sch, bus_clock, false); @@ -937,6 +949,9 @@ sdmmc_mem_mmc_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) uint32_t sectors = 0; bool ddr = false; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + sc->sc_transfer_mode = NULL; /* change bus clock */ @@ -1167,6 +1182,9 @@ sdmmc_mem_send_cid(struct sdmmc_softc *sc, sdmmc_response *resp) struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { memset(&cmd, 0, sizeof cmd); cmd.c_opcode = MMC_ALL_SEND_CID; @@ -1194,6 +1212,9 @@ sdmmc_mem_send_csd(struct sdmmc_softc *sc, struct sdmmc_function *sf, struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { memset(&cmd, 0, sizeof cmd); cmd.c_opcode = MMC_SEND_CSD; @@ -1226,7 +1247,8 @@ sdmmc_mem_send_scr(struct sdmmc_softc *sc, struct sdmmc_function *sf, int rseg; int error = 0; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); if (ISSET(sc->sc_caps, SMC_CAPS_DMA)) { error = bus_dmamem_alloc(sc->sc_dmat, datalen, PAGE_SIZE, 0, @@ -1297,6 +1319,8 @@ sdmmc_mem_decode_scr(struct sdmmc_softc *sc, struct sdmmc_function *sf) sdmmc_response resp; int ver; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + memset(resp, 0, sizeof(resp)); /* * Change the raw-scr received from the DMA stream to resp. @@ -1333,7 +1357,8 @@ sdmmc_mem_send_ssr(struct sdmmc_softc *sc, struct sdmmc_function *sf, int rseg; int error = 0; - /* Don't lock */ + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); if (ISSET(sc->sc_caps, SMC_CAPS_DMA)) { error = bus_dmamem_alloc(sc->sc_dmat, datalen, PAGE_SIZE, 0, @@ -1460,6 +1485,9 @@ sdmmc_mem_send_cxd_data(struct sdmmc_softc *sc, int opcode, void *data, int rseg; int error = 0; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (ISSET(sc->sc_caps, SMC_CAPS_DMA)) { error = bus_dmamem_alloc(sc->sc_dmat, datalen, PAGE_SIZE, 0, ds, 1, &rseg, BUS_DMA_NOWAIT); @@ -1530,6 +1558,8 @@ sdmmc_set_bus_width(struct sdmmc_function *sf, int width) struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_mtx)); + if (ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) return ENODEV; @@ -1567,6 +1597,9 @@ sdmmc_mem_sd_switch(struct sdmmc_function *sf, int mode, int group, int gsft, rseg, error = 0; const int statlen = 64; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + if (sf->scr.sd_spec >= SCR_SD_SPEC_VER_1_10 && !ISSET(sf->csd.ccc, SD_CSD_CCC_SWITCH)) return EINVAL; @@ -1646,6 +1679,8 @@ sdmmc_mem_mmc_switch(struct sdmmc_function *sf, uint8_t set, uint8_t index, struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_mtx)); + memset(&cmd, 0, sizeof(cmd)); cmd.c_opcode = MMC_SWITCH; cmd.c_arg = (MMC_SWITCH_MODE_WRITE_BYTE << 24) | @@ -1698,6 +1733,8 @@ sdmmc_mem_spi_read_ocr(struct sdmmc_softc *sc, uint32_t hcs, uint32_t *card_ocr) struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_mtx)); + memset(&cmd, 0, sizeof(cmd)); cmd.c_opcode = MMC_READ_OCR; cmd.c_arg = hcs ? MMC_OCR_HCS : 0; @@ -1725,6 +1762,7 @@ sdmmc_mem_single_read_block(struct sdmmc_function *sf, uint32_t blkno, KASSERT((datalen % SDMMC_SECTOR_SIZE) == 0); KASSERT(!ISSET(sc->sc_caps, SMC_CAPS_DMA)); + KASSERT(mutex_owned(&sc->sc_mtx)); for (i = 0; i < datalen / SDMMC_SECTOR_SIZE; i++) { error = sdmmc_mem_read_block_subr(sf, sc->sc_dmap, blkno + i, @@ -1747,6 +1785,8 @@ sdmmc_mem_single_segment_dma_read_block(struct sdmmc_function *sf, int error = 0; int i; + KASSERT(mutex_owned(&sc->sc_mtx)); + for (i = 0; i < sc->sc_dmap->dm_nsegs; i++) { size_t len = sc->sc_dmap->dm_segs[i].ds_len; if ((len % SDMMC_SECTOR_SIZE) != 0) { @@ -1811,6 +1851,8 @@ sdmmc_mem_read_block_subr(struct sdmmc_function *sf, bus_dmamap_t dmap, struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_mtx)); + if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { error = sdmmc_select_card(sc, sf); if (error) @@ -1884,7 +1926,6 @@ sdmmc_mem_read_block(struct sdmmc_function *sf, uint32_t blkno, u_char *data, struct sdmmc_softc *sc = sf->sc; int error; - SDMMC_LOCK(sc); mutex_enter(&sc->sc_mtx); if (ISSET(sc->sc_caps, SMC_CAPS_SINGLE_ONLY)) { @@ -1935,7 +1976,6 @@ unload: out: mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); return error; } @@ -1951,6 +1991,7 @@ sdmmc_mem_single_write_block(struct sdmmc_function *sf, uint32_t blkno, KASSERT((datalen % SDMMC_SECTOR_SIZE) == 0); KASSERT(!ISSET(sc->sc_caps, SMC_CAPS_DMA)); + KASSERT(mutex_owned(&sc->sc_mtx)); for (i = 0; i < datalen / SDMMC_SECTOR_SIZE; i++) { error = sdmmc_mem_write_block_subr(sf, sc->sc_dmap, blkno + i, @@ -1973,6 +2014,8 @@ sdmmc_mem_single_segment_dma_write_block(struct sdmmc_function *sf, int error = 0; int i; + KASSERT(mutex_owned(&sc->sc_mtx)); + for (i = 0; i < sc->sc_dmap->dm_nsegs; i++) { size_t len = sc->sc_dmap->dm_segs[i].ds_len; if ((len % SDMMC_SECTOR_SIZE) != 0) { @@ -2038,6 +2081,8 @@ sdmmc_mem_write_block_subr(struct sdmmc_function *sf, bus_dmamap_t dmap, struct sdmmc_command cmd; int error; + KASSERT(mutex_owned(&sc->sc_mtx)); + if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { error = sdmmc_select_card(sc, sf); if (error) @@ -2122,7 +2167,6 @@ sdmmc_mem_write_block(struct sdmmc_function *sf, uint32_t blkno, u_char *data, struct sdmmc_softc *sc = sf->sc; int error; - SDMMC_LOCK(sc); mutex_enter(&sc->sc_mtx); if (ISSET(sc->sc_flags, SMF_SD_MODE) && @@ -2182,7 +2226,6 @@ unload: out: mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); return error; } @@ -2200,7 +2243,6 @@ sdmmc_mem_discard(struct sdmmc_function *sf, uint32_t sblkno, uint32_t eblkno) if (eblkno < sblkno) return EINVAL; - SDMMC_LOCK(sc); mutex_enter(&sc->sc_mtx); /* Set the address of the first write block to be erased */ @@ -2237,7 +2279,6 @@ sdmmc_mem_discard(struct sdmmc_function *sf, uint32_t sblkno, uint32_t eblkno) out: mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); #ifdef SDMMC_DEBUG device_printf(sc->sc_dev, "discard blk %u-%u error %d\n", @@ -2256,15 +2297,11 @@ sdmmc_mem_flush_cache(struct sdmmc_function *sf, bool poll) if (!ISSET(sf->flags, SFF_CACHE_ENABLED)) return 0; - SDMMC_LOCK(sc); mutex_enter(&sc->sc_mtx); - error = sdmmc_mem_mmc_switch(sf, EXT_CSD_CMD_SET_NORMAL, EXT_CSD_FLUSH_CACHE, EXT_CSD_FLUSH_CACHE_FLUSH, poll); - mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); #ifdef SDMMC_DEBUG device_printf(sc->sc_dev, "mmc flush cache error %d\n", error); diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index 0c2d553294e3..b4c95da3f0bb 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -222,7 +222,7 @@ struct sdmmc_softc { #define SDMMC_MAXNSEGS ((MAXPHYS / PAGE_SIZE) + 1) struct kmutex sc_mtx; /* lock around host controller */ - int sc_dying; /* bus driver is shutting down */ + int sc_tskq_dying; /* bus driver is shutting down */ uint32_t sc_flags; #define SMF_INITED 0x0001 @@ -268,9 +268,9 @@ struct sdmmc_softc { struct kcondvar sc_tskq_cv; struct sdmmc_task *sc_curtask; - /* discover task */ - struct sdmmc_task sc_discover_task; /* card attach/detach task */ - struct kmutex sc_discover_task_mtx; + /* former discovery task */ + struct sdmmc_task sc_unused_task; + struct kmutex sc_unused_mtx; /* interrupt task */ struct sdmmc_task sc_intr_task; /* card interrupt task */ @@ -292,6 +292,17 @@ struct sdmmc_softc { struct evcnt sc_ev_xfer_error; /* error xfer count */ uint32_t sc_max_seg; /* maximum segment size */ + + struct kmutex sc_attach_mtx; /* serializes child attach/detach */ + bool sc_detaching; /* set on sdmmc_detach */ + + struct kmutex sc_intr_mtx; /* serializes intr task scheduling */ + bool sc_intr_enabled; /* true if discover/cardintr allowed */ + + /* card discovery */ + struct lwp *sc_discover_lwp; + struct kcondvar sc_discover_cv; + bool sc_discover_needed; }; /* @@ -318,9 +329,6 @@ struct sdmmc_product { #define splsdmmc() splbio() #endif -#define SDMMC_LOCK(sc) -#define SDMMC_UNLOCK(sc) - #ifdef SDMMC_DEBUG extern int sdmmcdebug; #endif @@ -330,6 +338,7 @@ bool sdmmc_del_task(struct sdmmc_softc *, struct sdmmc_task *, kmutex_t *); struct sdmmc_function *sdmmc_function_alloc(struct sdmmc_softc *); void sdmmc_function_free(struct sdmmc_function *); +void sdmmc_function_add(struct sdmmc_softc *, struct sdmmc_function *); int sdmmc_set_bus_power(struct sdmmc_softc *, uint32_t, uint32_t); int sdmmc_mmc_command(struct sdmmc_softc *, struct sdmmc_command *); int sdmmc_app_command(struct sdmmc_softc *, struct sdmmc_function *,