diff options
author | Hannes Frederic Sowa <hannes@stressinduktion.org> | 2013-10-23 05:06:56 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-10-23 17:01:40 -0400 |
commit | b1190570b451fb9fd77be8c115fcdb418c5108a5 (patch) | |
tree | 6b9d50498ba3b2904b021da902362f90105ded8d /net/ipv6 | |
parent | e7b519ba55aeb675daee1d304e80d752c385f7f0 (diff) |
ipv6: split inet6_hash_frag for netfilter and initialize secrets with net_get_random_once
Defer the fragmentation hash secret initialization for IPv6 like the
previous patch did for IPv4.
Because the netfilter logic reuses the hash secret we have to split it
first. Thus introduce a new nf_hash_frag function which takes care to
seed the hash secret.
Cc: David S. Miller <davem@davemloft.net>
Cc: Eric Dumazet <edumazet@google.com>
Cc: Pablo Neira Ayuso <pablo@netfilter.org>
Signed-off-by: Hannes Frederic Sowa <hannes@stressinduktion.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv6')
-rw-r--r-- | net/ipv6/netfilter/nf_conntrack_reasm.c | 16 | ||||
-rw-r--r-- | net/ipv6/reassembly.c | 12 |
2 files changed, 20 insertions, 8 deletions
diff --git a/net/ipv6/netfilter/nf_conntrack_reasm.c b/net/ipv6/netfilter/nf_conntrack_reasm.c index dffdc1a389c5..4a258263d8ec 100644 --- a/net/ipv6/netfilter/nf_conntrack_reasm.c +++ b/net/ipv6/netfilter/nf_conntrack_reasm.c | |||
@@ -144,12 +144,24 @@ static inline u8 ip6_frag_ecn(const struct ipv6hdr *ipv6h) | |||
144 | return 1 << (ipv6_get_dsfield(ipv6h) & INET_ECN_MASK); | 144 | return 1 << (ipv6_get_dsfield(ipv6h) & INET_ECN_MASK); |
145 | } | 145 | } |
146 | 146 | ||
147 | static unsigned int nf_hash_frag(__be32 id, const struct in6_addr *saddr, | ||
148 | const struct in6_addr *daddr) | ||
149 | { | ||
150 | u32 c; | ||
151 | |||
152 | net_get_random_once(&nf_frags.rnd, sizeof(nf_frags.rnd)); | ||
153 | c = jhash_3words(ipv6_addr_hash(saddr), ipv6_addr_hash(daddr), | ||
154 | (__force u32)id, nf_frags.rnd); | ||
155 | return c & (INETFRAGS_HASHSZ - 1); | ||
156 | } | ||
157 | |||
158 | |||
147 | static unsigned int nf_hashfn(struct inet_frag_queue *q) | 159 | static unsigned int nf_hashfn(struct inet_frag_queue *q) |
148 | { | 160 | { |
149 | const struct frag_queue *nq; | 161 | const struct frag_queue *nq; |
150 | 162 | ||
151 | nq = container_of(q, struct frag_queue, q); | 163 | nq = container_of(q, struct frag_queue, q); |
152 | return inet6_hash_frag(nq->id, &nq->saddr, &nq->daddr, nf_frags.rnd); | 164 | return nf_hash_frag(nq->id, &nq->saddr, &nq->daddr); |
153 | } | 165 | } |
154 | 166 | ||
155 | static void nf_skb_free(struct sk_buff *skb) | 167 | static void nf_skb_free(struct sk_buff *skb) |
@@ -185,7 +197,7 @@ static inline struct frag_queue *fq_find(struct net *net, __be32 id, | |||
185 | arg.ecn = ecn; | 197 | arg.ecn = ecn; |
186 | 198 | ||
187 | read_lock_bh(&nf_frags.lock); | 199 | read_lock_bh(&nf_frags.lock); |
188 | hash = inet6_hash_frag(id, src, dst, nf_frags.rnd); | 200 | hash = nf_hash_frag(id, src, dst); |
189 | 201 | ||
190 | q = inet_frag_find(&net->nf_frag.frags, &nf_frags, &arg, hash); | 202 | q = inet_frag_find(&net->nf_frag.frags, &nf_frags, &arg, hash); |
191 | local_bh_enable(); | 203 | local_bh_enable(); |
diff --git a/net/ipv6/reassembly.c b/net/ipv6/reassembly.c index 1aeb473b2cc6..cc85a9ba5010 100644 --- a/net/ipv6/reassembly.c +++ b/net/ipv6/reassembly.c | |||
@@ -82,24 +82,24 @@ static int ip6_frag_reasm(struct frag_queue *fq, struct sk_buff *prev, | |||
82 | * callers should be careful not to use the hash value outside the ipfrag_lock | 82 | * callers should be careful not to use the hash value outside the ipfrag_lock |
83 | * as doing so could race with ipfrag_hash_rnd being recalculated. | 83 | * as doing so could race with ipfrag_hash_rnd being recalculated. |
84 | */ | 84 | */ |
85 | unsigned int inet6_hash_frag(__be32 id, const struct in6_addr *saddr, | 85 | static unsigned int inet6_hash_frag(__be32 id, const struct in6_addr *saddr, |
86 | const struct in6_addr *daddr, u32 rnd) | 86 | const struct in6_addr *daddr) |
87 | { | 87 | { |
88 | u32 c; | 88 | u32 c; |
89 | 89 | ||
90 | net_get_random_once(&ip6_frags.rnd, sizeof(ip6_frags.rnd)); | ||
90 | c = jhash_3words(ipv6_addr_hash(saddr), ipv6_addr_hash(daddr), | 91 | c = jhash_3words(ipv6_addr_hash(saddr), ipv6_addr_hash(daddr), |
91 | (__force u32)id, rnd); | 92 | (__force u32)id, ip6_frags.rnd); |
92 | 93 | ||
93 | return c & (INETFRAGS_HASHSZ - 1); | 94 | return c & (INETFRAGS_HASHSZ - 1); |
94 | } | 95 | } |
95 | EXPORT_SYMBOL_GPL(inet6_hash_frag); | ||
96 | 96 | ||
97 | static unsigned int ip6_hashfn(struct inet_frag_queue *q) | 97 | static unsigned int ip6_hashfn(struct inet_frag_queue *q) |
98 | { | 98 | { |
99 | struct frag_queue *fq; | 99 | struct frag_queue *fq; |
100 | 100 | ||
101 | fq = container_of(q, struct frag_queue, q); | 101 | fq = container_of(q, struct frag_queue, q); |
102 | return inet6_hash_frag(fq->id, &fq->saddr, &fq->daddr, ip6_frags.rnd); | 102 | return inet6_hash_frag(fq->id, &fq->saddr, &fq->daddr); |
103 | } | 103 | } |
104 | 104 | ||
105 | bool ip6_frag_match(struct inet_frag_queue *q, void *a) | 105 | bool ip6_frag_match(struct inet_frag_queue *q, void *a) |
@@ -193,7 +193,7 @@ fq_find(struct net *net, __be32 id, const struct in6_addr *src, | |||
193 | arg.ecn = ecn; | 193 | arg.ecn = ecn; |
194 | 194 | ||
195 | read_lock(&ip6_frags.lock); | 195 | read_lock(&ip6_frags.lock); |
196 | hash = inet6_hash_frag(id, src, dst, ip6_frags.rnd); | 196 | hash = inet6_hash_frag(id, src, dst); |
197 | 197 | ||
198 | q = inet_frag_find(&net->ipv6.frags, &ip6_frags, &arg, hash); | 198 | q = inet_frag_find(&net->ipv6.frags, &ip6_frags, &arg, hash); |
199 | if (IS_ERR_OR_NULL(q)) { | 199 | if (IS_ERR_OR_NULL(q)) { |