From: Jit Loon Lim <jit.loon.lim@intel.com>
To: u-boot@lists.denx.de
Cc: Jagan Teki <jagan@amarulasolutions.com>,
Vignesh R <vigneshr@ti.com>, Marek <marex@denx.de>,
Simon <simon.k.r.goldschmidt@gmail.com>,
Tien Fong <tien.fong.chee@intel.com>,
Kok Kiang <kok.kiang.hea@intel.com>,
Raaj <raaj.lokanathan@intel.com>,
Dinesh <dinesh.maniyam@intel.com>,
Boon Khai <boon.khai.ng@intel.com>,
Alif <alif.zakuan.yuslaimi@intel.com>,
Teik Heng <teik.heng.chong@intel.com>,
Hazim <muhammad.hazim.izzat.zamri@intel.com>,
Jit Loon Lim <jit.loon.lim@intel.com>,
Sieu Mun Tang <sieu.mun.tang@intel.com>
Subject: [PATCH v1 11/17] drivers: mmc: add mmc/cadence driver for agilex5
Date: Wed, 21 Jun 2023 11:16:04 +0800 [thread overview]
Message-ID: <20230621031610.28401-12-jit.loon.lim@intel.com> (raw)
In-Reply-To: <20230621031610.28401-1-jit.loon.lim@intel.com>
This is for new platform enablement for agilex5.
Add mmc and cadence host driver for new platform.
Signed-off-by: Jit Loon Lim <jit.loon.lim@intel.com>
---
drivers/mmc/mmc.c | 27 +++---
drivers/mmc/sdhci-cadence.c | 164 ++++++++++++++++++++++++++++++++----
2 files changed, 160 insertions(+), 31 deletions(-)
diff --git a/drivers/mmc/mmc.c b/drivers/mmc/mmc.c
index 1af6af82e6..88c674d44f 100644
--- a/drivers/mmc/mmc.c
+++ b/drivers/mmc/mmc.c
@@ -26,6 +26,7 @@
#include <div64.h>
#include "mmc_private.h"
+#define TIMEOUT_TEN_MS 10
#define DEFAULT_CMD6_TIMEOUT_MS 500
static int mmc_set_signal_voltage(struct mmc *mmc, uint signal_voltage);
@@ -247,7 +248,7 @@ static int mmc_send_cmd_retry(struct mmc *mmc, struct mmc_cmd *cmd,
static int mmc_send_cmd_quirks(struct mmc *mmc, struct mmc_cmd *cmd,
struct mmc_data *data, u32 quirk, uint retries)
{
- if (IS_ENABLED(CONFIG_MMC_QUIRKS) && mmc->quirks & quirk)
+ if (CONFIG_IS_ENABLED(MMC_QUIRKS) && mmc->quirks & quirk)
return mmc_send_cmd_retry(mmc, cmd, data, retries);
else
return mmc_send_cmd(mmc, cmd, data);
@@ -597,6 +598,11 @@ static int sd_send_op_cond(struct mmc *mmc, bool uhs_en)
int err;
struct mmc_cmd cmd;
+ /* lower timeout, to speed up mmc init since both uses same flow */
+ if (IS_ENABLED(CONFIG_TARGET_SOCFPGA_AGILEX5_EMU) ||
+ IS_ENABLED(CONFIG_TARGET_SOCFPGA_AGILEX5_SIMICS))
+ timeout = TIMEOUT_TEN_MS;
+
while (1) {
cmd.cmdidx = MMC_CMD_APP_CMD;
cmd.resp_type = MMC_RSP_R1;
@@ -635,7 +641,7 @@ static int sd_send_op_cond(struct mmc *mmc, bool uhs_en)
break;
if (timeout-- <= 0)
- return -EOPNOTSUPP;
+ return -ETIMEDOUT;
udelay(1000);
}
@@ -2432,9 +2438,6 @@ static int mmc_startup_v4(struct mmc *mmc)
mmc->wr_rel_set = ext_csd[EXT_CSD_WR_REL_SET];
- mmc->can_trim =
- !!(ext_csd[EXT_CSD_SEC_FEATURE] & EXT_CSD_SEC_FEATURE_TRIM_EN);
-
return 0;
error:
if (mmc->ext_csd) {
@@ -3130,10 +3133,9 @@ int mmc_init_device(int num)
#endif
#ifdef CONFIG_CMD_BKOPS_ENABLE
-int mmc_set_bkops_enable(struct mmc *mmc, bool autobkops, bool enable)
+int mmc_set_bkops_enable(struct mmc *mmc)
{
int err;
- u32 bit = autobkops ? BIT(1) : BIT(0);
ALLOC_CACHE_ALIGN_BUFFER(u8, ext_csd, MMC_MAX_BLOCK_LEN);
err = mmc_send_ext_csd(mmc, ext_csd);
@@ -3147,21 +3149,18 @@ int mmc_set_bkops_enable(struct mmc *mmc, bool autobkops, bool enable)
return -EMEDIUMTYPE;
}
- if (enable && (ext_csd[EXT_CSD_BKOPS_EN] & bit)) {
+ if (ext_csd[EXT_CSD_BKOPS_EN] & 0x1) {
puts("Background operations already enabled\n");
return 0;
}
- err = mmc_switch(mmc, EXT_CSD_CMD_SET_NORMAL, EXT_CSD_BKOPS_EN,
- enable ? bit : 0);
+ err = mmc_switch(mmc, EXT_CSD_CMD_SET_NORMAL, EXT_CSD_BKOPS_EN, 1);
if (err) {
- printf("Failed to %sable manual background operations\n",
- enable ? "en" : "dis");
+ puts("Failed to enable manual background operations\n");
return err;
}
- printf("%sabled %s background operations\n",
- enable ? "En" : "Dis", autobkops ? "auto" : "manual");
+ puts("Enabled manual background operations\n");
return 0;
}
diff --git a/drivers/mmc/sdhci-cadence.c b/drivers/mmc/sdhci-cadence.c
index 327a05ad11..4154fd33da 100644
--- a/drivers/mmc/sdhci-cadence.c
+++ b/drivers/mmc/sdhci-cadence.c
@@ -5,7 +5,10 @@
*/
#include <common.h>
+#include <clk.h>
#include <dm.h>
+#include <generic-phy.h>
+#include <asm/arch/clock_manager.h>
#include <asm/global_data.h>
#include <dm/device_compat.h>
#include <linux/bitfield.h>
@@ -16,10 +19,15 @@
#include <linux/sizes.h>
#include <linux/libfdt.h>
#include <mmc.h>
+#include <reset-uclass.h>
#include <sdhci.h>
+/* General define */
+#define SD_MIN_CLK 400000
+
/* HRS - Host Register Set (specific to Cadence) */
#define SDHCI_CDNS_HRS04 0x10 /* PHY access port */
+#define SDHCI_CDNS_HRS05 0x14 /* PHY data access port */
#define SDHCI_CDNS_HRS04_ACK BIT(26)
#define SDHCI_CDNS_HRS04_RD BIT(25)
#define SDHCI_CDNS_HRS04_WR BIT(24)
@@ -66,6 +74,16 @@ struct sdhci_cdns_plat {
struct mmc_config cfg;
struct mmc mmc;
void __iomem *hrs_addr;
+ struct udevice *udev;
+ struct phy phy_dev;
+ bool phy_enabled;
+ struct reset_ctl softreset_ctl;
+};
+
+/* socfpga implementation specific driver private data */
+struct sdhci_socfpga_priv_data {
+ struct sdhci_host host;
+ struct phy phy;
};
struct sdhci_cdns_phy_cfg {
@@ -94,25 +112,45 @@ static int sdhci_cdns_write_phy_reg(struct sdhci_cdns_plat *plat,
u32 tmp;
int ret;
- tmp = FIELD_PREP(SDHCI_CDNS_HRS04_WDATA, data) |
- FIELD_PREP(SDHCI_CDNS_HRS04_ADDR, addr);
- writel(tmp, reg);
+ if (plat->phy_enabled) {
+ /* retrieve reg. addr */
+ tmp = FIELD_PREP(SDHCI_CDNS_HRS04_ADDR, addr);
- tmp |= SDHCI_CDNS_HRS04_WR;
- writel(tmp, reg);
+ ret = writel(tmp, reg);
+ debug("%s: register = 0x%08x\n", __func__, readl(reg));
- ret = readl_poll_timeout(reg, tmp, tmp & SDHCI_CDNS_HRS04_ACK, 10);
- if (ret)
- return ret;
+ /* read existing value, mask it */
+ reg = plat->hrs_addr + SDHCI_CDNS_HRS05;
+ tmp = readl(reg);
+ debug("%s: register = 0x%08x\n", __func__, readl(reg));
- tmp &= ~SDHCI_CDNS_HRS04_WR;
- writel(tmp, reg);
+ tmp &= ~data;
+ tmp |= data;
+
+ /* write operation */
+ ret = writel(tmp, reg);
+ debug("%s: register = 0x%08x\n", __func__, readl(reg));
+ } else {
+ tmp = FIELD_PREP(SDHCI_CDNS_HRS04_WDATA, data) |
+ FIELD_PREP(SDHCI_CDNS_HRS04_ADDR, addr);
+ writel(tmp, reg);
+
+ tmp |= SDHCI_CDNS_HRS04_WR;
+ writel(tmp, reg);
+
+ ret = readl_poll_timeout(reg, tmp, tmp & SDHCI_CDNS_HRS04_ACK, 10);
+ if (ret)
+ return ret;
+
+ tmp &= ~SDHCI_CDNS_HRS04_WR;
+ writel(tmp, reg);
+ }
return 0;
}
static int sdhci_cdns_phy_init(struct sdhci_cdns_plat *plat,
- const void *fdt, int nodeoffset)
+ const void *fdt, int nodeoffset)
{
const fdt32_t *prop;
int ret, i;
@@ -126,6 +164,7 @@ static int sdhci_cdns_phy_init(struct sdhci_cdns_plat *plat,
ret = sdhci_cdns_write_phy_reg(plat,
sdhci_cdns_phy_cfgs[i].addr,
fdt32_to_cpu(*prop));
+
if (ret)
return ret;
}
@@ -162,7 +201,14 @@ static void sdhci_cdns_set_control_reg(struct sdhci_host *host)
tmp = readl(plat->hrs_addr + SDHCI_CDNS_HRS06);
tmp &= ~SDHCI_CDNS_HRS06_MODE;
tmp |= FIELD_PREP(SDHCI_CDNS_HRS06_MODE, mode);
+
writel(tmp, plat->hrs_addr + SDHCI_CDNS_HRS06);
+ debug("%s: register = 0x%x\n", __func__,
+ readl(plat->hrs_addr + SDHCI_CDNS_HRS06));
+
+ /* program phy based on generated settings, input through device tree */
+ if (plat->phy_enabled)
+ generic_phy_configure(&plat->phy_dev, NULL);
}
static const struct sdhci_ops sdhci_cdns_ops = {
@@ -192,6 +238,8 @@ static int sdhci_cdns_set_tune_val(struct sdhci_cdns_plat *plat,
tmp |= SDHCI_CDNS_HRS06_TUNE_UP;
writel(tmp, reg);
+ debug("%s: register = 0x%08x\n", __func__, readl(reg));
+
ret = readl_poll_timeout(reg, tmp,
!(tmp & SDHCI_CDNS_HRS06_TUNE_UP), 1);
if (ret)
@@ -252,15 +300,49 @@ static int sdhci_cdns_bind(struct udevice *dev)
return sdhci_bind(dev, &plat->mmc, &plat->cfg);
}
+static int socfpga_sdhci_get_clk_rate(struct udevice *dev)
+{
+ struct sdhci_socfpga_priv_data *priv = dev_get_priv(dev);
+ struct sdhci_host *host = &priv->host;
+
+#if (IS_ENABLED(CONFIG_CLK))
+ struct clk clk;
+ int ret;
+
+ ret = clk_get_by_index(dev, 1, &clk);
+ if (ret)
+ return ret;
+
+ host->max_clk = clk_get_rate(&clk);
+
+ clk_free(&clk);
+#else
+ /* Fixed clock divide by 4 which due to the SDMMC wrapper */
+ host->max_clk = cm_get_mmc_controller_clk_hz();
+#endif
+
+ if (!host->max_clk) {
+ debug("SDHCI: MMC clock is zero!");
+ return -EINVAL;
+ }
+ debug("max_clk: %d\n", host->max_clk);
+
+ return 0;
+}
+
static int sdhci_cdns_probe(struct udevice *dev)
{
DECLARE_GLOBAL_DATA_PTR;
struct mmc_uclass_priv *upriv = dev_get_uclass_priv(dev);
struct sdhci_cdns_plat *plat = dev_get_plat(dev);
- struct sdhci_host *host = dev_get_priv(dev);
+ struct sdhci_socfpga_priv_data *priv = dev_get_priv(dev);
+ struct sdhci_host *host = &priv->host;
+ const char *phy_name = dev_read_string(dev, "phy-names");
fdt_addr_t base;
int ret;
+ plat->phy_enabled = false;
+
base = dev_read_addr(dev);
if (base == FDT_ADDR_T_NONE)
return -EINVAL;
@@ -269,6 +351,43 @@ static int sdhci_cdns_probe(struct udevice *dev)
if (!plat->hrs_addr)
return -ENOMEM;
+ if (!phy_name)
+ return -EINVAL;
+
+ /* get SDMMC softreset */
+ ret = reset_get_by_name(dev, "reset", &plat->softreset_ctl);
+ if (ret)
+ pr_err("can't get soft reset for %s (%d)", dev->name, ret);
+
+ /* assert & deassert softreset */
+ ret = reset_assert(&plat->softreset_ctl);
+ if (ret < 0) {
+ pr_err("SDMMC soft reset deassert failed: %d", ret);
+ return ret;
+ }
+
+ ret = reset_deassert(&plat->softreset_ctl);
+ if (ret < 0) {
+ pr_err("SDMMC soft reset deassert failed: %d", ret);
+ return ret;
+ }
+
+ /* probe ComboPHY */
+ ret = generic_phy_get_by_name(dev, "combo-phy", &plat->phy_dev);
+ if (ret) {
+ printf("ComboPHY probe failed: %d\n", ret);
+ return ret;
+ }
+ debug("ComboPHY probe success\n");
+
+ ret = generic_phy_init(&plat->phy_dev);
+ if (ret) {
+ printf("ComboPHY init failed: %d\n", ret);
+ return ret;
+ }
+ debug("ComboPHY init success\n");
+
+ plat->phy_enabled = true;
host->name = dev->name;
host->ioaddr = plat->hrs_addr + SDHCI_CDNS_SRS_BASE;
host->ops = &sdhci_cdns_ops;
@@ -282,18 +401,29 @@ static int sdhci_cdns_probe(struct udevice *dev)
if (ret)
return ret;
- ret = sdhci_cdns_phy_init(plat, gd->fdt_blob, dev_of_offset(dev));
+ /* get max clk */
+ ret = socfpga_sdhci_get_clk_rate(dev);
if (ret)
return ret;
- host->mmc = &plat->mmc;
- host->mmc->dev = dev;
- ret = sdhci_setup_cfg(&plat->cfg, host, 0, 0);
+ ret = sdhci_cdns_phy_init(plat, gd->fdt_blob, dev_of_offset(dev));
if (ret)
return ret;
+ host->mmc = &plat->mmc;
upriv->mmc = &plat->mmc;
host->mmc->priv = host;
+ host->mmc->dev = dev;
+
+#if (IS_ENABLED(CONFIG_BLK))
+ ret = sdhci_setup_cfg(&plat->cfg, host, host->max_clk, SD_MIN_CLK);
+ if (ret)
+ return ret;
+#else
+ ret = add_sdhci(host, host->max_clk, SD_MIN_CLK);
+ if (ret)
+ return ret;
+#endif
return sdhci_probe(dev);
}
@@ -310,7 +440,7 @@ U_BOOT_DRIVER(sdhci_cdns) = {
.of_match = sdhci_cdns_match,
.bind = sdhci_cdns_bind,
.probe = sdhci_cdns_probe,
- .priv_auto = sizeof(struct sdhci_host),
+ .priv_auto = sizeof(struct sdhci_socfpga_priv_data),
.plat_auto = sizeof(struct sdhci_cdns_plat),
.ops = &sdhci_cdns_mmc_ops,
};
--
2.26.2
next prev parent reply other threads:[~2023-06-21 3:18 UTC|newest]
Thread overview: 23+ messages / expand[flat|nested] mbox.gz Atom feed top
2023-06-21 3:15 [PATCH v1 00/17] Agilex5 Platform Enablement Jit Loon Lim
2023-06-21 3:15 ` [PATCH v1 01/17] arch: arm: update kconfig for new platform agilex5 Jit Loon Lim
2023-06-28 8:17 ` Chee, Tien Fong
2023-06-30 8:18 ` Lim, Jit Loon
2023-06-21 3:15 ` [PATCH v1 02/17] arch: arm: dts: add dts and dtsi " Jit Loon Lim
2023-06-28 9:13 ` Chee, Tien Fong
2023-06-30 8:29 ` Lim, Jit Loon
2023-06-21 3:15 ` [PATCH v1 03/17] arch: arm: mach-socfpga: add new platform agilex5 mach-socfpga enablement Jit Loon Lim
2023-06-28 9:49 ` Chee, Tien Fong
2023-06-21 3:15 ` [PATCH v1 04/17] arch: arm: mach-socfpga: include: mach: " Jit Loon Lim
2023-06-21 3:15 ` [PATCH v1 05/17] board: intel: add new platform agilex5 socdk Jit Loon Lim
2023-06-21 3:15 ` [PATCH v1 06/17] configs: add new platform agilex5 defconfig Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 07/17] doc: device-tree-bindings: misc: add secreg text file for agilex5 Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 08/17] drivers: ddr: altera: add ddr support " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 09/17] drivers: clk: altera: add clock " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 10/17] drivers: misc: update driver misc " Jit Loon Lim
2023-06-21 3:16 ` Jit Loon Lim [this message]
2023-06-21 3:16 ` [PATCH v1 12/17] drivers: phy: add combo phy driver " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 13/17] drivers: reset: add reset " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 14/17] drivers: sysreset: add system " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 15/17] drivers: watchdog: update watchdog " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 16/17] includes: add and update configuration " Jit Loon Lim
2023-06-21 3:16 ` [PATCH v1 17/17] tools: binman: update binman tool " Jit Loon Lim
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20230621031610.28401-12-jit.loon.lim@intel.com \
--to=jit.loon.lim@intel.com \
--cc=alif.zakuan.yuslaimi@intel.com \
--cc=boon.khai.ng@intel.com \
--cc=dinesh.maniyam@intel.com \
--cc=jagan@amarulasolutions.com \
--cc=kok.kiang.hea@intel.com \
--cc=marex@denx.de \
--cc=muhammad.hazim.izzat.zamri@intel.com \
--cc=raaj.lokanathan@intel.com \
--cc=sieu.mun.tang@intel.com \
--cc=simon.k.r.goldschmidt@gmail.com \
--cc=teik.heng.chong@intel.com \
--cc=tien.fong.chee@intel.com \
--cc=u-boot@lists.denx.de \
--cc=vigneshr@ti.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox