18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Rockchip PCIe PHY driver
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Copyright (C) 2016 Shawn Lin <shawn.lin@rock-chips.com>
68c2ecf20Sopenharmony_ci * Copyright (C) 2016 ROCKCHIP, Inc.
78c2ecf20Sopenharmony_ci */
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci#include <linux/clk.h>
108c2ecf20Sopenharmony_ci#include <linux/delay.h>
118c2ecf20Sopenharmony_ci#include <linux/io.h>
128c2ecf20Sopenharmony_ci#include <linux/mfd/syscon.h>
138c2ecf20Sopenharmony_ci#include <linux/module.h>
148c2ecf20Sopenharmony_ci#include <linux/of.h>
158c2ecf20Sopenharmony_ci#include <linux/of_address.h>
168c2ecf20Sopenharmony_ci#include <linux/of_platform.h>
178c2ecf20Sopenharmony_ci#include <linux/phy/phy.h>
188c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
198c2ecf20Sopenharmony_ci#include <linux/regmap.h>
208c2ecf20Sopenharmony_ci#include <linux/reset.h>
218c2ecf20Sopenharmony_ci
228c2ecf20Sopenharmony_ci/*
238c2ecf20Sopenharmony_ci * The higher 16-bit of this register is used for write protection
248c2ecf20Sopenharmony_ci * only if BIT(x + 16) set to 1 the BIT(x) can be written.
258c2ecf20Sopenharmony_ci */
268c2ecf20Sopenharmony_ci#define HIWORD_UPDATE(val, mask, shift) \
278c2ecf20Sopenharmony_ci		((val) << (shift) | (mask) << ((shift) + 16))
288c2ecf20Sopenharmony_ci
298c2ecf20Sopenharmony_ci#define PHY_MAX_LANE_NUM      4
308c2ecf20Sopenharmony_ci#define PHY_CFG_DATA_SHIFT    7
318c2ecf20Sopenharmony_ci#define PHY_CFG_ADDR_SHIFT    1
328c2ecf20Sopenharmony_ci#define PHY_CFG_DATA_MASK     0xf
338c2ecf20Sopenharmony_ci#define PHY_CFG_ADDR_MASK     0x3f
348c2ecf20Sopenharmony_ci#define PHY_CFG_RD_MASK       0x3ff
358c2ecf20Sopenharmony_ci#define PHY_CFG_WR_ENABLE     1
368c2ecf20Sopenharmony_ci#define PHY_CFG_WR_DISABLE    1
378c2ecf20Sopenharmony_ci#define PHY_CFG_WR_SHIFT      0
388c2ecf20Sopenharmony_ci#define PHY_CFG_WR_MASK       1
398c2ecf20Sopenharmony_ci#define PHY_CFG_PLL_LOCK      0x10
408c2ecf20Sopenharmony_ci#define PHY_CFG_CLK_TEST      0x10
418c2ecf20Sopenharmony_ci#define PHY_CFG_CLK_SCC       0x12
428c2ecf20Sopenharmony_ci#define PHY_CFG_SEPE_RATE     BIT(3)
438c2ecf20Sopenharmony_ci#define PHY_CFG_PLL_100M      BIT(3)
448c2ecf20Sopenharmony_ci#define PHY_PLL_LOCKED        BIT(9)
458c2ecf20Sopenharmony_ci#define PHY_PLL_OUTPUT        BIT(10)
468c2ecf20Sopenharmony_ci#define PHY_LANE_A_STATUS     0x30
478c2ecf20Sopenharmony_ci#define PHY_LANE_B_STATUS     0x31
488c2ecf20Sopenharmony_ci#define PHY_LANE_C_STATUS     0x32
498c2ecf20Sopenharmony_ci#define PHY_LANE_D_STATUS     0x33
508c2ecf20Sopenharmony_ci#define PHY_LANE_RX_DET_SHIFT 11
518c2ecf20Sopenharmony_ci#define PHY_LANE_RX_DET_TH    0x1
528c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_OFF     0x1
538c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_MASK    0x1
548c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_A_SHIFT 3
558c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_B_SHIFT 4
568c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_C_SHIFT 5
578c2ecf20Sopenharmony_ci#define PHY_LANE_IDLE_D_SHIFT 6
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_cistruct rockchip_pcie_data {
608c2ecf20Sopenharmony_ci	unsigned int pcie_conf;
618c2ecf20Sopenharmony_ci	unsigned int pcie_status;
628c2ecf20Sopenharmony_ci	unsigned int pcie_laneoff;
638c2ecf20Sopenharmony_ci};
648c2ecf20Sopenharmony_ci
658c2ecf20Sopenharmony_cistruct rockchip_pcie_phy {
668c2ecf20Sopenharmony_ci	struct rockchip_pcie_data *phy_data;
678c2ecf20Sopenharmony_ci	struct regmap *reg_base;
688c2ecf20Sopenharmony_ci	struct phy_pcie_instance {
698c2ecf20Sopenharmony_ci		struct phy *phy;
708c2ecf20Sopenharmony_ci		u32 index;
718c2ecf20Sopenharmony_ci	} phys[PHY_MAX_LANE_NUM];
728c2ecf20Sopenharmony_ci	struct mutex pcie_mutex;
738c2ecf20Sopenharmony_ci	struct reset_control *phy_rst;
748c2ecf20Sopenharmony_ci	struct clk *clk_pciephy_ref;
758c2ecf20Sopenharmony_ci	int pwr_cnt;
768c2ecf20Sopenharmony_ci	int init_cnt;
778c2ecf20Sopenharmony_ci};
788c2ecf20Sopenharmony_ci
798c2ecf20Sopenharmony_cistatic struct rockchip_pcie_phy *to_pcie_phy(struct phy_pcie_instance *inst)
808c2ecf20Sopenharmony_ci{
818c2ecf20Sopenharmony_ci	return container_of(inst, struct rockchip_pcie_phy,
828c2ecf20Sopenharmony_ci					phys[inst->index]);
838c2ecf20Sopenharmony_ci}
848c2ecf20Sopenharmony_ci
858c2ecf20Sopenharmony_cistatic struct phy *rockchip_pcie_phy_of_xlate(struct device *dev,
868c2ecf20Sopenharmony_ci					      struct of_phandle_args *args)
878c2ecf20Sopenharmony_ci{
888c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy = dev_get_drvdata(dev);
898c2ecf20Sopenharmony_ci
908c2ecf20Sopenharmony_ci	if (args->args_count == 0)
918c2ecf20Sopenharmony_ci		return rk_phy->phys[0].phy;
928c2ecf20Sopenharmony_ci
938c2ecf20Sopenharmony_ci	if (WARN_ON(args->args[0] >= PHY_MAX_LANE_NUM))
948c2ecf20Sopenharmony_ci		return ERR_PTR(-ENODEV);
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_ci	return rk_phy->phys[args->args[0]].phy;
978c2ecf20Sopenharmony_ci}
988c2ecf20Sopenharmony_ci
998c2ecf20Sopenharmony_ci
1008c2ecf20Sopenharmony_cistatic inline void phy_wr_cfg(struct rockchip_pcie_phy *rk_phy,
1018c2ecf20Sopenharmony_ci			      u32 addr, u32 data)
1028c2ecf20Sopenharmony_ci{
1038c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
1048c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(data,
1058c2ecf20Sopenharmony_ci				   PHY_CFG_DATA_MASK,
1068c2ecf20Sopenharmony_ci				   PHY_CFG_DATA_SHIFT) |
1078c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(addr,
1088c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_MASK,
1098c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_SHIFT));
1108c2ecf20Sopenharmony_ci	udelay(1);
1118c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
1128c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(PHY_CFG_WR_ENABLE,
1138c2ecf20Sopenharmony_ci				   PHY_CFG_WR_MASK,
1148c2ecf20Sopenharmony_ci				   PHY_CFG_WR_SHIFT));
1158c2ecf20Sopenharmony_ci	udelay(1);
1168c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
1178c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(PHY_CFG_WR_DISABLE,
1188c2ecf20Sopenharmony_ci				   PHY_CFG_WR_MASK,
1198c2ecf20Sopenharmony_ci				   PHY_CFG_WR_SHIFT));
1208c2ecf20Sopenharmony_ci}
1218c2ecf20Sopenharmony_ci
1228c2ecf20Sopenharmony_cistatic inline u32 phy_rd_cfg(struct rockchip_pcie_phy *rk_phy,
1238c2ecf20Sopenharmony_ci			     u32 addr)
1248c2ecf20Sopenharmony_ci{
1258c2ecf20Sopenharmony_ci	u32 val;
1268c2ecf20Sopenharmony_ci
1278c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
1288c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(addr,
1298c2ecf20Sopenharmony_ci				   PHY_CFG_RD_MASK,
1308c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_SHIFT));
1318c2ecf20Sopenharmony_ci	regmap_read(rk_phy->reg_base,
1328c2ecf20Sopenharmony_ci		    rk_phy->phy_data->pcie_status,
1338c2ecf20Sopenharmony_ci		    &val);
1348c2ecf20Sopenharmony_ci	return val;
1358c2ecf20Sopenharmony_ci}
1368c2ecf20Sopenharmony_ci
1378c2ecf20Sopenharmony_cistatic int rockchip_pcie_phy_power_off(struct phy *phy)
1388c2ecf20Sopenharmony_ci{
1398c2ecf20Sopenharmony_ci	struct phy_pcie_instance *inst = phy_get_drvdata(phy);
1408c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy = to_pcie_phy(inst);
1418c2ecf20Sopenharmony_ci	int err = 0;
1428c2ecf20Sopenharmony_ci
1438c2ecf20Sopenharmony_ci	mutex_lock(&rk_phy->pcie_mutex);
1448c2ecf20Sopenharmony_ci
1458c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base,
1468c2ecf20Sopenharmony_ci		     rk_phy->phy_data->pcie_laneoff,
1478c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(PHY_LANE_IDLE_OFF,
1488c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_MASK,
1498c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_A_SHIFT + inst->index));
1508c2ecf20Sopenharmony_ci
1518c2ecf20Sopenharmony_ci	if (--rk_phy->pwr_cnt)
1528c2ecf20Sopenharmony_ci		goto err_out;
1538c2ecf20Sopenharmony_ci
1548c2ecf20Sopenharmony_ci	err = reset_control_assert(rk_phy->phy_rst);
1558c2ecf20Sopenharmony_ci	if (err) {
1568c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "assert phy_rst err %d\n", err);
1578c2ecf20Sopenharmony_ci		goto err_restore;
1588c2ecf20Sopenharmony_ci	}
1598c2ecf20Sopenharmony_ci
1608c2ecf20Sopenharmony_cierr_out:
1618c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
1628c2ecf20Sopenharmony_ci	return 0;
1638c2ecf20Sopenharmony_ci
1648c2ecf20Sopenharmony_cierr_restore:
1658c2ecf20Sopenharmony_ci	rk_phy->pwr_cnt++;
1668c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base,
1678c2ecf20Sopenharmony_ci		     rk_phy->phy_data->pcie_laneoff,
1688c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(!PHY_LANE_IDLE_OFF,
1698c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_MASK,
1708c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_A_SHIFT + inst->index));
1718c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
1728c2ecf20Sopenharmony_ci	return err;
1738c2ecf20Sopenharmony_ci}
1748c2ecf20Sopenharmony_ci
1758c2ecf20Sopenharmony_cistatic int rockchip_pcie_phy_power_on(struct phy *phy)
1768c2ecf20Sopenharmony_ci{
1778c2ecf20Sopenharmony_ci	struct phy_pcie_instance *inst = phy_get_drvdata(phy);
1788c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy = to_pcie_phy(inst);
1798c2ecf20Sopenharmony_ci	int err = 0;
1808c2ecf20Sopenharmony_ci	u32 status;
1818c2ecf20Sopenharmony_ci	unsigned long timeout;
1828c2ecf20Sopenharmony_ci
1838c2ecf20Sopenharmony_ci	mutex_lock(&rk_phy->pcie_mutex);
1848c2ecf20Sopenharmony_ci
1858c2ecf20Sopenharmony_ci	if (rk_phy->pwr_cnt++)
1868c2ecf20Sopenharmony_ci		goto err_out;
1878c2ecf20Sopenharmony_ci
1888c2ecf20Sopenharmony_ci	err = reset_control_deassert(rk_phy->phy_rst);
1898c2ecf20Sopenharmony_ci	if (err) {
1908c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "deassert phy_rst err %d\n", err);
1918c2ecf20Sopenharmony_ci		goto err_pwr_cnt;
1928c2ecf20Sopenharmony_ci	}
1938c2ecf20Sopenharmony_ci
1948c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
1958c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(PHY_CFG_PLL_LOCK,
1968c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_MASK,
1978c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_SHIFT));
1988c2ecf20Sopenharmony_ci
1998c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base,
2008c2ecf20Sopenharmony_ci		     rk_phy->phy_data->pcie_laneoff,
2018c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(!PHY_LANE_IDLE_OFF,
2028c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_MASK,
2038c2ecf20Sopenharmony_ci				   PHY_LANE_IDLE_A_SHIFT + inst->index));
2048c2ecf20Sopenharmony_ci
2058c2ecf20Sopenharmony_ci	/*
2068c2ecf20Sopenharmony_ci	 * No documented timeout value for phy operation below,
2078c2ecf20Sopenharmony_ci	 * so we make it large enough here. And we use loop-break
2088c2ecf20Sopenharmony_ci	 * method which should not be harmful.
2098c2ecf20Sopenharmony_ci	 */
2108c2ecf20Sopenharmony_ci	timeout = jiffies + msecs_to_jiffies(1000);
2118c2ecf20Sopenharmony_ci
2128c2ecf20Sopenharmony_ci	err = -EINVAL;
2138c2ecf20Sopenharmony_ci	while (time_before(jiffies, timeout)) {
2148c2ecf20Sopenharmony_ci		regmap_read(rk_phy->reg_base,
2158c2ecf20Sopenharmony_ci			    rk_phy->phy_data->pcie_status,
2168c2ecf20Sopenharmony_ci			    &status);
2178c2ecf20Sopenharmony_ci		if (status & PHY_PLL_LOCKED) {
2188c2ecf20Sopenharmony_ci			dev_dbg(&phy->dev, "pll locked!\n");
2198c2ecf20Sopenharmony_ci			err = 0;
2208c2ecf20Sopenharmony_ci			break;
2218c2ecf20Sopenharmony_ci		}
2228c2ecf20Sopenharmony_ci		msleep(20);
2238c2ecf20Sopenharmony_ci	}
2248c2ecf20Sopenharmony_ci
2258c2ecf20Sopenharmony_ci	if (err) {
2268c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "pll lock timeout!\n");
2278c2ecf20Sopenharmony_ci		goto err_pll_lock;
2288c2ecf20Sopenharmony_ci	}
2298c2ecf20Sopenharmony_ci
2308c2ecf20Sopenharmony_ci	phy_wr_cfg(rk_phy, PHY_CFG_CLK_TEST, PHY_CFG_SEPE_RATE);
2318c2ecf20Sopenharmony_ci	phy_wr_cfg(rk_phy, PHY_CFG_CLK_SCC, PHY_CFG_PLL_100M);
2328c2ecf20Sopenharmony_ci
2338c2ecf20Sopenharmony_ci	err = -ETIMEDOUT;
2348c2ecf20Sopenharmony_ci	while (time_before(jiffies, timeout)) {
2358c2ecf20Sopenharmony_ci		regmap_read(rk_phy->reg_base,
2368c2ecf20Sopenharmony_ci			    rk_phy->phy_data->pcie_status,
2378c2ecf20Sopenharmony_ci			    &status);
2388c2ecf20Sopenharmony_ci		if (!(status & PHY_PLL_OUTPUT)) {
2398c2ecf20Sopenharmony_ci			dev_dbg(&phy->dev, "pll output enable done!\n");
2408c2ecf20Sopenharmony_ci			err = 0;
2418c2ecf20Sopenharmony_ci			break;
2428c2ecf20Sopenharmony_ci		}
2438c2ecf20Sopenharmony_ci		msleep(20);
2448c2ecf20Sopenharmony_ci	}
2458c2ecf20Sopenharmony_ci
2468c2ecf20Sopenharmony_ci	if (err) {
2478c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "pll output enable timeout!\n");
2488c2ecf20Sopenharmony_ci		goto err_pll_lock;
2498c2ecf20Sopenharmony_ci	}
2508c2ecf20Sopenharmony_ci
2518c2ecf20Sopenharmony_ci	regmap_write(rk_phy->reg_base, rk_phy->phy_data->pcie_conf,
2528c2ecf20Sopenharmony_ci		     HIWORD_UPDATE(PHY_CFG_PLL_LOCK,
2538c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_MASK,
2548c2ecf20Sopenharmony_ci				   PHY_CFG_ADDR_SHIFT));
2558c2ecf20Sopenharmony_ci	err = -EINVAL;
2568c2ecf20Sopenharmony_ci	while (time_before(jiffies, timeout)) {
2578c2ecf20Sopenharmony_ci		regmap_read(rk_phy->reg_base,
2588c2ecf20Sopenharmony_ci			    rk_phy->phy_data->pcie_status,
2598c2ecf20Sopenharmony_ci			    &status);
2608c2ecf20Sopenharmony_ci		if (status & PHY_PLL_LOCKED) {
2618c2ecf20Sopenharmony_ci			dev_dbg(&phy->dev, "pll relocked!\n");
2628c2ecf20Sopenharmony_ci			err = 0;
2638c2ecf20Sopenharmony_ci			break;
2648c2ecf20Sopenharmony_ci		}
2658c2ecf20Sopenharmony_ci		msleep(20);
2668c2ecf20Sopenharmony_ci	}
2678c2ecf20Sopenharmony_ci
2688c2ecf20Sopenharmony_ci	if (err) {
2698c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "pll relock timeout!\n");
2708c2ecf20Sopenharmony_ci		goto err_pll_lock;
2718c2ecf20Sopenharmony_ci	}
2728c2ecf20Sopenharmony_ci
2738c2ecf20Sopenharmony_cierr_out:
2748c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
2758c2ecf20Sopenharmony_ci	return 0;
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_cierr_pll_lock:
2788c2ecf20Sopenharmony_ci	reset_control_assert(rk_phy->phy_rst);
2798c2ecf20Sopenharmony_cierr_pwr_cnt:
2808c2ecf20Sopenharmony_ci	rk_phy->pwr_cnt--;
2818c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
2828c2ecf20Sopenharmony_ci	return err;
2838c2ecf20Sopenharmony_ci}
2848c2ecf20Sopenharmony_ci
2858c2ecf20Sopenharmony_cistatic int rockchip_pcie_phy_init(struct phy *phy)
2868c2ecf20Sopenharmony_ci{
2878c2ecf20Sopenharmony_ci	struct phy_pcie_instance *inst = phy_get_drvdata(phy);
2888c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy = to_pcie_phy(inst);
2898c2ecf20Sopenharmony_ci	int err = 0;
2908c2ecf20Sopenharmony_ci
2918c2ecf20Sopenharmony_ci	mutex_lock(&rk_phy->pcie_mutex);
2928c2ecf20Sopenharmony_ci
2938c2ecf20Sopenharmony_ci	if (rk_phy->init_cnt++)
2948c2ecf20Sopenharmony_ci		goto err_out;
2958c2ecf20Sopenharmony_ci
2968c2ecf20Sopenharmony_ci	err = clk_prepare_enable(rk_phy->clk_pciephy_ref);
2978c2ecf20Sopenharmony_ci	if (err) {
2988c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "Fail to enable pcie ref clock.\n");
2998c2ecf20Sopenharmony_ci		goto err_refclk;
3008c2ecf20Sopenharmony_ci	}
3018c2ecf20Sopenharmony_ci
3028c2ecf20Sopenharmony_ci	err = reset_control_assert(rk_phy->phy_rst);
3038c2ecf20Sopenharmony_ci	if (err) {
3048c2ecf20Sopenharmony_ci		dev_err(&phy->dev, "assert phy_rst err %d\n", err);
3058c2ecf20Sopenharmony_ci		goto err_reset;
3068c2ecf20Sopenharmony_ci	}
3078c2ecf20Sopenharmony_ci
3088c2ecf20Sopenharmony_cierr_out:
3098c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
3108c2ecf20Sopenharmony_ci	return 0;
3118c2ecf20Sopenharmony_ci
3128c2ecf20Sopenharmony_cierr_reset:
3138c2ecf20Sopenharmony_ci
3148c2ecf20Sopenharmony_ci	clk_disable_unprepare(rk_phy->clk_pciephy_ref);
3158c2ecf20Sopenharmony_cierr_refclk:
3168c2ecf20Sopenharmony_ci	rk_phy->init_cnt--;
3178c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
3188c2ecf20Sopenharmony_ci	return err;
3198c2ecf20Sopenharmony_ci}
3208c2ecf20Sopenharmony_ci
3218c2ecf20Sopenharmony_cistatic int rockchip_pcie_phy_exit(struct phy *phy)
3228c2ecf20Sopenharmony_ci{
3238c2ecf20Sopenharmony_ci	struct phy_pcie_instance *inst = phy_get_drvdata(phy);
3248c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy = to_pcie_phy(inst);
3258c2ecf20Sopenharmony_ci
3268c2ecf20Sopenharmony_ci	mutex_lock(&rk_phy->pcie_mutex);
3278c2ecf20Sopenharmony_ci
3288c2ecf20Sopenharmony_ci	if (--rk_phy->init_cnt)
3298c2ecf20Sopenharmony_ci		goto err_init_cnt;
3308c2ecf20Sopenharmony_ci
3318c2ecf20Sopenharmony_ci	clk_disable_unprepare(rk_phy->clk_pciephy_ref);
3328c2ecf20Sopenharmony_ci
3338c2ecf20Sopenharmony_cierr_init_cnt:
3348c2ecf20Sopenharmony_ci	mutex_unlock(&rk_phy->pcie_mutex);
3358c2ecf20Sopenharmony_ci	return 0;
3368c2ecf20Sopenharmony_ci}
3378c2ecf20Sopenharmony_ci
3388c2ecf20Sopenharmony_cistatic const struct phy_ops ops = {
3398c2ecf20Sopenharmony_ci	.init		= rockchip_pcie_phy_init,
3408c2ecf20Sopenharmony_ci	.exit		= rockchip_pcie_phy_exit,
3418c2ecf20Sopenharmony_ci	.power_on	= rockchip_pcie_phy_power_on,
3428c2ecf20Sopenharmony_ci	.power_off	= rockchip_pcie_phy_power_off,
3438c2ecf20Sopenharmony_ci	.owner		= THIS_MODULE,
3448c2ecf20Sopenharmony_ci};
3458c2ecf20Sopenharmony_ci
3468c2ecf20Sopenharmony_cistatic const struct rockchip_pcie_data rk3399_pcie_data = {
3478c2ecf20Sopenharmony_ci	.pcie_conf = 0xe220,
3488c2ecf20Sopenharmony_ci	.pcie_status = 0xe2a4,
3498c2ecf20Sopenharmony_ci	.pcie_laneoff = 0xe214,
3508c2ecf20Sopenharmony_ci};
3518c2ecf20Sopenharmony_ci
3528c2ecf20Sopenharmony_cistatic const struct of_device_id rockchip_pcie_phy_dt_ids[] = {
3538c2ecf20Sopenharmony_ci	{
3548c2ecf20Sopenharmony_ci		.compatible = "rockchip,rk3399-pcie-phy",
3558c2ecf20Sopenharmony_ci		.data = &rk3399_pcie_data,
3568c2ecf20Sopenharmony_ci	},
3578c2ecf20Sopenharmony_ci	{}
3588c2ecf20Sopenharmony_ci};
3598c2ecf20Sopenharmony_ci
3608c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, rockchip_pcie_phy_dt_ids);
3618c2ecf20Sopenharmony_ci
3628c2ecf20Sopenharmony_cistatic int rockchip_pcie_phy_probe(struct platform_device *pdev)
3638c2ecf20Sopenharmony_ci{
3648c2ecf20Sopenharmony_ci	struct device *dev = &pdev->dev;
3658c2ecf20Sopenharmony_ci	struct rockchip_pcie_phy *rk_phy;
3668c2ecf20Sopenharmony_ci	struct phy_provider *phy_provider;
3678c2ecf20Sopenharmony_ci	struct regmap *grf;
3688c2ecf20Sopenharmony_ci	const struct of_device_id *of_id;
3698c2ecf20Sopenharmony_ci	int i;
3708c2ecf20Sopenharmony_ci	u32 phy_num;
3718c2ecf20Sopenharmony_ci
3728c2ecf20Sopenharmony_ci	grf = syscon_node_to_regmap(dev->parent->of_node);
3738c2ecf20Sopenharmony_ci	if (IS_ERR(grf)) {
3748c2ecf20Sopenharmony_ci		dev_err(dev, "Cannot find GRF syscon\n");
3758c2ecf20Sopenharmony_ci		return PTR_ERR(grf);
3768c2ecf20Sopenharmony_ci	}
3778c2ecf20Sopenharmony_ci
3788c2ecf20Sopenharmony_ci	rk_phy = devm_kzalloc(dev, sizeof(*rk_phy), GFP_KERNEL);
3798c2ecf20Sopenharmony_ci	if (!rk_phy)
3808c2ecf20Sopenharmony_ci		return -ENOMEM;
3818c2ecf20Sopenharmony_ci
3828c2ecf20Sopenharmony_ci	of_id = of_match_device(rockchip_pcie_phy_dt_ids, &pdev->dev);
3838c2ecf20Sopenharmony_ci	if (!of_id)
3848c2ecf20Sopenharmony_ci		return -EINVAL;
3858c2ecf20Sopenharmony_ci
3868c2ecf20Sopenharmony_ci	rk_phy->phy_data = (struct rockchip_pcie_data *)of_id->data;
3878c2ecf20Sopenharmony_ci	rk_phy->reg_base = grf;
3888c2ecf20Sopenharmony_ci
3898c2ecf20Sopenharmony_ci	mutex_init(&rk_phy->pcie_mutex);
3908c2ecf20Sopenharmony_ci
3918c2ecf20Sopenharmony_ci	rk_phy->phy_rst = devm_reset_control_get(dev, "phy");
3928c2ecf20Sopenharmony_ci	if (IS_ERR(rk_phy->phy_rst)) {
3938c2ecf20Sopenharmony_ci		if (PTR_ERR(rk_phy->phy_rst) != -EPROBE_DEFER)
3948c2ecf20Sopenharmony_ci			dev_err(dev,
3958c2ecf20Sopenharmony_ci				"missing phy property for reset controller\n");
3968c2ecf20Sopenharmony_ci		return PTR_ERR(rk_phy->phy_rst);
3978c2ecf20Sopenharmony_ci	}
3988c2ecf20Sopenharmony_ci
3998c2ecf20Sopenharmony_ci	rk_phy->clk_pciephy_ref = devm_clk_get(dev, "refclk");
4008c2ecf20Sopenharmony_ci	if (IS_ERR(rk_phy->clk_pciephy_ref)) {
4018c2ecf20Sopenharmony_ci		dev_err(dev, "refclk not found.\n");
4028c2ecf20Sopenharmony_ci		return PTR_ERR(rk_phy->clk_pciephy_ref);
4038c2ecf20Sopenharmony_ci	}
4048c2ecf20Sopenharmony_ci
4058c2ecf20Sopenharmony_ci	/* parse #phy-cells to see if it's legacy PHY model */
4068c2ecf20Sopenharmony_ci	if (of_property_read_u32(dev->of_node, "#phy-cells", &phy_num))
4078c2ecf20Sopenharmony_ci		return -ENOENT;
4088c2ecf20Sopenharmony_ci
4098c2ecf20Sopenharmony_ci	phy_num = (phy_num == 0) ? 1 : PHY_MAX_LANE_NUM;
4108c2ecf20Sopenharmony_ci	dev_dbg(dev, "phy number is %d\n", phy_num);
4118c2ecf20Sopenharmony_ci
4128c2ecf20Sopenharmony_ci	for (i = 0; i < phy_num; i++) {
4138c2ecf20Sopenharmony_ci		rk_phy->phys[i].phy = devm_phy_create(dev, dev->of_node, &ops);
4148c2ecf20Sopenharmony_ci		if (IS_ERR(rk_phy->phys[i].phy)) {
4158c2ecf20Sopenharmony_ci			dev_err(dev, "failed to create PHY%d\n", i);
4168c2ecf20Sopenharmony_ci			return PTR_ERR(rk_phy->phys[i].phy);
4178c2ecf20Sopenharmony_ci		}
4188c2ecf20Sopenharmony_ci		rk_phy->phys[i].index = i;
4198c2ecf20Sopenharmony_ci		phy_set_drvdata(rk_phy->phys[i].phy, &rk_phy->phys[i]);
4208c2ecf20Sopenharmony_ci	}
4218c2ecf20Sopenharmony_ci
4228c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, rk_phy);
4238c2ecf20Sopenharmony_ci	phy_provider = devm_of_phy_provider_register(dev,
4248c2ecf20Sopenharmony_ci					rockchip_pcie_phy_of_xlate);
4258c2ecf20Sopenharmony_ci
4268c2ecf20Sopenharmony_ci	return PTR_ERR_OR_ZERO(phy_provider);
4278c2ecf20Sopenharmony_ci}
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_cistatic struct platform_driver rockchip_pcie_driver = {
4308c2ecf20Sopenharmony_ci	.probe		= rockchip_pcie_phy_probe,
4318c2ecf20Sopenharmony_ci	.driver		= {
4328c2ecf20Sopenharmony_ci		.name	= "rockchip-pcie-phy",
4338c2ecf20Sopenharmony_ci		.of_match_table = rockchip_pcie_phy_dt_ids,
4348c2ecf20Sopenharmony_ci	},
4358c2ecf20Sopenharmony_ci};
4368c2ecf20Sopenharmony_ci
4378c2ecf20Sopenharmony_cimodule_platform_driver(rockchip_pcie_driver);
4388c2ecf20Sopenharmony_ci
4398c2ecf20Sopenharmony_ciMODULE_AUTHOR("Shawn Lin <shawn.lin@rock-chips.com>");
4408c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Rockchip PCIe PHY driver");
4418c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2");
442