diff options
Diffstat (limited to 'net/ipv6/exthdrs.c')
| -rw-r--r-- | net/ipv6/exthdrs.c | 34 |
1 files changed, 28 insertions, 6 deletions
diff --git a/net/ipv6/exthdrs.c b/net/ipv6/exthdrs.c index 8a659f92d17a..262f105d23b9 100644 --- a/net/ipv6/exthdrs.c +++ b/net/ipv6/exthdrs.c | |||
| @@ -312,6 +312,7 @@ static int ipv6_destopt_rcv(struct sk_buff *skb) | |||
| 312 | Routing header. | 312 | Routing header. |
| 313 | ********************************/ | 313 | ********************************/ |
| 314 | 314 | ||
| 315 | /* called with rcu_read_lock() */ | ||
| 315 | static int ipv6_rthdr_rcv(struct sk_buff *skb) | 316 | static int ipv6_rthdr_rcv(struct sk_buff *skb) |
| 316 | { | 317 | { |
| 317 | struct inet6_skb_parm *opt = IP6CB(skb); | 318 | struct inet6_skb_parm *opt = IP6CB(skb); |
| @@ -324,12 +325,9 @@ static int ipv6_rthdr_rcv(struct sk_buff *skb) | |||
| 324 | struct net *net = dev_net(skb->dev); | 325 | struct net *net = dev_net(skb->dev); |
| 325 | int accept_source_route = net->ipv6.devconf_all->accept_source_route; | 326 | int accept_source_route = net->ipv6.devconf_all->accept_source_route; |
| 326 | 327 | ||
| 327 | idev = in6_dev_get(skb->dev); | 328 | idev = __in6_dev_get(skb->dev); |
| 328 | if (idev) { | 329 | if (idev && accept_source_route > idev->cnf.accept_source_route) |
| 329 | if (accept_source_route > idev->cnf.accept_source_route) | 330 | accept_source_route = idev->cnf.accept_source_route; |
| 330 | accept_source_route = idev->cnf.accept_source_route; | ||
| 331 | in6_dev_put(idev); | ||
| 332 | } | ||
| 333 | 331 | ||
| 334 | if (!pskb_may_pull(skb, skb_transport_offset(skb) + 8) || | 332 | if (!pskb_may_pull(skb, skb_transport_offset(skb) + 8) || |
| 335 | !pskb_may_pull(skb, (skb_transport_offset(skb) + | 333 | !pskb_may_pull(skb, (skb_transport_offset(skb) + |
| @@ -874,3 +872,27 @@ struct ipv6_txoptions *ipv6_fixup_options(struct ipv6_txoptions *opt_space, | |||
| 874 | return opt; | 872 | return opt; |
| 875 | } | 873 | } |
| 876 | 874 | ||
| 875 | /** | ||
| 876 | * fl6_update_dst - update flowi destination address with info given | ||
| 877 | * by srcrt option, if any. | ||
| 878 | * | ||
| 879 | * @fl: flowi for which fl6_dst is to be updated | ||
| 880 | * @opt: struct ipv6_txoptions in which to look for srcrt opt | ||
| 881 | * @orig: copy of original fl6_dst address if modified | ||
| 882 | * | ||
| 883 | * Returns NULL if no txoptions or no srcrt, otherwise returns orig | ||
| 884 | * and initial value of fl->fl6_dst set in orig | ||
| 885 | */ | ||
| 886 | struct in6_addr *fl6_update_dst(struct flowi *fl, | ||
| 887 | const struct ipv6_txoptions *opt, | ||
| 888 | struct in6_addr *orig) | ||
| 889 | { | ||
| 890 | if (!opt || !opt->srcrt) | ||
| 891 | return NULL; | ||
| 892 | |||
| 893 | ipv6_addr_copy(orig, &fl->fl6_dst); | ||
| 894 | ipv6_addr_copy(&fl->fl6_dst, ((struct rt0_hdr *)opt->srcrt)->addr); | ||
| 895 | return orig; | ||
| 896 | } | ||
| 897 | |||
| 898 | EXPORT_SYMBOL_GPL(fl6_update_dst); | ||
