Skip to content

Commit e227701

Browse files
committed
Merge branch 'net-ICW-sendmsg-recvmsg'
Paolo Abeni says: ==================== net: use ICW for sk_proto->{send,recv}msg This series extends ICW usage to one of the few remaining spots in fast-path still hitting per packet retpoline overhead, namely the sk_proto->{send,recv}msg calls. The first 3 patches in this series refactor the existing code so that applying the ICW macros is straight-forward: we demux inet_{recv,send}msg in ipv4 and ipv6 variants so that each of them can easily select the appropriate TCP or UDP direct call. While at it, a new helper is created to avoid excessive code duplication, and the current ICWs for inet_{recv,send}msg are adjusted accordingly. The last 2 patches really introduce the new ICW use-case, respectively for the ipv6 and the ipv4 code path. This gives up to 5% performance improvement under UDP flood, and smaller but measurable gains for TCP RR workloads. v1 -> v2: - drop inet6_{recv,send}msg declaration from header file, prefer ICW macro instead - avoid unneeded reclaration for udp_sendmsg, as suggested by Willem ==================== Acked-by: Willem de Bruijn <[email protected]> Signed-off-by: David S. Miller <[email protected]>
2 parents 2559d7c + 6f24080 commit e227701

File tree

4 files changed

+69
-27
lines changed

4 files changed

+69
-27
lines changed

include/net/inet_common.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ int inet_dgram_connect(struct socket *sock, struct sockaddr *uaddr,
2525
int addr_len, int flags);
2626
int inet_accept(struct socket *sock, struct socket *newsock, int flags,
2727
bool kern);
28+
int inet_send_prepare(struct sock *sk);
2829
int inet_sendmsg(struct socket *sock, struct msghdr *msg, size_t size);
2930
ssize_t inet_sendpage(struct socket *sock, struct page *page, int offset,
3031
size_t size, int flags);

net/ipv4/af_inet.c

Lines changed: 20 additions & 11 deletions
Original file line numberDiff line numberDiff line change
@@ -784,18 +784,28 @@ int inet_getname(struct socket *sock, struct sockaddr *uaddr,
784784
}
785785
EXPORT_SYMBOL(inet_getname);
786786

787-
int inet_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
787+
int inet_send_prepare(struct sock *sk)
788788
{
789-
struct sock *sk = sock->sk;
790-
791789
sock_rps_record_flow(sk);
792790

793791
/* We may need to bind the socket. */
794792
if (!inet_sk(sk)->inet_num && !sk->sk_prot->no_autobind &&
795793
inet_autobind(sk))
796794
return -EAGAIN;
797795

798-
return sk->sk_prot->sendmsg(sk, msg, size);
796+
return 0;
797+
}
798+
EXPORT_SYMBOL_GPL(inet_send_prepare);
799+
800+
int inet_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
801+
{
802+
struct sock *sk = sock->sk;
803+
804+
if (unlikely(inet_send_prepare(sk)))
805+
return -EAGAIN;
806+
807+
return INDIRECT_CALL_2(sk->sk_prot->sendmsg, tcp_sendmsg, udp_sendmsg,
808+
sk, msg, size);
799809
}
800810
EXPORT_SYMBOL(inet_sendmsg);
801811

