aboutsummaryrefslogtreecommitdiffstats
path: root/net/tipc/node.c
diff options
context:
space:
mode:
authorJohannes Berg <johannes.berg@intel.com>2019-04-26 08:07:28 -0400
committerDavid S. Miller <davem@davemloft.net>2019-04-27 17:07:21 -0400
commit8cb081746c031fb164089322e2336a0bf5b3070c (patch)
tree1404b16c0859f9d67e67247a10833050c870a3cf /net/tipc/node.c
parent6f455f5f4e9c28aefaefbe18ce7304b499645d75 (diff)
netlink: make validation more configurable for future strictness
We currently have two levels of strict validation: 1) liberal (default) - undefined (type >= max) & NLA_UNSPEC attributes accepted - attribute length >= expected accepted - garbage at end of message accepted 2) strict (opt-in) - NLA_UNSPEC attributes accepted - attribute length >= expected accepted Split out parsing strictness into four different options: * TRAILING - check that there's no trailing data after parsing attributes (in message or nested) * MAXTYPE - reject attrs > max known type * UNSPEC - reject attributes with NLA_UNSPEC policy entries * STRICT_ATTRS - strictly validate attribute size The default for future things should be *everything*. The current *_strict() is a combination of TRAILING and MAXTYPE, and is renamed to _deprecated_strict(). The current regular parsing has none of this, and is renamed to *_parse_deprecated(). Additionally it allows us to selectively set one of the new flags even on old policies. Notably, the UNSPEC flag could be useful in this case, since it can be arranged (by filling in the policy) to not be an incompatible userspace ABI change, but would then going forward prevent forgetting attribute entries. Similar can apply to the POLICY flag. We end up with the following renames: * nla_parse -> nla_parse_deprecated * nla_parse_strict -> nla_parse_deprecated_strict * nlmsg_parse -> nlmsg_parse_deprecated * nlmsg_parse_strict -> nlmsg_parse_deprecated_strict * nla_parse_nested -> nla_parse_nested_deprecated * nla_validate_nested -> nla_validate_nested_deprecated Using spatch, of course: @@ expression TB, MAX, HEAD, LEN, POL, EXT; @@ -nla_parse(TB, MAX, HEAD, LEN, POL, EXT) +nla_parse_deprecated(TB, MAX, HEAD, LEN, POL, EXT) @@ expression NLH, HDRLEN, TB, MAX, POL, EXT; @@ -nlmsg_parse(NLH, HDRLEN, TB, MAX, POL, EXT) +nlmsg_parse_deprecated(NLH, HDRLEN, TB, MAX, POL, EXT) @@ expression NLH, HDRLEN, TB, MAX, POL, EXT; @@ -nlmsg_parse_strict(NLH, HDRLEN, TB, MAX, POL, EXT) +nlmsg_parse_deprecated_strict(NLH, HDRLEN, TB, MAX, POL, EXT) @@ expression TB, MAX, NLA, POL, EXT; @@ -nla_parse_nested(TB, MAX, NLA, POL, EXT) +nla_parse_nested_deprecated(TB, MAX, NLA, POL, EXT) @@ expression START, MAX, POL, EXT; @@ -nla_validate_nested(START, MAX, POL, EXT) +nla_validate_nested_deprecated(START, MAX, POL, EXT) @@ expression NLH, HDRLEN, MAX, POL, EXT; @@ -nlmsg_validate(NLH, HDRLEN, MAX, POL, EXT) +nlmsg_validate_deprecated(NLH, HDRLEN, MAX, POL, EXT) For this patch, don't actually add the strict, non-renamed versions yet so that it breaks compile if I get it wrong. Also, while at it, make nla_validate and nla_parse go down to a common __nla_validate_parse() function to avoid code duplication. Ultimately, this allows us to have very strict validation for every new caller of nla_parse()/nlmsg_parse() etc as re-introduced in the next patch, while existing things will continue to work as is. In effect then, this adds fully strict validation for any new command. Signed-off-by: Johannes Berg <johannes.berg@intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/tipc/node.c')
-rw-r--r--net/tipc/node.c38
1 files changed, 20 insertions, 18 deletions
diff --git a/net/tipc/node.c b/net/tipc/node.c
index 3777254a508f..0eb1bf850219 100644
--- a/net/tipc/node.c
+++ b/net/tipc/node.c
@@ -1885,9 +1885,9 @@ int tipc_nl_peer_rm(struct sk_buff *skb, struct genl_info *info)
1885 if (!info->attrs[TIPC_NLA_NET]) 1885 if (!info->attrs[TIPC_NLA_NET])
1886 return -EINVAL; 1886 return -EINVAL;
1887 1887
1888 err = nla_parse_nested(attrs, TIPC_NLA_NET_MAX, 1888 err = nla_parse_nested_deprecated(attrs, TIPC_NLA_NET_MAX,
1889 info->attrs[TIPC_NLA_NET], tipc_nl_net_policy, 1889 info->attrs[TIPC_NLA_NET],
1890 info->extack); 1890 tipc_nl_net_policy, info->extack);
1891 if (err) 1891 if (err)
1892 return err; 1892 return err;
1893 1893
@@ -2043,9 +2043,9 @@ int tipc_nl_node_set_link(struct sk_buff *skb, struct genl_info *info)
2043 if (!info->attrs[TIPC_NLA_LINK]) 2043 if (!info->attrs[TIPC_NLA_LINK])
2044 return -EINVAL; 2044 return -EINVAL;
2045 2045
2046 err = nla_parse_nested(attrs, TIPC_NLA_LINK_MAX, 2046 err = nla_parse_nested_deprecated(attrs, TIPC_NLA_LINK_MAX,
2047 info->attrs[TIPC_NLA_LINK], 2047 info->attrs[TIPC_NLA_LINK],
2048 tipc_nl_link_policy, info->extack); 2048 tipc_nl_link_policy, info->extack);
2049 if (err) 2049 if (err)
2050 return err; 2050 return err;
2051 2051
@@ -2119,9 +2119,9 @@ int tipc_nl_node_get_link(struct sk_buff *skb, struct genl_info *info)
2119 if (!info->attrs[TIPC_NLA_LINK]) 2119 if (!info->attrs[TIPC_NLA_LINK])
2120 return -EINVAL; 2120 return -EINVAL;
2121 2121
2122 err = nla_parse_nested(attrs, TIPC_NLA_LINK_MAX, 2122 err = nla_parse_nested_deprecated(attrs, TIPC_NLA_LINK_MAX,
2123 info->attrs[TIPC_NLA_LINK], 2123 info->attrs[TIPC_NLA_LINK],
2124 tipc_nl_link_policy, info->extack); 2124 tipc_nl_link_policy, info->extack);
2125 if (err) 2125 if (err)
2126 return err; 2126 return err;
2127 2127
@@ -2184,9 +2184,9 @@ int tipc_nl_node_reset_link_stats(struct sk_buff *skb, struct genl_info *info)
2184 if (!info->attrs[TIPC_NLA_LINK]) 2184 if (!info->attrs[TIPC_NLA_LINK])
2185 return -EINVAL; 2185 return -EINVAL;
2186 2186
2187 err = nla_parse_nested(attrs, TIPC_NLA_LINK_MAX, 2187 err = nla_parse_nested_deprecated(attrs, TIPC_NLA_LINK_MAX,
2188 info->attrs[TIPC_NLA_LINK], 2188 info->attrs[TIPC_NLA_LINK],
2189 tipc_nl_link_policy, info->extack); 2189 tipc_nl_link_policy, info->extack);
2190 if (err) 2190 if (err)
2191 return err; 2191 return err;
2192 2192
@@ -2324,9 +2324,10 @@ int tipc_nl_node_set_monitor(struct sk_buff *skb, struct genl_info *info)
2324 if (!info->attrs[TIPC_NLA_MON]) 2324 if (!info->attrs[TIPC_NLA_MON])
2325 return -EINVAL; 2325 return -EINVAL;
2326 2326
2327 err = nla_parse_nested(attrs, TIPC_NLA_MON_MAX, 2327 err = nla_parse_nested_deprecated(attrs, TIPC_NLA_MON_MAX,
2328 info->attrs[TIPC_NLA_MON], 2328 info->attrs[TIPC_NLA_MON],
2329 tipc_nl_monitor_policy, info->extack); 2329 tipc_nl_monitor_policy,
2330 info->extack);
2330 if (err) 2331 if (err)
2331 return err; 2332 return err;
2332 2333
@@ -2444,9 +2445,10 @@ int tipc_nl_node_dump_monitor_peer(struct sk_buff *skb,
2444 if (!attrs[TIPC_NLA_MON]) 2445 if (!attrs[TIPC_NLA_MON])
2445 return -EINVAL; 2446 return -EINVAL;
2446 2447
2447 err = nla_parse_nested(mon, TIPC_NLA_MON_MAX, 2448 err = nla_parse_nested_deprecated(mon, TIPC_NLA_MON_MAX,
2448 attrs[TIPC_NLA_MON], 2449 attrs[TIPC_NLA_MON],
2449 tipc_nl_monitor_policy, NULL); 2450 tipc_nl_monitor_policy,
2451 NULL);
2450 if (err) 2452 if (err)
2451 return err; 2453 return err;
2452 2454