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