1
0
Fork 0

net: Use flowi4 and flowi6 in xfrm layer.

Signed-off-by: David S. Miller <davem@davemloft.net>
wifi-calibration
David S. Miller 2011-03-12 02:42:11 -05:00
parent 2032656e76
commit 7e1dc7b6f7
6 changed files with 89 additions and 76 deletions

View File

@ -1142,9 +1142,9 @@ xfrm_address_t *xfrm_flowi_daddr(const struct flowi *fl, unsigned short family)
{ {
switch (family){ switch (family){
case AF_INET: case AF_INET:
return (xfrm_address_t *)&fl->fl4_dst; return (xfrm_address_t *)&fl->u.ip4.daddr;
case AF_INET6: case AF_INET6:
return (xfrm_address_t *)&fl->fl6_dst; return (xfrm_address_t *)&fl->u.ip6.daddr;
} }
return NULL; return NULL;
} }
@ -1154,9 +1154,9 @@ xfrm_address_t *xfrm_flowi_saddr(const struct flowi *fl, unsigned short family)
{ {
switch (family){ switch (family){
case AF_INET: case AF_INET:
return (xfrm_address_t *)&fl->fl4_src; return (xfrm_address_t *)&fl->u.ip4.saddr;
case AF_INET6: case AF_INET6:
return (xfrm_address_t *)&fl->fl6_src; return (xfrm_address_t *)&fl->u.ip6.saddr;
} }
return NULL; return NULL;
} }
@ -1168,12 +1168,12 @@ void xfrm_flowi_addr_get(const struct flowi *fl,
{ {
switch(family) { switch(family) {
case AF_INET: case AF_INET:
memcpy(&saddr->a4, &fl->fl4_src, sizeof(saddr->a4)); memcpy(&saddr->a4, &fl->u.ip4.saddr, sizeof(saddr->a4));
memcpy(&daddr->a4, &fl->fl4_dst, sizeof(daddr->a4)); memcpy(&daddr->a4, &fl->u.ip4.daddr, sizeof(daddr->a4));
break; break;
case AF_INET6: case AF_INET6:
ipv6_addr_copy((struct in6_addr *)&saddr->a6, &fl->fl6_src); ipv6_addr_copy((struct in6_addr *)&saddr->a6, &fl->u.ip6.saddr);
ipv6_addr_copy((struct in6_addr *)&daddr->a6, &fl->fl6_dst); ipv6_addr_copy((struct in6_addr *)&daddr->a6, &fl->u.ip6.daddr);
break; break;
} }
} }
@ -1221,12 +1221,12 @@ xfrm_state_addr_flow_check(const struct xfrm_state *x, const struct flowi *fl,
switch (family) { switch (family) {
case AF_INET: case AF_INET:
return __xfrm4_state_addr_check(x, return __xfrm4_state_addr_check(x,
(const xfrm_address_t *)&fl->fl4_dst, (const xfrm_address_t *)&fl->u.ip4.daddr,
(const xfrm_address_t *)&fl->fl4_src); (const xfrm_address_t *)&fl->u.ip4.saddr);
case AF_INET6: case AF_INET6:
return __xfrm6_state_addr_check(x, return __xfrm6_state_addr_check(x,
(const xfrm_address_t *)&fl->fl6_dst, (const xfrm_address_t *)&fl->u.ip6.daddr,
(const xfrm_address_t *)&fl->fl6_src); (const xfrm_address_t *)&fl->u.ip6.saddr);
} }
return 0; return 0;
} }

View File

