diff options
Diffstat (limited to 'net/ipv4/esp4.c')
| -rw-r--r-- | net/ipv4/esp4.c | 53 | 
1 files changed, 7 insertions, 46 deletions
| diff --git a/net/ipv4/esp4.c b/net/ipv4/esp4.c index 0e4076866c0a..f14a41ee4aa1 100644 --- a/net/ipv4/esp4.c +++ b/net/ipv4/esp4.c @@ -120,47 +120,16 @@ static void esp_ssg_unref(struct xfrm_state *x, void *tmp, struct sk_buff *skb)  }  #ifdef CONFIG_INET_ESPINTCP -struct esp_tcp_sk { -	struct sock *sk; -	struct rcu_head rcu; -}; - -static void esp_free_tcp_sk(struct rcu_head *head) -{ -	struct esp_tcp_sk *esk = container_of(head, struct esp_tcp_sk, rcu); - -	sock_put(esk->sk); -	kfree(esk); -} -  static struct sock *esp_find_tcp_sk(struct xfrm_state *x)  {  	struct xfrm_encap_tmpl *encap = x->encap;  	struct net *net = xs_net(x); -	struct esp_tcp_sk *esk;  	__be16 sport, dport; -	struct sock *nsk;  	struct sock *sk; -	sk = rcu_dereference(x->encap_sk); -	if (sk && sk->sk_state == TCP_ESTABLISHED) -		return sk; -  	spin_lock_bh(&x->lock);  	sport = encap->encap_sport;  	dport = encap->encap_dport; -	nsk = rcu_dereference_protected(x->encap_sk, -					lockdep_is_held(&x->lock)); -	if (sk && sk == nsk) { -		esk = kmalloc(sizeof(*esk), GFP_ATOMIC); -		if (!esk) { -			spin_unlock_bh(&x->lock); -			return ERR_PTR(-ENOMEM); -		} -		RCU_INIT_POINTER(x->encap_sk, NULL); -		esk->sk = sk; -		call_rcu(&esk->rcu, esp_free_tcp_sk); -	}  	spin_unlock_bh(&x->lock);  	sk = inet_lookup_established(net, net->ipv4.tcp_death_row.hashinfo, x->id.daddr.a4, @@ -173,20 +142,6 @@ static struct sock *esp_find_tcp_sk(struct xfrm_state *x)  		return ERR_PTR(-EINVAL);  	} -	spin_lock_bh(&x->lock); -	nsk = rcu_dereference_protected(x->encap_sk, -					lockdep_is_held(&x->lock)); -	if (encap->encap_sport != sport || -	    encap->encap_dport != dport) { -		sock_put(sk); -		sk = nsk ?: ERR_PTR(-EREMCHG); -	} else if (sk == nsk) { -		sock_put(sk); -	} else { -		rcu_assign_pointer(x->encap_sk, sk); -	} -	spin_unlock_bh(&x->lock); -  	return sk;  } @@ -199,8 +154,10 @@ static int esp_output_tcp_finish(struct xfrm_state *x, struct sk_buff *skb)  	sk = esp_find_tcp_sk(x);  	err = PTR_ERR_OR_ZERO(sk); -	if (err) +	if (err) { +		kfree_skb(skb);  		goto out; +	}  	bh_lock_sock(sk);  	if (sock_owned_by_user(sk)) @@ -209,6 +166,8 @@ static int esp_output_tcp_finish(struct xfrm_state *x, struct sk_buff *skb)  		err = espintcp_push_skb(sk, skb);  	bh_unlock_sock(sk); +	sock_put(sk); +  out:  	rcu_read_unlock();  	return err; @@ -392,6 +351,8 @@ static struct ip_esp_hdr *esp_output_tcp_encap(struct xfrm_state *x,  	if (IS_ERR(sk))  		return ERR_CAST(sk); +	sock_put(sk); +  	*lenp = htons(len);  	esph = (struct ip_esp_hdr *)(lenp + 1); | 
