On Mon, 2019-06-10 at 09:20 +0100, Marc Zyngier wrote:
> Hi Zeev,
> 
> On 07/06/2019 00:17, Zeev Zilberman wrote:
> > The patch adds support for Amazon Graviton custom variant of GICv2m, where
> > hw irq is encoded using the MSI message address, as opposed to standard
> > GICv2m, where hw irq is encoded in the MSI message data.
> > In addition, the Graviton flavor of GICv2m is used along GICv3 (and not
> > GICv2).
> > 
> > Signed-off-by: Zeev Zilberman <z...@amazon.com>
> > Signed-off-by: Benjamin Herrenschmidt <b...@kernel.crashing.org>
> 
> There seem to be some confusion about who is the author of this patch.
> As you're the one posting the patch, your SoB tag should be the last
> one. And assuming the patch has been developed together with Ben, it
> should read:
> 
> Co-developed-by: Benjamin Herrenschmidt <b...@kernel.crashing.org>
> Signed-off-by: Benjamin Herrenschmidt <b...@kernel.crashing.org>
> Signed-off-by: Zeev Zilberman <z...@amazon.com>

It was his patch originally. I shuffled a few things around to make it
less intrusive, then Zeev picked it back up and addresses your previous
comments. I'm happy for him to take full ownership.

