Hi Subhash, > -----Original Message----- > From: subhashj@xxxxxxxxxxxxxx [mailto:subhashj@xxxxxxxxxxxxxx] > Sent: Thursday, March 10, 2011 7:28 PM > To: Nath, Arindam; cjb@xxxxxxxxxx > Cc: zhangfei.gao@xxxxxxxxx; prakity@xxxxxxxxxxx; linux- > mmc@xxxxxxxxxxxxxxx; Su, Henry; Lu, Aaron; anath.amd@xxxxxxxxx > Subject: RE: [PATCH v2 12/12] mmc: sdhci: add support for retuning mode > 1 > > > > > -----Original Message----- > > From: Arindam Nath [mailto:anath.amd@xxxxxxxxx] On Behalf Of Arindam > > Nath > > Sent: Friday, March 04, 2011 5:03 PM > > To: cjb@xxxxxxxxxx > > Cc: zhangfei.gao@xxxxxxxxx; prakity@xxxxxxxxxxx; > > subhashj@xxxxxxxxxxxxxx; linux-mmc@xxxxxxxxxxxxxxx; henry.su@xxxxxxx; > > aaron.lu@xxxxxxx; anath.amd@xxxxxxxxx; Arindam Nath > > Subject: [PATCH v2 12/12] mmc: sdhci: add support for retuning mode 1 > > > > Host Controller v3.00 can support retuning modes 1,2 or 3 depending > on > > the bits 46-47 of the Capabilities register. Also, the timer count > for > > retuning is indicated by bits 40-43 of the same register. We > initialize > > timer_list for retuning after successfull UHS-I initialization. Since > > retuning mode 1 sets a limit of 4MB on the maximum data length, we > set > > max_blk_count appropriately. Once the tuning timer expires, we set > > SDHCI_NEEDS_RETUNING flag, and if the flag is set, we execute tuning > > procedure before sending the next command. We need to restore > > mmc_request > > structure after executing retuning procedure since host->mrq is used > > inside the procedure to send CMD19. We also disable and re-enable > this > > flag during suspend and resume respectively, as per the spec v3.00. > > > > Signed-off-by: Arindam Nath <arindam.nath@xxxxxxx> > > --- > > drivers/mmc/core/sd.c | 5 ++ > > drivers/mmc/host/sdhci.c | 109 > > ++++++++++++++++++++++++++++++++++++++++++++- > > drivers/mmc/host/sdhci.h | 6 ++- > > include/linux/mmc/host.h | 1 + > > include/linux/mmc/sdhci.h | 6 +++ > > 5 files changed, 124 insertions(+), 3 deletions(-) > > > > diff --git a/drivers/mmc/core/sd.c b/drivers/mmc/core/sd.c > > index ae7a771..1f3cf57 100644 > > --- a/drivers/mmc/core/sd.c > > +++ b/drivers/mmc/core/sd.c > > @@ -641,6 +641,11 @@ static int mmc_sd_init_uhs_card(struct mmc_card > > *card) > > if (!mmc_host_is_spi(card->host) && card->host->ops- > > >execute_tuning) > > card->host->ops->execute_tuning(card->host); > > > > + /* Initialize and start re-tuning timer */ > > + if (!mmc_host_is_spi(card->host) && > > + card->host->ops->start_retuning_timer) > > + card->host->ops->start_retuning_timer(card->host); > > Why you need extra mmc_ops for this? You can start the retuning timer > when > first time your driver's "execute_tuning" ops gets executed. Thanks for the comment. I agree with you. Will change it for next version. Arindam > > > + > > out: > > kfree(status); > > > > diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c > > index 2ffb0c4..8bf0408 100644 > > --- a/drivers/mmc/host/sdhci.c > > +++ b/drivers/mmc/host/sdhci.c > > @@ -48,6 +48,8 @@ static void sdhci_finish_data(struct sdhci_host *); > > > > static void sdhci_send_command(struct sdhci_host *, struct > mmc_command > > *); > > static void sdhci_finish_command(struct sdhci_host *); > > +static void sdhci_execute_tuning(struct mmc_host *mmc); > > +static void sdhci_tuning_timer(unsigned long data); > > > > static void sdhci_dumpregs(struct sdhci_host *host) > > { > > @@ -1240,8 +1242,34 @@ static void sdhci_request(struct mmc_host > *mmc, > > struct mmc_request *mrq) > > if (!present || host->flags & SDHCI_DEVICE_DEAD) { > > host->mrq->cmd->error = -ENOMEDIUM; > > tasklet_schedule(&host->finish_tasklet); > > - } else > > + } else { > > + u32 present_state; > > + > > + present_state = sdhci_readl(host, SDHCI_PRESENT_STATE); > > + /* > > + * Check if the re-tuning timer has already expired and > > there > > + * is no on-going data transfer. If so, we need to execute > > + * tuning procedure before sending command. > > + */ > > + if ((host->flags & SDHCI_NEEDS_RETUNING) && > > + !(present_state & (SDHCI_DOING_WRITE | > > + SDHCI_DOING_READ))) { > > + host->flags &= ~SDHCI_NEEDS_RETUNING; > > + /* Reload the new initial value for timer */ > > + if (host->tuning_mode == SDHCI_TUNING_MODE_1) > > + mod_timer(&host->tuning_timer, jiffies + > > + host->tuning_count * HZ); > > + > > + spin_unlock_irqrestore(&host->lock, flags); > > + sdhci_execute_tuning(mmc); > > + spin_lock_irqsave(&host->lock, flags); > > + > > + /* Restore original mmc_request structure */ > > + host->mrq = mrq; > > + } > > + > > sdhci_send_command(host, mrq->cmd); > > + } > > > > mmiowb(); > > spin_unlock_irqrestore(&host->lock, flags); > > @@ -1667,6 +1695,26 @@ static void sdhci_disable_preset_value(struct > > sdhci_host *host) > > sdhci_writew(host, ctrl, SDHCI_HOST_CONTROL2); > > } > > > > +static void sdhci_start_retuning_timer(struct mmc_host *mmc) > > +{ > > + struct sdhci_host *host; > > + unsigned long flags; > > + > > + host = mmc_priv(mmc); > > + > > + spin_lock_irqsave(&host->lock, flags); > > + > > + if (host->tuning_count && > > + (host->tuning_mode == SDHCI_TUNING_MODE_1)) { > > + mod_timer(&host->tuning_timer, jiffies + host->tuning_count > > * > > + HZ); > > + /* Tuning mode 1 limits the maximum data length to 4MB */ > > + mmc->max_blk_count = (4 * 1024 * 1024) / mmc->max_blk_size; > > + } > > + > > + spin_unlock_irqrestore(&host->lock, flags); > > +} > > + > > static const struct mmc_host_ops sdhci_ops = { > > .request = sdhci_request, > > .set_ios = sdhci_set_ios, > > @@ -1676,6 +1724,7 @@ static const struct mmc_host_ops sdhci_ops = { > > .get_max_current_180 = sdhci_get_max_current_180, > > .execute_tuning = sdhci_execute_tuning, > > .enable_preset_value = sdhci_enable_preset_value, > > + .start_retuning_timer = sdhci_start_retuning_timer, > > }; > > > > > > > /********************************************************************** > > *******\ > > @@ -1725,6 +1774,10 @@ static void sdhci_tasklet_finish(unsigned long > > param) > > > > del_timer(&host->timer); > > > > + if (host->version >= SDHCI_SPEC_300) > > + del_timer(&host->tuning_timer); > > + > > + > > mrq = host->mrq; > > > > /* > > @@ -1799,6 +1852,20 @@ static void sdhci_timeout_timer(unsigned long > > data) > > spin_unlock_irqrestore(&host->lock, flags); > > } > > > > +static void sdhci_tuning_timer(unsigned long data) > > +{ > > + struct sdhci_host *host; > > + unsigned long flags; > > + > > + host = (struct sdhci_host *)data; > > + > > + spin_lock_irqsave(&host->lock, flags); > > + > > + host->flags |= SDHCI_NEEDS_RETUNING; > > + > > + spin_unlock_irqrestore(&host->lock, flags); > > +} > > + > > > > > /********************************************************************** > > *******\ > > * > > * > > * Interrupt handling > > * > > @@ -2057,6 +2124,15 @@ int sdhci_suspend_host(struct sdhci_host > *host, > > pm_message_t state) > > > > sdhci_disable_card_detection(host); > > > > + /* Disable tuning since we are suspending */ > > + if ((host->version >= SDHCI_SPEC_300) && > > + host->tuning_count && > > + (host->tuning_mode == SDHCI_TUNING_MODE_1)) { > > + host->flags &= ~SDHCI_NEEDS_RETUNING; > > + mod_timer(&host->tuning_timer, jiffies + > > + host->tuning_count * HZ); > > + } > > + > > ret = mmc_suspend_host(host->mmc); > > if (ret) > > return ret; > > @@ -2098,6 +2174,12 @@ int sdhci_resume_host(struct sdhci_host *host) > > ret = mmc_resume_host(host->mmc); > > sdhci_enable_card_detection(host); > > > > + /* Set the re-tuning expiration flag */ > > + if ((host->version >= SDHCI_SPEC_300) && > > + host->tuning_count && > > + (host->tuning_mode == SDHCI_TUNING_MODE_1)) > > + host->flags |= SDHCI_NEEDS_RETUNING; > > + > > return ret; > > } > > > > @@ -2353,6 +2435,21 @@ int sdhci_add_host(struct sdhci_host *host) > > if (caps[1] & SDHCI_DRIVER_TYPE_D) > > mmc->caps |= MMC_CAP_DRIVER_TYPE_D; > > > > + /* Initial value for re-tuning timer count */ > > + host->tuning_count = (caps[1] & SDHCI_RETUNING_TIMER_COUNT_MASK) > > >> > > + SDHCI_RETUNING_TIMER_COUNT_SHIFT; > > + > > + /* > > + * In case Re-tuning Timer is not disabled, the actual value of > > + * re-tuning timer will be 2 ^ (n - 1). > > + */ > > + if (host->tuning_count) > > + host->tuning_count = 1 << (host->tuning_count - 1); > > + > > + /* Re-tuning mode supported by the Host Controller */ > > + host->tuning_mode = (caps[1] & SDHCI_RETUNING_MODE_MASK) >> > > + SDHCI_RETUNING_MODE_SHIFT; > > + > > ocr_avail = 0; > > /* > > * According to SD Host Controller spec v3.00, if the Host System > > @@ -2488,9 +2585,15 @@ int sdhci_add_host(struct sdhci_host *host) > > > > setup_timer(&host->timer, sdhci_timeout_timer, (unsigned > > long)host); > > > > - if (host->version >= SDHCI_SPEC_300) > > + if (host->version >= SDHCI_SPEC_300) { > > init_waitqueue_head(&host->buf_ready_int); > > > > + /* Initialize re-tuning timer */ > > + init_timer(&host->tuning_timer); > > + host->tuning_timer.data = (unsigned long)host; > > + host->tuning_timer.function = sdhci_tuning_timer; > > + } > > + > > ret = request_irq(host->irq, sdhci_irq, IRQF_SHARED, > > mmc_hostname(mmc), host); > > if (ret) > > @@ -2584,6 +2687,8 @@ void sdhci_remove_host(struct sdhci_host *host, > > int dead) > > free_irq(host->irq, host); > > > > del_timer_sync(&host->timer); > > + if (host->version >= SDHCI_SPEC_300) > > + del_timer_sync(&host->tuning_timer); > > > > tasklet_kill(&host->card_tasklet); > > tasklet_kill(&host->finish_tasklet); > > diff --git a/drivers/mmc/host/sdhci.h b/drivers/mmc/host/sdhci.h > > index 37a8c32..53d5909 100644 > > --- a/drivers/mmc/host/sdhci.h > > +++ b/drivers/mmc/host/sdhci.h > > @@ -190,7 +190,11 @@ > > #define SDHCI_DRIVER_TYPE_A 0x00000010 > > #define SDHCI_DRIVER_TYPE_C 0x00000020 > > #define SDHCI_DRIVER_TYPE_D 0x00000040 > > -#define SDHCI_USE_SDR50_TUNING 0x00002000 > > +#define SDHCI_RETUNING_TIMER_COUNT_MASK 0x00000F00 > > +#define SDHCI_RETUNING_TIMER_COUNT_SHIFT 8 > > +#define SDHCI_USE_SDR50_TUNING 0x00002000 > > +#define SDHCI_RETUNING_MODE_MASK 0x0000C000 > > +#define SDHCI_RETUNING_MODE_SHIFT 14 > > #define SDHCI_CLOCK_MUL_MASK 0x00FF0000 > > #define SDHCI_CLOCK_MUL_SHIFT 16 > > > > diff --git a/include/linux/mmc/host.h b/include/linux/mmc/host.h > > index e63e063..09f5d03 100644 > > --- a/include/linux/mmc/host.h > > +++ b/include/linux/mmc/host.h > > @@ -131,6 +131,7 @@ struct mmc_host_ops { > > int (*get_max_current_180)(struct mmc_host *mmc); > > void (*execute_tuning)(struct mmc_host *host); > > void (*enable_preset_value)(struct mmc_host *host); > > + void (*start_retuning_timer)(struct mmc_host *host); > > }; > > > > struct mmc_card; > > diff --git a/include/linux/mmc/sdhci.h b/include/linux/mmc/sdhci.h > > index 4be4022..c6539be 100644 > > --- a/include/linux/mmc/sdhci.h > > +++ b/include/linux/mmc/sdhci.h > > @@ -110,6 +110,7 @@ struct sdhci_host { > > #define SDHCI_REQ_USE_DMA (1<<2) /* Use DMA for this req. */ > > #define SDHCI_DEVICE_DEAD (1<<3) /* Device unresponsive */ > > #define SDHCI_SDR50_NEEDS_TUNING (1<<4) /* SDR50 needs tuning > */ > > +#define SDHCI_NEEDS_RETUNING (1<<5) /* Host needs retuning > */ > > > > unsigned int version; /* SDHCI spec. version */ > > > > @@ -152,6 +153,11 @@ struct sdhci_host { > > wait_queue_head_t buf_ready_int; /* Waitqueue for Buffer Read > > Ready interrupt */ > > unsigned int tuning_done; /* Condition flag set > > when CMD19 succeeds */ > > > > + unsigned int tuning_count; /* Timer count for re- > > tuning */ > > + unsigned int tuning_mode; /* Re-tuning mode > > supported by host */ > > +#define SDHCI_TUNING_MODE_1 0 > > + struct timer_list tuning_timer; /* Timer for tuning */ > > + > > unsigned long private[0] ____cacheline_aligned; > > }; > > #endif /* __SDHCI_H */ > > -- > > 1.7.1 > > -- To unsubscribe from this list: send the line "unsubscribe linux-mmc" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html