Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/irqchip/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 14 kB image not shown  

Quelle  irq-loongson-eiointc.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
/*
 * Loongson Extend I/O Interrupt Controller support
 *
 * Copyright (C) 2020-2022 Loongson Technology Corporation Limited
 */


#define pr_fmt(fmt) "eiointc: " fmt

#include <linux/cpuhotplug.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <linux/irqchip.h>
#include <linux/irqdomain.h>
#include <linux/irqchip/chained_irq.h>
#include <linux/kernel.h>
#include <linux/kvm_para.h>
#include <linux/syscore_ops.h>
#include <asm/numa.h>

#include "irq-loongson.h"

#define EIOINTC_REG_NODEMAP 0x14a0
#define EIOINTC_REG_IPMAP 0x14c0
#define EIOINTC_REG_ENABLE 0x1600
#define EIOINTC_REG_BOUNCE 0x1680
#define EIOINTC_REG_ISR  0x1800
#define EIOINTC_REG_ROUTE 0x1c00

#define EXTIOI_VIRT_FEATURES           0x40000000
#define  EXTIOI_HAS_VIRT_EXTENSION     BIT(0)
#define  EXTIOI_HAS_ENABLE_OPTION      BIT(1)
#define  EXTIOI_HAS_INT_ENCODE         BIT(2)
#define  EXTIOI_HAS_CPU_ENCODE         BIT(3)
#define EXTIOI_VIRT_CONFIG             0x40000004
#define  EXTIOI_ENABLE                 BIT(1)
#define  EXTIOI_ENABLE_INT_ENCODE      BIT(2)
#define  EXTIOI_ENABLE_CPU_ENCODE      BIT(3)

#define VEC_REG_COUNT  4
#define VEC_COUNT_PER_REG 64
#define VEC_COUNT  (VEC_REG_COUNT * VEC_COUNT_PER_REG)
#define VEC_REG_IDX(irq_id) ((irq_id) / VEC_COUNT_PER_REG)
#define VEC_REG_BIT(irq_id)     ((irq_id) % VEC_COUNT_PER_REG)
#define EIOINTC_ALL_ENABLE 0xffffffff
#define EIOINTC_ALL_ENABLE_VEC_MASK(vector) (EIOINTC_ALL_ENABLE & ~BIT(vector &&nbsp;0x1f))
#define EIOINTC_REG_ENABLE_VEC(vector)  (EIOINTC_REG_ENABLE + ((vector >> 5) << 2))
#define EIOINTC_USE_CPU_ENCODE   BIT(0)

#define MAX_EIO_NODES  (NR_CPUS / CORES_PER_EIO_NODE)

/*
 * Routing registers are 32bit, and there is 8-bit route setting for every
 * interrupt vector. So one Route register contains four vectors routing
 * information.
 */

#define EIOINTC_REG_ROUTE_VEC(vector)  (EIOINTC_REG_ROUTE + (vector & ~0x03))
#define EIOINTC_REG_ROUTE_VEC_SHIFT(vector) ((vector & 0x03) << 3)
#define EIOINTC_REG_ROUTE_VEC_MASK(vector) (0xff << EIOINTC_REG_ROUTE_VEC_SHIFT(vector))

static int nr_pics;

struct eiointc_priv {
 u32   node;
 u32   vec_count;
 nodemask_t  node_map;
 cpumask_t  cpuspan_map;
 struct fwnode_handle *domain_handle;
 struct irq_domain *eiointc_domain;
 int   flags;
 irq_hw_number_t  parent_hwirq;
};

static struct eiointc_priv *eiointc_priv[MAX_IO_PICS];

static void eiointc_enable(void)
{
 uint64_t misc;

 misc = iocsr_read64(LOONGARCH_IOCSR_MISC_FUNC);
 misc |= IOCSR_MISC_FUNC_EXT_IOI_EN;
 iocsr_write64(misc, LOONGARCH_IOCSR_MISC_FUNC);
}

static int cpu_to_eio_node(int cpu)
{
 if (!kvm_para_has_feature(KVM_FEATURE_VIRT_EXTIOI))
  return cpu_logical_map(cpu) / CORES_PER_EIO_NODE;
 else
  return cpu_logical_map(cpu) / CORES_PER_VEIO_NODE;
}

