diff options
author | David S. Miller <davem@davemloft.net> | 2011-03-24 21:06:47 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2011-03-24 21:06:47 -0400 |
commit | 37e826c513883099c298317bad1b3b677b2905fb (patch) | |
tree | a032cfe69e0cde179d86eba157cf9dc85d67df86 /net/ipv4/fib_semantics.c | |
parent | 436c3b66ec9824a633724ae42de1c416af4f2063 (diff) |
ipv4: Fix nexthop caching wrt. scoping.
Move the scope value out of the fib alias entries and into fib_info,
so that we always use the correct scope when recomputing the nexthop
cached source address.
Reported-by: Julian Anastasov <ja@ssi.bg>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4/fib_semantics.c')
-rw-r--r-- | net/ipv4/fib_semantics.c | 15 |
1 files changed, 8 insertions, 7 deletions
diff --git a/net/ipv4/fib_semantics.c b/net/ipv4/fib_semantics.c index 2d4bebca671a..641a5a2a9f9c 100644 --- a/net/ipv4/fib_semantics.c +++ b/net/ipv4/fib_semantics.c | |||
@@ -222,7 +222,7 @@ static inline unsigned int fib_info_hashfn(const struct fib_info *fi) | |||
222 | unsigned int mask = (fib_info_hash_size - 1); | 222 | unsigned int mask = (fib_info_hash_size - 1); |
223 | unsigned int val = fi->fib_nhs; | 223 | unsigned int val = fi->fib_nhs; |
224 | 224 | ||
225 | val ^= fi->fib_protocol; | 225 | val ^= (fi->fib_protocol << 8) | fi->fib_scope; |
226 | val ^= (__force u32)fi->fib_prefsrc; | 226 | val ^= (__force u32)fi->fib_prefsrc; |
227 | val ^= fi->fib_priority; | 227 | val ^= fi->fib_priority; |
228 | for_nexthops(fi) { | 228 | for_nexthops(fi) { |
@@ -248,6 +248,7 @@ static struct fib_info *fib_find_info(const struct fib_info *nfi) | |||
248 | if (fi->fib_nhs != nfi->fib_nhs) | 248 | if (fi->fib_nhs != nfi->fib_nhs) |
249 | continue; | 249 | continue; |
250 | if (nfi->fib_protocol == fi->fib_protocol && | 250 | if (nfi->fib_protocol == fi->fib_protocol && |
251 | nfi->fib_scope == fi->fib_scope && | ||
251 | nfi->fib_prefsrc == fi->fib_prefsrc && | 252 | nfi->fib_prefsrc == fi->fib_prefsrc && |
252 | nfi->fib_priority == fi->fib_priority && | 253 | nfi->fib_priority == fi->fib_priority && |
253 | memcmp(nfi->fib_metrics, fi->fib_metrics, | 254 | memcmp(nfi->fib_metrics, fi->fib_metrics, |
@@ -328,7 +329,7 @@ void rtmsg_fib(int event, __be32 key, struct fib_alias *fa, | |||
328 | goto errout; | 329 | goto errout; |
329 | 330 | ||
330 | err = fib_dump_info(skb, info->pid, seq, event, tb_id, | 331 | err = fib_dump_info(skb, info->pid, seq, event, tb_id, |
331 | fa->fa_type, fa->fa_scope, key, dst_len, | 332 | fa->fa_type, key, dst_len, |
332 | fa->fa_tos, fa->fa_info, nlm_flags); | 333 | fa->fa_tos, fa->fa_info, nlm_flags); |
333 | if (err < 0) { | 334 | if (err < 0) { |
334 | /* -EMSGSIZE implies BUG in fib_nlmsg_size() */ | 335 | /* -EMSGSIZE implies BUG in fib_nlmsg_size() */ |
@@ -699,7 +700,7 @@ __be32 fib_info_update_nh_saddr(struct net *net, struct fib_nh *nh) | |||
699 | { | 700 | { |
700 | nh->nh_saddr = inet_select_addr(nh->nh_dev, | 701 | nh->nh_saddr = inet_select_addr(nh->nh_dev, |
701 | nh->nh_gw, | 702 | nh->nh_gw, |
702 | nh->nh_cfg_scope); | 703 | nh->nh_parent->fib_scope); |
703 | nh->nh_saddr_genid = atomic_read(&net->ipv4.dev_addr_genid); | 704 | nh->nh_saddr_genid = atomic_read(&net->ipv4.dev_addr_genid); |
704 | 705 | ||
705 | return nh->nh_saddr; | 706 | return nh->nh_saddr; |
@@ -763,6 +764,7 @@ struct fib_info *fib_create_info(struct fib_config *cfg) | |||
763 | 764 | ||
764 | fi->fib_net = hold_net(net); | 765 | fi->fib_net = hold_net(net); |
765 | fi->fib_protocol = cfg->fc_protocol; | 766 | fi->fib_protocol = cfg->fc_protocol; |
767 | fi->fib_scope = cfg->fc_scope; | ||
766 | fi->fib_flags = cfg->fc_flags; | 768 | fi->fib_flags = cfg->fc_flags; |
767 | fi->fib_priority = cfg->fc_priority; | 769 | fi->fib_priority = cfg->fc_priority; |
768 | fi->fib_prefsrc = cfg->fc_prefsrc; | 770 | fi->fib_prefsrc = cfg->fc_prefsrc; |
@@ -864,7 +866,6 @@ struct fib_info *fib_create_info(struct fib_config *cfg) | |||
864 | } | 866 | } |
865 | 867 | ||
866 | change_nexthops(fi) { | 868 | change_nexthops(fi) { |
867 | nexthop_nh->nh_cfg_scope = cfg->fc_scope; | ||
868 | fib_info_update_nh_saddr(net, nexthop_nh); | 869 | fib_info_update_nh_saddr(net, nexthop_nh); |
869 | } endfor_nexthops(fi) | 870 | } endfor_nexthops(fi) |
870 | 871 | ||
@@ -914,7 +915,7 @@ failure: | |||
914 | } | 915 | } |
915 | 916 | ||
916 | int fib_dump_info(struct sk_buff *skb, u32 pid, u32 seq, int event, | 917 | int fib_dump_info(struct sk_buff *skb, u32 pid, u32 seq, int event, |
917 | u32 tb_id, u8 type, u8 scope, __be32 dst, int dst_len, u8 tos, | 918 | u32 tb_id, u8 type, __be32 dst, int dst_len, u8 tos, |
918 | struct fib_info *fi, unsigned int flags) | 919 | struct fib_info *fi, unsigned int flags) |
919 | { | 920 | { |
920 | struct nlmsghdr *nlh; | 921 | struct nlmsghdr *nlh; |
@@ -936,7 +937,7 @@ int fib_dump_info(struct sk_buff *skb, u32 pid, u32 seq, int event, | |||
936 | NLA_PUT_U32(skb, RTA_TABLE, tb_id); | 937 | NLA_PUT_U32(skb, RTA_TABLE, tb_id); |
937 | rtm->rtm_type = type; | 938 | rtm->rtm_type = type; |
938 | rtm->rtm_flags = fi->fib_flags; | 939 | rtm->rtm_flags = fi->fib_flags; |
939 | rtm->rtm_scope = scope; | 940 | rtm->rtm_scope = fi->fib_scope; |
940 | rtm->rtm_protocol = fi->fib_protocol; | 941 | rtm->rtm_protocol = fi->fib_protocol; |
941 | 942 | ||
942 | if (rtm->rtm_dst_len) | 943 | if (rtm->rtm_dst_len) |
@@ -1092,7 +1093,7 @@ void fib_select_default(struct fib_result *res) | |||
1092 | list_for_each_entry_rcu(fa, fa_head, fa_list) { | 1093 | list_for_each_entry_rcu(fa, fa_head, fa_list) { |
1093 | struct fib_info *next_fi = fa->fa_info; | 1094 | struct fib_info *next_fi = fa->fa_info; |
1094 | 1095 | ||
1095 | if (fa->fa_scope != res->scope || | 1096 | if (next_fi->fib_scope != res->scope || |
1096 | fa->fa_type != RTN_UNICAST) | 1097 | fa->fa_type != RTN_UNICAST) |
1097 | continue; | 1098 | continue; |
1098 | 1099 | ||