[PATCH] nvme-pci: expose active quirks in sysfs
Chaitanya Kulkarni
chaitanyak at nvidia.com
Wed Oct 29 16:25:50 PDT 2025
On 10/29/25 01:02, Maurizio Lombardi wrote:
> On Wed Oct 29, 2025 at 6:47 AM CET, Chaitanya Kulkarni wrote:
>> On 10/28/25 10:00, Maurizio Lombardi wrote:
>>> On Tue Oct 28, 2025 at 5:32 PM CET, Bart Van Assche wrote:
>>>> On 10/28/25 7:29 AM, Maurizio Lombardi wrote:
>>>>> Currently, there is no straightforward way for a user to inspect
>>>>> the quirks value from userspace.
>>>>> Add a new read-only sysfs attribute "quirks";
>>>>> reading this file will display the hexadecimal
>>>>> value of the ctrl->quirks bitmask for the given NVMe device.
>>>> This patch changes the constants in enum nvme_quirks from
>>>> kernel-internal constants into an ABI. I'm not sure that's what we
>>>> want.
>>> I understand your concern.
>>> I could respin it to export the names of the active quirks
>>> rather than the raw bitmask, if there is an interest in having
>>> this feature.
>>>
>>> Maurizio
>>>
>> question here: do we need to care about some kind of security before we
>> expose controller information that is internal to the driver ?
>> CAP_SYS_ADMIN ? -ck
> IMO I don't think it's an information that needs to be hidden.
> After all, any user could run "lscpi -nn", get the vendor and device
> id and cross-reference them with the pci table in the source code
> to get the enabled quirks, this sysfs entry would just make it faster.
>
> Maurizio
Yes it is definitely useful after only decoding :-
linux-block (for-next) # cat /sys/class/nvme/nvme0/quirks
0x0000000000040000
Active quirks:
BOGUS_NID
linux-block (for-next) #
diff --git a/drivers/nvme/host/pci.c b/drivers/nvme/host/pci.c
index c916176bd9f0..1b9ad818b12d 100644
--- a/drivers/nvme/host/pci.c
+++ b/drivers/nvme/host/pci.c
@@ -2497,6 +2497,80 @@ static ssize_t cmbsz_show(struct device *dev, struct device_attribute *attr,
}
static DEVICE_ATTR_RO(cmbsz);
+static int nvme_get_quirks_string(unsigned long quirks, char *buf, size_t buf_len)
+{
+ size_t len = 0;
+
+ if (!quirks) {
+ len = scnprintf(buf, buf_len, "No quirks enabled\n");
+ return len;
+ }
+
+ len += scnprintf(buf + len, buf_len - len, "Active quirks:\n");
+
+ if (quirks & NVME_QUIRK_STRIPE_SIZE)
+ len += scnprintf(buf + len, buf_len - len, " STRIPE_SIZE\n");
+ if (quirks & NVME_QUIRK_IDENTIFY_CNS)
+ len += scnprintf(buf + len, buf_len - len, " IDENTIFY_CNS\n");
+ if (quirks & NVME_QUIRK_DEALLOCATE_ZEROES)
+ len += scnprintf(buf + len, buf_len - len, " DEALLOCATE_ZEROES\n");
+ if (quirks & NVME_QUIRK_DELAY_BEFORE_CHK_RDY)
+ len += scnprintf(buf + len, buf_len - len, " DELAY_BEFORE_CHK_RDY\n");
+ if (quirks & NVME_QUIRK_NO_APST)
+ len += scnprintf(buf + len, buf_len - len, " NO_APST\n");
+ if (quirks & NVME_QUIRK_NO_DEEPEST_PS)
+ len += scnprintf(buf + len, buf_len - len, " NO_DEEPEST_PS\n");
+ if (quirks & NVME_QUIRK_QDEPTH_ONE)
+ len += scnprintf(buf + len, buf_len - len, " QDEPTH_ONE\n");
+ if (quirks & NVME_QUIRK_MEDIUM_PRIO_SQ)
+ len += scnprintf(buf + len, buf_len - len, " MEDIUM_PRIO_SQ\n");
+ if (quirks & NVME_QUIRK_IGNORE_DEV_SUBNQN)
+ len += scnprintf(buf + len, buf_len - len, " IGNORE_DEV_SUBNQN\n");
+ if (quirks & NVME_QUIRK_DISABLE_WRITE_ZEROES)
+ len += scnprintf(buf + len, buf_len - len, " DISABLE_WRITE_ZEROES\n");
+ if (quirks & NVME_QUIRK_SIMPLE_SUSPEND)
+ len += scnprintf(buf + len, buf_len - len, " SIMPLE_SUSPEND\n");
+ if (quirks & NVME_QUIRK_SINGLE_VECTOR)
+ len += scnprintf(buf + len, buf_len - len, " SINGLE_VECTOR\n");
+ if (quirks & NVME_QUIRK_128_BYTES_SQES)
+ len += scnprintf(buf + len, buf_len - len, " 128_BYTES_SQES\n");
+ if (quirks & NVME_QUIRK_SHARED_TAGS)
+ len += scnprintf(buf + len, buf_len - len, " SHARED_TAGS\n");
+ if (quirks & NVME_QUIRK_NO_TEMP_THRESH_CHANGE)
+ len += scnprintf(buf + len, buf_len - len, " NO_TEMP_THRESH_CHANGE\n");
+ if (quirks & NVME_QUIRK_NO_NS_DESC_LIST)
+ len += scnprintf(buf + len, buf_len - len, " NO_NS_DESC_LIST\n");
+ if (quirks & NVME_QUIRK_DMA_ADDRESS_BITS_48)
+ len += scnprintf(buf + len, buf_len - len, " DMA_ADDRESS_BITS_48\n");
+ if (quirks & NVME_QUIRK_SKIP_CID_GEN)
+ len += scnprintf(buf + len, buf_len - len, " SKIP_CID_GEN\n");
+ if (quirks & NVME_QUIRK_BOGUS_NID)
+ len += scnprintf(buf + len, buf_len - len, " BOGUS_NID\n");
+ if (quirks & NVME_QUIRK_NO_SECONDARY_TEMP_THRESH)
+ len += scnprintf(buf + len, buf_len - len, " NO_SECONDARY_TEMP_THRESH\n");
+ if (quirks & NVME_QUIRK_FORCE_NO_SIMPLE_SUSPEND)
+ len += scnprintf(buf + len, buf_len - len, " FORCE_NO_SIMPLE_SUSPEND\n");
+ if (quirks & NVME_QUIRK_BROKEN_MSI)
+ len += scnprintf(buf + len, buf_len - len, " BROKEN_MSI\n");
+ if (quirks & NVME_QUIRK_DMAPOOL_ALIGN_512)
+ len += scnprintf(buf + len, buf_len - len, " DMAPOOL_ALIGN_512\n");
+
+ return len;
+}
+
+static ssize_t quirks_show(struct device *dev, struct device_attribute *attr,
+ char *buf)
+{
+ struct nvme_dev *ndev = to_nvme_dev(dev_get_drvdata(dev));
+ int len;
+
+ len = sysfs_emit(buf, "0x%016lx\n\n", ndev->ctrl.quirks);
+ len += nvme_get_quirks_string(ndev->ctrl.quirks, buf + len, PAGE_SIZE - len);
+
+ return len;
+}
+static DEVICE_ATTR_RO(quirks);
+
static ssize_t hmb_show(struct device *dev, struct device_attribute *attr,
char *buf)
{
@@ -2557,6 +2631,7 @@ static struct attribute *nvme_pci_attrs[] = {
&dev_attr_cmbloc.attr,
&dev_attr_cmbsz.attr,
&dev_attr_hmb.attr,
+ &dev_attr_quirks.attr,
NULL,
};
--
2.40.0
More information about the Linux-nvme
mailing list