cregit-Linux how code gets into the kernel

Release 4.14 arch/x86/platform/uv/uv_irq.c

/*
 * This file is subject to the terms and conditions of the GNU General Public
 * License.  See the file "COPYING" in the main directory of this archive
 * for more details.
 *
 * SGI UV IRQ functions
 *
 * Copyright (C) 2008 Silicon Graphics, Inc. All rights reserved.
 */

#include <linux/export.h>
#include <linux/rbtree.h>
#include <linux/slab.h>
#include <linux/irq.h>

#include <asm/irqdomain.h>
#include <asm/apic.h>
#include <asm/uv/uv_irq.h>
#include <asm/uv/uv_hub.h>

/* MMR offset and pnode of hub sourcing interrupts for a given irq */

struct uv_irq_2_mmr_pnode {
	
unsigned long		offset;
	
int			pnode;
};


static void uv_program_mmr(struct irq_cfg *cfg, struct uv_irq_2_mmr_pnode *info) { unsigned long mmr_value; struct uv_IO_APIC_route_entry *entry; BUILD_BUG_ON(sizeof(struct uv_IO_APIC_route_entry) != sizeof(unsigned long)); mmr_value = 0; entry = (struct uv_IO_APIC_route_entry *)&mmr_value; entry->vector = cfg->vector; entry->delivery_mode = apic->irq_delivery_mode; entry->dest_mode = apic->irq_dest_mode; entry->polarity = 0; entry->trigger = 0; entry->mask = 0; entry->dest = cfg->dest_apicid; uv_write_global_mmr64(info->pnode, info->offset, mmr_value); }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu10993.16%133.33%
Dimitri Sivanich65.13%133.33%
Thomas Gleixner21.71%133.33%
Total117100.00%3100.00%


static void uv_noop(struct irq_data *data) { }

Contributors

PersonTokensPropCommitsCommitProp
Dean Nelson660.00%150.00%
Thomas Gleixner440.00%150.00%
Total10100.00%2100.00%


static void uv_ack_apic(struct irq_data *data) { ack_APIC_irq(); }

Contributors

PersonTokensPropCommitsCommitProp
Dean Nelson1071.43%150.00%
Thomas Gleixner428.57%150.00%
Total14100.00%2100.00%


