Re: [PATCH v5 3/3] PCI: Introduce the disable_acs_redir parameter

From: Alex Williamson
Date: Tue Jul 10 2018 - 15:19:23 EST


On Mon, 9 Jul 2018 16:27:40 -0600
Logan Gunthorpe <logang@xxxxxxxxxxxx> wrote:

> Hey Alex,
>
> On 06/07/18 04:56 PM, Alex Williamson wrote:
> > Maybe we track if we enabled ACS via a device specific quirk and
> > minimally print an incompatibility error if it's also specified for
> > disable_acs_redir? Thanks,
>
> Ok, I dug into this a bit and I think tracking if we enabled via a
> device specific quirk does not work. If 'pci_acs_enable' is not set, we
> wouldn't know if we used a device specific quirk and still try to
> disable a quirked device the standard way.
>
> Instead, I've looked at adding a device specific disable_acs_redir
> function next to enable_acs. In this way, the device specific quirks can
> override the operation. See the diff below.
>
> Implementing the Intel SPT PCH version is pretty trivial, so I've done
> that. The Intel PCH variant I've just left as unsupported and printed a
> warning.
>
> If this sounds good to you, I'll fold it into my patchset and resubmit a v6.
>
> Thanks,
>
> Logan
>
> --
>
> diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c
> index 079f7c911e09..54001b307496 100644
> --- a/drivers/pci/pci.c
> +++ b/drivers/pci/pci.c
> @@ -3028,6 +3028,9 @@ static void pci_disable_acs_redir(struct pci_dev *dev)
> if (!pos)
> return;
>
> + if (!pci_dev_specific_disable_acs_redir(dev))
> + return;
> +
> pci_read_config_word(dev, pos + PCI_ACS_CTRL, &ctrl);
>
> /* P2P Request & Completion Redirect */
> diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c
> index f439de848658..c976a025ae92 100644
> --- a/drivers/pci/quirks.c
> +++ b/drivers/pci/quirks.c
> @@ -4526,6 +4526,16 @@ static int pci_quirk_enable_intel_pch_acs(struct
> pci_dev *dev)
> return 0;
> }
>
> +static int pci_quirk_disable_intel_pch_acs_redir(struct pci_dev *dev)
> +{
> + if (!pci_quirk_intel_pch_acs_match(dev))
> + return -ENOTTY;
> +
> + pci_warn(dev, "Disabling ACS redirect on the Intel PCH root port is
> not supported\n");
> +
> + return 0;
> +}

Note that these devices don't have an ACS capability, so they should
drop out just as any other device without an ACS capability would.
Should pci_disable_acs_redir() perhaps issue the pci_warn() for all
such devices, removing this device specific disable function?

> +
> static int pci_quirk_enable_intel_spt_pch_acs(struct pci_dev *dev)
> {
> int pos;
> @@ -4553,22 +4563,53 @@ static int
> pci_quirk_enable_intel_spt_pch_acs(struct pci_dev *dev)
> return 0;
> }
>
> -static const struct pci_dev_enable_acs {
> +static int pci_quirk_disable_intel_spt_pch_acs_redir(struct pci_dev *dev)
> +{
> + int pos;
> + u32 cap, ctrl;
> +
> + if (!pci_quirk_intel_spt_pch_acs_match(dev))
> + return -ENOTTY;
> +
> + pos = pci_find_ext_capability(dev, PCI_EXT_CAP_ID_ACS);
> + if (!pos)
> + return -ENOTTY;
> +
> + pci_read_config_dword(dev, pos + PCI_ACS_CAP, &cap);
> + pci_read_config_dword(dev, pos + INTEL_SPT_ACS_CTRL, &ctrl);
> +
> + ctrl &= ~(PCI_ACS_RR | PCI_ACS_CR | PCI_ACS_EC);
> +
> + pci_write_config_dword(dev, pos + INTEL_SPT_ACS_CTRL, ctrl);
> +
> + pci_info(dev, "Intel SPT PCH root port workaround: disabled ACS
> redirect\n");
> +
> + return 0;
> +}
> +
> +static const struct pci_dev_acs_ops {
> u16 vendor;
> u16 device;
> int (*enable_acs)(struct pci_dev *dev);
> -} pci_dev_enable_acs[] = {
> - { PCI_VENDOR_ID_INTEL, PCI_ANY_ID, pci_quirk_enable_intel_pch_acs },
> - { PCI_VENDOR_ID_INTEL, PCI_ANY_ID, pci_quirk_enable_intel_spt_pch_acs },
> + int (*disable_acs_redir)(struct pci_dev *dev);
> +} pci_dev_acs_ops[] = {
> + { PCI_VENDOR_ID_INTEL, PCI_ANY_ID,
> + .enable_acs = pci_quirk_enable_intel_pch_acs,
> + .disable_acs_redir = pci_quirk_disable_intel_pch_acs_redir,
> + },
> + { PCI_VENDOR_ID_INTEL, PCI_ANY_ID,
> + .enable_acs = pci_quirk_enable_intel_spt_pch_acs,
> + .disable_acs_redir = pci_quirk_disable_intel_spt_pch_acs_redir
> + },

Kind of cumbersome, and as above, maybe the reverse path is optional.
I wonder if there's a better callback we should use or if we should not
rely on quirks providing both.

> { 0 }
> };
>
> int pci_dev_specific_enable_acs(struct pci_dev *dev)
> {
> - const struct pci_dev_enable_acs *i;
> + const struct pci_dev_acs_ops *i;
> int ret;
>
> - for (i = pci_dev_enable_acs; i->enable_acs; i++) {
> + for (i = pci_dev_acs_ops; i->enable_acs; i++) {

Perhaps this would walk via ARRAY_SIZE if we decide one or the other
callback is optional.

> if ((i->vendor == dev->vendor ||
> i->vendor == (u16)PCI_ANY_ID) &&
> (i->device == dev->device ||
> @@ -4582,6 +4623,25 @@ int pci_dev_specific_enable_acs(struct pci_dev *dev)
> return -ENOTTY;
> }
>
> +int pci_dev_specific_disable_acs_redir(struct pci_dev *dev)
> +{
> + const struct pci_dev_acs_ops *i;
> + int ret;
> +
> + for (i = pci_dev_acs_ops; i->enable_acs; i++) {

Test i->disable_acs_redir?

> + if ((i->vendor == dev->vendor ||
> + i->vendor == (u16)PCI_ANY_ID) &&
> + (i->device == dev->device ||
> + i->device == (u16)PCI_ANY_ID)) {
> + ret = i->disable_acs_redir(dev);
> + if (ret >= 0)
> + return ret;
> + }
> + }
> +
> + return -ENOTTY;
> +}
> +
> /*
> * The PCI capabilities list for Intel DH895xCC VFs (device ID 0x0443) with
> * QuickAssist Technology (QAT) is prematurely terminated in hardware. The
> diff --git a/include/linux/pci.h b/include/linux/pci.h
> index 340029b2fb38..7ee208aa1a31 100644
> --- a/include/linux/pci.h
> +++ b/include/linux/pci.h
> @@ -1878,6 +1878,7 @@ enum pci_fixup_pass {
> void pci_fixup_device(enum pci_fixup_pass pass, struct pci_dev *dev);
> int pci_dev_specific_acs_enabled(struct pci_dev *dev, u16 acs_flags);
> int pci_dev_specific_enable_acs(struct pci_dev *dev);
> +int pci_dev_specific_disable_acs_redir(struct pci_dev *dev);

static inline version for !CONFIG_PCI_QUIRKS? Thanks,

Alex

> #else
> static inline void pci_fixup_device(enum pci_fixup_pass pass,
> struct pci_dev *dev) { }