mirror of
				git://git.openwrt.org/openwrt/openwrt.git
				synced 2025-10-25 19:14:26 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			97 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
			
		
		
	
	
			97 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Diff
		
	
	
	
	
	
| From e16e888b525503be05b3aea64190e8b3bdef44d0 Mon Sep 17 00:00:00 2001
 | |
| From: Markus Stenberg <markus.stenberg@iki.fi>
 | |
| Date: Tue, 5 May 2015 13:36:59 +0300
 | |
| Subject: [PATCH] ipv6: Fixed source specific default route handling.
 | |
| 
 | |
| If there are only IPv6 source specific default routes present, the
 | |
| host gets -ENETUNREACH on e.g. connect() because ip6_dst_lookup_tail
 | |
| calls ip6_route_output first, and given source address any, it fails,
 | |
| and ip6_route_get_saddr is never called.
 | |
| 
 | |
| The change is to use the ip6_route_get_saddr, even if the initial
 | |
| ip6_route_output fails, and then doing ip6_route_output _again_ after
 | |
| we have appropriate source address available.
 | |
| 
 | |
| Note that this is '99% fix' to the problem; a correct fix would be to
 | |
| do route lookups only within addrconf.c when picking a source address,
 | |
| and never call ip6_route_output before source address has been
 | |
| populated.
 | |
| 
 | |
| Signed-off-by: Markus Stenberg <markus.stenberg@iki.fi>
 | |
| Signed-off-by: David S. Miller <davem@davemloft.net>
 | |
| ---
 | |
|  net/ipv6/ip6_output.c | 39 +++++++++++++++++++++++++++++++--------
 | |
|  net/ipv6/route.c      |  5 +++--
 | |
|  2 files changed, 34 insertions(+), 10 deletions(-)
 | |
| 
 | |
| --- a/net/ipv6/ip6_output.c
 | |
| +++ b/net/ipv6/ip6_output.c
 | |
| @@ -903,21 +903,45 @@ static int ip6_dst_lookup_tail(struct so
 | |
|  #endif
 | |
|  	int err;
 | |
|  
 | |
| -	if (*dst == NULL)
 | |
| -		*dst = ip6_route_output(net, sk, fl6);
 | |
| -
 | |
| -	if ((err = (*dst)->error))
 | |
| -		goto out_err_release;
 | |
| +	/* The correct way to handle this would be to do
 | |
| +	 * ip6_route_get_saddr, and then ip6_route_output; however,
 | |
| +	 * the route-specific preferred source forces the
 | |
| +	 * ip6_route_output call _before_ ip6_route_get_saddr.
 | |
| +	 *
 | |
| +	 * In source specific routing (no src=any default route),
 | |
| +	 * ip6_route_output will fail given src=any saddr, though, so
 | |
| +	 * that's why we try it again later.
 | |
| +	 */
 | |
| +	if (ipv6_addr_any(&fl6->saddr) && (!*dst || !(*dst)->error)) {
 | |
| +		struct rt6_info *rt;
 | |
| +		bool had_dst = *dst != NULL;
 | |
|  
 | |
| -	if (ipv6_addr_any(&fl6->saddr)) {
 | |
| -		struct rt6_info *rt = (struct rt6_info *) *dst;
 | |
| +		if (!had_dst)
 | |
| +			*dst = ip6_route_output(net, sk, fl6);
 | |
| +		rt = (*dst)->error ? NULL : (struct rt6_info *)*dst;
 | |
|  		err = ip6_route_get_saddr(net, rt, &fl6->daddr,
 | |
|  					  sk ? inet6_sk(sk)->srcprefs : 0,
 | |
|  					  &fl6->saddr);
 | |
|  		if (err)
 | |
|  			goto out_err_release;
 | |
| +
 | |
| +		/* If we had an erroneous initial result, pretend it
 | |
| +		 * never existed and let the SA-enabled version take
 | |
| +		 * over.
 | |
| +		 */
 | |
| +		if (!had_dst && (*dst)->error) {
 | |
| +			dst_release(*dst);
 | |
| +			*dst = NULL;
 | |
| +		}
 | |
|  	}
 | |
|  
 | |
| +	if (!*dst)
 | |
| +		*dst = ip6_route_output(net, sk, fl6);
 | |
| +
 | |
| +	err = (*dst)->error;
 | |
| +	if (err)
 | |
| +		goto out_err_release;
 | |
| +
 | |
|  #ifdef CONFIG_IPV6_OPTIMISTIC_DAD
 | |
|  	/*
 | |
|  	 * Here if the dst entry we've looked up
 | |
| --- a/net/ipv6/route.c
 | |
| +++ b/net/ipv6/route.c
 | |
| @@ -2182,9 +2182,10 @@ int ip6_route_get_saddr(struct net *net,
 | |
|  			unsigned int prefs,
 | |
|  			struct in6_addr *saddr)
 | |
|  {
 | |
| -	struct inet6_dev *idev = ip6_dst_idev((struct dst_entry *)rt);
 | |
| +	struct inet6_dev *idev =
 | |
| +		rt ? ip6_dst_idev((struct dst_entry *)rt) : NULL;
 | |
|  	int err = 0;
 | |
| -	if (rt->rt6i_prefsrc.plen)
 | |
| +	if (rt && rt->rt6i_prefsrc.plen)
 | |
|  		*saddr = rt->rt6i_prefsrc.addr;
 | |
|  	else
 | |
|  		err = ipv6_dev_get_saddr(net, idev ? idev->dev : NULL,
 |