From: Suravee Suthikulpanit <suravee.suthikulpa...@amd.com>

By default, PCI segment is zero and can be omitted. To support system
with non-zero PCI segment ID, modify the parsing functions to allow
PCI segment ID.

Co-developed-by: Vasant Hegde <vasant.he...@amd.com>
Signed-off-by: Vasant Hegde <vasant.he...@amd.com>
Signed-off-by: Suravee Suthikulpanit <suravee.suthikulpa...@amd.com>
---
 .../admin-guide/kernel-parameters.txt         | 34 ++++++++++----
 drivers/iommu/amd/init.c                      | 44 ++++++++++++-------
 2 files changed, 52 insertions(+), 26 deletions(-)

diff --git a/Documentation/admin-guide/kernel-parameters.txt 
b/Documentation/admin-guide/kernel-parameters.txt
index f5a27f067db9..cc8f0c82ff55 100644
--- a/Documentation/admin-guide/kernel-parameters.txt
+++ b/Documentation/admin-guide/kernel-parameters.txt
@@ -2208,23 +2208,39 @@
 
        ivrs_ioapic     [HW,X86-64]
                        Provide an override to the IOAPIC-ID<->DEVICE-ID
-                       mapping provided in the IVRS ACPI table. For
-                       example, to map IOAPIC-ID decimal 10 to
-                       PCI device 00:14.0 write the parameter as:
+                       mapping provided in the IVRS ACPI table.
+                       By default, PCI segment is 0, and can be omitted.
+                       For example:
+                       * To map IOAPIC-ID decimal 10 to PCI device 00:14.0
+                         write the parameter as:
                                ivrs_ioapic[10]=00:14.0
+                       * To map IOAPIC-ID decimal 10 to PCI segment 0x1 and
+                         PCI device 00:14.0 write the parameter as:
+                               ivrs_ioapic[10]=0001:00:14.0
 
        ivrs_hpet       [HW,X86-64]
                        Provide an override to the HPET-ID<->DEVICE-ID
-                       mapping provided in the IVRS ACPI table. For
-                       example, to map HPET-ID decimal 0 to
-                       PCI device 00:14.0 write the parameter as:
+                       mapping provided in the IVRS ACPI table.
+                       By default, PCI segment is 0, and can be omitted.
+                       For example:
+                       * To map HPET-ID decimal 0 to PCI device 00:14.0
+                         write the parameter as:
                                ivrs_hpet[0]=00:14.0
+                       * To map HPET-ID decimal 10 to PCI segment 0x1 and
+                         PCI device 00:14.0 write the parameter as:
+                               ivrs_ioapic[10]=0001:00:14.0
 
        ivrs_acpihid    [HW,X86-64]
                        Provide an override to the ACPI-HID:UID<->DEVICE-ID
-                       mapping provided in the IVRS ACPI table. For
-                       example, to map UART-HID:UID AMD0020:0 to
-                       PCI device 00:14.5 write the parameter as:
+                       mapping provided in the IVRS ACPI table.
+
+                       For example, to map UART-HID:UID AMD0020:0 to
+                       PCI segment 0x1 and PCI device ID 00:14.5,
+                       write the parameter as:
+                               ivrs_acpihid[0001:00:14.5]=AMD0020:0
+
+                       By default, PCI segment is 0, and can be omitted.
+                       For example, PCI device 00:14.5 write the parameter as:
                                ivrs_acpihid[00:14.5]=AMD0020:0
 
        js=             [HW,JOY] Analog joystick
diff --git a/drivers/iommu/amd/init.c b/drivers/iommu/amd/init.c
index ca79637560a3..969b496f7e74 100644
--- a/drivers/iommu/amd/init.c
+++ b/drivers/iommu/amd/init.c
@@ -85,6 +85,10 @@
 #define ACPI_DEVFLAG_ATSDIS             0x10000000
 
 #define LOOP_TIMEOUT   100000
