2006-12-28 18:22:32 +01:00
/*
* PCIMT specific code
*
* 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 .
*
* Copyright ( C ) 1996 , 97 , 98 , 2000 , 03 , 04 , 06 Ralf Baechle ( ralf @ linux - mips . org )
2007-05-17 14:51:47 +02:00
* Copyright ( C ) 2006 , 2007 Thomas Bogendoerfer ( tsbogend @ alpha . franken . de )
2006-12-28 18:22:32 +01:00
*/
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/pci.h>
# include <linux/serial_8250.h>
# include <asm/sni.h>
# include <asm/time.h>
# include <asm/i8259.h>
# include <asm/irq_cpu.h>
# define cacheconf (*(volatile unsigned int *)PCIMT_CACHECONF)
# define invspace (*(volatile unsigned int *)PCIMT_INVSPACE)
static void __init sni_pcimt_sc_init ( void )
{
unsigned int scsiz , sc_size ;
scsiz = cacheconf & 7 ;
if ( scsiz = = 0 ) {
printk ( " Second level cache is deactived. \n " ) ;
return ;
}
if ( scsiz > = 6 ) {
printk ( " Invalid second level cache size configured, "
" deactivating second level cache. \n " ) ;
cacheconf = 0 ;
return ;
}
sc_size = 128 < < scsiz ;
printk ( " %dkb second level cache detected, deactivating. \n " , sc_size ) ;
cacheconf = 0 ;
}
/*
* A bit more gossip about the iron we ' re running on . . .
*/
static inline void sni_pcimt_detect ( void )
{
char boardtype [ 80 ] ;
unsigned char csmsr ;
char * p = boardtype ;
unsigned int asic ;
csmsr = * ( volatile unsigned char * ) PCIMT_CSMSR ;
p + = sprintf ( p , " %s PCI " , ( csmsr & 0x80 ) ? " RM200 " : " RM300 " ) ;
if ( ( csmsr & 0x80 ) = = 0 )
p + = sprintf ( p , " , board revision %s " ,
( csmsr & 0x20 ) ? " D " : " C " ) ;
asic = csmsr & 0x80 ;
asic = ( csmsr & 0x08 ) ? asic : ! asic ;
p + = sprintf ( p , " , ASIC PCI Rev %s " , asic ? " 1.0 " : " 1.1 " ) ;
printk ( " %s. \n " , boardtype ) ;
}
# define PORT(_base,_irq) \
{ \
. iobase = _base , \
. irq = _irq , \
. uartclk = 1843200 , \
. iotype = UPIO_PORT , \
. flags = UPF_BOOT_AUTOCONF , \
}
static struct plat_serial8250_port pcimt_data [ ] = {
PORT ( 0x3f8 , 4 ) ,
PORT ( 0x2f8 , 3 ) ,
{ } ,
} ;
static struct platform_device pcimt_serial8250_device = {
. name = " serial8250 " ,
. id = PLAT8250_DEV_PLATFORM ,
. dev = {
. platform_data = pcimt_data ,
} ,
} ;
2007-06-20 23:36:47 +02:00
static struct resource pcimt_cmos_rsrc [ ] = {
{
. start = 0x70 ,
. end = 0x71 ,
. flags = IORESOURCE_IO
} ,
{
. start = 8 ,
. end = 8 ,
. flags = IORESOURCE_IRQ
}
} ;
static struct platform_device pcimt_cmos_device = {
. name = " rtc_cmos " ,
. num_resources = ARRAY_SIZE ( pcimt_cmos_rsrc ) ,
. resource = pcimt_cmos_rsrc
} ;
2006-12-28 18:22:32 +01:00
static struct resource sni_io_resource = {
2007-04-08 13:34:57 +02:00
. start = 0x00000000UL ,
2006-12-28 18:22:32 +01:00
. end = 0x03bfffffUL ,
. name = " PCIMT IO MEM " ,
. flags = IORESOURCE_IO ,
} ;
static struct resource pcimt_io_resources [ ] = {
{
. start = 0x00 ,
. end = 0x1f ,
. name = " dma1 " ,
. flags = IORESOURCE_BUSY
} , {
. start = 0x40 ,
. end = 0x5f ,
. name = " timer " ,
. flags = IORESOURCE_BUSY
} , {
. start = 0x60 ,
. end = 0x6f ,
. name = " keyboard " ,
. flags = IORESOURCE_BUSY
} , {
. start = 0x80 ,
. end = 0x8f ,
. name = " dma page reg " ,
. flags = IORESOURCE_BUSY
} , {
. start = 0xc0 ,
. end = 0xdf ,
. name = " dma2 " ,
. flags = IORESOURCE_BUSY
} , {
. start = 0xcfc ,
. end = 0xcff ,
. name = " PCI config data " ,
. flags = IORESOURCE_BUSY
}
} ;
2007-05-17 14:51:47 +02:00
static struct resource pcimt_mem_resources [ ] = {
{
/*
* this region should only be 4 bytes long ,
* but it ' s 16 MB on all RM300C I ' ve checked
*/
. start = 0x1a000000 ,
. end = 0x1affffff ,
. name = " PCI INT ACK " ,
. flags = IORESOURCE_BUSY
}
} ;
2006-12-28 18:22:32 +01:00
static struct resource sni_mem_resource = {
2007-04-08 13:34:57 +02:00
. start = 0x18000000UL ,
. end = 0x1fbfffffUL ,
2006-12-28 18:22:32 +01:00
. name = " PCIMT PCI MEM " ,
. flags = IORESOURCE_MEM
} ;
static void __init sni_pcimt_resource_init ( void )
{
int i ;
/* request I/O space for devices used on all i[345]86 PCs */
for ( i = 0 ; i < ARRAY_SIZE ( pcimt_io_resources ) ; i + + )
2007-04-08 13:34:57 +02:00
request_resource ( & sni_io_resource , pcimt_io_resources + i ) ;
2007-05-17 14:51:47 +02:00
/* request MEM space for devices used on all i[345]86 PCs */
for ( i = 0 ; i < ARRAY_SIZE ( pcimt_mem_resources ) ; i + + )
request_resource ( & sni_mem_resource , pcimt_mem_resources + i ) ;
2006-12-28 18:22:32 +01:00
}
extern struct pci_ops sni_pcimt_ops ;
static struct pci_controller sni_controller = {
. pci_ops = & sni_pcimt_ops ,
. mem_resource = & sni_mem_resource ,
2007-04-08 13:34:57 +02:00
. mem_offset = 0x00000000UL ,
2006-12-28 18:22:32 +01:00
. io_resource = & sni_io_resource ,
2007-04-08 13:34:57 +02:00
. io_offset = 0x00000000UL ,
. io_map_base = SNI_PORT_BASE
2006-12-28 18:22:32 +01:00
} ;
static void enable_pcimt_irq ( unsigned int irq )
{
unsigned int mask = 1 < < ( irq - PCIMT_IRQ_INT2 ) ;
* ( volatile u8 * ) PCIMT_IRQSEL | = mask ;
}
void disable_pcimt_irq ( unsigned int irq )
{
unsigned int mask = ~ ( 1 < < ( irq - PCIMT_IRQ_INT2 ) ) ;
* ( volatile u8 * ) PCIMT_IRQSEL & = mask ;
}
static void end_pcimt_irq ( unsigned int irq )
{
if ( ! ( irq_desc [ irq ] . status & ( IRQ_DISABLED | IRQ_INPROGRESS ) ) )
enable_pcimt_irq ( irq ) ;
}
static struct irq_chip pcimt_irq_type = {
. typename = " PCIMT " ,
. ack = disable_pcimt_irq ,
. mask = disable_pcimt_irq ,
. mask_ack = disable_pcimt_irq ,
. unmask = enable_pcimt_irq ,
. end = end_pcimt_irq ,
} ;
/*
* hwint0 should deal with MP agent , ASIC PCI , EISA NMI and debug
* button interrupts . Later . . .
*/
static void pcimt_hwint0 ( void )
{
panic ( " Received int0 but no handler yet ... " ) ;
}
/*
* hwint 1 deals with EISA and SCSI interrupts ,
*
* The EISA_INT bit in CSITPEND is high active , all others are low active .
*/
static void pcimt_hwint1 ( void )
{
u8 pend = * ( volatile char * ) PCIMT_CSITPEND ;
unsigned long flags ;
if ( pend & IT_EISA ) {
int irq ;
/*
2007-11-12 17:32:48 +00:00
* Note : ASIC PCI ' s builtin interrupt acknowledge feature is
2006-12-28 18:22:32 +01:00
* broken . Using it may result in loss of some or all i8259
2007-10-19 23:10:43 +02:00
* interrupts , so don ' t use PCIMT_INT_ACKNOWLEDGE . . .
2006-12-28 18:22:32 +01:00
*/
irq = i8259_irq ( ) ;
if ( unlikely ( irq < 0 ) )
return ;
do_IRQ ( irq ) ;
}
if ( ! ( pend & IT_SCSI ) ) {
flags = read_c0_status ( ) ;
clear_c0_status ( ST0_IM ) ;
do_IRQ ( PCIMT_IRQ_SCSI ) ;
write_c0_status ( flags ) ;
}
}
/*
* hwint 3 should deal with the PCI A - D interrupts ,
*/
static void pcimt_hwint3 ( void )
{
u8 pend = * ( volatile char * ) PCIMT_CSITPEND ;
int irq ;
pend & = ( IT_INTA | IT_INTB | IT_INTC | IT_INTD ) ;
pend ^ = ( IT_INTA | IT_INTB | IT_INTC | IT_INTD ) ;
clear_c0_status ( IE_IRQ3 ) ;
irq = PCIMT_IRQ_INT2 + ffs ( pend ) - 1 ;
do_IRQ ( irq ) ;
set_c0_status ( IE_IRQ3 ) ;
}
static void sni_pcimt_hwint ( void )
{
2007-03-19 00:13:37 +00:00
u32 pending = read_c0_cause ( ) & read_c0_status ( ) ;
2006-12-28 18:22:32 +01:00
if ( pending & C_IRQ5 )
2007-10-11 23:46:15 +01:00
do_IRQ ( MIPS_CPU_IRQ_BASE + 7 ) ;
2006-12-28 18:22:32 +01:00
else if ( pending & C_IRQ4 )
2007-10-11 23:46:15 +01:00
do_IRQ ( MIPS_CPU_IRQ_BASE + 6 ) ;
2006-12-28 18:22:32 +01:00
else if ( pending & C_IRQ3 )
pcimt_hwint3 ( ) ;
else if ( pending & C_IRQ1 )
pcimt_hwint1 ( ) ;
else if ( pending & C_IRQ0 ) {
pcimt_hwint0 ( ) ;
}
}
void __init sni_pcimt_irq_init ( void )
{
int i ;
* ( volatile u8 * ) PCIMT_IRQSEL = IT_ETH | IT_EISA ;
mips_cpu_irq_init ( ) ;
/* Actually we've got more interrupts to handle ... */
for ( i = PCIMT_IRQ_INT2 ; i < = PCIMT_IRQ_SCSI ; i + + )
set_irq_chip ( i , & pcimt_irq_type ) ;
sni_hwint = sni_pcimt_hwint ;
change_c0_status ( ST0_IM , IE_IRQ1 | IE_IRQ3 ) ;
}
2007-06-20 23:36:47 +02:00
void __init sni_pcimt_init ( void )
2006-12-28 18:22:32 +01:00
{
sni_pcimt_detect ( ) ;
sni_pcimt_sc_init ( ) ;
2007-04-08 13:34:57 +02:00
ioport_resource . end = sni_io_resource . end ;
2006-12-28 18:22:32 +01:00
# ifdef CONFIG_PCI
2007-04-08 13:34:57 +02:00
PCIBIOS_MIN_IO = 0x9000 ;
2006-12-28 18:22:32 +01:00
register_pci_controller ( & sni_controller ) ;
# endif
2007-04-08 13:34:57 +02:00
sni_pcimt_resource_init ( ) ;
2006-12-28 18:22:32 +01:00
}
static int __init snirm_pcimt_setup_devinit ( void )
{
switch ( sni_brd_type ) {
case SNI_BRD_PCI_MTOWER :
case SNI_BRD_PCI_DESKTOP :
case SNI_BRD_PCI_MTOWER_CPLUS :
platform_device_register ( & pcimt_serial8250_device ) ;
2007-06-20 23:36:47 +02:00
platform_device_register ( & pcimt_cmos_device ) ;
2006-12-28 18:22:32 +01:00
break ;
}
return 0 ;
}
device_initcall ( snirm_pcimt_setup_devinit ) ;