|
@@ -14,16 +14,34 @@
|
|
|
#include <linux/clk.h>
|
|
|
#include <linux/clk-provider.h>
|
|
|
#include <linux/device.h>
|
|
|
+#include <linux/gpio/consumer.h>
|
|
|
#include <linux/ethtool.h>
|
|
|
#include <linux/io.h>
|
|
|
+#include <linux/iopoll.h>
|
|
|
#include <linux/ioport.h>
|
|
|
#include <linux/module.h>
|
|
|
+#include <linux/of_device.h>
|
|
|
#include <linux/of_net.h>
|
|
|
#include <linux/mfd/syscon.h>
|
|
|
#include <linux/platform_device.h>
|
|
|
+#include <linux/reset.h>
|
|
|
#include <linux/stmmac.h>
|
|
|
|
|
|
#include "stmmac_platform.h"
|
|
|
+#include "dwmac4.h"
|
|
|
+
|
|
|
+struct tegra_eqos {
|
|
|
+ struct device *dev;
|
|
|
+ void __iomem *regs;
|
|
|
+
|
|
|
+ struct reset_control *rst;
|
|
|
+ struct clk *clk_master;
|
|
|
+ struct clk *clk_slave;
|
|
|
+ struct clk *clk_tx;
|
|
|
+ struct clk *clk_rx;
|
|
|
+
|
|
|
+ struct gpio_desc *reset;
|
|
|
+};
|
|
|
|
|
|
static int dwc_eth_dwmac_config_dt(struct platform_device *pdev,
|
|
|
struct plat_stmmacenet_data *plat_dat)
|
|
@@ -106,13 +124,309 @@ static int dwc_eth_dwmac_config_dt(struct platform_device *pdev,
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+static void *dwc_qos_probe(struct platform_device *pdev,
|
|
|
+ struct plat_stmmacenet_data *plat_dat,
|
|
|
+ struct stmmac_resources *stmmac_res)
|
|
|
+{
|
|
|
+ int err;
|
|
|
+
|
|
|
+ plat_dat->stmmac_clk = devm_clk_get(&pdev->dev, "apb_pclk");
|
|
|
+ if (IS_ERR(plat_dat->stmmac_clk)) {
|
|
|
+ dev_err(&pdev->dev, "apb_pclk clock not found.\n");
|
|
|
+ return ERR_CAST(plat_dat->stmmac_clk);
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_prepare_enable(plat_dat->stmmac_clk);
|
|
|
+ if (err < 0) {
|
|
|
+ dev_err(&pdev->dev, "failed to enable apb_pclk clock: %d\n",
|
|
|
+ err);
|
|
|
+ return ERR_PTR(err);
|
|
|
+ }
|
|
|
+
|
|
|
+ plat_dat->pclk = devm_clk_get(&pdev->dev, "phy_ref_clk");
|
|
|
+ if (IS_ERR(plat_dat->pclk)) {
|
|
|
+ dev_err(&pdev->dev, "phy_ref_clk clock not found.\n");
|
|
|
+ err = PTR_ERR(plat_dat->pclk);
|
|
|
+ goto disable;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_prepare_enable(plat_dat->pclk);
|
|
|
+ if (err < 0) {
|
|
|
+ dev_err(&pdev->dev, "failed to enable phy_ref clock: %d\n",
|
|
|
+ err);
|
|
|
+ goto disable;
|
|
|
+ }
|
|
|
+
|
|
|
+ return NULL;
|
|
|
+
|
|
|
+disable:
|
|
|
+ clk_disable_unprepare(plat_dat->stmmac_clk);
|
|
|
+ return ERR_PTR(err);
|
|
|
+}
|
|
|
+
|
|
|
+static int dwc_qos_remove(struct platform_device *pdev)
|
|
|
+{
|
|
|
+ struct net_device *ndev = platform_get_drvdata(pdev);
|
|
|
+ struct stmmac_priv *priv = netdev_priv(ndev);
|
|
|
+
|
|
|
+ clk_disable_unprepare(priv->plat->pclk);
|
|
|
+ clk_disable_unprepare(priv->plat->stmmac_clk);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+#define SDMEMCOMPPADCTRL 0x8800
|
|
|
+#define SDMEMCOMPPADCTRL_PAD_E_INPUT_OR_E_PWRD BIT(31)
|
|
|
+
|
|
|
+#define AUTO_CAL_CONFIG 0x8804
|
|
|
+#define AUTO_CAL_CONFIG_START BIT(31)
|
|
|
+#define AUTO_CAL_CONFIG_ENABLE BIT(29)
|
|
|
+
|
|
|
+#define AUTO_CAL_STATUS 0x880c
|
|
|
+#define AUTO_CAL_STATUS_ACTIVE BIT(31)
|
|
|
+
|
|
|
+static void tegra_eqos_fix_speed(void *priv, unsigned int speed)
|
|
|
+{
|
|
|
+ struct tegra_eqos *eqos = priv;
|
|
|
+ unsigned long rate = 125000000;
|
|
|
+ bool needs_calibration = false;
|
|
|
+ u32 value;
|
|
|
+ int err;
|
|
|
+
|
|
|
+ switch (speed) {
|
|
|
+ case SPEED_1000:
|
|
|
+ needs_calibration = true;
|
|
|
+ rate = 125000000;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case SPEED_100:
|
|
|
+ needs_calibration = true;
|
|
|
+ rate = 25000000;
|
|
|
+ break;
|
|
|
+
|
|
|
+ case SPEED_10:
|
|
|
+ rate = 2500000;
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ dev_err(eqos->dev, "invalid speed %u\n", speed);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (needs_calibration) {
|
|
|
+ /* calibrate */
|
|
|
+ value = readl(eqos->regs + SDMEMCOMPPADCTRL);
|
|
|
+ value |= SDMEMCOMPPADCTRL_PAD_E_INPUT_OR_E_PWRD;
|
|
|
+ writel(value, eqos->regs + SDMEMCOMPPADCTRL);
|
|
|
+
|
|
|
+ udelay(1);
|
|
|
+
|
|
|
+ value = readl(eqos->regs + AUTO_CAL_CONFIG);
|
|
|
+ value |= AUTO_CAL_CONFIG_START | AUTO_CAL_CONFIG_ENABLE;
|
|
|
+ writel(value, eqos->regs + AUTO_CAL_CONFIG);
|
|
|
+
|
|
|
+ err = readl_poll_timeout_atomic(eqos->regs + AUTO_CAL_STATUS,
|
|
|
+ value,
|
|
|
+ value & AUTO_CAL_STATUS_ACTIVE,
|
|
|
+ 1, 10);
|
|
|
+ if (err < 0) {
|
|
|
+ dev_err(eqos->dev, "calibration did not start\n");
|
|
|
+ goto failed;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = readl_poll_timeout_atomic(eqos->regs + AUTO_CAL_STATUS,
|
|
|
+ value,
|
|
|
+ (value & AUTO_CAL_STATUS_ACTIVE) == 0,
|
|
|
+ 20, 200);
|
|
|
+ if (err < 0) {
|
|
|
+ dev_err(eqos->dev, "calibration didn't finish\n");
|
|
|
+ goto failed;
|
|
|
+ }
|
|
|
+
|
|
|
+ failed:
|
|
|
+ value = readl(eqos->regs + SDMEMCOMPPADCTRL);
|
|
|
+ value &= ~SDMEMCOMPPADCTRL_PAD_E_INPUT_OR_E_PWRD;
|
|
|
+ writel(value, eqos->regs + SDMEMCOMPPADCTRL);
|
|
|
+ } else {
|
|
|
+ value = readl(eqos->regs + AUTO_CAL_CONFIG);
|
|
|
+ value &= ~AUTO_CAL_CONFIG_ENABLE;
|
|
|
+ writel(value, eqos->regs + AUTO_CAL_CONFIG);
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_set_rate(eqos->clk_tx, rate);
|
|
|
+ if (err < 0)
|
|
|
+ dev_err(eqos->dev, "failed to set TX rate: %d\n", err);
|
|
|
+}
|
|
|
+
|
|
|
+static int tegra_eqos_init(struct platform_device *pdev, void *priv)
|
|
|
+{
|
|
|
+ struct tegra_eqos *eqos = priv;
|
|
|
+ unsigned long rate;
|
|
|
+ u32 value;
|
|
|
+
|
|
|
+ rate = clk_get_rate(eqos->clk_slave);
|
|
|
+
|
|
|
+ value = (rate / 1000000) - 1;
|
|
|
+ writel(value, eqos->regs + GMAC_1US_TIC_COUNTER);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+static void *tegra_eqos_probe(struct platform_device *pdev,
|
|
|
+ struct plat_stmmacenet_data *data,
|
|
|
+ struct stmmac_resources *res)
|
|
|
+{
|
|
|
+ struct tegra_eqos *eqos;
|
|
|
+ int err;
|
|
|
+
|
|
|
+ eqos = devm_kzalloc(&pdev->dev, sizeof(*eqos), GFP_KERNEL);
|
|
|
+ if (!eqos) {
|
|
|
+ err = -ENOMEM;
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ eqos->dev = &pdev->dev;
|
|
|
+ eqos->regs = res->addr;
|
|
|
+
|
|
|
+ eqos->clk_master = devm_clk_get(&pdev->dev, "master_bus");
|
|
|
+ if (IS_ERR(eqos->clk_master)) {
|
|
|
+ err = PTR_ERR(eqos->clk_master);
|
|
|
+ goto error;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_prepare_enable(eqos->clk_master);
|
|
|
+ if (err < 0)
|
|
|
+ goto error;
|
|
|
+
|
|
|
+ eqos->clk_slave = devm_clk_get(&pdev->dev, "slave_bus");
|
|
|
+ if (IS_ERR(eqos->clk_slave)) {
|
|
|
+ err = PTR_ERR(eqos->clk_slave);
|
|
|
+ goto disable_master;
|
|
|
+ }
|
|
|
+
|
|
|
+ data->stmmac_clk = eqos->clk_slave;
|
|
|
+
|
|
|
+ err = clk_prepare_enable(eqos->clk_slave);
|
|
|
+ if (err < 0)
|
|
|
+ goto disable_master;
|
|
|
+
|
|
|
+ eqos->clk_rx = devm_clk_get(&pdev->dev, "rx");
|
|
|
+ if (IS_ERR(eqos->clk_rx)) {
|
|
|
+ err = PTR_ERR(eqos->clk_rx);
|
|
|
+ goto disable_slave;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_prepare_enable(eqos->clk_rx);
|
|
|
+ if (err < 0)
|
|
|
+ goto disable_slave;
|
|
|
+
|
|
|
+ eqos->clk_tx = devm_clk_get(&pdev->dev, "tx");
|
|
|
+ if (IS_ERR(eqos->clk_tx)) {
|
|
|
+ err = PTR_ERR(eqos->clk_tx);
|
|
|
+ goto disable_rx;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = clk_prepare_enable(eqos->clk_tx);
|
|
|
+ if (err < 0)
|
|
|
+ goto disable_rx;
|
|
|
+
|
|
|
+ eqos->reset = devm_gpiod_get(&pdev->dev, "phy-reset", GPIOD_OUT_HIGH);
|
|
|
+ if (IS_ERR(eqos->reset)) {
|
|
|
+ err = PTR_ERR(eqos->reset);
|
|
|
+ goto disable_tx;
|
|
|
+ }
|
|
|
+
|
|
|
+ usleep_range(2000, 4000);
|
|
|
+ gpiod_set_value(eqos->reset, 0);
|
|
|
+
|
|
|
+ eqos->rst = devm_reset_control_get(&pdev->dev, "eqos");
|
|
|
+ if (IS_ERR(eqos->rst)) {
|
|
|
+ err = PTR_ERR(eqos->rst);
|
|
|
+ goto reset_phy;
|
|
|
+ }
|
|
|
+
|
|
|
+ err = reset_control_assert(eqos->rst);
|
|
|
+ if (err < 0)
|
|
|
+ goto reset_phy;
|
|
|
+
|
|
|
+ usleep_range(2000, 4000);
|
|
|
+
|
|
|
+ err = reset_control_deassert(eqos->rst);
|
|
|
+ if (err < 0)
|
|
|
+ goto reset_phy;
|
|
|
+
|
|
|
+ usleep_range(2000, 4000);
|
|
|
+
|
|
|
+ data->fix_mac_speed = tegra_eqos_fix_speed;
|
|
|
+ data->init = tegra_eqos_init;
|
|
|
+ data->bsp_priv = eqos;
|
|
|
+
|
|
|
+ err = tegra_eqos_init(pdev, eqos);
|
|
|
+ if (err < 0)
|
|
|
+ goto reset;
|
|
|
+
|
|
|
+out:
|
|
|
+ return eqos;
|
|
|
+
|
|
|
+reset:
|
|
|
+ reset_control_assert(eqos->rst);
|
|
|
+reset_phy:
|
|
|
+ gpiod_set_value(eqos->reset, 1);
|
|
|
+disable_tx:
|
|
|
+ clk_disable_unprepare(eqos->clk_tx);
|
|
|
+disable_rx:
|
|
|
+ clk_disable_unprepare(eqos->clk_rx);
|
|
|
+disable_slave:
|
|
|
+ clk_disable_unprepare(eqos->clk_slave);
|
|
|
+disable_master:
|
|
|
+ clk_disable_unprepare(eqos->clk_master);
|
|
|
+error:
|
|
|
+ eqos = ERR_PTR(err);
|
|
|
+ goto out;
|
|
|
+}
|
|
|
+
|
|
|
+static int tegra_eqos_remove(struct platform_device *pdev)
|
|
|
+{
|
|
|
+ struct tegra_eqos *eqos = get_stmmac_bsp_priv(&pdev->dev);
|
|
|
+
|
|
|
+ reset_control_assert(eqos->rst);
|
|
|
+ gpiod_set_value(eqos->reset, 1);
|
|
|
+ clk_disable_unprepare(eqos->clk_tx);
|
|
|
+ clk_disable_unprepare(eqos->clk_rx);
|
|
|
+ clk_disable_unprepare(eqos->clk_slave);
|
|
|
+ clk_disable_unprepare(eqos->clk_master);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+struct dwc_eth_dwmac_data {
|
|
|
+ void *(*probe)(struct platform_device *pdev,
|
|
|
+ struct plat_stmmacenet_data *data,
|
|
|
+ struct stmmac_resources *res);
|
|
|
+ int (*remove)(struct platform_device *pdev);
|
|
|
+};
|
|
|
+
|
|
|
+static const struct dwc_eth_dwmac_data dwc_qos_data = {
|
|
|
+ .probe = dwc_qos_probe,
|
|
|
+ .remove = dwc_qos_remove,
|
|
|
+};
|
|
|
+
|
|
|
+static const struct dwc_eth_dwmac_data tegra_eqos_data = {
|
|
|
+ .probe = tegra_eqos_probe,
|
|
|
+ .remove = tegra_eqos_remove,
|
|
|
+};
|
|
|
+
|
|
|
static int dwc_eth_dwmac_probe(struct platform_device *pdev)
|
|
|
{
|
|
|
+ const struct dwc_eth_dwmac_data *data;
|
|
|
struct plat_stmmacenet_data *plat_dat;
|
|
|
struct stmmac_resources stmmac_res;
|
|
|
struct resource *res;
|
|
|
+ void *priv;
|
|
|
int ret;
|
|
|
|
|
|
+ data = of_device_get_match_data(&pdev->dev);
|
|
|
+
|
|
|
memset(&stmmac_res, 0, sizeof(struct stmmac_resources));
|
|
|
|
|
|
/**
|
|
@@ -138,39 +452,26 @@ static int dwc_eth_dwmac_probe(struct platform_device *pdev)
|
|
|
if (IS_ERR(plat_dat))
|
|
|
return PTR_ERR(plat_dat);
|
|
|
|
|
|
- plat_dat->stmmac_clk = devm_clk_get(&pdev->dev, "apb_pclk");
|
|
|
- if (IS_ERR(plat_dat->stmmac_clk)) {
|
|
|
- dev_err(&pdev->dev, "apb_pclk clock not found.\n");
|
|
|
- ret = PTR_ERR(plat_dat->stmmac_clk);
|
|
|
- plat_dat->stmmac_clk = NULL;
|
|
|
- goto err_remove_config_dt;
|
|
|
- }
|
|
|
- clk_prepare_enable(plat_dat->stmmac_clk);
|
|
|
-
|
|
|
- plat_dat->pclk = devm_clk_get(&pdev->dev, "phy_ref_clk");
|
|
|
- if (IS_ERR(plat_dat->pclk)) {
|
|
|
- dev_err(&pdev->dev, "phy_ref_clk clock not found.\n");
|
|
|
- ret = PTR_ERR(plat_dat->pclk);
|
|
|
- plat_dat->pclk = NULL;
|
|
|
- goto err_out_clk_dis_phy;
|
|
|
+ priv = data->probe(pdev, plat_dat, &stmmac_res);
|
|
|
+ if (IS_ERR(priv)) {
|
|
|
+ ret = PTR_ERR(priv);
|
|
|
+ dev_err(&pdev->dev, "failed to probe subdriver: %d\n", ret);
|
|
|
+ goto remove_config;
|
|
|
}
|
|
|
- clk_prepare_enable(plat_dat->pclk);
|
|
|
|
|
|
ret = dwc_eth_dwmac_config_dt(pdev, plat_dat);
|
|
|
if (ret)
|
|
|
- goto err_out_clk_dis_aper;
|
|
|
+ goto remove;
|
|
|
|
|
|
ret = stmmac_dvr_probe(&pdev->dev, plat_dat, &stmmac_res);
|
|
|
if (ret)
|
|
|
- goto err_out_clk_dis_aper;
|
|
|
+ goto remove;
|
|
|
|
|
|
- return 0;
|
|
|
+ return ret;
|
|
|
|
|
|
-err_out_clk_dis_aper:
|
|
|
- clk_disable_unprepare(plat_dat->pclk);
|
|
|
-err_out_clk_dis_phy:
|
|
|
- clk_disable_unprepare(plat_dat->stmmac_clk);
|
|
|
-err_remove_config_dt:
|
|
|
+remove:
|
|
|
+ data->remove(pdev);
|
|
|
+remove_config:
|
|
|
stmmac_remove_config_dt(pdev, plat_dat);
|
|
|
|
|
|
return ret;
|
|
@@ -178,11 +479,29 @@ err_remove_config_dt:
|
|
|
|
|
|
static int dwc_eth_dwmac_remove(struct platform_device *pdev)
|
|
|
{
|
|
|
- return stmmac_pltfr_remove(pdev);
|
|
|
+ struct net_device *ndev = platform_get_drvdata(pdev);
|
|
|
+ struct stmmac_priv *priv = netdev_priv(ndev);
|
|
|
+ const struct dwc_eth_dwmac_data *data;
|
|
|
+ int err;
|
|
|
+
|
|
|
+ data = of_device_get_match_data(&pdev->dev);
|
|
|
+
|
|
|
+ err = stmmac_dvr_remove(&pdev->dev);
|
|
|
+ if (err < 0)
|
|
|
+ dev_err(&pdev->dev, "failed to remove platform: %d\n", err);
|
|
|
+
|
|
|
+ err = data->remove(pdev);
|
|
|
+ if (err < 0)
|
|
|
+ dev_err(&pdev->dev, "failed to remove subdriver: %d\n", err);
|
|
|
+
|
|
|
+ stmmac_remove_config_dt(pdev, priv->plat);
|
|
|
+
|
|
|
+ return err;
|
|
|
}
|
|
|
|
|
|
static const struct of_device_id dwc_eth_dwmac_match[] = {
|
|
|
- { .compatible = "snps,dwc-qos-ethernet-4.10", },
|
|
|
+ { .compatible = "snps,dwc-qos-ethernet-4.10", .data = &dwc_qos_data },
|
|
|
+ { .compatible = "nvidia,tegra186-eqos", .data = &tegra_eqos_data },
|
|
|
{ }
|
|
|
};
|
|
|
MODULE_DEVICE_TABLE(of, dwc_eth_dwmac_match);
|