+ Adrian On Fri, 7 Jun 2019 at 18:05, Raul Rangel <rrangel@xxxxxxxxxxxx> wrote: > > On Tue, May 28, 2019 at 09:38:20AM +0200, Ulf Hansson wrote: > > On Wed, 1 May 2019 at 19:55, Raul E Rangel <rrangel@xxxxxxxxxxxx> wrote: > > First off, thanks for the review. > > > > > > > There is a race condition between resetting the SDHCI controller and > > > disconnecting the card. > > > > > > For example: > > > 0) Card is connected and transferring data > > > 1) mmc_sd_reset is called to reset the controller due to a data error > > > > I assume you refer to mmc_sd_hw_reset()? In that case, I think you > > have interpreted the purpose of mmc_sd_hw_reset() slightly wrong. It's > > responsibility is to reset the SD-card and not the host/controller. > You are correct. I was looking at a 4.14 kernel where it's called > mmc_sd_reset. 4.19 and above call it mmc_sd_hw_reset. > > All I was trying to convey here was that a block error will eventually > call sdhci_set_ios with SOFT_RESET_ALL via: > mmc_blk_reset > mmc_hw_reset > mmc_sd_hw_reset > mmc_power_cycle > mmc_power_off > mmc_set_initial_state > sdhci_set_ios > sdhci_reinit > sdhci_init > sdhci_do_reset(host, SDHCI_RESET_ALL); > > > > > Whether there some additional "reset" of the controller needed, that > > is assumed by the core, to be managed via the ->set_ios() callback for > > the host. > > > > > 2) sdhci_set_ios calls sdhci_do_reset > > > 3) SOFT_RESET_ALL is toggled which clears the IRQs the controller has > > > configured. > > > 4) Wait for SOFT_RESET_ALL to clear > > > 5) CD logic notices card is gone and CARD_PRESENT goes low, but since the > > > IRQs are not configured a CARD_REMOVED interrupt is never raised. > > > 6) IRQs are enabled again > > > 7) mmc layer never notices the device is disconnected. The SDHCI layer > > > will keep returning -ENOMEDIUM. This results in a card that is always > > > present and not functional. > > > > This sounds like host specific problems, which most likely should be > > fixed in host driver, solely. Unless I am missing something, of > > course. > > So in sdhci_do_reset we call the reset callback which is typically > sdhci_reset. sdhci_reset can wait for up to 100ms waiting for the > controller to reset. If SDHCI_RESET_ALL was passed as the flag, the > controller will clear the IRQ mask. If during that 100ms the card is > removed there is no notification to the MMC system that the card was > removed. So it seems like the card is always present. So you are saying that the problem occurs when the card is removed during this 100ms period? I assume there a way for sdhci to re-validate whether the card has been removed during this period, right? > > > > drivers/mmc/core/sd.c | 20 ++++++++++++++++++++ > > > 1 file changed, 20 insertions(+) > > > > > > diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c > > > index 265e1aeeb9d8..9206c4297d66 100644 > > > --- a/drivers/mmc/core/sd.c > > > +++ b/drivers/mmc/core/sd.c > > > @@ -1242,7 +1242,27 @@ static int mmc_sd_runtime_resume(struct mmc_host *host) > > > > > > static int mmc_sd_hw_reset(struct mmc_host *host) > > > { > > > + int present; > > > mmc_power_cycle(host, host->card->ocr); > > > + > > > + present = host->ops->get_cd(host); > > > + > > > + /* The card status could have changed while resetting. */ > > > + if ((mmc_card_removed(host->card) && present) || > > > + (!mmc_card_removed(host->card) && !present)) { > > > + pr_info("%s: card status changed during reset\n", > > > + mmc_hostname(host)); > > > + host->ops->card_event(host); > > > + mmc_detect_change(host, 0); > > > + } > > > + > > > + /* Don't perform unnecessary transactions if the card is missing. */ > > > + if (!present) { > > > + pr_info("%s: card was removed during reset\n", > > > + mmc_hostname(host)); > > > + return -ENOMEDIUM; > > > + } > > > + > > > > When doing a mmc_hw_reset() (which ends up calling mmc_sd_hw_reset() > > in case of SD cards), we are making a final attempt to make the card > > functional again, via a power cycle and a re-init of it. > > > > In this path, we don't care whether the card is removed, as that > > should have been detected already when the block layer calls > > mmc_detect_card_removed(). > > mmc_detect_card_removed has the guard `host->detect_change` to > prevent it from checking the card status again. So maybe the fix to the > missing interrupt/race condition is to set `host->detect_change = 1` > from sdhci_do_reset after calling the reset handler. This way there will > always be a single poll after a full reset so the correct card status can > be detected? It sounds like you should call mmc_detect_change(), after the reset to let the core check for cards that may have been removed/inserted. Would that work? [...] Kind regards Uffe