]> git.itanic.dy.fi Git - linux-stable/commitdiff
mtd: cfi_cmdset_0001: Byte swap OTP info
authorLinus Walleij <linus.walleij@linaro.org>
Fri, 20 Oct 2023 20:30:29 +0000 (22:30 +0200)
committerMiquel Raynal <miquel.raynal@bootlin.com>
Fri, 27 Oct 2023 17:45:11 +0000 (19:45 +0200)
Currently the offset into the device when looking for OTP
bits can go outside of the address of the MTD NOR devices,
and if that memory isn't readable, bad things happen
on the IXP4xx (added prints that illustrate the problem before
the crash):

cfi_intelext_otp_walk walk OTP on chip 0 start at reg_prot_offset 0x00000100
ixp4xx_copy_from copy from 0x00000100 to 0xc880dd78
cfi_intelext_otp_walk walk OTP on chip 0 start at reg_prot_offset 0x12000000
ixp4xx_copy_from copy from 0x12000000 to 0xc880dd78
8<--- cut here ---
Unable to handle kernel paging request at virtual address db000000
[db000000] *pgd=00000000
(...)

This happens in this case because the IXP4xx is big endian and
the 32- and 16-bit fields in the struct cfi_intelext_otpinfo are not
properly byteswapped. Compare to how the code in read_pri_intelext()
byteswaps the fields in struct cfi_pri_intelext.

Adding a small byte swapping loop for the OTP in read_pri_intelext()
and the crash goes away.

The problem went unnoticed for many years until I enabled
CONFIG_MTD_OTP on the IXP4xx as well, triggering the bug.

Cc: stable@vger.kernel.org
Reviewed-by: Nicolas Pitre <nico@fluxnic.net>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Signed-off-by: Miquel Raynal <miquel.raynal@bootlin.com>
Link: https://lore.kernel.org/linux-mtd/20231020-mtd-otp-byteswap-v4-1-0d132c06aa9d@linaro.org
drivers/mtd/chips/cfi_cmdset_0001.c

index 11b06fefaa0e297ef82e88401bedd48a1c95e657..c10693ba265baef4be26f38e705755d333794c1c 100644 (file)
@@ -422,9 +422,25 @@ read_pri_intelext(struct map_info *map, __u16 adr)
                extra_size = 0;
 
                /* Protection Register info */
-               if (extp->NumProtectionFields)
+               if (extp->NumProtectionFields) {
+                       struct cfi_intelext_otpinfo *otp =
+                               (struct cfi_intelext_otpinfo *)&extp->extra[0];
+
                        extra_size += (extp->NumProtectionFields - 1) *
-                                     sizeof(struct cfi_intelext_otpinfo);
+                               sizeof(struct cfi_intelext_otpinfo);
+
+                       if (extp_size >= sizeof(*extp) + extra_size) {
+                               int i;
+
+                               /* Do some byteswapping if necessary */
+                               for (i = 0; i < extp->NumProtectionFields - 1; i++) {
+                                       otp->ProtRegAddr = le32_to_cpu(otp->ProtRegAddr);
+                                       otp->FactGroups = le16_to_cpu(otp->FactGroups);
+                                       otp->UserGroups = le16_to_cpu(otp->UserGroups);
+                                       otp++;
+                               }
+                       }
+               }
        }
 
        if (extp->MinorVersion >= '1') {