Re: [PATCH v2] scsi: sd: Fix build warning in sd_revalidate_disk()

From: Damien Le Moal
Date: Fri Aug 08 2025 - 09:38:46 EST


On 8/8/25 20:30, Abinash Singh wrote:
> A build warning was triggered due to excessive stack usage in
> sd_revalidate_disk():
>
> drivers/scsi/sd.c: In function ‘sd_revalidate_disk.isra’:
> drivers/scsi/sd.c:3824:1: warning: the frame size of 1160 bytes is larger than 1024 bytes [-Wframe-larger-than=]
>
> This is caused by a large local struct queue_limits (~400B) allocated
> on the stack. Replacing it with a heap allocation using kmalloc()
> significantly reduces frame usage. Kernel stack is limited (~8 KB),
> and allocating large structs on the stack is discouraged.
> As the function already performs heap allocations (e.g. for buffer),
> this change fits well.
>
> Signed-off-by: Abinash Singh <abinashsinghlalotra@xxxxxxxxx>
> ---
>
> Hi,
>
> Thank you very much for your comments.
> I have addressed all your suggestions from v1.
>
> As you mentioned concerns regarding the readability of
> the __free(kfree) attribute, I have used the classic
> approach in v2. Additionally, I will also send v3
> where the __free() attribute is used instead.
>
> We can proceed with patch version you
> find more suitable, and do let me know if you have
> any further feedback.
>
> changelog v1->v2:
> moved declarations together
> avoided "unreadable" cleanup attribute
> splited long line
> changed the log message to diiferentiate with buffer allocation
>
> Thanks,
>
> ---
> drivers/scsi/sd.c | 49 +++++++++++++++++++++++++++++------------------
> 1 file changed, 30 insertions(+), 19 deletions(-)
>
> diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
> index 4a68b2ab2804..f5ab2a422df6 100644
> --- a/drivers/scsi/sd.c
> +++ b/drivers/scsi/sd.c
> @@ -3696,7 +3696,7 @@ static int sd_revalidate_disk(struct gendisk *disk)
> struct scsi_disk *sdkp = scsi_disk(disk);
> struct scsi_device *sdp = sdkp->device;
> sector_t old_capacity = sdkp->capacity;
> - struct queue_limits lim;
> + struct queue_limits *lim;
> unsigned char *buffer;
> unsigned int dev_max;
> int err;

If you change this to "int err = 0", then...

> @@ -3711,23 +3711,30 @@ static int sd_revalidate_disk(struct gendisk *disk)
> if (!scsi_device_online(sdp))
> goto out;
>
> + lim = kmalloc(sizeof(*lim), GFP_KERNEL);
> + if (!lim) {
> + sd_printk(KERN_WARNING, sdkp,
> + "sd_revalidate_disk: Disk limit allocation failure.\n");
> + goto out;
> + }
> +
> buffer = kmalloc(SD_BUF_SIZE, GFP_KERNEL);
> if (!buffer) {
> sd_printk(KERN_WARNING, sdkp, "sd_revalidate_disk: Memory "
> "allocation failure.\n");
> - goto out;
> + goto free_lim;

Yout can do a "goto out" here...

> }
>

[...]

> set_capacity_and_notify(disk, logical_to_sectors(sdp, sdkp->capacity));
> - sd_config_write_same(sdkp, &lim);
> + sd_config_write_same(sdkp, lim);
> kfree(buffer);

Move this line after the "out" label...

>
> - err = queue_limits_commit_update_frozen(sdkp->disk->queue, &lim);
> - if (err)
> + err = queue_limits_commit_update_frozen(sdkp->disk->queue, lim);
> + if (err) {

just do a goto out here...

> + kfree(lim);
> return err;
> + }
>
> /*
> * Query concurrent positioning ranges after
> @@ -3819,6 +3828,8 @@ static int sd_revalidate_disk(struct gendisk *disk)
> if (sd_zbc_revalidate_zones(sdkp))
> set_capacity_and_notify(disk, 0);
>
> + free_lim:
> + kfree(lim);
> out:

And this becomes:

out:
kfree(lim);
kfree(buffer)

return err;

Much cleaner :)

> return 0;
> }


--
Damien Le Moal
Western Digital Research