#ifdef CONFIG_SMP
static void eiointc_set_irq_route(int pos, unsigned int cpu, unsigned int mnode, nodemask_t *node_map)
{
 int i, node, cpu_node, route_node;
 unsigned char coremap;
 uint32_t pos_off, data, data_byte, data_mask;

 pos_off = pos & ~3;
 data_byte = pos & 3;
 data_mask = ~BIT_MASK(data_byte) & 0xf;

 /* Calculate node and coremap of target irq */
 cpu_node = cpu_logical_map(cpu) / CORES_PER_EIO_NODE;
 coremap = BIT(cpu_logical_map(cpu) % CORES_PER_EIO_NODE);

 for_each_online_cpu(i) {
  node = cpu_to_eio_node(i);
  if (!node_isset(node, *node_map))
   continue;

  /* EIO node 0 is in charge of inter-node interrupt dispatch */
  route_node = (node == mnode) ? cpu_node : node;
  data = ((coremap | (route_node << 4)) << (data_byte * 8));
  csr_any_send(EIOINTC_REG_ROUTE + pos_off, data, data_mask, node * CORES_PER_EIO_NODE);
 }
}

static void veiointc_set_irq_route(unsigned int vector, unsigned int cpu)
{
 unsigned long reg = EIOINTC_REG_ROUTE_VEC(vector);
 unsigned int data;

 data = iocsr_read32(reg);
 data &= ~EIOINTC_REG_ROUTE_VEC_MASK(vector);
 data |= cpu_logical_map(cpu) << EIOINTC_REG_ROUTE_VEC_SHIFT(vector);
 iocsr_write32(data, reg);
}

static DEFINE_RAW_SPINLOCK(affinity_lock);

static int eiointc_set_irq_affinity(struct irq_data *d, const struct cpumask *affinity, bool force)
{
 unsigned int cpu;
 unsigned long flags;
 uint32_t vector, regaddr;
 struct eiointc_priv *priv = d->domain->host_data;

 raw_spin_lock_irqsave(&affinity_lock, flags);

 cpu = cpumask_first_and_and(&priv->cpuspan_map, affinity, cpu_online_mask);
 if (cpu >= nr_cpu_ids) {
  raw_spin_unlock_irqrestore(&affinity_lock, flags);
  return -EINVAL;
 }

 vector = d->hwirq;
 regaddr = EIOINTC_REG_ENABLE_VEC(vector);

 if (priv->flags & EIOINTC_USE_CPU_ENCODE) {
  iocsr_write32(EIOINTC_ALL_ENABLE_VEC_MASK(vector), regaddr);
  veiointc_set_irq_route(vector, cpu);
  iocsr_write32(EIOINTC_ALL_ENABLE, regaddr);
 } else {
  /* Mask target vector */
  csr_any_send(regaddr, EIOINTC_ALL_ENABLE_VEC_MASK(vector),
        0x0, priv->node * CORES_PER_EIO_NODE);

  /* Set route for target vector */
  eiointc_set_irq_route(vector, cpu, priv->node, &priv->node_map);

  /* Unmask target vector */
  csr_any_send(regaddr, EIOINTC_ALL_ENABLE,
        0x0, priv->node * CORES_PER_EIO_NODE);
 }

 irq_data_update_effective_affinity(d, cpumask_of(cpu));

 raw_spin_unlock_irqrestore(&affinity_lock, flags);

 return IRQ_SET_MASK_OK;
}
#endif

static int eiointc_index(int node)
{
 int i;

 for (i = 0; i < nr_pics; i++) {
  if (node_isset(node, eiointc_priv[i]->node_map))
   return i;
 }

 return -1;
}

