Re: Re: [PATCH] ata: libahci: fix panic when accessing ports beyond MMIO region
From: Niklas Cassel
Date: Thu Apr 23 2026 - 13:22:02 EST
Hello liyouhong,
On Thu, Apr 23, 2026 at 05:48:54PM +0800, 李佑鸿 wrote:
> However, I have already confirmed with the BIOS supplier that when the
> BIOS disables all SATA ports, it does indeed initialize the values of
> the HOST_CAP and HOSTPorts_IMPL registers in accordance with the specifications.
>
>
> /* Register values after disabling SATA in BIOS */
> HOST_CAP (0x00) = 0xffffffff
> HOST_PORTS_IMPL (0x0c) = 0xffffffff
> HOST_VERSION (0x10) = 0xffffffff
> MMIO_SIZE = 4096
I am actually very surprised to see e.g. CAP (0x00) and AHCI VERSION (0x10)
being uninitialized.
These registers are not mentioned in:
AHCI 1.3.1 - 10.1.1 Firmware Specific Initialization
And I would have expected them to have fixed value regardless if BIOS
has enabled the controller or not, because e.g. CAP.NP (number of ports)
and the AHCI version must obviously be known when synthesizing the IP.
> lspci -vvvnns 05:00.0
> 05:00.0 SATA controller [0106]: Phytium Technology Co., Ltd. Device [1db7:d001] (rev 01) (prog-if 01 [AHCI 1.0])
> Region 5: Memory at 5b800000 (32-bit, non-prefetchable) [size=4K]
So the BAR size is 4K (when the controller is disabled).
Could you please enable the controller in BIOS, and then dump the value of
the CAP (0x00) register.
And also double check that the BAR size is still 4K when the controller is
enabled?
We could then add a quirk to drivers/ata/ahci.c, something like:
diff --git a/drivers/ata/ahci.c b/drivers/ata/ahci.c
index 1d73a53370cf..e7250781e9b7 100644
--- a/drivers/ata/ahci.c
+++ b/drivers/ata/ahci.c
@@ -2003,6 +2003,10 @@ static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
if (pdev->vendor == 0x177d && pdev->device == 0xa01c)
hpriv->irq_handler = ahci_thunderx_irq_handler;
+
+ /* Phytium SATA controller has empty CAP register */
+ if (pdev->vendor == 0x1db7 && pdev->device == 0xd001)
+ hpriv->saved_cap = 0xC734FF02;
#endif
/* save initial config */
Where you replace 0xC734FF02 with whatever you get when reading the
CAP (0x00) register when the controller is enabled.
Kind regards,
Niklas