diff options
Diffstat (limited to 'net/sctp/socket.c')
| -rw-r--r-- | net/sctp/socket.c | 56 | 
1 files changed, 28 insertions, 28 deletions
| diff --git a/net/sctp/socket.c b/net/sctp/socket.c index 2b6c88b9a038..4d60f1e42f42 100644 --- a/net/sctp/socket.c +++ b/net/sctp/socket.c @@ -185,13 +185,13 @@ static void sctp_for_each_tx_datachunk(struct sctp_association *asoc,  		list_for_each_entry(chunk, &t->transmitted, transmitted_list)  			cb(chunk); -	list_for_each_entry(chunk, &q->retransmit, list) +	list_for_each_entry(chunk, &q->retransmit, transmitted_list)  		cb(chunk); -	list_for_each_entry(chunk, &q->sacked, list) +	list_for_each_entry(chunk, &q->sacked, transmitted_list)  		cb(chunk); -	list_for_each_entry(chunk, &q->abandoned, list) +	list_for_each_entry(chunk, &q->abandoned, transmitted_list)  		cb(chunk);  	list_for_each_entry(chunk, &q->out_chunk_list, list) @@ -352,6 +352,18 @@ static struct sctp_af *sctp_sockaddr_af(struct sctp_sock *opt,  	return af;  } +static void sctp_auto_asconf_init(struct sctp_sock *sp) +{ +	struct net *net = sock_net(&sp->inet.sk); + +	if (net->sctp.default_auto_asconf) { +		spin_lock(&net->sctp.addr_wq_lock); +		list_add_tail(&sp->auto_asconf_list, &net->sctp.auto_asconf_splist); +		spin_unlock(&net->sctp.addr_wq_lock); +		sp->do_auto_asconf = 1; +	} +} +  /* Bind a local address either to an endpoint or to an association.  */  static int sctp_do_bind(struct sock *sk, union sctp_addr *addr, int len)  { @@ -414,8 +426,10 @@ static int sctp_do_bind(struct sock *sk, union sctp_addr *addr, int len)  	}  	/* Refresh ephemeral port.  */ -	if (!bp->port) +	if (!bp->port) {  		bp->port = inet_sk(sk)->inet_num; +		sctp_auto_asconf_init(sp); +	}  	/* Add the address to the bind address list.  	 * Use GFP_ATOMIC since BHs will be disabled. @@ -4161,19 +4175,6 @@ static int sctp_init_sock(struct sock *sk)  	sk_sockets_allocated_inc(sk);  	sock_prot_inuse_add(net, sk->sk_prot, 1); -	/* Nothing can fail after this block, otherwise -	 * sctp_destroy_sock() will be called without addr_wq_lock held -	 */ -	if (net->sctp.default_auto_asconf) { -		spin_lock(&sock_net(sk)->sctp.addr_wq_lock); -		list_add_tail(&sp->auto_asconf_list, -		    &net->sctp.auto_asconf_splist); -		sp->do_auto_asconf = 1; -		spin_unlock(&sock_net(sk)->sctp.addr_wq_lock); -	} else { -		sp->do_auto_asconf = 0; -	} -  	local_bh_enable();  	return 0; @@ -6170,9 +6171,10 @@ static int sctp_getsockopt(struct sock *sk, int level, int optname,  	return retval;  } -static void sctp_hash(struct sock *sk) +static int sctp_hash(struct sock *sk)  {  	/* STUB */ +	return 0;  }  static void sctp_unhash(struct sock *sk) @@ -6206,8 +6208,6 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)  	pr_debug("%s: begins, snum:%d\n", __func__, snum); -	local_bh_disable(); -  	if (snum == 0) {  		/* Search for an available port. */  		int low, high, remaining, index; @@ -6226,20 +6226,21 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)  				continue;  			index = sctp_phashfn(sock_net(sk), rover);  			head = &sctp_port_hashtable[index]; -			spin_lock(&head->lock); +			spin_lock_bh(&head->lock);  			sctp_for_each_hentry(pp, &head->chain)  				if ((pp->port == rover) &&  				    net_eq(sock_net(sk), pp->net))  					goto next;  			break;  		next: -			spin_unlock(&head->lock); +			spin_unlock_bh(&head->lock); +			cond_resched();  		} while (--remaining > 0);  		/* Exhausted local port range during search? */  		ret = 1;  		if (remaining <= 0) -			goto fail; +			return ret;  		/* OK, here is the one we will use.  HEAD (the port  		 * hash table list entry) is non-NULL and we hold it's @@ -6254,7 +6255,7 @@ static long sctp_get_port_local(struct sock *sk, union sctp_addr *addr)  		 * port iterator, pp being NULL.  		 */  		head = &sctp_port_hashtable[sctp_phashfn(sock_net(sk), snum)]; -		spin_lock(&head->lock); +		spin_lock_bh(&head->lock);  		sctp_for_each_hentry(pp, &head->chain) {  			if ((pp->port == snum) && net_eq(pp->net, sock_net(sk)))  				goto pp_found; @@ -6338,10 +6339,7 @@ success:  	ret = 0;  fail_unlock: -	spin_unlock(&head->lock); - -fail: -	local_bh_enable(); +	spin_unlock_bh(&head->lock);  	return ret;  } @@ -7341,6 +7339,8 @@ static void sctp_sock_migrate(struct sock *oldsk, struct sock *newsk,  	sctp_bind_addr_dup(&newsp->ep->base.bind_addr,  				&oldsp->ep->base.bind_addr, GFP_KERNEL); +	sctp_auto_asconf_init(newsp); +  	/* Move any messages in the old socket's receive queue that are for the  	 * peeled off association to the new socket's receive queue.  	 */ | 
