fs/fat: Avoid corruption of sectors following the FAT
authorStefan Brüns <stefan.bruens@rwth-aachen.de>
Fri, 16 Dec 2016 23:27:50 +0000 (00:27 +0100)
committerTom Rini <trini@konsulko.com>
Tue, 27 Dec 2016 16:24:13 +0000 (11:24 -0500)
The FAT is read/flushed in segments of 6 (FATBUFBLOCKS) disk sectors. The
last segment may be less than 6 sectors, cap the length.

Signed-off-by: Stefan Brüns <stefan.bruens@rwth-aachen.de>
Reviewed-by: Benoît Thébaudeau <benoit.thebaudeau.dev@gmail.com>
fs/fat/fat.c
fs/fat/fat_write.c

index df9f2b56562aafae900974fb85ac236d3d8a2975..631958140674ae9544affcb875d03d6f695c92c5 100644 (file)
@@ -202,6 +202,7 @@ static __u32 get_fatent(fsdata *mydata, __u32 entry)
                __u32 fatlength = mydata->fatlength;
                __u32 startblock = bufnum * FATBUFBLOCKS;
 
+               /* Cap length if fatlength is not a multiple of FATBUFBLOCKS */
                if (startblock + getsize > fatlength)
                        getsize = fatlength - startblock;
 
index aab0b0e71d8ed3e1bf27be5d8444bb97032f8af4..4605020854c705c669fe2abc276c49b32d7417b5 100644 (file)
@@ -117,10 +117,11 @@ static int flush_dirty_fat_buffer(fsdata *mydata)
        if ((!mydata->fat_dirty) || (mydata->fatbufnum == -1))
                return 0;
 
-       startblock += mydata->fat_sect;
+       /* Cap length if fatlength is not a multiple of FATBUFBLOCKS */
+       if (startblock + getsize > fatlength)
+               getsize = fatlength - startblock;
 
-       if (getsize > fatlength)
-               getsize = fatlength;
+       startblock += mydata->fat_sect;
 
        /* Write FAT buf */
        if (disk_write(startblock, getsize, bufptr) < 0) {
@@ -187,8 +188,9 @@ static __u32 get_fatent_value(fsdata *mydata, __u32 entry)
                __u32 fatlength = mydata->fatlength;
                __u32 startblock = bufnum * FATBUFBLOCKS;
 
-               if (getsize > fatlength)
-                       getsize = fatlength;
+               /* Cap length if fatlength is not a multiple of FATBUFBLOCKS */
+               if (startblock + getsize > fatlength)
+                       getsize = fatlength - startblock;
 
                startblock += mydata->fat_sect; /* Offset from start of disk */
 
@@ -499,15 +501,15 @@ static int set_fatent_value(fsdata *mydata, __u32 entry, __u32 entry_value)
                __u32 fatlength = mydata->fatlength;
                __u32 startblock = bufnum * FATBUFBLOCKS;
 
-               fatlength *= mydata->sect_size;
-               startblock += mydata->fat_sect;
-
-               if (getsize > fatlength)
-                       getsize = fatlength;
+               /* Cap length if fatlength is not a multiple of FATBUFBLOCKS */
+               if (startblock + getsize > fatlength)
+                       getsize = fatlength - startblock;
 
                if (flush_dirty_fat_buffer(mydata) < 0)
                        return -1;
 
+               startblock += mydata->fat_sect;
+
                if (disk_read(startblock, getsize, bufptr) < 0) {
                        debug("Error reading FAT blocks\n");
                        return -1;