2007-09-18 23:12:50 +04:00
/*
* Sonics Silicon Backplane
* Broadcom MIPS core driver
*
* Copyright 2005 , Broadcom Corporation
2011-07-04 22:50:05 +04:00
* Copyright 2006 , 2007 , Michael Buesch < m @ bues . ch >
2007-09-18 23:12:50 +04:00
*
* Licensed under the GNU / GPL . See COPYING for details .
*/
# include <linux/ssb/ssb.h>
# include <linux/serial.h>
# include <linux/serial_core.h>
# include <linux/serial_reg.h>
# include <linux/time.h>
# include "ssb_private.h"
static inline u32 mips_read32 ( struct ssb_mipscore * mcore ,
u16 offset )
{
return ssb_read32 ( mcore - > dev , offset ) ;
}
static inline void mips_write32 ( struct ssb_mipscore * mcore ,
u16 offset ,
u32 value )
{
ssb_write32 ( mcore - > dev , offset , value ) ;
}
static const u32 ipsflag_irq_mask [ ] = {
0 ,
SSB_IPSFLAG_IRQ1 ,
SSB_IPSFLAG_IRQ2 ,
SSB_IPSFLAG_IRQ3 ,
SSB_IPSFLAG_IRQ4 ,
} ;
static const u32 ipsflag_irq_shift [ ] = {
0 ,
SSB_IPSFLAG_IRQ1_SHIFT ,
SSB_IPSFLAG_IRQ2_SHIFT ,
SSB_IPSFLAG_IRQ3_SHIFT ,
SSB_IPSFLAG_IRQ4_SHIFT ,
} ;
static inline u32 ssb_irqflag ( struct ssb_device * dev )
{
2009-07-01 01:04:55 +04:00
u32 tpsflag = ssb_read32 ( dev , SSB_TPSFLAG ) ;
if ( tpsflag )
return ssb_read32 ( dev , SSB_TPSFLAG ) & SSB_TPSFLAG_BPFLAG ;
else
/* not irq supported */
return 0x3f ;
}
static struct ssb_device * find_device ( struct ssb_device * rdev , int irqflag )
{
struct ssb_bus * bus = rdev - > bus ;
int i ;
for ( i = 0 ; i < bus - > nr_devices ; i + + ) {
struct ssb_device * dev ;
dev = & ( bus - > devices [ i ] ) ;
if ( ssb_irqflag ( dev ) = = irqflag )
return dev ;
}
return NULL ;
2007-09-18 23:12:50 +04:00
}
/* Get the MIPS IRQ assignment for a specified device.
* If unassigned , 0 is returned .
2009-07-01 01:04:55 +04:00
* If disabled , 5 is returned .
* If not supported , 6 is returned .
2007-09-18 23:12:50 +04:00
*/
unsigned int ssb_mips_irq ( struct ssb_device * dev )
{
struct ssb_bus * bus = dev - > bus ;
2009-07-01 01:04:55 +04:00
struct ssb_device * mdev = bus - > mipscore . dev ;
2007-09-18 23:12:50 +04:00
u32 irqflag ;
u32 ipsflag ;
u32 tmp ;
unsigned int irq ;
irqflag = ssb_irqflag ( dev ) ;
2009-07-01 01:04:55 +04:00
if ( irqflag = = 0x3f )
return 6 ;
2007-09-18 23:12:50 +04:00
ipsflag = ssb_read32 ( bus - > mipscore . dev , SSB_IPSFLAG ) ;
for ( irq = 1 ; irq < = 4 ; irq + + ) {
tmp = ( ( ipsflag & ipsflag_irq_mask [ irq ] ) > > ipsflag_irq_shift [ irq ] ) ;
if ( tmp = = irqflag )
break ;
}
2009-07-01 01:04:55 +04:00
if ( irq = = 5 ) {
if ( ( 1 < < irqflag ) & ssb_read32 ( mdev , SSB_INTVEC ) )
irq = 0 ;
}
2007-09-18 23:12:50 +04:00
return irq ;
}
static void clear_irq ( struct ssb_bus * bus , unsigned int irq )
{
struct ssb_device * dev = bus - > mipscore . dev ;
/* Clear the IRQ in the MIPScore backplane registers */
if ( irq = = 0 ) {
ssb_write32 ( dev , SSB_INTVEC , 0 ) ;
} else {
ssb_write32 ( dev , SSB_IPSFLAG ,
ssb_read32 ( dev , SSB_IPSFLAG ) |
ipsflag_irq_mask [ irq ] ) ;
}
}
static void set_irq ( struct ssb_device * dev , unsigned int irq )
{
unsigned int oldirq = ssb_mips_irq ( dev ) ;
struct ssb_bus * bus = dev - > bus ;
struct ssb_device * mdev = bus - > mipscore . dev ;
u32 irqflag = ssb_irqflag ( dev ) ;
2009-07-01 01:04:55 +04:00
BUG_ON ( oldirq = = 6 ) ;
2007-09-18 23:12:50 +04:00
dev - > irq = irq + 2 ;
/* clear the old irq */
if ( oldirq = = 0 )
ssb_write32 ( mdev , SSB_INTVEC , ( ~ ( 1 < < irqflag ) & ssb_read32 ( mdev , SSB_INTVEC ) ) ) ;
2009-07-01 01:04:55 +04:00
else if ( oldirq ! = 5 )
2007-09-18 23:12:50 +04:00
clear_irq ( bus , oldirq ) ;
/* assign the new one */
2008-04-08 13:17:29 +04:00
if ( irq = = 0 ) {
ssb_write32 ( mdev , SSB_INTVEC , ( ( 1 < < irqflag ) | ssb_read32 ( mdev , SSB_INTVEC ) ) ) ;
} else {
2009-07-01 01:04:55 +04:00
u32 ipsflag = ssb_read32 ( mdev , SSB_IPSFLAG ) ;
if ( ( ipsflag & ipsflag_irq_mask [ irq ] ) ! = ipsflag_irq_mask [ irq ] ) {
u32 oldipsflag = ( ipsflag & ipsflag_irq_mask [ irq ] ) > > ipsflag_irq_shift [ irq ] ;
struct ssb_device * olddev = find_device ( dev , oldipsflag ) ;
if ( olddev )
set_irq ( olddev , 0 ) ;
}
2008-04-08 13:17:29 +04:00
irqflag < < = ipsflag_irq_shift [ irq ] ;
2009-07-01 01:04:55 +04:00
irqflag | = ( ipsflag & ~ ipsflag_irq_mask [ irq ] ) ;
2008-04-08 13:17:29 +04:00
ssb_write32 ( mdev , SSB_IPSFLAG , irqflag ) ;
}
2009-07-01 01:04:55 +04:00
ssb_dprintk ( KERN_INFO PFX
" set_irq: core 0x%04x, irq %d => %d \n " ,
dev - > id . coreid , oldirq + 2 , irq + 2 ) ;
}
static void print_irq ( struct ssb_device * dev , unsigned int irq )
{
int i ;
static const char * irq_name [ ] = { " 2(S) " , " 3 " , " 4 " , " 5 " , " 6 " , " D " , " I " } ;
ssb_dprintk ( KERN_INFO PFX
" core 0x%04x, irq : " , dev - > id . coreid ) ;
for ( i = 0 ; i < = 6 ; i + + ) {
ssb_dprintk ( " %s%s " , irq_name [ i ] , i = = irq ? " * " : " " ) ;
}
ssb_dprintk ( " \n " ) ;
}
static void dump_irq ( struct ssb_bus * bus )
{
int i ;
for ( i = 0 ; i < bus - > nr_devices ; i + + ) {
struct ssb_device * dev ;
dev = & ( bus - > devices [ i ] ) ;
print_irq ( dev , ssb_mips_irq ( dev ) ) ;
}
2007-09-18 23:12:50 +04:00
}
static void ssb_mips_serial_init ( struct ssb_mipscore * mcore )
{
struct ssb_bus * bus = mcore - > dev - > bus ;
if ( bus - > extif . dev )
mcore - > nr_serial_ports = ssb_extif_serial_init ( & bus - > extif , mcore - > serial_ports ) ;
else if ( bus - > chipco . dev )
mcore - > nr_serial_ports = ssb_chipco_serial_init ( & bus - > chipco , mcore - > serial_ports ) ;
else
mcore - > nr_serial_ports = 0 ;
}
static void ssb_mips_flash_detect ( struct ssb_mipscore * mcore )
{
struct ssb_bus * bus = mcore - > dev - > bus ;
mcore - > flash_buswidth = 2 ;
if ( bus - > chipco . dev ) {
mcore - > flash_window = 0x1c000000 ;
mcore - > flash_window_size = 0x02000000 ;
if ( ( ssb_read32 ( bus - > chipco . dev , SSB_CHIPCO_FLASH_CFG )
& SSB_CHIPCO_CFG_DS16 ) = = 0 )
mcore - > flash_buswidth = 1 ;
} else {
mcore - > flash_window = 0x1fc00000 ;
mcore - > flash_window_size = 0x00400000 ;
}
}
u32 ssb_cpu_clock ( struct ssb_mipscore * mcore )
{
struct ssb_bus * bus = mcore - > dev - > bus ;
u32 pll_type , n , m , rate = 0 ;
2012-02-01 03:13:56 +04:00
if ( bus - > chipco . capabilities & SSB_CHIPCO_CAP_PMU )
return ssb_pmu_get_cpu_clock ( & bus - > chipco ) ;
2007-09-18 23:12:50 +04:00
if ( bus - > extif . dev ) {
ssb_extif_get_clockcontrol ( & bus - > extif , & pll_type , & n , & m ) ;
} else if ( bus - > chipco . dev ) {
ssb_chipco_get_clockcpu ( & bus - > chipco , & pll_type , & n , & m ) ;
} else
return 0 ;
if ( ( pll_type = = SSB_PLLTYPE_5 ) | | ( bus - > chip_id = = 0x5365 ) ) {
rate = 200000000 ;
} else {
rate = ssb_calc_clock_rate ( pll_type , n , m ) ;
}
if ( pll_type = = SSB_PLLTYPE_6 ) {
rate * = 2 ;
}
return rate ;
}
void ssb_mipscore_init ( struct ssb_mipscore * mcore )
{
2007-10-14 23:04:22 +04:00
struct ssb_bus * bus ;
2007-09-18 23:12:50 +04:00
struct ssb_device * dev ;
unsigned long hz , ns ;
unsigned int irq , i ;
if ( ! mcore - > dev )
return ; /* We don't have a MIPS core */
ssb_dprintk ( KERN_INFO PFX " Initializing MIPS core... \n " ) ;
2007-10-14 23:04:22 +04:00
bus = mcore - > dev - > bus ;
2007-09-18 23:12:50 +04:00
hz = ssb_clockspeed ( bus ) ;
if ( ! hz )
hz = 100000000 ;
ns = 1000000000 / hz ;
if ( bus - > extif . dev )
ssb_extif_timing_init ( & bus - > extif , ns ) ;
else if ( bus - > chipco . dev )
ssb_chipco_timing_init ( & bus - > chipco , ns ) ;
/* Assign IRQs to all cores on the bus, start with irq line 2, because serial usually takes 1 */
for ( irq = 2 , i = 0 ; i < bus - > nr_devices ; i + + ) {
2009-07-01 01:04:55 +04:00
int mips_irq ;
2007-09-18 23:12:50 +04:00
dev = & ( bus - > devices [ i ] ) ;
2009-07-01 01:04:55 +04:00
mips_irq = ssb_mips_irq ( dev ) ;
if ( mips_irq > 4 )
dev - > irq = 0 ;
else
dev - > irq = mips_irq + 2 ;
if ( dev - > irq > 5 )
continue ;
2007-09-18 23:12:50 +04:00
switch ( dev - > id . coreid ) {
case SSB_DEV_USB11_HOST :
/* shouldn't need a separate irq line for non-4710, most of them have a proper
* external usb controller on the pci */
if ( ( bus - > chip_id = = 0x4710 ) & & ( irq < = 4 ) ) {
set_irq ( dev , irq + + ) ;
}
2009-07-01 01:04:55 +04:00
break ;
2007-09-18 23:12:50 +04:00
case SSB_DEV_PCI :
case SSB_DEV_ETHERNET :
2008-02-29 13:36:12 +03:00
case SSB_DEV_ETHERNET_GBIT :
2007-09-18 23:12:50 +04:00
case SSB_DEV_80211 :
case SSB_DEV_USB20_HOST :
/* These devices get their own IRQ line if available, the rest goes on IRQ0 */
if ( irq < = 4 ) {
set_irq ( dev , irq + + ) ;
break ;
}
2010-02-03 23:28:11 +03:00
/* fallthrough */
case SSB_DEV_EXTIF :
set_irq ( dev , 0 ) ;
break ;
2007-09-18 23:12:50 +04:00
}
}
2009-07-01 01:04:55 +04:00
ssb_dprintk ( KERN_INFO PFX " after irq reconfiguration \n " ) ;
dump_irq ( bus ) ;
2007-09-18 23:12:50 +04:00
ssb_mips_serial_init ( mcore ) ;
ssb_mips_flash_detect ( mcore ) ;
}