metrics.c 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. #include <linux/netlink.h>
  3. #include <linux/nospec.h>
  4. #include <linux/rtnetlink.h>
  5. #include <linux/types.h>
  6. #include <net/ip.h>
  7. #include <net/net_namespace.h>
  8. #include <net/tcp.h>
  9. static int ip_metrics_convert(struct net *net, struct nlattr *fc_mx,
  10. int fc_mx_len, u32 *metrics,
  11. struct netlink_ext_ack *extack)
  12. {
  13. bool ecn_ca = false;
  14. struct nlattr *nla;
  15. int remaining;
  16. if (!fc_mx)
  17. return 0;
  18. nla_for_each_attr(nla, fc_mx, fc_mx_len, remaining) {
  19. int type = nla_type(nla);
  20. u32 val;
  21. if (!type)
  22. continue;
  23. if (type > RTAX_MAX) {
  24. NL_SET_ERR_MSG(extack, "Invalid metric type");
  25. return -EINVAL;
  26. }
  27. type = array_index_nospec(type, RTAX_MAX + 1);
  28. if (type == RTAX_CC_ALGO) {
  29. char tmp[TCP_CA_NAME_MAX];
  30. nla_strscpy(tmp, nla, sizeof(tmp));
  31. val = tcp_ca_get_key_by_name(net, tmp, &ecn_ca);
  32. if (val == TCP_CA_UNSPEC) {
  33. NL_SET_ERR_MSG(extack, "Unknown tcp congestion algorithm");
  34. return -EINVAL;
  35. }
  36. } else {
  37. if (nla_len(nla) != sizeof(u32)) {
  38. NL_SET_ERR_MSG_ATTR(extack, nla,
  39. "Invalid attribute in metrics");
  40. return -EINVAL;
  41. }
  42. val = nla_get_u32(nla);
  43. }
  44. if (type == RTAX_ADVMSS && val > 65535 - 40)
  45. val = 65535 - 40;
  46. if (type == RTAX_MTU && val > 65535 - 15)
  47. val = 65535 - 15;
  48. if (type == RTAX_HOPLIMIT && val > 255)
  49. val = 255;
  50. if (type == RTAX_FEATURES && (val & ~RTAX_FEATURE_MASK)) {
  51. NL_SET_ERR_MSG(extack, "Unknown flag set in feature mask in metrics attribute");
  52. return -EINVAL;
  53. }
  54. metrics[type - 1] = val;
  55. }
  56. if (ecn_ca)
  57. metrics[RTAX_FEATURES - 1] |= DST_FEATURE_ECN_CA;
  58. return 0;
  59. }
  60. struct dst_metrics *ip_fib_metrics_init(struct net *net, struct nlattr *fc_mx,
  61. int fc_mx_len,
  62. struct netlink_ext_ack *extack)
  63. {
  64. struct dst_metrics *fib_metrics;
  65. int err;
  66. if (!fc_mx)
  67. return (struct dst_metrics *)&dst_default_metrics;
  68. fib_metrics = kzalloc(sizeof(*fib_metrics), GFP_KERNEL);
  69. if (unlikely(!fib_metrics))
  70. return ERR_PTR(-ENOMEM);
  71. err = ip_metrics_convert(net, fc_mx, fc_mx_len, fib_metrics->metrics,
  72. extack);
  73. if (!err) {
  74. refcount_set(&fib_metrics->refcnt, 1);
  75. } else {
  76. kfree(fib_metrics);
  77. fib_metrics = ERR_PTR(err);
  78. }
  79. return fib_metrics;
  80. }
  81. EXPORT_SYMBOL_GPL(ip_fib_metrics_init);