@@ -804,11 +814,7 @@ ssize_t inet_sendpage(struct socket *sock, struct page *page, int offset,
804814
{
805815
struct sock *sk = sock->sk;
806816

807-
sock_rps_record_flow(sk);
808-
809-
/* We may need to bind the socket. */
810-
if (!inet_sk(sk)->inet_num && !sk->sk_prot->no_autobind &&
811-
inet_autobind(sk))
817+
if (unlikely(inet_send_prepare(sk)))
812818
return -EAGAIN;
813819

814820
if (sk->sk_prot->sendpage)
@@ -817,6 +823,8 @@ ssize_t inet_sendpage(struct socket *sock, struct page *page, int offset,
817823
}
818824
EXPORT_SYMBOL(inet_sendpage);
819825

826+
INDIRECT_CALLABLE_DECLARE(int udp_recvmsg(struct sock *, struct msghdr *,
827+
size_t, int, int, int *));
820828
int inet_recvmsg(struct socket *sock, struct msghdr *msg, size_t size,
821829
int flags)
822830
{
@@ -827,8 +835,9 @@ int inet_recvmsg(struct socket *sock, struct msghdr *msg, size_t size,
827835
if (likely(!(flags & MSG_ERRQUEUE)))
828836
sock_rps_record_flow(sk);
829837

830-
err = sk->sk_prot->recvmsg(sk, msg, size, flags & MSG_DONTWAIT,
831-
flags & ~MSG_DONTWAIT, &addr_len);
838+
err = INDIRECT_CALL_2(sk->sk_prot->recvmsg, tcp_recvmsg, udp_recvmsg,
839+
sk, msg, size, flags & MSG_DONTWAIT,
840+
flags & ~MSG_DONTWAIT, &addr_len);
832841
if (err >= 0)
833842
msg->msg_namelen = addr_len;
834843
return err;

net/ipv6/af_inet6.c

Lines changed: 37 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -564,6 +564,39 @@ int inet6_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg)
564564
}
565565
EXPORT_SYMBOL(inet6_ioctl);
566566

567+
INDIRECT_CALLABLE_DECLARE(int udpv6_sendmsg(struct sock *, struct msghdr *,
568+
size_t));
569+
int inet6_sendmsg(struct socket *sock, struct msghdr *msg, size_t size)
570+
{
571+
struct sock *sk = sock->sk;
572+
573+
if (unlikely(inet_send_prepare(sk)))
574+
return -EAGAIN;
575+
576+
return INDIRECT_CALL_2(sk->sk_prot->sendmsg, tcp_sendmsg, udpv6_sendmsg,
577+
sk, msg, size);
578+
}
579+
580+
INDIRECT_CALLABLE_DECLARE(int udpv6_recvmsg(struct sock *, struct msghdr *,
581+
size_t, int, int, int *));
582+
int inet6_recvmsg(struct socket *sock, struct msghdr *msg, size_t size,
583+
int flags)
584+
{
585+
struct sock *sk = sock->sk;
586+
int addr_len = 0;
587+
int err;
588+
589+
if (likely(!(flags & MSG_ERRQUEUE)))
590+
sock_rps_record_flow(sk);
591+
592+
err = INDIRECT_CALL_2(sk->sk_prot->recvmsg, tcp_recvmsg, udpv6_recvmsg,
593+
sk, msg, size, flags & MSG_DONTWAIT,
594+
flags & ~MSG_DONTWAIT, &addr_len);
595+
if (err >= 0)
596+
msg->msg_namelen = addr_len;
597+
return err;
598+
}
599+
567600
const struct proto_ops inet6_stream_ops = {
568601
.family = PF_INET6,
569602
.owner = THIS_MODULE,
@@ -580,8 +613,8 @@ const struct proto_ops inet6_stream_ops = {
580613
.shutdown = inet_shutdown, /* ok */
581614
.setsockopt = sock_common_setsockopt, /* ok */
582615
.getsockopt = sock_common_getsockopt, /* ok */
583-
.sendmsg = inet_sendmsg, /* ok */
584-
.recvmsg = inet_recvmsg, /* ok */
616+
.sendmsg = inet6_sendmsg, /* retpoline's sake */
617+
.recvmsg = inet6_recvmsg, /* retpoline's sake */
585618
#ifdef CONFIG_MMU
586619
.mmap = tcp_mmap,
587620
#endif
@@ -614,8 +647,8 @@ const struct proto_ops inet6_dgram_ops = {
614647
.shutdown = inet_shutdown, /* ok */
615648
.setsockopt = sock_common_setsockopt, /* ok */
616649
.getsockopt = sock_common_getsockopt, /* ok */
617-
.sendmsg = inet_sendmsg, /* ok */
618-
.recvmsg = inet_recvmsg, /* ok */
650+
.sendmsg = inet6_sendmsg, /* retpoline's sake */
651+
.recvmsg = inet6_recvmsg, /* retpoline's sake */
619652
.mmap = sock_no_mmap,
620653
.sendpage = sock_no_sendpage,
621654
.set_peek_off = sk_set_peek_off,

net/socket.c

Lines changed: 11 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -103,13 +103,6 @@
103103
#include <net/busy_poll.h>
104104
#include <linux/errqueue.h>
105105

106-
/* proto_ops for ipv4 and ipv6 use the same {recv,send}msg function */
107-
#if IS_ENABLED(CONFIG_INET)
108-
#define INDIRECT_CALL_INET4(f, f1, ...) INDIRECT_CALL_1(f, f1, __VA_ARGS__)
109-
#else
110-
#define INDIRECT_CALL_INET4(f, f1, ...) f(__VA_ARGS__)
111-
#endif
112-
113106
#ifdef CONFIG_NET_RX_BUSY_POLL
114107
unsigned int sysctl_net_busy_read __read_mostly;
115108
unsigned int sysctl_net_busy_poll __read_mostly;
@@ -641,10 +634,13 @@ EXPORT_SYMBOL(__sock_tx_timestamp);
641634

642635
INDIRECT_CALLABLE_DECLARE(int inet_sendmsg(struct socket *, struct msghdr *,
643636
size_t));
637+
INDIRECT_CALLABLE_DECLARE(int inet6_sendmsg(struct socket *, struct msghdr *,
638+
size_t));
644639
static inline int sock_sendmsg_nosec(struct socket *sock, struct msghdr *msg)
645640
{
646-
int ret = INDIRECT_CALL_INET4(sock->ops->sendmsg, inet_sendmsg, sock,
647-
msg, msg_data_left(msg));
641+
int ret = INDIRECT_CALL_INET(sock->ops->sendmsg, inet6_sendmsg,
642+
inet_sendmsg, sock, msg,
643+
msg_data_left(msg));
648644
BUG_ON(ret == -EIOCBQUEUED);
649645
return ret;
650646
}
@@ -870,12 +866,15 @@ void __sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
870866
EXPORT_SYMBOL_GPL(__sock_recv_ts_and_drops);
871867

872868
INDIRECT_CALLABLE_DECLARE(int inet_recvmsg(struct socket *, struct msghdr *,
873-
size_t , int ));
869+
size_t, int));
870+
INDIRECT_CALLABLE_DECLARE(int inet6_recvmsg(struct socket *, struct msghdr *,
871+
size_t, int));
874872
static inline int sock_recvmsg_nosec(struct socket *sock, struct msghdr *msg,
875873
int flags)
876874
{
877-
return INDIRECT_CALL_INET4(sock->ops->recvmsg, inet_recvmsg, sock, msg,
878-
msg_data_left(msg), flags);
875+
return INDIRECT_CALL_INET(sock->ops->recvmsg, inet6_recvmsg,
876+
inet_recvmsg, sock, msg, msg_data_left(msg),
877+
flags);
879878
}
880879

881880
/**

0 commit comments

Comments
 (0)