Hi Subhash,

> -----Original Message-----
> From: subha...@codeaurora.org [mailto:subha...@codeaurora.org]
> Sent: Thursday, March 10, 2011 7:28 PM
> To: Nath, Arindam; c...@laptop.org
> Cc: zhangfei....@gmail.com; prak...@marvell.com; linux-
> m...@vger.kernel.org; Su, Henry; Lu, Aaron; anath....@gmail.com
> Subject: RE: [PATCH v2 12/12] mmc: sdhci: add support for retuning mode
> 1
> 
> 
> 
> > -----Original Message-----
> > From: Arindam Nath [mailto:anath....@gmail.com] On Behalf Of Arindam
> > Nath
> > Sent: Friday, March 04, 2011 5:03 PM
> > To: c...@laptop.org
> > Cc: zhangfei....@gmail.com; prak...@marvell.com;
> > subha...@codeaurora.org; linux-mmc@vger.kernel.org; henry...@amd.com;
> > aaron...@amd.com; anath....@gmail.com; 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.n...@amd.com>
> > ---
> >  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 majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to