Takashi Iwai 8f2311
From e41473543f75f7dbc5d605007e6f883f1bd13b9a Mon Sep 17 00:00:00 2001
Takashi Iwai 8f2311
From: Daniel Jordan <daniel.m.jordan@oracle.com>
Takashi Iwai 8f2311
Date: Wed, 27 Oct 2021 17:59:20 -0400
Takashi Iwai 8f2311
Subject: [PATCH] net/tls: Fix flipped sign in tls_err_abort() calls
Takashi Iwai 8f2311
Git-commit: da353fac65fede6b8b4cfe207f0d9408e3121105
Takashi Iwai 8f2311
Patch-mainline: v5.15
Takashi Iwai 8f2311
References: stable-5.14.16
Takashi Iwai 8f2311
Takashi Iwai 8f2311
commit da353fac65fede6b8b4cfe207f0d9408e3121105 upstream.
Takashi Iwai 8f2311
Takashi Iwai 8f2311
sk->sk_err appears to expect a positive value, a convention that ktls
Takashi Iwai 8f2311
doesn't always follow and that leads to memory corruption in other code.
Takashi Iwai 8f2311
For instance,
Takashi Iwai 8f2311
Takashi Iwai 8f2311
    [kworker]
Takashi Iwai 8f2311
    tls_encrypt_done(..., err=<negative error from crypto request>)
Takashi Iwai 8f2311
      tls_err_abort(.., err)
Takashi Iwai 8f2311
        sk->sk_err = err;
Takashi Iwai 8f2311
Takashi Iwai 8f2311
    [task]
Takashi Iwai 8f2311
    splice_from_pipe_feed
Takashi Iwai 8f2311
      ...
Takashi Iwai 8f2311
        tls_sw_do_sendpage
Takashi Iwai 8f2311
          if (sk->sk_err) {
Takashi Iwai 8f2311
            ret = -sk->sk_err;  // ret is positive
Takashi Iwai 8f2311
Takashi Iwai 8f2311
    splice_from_pipe_feed (continued)
Takashi Iwai 8f2311
      ret = actor(...)  // ret is still positive and interpreted as bytes
Takashi Iwai 8f2311
                        // written, resulting in underflow of buf->len and
Takashi Iwai 8f2311
                        // sd->len, leading to huge buf->offset and bogus
Takashi Iwai 8f2311
                        // addresses computed in later calls to actor()
Takashi Iwai 8f2311
Takashi Iwai 8f2311
Fix all tls_err_abort() callers to pass a negative error code
Takashi Iwai 8f2311
consistently and centralize the error-prone sign flip there, throwing in
Takashi Iwai 8f2311
a warning to catch future misuse and uninlining the function so it
Takashi Iwai 8f2311
really does only warn once.
Takashi Iwai 8f2311
Takashi Iwai 8f2311
Cc: stable@vger.kernel.org
Takashi Iwai 8f2311
Fixes: c46234ebb4d1e ("tls: RX path for ktls")
Takashi Iwai 8f2311
Reported-by: syzbot+b187b77c8474f9648fae@syzkaller.appspotmail.com
Takashi Iwai 8f2311
Signed-off-by: Daniel Jordan <daniel.m.jordan@oracle.com>
Takashi Iwai 8f2311
Signed-off-by: David S. Miller <davem@davemloft.net>
Takashi Iwai 8f2311
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Takashi Iwai 8f2311
Acked-by: Takashi Iwai <tiwai@suse.de>
Takashi Iwai 8f2311
Takashi Iwai 8f2311
---
Takashi Iwai 8f2311
 include/net/tls.h |  9 ++-------
Takashi Iwai 8f2311
 net/tls/tls_sw.c  | 17 +++++++++++++----
Takashi Iwai 8f2311
 2 files changed, 15 insertions(+), 11 deletions(-)
Takashi Iwai 8f2311
Takashi Iwai 8f2311
diff --git a/include/net/tls.h b/include/net/tls.h
Takashi Iwai 8f2311
index be4b3e1cac46..64217c9873c9 100644
Takashi Iwai 8f2311
--- a/include/net/tls.h
Takashi Iwai 8f2311
+++ b/include/net/tls.h
Takashi Iwai 8f2311
@@ -358,6 +358,7 @@ int tls_sk_query(struct sock *sk, int optname, char __user *optval,
Takashi Iwai 8f2311
 		int __user *optlen);
Takashi Iwai 8f2311
 int tls_sk_attach(struct sock *sk, int optname, char __user *optval,
Takashi Iwai 8f2311
 		  unsigned int optlen);
Takashi Iwai 8f2311
+void tls_err_abort(struct sock *sk, int err);
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
 int tls_set_sw_offload(struct sock *sk, struct tls_context *ctx, int tx);
Takashi Iwai 8f2311
 void tls_sw_strparser_arm(struct sock *sk, struct tls_context *ctx);
Takashi Iwai 8f2311
@@ -466,12 +467,6 @@ static inline bool tls_is_sk_tx_device_offloaded(struct sock *sk)
Takashi Iwai 8f2311
 #endif
Takashi Iwai 8f2311
 }
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
-static inline void tls_err_abort(struct sock *sk, int err)
Takashi Iwai 8f2311
-{
Takashi Iwai 8f2311
-	sk->sk_err = err;
Takashi Iwai 8f2311
-	sk_error_report(sk);
Takashi Iwai 8f2311
-}
Takashi Iwai 8f2311
-
Takashi Iwai 8f2311
 static inline bool tls_bigint_increment(unsigned char *seq, int len)
