Message ID | 20250402042020.48681-3-18255117159@163.com (mailing list archive) |
---|---|
State | Superseded |
Headers | show |
Series | Refactor capability search into common macros | expand |
Hi Hans,
kernel test robot noticed the following build errors:
[auto build test ERROR on acb4f33713b9f6cadb6143f211714c343465411c]
url: https://github.com/intel-lab-lkp/linux/commits/Hans-Zhang/PCI-Refactor-capability-search-into-common-macros/20250402-122544
base: acb4f33713b9f6cadb6143f211714c343465411c
patch link: https://lore.kernel.org/r/20250402042020.48681-3-18255117159%40163.com
patch subject: [v7 2/5] PCI: Refactor capability search functions to eliminate code duplication
config: loongarch-randconfig-001-20250402 (https://download.01.org/0day-ci/archive/20250402/202504021623.LgnqoZPE-lkp@intel.com/config)
compiler: loongarch64-linux-gcc (GCC) 14.2.0
reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20250402/202504021623.LgnqoZPE-lkp@intel.com/reproduce)
If you fix the issue in a separate patch/commit (i.e. not just a new version of
the same patch/commit), kindly add following tags
| Reported-by: kernel test robot <lkp@intel.com>
| Closes: https://lore.kernel.org/oe-kbuild-all/202504021623.LgnqoZPE-lkp@intel.com/
All errors (new ones prefixed by >>):
drivers/pci/pci.c: In function '__pci_find_next_ht_cap':
>> drivers/pci/pci.c:627:17: error: 'rc' undeclared (first use in this function); did you mean 'rq'?
627 | rc = pci_read_config_byte(dev, pos + 3, &cap);
| ^~
| rq
drivers/pci/pci.c:627:17: note: each undeclared identifier is reported only once for each function it appears in
vim +627 drivers/pci/pci.c
70c0923b0ef10b Jacob Keller 2020-03-02 614
f646c2a0a6685a Puranjay Mohan 2020-11-29 615 static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap)
687d5fe3dc3379 Michael Ellerman 2006-11-22 616 {
687d5fe3dc3379 Michael Ellerman 2006-11-22 617 u8 cap, mask;
687d5fe3dc3379 Michael Ellerman 2006-11-22 618
687d5fe3dc3379 Michael Ellerman 2006-11-22 619 if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST)
687d5fe3dc3379 Michael Ellerman 2006-11-22 620 mask = HT_3BIT_CAP_MASK;
687d5fe3dc3379 Michael Ellerman 2006-11-22 621 else
687d5fe3dc3379 Michael Ellerman 2006-11-22 622 mask = HT_5BIT_CAP_MASK;
687d5fe3dc3379 Michael Ellerman 2006-11-22 623
687d5fe3dc3379 Michael Ellerman 2006-11-22 624 pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos,
2675dfd086c109 Hans Zhang 2025-04-02 625 PCI_CAP_ID_HT);
687d5fe3dc3379 Michael Ellerman 2006-11-22 626 while (pos) {
687d5fe3dc3379 Michael Ellerman 2006-11-22 @627 rc = pci_read_config_byte(dev, pos + 3, &cap);
687d5fe3dc3379 Michael Ellerman 2006-11-22 628 if (rc != PCIBIOS_SUCCESSFUL)
687d5fe3dc3379 Michael Ellerman 2006-11-22 629 return 0;
687d5fe3dc3379 Michael Ellerman 2006-11-22 630
687d5fe3dc3379 Michael Ellerman 2006-11-22 631 if ((cap & mask) == ht_cap)
687d5fe3dc3379 Michael Ellerman 2006-11-22 632 return pos;
687d5fe3dc3379 Michael Ellerman 2006-11-22 633
47a4d5be7c50b2 Brice Goglin 2007-01-10 634 pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn,
47a4d5be7c50b2 Brice Goglin 2007-01-10 635 pos + PCI_CAP_LIST_NEXT,
2675dfd086c109 Hans Zhang 2025-04-02 636 PCI_CAP_ID_HT);
687d5fe3dc3379 Michael Ellerman 2006-11-22 637 }
687d5fe3dc3379 Michael Ellerman 2006-11-22 638
687d5fe3dc3379 Michael Ellerman 2006-11-22 639 return 0;
687d5fe3dc3379 Michael Ellerman 2006-11-22 640 }
f646c2a0a6685a Puranjay Mohan 2020-11-29 641
On 2025/4/2 17:19, kernel test robot wrote: > Hi Hans, > > kernel test robot noticed the following build errors: > > [auto build test ERROR on acb4f33713b9f6cadb6143f211714c343465411c] > > url: https://github.com/intel-lab-lkp/linux/commits/Hans-Zhang/PCI-Refactor-capability-search-into-common-macros/20250402-122544 > base: acb4f33713b9f6cadb6143f211714c343465411c > patch link: https://lore.kernel.org/r/20250402042020.48681-3-18255117159%40163.com > patch subject: [v7 2/5] PCI: Refactor capability search functions to eliminate code duplication > config: loongarch-randconfig-001-20250402 (https://download.01.org/0day-ci/archive/20250402/202504021623.LgnqoZPE-lkp@intel.com/config) > compiler: loongarch64-linux-gcc (GCC) 14.2.0 > reproduce (this is a W=1 build): (https://download.01.org/0day-ci/archive/20250402/202504021623.LgnqoZPE-lkp@intel.com/reproduce) > > If you fix the issue in a separate patch/commit (i.e. not just a new version of > the same patch/commit), kindly add following tags > | Reported-by: kernel test robot <lkp@intel.com> > | Closes: https://lore.kernel.org/oe-kbuild-all/202504021623.LgnqoZPE-lkp@intel.com/ > > All errors (new ones prefixed by >>): > > drivers/pci/pci.c: In function '__pci_find_next_ht_cap': >>> drivers/pci/pci.c:627:17: error: 'rc' undeclared (first use in this function); did you mean 'rq'? > 627 | rc = pci_read_config_byte(dev, pos + 3, &cap); > | ^~ > | rq > drivers/pci/pci.c:627:17: note: each undeclared identifier is reported only once for each function it appears in > static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) { - int rc, ttl = PCI_FIND_CAP_TTL; u8 cap, mask; The rc variable was deleted by mistake and will be changed in the next version. static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) { - int rc; u8 cap, mask; Best regards, Hans > > vim +627 drivers/pci/pci.c > > 70c0923b0ef10b Jacob Keller 2020-03-02 614 > f646c2a0a6685a Puranjay Mohan 2020-11-29 615 static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) > 687d5fe3dc3379 Michael Ellerman 2006-11-22 616 { > 687d5fe3dc3379 Michael Ellerman 2006-11-22 617 u8 cap, mask; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 618 > 687d5fe3dc3379 Michael Ellerman 2006-11-22 619 if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) > 687d5fe3dc3379 Michael Ellerman 2006-11-22 620 mask = HT_3BIT_CAP_MASK; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 621 else > 687d5fe3dc3379 Michael Ellerman 2006-11-22 622 mask = HT_5BIT_CAP_MASK; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 623 > 687d5fe3dc3379 Michael Ellerman 2006-11-22 624 pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, > 2675dfd086c109 Hans Zhang 2025-04-02 625 PCI_CAP_ID_HT); > 687d5fe3dc3379 Michael Ellerman 2006-11-22 626 while (pos) { > 687d5fe3dc3379 Michael Ellerman 2006-11-22 @627 rc = pci_read_config_byte(dev, pos + 3, &cap); > 687d5fe3dc3379 Michael Ellerman 2006-11-22 628 if (rc != PCIBIOS_SUCCESSFUL) > 687d5fe3dc3379 Michael Ellerman 2006-11-22 629 return 0; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 630 > 687d5fe3dc3379 Michael Ellerman 2006-11-22 631 if ((cap & mask) == ht_cap) > 687d5fe3dc3379 Michael Ellerman 2006-11-22 632 return pos; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 633 > 47a4d5be7c50b2 Brice Goglin 2007-01-10 634 pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, > 47a4d5be7c50b2 Brice Goglin 2007-01-10 635 pos + PCI_CAP_LIST_NEXT, > 2675dfd086c109 Hans Zhang 2025-04-02 636 PCI_CAP_ID_HT); > 687d5fe3dc3379 Michael Ellerman 2006-11-22 637 } > 687d5fe3dc3379 Michael Ellerman 2006-11-22 638 > 687d5fe3dc3379 Michael Ellerman 2006-11-22 639 return 0; > 687d5fe3dc3379 Michael Ellerman 2006-11-22 640 } > f646c2a0a6685a Puranjay Mohan 2020-11-29 641 >
On Wed, 2 Apr 2025, Hans Zhang wrote: > Refactor the PCI capability and extended capability search functions > by consolidating common code patterns into reusable macros > (PCI_FIND_NEXT_CAP_TTL and PCI_FIND_NEXT_EXT_CAPABILITY). The main > changes include: > > 1. Introducing a unified config space read helper (__pci_bus_read_config). > 2. Removing duplicate search logic from __pci_find_next_cap_ttl and > pci_find_next_ext_capability. > 3. Implementing consistent capability discovery using the new macros. > 4. Simplifying HyperTransport capability lookup by leveraging the > refactored code. > > The refactoring maintains existing functionality while reducing code > duplication and improving maintainability. By centralizing the search > logic, we achieve better code consistency and make future updates easier. > > This change has been verified to maintain backward compatibility with > existing capability discovery patterns through thorough testing of PCI > device enumeration and capability probing. > > Signed-off-by: Hans Zhang <18255117159@163.com> > --- > drivers/pci/pci.c | 79 +++++++++++++---------------------------------- > 1 file changed, 22 insertions(+), 57 deletions(-) > > diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c > index 869d204a70a3..521096c73686 100644 > --- a/drivers/pci/pci.c > +++ b/drivers/pci/pci.c > @@ -423,36 +423,33 @@ static int pci_dev_str_match(struct pci_dev *dev, const char *p, > return 1; > } > > -static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, > - u8 pos, int cap, int *ttl) > +static int __pci_bus_read_config(void *priv, unsigned int devfn, int where, > + u32 size, u32 *val) This probably should be where the other accessors are so in access.c. I'd put its prototype into drivers/pci/pci.h only for now. > { > - u8 id; > - u16 ent; > + struct pci_bus *bus = priv; > + int ret; > > - pci_bus_read_config_byte(bus, devfn, pos, &pos); > + if (size == 1) > + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); > + else if (size == 2) > + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); > + else > + ret = pci_bus_read_config_dword(bus, devfn, where, val); > > - while ((*ttl)--) { > - if (pos < 0x40) > - break; > - pos &= ~3; > - pci_bus_read_config_word(bus, devfn, pos, &ent); > + return ret; > +} > > - id = ent & 0xff; > - if (id == 0xff) > - break; > - if (id == cap) > - return pos; > - pos = (ent >> 8); > - } > - return 0; > +static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, > + u8 pos, int cap) > +{ > + return PCI_FIND_NEXT_CAP_TTL(__pci_bus_read_config, pos, cap, bus, > + devfn); > } > > static u8 __pci_find_next_cap(struct pci_bus *bus, unsigned int devfn, > u8 pos, int cap) > { > - int ttl = PCI_FIND_CAP_TTL; > - > - return __pci_find_next_cap_ttl(bus, devfn, pos, cap, &ttl); > + return __pci_find_next_cap_ttl(bus, devfn, pos, cap); > } > > u8 pci_find_next_capability(struct pci_dev *dev, u8 pos, int cap) > @@ -553,42 +550,11 @@ EXPORT_SYMBOL(pci_bus_find_capability); > */ > u16 pci_find_next_ext_capability(struct pci_dev *dev, u16 start, int cap) > { > - u32 header; > - int ttl; > - u16 pos = PCI_CFG_SPACE_SIZE; > - > - /* minimum 8 bytes per capability */ > - ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; > - > if (dev->cfg_size <= PCI_CFG_SPACE_SIZE) > return 0; > > - if (start) > - pos = start; > - > - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) > - return 0; > - > - /* > - * If we have no capabilities, this is indicated by cap ID, > - * cap version and next pointer all being 0. > - */ > - if (header == 0) > - return 0; > - > - while (ttl-- > 0) { > - if (PCI_EXT_CAP_ID(header) == cap && pos != start) > - return pos; > - > - pos = PCI_EXT_CAP_NEXT(header); > - if (pos < PCI_CFG_SPACE_SIZE) > - break; > - > - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) > - break; > - } > - > - return 0; > + return PCI_FIND_NEXT_EXT_CAPABILITY(__pci_bus_read_config, start, cap, > + dev->bus, dev->devfn); I don't like how 1 & 2 patches are split into two. IMO, they mostly belong together. However, (IMO) you can introduce the new all-size config space accessor in a separate patch before the combined patch. > } > EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); > > @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); > > static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) > { > - int rc, ttl = PCI_FIND_CAP_TTL; > u8 cap, mask; > > if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) > @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) > mask = HT_5BIT_CAP_MASK; > > pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, > - PCI_CAP_ID_HT, &ttl); > + PCI_CAP_ID_HT); > while (pos) { > rc = pci_read_config_byte(dev, pos + 3, &cap); > if (rc != PCIBIOS_SUCCESSFUL) > @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) > > pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, > pos + PCI_CAP_LIST_NEXT, > - PCI_CAP_ID_HT, &ttl); > + PCI_CAP_ID_HT); This function kind of had the idea to share the ttl but I suppose that was just a final safeguard to make sure the loop will always terminate in case the config space is corrupted so the unsharing is not a big issue. > } > > return 0; >
On 2025/4/2 20:38, Ilpo Järvinen wrote: > On Wed, 2 Apr 2025, Hans Zhang wrote: > >> Refactor the PCI capability and extended capability search functions >> by consolidating common code patterns into reusable macros >> (PCI_FIND_NEXT_CAP_TTL and PCI_FIND_NEXT_EXT_CAPABILITY). The main >> changes include: >> >> 1. Introducing a unified config space read helper (__pci_bus_read_config). >> 2. Removing duplicate search logic from __pci_find_next_cap_ttl and >> pci_find_next_ext_capability. >> 3. Implementing consistent capability discovery using the new macros. >> 4. Simplifying HyperTransport capability lookup by leveraging the >> refactored code. >> >> The refactoring maintains existing functionality while reducing code >> duplication and improving maintainability. By centralizing the search >> logic, we achieve better code consistency and make future updates easier. >> >> This change has been verified to maintain backward compatibility with >> existing capability discovery patterns through thorough testing of PCI >> device enumeration and capability probing. >> >> Signed-off-by: Hans Zhang <18255117159@163.com> >> --- >> drivers/pci/pci.c | 79 +++++++++++++---------------------------------- >> 1 file changed, 22 insertions(+), 57 deletions(-) >> >> diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c >> index 869d204a70a3..521096c73686 100644 >> --- a/drivers/pci/pci.c >> +++ b/drivers/pci/pci.c >> @@ -423,36 +423,33 @@ static int pci_dev_str_match(struct pci_dev *dev, const char *p, >> return 1; >> } >> >> -static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, >> - u8 pos, int cap, int *ttl) >> +static int __pci_bus_read_config(void *priv, unsigned int devfn, int where, >> + u32 size, u32 *val) > > This probably should be where the other accessors are so in access.c. I'd > put its prototype into drivers/pci/pci.h only for now. > Hi Ilpo, Thank you very much for your guidance. Will change. >> { >> - u8 id; >> - u16 ent; >> + struct pci_bus *bus = priv; >> + int ret; >> >> - pci_bus_read_config_byte(bus, devfn, pos, &pos); >> + if (size == 1) >> + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); >> + else if (size == 2) >> + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); >> + else >> + ret = pci_bus_read_config_dword(bus, devfn, where, val); >> >> - while ((*ttl)--) { >> - if (pos < 0x40) >> - break; >> - pos &= ~3; >> - pci_bus_read_config_word(bus, devfn, pos, &ent); >> + return ret; >> +} >> >> - id = ent & 0xff; >> - if (id == 0xff) >> - break; >> - if (id == cap) >> - return pos; >> - pos = (ent >> 8); >> - } >> - return 0; >> +static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, >> + u8 pos, int cap) >> +{ >> + return PCI_FIND_NEXT_CAP_TTL(__pci_bus_read_config, pos, cap, bus, >> + devfn); >> } >> >> static u8 __pci_find_next_cap(struct pci_bus *bus, unsigned int devfn, >> u8 pos, int cap) >> { >> - int ttl = PCI_FIND_CAP_TTL; >> - >> - return __pci_find_next_cap_ttl(bus, devfn, pos, cap, &ttl); >> + return __pci_find_next_cap_ttl(bus, devfn, pos, cap); >> } >> >> u8 pci_find_next_capability(struct pci_dev *dev, u8 pos, int cap) >> @@ -553,42 +550,11 @@ EXPORT_SYMBOL(pci_bus_find_capability); >> */ >> u16 pci_find_next_ext_capability(struct pci_dev *dev, u16 start, int cap) >> { >> - u32 header; >> - int ttl; >> - u16 pos = PCI_CFG_SPACE_SIZE; >> - >> - /* minimum 8 bytes per capability */ >> - ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; >> - >> if (dev->cfg_size <= PCI_CFG_SPACE_SIZE) >> return 0; >> >> - if (start) >> - pos = start; >> - >> - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) >> - return 0; >> - >> - /* >> - * If we have no capabilities, this is indicated by cap ID, >> - * cap version and next pointer all being 0. >> - */ >> - if (header == 0) >> - return 0; >> - >> - while (ttl-- > 0) { >> - if (PCI_EXT_CAP_ID(header) == cap && pos != start) >> - return pos; >> - >> - pos = PCI_EXT_CAP_NEXT(header); >> - if (pos < PCI_CFG_SPACE_SIZE) >> - break; >> - >> - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) >> - break; >> - } >> - >> - return 0; >> + return PCI_FIND_NEXT_EXT_CAPABILITY(__pci_bus_read_config, start, cap, >> + dev->bus, dev->devfn); > > I don't like how 1 & 2 patches are split into two. IMO, they mostly belong > together. However, (IMO) you can introduce the new all-size config space > accessor in a separate patch before the combined patch. > Ok. I'll change it to the following. The rest I'll combine into a patch. diff --git a/drivers/pci/access.c b/drivers/pci/access.c index b123da16b63b..bb2e26c2eb81 100644 --- a/drivers/pci/access.c +++ b/drivers/pci/access.c @@ -85,6 +85,23 @@ EXPORT_SYMBOL(pci_bus_write_config_byte); EXPORT_SYMBOL(pci_bus_write_config_word); EXPORT_SYMBOL(pci_bus_write_config_dword); + +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, + u32 *val) +{ + struct pci_bus *bus = priv; + int ret; + + if (size == 1) + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); + else if (size == 2) + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); + else + ret = pci_bus_read_config_dword(bus, devfn, where, val); + + return ret; +} + int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn, int where, int size, u32 *val) { diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 2e9cf26a9ee9..6a7c88b9cd35 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h @@ -88,6 +88,8 @@ extern bool pci_early_dump; bool pcie_cap_has_lnkctl(const struct pci_dev *dev); bool pcie_cap_has_lnkctl2(const struct pci_dev *dev); bool pcie_cap_has_rtctl(const struct pci_dev *dev); +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, + u32 *val); /* Functions internal to the PCI core code */ >> } >> EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); >> >> @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); >> >> static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) >> { >> - int rc, ttl = PCI_FIND_CAP_TTL; >> u8 cap, mask; >> >> if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) >> @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) >> mask = HT_5BIT_CAP_MASK; >> >> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, >> - PCI_CAP_ID_HT, &ttl); >> + PCI_CAP_ID_HT); >> while (pos) { >> rc = pci_read_config_byte(dev, pos + 3, &cap); >> if (rc != PCIBIOS_SUCCESSFUL) >> @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) >> >> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, >> pos + PCI_CAP_LIST_NEXT, >> - PCI_CAP_ID_HT, &ttl); >> + PCI_CAP_ID_HT); > > This function kind of had the idea to share the ttl but I suppose that was > just a final safeguard to make sure the loop will always terminate in case > the config space is corrupted so the unsharing is not a big issue. > __pci_find_next_cap_ttl // This macro definition already has ttl loop restrictions inside it. PCI_FIND_NEXT_CAP_TTL Do I understand that you agree to remove ttl initialization and parameter passing? Best regards, Hans >> } >> >> return 0; >> >
On Wed, 2 Apr 2025, Hans Zhang wrote: > On 2025/4/2 20:38, Ilpo Järvinen wrote: > > On Wed, 2 Apr 2025, Hans Zhang wrote: > > > > > Refactor the PCI capability and extended capability search functions > > > by consolidating common code patterns into reusable macros > > > (PCI_FIND_NEXT_CAP_TTL and PCI_FIND_NEXT_EXT_CAPABILITY). The main > > > changes include: > > > > > > 1. Introducing a unified config space read helper (__pci_bus_read_config). > > > 2. Removing duplicate search logic from __pci_find_next_cap_ttl and > > > pci_find_next_ext_capability. > > > 3. Implementing consistent capability discovery using the new macros. > > > 4. Simplifying HyperTransport capability lookup by leveraging the > > > refactored code. > > > > > > The refactoring maintains existing functionality while reducing code > > > duplication and improving maintainability. By centralizing the search > > > logic, we achieve better code consistency and make future updates easier. > > > > > > This change has been verified to maintain backward compatibility with > > > existing capability discovery patterns through thorough testing of PCI > > > device enumeration and capability probing. > > > > > > Signed-off-by: Hans Zhang <18255117159@163.com> > > > --- > > > drivers/pci/pci.c | 79 +++++++++++++---------------------------------- > > > 1 file changed, 22 insertions(+), 57 deletions(-) > > > > > > diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c > > > index 869d204a70a3..521096c73686 100644 > > > --- a/drivers/pci/pci.c > > > +++ b/drivers/pci/pci.c > > > @@ -423,36 +423,33 @@ static int pci_dev_str_match(struct pci_dev *dev, > > > const char *p, > > > return 1; > > > } > > > -static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int > > > devfn, > > > - u8 pos, int cap, int *ttl) > > > +static int __pci_bus_read_config(void *priv, unsigned int devfn, int > > > where, > > > + u32 size, u32 *val) > > > > This probably should be where the other accessors are so in access.c. I'd > > put its prototype into drivers/pci/pci.h only for now. > > > > Hi Ilpo, > > Thank you very much for your guidance. Will change. > > > > > { > > > - u8 id; > > > - u16 ent; > > > + struct pci_bus *bus = priv; > > > + int ret; > > > - pci_bus_read_config_byte(bus, devfn, pos, &pos); > > > + if (size == 1) > > > + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); > > > + else if (size == 2) > > > + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); > > > + else > > > + ret = pci_bus_read_config_dword(bus, devfn, where, val); > > > - while ((*ttl)--) { > > > - if (pos < 0x40) > > > - break; > > > - pos &= ~3; > > > - pci_bus_read_config_word(bus, devfn, pos, &ent); > > > + return ret; > > > +} > > > - id = ent & 0xff; > > > - if (id == 0xff) > > > - break; > > > - if (id == cap) > > > - return pos; > > > - pos = (ent >> 8); > > > - } > > > - return 0; > > > +static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int > > > devfn, > > > + u8 pos, int cap) > > > +{ > > > + return PCI_FIND_NEXT_CAP_TTL(__pci_bus_read_config, pos, cap, bus, > > > + devfn); > > > } > > > static u8 __pci_find_next_cap(struct pci_bus *bus, unsigned int devfn, > > > u8 pos, int cap) > > > { > > > - int ttl = PCI_FIND_CAP_TTL; > > > - > > > - return __pci_find_next_cap_ttl(bus, devfn, pos, cap, &ttl); > > > + return __pci_find_next_cap_ttl(bus, devfn, pos, cap); > > > } > > > u8 pci_find_next_capability(struct pci_dev *dev, u8 pos, int cap) > > > @@ -553,42 +550,11 @@ EXPORT_SYMBOL(pci_bus_find_capability); > > > */ > > > u16 pci_find_next_ext_capability(struct pci_dev *dev, u16 start, int > > > cap) > > > { > > > - u32 header; > > > - int ttl; > > > - u16 pos = PCI_CFG_SPACE_SIZE; > > > - > > > - /* minimum 8 bytes per capability */ > > > - ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; > > > - > > > if (dev->cfg_size <= PCI_CFG_SPACE_SIZE) > > > return 0; > > > - if (start) > > > - pos = start; > > > - > > > - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) > > > - return 0; > > > - > > > - /* > > > - * If we have no capabilities, this is indicated by cap ID, > > > - * cap version and next pointer all being 0. > > > - */ > > > - if (header == 0) > > > - return 0; > > > - > > > - while (ttl-- > 0) { > > > - if (PCI_EXT_CAP_ID(header) == cap && pos != start) > > > - return pos; > > > - > > > - pos = PCI_EXT_CAP_NEXT(header); > > > - if (pos < PCI_CFG_SPACE_SIZE) > > > - break; > > > - > > > - if (pci_read_config_dword(dev, pos, &header) != > > > PCIBIOS_SUCCESSFUL) > > > - break; > > > - } > > > - > > > - return 0; > > > + return PCI_FIND_NEXT_EXT_CAPABILITY(__pci_bus_read_config, start, cap, > > > + dev->bus, dev->devfn); > > > > I don't like how 1 & 2 patches are split into two. IMO, they mostly belong > > together. However, (IMO) you can introduce the new all-size config space > > accessor in a separate patch before the combined patch. > > > > Ok. I'll change it to the following. The rest I'll combine into a patch. > > diff --git a/drivers/pci/access.c b/drivers/pci/access.c > index b123da16b63b..bb2e26c2eb81 100644 > --- a/drivers/pci/access.c > +++ b/drivers/pci/access.c > @@ -85,6 +85,23 @@ EXPORT_SYMBOL(pci_bus_write_config_byte); > EXPORT_SYMBOL(pci_bus_write_config_word); > EXPORT_SYMBOL(pci_bus_write_config_dword); > > + Extra newline > +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, > + u32 *val) > +{ > + struct pci_bus *bus = priv; > + int ret; > + > + if (size == 1) > + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); > + else if (size == 2) > + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); > + else > + ret = pci_bus_read_config_dword(bus, devfn, where, val); > + > + return ret; > +} > + > int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn, > int where, int size, u32 *val) > { > diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h > index 2e9cf26a9ee9..6a7c88b9cd35 100644 > --- a/drivers/pci/pci.h > +++ b/drivers/pci/pci.h > @@ -88,6 +88,8 @@ extern bool pci_early_dump; > bool pcie_cap_has_lnkctl(const struct pci_dev *dev); > bool pcie_cap_has_lnkctl2(const struct pci_dev *dev); > bool pcie_cap_has_rtctl(const struct pci_dev *dev); > +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, > + u32 *val); > > /* Functions internal to the PCI core code */ > > > > > } > > > EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); > > > @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); > > > static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int > > > ht_cap) > > > { > > > - int rc, ttl = PCI_FIND_CAP_TTL; > > > u8 cap, mask; > > > if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) > > > @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, > > > u8 pos, int ht_cap) > > > mask = HT_5BIT_CAP_MASK; > > > pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, > > > - PCI_CAP_ID_HT, &ttl); > > > + PCI_CAP_ID_HT); > > > while (pos) { > > > rc = pci_read_config_byte(dev, pos + 3, &cap); > > > if (rc != PCIBIOS_SUCCESSFUL) > > > @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, > > > u8 pos, int ht_cap) > > > pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, > > > pos + PCI_CAP_LIST_NEXT, > > > - PCI_CAP_ID_HT, &ttl); > > > + PCI_CAP_ID_HT); > > > > This function kind of had the idea to share the ttl but I suppose that was > > just a final safeguard to make sure the loop will always terminate in case > > the config space is corrupted so the unsharing is not a big issue. > > > > __pci_find_next_cap_ttl > // This macro definition already has ttl loop restrictions inside it. > PCI_FIND_NEXT_CAP_TTL > > Do I understand that you agree to remove ttl initialization and parameter > passing? Yes, I agree with it but doing anything like this (although I'd mention the reasoning in the changelog myself).
On 2025/4/3 17:15, Ilpo Järvinen wrote: >>> I don't like how 1 & 2 patches are split into two. IMO, they mostly belong >>> together. However, (IMO) you can introduce the new all-size config space >>> accessor in a separate patch before the combined patch. >>> >> >> Ok. I'll change it to the following. The rest I'll combine into a patch. >> >> diff --git a/drivers/pci/access.c b/drivers/pci/access.c >> index b123da16b63b..bb2e26c2eb81 100644 >> --- a/drivers/pci/access.c >> +++ b/drivers/pci/access.c >> @@ -85,6 +85,23 @@ EXPORT_SYMBOL(pci_bus_write_config_byte); >> EXPORT_SYMBOL(pci_bus_write_config_word); >> EXPORT_SYMBOL(pci_bus_write_config_dword); >> >> + > > Extra newline > Hi Ilpo, Thanks your for reply. Will delete. >> +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, >> + u32 *val) >> +{ >> + struct pci_bus *bus = priv; >> + int ret; >> + >> + if (size == 1) >> + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); >> + else if (size == 2) >> + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); >> + else >> + ret = pci_bus_read_config_dword(bus, devfn, where, val); >> + >> + return ret; >> +} >> + >> int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn, >> int where, int size, u32 *val) >> { >> diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h >> index 2e9cf26a9ee9..6a7c88b9cd35 100644 >> --- a/drivers/pci/pci.h >> +++ b/drivers/pci/pci.h >> @@ -88,6 +88,8 @@ extern bool pci_early_dump; >> bool pcie_cap_has_lnkctl(const struct pci_dev *dev); >> bool pcie_cap_has_lnkctl2(const struct pci_dev *dev); >> bool pcie_cap_has_rtctl(const struct pci_dev *dev); >> +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, >> + u32 *val); >> >> /* Functions internal to the PCI core code */ >> >> >>>> } >>>> EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); >>>> @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); >>>> static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int >>>> ht_cap) >>>> { >>>> - int rc, ttl = PCI_FIND_CAP_TTL; >>>> u8 cap, mask; >>>> if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) >>>> @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, >>>> u8 pos, int ht_cap) >>>> mask = HT_5BIT_CAP_MASK; >>>> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, >>>> - PCI_CAP_ID_HT, &ttl); >>>> + PCI_CAP_ID_HT); >>>> while (pos) { >>>> rc = pci_read_config_byte(dev, pos + 3, &cap); >>>> if (rc != PCIBIOS_SUCCESSFUL) >>>> @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, >>>> u8 pos, int ht_cap) >>>> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, >>>> pos + PCI_CAP_LIST_NEXT, >>>> - PCI_CAP_ID_HT, &ttl); >>>> + PCI_CAP_ID_HT); >>> >>> This function kind of had the idea to share the ttl but I suppose that was >>> just a final safeguard to make sure the loop will always terminate in case >>> the config space is corrupted so the unsharing is not a big issue. >>> >> >> __pci_find_next_cap_ttl >> // This macro definition already has ttl loop restrictions inside it. >> PCI_FIND_NEXT_CAP_TTL >> >> Do I understand that you agree to remove ttl initialization and parameter >> passing? > > Yes, I agree with it but doing anything like this (although I'd mention > the reasoning in the changelog myself). > Ok, I see. I will give the reasons. Best regards, Hans
On 2025/4/3 20:24, Hans Zhang wrote: > > > On 2025/4/3 17:15, Ilpo Järvinen wrote: >>>> I don't like how 1 & 2 patches are split into two. IMO, they mostly >>>> belong >>>> together. However, (IMO) you can introduce the new all-size config >>>> space >>>> accessor in a separate patch before the combined patch. >>>> >>> >>> Ok. I'll change it to the following. The rest I'll combine into a patch. >>> >>> diff --git a/drivers/pci/access.c b/drivers/pci/access.c >>> index b123da16b63b..bb2e26c2eb81 100644 >>> --- a/drivers/pci/access.c >>> +++ b/drivers/pci/access.c >>> @@ -85,6 +85,23 @@ EXPORT_SYMBOL(pci_bus_write_config_byte); >>> EXPORT_SYMBOL(pci_bus_write_config_word); >>> EXPORT_SYMBOL(pci_bus_write_config_dword); >>> >>> + >> >> Extra newline >> > > Hi Ilpo, > > Thanks your for reply. Will delete. > Hi Ilpo, The [v9 1/6]patch I plan to submit is as follows, please review it. From c099691ff1e980ff4633c55e94abcd888000e2cc Mon Sep 17 00:00:00 2001 From: Hans Zhang <18255117159@163.com> Date: Fri, 4 Apr 2025 00:19:32 +0800 Subject: [v9 1/6] PCI: Introduce generic bus config read helper function The primary PCI config space accessors are tied to the size of the read (byte/word/dword). Upcoming refactoring of PCI capability discovery logic requires passing a config accessor function that must be able to perform read with different sizes. Add any size config space read accessor pci_bus_read_config() to allow giving it as the config space accessor to the upcoming PCI capability discovery macro. Reconstructs the PCI function discovery logic to prepare for unified configuration of access modes. No function changes are intended. Signed-off-by: Hans Zhang <18255117159@163.com> --- drivers/pci/access.c | 17 +++++++++++++++++ drivers/pci/pci.h | 2 ++ 2 files changed, 19 insertions(+) diff --git a/drivers/pci/access.c b/drivers/pci/access.c index b123da16b63b..603332658ab3 100644 --- a/drivers/pci/access.c +++ b/drivers/pci/access.c @@ -85,6 +85,23 @@ EXPORT_SYMBOL(pci_bus_write_config_byte); EXPORT_SYMBOL(pci_bus_write_config_word); EXPORT_SYMBOL(pci_bus_write_config_dword); +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, + u32 *val) +{ + struct pci_bus *bus = priv; + int ret; + + if (size == 1) + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); + else if (size == 2) + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); + else + ret = pci_bus_read_config_dword(bus, devfn, where, val); + + return ret; +} +EXPORT_SYMBOL_GPL(pci_bus_read_config); + int pci_generic_config_read(struct pci_bus *bus, unsigned int devfn, int where, int size, u32 *val) { diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 2e9cf26a9ee9..6a7c88b9cd35 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h @@ -88,6 +88,8 @@ extern bool pci_early_dump; bool pcie_cap_has_lnkctl(const struct pci_dev *dev); bool pcie_cap_has_lnkctl2(const struct pci_dev *dev); bool pcie_cap_has_rtctl(const struct pci_dev *dev); +int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, + u32 *val); /* Functions internal to the PCI core code */ Best regards, Hans
On 2025/4/3 20:24, Hans Zhang wrote: >>>>> } >>>>> EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); >>>>> @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); >>>>> static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int >>>>> ht_cap) >>>>> { >>>>> - int rc, ttl = PCI_FIND_CAP_TTL; >>>>> u8 cap, mask; >>>>> if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) >>>>> @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev >>>>> *dev, >>>>> u8 pos, int ht_cap) >>>>> mask = HT_5BIT_CAP_MASK; >>>>> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, >>>>> - PCI_CAP_ID_HT, &ttl); >>>>> + PCI_CAP_ID_HT); >>>>> while (pos) { >>>>> rc = pci_read_config_byte(dev, pos + 3, &cap); >>>>> if (rc != PCIBIOS_SUCCESSFUL) >>>>> @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev >>>>> *dev, >>>>> u8 pos, int ht_cap) >>>>> pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, >>>>> pos + PCI_CAP_LIST_NEXT, >>>>> - PCI_CAP_ID_HT, &ttl); >>>>> + PCI_CAP_ID_HT); >>>> >>>> This function kind of had the idea to share the ttl but I suppose >>>> that was >>>> just a final safeguard to make sure the loop will always terminate >>>> in case >>>> the config space is corrupted so the unsharing is not a big issue. >>>> >>> >>> __pci_find_next_cap_ttl >>> // This macro definition already has ttl loop restrictions inside it. >>> PCI_FIND_NEXT_CAP_TTL >>> >>> Do I understand that you agree to remove ttl initialization and >>> parameter >>> passing? >> >> Yes, I agree with it but doing anything like this (although I'd mention >> the reasoning in the changelog myself). >> > > Ok, I see. I will give the reasons. Hi Ilpo, The [v9 3/6]patch I plan to submit is as follows, please review it. From 6da415d130e76b57ecf401f14bf0b66f20407839 Mon Sep 17 00:00:00 2001 From: Hans Zhang <18255117159@163.com> Date: Fri, 4 Apr 2025 00:20:29 +0800 Subject: [v9 3/6] PCI: Refactor capability search into common macros - Capability search is done both in PCI core and some controller drivers. - PCI core's cap search func requires PCI device and bus structs exist. - Controller drivers cannot use PCI core's cap search func as they need to find capabilities before they instantiated the PCI device & bus structs. - Move capability search into a macro so it can be reused where normal PCI config space accessors cannot yet be used due to lack of the instantiated PCI dev. - Instead, give the config space reading function as an argument to the new macro. - Convert PCI core to use the new macro. The macros now implement, parameterized by the config access method. The PCI core functions are converted to utilize these macros with the standard pci_bus_read_config accessors. Controller drivers can later use the same macros with their early access mechanisms while maintaining the existing protection against infinite loops through preserved TTL checks. The ttl parameter was originally an additional safeguard to prevent infinite loops in corrupted config space. However, the PCI_FIND_NEXT_CAP_TTL macro already enforces a TTL limit internally. Removing redundant ttl handling simplifies the interface while maintaining the safety guarantee. This aligns with the macro's design intent of encapsulating TTL management. Signed-off-by: Hans Zhang <18255117159@163.com> --- drivers/pci/pci.c | 70 +++++--------------------------------- drivers/pci/pci.h | 86 +++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 95 insertions(+), 61 deletions(-) diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index e4d3719b653d..bef242d84ab4 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c @@ -9,7 +9,6 @@ */ #include <linux/acpi.h> -#include <linux/align.h> #include <linux/kernel.h> #include <linux/delay.h> #include <linux/dmi.h> @@ -31,7 +30,6 @@ #include <asm/dma.h> #include <linux/aer.h> #include <linux/bitfield.h> -#include <uapi/linux/pci_regs.h> #include "pci.h" DEFINE_MUTEX(pci_slot_mutex); @@ -426,35 +424,16 @@ static int pci_dev_str_match(struct pci_dev *dev, const char *p, } static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, - u8 pos, int cap, int *ttl) + u8 pos, int cap) { - u8 id; - u16 ent; - - pci_bus_read_config_byte(bus, devfn, pos, &pos); - - while ((*ttl)--) { - if (pos < PCI_STD_HEADER_SIZEOF) - break; - pos = ALIGN_DOWN(pos, 4); - pci_bus_read_config_word(bus, devfn, pos, &ent); - - id = FIELD_GET(PCI_CAP_ID_MASK, ent); - if (id == 0xff) - break; - if (id == cap) - return pos; - pos = FIELD_GET(PCI_CAP_LIST_NEXT_MASK, ent); - } - return 0; + return PCI_FIND_NEXT_CAP_TTL(pci_bus_read_config, pos, cap, bus, + devfn); } static u8 __pci_find_next_cap(struct pci_bus *bus, unsigned int devfn, u8 pos, int cap) { - int ttl = PCI_FIND_CAP_TTL; - - return __pci_find_next_cap_ttl(bus, devfn, pos, cap, &ttl); + return __pci_find_next_cap_ttl(bus, devfn, pos, cap); } u8 pci_find_next_capability(struct pci_dev *dev, u8 pos, int cap) @@ -555,42 +534,11 @@ EXPORT_SYMBOL(pci_bus_find_capability); */ u16 pci_find_next_ext_capability(struct pci_dev *dev, u16 start, int cap) { - u32 header; - int ttl; - u16 pos = PCI_CFG_SPACE_SIZE; - - /* minimum 8 bytes per capability */ - ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; - if (dev->cfg_size <= PCI_CFG_SPACE_SIZE) return 0; - if (start) - pos = start; - - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) - return 0; - - /* - * If we have no capabilities, this is indicated by cap ID, - * cap version and next pointer all being 0. - */ - if (header == 0) - return 0; - - while (ttl-- > 0) { - if (PCI_EXT_CAP_ID(header) == cap && pos != start) - return pos; - - pos = PCI_EXT_CAP_NEXT(header); - if (pos < PCI_CFG_SPACE_SIZE) - break; - - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) - break; - } - - return 0; + return PCI_FIND_NEXT_EXT_CAPABILITY(pci_bus_read_config, start, cap, + dev->bus, dev->devfn); } EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); @@ -650,7 +598,7 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) { - int rc, ttl = PCI_FIND_CAP_TTL; + int rc; u8 cap, mask; if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) @@ -659,7 +607,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) mask = HT_5BIT_CAP_MASK; pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, - PCI_CAP_ID_HT, &ttl); + PCI_CAP_ID_HT); while (pos) { rc = pci_read_config_byte(dev, pos + 3, &cap); if (rc != PCIBIOS_SUCCESSFUL) @@ -670,7 +618,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos + PCI_CAP_LIST_NEXT, - PCI_CAP_ID_HT, &ttl); + PCI_CAP_ID_HT); } return 0; diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 6a7c88b9cd35..b204ebeeb1cf 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h @@ -2,7 +2,9 @@ #ifndef DRIVERS_PCI_H #define DRIVERS_PCI_H +#include <linux/align.h> #include <linux/pci.h> +#include <uapi/linux/pci_regs.h> struct pcie_tlp_log; @@ -91,6 +93,90 @@ bool pcie_cap_has_rtctl(const struct pci_dev *dev); int pci_bus_read_config(void *priv, unsigned int devfn, int where, u32 size, u32 *val); +/* Standard Capability finder */ +/** + * PCI_FIND_NEXT_CAP_TTL - Find a PCI standard capability + * @read_cfg: Function pointer for reading PCI config space + * @start: Starting position to begin search + * @cap: Capability ID to find + * @args: Arguments to pass to read_cfg function + * + * Iterates through the capability list in PCI config space to find + * the specified capability. Implements TTL (time-to-live) protection + * against infinite loops. + * + * Returns: Position of the capability if found, 0 otherwise. + */ +#define PCI_FIND_NEXT_CAP_TTL(read_cfg, start, cap, args...) \ +({ \ + int __ttl = PCI_FIND_CAP_TTL; \ + u8 __id, __found_pos = 0; \ + u8 __pos = (start); \ + u16 __ent; \ + \ + read_cfg(args, __pos, 1, (u32 *)&__pos); \ + \ + while (__ttl--) { \ + if (__pos < PCI_STD_HEADER_SIZEOF) \ + break; \ + \ + __pos = ALIGN_DOWN(__pos, 4); \ + read_cfg(args, __pos, 2, (u32 *)&__ent); \ + \ + __id = FIELD_GET(PCI_CAP_ID_MASK, __ent); \ + if (__id == 0xff) \ + break; \ + \ + if (__id == (cap)) { \ + __found_pos = __pos; \ + break; \ + } \ + \ + __pos = FIELD_GET(PCI_CAP_LIST_NEXT_MASK, __ent); \ + } \ + __found_pos; \ +}) + +/* Extended Capability finder */ +/** + * PCI_FIND_NEXT_EXT_CAPABILITY - Find a PCI extended capability + * @read_cfg: Function pointer for reading PCI config space + * @start: Starting position to begin search (0 for initial search) + * @cap: Extended capability ID to find + * @args: Arguments to pass to read_cfg function + * + * Searches the extended capability space in PCI config registers + * for the specified capability. Implements TTL protection against + * infinite loops using a calculated maximum search count. + * + * Returns: Position of the capability if found, 0 otherwise. + */ +#define PCI_FIND_NEXT_EXT_CAPABILITY(read_cfg, start, cap, args...) \ +({ \ + u16 __pos = (start) ?: PCI_CFG_SPACE_SIZE; \ + u16 __found_pos = 0; \ + int __ttl, __ret; \ + u32 __header; \ + \ + __ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; \ + while (__ttl-- > 0 && __pos >= PCI_CFG_SPACE_SIZE) { \ + __ret = read_cfg(args, __pos, 4, &__header); \ + if (__ret != PCIBIOS_SUCCESSFUL) \ + break; \ + \ + if (__header == 0) \ + break; \ + \ + if (PCI_EXT_CAP_ID(__header) == (cap) && __pos != start) { \ + __found_pos = __pos; \ + break; \ + } \ + \ + __pos = PCI_EXT_CAP_NEXT(__header); \ + } \ + __found_pos; \ +}) + /* Functions internal to the PCI core code */ #ifdef CONFIG_DMI Best regards, Hans
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 869d204a70a3..521096c73686 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c @@ -423,36 +423,33 @@ static int pci_dev_str_match(struct pci_dev *dev, const char *p, return 1; } -static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, - u8 pos, int cap, int *ttl) +static int __pci_bus_read_config(void *priv, unsigned int devfn, int where, + u32 size, u32 *val) { - u8 id; - u16 ent; + struct pci_bus *bus = priv; + int ret; - pci_bus_read_config_byte(bus, devfn, pos, &pos); + if (size == 1) + ret = pci_bus_read_config_byte(bus, devfn, where, (u8 *)val); + else if (size == 2) + ret = pci_bus_read_config_word(bus, devfn, where, (u16 *)val); + else + ret = pci_bus_read_config_dword(bus, devfn, where, val); - while ((*ttl)--) { - if (pos < 0x40) - break; - pos &= ~3; - pci_bus_read_config_word(bus, devfn, pos, &ent); + return ret; +} - id = ent & 0xff; - if (id == 0xff) - break; - if (id == cap) - return pos; - pos = (ent >> 8); - } - return 0; +static u8 __pci_find_next_cap_ttl(struct pci_bus *bus, unsigned int devfn, + u8 pos, int cap) +{ + return PCI_FIND_NEXT_CAP_TTL(__pci_bus_read_config, pos, cap, bus, + devfn); } static u8 __pci_find_next_cap(struct pci_bus *bus, unsigned int devfn, u8 pos, int cap) { - int ttl = PCI_FIND_CAP_TTL; - - return __pci_find_next_cap_ttl(bus, devfn, pos, cap, &ttl); + return __pci_find_next_cap_ttl(bus, devfn, pos, cap); } u8 pci_find_next_capability(struct pci_dev *dev, u8 pos, int cap) @@ -553,42 +550,11 @@ EXPORT_SYMBOL(pci_bus_find_capability); */ u16 pci_find_next_ext_capability(struct pci_dev *dev, u16 start, int cap) { - u32 header; - int ttl; - u16 pos = PCI_CFG_SPACE_SIZE; - - /* minimum 8 bytes per capability */ - ttl = (PCI_CFG_SPACE_EXP_SIZE - PCI_CFG_SPACE_SIZE) / 8; - if (dev->cfg_size <= PCI_CFG_SPACE_SIZE) return 0; - if (start) - pos = start; - - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) - return 0; - - /* - * If we have no capabilities, this is indicated by cap ID, - * cap version and next pointer all being 0. - */ - if (header == 0) - return 0; - - while (ttl-- > 0) { - if (PCI_EXT_CAP_ID(header) == cap && pos != start) - return pos; - - pos = PCI_EXT_CAP_NEXT(header); - if (pos < PCI_CFG_SPACE_SIZE) - break; - - if (pci_read_config_dword(dev, pos, &header) != PCIBIOS_SUCCESSFUL) - break; - } - - return 0; + return PCI_FIND_NEXT_EXT_CAPABILITY(__pci_bus_read_config, start, cap, + dev->bus, dev->devfn); } EXPORT_SYMBOL_GPL(pci_find_next_ext_capability); @@ -648,7 +614,6 @@ EXPORT_SYMBOL_GPL(pci_get_dsn); static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) { - int rc, ttl = PCI_FIND_CAP_TTL; u8 cap, mask; if (ht_cap == HT_CAPTYPE_SLAVE || ht_cap == HT_CAPTYPE_HOST) @@ -657,7 +622,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) mask = HT_5BIT_CAP_MASK; pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos, - PCI_CAP_ID_HT, &ttl); + PCI_CAP_ID_HT); while (pos) { rc = pci_read_config_byte(dev, pos + 3, &cap); if (rc != PCIBIOS_SUCCESSFUL) @@ -668,7 +633,7 @@ static u8 __pci_find_next_ht_cap(struct pci_dev *dev, u8 pos, int ht_cap) pos = __pci_find_next_cap_ttl(dev->bus, dev->devfn, pos + PCI_CAP_LIST_NEXT, - PCI_CAP_ID_HT, &ttl); + PCI_CAP_ID_HT); } return 0;
Refactor the PCI capability and extended capability search functions by consolidating common code patterns into reusable macros (PCI_FIND_NEXT_CAP_TTL and PCI_FIND_NEXT_EXT_CAPABILITY). The main changes include: 1. Introducing a unified config space read helper (__pci_bus_read_config). 2. Removing duplicate search logic from __pci_find_next_cap_ttl and pci_find_next_ext_capability. 3. Implementing consistent capability discovery using the new macros. 4. Simplifying HyperTransport capability lookup by leveraging the refactored code. The refactoring maintains existing functionality while reducing code duplication and improving maintainability. By centralizing the search logic, we achieve better code consistency and make future updates easier. This change has been verified to maintain backward compatibility with existing capability discovery patterns through thorough testing of PCI device enumeration and capability probing. Signed-off-by: Hans Zhang <18255117159@163.com> --- drivers/pci/pci.c | 79 +++++++++++++---------------------------------- 1 file changed, 22 insertions(+), 57 deletions(-)