162 lines
3.2 KiB
C
162 lines
3.2 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
/*
|
||
|
* ARM Generic Interrupt Controller (GIC) support
|
||
|
*/
|
||
|
|
||
|
#include <errno.h>
|
||
|
#include <linux/bits.h>
|
||
|
#include <linux/sizes.h>
|
||
|
|
||
|
#include "kvm_util.h"
|
||
|
|
||
|
#include <gic.h>
|
||
|
#include "gic_private.h"
|
||
|
#include "processor.h"
|
||
|
#include "spinlock.h"
|
||
|
|
||
|
static const struct gic_common_ops *gic_common_ops;
|
||
|
static struct spinlock gic_lock;
|
||
|
|
||
|
static void gic_cpu_init(unsigned int cpu, void *redist_base)
|
||
|
{
|
||
|
gic_common_ops->gic_cpu_init(cpu, redist_base);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
gic_dist_init(enum gic_type type, unsigned int nr_cpus, void *dist_base)
|
||
|
{
|
||
|
const struct gic_common_ops *gic_ops = NULL;
|
||
|
|
||
|
spin_lock(&gic_lock);
|
||
|
|
||
|
/* Distributor initialization is needed only once per VM */
|
||
|
if (gic_common_ops) {
|
||
|
spin_unlock(&gic_lock);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (type == GIC_V3)
|
||
|
gic_ops = &gicv3_ops;
|
||
|
|
||
|
GUEST_ASSERT(gic_ops);
|
||
|
|
||
|
gic_ops->gic_init(nr_cpus, dist_base);
|
||
|
gic_common_ops = gic_ops;
|
||
|
|
||
|
/* Make sure that the initialized data is visible to all the vCPUs */
|
||
|
dsb(sy);
|
||
|
|
||
|
spin_unlock(&gic_lock);
|
||
|
}
|
||
|
|
||
|
void gic_init(enum gic_type type, unsigned int nr_cpus,
|
||
|
void *dist_base, void *redist_base)
|
||
|
{
|
||
|
uint32_t cpu = guest_get_vcpuid();
|
||
|
|
||
|
GUEST_ASSERT(type < GIC_TYPE_MAX);
|
||
|
GUEST_ASSERT(dist_base);
|
||
|
GUEST_ASSERT(redist_base);
|
||
|
GUEST_ASSERT(nr_cpus);
|
||
|
|
||
|
gic_dist_init(type, nr_cpus, dist_base);
|
||
|
gic_cpu_init(cpu, redist_base);
|
||
|
}
|
||
|
|
||
|
void gic_irq_enable(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_enable(intid);
|
||
|
}
|
||
|
|
||
|
void gic_irq_disable(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_disable(intid);
|
||
|
}
|
||
|
|
||
|
unsigned int gic_get_and_ack_irq(void)
|
||
|
{
|
||
|
uint64_t irqstat;
|
||
|
unsigned int intid;
|
||
|
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
|
||
|
irqstat = gic_common_ops->gic_read_iar();
|
||
|
intid = irqstat & GENMASK(23, 0);
|
||
|
|
||
|
return intid;
|
||
|
}
|
||
|
|
||
|
void gic_set_eoi(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_write_eoir(intid);
|
||
|
}
|
||
|
|
||
|
void gic_set_dir(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_write_dir(intid);
|
||
|
}
|
||
|
|
||
|
void gic_set_eoi_split(bool split)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_set_eoi_split(split);
|
||
|
}
|
||
|
|
||
|
void gic_set_priority_mask(uint64_t pmr)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_set_priority_mask(pmr);
|
||
|
}
|
||
|
|
||
|
void gic_set_priority(unsigned int intid, unsigned int prio)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_set_priority(intid, prio);
|
||
|
}
|
||
|
|
||
|
void gic_irq_set_active(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_set_active(intid);
|
||
|
}
|
||
|
|
||
|
void gic_irq_clear_active(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_clear_active(intid);
|
||
|
}
|
||
|
|
||
|
bool gic_irq_get_active(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
return gic_common_ops->gic_irq_get_active(intid);
|
||
|
}
|
||
|
|
||
|
void gic_irq_set_pending(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_set_pending(intid);
|
||
|
}
|
||
|
|
||
|
void gic_irq_clear_pending(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_clear_pending(intid);
|
||
|
}
|
||
|
|
||
|
bool gic_irq_get_pending(unsigned int intid)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
return gic_common_ops->gic_irq_get_pending(intid);
|
||
|
}
|
||
|
|
||
|
void gic_irq_set_config(unsigned int intid, bool is_edge)
|
||
|
{
|
||
|
GUEST_ASSERT(gic_common_ops);
|
||
|
gic_common_ops->gic_irq_set_config(intid, is_edge);
|
||
|
}
|