+
+#define IVRS_GET_SBDF_ID(seg, bus, dev, fd)    (((seg & 0xffff) << 16) | ((bus 
& 0xff) << 8) \
+                                                | ((dev & 0x1f) << 3) | (fn & 
0x7))
+
 /*
  * ACPI table definitions
  *
@@ -3287,15 +3291,17 @@ static int __init parse_amd_iommu_options(char *str)
 
 static int __init parse_ivrs_ioapic(char *str)
 {
-       unsigned int bus, dev, fn;
+       u32 seg = 0, bus, dev, fn;
        int ret, id, i;
-       u16 devid;
+       u32 devid;
 
        ret = sscanf(str, "[%d]=%x:%x.%x", &id, &bus, &dev, &fn);
-
        if (ret != 4) {
-               pr_err("Invalid command line: ivrs_ioapic%s\n", str);
-               return 1;
+               ret = sscanf(str, "[%d]=%x:%x:%x.%x", &id, &seg, &bus, &dev, 
&fn);
+               if (ret != 5) {
+                       pr_err("Invalid command line: ivrs_ioapic%s\n", str);
+                       return 1;
+               }
        }
 
        if (early_ioapic_map_size == EARLY_MAP_SIZE) {
@@ -3304,7 +3310,7 @@ static int __init parse_ivrs_ioapic(char *str)
                return 1;
        }
 
-       devid = ((bus & 0xff) << 8) | ((dev & 0x1f) << 3) | (fn & 0x7);
+       devid = IVRS_GET_SBDF_ID(seg, bus, dev, fn);
 
        cmdline_maps                    = true;
        i                               = early_ioapic_map_size++;
@@ -3317,15 +3323,17 @@ static int __init parse_ivrs_ioapic(char *str)
 
 static int __init parse_ivrs_hpet(char *str)
 {
-       unsigned int bus, dev, fn;
+       u32 seg = 0, bus, dev, fn;
        int ret, id, i;
-       u16 devid;
+       u32 devid;
 
        ret = sscanf(str, "[%d]=%x:%x.%x", &id, &bus, &dev, &fn);
-
        if (ret != 4) {
-               pr_err("Invalid command line: ivrs_hpet%s\n", str);
-               return 1;
+               ret = sscanf(str, "[%d]=%x:%x:%x.%x", &id, &seg, &bus, &dev, 
&fn);
+               if (ret != 5) {
+                       pr_err("Invalid command line: ivrs_hpet%s\n", str);
+                       return 1;
+               }
        }
 
        if (early_hpet_map_size == EARLY_MAP_SIZE) {
@@ -3334,7 +3342,7 @@ static int __init parse_ivrs_hpet(char *str)
                return 1;
        }
 
-       devid = ((bus & 0xff) << 8) | ((dev & 0x1f) << 3) | (fn & 0x7);
+       devid = IVRS_GET_SBDF_ID(seg, bus, dev, fn);
 
        cmdline_maps                    = true;
        i                               = early_hpet_map_size++;
@@ -3347,15 +3355,18 @@ static int __init parse_ivrs_hpet(char *str)
 
 static int __init parse_ivrs_acpihid(char *str)
 {
-       u32 bus, dev, fn;
+       u32 seg = 0, bus, dev, fn;
        char *hid, *uid, *p;
        char acpiid[ACPIHID_UID_LEN + ACPIHID_HID_LEN] = {0};
        int ret, i;
 
        ret = sscanf(str, "[%x:%x.%x]=%s", &bus, &dev, &fn, acpiid);
        if (ret != 4) {
-               pr_err("Invalid command line: ivrs_acpihid(%s)\n", str);
-               return 1;
+               ret = sscanf(str, "[%x:%x:%x.%x]=%s", &seg, &bus, &dev, &fn, 
acpiid);
+               if (ret != 5) {
+                       pr_err("Invalid command line: ivrs_acpihid(%s)\n", str);
+                       return 1;
+               }
        }
 
        p = acpiid;
@@ -3370,8 +3381,7 @@ static int __init parse_ivrs_acpihid(char *str)
        i = early_acpihid_map_size++;
        memcpy(early_acpihid_map[i].hid, hid, strlen(hid));
        memcpy(early_acpihid_map[i].uid, uid, strlen(uid));
-       early_acpihid_map[i].devid =
-               ((bus & 0xff) << 8) | ((dev & 0x1f) << 3) | (fn & 0x7);
+       early_acpihid_map[i].devid = IVRS_GET_SBDF_ID(seg, bus, dev, fn);
        early_acpihid_map[i].cmd_line   = true;
 
        return 1;
-- 
2.27.0

_______________________________________________
iommu mailing list
iommu@lists.linux-foundation.org
https://lists.linuxfoundation.org/mailman/listinfo/iommu

Reply via email to