mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-06 05:02:31 +00:00
7559e7572c
The DT of_device.h and of_platform.h date back to the separate of_platform_bus_type before it as merged into the regular platform bus. As part of that merge prepping Arm DT support 13 years ago, they "temporarily" include each other. They also include platform_device.h and of.h. As a result, there's a pretty much random mix of those include files used throughout the tree. In order to detangle these headers and replace the implicit includes with struct declarations, users need to explicitly include the correct includes. Signed-off-by: Rob Herring <robh@kernel.org> Acked-by: Marc Kleine-Budde <mkl@pengutronix.de> # for drivers/phy/phy-can-transceiver.c Acked-by: Heiko Stuebner <heiko@sntech.de> Acked-by: Sergio Paracuellos <sergio.paracuellos@gmail.com> Link: https://lore.kernel.org/r/20230714174841.4061919-1-robh@kernel.org Signed-off-by: Vinod Koul <vkoul@kernel.org>
360 lines
9.7 KiB
C
360 lines
9.7 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Mediatek MT7621 PCI PHY Driver
|
|
* Author: Sergio Paracuellos <sergio.paracuellos@gmail.com>
|
|
*/
|
|
|
|
#include <dt-bindings/phy/phy.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/bitfield.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of.h>
|
|
#include <linux/phy/phy.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/sys_soc.h>
|
|
|
|
#define RG_PE1_PIPE_REG 0x02c
|
|
#define RG_PE1_PIPE_RST BIT(12)
|
|
#define RG_PE1_PIPE_CMD_FRC BIT(4)
|
|
|
|
#define RG_P0_TO_P1_WIDTH 0x100
|
|
#define RG_PE1_H_LCDDS_REG 0x49c
|
|
#define RG_PE1_H_LCDDS_PCW GENMASK(30, 0)
|
|
|
|
#define RG_PE1_FRC_H_XTAL_REG 0x400
|
|
#define RG_PE1_FRC_H_XTAL_TYPE BIT(8)
|
|
#define RG_PE1_H_XTAL_TYPE GENMASK(10, 9)
|
|
|
|
#define RG_PE1_FRC_PHY_REG 0x000
|
|
#define RG_PE1_FRC_PHY_EN BIT(4)
|
|
#define RG_PE1_PHY_EN BIT(5)
|
|
|
|
#define RG_PE1_H_PLL_REG 0x490
|
|
#define RG_PE1_H_PLL_BC GENMASK(23, 22)
|
|
#define RG_PE1_H_PLL_BP GENMASK(21, 18)
|
|
#define RG_PE1_H_PLL_IR GENMASK(15, 12)
|
|
#define RG_PE1_H_PLL_IC GENMASK(11, 8)
|
|
#define RG_PE1_H_PLL_PREDIV GENMASK(7, 6)
|
|
#define RG_PE1_PLL_DIVEN GENMASK(3, 1)
|
|
|
|
#define RG_PE1_H_PLL_FBKSEL_REG 0x4bc
|
|
#define RG_PE1_H_PLL_FBKSEL GENMASK(5, 4)
|
|
|
|
#define RG_PE1_H_LCDDS_SSC_PRD_REG 0x4a4
|
|
#define RG_PE1_H_LCDDS_SSC_PRD GENMASK(15, 0)
|
|
|
|
#define RG_PE1_H_LCDDS_SSC_DELTA_REG 0x4a8
|
|
#define RG_PE1_H_LCDDS_SSC_DELTA GENMASK(11, 0)
|
|
#define RG_PE1_H_LCDDS_SSC_DELTA1 GENMASK(27, 16)
|
|
|
|
#define RG_PE1_LCDDS_CLK_PH_INV_REG 0x4a0
|
|
#define RG_PE1_LCDDS_CLK_PH_INV BIT(5)
|
|
|
|
#define RG_PE1_H_PLL_BR_REG 0x4ac
|
|
#define RG_PE1_H_PLL_BR GENMASK(18, 16)
|
|
|
|
#define RG_PE1_MSTCKDIV_REG 0x414
|
|
#define RG_PE1_MSTCKDIV GENMASK(7, 6)
|
|
|
|
#define RG_PE1_FRC_MSTCKDIV BIT(5)
|
|
|
|
#define MAX_PHYS 2
|
|
|
|
/**
|
|
* struct mt7621_pci_phy - Mt7621 Pcie PHY core
|
|
* @dev: pointer to device
|
|
* @regmap: kernel regmap pointer
|
|
* @phy: pointer to the kernel PHY device
|
|
* @sys_clk: pointer to the system XTAL clock
|
|
* @port_base: base register
|
|
* @has_dual_port: if the phy has dual ports.
|
|
* @bypass_pipe_rst: mark if 'mt7621_bypass_pipe_rst'
|
|
* needs to be executed. Depends on chip revision.
|
|
*/
|
|
struct mt7621_pci_phy {
|
|
struct device *dev;
|
|
struct regmap *regmap;
|
|
struct phy *phy;
|
|
struct clk *sys_clk;
|
|
void __iomem *port_base;
|
|
bool has_dual_port;
|
|
bool bypass_pipe_rst;
|
|
};
|
|
|
|
static inline void mt7621_phy_rmw(struct mt7621_pci_phy *phy,
|
|
u32 reg, u32 clr, u32 set)
|
|
{
|
|
u32 val;
|
|
|
|
/*
|
|
* We cannot use 'regmap_write_bits' here because internally
|
|
* 'set' is masked before is set to the value that will be
|
|
* written to the register. That way results in no reliable
|
|
* pci setup. Avoid to mask 'set' before set value to 'val'
|
|
* completely avoid the problem.
|
|
*/
|
|
regmap_read(phy->regmap, reg, &val);
|
|
val &= ~clr;
|
|
val |= set;
|
|
regmap_write(phy->regmap, reg, val);
|
|
}
|
|
|
|
static void mt7621_bypass_pipe_rst(struct mt7621_pci_phy *phy)
|
|
{
|
|
mt7621_phy_rmw(phy, RG_PE1_PIPE_REG, 0, RG_PE1_PIPE_RST);
|
|
mt7621_phy_rmw(phy, RG_PE1_PIPE_REG, 0, RG_PE1_PIPE_CMD_FRC);
|
|
|
|
if (phy->has_dual_port) {
|
|
mt7621_phy_rmw(phy, RG_PE1_PIPE_REG + RG_P0_TO_P1_WIDTH,
|
|
0, RG_PE1_PIPE_RST);
|
|
mt7621_phy_rmw(phy, RG_PE1_PIPE_REG + RG_P0_TO_P1_WIDTH,
|
|
0, RG_PE1_PIPE_CMD_FRC);
|
|
}
|
|
}
|
|
|
|
static int mt7621_set_phy_for_ssc(struct mt7621_pci_phy *phy)
|
|
{
|
|
struct device *dev = phy->dev;
|
|
unsigned long clk_rate;
|
|
|
|
clk_rate = clk_get_rate(phy->sys_clk);
|
|
if (!clk_rate)
|
|
return -EINVAL;
|
|
|
|
/* Set PCIe Port PHY to disable SSC */
|
|
/* Debug Xtal Type */
|
|
mt7621_phy_rmw(phy, RG_PE1_FRC_H_XTAL_REG,
|
|
RG_PE1_FRC_H_XTAL_TYPE | RG_PE1_H_XTAL_TYPE,
|
|
RG_PE1_FRC_H_XTAL_TYPE |
|
|
FIELD_PREP(RG_PE1_H_XTAL_TYPE, 0x00));
|
|
|
|
/* disable port */
|
|
mt7621_phy_rmw(phy, RG_PE1_FRC_PHY_REG, RG_PE1_PHY_EN,
|
|
RG_PE1_FRC_PHY_EN);
|
|
|
|
if (phy->has_dual_port) {
|
|
mt7621_phy_rmw(phy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
|
|
RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
|
|
}
|
|
|
|
if (clk_rate == 40000000) { /* 40MHz Xtal */
|
|
/* Set Pre-divider ratio (for host mode) */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG, RG_PE1_H_PLL_PREDIV,
|
|
FIELD_PREP(RG_PE1_H_PLL_PREDIV, 0x01));
|
|
|
|
dev_dbg(dev, "Xtal is 40MHz\n");
|
|
} else if (clk_rate == 25000000) { /* 25MHz Xal */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG, RG_PE1_H_PLL_PREDIV,
|
|
FIELD_PREP(RG_PE1_H_PLL_PREDIV, 0x00));
|
|
|
|
/* Select feedback clock */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_FBKSEL_REG,
|
|
RG_PE1_H_PLL_FBKSEL,
|
|
FIELD_PREP(RG_PE1_H_PLL_FBKSEL, 0x01));
|
|
|
|
/* DDS NCPO PCW (for host mode) */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_PRD_REG,
|
|
RG_PE1_H_LCDDS_SSC_PRD,
|
|
FIELD_PREP(RG_PE1_H_LCDDS_SSC_PRD, 0x00));
|
|
|
|
/* DDS SSC dither period control */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_PRD_REG,
|
|
RG_PE1_H_LCDDS_SSC_PRD,
|
|
FIELD_PREP(RG_PE1_H_LCDDS_SSC_PRD, 0x18d));
|
|
|
|
/* DDS SSC dither amplitude control */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_LCDDS_SSC_DELTA_REG,
|
|
RG_PE1_H_LCDDS_SSC_DELTA |
|
|
RG_PE1_H_LCDDS_SSC_DELTA1,
|
|
FIELD_PREP(RG_PE1_H_LCDDS_SSC_DELTA, 0x4a) |
|
|
FIELD_PREP(RG_PE1_H_LCDDS_SSC_DELTA1, 0x4a));
|
|
|
|
dev_dbg(dev, "Xtal is 25MHz\n");
|
|
} else { /* 20MHz Xtal */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG, RG_PE1_H_PLL_PREDIV,
|
|
FIELD_PREP(RG_PE1_H_PLL_PREDIV, 0x00));
|
|
|
|
dev_dbg(dev, "Xtal is 20MHz\n");
|
|
}
|
|
|
|
/* DDS clock inversion */
|
|
mt7621_phy_rmw(phy, RG_PE1_LCDDS_CLK_PH_INV_REG,
|
|
RG_PE1_LCDDS_CLK_PH_INV, RG_PE1_LCDDS_CLK_PH_INV);
|
|
|
|
/* Set PLL bits */
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_REG,
|
|
RG_PE1_H_PLL_BC | RG_PE1_H_PLL_BP | RG_PE1_H_PLL_IR |
|
|
RG_PE1_H_PLL_IC | RG_PE1_PLL_DIVEN,
|
|
FIELD_PREP(RG_PE1_H_PLL_BC, 0x02) |
|
|
FIELD_PREP(RG_PE1_H_PLL_BP, 0x06) |
|
|
FIELD_PREP(RG_PE1_H_PLL_IR, 0x02) |
|
|
FIELD_PREP(RG_PE1_H_PLL_IC, 0x01) |
|
|
FIELD_PREP(RG_PE1_PLL_DIVEN, 0x02));
|
|
|
|
mt7621_phy_rmw(phy, RG_PE1_H_PLL_BR_REG, RG_PE1_H_PLL_BR,
|
|
FIELD_PREP(RG_PE1_H_PLL_BR, 0x00));
|
|
|
|
if (clk_rate == 40000000) { /* 40MHz Xtal */
|
|
/* set force mode enable of da_pe1_mstckdiv */
|
|
mt7621_phy_rmw(phy, RG_PE1_MSTCKDIV_REG,
|
|
RG_PE1_MSTCKDIV | RG_PE1_FRC_MSTCKDIV,
|
|
FIELD_PREP(RG_PE1_MSTCKDIV, 0x01) |
|
|
RG_PE1_FRC_MSTCKDIV);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mt7621_pci_phy_init(struct phy *phy)
|
|
{
|
|
struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
|
|
|
|
if (mphy->bypass_pipe_rst)
|
|
mt7621_bypass_pipe_rst(mphy);
|
|
|
|
return mt7621_set_phy_for_ssc(mphy);
|
|
}
|
|
|
|
static int mt7621_pci_phy_power_on(struct phy *phy)
|
|
{
|
|
struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
|
|
|
|
/* Enable PHY and disable force mode */
|
|
mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG,
|
|
RG_PE1_FRC_PHY_EN, RG_PE1_PHY_EN);
|
|
|
|
if (mphy->has_dual_port) {
|
|
mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
|
|
RG_PE1_FRC_PHY_EN, RG_PE1_PHY_EN);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mt7621_pci_phy_power_off(struct phy *phy)
|
|
{
|
|
struct mt7621_pci_phy *mphy = phy_get_drvdata(phy);
|
|
|
|
/* Disable PHY */
|
|
mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG,
|
|
RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
|
|
|
|
if (mphy->has_dual_port) {
|
|
mt7621_phy_rmw(mphy, RG_PE1_FRC_PHY_REG + RG_P0_TO_P1_WIDTH,
|
|
RG_PE1_PHY_EN, RG_PE1_FRC_PHY_EN);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mt7621_pci_phy_exit(struct phy *phy)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static const struct phy_ops mt7621_pci_phy_ops = {
|
|
.init = mt7621_pci_phy_init,
|
|
.exit = mt7621_pci_phy_exit,
|
|
.power_on = mt7621_pci_phy_power_on,
|
|
.power_off = mt7621_pci_phy_power_off,
|
|
.owner = THIS_MODULE,
|
|
};
|
|
|
|
static struct phy *mt7621_pcie_phy_of_xlate(struct device *dev,
|
|
struct of_phandle_args *args)
|
|
{
|
|
struct mt7621_pci_phy *mt7621_phy = dev_get_drvdata(dev);
|
|
|
|
if (WARN_ON(args->args[0] >= MAX_PHYS))
|
|
return ERR_PTR(-ENODEV);
|
|
|
|
mt7621_phy->has_dual_port = args->args[0];
|
|
|
|
dev_dbg(dev, "PHY for 0x%px (dual port = %d)\n",
|
|
mt7621_phy->port_base, mt7621_phy->has_dual_port);
|
|
|
|
return mt7621_phy->phy;
|
|
}
|
|
|
|
static const struct soc_device_attribute mt7621_pci_quirks_match[] = {
|
|
{ .soc_id = "mt7621", .revision = "E2" },
|
|
{ /* sentinel */ }
|
|
};
|
|
|
|
static const struct regmap_config mt7621_pci_phy_regmap_config = {
|
|
.reg_bits = 32,
|
|
.val_bits = 32,
|
|
.reg_stride = 4,
|
|
.max_register = 0x700,
|
|
};
|
|
|
|
static int mt7621_pci_phy_probe(struct platform_device *pdev)
|
|
{
|
|
struct device *dev = &pdev->dev;
|
|
const struct soc_device_attribute *attr;
|
|
struct phy_provider *provider;
|
|
struct mt7621_pci_phy *phy;
|
|
|
|
phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL);
|
|
if (!phy)
|
|
return -ENOMEM;
|
|
|
|
attr = soc_device_match(mt7621_pci_quirks_match);
|
|
if (attr)
|
|
phy->bypass_pipe_rst = true;
|
|
|
|
phy->dev = dev;
|
|
platform_set_drvdata(pdev, phy);
|
|
|
|
phy->port_base = devm_platform_ioremap_resource(pdev, 0);
|
|
if (IS_ERR(phy->port_base)) {
|
|
dev_err(dev, "failed to remap phy regs\n");
|
|
return PTR_ERR(phy->port_base);
|
|
}
|
|
|
|
phy->regmap = devm_regmap_init_mmio(phy->dev, phy->port_base,
|
|
&mt7621_pci_phy_regmap_config);
|
|
if (IS_ERR(phy->regmap))
|
|
return PTR_ERR(phy->regmap);
|
|
|
|
phy->phy = devm_phy_create(dev, dev->of_node, &mt7621_pci_phy_ops);
|
|
if (IS_ERR(phy->phy)) {
|
|
dev_err(dev, "failed to create phy\n");
|
|
return PTR_ERR(phy->phy);
|
|
}
|
|
|
|
phy->sys_clk = devm_clk_get(dev, NULL);
|
|
if (IS_ERR(phy->sys_clk)) {
|
|
dev_err(dev, "failed to get phy clock\n");
|
|
return PTR_ERR(phy->sys_clk);
|
|
}
|
|
|
|
phy_set_drvdata(phy->phy, phy);
|
|
|
|
provider = devm_of_phy_provider_register(dev, mt7621_pcie_phy_of_xlate);
|
|
|
|
return PTR_ERR_OR_ZERO(provider);
|
|
}
|
|
|
|
static const struct of_device_id mt7621_pci_phy_ids[] = {
|
|
{ .compatible = "mediatek,mt7621-pci-phy" },
|
|
{},
|
|
};
|
|
MODULE_DEVICE_TABLE(of, mt7621_pci_phy_ids);
|
|
|
|
static struct platform_driver mt7621_pci_phy_driver = {
|
|
.probe = mt7621_pci_phy_probe,
|
|
.driver = {
|
|
.name = "mt7621-pci-phy",
|
|
.of_match_table = mt7621_pci_phy_ids,
|
|
},
|
|
};
|
|
|
|
builtin_platform_driver(mt7621_pci_phy_driver);
|
|
|
|
MODULE_AUTHOR("Sergio Paracuellos <sergio.paracuellos@gmail.com>");
|
|
MODULE_DESCRIPTION("MediaTek MT7621 PCIe PHY driver");
|
|
MODULE_LICENSE("GPL v2");
|