diff options
author | Eric Dumazet <edumazet@google.com> | 2017-10-05 05:50:07 -0400 |
---|---|---|
committer | Pablo Neira Ayuso <pablo@netfilter.org> | 2017-10-06 09:04:05 -0400 |
commit | e466af75c074e76107ae1cd5a2823e9c61894ffb (patch) | |
tree | d89583ebd5b7d23418dbc0d44472f60eecd66ccd /net | |
parent | 5f9bfe0ef622a7bb9707c22ceb4b6451e1e2cb7b (diff) |
netfilter: x_tables: avoid stack-out-of-bounds read in xt_copy_counters_from_user
syzkaller reports an out of bound read in strlcpy(), triggered
by xt_copy_counters_from_user()
Fix this by using memcpy(), then forcing a zero byte at the last position
of the destination, as Florian did for the non COMPAT code.
Fixes: d7591f0c41ce ("netfilter: x_tables: introduce and use xt_copy_counters_from_user")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Cc: Willem de Bruijn <willemb@google.com>
Acked-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
Diffstat (limited to 'net')
-rw-r--r-- | net/netfilter/x_tables.c | 4 |
1 files changed, 2 insertions, 2 deletions
diff --git a/net/netfilter/x_tables.c b/net/netfilter/x_tables.c index c83a3b5e1c6c..d8571f414208 100644 --- a/net/netfilter/x_tables.c +++ b/net/netfilter/x_tables.c | |||
@@ -892,7 +892,7 @@ void *xt_copy_counters_from_user(const void __user *user, unsigned int len, | |||
892 | if (copy_from_user(&compat_tmp, user, sizeof(compat_tmp)) != 0) | 892 | if (copy_from_user(&compat_tmp, user, sizeof(compat_tmp)) != 0) |
893 | return ERR_PTR(-EFAULT); | 893 | return ERR_PTR(-EFAULT); |
894 | 894 | ||
895 | strlcpy(info->name, compat_tmp.name, sizeof(info->name)); | 895 | memcpy(info->name, compat_tmp.name, sizeof(info->name) - 1); |
896 | info->num_counters = compat_tmp.num_counters; | 896 | info->num_counters = compat_tmp.num_counters; |
897 | user += sizeof(compat_tmp); | 897 | user += sizeof(compat_tmp); |
898 | } else | 898 | } else |
@@ -905,9 +905,9 @@ void *xt_copy_counters_from_user(const void __user *user, unsigned int len, | |||
905 | if (copy_from_user(info, user, sizeof(*info)) != 0) | 905 | if (copy_from_user(info, user, sizeof(*info)) != 0) |
906 | return ERR_PTR(-EFAULT); | 906 | return ERR_PTR(-EFAULT); |
907 | 907 | ||
908 | info->name[sizeof(info->name) - 1] = '\0'; | ||
909 | user += sizeof(*info); | 908 | user += sizeof(*info); |
910 | } | 909 | } |
910 | info->name[sizeof(info->name) - 1] = '\0'; | ||
911 | 911 | ||
912 | size = sizeof(struct xt_counters); | 912 | size = sizeof(struct xt_counters); |
913 | size *= info->num_counters; | 913 | size *= info->num_counters; |