@ -56,7 +56,7 @@ static int xfrm4_get_saddr(struct net *net,
static int xfrm4_get_tos(const struct flowi *fl) static int xfrm4_get_tos(const struct flowi *fl)
{ {
return IPTOS_RT_MASK & fl->fl4_tos; /* Strip ECN bits */ return IPTOS_RT_MASK & fl->u.ip4.flowi4_tos; /* Strip ECN bits */
} }
static int xfrm4_init_path(struct xfrm_dst *path, struct dst_entry *dst, static int xfrm4_init_path(struct xfrm_dst *path, struct dst_entry *dst,
@ -69,13 +69,14 @@ static int xfrm4_fill_dst(struct xfrm_dst *xdst, struct net_device *dev,
const struct flowi *fl) const struct flowi *fl)
{ {
struct rtable *rt = (struct rtable *)xdst->route; struct rtable *rt = (struct rtable *)xdst->route;
const struct flowi4 *fl4 = &fl->u.ip4;
rt->rt_key_dst = fl->fl4_dst; rt->rt_key_dst = fl4->daddr;
rt->rt_key_src = fl->fl4_src; rt->rt_key_src = fl4->saddr;
rt->rt_tos = fl->fl4_tos; rt->rt_tos = fl4->flowi4_tos;
rt->rt_iif = fl->flowi_iif; rt->rt_iif = fl4->flowi4_iif;
rt->rt_oif = fl->flowi_oif; rt->rt_oif = fl4->flowi4_oif;
rt->rt_mark = fl->flowi_mark; rt->rt_mark = fl4->flowi4_mark;
xdst->u.dst.dev = dev; xdst->u.dst.dev = dev;
dev_hold(dev); dev_hold(dev);
@ -102,9 +103,10 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
{ {
struct iphdr *iph = ip_hdr(skb); struct iphdr *iph = ip_hdr(skb);
u8 *xprth = skb_network_header(skb) + iph->ihl * 4; u8 *xprth = skb_network_header(skb) + iph->ihl * 4;
struct flowi4 *fl4 = &fl->u.ip4;
memset(fl, 0, sizeof(struct flowi)); memset(fl4, 0, sizeof(struct flowi4));
fl->flowi_mark = skb->mark; fl4->flowi4_mark = skb->mark;
if (!(iph->frag_off & htons(IP_MF | IP_OFFSET))) { if (!(iph->frag_off & htons(IP_MF | IP_OFFSET))) {
switch (iph->protocol) { switch (iph->protocol) {
@ -117,8 +119,8 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
pskb_may_pull(skb, xprth + 4 - skb->data)) { pskb_may_pull(skb, xprth + 4 - skb->data)) {
__be16 *ports = (__be16 *)xprth; __be16 *ports = (__be16 *)xprth;
fl->fl4_sport = ports[!!reverse]; fl4->uli.ports.sport = ports[!!reverse];
fl->fl4_dport = ports[!reverse]; fl4->uli.ports.dport = ports[!reverse];
} }
break; break;
@ -126,8 +128,8 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
if (pskb_may_pull(skb, xprth + 2 - skb->data)) { if (pskb_may_pull(skb, xprth + 2 - skb->data)) {
u8 *icmp = xprth; u8 *icmp = xprth;
fl->fl4_icmp_type = icmp[0]; fl4->uli.icmpt.type = icmp[0];
fl->fl4_icmp_code = icmp[1]; fl4->uli.icmpt.code = icmp[1];
} }
break; break;
@ -135,7 +137,7 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
if (pskb_may_pull(skb, xprth + 4 - skb->data)) { if (pskb_may_pull(skb, xprth + 4 - skb->data)) {
__be32 *ehdr = (__be32 *)xprth; __be32 *ehdr = (__be32 *)xprth;
fl->fl4_ipsec_spi = ehdr[0]; fl4->uli.spi = ehdr[0];
} }
break; break;
@ -143,7 +145,7 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
if (pskb_may_pull(skb, xprth + 8 - skb->data)) { if (pskb_may_pull(skb, xprth + 8 - skb->data)) {
__be32 *ah_hdr = (__be32*)xprth; __be32 *ah_hdr = (__be32*)xprth;
fl->fl4_ipsec_spi = ah_hdr[1]; fl4->uli.spi = ah_hdr[1];
} }
break; break;
@ -151,7 +153,7 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
if (pskb_may_pull(skb, xprth + 4 - skb->data)) { if (pskb_may_pull(skb, xprth + 4 - skb->data)) {
__be16 *ipcomp_hdr = (__be16 *)xprth; __be16 *ipcomp_hdr = (__be16 *)xprth;
fl->fl4_ipsec_spi = htonl(ntohs(ipcomp_hdr[1])); fl4->uli.spi = htonl(ntohs(ipcomp_hdr[1]));
} }
break; break;
@ -163,20 +165,20 @@ _decode_session4(struct sk_buff *skb, struct flowi *fl, int reverse)
if (greflags[0] & GRE_KEY) { if (greflags[0] & GRE_KEY) {
if (greflags[0] & GRE_CSUM) if (greflags[0] & GRE_CSUM)
gre_hdr++; gre_hdr++;
fl->fl4_gre_key = gre_hdr[1]; fl4->uli.gre_key = gre_hdr[1];
} }
} }
break; break;
default: default:
fl->fl4_ipsec_spi = 0; fl4->uli.spi = 0;
break; break;
} }
} }
fl->flowi_proto = iph->protocol; fl4->flowi4_proto = iph->protocol;
fl->fl4_dst = reverse ? iph->saddr : iph->daddr; fl4->daddr = reverse ? iph->saddr : iph->daddr;
fl->fl4_src = reverse ? iph->daddr : iph->saddr; fl4->saddr = reverse ? iph->daddr : iph->saddr;
fl->fl4_tos = iph->tos; fl4->flowi4_tos = iph->tos;
} }
static inline int xfrm4_garbage_collect(struct dst_ops *ops) static inline int xfrm4_garbage_collect(struct dst_ops *ops)

