diff options
-rw-r--r-- | include/net/netns/ipv4.h | 2 | ||||
-rw-r--r-- | net/ipv4/tcp_metrics.c | 25 |
2 files changed, 11 insertions, 16 deletions
diff --git a/include/net/netns/ipv4.h b/include/net/netns/ipv4.h index d909c7fc3da1..0ffb8e31f3cd 100644 --- a/include/net/netns/ipv4.h +++ b/include/net/netns/ipv4.h | |||
@@ -40,7 +40,7 @@ struct netns_ipv4 { | |||
40 | struct sock **icmp_sk; | 40 | struct sock **icmp_sk; |
41 | struct inet_peer_base *peers; | 41 | struct inet_peer_base *peers; |
42 | struct tcpm_hash_bucket *tcp_metrics_hash; | 42 | struct tcpm_hash_bucket *tcp_metrics_hash; |
43 | unsigned int tcp_metrics_hash_mask; | 43 | unsigned int tcp_metrics_hash_log; |
44 | struct netns_frags frags; | 44 | struct netns_frags frags; |
45 | #ifdef CONFIG_NETFILTER | 45 | #ifdef CONFIG_NETFILTER |
46 | struct xt_table *iptable_filter; | 46 | struct xt_table *iptable_filter; |
diff --git a/net/ipv4/tcp_metrics.c b/net/ipv4/tcp_metrics.c index 99779ae44f64..992f1bff4fc6 100644 --- a/net/ipv4/tcp_metrics.c +++ b/net/ipv4/tcp_metrics.c | |||
@@ -7,6 +7,7 @@ | |||
7 | #include <linux/slab.h> | 7 | #include <linux/slab.h> |
8 | #include <linux/init.h> | 8 | #include <linux/init.h> |
9 | #include <linux/tcp.h> | 9 | #include <linux/tcp.h> |
10 | #include <linux/hash.h> | ||
10 | 11 | ||
11 | #include <net/inet_connection_sock.h> | 12 | #include <net/inet_connection_sock.h> |
12 | #include <net/net_namespace.h> | 13 | #include <net/net_namespace.h> |
@@ -228,10 +229,8 @@ static struct tcp_metrics_block *__tcp_get_metrics_req(struct request_sock *req, | |||
228 | return NULL; | 229 | return NULL; |
229 | } | 230 | } |
230 | 231 | ||
231 | hash ^= (hash >> 24) ^ (hash >> 16) ^ (hash >> 8); | ||
232 | |||
233 | net = dev_net(dst->dev); | 232 | net = dev_net(dst->dev); |
234 | hash &= net->ipv4.tcp_metrics_hash_mask; | 233 | hash = hash_32(hash, net->ipv4.tcp_metrics_hash_log); |
235 | 234 | ||
236 | for (tm = rcu_dereference(net->ipv4.tcp_metrics_hash[hash].chain); tm; | 235 | for (tm = rcu_dereference(net->ipv4.tcp_metrics_hash[hash].chain); tm; |
237 | tm = rcu_dereference(tm->tcpm_next)) { | 236 | tm = rcu_dereference(tm->tcpm_next)) { |
@@ -265,10 +264,8 @@ static struct tcp_metrics_block *__tcp_get_metrics_tw(struct inet_timewait_sock | |||
265 | return NULL; | 264 | return NULL; |
266 | } | 265 | } |
267 | 266 | ||
268 | hash ^= (hash >> 24) ^ (hash >> 16) ^ (hash >> 8); | ||
269 | |||
270 | net = twsk_net(tw); | 267 | net = twsk_net(tw); |
271 | hash &= net->ipv4.tcp_metrics_hash_mask; | 268 | hash = hash_32(hash, net->ipv4.tcp_metrics_hash_log); |
272 | 269 | ||
273 | for (tm = rcu_dereference(net->ipv4.tcp_metrics_hash[hash].chain); tm; | 270 | for (tm = rcu_dereference(net->ipv4.tcp_metrics_hash[hash].chain); tm; |
274 | tm = rcu_dereference(tm->tcpm_next)) { | 271 | tm = rcu_dereference(tm->tcpm_next)) { |
@@ -302,10 +299,8 @@ static struct tcp_metrics_block *tcp_get_metrics(struct sock *sk, | |||
302 | return NULL; | 299 | return NULL; |
303 | } | 300 | } |
304 | 301 | ||
305 | hash ^= (hash >> 24) ^ (hash >> 16) ^ (hash >> 8); | ||
306 | |||
307 | net = dev_net(dst->dev); | 302 | net = dev_net(dst->dev); |
308 | hash &= net->ipv4.tcp_metrics_hash_mask; | 303 | hash = hash_32(hash, net->ipv4.tcp_metrics_hash_log); |
309 | 304 | ||
310 | tm = __tcp_get_metrics(&addr, net, hash); | 305 | tm = __tcp_get_metrics(&addr, net, hash); |
311 | reclaim = false; | 306 | reclaim = false; |
@@ -694,7 +689,7 @@ void tcp_fastopen_cache_set(struct sock *sk, u16 mss, | |||
694 | rcu_read_unlock(); | 689 | rcu_read_unlock(); |
695 | } | 690 | } |
696 | 691 | ||
697 | static unsigned long tcpmhash_entries; | 692 | static unsigned int tcpmhash_entries; |
698 | static int __init set_tcpmhash_entries(char *str) | 693 | static int __init set_tcpmhash_entries(char *str) |
699 | { | 694 | { |
700 | ssize_t ret; | 695 | ssize_t ret; |
@@ -702,7 +697,7 @@ static int __init set_tcpmhash_entries(char *str) | |||
702 | if (!str) | 697 | if (!str) |
703 | return 0; | 698 | return 0; |
704 | 699 | ||
705 | ret = kstrtoul(str, 0, &tcpmhash_entries); | 700 | ret = kstrtouint(str, 0, &tcpmhash_entries); |
706 | if (ret) | 701 | if (ret) |
707 | return 0; | 702 | return 0; |
708 | 703 | ||
@@ -712,7 +707,8 @@ __setup("tcpmhash_entries=", set_tcpmhash_entries); | |||
712 | 707 | ||
713 | static int __net_init tcp_net_metrics_init(struct net *net) | 708 | static int __net_init tcp_net_metrics_init(struct net *net) |
714 | { | 709 | { |
715 | int slots, size; | 710 | size_t size; |
711 | unsigned int slots; | ||
716 | 712 | ||
717 | slots = tcpmhash_entries; | 713 | slots = tcpmhash_entries; |
718 | if (!slots) { | 714 | if (!slots) { |
@@ -722,14 +718,13 @@ static int __net_init tcp_net_metrics_init(struct net *net) | |||
722 | slots = 8 * 1024; | 718 | slots = 8 * 1024; |
723 | } | 719 | } |
724 | 720 | ||
725 | size = slots * sizeof(struct tcpm_hash_bucket); | 721 | net->ipv4.tcp_metrics_hash_log = order_base_2(slots); |
722 | size = sizeof(struct tcpm_hash_bucket) << net->ipv4.tcp_metrics_hash_log; | ||
726 | 723 | ||
727 | net->ipv4.tcp_metrics_hash = kzalloc(size, GFP_KERNEL); | 724 | net->ipv4.tcp_metrics_hash = kzalloc(size, GFP_KERNEL); |
728 | if (!net->ipv4.tcp_metrics_hash) | 725 | if (!net->ipv4.tcp_metrics_hash) |
729 | return -ENOMEM; | 726 | return -ENOMEM; |
730 | 727 | ||
731 | net->ipv4.tcp_metrics_hash_mask = (slots - 1); | ||
732 | |||
733 | return 0; | 728 | return 0; |
734 | } | 729 | } |
735 | 730 | ||