2005-04-16 15:20:36 -07:00
/*
* arch / m68k / q40 / q40ints . c
*
* Copyright ( C ) 1999 , 2001 Richard Zidlicky
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file COPYING in the main directory of this archive
* for more details .
*
* . . used to be loosely based on bvme6000ints . c
*
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/interrupt.h>
2011-08-18 14:47:16 +02:00
# include <linux/irq.h>
2005-04-16 15:20:36 -07:00
# include <asm/ptrace.h>
# include <asm/traps.h>
# include <asm/q40_master.h>
# include <asm/q40ints.h>
/*
* Q40 IRQs are defined as follows :
* 3 , 4 , 5 , 6 , 7 , 10 , 11 , 14 , 15 : ISA dev IRQs
* 16 - 31 : reserved
* 32 : keyboard int
* 33 : frame int ( 50 / 200 Hz periodic timer )
* 34 : sample int ( 10 / 20 KHz periodic timer )
*
*/
2006-06-25 05:47:05 -07:00
static void q40_irq_handler ( unsigned int , struct pt_regs * fp ) ;
2011-04-17 22:53:04 +02:00
static void q40_irq_enable ( struct irq_data * data ) ;
static void q40_irq_disable ( struct irq_data * data ) ;
2005-04-16 15:20:36 -07:00
2006-06-25 05:47:05 -07:00
unsigned short q40_ablecount [ 35 ] ;
unsigned short q40_state [ 35 ] ;
2005-04-16 15:20:36 -07:00
2011-04-17 22:53:04 +02:00
static unsigned int q40_irq_startup ( struct irq_data * data )
2006-06-25 05:47:05 -07:00
{
2011-04-17 22:53:04 +02:00
unsigned int irq = data - > irq ;
2006-06-25 05:47:05 -07:00
/* test for ISA ints not implemented by HW */
switch ( irq ) {
case 1 : case 2 : case 8 : case 9 :
case 11 : case 12 : case 13 :
2008-04-28 02:13:49 -07:00
printk ( " %s: ISA IRQ %d not implemented by HW \n " , __func__ , irq ) ;
2011-04-13 22:31:28 +02:00
/* FIXME return -ENXIO; */
2006-06-25 05:47:05 -07:00
}
return 0 ;
}
2005-04-16 15:20:36 -07:00
2011-04-17 22:53:04 +02:00
static void q40_irq_shutdown ( struct irq_data * data )
2006-06-25 05:47:05 -07:00
{
}
2005-04-16 15:20:36 -07:00
2011-04-13 22:31:28 +02:00
static struct irq_chip q40_irq_chip = {
2006-06-25 05:47:05 -07:00
. name = " q40 " ,
2011-04-13 22:31:28 +02:00
. irq_startup = q40_irq_startup ,
. irq_shutdown = q40_irq_shutdown ,
2011-04-17 22:53:04 +02:00
. irq_enable = q40_irq_enable ,
. irq_disable = q40_irq_disable ,
2006-06-25 05:47:05 -07:00
} ;
2005-04-16 15:20:36 -07:00
/*
* void q40_init_IRQ ( void )
*
* Parameters : None
*
* Returns : Nothing
*
* This function is called during kernel startup to initialize
* the q40 IRQ handling routines .
*/
2006-06-25 05:47:05 -07:00
static int disabled ;
2005-04-16 15:20:36 -07:00
2007-07-20 04:33:28 +01:00
void __init q40_init_IRQ ( void )
2005-04-16 15:20:36 -07:00
{
2011-06-01 11:15:21 +02:00
m68k_setup_irq_controller ( & q40_irq_chip , handle_simple_irq , 1 ,
Q40_IRQ_MAX ) ;
2005-04-16 15:20:36 -07:00
/* setup handler for ISA ints */
2006-06-25 05:47:05 -07:00
m68k_setup_auto_interrupt ( q40_irq_handler ) ;
2011-04-17 22:53:04 +02:00
m68k_irq_startup_irq ( IRQ_AUTO_2 ) ;
m68k_irq_startup_irq ( IRQ_AUTO_4 ) ;
2005-04-16 15:20:36 -07:00
/* now enable some ints.. */
2006-06-25 05:47:05 -07:00
master_outb ( 1 , EXT_ENABLE_REG ) ; /* ISA IRQ 5-15 */
2005-04-16 15:20:36 -07:00
/* make sure keyboard IRQ is disabled */
2006-06-25 05:47:05 -07:00
master_outb ( 0 , KEY_IRQ_ENABLE_REG ) ;
2005-04-16 15:20:36 -07:00
}
/*
* this stuff doesn ' t really belong here . .
2006-06-25 05:47:05 -07:00
*/
2005-04-16 15:20:36 -07:00
int ql_ticks ; /* 200Hz ticks since last jiffie */
static int sound_ticks ;
# define SVOL 45
void q40_mksound ( unsigned int hz , unsigned int ticks )
{
2006-06-25 05:47:05 -07:00
/* for now ignore hz, except that hz==0 switches off sound */
/* simply alternate the ampl (128-SVOL)-(128+SVOL)-..-.. at 200Hz */
if ( hz = = 0 ) {
if ( sound_ticks )
sound_ticks = 1 ;
* DAC_LEFT = 128 ;
* DAC_RIGHT = 128 ;
return ;
}
/* sound itself is done in q40_timer_int */
if ( sound_ticks = = 0 )
sound_ticks = 1000 ; /* pretty long beep */
sound_ticks = ticks < < 1 ;
2005-04-16 15:20:36 -07:00
}
2006-10-09 12:19:47 +01:00
static irq_handler_t q40_timer_routine ;
2005-04-16 15:20:36 -07:00
2006-10-07 14:16:45 +01:00
static irqreturn_t q40_timer_int ( int irq , void * dev )
2005-04-16 15:20:36 -07:00
{
2006-06-25 05:47:05 -07:00
ql_ticks = ql_ticks ? 0 : 1 ;
if ( sound_ticks ) {
unsigned char sval = ( sound_ticks & 1 ) ? 128 - SVOL : 128 + SVOL ;
sound_ticks - - ;
* DAC_LEFT = sval ;
* DAC_RIGHT = sval ;
}
if ( ! ql_ticks )
2006-10-07 14:16:45 +01:00
q40_timer_routine ( irq , dev ) ;
2006-06-25 05:47:05 -07:00
return IRQ_HANDLED ;
2005-04-16 15:20:36 -07:00
}
2006-10-09 12:19:47 +01:00
void q40_sched_init ( irq_handler_t timer_routine )
2005-04-16 15:20:36 -07:00
{
2006-06-25 05:47:05 -07:00
int timer_irq ;
2005-04-16 15:20:36 -07:00
2006-06-25 05:47:05 -07:00
q40_timer_routine = timer_routine ;
timer_irq = Q40_IRQ_FRAME ;
2005-04-16 15:20:36 -07:00
2006-06-25 05:47:05 -07:00
if ( request_irq ( timer_irq , q40_timer_int , 0 ,
2005-04-16 15:20:36 -07:00
" timer " , q40_timer_int ) )
2006-06-25 05:47:05 -07:00
panic ( " Couldn't register timer int " ) ;
2005-04-16 15:20:36 -07:00
2006-06-25 05:47:05 -07:00
master_outb ( - 1 , FRAME_CLEAR_REG ) ;
master_outb ( 1 , FRAME_RATE_REG ) ;
2005-04-16 15:20:36 -07:00
}
/*
* tables to translate bits into IRQ numbers
* it is a good idea to order the entries by priority
*
*/
struct IRQ_TABLE { unsigned mask ; int irq ; } ;
#if 0
static struct IRQ_TABLE iirqs [ ] = {
{ Q40_IRQ_FRAME_MASK , Q40_IRQ_FRAME } ,
{ Q40_IRQ_KEYB_MASK , Q40_IRQ_KEYBOARD } ,
{ 0 , 0 } } ;
# endif
static struct IRQ_TABLE eirqs [ ] = {
{ . mask = Q40_IRQ3_MASK , . irq = 3 } , /* ser 1 */
{ . mask = Q40_IRQ4_MASK , . irq = 4 } , /* ser 2 */
{ . mask = Q40_IRQ14_MASK , . irq = 14 } , /* IDE 1 */
{ . mask = Q40_IRQ15_MASK , . irq = 15 } , /* IDE 2 */
{ . mask = Q40_IRQ6_MASK , . irq = 6 } , /* floppy, handled elsewhere */
{ . mask = Q40_IRQ7_MASK , . irq = 7 } , /* par */
{ . mask = Q40_IRQ5_MASK , . irq = 5 } ,
{ . mask = Q40_IRQ10_MASK , . irq = 10 } ,
{ 0 , 0 }
} ;
/* complain only this many times about spurious ints : */
2007-10-20 01:20:32 +02:00
static int ccleirq = 60 ; /* ISA dev IRQs*/
2005-04-16 15:20:36 -07:00
/*static int cclirq=60;*/ /* internal */
/* FIXME: add shared ints,mask,unmask,probing.... */
# define IRQ_INPROGRESS 1
/*static unsigned short saved_mask;*/
//static int do_tint=0;
# define DEBUG_Q40INT
/*#define IP_USE_DISABLE */ /* would be nice, but crashes ???? */
static int mext_disabled = 0 ; /* ext irq disabled by master chip? */
static int aliased_irq = 0 ; /* how many times inside handler ?*/
2006-06-25 05:47:05 -07:00
/* got interrupt, dispatch to ISA or keyboard/timer IRQs */
static void q40_irq_handler ( unsigned int irq , struct pt_regs * fp )
2005-04-16 15:20:36 -07:00
{
2006-06-25 05:47:05 -07:00
unsigned mir , mer ;
int i ;
2005-04-16 15:20:36 -07:00
//repeat:
2006-06-25 05:47:05 -07:00
mir = master_inb ( IIRQ_REG ) ;
# ifdef CONFIG_BLK_DEV_FD
if ( ( mir & Q40_IRQ_EXT_MASK ) & &
( master_inb ( EIRQ_REG ) & Q40_IRQ6_MASK ) ) {
floppy_hardint ( ) ;
return ;
}
# endif
switch ( irq ) {
case 4 :
case 6 :
2011-07-01 20:39:19 +02:00
do_IRQ ( Q40_IRQ_SAMPLE , fp ) ;
2006-06-25 05:47:05 -07:00
return ;
}
if ( mir & Q40_IRQ_FRAME_MASK ) {
2011-07-01 20:39:19 +02:00
do_IRQ ( Q40_IRQ_FRAME , fp ) ;
2006-06-25 05:47:05 -07:00
master_outb ( - 1 , FRAME_CLEAR_REG ) ;
}
if ( ( mir & Q40_IRQ_SER_MASK ) | | ( mir & Q40_IRQ_EXT_MASK ) ) {
mer = master_inb ( EIRQ_REG ) ;
for ( i = 0 ; eirqs [ i ] . mask ; i + + ) {
if ( mer & eirqs [ i ] . mask ) {
irq = eirqs [ i ] . irq ;
2005-04-16 15:20:36 -07:00
/*
* There is a little mess wrt which IRQ really caused this irq request . The
* main problem is that IIRQ_REG and EIRQ_REG reflect the state when they
* are read - which is long after the request came in . In theory IRQs should
2007-10-20 01:20:32 +02:00
* not just go away but they occasionally do
2005-04-16 15:20:36 -07:00
*/
2006-06-25 05:47:05 -07:00
if ( irq > 4 & & irq < = 15 & & mext_disabled ) {
/*aliased_irq++;*/
goto iirq ;
}
if ( q40_state [ irq ] & IRQ_INPROGRESS ) {
/* some handlers do local_irq_enable() for irq latency reasons, */
/* however reentering an active irq handler is not permitted */
2005-04-16 15:20:36 -07:00
# ifdef IP_USE_DISABLE
2006-06-25 05:47:05 -07:00
/* in theory this is the better way to do it because it still */
/* lets through eg the serial irqs, unfortunately it crashes */
disable_irq ( irq ) ;
disabled = 1 ;
2005-04-16 15:20:36 -07:00
# else
2006-06-25 05:47:05 -07:00
/*printk("IRQ_INPROGRESS detected for irq %d, disabling - %s disabled\n",
irq , disabled ? " already " : " not yet " ) ; */
fp - > sr = ( ( ( fp - > sr ) & ( ~ 0x700 ) ) + 0x200 ) ;
disabled = 1 ;
2005-04-16 15:20:36 -07:00
# endif
2006-06-25 05:47:05 -07:00
goto iirq ;
}
q40_state [ irq ] | = IRQ_INPROGRESS ;
2011-07-01 20:39:19 +02:00
do_IRQ ( irq , fp ) ;
2006-06-25 05:47:05 -07:00
q40_state [ irq ] & = ~ IRQ_INPROGRESS ;
/* naively enable everything, if that fails than */
/* this function will be reentered immediately thus */
/* getting another chance to disable the IRQ */
if ( disabled ) {
2005-04-16 15:20:36 -07:00
# ifdef IP_USE_DISABLE
2006-06-25 05:47:05 -07:00
if ( irq > 4 ) {
disabled = 0 ;
enable_irq ( irq ) ;
}
2005-04-16 15:20:36 -07:00
# else
2006-06-25 05:47:05 -07:00
disabled = 0 ;
/*printk("reenabling irq %d\n", irq); */
2005-04-16 15:20:36 -07:00
# endif
2006-06-25 05:47:05 -07:00
}
2005-04-16 15:20:36 -07:00
// used to do 'goto repeat;' here, this delayed bh processing too long
2006-06-25 05:47:05 -07:00
return ;
}
}
if ( mer & & ccleirq > 0 & & ! aliased_irq ) {
printk ( " ISA interrupt from unknown source? EIRQ_REG = %x \n " , mer ) ;
ccleirq - - ;
}
2005-04-16 15:20:36 -07:00
}
2006-06-25 05:47:05 -07:00
iirq :
mir = master_inb ( IIRQ_REG ) ;
/* should test whether keyboard irq is really enabled, doing it in defhand */
if ( mir & Q40_IRQ_KEYB_MASK )
2011-07-01 20:39:19 +02:00
do_IRQ ( Q40_IRQ_KEYBOARD , fp ) ;
2005-04-16 15:20:36 -07:00
2006-06-25 05:47:05 -07:00
return ;
2005-04-16 15:20:36 -07:00
}
2011-04-17 22:53:04 +02:00
void q40_irq_enable ( struct irq_data * data )
2005-04-16 15:20:36 -07:00
{
2011-04-17 22:53:04 +02:00
unsigned int irq = data - > irq ;
2006-06-25 05:47:05 -07:00
if ( irq > = 5 & & irq < = 15 ) {
mext_disabled - - ;
if ( mext_disabled > 0 )
2011-04-17 22:53:04 +02:00
printk ( " q40_irq_enable : nested disable/enable \n " ) ;
2006-06-25 05:47:05 -07:00
if ( mext_disabled = = 0 )
master_outb ( 1 , EXT_ENABLE_REG ) ;
}
2005-04-16 15:20:36 -07:00
}
2011-04-17 22:53:04 +02:00
void q40_irq_disable ( struct irq_data * data )
2005-04-16 15:20:36 -07:00
{
2011-04-17 22:53:04 +02:00
unsigned int irq = data - > irq ;
2006-06-25 05:47:05 -07:00
/* disable ISA iqs : only do something if the driver has been
* verified to be Q40 " compatible " - right now IDE , NE2K
* Any driver should not attempt to sleep across disable_irq ! !
*/
if ( irq > = 5 & & irq < = 15 ) {
master_outb ( 0 , EXT_ENABLE_REG ) ;
mext_disabled + + ;
if ( mext_disabled > 1 )
printk ( " disable_irq nesting count %d \n " , mext_disabled ) ;
}
2005-04-16 15:20:36 -07:00
}