[PATCH v1] drivers: pci: introduce configurable delay for Rockchip PCIe bus scan
Bjorn Helgaas
helgaas at kernel.org
Tue May 9 14:19:02 PDT 2023
Hi Vincenzo,
Thanks for raising this issue. Let's see what we can do to address
it.
On Tue, May 09, 2023 at 05:39:12PM +0200, Vincenzo Palazzo wrote:
> Add a configurable delay to the Rockchip PCIe driver to address
> crashes that occur on some old devices, such as the Pine64 RockPro64.
>
> This issue is affecting the ARM community, but there is no
> upstream solution for it yet.
It sounds like this happens with several endpoints, right? And I
assume the endpoints work fine in other non-Rockchip systems? If
that's the case, my guess is the problem is with the Rockchip host
controller and how it's initialized, not with the endpoints.
The only delays and timeouts I see in the driver now are in
rockchip_pcie_host_init_port(), where it waits for link training to
complete. I assume the link training did completely successfully
since you don't mention either a gen1 or gen2 timeout (although the
gen2 message is a dev_dbg() that normally wouldn't go to the console).
I don't know that the spec contains a retrain timeout value. Several
other drivers use 1 second, while rockchip uses 500ms (for example,
see LINK_RETRAIN_TIMEOUT and LINK_UP_TIMEOUT).
I think we need to understand the issue better before adding a DT
property and a module parameter. Those are hard for users to deal
with. If we can figure out a value that works for everybody, it would
be better to just hard-code it in the driver and use that all the
time.
A few minor style/formatting comments below just for future reference:
> Crash dump (customized Manjaro kernel before this patch):
> [ 1.229856] SError Interrupt on CPU4, code 0xbf000002 -- SError
> [ 1.229860] CPU: 4 PID: 1 Comm: swapper/0 Not tainted 5.9.9-2.0-MANJARO-ARM #1
> [ 1.229862] Hardware name: Pine64 RockPro64 v2.1 (DT)
> [ 1.229864] pstate: 60000085 (nZCv daIf -PAN -UAO BTYPE=--)
> [ 1.229866] pc : rockchip_pcie_rd_conf+0xb4/0x270
> [ 1.229868] lr : rockchip_pcie_rd_conf+0x1b4/0x270
> [ 1.229870] sp : ffff80001004b850
> [ 1.229872] x29: ffff80001004b850 x28: 0000000000000001
> [ 1.229877] x27: 0000000000000000 x26: ffff00007a795000
> [ 1.229882] x25: ffff00007a7910b0 x24: 0000000000000000
> [ 1.229887] x23: 0000000000000000 x22: ffff00007b3a4380
> [ 1.229891] x21: ffff80001004b8c4 x20: 0000000000000004
> [ 1.229895] x19: 0000000000100000 x18: 0000000000000020
> [ 1.229900] x17: 0000000000000001 x16: 0000000000000019
> [ 1.229904] x15: ffff00007b222fd8 x14: ffffffffffffffff
> [ 1.229908] x13: ffff00007a79ba1c x12: ffff00007a79b290
> [ 1.229912] x11: 0101010101010101 x10: 7f7f7f7f7f7f7f7f
> [ 1.229917] x9 : ff72646268756463 x8 : 0000000000000391
> [ 1.229921] x7 : ffff80001004b880 x6 : 0000000000000001
> [ 1.229925] x5 : 0000000000000000 x4 : 0000000000000000
> [ 1.229930] x3 : 0000000000c00008 x2 : 000000000080000a
> [ 1.229934] x1 : 0000000000000000 x0 : ffff800014000000
> [ 1.229939] Kernel panic - not syncing: Asynchronous SError Interrupt
> [ 1.229942] CPU: 4 PID: 1 Comm: swapper/0 Not tainted 5.9.9-2.0-MANJARO-ARM #1
> [ 1.229944] Hardware name: Pine64 RockPro64 v2.1 (DT)
> [ 1.229946] Call trace:
> [ 1.229948] dump_backtrace+0x0/0x1d0
> [ 1.229949] show_stack+0x18/0x24
> [ 1.229951] dump_stack+0xc0/0x118
> [ 1.229953] panic+0x148/0x320
> [ 1.229955] nmi_panic+0x8c/0x90
> [ 1.229956] arm64_serror_panic+0x78/0x84
> [ 1.229958] do_serror+0x15c/0x160
> [ 1.229960] el1_error+0x84/0x100
> [ 1.229962] rockchip_pcie_rd_conf+0xb4/0x270
> [ 1.229964] pci_bus_read_config_dword+0x6c/0xd0
> [ 1.229966] pci_bus_generic_read_dev_vendor_id+0x34/0x1b0
> [ 1.229968] pci_scan_single_device+0xa4/0x144
> [ 1.229970] pci_scan_slot+0x40/0x12c
> [ 1.229972] pci_scan_child_bus_extend+0x58/0x34c
> [ 1.229974] pci_scan_bridge_extend+0x310/0x590
> [ 1.229976] pci_scan_child_bus_extend+0x210/0x34c
> [ 1.229978] pci_scan_root_bus_bridge+0x68/0xdc
> [ 1.229980] pci_host_probe+0x18/0xc4
> [ 1.229981] rockchip_pcie_probe+0x204/0x330
Include only the parts of the crash dump that are needed to debug the
problem or identify the problem enough to find this patch. Timestamps
probably aren't necessary. Register contents -- probably not either.
The rest of the backtrace (below here) probably isn't useful.
> [ 1.229984] platform_drv_probe+0x54/0xb0
> [ 1.229985] really_probe+0xe8/0x500
> [ 1.229987] driver_probe_device+0xd8/0xf0
> [ 1.229989] device_driver_attach+0xc0/0xcc
> [ 1.229991] __driver_attach+0xa4/0x170
> [ 1.229993] bus_for_each_dev+0x70/0xc0
> [ 1.229994] driver_attach+0x24/0x30
> [ 1.229996] bus_add_driver+0x140/0x234
> [ 1.229998] driver_register+0x78/0x130
> [ 1.230000] __platform_driver_register+0x4c/0x60
> [ 1.230002] rockchip_pcie_driver_init+0x1c/0x28
> [ 1.230004] do_one_initcall+0x54/0x1c0
> [ 1.230005] do_initcalls+0xf4/0x130
> [ 1.230007] kernel_init_freeable+0x144/0x19c
> [ 1.230009] kernel_init+0x14/0x11c
> [ 1.230011] ret_from_fork+0x10/0x34
> [ 1.230035] SMP: stopping secondary CPUs
> [ 1.230037] Kernel Offset: disabled
> [ 1.230039] CPU features: 0x0240022,2100200c
> [ 1.230041] Memory Limit: none
> +++ b/Documentation/admin-guide/kernel-parameters.txt
> @@ -4286,6 +4286,14 @@
> any pair of devices, possibly at the cost of
> reduced performance. This also guarantees
> that hot-added devices will work.
> + pcie_rockchip_host.bus_scan_delay= [PCIE] Delay in ms before
> + scanning PCIe bus in Rockchip PCIe host driver. Some PCIe
> + cards seem to need delays that can be several hundred ms.
> + If set to greater than or equal to 0 this parameter will
> + override delay that can be set in device tree.
> + Values less than 0 the module will hit an assertion
> + during the init.
> + The default value is 0.
Generally speaking module-specific stuff like this doesn't get
documented in kernel-parameters.txt.
> +++ b/arch/arm64/boot/dts/rockchip/rk3399-rockpro64.dtsi
> @@ -663,7 +663,8 @@ &pcie0 {
> pinctrl-0 = <&pcie_perst>;
> vpcie12v-supply = <&vcc12v_dcin>;
> vpcie3v3-supply = <&vcc3v3_pcie>;
> - status = "okay";
> + bus-scan-delay-ms = <0>;
> + status = "okay";
Please don't add arbitrary whitespace changes (it looks like this
uses leading spaces instead of tabs).
> +/* bus_scan_delay - module parameter to override the
> + * device tree value, which is 0 by default. */
Please follow comment style in the file, i.e.,
/*
* bus_scan_delay - ...
*/
Wrap comments to fill 78 columns or so to match the rest of the file.
> @@ -987,6 +996,23 @@ static int rockchip_pcie_probe(struct platform_device *pdev)
>
> rockchip_pcie_enable_interrupts(rockchip);
>
> + prob_delay = rockchip->bus_scan_delay;
> + if (bus_scan_delay)
> + prob_delay = bus_scan_delay;
> +
> + /*
> + * FIXME: This is a workaround for some devices that crash on calls to pci_host_probe()
> + * or pci_scan_root_bus_bridge(). We add a delay before bus scanning to avoid the crash.
> + * The call trace reaches rockchip_pcie_rd_conf() while attempting to read the vendor ID
> + * (pci_bus_generic_read_dev_vendor_id() is in the call stack) before panicking.
> + *
> + * I'm not sure why this workaround is effective or what causes the panic. It may be related
> + * to the cansleep value.
Wrap comments to fit in 78 columns to match the rest of the file.
> + */
> + dev_info(dev, "wait %u ms before bus scan\n", prob_delay);
> + if (prob_delay > 0)
> + msleep(prob_delay);
I don't think we want to just add a random delay here that's not
connected to anything else. I assume it could go in
rockchip_pcie_host_init_port() or perhaps rockchip_pcie_init_port()
(which deasserts resets, and there are usually timing constraints
related to deasserting resets). Hopefully Shawn can shed some light
on this.
> err = pci_host_probe(bridge);
> if (err < 0)
> goto err_remove_irq_domain;
> @@ -1055,6 +1081,11 @@ static struct platform_driver rockchip_pcie_driver = {
> };
> module_platform_driver(rockchip_pcie_driver);
>
> +/** Allow to override the device tree default configuration with
> + * a command line argument.
> + **/
Use multi-line comment style that matches the rest of the file.
> +module_param_named(bus_scan_delay, bus_scan_delay, int, S_IRUGO);
This should go right next to the bus_scan_delay definition above.
> +++ b/drivers/pci/controller/pcie-rockchip.c
> @@ -149,6 +149,11 @@ int rockchip_pcie_parse_dt(struct rockchip_pcie *rockchip)
> return PTR_ERR(rockchip->clk_pcie_pm);
> }
>
> + err = of_property_read_u32(node, "bus-scan-delay-ms", &rockchip->bus_scan_delay);
> + if (err) {
> + dev_info(dev, "no bus scan delay, default to 0 ms\n");
> + rockchip->bus_scan_delay = 0;
I hope we don't need this property at all, but if we do, I assume it
should be optional, with no message needed if it's not present.
> +++ b/drivers/pci/controller/pcie-rockchip.h
> @@ -299,6 +299,16 @@ struct rockchip_pcie {
> phys_addr_t msg_bus_addr;
> bool is_rc;
> struct resource *mem_res;
> +
> + /* It seems that the driver crashes on some
> + * older devices. To work around this, we
> + * should add a sleep delay before probing.
> + *
> + * FIXME: need more investigated with an,
> + * but looks like the problem can be related with
> + * the cansleep value?
> + **/
We need better understanding of what's going on here. Then this
comment could be made more specific, shorter, and formatted like
others.
> + u32 bus_scan_delay;
More information about the linux-arm-kernel
mailing list