View File

@ -23,17 +23,19 @@ static int xfrm4_init_flags(struct xfrm_state *x)
static void static void
__xfrm4_init_tempsel(struct xfrm_selector *sel, const struct flowi *fl) __xfrm4_init_tempsel(struct xfrm_selector *sel, const struct flowi *fl)
{ {
sel->daddr.a4 = fl->fl4_dst; const struct flowi4 *fl4 = &fl->u.ip4;
sel->saddr.a4 = fl->fl4_src;
sel->dport = xfrm_flowi_dport(fl, &fl->u.ip4.uli); sel->daddr.a4 = fl4->daddr;
sel->saddr.a4 = fl4->saddr;
sel->dport = xfrm_flowi_dport(fl, &fl4->uli);
sel->dport_mask = htons(0xffff); sel->dport_mask = htons(0xffff);
sel->sport = xfrm_flowi_sport(fl, &fl->u.ip4.uli); sel->sport = xfrm_flowi_sport(fl, &fl4->uli);
sel->sport_mask = htons(0xffff); sel->sport_mask = htons(0xffff);
sel->family = AF_INET; sel->family = AF_INET;
sel->prefixlen_d = 32; sel->prefixlen_d = 32;
sel->prefixlen_s = 32; sel->prefixlen_s = 32;
sel->proto = fl->flowi_proto; sel->proto = fl4->flowi4_proto;
sel->ifindex = fl->flowi_oif; sel->ifindex = fl4->flowi4_oif;
} }
static void static void

View File

