sctp: Set source addresses on the association before adding transports
Recent commit 8da645e101
	sctp: Get rid of an extra routing lookup when adding a transport
introduced a regression in the connection setup.  The behavior was
different between IPv4 and IPv6.  IPv4 case ended up working because the
route lookup routing returned a NULL route, which triggered another
route lookup later in the output patch that succeeded.  In the IPv6 case,
a valid route was returned for first call, but we could not find a valid
source address at the time since the source addresses were not set on the
association yet.  Thus resulted in a hung connection.
The solution is to set the source addresses on the association prior to
adding peers.
Signed-off-by: Vlad Yasevich <vladislav.yasevich@hp.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
	
	
This commit is contained in:
		
					parent
					
						
							
								d792c1006f
							
						
					
				
			
			
				commit
				
					
						409b95aff3
					
				
			
		
					 4 changed files with 23 additions and 20 deletions
				
			
		|  | @ -1980,7 +1980,7 @@ void sctp_assoc_set_primary(struct sctp_association *, | ||||||
| void sctp_assoc_del_nonprimary_peers(struct sctp_association *, | void sctp_assoc_del_nonprimary_peers(struct sctp_association *, | ||||||
| 				    struct sctp_transport *); | 				    struct sctp_transport *); | ||||||
| int sctp_assoc_set_bind_addr_from_ep(struct sctp_association *, | int sctp_assoc_set_bind_addr_from_ep(struct sctp_association *, | ||||||
| 				     gfp_t); | 				     sctp_scope_t, gfp_t); | ||||||
| int sctp_assoc_set_bind_addr_from_cookie(struct sctp_association *, | int sctp_assoc_set_bind_addr_from_cookie(struct sctp_association *, | ||||||
| 					 struct sctp_cookie*, | 					 struct sctp_cookie*, | ||||||
| 					 gfp_t gfp); | 					 gfp_t gfp); | ||||||
|  |  | ||||||
|  | @ -1485,15 +1485,13 @@ void sctp_assoc_rwnd_decrease(struct sctp_association *asoc, unsigned len) | ||||||
|  * local endpoint and the remote peer. |  * local endpoint and the remote peer. | ||||||
|  */ |  */ | ||||||
| int sctp_assoc_set_bind_addr_from_ep(struct sctp_association *asoc, | int sctp_assoc_set_bind_addr_from_ep(struct sctp_association *asoc, | ||||||
| 				     gfp_t gfp) | 				     sctp_scope_t scope, gfp_t gfp) | ||||||
| { | { | ||||||
| 	sctp_scope_t scope; |  | ||||||
| 	int flags; | 	int flags; | ||||||
| 
 | 
 | ||||||
| 	/* Use scoping rules to determine the subset of addresses from
 | 	/* Use scoping rules to determine the subset of addresses from
 | ||||||
| 	 * the endpoint. | 	 * the endpoint. | ||||||
| 	 */ | 	 */ | ||||||
| 	scope = sctp_scope(&asoc->peer.active_path->ipaddr); |  | ||||||
| 	flags = (PF_INET6 == asoc->base.sk->sk_family) ? SCTP_ADDR6_ALLOWED : 0; | 	flags = (PF_INET6 == asoc->base.sk->sk_family) ? SCTP_ADDR6_ALLOWED : 0; | ||||||
| 	if (asoc->peer.ipv4_address) | 	if (asoc->peer.ipv4_address) | ||||||
| 		flags |= SCTP_ADDR4_PEERSUPP; | 		flags |= SCTP_ADDR4_PEERSUPP; | ||||||
|  |  | ||||||
|  | @ -384,6 +384,11 @@ sctp_disposition_t sctp_sf_do_5_1B_init(const struct sctp_endpoint *ep, | ||||||
| 	if (!new_asoc) | 	if (!new_asoc) | ||||||
| 		goto nomem; | 		goto nomem; | ||||||
| 
 | 
 | ||||||
|  | 	if (sctp_assoc_set_bind_addr_from_ep(new_asoc, | ||||||
|  | 					     sctp_scope(sctp_source(chunk)), | ||||||
|  | 					     GFP_ATOMIC) < 0) | ||||||
|  | 		goto nomem_init; | ||||||
|  | 
 | ||||||
| 	/* The call, sctp_process_init(), can fail on memory allocation.  */ | 	/* The call, sctp_process_init(), can fail on memory allocation.  */ | ||||||
| 	if (!sctp_process_init(new_asoc, chunk->chunk_hdr->type, | 	if (!sctp_process_init(new_asoc, chunk->chunk_hdr->type, | ||||||
| 			       sctp_source(chunk), | 			       sctp_source(chunk), | ||||||
|  | @ -401,9 +406,6 @@ sctp_disposition_t sctp_sf_do_5_1B_init(const struct sctp_endpoint *ep, | ||||||
| 		len = ntohs(err_chunk->chunk_hdr->length) - | 		len = ntohs(err_chunk->chunk_hdr->length) - | ||||||
| 			sizeof(sctp_chunkhdr_t); | 			sizeof(sctp_chunkhdr_t); | ||||||
| 
 | 
 | ||||||
| 	if (sctp_assoc_set_bind_addr_from_ep(new_asoc, GFP_ATOMIC) < 0) |  | ||||||
| 		goto nomem_init; |  | ||||||
| 
 |  | ||||||
| 	repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC, len); | 	repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC, len); | ||||||
| 	if (!repl) | 	if (!repl) | ||||||
| 		goto nomem_init; | 		goto nomem_init; | ||||||
|  | @ -1452,6 +1454,10 @@ static sctp_disposition_t sctp_sf_do_unexpected_init( | ||||||
| 	if (!new_asoc) | 	if (!new_asoc) | ||||||
| 		goto nomem; | 		goto nomem; | ||||||
| 
 | 
 | ||||||
|  | 	if (sctp_assoc_set_bind_addr_from_ep(new_asoc, | ||||||
|  | 				sctp_scope(sctp_source(chunk)), GFP_ATOMIC) < 0) | ||||||
|  | 		goto nomem; | ||||||
|  | 
 | ||||||
| 	/* In the outbound INIT ACK the endpoint MUST copy its current
 | 	/* In the outbound INIT ACK the endpoint MUST copy its current
 | ||||||
| 	 * Verification Tag and Peers Verification tag into a reserved | 	 * Verification Tag and Peers Verification tag into a reserved | ||||||
| 	 * place (local tie-tag and per tie-tag) within the state cookie. | 	 * place (local tie-tag and per tie-tag) within the state cookie. | ||||||
|  | @ -1488,9 +1494,6 @@ static sctp_disposition_t sctp_sf_do_unexpected_init( | ||||||
| 			sizeof(sctp_chunkhdr_t); | 			sizeof(sctp_chunkhdr_t); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (sctp_assoc_set_bind_addr_from_ep(new_asoc, GFP_ATOMIC) < 0) |  | ||||||
| 		goto nomem; |  | ||||||
| 
 |  | ||||||
| 	repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC, len); | 	repl = sctp_make_init_ack(new_asoc, chunk, GFP_ATOMIC, len); | ||||||
| 	if (!repl) | 	if (!repl) | ||||||
| 		goto nomem; | 		goto nomem; | ||||||
|  |  | ||||||
|  | @ -1080,6 +1080,13 @@ static int __sctp_connect(struct sock* sk, | ||||||
| 				err = -ENOMEM; | 				err = -ENOMEM; | ||||||
| 				goto out_free; | 				goto out_free; | ||||||
| 			} | 			} | ||||||
|  | 
 | ||||||
|  | 			err = sctp_assoc_set_bind_addr_from_ep(asoc, scope, | ||||||
|  | 							      GFP_KERNEL); | ||||||
|  | 			if (err < 0) { | ||||||
|  | 				goto out_free; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* Prime the peer's transport structures.  */ | 		/* Prime the peer's transport structures.  */ | ||||||
|  | @ -1095,11 +1102,6 @@ static int __sctp_connect(struct sock* sk, | ||||||
| 		walk_size += af->sockaddr_len; | 		walk_size += af->sockaddr_len; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	err = sctp_assoc_set_bind_addr_from_ep(asoc, GFP_KERNEL); |  | ||||||
| 	if (err < 0) { |  | ||||||
| 		goto out_free; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* In case the user of sctp_connectx() wants an association
 | 	/* In case the user of sctp_connectx() wants an association
 | ||||||
| 	 * id back, assign one now. | 	 * id back, assign one now. | ||||||
| 	 */ | 	 */ | ||||||
|  | @ -1689,6 +1691,11 @@ SCTP_STATIC int sctp_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 			goto out_unlock; | 			goto out_unlock; | ||||||
| 		} | 		} | ||||||
| 		asoc = new_asoc; | 		asoc = new_asoc; | ||||||
|  | 		err = sctp_assoc_set_bind_addr_from_ep(asoc, scope, GFP_KERNEL); | ||||||
|  | 		if (err < 0) { | ||||||
|  | 			err = -ENOMEM; | ||||||
|  | 			goto out_free; | ||||||
|  | 		} | ||||||
| 
 | 
 | ||||||
| 		/* If the SCTP_INIT ancillary data is specified, set all
 | 		/* If the SCTP_INIT ancillary data is specified, set all
 | ||||||
| 		 * the association init values accordingly. | 		 * the association init values accordingly. | ||||||
|  | @ -1718,11 +1725,6 @@ SCTP_STATIC int sctp_sendmsg(struct kiocb *iocb, struct sock *sk, | ||||||
| 			err = -ENOMEM; | 			err = -ENOMEM; | ||||||
| 			goto out_free; | 			goto out_free; | ||||||
| 		} | 		} | ||||||
| 		err = sctp_assoc_set_bind_addr_from_ep(asoc, GFP_KERNEL); |  | ||||||
| 		if (err < 0) { |  | ||||||
| 			err = -ENOMEM; |  | ||||||
| 			goto out_free; |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* ASSERT: we have a valid association at this point.  */ | 	/* ASSERT: we have a valid association at this point.  */ | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Vlad Yasevich
				Vlad Yasevich