src/fw/pciinit.c | 49 ++++++++++++++++++++++++++++++++---------------- 1 file changed, 33 insertions(+), 16 deletions(-)
hotplug of a PCI device to empty at startup pcie-pci-bridge fails when
ACPI PCI hotplug (default since 6.1) is enabled due to lack of resources.
Reproduced with:
#QEMU-6.2-rc1 -machine q35 -device pcie-pci-bridge,id=pcie-pci-bridge-0,addr=0x4
once linux guest is booted (test used Fedora 34), hotplug NIC from
QEMU monitor:
(qemu) device_add rtl8139,bus=pcie-pci-bridge-0,addr=0x2
guest fails hotplug with:
pci 0000:01:02.0: [10ec:8139] type 00 class 0x020000
pci 0000:01:02.0: reg 0x10: [io 0x0000-0x00ff]
pci 0000:01:02.0: reg 0x14: [mem 0x00000000-0x000000ff]
pci 0000:01:02.0: reg 0x30: [mem 0x00000000-0x0003ffff pref]
pci 0000:01:02.0: BAR 6: no space for [mem size 0x00040000 pref]
pci 0000:01:02.0: BAR 6: failed to assign [mem size 0x00040000 pref]
pci 0000:01:02.0: BAR 0: no space for [io size 0x0100]
pci 0000:01:02.0: BAR 0: failed to assign [io size 0x0100]
pci 0000:01:02.0: BAR 1: no space for [mem size 0x00000100]
pci 0000:01:02.0: BAR 1: failed to assign [mem size 0x00000100]
8139cp: 8139cp: 10/100 PCI Ethernet driver v1.3 (Mar 22, 2004)
PCI Interrupt Link [GSIG] enabled at IRQ 22
8139cp 0000:01:02.0: no MMIO resource
8139cp: probe of 0000:01:02.0 failed with error -5
Reason for this is that commit [1] didn't take into account
pcie-pci-bridge, marking bridge as non hotpluggable instead of
handling it as possibly SHPC capable bridge.
Fix issue by checking if pcie-pci-bridge is SHPC capable and
if it is mark it as hotpluggable.
With this hotplug of rtl8139 succeeds, with caveat that it fail
initialize IO bar, which is caused by [2] which makes firmware
skip IO reservation for any PCI device which isn't correct in case
of pcie-pci-bridge.
Fix it by exposing hotplug type and making resource optional
only if PCIe hotplug is in use.
Fixes regression in QEMU-6.1 and later, since it's switched to ACPI
based PCI hotplug on Q35 by default at that time.
RHBZ: https://bugzilla.redhat.com/show_bug.cgi?id=2001732
[1]
Fixes: 3aa31d7d637 ("hw/pci: reserve IO and mem for pci express downstream ports with no devices attached")
[2]
Fixes: 76327b9f32a ("fw/pci: do not automatically allocate IO region for PCIe bridges")
Signed-off-by: Igor Mammedov <imammedo@redhat.com>
CC: mapfelba@redhat.com
CC: kraxel@redhat.com
CC: mst@redhat.com
CC: lvivier@redhat.com
CC: jusual@redhat.com
---
src/fw/pciinit.c | 49 ++++++++++++++++++++++++++++++++----------------
1 file changed, 33 insertions(+), 16 deletions(-)
diff --git a/src/fw/pciinit.c b/src/fw/pciinit.c
index d25931bb..3107a0e6 100644
--- a/src/fw/pciinit.c
+++ b/src/fw/pciinit.c
@@ -793,7 +793,13 @@ pci_region_create_entry(struct pci_bus *bus, struct pci_device *dev,
return entry;
}
-static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
+typedef enum hotplug_type_t {
+ HOTPLUG_NO_SUPPORTED = 0,
+ HOTPLUG_PCIE,
+ HOTPLUG_SHPC
+} hotplug_type_t;
+
+static hotplug_type_t pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
{
u8 shpc_cap;
@@ -804,22 +810,31 @@ static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
(__builtin_ffs(PCI_EXP_FLAGS_TYPE) - 1));
u8 downstream_port = (port_type == PCI_EXP_TYPE_DOWNSTREAM) ||
(port_type == PCI_EXP_TYPE_ROOT_PORT);
- /*
- * PCI Express SPEC, 7.8.2:
- * Slot Implemented – When Set, this bit indicates that the Link
- * HwInit associated with this Port is connected to a slot (as
- * compared to being connected to a system-integrated device or
- * being disabled).
- * This bit is valid for Downstream Ports. This bit is undefined
- * for Upstream Ports.
- */
- u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
-
- return downstream_port && slot_implemented;
+
+ switch (port_type) {
+ case PCI_EXP_TYPE_PCI_BRIDGE:
+ /* fall-through to check if SHPC is enabled on bridge */
+ break;
+ default: {
+ /*
+ * PCI Express SPEC, 7.8.2:
+ * Slot Implemented – When Set, this bit indicates that the Link
+ * HwInit associated with this Port is connected to a slot (as
+ * compared to being connected to a system-integrated device or
+ * being disabled).
+ * This bit is valid for Downstream Ports. This bit is undefined
+ * for Upstream Ports.
+ */
+ u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
+ return downstream_port && slot_implemented ?
+ HOTPLUG_PCIE : HOTPLUG_NO_SUPPORTED;
+ break;
+ }
+ }
}
shpc_cap = pci_find_capability(bus->bus_dev->bdf, PCI_CAP_ID_SHPC, 0);
- return !!shpc_cap;
+ return !!shpc_cap ? HOTPLUG_SHPC : HOTPLUG_NO_SUPPORTED;
}
/* Test whether bridge support forwarding of transactions
@@ -904,7 +919,7 @@ static int pci_bios_check_devices(struct pci_bus *busses)
u8 pcie_cap = pci_find_capability(bdf, PCI_CAP_ID_EXP, 0);
u8 qemu_cap = pci_find_resource_reserve_capability(bdf);
- int hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
+ hotplug_type_t hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
for (type = 0; type < PCI_REGION_TYPE_COUNT; type++) {
u64 align = (type == PCI_REGION_TYPE_IO) ?
PCI_BRIDGE_IO_MIN : PCI_BRIDGE_MEM_MIN;
@@ -948,7 +963,9 @@ static int pci_bios_check_devices(struct pci_bus *busses)
if (pci_region_align(&s->r[type]) > align)
align = pci_region_align(&s->r[type]);
u64 sum = pci_region_sum(&s->r[type]);
- int resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
+ int resource_optional = 0;
+ if (hotplug_support == HOTPLUG_PCIE)
+ resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
if (!sum && hotplug_support && !resource_optional)
sum = align; /* reserve min size for hot-plug */
if (size > sum) {
--
2.27.0
_______________________________________________
SeaBIOS mailing list -- seabios@seabios.org
To unsubscribe send an email to seabios-leave@seabios.org
Dear Igor,
Am 26.11.21 um 19:46 schrieb Igor Mammedov:
> hotplug of a PCI device to empty at startup pcie-pci-bridge fails when
“to empty”, what does that mean?
> ACPI PCI hotplug (default since 6.1) is enabled due to lack of resources.
QEMU 6.1?
> Reproduced with:
>
> #QEMU-6.2-rc1 -machine q35 -device pcie-pci-bridge,id=pcie-pci-bridge-0,addr=0x4
>
> once linux guest is booted (test used Fedora 34), hotplug NIC from
> QEMU monitor:
> (qemu) device_add rtl8139,bus=pcie-pci-bridge-0,addr=0x2
>
> guest fails hotplug with:
> pci 0000:01:02.0: [10ec:8139] type 00 class 0x020000
> pci 0000:01:02.0: reg 0x10: [io 0x0000-0x00ff]
> pci 0000:01:02.0: reg 0x14: [mem 0x00000000-0x000000ff]
> pci 0000:01:02.0: reg 0x30: [mem 0x00000000-0x0003ffff pref]
> pci 0000:01:02.0: BAR 6: no space for [mem size 0x00040000 pref]
> pci 0000:01:02.0: BAR 6: failed to assign [mem size 0x00040000 pref]
> pci 0000:01:02.0: BAR 0: no space for [io size 0x0100]
> pci 0000:01:02.0: BAR 0: failed to assign [io size 0x0100]
> pci 0000:01:02.0: BAR 1: no space for [mem size 0x00000100]
> pci 0000:01:02.0: BAR 1: failed to assign [mem size 0x00000100]
> 8139cp: 8139cp: 10/100 PCI Ethernet driver v1.3 (Mar 22, 2004)
> PCI Interrupt Link [GSIG] enabled at IRQ 22
> 8139cp 0000:01:02.0: no MMIO resource
> 8139cp: probe of 0000:01:02.0 failed with error -5
>
> Reason for this is that commit [1] didn't take into account
> pcie-pci-bridge, marking bridge as non hotpluggable instead of
> handling it as possibly SHPC capable bridge.
> Fix issue by checking if pcie-pci-bridge is SHPC capable and
> if it is mark it as hotpluggable.
>
> With this hotplug of rtl8139 succeeds, with caveat that it fail
Nit: fail*s* *to*
> initialize IO bar, which is caused by [2] which makes firmware
> skip IO reservation for any PCI device which isn't correct in case
> of pcie-pci-bridge.
> Fix it by exposing hotplug type and making resource optional
> only if PCIe hotplug is in use.
>
> Fixes regression in QEMU-6.1 and later, since it's switched to ACPI
Nit: was switched
> based PCI hotplug on Q35 by default at that time.
>
> RHBZ: https://bugzilla.redhat.com/show_bug.cgi?id=2001732
> [1]
> Fixes: 3aa31d7d637 ("hw/pci: reserve IO and mem for pci express downstream ports with no devices attached")
> [2]
> Fixes: 76327b9f32a ("fw/pci: do not automatically allocate IO region for PCIe bridges")
> Signed-off-by: Igor Mammedov <imammedo@redhat.com>
> CC: mapfelba@redhat.com
> CC: kraxel@redhat.com
> CC: mst@redhat.com
> CC: lvivier@redhat.com
> CC: jusual@redhat.com
> ---
> src/fw/pciinit.c | 49 ++++++++++++++++++++++++++++++++----------------
> 1 file changed, 33 insertions(+), 16 deletions(-)
>
> diff --git a/src/fw/pciinit.c b/src/fw/pciinit.c
> index d25931bb..3107a0e6 100644
> --- a/src/fw/pciinit.c
> +++ b/src/fw/pciinit.c
> @@ -793,7 +793,13 @@ pci_region_create_entry(struct pci_bus *bus, struct pci_device *dev,
> return entry;
> }
>
> -static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> +typedef enum hotplug_type_t {
> + HOTPLUG_NO_SUPPORTED = 0,
> + HOTPLUG_PCIE,
> + HOTPLUG_SHPC
> +} hotplug_type_t;
> +
> +static hotplug_type_t pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> {
> u8 shpc_cap;
>
> @@ -804,22 +810,31 @@ static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> (__builtin_ffs(PCI_EXP_FLAGS_TYPE) - 1));
> u8 downstream_port = (port_type == PCI_EXP_TYPE_DOWNSTREAM) ||
> (port_type == PCI_EXP_TYPE_ROOT_PORT);
> - /*
> - * PCI Express SPEC, 7.8.2:
> - * Slot Implemented – When Set, this bit indicates that the Link
> - * HwInit associated with this Port is connected to a slot (as
> - * compared to being connected to a system-integrated device or
> - * being disabled).
> - * This bit is valid for Downstream Ports. This bit is undefined
> - * for Upstream Ports.
> - */
> - u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
> -
> - return downstream_port && slot_implemented;
> +
> + switch (port_type) {
> + case PCI_EXP_TYPE_PCI_BRIDGE:
> + /* fall-through to check if SHPC is enabled on bridge */
Doesn’t fall-through for switch statements mean, there is no break
statement?
> + break;
> + default: {
> + /*
> + * PCI Express SPEC, 7.8.2:
> + * Slot Implemented – When Set, this bit indicates that the Link
> + * HwInit associated with this Port is connected to a slot (as
> + * compared to being connected to a system-integrated device or
> + * being disabled).
> + * This bit is valid for Downstream Ports. This bit is undefined
> + * for Upstream Ports.
> + */
> + u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
> + return downstream_port && slot_implemented ?
> + HOTPLUG_PCIE : HOTPLUG_NO_SUPPORTED;
> + break;
> + }
> + }
> }
>
> shpc_cap = pci_find_capability(bus->bus_dev->bdf, PCI_CAP_ID_SHPC, 0);
> - return !!shpc_cap;
> + return !!shpc_cap ? HOTPLUG_SHPC : HOTPLUG_NO_SUPPORTED;
> }
>
> /* Test whether bridge support forwarding of transactions
> @@ -904,7 +919,7 @@ static int pci_bios_check_devices(struct pci_bus *busses)
> u8 pcie_cap = pci_find_capability(bdf, PCI_CAP_ID_EXP, 0);
> u8 qemu_cap = pci_find_resource_reserve_capability(bdf);
>
> - int hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
> + hotplug_type_t hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
> for (type = 0; type < PCI_REGION_TYPE_COUNT; type++) {
> u64 align = (type == PCI_REGION_TYPE_IO) ?
> PCI_BRIDGE_IO_MIN : PCI_BRIDGE_MEM_MIN;
> @@ -948,7 +963,9 @@ static int pci_bios_check_devices(struct pci_bus *busses)
> if (pci_region_align(&s->r[type]) > align)
> align = pci_region_align(&s->r[type]);
> u64 sum = pci_region_sum(&s->r[type]);
> - int resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
> + int resource_optional = 0;
> + if (hotplug_support == HOTPLUG_PCIE)
> + resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
> if (!sum && hotplug_support && !resource_optional)
> sum = align; /* reserve min size for hot-plug */
> if (size > sum) {
>
Rest looks good.
Kind regards,
Paul
_______________________________________________
SeaBIOS mailing list -- seabios@seabios.org
To unsubscribe send an email to seabios-leave@seabios.org
On Sat, 27 Nov 2021 11:19:35 +0100
Paul Menzel <pmenzel@molgen.mpg.de> wrote:
> Dear Igor,
>
>
> Am 26.11.21 um 19:46 schrieb Igor Mammedov:
> > hotplug of a PCI device to empty at startup pcie-pci-bridge fails when
>
> “to empty”, what does that mean?
meaning is pcie-pci-bridge that does not have any devices plugged into it.
maybe 'not populated' would be more clear?
>
> > ACPI PCI hotplug (default since 6.1) is enabled due to lack of resources.
>
> QEMU 6.1?
>
> > Reproduced with:
> >
> > #QEMU-6.2-rc1 -machine q35 -device pcie-pci-bridge,id=pcie-pci-bridge-0,addr=0x4
> >
> > once linux guest is booted (test used Fedora 34), hotplug NIC from
> > QEMU monitor:
> > (qemu) device_add rtl8139,bus=pcie-pci-bridge-0,addr=0x2
> >
> > guest fails hotplug with:
> > pci 0000:01:02.0: [10ec:8139] type 00 class 0x020000
> > pci 0000:01:02.0: reg 0x10: [io 0x0000-0x00ff]
> > pci 0000:01:02.0: reg 0x14: [mem 0x00000000-0x000000ff]
> > pci 0000:01:02.0: reg 0x30: [mem 0x00000000-0x0003ffff pref]
> > pci 0000:01:02.0: BAR 6: no space for [mem size 0x00040000 pref]
> > pci 0000:01:02.0: BAR 6: failed to assign [mem size 0x00040000 pref]
> > pci 0000:01:02.0: BAR 0: no space for [io size 0x0100]
> > pci 0000:01:02.0: BAR 0: failed to assign [io size 0x0100]
> > pci 0000:01:02.0: BAR 1: no space for [mem size 0x00000100]
> > pci 0000:01:02.0: BAR 1: failed to assign [mem size 0x00000100]
> > 8139cp: 8139cp: 10/100 PCI Ethernet driver v1.3 (Mar 22, 2004)
> > PCI Interrupt Link [GSIG] enabled at IRQ 22
> > 8139cp 0000:01:02.0: no MMIO resource
> > 8139cp: probe of 0000:01:02.0 failed with error -5
> >
> > Reason for this is that commit [1] didn't take into account
> > pcie-pci-bridge, marking bridge as non hotpluggable instead of
> > handling it as possibly SHPC capable bridge.
> > Fix issue by checking if pcie-pci-bridge is SHPC capable and
> > if it is mark it as hotpluggable.
> >
> > With this hotplug of rtl8139 succeeds, with caveat that it fail
>
> Nit: fail*s* *to*
>
> > initialize IO bar, which is caused by [2] which makes firmware
> > skip IO reservation for any PCI device which isn't correct in case
> > of pcie-pci-bridge.
> > Fix it by exposing hotplug type and making resource optional
> > only if PCIe hotplug is in use.
> >
> > Fixes regression in QEMU-6.1 and later, since it's switched to ACPI
>
> Nit: was switched
>
> > based PCI hotplug on Q35 by default at that time.
> >
> > RHBZ: https://bugzilla.redhat.com/show_bug.cgi?id=2001732
> > [1]
> > Fixes: 3aa31d7d637 ("hw/pci: reserve IO and mem for pci express downstream ports with no devices attached")
> > [2]
> > Fixes: 76327b9f32a ("fw/pci: do not automatically allocate IO region for PCIe bridges")
> > Signed-off-by: Igor Mammedov <imammedo@redhat.com>
> > CC: mapfelba@redhat.com
> > CC: kraxel@redhat.com
> > CC: mst@redhat.com
> > CC: lvivier@redhat.com
> > CC: jusual@redhat.com
> > ---
> > src/fw/pciinit.c | 49 ++++++++++++++++++++++++++++++++----------------
> > 1 file changed, 33 insertions(+), 16 deletions(-)
> >
> > diff --git a/src/fw/pciinit.c b/src/fw/pciinit.c
> > index d25931bb..3107a0e6 100644
> > --- a/src/fw/pciinit.c
> > +++ b/src/fw/pciinit.c
> > @@ -793,7 +793,13 @@ pci_region_create_entry(struct pci_bus *bus, struct pci_device *dev,
> > return entry;
> > }
> >
> > -static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> > +typedef enum hotplug_type_t {
> > + HOTPLUG_NO_SUPPORTED = 0,
> > + HOTPLUG_PCIE,
> > + HOTPLUG_SHPC
> > +} hotplug_type_t;
> > +
> > +static hotplug_type_t pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> > {
> > u8 shpc_cap;
> >
> > @@ -804,22 +810,31 @@ static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
> > (__builtin_ffs(PCI_EXP_FLAGS_TYPE) - 1));
> > u8 downstream_port = (port_type == PCI_EXP_TYPE_DOWNSTREAM) ||
> > (port_type == PCI_EXP_TYPE_ROOT_PORT);
> > - /*
> > - * PCI Express SPEC, 7.8.2:
> > - * Slot Implemented – When Set, this bit indicates that the Link
> > - * HwInit associated with this Port is connected to a slot (as
> > - * compared to being connected to a system-integrated device or
> > - * being disabled).
> > - * This bit is valid for Downstream Ports. This bit is undefined
> > - * for Upstream Ports.
> > - */
> > - u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
> > -
> > - return downstream_port && slot_implemented;
> > +
> > + switch (port_type) {
> > + case PCI_EXP_TYPE_PCI_BRIDGE:
> > + /* fall-through to check if SHPC is enabled on bridge */
>
> Doesn’t fall-through for switch statements mean, there is no break
> statement?
how about
/* do nothing and check later if SHPC is enabled */
>
> > + break;
> > + default: {
> > + /*
> > + * PCI Express SPEC, 7.8.2:
> > + * Slot Implemented – When Set, this bit indicates that the Link
> > + * HwInit associated with this Port is connected to a slot (as
> > + * compared to being connected to a system-integrated device or
> > + * being disabled).
> > + * This bit is valid for Downstream Ports. This bit is undefined
> > + * for Upstream Ports.
> > + */
> > + u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
> > + return downstream_port && slot_implemented ?
> > + HOTPLUG_PCIE : HOTPLUG_NO_SUPPORTED;
> > + break;
> > + }
> > + }
> > }
> >
> > shpc_cap = pci_find_capability(bus->bus_dev->bdf, PCI_CAP_ID_SHPC, 0);
> > - return !!shpc_cap;
> > + return !!shpc_cap ? HOTPLUG_SHPC : HOTPLUG_NO_SUPPORTED;
> > }
> >
> > /* Test whether bridge support forwarding of transactions
> > @@ -904,7 +919,7 @@ static int pci_bios_check_devices(struct pci_bus *busses)
> > u8 pcie_cap = pci_find_capability(bdf, PCI_CAP_ID_EXP, 0);
> > u8 qemu_cap = pci_find_resource_reserve_capability(bdf);
> >
> > - int hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
> > + hotplug_type_t hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
> > for (type = 0; type < PCI_REGION_TYPE_COUNT; type++) {
> > u64 align = (type == PCI_REGION_TYPE_IO) ?
> > PCI_BRIDGE_IO_MIN : PCI_BRIDGE_MEM_MIN;
> > @@ -948,7 +963,9 @@ static int pci_bios_check_devices(struct pci_bus *busses)
> > if (pci_region_align(&s->r[type]) > align)
> > align = pci_region_align(&s->r[type]);
> > u64 sum = pci_region_sum(&s->r[type]);
> > - int resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
> > + int resource_optional = 0;
> > + if (hotplug_support == HOTPLUG_PCIE)
> > + resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
> > if (!sum && hotplug_support && !resource_optional)
> > sum = align; /* reserve min size for hot-plug */
> > if (size > sum) {
> >
>
> Rest looks good.
>
>
> Kind regards,
>
> Paul
>
_______________________________________________
SeaBIOS mailing list -- seabios@seabios.org
To unsubscribe send an email to seabios-leave@seabios.org
Dear Igor,
Am 27.11.21 um 12:16 schrieb Igor Mammedov:
> On Sat, 27 Nov 2021 11:19:35 +0100 wrote:
>> Am 26.11.21 um 19:46 schrieb Igor Mammedov:
>>> hotplug of a PCI device to empty at startup pcie-pci-bridge fails when
>>
>> “to empty”, what does that mean?
>
> meaning is pcie-pci-bridge that does not have any devices plugged into it.
> maybe 'not populated' would be more clear?
Ah, the “at startup“ confused me then. Maybe (but I am no native speaker):
Hotplugging a PCI device to a since startup unpopulated pcee-pci-bridge
fails …
>>> ACPI PCI hotplug (default since 6.1) is enabled due to lack of resources.
>>
>> QEMU 6.1?
>>
>>> Reproduced with:
>>>
>>> #QEMU-6.2-rc1 -machine q35 -device pcie-pci-bridge,id=pcie-pci-bridge-0,addr=0x4
>>>
>>> once linux guest is booted (test used Fedora 34), hotplug NIC from
>>> QEMU monitor:
>>> (qemu) device_add rtl8139,bus=pcie-pci-bridge-0,addr=0x2
>>>
>>> guest fails hotplug with:
>>> pci 0000:01:02.0: [10ec:8139] type 00 class 0x020000
>>> pci 0000:01:02.0: reg 0x10: [io 0x0000-0x00ff]
>>> pci 0000:01:02.0: reg 0x14: [mem 0x00000000-0x000000ff]
>>> pci 0000:01:02.0: reg 0x30: [mem 0x00000000-0x0003ffff pref]
>>> pci 0000:01:02.0: BAR 6: no space for [mem size 0x00040000 pref]
>>> pci 0000:01:02.0: BAR 6: failed to assign [mem size 0x00040000 pref]
>>> pci 0000:01:02.0: BAR 0: no space for [io size 0x0100]
>>> pci 0000:01:02.0: BAR 0: failed to assign [io size 0x0100]
>>> pci 0000:01:02.0: BAR 1: no space for [mem size 0x00000100]
>>> pci 0000:01:02.0: BAR 1: failed to assign [mem size 0x00000100]
>>> 8139cp: 8139cp: 10/100 PCI Ethernet driver v1.3 (Mar 22, 2004)
>>> PCI Interrupt Link [GSIG] enabled at IRQ 22
>>> 8139cp 0000:01:02.0: no MMIO resource
>>> 8139cp: probe of 0000:01:02.0 failed with error -5
>>>
>>> Reason for this is that commit [1] didn't take into account
>>> pcie-pci-bridge, marking bridge as non hotpluggable instead of
>>> handling it as possibly SHPC capable bridge.
>>> Fix issue by checking if pcie-pci-bridge is SHPC capable and
>>> if it is mark it as hotpluggable.
>>>
>>> With this hotplug of rtl8139 succeeds, with caveat that it fail
>>
>> Nit: fail*s* *to*
>>
>>> initialize IO bar, which is caused by [2] which makes firmware
>>> skip IO reservation for any PCI device which isn't correct in case
>>> of pcie-pci-bridge.
>>> Fix it by exposing hotplug type and making resource optional
>>> only if PCIe hotplug is in use.
>>>
>>> Fixes regression in QEMU-6.1 and later, since it's switched to ACPI
>>
>> Nit: was switched
>>
>>> based PCI hotplug on Q35 by default at that time.
>>>
>>> RHBZ: https://bugzilla.redhat.com/show_bug.cgi?id=2001732
>>> [1]
>>> Fixes: 3aa31d7d637 ("hw/pci: reserve IO and mem for pci express downstream ports with no devices attached")
>>> [2]
>>> Fixes: 76327b9f32a ("fw/pci: do not automatically allocate IO region for PCIe bridges")
>>> Signed-off-by: Igor Mammedov <imammedo@redhat.com>
>>> CC: mapfelba@redhat.com
>>> CC: kraxel@redhat.com
>>> CC: mst@redhat.com
>>> CC: lvivier@redhat.com
>>> CC: jusual@redhat.com
>>> ---
>>> src/fw/pciinit.c | 49 ++++++++++++++++++++++++++++++++----------------
>>> 1 file changed, 33 insertions(+), 16 deletions(-)
>>>
>>> diff --git a/src/fw/pciinit.c b/src/fw/pciinit.c
>>> index d25931bb..3107a0e6 100644
>>> --- a/src/fw/pciinit.c
>>> +++ b/src/fw/pciinit.c
>>> @@ -793,7 +793,13 @@ pci_region_create_entry(struct pci_bus *bus, struct pci_device *dev,
>>> return entry;
>>> }
>>>
>>> -static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
>>> +typedef enum hotplug_type_t {
>>> + HOTPLUG_NO_SUPPORTED = 0,
>>> + HOTPLUG_PCIE,
>>> + HOTPLUG_SHPC
>>> +} hotplug_type_t;
>>> +
>>> +static hotplug_type_t pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
>>> {
>>> u8 shpc_cap;
>>>
>>> @@ -804,22 +810,31 @@ static int pci_bus_hotplug_support(struct pci_bus *bus, u8 pcie_cap)
>>> (__builtin_ffs(PCI_EXP_FLAGS_TYPE) - 1));
>>> u8 downstream_port = (port_type == PCI_EXP_TYPE_DOWNSTREAM) ||
>>> (port_type == PCI_EXP_TYPE_ROOT_PORT);
>>> - /*
>>> - * PCI Express SPEC, 7.8.2:
>>> - * Slot Implemented – When Set, this bit indicates that the Link
>>> - * HwInit associated with this Port is connected to a slot (as
>>> - * compared to being connected to a system-integrated device or
>>> - * being disabled).
>>> - * This bit is valid for Downstream Ports. This bit is undefined
>>> - * for Upstream Ports.
>>> - */
>>> - u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
>>> -
>>> - return downstream_port && slot_implemented;
>>> +
>>> + switch (port_type) {
>>> + case PCI_EXP_TYPE_PCI_BRIDGE:
>>> + /* fall-through to check if SHPC is enabled on bridge */
>>
>> Doesn’t fall-through for switch statements mean, there is no break
>> statement?
>
> how about
> /* do nothing and check later if SHPC is enabled */
Sounds good.
Kind regards,
Paul
>>> + break;
>>> + default: {
>>> + /*
>>> + * PCI Express SPEC, 7.8.2:
>>> + * Slot Implemented – When Set, this bit indicates that the Link
>>> + * HwInit associated with this Port is connected to a slot (as
>>> + * compared to being connected to a system-integrated device or
>>> + * being disabled).
>>> + * This bit is valid for Downstream Ports. This bit is undefined
>>> + * for Upstream Ports.
>>> + */
>>> + u16 slot_implemented = pcie_flags & PCI_EXP_FLAGS_SLOT;
>>> + return downstream_port && slot_implemented ?
>>> + HOTPLUG_PCIE : HOTPLUG_NO_SUPPORTED;
>>> + break;
>>> + }
>>> + }
>>> }
>>>
>>> shpc_cap = pci_find_capability(bus->bus_dev->bdf, PCI_CAP_ID_SHPC, 0);
>>> - return !!shpc_cap;
>>> + return !!shpc_cap ? HOTPLUG_SHPC : HOTPLUG_NO_SUPPORTED;
>>> }
>>>
>>> /* Test whether bridge support forwarding of transactions
>>> @@ -904,7 +919,7 @@ static int pci_bios_check_devices(struct pci_bus *busses)
>>> u8 pcie_cap = pci_find_capability(bdf, PCI_CAP_ID_EXP, 0);
>>> u8 qemu_cap = pci_find_resource_reserve_capability(bdf);
>>>
>>> - int hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
>>> + hotplug_type_t hotplug_support = pci_bus_hotplug_support(s, pcie_cap);
>>> for (type = 0; type < PCI_REGION_TYPE_COUNT; type++) {
>>> u64 align = (type == PCI_REGION_TYPE_IO) ?
>>> PCI_BRIDGE_IO_MIN : PCI_BRIDGE_MEM_MIN;
>>> @@ -948,7 +963,9 @@ static int pci_bios_check_devices(struct pci_bus *busses)
>>> if (pci_region_align(&s->r[type]) > align)
>>> align = pci_region_align(&s->r[type]);
>>> u64 sum = pci_region_sum(&s->r[type]);
>>> - int resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
>>> + int resource_optional = 0;
>>> + if (hotplug_support == HOTPLUG_PCIE)
>>> + resource_optional = pcie_cap && (type == PCI_REGION_TYPE_IO);
>>> if (!sum && hotplug_support && !resource_optional)
>>> sum = align; /* reserve min size for hot-plug */
>>> if (size > sum) {
>>>
_______________________________________________
SeaBIOS mailing list -- seabios@seabios.org
To unsubscribe send an email to seabios-leave@seabios.org
© 2016 - 2026 Red Hat, Inc.