@ -30,15 +30,17 @@ static struct dst_entry *xfrm6_dst_lookup(struct net *net, int tos,
const xfrm_address_t *saddr, const xfrm_address_t *saddr,
const xfrm_address_t *daddr) const xfrm_address_t *daddr)
{ {
struct flowi fl = {}; struct flowi6 fl6;
struct dst_entry *dst; struct dst_entry *dst;
int err; int err;
memcpy(&fl.fl6_dst, daddr, sizeof(fl.fl6_dst)); memset(&fl6, 0, sizeof(fl6));
memcpy(&fl6.daddr, daddr, sizeof(fl6.daddr));
if (saddr) if (saddr)
memcpy(&fl.fl6_src, saddr, sizeof(fl.fl6_src)); memcpy(&fl6.saddr, saddr, sizeof(fl6.saddr));
dst = ip6_route_output(net, NULL, &fl); dst = ip6_route_output(net, NULL,
flowi6_to_flowi(&fl6));
err = dst->error; err = dst->error;
if (dst->error) { if (dst->error) {
@ -120,6 +122,7 @@ static int xfrm6_fill_dst(struct xfrm_dst *xdst, struct net_device *dev,
static inline void static inline void
_decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse) _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse)
{ {
struct flowi6 *fl6 = &fl->u.ip6;
int onlyproto = 0; int onlyproto = 0;
u16 offset = skb_network_header_len(skb); u16 offset = skb_network_header_len(skb);
struct ipv6hdr *hdr = ipv6_hdr(skb); struct ipv6hdr *hdr = ipv6_hdr(skb);
@ -127,11 +130,11 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse)
const unsigned char *nh = skb_network_header(skb); const unsigned char *nh = skb_network_header(skb);
u8 nexthdr = nh[IP6CB(skb)->nhoff]; u8 nexthdr = nh[IP6CB(skb)->nhoff];
memset(fl, 0, sizeof(struct flowi)); memset(fl6, 0, sizeof(struct flowi6));
fl->flowi_mark = skb->mark; fl6->flowi6_mark = skb->mark;
ipv6_addr_copy(&fl->fl6_dst, reverse ? &hdr->saddr : &hdr->daddr); ipv6_addr_copy(&fl6->daddr, reverse ? &hdr->saddr : &hdr->daddr);
ipv6_addr_copy(&fl->fl6_src, reverse ? &hdr->daddr : &hdr->saddr); ipv6_addr_copy(&fl6->saddr, reverse ? &hdr->daddr : &hdr->saddr);
while (nh + offset + 1 < skb->data || while (nh + offset + 1 < skb->data ||
pskb_may_pull(skb, nh + offset + 1 - skb->data)) { pskb_may_pull(skb, nh + offset + 1 - skb->data)) {
@ -158,20 +161,20 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse)
pskb_may_pull(skb, nh + offset + 4 - skb->data))) { pskb_may_pull(skb, nh + offset + 4 - skb->data))) {
__be16 *ports = (__be16 *)exthdr; __be16 *ports = (__be16 *)exthdr;
fl->fl6_sport = ports[!!reverse]; fl6->uli.ports.sport = ports[!!reverse];
fl->fl6_dport = ports[!reverse]; fl6->uli.ports.dport = ports[!reverse];
} }
fl->flowi_proto = nexthdr; fl6->flowi6_proto = nexthdr;
return; return;
case IPPROTO_ICMPV6: case IPPROTO_ICMPV6:
if (!onlyproto && pskb_may_pull(skb, nh + offset + 2 - skb->data)) { if (!onlyproto && pskb_may_pull(skb, nh + offset + 2 - skb->data)) {
u8 *icmp = (u8 *)exthdr; u8 *icmp = (u8 *)exthdr;
fl->fl6_icmp_type = icmp[0]; fl6->uli.icmpt.type = icmp[0];
fl->fl6_icmp_code = icmp[1]; fl6->uli.icmpt.code = icmp[1];
} }
fl->flowi_proto = nexthdr; fl6->flowi6_proto = nexthdr;
return; return;
#if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE) #if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE)
@ -180,9 +183,9 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse)
struct ip6_mh *mh; struct ip6_mh *mh;
mh = (struct ip6_mh *)exthdr; mh = (struct ip6_mh *)exthdr;
fl->fl6_mh_type = mh->ip6mh_type; fl6->uli.mht.type = mh->ip6mh_type;
} }
fl->flowi_proto = nexthdr; fl6->flowi6_proto = nexthdr;
return; return;
#endif #endif
@ -191,8 +194,8 @@ _decode_session6(struct sk_buff *skb, struct flowi *fl, int reverse)
case IPPROTO_ESP: case IPPROTO_ESP:
case IPPROTO_COMP: case IPPROTO_COMP:
default: default:
fl->fl6_ipsec_spi = 0; fl6->uli.spi = 0;
fl->flowi_proto = nexthdr; fl6->flowi6_proto = nexthdr;
return; return;
} }
} }

View File