static int eiointc_router_init(unsigned int cpu)
{
 int i, bit, cores, index, node;
 unsigned int data;

 node = cpu_to_eio_node(cpu);
 index = eiointc_index(node);

 if (index < 0) {
  pr_err("Error: invalid nodemap!\n");
  return -EINVAL;
 }

 if (!(eiointc_priv[index]->flags & EIOINTC_USE_CPU_ENCODE))
  cores = CORES_PER_EIO_NODE;
 else
  cores = CORES_PER_VEIO_NODE;

 if ((cpu_logical_map(cpu) % cores) == 0) {
  eiointc_enable();

  for (i = 0; i < eiointc_priv[0]->vec_count / 32; i++) {
   data = (((1 << (i * 2 + 1)) << 16) | (1 << (i * 2)));
   iocsr_write32(data, EIOINTC_REG_NODEMAP + i * 4);
  }

  for (i = 0; i < eiointc_priv[0]->vec_count / 32 / 4; i++) {
   /*
 * Route to interrupt pin, relative offset used here
 * Offset 0 means routing to IP0 and so on
 * Every 32 vector routing to one interrupt pin
 */

   bit = BIT(eiointc_priv[index]->parent_hwirq - INT_HWI0);
   data = bit | (bit << 8) | (bit << 16) | (bit << 24);
   iocsr_write32(data, EIOINTC_REG_IPMAP + i * 4);
  }

  for (i = 0; i < eiointc_priv[0]->vec_count / 4; i++) {
   /* Route to Node-0 Core-0 */
   if (eiointc_priv[index]->flags & EIOINTC_USE_CPU_ENCODE)
    bit = cpu_logical_map(0);
   else if (index == 0)
    bit = BIT(cpu_logical_map(0));
   else
    bit = (eiointc_priv[index]->node << 4) | 1;

   data = bit | (bit << 8) | (bit << 16) | (bit << 24);
   iocsr_write32(data, EIOINTC_REG_ROUTE + i * 4);
  }

  for (i = 0; i < eiointc_priv[0]->vec_count / 32; i++) {
   data = 0xffffffff;
   iocsr_write32(data, EIOINTC_REG_ENABLE + i * 4);
   iocsr_write32(data, EIOINTC_REG_BOUNCE + i * 4);
  }
 }

 return 0;
}

static void eiointc_irq_dispatch(struct irq_desc *desc)
{
 int i;
 u64 pending;
 bool handled = false;
 struct irq_chip *chip = irq_desc_get_chip(desc);
 struct eiointc_priv *priv = irq_desc_get_handler_data(desc);

 chained_irq_enter(chip, desc);

 for (i = 0; i < eiointc_priv[0]->vec_count / VEC_COUNT_PER_REG; i++) {
  pending = iocsr_read64(EIOINTC_REG_ISR + (i << 3));

  /* Skip handling if pending bitmap is zero */
  if (!pending)
   continue;

  /* Clear the IRQs */
  iocsr_write64(pending, EIOINTC_REG_ISR + (i << 3));
  while (pending) {
   int bit = __ffs(pending);
   int irq = bit + VEC_COUNT_PER_REG * i;

   generic_handle_domain_irq(priv->eiointc_domain, irq);
   pending &= ~BIT(bit);
   handled = true;
  }
 }

 if (!handled)
  spurious_interrupt();

 chained_irq_exit(chip, desc);
}

static void eiointc_ack_irq(struct irq_data *d)
{
}

static void eiointc_mask_irq(struct irq_data *d)
{
}

static void eiointc_unmask_irq(struct irq_data *d)
{
}

static struct irq_chip eiointc_irq_chip = {
 .name   = "EIOINTC",
 .irq_ack  = eiointc_ack_irq,
 .irq_mask  = eiointc_mask_irq,
 .irq_unmask  = eiointc_unmask_irq,
#ifdef CONFIG_SMP
 .irq_set_affinity = eiointc_set_irq_affinity,
#endif
};

static int eiointc_domain_alloc(struct irq_domain *domain, unsigned int virq,
    unsigned int nr_irqs, void *arg)
{
 int ret;
 unsigned int i, type;
 unsigned long hwirq = 0;
 struct eiointc_priv *priv = domain->host_data;

 ret = irq_domain_translate_onecell(domain, arg, &hwirq, &type);
 if (ret)
  return ret;

 for (i = 0; i < nr_irqs; i++) {
  irq_domain_set_info(domain, virq + i, hwirq + i, &eiointc_irq_chip,
     priv, handle_edge_irq, NULL, NULL);
 }

 return 0;
}

