Re: [PATCH 3/4 V3] irqchip: gic: Add supports for ARM GICv2m MSI(-X)

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On 7/30/2014 9:57 AM, Marc Zyngier wrote:
On Thu, Jul 10 2014 at 12:05:03 am BST, "suravee.suthikulpanit@xxxxxxx" <suravee.suthikulpanit@xxxxxxx> wrote:

Hi Suravee,

From: Suravee Suthikulpanit <Suravee.Suthikulpanit@xxxxxxx>

>> ......
>>
-  first region is the GIC distributor register base and size. The 2nd region is
-  the GIC cpu interface register base and size.
+- reg : Specifies base physical address(s) and size of the GIC register frames.
+
+  Region | Description
+  Index  |
+  -------------------------------------------------------------------
+     0   | GIC distributor register base and size
+     1   | GIC cpu interface register base and size
+     2   | VGIC interface control register base and size (Optional)
+     3   | VGIC CPU interface register base and size (Optional)
+     4   | GICv2m MSI interface register base and size (Optional)

Given that the v2m block is somehow bolted on the side of a standard
GICv2, I'd rather see it as a subnode of the main GIC.

Then you could directly call into the v2m layer to initialize it,
instead of the odd "reverse probing" you're using here...

[Suravee] IIUC, you don't think we should introduce the "gic-400-v2m" compatible ID. Instead, you want to see something like:

    gic @ 0x00f00000 {
        .....
        v2m {
            msi-controller;
            reg = < .... >; /* v2m reg frame */
        }
    }

If so, I can change this.


+
+static int __init
+gicv2m_msi_init(struct device_node *node, struct v2m_data *v2m)
+{
+       unsigned int val;
+
+       if (of_address_to_resource(node, GIC_OF_MSIV2M_RANGE_INDEX,
+                                  &v2m->res)) {
+               pr_err("GICv2m: Failed locate GICv2m MSI register frame\n");
+               return -EINVAL;
+       }
+
+       v2m->base = of_iomap(node, GIC_OF_MSIV2M_RANGE_INDEX);
+       if (!v2m->base) {
+               pr_err("GICv2m: Failed to map GIC MSI registers\n");
+               return -EINVAL;
+       }
+
+       val = readl_relaxed(v2m->base + V2M_MSI_TYPER);
+       if (!val) {
+               pr_warn("GICv2m: Failed to read V2M_MSI_TYPER register\n");
+               return -EINVAL;
+       }
+
+       v2m->spi_start = (val >> V2M_MSI_TYPER_BASE_SHIFT) &
+                               V2M_MSI_TYPER_BASE_MASK;
+       v2m->nr_spis = val & V2M_MSI_TYPER_NUM_MASK;
+       if ((v2m->spi_start < V2M_MIN_SPI) || (v2m->nr_spis >= V2M_MAX_SPI)) {
+                       pr_err("GICv2m: Invalid MSI_TYPER (%#x)\n", val);
+                       return -EINVAL;
+       }
+
+       v2m->bm = kzalloc(sizeof(long) * BITS_TO_LONGS(v2m->nr_spis),
+                         GFP_KERNEL);
+       if (!v2m->bm) {
+               pr_err("GICv2m: Failed to allocate MSI bitmap\n");
+               return -ENOMEM;
+       }
+
+       spin_lock_init(&v2m->msi_cnt_lock);
+
+       pr_info("GICv2m: SPI range [%d:%d]\n",
+               v2m->spi_start, (v2m->spi_start + v2m->nr_spis));
+
+       return 0;
+}

This function is assuming that you will only see one single MSI frame
here. Is there any chance that there would be more than one in a system?


[Suravee] If I would imagine such SOC, where there are multiple MSI frames (hence multiple msi-controllers), can we currently support this with the current msichip interface? Currently, each PCI RC connects to an "interrupt-parrent", which is also an MSI controller. We would need to have a way for PCI RC to specify which of the msichips within an interrupt-controller it wants to use.

Currently, I am not aware if there is a GIC w/ multiple MSI frames. Could you check if there is such product for ARM GICs?

