]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
ata: libata-scsi: Fix CDL control
authorIgor Pylypiv <ipylypiv@google.com>
Thu, 14 Aug 2025 02:22:56 +0000 (19:22 -0700)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 28 Aug 2025 14:30:57 +0000 (16:30 +0200)
commit 58768b0563916ddcb73d8ed26ede664915f8df31 upstream.

Delete extra checks for the ATA_DFLAG_CDL_ENABLED flag that prevent
SET FEATURES command from being issued to a drive when NCQ commands
are active.

ata_mselect_control_ata_feature() sets / clears the ATA_DFLAG_CDL_ENABLED
flag during the translation of MODE SELECT to SET FEATURES. If SET FEATURES
gets deferred due to outstanding NCQ commands, the original MODE SELECT
command will be re-queued. When the re-queued MODE SELECT goes through
the ata_mselect_control_ata_feature() translation again, SET FEATURES
will not be issued because ATA_DFLAG_CDL_ENABLED has been already set or
cleared by the initial translation of MODE SELECT.

The ATA_DFLAG_CDL_ENABLED checks in ata_mselect_control_ata_feature()
are safe to remove because scsi_cdl_enable() implements a similar logic
that avoids enabling CDL if it has been enabled already.

Fixes: 17e897a45675 ("ata: libata-scsi: Improve CDL control")
Cc: stable@vger.kernel.org
Signed-off-by: Igor Pylypiv <ipylypiv@google.com>
Reviewed-by: Niklas Cassel <cassel@kernel.org>
Signed-off-by: Damien Le Moal <dlemoal@kernel.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/ata/libata-scsi.c

index cf7ca3c9eca28f919590f19d6b8dab459e8f3614..72819da71930d69fde7b7dda826b93539748ce23 100644 (file)
@@ -3752,21 +3752,16 @@ static int ata_mselect_control_ata_feature(struct ata_queued_cmd *qc,
        /* Check cdl_ctrl */
        switch (buf[0] & 0x03) {
        case 0:
-               /* Disable CDL if it is enabled */
-               if (!(dev->flags & ATA_DFLAG_CDL_ENABLED))
-                       return 0;
+               /* Disable CDL */
                ata_dev_dbg(dev, "Disabling CDL\n");
                cdl_action = 0;
                dev->flags &= ~ATA_DFLAG_CDL_ENABLED;
                break;
        case 0x02:
                /*
-                * Enable CDL if not already enabled. Since this is mutually
-                * exclusive with NCQ priority, allow this only if NCQ priority
-                * is disabled.
+                * Enable CDL. Since CDL is mutually exclusive with NCQ
+                * priority, allow this only if NCQ priority is disabled.
                 */
-               if (dev->flags & ATA_DFLAG_CDL_ENABLED)
-                       return 0;
                if (dev->flags & ATA_DFLAG_NCQ_PRIO_ENABLED) {
                        ata_dev_err(dev,
                                "NCQ priority must be disabled to enable CDL\n");