static void eiointc_domain_free(struct irq_domain *domain, unsigned int virq,
    unsigned int nr_irqs)
{
 int i;

 for (i = 0; i < nr_irqs; i++) {
  struct irq_data *d = irq_domain_get_irq_data(domain, virq + i);

  irq_set_handler(virq + i, NULL);
  irq_domain_reset_irq_data(d);
 }
}

static const struct irq_domain_ops eiointc_domain_ops = {
 .translate = irq_domain_translate_onecell,
 .alloc  = eiointc_domain_alloc,
 .free  = eiointc_domain_free,
};

static void acpi_set_vec_parent(int node, struct irq_domain *parent, struct acpi_vector_group *vec_group)
{
 int i;

 for (i = 0; i < MAX_IO_PICS; i++) {
  if (node == vec_group[i].node) {
   vec_group[i].parent = parent;
   return;
  }
 }
}

static struct irq_domain *acpi_get_vec_parent(int node, struct acpi_vector_group *vec_group)
{
 int i;

 for (i = 0; i < MAX_IO_PICS; i++) {
  if (node == vec_group[i].node)
   return vec_group[i].parent;
 }
 return NULL;
}

static int eiointc_suspend(void)
{
 return 0;
}

static void eiointc_resume(void)
{
 eiointc_router_init(0);
}

static struct syscore_ops eiointc_syscore_ops = {
 .suspend = eiointc_suspend,
 .resume = eiointc_resume,
};

static int __init pch_pic_parse_madt(union acpi_subtable_headers *header,
     const unsigned long end)
{
 struct acpi_madt_bio_pic *pchpic_entry = (struct acpi_madt_bio_pic *)header;
 unsigned int node = (pchpic_entry->address >> 44) & 0xf;
 struct irq_domain *parent = acpi_get_vec_parent(node, pch_group);

 if (parent)
  return pch_pic_acpi_init(parent, pchpic_entry);

 return 0;
}

static int __init pch_msi_parse_madt(union acpi_subtable_headers *header,
     const unsigned long end)
{
 struct irq_domain *parent;
 struct acpi_madt_msi_pic *pchmsi_entry = (struct acpi_madt_msi_pic *)header;
 int node;

 if (cpu_has_flatmode)
  node = early_cpu_to_node(eiointc_priv[nr_pics - 1]->node * CORES_PER_EIO_NODE);
 else
  node = eiointc_priv[nr_pics - 1]->node;

 parent = acpi_get_vec_parent(node, msi_group);

 if (parent)
  return pch_msi_acpi_init(parent, pchmsi_entry);

 return 0;
}

static int __init acpi_cascade_irqdomain_init(void)
{
 int r;

 r = acpi_table_parse_madt(ACPI_MADT_TYPE_BIO_PIC, pch_pic_parse_madt, 0);
 if (r < 0)
  return r;

 if (cpu_has_avecint)
  return 0;

 r = acpi_table_parse_madt(ACPI_MADT_TYPE_MSI_PIC, pch_msi_parse_madt, 1);
 if (r < 0)
  return r;

 return 0;
}

static int __init eiointc_init(struct eiointc_priv *priv, int parent_irq,
          u64 node_map)
{
 int i, val;

 node_map = node_map ? node_map : -1ULL;
 for_each_possible_cpu(i) {
  if (node_map & (1ULL << (cpu_to_eio_node(i)))) {
   node_set(cpu_to_eio_node(i), priv->node_map);
   cpumask_or(&priv->cpuspan_map, &priv->cpuspan_map,
       cpumask_of(i));
  }
 }

 priv->eiointc_domain = irq_domain_create_linear(priv->domain_handle,
       priv->vec_count,
       &eiointc_domain_ops,
       priv);
 if (!priv->eiointc_domain) {
  pr_err("loongson-extioi: cannot add IRQ domain\n");
  return -ENOMEM;
 }

 if (kvm_para_has_feature(KVM_FEATURE_VIRT_EXTIOI)) {
  val = iocsr_read32(EXTIOI_VIRT_FEATURES);
  /*
 * With EXTIOI_ENABLE_CPU_ENCODE set
 * interrupts can route to 256 vCPUs.
 */

  if (val & EXTIOI_HAS_CPU_ENCODE) {
   val = iocsr_read32(EXTIOI_VIRT_CONFIG);
   val |= EXTIOI_ENABLE_CPU_ENCODE;
   iocsr_write32(val, EXTIOI_VIRT_CONFIG);
   priv->flags = EIOINTC_USE_CPU_ENCODE;
  }
 }

 eiointc_priv[nr_pics++] = priv;
 eiointc_router_init(0);
 irq_set_chained_handler_and_data(parent_irq, eiointc_irq_dispatch, priv);

 if (nr_pics == 1) {
  register_syscore_ops(&eiointc_syscore_ops);
  cpuhp_setup_state_nocalls(CPUHP_AP_IRQ_EIOINTC_STARTING,
       "irqchip/loongarch/eiointc:starting",
       eiointc_router_init, NULL);
 }

 return 0;
}

