2016-07-15 14:43:30 +03:00
/*
* GICv3 ITS emulation
*
* Copyright ( C ) 2015 , 2016 ARM Ltd .
* Author : Andre Przywara < andre . przywara @ 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 , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/cpu.h>
# include <linux/kvm.h>
# include <linux/kvm_host.h>
# include <linux/interrupt.h>
2016-07-15 14:43:32 +03:00
# include <linux/list.h>
2016-07-15 14:43:31 +03:00
# include <linux/uaccess.h>
2017-01-09 18:27:07 +03:00
# include <linux/list_sort.h>
2016-07-15 14:43:30 +03:00
# include <linux/irqchip/arm-gic-v3.h>
# include <asm/kvm_emulate.h>
# include <asm/kvm_arm.h>
# include <asm/kvm_mmu.h>
# include "vgic.h"
# include "vgic-mmio.h"
2017-04-13 10:06:20 +03:00
static int vgic_its_save_tables_v0 ( struct vgic_its * its ) ;
static int vgic_its_restore_tables_v0 ( struct vgic_its * its ) ;
static int vgic_its_commit_v0 ( struct vgic_its * its ) ;
2017-05-04 12:36:32 +03:00
static int update_lpi_config ( struct kvm * kvm , struct vgic_irq * irq ,
struct kvm_vcpu * filter_vcpu ) ;
2017-04-13 10:06:20 +03:00
2016-07-15 14:43:36 +03:00
/*
* Creates a new ( reference to a ) struct vgic_irq for a given LPI .
* If this LPI is already mapped on another ITS , we increase its refcount
* and return a pointer to the existing structure .
* If this is a " new " LPI , we allocate and initialize a new struct vgic_irq .
* This function returns a pointer to the _unlocked_ structure .
*/
2017-05-04 12:36:32 +03:00
static struct vgic_irq * vgic_add_lpi ( struct kvm * kvm , u32 intid ,
struct kvm_vcpu * vcpu )
2016-07-15 14:43:36 +03:00
{
struct vgic_dist * dist = & kvm - > arch . vgic ;
struct vgic_irq * irq = vgic_get_irq ( kvm , NULL , intid ) , * oldirq ;
2017-05-04 12:36:32 +03:00
int ret ;
2016-07-15 14:43:36 +03:00
/* In this case there is no put, since we keep the reference. */
if ( irq )
return irq ;
irq = kzalloc ( sizeof ( struct vgic_irq ) , GFP_KERNEL ) ;
if ( ! irq )
2016-08-01 21:25:33 +03:00
return ERR_PTR ( - ENOMEM ) ;
2016-07-15 14:43:36 +03:00
INIT_LIST_HEAD ( & irq - > lpi_list ) ;
INIT_LIST_HEAD ( & irq - > ap_list ) ;
spin_lock_init ( & irq - > irq_lock ) ;
irq - > config = VGIC_CONFIG_EDGE ;
kref_init ( & irq - > refcount ) ;
irq - > intid = intid ;
2017-05-04 12:36:32 +03:00
irq - > target_vcpu = vcpu ;
2016-07-15 14:43:36 +03:00
spin_lock ( & dist - > lpi_list_lock ) ;
/*
* There could be a race with another vgic_add_lpi ( ) , so we need to
* check that we don ' t add a second list entry with the same LPI .
*/
list_for_each_entry ( oldirq , & dist - > lpi_list_head , lpi_list ) {
if ( oldirq - > intid ! = intid )
continue ;
/* Someone was faster with adding this LPI, lets use that. */
kfree ( irq ) ;
irq = oldirq ;
/*
* This increases the refcount , the caller is expected to
* call vgic_put_irq ( ) on the returned pointer once it ' s
* finished with the IRQ .
*/
2016-07-17 13:27:23 +03:00
vgic_get_irq_kref ( irq ) ;
2016-07-15 14:43:36 +03:00
goto out_unlock ;
}
list_add_tail ( & irq - > lpi_list , & dist - > lpi_list_head ) ;
dist - > lpi_list_count + + ;
out_unlock :
spin_unlock ( & dist - > lpi_list_lock ) ;
2017-05-04 12:36:32 +03:00
/*
* We " cache " the configuration table entries in our struct vgic_irq ' s .
* However we only have those structs for mapped IRQs , so we read in
* the respective config data from memory here upon mapping the LPI .
*/
ret = update_lpi_config ( kvm , irq , NULL ) ;
if ( ret )
return ERR_PTR ( ret ) ;
ret = vgic_v3_lpi_sync_pending_status ( kvm , irq ) ;
if ( ret )
return ERR_PTR ( ret ) ;
2016-07-15 14:43:36 +03:00
return irq ;
}
2016-07-15 14:43:32 +03:00
struct its_device {
struct list_head dev_list ;
/* the head for the list of ITTEs */
struct list_head itt_head ;
2016-12-22 20:14:14 +03:00
u32 num_eventid_bits ;
2017-02-02 15:45:45 +03:00
gpa_t itt_addr ;
2016-07-15 14:43:32 +03:00
u32 device_id ;
} ;
# define COLLECTION_NOT_MAPPED ((u32)~0)
struct its_collection {
struct list_head coll_list ;
u32 collection_id ;
u32 target_addr ;
} ;
# define its_is_collection_mapped(coll) ((coll) && \
( ( coll ) - > target_addr ! = COLLECTION_NOT_MAPPED ) )
2017-02-08 08:09:29 +03:00
struct its_ite {
struct list_head ite_list ;
2016-07-15 14:43:32 +03:00
2016-07-15 14:43:33 +03:00
struct vgic_irq * irq ;
2016-07-15 14:43:32 +03:00
struct its_collection * collection ;
u32 lpi ;
u32 event_id ;
} ;
2017-04-13 10:06:20 +03:00
/**
* struct vgic_its_abi - ITS abi ops and settings
* @ cte_esz : collection table entry size
* @ dte_esz : device table entry size
* @ ite_esz : interrupt translation table entry size
* @ save tables : save the ITS tables into guest RAM
* @ restore_tables : restore the ITS internal structs from tables
* stored in guest RAM
* @ commit : initialize the registers which expose the ABI settings ,
* especially the entry sizes
*/
struct vgic_its_abi {
int cte_esz ;
int dte_esz ;
int ite_esz ;
int ( * save_tables ) ( struct vgic_its * its ) ;
int ( * restore_tables ) ( struct vgic_its * its ) ;
int ( * commit ) ( struct vgic_its * its ) ;
} ;
static const struct vgic_its_abi its_table_abi_versions [ ] = {
[ 0 ] = { . cte_esz = 8 , . dte_esz = 8 , . ite_esz = 8 ,
. save_tables = vgic_its_save_tables_v0 ,
. restore_tables = vgic_its_restore_tables_v0 ,
. commit = vgic_its_commit_v0 ,
} ,
} ;
# define NR_ITS_ABIS ARRAY_SIZE(its_table_abi_versions)
inline const struct vgic_its_abi * vgic_its_get_abi ( struct vgic_its * its )
{
return & its_table_abi_versions [ its - > abi_rev ] ;
}
int vgic_its_set_abi ( struct vgic_its * its , int rev )
{
const struct vgic_its_abi * abi ;
its - > abi_rev = rev ;
abi = vgic_its_get_abi ( its ) ;
return abi - > commit ( its ) ;
}
2016-07-15 14:43:36 +03:00
/*
* Find and returns a device in the device table for an ITS .
* Must be called with the its_lock mutex held .
*/
static struct its_device * find_its_device ( struct vgic_its * its , u32 device_id )
{
struct its_device * device ;
list_for_each_entry ( device , & its - > device_list , dev_list )
if ( device_id = = device - > device_id )
return device ;
return NULL ;
}
/*
* Find and returns an interrupt translation table entry ( ITTE ) for a given
* Device ID / Event ID pair on an ITS .
* Must be called with the its_lock mutex held .
*/
2017-02-08 08:09:29 +03:00
static struct its_ite * find_ite ( struct vgic_its * its , u32 device_id ,
2016-07-15 14:43:36 +03:00
u32 event_id )
{
struct its_device * device ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
device = find_its_device ( its , device_id ) ;
if ( device = = NULL )
return NULL ;
2017-02-08 08:09:29 +03:00
list_for_each_entry ( ite , & device - > itt_head , ite_list )
if ( ite - > event_id = = event_id )
return ite ;
2016-07-15 14:43:36 +03:00
return NULL ;
}
/* To be used as an iterator this macro misses the enclosing parentheses */
2017-02-08 08:09:29 +03:00
# define for_each_lpi_its(dev, ite, its) \
2016-07-15 14:43:36 +03:00
list_for_each_entry ( dev , & ( its ) - > device_list , dev_list ) \
2017-02-08 08:09:29 +03:00
list_for_each_entry ( ite , & ( dev ) - > itt_head , ite_list )
2016-07-15 14:43:36 +03:00
2016-07-15 14:43:32 +03:00
/*
* We only implement 48 bits of PA at the moment , although the ITS
* supports more . Let ' s be restrictive here .
*/
2016-07-15 14:43:36 +03:00
# define BASER_ADDRESS(x) ((x) & GENMASK_ULL(47, 16))
2016-07-15 14:43:32 +03:00
# define CBASER_ADDRESS(x) ((x) & GENMASK_ULL(47, 12))
2016-07-15 14:43:35 +03:00
# define GIC_LPI_OFFSET 8192
2016-12-22 20:14:14 +03:00
# define VITS_TYPER_IDBITS 16
2017-02-02 16:37:33 +03:00
# define VITS_TYPER_DEVBITS 16
2017-02-08 07:20:04 +03:00
# define VITS_DTE_MAX_DEVID_OFFSET (BIT(14) - 1)
# define VITS_ITE_MAX_EVENTID_OFFSET (BIT(16) - 1)
2016-12-22 20:14:14 +03:00
2016-07-15 14:43:36 +03:00
/*
* Finds and returns a collection in the ITS collection table .
* Must be called with the its_lock mutex held .
*/
static struct its_collection * find_collection ( struct vgic_its * its , int coll_id )
{
struct its_collection * collection ;
list_for_each_entry ( collection , & its - > collection_list , coll_list ) {
if ( coll_id = = collection - > collection_id )
return collection ;
}
return NULL ;
}
2016-07-15 14:43:35 +03:00
# define LPI_PROP_ENABLE_BIT(p) ((p) & LPI_PROP_ENABLED)
# define LPI_PROP_PRIORITY(p) ((p) & 0xfc)
/*
* Reads the configuration data for a given LPI from guest memory and
* updates the fields in struct vgic_irq .
* If filter_vcpu is not NULL , applies only if the IRQ is targeting this
* VCPU . Unconditionally applies if filter_vcpu is NULL .
*/
static int update_lpi_config ( struct kvm * kvm , struct vgic_irq * irq ,
struct kvm_vcpu * filter_vcpu )
{
2017-05-04 12:19:52 +03:00
u64 propbase = GICR_PROPBASER_ADDRESS ( kvm - > arch . vgic . propbaser ) ;
2016-07-15 14:43:35 +03:00
u8 prop ;
int ret ;
ret = kvm_read_guest ( kvm , propbase + irq - > intid - GIC_LPI_OFFSET ,
& prop , 1 ) ;
if ( ret )
return ret ;
spin_lock ( & irq - > irq_lock ) ;
if ( ! filter_vcpu | | filter_vcpu = = irq - > target_vcpu ) {
irq - > priority = LPI_PROP_PRIORITY ( prop ) ;
irq - > enabled = LPI_PROP_ENABLE_BIT ( prop ) ;
vgic_queue_irq_unlock ( kvm , irq ) ;
} else {
spin_unlock ( & irq - > irq_lock ) ;
}
return 0 ;
}
2016-07-15 14:43:34 +03:00
/*
2017-04-12 15:13:27 +03:00
* Create a snapshot of the current LPIs targeting @ vcpu , so that we can
* enumerate those LPIs without holding any lock .
* Returns their number and puts the kmalloc ' ed array into intid_ptr .
2016-07-15 14:43:34 +03:00
*/
2017-04-12 15:13:27 +03:00
static int vgic_copy_lpi_list ( struct kvm_vcpu * vcpu , u32 * * intid_ptr )
2016-07-15 14:43:34 +03:00
{
2017-04-12 15:13:27 +03:00
struct vgic_dist * dist = & vcpu - > kvm - > arch . vgic ;
2016-07-15 14:43:34 +03:00
struct vgic_irq * irq ;
u32 * intids ;
int irq_count = dist - > lpi_list_count , i = 0 ;
/*
* We use the current value of the list length , which may change
* after the kmalloc . We don ' t care , because the guest shouldn ' t
* change anything while the command handling is still running ,
* and in the worst case we would miss a new IRQ , which one wouldn ' t
* expect to be covered by this command anyway .
*/
intids = kmalloc_array ( irq_count , sizeof ( intids [ 0 ] ) , GFP_KERNEL ) ;
if ( ! intids )
return - ENOMEM ;
spin_lock ( & dist - > lpi_list_lock ) ;
list_for_each_entry ( irq , & dist - > lpi_list_head , lpi_list ) {
/* We don't need to "get" the IRQ, as we hold the list lock. */
2017-04-12 15:13:27 +03:00
if ( irq - > target_vcpu ! = vcpu )
continue ;
intids [ i + + ] = irq - > intid ;
2016-07-15 14:43:34 +03:00
}
spin_unlock ( & dist - > lpi_list_lock ) ;
* intid_ptr = intids ;
2017-04-12 15:13:27 +03:00
return i ;
2016-07-15 14:43:34 +03:00
}
2016-07-15 14:43:36 +03:00
/*
* Promotes the ITS view of affinity of an ITTE ( which redistributor this LPI
* is targeting ) to the VGIC ' s view , which deals with target VCPUs .
* Needs to be called whenever either the collection for a LPIs has
* changed or the collection itself got retargeted .
*/
2017-02-08 08:09:29 +03:00
static void update_affinity_ite ( struct kvm * kvm , struct its_ite * ite )
2016-07-15 14:43:36 +03:00
{
struct kvm_vcpu * vcpu ;
2017-02-08 08:09:29 +03:00
if ( ! its_is_collection_mapped ( ite - > collection ) )
2016-07-15 14:43:36 +03:00
return ;
2017-02-08 08:09:29 +03:00
vcpu = kvm_get_vcpu ( kvm , ite - > collection - > target_addr ) ;
2016-07-15 14:43:36 +03:00
2017-02-08 08:09:29 +03:00
spin_lock ( & ite - > irq - > irq_lock ) ;
ite - > irq - > target_vcpu = vcpu ;
spin_unlock ( & ite - > irq - > irq_lock ) ;
2016-07-15 14:43:36 +03:00
}
/*
* Updates the target VCPU for every LPI targeting this collection .
* Must be called with the its_lock mutex held .
*/
static void update_affinity_collection ( struct kvm * kvm , struct vgic_its * its ,
struct its_collection * coll )
{
struct its_device * device ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
2017-02-08 08:09:29 +03:00
for_each_lpi_its ( device , ite , its ) {
if ( ! ite - > collection | | coll ! = ite - > collection )
2016-07-15 14:43:36 +03:00
continue ;
2017-02-08 08:09:29 +03:00
update_affinity_ite ( kvm , ite ) ;
2016-07-15 14:43:36 +03:00
}
}
static u32 max_lpis_propbaser ( u64 propbaser )
{
int nr_idbits = ( propbaser & 0x1f ) + 1 ;
return 1U < < min ( nr_idbits , INTERRUPT_ID_BITS_ITS ) ;
}
2016-07-15 14:43:34 +03:00
/*
2017-04-12 15:13:27 +03:00
* Sync the pending table pending bit of LPIs targeting @ vcpu
2016-07-15 14:43:34 +03:00
* with our own data structures . This relies on the LPI being
* mapped before .
*/
static int its_sync_lpi_pending_table ( struct kvm_vcpu * vcpu )
{
2017-05-04 12:19:52 +03:00
gpa_t pendbase = GICR_PENDBASER_ADDRESS ( vcpu - > arch . vgic_cpu . pendbaser ) ;
2016-07-15 14:43:34 +03:00
struct vgic_irq * irq ;
int last_byte_offset = - 1 ;
int ret = 0 ;
u32 * intids ;
int nr_irqs , i ;
2017-04-12 15:13:27 +03:00
nr_irqs = vgic_copy_lpi_list ( vcpu , & intids ) ;
2016-07-15 14:43:34 +03:00
if ( nr_irqs < 0 )
return nr_irqs ;
for ( i = 0 ; i < nr_irqs ; i + + ) {
int byte_offset , bit_nr ;
u8 pendmask ;
byte_offset = intids [ i ] / BITS_PER_BYTE ;
bit_nr = intids [ i ] % BITS_PER_BYTE ;
/*
* For contiguously allocated LPIs chances are we just read
* this very same byte in the last iteration . Reuse that .
*/
if ( byte_offset ! = last_byte_offset ) {
ret = kvm_read_guest ( vcpu - > kvm , pendbase + byte_offset ,
& pendmask , 1 ) ;
if ( ret ) {
kfree ( intids ) ;
return ret ;
}
last_byte_offset = byte_offset ;
}
irq = vgic_get_irq ( vcpu - > kvm , NULL , intids [ i ] ) ;
spin_lock ( & irq - > irq_lock ) ;
2017-01-23 16:07:18 +03:00
irq - > pending_latch = pendmask & ( 1U < < bit_nr ) ;
2016-07-15 14:43:34 +03:00
vgic_queue_irq_unlock ( vcpu - > kvm , irq ) ;
vgic_put_irq ( vcpu - > kvm , irq ) ;
}
kfree ( intids ) ;
return ret ;
}
2016-07-15 14:43:32 +03:00
static unsigned long vgic_mmio_read_its_typer ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
2017-04-13 10:06:20 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
2016-07-15 14:43:32 +03:00
u64 reg = GITS_TYPER_PLPIS ;
/*
* We use linear CPU numbers for redistributor addressing ,
* so GITS_TYPER . PTA is 0.
* Also we force all PROPBASER registers to be the same , so
* CommonLPIAff is 0 as well .
* To avoid memory waste in the guest , we keep the number of IDBits and
* DevBits low - as least for the time being .
*/
2017-02-02 16:37:33 +03:00
reg | = GIC_ENCODE_SZ ( VITS_TYPER_DEVBITS , 5 ) < < GITS_TYPER_DEVBITS_SHIFT ;
2016-12-22 20:14:14 +03:00
reg | = GIC_ENCODE_SZ ( VITS_TYPER_IDBITS , 5 ) < < GITS_TYPER_IDBITS_SHIFT ;
2017-04-13 10:06:20 +03:00
reg | = GIC_ENCODE_SZ ( abi - > ite_esz , 4 ) < < GITS_TYPER_ITT_ENTRY_SIZE_SHIFT ;
2016-07-15 14:43:32 +03:00
return extract_bytes ( reg , addr & 7 , len ) ;
}
static unsigned long vgic_mmio_read_its_iidr ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
2017-03-23 17:14:00 +03:00
u32 val ;
val = ( its - > abi_rev < < GITS_IIDR_REV_SHIFT ) & GITS_IIDR_REV_MASK ;
val | = ( PRODUCT_ID_KVM < < GITS_IIDR_PRODUCTID_SHIFT ) | IMPLEMENTER_ARM ;
return val ;
}
static int vgic_mmio_uaccess_write_its_iidr ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
u32 rev = GITS_IIDR_REV ( val ) ;
if ( rev > = NR_ITS_ABIS )
return - EINVAL ;
return vgic_its_set_abi ( its , rev ) ;
2016-07-15 14:43:32 +03:00
}
static unsigned long vgic_mmio_read_its_idregs ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
switch ( addr & 0xffff ) {
case GITS_PIDR0 :
return 0x92 ; /* part number, bits[7:0] */
case GITS_PIDR1 :
return 0xb4 ; /* part number, bits[11:8] */
case GITS_PIDR2 :
return GIC_PIDR2_ARCH_GICv3 | 0x0b ;
case GITS_PIDR4 :
return 0x40 ; /* This is a 64K software visible page */
/* The following are the ID registers for (any) GIC. */
case GITS_CIDR0 :
return 0x0d ;
case GITS_CIDR1 :
return 0xf0 ;
case GITS_CIDR2 :
return 0x05 ;
case GITS_CIDR3 :
return 0xb1 ;
}
return 0 ;
}
2016-07-15 14:43:37 +03:00
/*
* Find the target VCPU and the LPI number for a given devid / eventid pair
* and make this IRQ pending , possibly injecting it .
* Must be called with the its_lock mutex held .
2016-08-08 19:29:28 +03:00
* Returns 0 on success , a positive error value for any ITS mapping
* related errors and negative error values for generic errors .
2016-07-15 14:43:37 +03:00
*/
2016-08-08 19:29:28 +03:00
static int vgic_its_trigger_msi ( struct kvm * kvm , struct vgic_its * its ,
u32 devid , u32 eventid )
2016-07-15 14:43:37 +03:00
{
2016-08-08 19:29:28 +03:00
struct kvm_vcpu * vcpu ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:37 +03:00
if ( ! its - > enabled )
2016-08-08 19:29:28 +03:00
return - EBUSY ;
2016-07-15 14:43:37 +03:00
2017-02-08 08:09:29 +03:00
ite = find_ite ( its , devid , eventid ) ;
if ( ! ite | | ! its_is_collection_mapped ( ite - > collection ) )
2016-08-08 19:29:28 +03:00
return E_ITS_INT_UNMAPPED_INTERRUPT ;
2017-02-08 08:09:29 +03:00
vcpu = kvm_get_vcpu ( kvm , ite - > collection - > target_addr ) ;
2016-08-08 19:29:28 +03:00
if ( ! vcpu )
return E_ITS_INT_UNMAPPED_INTERRUPT ;
if ( ! vcpu - > arch . vgic_cpu . lpis_enabled )
return - EBUSY ;
2017-02-08 08:09:29 +03:00
spin_lock ( & ite - > irq - > irq_lock ) ;
ite - > irq - > pending_latch = true ;
vgic_queue_irq_unlock ( kvm , ite - > irq ) ;
2016-08-08 19:29:28 +03:00
return 0 ;
2016-07-15 14:43:37 +03:00
}
2016-08-09 12:54:29 +03:00
static struct vgic_io_device * vgic_get_its_iodev ( struct kvm_io_device * dev )
{
struct vgic_io_device * iodev ;
if ( dev - > ops ! = & kvm_io_gic_ops )
return NULL ;
iodev = container_of ( dev , struct vgic_io_device , dev ) ;
if ( iodev - > iodev_type ! = IODEV_ITS )
return NULL ;
return iodev ;
}
2016-07-15 14:43:37 +03:00
/*
* Queries the KVM IO bus framework to get the ITS pointer from the given
* doorbell address .
* We then call vgic_its_trigger_msi ( ) with the decoded data .
2016-08-08 19:29:28 +03:00
* According to the KVM_SIGNAL_MSI API description returns 1 on success .
2016-07-15 14:43:37 +03:00
*/
int vgic_its_inject_msi ( struct kvm * kvm , struct kvm_msi * msi )
{
u64 address ;
struct kvm_io_device * kvm_io_dev ;
struct vgic_io_device * iodev ;
2016-08-08 19:29:28 +03:00
int ret ;
2016-07-15 14:43:37 +03:00
if ( ! vgic_has_its ( kvm ) )
return - ENODEV ;
if ( ! ( msi - > flags & KVM_MSI_VALID_DEVID ) )
return - EINVAL ;
address = ( u64 ) msi - > address_hi < < 32 | msi - > address_lo ;
kvm_io_dev = kvm_io_bus_get_dev ( kvm , KVM_MMIO_BUS , address ) ;
if ( ! kvm_io_dev )
2016-08-09 12:54:29 +03:00
return - EINVAL ;
2016-07-15 14:43:37 +03:00
2016-08-09 12:54:29 +03:00
iodev = vgic_get_its_iodev ( kvm_io_dev ) ;
if ( ! iodev )
return - EINVAL ;
2016-07-15 14:43:37 +03:00
mutex_lock ( & iodev - > its - > its_lock ) ;
2016-08-08 19:29:28 +03:00
ret = vgic_its_trigger_msi ( kvm , iodev - > its , msi - > devid , msi - > data ) ;
2016-07-15 14:43:37 +03:00
mutex_unlock ( & iodev - > its - > its_lock ) ;
2016-08-08 19:29:28 +03:00
if ( ret < 0 )
return ret ;
/*
* KVM_SIGNAL_MSI demands a return value > 0 for success and 0
* if the guest has blocked the MSI . So we map any LPI mapping
* related error to that .
*/
if ( ret )
return 0 ;
else
return 1 ;
2016-07-15 14:43:37 +03:00
}
2016-07-15 14:43:32 +03:00
/* Requires the its_lock to be held. */
2017-02-08 08:09:29 +03:00
static void its_free_ite ( struct kvm * kvm , struct its_ite * ite )
2016-07-15 14:43:32 +03:00
{
2017-02-08 08:09:29 +03:00
list_del ( & ite - > ite_list ) ;
2016-07-15 14:43:33 +03:00
/* This put matches the get in vgic_add_lpi. */
2017-02-08 08:09:29 +03:00
if ( ite - > irq )
vgic_put_irq ( kvm , ite - > irq ) ;
2016-07-15 14:43:33 +03:00
2017-02-08 08:09:29 +03:00
kfree ( ite ) ;
2016-07-15 14:43:32 +03:00
}
2016-07-15 14:43:36 +03:00
static u64 its_cmd_mask_field ( u64 * its_cmd , int word , int shift , int size )
{
return ( le64_to_cpu ( its_cmd [ word ] ) > > shift ) & ( BIT_ULL ( size ) - 1 ) ;
}
# define its_cmd_get_command(cmd) its_cmd_mask_field(cmd, 0, 0, 8)
# define its_cmd_get_deviceid(cmd) its_cmd_mask_field(cmd, 0, 32, 32)
2016-12-22 20:14:14 +03:00
# define its_cmd_get_size(cmd) (its_cmd_mask_field(cmd, 1, 0, 5) + 1)
2016-07-15 14:43:36 +03:00
# define its_cmd_get_id(cmd) its_cmd_mask_field(cmd, 1, 0, 32)
# define its_cmd_get_physical_id(cmd) its_cmd_mask_field(cmd, 1, 32, 32)
# define its_cmd_get_collection(cmd) its_cmd_mask_field(cmd, 2, 0, 16)
2017-02-02 15:45:45 +03:00
# define its_cmd_get_ittaddr(cmd) (its_cmd_mask_field(cmd, 2, 8, 44) << 8)
2016-07-15 14:43:36 +03:00
# define its_cmd_get_target_addr(cmd) its_cmd_mask_field(cmd, 2, 16, 32)
# define its_cmd_get_validbit(cmd) its_cmd_mask_field(cmd, 2, 63, 1)
/*
* The DISCARD command frees an Interrupt Translation Table Entry ( ITTE ) .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_discard ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
u32 event_id = its_cmd_get_id ( its_cmd ) ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
2017-02-08 08:09:29 +03:00
ite = find_ite ( its , device_id , event_id ) ;
if ( ite & & ite - > collection ) {
2016-07-15 14:43:36 +03:00
/*
* Though the spec talks about removing the pending state , we
* don ' t bother here since we clear the ITTE anyway and the
* pending state is a property of the ITTE struct .
*/
2017-02-08 08:09:29 +03:00
its_free_ite ( kvm , ite ) ;
2016-07-15 14:43:36 +03:00
return 0 ;
}
return E_ITS_DISCARD_UNMAPPED_INTERRUPT ;
}
/*
* The MOVI command moves an ITTE to a different collection .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_movi ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
u32 event_id = its_cmd_get_id ( its_cmd ) ;
u32 coll_id = its_cmd_get_collection ( its_cmd ) ;
struct kvm_vcpu * vcpu ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
struct its_collection * collection ;
2017-02-08 08:09:29 +03:00
ite = find_ite ( its , device_id , event_id ) ;
if ( ! ite )
2016-07-15 14:43:36 +03:00
return E_ITS_MOVI_UNMAPPED_INTERRUPT ;
2017-02-08 08:09:29 +03:00
if ( ! its_is_collection_mapped ( ite - > collection ) )
2016-07-15 14:43:36 +03:00
return E_ITS_MOVI_UNMAPPED_COLLECTION ;
collection = find_collection ( its , coll_id ) ;
if ( ! its_is_collection_mapped ( collection ) )
return E_ITS_MOVI_UNMAPPED_COLLECTION ;
2017-02-08 08:09:29 +03:00
ite - > collection = collection ;
2016-07-15 14:43:36 +03:00
vcpu = kvm_get_vcpu ( kvm , collection - > target_addr ) ;
2017-02-08 08:09:29 +03:00
spin_lock ( & ite - > irq - > irq_lock ) ;
ite - > irq - > target_vcpu = vcpu ;
spin_unlock ( & ite - > irq - > irq_lock ) ;
2016-07-15 14:43:36 +03:00
return 0 ;
}
2016-07-17 23:52:55 +03:00
/*
* Check whether an ID can be stored into the corresponding guest table .
* For a direct table this is pretty easy , but gets a bit nasty for
* indirect tables . We check whether the resulting guest physical address
2017-02-02 16:37:33 +03:00
* is actually valid ( covered by a memslot and guest accessible ) .
2016-07-17 23:52:55 +03:00
* For this we have to read the respective first level entry .
*/
2017-01-31 16:36:14 +03:00
static bool vgic_its_check_id ( struct vgic_its * its , u64 baser , u32 id ,
gpa_t * eaddr )
2016-07-17 23:52:55 +03:00
{
int l1_tbl_size = GITS_BASER_NR_PAGES ( baser ) * SZ_64K ;
2017-02-02 16:37:33 +03:00
u64 indirect_ptr , type = GITS_BASER_TYPE ( baser ) ;
int esz = GITS_BASER_ENTRY_SIZE ( baser ) ;
2016-07-17 23:52:55 +03:00
int index ;
gfn_t gfn ;
2017-02-02 16:37:33 +03:00
switch ( type ) {
case GITS_BASER_TYPE_DEVICE :
if ( id > = BIT_ULL ( VITS_TYPER_DEVBITS ) )
return false ;
break ;
case GITS_BASER_TYPE_COLLECTION :
/* as GITS_TYPER.CIL == 0, ITS supports 16-bit collection ID */
if ( id > = BIT_ULL ( 16 ) )
return false ;
break ;
default :
return false ;
}
2016-07-17 23:52:55 +03:00
if ( ! ( baser & GITS_BASER_INDIRECT ) ) {
phys_addr_t addr ;
2016-11-02 14:55:33 +03:00
if ( id > = ( l1_tbl_size / esz ) )
2016-07-17 23:52:55 +03:00
return false ;
2016-11-02 14:55:33 +03:00
addr = BASER_ADDRESS ( baser ) + id * esz ;
2016-07-17 23:52:55 +03:00
gfn = addr > > PAGE_SHIFT ;
2017-01-31 16:36:14 +03:00
if ( eaddr )
* eaddr = addr ;
2016-07-17 23:52:55 +03:00
return kvm_is_visible_gfn ( its - > dev - > kvm , gfn ) ;
}
/* calculate and check the index into the 1st level */
2016-11-02 14:55:33 +03:00
index = id / ( SZ_64K / esz ) ;
2016-07-17 23:52:55 +03:00
if ( index > = ( l1_tbl_size / sizeof ( u64 ) ) )
return false ;
/* Each 1st level entry is represented by a 64-bit value. */
if ( kvm_read_guest ( its - > dev - > kvm ,
BASER_ADDRESS ( baser ) + index * sizeof ( indirect_ptr ) ,
& indirect_ptr , sizeof ( indirect_ptr ) ) )
return false ;
indirect_ptr = le64_to_cpu ( indirect_ptr ) ;
/* check the valid bit of the first level entry */
if ( ! ( indirect_ptr & BIT_ULL ( 63 ) ) )
return false ;
/*
* Mask the guest physical address and calculate the frame number .
* Any address beyond our supported 48 bits of PA will be caught
* by the actual check in the final step .
*/
indirect_ptr & = GENMASK_ULL ( 51 , 16 ) ;
/* Find the address of the actual entry */
2016-11-02 14:55:33 +03:00
index = id % ( SZ_64K / esz ) ;
indirect_ptr + = index * esz ;
2016-07-17 23:52:55 +03:00
gfn = indirect_ptr > > PAGE_SHIFT ;
2017-01-31 16:36:14 +03:00
if ( eaddr )
* eaddr = indirect_ptr ;
2016-07-17 23:52:55 +03:00
return kvm_is_visible_gfn ( its - > dev - > kvm , gfn ) ;
}
2016-07-17 22:01:46 +03:00
static int vgic_its_alloc_collection ( struct vgic_its * its ,
struct its_collection * * colp ,
2016-07-15 14:43:36 +03:00
u32 coll_id )
{
2016-07-17 22:01:46 +03:00
struct its_collection * collection ;
2017-01-31 16:36:14 +03:00
if ( ! vgic_its_check_id ( its , its - > baser_coll_table , coll_id , NULL ) )
2016-07-17 23:52:55 +03:00
return E_ITS_MAPC_COLLECTION_OOR ;
2016-07-17 22:01:46 +03:00
collection = kzalloc ( sizeof ( * collection ) , GFP_KERNEL ) ;
2016-07-15 14:43:36 +03:00
collection - > collection_id = coll_id ;
collection - > target_addr = COLLECTION_NOT_MAPPED ;
list_add_tail ( & collection - > coll_list , & its - > collection_list ) ;
2016-07-17 22:01:46 +03:00
* colp = collection ;
return 0 ;
}
static void vgic_its_free_collection ( struct vgic_its * its , u32 coll_id )
{
struct its_collection * collection ;
struct its_device * device ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-17 22:01:46 +03:00
/*
* Clearing the mapping for that collection ID removes the
* entry from the list . If there wasn ' t any before , we can
* go home early .
*/
collection = find_collection ( its , coll_id ) ;
if ( ! collection )
return ;
2017-02-08 08:09:29 +03:00
for_each_lpi_its ( device , ite , its )
if ( ite - > collection & &
ite - > collection - > collection_id = = coll_id )
ite - > collection = NULL ;
2016-07-17 22:01:46 +03:00
list_del ( & collection - > coll_list ) ;
kfree ( collection ) ;
2016-07-15 14:43:36 +03:00
}
2016-12-25 20:57:54 +03:00
/* Must be called with its_lock mutex held */
static struct its_ite * vgic_its_alloc_ite ( struct its_device * device ,
struct its_collection * collection ,
u32 lpi_id , u32 event_id )
{
struct its_ite * ite ;
ite = kzalloc ( sizeof ( * ite ) , GFP_KERNEL ) ;
if ( ! ite )
return ERR_PTR ( - ENOMEM ) ;
ite - > event_id = event_id ;
ite - > collection = collection ;
ite - > lpi = lpi_id ;
list_add_tail ( & ite - > ite_list , & device - > itt_head ) ;
return ite ;
}
2016-07-15 14:43:36 +03:00
/*
* The MAPTI and MAPI commands map LPIs to ITTEs .
* Must be called with its_lock mutex held .
*/
static int vgic_its_cmd_handle_mapi ( struct kvm * kvm , struct vgic_its * its ,
2016-07-18 00:38:32 +03:00
u64 * its_cmd )
2016-07-15 14:43:36 +03:00
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
u32 event_id = its_cmd_get_id ( its_cmd ) ;
u32 coll_id = its_cmd_get_collection ( its_cmd ) ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2017-05-04 12:36:32 +03:00
struct kvm_vcpu * vcpu = NULL ;
2016-07-15 14:43:36 +03:00
struct its_device * device ;
struct its_collection * collection , * new_coll = NULL ;
2016-08-01 21:25:33 +03:00
struct vgic_irq * irq ;
2016-12-25 20:57:54 +03:00
int lpi_nr ;
2016-07-15 14:43:36 +03:00
device = find_its_device ( its , device_id ) ;
if ( ! device )
return E_ITS_MAPTI_UNMAPPED_DEVICE ;
2016-12-22 20:14:14 +03:00
if ( event_id > = BIT_ULL ( device - > num_eventid_bits ) )
return E_ITS_MAPTI_ID_OOR ;
2016-07-18 00:38:32 +03:00
if ( its_cmd_get_command ( its_cmd ) = = GITS_CMD_MAPTI )
2016-07-15 14:43:36 +03:00
lpi_nr = its_cmd_get_physical_id ( its_cmd ) ;
else
lpi_nr = event_id ;
if ( lpi_nr < GIC_LPI_OFFSET | |
2016-07-18 18:27:14 +03:00
lpi_nr > = max_lpis_propbaser ( kvm - > arch . vgic . propbaser ) )
return E_ITS_MAPTI_PHYSICALID_OOR ;
2016-08-16 19:51:06 +03:00
/* If there is an existing mapping, behavior is UNPREDICTABLE. */
2017-02-08 08:09:29 +03:00
if ( find_ite ( its , device_id , event_id ) )
2016-08-16 19:51:06 +03:00
return 0 ;
2016-07-18 18:27:14 +03:00
collection = find_collection ( its , coll_id ) ;
if ( ! collection ) {
int ret = vgic_its_alloc_collection ( its , & collection , coll_id ) ;
if ( ret )
return ret ;
new_coll = collection ;
2016-07-15 14:43:36 +03:00
}
2016-12-25 20:57:54 +03:00
ite = vgic_its_alloc_ite ( device , collection , lpi_nr , event_id ) ;
if ( IS_ERR ( ite ) ) {
2016-08-16 19:51:06 +03:00
if ( new_coll )
vgic_its_free_collection ( its , coll_id ) ;
2016-12-25 20:57:54 +03:00
return PTR_ERR ( ite ) ;
2016-07-15 14:43:36 +03:00
}
2017-05-04 12:36:32 +03:00
if ( its_is_collection_mapped ( collection ) )
vcpu = kvm_get_vcpu ( kvm , collection - > target_addr ) ;
irq = vgic_add_lpi ( kvm , lpi_nr , vcpu ) ;
2016-08-01 21:25:33 +03:00
if ( IS_ERR ( irq ) ) {
if ( new_coll )
vgic_its_free_collection ( its , coll_id ) ;
2017-02-08 08:09:29 +03:00
its_free_ite ( kvm , ite ) ;
2016-08-01 21:25:33 +03:00
return PTR_ERR ( irq ) ;
}
2017-02-08 08:09:29 +03:00
ite - > irq = irq ;
2016-08-01 21:25:33 +03:00
2016-07-15 14:43:36 +03:00
return 0 ;
}
/* Requires the its_lock to be held. */
static void vgic_its_unmap_device ( struct kvm * kvm , struct its_device * device )
{
2017-02-08 08:09:29 +03:00
struct its_ite * ite , * temp ;
2016-07-15 14:43:36 +03:00
/*
* The spec says that unmapping a device with still valid
* ITTEs associated is UNPREDICTABLE . We remove all ITTEs ,
* since we cannot leave the memory unreferenced .
*/
2017-02-08 08:09:29 +03:00
list_for_each_entry_safe ( ite , temp , & device - > itt_head , ite_list )
its_free_ite ( kvm , ite ) ;
2016-07-15 14:43:36 +03:00
list_del ( & device - > dev_list ) ;
kfree ( device ) ;
}
2016-12-25 20:57:54 +03:00
/* Must be called with its_lock mutex held */
static struct its_device * vgic_its_alloc_device ( struct vgic_its * its ,
u32 device_id , gpa_t itt_addr ,
u8 num_eventid_bits )
{
struct its_device * device ;
device = kzalloc ( sizeof ( * device ) , GFP_KERNEL ) ;
if ( ! device )
return ERR_PTR ( - ENOMEM ) ;
device - > device_id = device_id ;
device - > itt_addr = itt_addr ;
device - > num_eventid_bits = num_eventid_bits ;
INIT_LIST_HEAD ( & device - > itt_head ) ;
list_add_tail ( & device - > dev_list , & its - > device_list ) ;
return device ;
}
2016-07-15 14:43:36 +03:00
/*
* MAPD maps or unmaps a device ID to Interrupt Translation Tables ( ITTs ) .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_mapd ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
bool valid = its_cmd_get_validbit ( its_cmd ) ;
2016-12-22 20:14:14 +03:00
u8 num_eventid_bits = its_cmd_get_size ( its_cmd ) ;
2017-02-02 15:45:45 +03:00
gpa_t itt_addr = its_cmd_get_ittaddr ( its_cmd ) ;
2016-07-15 14:43:36 +03:00
struct its_device * device ;
2017-01-31 16:36:14 +03:00
if ( ! vgic_its_check_id ( its , its - > baser_device_table , device_id , NULL ) )
2016-07-15 14:43:36 +03:00
return E_ITS_MAPD_DEVICE_OOR ;
2016-12-22 20:14:14 +03:00
if ( valid & & num_eventid_bits > VITS_TYPER_IDBITS )
return E_ITS_MAPD_ITTSIZE_OOR ;
2016-07-15 14:43:36 +03:00
device = find_its_device ( its , device_id ) ;
/*
* The spec says that calling MAPD on an already mapped device
* invalidates all cached data for this device . We implement this
* by removing the mapping and re - establishing it .
*/
if ( device )
vgic_its_unmap_device ( kvm , device ) ;
/*
* The spec does not say whether unmapping a not - mapped device
* is an error , so we are done in any case .
*/
if ( ! valid )
return 0 ;
2016-12-25 20:57:54 +03:00
device = vgic_its_alloc_device ( its , device_id , itt_addr ,
num_eventid_bits ) ;
if ( IS_ERR ( device ) )
return PTR_ERR ( device ) ;
2016-07-15 14:43:36 +03:00
return 0 ;
}
/*
* The MAPC command maps collection IDs to redistributors .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_mapc ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u16 coll_id ;
u32 target_addr ;
struct its_collection * collection ;
bool valid ;
valid = its_cmd_get_validbit ( its_cmd ) ;
coll_id = its_cmd_get_collection ( its_cmd ) ;
target_addr = its_cmd_get_target_addr ( its_cmd ) ;
if ( target_addr > = atomic_read ( & kvm - > online_vcpus ) )
return E_ITS_MAPC_PROCNUM_OOR ;
if ( ! valid ) {
2016-07-17 22:01:46 +03:00
vgic_its_free_collection ( its , coll_id ) ;
2016-07-15 14:43:36 +03:00
} else {
2016-07-17 22:01:46 +03:00
collection = find_collection ( its , coll_id ) ;
2016-07-15 14:43:36 +03:00
if ( ! collection ) {
2016-07-17 22:01:46 +03:00
int ret ;
2016-07-15 14:43:36 +03:00
2016-07-17 22:01:46 +03:00
ret = vgic_its_alloc_collection ( its , & collection ,
coll_id ) ;
if ( ret )
return ret ;
2016-07-15 14:43:36 +03:00
collection - > target_addr = target_addr ;
} else {
collection - > target_addr = target_addr ;
update_affinity_collection ( kvm , its , collection ) ;
}
}
return 0 ;
}
/*
* The CLEAR command removes the pending state for a particular LPI .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_clear ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
u32 event_id = its_cmd_get_id ( its_cmd ) ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
2017-02-08 08:09:29 +03:00
ite = find_ite ( its , device_id , event_id ) ;
if ( ! ite )
2016-07-15 14:43:36 +03:00
return E_ITS_CLEAR_UNMAPPED_INTERRUPT ;
2017-02-08 08:09:29 +03:00
ite - > irq - > pending_latch = false ;
2016-07-15 14:43:36 +03:00
return 0 ;
}
/*
* The INV command syncs the configuration bits from the memory table .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_inv ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 device_id = its_cmd_get_deviceid ( its_cmd ) ;
u32 event_id = its_cmd_get_id ( its_cmd ) ;
2017-02-08 08:09:29 +03:00
struct its_ite * ite ;
2016-07-15 14:43:36 +03:00
2017-02-08 08:09:29 +03:00
ite = find_ite ( its , device_id , event_id ) ;
if ( ! ite )
2016-07-15 14:43:36 +03:00
return E_ITS_INV_UNMAPPED_INTERRUPT ;
2017-02-08 08:09:29 +03:00
return update_lpi_config ( kvm , ite - > irq , NULL ) ;
2016-07-15 14:43:36 +03:00
}
/*
* The INVALL command requests flushing of all IRQ data in this collection .
* Find the VCPU mapped to that collection , then iterate over the VM ' s list
* of mapped LPIs and update the configuration for each IRQ which targets
* the specified vcpu . The configuration will be read from the in - memory
* configuration table .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_invall ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 coll_id = its_cmd_get_collection ( its_cmd ) ;
struct its_collection * collection ;
struct kvm_vcpu * vcpu ;
struct vgic_irq * irq ;
u32 * intids ;
int irq_count , i ;
collection = find_collection ( its , coll_id ) ;
if ( ! its_is_collection_mapped ( collection ) )
return E_ITS_INVALL_UNMAPPED_COLLECTION ;
vcpu = kvm_get_vcpu ( kvm , collection - > target_addr ) ;
2017-04-12 15:13:27 +03:00
irq_count = vgic_copy_lpi_list ( vcpu , & intids ) ;
2016-07-15 14:43:36 +03:00
if ( irq_count < 0 )
return irq_count ;
for ( i = 0 ; i < irq_count ; i + + ) {
irq = vgic_get_irq ( kvm , NULL , intids [ i ] ) ;
if ( ! irq )
continue ;
update_lpi_config ( kvm , irq , vcpu ) ;
vgic_put_irq ( kvm , irq ) ;
}
kfree ( intids ) ;
return 0 ;
}
/*
* The MOVALL command moves the pending state of all IRQs targeting one
* redistributor to another . We don ' t hold the pending state in the VCPUs ,
* but in the IRQs instead , so there is really not much to do for us here .
* However the spec says that no IRQ must target the old redistributor
* afterwards , so we make sure that no LPI is using the associated target_vcpu .
* This command affects all LPIs in the system that target that redistributor .
*/
static int vgic_its_cmd_handle_movall ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
struct vgic_dist * dist = & kvm - > arch . vgic ;
u32 target1_addr = its_cmd_get_target_addr ( its_cmd ) ;
u32 target2_addr = its_cmd_mask_field ( its_cmd , 3 , 16 , 32 ) ;
struct kvm_vcpu * vcpu1 , * vcpu2 ;
struct vgic_irq * irq ;
if ( target1_addr > = atomic_read ( & kvm - > online_vcpus ) | |
target2_addr > = atomic_read ( & kvm - > online_vcpus ) )
return E_ITS_MOVALL_PROCNUM_OOR ;
if ( target1_addr = = target2_addr )
return 0 ;
vcpu1 = kvm_get_vcpu ( kvm , target1_addr ) ;
vcpu2 = kvm_get_vcpu ( kvm , target2_addr ) ;
spin_lock ( & dist - > lpi_list_lock ) ;
list_for_each_entry ( irq , & dist - > lpi_list_head , lpi_list ) {
spin_lock ( & irq - > irq_lock ) ;
if ( irq - > target_vcpu = = vcpu1 )
irq - > target_vcpu = vcpu2 ;
spin_unlock ( & irq - > irq_lock ) ;
}
spin_unlock ( & dist - > lpi_list_lock ) ;
return 0 ;
}
2016-07-15 14:43:37 +03:00
/*
* The INT command injects the LPI associated with that DevID / EvID pair .
* Must be called with the its_lock mutex held .
*/
static int vgic_its_cmd_handle_int ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
u32 msi_data = its_cmd_get_id ( its_cmd ) ;
u64 msi_devid = its_cmd_get_deviceid ( its_cmd ) ;
2016-08-08 19:29:28 +03:00
return vgic_its_trigger_msi ( kvm , its , msi_devid , msi_data ) ;
2016-07-15 14:43:37 +03:00
}
2016-07-15 14:43:36 +03:00
/*
* This function is called with the its_cmd lock held , but the ITS data
* structure lock dropped .
*/
2016-07-15 14:43:32 +03:00
static int vgic_its_handle_command ( struct kvm * kvm , struct vgic_its * its ,
u64 * its_cmd )
{
2016-07-15 14:43:36 +03:00
int ret = - ENODEV ;
mutex_lock ( & its - > its_lock ) ;
2016-07-18 00:38:32 +03:00
switch ( its_cmd_get_command ( its_cmd ) ) {
2016-07-15 14:43:36 +03:00
case GITS_CMD_MAPD :
ret = vgic_its_cmd_handle_mapd ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_MAPC :
ret = vgic_its_cmd_handle_mapc ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_MAPI :
2016-07-18 00:38:32 +03:00
ret = vgic_its_cmd_handle_mapi ( kvm , its , its_cmd ) ;
2016-07-15 14:43:36 +03:00
break ;
case GITS_CMD_MAPTI :
2016-07-18 00:38:32 +03:00
ret = vgic_its_cmd_handle_mapi ( kvm , its , its_cmd ) ;
2016-07-15 14:43:36 +03:00
break ;
case GITS_CMD_MOVI :
ret = vgic_its_cmd_handle_movi ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_DISCARD :
ret = vgic_its_cmd_handle_discard ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_CLEAR :
ret = vgic_its_cmd_handle_clear ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_MOVALL :
ret = vgic_its_cmd_handle_movall ( kvm , its , its_cmd ) ;
break ;
2016-07-15 14:43:37 +03:00
case GITS_CMD_INT :
ret = vgic_its_cmd_handle_int ( kvm , its , its_cmd ) ;
break ;
2016-07-15 14:43:36 +03:00
case GITS_CMD_INV :
ret = vgic_its_cmd_handle_inv ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_INVALL :
ret = vgic_its_cmd_handle_invall ( kvm , its , its_cmd ) ;
break ;
case GITS_CMD_SYNC :
/* we ignore this command: we are in sync all of the time */
ret = 0 ;
break ;
}
mutex_unlock ( & its - > its_lock ) ;
return ret ;
2016-07-15 14:43:32 +03:00
}
static u64 vgic_sanitise_its_baser ( u64 reg )
{
reg = vgic_sanitise_field ( reg , GITS_BASER_SHAREABILITY_MASK ,
GITS_BASER_SHAREABILITY_SHIFT ,
vgic_sanitise_shareability ) ;
reg = vgic_sanitise_field ( reg , GITS_BASER_INNER_CACHEABILITY_MASK ,
GITS_BASER_INNER_CACHEABILITY_SHIFT ,
vgic_sanitise_inner_cacheability ) ;
reg = vgic_sanitise_field ( reg , GITS_BASER_OUTER_CACHEABILITY_MASK ,
GITS_BASER_OUTER_CACHEABILITY_SHIFT ,
vgic_sanitise_outer_cacheability ) ;
/* Bits 15:12 contain bits 51:48 of the PA, which we don't support. */
reg & = ~ GENMASK_ULL ( 15 , 12 ) ;
/* We support only one (ITS) page size: 64K */
reg = ( reg & ~ GITS_BASER_PAGE_SIZE_MASK ) | GITS_BASER_PAGE_SIZE_64K ;
return reg ;
}
static u64 vgic_sanitise_its_cbaser ( u64 reg )
{
reg = vgic_sanitise_field ( reg , GITS_CBASER_SHAREABILITY_MASK ,
GITS_CBASER_SHAREABILITY_SHIFT ,
vgic_sanitise_shareability ) ;
reg = vgic_sanitise_field ( reg , GITS_CBASER_INNER_CACHEABILITY_MASK ,
GITS_CBASER_INNER_CACHEABILITY_SHIFT ,
vgic_sanitise_inner_cacheability ) ;
reg = vgic_sanitise_field ( reg , GITS_CBASER_OUTER_CACHEABILITY_MASK ,
GITS_CBASER_OUTER_CACHEABILITY_SHIFT ,
vgic_sanitise_outer_cacheability ) ;
/*
* Sanitise the physical address to be 64 k aligned .
* Also limit the physical addresses to 48 bits .
*/
reg & = ~ ( GENMASK_ULL ( 51 , 48 ) | GENMASK_ULL ( 15 , 12 ) ) ;
return reg ;
}
static unsigned long vgic_mmio_read_its_cbaser ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
return extract_bytes ( its - > cbaser , addr & 7 , len ) ;
}
static void vgic_mmio_write_its_cbaser ( struct kvm * kvm , struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
/* When GITS_CTLR.Enable is 1, this register is RO. */
if ( its - > enabled )
return ;
mutex_lock ( & its - > cmd_lock ) ;
its - > cbaser = update_64bit_reg ( its - > cbaser , addr & 7 , len , val ) ;
its - > cbaser = vgic_sanitise_its_cbaser ( its - > cbaser ) ;
its - > creadr = 0 ;
/*
* CWRITER is architecturally UNKNOWN on reset , but we need to reset
* it to CREADR to make sure we start with an empty command buffer .
*/
its - > cwriter = its - > creadr ;
mutex_unlock ( & its - > cmd_lock ) ;
}
# define ITS_CMD_BUFFER_SIZE(baser) ((((baser) & 0xff) + 1) << 12)
# define ITS_CMD_SIZE 32
# define ITS_CMD_OFFSET(reg) ((reg) & GENMASK(19, 5))
2017-02-16 13:41:20 +03:00
/* Must be called with the cmd_lock held. */
static void vgic_its_process_commands ( struct kvm * kvm , struct vgic_its * its )
2016-07-15 14:43:32 +03:00
{
gpa_t cbaser ;
u64 cmd_buf [ 4 ] ;
2017-02-16 13:41:20 +03:00
/* Commands are only processed when the ITS is enabled. */
if ( ! its - > enabled )
2016-07-15 14:43:32 +03:00
return ;
cbaser = CBASER_ADDRESS ( its - > cbaser ) ;
while ( its - > cwriter ! = its - > creadr ) {
int ret = kvm_read_guest ( kvm , cbaser + its - > creadr ,
cmd_buf , ITS_CMD_SIZE ) ;
/*
* If kvm_read_guest ( ) fails , this could be due to the guest
* programming a bogus value in CBASER or something else going
* wrong from which we cannot easily recover .
* According to section 6.3 .2 in the GICv3 spec we can just
* ignore that command then .
*/
if ( ! ret )
vgic_its_handle_command ( kvm , its , cmd_buf ) ;
its - > creadr + = ITS_CMD_SIZE ;
if ( its - > creadr = = ITS_CMD_BUFFER_SIZE ( its - > cbaser ) )
its - > creadr = 0 ;
}
2017-02-16 13:41:20 +03:00
}
/*
* By writing to CWRITER the guest announces new commands to be processed .
* To avoid any races in the first place , we take the its_cmd lock , which
* protects our ring buffer variables , so that there is only one user
* per ITS handling commands at a given time .
*/
static void vgic_mmio_write_its_cwriter ( struct kvm * kvm , struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
u64 reg ;
if ( ! its )
return ;
mutex_lock ( & its - > cmd_lock ) ;
reg = update_64bit_reg ( its - > cwriter , addr & 7 , len , val ) ;
reg = ITS_CMD_OFFSET ( reg ) ;
if ( reg > = ITS_CMD_BUFFER_SIZE ( its - > cbaser ) ) {
mutex_unlock ( & its - > cmd_lock ) ;
return ;
}
its - > cwriter = reg ;
vgic_its_process_commands ( kvm , its ) ;
2016-07-15 14:43:32 +03:00
mutex_unlock ( & its - > cmd_lock ) ;
}
static unsigned long vgic_mmio_read_its_cwriter ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
return extract_bytes ( its - > cwriter , addr & 0x7 , len ) ;
}
static unsigned long vgic_mmio_read_its_creadr ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
return extract_bytes ( its - > creadr , addr & 0x7 , len ) ;
}
2017-01-04 13:58:41 +03:00
static int vgic_mmio_uaccess_write_its_creadr ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
u32 cmd_offset ;
int ret = 0 ;
mutex_lock ( & its - > cmd_lock ) ;
if ( its - > enabled ) {
ret = - EBUSY ;
goto out ;
}
cmd_offset = ITS_CMD_OFFSET ( val ) ;
if ( cmd_offset > = ITS_CMD_BUFFER_SIZE ( its - > cbaser ) ) {
ret = - EINVAL ;
goto out ;
}
its - > creadr = cmd_offset ;
out :
mutex_unlock ( & its - > cmd_lock ) ;
return ret ;
}
2016-07-15 14:43:32 +03:00
# define BASER_INDEX(addr) (((addr) / sizeof(u64)) & 0x7)
static unsigned long vgic_mmio_read_its_baser ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
u64 reg ;
switch ( BASER_INDEX ( addr ) ) {
case 0 :
reg = its - > baser_device_table ;
break ;
case 1 :
reg = its - > baser_coll_table ;
break ;
default :
reg = 0 ;
break ;
}
return extract_bytes ( reg , addr & 7 , len ) ;
}
# define GITS_BASER_RO_MASK (GENMASK_ULL(52, 48) | GENMASK_ULL(58, 56))
static void vgic_mmio_write_its_baser ( struct kvm * kvm ,
struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
2017-04-13 10:06:20 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
2016-07-15 14:43:32 +03:00
u64 entry_size , device_type ;
u64 reg , * regptr , clearbits = 0 ;
/* When GITS_CTLR.Enable is 1, we ignore write accesses. */
if ( its - > enabled )
return ;
switch ( BASER_INDEX ( addr ) ) {
case 0 :
regptr = & its - > baser_device_table ;
2017-04-13 10:06:20 +03:00
entry_size = abi - > dte_esz ;
2016-07-15 14:43:32 +03:00
device_type = GITS_BASER_TYPE_DEVICE ;
break ;
case 1 :
regptr = & its - > baser_coll_table ;
2017-04-13 10:06:20 +03:00
entry_size = abi - > cte_esz ;
2016-07-15 14:43:32 +03:00
device_type = GITS_BASER_TYPE_COLLECTION ;
clearbits = GITS_BASER_INDIRECT ;
break ;
default :
return ;
}
reg = update_64bit_reg ( * regptr , addr & 7 , len , val ) ;
reg & = ~ GITS_BASER_RO_MASK ;
reg & = ~ clearbits ;
reg | = ( entry_size - 1 ) < < GITS_BASER_ENTRY_SIZE_SHIFT ;
reg | = device_type < < GITS_BASER_TYPE_SHIFT ;
reg = vgic_sanitise_its_baser ( reg ) ;
* regptr = reg ;
}
2017-02-16 13:41:20 +03:00
static unsigned long vgic_mmio_read_its_ctlr ( struct kvm * vcpu ,
struct vgic_its * its ,
gpa_t addr , unsigned int len )
{
u32 reg = 0 ;
mutex_lock ( & its - > cmd_lock ) ;
if ( its - > creadr = = its - > cwriter )
reg | = GITS_CTLR_QUIESCENT ;
if ( its - > enabled )
reg | = GITS_CTLR_ENABLE ;
mutex_unlock ( & its - > cmd_lock ) ;
return reg ;
}
static void vgic_mmio_write_its_ctlr ( struct kvm * kvm , struct vgic_its * its ,
gpa_t addr , unsigned int len ,
unsigned long val )
{
mutex_lock ( & its - > cmd_lock ) ;
its - > enabled = ! ! ( val & GITS_CTLR_ENABLE ) ;
/*
* Try to process any pending commands . This function bails out early
* if the ITS is disabled or no commands have been queued .
*/
vgic_its_process_commands ( kvm , its ) ;
mutex_unlock ( & its - > cmd_lock ) ;
}
2016-07-15 14:43:30 +03:00
# define REGISTER_ITS_DESC(off, rd, wr, length, acc) \
{ \
. reg_offset = off , \
. len = length , \
. access_flags = acc , \
. its_read = rd , \
. its_write = wr , \
}
2017-01-04 13:58:41 +03:00
# define REGISTER_ITS_DESC_UACCESS(off, rd, wr, uwr, length, acc)\
{ \
. reg_offset = off , \
. len = length , \
. access_flags = acc , \
. its_read = rd , \
. its_write = wr , \
. uaccess_its_write = uwr , \
}
2016-07-15 14:43:30 +03:00
static void its_mmio_write_wi ( struct kvm * kvm , struct vgic_its * its ,
gpa_t addr , unsigned int len , unsigned long val )
{
/* Ignore */
}
static struct vgic_register_region its_registers [ ] = {
REGISTER_ITS_DESC ( GITS_CTLR ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_ctlr , vgic_mmio_write_its_ctlr , 4 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_32bit ) ,
2017-03-23 17:14:00 +03:00
REGISTER_ITS_DESC_UACCESS ( GITS_IIDR ,
vgic_mmio_read_its_iidr , its_mmio_write_wi ,
vgic_mmio_uaccess_write_its_iidr , 4 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_32bit ) ,
REGISTER_ITS_DESC ( GITS_TYPER ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_typer , its_mmio_write_wi , 8 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_64bit | VGIC_ACCESS_32bit ) ,
REGISTER_ITS_DESC ( GITS_CBASER ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_cbaser , vgic_mmio_write_its_cbaser , 8 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_64bit | VGIC_ACCESS_32bit ) ,
REGISTER_ITS_DESC ( GITS_CWRITER ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_cwriter , vgic_mmio_write_its_cwriter , 8 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_64bit | VGIC_ACCESS_32bit ) ,
2017-01-04 13:58:41 +03:00
REGISTER_ITS_DESC_UACCESS ( GITS_CREADR ,
vgic_mmio_read_its_creadr , its_mmio_write_wi ,
vgic_mmio_uaccess_write_its_creadr , 8 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_64bit | VGIC_ACCESS_32bit ) ,
REGISTER_ITS_DESC ( GITS_BASER ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_baser , vgic_mmio_write_its_baser , 0x40 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_64bit | VGIC_ACCESS_32bit ) ,
REGISTER_ITS_DESC ( GITS_IDREGS_BASE ,
2016-07-15 14:43:32 +03:00
vgic_mmio_read_its_idregs , its_mmio_write_wi , 0x30 ,
2016-07-15 14:43:30 +03:00
VGIC_ACCESS_32bit ) ,
} ;
2016-07-15 14:43:34 +03:00
/* This is called on setting the LPI enable bit in the redistributor. */
void vgic_enable_lpis ( struct kvm_vcpu * vcpu )
{
if ( ! ( vcpu - > arch . vgic_cpu . pendbaser & GICR_PENDBASER_PTZ ) )
its_sync_lpi_pending_table ( vcpu ) ;
}
2017-05-08 14:14:57 +03:00
static int vgic_register_its_iodev ( struct kvm * kvm , struct vgic_its * its ,
u64 addr )
2016-07-15 14:43:30 +03:00
{
struct vgic_io_device * iodev = & its - > iodev ;
int ret ;
2017-05-08 14:14:57 +03:00
mutex_lock ( & kvm - > slots_lock ) ;
if ( ! IS_VGIC_ADDR_UNDEF ( its - > vgic_its_base ) ) {
ret = - EBUSY ;
goto out ;
}
2016-07-15 14:43:30 +03:00
2017-05-08 14:14:57 +03:00
its - > vgic_its_base = addr ;
2016-07-15 14:43:30 +03:00
iodev - > regions = its_registers ;
iodev - > nr_regions = ARRAY_SIZE ( its_registers ) ;
kvm_iodevice_init ( & iodev - > dev , & kvm_io_gic_ops ) ;
iodev - > base_addr = its - > vgic_its_base ;
iodev - > iodev_type = IODEV_ITS ;
iodev - > its = its ;
ret = kvm_io_bus_register_dev ( kvm , KVM_MMIO_BUS , iodev - > base_addr ,
KVM_VGIC_V3_ITS_SIZE , & iodev - > dev ) ;
2017-05-08 14:14:57 +03:00
out :
2016-07-15 14:43:30 +03:00
mutex_unlock ( & kvm - > slots_lock ) ;
return ret ;
}
2016-07-15 14:43:31 +03:00
2016-07-15 14:43:32 +03:00
# define INITIAL_BASER_VALUE \
( GIC_BASER_CACHEABILITY ( GITS_BASER , INNER , RaWb ) | \
GIC_BASER_CACHEABILITY ( GITS_BASER , OUTER , SameAsInner ) | \
GIC_BASER_SHAREABILITY ( GITS_BASER , InnerShareable ) | \
GITS_BASER_PAGE_SIZE_64K )
# define INITIAL_PROPBASER_VALUE \
( GIC_BASER_CACHEABILITY ( GICR_PROPBASER , INNER , RaWb ) | \
GIC_BASER_CACHEABILITY ( GICR_PROPBASER , OUTER , SameAsInner ) | \
GIC_BASER_SHAREABILITY ( GICR_PROPBASER , InnerShareable ) )
2016-07-15 14:43:31 +03:00
static int vgic_its_create ( struct kvm_device * dev , u32 type )
{
struct vgic_its * its ;
if ( type ! = KVM_DEV_TYPE_ARM_VGIC_ITS )
return - ENODEV ;
its = kzalloc ( sizeof ( struct vgic_its ) , GFP_KERNEL ) ;
if ( ! its )
return - ENOMEM ;
2016-07-15 14:43:32 +03:00
mutex_init ( & its - > its_lock ) ;
mutex_init ( & its - > cmd_lock ) ;
2016-07-15 14:43:31 +03:00
its - > vgic_its_base = VGIC_ADDR_UNDEF ;
2016-07-15 14:43:32 +03:00
INIT_LIST_HEAD ( & its - > device_list ) ;
INIT_LIST_HEAD ( & its - > collection_list ) ;
2016-07-15 14:43:31 +03:00
dev - > kvm - > arch . vgic . has_its = true ;
its - > enabled = false ;
2016-07-17 23:35:07 +03:00
its - > dev = dev ;
2016-07-15 14:43:31 +03:00
2016-07-15 14:43:32 +03:00
its - > baser_device_table = INITIAL_BASER_VALUE |
( ( u64 ) GITS_BASER_TYPE_DEVICE < < GITS_BASER_TYPE_SHIFT ) ;
its - > baser_coll_table = INITIAL_BASER_VALUE |
( ( u64 ) GITS_BASER_TYPE_COLLECTION < < GITS_BASER_TYPE_SHIFT ) ;
dev - > kvm - > arch . vgic . propbaser = INITIAL_PROPBASER_VALUE ;
2016-07-15 14:43:31 +03:00
dev - > private = its ;
2017-04-13 10:06:20 +03:00
return vgic_its_set_abi ( its , NR_ITS_ABIS - 1 ) ;
2016-07-15 14:43:31 +03:00
}
2017-05-08 14:31:12 +03:00
static void vgic_its_free_device ( struct kvm * kvm , struct its_device * dev )
{
struct its_ite * ite , * tmp ;
list_for_each_entry_safe ( ite , tmp , & dev - > itt_head , ite_list )
its_free_ite ( kvm , ite ) ;
list_del ( & dev - > dev_list ) ;
kfree ( dev ) ;
}
2016-07-15 14:43:31 +03:00
static void vgic_its_destroy ( struct kvm_device * kvm_dev )
{
2016-07-15 14:43:32 +03:00
struct kvm * kvm = kvm_dev - > kvm ;
2016-07-15 14:43:31 +03:00
struct vgic_its * its = kvm_dev - > private ;
2016-07-15 14:43:32 +03:00
struct list_head * cur , * temp ;
/*
* We may end up here without the lists ever having been initialized .
* Check this and bail out early to avoid dereferencing a NULL pointer .
*/
if ( ! its - > device_list . next )
return ;
mutex_lock ( & its - > its_lock ) ;
2017-05-08 14:31:12 +03:00
list_for_each_safe ( cur , temp , & its - > device_list ) {
struct its_device * dev ;
dev = list_entry ( cur , struct its_device , dev_list ) ;
vgic_its_free_device ( kvm , dev ) ;
2016-07-15 14:43:32 +03:00
}
list_for_each_safe ( cur , temp , & its - > collection_list ) {
2017-05-08 14:31:12 +03:00
struct its_collection * coll ;
coll = list_entry ( cur , struct its_collection , coll_list ) ;
2016-07-15 14:43:32 +03:00
list_del ( cur ) ;
2017-05-08 14:31:12 +03:00
kfree ( coll ) ;
2016-07-15 14:43:32 +03:00
}
mutex_unlock ( & its - > its_lock ) ;
2016-07-15 14:43:31 +03:00
kfree ( its ) ;
}
2016-12-20 09:36:35 +03:00
int vgic_its_has_attr_regs ( struct kvm_device * dev ,
struct kvm_device_attr * attr )
{
2016-12-20 11:33:13 +03:00
const struct vgic_register_region * region ;
gpa_t offset = attr - > attr ;
int align ;
align = ( offset < GITS_TYPER ) | | ( offset > = GITS_PIDR4 ) ? 0x3 : 0x7 ;
if ( offset & align )
return - EINVAL ;
region = vgic_find_mmio_region ( its_registers ,
ARRAY_SIZE ( its_registers ) ,
offset ) ;
if ( ! region )
return - ENXIO ;
return 0 ;
2016-12-20 09:36:35 +03:00
}
int vgic_its_attr_regs_access ( struct kvm_device * dev ,
struct kvm_device_attr * attr ,
u64 * reg , bool is_write )
{
2016-12-20 11:33:13 +03:00
const struct vgic_register_region * region ;
struct vgic_its * its ;
gpa_t addr , offset ;
unsigned int len ;
int align , ret = 0 ;
its = dev - > private ;
offset = attr - > attr ;
/*
* Although the spec supports upper / lower 32 - bit accesses to
* 64 - bit ITS registers , the userspace ABI requires 64 - bit
* accesses to all 64 - bit wide registers . We therefore only
* support 32 - bit accesses to GITS_CTLR , GITS_IIDR and GITS ID
* registers
*/
if ( ( offset < GITS_TYPER ) | | ( offset > = GITS_PIDR4 ) )
align = 0x3 ;
else
align = 0x7 ;
if ( offset & align )
return - EINVAL ;
mutex_lock ( & dev - > kvm - > lock ) ;
if ( IS_VGIC_ADDR_UNDEF ( its - > vgic_its_base ) ) {
ret = - ENXIO ;
goto out ;
}
region = vgic_find_mmio_region ( its_registers ,
ARRAY_SIZE ( its_registers ) ,
offset ) ;
if ( ! region ) {
ret = - ENXIO ;
goto out ;
}
if ( ! lock_all_vcpus ( dev - > kvm ) ) {
ret = - EBUSY ;
goto out ;
}
addr = its - > vgic_its_base + offset ;
len = region - > access_flags & VGIC_ACCESS_64bit ? 8 : 4 ;
if ( is_write ) {
if ( region - > uaccess_its_write )
ret = region - > uaccess_its_write ( dev - > kvm , its , addr ,
len , * reg ) ;
else
region - > its_write ( dev - > kvm , its , addr , len , * reg ) ;
} else {
* reg = region - > its_read ( dev - > kvm , its , addr , len ) ;
}
unlock_all_vcpus ( dev - > kvm ) ;
out :
mutex_unlock ( & dev - > kvm - > lock ) ;
return ret ;
2016-12-20 09:36:35 +03:00
}
2017-01-09 18:27:07 +03:00
static u32 compute_next_devid_offset ( struct list_head * h ,
struct its_device * dev )
2017-02-08 07:20:04 +03:00
{
struct its_device * next ;
u32 next_offset ;
if ( list_is_last ( & dev - > dev_list , h ) )
return 0 ;
next = list_next_entry ( dev , dev_list ) ;
next_offset = next - > device_id - dev - > device_id ;
return min_t ( u32 , next_offset , VITS_DTE_MAX_DEVID_OFFSET ) ;
}
2017-05-03 18:38:01 +03:00
static u32 compute_next_eventid_offset ( struct list_head * h , struct its_ite * ite )
2017-02-08 07:20:04 +03:00
{
struct its_ite * next ;
u32 next_offset ;
if ( list_is_last ( & ite - > ite_list , h ) )
return 0 ;
next = list_next_entry ( ite , ite_list ) ;
next_offset = next - > event_id - ite - > event_id ;
return min_t ( u32 , next_offset , VITS_ITE_MAX_EVENTID_OFFSET ) ;
}
/**
* entry_fn_t - Callback called on a table entry restore path
* @ its : its handle
* @ id : id of the entry
* @ entry : pointer to the entry
* @ opaque : pointer to an opaque data
*
* Return : < 0 on error , 0 if last element was identified , id offset to next
* element otherwise
*/
typedef int ( * entry_fn_t ) ( struct vgic_its * its , u32 id , void * entry ,
void * opaque ) ;
/**
* scan_its_table - Scan a contiguous table in guest RAM and applies a function
* to each entry
*
* @ its : its handle
* @ base : base gpa of the table
* @ size : size of the table in bytes
* @ esz : entry size in bytes
* @ start_id : the ID of the first entry in the table
* ( non zero for 2 d level tables )
* @ fn : function to apply on each entry
*
* Return : < 0 on error , 0 if last element was identified , 1 otherwise
* ( the last element may not be found on second level tables )
*/
2017-01-09 18:27:07 +03:00
static int scan_its_table ( struct vgic_its * its , gpa_t base , int size , int esz ,
int start_id , entry_fn_t fn , void * opaque )
2017-02-08 07:20:04 +03:00
{
void * entry = kzalloc ( esz , GFP_KERNEL ) ;
struct kvm * kvm = its - > dev - > kvm ;
unsigned long len = size ;
int id = start_id ;
gpa_t gpa = base ;
int ret ;
while ( len > 0 ) {
int next_offset ;
size_t byte_offset ;
ret = kvm_read_guest ( kvm , gpa , entry , esz ) ;
if ( ret )
goto out ;
next_offset = fn ( its , id , entry , opaque ) ;
if ( next_offset < = 0 ) {
ret = next_offset ;
goto out ;
}
byte_offset = next_offset * esz ;
id + = next_offset ;
gpa + = byte_offset ;
len - = byte_offset ;
}
ret = 1 ;
out :
kfree ( entry ) ;
return ret ;
}
2017-05-03 18:38:01 +03:00
/**
* vgic_its_save_ite - Save an interrupt translation entry at @ gpa
*/
static int vgic_its_save_ite ( struct vgic_its * its , struct its_device * dev ,
struct its_ite * ite , gpa_t gpa , int ite_esz )
{
struct kvm * kvm = its - > dev - > kvm ;
u32 next_offset ;
u64 val ;
next_offset = compute_next_eventid_offset ( & dev - > itt_head , ite ) ;
val = ( ( u64 ) next_offset < < KVM_ITS_ITE_NEXT_SHIFT ) |
( ( u64 ) ite - > lpi < < KVM_ITS_ITE_PINTID_SHIFT ) |
ite - > collection - > collection_id ;
val = cpu_to_le64 ( val ) ;
return kvm_write_guest ( kvm , gpa , & val , ite_esz ) ;
}
/**
* vgic_its_restore_ite - restore an interrupt translation entry
* @ event_id : id used for indexing
* @ ptr : pointer to the ITE entry
* @ opaque : pointer to the its_device
*/
static int vgic_its_restore_ite ( struct vgic_its * its , u32 event_id ,
void * ptr , void * opaque )
{
struct its_device * dev = ( struct its_device * ) opaque ;
struct its_collection * collection ;
struct kvm * kvm = its - > dev - > kvm ;
struct kvm_vcpu * vcpu = NULL ;
u64 val ;
u64 * p = ( u64 * ) ptr ;
struct vgic_irq * irq ;
u32 coll_id , lpi_id ;
struct its_ite * ite ;
u32 offset ;
val = * p ;
val = le64_to_cpu ( val ) ;
coll_id = val & KVM_ITS_ITE_ICID_MASK ;
lpi_id = ( val & KVM_ITS_ITE_PINTID_MASK ) > > KVM_ITS_ITE_PINTID_SHIFT ;
if ( ! lpi_id )
return 1 ; /* invalid entry, no choice but to scan next entry */
if ( lpi_id < VGIC_MIN_LPI )
return - EINVAL ;
offset = val > > KVM_ITS_ITE_NEXT_SHIFT ;
if ( event_id + offset > = BIT_ULL ( dev - > num_eventid_bits ) )
return - EINVAL ;
collection = find_collection ( its , coll_id ) ;
if ( ! collection )
return - EINVAL ;
ite = vgic_its_alloc_ite ( dev , collection , lpi_id , event_id ) ;
if ( IS_ERR ( ite ) )
return PTR_ERR ( ite ) ;
if ( its_is_collection_mapped ( collection ) )
vcpu = kvm_get_vcpu ( kvm , collection - > target_addr ) ;
irq = vgic_add_lpi ( kvm , lpi_id , vcpu ) ;
if ( IS_ERR ( irq ) )
return PTR_ERR ( irq ) ;
ite - > irq = irq ;
return offset ;
}
static int vgic_its_ite_cmp ( void * priv , struct list_head * a ,
struct list_head * b )
{
struct its_ite * itea = container_of ( a , struct its_ite , ite_list ) ;
struct its_ite * iteb = container_of ( b , struct its_ite , ite_list ) ;
if ( itea - > event_id < iteb - > event_id )
return - 1 ;
else
return 1 ;
}
2017-01-09 18:27:07 +03:00
static int vgic_its_save_itt ( struct vgic_its * its , struct its_device * device )
{
2017-05-03 18:38:01 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
gpa_t base = device - > itt_addr ;
struct its_ite * ite ;
int ret ;
int ite_esz = abi - > ite_esz ;
list_sort ( NULL , & device - > itt_head , vgic_its_ite_cmp ) ;
list_for_each_entry ( ite , & device - > itt_head , ite_list ) {
gpa_t gpa = base + ite - > event_id * ite_esz ;
ret = vgic_its_save_ite ( its , device , ite , gpa , ite_esz ) ;
if ( ret )
return ret ;
}
return 0 ;
2017-01-09 18:27:07 +03:00
}
static int vgic_its_restore_itt ( struct vgic_its * its , struct its_device * dev )
{
2017-05-03 18:38:01 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
gpa_t base = dev - > itt_addr ;
int ret ;
int ite_esz = abi - > ite_esz ;
size_t max_size = BIT_ULL ( dev - > num_eventid_bits ) * ite_esz ;
ret = scan_its_table ( its , base , max_size , ite_esz , 0 ,
vgic_its_restore_ite , dev ) ;
return ret ;
2017-01-09 18:27:07 +03:00
}
/**
* vgic_its_save_dte - Save a device table entry at a given GPA
*
* @ its : ITS handle
* @ dev : ITS device
* @ ptr : GPA
*/
static int vgic_its_save_dte ( struct vgic_its * its , struct its_device * dev ,
gpa_t ptr , int dte_esz )
{
struct kvm * kvm = its - > dev - > kvm ;
u64 val , itt_addr_field ;
u32 next_offset ;
itt_addr_field = dev - > itt_addr > > 8 ;
next_offset = compute_next_devid_offset ( & its - > device_list , dev ) ;
val = ( 1ULL < < KVM_ITS_DTE_VALID_SHIFT |
( ( u64 ) next_offset < < KVM_ITS_DTE_NEXT_SHIFT ) |
( itt_addr_field < < KVM_ITS_DTE_ITTADDR_SHIFT ) |
( dev - > num_eventid_bits - 1 ) ) ;
val = cpu_to_le64 ( val ) ;
return kvm_write_guest ( kvm , ptr , & val , dte_esz ) ;
}
/**
* vgic_its_restore_dte - restore a device table entry
*
* @ its : its handle
* @ id : device id the DTE corresponds to
* @ ptr : kernel VA where the 8 byte DTE is located
* @ opaque : unused
*
* Return : < 0 on error , 0 if the dte is the last one , id offset to the
* next dte otherwise
*/
static int vgic_its_restore_dte ( struct vgic_its * its , u32 id ,
void * ptr , void * opaque )
{
struct its_device * dev ;
gpa_t itt_addr ;
u8 num_eventid_bits ;
u64 entry = * ( u64 * ) ptr ;
bool valid ;
u32 offset ;
int ret ;
entry = le64_to_cpu ( entry ) ;
valid = entry > > KVM_ITS_DTE_VALID_SHIFT ;
num_eventid_bits = ( entry & KVM_ITS_DTE_SIZE_MASK ) + 1 ;
itt_addr = ( ( entry & KVM_ITS_DTE_ITTADDR_MASK )
> > KVM_ITS_DTE_ITTADDR_SHIFT ) < < 8 ;
if ( ! valid )
return 1 ;
/* dte entry is valid */
offset = ( entry & KVM_ITS_DTE_NEXT_MASK ) > > KVM_ITS_DTE_NEXT_SHIFT ;
dev = vgic_its_alloc_device ( its , id , itt_addr , num_eventid_bits ) ;
if ( IS_ERR ( dev ) )
return PTR_ERR ( dev ) ;
ret = vgic_its_restore_itt ( its , dev ) ;
2017-05-08 14:31:12 +03:00
if ( ret ) {
vgic_its_free_device ( its - > dev - > kvm , dev ) ;
2017-01-09 18:27:07 +03:00
return ret ;
2017-05-08 14:31:12 +03:00
}
2017-01-09 18:27:07 +03:00
return offset ;
}
static int vgic_its_device_cmp ( void * priv , struct list_head * a ,
struct list_head * b )
{
struct its_device * deva = container_of ( a , struct its_device , dev_list ) ;
struct its_device * devb = container_of ( b , struct its_device , dev_list ) ;
if ( deva - > device_id < devb - > device_id )
return - 1 ;
else
return 1 ;
}
2016-12-24 20:48:04 +03:00
/**
* vgic_its_save_device_tables - Save the device table and all ITT
* into guest RAM
2017-01-09 18:27:07 +03:00
*
* L1 / L2 handling is hidden by vgic_its_check_id ( ) helper which directly
* returns the GPA of the device entry
2016-12-24 20:48:04 +03:00
*/
static int vgic_its_save_device_tables ( struct vgic_its * its )
{
2017-01-09 18:27:07 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
struct its_device * dev ;
int dte_esz = abi - > dte_esz ;
u64 baser ;
baser = its - > baser_device_table ;
list_sort ( NULL , & its - > device_list , vgic_its_device_cmp ) ;
list_for_each_entry ( dev , & its - > device_list , dev_list ) {
int ret ;
gpa_t eaddr ;
if ( ! vgic_its_check_id ( its , baser ,
dev - > device_id , & eaddr ) )
return - EINVAL ;
ret = vgic_its_save_itt ( its , dev ) ;
if ( ret )
return ret ;
ret = vgic_its_save_dte ( its , dev , eaddr , dte_esz ) ;
if ( ret )
return ret ;
}
return 0 ;
}
/**
* handle_l1_dte - callback used for L1 device table entries ( 2 stage case )
*
* @ its : its handle
* @ id : index of the entry in the L1 table
* @ addr : kernel VA
* @ opaque : unused
*
* L1 table entries are scanned by steps of 1 entry
* Return < 0 if error , 0 if last dte was found when scanning the L2
* table , + 1 otherwise ( meaning next L1 entry must be scanned )
*/
static int handle_l1_dte ( struct vgic_its * its , u32 id , void * addr ,
void * opaque )
{
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
int l2_start_id = id * ( SZ_64K / abi - > dte_esz ) ;
u64 entry = * ( u64 * ) addr ;
int dte_esz = abi - > dte_esz ;
gpa_t gpa ;
int ret ;
entry = le64_to_cpu ( entry ) ;
if ( ! ( entry & KVM_ITS_L1E_VALID_MASK ) )
return 1 ;
gpa = entry & KVM_ITS_L1E_ADDR_MASK ;
ret = scan_its_table ( its , gpa , SZ_64K , dte_esz ,
l2_start_id , vgic_its_restore_dte , NULL ) ;
if ( ret < = 0 )
return ret ;
return 1 ;
2016-12-24 20:48:04 +03:00
}
/**
* vgic_its_restore_device_tables - Restore the device table and all ITT
* from guest RAM to internal data structs
*/
static int vgic_its_restore_device_tables ( struct vgic_its * its )
{
2017-01-09 18:27:07 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
u64 baser = its - > baser_device_table ;
int l1_esz , ret ;
int l1_tbl_size = GITS_BASER_NR_PAGES ( baser ) * SZ_64K ;
gpa_t l1_gpa ;
if ( ! ( baser & GITS_BASER_VALID ) )
return 0 ;
l1_gpa = BASER_ADDRESS ( baser ) ;
if ( baser & GITS_BASER_INDIRECT ) {
l1_esz = GITS_LVL1_ENTRY_SIZE ;
ret = scan_its_table ( its , l1_gpa , l1_tbl_size , l1_esz , 0 ,
handle_l1_dte , NULL ) ;
} else {
l1_esz = abi - > dte_esz ;
ret = scan_its_table ( its , l1_gpa , l1_tbl_size , l1_esz , 0 ,
vgic_its_restore_dte , NULL ) ;
}
if ( ret > 0 )
ret = - EINVAL ;
return ret ;
2016-12-24 20:48:04 +03:00
}
2017-01-09 18:19:41 +03:00
static int vgic_its_save_cte ( struct vgic_its * its ,
struct its_collection * collection ,
gpa_t gpa , int esz )
{
u64 val ;
val = ( 1ULL < < KVM_ITS_CTE_VALID_SHIFT |
( ( u64 ) collection - > target_addr < < KVM_ITS_CTE_RDBASE_SHIFT ) |
collection - > collection_id ) ;
val = cpu_to_le64 ( val ) ;
return kvm_write_guest ( its - > dev - > kvm , gpa , & val , esz ) ;
}
static int vgic_its_restore_cte ( struct vgic_its * its , gpa_t gpa , int esz )
{
struct its_collection * collection ;
struct kvm * kvm = its - > dev - > kvm ;
u32 target_addr , coll_id ;
u64 val ;
int ret ;
BUG_ON ( esz > sizeof ( val ) ) ;
ret = kvm_read_guest ( kvm , gpa , & val , esz ) ;
if ( ret )
return ret ;
val = le64_to_cpu ( val ) ;
if ( ! ( val & KVM_ITS_CTE_VALID_MASK ) )
return 0 ;
target_addr = ( u32 ) ( val > > KVM_ITS_CTE_RDBASE_SHIFT ) ;
coll_id = val & KVM_ITS_CTE_ICID_MASK ;
if ( target_addr > = atomic_read ( & kvm - > online_vcpus ) )
return - EINVAL ;
collection = find_collection ( its , coll_id ) ;
if ( collection )
return - EEXIST ;
ret = vgic_its_alloc_collection ( its , & collection , coll_id ) ;
if ( ret )
return ret ;
collection - > target_addr = target_addr ;
return 1 ;
}
2016-12-24 20:48:04 +03:00
/**
* vgic_its_save_collection_table - Save the collection table into
* guest RAM
*/
static int vgic_its_save_collection_table ( struct vgic_its * its )
{
2017-01-09 18:19:41 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
struct its_collection * collection ;
u64 val ;
gpa_t gpa ;
size_t max_size , filled = 0 ;
int ret , cte_esz = abi - > cte_esz ;
gpa = BASER_ADDRESS ( its - > baser_coll_table ) ;
if ( ! gpa )
return 0 ;
max_size = GITS_BASER_NR_PAGES ( its - > baser_coll_table ) * SZ_64K ;
list_for_each_entry ( collection , & its - > collection_list , coll_list ) {
ret = vgic_its_save_cte ( its , collection , gpa , cte_esz ) ;
if ( ret )
return ret ;
gpa + = cte_esz ;
filled + = cte_esz ;
}
if ( filled = = max_size )
return 0 ;
/*
* table is not fully filled , add a last dummy element
* with valid bit unset
*/
val = 0 ;
BUG_ON ( cte_esz > sizeof ( val ) ) ;
ret = kvm_write_guest ( its - > dev - > kvm , gpa , & val , cte_esz ) ;
return ret ;
2016-12-24 20:48:04 +03:00
}
/**
* vgic_its_restore_collection_table - reads the collection table
* in guest memory and restores the ITS internal state . Requires the
* BASER registers to be restored before .
*/
static int vgic_its_restore_collection_table ( struct vgic_its * its )
{
2017-01-09 18:19:41 +03:00
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
int cte_esz = abi - > cte_esz ;
size_t max_size , read = 0 ;
gpa_t gpa ;
int ret ;
if ( ! ( its - > baser_coll_table & GITS_BASER_VALID ) )
return 0 ;
gpa = BASER_ADDRESS ( its - > baser_coll_table ) ;
max_size = GITS_BASER_NR_PAGES ( its - > baser_coll_table ) * SZ_64K ;
while ( read < max_size ) {
ret = vgic_its_restore_cte ( its , gpa , cte_esz ) ;
if ( ret < = 0 )
break ;
gpa + = cte_esz ;
read + = cte_esz ;
}
return ret ;
2016-12-24 20:48:04 +03:00
}
2017-04-13 10:06:20 +03:00
/**
* vgic_its_save_tables_v0 - Save the ITS tables into guest ARM
* according to v0 ABI
*/
static int vgic_its_save_tables_v0 ( struct vgic_its * its )
{
2016-12-24 20:48:04 +03:00
struct kvm * kvm = its - > dev - > kvm ;
int ret ;
mutex_lock ( & kvm - > lock ) ;
mutex_lock ( & its - > its_lock ) ;
if ( ! lock_all_vcpus ( kvm ) ) {
mutex_unlock ( & its - > its_lock ) ;
mutex_unlock ( & kvm - > lock ) ;
return - EBUSY ;
}
ret = vgic_its_save_device_tables ( its ) ;
if ( ret )
goto out ;
ret = vgic_its_save_collection_table ( its ) ;
out :
unlock_all_vcpus ( kvm ) ;
mutex_unlock ( & its - > its_lock ) ;
mutex_unlock ( & kvm - > lock ) ;
return ret ;
2017-04-13 10:06:20 +03:00
}
/**
* vgic_its_restore_tables_v0 - Restore the ITS tables from guest RAM
* to internal data structs according to V0 ABI
*
*/
static int vgic_its_restore_tables_v0 ( struct vgic_its * its )
{
2016-12-24 20:48:04 +03:00
struct kvm * kvm = its - > dev - > kvm ;
int ret ;
mutex_lock ( & kvm - > lock ) ;
mutex_lock ( & its - > its_lock ) ;
if ( ! lock_all_vcpus ( kvm ) ) {
mutex_unlock ( & its - > its_lock ) ;
mutex_unlock ( & kvm - > lock ) ;
return - EBUSY ;
}
ret = vgic_its_restore_collection_table ( its ) ;
if ( ret )
goto out ;
ret = vgic_its_restore_device_tables ( its ) ;
out :
unlock_all_vcpus ( kvm ) ;
mutex_unlock ( & its - > its_lock ) ;
mutex_unlock ( & kvm - > lock ) ;
2017-05-08 15:43:45 +03:00
return ret ;
2017-04-13 10:06:20 +03:00
}
static int vgic_its_commit_v0 ( struct vgic_its * its )
{
const struct vgic_its_abi * abi ;
abi = vgic_its_get_abi ( its ) ;
its - > baser_coll_table & = ~ GITS_BASER_ENTRY_SIZE_MASK ;
its - > baser_device_table & = ~ GITS_BASER_ENTRY_SIZE_MASK ;
its - > baser_coll_table | = ( GIC_ENCODE_SZ ( abi - > cte_esz , 5 )
< < GITS_BASER_ENTRY_SIZE_SHIFT ) ;
its - > baser_device_table | = ( GIC_ENCODE_SZ ( abi - > dte_esz , 5 )
< < GITS_BASER_ENTRY_SIZE_SHIFT ) ;
return 0 ;
}
2016-07-15 14:43:31 +03:00
static int vgic_its_has_attr ( struct kvm_device * dev ,
struct kvm_device_attr * attr )
{
switch ( attr - > group ) {
case KVM_DEV_ARM_VGIC_GRP_ADDR :
switch ( attr - > attr ) {
case KVM_VGIC_ITS_ADDR_TYPE :
return 0 ;
}
break ;
case KVM_DEV_ARM_VGIC_GRP_CTRL :
switch ( attr - > attr ) {
case KVM_DEV_ARM_VGIC_CTRL_INIT :
return 0 ;
2016-12-24 20:48:04 +03:00
case KVM_DEV_ARM_ITS_SAVE_TABLES :
return 0 ;
case KVM_DEV_ARM_ITS_RESTORE_TABLES :
return 0 ;
2016-07-15 14:43:31 +03:00
}
break ;
2016-12-20 09:36:35 +03:00
case KVM_DEV_ARM_VGIC_GRP_ITS_REGS :
return vgic_its_has_attr_regs ( dev , attr ) ;
2016-07-15 14:43:31 +03:00
}
return - ENXIO ;
}
static int vgic_its_set_attr ( struct kvm_device * dev ,
struct kvm_device_attr * attr )
{
struct vgic_its * its = dev - > private ;
int ret ;
switch ( attr - > group ) {
case KVM_DEV_ARM_VGIC_GRP_ADDR : {
u64 __user * uaddr = ( u64 __user * ) ( long ) attr - > addr ;
unsigned long type = ( unsigned long ) attr - > attr ;
u64 addr ;
if ( type ! = KVM_VGIC_ITS_ADDR_TYPE )
return - ENODEV ;
if ( copy_from_user ( & addr , uaddr , sizeof ( addr ) ) )
return - EFAULT ;
ret = vgic_check_ioaddr ( dev - > kvm , & its - > vgic_its_base ,
addr , SZ_64K ) ;
if ( ret )
return ret ;
2017-05-08 14:14:57 +03:00
return vgic_register_its_iodev ( dev - > kvm , its , addr ) ;
2016-07-15 14:43:31 +03:00
}
2016-12-24 20:48:04 +03:00
case KVM_DEV_ARM_VGIC_GRP_CTRL : {
const struct vgic_its_abi * abi = vgic_its_get_abi ( its ) ;
2016-07-15 14:43:31 +03:00
switch ( attr - > attr ) {
case KVM_DEV_ARM_VGIC_CTRL_INIT :
2017-05-08 20:15:40 +03:00
/* Nothing to do */
2016-08-08 18:45:43 +03:00
return 0 ;
2016-12-24 20:48:04 +03:00
case KVM_DEV_ARM_ITS_SAVE_TABLES :
return abi - > save_tables ( its ) ;
case KVM_DEV_ARM_ITS_RESTORE_TABLES :
return abi - > restore_tables ( its ) ;
2016-07-15 14:43:31 +03:00
}
2016-12-24 20:48:04 +03:00
}
2016-12-20 09:36:35 +03:00
case KVM_DEV_ARM_VGIC_GRP_ITS_REGS : {
u64 __user * uaddr = ( u64 __user * ) ( long ) attr - > addr ;
u64 reg ;
if ( get_user ( reg , uaddr ) )
return - EFAULT ;
return vgic_its_attr_regs_access ( dev , attr , & reg , true ) ;
}
2016-07-15 14:43:31 +03:00
}
return - ENXIO ;
}
static int vgic_its_get_attr ( struct kvm_device * dev ,
struct kvm_device_attr * attr )
{
switch ( attr - > group ) {
case KVM_DEV_ARM_VGIC_GRP_ADDR : {
struct vgic_its * its = dev - > private ;
u64 addr = its - > vgic_its_base ;
u64 __user * uaddr = ( u64 __user * ) ( long ) attr - > addr ;
unsigned long type = ( unsigned long ) attr - > attr ;
if ( type ! = KVM_VGIC_ITS_ADDR_TYPE )
return - ENODEV ;
if ( copy_to_user ( uaddr , & addr , sizeof ( addr ) ) )
return - EFAULT ;
break ;
2016-12-20 09:36:35 +03:00
}
case KVM_DEV_ARM_VGIC_GRP_ITS_REGS : {
u64 __user * uaddr = ( u64 __user * ) ( long ) attr - > addr ;
u64 reg ;
int ret ;
ret = vgic_its_attr_regs_access ( dev , attr , & reg , false ) ;
if ( ret )
return ret ;
return put_user ( reg , uaddr ) ;
}
2016-07-15 14:43:31 +03:00
default :
return - ENXIO ;
}
return 0 ;
}
static struct kvm_device_ops kvm_arm_vgic_its_ops = {
. name = " kvm-arm-vgic-its " ,
. create = vgic_its_create ,
. destroy = vgic_its_destroy ,
. set_attr = vgic_its_set_attr ,
. get_attr = vgic_its_get_attr ,
. has_attr = vgic_its_has_attr ,
} ;
int kvm_vgic_register_its_device ( void )
{
return kvm_register_device_ops ( & kvm_arm_vgic_its_ops ,
KVM_DEV_TYPE_ARM_VGIC_ITS ) ;
}