static int uv_set_irq_affinity(struct irq_data *data, const struct cpumask *mask, bool force) { struct irq_data *parent = data->parent_data; struct irq_cfg *cfg = irqd_cfg(data); int ret; ret = parent->chip->irq_set_affinity(parent, mask, force); if (ret >= 0) { uv_program_mmr(cfg, data->chip_data); send_cleanup_vector(cfg); } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu82100.00%1100.00%
Total82100.00%1100.00%

static struct irq_chip uv_irq_chip = { .name = "UV-CORE", .irq_mask = uv_noop, .irq_unmask = uv_noop, .irq_eoi = uv_ack_apic, .irq_set_affinity = uv_set_irq_affinity, };
static int uv_domain_alloc(struct irq_domain *domain, unsigned int virq, unsigned int nr_irqs, void *arg) { struct uv_irq_2_mmr_pnode *chip_data; struct irq_alloc_info *info = arg; struct irq_data *irq_data = irq_domain_get_irq_data(domain, virq); int ret; if (nr_irqs > 1 || !info || info->type != X86_IRQ_ALLOC_TYPE_UV) return -EINVAL; chip_data = kmalloc_node(sizeof(*chip_data), GFP_KERNEL, irq_data_get_node(irq_data)); if (!chip_data) return -ENOMEM; ret = irq_domain_alloc_irqs_parent(domain, virq, nr_irqs, arg); if (ret >= 0) { if (info->uv_limit == UV_AFFINITY_CPU) irq_set_status_flags(virq, IRQ_NO_BALANCING); else irq_set_status_flags(virq, IRQ_MOVE_PCNTXT); chip_data->pnode = uv_blade_to_pnode(info->uv_blade); chip_data->offset = info->uv_offset; irq_domain_set_info(domain, virq, virq, &uv_irq_chip, chip_data, handle_percpu_irq, NULL, info->uv_name); } else { kfree(chip_data); } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu11459.38%266.67%
Dimitri Sivanich7840.62%133.33%
Total192100.00%3100.00%


static void uv_domain_free(struct irq_domain *domain, unsigned int virq, unsigned int nr_irqs) { struct irq_data *irq_data = irq_domain_get_irq_data(domain, virq); BUG_ON(nr_irqs != 1); kfree(irq_data->chip_data); irq_clear_status_flags(virq, IRQ_MOVE_PCNTXT); irq_clear_status_flags(virq, IRQ_NO_BALANCING); irq_domain_free_irqs_top(domain, virq, nr_irqs); }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu4160.29%150.00%
Dimitri Sivanich2739.71%150.00%
Total68100.00%2100.00%

/* * Re-target the irq to the specified CPU and enable the specified MMR located * on the specified blade to allow the sending of MSIs to the specified CPU. */
static void uv_domain_activate(struct irq_domain *domain, struct irq_data *irq_data) { uv_program_mmr(irqd_cfg(irq_data), irq_data->chip_data); }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu1553.57%250.00%
Dimitri Sivanich1035.71%125.00%
Thomas Gleixner310.71%125.00%
Total28100.00%4100.00%

/* * Disable the specified MMR located on the specified blade so that MSIs are * longer allowed to be sent. */
static void uv_domain_deactivate(struct irq_domain *domain, struct irq_data *irq_data) { unsigned long mmr_value; struct uv_IO_APIC_route_entry *entry; mmr_value = 0; entry = (struct uv_IO_APIC_route_entry *)&mmr_value; entry->mask = 1; uv_program_mmr(irqd_cfg(irq_data), irq_data->chip_data); }

Contributors

PersonTokensPropCommitsCommitProp
Dimitri Sivanich3561.40%133.33%
Jiang Liu2035.09%133.33%
Thomas Gleixner23.51%133.33%
Total57100.00%3100.00%

static const struct irq_domain_ops uv_domain_ops = { .alloc = uv_domain_alloc, .free = uv_domain_free, .activate = uv_domain_activate, .deactivate = uv_domain_deactivate, };
static struct irq_domain *uv_get_irq_domain(void) { static struct irq_domain *uv_domain; static DEFINE_MUTEX(uv_lock); struct fwnode_handle *fn; mutex_lock(&uv_lock); if (uv_domain) goto out; fn = irq_domain_alloc_named_fwnode("UV-CORE"); if (!fn) goto out; uv_domain = irq_domain_create_tree(fn, &uv_domain_ops, NULL); irq_domain_free_fwnode(fn); if (uv_domain) uv_domain->parent = x86_vector_domain; out: mutex_unlock(&uv_lock); return uv_domain; }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu4245.16%133.33%
Thomas Gleixner3234.41%133.33%
Dimitri Sivanich1920.43%133.33%
Total93100.00%3100.00%

/* * Set up a mapping of an available irq and vector, and enable the specified * MMR that defines the MSI that is to be sent to the specified CPU when an * interrupt is raised. */
int uv_setup_irq(char *irq_name, int cpu, int mmr_blade, unsigned long mmr_offset, int limit) { struct irq_alloc_info info; struct irq_domain *domain = uv_get_irq_domain(); if (!domain) return -ENOMEM; init_irq_alloc_info(&info, cpumask_of(cpu)); info.type = X86_IRQ_ALLOC_TYPE_UV; info.uv_limit = limit; info.uv_blade = mmr_blade; info.uv_offset = mmr_offset; info.uv_name = irq_name; return irq_domain_alloc_irqs(domain, 1, uv_blade_to_memory_nid(mmr_blade), &info); }

Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu6666.00%240.00%
Dean Nelson2525.00%120.00%
Dimitri Sivanich88.00%120.00%
Randy Dunlap11.00%120.00%
Total100100.00%5100.00%

EXPORT_SYMBOL_GPL(uv_setup_irq); /* * Tear down a mapping of an irq and vector, and disable the specified MMR that * defined the MSI that was to be sent to the specified CPU when an interrupt * was raised. * * Set mmr_blade and mmr_offset to what was passed in on uv_setup_irq(). */
void uv_teardown_irq(unsigned int irq) { irq_domain_free_irqs(irq, 1); }

Contributors

PersonTokensPropCommitsCommitProp
Dean Nelson1168.75%133.33%
Jiang Liu318.75%133.33%
Dimitri Sivanich212.50%133.33%
Total16100.00%3100.00%

EXPORT_SYMBOL_GPL(uv_teardown_irq);

Overall Contributors

PersonTokensPropCommitsCommitProp
Jiang Liu52158.61%428.57%
Dimitri Sivanich20923.51%214.29%
Dean Nelson9811.02%17.14%
Thomas Gleixner525.85%321.43%
Tejun Heo30.34%17.14%
Ingo Molnar30.34%17.14%
Randy Dunlap20.22%17.14%
Paul Gortmaker10.11%17.14%
Total889100.00%14100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.