2007-07-06 12:20:49 +03:00
/*
* 8259 interrupt controller emulation
*
* Copyright ( c ) 2003 - 2004 Fabrice Bellard
* Copyright ( c ) 2007 Intel Corporation
*
* Permission is hereby granted , free of charge , to any person obtaining a copy
* of this software and associated documentation files ( the " Software " ) , to deal
* in the Software without restriction , including without limitation the rights
* to use , copy , modify , merge , publish , distribute , sublicense , and / or sell
* copies of the Software , and to permit persons to whom the Software is
* furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
* LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM ,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE .
* Authors :
* Yaozu ( Eddie ) Dong < Eddie . dong @ intel . com >
* Port from Qemu .
*/
# include <linux/mm.h>
2008-12-21 22:48:32 +02:00
# include <linux/bitops.h>
2007-07-06 12:20:49 +03:00
# include "irq.h"
2007-12-16 11:02:48 +02:00
# include <linux/kvm_host.h>
2007-07-06 12:20:49 +03:00
2008-12-21 22:48:32 +02:00
static void pic_lock ( struct kvm_pic * s )
2009-02-21 02:18:13 +01:00
__acquires ( & s - > lock )
2008-12-21 22:48:32 +02:00
{
spin_lock ( & s - > lock ) ;
}
static void pic_unlock ( struct kvm_pic * s )
2009-02-21 02:18:13 +01:00
__releases ( & s - > lock )
2008-12-21 22:48:32 +02:00
{
struct kvm * kvm = s - > kvm ;
unsigned acks = s - > pending_acks ;
bool wakeup = s - > wakeup_needed ;
struct kvm_vcpu * vcpu ;
s - > pending_acks = 0 ;
s - > wakeup_needed = false ;
spin_unlock ( & s - > lock ) ;
while ( acks ) {
2009-01-27 15:12:38 -02:00
kvm_notify_acked_irq ( kvm , SELECT_PIC ( __ffs ( acks ) ) ,
__ffs ( acks ) ) ;
2008-12-21 22:48:32 +02:00
acks & = acks - 1 ;
}
if ( wakeup ) {
2009-06-09 15:56:26 +03:00
vcpu = s - > kvm - > bsp_vcpu ;
2008-12-21 22:48:32 +02:00
if ( vcpu )
kvm_vcpu_kick ( vcpu ) ;
}
}
2008-07-07 14:45:39 +03:00
static void pic_clear_isr ( struct kvm_kpic_state * s , int irq )
{
s - > isr & = ~ ( 1 < < irq ) ;
2008-09-24 20:28:34 -03:00
s - > isr_ack | = ( 1 < < irq ) ;
}
void kvm_pic_clear_isr_ack ( struct kvm * kvm )
{
struct kvm_pic * s = pic_irqchip ( kvm ) ;
2009-06-04 15:08:21 -03:00
pic_lock ( s ) ;
2008-09-24 20:28:34 -03:00
s - > pics [ 0 ] . isr_ack = 0xff ;
s - > pics [ 1 ] . isr_ack = 0xff ;
2009-06-04 15:08:21 -03:00
pic_unlock ( s ) ;
2008-07-07 14:45:39 +03:00
}
2007-07-06 12:20:49 +03:00
/*
* set irq level . If an edge is detected , then the IRR is set to 1
*/
2009-02-04 17:28:14 +02:00
static inline int pic_set_irq1 ( struct kvm_kpic_state * s , int irq , int level )
2007-07-06 12:20:49 +03:00
{
2009-02-04 17:28:14 +02:00
int mask , ret = 1 ;
2007-07-06 12:20:49 +03:00
mask = 1 < < irq ;
if ( s - > elcr & mask ) /* level triggered */
if ( level ) {
2009-02-04 17:28:14 +02:00
ret = ! ( s - > irr & mask ) ;
2007-07-06 12:20:49 +03:00
s - > irr | = mask ;
s - > last_irr | = mask ;
} else {
s - > irr & = ~ mask ;
s - > last_irr & = ~ mask ;
}
else /* edge triggered */
if ( level ) {
2009-02-04 17:28:14 +02:00
if ( ( s - > last_irr & mask ) = = 0 ) {
ret = ! ( s - > irr & mask ) ;
2007-07-06 12:20:49 +03:00
s - > irr | = mask ;
2009-02-04 17:28:14 +02:00
}
2007-07-06 12:20:49 +03:00
s - > last_irr | = mask ;
} else
s - > last_irr & = ~ mask ;
2009-02-04 17:28:14 +02:00
return ( s - > imr & mask ) ? - 1 : ret ;
2007-07-06 12:20:49 +03:00
}
/*
* return the highest priority found in mask ( highest = smallest
* number ) . Return 8 if no irq
*/
static inline int get_priority ( struct kvm_kpic_state * s , int mask )
{
int priority ;
if ( mask = = 0 )
return 8 ;
priority = 0 ;
while ( ( mask & ( 1 < < ( ( priority + s - > priority_add ) & 7 ) ) ) = = 0 )
priority + + ;
return priority ;
}
/*
* return the pic wanted interrupt . return - 1 if none
*/
static int pic_get_irq ( struct kvm_kpic_state * s )
{
int mask , cur_priority , priority ;
mask = s - > irr & ~ s - > imr ;
priority = get_priority ( s , mask ) ;
if ( priority = = 8 )
return - 1 ;
/*
* compute current priority . If special fully nested mode on the
* master , the IRQ coming from the slave is not taken into account
* for the priority computation .
*/
mask = s - > isr ;
if ( s - > special_fully_nested_mode & & s = = & s - > pics_state - > pics [ 0 ] )
mask & = ~ ( 1 < < 2 ) ;
cur_priority = get_priority ( s , mask ) ;
if ( priority < cur_priority )
/*
* higher priority found : an irq should be generated
*/
return ( priority + s - > priority_add ) & 7 ;
else
return - 1 ;
}
/*
* raise irq to CPU if necessary . must be called every time the active
* irq may change
*/
static void pic_update_irq ( struct kvm_pic * s )
{
int irq2 , irq ;
irq2 = pic_get_irq ( & s - > pics [ 1 ] ) ;
if ( irq2 > = 0 ) {
/*
* if irq request by slave pic , signal master PIC
*/
pic_set_irq1 ( & s - > pics [ 0 ] , 2 , 1 ) ;
pic_set_irq1 ( & s - > pics [ 0 ] , 2 , 0 ) ;
}
irq = pic_get_irq ( & s - > pics [ 0 ] ) ;
if ( irq > = 0 )
s - > irq_request ( s - > irq_request_opaque , 1 ) ;
else
s - > irq_request ( s - > irq_request_opaque , 0 ) ;
}
2007-07-26 11:05:18 +03:00
void kvm_pic_update_irq ( struct kvm_pic * s )
{
2008-12-21 22:48:32 +02:00
pic_lock ( s ) ;
2007-07-26 11:05:18 +03:00
pic_update_irq ( s ) ;
2008-12-21 22:48:32 +02:00
pic_unlock ( s ) ;
2007-07-26 11:05:18 +03:00
}
2009-02-04 17:28:14 +02:00
int kvm_pic_set_irq ( void * opaque , int irq , int level )
2007-07-06 12:20:49 +03:00
{
struct kvm_pic * s = opaque ;
2009-02-04 17:28:14 +02:00
int ret = - 1 ;
2007-07-06 12:20:49 +03:00
2008-12-21 22:48:32 +02:00
pic_lock ( s ) ;
2008-07-06 17:15:07 +03:00
if ( irq > = 0 & & irq < PIC_NUM_PINS ) {
2009-02-04 17:28:14 +02:00
ret = pic_set_irq1 ( & s - > pics [ irq > > 3 ] , irq & 7 , level ) ;
2008-07-06 17:15:07 +03:00
pic_update_irq ( s ) ;
}
2008-12-21 22:48:32 +02:00
pic_unlock ( s ) ;
2009-02-04 17:28:14 +02:00
return ret ;
2007-07-06 12:20:49 +03:00
}
/*
* acknowledge interrupt ' irq '
*/
static inline void pic_intack ( struct kvm_kpic_state * s , int irq )
{
2008-07-07 14:45:39 +03:00
s - > isr | = 1 < < irq ;
2007-07-06 12:20:49 +03:00
if ( s - > auto_eoi ) {
if ( s - > rotate_on_auto_eoi )
s - > priority_add = ( irq + 1 ) & 7 ;
2008-07-07 14:45:39 +03:00
pic_clear_isr ( s , irq ) ;
}
2007-07-06 12:20:49 +03:00
/*
* We don ' t clear a level sensitive interrupt here
*/
if ( ! ( s - > elcr & ( 1 < < irq ) ) )
s - > irr & = ~ ( 1 < < irq ) ;
}
2008-07-26 17:01:00 -03:00
int kvm_pic_read_irq ( struct kvm * kvm )
2007-07-06 12:20:49 +03:00
{
int irq , irq2 , intno ;
2008-07-26 17:01:00 -03:00
struct kvm_pic * s = pic_irqchip ( kvm ) ;
2007-07-06 12:20:49 +03:00
2008-12-21 22:48:32 +02:00
pic_lock ( s ) ;
2007-07-06 12:20:49 +03:00
irq = pic_get_irq ( & s - > pics [ 0 ] ) ;
if ( irq > = 0 ) {
pic_intack ( & s - > pics [ 0 ] , irq ) ;
if ( irq = = 2 ) {
irq2 = pic_get_irq ( & s - > pics [ 1 ] ) ;
if ( irq2 > = 0 )
pic_intack ( & s - > pics [ 1 ] , irq2 ) ;
else
/*
* spurious IRQ on slave controller
*/
irq2 = 7 ;
intno = s - > pics [ 1 ] . irq_base + irq2 ;
irq = irq2 + 8 ;
} else
intno = s - > pics [ 0 ] . irq_base + irq ;
} else {
/*
* spurious IRQ on host controller
*/
irq = 7 ;
intno = s - > pics [ 0 ] . irq_base + irq ;
}
pic_update_irq ( s ) ;
2008-12-21 22:48:32 +02:00
pic_unlock ( s ) ;
2009-01-27 15:12:38 -02:00
kvm_notify_acked_irq ( kvm , SELECT_PIC ( irq ) , irq ) ;
2007-07-06 12:20:49 +03:00
return intno ;
}
2007-10-10 12:14:25 +02:00
void kvm_pic_reset ( struct kvm_kpic_state * s )
2007-07-06 12:20:49 +03:00
{
2008-12-21 22:48:32 +02:00
int irq , irqbase , n ;
2008-07-26 17:01:00 -03:00
struct kvm * kvm = s - > pics_state - > irq_request_opaque ;
2009-06-09 15:56:26 +03:00
struct kvm_vcpu * vcpu0 = kvm - > bsp_vcpu ;
2008-07-26 17:01:00 -03:00
2008-08-14 20:53:25 -03:00
if ( s = = & s - > pics_state - > pics [ 0 ] )
irqbase = 0 ;
else
irqbase = 8 ;
for ( irq = 0 ; irq < PIC_NUM_PINS / 2 ; irq + + ) {
if ( vcpu0 & & kvm_apic_accept_pic_intr ( vcpu0 ) )
2008-12-21 22:48:32 +02:00
if ( s - > irr & ( 1 < < irq ) | | s - > isr & ( 1 < < irq ) ) {
n = irq + irqbase ;
s - > pics_state - > pending_acks | = 1 < < n ;
}
2008-07-26 17:01:00 -03:00
}
2007-07-06 12:20:49 +03:00
s - > last_irr = 0 ;
s - > irr = 0 ;
s - > imr = 0 ;
s - > isr = 0 ;
2008-09-24 20:28:34 -03:00
s - > isr_ack = 0xff ;
2007-07-06 12:20:49 +03:00
s - > priority_add = 0 ;
s - > irq_base = 0 ;
s - > read_reg_select = 0 ;
s - > poll = 0 ;
s - > special_mask = 0 ;
s - > init_state = 0 ;
s - > auto_eoi = 0 ;
s - > rotate_on_auto_eoi = 0 ;
s - > special_fully_nested_mode = 0 ;
s - > init4 = 0 ;
}
static void pic_ioport_write ( void * opaque , u32 addr , u32 val )
{
struct kvm_kpic_state * s = opaque ;
int priority , cmd , irq ;
addr & = 1 ;
if ( addr = = 0 ) {
if ( val & 0x10 ) {
2007-10-10 12:14:25 +02:00
kvm_pic_reset ( s ) ; /* init */
2007-07-06 12:20:49 +03:00
/*
* deassert a pending interrupt
*/
s - > pics_state - > irq_request ( s - > pics_state - >
irq_request_opaque , 0 ) ;
s - > init_state = 1 ;
s - > init4 = val & 1 ;
if ( val & 0x02 )
printk ( KERN_ERR " single mode not supported " ) ;
if ( val & 0x08 )
printk ( KERN_ERR
" level sensitive irq not supported " ) ;
} else if ( val & 0x08 ) {
if ( val & 0x04 )
s - > poll = 1 ;
if ( val & 0x02 )
s - > read_reg_select = val & 1 ;
if ( val & 0x40 )
s - > special_mask = ( val > > 5 ) & 1 ;
} else {
cmd = val > > 5 ;
switch ( cmd ) {
case 0 :
case 4 :
s - > rotate_on_auto_eoi = cmd > > 2 ;
break ;
case 1 : /* end of interrupt */
case 5 :
priority = get_priority ( s , s - > isr ) ;
if ( priority ! = 8 ) {
irq = ( priority + s - > priority_add ) & 7 ;
2008-07-07 14:45:39 +03:00
pic_clear_isr ( s , irq ) ;
2007-07-06 12:20:49 +03:00
if ( cmd = = 5 )
s - > priority_add = ( irq + 1 ) & 7 ;
pic_update_irq ( s - > pics_state ) ;
}
break ;
case 3 :
irq = val & 7 ;
2008-07-07 14:45:39 +03:00
pic_clear_isr ( s , irq ) ;
2007-07-06 12:20:49 +03:00
pic_update_irq ( s - > pics_state ) ;
break ;
case 6 :
s - > priority_add = ( val + 1 ) & 7 ;
pic_update_irq ( s - > pics_state ) ;
break ;
case 7 :
irq = val & 7 ;
s - > priority_add = ( irq + 1 ) & 7 ;
2008-07-07 14:45:39 +03:00
pic_clear_isr ( s , irq ) ;
2007-07-06 12:20:49 +03:00
pic_update_irq ( s - > pics_state ) ;
break ;
default :
break ; /* no operation */
}
}
} else
switch ( s - > init_state ) {
case 0 : /* normal mode */
s - > imr = val ;
pic_update_irq ( s - > pics_state ) ;
break ;
case 1 :
s - > irq_base = val & 0xf8 ;
s - > init_state = 2 ;
break ;
case 2 :
if ( s - > init4 )
s - > init_state = 3 ;
else
s - > init_state = 0 ;
break ;
case 3 :
s - > special_fully_nested_mode = ( val > > 4 ) & 1 ;
s - > auto_eoi = ( val > > 1 ) & 1 ;
s - > init_state = 0 ;
break ;
}
}
static u32 pic_poll_read ( struct kvm_kpic_state * s , u32 addr1 )
{
int ret ;
ret = pic_get_irq ( s ) ;
if ( ret > = 0 ) {
if ( addr1 > > 7 ) {
s - > pics_state - > pics [ 0 ] . isr & = ~ ( 1 < < 2 ) ;
s - > pics_state - > pics [ 0 ] . irr & = ~ ( 1 < < 2 ) ;
}
s - > irr & = ~ ( 1 < < ret ) ;
2008-07-07 14:45:39 +03:00
pic_clear_isr ( s , ret ) ;
2007-07-06 12:20:49 +03:00
if ( addr1 > > 7 | | ret ! = 2 )
pic_update_irq ( s - > pics_state ) ;
} else {
ret = 0x07 ;
pic_update_irq ( s - > pics_state ) ;
}
return ret ;
}
static u32 pic_ioport_read ( void * opaque , u32 addr1 )
{
struct kvm_kpic_state * s = opaque ;
unsigned int addr ;
int ret ;
addr = addr1 ;
addr & = 1 ;
if ( s - > poll ) {
ret = pic_poll_read ( s , addr1 ) ;
s - > poll = 0 ;
} else
if ( addr = = 0 )
if ( s - > read_reg_select )
ret = s - > isr ;
else
ret = s - > irr ;
else
ret = s - > imr ;
return ret ;
}
static void elcr_ioport_write ( void * opaque , u32 addr , u32 val )
{
struct kvm_kpic_state * s = opaque ;
s - > elcr = val & s - > elcr_mask ;
}
static u32 elcr_ioport_read ( void * opaque , u32 addr1 )
{
struct kvm_kpic_state * s = opaque ;
return s - > elcr ;
}
2008-05-30 16:05:53 +02:00
static int picdev_in_range ( struct kvm_io_device * this , gpa_t addr ,
int len , int is_write )
2007-07-06 12:20:49 +03:00
{
switch ( addr ) {
case 0x20 :
case 0x21 :
case 0xa0 :
case 0xa1 :
case 0x4d0 :
case 0x4d1 :
return 1 ;
default :
return 0 ;
}
}
2009-06-01 12:54:50 -04:00
static inline struct kvm_pic * to_pic ( struct kvm_io_device * dev )
{
return container_of ( dev , struct kvm_pic , dev ) ;
}
2007-07-06 12:20:49 +03:00
static void picdev_write ( struct kvm_io_device * this ,
gpa_t addr , int len , const void * val )
{
2009-06-01 12:54:50 -04:00
struct kvm_pic * s = to_pic ( this ) ;
2007-07-06 12:20:49 +03:00
unsigned char data = * ( unsigned char * ) val ;
if ( len ! = 1 ) {
if ( printk_ratelimit ( ) )
printk ( KERN_ERR " PIC: non byte write \n " ) ;
return ;
}
2008-12-21 22:48:32 +02:00
pic_lock ( s ) ;
2007-07-06 12:20:49 +03:00
switch ( addr ) {
case 0x20 :
case 0x21 :
case 0xa0 :
case 0xa1 :
pic_ioport_write ( & s - > pics [ addr > > 7 ] , addr , data ) ;
break ;
case 0x4d0 :
case 0x4d1 :
elcr_ioport_write ( & s - > pics [ addr & 1 ] , addr , data ) ;
break ;
}
2008-12-21 22:48:32 +02:00
pic_unlock ( s ) ;
2007-07-06 12:20:49 +03:00
}
static void picdev_read ( struct kvm_io_device * this ,
gpa_t addr , int len , void * val )
{
2009-06-01 12:54:50 -04:00
struct kvm_pic * s = to_pic ( this ) ;
2007-07-06 12:20:49 +03:00
unsigned char data = 0 ;
if ( len ! = 1 ) {
if ( printk_ratelimit ( ) )
printk ( KERN_ERR " PIC: non byte read \n " ) ;
return ;
}
2008-12-21 22:48:32 +02:00
pic_lock ( s ) ;
2007-07-06 12:20:49 +03:00
switch ( addr ) {
case 0x20 :
case 0x21 :
case 0xa0 :
case 0xa1 :
data = pic_ioport_read ( & s - > pics [ addr > > 7 ] , addr ) ;
break ;
case 0x4d0 :
case 0x4d1 :
data = elcr_ioport_read ( & s - > pics [ addr & 1 ] , addr ) ;
break ;
}
* ( unsigned char * ) val = data ;
2008-12-21 22:48:32 +02:00
pic_unlock ( s ) ;
2007-07-06 12:20:49 +03:00
}
/*
* callback when PIC0 irq status changed
*/
static void pic_irq_request ( void * opaque , int level )
{
struct kvm * kvm = opaque ;
2009-06-09 15:56:26 +03:00
struct kvm_vcpu * vcpu = kvm - > bsp_vcpu ;
2008-09-24 20:28:34 -03:00
struct kvm_pic * s = pic_irqchip ( kvm ) ;
int irq = pic_get_irq ( & s - > pics [ 0 ] ) ;
2007-07-06 12:20:49 +03:00
2008-09-24 20:28:34 -03:00
s - > output = level ;
if ( vcpu & & level & & ( s - > pics [ 0 ] . isr_ack & ( 1 < < irq ) ) ) {
s - > pics [ 0 ] . isr_ack & = ~ ( 1 < < irq ) ;
2008-12-21 22:48:32 +02:00
s - > wakeup_needed = true ;
2008-09-24 20:28:34 -03:00
}
2007-07-06 12:20:49 +03:00
}
2009-06-01 12:54:50 -04:00
static const struct kvm_io_device_ops picdev_ops = {
. read = picdev_read ,
. write = picdev_write ,
. in_range = picdev_in_range ,
} ;
2007-07-06 12:20:49 +03:00
struct kvm_pic * kvm_create_pic ( struct kvm * kvm )
{
struct kvm_pic * s ;
s = kzalloc ( sizeof ( struct kvm_pic ) , GFP_KERNEL ) ;
if ( ! s )
return NULL ;
2008-12-21 22:48:32 +02:00
spin_lock_init ( & s - > lock ) ;
s - > kvm = kvm ;
2007-07-06 12:20:49 +03:00
s - > pics [ 0 ] . elcr_mask = 0xf8 ;
s - > pics [ 1 ] . elcr_mask = 0xde ;
s - > irq_request = pic_irq_request ;
s - > irq_request_opaque = kvm ;
s - > pics [ 0 ] . pics_state = s ;
s - > pics [ 1 ] . pics_state = s ;
/*
* Initialize PIO device
*/
2009-06-01 12:54:50 -04:00
kvm_iodevice_init ( & s - > dev , & picdev_ops ) ;
2007-07-06 12:20:49 +03:00
kvm_io_bus_register_dev ( & kvm - > pio_bus , & s - > dev ) ;
return s ;
}