2013-01-22 04:36:12 +04:00
/*
* Copyright ( C ) 2012 ARM Ltd .
* Author : Marc Zyngier < marc . zyngier @ arm . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# ifndef __ASM_ARM_KVM_VGIC_H
# define __ASM_ARM_KVM_VGIC_H
2013-01-22 04:36:14 +04:00
# include <linux/kernel.h>
# include <linux/kvm.h>
# include <linux/irqreturn.h>
# include <linux/spinlock.h>
# include <linux/types.h>
2015-03-26 17:39:34 +03:00
# include <kvm/iodev.h>
2013-01-22 04:36:12 +04:00
2014-07-08 15:09:05 +04:00
# define VGIC_NR_IRQS_LEGACY 256
2013-01-22 04:36:14 +04:00
# define VGIC_NR_SGIS 16
# define VGIC_NR_PPIS 16
# define VGIC_NR_PRIVATE_IRQS (VGIC_NR_SGIS + VGIC_NR_PPIS)
2014-02-04 22:13:03 +04:00
# define VGIC_V2_MAX_LRS (1 << 6)
2013-07-12 18:15:23 +04:00
# define VGIC_V3_MAX_LRS 16
2014-07-08 15:09:04 +04:00
# define VGIC_MAX_IRQS 1024
2014-06-02 18:26:01 +04:00
# define VGIC_V2_MAX_CPUS 8
2013-01-22 04:36:14 +04:00
/* Sanity checks... */
2014-06-03 12:26:30 +04:00
# if (KVM_MAX_VCPUS > 255)
# error Too many KVM VCPUs, the VGIC only supports up to 255 VCPUs for now
2013-01-22 04:36:14 +04:00
# endif
2014-07-08 15:09:05 +04:00
# if (VGIC_NR_IRQS_LEGACY & 31)
2013-01-22 04:36:14 +04:00
# error "VGIC_NR_IRQS must be a multiple of 32"
# endif
2014-07-08 15:09:05 +04:00
# if (VGIC_NR_IRQS_LEGACY > VGIC_MAX_IRQS)
2013-01-22 04:36:14 +04:00
# error "VGIC_NR_IRQS must be <= 1024"
# endif
/*
* The GIC distributor registers describing interrupts have two parts :
* - 32 per - CPU interrupts ( SGI + PPI )
* - a bunch of shared interrupts ( SPI )
*/
struct vgic_bitmap {
2014-07-08 15:09:01 +04:00
/*
* - One UL per VCPU for private interrupts ( assumes UL is at
* least 32 bits )
* - As many UL as necessary for shared interrupts .
*
* The private interrupts are accessed via the " private "
* field , one UL per vcpu ( the state for vcpu n is in
* private [ n ] ) . The shared interrupts are accessed via the
* " shared " pointer ( IRQn state is at bit n - 32 in the bitmap ) .
*/
unsigned long * private ;
unsigned long * shared ;
2013-01-22 04:36:14 +04:00
} ;
struct vgic_bytemap {
2014-07-08 15:09:01 +04:00
/*
* - 8 u32 per VCPU for private interrupts
* - As many u32 as necessary for shared interrupts .
*
* The private interrupts are accessed via the " private "
* field , ( the state for vcpu n is in private [ n * 8 ] to
* private [ n * 8 + 7 ] ) . The shared interrupts are accessed via
* the " shared " pointer ( IRQn state is at byte ( n - 32 ) % 4 of the
* shared [ ( n - 32 ) / 4 ] word ) .
*/
u32 * private ;
u32 * shared ;
2013-01-22 04:36:14 +04:00
} ;
2013-06-03 18:55:02 +04:00
struct kvm_vcpu ;
2013-06-21 14:57:56 +04:00
enum vgic_type {
VGIC_V2 , /* Good ol' GICv2 */
2013-07-12 18:15:23 +04:00
VGIC_V3 , /* New fancy GICv3 */
2013-06-21 14:57:56 +04:00
} ;
2013-06-03 18:55:02 +04:00
# define LR_STATE_PENDING (1 << 0)
# define LR_STATE_ACTIVE (1 << 1)
# define LR_STATE_MASK (3 << 0)
# define LR_EOI_INT (1 << 2)
2015-06-08 17:21:32 +03:00
# define LR_HW (1 << 3)
2013-06-03 18:55:02 +04:00
struct vgic_lr {
2015-06-08 17:21:32 +03:00
unsigned irq : 10 ;
union {
unsigned hwirq : 10 ;
unsigned source : 3 ;
} ;
unsigned state : 4 ;
2013-06-03 18:55:02 +04:00
} ;
2014-02-04 21:48:10 +04:00
struct vgic_vmcr {
u32 ctlr ;
u32 abpr ;
u32 bpr ;
u32 pmr ;
} ;
2013-06-03 18:55:02 +04:00
struct vgic_ops {
struct vgic_lr ( * get_lr ) ( const struct kvm_vcpu * , int ) ;
void ( * set_lr ) ( struct kvm_vcpu * , int , struct vgic_lr ) ;
2013-06-04 13:29:39 +04:00
void ( * sync_lr_elrsr ) ( struct kvm_vcpu * , int , struct vgic_lr ) ;
u64 ( * get_elrsr ) ( const struct kvm_vcpu * vcpu ) ;
2013-06-04 13:33:43 +04:00
u64 ( * get_eisr ) ( const struct kvm_vcpu * vcpu ) ;
2015-03-13 20:02:56 +03:00
void ( * clear_eisr ) ( struct kvm_vcpu * vcpu ) ;
2013-06-04 14:02:10 +04:00
u32 ( * get_interrupt_status ) ( const struct kvm_vcpu * vcpu ) ;
2013-06-04 14:24:17 +04:00
void ( * enable_underflow ) ( struct kvm_vcpu * vcpu ) ;
void ( * disable_underflow ) ( struct kvm_vcpu * vcpu ) ;
2014-02-04 21:48:10 +04:00
void ( * get_vmcr ) ( struct kvm_vcpu * vcpu , struct vgic_vmcr * vmcr ) ;
void ( * set_vmcr ) ( struct kvm_vcpu * vcpu , struct vgic_vmcr * vmcr ) ;
2013-06-04 14:36:38 +04:00
void ( * enable ) ( struct kvm_vcpu * vcpu ) ;
2013-06-03 18:55:02 +04:00
} ;
2013-06-18 22:17:28 +04:00
struct vgic_params {
2013-06-21 14:57:56 +04:00
/* vgic type */
enum vgic_type type ;
2013-06-18 22:17:28 +04:00
/* Physical address of vgic virtual cpu interface */
phys_addr_t vcpu_base ;
/* Number of list registers */
u32 nr_lr ;
/* Interrupt number */
unsigned int maint_irq ;
/* Virtual control interface base address */
void __iomem * vctrl_base ;
2014-06-02 18:26:01 +04:00
int max_gic_vcpus ;
2014-06-03 12:26:03 +04:00
/* Only needed for the legacy KVM_CREATE_IRQCHIP */
bool can_emulate_gicv2 ;
2013-06-18 22:17:28 +04:00
} ;
2014-06-02 18:19:12 +04:00
struct vgic_vm_ops {
bool ( * queue_sgi ) ( struct kvm_vcpu * , int irq ) ;
void ( * add_sgi_source ) ( struct kvm_vcpu * , int irq , int source ) ;
int ( * init_model ) ( struct kvm * ) ;
int ( * map_resources ) ( struct kvm * , const struct vgic_params * ) ;
} ;
2015-03-26 17:39:34 +03:00
struct vgic_io_device {
gpa_t addr ;
int len ;
const struct vgic_io_range * reg_ranges ;
struct kvm_vcpu * redist_vcpu ;
struct kvm_io_device dev ;
} ;
2014-06-23 20:37:18 +04:00
struct irq_phys_map {
u32 virt_irq ;
u32 phys_irq ;
u32 irq ;
bool active ;
} ;
struct irq_phys_map_entry {
struct list_head entry ;
struct rcu_head rcu ;
struct irq_phys_map map ;
} ;
2013-01-22 04:36:12 +04:00
struct vgic_dist {
2013-01-22 04:36:14 +04:00
spinlock_t lock ;
2014-05-15 13:03:25 +04:00
bool in_kernel ;
2013-01-22 04:36:16 +04:00
bool ready ;
2013-01-22 04:36:14 +04:00
2014-06-03 11:33:10 +04:00
/* vGIC model the kernel emulates for the guest (GICv2 or GICv3) */
u32 vgic_model ;
2014-07-08 15:09:01 +04:00
int nr_cpus ;
int nr_irqs ;
2013-01-22 04:36:14 +04:00
/* Virtual control interface mapping */
void __iomem * vctrl_base ;
2013-01-22 04:36:13 +04:00
/* Distributor and vcpu interface mapping in the guest */
phys_addr_t vgic_dist_base ;
2014-06-07 02:54:51 +04:00
/* GICv2 and GICv3 use different mapped register blocks */
union {
phys_addr_t vgic_cpu_base ;
phys_addr_t vgic_redist_base ;
} ;
2013-01-22 04:36:14 +04:00
/* Distributor enabled */
u32 enabled ;
/* Interrupt enabled (one bit per IRQ) */
struct vgic_bitmap irq_enabled ;
2014-06-14 23:54:51 +04:00
/* Level-triggered interrupt external input is asserted */
struct vgic_bitmap irq_level ;
/*
* Interrupt state is pending on the distributor
*/
2014-06-09 14:27:18 +04:00
struct vgic_bitmap irq_pending ;
2013-01-22 04:36:14 +04:00
2014-06-14 23:54:51 +04:00
/*
* Tracks writes to GICD_ISPENDRn and GICD_ICPENDRn for level - triggered
* interrupts . Essentially holds the state of the flip - flop in
* Figure 4 - 10 on page 4 - 101 in ARM IHI 004 8 B . b .
* Once set , it is only cleared for level - triggered interrupts on
* guest ACKs ( when we queue it ) or writes to GICD_ICPENDRn .
*/
struct vgic_bitmap irq_soft_pend ;
2014-06-09 14:55:13 +04:00
/* Level-triggered interrupt queued on VCPU interface */
struct vgic_bitmap irq_queued ;
2013-01-22 04:36:14 +04:00
2015-03-13 20:02:54 +03:00
/* Interrupt was active when unqueue from VCPU interface */
struct vgic_bitmap irq_active ;
2013-01-22 04:36:14 +04:00
/* Interrupt priority. Not used yet. */
struct vgic_bytemap irq_priority ;
/* Level/edge triggered */
struct vgic_bitmap irq_cfg ;
2014-07-08 15:09:01 +04:00
/*
* Source CPU per SGI and target CPU :
*
* Each byte represent a SGI observable on a VCPU , each bit of
* this byte indicating if the corresponding VCPU has
* generated this interrupt . This is a GICv2 feature only .
*
* For VCPUn ( n < 8 ) , irq_sgi_sources [ n * 16 ] to [ n * 16 + 15 ] are
* the SGIs observable on VCPUn .
*/
u8 * irq_sgi_sources ;
2013-01-22 04:36:14 +04:00
2014-07-08 15:09:01 +04:00
/*
* Target CPU for each SPI :
*
* Array of available SPI , each byte indicating the target
* VCPU for SPI . IRQn ( n > = 32 ) is at irq_spi_cpu [ n - 32 ] .
*/
u8 * irq_spi_cpu ;
/*
* Reverse lookup of irq_spi_cpu for faster compute pending :
*
* Array of bitmaps , one per VCPU , describing if IRQn is
* routed to a particular VCPU .
*/
struct vgic_bitmap * irq_spi_target ;
2013-01-22 04:36:14 +04:00
2014-06-07 02:54:51 +04:00
/* Target MPIDR for each IRQ (needed for GICv3 IROUTERn) only */
u32 * irq_spi_mpidr ;
2013-01-22 04:36:14 +04:00
/* Bitmap indicating which CPU has something pending */
2014-07-08 15:09:01 +04:00
unsigned long * irq_pending_on_cpu ;
2014-06-02 18:19:12 +04:00
2015-03-13 20:02:54 +03:00
/* Bitmap indicating which CPU has active IRQs */
unsigned long * irq_active_on_cpu ;
2014-06-02 18:19:12 +04:00
struct vgic_vm_ops vm_ops ;
2015-03-26 17:39:35 +03:00
struct vgic_io_device dist_iodev ;
2015-03-26 17:39:37 +03:00
struct vgic_io_device * redist_iodevs ;
2014-06-23 20:37:18 +04:00
/* Virtual irq to hwirq mapping */
spinlock_t irq_phys_map_lock ;
struct list_head irq_phys_map_list ;
2013-01-22 04:36:12 +04:00
} ;
2013-05-30 13:20:36 +04:00
struct vgic_v2_cpu_if {
u32 vgic_hcr ;
u32 vgic_vmcr ;
u32 vgic_misr ; /* Saved only */
2014-09-28 18:04:26 +04:00
u64 vgic_eisr ; /* Saved only */
u64 vgic_elrsr ; /* Saved only */
2013-05-30 13:20:36 +04:00
u32 vgic_apr ;
2014-02-04 22:13:03 +04:00
u32 vgic_lr [ VGIC_V2_MAX_LRS ] ;
2013-05-30 13:20:36 +04:00
} ;
2013-07-12 18:15:23 +04:00
struct vgic_v3_cpu_if {
# ifdef CONFIG_ARM_GIC_V3
u32 vgic_hcr ;
u32 vgic_vmcr ;
2014-06-03 10:58:15 +04:00
u32 vgic_sre ; /* Restored only, change ignored */
2013-07-12 18:15:23 +04:00
u32 vgic_misr ; /* Saved only */
u32 vgic_eisr ; /* Saved only */
u32 vgic_elrsr ; /* Saved only */
u32 vgic_ap0r [ 4 ] ;
u32 vgic_ap1r [ 4 ] ;
u64 vgic_lr [ VGIC_V3_MAX_LRS ] ;
# endif
} ;
2013-01-22 04:36:12 +04:00
struct vgic_cpu {
2013-01-22 04:36:14 +04:00
/* per IRQ to LR mapping */
2014-07-08 15:09:01 +04:00
u8 * vgic_irq_lr_map ;
2013-01-22 04:36:14 +04:00
2015-03-13 20:02:54 +03:00
/* Pending/active/both interrupts on this VCPU */
2013-01-22 04:36:14 +04:00
DECLARE_BITMAP ( pending_percpu , VGIC_NR_PRIVATE_IRQS ) ;
2015-03-13 20:02:54 +03:00
DECLARE_BITMAP ( active_percpu , VGIC_NR_PRIVATE_IRQS ) ;
DECLARE_BITMAP ( pend_act_percpu , VGIC_NR_PRIVATE_IRQS ) ;
/* Pending/active/both shared interrupts, dynamically sized */
2014-07-08 15:09:01 +04:00
unsigned long * pending_shared ;
2015-03-13 20:02:54 +03:00
unsigned long * active_shared ;
unsigned long * pend_act_shared ;
2013-01-22 04:36:14 +04:00
/* Bitmap of used/free list registers */
2014-02-04 22:13:03 +04:00
DECLARE_BITMAP ( lr_used , VGIC_V2_MAX_LRS ) ;
2013-01-22 04:36:14 +04:00
/* Number of list registers on this CPU */
int nr_lr ;
/* CPU vif control registers for world switch */
2013-05-30 13:20:36 +04:00
union {
struct vgic_v2_cpu_if vgic_v2 ;
2013-07-12 18:15:23 +04:00
struct vgic_v3_cpu_if vgic_v3 ;
2013-05-30 13:20:36 +04:00
} ;
2014-06-23 20:37:18 +04:00
/* Protected by the distributor's irq_phys_map_lock */
struct list_head irq_phys_map_list ;
2013-01-22 04:36:12 +04:00
} ;
2013-01-22 04:36:14 +04:00
# define LR_EMPTY 0xff
2013-06-04 14:02:10 +04:00
# define INT_STATUS_EOI (1 << 0)
# define INT_STATUS_UNDERFLOW (1 << 1)
2013-01-22 04:36:12 +04:00
struct kvm ;
struct kvm_vcpu ;
2013-09-24 01:55:56 +04:00
int kvm_vgic_addr ( struct kvm * kvm , unsigned long type , u64 * addr , bool write ) ;
2013-01-22 04:36:16 +04:00
int kvm_vgic_hyp_init ( void ) ;
2014-12-04 18:02:24 +03:00
int kvm_vgic_map_resources ( struct kvm * kvm ) ;
2014-06-02 18:26:01 +04:00
int kvm_vgic_get_max_vcpus ( void ) ;
2014-06-23 20:37:18 +04:00
void kvm_vgic_early_init ( struct kvm * kvm ) ;
2014-06-03 11:33:10 +04:00
int kvm_vgic_create ( struct kvm * kvm , u32 type ) ;
2014-07-08 15:09:01 +04:00
void kvm_vgic_destroy ( struct kvm * kvm ) ;
2014-06-23 20:37:18 +04:00
void kvm_vgic_vcpu_early_init ( struct kvm_vcpu * vcpu ) ;
2014-07-08 15:09:01 +04:00
void kvm_vgic_vcpu_destroy ( struct kvm_vcpu * vcpu ) ;
2013-01-22 04:36:14 +04:00
void kvm_vgic_flush_hwstate ( struct kvm_vcpu * vcpu ) ;
void kvm_vgic_sync_hwstate ( struct kvm_vcpu * vcpu ) ;
2013-01-22 04:36:15 +04:00
int kvm_vgic_inject_irq ( struct kvm * kvm , int cpuid , unsigned int irq_num ,
bool level ) ;
2015-07-24 13:30:43 +03:00
int kvm_vgic_inject_mapped_irq ( struct kvm * kvm , int cpuid ,
struct irq_phys_map * map , bool level ) ;
2014-06-03 12:13:13 +04:00
void vgic_v3_dispatch_sgi ( struct kvm_vcpu * vcpu , u64 reg ) ;
2013-01-22 04:36:14 +04:00
int kvm_vgic_vcpu_pending_irq ( struct kvm_vcpu * vcpu ) ;
2015-03-13 20:02:54 +03:00
int kvm_vgic_vcpu_active_irq ( struct kvm_vcpu * vcpu ) ;
2014-06-23 20:37:18 +04:00
struct irq_phys_map * kvm_vgic_map_phys_irq ( struct kvm_vcpu * vcpu ,
int virt_irq , int irq ) ;
int kvm_vgic_unmap_phys_irq ( struct kvm_vcpu * vcpu , struct irq_phys_map * map ) ;
2015-06-08 18:13:30 +03:00
bool kvm_vgic_get_phys_irq_active ( struct irq_phys_map * map ) ;
void kvm_vgic_set_phys_irq_active ( struct irq_phys_map * map , bool active ) ;
2013-01-22 04:36:12 +04:00
2014-05-15 13:03:25 +04:00
# define irqchip_in_kernel(k) (!!((k)->arch.vgic.in_kernel))
2014-12-09 16:30:36 +03:00
# define vgic_initialized(k) (!!((k)->arch.vgic.nr_cpus))
2014-12-09 16:28:09 +03:00
# define vgic_ready(k) ((k)->arch.vgic.ready)
2013-01-22 04:36:14 +04:00
2014-02-04 22:13:03 +04:00
int vgic_v2_probe ( struct device_node * vgic_node ,
const struct vgic_ops * * ops ,
const struct vgic_params * * params ) ;
2013-07-12 18:15:23 +04:00
# ifdef CONFIG_ARM_GIC_V3
int vgic_v3_probe ( struct device_node * vgic_node ,
const struct vgic_ops * * ops ,
const struct vgic_params * * params ) ;
# else
static inline int vgic_v3_probe ( struct device_node * vgic_node ,
const struct vgic_ops * * ops ,
const struct vgic_params * * params )
{
return - ENODEV ;
}
# endif
2014-02-04 22:13:03 +04:00
2013-01-22 04:36:12 +04:00
# endif