[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20200916223143.GA1590208@bjorn-Precision-5520>
Date: Wed, 16 Sep 2020 17:31:43 -0500
From: Bjorn Helgaas <helgaas@...nel.org>
To: Krzysztof Wilczyński <kw@...ux.com>
Cc: Bjorn Helgaas <bhelgaas@...gle.com>,
"David S. Miller" <davem@...emloft.net>,
Jakub Kicinski <kuba@...nel.org>,
Jay Cliburn <jcliburn@...il.com>,
Chris Snook <chris.snook@...il.com>,
Edward Cree <ecree@...arflare.com>,
Martin Habets <mhabets@...arflare.com>, netdev@...r.kernel.org,
linux-pci@...r.kernel.org, linux-net-drivers@...arflare.com,
Kai-Heng Feng <kai.heng.feng@...onical.com>,
Colin Ian King <colin.king@...onical.com>,
Myron Stowe <myron.stowe@...hat.com>
Subject: Re: [PATCH] Convert enum pci_dev_flags to bit fields in struct
pci_dev
[+cc Kai-Heng, Colin, Myron]
On Mon, Sep 14, 2020 at 03:57:56AM +0000, Krzysztof Wilczyński wrote:
> All the flags defined in the enum pci_dev_flags are used to determine
> whether a particular feature of an underlying PCI device should be used
> or not - features are also often disabled via a device-specific quirk.
>
> These flags are tightly coupled with a PCI device and primarily used in
> simple binary on/off manner to check whether something is enabled or
> disabled, and have almost no other users (aside of two network drivers)
> outside of the PCI device drivers space.
>
> Therefore, convert enum pci_dev_flags into a set of bit fields in the
> struct pci_dev, and then drop said enum and the typedef pci_dev_flags_t.
>
> This will keep PCI device-specific features as part of the struct
> pci_dev and make the code that used to use flags simpler.
>
> Suggested-by: Bjorn Helgaas <bhelgaas@...gle.com>
> Signed-off-by: Krzysztof Wilczyński <kw@...ux.com>
I like this because we currently have two styles for setting per-PCI
dev flags:
pdev->dev_flags |= PCI_DEV_FLAGS_NO_D3
pdev->no_d3cold = true;
and there's no obvious reason to choose one way over the other. This
patch converts everything to the second style.
We might look at doing pci_bus_flags_t at the same time.
How much heartburn does this cause distro folks? You can let me know
off-list if you want :)
Generally we don't worry too much in the upstream world about breaking
out-of-tree modules, and these are trivial changes that affect very
few drivers anyway, but I don't want to gratuitously make things hard
for distros.
> ---
> drivers/net/ethernet/atheros/alx/main.c | 2 +-
> drivers/net/ethernet/sfc/ef10_sriov.c | 3 +-
> drivers/pci/msi.c | 2 +-
> drivers/pci/pci.c | 22 ++++++------
> drivers/pci/probe.c | 2 +-
> drivers/pci/quirks.c | 24 ++++++-------
> drivers/pci/search.c | 4 +--
> drivers/pci/vpd.c | 4 +--
> include/linux/pci.h | 47 +++++++++----------------
> 9 files changed, 47 insertions(+), 63 deletions(-)
>
> diff --git a/drivers/net/ethernet/atheros/alx/main.c b/drivers/net/ethernet/atheros/alx/main.c
> index 9b7f1af5f574..c52669f8ec26 100644
> --- a/drivers/net/ethernet/atheros/alx/main.c
> +++ b/drivers/net/ethernet/atheros/alx/main.c
> @@ -1763,7 +1763,7 @@ static int alx_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
> netdev->watchdog_timeo = ALX_WATCHDOG_TIME;
>
> if (ent->driver_data & ALX_DEV_QUIRK_MSI_INTX_DISABLE_BUG)
> - pdev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
> + pdev->msi_intx_disabled = 1;
>
> err = alx_init_sw(alx);
> if (err) {
> diff --git a/drivers/net/ethernet/sfc/ef10_sriov.c b/drivers/net/ethernet/sfc/ef10_sriov.c
> index 21fa6c0e8873..9af7e11ea113 100644
> --- a/drivers/net/ethernet/sfc/ef10_sriov.c
> +++ b/drivers/net/ethernet/sfc/ef10_sriov.c
> @@ -122,8 +122,7 @@ static void efx_ef10_sriov_free_vf_vports(struct efx_nic *efx)
> struct ef10_vf *vf = nic_data->vf + i;
>
> /* If VF is assigned, do not free the vport */
> - if (vf->pci_dev &&
> - vf->pci_dev->dev_flags & PCI_DEV_FLAGS_ASSIGNED)
> + if (vf->pci_dev && vf->pci_dev->flags_assigned)
> continue;
>
> if (vf->vport_assigned) {
> diff --git a/drivers/pci/msi.c b/drivers/pci/msi.c
> index 30ae4ffda5c1..719ae72d9028 100644
> --- a/drivers/pci/msi.c
> +++ b/drivers/pci/msi.c
> @@ -405,7 +405,7 @@ static void free_msi_irqs(struct pci_dev *dev)
>
> static void pci_intx_for_msi(struct pci_dev *dev, int enable)
> {
> - if (!(dev->dev_flags & PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG))
> + if (!dev->msi_intx_disabled)
> pci_intx(dev, enable);
> }
>
> diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c
> index e39c5499770f..08ffe872c34c 100644
> --- a/drivers/pci/pci.c
> +++ b/drivers/pci/pci.c
> @@ -1320,7 +1320,7 @@ int pci_set_power_state(struct pci_dev *dev, pci_power_t state)
> * This device is quirked not to be put into D3, so don't put it in
> * D3
> */
> - if (state >= PCI_D3hot && (dev->dev_flags & PCI_DEV_FLAGS_NO_D3))
> + if (state >= PCI_D3hot && dev->no_d3)
> return 0;
>
> /*
> @@ -4528,7 +4528,7 @@ bool pcie_has_flr(struct pci_dev *dev)
> {
> u32 cap;
>
> - if (dev->dev_flags & PCI_DEV_FLAGS_NO_FLR_RESET)
> + if (dev->no_flr_reset)
> return false;
>
> pcie_capability_read_dword(dev, PCI_EXP_DEVCAP, &cap);
> @@ -4574,7 +4574,7 @@ static int pci_af_flr(struct pci_dev *dev, int probe)
> if (!pos)
> return -ENOTTY;
>
> - if (dev->dev_flags & PCI_DEV_FLAGS_NO_FLR_RESET)
> + if (dev->no_flr_reset)
> return -ENOTTY;
>
> pci_read_config_byte(dev, pos + PCI_AF_CAP, &cap);
> @@ -4628,7 +4628,7 @@ static int pci_pm_reset(struct pci_dev *dev, int probe)
> {
> u16 csr;
>
> - if (!dev->pm_cap || dev->dev_flags & PCI_DEV_FLAGS_NO_PM_RESET)
> + if (!dev->pm_cap || dev->no_pm_reset)
> return -ENOTTY;
>
> pci_read_config_word(dev, dev->pm_cap + PCI_PM_CTRL, &csr);
> @@ -4890,7 +4890,7 @@ static int pci_parent_bus_reset(struct pci_dev *dev, int probe)
> struct pci_dev *pdev;
>
> if (pci_is_root_bus(dev->bus) || dev->subordinate ||
> - !dev->bus->self || dev->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET)
> + !dev->bus->self || dev->no_bus_reset)
> return -ENOTTY;
>
> list_for_each_entry(pdev, &dev->bus->devices, bus_list)
> @@ -4922,8 +4922,7 @@ static int pci_dev_reset_slot_function(struct pci_dev *dev, int probe)
> {
> struct pci_dev *pdev;
>
> - if (dev->subordinate || !dev->slot ||
> - dev->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET)
> + if (dev->subordinate || !dev->slot || dev->no_bus_reset)
> return -ENOTTY;
>
> list_for_each_entry(pdev, &dev->bus->devices, bus_list)
> @@ -5195,11 +5194,11 @@ static bool pci_bus_resetable(struct pci_bus *bus)
> struct pci_dev *dev;
>
>
> - if (bus->self && (bus->self->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET))
> + if (bus->self && bus->self->no_bus_reset)
> return false;
>
> list_for_each_entry(dev, &bus->devices, bus_list) {
> - if (dev->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET ||
> + if (dev->no_bus_reset ||
> (dev->subordinate && !pci_bus_resetable(dev->subordinate)))
> return false;
> }
> @@ -5262,14 +5261,13 @@ static bool pci_slot_resetable(struct pci_slot *slot)
> {
> struct pci_dev *dev;
>
> - if (slot->bus->self &&
> - (slot->bus->self->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET))
> + if (slot->bus->self && slot->bus->self->no_bus_reset)
> return false;
>
> list_for_each_entry(dev, &slot->bus->devices, bus_list) {
> if (!dev->slot || dev->slot != slot)
> continue;
> - if (dev->dev_flags & PCI_DEV_FLAGS_NO_BUS_RESET ||
> + if (dev->no_bus_reset ||
> (dev->subordinate && !pci_bus_resetable(dev->subordinate)))
> return false;
> }
> diff --git a/drivers/pci/probe.c b/drivers/pci/probe.c
> index 03d37128a24f..439cd35fe8f9 100644
> --- a/drivers/pci/probe.c
> +++ b/drivers/pci/probe.c
> @@ -2089,7 +2089,7 @@ static void pci_configure_relaxed_ordering(struct pci_dev *dev)
> if (!root)
> return;
>
> - if (root->dev_flags & PCI_DEV_FLAGS_NO_RELAXED_ORDERING) {
> + if (root->no_relaxed_ordering) {
> pcie_capability_clear_word(dev, PCI_EXP_DEVCTL,
> PCI_EXP_DEVCTL_RELAX_EN);
> pci_info(dev, "Relaxed Ordering disabled because the Root Port didn't support it\n");
> diff --git a/drivers/pci/quirks.c b/drivers/pci/quirks.c
> index 2a589b6d6ed8..e23290619683 100644
> --- a/drivers/pci/quirks.c
> +++ b/drivers/pci/quirks.c
> @@ -1341,7 +1341,7 @@ DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82801CA_10, qui
> /* Some ATA devices break if put into D3 */
> static void quirk_no_ata_d3(struct pci_dev *pdev)
> {
> - pdev->dev_flags |= PCI_DEV_FLAGS_NO_D3;
> + pdev->no_d3 = 1;
> }
> /* Quirk the legacy ATA devices only. The AHCI ones are ok */
> DECLARE_PCI_FIXUP_CLASS_EARLY(PCI_VENDOR_ID_SERVERWORKS, PCI_ANY_ID,
> @@ -2962,7 +2962,7 @@ DECLARE_PCI_FIXUP_RESUME_EARLY(PCI_VENDOR_ID_NVIDIA, PCI_ANY_ID, nv_msi_ht_cap_q
>
> static void quirk_msi_intx_disable_bug(struct pci_dev *dev)
> {
> - dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
> + dev->msi_intx_disabled = 1;
> }
>
> static void quirk_msi_intx_disable_ati_bug(struct pci_dev *dev)
> @@ -2980,7 +2980,7 @@ static void quirk_msi_intx_disable_ati_bug(struct pci_dev *dev)
> return;
>
> if ((p->revision < 0x3B) && (p->revision >= 0x30))
> - dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
> + dev->msi_intx_disabled = 1;
> pci_dev_put(p);
> }
>
> @@ -2989,7 +2989,7 @@ static void quirk_msi_intx_disable_qca_bug(struct pci_dev *dev)
> /* AR816X/AR817X/E210X MSI is fixed at HW level from revision 0x18 */
> if (dev->revision < 0x18) {
> pci_info(dev, "set MSI_INTX_DISABLE_BUG flag\n");
> - dev->dev_flags |= PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG;
> + dev->msi_intx_disabled = 1;
> }
> }
> DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_BROADCOM,
> @@ -3554,7 +3554,7 @@ DECLARE_PCI_FIXUP_FINAL(PCI_VENDOR_ID_MELLANOX, PCI_ANY_ID,
>
> static void quirk_no_bus_reset(struct pci_dev *dev)
> {
> - dev->dev_flags |= PCI_DEV_FLAGS_NO_BUS_RESET;
> + dev->no_bus_reset = 1;
> }
>
> /*
> @@ -3584,7 +3584,7 @@ static void quirk_no_pm_reset(struct pci_dev *dev)
> * PM reset may be better than nothing.
> */
> if (!pci_is_root_bus(dev->bus))
> - dev->dev_flags |= PCI_DEV_FLAGS_NO_PM_RESET;
> + dev->no_pm_reset = 1;
> }
>
> /*
> @@ -4124,7 +4124,7 @@ static void quirk_use_pcie_bridge_dma_alias(struct pci_dev *pdev)
> pdev->hdr_type == PCI_HEADER_TYPE_BRIDGE &&
> !pci_is_pcie(pdev) && pci_is_pcie(pdev->bus->self) &&
> pci_pcie_type(pdev->bus->self) != PCI_EXP_TYPE_PCI_BRIDGE)
> - pdev->dev_flags |= PCI_DEV_FLAG_PCIE_BRIDGE_ALIAS;
> + pdev->pcie_bridge_alias = 1;
> }
> /* ASM1083/1085, https://bugzilla.kernel.org/show_bug.cgi?id=44881#c46 */
> DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_ASMEDIA, 0x1080,
> @@ -4189,7 +4189,7 @@ DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, 0x295A, quirk_pex_vca_alias);
> */
> static void quirk_bridge_cavm_thrx2_pcie_root(struct pci_dev *pdev)
> {
> - pdev->dev_flags |= PCI_DEV_FLAGS_BRIDGE_XLATE_ROOT;
> + pdev->bridge_xlate_root = 1;
> }
> DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_BROADCOM, 0x9000,
> quirk_bridge_cavm_thrx2_pcie_root);
> @@ -4225,7 +4225,7 @@ DECLARE_PCI_FIXUP_CLASS_EARLY(0x1797, 0x6869, PCI_CLASS_NOT_DEFINED, 8,
> */
> static void quirk_relaxedordering_disable(struct pci_dev *dev)
> {
> - dev->dev_flags |= PCI_DEV_FLAGS_NO_RELAXED_ORDERING;
> + dev->no_relaxed_ordering = 1;
> pci_info(dev, "Disable Relaxed Ordering Attributes to avoid PCIe Completion erratum\n");
> }
>
> @@ -4557,7 +4557,7 @@ static int pci_quirk_intel_pch_acs(struct pci_dev *dev, u16 acs_flags)
> if (!pci_quirk_intel_pch_acs_match(dev))
> return -ENOTTY;
>
> - if (dev->dev_flags & PCI_DEV_FLAGS_ACS_ENABLED_QUIRK)
> + if (dev->acs_quirk_enabled)
> return pci_acs_ctrl_enabled(acs_flags,
> PCI_ACS_SV | PCI_ACS_RR | PCI_ACS_CR | PCI_ACS_UF);
>
> @@ -4961,7 +4961,7 @@ static int pci_quirk_enable_intel_pch_acs(struct pci_dev *dev)
>
> pci_quirk_enable_intel_rp_mpc_acs(dev);
>
> - dev->dev_flags |= PCI_DEV_FLAGS_ACS_ENABLED_QUIRK;
> + dev->acs_quirk_enabled = 1;
>
> pci_info(dev, "Intel PCH root port ACS workaround enabled\n");
>
> @@ -5171,7 +5171,7 @@ DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_INTEL, 0x443, quirk_intel_qat_vf_cap);
> */
> static void quirk_no_flr(struct pci_dev *dev)
> {
> - dev->dev_flags |= PCI_DEV_FLAGS_NO_FLR_RESET;
> + dev->no_flr_reset = 1;
> }
> DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_AMD, 0x1487, quirk_no_flr);
> DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_AMD, 0x148c, quirk_no_flr);
> diff --git a/drivers/pci/search.c b/drivers/pci/search.c
> index 2061672954ee..b26b12e2cc3f 100644
> --- a/drivers/pci/search.c
> +++ b/drivers/pci/search.c
> @@ -67,7 +67,7 @@ int pci_for_each_dma_alias(struct pci_dev *pdev,
> tmp = bus->self;
>
> /* stop at bridge where translation unit is associated */
> - if (tmp->dev_flags & PCI_DEV_FLAGS_BRIDGE_XLATE_ROOT)
> + if (tmp->bridge_xlate_root)
> return ret;
>
> /*
> @@ -99,7 +99,7 @@ int pci_for_each_dma_alias(struct pci_dev *pdev,
> continue;
> }
> } else {
> - if (tmp->dev_flags & PCI_DEV_FLAG_PCIE_BRIDGE_ALIAS)
> + if (tmp->pcie_bridge_alias)
> ret = fn(tmp,
> PCI_DEVID(tmp->subordinate->number,
> PCI_DEVFN(0, 0)), data);
> diff --git a/drivers/pci/vpd.c b/drivers/pci/vpd.c
> index 7915d10f9aa1..5c4366362bd7 100644
> --- a/drivers/pci/vpd.c
> +++ b/drivers/pci/vpd.c
> @@ -380,7 +380,7 @@ int pci_vpd_init(struct pci_dev *dev)
> return -ENOMEM;
>
> vpd->len = PCI_VPD_MAX_SIZE;
> - if (dev->dev_flags & PCI_DEV_FLAGS_VPD_REF_F0)
> + if (dev->pci_vpd_f0)
> vpd->ops = &pci_vpd_f0_ops;
> else
> vpd->ops = &pci_vpd_ops;
> @@ -536,7 +536,7 @@ static void quirk_f0_vpd_link(struct pci_dev *dev)
>
> if (f0->vpd && dev->class == f0->class &&
> dev->vendor == f0->vendor && dev->device == f0->device)
> - dev->dev_flags |= PCI_DEV_FLAGS_VPD_REF_F0;
> + dev->pci_vpd_f0 = 1;
>
> pci_dev_put(f0);
> }
> diff --git a/include/linux/pci.h b/include/linux/pci.h
> index 835530605c0d..09b15f41a0ab 100644
> --- a/include/linux/pci.h
> +++ b/include/linux/pci.h
> @@ -203,32 +203,6 @@ enum pcie_reset_state {
> pcie_hot_reset = (__force pcie_reset_state_t) 3
> };
>
> -typedef unsigned short __bitwise pci_dev_flags_t;
> -enum pci_dev_flags {
> - /* INTX_DISABLE in PCI_COMMAND register disables MSI too */
> - PCI_DEV_FLAGS_MSI_INTX_DISABLE_BUG = (__force pci_dev_flags_t) (1 << 0),
> - /* Device configuration is irrevocably lost if disabled into D3 */
> - PCI_DEV_FLAGS_NO_D3 = (__force pci_dev_flags_t) (1 << 1),
> - /* Provide indication device is assigned by a Virtual Machine Manager */
> - PCI_DEV_FLAGS_ASSIGNED = (__force pci_dev_flags_t) (1 << 2),
> - /* Flag for quirk use to store if quirk-specific ACS is enabled */
> - PCI_DEV_FLAGS_ACS_ENABLED_QUIRK = (__force pci_dev_flags_t) (1 << 3),
> - /* Use a PCIe-to-PCI bridge alias even if !pci_is_pcie */
> - PCI_DEV_FLAG_PCIE_BRIDGE_ALIAS = (__force pci_dev_flags_t) (1 << 5),
> - /* Do not use bus resets for device */
> - PCI_DEV_FLAGS_NO_BUS_RESET = (__force pci_dev_flags_t) (1 << 6),
> - /* Do not use PM reset even if device advertises NoSoftRst- */
> - PCI_DEV_FLAGS_NO_PM_RESET = (__force pci_dev_flags_t) (1 << 7),
> - /* Get VPD from function 0 VPD */
> - PCI_DEV_FLAGS_VPD_REF_F0 = (__force pci_dev_flags_t) (1 << 8),
> - /* A non-root bridge where translation occurs, stop alias search here */
> - PCI_DEV_FLAGS_BRIDGE_XLATE_ROOT = (__force pci_dev_flags_t) (1 << 9),
> - /* Do not use FLR even if device advertises PCI_AF_CAP */
> - PCI_DEV_FLAGS_NO_FLR_RESET = (__force pci_dev_flags_t) (1 << 10),
> - /* Don't use Relaxed Ordering for TLPs directed at this device */
> - PCI_DEV_FLAGS_NO_RELAXED_ORDERING = (__force pci_dev_flags_t) (1 << 11),
> -};
> -
> enum pci_irq_reroute_variant {
> INTEL_IRQ_REROUTE_VARIANT = 1,
> MAX_IRQ_REROUTE_VARIANTS = 3
> @@ -445,7 +419,20 @@ struct pci_dev {
> unsigned int is_probed:1; /* Device probing in progress */
> unsigned int link_active_reporting:1;/* Device capable of reporting link active */
> unsigned int no_vf_scan:1; /* Don't scan for VFs after IOV enablement */
> - pci_dev_flags_t dev_flags;
> +
> + /* PCI device flags */
> + unsigned int flags_assigned:1; /* Provide indication device is assigned by a Virtual Machine Manager */
> + unsigned int msi_intx_disabled:1; /* INTX_DISABLE in PCI_COMMAND register disables MSI too */
> + unsigned int acs_quirk_enabled:1; /* Flag for quirk use to store if quirk-specific ACS is enabled */
> + unsigned int no_d3:1; /* Device configuration is irrevocably lost if disabled into D3 */
> + unsigned int no_bus_reset:1; /* Don't use bus resets for device */
> + unsigned int no_pm_reset:1; /* Don't use PM reset even if device advertises NoSoftRst- */
> + unsigned int no_flr_reset:1; /* Don't use FLR even if device advertises PCI_AF_CAP */
> + unsigned int no_relaxed_ordering:1; /* Don't use Relaxed Ordering for TLPs directed at this device */
> + unsigned int pcie_bridge_alias:1; /* Use a PCIe-to-PCI bridge alias even if !pci_is_pcie */
> + unsigned int bridge_xlate_root:1; /* A non-root bridge where translation occurs, stop alias search here */
> + unsigned int pci_vpd_f0:1; /* Get VPD from function 0 VPD */
> +
> atomic_t enable_cnt; /* pci_enable_device has been called */
>
> u32 saved_config_space[16]; /* Config space saved at suspend time */
> @@ -2350,15 +2337,15 @@ int pci_for_each_dma_alias(struct pci_dev *pdev,
> /* Helper functions for operation of device flag */
> static inline void pci_set_dev_assigned(struct pci_dev *pdev)
> {
> - pdev->dev_flags |= PCI_DEV_FLAGS_ASSIGNED;
> + pdev->flags_assigned = 1;
> }
> static inline void pci_clear_dev_assigned(struct pci_dev *pdev)
> {
> - pdev->dev_flags &= ~PCI_DEV_FLAGS_ASSIGNED;
> + pdev->flags_assigned = 0;
> }
> static inline bool pci_is_dev_assigned(struct pci_dev *pdev)
> {
> - return (pdev->dev_flags & PCI_DEV_FLAGS_ASSIGNED) == PCI_DEV_FLAGS_ASSIGNED;
> + return !!pdev->flags_assigned;
> }
>
> /**
> --
> 2.28.0
>
Powered by blists - more mailing lists