phy.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright 2023 Bootlin
  4. *
  5. */
  6. #include "common.h"
  7. #include "netlink.h"
  8. #include <linux/phy.h>
  9. #include <linux/phy_link_topology.h>
  10. #include <linux/sfp.h>
  11. #include <net/netdev_lock.h>
  12. struct phy_req_info {
  13. struct ethnl_req_info base;
  14. };
  15. struct phy_reply_data {
  16. struct ethnl_reply_data base;
  17. u32 phyindex;
  18. char *drvname;
  19. char *name;
  20. unsigned int upstream_type;
  21. char *upstream_sfp_name;
  22. unsigned int upstream_index;
  23. char *downstream_sfp_name;
  24. };
  25. #define PHY_REPDATA(__reply_base) \
  26. container_of(__reply_base, struct phy_reply_data, base)
  27. const struct nla_policy ethnl_phy_get_policy[ETHTOOL_A_PHY_HEADER + 1] = {
  28. [ETHTOOL_A_PHY_HEADER] = NLA_POLICY_NESTED(ethnl_header_policy),
  29. };
  30. static int phy_reply_size(const struct ethnl_req_info *req_info,
  31. const struct ethnl_reply_data *reply_data)
  32. {
  33. struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
  34. size_t size = 0;
  35. /* ETHTOOL_A_PHY_INDEX */
  36. size += nla_total_size(sizeof(u32));
  37. /* ETHTOOL_A_DRVNAME */
  38. if (rep_data->drvname)
  39. size += nla_total_size(strlen(rep_data->drvname) + 1);
  40. /* ETHTOOL_A_NAME */
  41. size += nla_total_size(strlen(rep_data->name) + 1);
  42. /* ETHTOOL_A_PHY_UPSTREAM_TYPE */
  43. size += nla_total_size(sizeof(u32));
  44. /* ETHTOOL_A_PHY_UPSTREAM_SFP_NAME */
  45. if (rep_data->upstream_sfp_name)
  46. size += nla_total_size(strlen(rep_data->upstream_sfp_name) + 1);
  47. /* ETHTOOL_A_PHY_UPSTREAM_INDEX */
  48. if (rep_data->upstream_index)
  49. size += nla_total_size(sizeof(u32));
  50. /* ETHTOOL_A_PHY_DOWNSTREAM_SFP_NAME */
  51. if (rep_data->downstream_sfp_name)
  52. size += nla_total_size(strlen(rep_data->downstream_sfp_name) + 1);
  53. return size;
  54. }
  55. static int phy_prepare_data(const struct ethnl_req_info *req_info,
  56. struct ethnl_reply_data *reply_data,
  57. const struct genl_info *info)
  58. {
  59. struct phy_link_topology *topo = reply_data->dev->link_topo;
  60. struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
  61. struct nlattr **tb = info->attrs;
  62. struct phy_device_node *pdn;
  63. struct phy_device *phydev;
  64. /* RTNL is held by the caller */
  65. phydev = ethnl_req_get_phydev(req_info, tb, ETHTOOL_A_PHY_HEADER,
  66. info->extack);
  67. if (IS_ERR_OR_NULL(phydev))
  68. return -EOPNOTSUPP;
  69. pdn = xa_load(&topo->phys, phydev->phyindex);
  70. if (!pdn)
  71. return -EOPNOTSUPP;
  72. rep_data->phyindex = phydev->phyindex;
  73. rep_data->name = kstrdup(dev_name(&phydev->mdio.dev), GFP_KERNEL);
  74. rep_data->drvname = kstrdup(phydev->drv->name, GFP_KERNEL);
  75. rep_data->upstream_type = pdn->upstream_type;
  76. if (pdn->upstream_type == PHY_UPSTREAM_PHY) {
  77. struct phy_device *upstream = pdn->upstream.phydev;
  78. rep_data->upstream_index = upstream->phyindex;
  79. }
  80. if (pdn->parent_sfp_bus)
  81. rep_data->upstream_sfp_name = kstrdup(sfp_get_name(pdn->parent_sfp_bus),
  82. GFP_KERNEL);
  83. if (phydev->sfp_bus)
  84. rep_data->downstream_sfp_name = kstrdup(sfp_get_name(phydev->sfp_bus),
  85. GFP_KERNEL);
  86. return 0;
  87. }
  88. static int phy_fill_reply(struct sk_buff *skb,
  89. const struct ethnl_req_info *req_info,
  90. const struct ethnl_reply_data *reply_data)
  91. {
  92. struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
  93. if (nla_put_u32(skb, ETHTOOL_A_PHY_INDEX, rep_data->phyindex) ||
  94. nla_put_string(skb, ETHTOOL_A_PHY_NAME, rep_data->name) ||
  95. nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_TYPE, rep_data->upstream_type))
  96. return -EMSGSIZE;
  97. if (rep_data->drvname &&
  98. nla_put_string(skb, ETHTOOL_A_PHY_DRVNAME, rep_data->drvname))
  99. return -EMSGSIZE;
  100. if (rep_data->upstream_index &&
  101. nla_put_u32(skb, ETHTOOL_A_PHY_UPSTREAM_INDEX,
  102. rep_data->upstream_index))
  103. return -EMSGSIZE;
  104. if (rep_data->upstream_sfp_name &&
  105. nla_put_string(skb, ETHTOOL_A_PHY_UPSTREAM_SFP_NAME,
  106. rep_data->upstream_sfp_name))
  107. return -EMSGSIZE;
  108. if (rep_data->downstream_sfp_name &&
  109. nla_put_string(skb, ETHTOOL_A_PHY_DOWNSTREAM_SFP_NAME,
  110. rep_data->downstream_sfp_name))
  111. return -EMSGSIZE;
  112. return 0;
  113. }
  114. static void phy_cleanup_data(struct ethnl_reply_data *reply_data)
  115. {
  116. struct phy_reply_data *rep_data = PHY_REPDATA(reply_data);
  117. kfree(rep_data->drvname);
  118. kfree(rep_data->name);
  119. kfree(rep_data->upstream_sfp_name);
  120. kfree(rep_data->downstream_sfp_name);
  121. }
  122. const struct ethnl_request_ops ethnl_phy_request_ops = {
  123. .request_cmd = ETHTOOL_MSG_PHY_GET,
  124. .reply_cmd = ETHTOOL_MSG_PHY_GET_REPLY,
  125. .hdr_attr = ETHTOOL_A_PHY_HEADER,
  126. .req_info_size = sizeof(struct phy_req_info),
  127. .reply_data_size = sizeof(struct phy_reply_data),
  128. .prepare_data = phy_prepare_data,
  129. .reply_size = phy_reply_size,
  130. .fill_reply = phy_fill_reply,
  131. .cleanup_data = phy_cleanup_data,
  132. };