int __init eiointc_acpi_init(struct irq_domain *parent,
         struct acpi_madt_eio_pic *acpi_eiointc)
{
 int parent_irq, ret;
 struct eiointc_priv *priv;
 int node;

 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
 if (!priv)
  return -ENOMEM;

 priv->domain_handle = irq_domain_alloc_named_id_fwnode("EIOPIC",
              acpi_eiointc->node);
 if (!priv->domain_handle) {
  pr_err("Unable to allocate domain handle\n");
  goto out_free_priv;
 }

 priv->vec_count = VEC_COUNT;
 priv->node = acpi_eiointc->node;
 priv->parent_hwirq = acpi_eiointc->cascade;
 parent_irq = irq_create_mapping(parent, acpi_eiointc->cascade);

 ret = eiointc_init(priv, parent_irq, acpi_eiointc->node_map);
 if (ret < 0)
  goto out_free_handle;

 if (cpu_has_flatmode)
  node = early_cpu_to_node(acpi_eiointc->node * CORES_PER_EIO_NODE);
 else
  node = acpi_eiointc->node;
 acpi_set_vec_parent(node, priv->eiointc_domain, pch_group);
 acpi_set_vec_parent(node, priv->eiointc_domain, msi_group);

 ret = acpi_cascade_irqdomain_init();
 if (ret < 0)
  goto out_free_handle;

 return ret;

out_free_handle:
 irq_domain_free_fwnode(priv->domain_handle);
 priv->domain_handle = NULL;
out_free_priv:
 kfree(priv);

 return -ENOMEM;
}

static int __init eiointc_of_init(struct device_node *of_node,
      struct device_node *parent)
{
 struct eiointc_priv *priv;
 struct irq_data *irq_data;
 int parent_irq, ret;

 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
 if (!priv)
  return -ENOMEM;

 parent_irq = irq_of_parse_and_map(of_node, 0);
 if (parent_irq <= 0) {
  ret = -ENODEV;
  goto out_free_priv;
 }

 ret = irq_set_handler_data(parent_irq, priv);
 if (ret < 0)
  goto out_free_priv;

 irq_data = irq_get_irq_data(parent_irq);
 if (!irq_data) {
  ret = -ENODEV;
  goto out_free_priv;
 }

 /*
 * In particular, the number of devices supported by the LS2K0500
 * extended I/O interrupt vector is 128.
 */

 if (of_device_is_compatible(of_node, "loongson,ls2k0500-eiointc"))
  priv->vec_count = 128;
 else
  priv->vec_count = VEC_COUNT;
 priv->parent_hwirq = irqd_to_hwirq(irq_data);
 priv->node = 0;
 priv->domain_handle = of_fwnode_handle(of_node);

 ret = eiointc_init(priv, parent_irq, 0);
 if (ret < 0)
  goto out_free_priv;

 return 0;

out_free_priv:
 kfree(priv);
 return ret;
}

IRQCHIP_DECLARE(loongson_ls2k0500_eiointc, "loongson,ls2k0500-eiointc", eiointc_of_init);
IRQCHIP_DECLARE(loongson_ls2k2000_eiointc, "loongson,ls2k2000-eiointc", eiointc_of_init);

Messung V0.5
C=95 H=94 G=94

¤ Dauer der Verarbeitung: 0.12 Sekunden  (vorverarbeitet)  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.