[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <20250910120734.2618011-1-yangzh0906@thundersoft.com>
Date: Wed, 10 Sep 2025 20:07:34 +0800
From: Albert Yang <yangzh0906@...ndersoft.com>
To: adrian.hunter@...el.com
Cc: arnd@...db.de,
gordon.ge@....ai,
linux-kernel@...r.kernel.org,
linux-mmc@...r.kernel.org,
robh@...nel.org,
ulf.hansson@...aro.org,
yangzh0906@...ndersoft.com
Subject: Re: [PATCH v3 5/8] mmc: sdhci: add Black Sesame Technologies BST C1200 controller driver
On Mon, Aug 18, 2025 at 09:16:55PM +0300, Adrian Hunter wrote:
> > diff --git a/drivers/mmc/host/sdhci-of-bst-c1200.c b/drivers/mmc/host/sdhci-of-bst-c1200.c
> > new file mode 100644
> > index 000000000000..6d2ba4232306
> > --- /dev/null
> > +++ b/drivers/mmc/host/sdhci-of-bst-c1200.c
>
> Unless you foresee more BST sdhci drivers, maybe sdhci-of-bst.c is an
> easier file name to deal with.
>
Thanks for the detailed review. I have addressed all the comments and renamed
the file to sdhci-of-bst.c as suggested.
> > +struct dwcmshc_priv {
>
> Name sdhci_bst_priv perhaps, see comment further below about
> names.
>
Renamed to sdhci_bst_priv.
> > + return ioread32(priv->crm_reg_base + offset);
>
> Are ioread32() / iowrite32() actually needed instead of readl() / writel()?
>
Changed to use readl() / writel() for consistency with standard kernel practices.
> > +static unsigned int bst_get_max_clock(struct sdhci_host *host)
> > +{
> > + return host->mmc->f_max;
> > +}
> > +
> > +static unsigned int bst_get_min_clock(struct sdhci_host *host)
> > +{
> > + return host->mmc->f_min;
>
> But what sets f_min? Should make sure it has a value.
>
Fixed by returning hardcoded values:
static unsigned int sdhci_bst_get_max_clock(struct sdhci_host *host)
{
return BST_DEFAULT_MAX_FREQ;
}
static unsigned int sdhci_bst_get_min_clock(struct sdhci_host *host)
{
return BST_DEFAULT_MIN_FREQ;
}
> > +}
> > +
> > +struct rx_ctrl {
>
> Looks like the intention is for this to be a union not a struct
>
Changed to union sdhci_bst_rx_ctrl as suggested.
> > + struct {
> > + u32 rx_revert:1;
> > + u32 rx_clk_sel_sec:1;
> > + u32 rx_clk_div:4;
> > + u32 rx_clk_phase_inner:2;
> > + u32 rx_clk_sel_first:1;
> > + u32 rx_clk_phase_out:2;
> > + u32 rx_clk_en:1;
> > + u32 res0:20;
> > + } bit;
>
> It isn't necessary for the struct to have a name, so like:
>
> union rx_ctrl {
> struct {
> u32 rx_revert:1,
> rx_clk_sel_sec:1,
> rx_clk_div:4,
> rx_clk_phase_inner:2,
> rx_clk_sel_first:1,
> rx_clk_phase_out:2,
> rx_clk_en:1,
> res0:20;
> };
> u32 reg;
> };
>
Updated as suggested.
> > + u32 reg;
> > +};
> > +
> > +struct sdmmc_iocfg {
>
> Not used
>
Removed the unused structure.
> > +static void sdhci_enable_bst_clk(struct sdhci_host *host, unsigned int clk)
>
> Function naming is a bit inconsistent. Please try to have
> a common prefix such as sdhci_bst, so for example
>
> sdhci_enable_bst_clk -> sdhci_bst_enable_clk
>
Updated all function names to use consistent sdhci_bst_ prefix.
> > + val = bst_crm_read(pltfm_host, SDEMMC_CRM_TIMER_DIV_CTRL);
> > + val &= ~BIT(8);
> > + bst_crm_write(pltfm_host, SDEMMC_CRM_TIMER_DIV_CTRL, val);
> > +
> > + val = bst_crm_read(pltfm_host, SDEMMC_CRM_TIMER_DIV_CTRL);
> > + val &= ~0xff;
> > + val |= 0x20;
>
> BIT() and other special values should be #define'd
> here and elsewhere
>
Added proper #define macros for all magic values:
#define BST_TIMER_LOAD_BIT BIT(8)
#define BST_TIMER_DIV_MASK GENMASK(7, 0)
#define BST_TIMER_DIV_VAL 0x20
> > +static void sdhci_set_bst_clock(struct sdhci_host *host, unsigned int clock)
>
> sdhci_bst_set_clock
>
> > +{
> > + if (clock == 0)
> > + return;
>
> The clock should be tuned off if it is 0. If there is a
> reason not to, then add a comment explaining.
>
Fixed to properly handle clock == 0 case by turning off clocks to save power:
if (!clock) {
clk_reg &= ~(SDHCI_CLOCK_CARD_EN | SDHCI_CLOCK_INT_EN | SDHCI_CLOCK_PLL_EN);
sdhci_writew(host, clk_reg, SDHCI_CLOCK_CONTROL);
return;
}
> > + sdhci_writew(host,
> > + sdhci_readw(host, emmc_ctrl_reg) & (~BIT(2)),
> > + emmc_ctrl_reg);
>
> Should #define BIT(2). Also read, update, write seems
> more readable e.g.
>
> reg = sdhci_readw(host, emmc_ctrl_reg);
> reg &= ~WHATEVER_IS_BIT_2;
> sdhci_writew(host, reg, emmc_ctrl_reg);
>
Updated to use read-modify-write pattern with proper #define:
reg = sdhci_readw(host, emmc_ctrl_reg);
reg &= ~BST_EMMC_CTRL_BIT2;
sdhci_writew(host, reg, emmc_ctrl_reg);
> > + sdhci_writew(host,
> > + (sdhci_readw(host, MBIU_CTRL) & (~0xf)) | BURST_EN,
> > + MBIU_CTRL);
>
> Doesn't look like it caters for mode == MMC_POWER_OFF
>
Added proper power off handling and improved code structure:
if (mode == MMC_POWER_OFF) {
/* Disable MBIU burst mode and turn off power supplies */
reg = sdhci_readw(host, MBIU_CTRL);
reg &= ~BURST_EN;
sdhci_writew(host, reg, MBIU_CTRL);
/* ... additional power off sequence ... */
} else {
/* Configure burst mode only when powered on */
reg = sdhci_readw(host, MBIU_CTRL);
reg &= ~MBIU_BURST_MASK;
reg |= BURST_EN;
sdhci_writew(host, reg, MBIU_CTRL);
}
> > +static int bst_sdhci_execute_tuning(struct sdhci_host *host, u32 opcode)
>
> sdhci_bst_execute_tuning
Renamed to sdhci_bst_execute_tuning.
>
> > + timeout = 20;
> > + while (!((clk = sdhci_readw(host, SDHCI_CLOCK_CONTROL)) &
> > + SDHCI_CLOCK_INT_STABLE)) {
> > + if (timeout == 0) {
> > + dev_err(mmc_dev(host->mmc), "Internal clock never stabilised\n");
> > + return -EBUSY;
> > + }
> > + timeout--;
> > + usleep_range(1000, 1100);
> > + }
>
> As Ulf already mentioned, read_poll_timeout() can be used e.g.
>
> if (read_poll_timeout(sdhci_readw, clock, (clock & SDHCI_CLOCK_INT_STABLE),
> 1000, 1100, false, host, SDHCI_CLOCK_CONTROL)) {
> dev_err(mmc_dev(host->mmc), "Internal clock never stabilised\n");
> return -EBUSY;
> }
>
>
> Same code as above, maybe make it a separate function.
>
Created a separate function sdhci_bst_wait_int_clk() using read_poll_timeout():
static int sdhci_bst_wait_int_clk(struct sdhci_host *host)
{
u16 clk;
if (read_poll_timeout(sdhci_readw, clk, (clk & SDHCI_CLOCK_INT_STABLE),
BST_CLK_STABLE_POLL_US, BST_CLK_STABLE_TIMEOUT_US, false,
host, SDHCI_CLOCK_CONTROL))
return -EBUSY;
return 0;
}
And replaced all instances with:
if (sdhci_bst_wait_int_clk(host)) {
dev_err(mmc_dev(host->mmc), "Internal clock never stabilised\n");
return -EBUSY;
}
> > +static const struct sdhci_ops sdhci_dwcmshc_ops = {
>
> sdhci_bst_ops
>
> > +static const struct sdhci_pltfm_data sdhci_dwcmshc_pdata = {
>
> sdhci_bst_pdata
Updated both structure names as suggested.
> > +static int bst_sdhci_reallocate_bounce_buffer(struct sdhci_host *host)
>
> sdhci_bst_reallocate_bounce_buffer
>
Renamed as suggested.
> > + /*
> > + * Cap the bounce buffer at 32KB. Using a bigger bounce buffer
> > + * has diminishing returns, this is probably because SD/MMC
> > + * cards are usually optimized to handle this size of requests.
> > + */
>
> That comment is copied from sdhci.c and makes less sense here.
> Presumably the size is fixed by hardware. Probably better
> to leave out the comment.
>
> > + bounce_size = SZ_32K;
> > + /*
> > + * Adjust downwards to maximum request size if this is less
> > + * than our segment size, else hammer down the maximum
> > + * request size to the maximum buffer size.
> > + */
> > + if (mmc->max_req_size < bounce_size)
> > + bounce_size = mmc->max_req_size;
>
> Similarly, 32K is your max request size, so there is no need
> of that logic or comment.
Replaced the generic comment with hardware-specific explanation.
> > +
> > + /* Lie about this since we're bouncing */
> > + mmc->max_segs = max_blocks;
> > + mmc->max_seg_size = bounce_size;
> > + mmc->max_req_size = bounce_size;
>
> If you make the change I suggest below to sdhci.c then
> the above 4 lines won't be needed.
>
Removed the unnecessary logic and the mmc->max_* assignments as suggested.
> > +static int dwcmshc_probe(struct platform_device *pdev)
>
> sdhci_bst_probe
>
> > + host = sdhci_pltfm_init(pdev, &sdhci_dwcmshc_pdata,
> > + sizeof(struct dwcmshc_priv));
>
> It is ok to use up to 100 columns, so line wrapping is not needed
> here.
>
Fixed line wrapping.
> > + err = bst_sdhci_reallocate_bounce_buffer(host);
> > + if (err) {
> > + dev_err(&pdev->dev, "Failed to allocate bounce buffer: %d\n", err);
> > + goto err_remove_host;
> > + }
>
> This would normally need to be done after sdhci_setup_host() and
> before __sdhci_add_host() because adding the host starts it.
>
> However, I would prefer to alter sdhci.c to allow it to be done
> before sdhci_add_host().
>
> Please make a separate patch for the change below, and then do
> the bounce buffer allocation before calling sdhci_add_host.
>
I will create a separate patch for the sdhci.c modification as you suggested, and
then move the bounce buffer allocation before calling sdhci_add_host.
> > +err_remove_host:
> > + sdhci_remove_host(host, 1);
> > +err:
> > + sdhci_pltfm_free(pdev);
>
> There is no sdhci_pltfm_free() anymore.
>
removed sdhci_pltfm_free() call.
> > +static void dwcmshc_remove(struct platform_device *pdev)
>
> sdhci_bst_remove
Updated to use sdhci_bst_remove
>
> > + /* Release reserved memory */
> > + of_reserved_mem_device_release(mmc_dev(host->mmc));
> > +
> > + sdhci_remove_host(host, 0);
>
> Because sdhci_pltfm_init() was used, sdhci_pltfm_remove() shoud be
> used here not sdhci_remove_host(host, 0) directly.
Fixed to use sdhci_pltfm_remove() in remove function.
> > + sdhci_pltfm_free(pdev);
> > +}
> > +
> > +static const struct of_device_id sdhci_dwcmshc_dt_ids[] = {
>
> sdhci_bst_ids
>
> > +
> > +static struct platform_driver sdhci_dwcmshc_driver = {
>
> sdhci_bst_driver
>
> > + .driver = {
> > + .name = "sdhci-dwcmshc",
>
> "sdhci-dwcmshc" has been used. Maybe "sdhci-bst"
Updated remove function to use sdhci_pltfm_remove() and renamed all remaining
structures to use sdhci_bst_ prefix. Changed driver name to "sdhci-bst".
Thank you for the comprehensive review. All the suggested changes have been
implemented. I will submit the sdhci.c patch separately as discussed.
Best Regards,
Albert
Powered by blists - more mailing lists