+
+static void gicv2m_mask_irq(struct irq_data *d)
+{
+       gic_mask_irq(d);
+       if (d->msi_desc)
+               mask_msi_irq(d);
+}
+
+static void gicv2m_unmask_irq(struct irq_data *d)
+{
+       gic_unmask_irq(d);
+       if (d->msi_desc)
+               unmask_msi_irq(d);
+}
+
+static struct irq_chip gicv2m_chip = {
+       .name                   = "GICv2m",
+       .irq_mask               = gicv2m_mask_irq,
+       .irq_unmask             = gicv2m_unmask_irq,
+       .irq_eoi                = gic_eoi_irq,
+       .irq_set_type           = gic_set_type,
+       .irq_retrigger          = gic_retrigger,

I think you can loose the retrigger here.

OK.

+#ifdef CONFIG_SMP
+       .irq_set_affinity       = gic_set_affinity,
+#endif
+#ifdef CONFIG_PM
+       .irq_set_wake           = gic_set_wake,
+#endif
+};
+
+#ifdef CONFIG_OF
+static int __init
+gicv2m_of_init(struct device_node *node, struct device_node *parent)
+{
+       struct gic_chip_data *gic;
+       int ret;
+
+       ret = _gic_of_init(node, parent, &gicv2m_chip, &gic);
+       if (ret) {
+               pr_err("GICv2m: Failed to initialize GIC\n");
+               return ret;
+       }
+
+       gic->msi_chip.owner = THIS_MODULE;
+       gic->msi_chip.of_node = node;
+       gic->msi_chip.setup_irq = gicv2m_setup_msi_irq;
+       gic->msi_chip.teardown_irq = gicv2m_teardown_msi_irq;
+       ret = of_pci_msi_chip_add(&gic->msi_chip);
+       if (ret) {
+               /* MSI is optional and not supported here */
+               pr_info("GICv2m: MSI is not supported.\n");
+               return 0;
+       }
+
+       ret = gicv2m_msi_init(node, &gic->v2m_data);
+       if (ret)
+               return ret;
+       return ret;
+}
+
+IRQCHIP_DECLARE(arm_gic_400_v2m, "arm,gic-400-v2m", gicv2m_of_init);

So if you follow my advise of reversing your probing and call into the
v2m init from the main GIC driver, you could take a irq_chip as a
parameter, and use it to populate the v2m irq_chip, only overriding the
two methods that actually differ.

This would have the net effect of completely dropping patch #2, which
becomes effectively useless.


[Suravee] Ok, lemme look into this.

  struct gic_chip_data {
         union gic_base dist_base;
         union gic_base cpu_base;
@@ -20,12 +31,23 @@ struct gic_chip_data {
  #endif
         struct irq_domain *domain;
         unsigned int gic_irqs;
-       struct irq_chip *irq_chip;
  #ifdef CONFIG_GIC_NON_BANKED
         void __iomem *(*get_base)(union gic_base *);
  #endif
+       struct irq_chip *irq_chip;
+       struct msi_chip msi_chip;
+#ifdef CONFIG_ARM_GIC_V2M
+       struct v2m_data v2m_data;
+#endif

Why isn't msi_chip directly part of v2m_data? I think that would make
some of the code slightly clearer, and avoid polluting unsuspecting
architectures...


[Suravee] I can do that.

  };

+#ifdef CONFIG_OF
+int _gic_of_init(struct device_node *node,
+                struct device_node *parent,
+                struct irq_chip *chip,
+                struct gic_chip_data **gic) __init;
+#endif
+
  void gic_mask_irq(struct irq_data *d);
  void gic_unmask_irq(struct irq_data *d);
  void gic_eoi_irq(struct irq_data *d);
@@ -42,11 +64,4 @@ int gic_set_affinity(struct irq_data *d,
  int gic_set_wake(struct irq_data *d, unsigned int on);
  #endif

-#ifdef CONFIG_OF
-int _gic_of_init(struct device_node *node,
-                struct device_node *parent,
-                struct irq_chip *chip,
-                struct gic_chip_data **gic) __init;
-#endif
-
  #endif /* _IRQ_GIC_H_ */

What is the purpose of this move?

[Suravee] No need. I might have accidentally moved it.

Thanks,

Suravee


--
To unsubscribe from this list: send the line "unsubscribe linux-pci" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [DMA Engine]     [Linux Coverity]     [Linux USB]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Greybus]

  Powered by Linux