From 0036147e0a1bebd40877820724b70ed81cba54a3 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 22:05:55 +0000 Subject: [PATCH 01/21] sdmmc: Use kthread_join, not a cv dance, to wait for task thread. Simpler this way. Would be needed in order to make sdmmc an unloadable module, because we need to ensure not just that a wakeup has been issued and mutex exited in the task thread, but that the task thread function is no longer executing on any CPU. No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 3127012022a2..79e667235142 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -206,9 +206,8 @@ sdmmc_detach(device_t self, int flags) mutex_enter(&sc->sc_tskq_mtx); sc->sc_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); @@ -349,7 +348,6 @@ sdmmc_task_thread(void *arg) } } /* time to die. */ - sc->sc_dying = 0; if (ISSET(sc->sc_flags, SMF_CARD_PRESENT)) { /* * sdmmc_card_detach() may issue commands, @@ -359,8 +357,6 @@ sdmmc_task_thread(void *arg) 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); } From baf423b4f961466a3abb082cd4799a663965bc0a Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 22:10:12 +0000 Subject: [PATCH 02/21] sdmmc: Simplify polling callout. The task will do card detect anyway, so no loss of functionality. Might slightly increase the cost of polling when the state hasn't changed, because that now requires scheduling the task thread. But it be simpler if we only have that logic in one place and in one type of context, and this will make subsequent changes to fix nearby races and deadlocks easier. No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 12 ++---------- 1 file changed, 2 insertions(+), 10 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 79e667235142..4c917245bb04 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -404,17 +404,9 @@ sdmmc_discover_task(void *arg) static void sdmmc_polling_card(void *arg) { - struct sdmmc_softc *sc = (struct sdmmc_softc *)arg; - int card_detect, card_present; - - 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); - - if (card_detect != card_present) - sdmmc_needs_discover(sc->sc_dev); + struct sdmmc_softc *sc = arg; + sdmmc_needs_discover(sc->sc_dev); callout_schedule(&sc->sc_card_detect_ch, hz); } From 8f02573b6df2674d4d0c225323e7d9569901b375 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 22:23:10 +0000 Subject: [PATCH 03/21] sdmmc: Rename sc_dying -> sc_tskq_dying. This is used only to notify the task thread, and access to it is serialized by sc_tskq_mtx, so make the naming clearly reflect that. - No functional change intended. - No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 4 ++-- sys/dev/sdmmc/sdmmcvar.h | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 4c917245bb04..7fbf29059618 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -204,7 +204,7 @@ sdmmc_detach(device_t self, int flags) int error, i; mutex_enter(&sc->sc_tskq_mtx); - sc->sc_dying = 1; + sc->sc_tskq_dying = 1; cv_signal(&sc->sc_tskq_cv); mutex_exit(&sc->sc_tskq_mtx); (void)kthread_join(sc->sc_tskq_lwp); @@ -342,7 +342,7 @@ 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); } diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index 0c2d553294e3..fabbb1780bae 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 From 035ae8ce81aac0dab56da4e17f8d841735177ac6 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 22:24:35 +0000 Subject: [PATCH 04/21] sdmmc: Narrow the scope of sc_tskq_mtx in sdmmc_task_thread. This doesn't serialize access to sc_flags, so it can't help to avoid races here. And currently sc_flags is only ever modified in the task thread, so no lock is needed at this point anyway -- there's only one task thread. - No functional change intended. - No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 10 +++------- 1 file changed, 3 insertions(+), 7 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 7fbf29059618..91dab2c0ecff 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -347,17 +347,13 @@ sdmmc_task_thread(void *arg) cv_wait(&sc->sc_tskq_cv, &sc->sc_tskq_mtx); } } + mutex_exit(&sc->sc_tskq_mtx); + /* time to die. */ 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); } - mutex_exit(&sc->sc_tskq_mtx); + kthread_exit(0); } From b1bfe34df8cbcb1a9afbb4972f804495a4d7c746 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 22:37:18 +0000 Subject: [PATCH 05/21] sdmmc: Serialize attach/detach in discover and sdmmc_detach. - New sc_attach_mtx serializes both sdmmc_discover_task and config_detach_children. - New sc_detaching makes sdmmc_discover_task treat card as yanked, without asking the hardware. This way, sdmmc_detach can temporarily suspend concurrent child attach/detach in the hardware discovery task while it runs config_detach_children to detach our software drivers. If config_detach_childen succeeds (which it always does with DETACH_FORCE, but not if we use `drvct -d sdmmcN'), sdmmc_detach can set sc_detaching to permanently treat the card as yanked so no new software drivers will attach children and we can proceed to free all the sdmmc resources. No module ABI change intended, because modules don't rely on sizeof(struct sdmmc_softc) and the new members go at the end of struct sdmmc_softc. --- sys/dev/sdmmc/sdmmc.c | 39 ++++++++++++++++++++++++++++++++++----- sys/dev/sdmmc/sdmmcvar.h | 3 +++ 2 files changed, 37 insertions(+), 5 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 91dab2c0ecff..9697e6e645ed 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -152,6 +152,7 @@ sdmmc_attach(device_t parent, device_t self, void *aux) 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); cv_init(&sc->sc_tskq_cv, "mmctaskq"); evcnt_attach_dynamic(&sc->sc_ev_xfer, EVCNT_TYPE_MISC, NULL, @@ -203,6 +204,28 @@ 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); + + /* + * Wait for queued tasks to complete and treat any remaining + * card as forcibly detached. + */ mutex_enter(&sc->sc_tskq_mtx); sc->sc_tskq_dying = 1; cv_signal(&sc->sc_tskq_cv); @@ -211,10 +234,6 @@ sdmmc_detach(device_t self, int flags) 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); @@ -229,6 +248,7 @@ sdmmc_detach(device_t self, int flags) sdmmc_del_task(sc, &sc->sc_discover_task, NULL); cv_destroy(&sc->sc_tskq_cv); + mutex_destroy(&sc->sc_attach_mtx); mutex_destroy(&sc->sc_discover_task_mtx); mutex_destroy(&sc->sc_tskq_mtx); mutex_destroy(&sc->sc_mtx); @@ -350,9 +370,11 @@ sdmmc_task_thread(void *arg) mutex_exit(&sc->sc_tskq_mtx); /* time to die. */ + mutex_enter(&sc->sc_attach_mtx); if (ISSET(sc->sc_flags, SMF_CARD_PRESENT)) { sdmmc_card_detach(sc, DETACH_FORCE); } + mutex_exit(&sc->sc_attach_mtx); kthread_exit(0); } @@ -374,8 +396,10 @@ sdmmc_discover_task(void *arg) struct sdmmc_softc *sc = (struct sdmmc_softc *)arg; int card_detect, card_present; + 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); + card_detect = sc->sc_detaching ? 0 : + sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); card_present = ISSET(sc->sc_flags, SMF_CARD_PRESENT); if (card_detect) SET(sc->sc_flags, SMF_CARD_PRESENT); @@ -395,6 +419,7 @@ sdmmc_discover_task(void *arg) if (card_present) sdmmc_card_detach(sc, DETACH_FORCE); } + mutex_exit(&sc->sc_attach_mtx); } static void @@ -418,6 +443,8 @@ sdmmc_card_attach(struct sdmmc_softc *sc) DPRINTF(1,("%s: attach card\n", DEVNAME(sc))); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + CLR(sc->sc_flags, SMF_CARD_ATTACHED); sdmmc_chip_hw_reset(sc->sc_sct, sc->sc_sch); @@ -484,6 +511,8 @@ sdmmc_card_detach(struct sdmmc_softc *sc, int flags) DPRINTF(1,("%s: detach card\n", DEVNAME(sc))); + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + if (ISSET(sc->sc_flags, SMF_CARD_ATTACHED)) { SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { if (sf->child != NULL) { diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index fabbb1780bae..411edfce27ee 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -292,6 +292,9 @@ 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 */ }; /* From 845eae11e40a2e2e5f46d3d899ee015db8b8043e Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 23:26:36 +0000 Subject: [PATCH 06/21] sdmmc: New sdmmc_function_add to put a function on the list. This lets us more easily sprinkle locking assertions, and, if necessary, adjust the locking around access to the function list. No module ABI change intended because modules shouldn't touch the function list directly. (Even if they do, a new kernel is compatible with existing modules -- the function list itself didn't change; we just added a new symbol to regulate access to it.) --- sys/dev/sdmmc/sdmmc.c | 20 ++++++++++++++++++++ sys/dev/sdmmc/sdmmc_io.c | 4 ++-- sys/dev/sdmmc/sdmmc_mem.c | 2 +- sys/dev/sdmmc/sdmmcvar.h | 1 + 4 files changed, 24 insertions(+), 3 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 9697e6e645ed..4a692ef6b74b 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -770,6 +770,26 @@ 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)); + + 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. diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c index 23cdd8aec984..144d6c009f8a 100644 --- a/sys/dev/sdmmc/sdmmc_io.c +++ b/sys/dev/sdmmc/sdmmc_io.c @@ -157,7 +157,7 @@ sdmmc_io_scan(struct sdmmc_softc *sc) 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,7 +175,7 @@ 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: diff --git a/sys/dev/sdmmc/sdmmc_mem.c b/sys/dev/sdmmc/sdmmc_mem.c index d58b27111387..aa74421e2bc2 100644 --- a/sys/dev/sdmmc/sdmmc_mem.c +++ b/sys/dev/sdmmc/sdmmc_mem.c @@ -383,7 +383,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)) diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index 411edfce27ee..67abc25fa54c 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -333,6 +333,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 *, From 29d42d92cbc5801df0ee35b47e5a6f035af1ccd5 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 23:49:12 +0000 Subject: [PATCH 07/21] sdmmc: Set SMF_CARD_ATTACHED before attaching children. This way the sf->child pointers can be nonnull only if SMF_CARD_ATTACHED is set, so we will be able to add assertions about that in a future sdmmc_childdet routine. - No functional change intended other than to enable stronger assertions. - No module ABI change intended because there was no safe way for modules to rely on at SMF_CARD_ATTACHED anyway, so any modules that did were broken. --- sys/dev/sdmmc/sdmmc.c | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 4a692ef6b74b..a7292d57dea9 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -479,6 +479,11 @@ sdmmc_card_attach(struct sdmmc_softc *sc) goto err; } + /* + * Card is attached. We can now attach child drivers. + */ + SET(sc->sc_flags, SMF_CARD_ATTACHED); + SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { if (ISSET(sc->sc_flags, SMF_IO_MODE) && sf->number < 1) continue; @@ -493,7 +498,6 @@ sdmmc_card_attach(struct sdmmc_softc *sc) config_found(sc->sc_dev, &saa, sdmmc_print, CFARGS_NONE); } - SET(sc->sc_flags, SMF_CARD_ATTACHED); return; err: @@ -523,6 +527,11 @@ sdmmc_card_detach(struct sdmmc_softc *sc, int flags) 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); } From 6decc1b6de709bf9cd8b50146f4e958df8099d7e Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Tue, 23 Jan 2024 23:51:55 +0000 Subject: [PATCH 08/21] sdmmc: Hold sc_mtx across sdmmc_card_attach/detach; support drvctl. Also hold it across changes to sc_flags. (XXX Need to audit reads from sc_flags, but this covers writes to it, at least.) However, release it around child attach/detach, when we call back into autoconf(9). This means we also hold sc_mtx across: 1. access to the function list (sc->sf_head, sf->sf_list) 2. access to the device_t child pointers (sf->child) Thus, we can safely read the function list under only sc_mtx (writing also requires sc_attach_mtx), and we can safely read and write device_t child pointers under only sc_mtx. New sdmmc_childdet uses this so we can safely manage the child pointers in the event of concurrent access like drvctl. No module ABI change intended because we release the lock around child attach/detach, so the child context is the same. --- sys/dev/sdmmc/sdmmc.c | 92 ++++++++++++++++++++++++++++++++++++++----- 1 file changed, 83 insertions(+), 9 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index a7292d57dea9..0034600c93cc 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -84,9 +84,11 @@ 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 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, /*rescan*/NULL, + sdmmc_childdet); static void sdmmc_doattach(device_t); static void sdmmc_task_thread(void *); @@ -262,6 +264,27 @@ sdmmc_detach(device_t self, int flags) 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) { @@ -400,6 +423,8 @@ sdmmc_discover_task(void *arg) mutex_enter(&sc->sc_discover_task_mtx); 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); @@ -416,9 +441,12 @@ sdmmc_discover_task(void *arg) mutex_exit(&sc->sc_discover_task_mtx); } } else { - if (card_present) + if (card_present) { sdmmc_card_detach(sc, DETACH_FORCE); + } } + mutex_exit(&sc->sc_mtx); + mutex_exit(&sc->sc_attach_mtx); } @@ -444,6 +472,7 @@ sdmmc_card_attach(struct sdmmc_softc *sc) DPRINTF(1,("%s: attach card\n", DEVNAME(sc))); KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); CLR(sc->sc_flags, SMF_CARD_ATTACHED); @@ -485,6 +514,8 @@ sdmmc_card_attach(struct sdmmc_softc *sc) SET(sc->sc_flags, SMF_CARD_ATTACHED); SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { + device_t child; + if (ISSET(sc->sc_flags, SMF_IO_MODE) && sf->number < 1) continue; @@ -494,8 +525,22 @@ 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 */ } return; @@ -516,13 +561,41 @@ sdmmc_card_detach(struct sdmmc_softc *sc, int flags) 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. + */ 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; - } + device_t child; + + /* + * If there's no child attached at this + * function, nothing to do -- it will stay this + * way as long as we hold sc_attach_mtx. + */ + if ((child = sf->child) == NULL) + continue; + + /* + * Acquire a reference to the child so we can + * drop sc_mtx to forcibly detach the child, in + * case the child driver tries to do sdmmc + * operations. + * + * Further sdmmc operations will probably fail + * in this case, but some drivers like ld(4) + * will try things like cache flushing anyway + * because in other circumstances it is + * critical to avoid data loss. + */ + device_acquire(child); + mutex_exit(&sc->sc_mtx); + config_detach_release(child, DETACH_FORCE); + child = NULL; /* paranoia */ + mutex_enter(&sc->sc_mtx); } KASSERT(TAILQ_EMPTY(&sc->sc_intrq)); @@ -795,6 +868,7 @@ 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)); SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list); } From 56b5f90c59171c74088fd9bf13be11bc161c00da Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:19:52 +0000 Subject: [PATCH 09/21] sdmmc: Sprinkle SMF_CARD_ATTACHED assertions. No functional change intended. --- sys/dev/sdmmc/sdmmc.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 0034600c93cc..a9262e105d45 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -444,6 +444,7 @@ sdmmc_discover_task(void *arg) if (card_present) { sdmmc_card_detach(sc, DETACH_FORCE); } + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); } mutex_exit(&sc->sc_mtx); @@ -473,6 +474,7 @@ sdmmc_card_attach(struct sdmmc_softc *sc) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); CLR(sc->sc_flags, SMF_CARD_ATTACHED); @@ -511,6 +513,7 @@ sdmmc_card_attach(struct sdmmc_softc *sc) /* * Card is attached. We can now attach child drivers. */ + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); SET(sc->sc_flags, SMF_CARD_ATTACHED); SIMPLEQ_FOREACH(sf, &sc->sf_head, sf_list) { @@ -543,10 +546,12 @@ sdmmc_card_attach(struct sdmmc_softc *sc) child = NULL; /* paranoia */ } + KASSERT(ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); return; err: sdmmc_card_detach(sc, DETACH_FORCE); + KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); } /* @@ -869,6 +874,7 @@ 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_ATTACHED)); SIMPLEQ_INSERT_TAIL(&sc->sf_head, sf, sf_list); } From da6298d6ee25d7460815d7409fa3d48ab97f6cec Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:20:29 +0000 Subject: [PATCH 10/21] sdmmc: Omit needless use of sc_discover_task_mtx. The call to sdmmc_chip_card_detect in sdmmc_discover_task is already serialized (both by running in a single thread, and by holding sc_attach_mtx), and access to sc_flags is serialized by sc_mtx rather than by sc_discover_task_mtx, so just nix the confusing sc_discover_task_mtx usage. We may repurpose sc_discover_task_mtx later to split discovery out of the task thread into its own thread in order to resolve PR kern/57870 properly, but for now let's just get it out of the way since it's not doing anything for us. No module ABI change intended, since we're keeping it in the softc so the layout doesn't change. (In principle modules could have used the mutex but in practice this would be a mistake, so modules that do are presumptively broken.) --- sys/dev/sdmmc/sdmmc.c | 4 ---- 1 file changed, 4 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index a9262e105d45..a3f37deaf3be 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -420,7 +420,6 @@ sdmmc_discover_task(void *arg) int card_detect, card_present; mutex_enter(&sc->sc_attach_mtx); - mutex_enter(&sc->sc_discover_task_mtx); card_detect = sc->sc_detaching ? 0 : sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); @@ -430,15 +429,12 @@ sdmmc_discover_task(void *arg) 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) { From b04b5d840f5cb035970b95cca947d7b585024bf8 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:31:13 +0000 Subject: [PATCH 11/21] sdmmc: Sprinkle SMF_CARD_PRESENT assertions. No functional change intended. --- sys/dev/sdmmc/sdmmc.c | 2 ++ 1 file changed, 2 insertions(+) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index a3f37deaf3be..e8c5605e3048 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -470,6 +470,7 @@ sdmmc_card_attach(struct sdmmc_softc *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); @@ -870,6 +871,7 @@ 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); From 3a308d21e4680d2ba989bba2c35c6e64023a7f8b Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:45:47 +0000 Subject: [PATCH 12/21] sdmmc: Use config_detach_children, not open-coded loop. We would need the open-coded loop if we ever had any children aside from sf->child for some sf on the function list, but we don't need that today, so let's keep it simpler. No functional change intended. --- sys/dev/sdmmc/sdmmc.c | 46 +++++++++++++++---------------------------- 1 file changed, 16 insertions(+), 30 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index e8c5605e3048..f7ac9bf38267 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -559,6 +559,7 @@ 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))); @@ -567,38 +568,23 @@ sdmmc_card_detach(struct sdmmc_softc *sc, int flags) /* * If the card had finished attaching, forcibly detach any - * children whose software drivers might have attached. + * 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) { - device_t child; - - /* - * If there's no child attached at this - * function, nothing to do -- it will stay this - * way as long as we hold sc_attach_mtx. - */ - if ((child = sf->child) == NULL) - continue; - - /* - * Acquire a reference to the child so we can - * drop sc_mtx to forcibly detach the child, in - * case the child driver tries to do sdmmc - * operations. - * - * Further sdmmc operations will probably fail - * in this case, but some drivers like ld(4) - * will try things like cache flushing anyway - * because in other circumstances it is - * critical to avoid data loss. - */ - device_acquire(child); - mutex_exit(&sc->sc_mtx); - config_detach_release(child, DETACH_FORCE); - child = NULL; /* paranoia */ - mutex_enter(&sc->sc_mtx); - } + mutex_exit(&sc->sc_mtx); + error = config_detach_children(sc->sc_dev, DETACH_FORCE); + mutex_enter(&sc->sc_mtx); + KASSERTMSG(error == 0, + "%s failed to forcibly detach children: %d", + SDMMCDEVNAME(sc), error); KASSERT(TAILQ_EMPTY(&sc->sc_intrq)); From ca25c10d028f7022faf920a68d9a44fa018d7e64 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:48:41 +0000 Subject: [PATCH 13/21] sdmmc: Use SIMPLEQ_FOREACH_SAFE instead of open-coding it. No functional change intended. --- sys/dev/sdmmc/sdmmc.c | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index f7ac9bf38267..381f8d2c00ea 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -600,8 +600,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); From 9e5277a2656843223a8bf15a670a46b796fd5223 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:49:09 +0000 Subject: [PATCH 14/21] sdmmc: Implement rescan (drvctl -r). No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 53 +++++++++++++++++++++++++++++++++++-------- 1 file changed, 43 insertions(+), 10 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 381f8d2c00ea..7da1e431d2d5 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -84,10 +84,11 @@ 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_DECL2_NEW(sdmmc, sizeof(struct sdmmc_softc), - sdmmc_match, sdmmc_attach, sdmmc_detach, /*activate*/NULL, /*rescan*/NULL, + sdmmc_match, sdmmc_attach, sdmmc_detach, /*activate*/NULL, sdmmc_rescan, sdmmc_childdet); static void sdmmc_doattach(device_t); @@ -95,6 +96,7 @@ static void sdmmc_task_thread(void *); static void sdmmc_discover_task(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 *); @@ -264,6 +266,20 @@ 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) { @@ -462,8 +478,6 @@ 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))); @@ -512,10 +526,36 @@ sdmmc_card_attach(struct sdmmc_softc *sc) */ 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; @@ -542,13 +582,6 @@ sdmmc_card_attach(struct sdmmc_softc *sc) device_release(child); child = NULL; /* paranoia */ } - - KASSERT(ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); - return; - -err: - sdmmc_card_detach(sc, DETACH_FORCE); - KASSERT(!ISSET(sc->sc_flags, SMF_CARD_ATTACHED)); } /* From 41b7a775d490c4c936f211a1214228615d83227a Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 00:51:24 +0000 Subject: [PATCH 15/21] sdmmc: Fix memory leak in sdmmc_io_scan error branch. No module ABI change intended. --- sys/dev/sdmmc/sdmmc_io.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c index 144d6c009f8a..a0430049aff6 100644 --- a/sys/dev/sdmmc/sdmmc_io.c +++ b/sys/dev/sdmmc/sdmmc_io.c @@ -153,7 +153,7 @@ sdmmc_io_scan(struct sdmmc_softc *sc) 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; From 942003e6baef0a8224331918838a42f2efdf23ab Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 01:06:57 +0000 Subject: [PATCH 16/21] sdmmc: Avoid races with interrupt handlers. - Create a kill switch for sdmmc_card_intr, sc_intr_enabled, with access serialized by new sc_intr_mtx. - Avoid racy access to sc_flags in sdmmc_needs_discover by using the same kill switch. - Flip the kill switch in sdmmc_detach _before_ we wait for pending tasks to drain. This way, we guarantee that there can be no more discovery or card interrupt tasks scheduled by the time we start waiting for tasks to drain. This should maybe use pserialize(9) rather than a mutex, but the discovery and card interrupt paths already require taking mutexes, so let's just keep it simple for now to get it correct before speeding it up. No module ABI change intended because this only adds to the end of struct sdmmc_softc. --- sys/dev/sdmmc/sdmmc.c | 26 ++++++++++++++++++++++---- sys/dev/sdmmc/sdmmc_io.c | 5 ++++- sys/dev/sdmmc/sdmmcvar.h | 3 +++ 3 files changed, 29 insertions(+), 5 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 7da1e431d2d5..2f65bf697052 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -157,6 +157,7 @@ sdmmc_attach(device_t parent, device_t self, void *aux) 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, @@ -182,6 +183,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, @@ -226,6 +231,18 @@ sdmmc_detach(device_t self, int flags) 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); + /* * Wait for queued tasks to complete and treat any remaining * card as forcibly detached. @@ -252,6 +269,7 @@ sdmmc_detach(device_t self, int flags) sdmmc_del_task(sc, &sc->sc_discover_task, NULL); cv_destroy(&sc->sc_tskq_cv); + mutex_destroy(&sc->sc_intr_mtx); mutex_destroy(&sc->sc_attach_mtx); mutex_destroy(&sc->sc_discover_task_mtx); mutex_destroy(&sc->sc_tskq_mtx); @@ -423,10 +441,10 @@ 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); + if (sc->sc_intr_enabled) + sdmmc_add_task(sc, &sc->sc_discover_task); + mutex_exit(&sc->sc_intr_mtx); } static void diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c index a0430049aff6..2bac0e1c7008 100644 --- a/sys/dev/sdmmc/sdmmc_io.c +++ b/sys/dev/sdmmc/sdmmc_io.c @@ -804,7 +804,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 diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index 67abc25fa54c..b8f7a6fa2e8a 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -295,6 +295,9 @@ struct sdmmc_softc { 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 */ }; /* From 26c3ecea49b0c7b41b851ad358cbc993faa97fd5 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 01:15:13 +0000 Subject: [PATCH 17/21] sdmmc: Halt the polling callout _before_ waiting to drain tasks. The callout may schedule discovery tasks anew, so it must be stopped before we can wait for tasks to finish. No module ABI change intended. --- sys/dev/sdmmc/sdmmc.c | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 2f65bf697052..9ae924ade345 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -243,6 +243,16 @@ sdmmc_detach(device_t self, int flags) sc->sc_intr_enabled = false; mutex_exit(&sc->sc_intr_mtx); + /* + * Prevent periodic polling to schedule discovery tasks. 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); + } + /* * Wait for queued tasks to complete and treat any remaining * card as forcibly detached. @@ -260,11 +270,6 @@ sdmmc_detach(device_t self, int flags) 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); From 6e1adf6c96cf006ce918a481584d3f5e6836034e Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 03:23:34 +0000 Subject: [PATCH 18/21] sdmmc: Split discovery into a separate thread. That way, child driver attach and detach routines can safely issue tasks and wait for them to complete. No module ABI change intended. This leaves some unused members of struct sdmmc_softc, which we can tidy up later; modules should not have used these members anyway. PR kern/57870 --- sys/dev/sdmmc/sdmmc.c | 80 +++++++++++++++++++++++++++------------- sys/dev/sdmmc/sdmmcvar.h | 11 ++++-- 2 files changed, 63 insertions(+), 28 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 9ae924ade345..34d120b8cfe3 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -93,7 +93,7 @@ CFATTACH_DECL2_NEW(sdmmc, sizeof(struct sdmmc_softc), 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 *); @@ -150,15 +150,14 @@ 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"); + cv_init(&sc->sc_discover_cv, "mmcdisc"); evcnt_attach_dynamic(&sc->sc_ev_xfer, EVCNT_TYPE_MISC, NULL, device_xname(self), "xfer"); @@ -244,15 +243,21 @@ sdmmc_detach(device_t self, int flags) mutex_exit(&sc->sc_intr_mtx); /* - * Prevent periodic polling to schedule discovery tasks. After - * this point, there may be discovery still scheduled, but it - * can't be newly scheduled again. + * 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. @@ -271,12 +276,11 @@ sdmmc_detach(device_t self, int flags) } sdmmc_del_task(sc, &sc->sc_intr_task, NULL); - sdmmc_del_task(sc, &sc->sc_discover_task, NULL); + cv_destroy(&sc->sc_discover_cv); cv_destroy(&sc->sc_tskq_cv); mutex_destroy(&sc->sc_intr_mtx); mutex_destroy(&sc->sc_attach_mtx); - mutex_destroy(&sc->sc_discover_task_mtx); mutex_destroy(&sc->sc_tskq_mtx); mutex_destroy(&sc->sc_mtx); @@ -330,7 +334,13 @@ 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"); } } @@ -408,9 +418,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); @@ -430,14 +437,6 @@ sdmmc_task_thread(void *arg) } } mutex_exit(&sc->sc_tskq_mtx); - - /* time to die. */ - mutex_enter(&sc->sc_attach_mtx); - if (ISSET(sc->sc_flags, SMF_CARD_PRESENT)) { - sdmmc_card_detach(sc, DETACH_FORCE); - } - mutex_exit(&sc->sc_attach_mtx); - kthread_exit(0); } @@ -447,18 +446,20 @@ sdmmc_needs_discover(device_t dev) struct sdmmc_softc *sc = device_private(dev); mutex_enter(&sc->sc_intr_mtx); - if (sc->sc_intr_enabled) - sdmmc_add_task(sc, &sc->sc_discover_task); + 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); + + ok = !sc->sc_detaching; card_detect = sc->sc_detaching ? 0 : sdmmc_chip_card_detect(sc->sc_sct, sc->sc_sch); @@ -484,6 +485,35 @@ sdmmc_discover_task(void *arg) mutex_exit(&sc->sc_mtx); mutex_exit(&sc->sc_attach_mtx); + + return ok; +} + +static void +sdmmc_discover_thread(void *arg) +{ + struct sdmmc_softc *sc = arg; + + /* + * 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); + + /* + * 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 diff --git a/sys/dev/sdmmc/sdmmcvar.h b/sys/dev/sdmmc/sdmmcvar.h index b8f7a6fa2e8a..2e27340323d4 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -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 */ @@ -298,6 +298,11 @@ struct sdmmc_softc { 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; }; /* From b4c9ddcdc68aa3b85ca7645fc22b7b84ae0b6911 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Wed, 24 Jan 2024 04:03:00 +0000 Subject: [PATCH 19/21] sdmmc: Serialize I/O operations and card selection by sc_mtx. Sprinkle locking assertions while here. No module ABI change intended. This just provides internal serialization for access to the sdmmc controller. No kind of transaction is exposed to callers by this -- it just means that, e.g., if a driver tries to use sdmmc_io_read/write on the same sdmmc controller (but possibly different functions) at the same time in different threads, the two I/O operations will happen in serial rather than interleaving calls into the sdmmc chipset functions. --- sys/dev/sdmmc/sdmmc.c | 25 ++++++-- sys/dev/sdmmc/sdmmc_io.c | 131 +++++++++++++++++++++++++++----------- sys/dev/sdmmc/sdmmc_mem.c | 69 ++++++++++++++++++-- 3 files changed, 177 insertions(+), 48 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 34d120b8cfe3..2bef7cec9265 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -749,6 +749,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. @@ -801,6 +804,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)) { @@ -956,6 +963,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)) @@ -983,6 +993,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)) { @@ -1035,7 +1048,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; @@ -1069,7 +1082,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)) { @@ -1107,7 +1120,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; @@ -1126,7 +1139,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; @@ -1144,7 +1157,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, @@ -1177,7 +1190,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 2bac0e1c7008..a179ec4f3ecd 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,6 +77,9 @@ sdmmc_io_enable(struct sdmmc_softc *sc) uint32_t card_ocr; int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); /* Set host mode to SD "combo" card. */ @@ -146,6 +151,9 @@ sdmmc_io_scan(struct sdmmc_softc *sc) int error; int i; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); sf0 = sdmmc_function_alloc(sc); @@ -192,14 +200,17 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) int error = 0; uint8_t reg; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); 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 +234,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 +262,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"); @@ -346,7 +361,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 +409,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 +448,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 +507,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 +518,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 +529,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 +542,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 +561,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 +571,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 +590,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 +601,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 +621,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 +631,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 +651,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 +675,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,6 +693,9 @@ 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 */ @@ -665,7 +713,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 +754,15 @@ sdmmc_intr_enable(struct sdmmc_function *sf) struct sdmmc_function *sf0 = sc->sc_fn0; uint8_t reg; + mutex_enter(&sc->sc_mtx); + SDMMC_LOCK(sc); - reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_INTEN); + 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_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); SDMMC_UNLOCK(sc); + + mutex_exit(&sc->sc_mtx); } void @@ -720,11 +772,15 @@ sdmmc_intr_disable(struct sdmmc_function *sf) struct sdmmc_function *sf0 = sc->sc_fn0; uint8_t reg; + mutex_enter(&sc->sc_mtx); + SDMMC_LOCK(sc); - reg = sdmmc_io_read_1(sf0, SD_IO_CCCR_FN_INTEN); + 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_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); SDMMC_UNLOCK(sc); + + mutex_exit(&sc->sc_mtx); } /* @@ -821,9 +877,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 @@ -834,15 +889,16 @@ sdmmc_io_set_blocklen(struct sdmmc_function *sf, struct sdmmc_function *sf0 = sc->sc_fn0; int error = EINVAL; + mutex_enter(&sc->sc_mtx); SDMMC_LOCK(sc); 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; @@ -850,6 +906,7 @@ sdmmc_io_set_blocklen(struct sdmmc_function *sf, 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 aa74421e2bc2..774418522569 100644 --- a/sys/dev/sdmmc/sdmmc_mem.c +++ b/sys/dev/sdmmc/sdmmc_mem.c @@ -142,6 +142,9 @@ sdmmc_mem_enable(struct sdmmc_softc *sc) uint32_t ocr = 0; int error; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); /* Set host mode to SD "combo" card or SD memory-only. */ @@ -275,6 +278,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,6 +331,9 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) int error; int retry; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); /* @@ -442,6 +451,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 +522,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,6 +592,9 @@ sdmmc_mem_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) { int error = 0; + KASSERT(mutex_owned(&sc->sc_attach_mtx)); + KASSERT(mutex_owned(&sc->sc_mtx)); + SDMMC_LOCK(sc); if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { @@ -613,7 +631,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 +689,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 +720,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 +778,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 +818,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 +961,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 +1194,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 +1224,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 +1259,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 +1331,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 +1369,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 +1497,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 +1570,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 +1609,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 +1691,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 +1745,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 +1774,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 +1797,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 +1863,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) @@ -1951,6 +2005,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 +2028,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 +2095,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) From 12af35532b4ab0c2381beca622bb7c209d310377 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Fri, 26 Jan 2024 22:56:00 +0000 Subject: [PATCH 20/21] sdmmc: Nix SDMMC_LOCK and SDMMC_UNLOCK. These macros -- defined as noops -- weren't used consistently to make a coherent locking scheme, so there is no use in keeping them. No module ABI change because these macros expanded to nothing. --- sys/dev/sdmmc/sdmmc_io.c | 30 ++---------------------------- sys/dev/sdmmc/sdmmc_mem.c | 22 ---------------------- sys/dev/sdmmc/sdmmcvar.h | 3 --- 3 files changed, 2 insertions(+), 53 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc_io.c b/sys/dev/sdmmc/sdmmc_io.c index a179ec4f3ecd..25922e47470a 100644 --- a/sys/dev/sdmmc/sdmmc_io.c +++ b/sys/dev/sdmmc/sdmmc_io.c @@ -80,8 +80,6 @@ sdmmc_io_enable(struct sdmmc_softc *sc) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - /* Set host mode to SD "combo" card. */ SET(sc->sc_flags, SMF_SD_MODE|SMF_IO_MODE|SMF_MEM_MODE); @@ -135,8 +133,6 @@ sdmmc_io_enable(struct sdmmc_softc *sc) } out: - SDMMC_UNLOCK(sc); - return error; } @@ -154,8 +150,6 @@ sdmmc_io_scan(struct sdmmc_softc *sc) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - sf0 = sdmmc_function_alloc(sc); sf0->number = 0; error = sdmmc_set_relative_addr(sc, sf0); @@ -187,7 +181,7 @@ sdmmc_io_scan(struct sdmmc_softc *sc) } out: - SDMMC_UNLOCK(sc); + return; } /* @@ -203,8 +197,6 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - sf->blklen = sdmmc_chip_host_maxblklen(sc->sc_sct, sc->sc_sch); if (sf->number == 0) { @@ -286,8 +278,6 @@ sdmmc_io_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) } out: - SDMMC_UNLOCK(sc); - return error; } @@ -304,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; } @@ -321,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) @@ -347,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 @@ -698,7 +682,7 @@ sdmmc_io_reset(struct sdmmc_softc *sc) 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); } /* @@ -755,13 +739,9 @@ sdmmc_intr_enable(struct sdmmc_function *sf) uint8_t reg; mutex_enter(&sc->sc_mtx); - - SDMMC_LOCK(sc); reg = sdmmc_io_read_1_locked(sf0, SD_IO_CCCR_FN_INTEN); reg |= 1 << sf->number; sdmmc_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); - SDMMC_UNLOCK(sc); - mutex_exit(&sc->sc_mtx); } @@ -773,13 +753,9 @@ sdmmc_intr_disable(struct sdmmc_function *sf) uint8_t reg; mutex_enter(&sc->sc_mtx); - - SDMMC_LOCK(sc); reg = sdmmc_io_read_1_locked(sf0, SD_IO_CCCR_FN_INTEN); reg &= ~(1 << sf->number); sdmmc_io_write_1_locked(sf0, SD_IO_CCCR_FN_INTEN, reg); - SDMMC_UNLOCK(sc); - mutex_exit(&sc->sc_mtx); } @@ -890,7 +866,6 @@ sdmmc_io_set_blocklen(struct sdmmc_function *sf, int error = EINVAL; mutex_enter(&sc->sc_mtx); - SDMMC_LOCK(sc); if (blklen <= 0 || blklen > sdmmc_chip_host_maxblklen(sc->sc_sct, sc->sc_sch)) @@ -905,7 +880,6 @@ sdmmc_io_set_blocklen(struct sdmmc_function *sf, 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 774418522569..7841c2488f24 100644 --- a/sys/dev/sdmmc/sdmmc_mem.c +++ b/sys/dev/sdmmc/sdmmc_mem.c @@ -145,8 +145,6 @@ sdmmc_mem_enable(struct sdmmc_softc *sc) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - /* Set host mode to SD "combo" card or SD memory-only. */ CLR(sc->sc_flags, SMF_UHS_MODE); SET(sc->sc_flags, SMF_SD_MODE|SMF_MEM_MODE); @@ -268,8 +266,6 @@ mmc_mode: } out: - SDMMC_UNLOCK(sc); - return error; } @@ -334,8 +330,6 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - /* * CMD2 is a broadcast command understood by SD cards and MMC * cards. All cards begin to respond to the command, but back @@ -425,8 +419,6 @@ sdmmc_mem_scan(struct sdmmc_softc *sc) sdmmc_print_cid(&sf->cid); #endif } - - SDMMC_UNLOCK(sc); } int @@ -595,8 +587,6 @@ sdmmc_mem_init(struct sdmmc_softc *sc, struct sdmmc_function *sf) KASSERT(mutex_owned(&sc->sc_attach_mtx)); KASSERT(mutex_owned(&sc->sc_mtx)); - SDMMC_LOCK(sc); - if (!ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { error = sdmmc_select_card(sc, sf); if (error) @@ -616,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; } @@ -1938,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)) { @@ -1989,7 +1976,6 @@ unload: out: mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); return error; } @@ -2181,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) && @@ -2241,7 +2226,6 @@ unload: out: mutex_exit(&sc->sc_mtx); - SDMMC_UNLOCK(sc); return error; } @@ -2259,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 */ @@ -2296,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", @@ -2315,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 2e27340323d4..b4c95da3f0bb 100644 --- a/sys/dev/sdmmc/sdmmcvar.h +++ b/sys/dev/sdmmc/sdmmcvar.h @@ -329,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 From d04c09c568b5910bb895482322aaf3768fce0f15 Mon Sep 17 00:00:00 2001 From: Taylor R Campbell Date: Mon, 29 Jan 2024 03:22:43 +0000 Subject: [PATCH 21/21] sdmmc: Fix missing newline in console messages. --- sys/dev/sdmmc/sdmmc.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/sys/dev/sdmmc/sdmmc.c b/sys/dev/sdmmc/sdmmc.c index 2bef7cec9265..7f866cd11975 100644 --- a/sys/dev/sdmmc/sdmmc.c +++ b/sys/dev/sdmmc/sdmmc.c @@ -1194,7 +1194,7 @@ sdmmc_select_card(struct sdmmc_softc *sc, struct sdmmc_function *sf) if (ISSET(sc->sc_caps, SMC_CAPS_SPI_MODE)) { device_printf(sc->sc_dev, - "sdmmc_select_card: SMC_CAPS_SPI_MODE set"); + "sdmmc_select_card: SMC_CAPS_SPI_MODE set\n"); return EIO; } @@ -1214,7 +1214,7 @@ sdmmc_select_card(struct sdmmc_softc *sc, struct sdmmc_function *sf) if (error) { device_printf(sc->sc_dev, - "sdmmc_select_card: error %d", error); + "sdmmc_select_card: error %d\n", error); } return error;