ieee802154: ca8210: fix uninitialised data read
authorHarry Morris <h.morris@cascoda.com>
Wed, 28 Mar 2018 10:54:27 +0000 (11:54 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 30 May 2018 05:52:27 +0000 (07:52 +0200)
[ Upstream commit 86674a97f5055f4c7f406563408096e8cf9364ff ]

In ca8210_test_int_user_write() a user can request the transfer of a
frame with a length field (command.length) that is longer than the
actual buffer provided (len). In this scenario the driver will copy
the buffer contents into the uninitialised command[] buffer, then
transfer <data.length> bytes over the SPI even though only <len> bytes
had been populated, potentially leaking sensitive kernel memory.

Also the first 6 bytes of the command buffer must be initialised in case
a malformed, short packet is written and the uninitialised bytes are
read in ca8210_test_check_upstream.

Reported-by: Domen Puncer Kugler <domen.puncer@samsung.com>
Signed-off-by: Harry Morris <h.morris@cascoda.com>
Tested-by: Harry Morris <h.morris@cascoda.com>
Signed-off-by: Stefan Schmidt <stefan@osg.samsung.com>
Signed-off-by: Sasha Levin <alexander.levin@microsoft.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/net/ieee802154/ca8210.c

index 24a1eabbbc9da3bf1e34a966a1b14b5f737f5a35..22e466ea919a8f309fb67767513e693b4fa74769 100644 (file)
@@ -2493,13 +2493,14 @@ static ssize_t ca8210_test_int_user_write(
        struct ca8210_priv *priv = filp->private_data;
        u8 command[CA8210_SPI_BUF_SIZE];
 
-       if (len > CA8210_SPI_BUF_SIZE) {
+       memset(command, SPI_IDLE, 6);
+       if (len > CA8210_SPI_BUF_SIZE || len < 2) {
                dev_warn(
                        &priv->spi->dev,
-                       "userspace requested erroneously long write (%zu)\n",
+                       "userspace requested erroneous write length (%zu)\n",
                        len
                );
-               return -EMSGSIZE;
+               return -EBADE;
        }
 
        ret = copy_from_user(command, in_buf, len);
@@ -2511,6 +2512,13 @@ static ssize_t ca8210_test_int_user_write(
                );
                return -EIO;
        }
+       if (len != command[1] + 2) {
+               dev_err(
+                       &priv->spi->dev,
+                       "write len does not match packet length field\n"
+               );
+               return -EBADE;
+       }
 
        ret = ca8210_test_check_upstream(command, priv->spi);
        if (ret == 0) {