2005-04-17 02:20:36 +04:00
/*
* Baboon Custom IC Management
*
* The Baboon custom IC controls the IDE , PCMCIA and media bay on the
* PowerBook 190. It multiplexes multiple interrupt sources onto the
* Nubus slot $ C interrupt .
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/ide.h>
# include <asm/traps.h>
# include <asm/bootinfo.h>
# include <asm/macintosh.h>
# include <asm/macints.h>
# include <asm/mac_baboon.h>
/* #define DEBUG_BABOON */
/* #define DEBUG_IRQS */
2007-05-02 00:32:58 +04:00
int baboon_present ;
2005-04-17 02:20:36 +04:00
volatile struct baboon * baboon ;
2006-10-07 17:16:45 +04:00
irqreturn_t baboon_irq ( int , void * ) ;
2005-04-17 02:20:36 +04:00
#if 0
extern int macide_ack_intr ( struct ata_channel * ) ;
# endif
/*
* Baboon initialization .
*/
void __init baboon_init ( void )
{
if ( macintosh_config - > ident ! = MAC_MODEL_PB190 ) {
baboon = NULL ;
baboon_present = 0 ;
return ;
}
baboon = ( struct baboon * ) BABOON_BASE ;
baboon_present = 1 ;
printk ( " Baboon detected at %p \n " , baboon ) ;
}
/*
* Register the Baboon interrupt dispatcher on nubus slot $ C .
*/
void __init baboon_register_interrupts ( void )
{
request_irq ( IRQ_NUBUS_C , baboon_irq , IRQ_FLG_LOCK | IRQ_FLG_FAST ,
" baboon " , ( void * ) baboon ) ;
}
/*
* Baboon interrupt handler . This works a lot like a VIA .
*/
2006-10-07 17:16:45 +04:00
irqreturn_t baboon_irq ( int irq , void * dev_id )
2005-04-17 02:20:36 +04:00
{
2007-05-02 00:32:56 +04:00
int irq_bit , irq_num ;
2005-04-17 02:20:36 +04:00
unsigned char events ;
# ifdef DEBUG_IRQS
2007-05-02 00:32:58 +04:00
printk ( " baboon_irq: mb_control %02X mb_ifr %02X mb_status %02X \n " ,
2005-04-17 02:20:36 +04:00
( uint ) baboon - > mb_control , ( uint ) baboon - > mb_ifr ,
2007-05-02 00:32:58 +04:00
( uint ) baboon - > mb_status ) ;
2005-04-17 02:20:36 +04:00
# endif
if ( ! ( events = baboon - > mb_ifr & 0x07 ) )
return IRQ_NONE ;
2007-05-02 00:32:56 +04:00
irq_num = IRQ_BABOON_0 ;
irq_bit = 1 ;
do {
2007-05-02 00:32:58 +04:00
if ( events & irq_bit ) {
2007-05-02 00:32:55 +04:00
baboon - > mb_ifr & = ~ irq_bit ;
2007-05-02 00:32:56 +04:00
m68k_handle_int ( irq_num ) ;
2005-04-17 02:20:36 +04:00
}
2007-05-02 00:32:56 +04:00
irq_bit < < = 1 ;
irq_num + + ;
} while ( events > = irq_bit ) ;
2005-04-17 02:20:36 +04:00
#if 0
if ( baboon - > mb_ifr & 0x02 ) macide_ack_intr ( NULL ) ;
/* for now we need to smash all interrupts */
baboon - > mb_ifr & = ~ events ;
# endif
return IRQ_HANDLED ;
}
void baboon_irq_enable ( int irq ) {
# ifdef DEBUG_IRQUSE
printk ( " baboon_irq_enable(%d) \n " , irq ) ;
# endif
2007-05-02 00:32:58 +04:00
/* FIXME: figure out how to mask and unmask baboon interrupt sources */
enable_irq ( IRQ_NUBUS_C ) ;
2005-04-17 02:20:36 +04:00
}
void baboon_irq_disable ( int irq ) {
# ifdef DEBUG_IRQUSE
printk ( " baboon_irq_disable(%d) \n " , irq ) ;
# endif
2007-05-02 00:32:58 +04:00
disable_irq ( IRQ_NUBUS_C ) ;
2005-04-17 02:20:36 +04:00
}
void baboon_irq_clear ( int irq ) {
int irq_idx = IRQ_IDX ( irq ) ;
baboon - > mb_ifr & = ~ ( 1 < < irq_idx ) ;
}
int baboon_irq_pending ( int irq )
{
int irq_idx = IRQ_IDX ( irq ) ;
return baboon - > mb_ifr & ( 1 < < irq_idx ) ;
}