mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/next/linux-next.git
synced 2025-01-15 13:15:57 +00:00
nvme: Adjust the Samsung APST quirk
I got a couple more reports: the Samsung APST issues appears to affect multiple 950-series devices in Dell XPS 15 9550 and Precision 5510 laptops. Change the quirk: rather than blacklisting the firmware on the first problematic SSD that was reported, disable APST on all 144d:a802 devices if they're installed in the two affected Dell models. While we're at it, disable only the deepest sleep state instead of all of them -- the reporters say that this is sufficient to fix the problem. (I have a device that appears to be entirely identical to one of the affected devices, but I have a different Dell laptop, so it's not the case that all Samsung devices with firmware BXW75D0Q are broken under all circumstances.) Samsung engineers have an affected system, and hopefully they'll give us a better workaround some time soon. In the mean time, this should minimize regressions. See https://bugs.launchpad.net/ubuntu/+source/linux/+bug/1678184 Cc: Kai-Heng Feng <kai.heng.feng@canonical.com> Signed-off-by: Andy Lutomirski <luto@kernel.org> Signed-off-by: Jens Axboe <axboe@fb.com>
This commit is contained in:
parent
4981d04dd8
commit
ff5350a86b
@ -1315,6 +1315,14 @@ static void nvme_configure_apst(struct nvme_ctrl *ctrl)
|
||||
if (target)
|
||||
table->entries[state] = target;
|
||||
|
||||
/*
|
||||
* Don't allow transitions to the deepest state
|
||||
* if it's quirked off.
|
||||
*/
|
||||
if (state == ctrl->npss &&
|
||||
(ctrl->quirks & NVME_QUIRK_NO_DEEPEST_PS))
|
||||
continue;
|
||||
|
||||
/*
|
||||
* Is this state a useful non-operational state for
|
||||
* higher-power states to autonomously transition to?
|
||||
@ -1387,16 +1395,6 @@ struct nvme_core_quirk_entry {
|
||||
};
|
||||
|
||||
static const struct nvme_core_quirk_entry core_quirks[] = {
|
||||
/*
|
||||
* Seen on a Samsung "SM951 NVMe SAMSUNG 256GB": using APST causes
|
||||
* the controller to go out to lunch. It dies when the watchdog
|
||||
* timer reads CSTS and gets 0xffffffff.
|
||||
*/
|
||||
{
|
||||
.vid = 0x144d,
|
||||
.fr = "BXW75D0Q",
|
||||
.quirks = NVME_QUIRK_NO_APST,
|
||||
},
|
||||
};
|
||||
|
||||
/* match is null-terminated but idstr is space-padded. */
|
||||
|
@ -83,6 +83,11 @@ enum nvme_quirks {
|
||||
* APST should not be used.
|
||||
*/
|
||||
NVME_QUIRK_NO_APST = (1 << 4),
|
||||
|
||||
/*
|
||||
* The deepest sleep state should not be used.
|
||||
*/
|
||||
NVME_QUIRK_NO_DEEPEST_PS = (1 << 5),
|
||||
};
|
||||
|
||||
/*
|
||||
|
@ -19,6 +19,7 @@
|
||||
#include <linux/blk-mq-pci.h>
|
||||
#include <linux/cpu.h>
|
||||
#include <linux/delay.h>
|
||||
#include <linux/dmi.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/fs.h>
|
||||
#include <linux/genhd.h>
|
||||
@ -1943,10 +1944,31 @@ static int nvme_dev_map(struct nvme_dev *dev)
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
static unsigned long check_dell_samsung_bug(struct pci_dev *pdev)
|
||||
{
|
||||
if (pdev->vendor == 0x144d && pdev->device == 0xa802) {
|
||||
/*
|
||||
* Several Samsung devices seem to drop off the PCIe bus
|
||||
* randomly when APST is on and uses the deepest sleep state.
|
||||
* This has been observed on a Samsung "SM951 NVMe SAMSUNG
|
||||
* 256GB", a "PM951 NVMe SAMSUNG 512GB", and a "Samsung SSD
|
||||
* 950 PRO 256GB", but it seems to be restricted to two Dell
|
||||
* laptops.
|
||||
*/
|
||||
if (dmi_match(DMI_SYS_VENDOR, "Dell Inc.") &&
|
||||
(dmi_match(DMI_PRODUCT_NAME, "XPS 15 9550") ||
|
||||
dmi_match(DMI_PRODUCT_NAME, "Precision 5510")))
|
||||
return NVME_QUIRK_NO_DEEPEST_PS;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int nvme_probe(struct pci_dev *pdev, const struct pci_device_id *id)
|
||||
{
|
||||
int node, result = -ENOMEM;
|
||||
struct nvme_dev *dev;
|
||||
unsigned long quirks = id->driver_data;
|
||||
|
||||
node = dev_to_node(&pdev->dev);
|
||||
if (node == NUMA_NO_NODE)
|
||||
@ -1978,8 +2000,10 @@ static int nvme_probe(struct pci_dev *pdev, const struct pci_device_id *id)
|
||||
if (result)
|
||||
goto put_pci;
|
||||
|
||||
quirks |= check_dell_samsung_bug(pdev);
|
||||
|
||||
result = nvme_init_ctrl(&dev->ctrl, &pdev->dev, &nvme_pci_ctrl_ops,
|
||||
id->driver_data);
|
||||
quirks);
|
||||
if (result)
|
||||
goto release_pools;
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user