> > ---
> > diff --git a/drivers/irqchip/irq-gic-v2m.c b/drivers/irqchip/irq-gic-v2m.c
> > index 3c77ab6..eeed19f 100644
> > --- a/drivers/irqchip/irq-gic-v2m.c
> > +++ b/drivers/irqchip/irq-gic-v2m.c
> > @@ -56,6 +56,7 @@
> >  
> >  /* List of flags for specific v2m implementation */
> >  #define GICV2M_NEEDS_SPI_OFFSET            0x00000001
> > +#define GICV2M_GRAVITON_ADDRESS_ONLY       0x00000002
> >  
> >  static LIST_HEAD(v2m_nodes);
> >  static DEFINE_SPINLOCK(v2m_lock);
> > @@ -98,15 +99,26 @@ static struct msi_domain_info gicv2m_msi_domain_info = {
> >     .chip   = &gicv2m_msi_irq_chip,
> >  };
> >  
> > +static phys_addr_t gicv2m_get_msi_addr(struct v2m_data *v2m, int hwirq)
> > +{
> > +   if (v2m->flags & GICV2M_GRAVITON_ADDRESS_ONLY)
> > +           return v2m->res.start | ((hwirq - 32) << 3);
> > +   else
> > +           return v2m->res.start + V2M_MSI_SETSPI_NS;
> > +}
> > +
> >  static void gicv2m_compose_msi_msg(struct irq_data *data, struct msi_msg 
> > *msg)
> >  {
> >     struct v2m_data *v2m = irq_data_get_irq_chip_data(data);
> > -   phys_addr_t addr = v2m->res.start + V2M_MSI_SETSPI_NS;
> > +   phys_addr_t addr = gicv2m_get_msi_addr(v2m, data->hwirq);
> >  
> >     msg->address_hi = upper_32_bits(addr);
> >     msg->address_lo = lower_32_bits(addr);
> > -   msg->data = data->hwirq;
> >  
> > +   if (v2m->flags & GICV2M_GRAVITON_ADDRESS_ONLY)
> > +           msg->data = 0;
> > +   else
> > +           msg->data = data->hwirq;
> >     if (v2m->flags & GICV2M_NEEDS_SPI_OFFSET)
> >             msg->data -= v2m->spi_offset;
> >  
> > @@ -188,7 +200,7 @@ static int gicv2m_irq_domain_alloc(struct irq_domain 
> > *domain, unsigned int virq,
> >     hwirq = v2m->spi_start + offset;
> >  
> >     err = iommu_dma_prepare_msi(info->desc,
> > -                               v2m->res.start + V2M_MSI_SETSPI_NS);
> > +                               gicv2m_get_msi_addr(v2m, hwirq));
> >     if (err)
> >             return err;
> >  
> > @@ -307,7 +319,7 @@ static int gicv2m_allocate_domains(struct irq_domain 
> > *parent)
> >  
> >  static int __init gicv2m_init_one(struct fwnode_handle *fwnode,
> >                               u32 spi_start, u32 nr_spis,
> > -                             struct resource *res)
> > +                             struct resource *res, u32 flags)
> >  {
> >     int ret;
> >     struct v2m_data *v2m;
> > @@ -320,6 +332,7 @@ static int __init gicv2m_init_one(struct fwnode_handle 
> > *fwnode,
> >  
> >     INIT_LIST_HEAD(&v2m->entry);
> >     v2m->fwnode = fwnode;
> > +   v2m->flags = flags;
> >  
> >     memcpy(&v2m->res, res, sizeof(struct resource));
> >  
> > @@ -334,7 +347,14 @@ static int __init gicv2m_init_one(struct fwnode_handle 
> > *fwnode,
> >             v2m->spi_start = spi_start;
> >             v2m->nr_spis = nr_spis;
> >     } else {
> > -           u32 typer = readl_relaxed(v2m->base + V2M_MSI_TYPER);
> > +           u32 typer;
> > +
> > +           /* Graviton should always have explicit spi_start/nr_spis */
> > +           if (v2m->flags & GICV2M_GRAVITON_ADDRESS_ONLY) {
> > +                   ret = -EINVAL;
> > +                   goto err_iounmap;
> > +           }
> > +           typer = readl_relaxed(v2m->base + V2M_MSI_TYPER);
> >  
> >             v2m->spi_start = V2M_MSI_TYPER_BASE_SPI(typer);
> >             v2m->nr_spis = V2M_MSI_TYPER_NUM_SPI(typer);
> > @@ -355,18 +375,21 @@ static int __init gicv2m_init_one(struct 
> > fwnode_handle *fwnode,
> >      *
> >      * Broadom NS2 GICv2m implementation has an erratum where the MSI data
> >      * is 'spi_number - 32'
> > +    *
> > +    * Reading that register fails on the Graviton implementation
> >      */
> > -   switch (readl_relaxed(v2m->base + V2M_MSI_IIDR)) {
> > -   case XGENE_GICV2M_MSI_IIDR:
> > -           v2m->flags |= GICV2M_NEEDS_SPI_OFFSET;
> > -           v2m->spi_offset = v2m->spi_start;
> > -           break;
> > -   case BCM_NS2_GICV2M_MSI_IIDR:
> > -           v2m->flags |= GICV2M_NEEDS_SPI_OFFSET;
> > -           v2m->spi_offset = 32;
> > -           break;
> > +   if (!(v2m->flags & GICV2M_GRAVITON_ADDRESS_ONLY)) {
> > +           switch (readl_relaxed(v2m->base + V2M_MSI_IIDR)) {
> > +           case XGENE_GICV2M_MSI_IIDR:
> > +                   v2m->flags |= GICV2M_NEEDS_SPI_OFFSET;
> > +                   v2m->spi_offset = v2m->spi_start;
> > +                   break;
> > +           case BCM_NS2_GICV2M_MSI_IIDR:
> > +                   v2m->flags |= GICV2M_NEEDS_SPI_OFFSET;
> > +                   v2m->spi_offset = 32;
> > +                   break;
> > +           }
> >     }
> > -
> >     v2m->bm = kcalloc(BITS_TO_LONGS(v2m->nr_spis), sizeof(long),
> >                       GFP_KERNEL);
> >     if (!v2m->bm) {
> > @@ -419,7 +442,8 @@ static int __init gicv2m_of_init(struct fwnode_handle 
> > *parent_handle,
> >                     pr_info("DT overriding V2M MSI_TYPER (base:%u, 
> > num:%u)\n",
> >                             spi_start, nr_spis);
> >  
> > -           ret = gicv2m_init_one(&child->fwnode, spi_start, nr_spis, &res);
> > +           ret = gicv2m_init_one(&child->fwnode, spi_start, nr_spis,
> > +                                 &res, 0);
> >             if (ret) {
> >                     of_node_put(child);
> >                     break;
> > @@ -451,6 +475,29 @@ static struct fwnode_handle *gicv2m_get_fwnode(struct 
> > device *dev)
> >     return data->fwnode;
> >  }
> >  
> > +#ifdef CONFIG_ACPI
> > +static bool acpi_check_amazon_graviton_quirks(void)
> > +{
> > +   static struct acpi_table_madt *madt;
> > +   acpi_status status;
> > +   bool rc = false;
> > +
> > +#define ACPI_AMZN_OEM_ID           "AMAZON"
> > +
> > +   status = acpi_get_table(ACPI_SIG_MADT, 0,
> > +                           (struct acpi_table_header **)&madt);
> > +
> > +   if (ACPI_FAILURE(status) || !madt)
> > +           return rc;
> > +   rc = !memcmp(madt->header.oem_id, ACPI_AMZN_OEM_ID, ACPI_OEM_ID_SIZE);
> > +   acpi_put_table((struct acpi_table_header *)madt);
> > +
> > +   return rc;
> > +}
> > +#else
> > +static inline bool acpi_check_amazon_graviton_quirks(void) { return false; 
> > }
> > +#endif
> 
> Aren't you already in a #ifdef CONFIG_ACPI block here? You could loose
> the #ifdefery altogether.
> 
> > +
> >  static int __init
> >  acpi_parse_madt_msi(union acpi_subtable_headers *header,
> >                 const unsigned long end)
> > @@ -460,6 +507,7 @@ acpi_parse_madt_msi(union acpi_subtable_headers *header,
> >     u32 spi_start = 0, nr_spis = 0;
> >     struct acpi_madt_generic_msi_frame *m;
> >     struct fwnode_handle *fwnode;
> > +   u32 flags = 0;
> >  
> >     m = (struct acpi_madt_generic_msi_frame *)header;
> >     if (BAD_MADT_ENTRY(m, end))
> > @@ -469,6 +517,13 @@ acpi_parse_madt_msi(union acpi_subtable_headers 
> > *header,
> >     res.end = m->base_address + SZ_4K - 1;
> >     res.flags = IORESOURCE_MEM;
> >  
> > +   if (acpi_check_amazon_graviton_quirks()) {
> > +           pr_info("applying Amazon Graviton quirk\n");
> > +           res.end = res.start + SZ_8K - 1;
> > +           flags |= GICV2M_GRAVITON_ADDRESS_ONLY;
> > +           gicv2m_msi_domain_info.flags &= ~MSI_FLAG_MULTI_PCI_MSI;
> > +   }
> > +
> >     if (m->flags & ACPI_MADT_OVERRIDE_SPI_VALUES) {
> >             spi_start = m->spi_base;
> >             nr_spis = m->spi_count;
> > @@ -483,7 +538,7 @@ acpi_parse_madt_msi(union acpi_subtable_headers *header,
> >             return -EINVAL;
> >     }
> >  
> > -   ret = gicv2m_init_one(fwnode, spi_start, nr_spis, &res);
> > +   ret = gicv2m_init_one(fwnode, spi_start, nr_spis, &res, flags);
> >     if (ret)
> >             irq_domain_free_fwnode(fwnode);
> >  
> > diff --git a/drivers/irqchip/irq-gic-v3.c b/drivers/irqchip/irq-gic-v3.c
> > index f44cd89..1282f81 100644
> > --- a/drivers/irqchip/irq-gic-v3.c
> > +++ b/drivers/irqchip/irq-gic-v3.c
> > @@ -1343,6 +1343,9 @@ static int __init gic_init_bases(void __iomem 
> > *dist_base,
> >     if (gic_dist_supports_lpis()) {
> >             its_init(handle, &gic_data.rdists, gic_data.domain);
> >             its_cpu_init();
> > +   } else {
> > +           if (IS_ENABLED(CONFIG_ARM_GIC_V2M))
> > +                   gicv2m_init(handle, gic_data.domain);
> >     }
> >  
> >     if (gic_prio_masking_enabled()) {
> > diff --git a/include/linux/irqchip/arm-gic-common.h 
> > b/include/linux/irqchip/arm-gic-common.h
> > index 9a1a479..62a8821 100644
> > --- a/include/linux/irqchip/arm-gic-common.h
> > +++ b/include/linux/irqchip/arm-gic-common.h
> > @@ -39,4 +39,9 @@ struct gic_kvm_info {
> >  
> >  const struct gic_kvm_info *gic_get_kvm_info(void);
> >  
> > +struct irq_domain;
> > +struct fwnode_handle;
> > +int gicv2m_init(struct fwnode_handle *parent_handle,
> > +           struct irq_domain *parent);
> > +
> >  #endif /* __LINUX_IRQCHIP_ARM_GIC_COMMON_H */
> > diff --git a/include/linux/irqchip/arm-gic.h 
> > b/include/linux/irqchip/arm-gic.h
> > index 0f049b3..7bd3bc6 100644
> > --- a/include/linux/irqchip/arm-gic.h
> > +++ b/include/linux/irqchip/arm-gic.h
> > @@ -160,9 +160,6 @@ int gic_of_init_child(struct device *dev, struct 
> > gic_chip_data **gic, int irq);
> >   */
> >  void gic_init(void __iomem *dist , void __iomem *cpu);
> >  
> > -int gicv2m_init(struct fwnode_handle *parent_handle,
> > -           struct irq_domain *parent);
> > -
> >  void gic_send_sgi(unsigned int cpu_id, unsigned int irq);
> >  int gic_get_cpu_id(unsigned int cpu);
> >  void gic_migrate_target(unsigned int new_cpu_id);
> > 
> 
> Otherwise, looks OK to me. If you repost it with the couple of nits
> above fixed, I'll take it into -next for a bit of a soak test.
> 
> Thanks,
> 
>       M.

Reply via email to