[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <20170615110613.GA15854@red-moon>
Date: Thu, 15 Jun 2017 12:06:13 +0100
From: Lorenzo Pieralisi <lorenzo.pieralisi@....com>
To: Khuong Dinh <kdinh@....com>
Cc: Marc Zyngier <marc.zyngier@....com>, msalter@...hat.com,
Bjorn Helgaas <bhelgaas@...gle.com>, linux-pci@...r.kernel.org,
jcm@...hat.com, patches@....com, linux-kernel@...r.kernel.org,
linux-arm-kernel@...ts.infradead.org, rjw@...ysocki.net
Subject: Re: [PATCH v2 pci] PCI/MSI: pci-xgene-msi: Enable MSI support in
ACPI boot for X-Gene v1
On Wed, Jun 14, 2017 at 10:43:23AM -0700, Khuong Dinh wrote:
[...]
> >> Thanks for helping to resolve this general ACPI dependence issue.
> >> I look forward for a version to test with.
> >>
> >> Can we separate the general dependence patch from the X-Gene MSI ACPI
> >> enable patch.
> >> This original patch was to enable ACPI support for PCIe MSI.
> >> The dependence issue can be resolved separately.
> >> Can you help to pull in the patch.
> >
> > Ok, pragmatically the only sane thing you can do is:
> >
> > (1) Add an X-gene specific hook in drivers/acpi/scan.c (acpi_scan_init())
> > that carries out fwnode registration (ie register the fwnode by
> > searching the MSI HID object - this does not depend on ACPI device
> > nodes order - you enforce the order by parsing the namespace before
> > ACPI core does it, at that point in time ACPI objects are created).
> > Not sure Rafael will be happy to see this code but that's the only
> > solution that does not rely on ACPI device nodes ordering and kernel
> > link ordering. You may achieve the same by extending the ACPI APD
> > code (drivers/acpi/acpi_apd.c) whether that's the best way forward is
> > to be seen.
>
> Hi Rafael,
> Do you have any idea about this suggestion?
I mentioned drivers/acpi/acpi_apd.c because is in a way "similar" to
what you need but it is not exactly what's needed (I am not sure what
"devices" acpi_apd.c is meant to manage at the moment but certainly not
MSI controllers). In your callback from ACPI code you can't add a scan
handler - you have to get the ACPI device corresponding to the MSI
controller and register the IRQ domain fwnode.
Honestly it is hackish but that's the only approach that is robust
enough to always work (ie it does not depend on ACPI tables nodes order
and it does not depend on kernel link order).
Lorenzo
> Otherwise, I'll follow this Lorenzo's approach (1).
>
> > (2) You can also add an xgene specific scan handler at arch_init_call()
> > but given that PCI root bridge is managed through a scan handler too you
> > would rely on ACPI devices nodes ordering in the DSDT. Let me explain
> > something for the benefit of everyone reading this thread: I do not want
> > X-gene ACPI tables to force the kernel to scan devices in any order
> > whatsover because this would mean we will _never_ be able to change the
> > scan order if we wanted to lest we trigger regressions. So this is
> > a non-option in my opinion.
> >
> > (3) Last option is to register the MSI fwnode either in PCI ECAM quirk
> > handling or in arm64 before probing the PCI root bus.
> >
> > I am not keen on (2) and (3) at all, so _if_ we have to find a solution
> > to this problem (1) is the only option you have for the time being as
> > far as I am concerned.
>
> Thank you very much for your suggestions, Lorenzo.
> If there's not any other idea, I'll follow your approach (1)
>
> Best regards,
> Khuong Dinh
>
> > Lorenzo
> >
> >>
> >> Best regards,
> >> Khuong Dinh
> >>
> >> > Thanks,
> >> > Lorenzo
> >> >
> >> >> Thanks,
> >> >> Khuong Dinh
> >> >>
> >> >> >> Finally, please execute this command on the vmlinux that "works"
> >> >> >> for you:
> >> >> >>
> >> >> >> nm vmlinux | grep -E '__initcall_(xgene_pcie_msi_init|acpi_init)'
> >> >> >
> >> >> > $ nm vmlinux | grep -E '__initcall_(xgene_pcie_msi_init|acpi_init)'
> >> >> > ffff000008dab2d8 t __initcall_acpi_init4
> >> >> > ffff000008dab2c8 t __initcall_xgene_pcie_msi_init4
> >> >> >
> >> >> > Best regards,
> >> >> > Khuong Dinh
> >> >> >
> >> >> >> Even by ordering devices in the ACPI tables (that I abhor) I still do
> >> >> >> not understand how this works (I mean without relying on kernel link
> >> >> >> order to ensure that the MSI driver is probed before PCI devices are
> >> >> >> enumerated in acpi_init()).
> >> >> >>
> >> >> >> Thanks,
> >> >> >> Lorenzo
> >> >> >>
> >> >> >>> Best regards,
> >> >> >>> Khuong
> >> >> >>>
> >> >> >>> On Fri, Apr 28, 2017 at 2:27 AM, Marc Zyngier <marc.zyngier@....com> wrote:
> >> >> >>> > On 28/04/17 01:54, Khuong Dinh wrote:
> >> >> >>> >> From: Khuong Dinh <kdinh@....com>
> >> >> >>> >>
> >> >> >>> >> This patch makes pci-xgene-msi driver ACPI-aware and provides
> >> >> >>> >> MSI capability for X-Gene v1 PCIe controllers in ACPI boot mode.
> >> >> >>> >>
> >> >> >>> >> Signed-off-by: Khuong Dinh <kdinh@....com>
> >> >> >>> >> Signed-off-by: Duc Dang <dhdang@....com>
> >> >> >>> >> Acked-by: Marc Zyngier <marc.zyngier@....com>
> >> >> >>> >> ---
> >> >> >>> >> v2:
> >> >> >>> >> - Verify with BIOS version 3.06.25 and 3.07.09
> >> >> >>> >> v1:
> >> >> >>> >> - Initial version
> >> >> >>> >> ---
> >> >> >>> >> drivers/pci/host/pci-xgene-msi.c | 35 ++++++++++++++++++++++++++++++++---
> >> >> >>> >> 1 files changed, 32 insertions(+), 3 deletions(-)
> >> >> >>> >>
> >> >> >>> >> diff --git a/drivers/pci/host/pci-xgene-msi.c b/drivers/pci/host/pci-xgene-msi.c
> >> >> >>> >> index f1b633b..00aaa3d 100644
> >> >> >>> >> --- a/drivers/pci/host/pci-xgene-msi.c
> >> >> >>> >> +++ b/drivers/pci/host/pci-xgene-msi.c
> >> >> >>> >> @@ -24,6 +24,7 @@
> >> >> >>> >> #include <linux/pci.h>
> >> >> >>> >> #include <linux/platform_device.h>
> >> >> >>> >> #include <linux/of_pci.h>
> >> >> >>> >> +#include <linux/acpi.h>
> >> >> >>> >>
> >> >> >>> >> #define MSI_IR0 0x000000
> >> >> >>> >> #define MSI_INT0 0x800000
> >> >> >>> >> @@ -39,7 +40,7 @@ struct xgene_msi_group {
> >> >> >>> >> };
> >> >> >>> >>
> >> >> >>> >> struct xgene_msi {
> >> >> >>> >> - struct device_node *node;
> >> >> >>> >> + struct fwnode_handle *fwnode;
> >> >> >>> >> struct irq_domain *inner_domain;
> >> >> >>> >> struct irq_domain *msi_domain;
> >> >> >>> >> u64 msi_addr;
> >> >> >>> >> @@ -249,6 +250,13 @@ static void xgene_irq_domain_free(struct irq_domain *domain,
> >> >> >>> >> .free = xgene_irq_domain_free,
> >> >> >>> >> };
> >> >> >>> >>
> >> >> >>> >> +#ifdef CONFIG_ACPI
> >> >> >>> >> +static struct fwnode_handle *xgene_msi_get_fwnode(struct device *dev)
> >> >> >>> >> +{
> >> >> >>> >> + return xgene_msi_ctrl.fwnode;
> >> >> >>> >> +}
> >> >> >>> >> +#endif
> >> >> >>> >> +
> >> >> >>> >> static int xgene_allocate_domains(struct xgene_msi *msi)
> >> >> >>> >> {
> >> >> >>> >> msi->inner_domain = irq_domain_add_linear(NULL, NR_MSI_VEC,
> >> >> >>> >> @@ -256,7 +264,7 @@ static int xgene_allocate_domains(struct xgene_msi *msi)
> >> >> >>> >> if (!msi->inner_domain)
> >> >> >>> >> return -ENOMEM;
> >> >> >>> >>
> >> >> >>> >> - msi->msi_domain = pci_msi_create_irq_domain(of_node_to_fwnode(msi->node),
> >> >> >>> >> + msi->msi_domain = pci_msi_create_irq_domain(msi->fwnode,
> >> >> >>> >> &xgene_msi_domain_info,
> >> >> >>> >> msi->inner_domain);
> >> >> >>> >>
> >> >> >>> >> @@ -265,6 +273,9 @@ static int xgene_allocate_domains(struct xgene_msi *msi)
> >> >> >>> >> return -ENOMEM;
> >> >> >>> >> }
> >> >> >>> >>
> >> >> >>> >> +#ifdef CONFIG_ACPI
> >> >> >>> >> + pci_msi_register_fwnode_provider(&xgene_msi_get_fwnode);
> >> >> >>> >> +#endif
> >> >> >>> >> return 0;
> >> >> >>> >> }
> >> >> >>> >>
> >> >> >>> >> @@ -449,6 +460,13 @@ static int xgene_msi_hwirq_free(unsigned int cpu)
> >> >> >>> >> {},
> >> >> >>> >> };
> >> >> >>> >>
> >> >> >>> >> +#ifdef CONFIG_ACPI
> >> >> >>> >> +static const struct acpi_device_id xgene_msi_acpi_ids[] = {
> >> >> >>> >> + {"APMC0D0E", 0},
> >> >> >>> >> + { },
> >> >> >>> >> +};
> >> >> >>> >> +#endif
> >> >> >>> >> +
> >> >> >>> >> static int xgene_msi_probe(struct platform_device *pdev)
> >> >> >>> >> {
> >> >> >>> >> struct resource *res;
> >> >> >>> >> @@ -469,7 +487,17 @@ static int xgene_msi_probe(struct platform_device *pdev)
> >> >> >>> >> goto error;
> >> >> >>> >> }
> >> >> >>> >> xgene_msi->msi_addr = res->start;
> >> >> >>> >> - xgene_msi->node = pdev->dev.of_node;
> >> >> >>> >> +
> >> >> >>> >> + xgene_msi->fwnode = of_node_to_fwnode(pdev->dev.of_node);
> >> >> >>> >> + if (!xgene_msi->fwnode) {
> >> >> >>> >> + xgene_msi->fwnode = irq_domain_alloc_fwnode(NULL);
> >> >> >>> >
> >> >> >>> > Please provide something other than NULL, such as the base address if
> >> >> >>> > the device. That's quite useful for debugging.
> >> >> >>> >
> >> >> >>> >> + if (!xgene_msi->fwnode) {
> >> >> >>> >> + dev_err(&pdev->dev, "Failed to create fwnode\n");
> >> >> >>> >> + rc = ENOMEM;
> >> >> >>> >> + goto error;
> >> >> >>> >> + }
> >> >> >>> >> + }
> >> >> >>> >> +
> >> >> >>> >> xgene_msi->num_cpus = num_possible_cpus();
> >> >> >>> >>
> >> >> >>> >> rc = xgene_msi_init_allocator(xgene_msi);
> >> >> >>> >> @@ -540,6 +568,7 @@ static int xgene_msi_probe(struct platform_device *pdev)
> >> >> >>> >> .driver = {
> >> >> >>> >> .name = "xgene-msi",
> >> >> >>> >> .of_match_table = xgene_msi_match_table,
> >> >> >>> >> + .acpi_match_table = ACPI_PTR(xgene_msi_acpi_ids),
> >> >> >>> >> },
> >> >> >>> >> .probe = xgene_msi_probe,
> >> >> >>> >> .remove = xgene_msi_remove,
> >> >> >>> >>
> >> >> >>> >
> >> >> >>> > The code is trivial, but relies on the MSI controller to probe before
> >> >> >>> > the PCI bus. What enforces this? How is it making sure that this is not
> >> >> >>> > going to break in the next kernel release? As far as I can tell, there
> >> >> >>> > is no explicit dependency between the two, making it the whole thing
> >> >> >>> > extremely fragile.
> >> >> >>> >
> >> >> >>> > Thanks,
> >> >> >>> >
> >> >> >>> > M.
> >> >> >>> > --
> >> >> >>> > Jazz is not dead. It just smells funny...
> >> >> >>>
> >> >> >>> --
Powered by blists - more mailing lists