Skip to content

Commit ae418e9

Browse files
committed
Merge branch 'ipv4-consolidate-route-lookups-from-ipv4-sockets'
Guillaume Nault says: ==================== ipv4: Consolidate route lookups from IPv4 sockets. Create inet_sk_init_flowi4() so that the different IPv4 code paths that need to do a route lookup based on an IPv4 socket don't need to reimplement that logic. ==================== Link: https://patch.msgid.link/[email protected] Signed-off-by: Jakub Kicinski <[email protected]>
2 parents 6ed3472 + c63e9f3 commit ae418e9

File tree

6 files changed

+44
-55
lines changed

6 files changed

+44
-55
lines changed

include/net/route.h

Lines changed: 28 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
#include <net/arp.h>
2929
#include <net/ndisc.h>
3030
#include <net/inet_dscp.h>
31+
#include <net/sock.h>
3132
#include <linux/in_route.h>
3233
#include <linux/rtnetlink.h>
3334
#include <linux/rcupdate.h>
@@ -129,6 +130,33 @@ struct in_device;
129130
int ip_rt_init(void);
130131
void rt_cache_flush(struct net *net);
131132
void rt_flush_dev(struct net_device *dev);
133+
134+
static inline void inet_sk_init_flowi4(const struct inet_sock *inet,
135+
struct flowi4 *fl4)
136+
{
137+
const struct ip_options_rcu *ip4_opt;
138+
const struct sock *sk;
139+
__be32 daddr;
140+
141+
rcu_read_lock();
142+
ip4_opt = rcu_dereference(inet->inet_opt);
143+
144+
/* Source routing option overrides the socket destination address */
145+
if (ip4_opt && ip4_opt->opt.srr)
146+
daddr = ip4_opt->opt.faddr;
147+
else
148+
daddr = inet->inet_daddr;
149+
rcu_read_unlock();
150+
151+
sk = &inet->sk;
152+
flowi4_init_output(fl4, sk->sk_bound_dev_if, READ_ONCE(sk->sk_mark),
153+
ip_sock_rt_tos(sk), ip_sock_rt_scope(sk),
154+
sk->sk_protocol, inet_sk_flowi_flags(sk), daddr,
155+
inet->inet_saddr, inet->inet_dport,
156+
inet->inet_sport, sk->sk_uid);
157+
security_sk_classify_flow(sk, flowi4_to_flowi_common(fl4));
158+
}
159+
132160
struct rtable *ip_route_output_key_hash(struct net *net, struct flowi4 *flp,
133161
const struct sk_buff *skb);
134162
struct rtable *ip_route_output_key_hash_rcu(struct net *net, struct flowi4 *flp,

net/ipv4/af_inet.c

Lines changed: 2 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -1309,8 +1309,6 @@ int inet_sk_rebuild_header(struct sock *sk)
13091309
{
13101310
struct rtable *rt = dst_rtable(__sk_dst_check(sk, 0));
13111311
struct inet_sock *inet = inet_sk(sk);
1312-
__be32 daddr;
1313-
struct ip_options_rcu *inet_opt;
13141312
struct flowi4 *fl4;
13151313
int err;
13161314

@@ -1319,17 +1317,9 @@ int inet_sk_rebuild_header(struct sock *sk)
13191317
return 0;
13201318

13211319
/* Reroute. */
1322-
rcu_read_lock();
1323-
inet_opt = rcu_dereference(inet->inet_opt);
1324-
daddr = inet->inet_daddr;
1325-
if (inet_opt && inet_opt->opt.srr)
1326-
daddr = inet_opt->opt.faddr;
1327-
rcu_read_unlock();
13281320
fl4 = &inet->cork.fl.u.ip4;
1329-
rt = ip_route_output_ports(sock_net(sk), fl4, sk, daddr, inet->inet_saddr,
1330-
inet->inet_dport, inet->inet_sport,
1331-
sk->sk_protocol, ip_sock_rt_tos(sk),
1332-
sk->sk_bound_dev_if);
1321+
inet_sk_init_flowi4(inet, fl4);
1322+
rt = ip_route_output_flow(sock_net(sk), fl4, sk);
13331323
if (!IS_ERR(rt)) {
13341324
err = 0;
13351325
sk_setup_caps(sk, &rt->dst);

net/ipv4/datagram.c

Lines changed: 2 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -102,8 +102,6 @@ EXPORT_SYMBOL(ip4_datagram_connect);
102102
void ip4_datagram_release_cb(struct sock *sk)
103103
{
104104
const struct inet_sock *inet = inet_sk(sk);
105-
const struct ip_options_rcu *inet_opt;
106-
__be32 daddr = inet->inet_daddr;
107105
struct dst_entry *dst;
108106
struct flowi4 fl4;
109107
struct rtable *rt;
@@ -115,14 +113,9 @@ void ip4_datagram_release_cb(struct sock *sk)
115113
rcu_read_unlock();
116114
return;
117115
}
118-
inet_opt = rcu_dereference(inet->inet_opt);
119-
if (inet_opt && inet_opt->opt.srr)
120-
daddr = inet_opt->opt.faddr;
121-
rt = ip_route_output_ports(sock_net(sk), &fl4, sk, daddr,
122-
inet->inet_saddr, inet->inet_dport,
123-
inet->inet_sport, sk->sk_protocol,
124-
ip_sock_rt_tos(sk), sk->sk_bound_dev_if);
125116

117+
inet_sk_init_flowi4(inet, &fl4);
118+
rt = ip_route_output_flow(sock_net(sk), &fl4, sk);
126119
dst = !IS_ERR(rt) ? &rt->dst : NULL;
127120
sk_dst_set(sk, dst);
128121

net/ipv4/inet_connection_sock.c

Lines changed: 2 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -1561,20 +1561,13 @@ EXPORT_SYMBOL_GPL(inet_csk_addr2sockaddr);
15611561
static struct dst_entry *inet_csk_rebuild_route(struct sock *sk, struct flowi *fl)
15621562
{
15631563
const struct inet_sock *inet = inet_sk(sk);
1564-
const struct ip_options_rcu *inet_opt;
1565-
__be32 daddr = inet->inet_daddr;
15661564
struct flowi4 *fl4;
15671565
struct rtable *rt;
15681566

15691567
rcu_read_lock();
1570-
inet_opt = rcu_dereference(inet->inet_opt);
1571-
if (inet_opt && inet_opt->opt.srr)
1572-
daddr = inet_opt->opt.faddr;
15731568
fl4 = &fl->u.ip4;
1574-
rt = ip_route_output_ports(sock_net(sk), fl4, sk, daddr,
1575-
inet->inet_saddr, inet->inet_dport,
1576-
inet->inet_sport, sk->sk_protocol,
1577-
ip_sock_rt_tos(sk), sk->sk_bound_dev_if);
1569+
inet_sk_init_flowi4(inet, fl4);
1570+
rt = ip_route_output_flow(sock_net(sk), fl4, sk);
15781571
if (IS_ERR(rt))
15791572
rt = NULL;
15801573
if (rt)

net/ipv4/ip_output.c

Lines changed: 4 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -478,24 +478,16 @@ int __ip_queue_xmit(struct sock *sk, struct sk_buff *skb, struct flowi *fl,
478478
/* Make sure we can route this packet. */
479479
rt = dst_rtable(__sk_dst_check(sk, 0));
480480
if (!rt) {
481-
__be32 daddr;
481+
inet_sk_init_flowi4(inet, fl4);
482482

483-
/* Use correct destination address if we have options. */
484-
daddr = inet->inet_daddr;
485-
if (inet_opt && inet_opt->opt.srr)
486-
daddr = inet_opt->opt.faddr;
483+
/* sctp_v4_xmit() uses its own DSCP value */
484+
fl4->flowi4_tos = tos & INET_DSCP_MASK;
487485

488486
/* If this fails, retransmit mechanism of transport layer will
489487
* keep trying until route appears or the connection times
490488
* itself out.
491489
*/
492-
rt = ip_route_output_ports(net, fl4, sk,
493-
daddr, inet->inet_saddr,
494-
inet->inet_dport,
495-
inet->inet_sport,
496-
sk->sk_protocol,
497-
tos & INET_DSCP_MASK,
498-
sk->sk_bound_dev_if);
490+
rt = ip_route_output_flow(net, fl4, sk);
499491
if (IS_ERR(rt))
500492
goto no_route;
501493
sk_setup_caps(sk, &rt->dst);

net/l2tp/l2tp_ip.c

Lines changed: 6 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -425,7 +425,6 @@ static int l2tp_ip_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)
425425
int rc;
426426
struct inet_sock *inet = inet_sk(sk);
427427
struct rtable *rt = NULL;
428-
struct flowi4 *fl4;
429428
int connected = 0;
430429
__be32 daddr;
431430

@@ -455,7 +454,6 @@ static int l2tp_ip_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)
455454
if (sk->sk_state != TCP_ESTABLISHED)
456455
goto out;
457456

458-
daddr = inet->inet_daddr;
459457
connected = 1;
460458
}
461459

@@ -482,29 +480,24 @@ static int l2tp_ip_sendmsg(struct sock *sk, struct msghdr *msg, size_t len)
482480
goto error;
483481
}
484482

485-
fl4 = &inet->cork.fl.u.ip4;
486483
if (connected)
487484
rt = dst_rtable(__sk_dst_check(sk, 0));
488485

489486
rcu_read_lock();
490487
if (!rt) {
491-
const struct ip_options_rcu *inet_opt;
488+
struct flowi4 *fl4 = &inet->cork.fl.u.ip4;
492489

493-
inet_opt = rcu_dereference(inet->inet_opt);
490+
inet_sk_init_flowi4(inet, fl4);
494491

495-
/* Use correct destination address if we have options. */
496-
if (inet_opt && inet_opt->opt.srr)
497-
daddr = inet_opt->opt.faddr;
492+
/* Overwrite ->daddr if msg->msg_name was provided */
493+
if (!connected)
494+
fl4->daddr = daddr;
498495

499496
/* If this fails, retransmit mechanism of transport layer will
500497
* keep trying until route appears or the connection times
501498
* itself out.
502499
*/
503-
rt = ip_route_output_ports(sock_net(sk), fl4, sk,
504-
daddr, inet->inet_saddr,
505-
inet->inet_dport, inet->inet_sport,
506-
sk->sk_protocol, ip_sock_rt_tos(sk),
507-
sk->sk_bound_dev_if);
500+
rt = ip_route_output_flow(sock_net(sk), fl4, sk);
508501
if (IS_ERR(rt))
509502
goto no_route;
510503
if (connected) {

0 commit comments

Comments
 (0)