@ -22,19 +22,21 @@
static void static void
__xfrm6_init_tempsel(struct xfrm_selector *sel, const struct flowi *fl) __xfrm6_init_tempsel(struct xfrm_selector *sel, const struct flowi *fl)
{ {
const struct flowi6 *fl6 = &fl->u.ip6;
/* Initialize temporary selector matching only /* Initialize temporary selector matching only
* to current session. */ * to current session. */
ipv6_addr_copy((struct in6_addr *)&sel->daddr, &fl->fl6_dst); ipv6_addr_copy((struct in6_addr *)&sel->daddr, &fl6->daddr);
ipv6_addr_copy((struct in6_addr *)&sel->saddr, &fl->fl6_src); ipv6_addr_copy((struct in6_addr *)&sel->saddr, &fl6->saddr);
sel->dport = xfrm_flowi_dport(fl, &fl->u.ip6.uli); sel->dport = xfrm_flowi_dport(fl, &fl6->uli);
sel->dport_mask = htons(0xffff); sel->dport_mask = htons(0xffff);
sel->sport = xfrm_flowi_sport(fl, &fl->u.ip6.uli); sel->sport = xfrm_flowi_sport(fl, &fl6->uli);
sel->sport_mask = htons(0xffff); sel->sport_mask = htons(0xffff);
sel->family = AF_INET6; sel->family = AF_INET6;
sel->prefixlen_d = 128; sel->prefixlen_d = 128;
sel->prefixlen_s = 128; sel->prefixlen_s = 128;
sel->proto = fl->flowi_proto; sel->proto = fl6->flowi6_proto;
sel->ifindex = fl->flowi_oif; sel->ifindex = fl6->flowi6_oif;
} }
static void static void

View File

@ -59,23 +59,27 @@ static struct xfrm_policy *__xfrm_policy_unlink(struct xfrm_policy *pol,
static inline int static inline int
__xfrm4_selector_match(const struct xfrm_selector *sel, const struct flowi *fl) __xfrm4_selector_match(const struct xfrm_selector *sel, const struct flowi *fl)
{ {
return addr_match(&fl->fl4_dst, &sel->daddr, sel->prefixlen_d) && const struct flowi4 *fl4 = &fl->u.ip4;
addr_match(&fl->fl4_src, &sel->saddr, sel->prefixlen_s) &&
!((xfrm_flowi_dport(fl, &fl->u.ip4.uli) ^ sel->dport) & sel->dport_mask) && return addr_match(&fl4->daddr, &sel->daddr, sel->prefixlen_d) &&
!((xfrm_flowi_sport(fl, &fl->u.ip4.uli) ^ sel->sport) & sel->sport_mask) && addr_match(&fl4->saddr, &sel->saddr, sel->prefixlen_s) &&
(fl->flowi_proto == sel->proto || !sel->proto) && !((xfrm_flowi_dport(fl, &fl4->uli) ^ sel->dport) & sel->dport_mask) &&
(fl->flowi_oif == sel->ifindex || !sel->ifindex); !((xfrm_flowi_sport(fl, &fl4->uli) ^ sel->sport) & sel->sport_mask) &&
(fl4->flowi4_proto == sel->proto || !sel->proto) &&
(fl4->flowi4_oif == sel->ifindex || !sel->ifindex);
} }
static inline int static inline int
__xfrm6_selector_match(const struct xfrm_selector *sel, const struct flowi *fl) __xfrm6_selector_match(const struct xfrm_selector *sel, const struct flowi *fl)
{ {
return addr_match(&fl->fl6_dst, &sel->daddr, sel->prefixlen_d) && const struct flowi6 *fl6 = &fl->u.ip6;
addr_match(&fl->fl6_src, &sel->saddr, sel->prefixlen_s) &&
!((xfrm_flowi_dport(fl, &fl->u.ip6.uli) ^ sel->dport) & sel->dport_mask) && return addr_match(&fl6->daddr, &sel->daddr, sel->prefixlen_d) &&
!((xfrm_flowi_sport(fl, &fl->u.ip6.uli) ^ sel->sport) & sel->sport_mask) && addr_match(&fl6->saddr, &sel->saddr, sel->prefixlen_s) &&
(fl->flowi_proto == sel->proto || !sel->proto) && !((xfrm_flowi_dport(fl, &fl6->uli) ^ sel->dport) & sel->dport_mask) &&
(fl->flowi_oif == sel->ifindex || !sel->ifindex); !((xfrm_flowi_sport(fl, &fl6->uli) ^ sel->sport) & sel->sport_mask) &&
(fl6->flowi6_proto == sel->proto || !sel->proto) &&
(fl6->flowi6_oif == sel->ifindex || !sel->ifindex);
} }
int xfrm_selector_match(const struct xfrm_selector *sel, const struct flowi *fl, int xfrm_selector_match(const struct xfrm_selector *sel, const struct flowi *fl,