[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-Id: <20250928062756.2188329-3-yilun.xu@linux.intel.com>
Date: Sun, 28 Sep 2025 14:27:55 +0800
From: Xu Yilun <yilun.xu@...ux.intel.com>
To: linux-coco@...ts.linux.dev,
linux-pci@...r.kernel.org,
dan.j.williams@...el.com
Cc: yilun.xu@...el.com,
yilun.xu@...ux.intel.com,
baolu.lu@...ux.intel.com,
zhenzhong.duan@...el.com,
aneesh.kumar@...nel.org,
bhelgaas@...gle.com,
aik@....com,
linux-kernel@...r.kernel.org
Subject: [PATCH 2/3] PCI/IDE: Add Address Association Register setup for RP
Add Address Association Register setup for Root Ports.
The address ranges for RP side Address Association Registers should
cover memory addresses for all PFs/VFs/downstream devices of the DSM
device. A simple solution is to get the aggregated 32-bit and 64-bit
address ranges from directly connected downstream port (either an RP or
a switch port) and set into 2 Address Association Register blocks.
There is a case the platform doesn't require Address Association
Registers setup and provides no register block for RP (AMD). Will skip
the setup in pci_ide_stream_setup().
Also imaging another case where there is only one block for RP.
Prioritize 64-bit address ranges setup for it. No strong reason for the
preference until a real use case comes.
The Address Association Register setup for Endpoint Side is still
uncertain so isn't supported in this patch.
Take the oppotunity to export some mini helpers for Address Association
Registers setup. TDX Connect needs the provided aggregated address
ranges but will use specific firmware calls for actual setup instead of
pci_ide_stream_setup().
Co-developed-by: Aneesh Kumar K.V <aneesh.kumar@...nel.org>
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@...nel.org>
Co-developed-by: Arto Merilainen <amerilainen@...dia.com>
Signed-off-by: Arto Merilainen <amerilainen@...dia.com>
Signed-off-by: Xu Yilun <yilun.xu@...ux.intel.com>
---
include/linux/pci-ide.h | 11 +++++++
drivers/pci/ide.c | 64 ++++++++++++++++++++++++++++++++++++++++-
2 files changed, 74 insertions(+), 1 deletion(-)
diff --git a/include/linux/pci-ide.h b/include/linux/pci-ide.h
index 5adbd8b81f65..ac84fb611963 100644
--- a/include/linux/pci-ide.h
+++ b/include/linux/pci-ide.h
@@ -6,6 +6,15 @@
#ifndef __PCI_IDE_H__
#define __PCI_IDE_H__
+#define SEL_ADDR1_LOWER GENMASK(31, 20)
+#define SEL_ADDR_UPPER GENMASK_ULL(63, 32)
+#define PREP_PCI_IDE_SEL_ADDR1(base, limit) \
+ (FIELD_PREP(PCI_IDE_SEL_ADDR_1_VALID, 1) | \
+ FIELD_PREP(PCI_IDE_SEL_ADDR_1_BASE_LOW, \
+ FIELD_GET(SEL_ADDR1_LOWER, (base))) | \
+ FIELD_PREP(PCI_IDE_SEL_ADDR_1_LIMIT_LOW, \
+ FIELD_GET(SEL_ADDR1_LOWER, (limit))))
+
#define PREP_PCI_IDE_SEL_RID_2(base, domain) \
(FIELD_PREP(PCI_IDE_SEL_RID_2_VALID, 1) | \
FIELD_PREP(PCI_IDE_SEL_RID_2_BASE, (base)) | \
@@ -42,6 +51,8 @@ struct pci_ide_partner {
unsigned int default_stream:1;
unsigned int setup:1;
unsigned int enable:1;
+ struct range mem32;
+ struct range mem64;
};
/**
diff --git a/drivers/pci/ide.c b/drivers/pci/ide.c
index 7633b8e52399..8db1163737e5 100644
--- a/drivers/pci/ide.c
+++ b/drivers/pci/ide.c
@@ -159,7 +159,11 @@ struct pci_ide *pci_ide_stream_alloc(struct pci_dev *pdev)
struct stream_index __stream[PCI_IDE_HB + 1];
struct pci_host_bridge *hb;
struct pci_dev *rp;
+ struct pci_dev *br;
int num_vf, rid_end;
+ struct range mem32 = {}, mem64 = {};
+ struct pci_bus_region region;
+ struct resource *res;
if (!pci_is_pcie(pdev))
return NULL;
@@ -206,6 +210,24 @@ struct pci_ide *pci_ide_stream_alloc(struct pci_dev *pdev)
else
rid_end = pci_dev_id(pdev);
+ br = pci_upstream_bridge(pdev);
+ if (!br)
+ return NULL;
+
+ res = &br->resource[PCI_BRIDGE_MEM_WINDOW];
+ if (res->flags & IORESOURCE_MEM) {
+ pcibios_resource_to_bus(br->bus, ®ion, res);
+ mem32.start = region.start;
+ mem32.end = region.end;
+ }
+
+ res = &br->resource[PCI_BRIDGE_PREF_MEM_WINDOW];
+ if (res->flags & IORESOURCE_PREFETCH) {
+ pcibios_resource_to_bus(br->bus, ®ion, res);
+ mem64.start = region.start;
+ mem64.end = region.end;
+ }
+
*ide = (struct pci_ide) {
.pdev = pdev,
.partner = {
@@ -218,6 +240,8 @@ struct pci_ide *pci_ide_stream_alloc(struct pci_dev *pdev)
.rid_start = pci_dev_id(pdev),
.rid_end = rid_end,
.stream_index = no_free_ptr(rp_stream)->stream_index,
+ .mem32 = mem32,
+ .mem64 = mem64,
},
},
.host_bridge_stream = no_free_ptr(hb_stream)->stream_index,
@@ -397,6 +421,21 @@ static void set_ide_sel_ctl(struct pci_dev *pdev, struct pci_ide *ide,
pci_write_config_dword(pdev, pos + PCI_IDE_SEL_CTL, val);
}
+static void set_ide_sel_addr(struct pci_dev *pdev, int pos, int assoc_idx,
+ struct range *mem)
+{
+ u32 val;
+
+ val = PREP_PCI_IDE_SEL_ADDR1(mem->start, mem->end);
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_1(assoc_idx), val);
+
+ val = FIELD_GET(SEL_ADDR_UPPER, mem->end);
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_2(assoc_idx), val);
+
+ val = FIELD_GET(SEL_ADDR_UPPER, mem->start);
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_3(assoc_idx), val);
+}
+
/**
* pci_ide_stream_setup() - program settings to Selective IDE Stream registers
* @pdev: PCIe device object for either a Root Port or Endpoint Partner Port
@@ -410,6 +449,7 @@ static void set_ide_sel_ctl(struct pci_dev *pdev, struct pci_ide *ide,
void pci_ide_stream_setup(struct pci_dev *pdev, struct pci_ide *ide)
{
struct pci_ide_partner *settings = pci_ide_to_settings(pdev, ide);
+ u8 assoc_idx = 0;
int pos;
u32 val;
@@ -424,6 +464,21 @@ void pci_ide_stream_setup(struct pci_dev *pdev, struct pci_ide *ide)
val = PREP_PCI_IDE_SEL_RID_2(settings->rid_start, pci_ide_domain(pdev));
pci_write_config_dword(pdev, pos + PCI_IDE_SEL_RID_2, val);
+ /*
+ * Feel free to change the default stratagy, Intel & AMD don't directly
+ * setup RP registers.
+ *
+ * 64 bit memory first, assuming it's more popular.
+ */
+ if (assoc_idx < pdev->nr_ide_mem && settings->mem64.end != 0) {
+ set_ide_sel_addr(pdev, pos, assoc_idx, &settings->mem64);
+ assoc_idx++;
+ }
+
+ /* 64 bit memory in lower block and 32 bit in higher block, any risk? */
+ if (assoc_idx < pdev->nr_ide_mem && settings->mem32.end != 0)
+ set_ide_sel_addr(pdev, pos, assoc_idx, &settings->mem32);
+
/*
* Setup control register early for devices that expect
* stream_id is set during key programming.
@@ -445,7 +500,7 @@ EXPORT_SYMBOL_GPL(pci_ide_stream_setup);
void pci_ide_stream_teardown(struct pci_dev *pdev, struct pci_ide *ide)
{
struct pci_ide_partner *settings = pci_ide_to_settings(pdev, ide);
- int pos;
+ int pos, i;
if (!settings)
return;
@@ -453,6 +508,13 @@ void pci_ide_stream_teardown(struct pci_dev *pdev, struct pci_ide *ide)
pos = sel_ide_offset(pdev, settings);
pci_write_config_dword(pdev, pos + PCI_IDE_SEL_CTL, 0);
+
+ for (i = 0; i < pdev->nr_ide_mem; i++) {
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_1(i), 0);
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_2(i), 0);
+ pci_write_config_dword(pdev, pos + PCI_IDE_SEL_ADDR_3(i), 0);
+ }
+
pci_write_config_dword(pdev, pos + PCI_IDE_SEL_RID_2, 0);
pci_write_config_dword(pdev, pos + PCI_IDE_SEL_RID_1, 0);
settings->setup = 0;
--
2.25.1
Powered by blists - more mailing lists