103-06-net-phy-qca808x-Add-QCA8084-probe-function.patch 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. From 9d0e22124d6f3ca901626dd5537b36c7c0c97812 Mon Sep 17 00:00:00 2001
  2. From: Luo Jie <[email protected]>
  3. Date: Mon, 29 Jan 2024 10:51:38 +0800
  4. Subject: [PATCH 06/50] net: phy: qca808x: Add QCA8084 probe function
  5. Add the PHY package probe function. The MDIO slave address of
  6. PHY, PCS and XPCS can be optionally customized by configuring
  7. the PHY package level register.
  8. In addition, enable system clock of PHY and de-assert PHY in
  9. the probe function so that the register of PHY device can be
  10. accessed, and the features of PHY can be acquired.
  11. Change-Id: I2251b9c5c398a21a4ef547a727189a934ad3a44c
  12. Signed-off-by: Luo Jie <[email protected]>
  13. ---
  14. drivers/net/phy/qcom/qca808x.c | 91 ++++++++++++++++++++++++++++++++++
  15. 1 file changed, 91 insertions(+)
  16. --- a/drivers/net/phy/qcom/qca808x.c
  17. +++ b/drivers/net/phy/qcom/qca808x.c
  18. @@ -2,6 +2,8 @@
  19. #include <linux/phy.h>
  20. #include <linux/module.h>
  21. +#include <linux/of.h>
  22. +#include <linux/clk.h>
  23. #include "qcom.h"
  24. @@ -127,6 +129,21 @@
  25. #define QCA8084_MII_SW_ADDR_MASK GENMASK(31, 24)
  26. #define QCA8084_MII_REG_DATA_UPPER_16_BITS BIT(1)
  27. +/* QCA8084 integrates 4 PHYs, PCS0 and PCS1(includes PCS and XPCS). */
  28. +#define QCA8084_MDIO_DEVICE_NUM 7
  29. +
  30. +#define QCA8084_PCS_CFG 0xc90f014
  31. +#define QCA8084_PCS_ADDR0_MASK GENMASK(4, 0)
  32. +#define QCA8084_PCS_ADDR1_MASK GENMASK(9, 5)
  33. +#define QCA8084_PCS_ADDR2_MASK GENMASK(14, 10)
  34. +
  35. +#define QCA8084_EPHY_CFG 0xc90f018
  36. +#define QCA8084_EPHY_ADDR0_MASK GENMASK(4, 0)
  37. +#define QCA8084_EPHY_ADDR1_MASK GENMASK(9, 5)
  38. +#define QCA8084_EPHY_ADDR2_MASK GENMASK(14, 10)
  39. +#define QCA8084_EPHY_ADDR3_MASK GENMASK(19, 15)
  40. +#define QCA8084_EPHY_LDO_EN GENMASK(21, 20)
  41. +
  42. MODULE_DESCRIPTION("Qualcomm Atheros QCA808X PHY driver");
  43. MODULE_AUTHOR("Matus Ujhelyi, Luo Jie");
  44. MODULE_LICENSE("GPL");
  45. @@ -836,6 +853,79 @@ static void qca8084_link_change_notify(s
  46. QCA8084_IPG_10_TO_11_EN : 0);
  47. }
  48. +static int qca8084_phy_package_probe_once(struct phy_device *phydev)
  49. +{
  50. + int addr[QCA8084_MDIO_DEVICE_NUM] = {0, 1, 2, 3, 4, 5, 6};
  51. + struct phy_package_shared *shared = phydev->shared;
  52. + int ret, clear, set;
  53. +
  54. + /* Program the MDIO address of PHY and PCS optionally, the MDIO
  55. + * address 0-6 is used for PHY and PCS MDIO devices by default.
  56. + */
  57. + ret = of_property_read_u32_array(shared->np,
  58. + "qcom,phy-addr-fixup",
  59. + addr, ARRAY_SIZE(addr));
  60. + if (ret && ret != -EINVAL)
  61. + return ret;
  62. +
  63. + /* Configure the MDIO addresses for the four PHY devices. */
  64. + clear = QCA8084_EPHY_ADDR0_MASK | QCA8084_EPHY_ADDR1_MASK |
  65. + QCA8084_EPHY_ADDR2_MASK | QCA8084_EPHY_ADDR3_MASK;
  66. + set = FIELD_PREP(QCA8084_EPHY_ADDR0_MASK, addr[0]);
  67. + set |= FIELD_PREP(QCA8084_EPHY_ADDR1_MASK, addr[1]);
  68. + set |= FIELD_PREP(QCA8084_EPHY_ADDR2_MASK, addr[2]);
  69. + set |= FIELD_PREP(QCA8084_EPHY_ADDR3_MASK, addr[3]);
  70. +
  71. + ret = qca8084_mii_modify(phydev, QCA8084_EPHY_CFG, clear, set);
  72. + if (ret)
  73. + return ret;
  74. +
  75. + /* Configure the MDIO addresses for PCS0 and PCS1 including
  76. + * PCS and XPCS.
  77. + */
  78. + clear = QCA8084_PCS_ADDR0_MASK | QCA8084_PCS_ADDR1_MASK |
  79. + QCA8084_PCS_ADDR2_MASK;
  80. + set = FIELD_PREP(QCA8084_PCS_ADDR0_MASK, addr[4]);
  81. + set |= FIELD_PREP(QCA8084_PCS_ADDR1_MASK, addr[5]);
  82. + set |= FIELD_PREP(QCA8084_PCS_ADDR2_MASK, addr[6]);
  83. +
  84. + return qca8084_mii_modify(phydev, QCA8084_PCS_CFG, clear, set);
  85. +}
  86. +
  87. +static int qca8084_probe(struct phy_device *phydev)
  88. +{
  89. + struct device *dev = &phydev->mdio.dev;
  90. + struct reset_control *rstc;
  91. + struct clk *clk;
  92. + int ret;
  93. +
  94. + ret = devm_of_phy_package_join(dev, phydev, 0);
  95. + if (ret)
  96. + return ret;
  97. +
  98. + if (phy_package_probe_once(phydev)) {
  99. + ret = qca8084_phy_package_probe_once(phydev);
  100. + if (ret)
  101. + return ret;
  102. + }
  103. +
  104. + /* Enable clock of PHY device, so that the PHY register
  105. + * can be accessed to get PHY features.
  106. + */
  107. + clk = devm_clk_get_enabled(dev, NULL);
  108. + if (IS_ERR(clk))
  109. + return dev_err_probe(dev, PTR_ERR(clk),
  110. + "Enable PHY clock failed\n");
  111. +
  112. + /* De-assert PHY reset after the clock of PHY enabled. */
  113. + rstc = devm_reset_control_get_exclusive(dev, NULL);
  114. + if (IS_ERR(rstc))
  115. + return dev_err_probe(dev, PTR_ERR(rstc),
  116. + "Get PHY reset failed\n");
  117. +
  118. + return reset_control_deassert(rstc);
  119. +}
  120. +
  121. static struct phy_driver qca808x_driver[] = {
  122. {
  123. /* Qualcomm QCA8081 */
  124. @@ -886,6 +976,7 @@ static struct phy_driver qca808x_driver[
  125. .cable_test_get_status = qca808x_cable_test_get_status,
  126. .config_init = qca8084_config_init,
  127. .link_change_notify = qca8084_link_change_notify,
  128. + .probe = qca8084_probe,
  129. }, };
  130. module_phy_driver(qca808x_driver);