Skip to content

Commit 7c85615

Browse files
scsi: sd: Fix capacity calculation with 32-bit sector_t
We previously made sure that the reported disk capacity was less than 0xffffffff blocks when the kernel was not compiled with large sector_t support (CONFIG_LBDAF). However, this check assumed that the capacity was reported in units of 512 bytes. Add a sanity check function to ensure that we only enable disks if the entire reported capacity can be expressed in terms of sector_t. Cc: <[email protected]> Reported-by: Steve Magnani <[email protected]> Cc: Bart Van Assche <[email protected]> Reviewed-by: Bart Van Assche <[email protected]> Signed-off-by: Martin K. Petersen <[email protected]>
1 parent bf6061b commit 7c85615

File tree

1 file changed

+18
-2
lines changed

1 file changed

+18
-2
lines changed

drivers/scsi/sd.c

Lines changed: 18 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2102,6 +2102,22 @@ static void read_capacity_error(struct scsi_disk *sdkp, struct scsi_device *sdp,
21022102

21032103
#define READ_CAPACITY_RETRIES_ON_RESET 10
21042104

2105+
/*
2106+
* Ensure that we don't overflow sector_t when CONFIG_LBDAF is not set
2107+
* and the reported logical block size is bigger than 512 bytes. Note
2108+
* that last_sector is a u64 and therefore logical_to_sectors() is not
2109+
* applicable.
2110+
*/
2111+
static bool sd_addressable_capacity(u64 lba, unsigned int sector_size)
2112+
{
2113+
u64 last_sector = (lba + 1ULL) << (ilog2(sector_size) - 9);
2114+
2115+
if (sizeof(sector_t) == 4 && last_sector > U32_MAX)
2116+
return false;
2117+
2118+
return true;
2119+
}
2120+
21052121
static int read_capacity_16(struct scsi_disk *sdkp, struct scsi_device *sdp,
21062122
unsigned char *buffer)
21072123
{
@@ -2167,7 +2183,7 @@ static int read_capacity_16(struct scsi_disk *sdkp, struct scsi_device *sdp,
21672183
return -ENODEV;
21682184
}
21692185

2170-
if ((sizeof(sdkp->capacity) == 4) && (lba >= 0xffffffffULL)) {
2186+
if (!sd_addressable_capacity(lba, sector_size)) {
21712187
sd_printk(KERN_ERR, sdkp, "Too big for this kernel. Use a "
21722188
"kernel compiled with support for large block "
21732189
"devices.\n");
@@ -2256,7 +2272,7 @@ static int read_capacity_10(struct scsi_disk *sdkp, struct scsi_device *sdp,
22562272
return sector_size;
22572273
}
22582274

2259-
if ((sizeof(sdkp->capacity) == 4) && (lba == 0xffffffff)) {
2275+
if (!sd_addressable_capacity(lba, sector_size)) {
22602276
sd_printk(KERN_ERR, sdkp, "Too big for this kernel. Use a "
22612277
"kernel compiled with support for large block "
22622278
"devices.\n");

0 commit comments

Comments
 (0)