ipv4: Create and use route lookup helpers.

The idea here is this minimizes the number of places one has to edit
in order to make changes to how flows are defined and used.

Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
David S. Miller 2011-03-12 00:00:52 -05:00
parent 1561747ddf
commit 78fbfd8a65
23 changed files with 204 additions and 324 deletions

View File

@ -183,17 +183,11 @@ static int addr4_resolve(struct sockaddr_in *src_in,
{ {
__be32 src_ip = src_in->sin_addr.s_addr; __be32 src_ip = src_in->sin_addr.s_addr;
__be32 dst_ip = dst_in->sin_addr.s_addr; __be32 dst_ip = dst_in->sin_addr.s_addr;
struct flowi fl;
struct rtable *rt; struct rtable *rt;
struct neighbour *neigh; struct neighbour *neigh;
int ret; int ret;
memset(&fl, 0, sizeof fl); rt = ip_route_output(&init_net, dst_ip, src_ip, 0, addr->bound_dev_if);
fl.nl_u.ip4_u.daddr = dst_ip;
fl.nl_u.ip4_u.saddr = src_ip;
fl.oif = addr->bound_dev_if;
rt = ip_route_output_key(&init_net, &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
ret = PTR_ERR(rt); ret = PTR_ERR(rt);
goto out; goto out;

View File

@ -338,23 +338,10 @@ static struct rtable *find_route(struct t3cdev *dev, __be32 local_ip,
__be16 peer_port, u8 tos) __be16 peer_port, u8 tos)
{ {
struct rtable *rt; struct rtable *rt;
struct flowi fl = {
.oif = 0,
.nl_u = {
.ip4_u = {
.daddr = peer_ip,
.saddr = local_ip,
.tos = tos}
},
.proto = IPPROTO_TCP,
.uli_u = {
.ports = {
.sport = local_port,
.dport = peer_port}
}
};
rt = ip_route_output_flow(&init_net, &fl, NULL); rt = ip_route_output_ports(&init_net, NULL, peer_ip, local_ip,
peer_port, local_port, IPPROTO_TCP,
tos, 0);
if (IS_ERR(rt)) if (IS_ERR(rt))
return NULL; return NULL;
return rt; return rt;

View File

@ -315,23 +315,10 @@ static struct rtable *find_route(struct c4iw_dev *dev, __be32 local_ip,
__be16 peer_port, u8 tos) __be16 peer_port, u8 tos)
{ {
struct rtable *rt; struct rtable *rt;
struct flowi fl = {
.oif = 0,
.nl_u = {
.ip4_u = {
.daddr = peer_ip,
.saddr = local_ip,
.tos = tos}
},
.proto = IPPROTO_TCP,
.uli_u = {
.ports = {
.sport = local_port,
.dport = peer_port}
}
};
rt = ip_route_output_flow(&init_net, &fl, NULL); rt = ip_route_output_ports(&init_net, NULL, peer_ip, local_ip,
peer_port, local_port, IPPROTO_TCP,
tos, 0);
if (IS_ERR(rt)) if (IS_ERR(rt))
return NULL; return NULL;
return rt; return rt;

View File

@ -1104,15 +1104,12 @@ static inline int mini_cm_accelerated(struct nes_cm_core *cm_core,
static int nes_addr_resolve_neigh(struct nes_vnic *nesvnic, u32 dst_ip, int arpindex) static int nes_addr_resolve_neigh(struct nes_vnic *nesvnic, u32 dst_ip, int arpindex)
{ {
struct rtable *rt; struct rtable *rt;
struct flowi fl;
struct neighbour *neigh; struct neighbour *neigh;
int rc = arpindex; int rc = arpindex;
struct net_device *netdev; struct net_device *netdev;
struct nes_adapter *nesadapter = nesvnic->nesdev->nesadapter; struct nes_adapter *nesadapter = nesvnic->nesdev->nesadapter;
memset(&fl, 0, sizeof fl); rt = ip_route_output(&init_net, htonl(dst_ip), 0, 0, 0);
fl.nl_u.ip4_u.daddr = htonl(dst_ip);
rt = ip_route_output_key(&init_net, &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
printk(KERN_ERR "%s: ip_route_output_key failed for 0x%08X\n", printk(KERN_ERR "%s: ip_route_output_key failed for 0x%08X\n",
__func__, dst_ip); __func__, dst_ip);

View File

@ -2676,7 +2676,6 @@ static void bond_arp_send_all(struct bonding *bond, struct slave *slave)
__be32 *targets = bond->params.arp_targets; __be32 *targets = bond->params.arp_targets;
struct vlan_entry *vlan; struct vlan_entry *vlan;
struct net_device *vlan_dev; struct net_device *vlan_dev;
struct flowi fl;
struct rtable *rt; struct rtable *rt;
for (i = 0; (i < BOND_MAX_ARP_TARGETS); i++) { for (i = 0; (i < BOND_MAX_ARP_TARGETS); i++) {
@ -2695,15 +2694,12 @@ static void bond_arp_send_all(struct bonding *bond, struct slave *slave)
* determine which VLAN interface would be used, so we * determine which VLAN interface would be used, so we
* can tag the ARP with the proper VLAN tag. * can tag the ARP with the proper VLAN tag.
*/ */
memset(&fl, 0, sizeof(fl)); rt = ip_route_output(dev_net(bond->dev), targets[i], 0,
fl.fl4_dst = targets[i]; RTO_ONLINK, 0);
fl.fl4_tos = RTO_ONLINK;
rt = ip_route_output_key(dev_net(bond->dev), &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
if (net_ratelimit()) { if (net_ratelimit()) {
pr_warning("%s: no route to arp_ip_target %pI4\n", pr_warning("%s: no route to arp_ip_target %pI4\n",
bond->dev->name, &fl.fl4_dst); bond->dev->name, &targets[i]);
} }
continue; continue;
} }
@ -2739,7 +2735,7 @@ static void bond_arp_send_all(struct bonding *bond, struct slave *slave)
if (net_ratelimit()) { if (net_ratelimit()) {
pr_warning("%s: no path to arp_ip_target %pI4 via rt.dev %s\n", pr_warning("%s: no path to arp_ip_target %pI4 via rt.dev %s\n",
bond->dev->name, &fl.fl4_dst, bond->dev->name, &targets[i],
rt->dst.dev ? rt->dst.dev->name : "NULL"); rt->dst.dev ? rt->dst.dev->name : "NULL");
} }
ip_rt_put(rt); ip_rt_put(rt);

View File

@ -3407,20 +3407,14 @@ static int cnic_get_v4_route(struct sockaddr_in *dst_addr,
struct dst_entry **dst) struct dst_entry **dst)
{ {
#if defined(CONFIG_INET) #if defined(CONFIG_INET)
struct flowi fl;
int err;
struct rtable *rt; struct rtable *rt;
memset(&fl, 0, sizeof(fl)); rt = ip_route_output(&init_net, dst_addr->sin_addr.s_addr, 0, 0, 0);
fl.nl_u.ip4_u.daddr = dst_addr->sin_addr.s_addr; if (!IS_ERR(rt)) {
rt = ip_route_output_key(&init_net, &fl);
err = 0;
if (!IS_ERR(rt))
*dst = &rt->dst; *dst = &rt->dst;
else return 0;
err = PTR_ERR(rt); }
return err; return PTR_ERR(rt);
#else #else
return -ENETUNREACH; return -ENETUNREACH;
#endif #endif

View File

@ -189,18 +189,14 @@ static int pptp_xmit(struct ppp_channel *chan, struct sk_buff *skb)
if (sk_pppox(po)->sk_state & PPPOX_DEAD) if (sk_pppox(po)->sk_state & PPPOX_DEAD)
goto tx_error; goto tx_error;
{ rt = ip_route_output_ports(&init_net, NULL,
struct flowi fl = { .oif = 0, opt->dst_addr.sin_addr.s_addr,
.nl_u = { opt->src_addr.sin_addr.s_addr,
.ip4_u = { 0, 0, IPPROTO_GRE,
.daddr = opt->dst_addr.sin_addr.s_addr, RT_TOS(0), 0);
.saddr = opt->src_addr.sin_addr.s_addr, if (IS_ERR(rt))
.tos = RT_TOS(0) } }, goto tx_error;
.proto = IPPROTO_GRE };
rt = ip_route_output_key(&init_net, &fl);
if (IS_ERR(rt))
goto tx_error;
}
tdev = rt->dst.dev; tdev = rt->dst.dev;
max_headroom = LL_RESERVED_SPACE(tdev) + sizeof(*iph) + sizeof(*hdr) + 2; max_headroom = LL_RESERVED_SPACE(tdev) + sizeof(*iph) + sizeof(*hdr) + 2;
@ -467,22 +463,17 @@ static int pptp_connect(struct socket *sock, struct sockaddr *uservaddr,
po->chan.private = sk; po->chan.private = sk;
po->chan.ops = &pptp_chan_ops; po->chan.ops = &pptp_chan_ops;
{ rt = ip_route_output_ports(&init_net, sk,
struct flowi fl = { opt->dst_addr.sin_addr.s_addr,
.nl_u = { opt->src_addr.sin_addr.s_addr,
.ip4_u = { 0, 0,
.daddr = opt->dst_addr.sin_addr.s_addr, IPPROTO_GRE, RT_CONN_FLAGS(sk), 0);
.saddr = opt->src_addr.sin_addr.s_addr, if (IS_ERR(rt)) {
.tos = RT_CONN_FLAGS(sk) } }, error = -EHOSTUNREACH;
.proto = IPPROTO_GRE }; goto end;
security_sk_classify_flow(sk, &fl);
rt = ip_route_output_key(&init_net, &fl);
if (IS_ERR(rt)) {
error = -EHOSTUNREACH;
goto end;
}
sk_setup_caps(sk, &rt->dst);
} }
sk_setup_caps(sk, &rt->dst);
po->chan.mtu = dst_mtu(&rt->dst); po->chan.mtu = dst_mtu(&rt->dst);
if (!po->chan.mtu) if (!po->chan.mtu)
po->chan.mtu = PPP_MTU; po->chan.mtu = PPP_MTU;

View File

@ -451,26 +451,12 @@ static struct cxgbi_sock *cxgbi_sock_create(struct cxgbi_device *cdev)
} }
static struct rtable *find_route_ipv4(__be32 saddr, __be32 daddr, static struct rtable *find_route_ipv4(__be32 saddr, __be32 daddr,
__be16 sport, __be16 dport, u8 tos) __be16 sport, __be16 dport, u8 tos)
{ {
struct rtable *rt; struct rtable *rt;
struct flowi fl = {
.oif = 0,
.nl_u = {
.ip4_u = {
.daddr = daddr,
.saddr = saddr,
.tos = tos }
},
.proto = IPPROTO_TCP,
.uli_u = {
.ports = {
.sport = sport,
.dport = dport }
}
};
rt = ip_route_output_flow(&init_net, &fl, NULL); rt = ip_route_output_ports(&init_net, NULL, daddr, saddr,
dport, sport, IPPROTO_TCP, tos, 0);
if (IS_ERR(rt)) if (IS_ERR(rt))
return NULL; return NULL;

View File

@ -132,6 +132,54 @@ static inline struct rtable *ip_route_output_key(struct net *net, struct flowi *
return ip_route_output_flow(net, flp, NULL); return ip_route_output_flow(net, flp, NULL);
} }
static inline struct rtable *ip_route_output(struct net *net, __be32 daddr,
__be32 saddr, u8 tos, int oif)
{
struct flowi fl = {
.oif = oif,
.fl4_dst = daddr,
.fl4_src = saddr,
.fl4_tos = tos,
};
return ip_route_output_key(net, &fl);
}
static inline struct rtable *ip_route_output_ports(struct net *net, struct sock *sk,
__be32 daddr, __be32 saddr,
__be16 dport, __be16 sport,
__u8 proto, __u8 tos, int oif)
{
struct flowi fl = {
.oif = oif,
.flags = sk ? inet_sk_flowi_flags(sk) : 0,
.mark = sk ? sk->sk_mark : 0,
.fl4_dst = daddr,
.fl4_src = saddr,
.fl4_tos = tos,
.proto = proto,
.fl_ip_dport = dport,
.fl_ip_sport = sport,
};
if (sk)
security_sk_classify_flow(sk, &fl);
return ip_route_output_flow(net, &fl, sk);
}
static inline struct rtable *ip_route_output_gre(struct net *net,
__be32 daddr, __be32 saddr,
__be32 gre_key, __u8 tos, int oif)
{
struct flowi fl = {
.oif = oif,
.fl4_dst = daddr,
.fl4_src = saddr,
.fl4_tos = tos,
.proto = IPPROTO_GRE,
.fl_gre_key = gre_key,
};
return ip_route_output_key(net, &fl);
}
extern int ip_route_input_common(struct sk_buff *skb, __be32 dst, __be32 src, extern int ip_route_input_common(struct sk_buff *skb, __be32 dst, __be32 src,
u8 tos, struct net_device *devin, bool noref); u8 tos, struct net_device *devin, bool noref);

View File

@ -502,8 +502,6 @@ static int clip_setentry(struct atm_vcc *vcc, __be32 ip)
struct atmarp_entry *entry; struct atmarp_entry *entry;
int error; int error;
struct clip_vcc *clip_vcc; struct clip_vcc *clip_vcc;
struct flowi fl = { .fl4_dst = ip,
.fl4_tos = 1 };
struct rtable *rt; struct rtable *rt;
if (vcc->push != clip_push) { if (vcc->push != clip_push) {
@ -520,7 +518,7 @@ static int clip_setentry(struct atm_vcc *vcc, __be32 ip)
unlink_clip_vcc(clip_vcc); unlink_clip_vcc(clip_vcc);
return 0; return 0;
} }
rt = ip_route_output_key(&init_net, &fl); rt = ip_route_output(&init_net, ip, 0, 1, 0);
if (IS_ERR(rt)) if (IS_ERR(rt))
return PTR_ERR(rt); return PTR_ERR(rt);
neigh = __neigh_lookup(&clip_tbl, &ip, rt->dst.dev, 1); neigh = __neigh_lookup(&clip_tbl, &ip, rt->dst.dev, 1);

View File

@ -412,10 +412,6 @@ static int br_nf_pre_routing_finish(struct sk_buff *skb)
nf_bridge->mask ^= BRNF_NF_BRIDGE_PREROUTING; nf_bridge->mask ^= BRNF_NF_BRIDGE_PREROUTING;
if (dnat_took_place(skb)) { if (dnat_took_place(skb)) {
if ((err = ip_route_input(skb, iph->daddr, iph->saddr, iph->tos, dev))) { if ((err = ip_route_input(skb, iph->daddr, iph->saddr, iph->tos, dev))) {
struct flowi fl = {
.fl4_dst = iph->daddr,
.fl4_tos = RT_TOS(iph->tos),
};
struct in_device *in_dev = __in_dev_get_rcu(dev); struct in_device *in_dev = __in_dev_get_rcu(dev);
/* If err equals -EHOSTUNREACH the error is due to a /* If err equals -EHOSTUNREACH the error is due to a
@ -428,7 +424,8 @@ static int br_nf_pre_routing_finish(struct sk_buff *skb)
if (err != -EHOSTUNREACH || !in_dev || IN_DEV_FORWARD(in_dev)) if (err != -EHOSTUNREACH || !in_dev || IN_DEV_FORWARD(in_dev))
goto free_skb; goto free_skb;
rt = ip_route_output_key(dev_net(dev), &fl); rt = ip_route_output(dev_net(dev), iph->daddr, 0,
RT_TOS(iph->tos), 0);
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
/* - Bridged-and-DNAT'ed traffic doesn't /* - Bridged-and-DNAT'ed traffic doesn't
* require ip_forwarding. */ * require ip_forwarding. */

View File

@ -1157,22 +1157,10 @@ int inet_sk_rebuild_header(struct sock *sk)
daddr = inet->inet_daddr; daddr = inet->inet_daddr;
if (inet->opt && inet->opt->srr) if (inet->opt && inet->opt->srr)
daddr = inet->opt->faddr; daddr = inet->opt->faddr;
{ rt = ip_route_output_ports(sock_net(sk), sk, daddr, inet->inet_saddr,
struct flowi fl = { inet->inet_dport, inet->inet_sport,
.oif = sk->sk_bound_dev_if, sk->sk_protocol, RT_CONN_FLAGS(sk),
.mark = sk->sk_mark, sk->sk_bound_dev_if);
.fl4_dst = daddr,
.fl4_src = inet->inet_saddr,
.fl4_tos = RT_CONN_FLAGS(sk),
.proto = sk->sk_protocol,
.flags = inet_sk_flowi_flags(sk),
.fl_ip_sport = inet->inet_sport,
.fl_ip_dport = inet->inet_dport,
};
security_sk_classify_flow(sk, &fl);
rt = ip_route_output_flow(sock_net(sk), &fl, sk);
}
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
err = 0; err = 0;
sk_setup_caps(sk, &rt->dst); sk_setup_caps(sk, &rt->dst);

View File

@ -433,14 +433,12 @@ static int arp_ignore(struct in_device *in_dev, __be32 sip, __be32 tip)
static int arp_filter(__be32 sip, __be32 tip, struct net_device *dev) static int arp_filter(__be32 sip, __be32 tip, struct net_device *dev)
{ {
struct flowi fl = { .fl4_dst = sip,
.fl4_src = tip };
struct rtable *rt; struct rtable *rt;
int flag = 0; int flag = 0;
/*unsigned long now; */ /*unsigned long now; */
struct net *net = dev_net(dev); struct net *net = dev_net(dev);
rt = ip_route_output_key(net, &fl); rt = ip_route_output(net, sip, tip, 0, 0);
if (IS_ERR(rt)) if (IS_ERR(rt))
return 1; return 1;
if (rt->dst.dev != dev) { if (rt->dst.dev != dev) {
@ -1062,9 +1060,7 @@ static int arp_req_set(struct net *net, struct arpreq *r,
if (r->arp_flags & ATF_PERM) if (r->arp_flags & ATF_PERM)
r->arp_flags |= ATF_COM; r->arp_flags |= ATF_COM;
if (dev == NULL) { if (dev == NULL) {
struct flowi fl = { .fl4_dst = ip, struct rtable *rt = ip_route_output(net, ip, 0, RTO_ONLINK, 0);
.fl4_tos = RTO_ONLINK };
struct rtable *rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
return PTR_ERR(rt); return PTR_ERR(rt);
@ -1185,9 +1181,7 @@ static int arp_req_delete(struct net *net, struct arpreq *r,
ip = ((struct sockaddr_in *)&r->arp_pa)->sin_addr.s_addr; ip = ((struct sockaddr_in *)&r->arp_pa)->sin_addr.s_addr;
if (dev == NULL) { if (dev == NULL) {
struct flowi fl = { .fl4_dst = ip, struct rtable *rt = ip_route_output(net, ip, 0, RTO_ONLINK, 0);
.fl4_tos = RTO_ONLINK };
struct rtable *rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
return PTR_ERR(rt); return PTR_ERR(rt);
dev = rt->dst.dev; dev = rt->dst.dev;

View File

@ -321,15 +321,12 @@ static struct sk_buff *igmpv3_newpack(struct net_device *dev, int size)
} }
igmp_skb_size(skb) = size; igmp_skb_size(skb) = size;
{ rt = ip_route_output_ports(net, NULL, IGMPV3_ALL_MCR, 0,
struct flowi fl = { .oif = dev->ifindex, 0, 0,
.fl4_dst = IGMPV3_ALL_MCR, IPPROTO_IGMP, 0, dev->ifindex);
.proto = IPPROTO_IGMP }; if (IS_ERR(rt)) {
rt = ip_route_output_key(net, &fl); kfree_skb(skb);
if (IS_ERR(rt)) { return NULL;
kfree_skb(skb);
return NULL;
}
} }
if (rt->rt_src == 0) { if (rt->rt_src == 0) {
kfree_skb(skb); kfree_skb(skb);
@ -667,14 +664,12 @@ static int igmp_send_report(struct in_device *in_dev, struct ip_mc_list *pmc,
else else
dst = group; dst = group;
{ rt = ip_route_output_ports(net, NULL, dst, 0,
struct flowi fl = { .oif = dev->ifindex, 0, 0,
.fl4_dst = dst, IPPROTO_IGMP, 0, dev->ifindex);
.proto = IPPROTO_IGMP }; if (IS_ERR(rt))
rt = ip_route_output_key(net, &fl); return -1;
if (IS_ERR(rt))
return -1;
}
if (rt->rt_src == 0) { if (rt->rt_src == 0) {
ip_rt_put(rt); ip_rt_put(rt);
return -1; return -1;
@ -1441,7 +1436,6 @@ void ip_mc_destroy_dev(struct in_device *in_dev)
/* RTNL is locked */ /* RTNL is locked */
static struct in_device *ip_mc_find_dev(struct net *net, struct ip_mreqn *imr) static struct in_device *ip_mc_find_dev(struct net *net, struct ip_mreqn *imr)
{ {
struct flowi fl = { .fl4_dst = imr->imr_multiaddr.s_addr };
struct net_device *dev = NULL; struct net_device *dev = NULL;
struct in_device *idev = NULL; struct in_device *idev = NULL;
@ -1456,7 +1450,9 @@ static struct in_device *ip_mc_find_dev(struct net *net, struct ip_mreqn *imr)
} }
if (!dev) { if (!dev) {
struct rtable *rt = ip_route_output_key(net, &fl); struct rtable *rt = ip_route_output(net,
imr->imr_multiaddr.s_addr,
0, 0, 0);
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
dev = rt->dst.dev; dev = rt->dst.dev;
ip_rt_put(rt); ip_rt_put(rt);

View File

@ -769,20 +769,12 @@ static netdev_tx_t ipgre_tunnel_xmit(struct sk_buff *skb, struct net_device *dev
tos = ipv6_get_dsfield((struct ipv6hdr *)old_iph); tos = ipv6_get_dsfield((struct ipv6hdr *)old_iph);
} }
{ rt = ip_route_output_gre(dev_net(dev), dst, tiph->saddr,
struct flowi fl = { tunnel->parms.o_key, RT_TOS(tos),
.oif = tunnel->parms.link, tunnel->parms.link);
.fl4_dst = dst, if (IS_ERR(rt)) {
.fl4_src = tiph->saddr, dev->stats.tx_carrier_errors++;
.fl4_tos = RT_TOS(tos), goto tx_error;
.proto = IPPROTO_GRE,
.fl_gre_key = tunnel->parms.o_key
};
rt = ip_route_output_key(dev_net(dev), &fl);
if (IS_ERR(rt)) {
dev->stats.tx_carrier_errors++;
goto tx_error;
}
} }
tdev = rt->dst.dev; tdev = rt->dst.dev;
@ -946,15 +938,11 @@ static int ipgre_tunnel_bind_dev(struct net_device *dev)
/* Guess output device to choose reasonable mtu and needed_headroom */ /* Guess output device to choose reasonable mtu and needed_headroom */
if (iph->daddr) { if (iph->daddr) {
struct flowi fl = { struct rtable *rt = ip_route_output_gre(dev_net(dev),
.oif = tunnel->parms.link, iph->daddr, iph->saddr,
.fl4_dst = iph->daddr, tunnel->parms.o_key,
.fl4_src = iph->saddr, RT_TOS(iph->tos),
.fl4_tos = RT_TOS(iph->tos), tunnel->parms.link);
.proto = IPPROTO_GRE,
.fl_gre_key = tunnel->parms.o_key
};
struct rtable *rt = ip_route_output_key(dev_net(dev), &fl);
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
tdev = rt->dst.dev; tdev = rt->dst.dev;
@ -1208,15 +1196,12 @@ static int ipgre_open(struct net_device *dev)
struct ip_tunnel *t = netdev_priv(dev); struct ip_tunnel *t = netdev_priv(dev);
if (ipv4_is_multicast(t->parms.iph.daddr)) { if (ipv4_is_multicast(t->parms.iph.daddr)) {
struct flowi fl = { struct rtable *rt = ip_route_output_gre(dev_net(dev),
.oif = t->parms.link, t->parms.iph.daddr,
.fl4_dst = t->parms.iph.daddr, t->parms.iph.saddr,
.fl4_src = t->parms.iph.saddr, t->parms.o_key,
.fl4_tos = RT_TOS(t->parms.iph.tos), RT_TOS(t->parms.iph.tos),
.proto = IPPROTO_GRE, t->parms.link);
.fl_gre_key = t->parms.o_key
};
struct rtable *rt = ip_route_output_key(dev_net(dev), &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
return -EADDRNOTAVAIL; return -EADDRNOTAVAIL;

View File

@ -339,26 +339,19 @@ int ip_queue_xmit(struct sk_buff *skb)
if(opt && opt->srr) if(opt && opt->srr)
daddr = opt->faddr; daddr = opt->faddr;
{ /* If this fails, retransmit mechanism of transport layer will
struct flowi fl = { .oif = sk->sk_bound_dev_if, * keep trying until route appears or the connection times
.mark = sk->sk_mark, * itself out.
.fl4_dst = daddr, */
.fl4_src = inet->inet_saddr, rt = ip_route_output_ports(sock_net(sk), sk,
.fl4_tos = RT_CONN_FLAGS(sk), daddr, inet->inet_saddr,
.proto = sk->sk_protocol, inet->inet_dport,
.flags = inet_sk_flowi_flags(sk), inet->inet_sport,
.fl_ip_sport = inet->inet_sport, sk->sk_protocol,
.fl_ip_dport = inet->inet_dport }; RT_CONN_FLAGS(sk),
sk->sk_bound_dev_if);
/* If this fails, retransmit mechanism of transport layer will if (IS_ERR(rt))
* keep trying until route appears or the connection times goto no_route;
* itself out.
*/
security_sk_classify_flow(sk, &fl);
rt = ip_route_output_flow(sock_net(sk), &fl, sk);
if (IS_ERR(rt))
goto no_route;
}
sk_setup_caps(sk, &rt->dst); sk_setup_caps(sk, &rt->dst);
} }
skb_dst_set_noref(skb, &rt->dst); skb_dst_set_noref(skb, &rt->dst);

View File

@ -460,20 +460,14 @@ static netdev_tx_t ipip_tunnel_xmit(struct sk_buff *skb, struct net_device *dev)
goto tx_error_icmp; goto tx_error_icmp;
} }
{ rt = ip_route_output_ports(dev_net(dev), NULL,
struct flowi fl = { dst, tiph->saddr,
.oif = tunnel->parms.link, 0, 0,
.fl4_dst = dst, IPPROTO_IPIP, RT_TOS(tos),
.fl4_src= tiph->saddr, tunnel->parms.link);
.fl4_tos = RT_TOS(tos), if (IS_ERR(rt)) {
.proto = IPPROTO_IPIP dev->stats.tx_carrier_errors++;
}; goto tx_error_icmp;
rt = ip_route_output_key(dev_net(dev), &fl);
if (IS_ERR(rt)) {
dev->stats.tx_carrier_errors++;
goto tx_error_icmp;
}
} }
tdev = rt->dst.dev; tdev = rt->dst.dev;
@ -584,14 +578,12 @@ static void ipip_tunnel_bind_dev(struct net_device *dev)
iph = &tunnel->parms.iph; iph = &tunnel->parms.iph;
if (iph->daddr) { if (iph->daddr) {
struct flowi fl = { struct rtable *rt = ip_route_output_ports(dev_net(dev), NULL,
.oif = tunnel->parms.link, iph->daddr, iph->saddr,
.fl4_dst = iph->daddr, 0, 0,
.fl4_src = iph->saddr, IPPROTO_IPIP,
.fl4_tos = RT_TOS(iph->tos), RT_TOS(iph->tos),
.proto = IPPROTO_IPIP tunnel->parms.link);
};
struct rtable *rt = ip_route_output_key(dev_net(dev), &fl);
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
tdev = rt->dst.dev; tdev = rt->dst.dev;

View File

@ -1611,25 +1611,19 @@ static void ipmr_queue_xmit(struct net *net, struct mr_table *mrt,
#endif #endif
if (vif->flags & VIFF_TUNNEL) { if (vif->flags & VIFF_TUNNEL) {
struct flowi fl = { rt = ip_route_output_ports(net, NULL,
.oif = vif->link, vif->remote, vif->local,
.fl4_dst = vif->remote, 0, 0,
.fl4_src = vif->local, IPPROTO_IPIP,
.fl4_tos = RT_TOS(iph->tos), RT_TOS(iph->tos), vif->link);
.proto = IPPROTO_IPIP
};
rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
goto out_free; goto out_free;
encap = sizeof(struct iphdr); encap = sizeof(struct iphdr);
} else { } else {
struct flowi fl = { rt = ip_route_output_ports(net, NULL, iph->daddr, 0,
.oif = vif->link, 0, 0,
.fl4_dst = iph->daddr, IPPROTO_IPIP,
.fl4_tos = RT_TOS(iph->tos), RT_TOS(iph->tos), vif->link);
.proto = IPPROTO_IPIP
};
rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
goto out_free; goto out_free;
} }

View File

@ -536,7 +536,6 @@ ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
int err; int err;
struct sk_buff *skb2; struct sk_buff *skb2;
struct iphdr *eiph; struct iphdr *eiph;
struct flowi fl;
struct rtable *rt; struct rtable *rt;
err = ip6_tnl_err(skb, IPPROTO_IPIP, opt, &rel_type, &rel_code, err = ip6_tnl_err(skb, IPPROTO_IPIP, opt, &rel_type, &rel_code,
@ -578,11 +577,10 @@ ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
eiph = ip_hdr(skb2); eiph = ip_hdr(skb2);
/* Try to guess incoming interface */ /* Try to guess incoming interface */
memset(&fl, 0, sizeof(fl)); rt = ip_route_output_ports(dev_net(skb->dev), NULL,
fl.fl4_dst = eiph->saddr; eiph->saddr, 0,
fl.fl4_tos = RT_TOS(eiph->tos); 0, 0,
fl.proto = IPPROTO_IPIP; IPPROTO_IPIP, RT_TOS(eiph->tos), 0);
rt = ip_route_output_key(dev_net(skb->dev), &fl);
if (IS_ERR(rt)) if (IS_ERR(rt))
goto out; goto out;
@ -592,10 +590,11 @@ ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt,
if (rt->rt_flags & RTCF_LOCAL) { if (rt->rt_flags & RTCF_LOCAL) {
ip_rt_put(rt); ip_rt_put(rt);
rt = NULL; rt = NULL;
fl.fl4_dst = eiph->daddr; rt = ip_route_output_ports(dev_net(skb->dev), NULL,
fl.fl4_src = eiph->saddr; eiph->daddr, eiph->saddr,
fl.fl4_tos = eiph->tos; 0, 0,
rt = ip_route_output_key(dev_net(skb->dev), &fl); IPPROTO_IPIP,
RT_TOS(eiph->tos), 0);
if (IS_ERR(rt) || if (IS_ERR(rt) ||
rt->dst.dev->type != ARPHRD_TUNNEL) { rt->dst.dev->type != ARPHRD_TUNNEL) {
if (!IS_ERR(rt)) if (!IS_ERR(rt))

View File

@ -732,17 +732,14 @@ static netdev_tx_t ipip6_tunnel_xmit(struct sk_buff *skb,
dst = addr6->s6_addr32[3]; dst = addr6->s6_addr32[3];
} }
{ rt = ip_route_output_ports(dev_net(dev), NULL,
struct flowi fl = { .fl4_dst = dst, dst, tiph->saddr,
.fl4_src = tiph->saddr, 0, 0,
.fl4_tos = RT_TOS(tos), IPPROTO_IPV6, RT_TOS(tos),
.oif = tunnel->parms.link, tunnel->parms.link);
.proto = IPPROTO_IPV6 }; if (IS_ERR(rt)) {
rt = ip_route_output_key(dev_net(dev), &fl); dev->stats.tx_carrier_errors++;
if (IS_ERR(rt)) { goto tx_error_icmp;
dev->stats.tx_carrier_errors++;
goto tx_error_icmp;
}
} }
if (rt->rt_type != RTN_UNICAST) { if (rt->rt_type != RTN_UNICAST) {
ip_rt_put(rt); ip_rt_put(rt);
@ -858,12 +855,12 @@ static void ipip6_tunnel_bind_dev(struct net_device *dev)
iph = &tunnel->parms.iph; iph = &tunnel->parms.iph;
if (iph->daddr) { if (iph->daddr) {
struct flowi fl = { .fl4_dst = iph->daddr, struct rtable *rt = ip_route_output_ports(dev_net(dev), NULL,
.fl4_src = iph->saddr, iph->daddr, iph->saddr,
.fl4_tos = RT_TOS(iph->tos), 0, 0,
.oif = tunnel->parms.link, IPPROTO_IPV6,
.proto = IPPROTO_IPV6 }; RT_TOS(iph->tos),
struct rtable *rt = ip_route_output_key(dev_net(dev), &fl); tunnel->parms.link);
if (!IS_ERR(rt)) { if (!IS_ERR(rt)) {
tdev = rt->dst.dev; tdev = rt->dst.dev;

View File

@ -475,25 +475,17 @@ static int l2tp_ip_sendmsg(struct kiocb *iocb, struct sock *sk, struct msghdr *m
if (opt && opt->srr) if (opt && opt->srr)
daddr = opt->faddr; daddr = opt->faddr;
{ /* If this fails, retransmit mechanism of transport layer will
struct flowi fl = { .oif = sk->sk_bound_dev_if, * keep trying until route appears or the connection times
.fl4_dst = daddr, * itself out.
.fl4_src = inet->inet_saddr, */
.fl4_tos = RT_CONN_FLAGS(sk), rt = ip_route_output_ports(sock_net(sk), sk,
.proto = sk->sk_protocol, daddr, inet->inet_saddr,
.flags = inet_sk_flowi_flags(sk), inet->inet_dport, inet->inet_sport,
.fl_ip_sport = inet->inet_sport, sk->sk_protocol, RT_CONN_FLAGS(sk),
.fl_ip_dport = inet->inet_dport }; sk->sk_bound_dev_if);
if (IS_ERR(rt))
/* If this fails, retransmit mechanism of transport layer will goto no_route;
* keep trying until route appears or the connection times
* itself out.
*/
security_sk_classify_flow(sk, &fl);
rt = ip_route_output_flow(sock_net(sk), &fl, sk);
if (IS_ERR(rt))
goto no_route;
}
sk_setup_caps(sk, &rt->dst); sk_setup_caps(sk, &rt->dst);
} }
skb_dst_set(skb, dst_clone(&rt->dst)); skb_dst_set(skb, dst_clone(&rt->dst));

View File

@ -98,12 +98,7 @@ __ip_vs_get_out_rt(struct sk_buff *skb, struct ip_vs_dest *dest,
spin_lock(&dest->dst_lock); spin_lock(&dest->dst_lock);
if (!(rt = (struct rtable *) if (!(rt = (struct rtable *)
__ip_vs_dst_check(dest, rtos))) { __ip_vs_dst_check(dest, rtos))) {
struct flowi fl = { rt = ip_route_output(net, dest->addr.ip, 0, rtos, 0);
.fl4_dst = dest->addr.ip,
.fl4_tos = rtos,
};
rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
spin_unlock(&dest->dst_lock); spin_unlock(&dest->dst_lock);
IP_VS_DBG_RL("ip_route_output error, dest: %pI4\n", IP_VS_DBG_RL("ip_route_output error, dest: %pI4\n",
@ -117,12 +112,7 @@ __ip_vs_get_out_rt(struct sk_buff *skb, struct ip_vs_dest *dest,
} }
spin_unlock(&dest->dst_lock); spin_unlock(&dest->dst_lock);
} else { } else {
struct flowi fl = { rt = ip_route_output(net, daddr, 0, rtos, 0);
.fl4_dst = daddr,
.fl4_tos = rtos,
};
rt = ip_route_output_key(net, &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
IP_VS_DBG_RL("ip_route_output error, dest: %pI4\n", IP_VS_DBG_RL("ip_route_output error, dest: %pI4\n",
&daddr); &daddr);

View File

@ -36,28 +36,13 @@ static void rxrpc_destroy_peer(struct work_struct *work);
static void rxrpc_assess_MTU_size(struct rxrpc_peer *peer) static void rxrpc_assess_MTU_size(struct rxrpc_peer *peer)
{ {
struct rtable *rt; struct rtable *rt;
struct flowi fl;
peer->if_mtu = 1500; peer->if_mtu = 1500;
memset(&fl, 0, sizeof(fl)); rt = ip_route_output_ports(&init_net, NULL,
peer->srx.transport.sin.sin_addr.s_addr, 0,
switch (peer->srx.transport.family) { htons(7000), htons(7001),
case AF_INET: IPPROTO_UDP, 0, 0);
fl.oif = 0;
fl.proto = IPPROTO_UDP,
fl.fl4_dst = peer->srx.transport.sin.sin_addr.s_addr;
fl.fl4_src = 0;
fl.fl4_tos = 0;
/* assume AFS.CM talking to AFS.FS */
fl.fl_ip_sport = htons(7001);
fl.fl_ip_dport = htons(7000);
break;
default:
BUG();
}
rt = ip_route_output_key(&init_net, &fl);
if (IS_ERR(rt)) { if (IS_ERR(rt)) {
_leave(" [route err %ld]", PTR_ERR(rt)); _leave(" [route err %ld]", PTR_ERR(rt));
return; return;