2008-09-16 10:57:47 +01:00
/*
2010-03-01 14:41:59 +00:00
* GE SBC610 board support
2008-09-16 10:57:47 +01:00
*
2010-03-01 14:41:59 +00:00
* Author : Martyn Welch < martyn . welch @ ge . com >
2008-09-16 10:57:47 +01:00
*
2010-03-01 14:41:59 +00:00
* Copyright 2008 GE Intelligent Platforms Embedded Systems , Inc .
2008-09-16 10:57:47 +01:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*
* Based on : mpc86xx_hpcn . c ( MPC86xx HPCN board specific routines )
* Copyright 2006 Freescale Semiconductor Inc .
2008-09-29 13:35:15 +01:00
*
* NEC fixup adapted from arch / mips / pci / fixup - lm2e . c
2008-09-16 10:57:47 +01:00
*/
# include <linux/stddef.h>
# include <linux/kernel.h>
# include <linux/pci.h>
# include <linux/kdev_t.h>
# include <linux/delay.h>
# include <linux/seq_file.h>
# include <linux/of_platform.h>
# include <asm/system.h>
# include <asm/time.h>
# include <asm/machdep.h>
# include <asm/pci-bridge.h>
# include <asm/prom.h>
# include <mm/mmu_decl.h>
# include <asm/udbg.h>
# include <asm/mpic.h>
2009-07-02 17:12:31 +01:00
# include <asm/nvram.h>
2008-09-16 10:57:47 +01:00
# include <sysdev/fsl_pci.h>
# include <sysdev/fsl_soc.h>
# include "mpc86xx.h"
2008-10-01 09:32:39 +01:00
# include "gef_pic.h"
2008-09-16 10:57:47 +01:00
# undef DEBUG
# ifdef DEBUG
# define DBG (fmt...) do { printk(KERN_ERR "SBC610: " fmt); } while (0)
# else
# define DBG (fmt...) do { } while (0)
# endif
2008-10-01 09:32:39 +01:00
void __iomem * sbc610_regs ;
static void __init gef_sbc610_init_irq ( void )
{
struct device_node * cascade_node = NULL ;
mpc86xx_init_irq ( ) ;
/*
* There is a simple interrupt handler in the main FPGA , this needs
* to be cascaded into the MPIC
*/
cascade_node = of_find_compatible_node ( NULL , NULL , " gef,fpga-pic " ) ;
if ( ! cascade_node ) {
printk ( KERN_WARNING " SBC610: No FPGA PIC \n " ) ;
return ;
}
gef_pic_init ( cascade_node ) ;
of_node_put ( cascade_node ) ;
}
2008-09-16 10:57:47 +01:00
static void __init gef_sbc610_setup_arch ( void )
{
2008-10-13 16:16:45 +01:00
struct device_node * regs ;
2008-09-16 10:57:47 +01:00
# ifdef CONFIG_PCI
struct device_node * np ;
for_each_compatible_node ( np , " pci " , " fsl,mpc8641-pcie " ) {
fsl_add_bridge ( np , 1 ) ;
}
# endif
2010-03-01 14:41:59 +00:00
printk ( KERN_INFO " GE Intelligent Platforms SBC610 6U VPX SBC \n " ) ;
2008-09-16 10:57:47 +01:00
# ifdef CONFIG_SMP
mpc86xx_smp_init ( ) ;
# endif
2008-10-13 16:16:45 +01:00
/* Remap basic board registers */
regs = of_find_compatible_node ( NULL , NULL , " gef,fpga-regs " ) ;
if ( regs ) {
sbc610_regs = of_iomap ( regs , 0 ) ;
if ( sbc610_regs = = NULL )
printk ( KERN_WARNING " Unable to map board registers \n " ) ;
of_node_put ( regs ) ;
}
2009-07-02 17:12:31 +01:00
# if defined(CONFIG_MMIO_NVRAM)
mmio_nvram_init ( ) ;
# endif
2008-10-13 16:16:45 +01:00
}
/* Return the PCB revision */
static unsigned int gef_sbc610_get_pcb_rev ( void )
{
unsigned int reg ;
reg = ioread32 ( sbc610_regs ) ;
return ( reg > > 8 ) & 0xff ;
}
/* Return the board (software) revision */
static unsigned int gef_sbc610_get_board_rev ( void )
{
unsigned int reg ;
reg = ioread32 ( sbc610_regs ) ;
return ( reg > > 16 ) & 0xff ;
2008-09-16 10:57:47 +01:00
}
2008-10-13 16:16:45 +01:00
/* Return the FPGA revision */
static unsigned int gef_sbc610_get_fpga_rev ( void )
{
unsigned int reg ;
reg = ioread32 ( sbc610_regs ) ;
return ( reg > > 24 ) & 0xf ;
}
2008-09-16 10:57:47 +01:00
static void gef_sbc610_show_cpuinfo ( struct seq_file * m )
{
uint svid = mfspr ( SPRN_SVR ) ;
2010-03-01 14:41:59 +00:00
seq_printf ( m , " Vendor \t \t : GE Intelligent Platforms \n " ) ;
2008-09-16 10:57:47 +01:00
2008-10-13 16:16:45 +01:00
seq_printf ( m , " Revision \t : %u%c \n " , gef_sbc610_get_pcb_rev ( ) ,
( ' A ' + gef_sbc610_get_board_rev ( ) - 1 ) ) ;
seq_printf ( m , " FPGA Revision \t : %u \n " , gef_sbc610_get_fpga_rev ( ) ) ;
2008-09-16 10:57:47 +01:00
seq_printf ( m , " SVR \t \t : 0x%x \n " , svid ) ;
}
2008-09-29 13:35:15 +01:00
static void __init gef_sbc610_nec_fixup ( struct pci_dev * pdev )
{
unsigned int val ;
2009-03-03 17:59:30 +00:00
/* Do not do the fixup on other platforms! */
if ( ! machine_is ( gef_sbc610 ) )
return ;
2008-09-29 13:35:15 +01:00
printk ( KERN_INFO " Running NEC uPD720101 Fixup \n " ) ;
/* Ensure ports 1, 2, 3, 4 & 5 are enabled */
pci_read_config_dword ( pdev , 0xe0 , & val ) ;
pci_write_config_dword ( pdev , 0xe0 , ( val & ~ 7 ) | 0x5 ) ;
/* System clock is 48-MHz Oscillator and EHCI Enabled. */
pci_write_config_dword ( pdev , 0xe4 , 1 < < 5 ) ;
}
DECLARE_PCI_FIXUP_HEADER ( PCI_VENDOR_ID_NEC , PCI_DEVICE_ID_NEC_USB ,
gef_sbc610_nec_fixup ) ;
2008-09-16 10:57:47 +01:00
/*
* Called very early , device - tree isn ' t unflattened
*
* This function is called to determine whether the BSP is compatible with the
* supplied device - tree , which is assumed to be the correct one for the actual
* board . It is expected thati , in the future , a kernel may support multiple
* boards .
*/
static int __init gef_sbc610_probe ( void )
{
unsigned long root = of_get_flat_dt_root ( ) ;
if ( of_flat_dt_is_compatible ( root , " gef,sbc610 " ) )
return 1 ;
return 0 ;
}
static long __init mpc86xx_time_init ( void )
{
unsigned int temp ;
/* Set the time base to zero */
mtspr ( SPRN_TBWL , 0 ) ;
mtspr ( SPRN_TBWU , 0 ) ;
temp = mfspr ( SPRN_HID0 ) ;
temp | = HID0_TBEN ;
mtspr ( SPRN_HID0 , temp ) ;
asm volatile ( " isync " ) ;
return 0 ;
}
static __initdata struct of_device_id of_bus_ids [ ] = {
{ . compatible = " simple-bus " , } ,
2009-03-19 21:01:51 +03:00
{ . compatible = " gianfar " , } ,
2008-09-16 10:57:47 +01:00
{ } ,
} ;
static int __init declare_of_platform_devices ( void )
{
printk ( KERN_DEBUG " Probe platform devices \n " ) ;
of_platform_bus_probe ( NULL , of_bus_ids , NULL ) ;
return 0 ;
}
machine_device_initcall ( gef_sbc610 , declare_of_platform_devices ) ;
define_machine ( gef_sbc610 ) {
2010-03-01 14:41:59 +00:00
. name = " GE SBC610 " ,
2008-09-16 10:57:47 +01:00
. probe = gef_sbc610_probe ,
. setup_arch = gef_sbc610_setup_arch ,
2008-10-01 09:32:39 +01:00
. init_IRQ = gef_sbc610_init_irq ,
2008-09-16 10:57:47 +01:00
. show_cpuinfo = gef_sbc610_show_cpuinfo ,
. get_irq = mpic_get_irq ,
. restart = fsl_rstcr_restart ,
. time_init = mpc86xx_time_init ,
. calibrate_decr = generic_calibrate_decr ,
. progress = udbg_progress ,
# ifdef CONFIG_PCI
. pcibios_fixup_bus = fsl_pcibios_fixup_bus ,
# endif
} ;