Takashi Iwai 8f2311
 {
Takashi Iwai 8f2311
 	int i;
Takashi Iwai 8f2311
@@ -512,7 +507,7 @@ static inline void tls_advance_record_sn(struct sock *sk,
Takashi Iwai 8f2311
 					 struct cipher_context *ctx)
Takashi Iwai 8f2311
 {
Takashi Iwai 8f2311
 	if (tls_bigint_increment(ctx->rec_seq, prot->rec_seq_size))
Takashi Iwai 8f2311
-		tls_err_abort(sk, EBADMSG);
Takashi Iwai 8f2311
+		tls_err_abort(sk, -EBADMSG);
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
 	if (prot->version != TLS_1_3_VERSION &&
Takashi Iwai 8f2311
 	    prot->cipher_type != TLS_CIPHER_CHACHA20_POLY1305)
Takashi Iwai 8f2311
diff --git a/net/tls/tls_sw.c b/net/tls/tls_sw.c
Takashi Iwai 8f2311
index 4feb95e34b64..50da7ce5f54b 100644
Takashi Iwai 8f2311
--- a/net/tls/tls_sw.c
Takashi Iwai 8f2311
+++ b/net/tls/tls_sw.c
Takashi Iwai 8f2311
@@ -35,6 +35,7 @@
Takashi Iwai 8f2311
  * SOFTWARE.
Takashi Iwai 8f2311
  */
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
+#include <linux/bug.h>
Takashi Iwai 8f2311
 #include <linux/sched/signal.h>
Takashi Iwai 8f2311
 #include <linux/module.h>
Takashi Iwai 8f2311
 #include <linux/splice.h>
Takashi Iwai 8f2311
@@ -43,6 +44,14 @@
Takashi Iwai 8f2311
 #include <net/strparser.h>
Takashi Iwai 8f2311
 #include <net/tls.h>
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
+noinline void tls_err_abort(struct sock *sk, int err)
Takashi Iwai 8f2311
+{
Takashi Iwai 8f2311
+	WARN_ON_ONCE(err >= 0);
Takashi Iwai 8f2311
+	/* sk->sk_err should contain a positive error code. */
Takashi Iwai 8f2311
+	sk->sk_err = -err;
Takashi Iwai 8f2311
+	sk_error_report(sk);
Takashi Iwai 8f2311
+}
Takashi Iwai 8f2311
+
Takashi Iwai 8f2311
 static int __skb_nsg(struct sk_buff *skb, int offset, int len,
Takashi Iwai 8f2311
                      unsigned int recursion_level)
Takashi Iwai 8f2311
 {
Takashi Iwai 8f2311
@@ -419,7 +428,7 @@ int tls_tx_records(struct sock *sk, int flags)
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
 tx_err:
Takashi Iwai 8f2311
 	if (rc < 0 && rc != -EAGAIN)
Takashi Iwai 8f2311
-		tls_err_abort(sk, EBADMSG);
Takashi Iwai 8f2311
+		tls_err_abort(sk, -EBADMSG);
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
 	return rc;
Takashi Iwai 8f2311
 }
Takashi Iwai 8f2311
@@ -763,7 +772,7 @@ static int tls_push_record(struct sock *sk, int flags,
Takashi Iwai 8f2311
 			       msg_pl->sg.size + prot->tail_size, i);
Takashi Iwai 8f2311
 	if (rc < 0) {
Takashi Iwai 8f2311
 		if (rc != -EINPROGRESS) {
Takashi Iwai 8f2311
-			tls_err_abort(sk, EBADMSG);
Takashi Iwai 8f2311
+			tls_err_abort(sk, -EBADMSG);
Takashi Iwai 8f2311
 			if (split) {
Takashi Iwai 8f2311
 				tls_ctx->pending_open_record_frags = true;
Takashi Iwai 8f2311
 				tls_merge_open_record(sk, rec, tmp, orig_end);
Takashi Iwai 8f2311
@@ -1827,7 +1836,7 @@ int tls_sw_recvmsg(struct sock *sk,
Takashi Iwai 8f2311
 		err = decrypt_skb_update(sk, skb, &msg->msg_iter,
Takashi Iwai 8f2311
 					 &chunk, &zc, async_capable);
Takashi Iwai 8f2311
 		if (err < 0 && err != -EINPROGRESS) {
Takashi Iwai 8f2311
-			tls_err_abort(sk, EBADMSG);
Takashi Iwai 8f2311
+			tls_err_abort(sk, -EBADMSG);
Takashi Iwai 8f2311
 			goto recv_end;
Takashi Iwai 8f2311
 		}
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
@@ -2007,7 +2016,7 @@ ssize_t tls_sw_splice_read(struct socket *sock,  loff_t *ppos,
Takashi Iwai 8f2311
 		}
Takashi Iwai 8f2311
 
Takashi Iwai 8f2311
 		if (err < 0) {
Takashi Iwai 8f2311
-			tls_err_abort(sk, EBADMSG);
Takashi Iwai 8f2311
+			tls_err_abort(sk, -EBADMSG);
Takashi Iwai 8f2311
 			goto splice_read_end;
Takashi Iwai 8f2311
 		}
Takashi Iwai 8f2311
 		ctx->decrypted = 1;
Takashi Iwai 8f2311
-- 
Takashi Iwai 8f2311
2.26.2
Takashi Iwai 8f2311