提交 d4cada4a 编写于 作者: E Eric Dumazet 提交者: David S. Miller

udp: split sk_hash into two u16 hashes

Union sk_hash with two u16 hashes for udp (no extra memory taken)

One 16 bits hash on (local port) value (the previous udp 'hash')

One 16 bits hash on (local address, local port) values, initialized
but not yet used. This second hash is using jenkin hash for better
distribution.

Because the 'port' is xored later, a partial hash is performed
on local address + net_hash_mix(net)
Signed-off-by: NEric Dumazet <eric.dumazet@gmail.com>
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
上级 fdcc8aa9
...@@ -55,6 +55,8 @@ static inline int udp_hashfn(struct net *net, unsigned num, unsigned mask) ...@@ -55,6 +55,8 @@ static inline int udp_hashfn(struct net *net, unsigned num, unsigned mask)
struct udp_sock { struct udp_sock {
/* inet_sock has to be the first member */ /* inet_sock has to be the first member */
struct inet_sock inet; struct inet_sock inet;
#define udp_port_hash inet.sk.__sk_common.skc_u16hashes[0]
#define udp_portaddr_hash inet.sk.__sk_common.skc_u16hashes[1]
int pending; /* Any pending frames ? */ int pending; /* Any pending frames ? */
unsigned int corkflag; /* Cork is required */ unsigned int corkflag; /* Cork is required */
__u16 encap_type; /* Is this an Encapsulation socket? */ __u16 encap_type; /* Is this an Encapsulation socket? */
......
...@@ -109,6 +109,7 @@ struct net; ...@@ -109,6 +109,7 @@ struct net;
* @skc_refcnt: reference count * @skc_refcnt: reference count
* @skc_tx_queue_mapping: tx queue number for this connection * @skc_tx_queue_mapping: tx queue number for this connection
* @skc_hash: hash value used with various protocol lookup tables * @skc_hash: hash value used with various protocol lookup tables
* @skc_u16hashes: two u16 hash values used by UDP lookup tables
* @skc_family: network address family * @skc_family: network address family
* @skc_state: Connection state * @skc_state: Connection state
* @skc_reuse: %SO_REUSEADDR setting * @skc_reuse: %SO_REUSEADDR setting
...@@ -131,7 +132,10 @@ struct sock_common { ...@@ -131,7 +132,10 @@ struct sock_common {
atomic_t skc_refcnt; atomic_t skc_refcnt;
int skc_tx_queue_mapping; int skc_tx_queue_mapping;
unsigned int skc_hash; union {
unsigned int skc_hash;
__u16 skc_u16hashes[2];
};
unsigned short skc_family; unsigned short skc_family;
volatile unsigned char skc_state; volatile unsigned char skc_state;
unsigned char skc_reuse; unsigned char skc_reuse;
......
...@@ -138,13 +138,14 @@ static int udp_lib_lport_inuse(struct net *net, __u16 num, ...@@ -138,13 +138,14 @@ static int udp_lib_lport_inuse(struct net *net, __u16 num,
sk_nulls_for_each(sk2, node, &hslot->head) sk_nulls_for_each(sk2, node, &hslot->head)
if (net_eq(sock_net(sk2), net) && if (net_eq(sock_net(sk2), net) &&
sk2 != sk && sk2 != sk &&
(bitmap || sk2->sk_hash == num) && (bitmap || udp_sk(sk2)->udp_port_hash == num) &&
(!sk2->sk_reuse || !sk->sk_reuse) && (!sk2->sk_reuse || !sk->sk_reuse) &&
(!sk2->sk_bound_dev_if || !sk->sk_bound_dev_if (!sk2->sk_bound_dev_if || !sk->sk_bound_dev_if
|| sk2->sk_bound_dev_if == sk->sk_bound_dev_if) && || sk2->sk_bound_dev_if == sk->sk_bound_dev_if) &&
(*saddr_comp)(sk, sk2)) { (*saddr_comp)(sk, sk2)) {
if (bitmap) if (bitmap)
__set_bit(sk2->sk_hash >> log, bitmap); __set_bit(udp_sk(sk2)->udp_port_hash >> log,
bitmap);
else else
return 1; return 1;
} }
...@@ -215,7 +216,8 @@ int udp_lib_get_port(struct sock *sk, unsigned short snum, ...@@ -215,7 +216,8 @@ int udp_lib_get_port(struct sock *sk, unsigned short snum,
} }
found: found:
inet_sk(sk)->inet_num = snum; inet_sk(sk)->inet_num = snum;
sk->sk_hash = snum; udp_sk(sk)->udp_port_hash = snum;
udp_sk(sk)->udp_portaddr_hash ^= snum;
if (sk_unhashed(sk)) { if (sk_unhashed(sk)) {
sk_nulls_add_node_rcu(sk, &hslot->head); sk_nulls_add_node_rcu(sk, &hslot->head);
hslot->count++; hslot->count++;
...@@ -238,8 +240,19 @@ static int ipv4_rcv_saddr_equal(const struct sock *sk1, const struct sock *sk2) ...@@ -238,8 +240,19 @@ static int ipv4_rcv_saddr_equal(const struct sock *sk1, const struct sock *sk2)
inet1->inet_rcv_saddr == inet2->inet_rcv_saddr)); inet1->inet_rcv_saddr == inet2->inet_rcv_saddr));
} }
static unsigned int udp4_portaddr_hash(struct net *net, __be32 saddr,
unsigned int port)
{
return jhash_1word(saddr, net_hash_mix(net)) ^ port;
}
int udp_v4_get_port(struct sock *sk, unsigned short snum) int udp_v4_get_port(struct sock *sk, unsigned short snum)
{ {
/* precompute partial secondary hash */
udp_sk(sk)->udp_portaddr_hash =
udp4_portaddr_hash(sock_net(sk),
inet_sk(sk)->inet_rcv_saddr,
0);
return udp_lib_get_port(sk, snum, ipv4_rcv_saddr_equal); return udp_lib_get_port(sk, snum, ipv4_rcv_saddr_equal);
} }
...@@ -249,7 +262,7 @@ static inline int compute_score(struct sock *sk, struct net *net, __be32 saddr, ...@@ -249,7 +262,7 @@ static inline int compute_score(struct sock *sk, struct net *net, __be32 saddr,
{ {
int score = -1; int score = -1;
if (net_eq(sock_net(sk), net) && sk->sk_hash == hnum && if (net_eq(sock_net(sk), net) && udp_sk(sk)->udp_port_hash == hnum &&
!ipv6_only_sock(sk)) { !ipv6_only_sock(sk)) {
struct inet_sock *inet = inet_sk(sk); struct inet_sock *inet = inet_sk(sk);
...@@ -360,7 +373,7 @@ static inline struct sock *udp_v4_mcast_next(struct net *net, struct sock *sk, ...@@ -360,7 +373,7 @@ static inline struct sock *udp_v4_mcast_next(struct net *net, struct sock *sk,
struct inet_sock *inet = inet_sk(s); struct inet_sock *inet = inet_sk(s);
if (!net_eq(sock_net(s), net) || if (!net_eq(sock_net(s), net) ||
s->sk_hash != hnum || udp_sk(s)->udp_port_hash != hnum ||
(inet->inet_daddr && inet->inet_daddr != rmt_addr) || (inet->inet_daddr && inet->inet_daddr != rmt_addr) ||
(inet->inet_dport != rmt_port && inet->inet_dport) || (inet->inet_dport != rmt_port && inet->inet_dport) ||
(inet->inet_rcv_saddr && (inet->inet_rcv_saddr &&
...@@ -1050,7 +1063,7 @@ void udp_lib_unhash(struct sock *sk) ...@@ -1050,7 +1063,7 @@ void udp_lib_unhash(struct sock *sk)
if (sk_hashed(sk)) { if (sk_hashed(sk)) {
struct udp_table *udptable = sk->sk_prot->h.udp_table; struct udp_table *udptable = sk->sk_prot->h.udp_table;
struct udp_hslot *hslot = udp_hashslot(udptable, sock_net(sk), struct udp_hslot *hslot = udp_hashslot(udptable, sock_net(sk),
sk->sk_hash); udp_sk(sk)->udp_port_hash);
spin_lock_bh(&hslot->lock); spin_lock_bh(&hslot->lock);
if (sk_nulls_del_node_init_rcu(sk)) { if (sk_nulls_del_node_init_rcu(sk)) {
......
...@@ -81,8 +81,30 @@ int ipv6_rcv_saddr_equal(const struct sock *sk, const struct sock *sk2) ...@@ -81,8 +81,30 @@ int ipv6_rcv_saddr_equal(const struct sock *sk, const struct sock *sk2)
return 0; return 0;
} }
static unsigned int udp6_portaddr_hash(struct net *net,
const struct in6_addr *addr6,
unsigned int port)
{
unsigned int hash, mix = net_hash_mix(net);
if (ipv6_addr_any(addr6))
hash = jhash_1word(0, mix);
else if (ipv6_addr_type(addr6) == IPV6_ADDR_MAPPED)
hash = jhash_1word(addr6->s6_addr32[3], mix);
else
hash = jhash2(addr6->s6_addr32, 4, mix);
return hash ^ port;
}
int udp_v6_get_port(struct sock *sk, unsigned short snum) int udp_v6_get_port(struct sock *sk, unsigned short snum)
{ {
/* precompute partial secondary hash */
udp_sk(sk)->udp_portaddr_hash =
udp6_portaddr_hash(sock_net(sk),
&inet6_sk(sk)->rcv_saddr,
0);
return udp_lib_get_port(sk, snum, ipv6_rcv_saddr_equal); return udp_lib_get_port(sk, snum, ipv6_rcv_saddr_equal);
} }
...@@ -94,7 +116,7 @@ static inline int compute_score(struct sock *sk, struct net *net, ...@@ -94,7 +116,7 @@ static inline int compute_score(struct sock *sk, struct net *net,
{ {
int score = -1; int score = -1;
if (net_eq(sock_net(sk), net) && sk->sk_hash == hnum && if (net_eq(sock_net(sk), net) && udp_sk(sk)->udp_port_hash == hnum &&
sk->sk_family == PF_INET6) { sk->sk_family == PF_INET6) {
struct ipv6_pinfo *np = inet6_sk(sk); struct ipv6_pinfo *np = inet6_sk(sk);
struct inet_sock *inet = inet_sk(sk); struct inet_sock *inet = inet_sk(sk);
...@@ -415,7 +437,8 @@ static struct sock *udp_v6_mcast_next(struct net *net, struct sock *sk, ...@@ -415,7 +437,8 @@ static struct sock *udp_v6_mcast_next(struct net *net, struct sock *sk,
if (!net_eq(sock_net(s), net)) if (!net_eq(sock_net(s), net))
continue; continue;
if (s->sk_hash == num && s->sk_family == PF_INET6) { if (udp_sk(s)->udp_port_hash == num &&
s->sk_family == PF_INET6) {
struct ipv6_pinfo *np = inet6_sk(s); struct ipv6_pinfo *np = inet6_sk(s);
if (inet->inet_dport) { if (inet->inet_dport) {
if (inet->inet_dport != rmt_port) if (inet->inet_dport != rmt_port)
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册