Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/phy/ralink/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 9 kB image not shown  

Quelle  phy-mt7621-pci.c   Sprache: 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,
         const 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 ");
MODULE_DESCRIPTION("MediaTek MT7621 PCIe PHY driver");
MODULE_LICENSE("GPL v2");

Messung V0.5
C=95 H=95 G=94

¤ Dauer der Verarbeitung: 0.4 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.