2006-11-23 00:46:51 +01:00
/*
* PS3 platform setup routines .
*
* Copyright ( C ) 2006 Sony Computer Entertainment Inc .
* Copyright 2006 Sony Corp .
*
* 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 ; version 2 of the License .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/kernel.h>
# include <linux/delay.h>
# include <linux/fs.h>
# include <linux/root_dev.h>
# include <linux/console.h>
# include <linux/kexec.h>
2007-02-12 00:55:22 -08:00
# include <linux/bootmem.h>
2006-11-23 00:46:51 +01:00
# include <asm/machdep.h>
# include <asm/firmware.h>
# include <asm/time.h>
# include <asm/iommu.h>
# include <asm/udbg.h>
# include <asm/prom.h>
# include <asm/lv1call.h>
# include "platform.h"
# if defined(DEBUG)
2007-06-16 07:19:23 +10:00
# define DBG udbg_printf
2006-11-23 00:46:51 +01:00
# else
2007-06-16 07:19:23 +10:00
# define DBG pr_debug
2006-11-23 00:46:51 +01:00
# endif
2007-02-07 12:20:01 -08:00
# if !defined(CONFIG_SMP)
static void smp_send_stop ( void ) { }
# endif
2007-06-16 07:18:48 +10:00
static union ps3_firmware_version ps3_firmware_version ;
void ps3_get_firmware_version ( union ps3_firmware_version * v )
2007-01-26 19:08:21 -08:00
{
2007-06-16 07:18:48 +10:00
* v = ps3_firmware_version ;
}
EXPORT_SYMBOL_GPL ( ps3_get_firmware_version ) ;
2007-01-26 19:08:21 -08:00
2007-06-16 07:18:48 +10:00
int ps3_compare_firmware_version ( u16 major , u16 minor , u16 rev )
{
union ps3_firmware_version x ;
x . pad = 0 ;
x . major = major ;
x . minor = minor ;
x . rev = rev ;
2007-01-26 19:08:21 -08:00
2007-08-29 20:30:25 +09:00
return ( ps3_firmware_version . raw > x . raw ) -
( ps3_firmware_version . raw < x . raw ) ;
2007-01-26 19:08:21 -08:00
}
2007-06-16 07:18:48 +10:00
EXPORT_SYMBOL_GPL ( ps3_compare_firmware_version ) ;
2007-01-26 19:08:21 -08:00
2006-11-23 00:46:51 +01:00
static void ps3_power_save ( void )
{
/*
* lv1_pause ( ) puts the PPE thread into inactive state until an
* irq on an unmasked plug exists . MSR [ EE ] has no effect .
* flags : 0 = wake on DEC interrupt , 1 = ignore DEC interrupt .
*/
lv1_pause ( 0 ) ;
}
2007-02-07 12:20:01 -08:00
static void ps3_restart ( char * cmd )
{
DBG ( " %s:%d cmd '%s' \n " , __func__ , __LINE__ , cmd ) ;
smp_send_stop ( ) ;
ps3_sys_manager_restart ( ) ; /* never returns */
}
static void ps3_power_off ( void )
{
DBG ( " %s:%d \n " , __func__ , __LINE__ ) ;
smp_send_stop ( ) ;
ps3_sys_manager_power_off ( ) ; /* never returns */
}
2006-11-23 00:46:51 +01:00
static void ps3_panic ( char * str )
{
DBG ( " %s:%d %s \n " , __func__ , __LINE__ , str ) ;
smp_send_stop ( ) ;
printk ( " \n " ) ;
printk ( " System does not reboot automatically. \n " ) ;
printk ( " Please press POWER button. \n " ) ;
printk ( " \n " ) ;
2007-02-07 12:20:01 -08:00
while ( 1 ) ;
2006-11-23 00:46:51 +01:00
}
2007-06-22 00:14:20 +10:00
# if defined(CONFIG_FB_PS3) || defined(CONFIG_FB_PS3_MODULE) || \
defined ( CONFIG_PS3_FLASH ) | | defined ( CONFIG_PS3_FLASH_MODULE )
2007-07-31 17:22:00 +10:00
static void __init prealloc ( struct ps3_prealloc * p )
2007-02-12 00:55:22 -08:00
{
if ( ! p - > size )
return ;
p - > address = __alloc_bootmem ( p - > size , p - > align , __pa ( MAX_DMA_ADDRESS ) ) ;
if ( ! p - > address ) {
printk ( KERN_ERR " %s: Cannot allocate %s \n " , __FUNCTION__ ,
p - > name ) ;
return ;
}
printk ( KERN_INFO " %s: %lu bytes at %p \n " , p - > name , p - > size ,
p - > address ) ;
}
2007-06-22 00:14:20 +10:00
# endif
2007-02-12 00:55:22 -08:00
2007-06-22 00:14:20 +10:00
# if defined(CONFIG_FB_PS3) || defined(CONFIG_FB_PS3_MODULE)
2007-02-12 00:55:22 -08:00
struct ps3_prealloc ps3fb_videomemory = {
2007-06-16 08:05:38 +10:00
. name = " ps3fb videomemory " ,
. size = CONFIG_FB_PS3_DEFAULT_SIZE_M * 1024 * 1024 ,
. align = 1024 * 1024 /* the GPU requires 1 MiB alignment */
2007-02-12 00:55:22 -08:00
} ;
2007-06-16 08:05:38 +10:00
EXPORT_SYMBOL_GPL ( ps3fb_videomemory ) ;
2007-02-12 00:55:22 -08:00
# define prealloc_ps3fb_videomemory() prealloc(&ps3fb_videomemory)
static int __init early_parse_ps3fb ( char * p )
{
if ( ! p )
return 1 ;
ps3fb_videomemory . size = _ALIGN_UP ( memparse ( p , & p ) ,
ps3fb_videomemory . align ) ;
return 0 ;
}
early_param ( " ps3fb " , early_parse_ps3fb ) ;
# else
# define prealloc_ps3fb_videomemory() do { } while (0)
# endif
2007-06-22 00:14:20 +10:00
# if defined(CONFIG_PS3_FLASH) || defined(CONFIG_PS3_FLASH_MODULE)
struct ps3_prealloc ps3flash_bounce_buffer = {
. name = " ps3flash bounce buffer " ,
. size = 256 * 1024 ,
. align = 256 * 1024
} ;
EXPORT_SYMBOL_GPL ( ps3flash_bounce_buffer ) ;
# define prealloc_ps3flash_bounce_buffer() prealloc(&ps3flash_bounce_buffer)
static int __init early_parse_ps3flash ( char * p )
{
if ( ! p )
return 1 ;
if ( ! strcmp ( p , " off " ) )
ps3flash_bounce_buffer . size = 0 ;
return 0 ;
}
early_param ( " ps3flash " , early_parse_ps3flash ) ;
# else
# define prealloc_ps3flash_bounce_buffer() do { } while (0)
# endif
2007-05-01 07:00:50 +10:00
static int ps3_set_dabr ( u64 dabr )
{
enum { DABR_USER = 1 , DABR_KERNEL = 2 , } ;
return lv1_set_dabr ( dabr , DABR_KERNEL | DABR_USER ) ? - 1 : 0 ;
}
2007-02-12 00:55:22 -08:00
2006-11-23 00:46:51 +01:00
static void __init ps3_setup_arch ( void )
{
2007-01-26 19:08:21 -08:00
2006-11-23 00:46:51 +01:00
DBG ( " -> %s:%d \n " , __func__ , __LINE__ ) ;
2007-06-16 07:18:48 +10:00
lv1_get_version_info ( & ps3_firmware_version . raw ) ;
printk ( KERN_INFO " PS3 firmware version %u.%u.%u \n " ,
ps3_firmware_version . major , ps3_firmware_version . minor ,
ps3_firmware_version . rev ) ;
2007-01-26 19:08:21 -08:00
2006-11-23 00:46:51 +01:00
ps3_spu_set_platform ( ) ;
ps3_map_htab ( ) ;
# ifdef CONFIG_SMP
smp_init_ps3 ( ) ;
# endif
# ifdef CONFIG_DUMMY_CONSOLE
conswitchp = & dummy_con ;
# endif
2007-02-12 00:55:22 -08:00
prealloc_ps3fb_videomemory ( ) ;
2007-06-22 00:14:20 +10:00
prealloc_ps3flash_bounce_buffer ( ) ;
2006-11-23 00:46:51 +01:00
ppc_md . power_save = ps3_power_save ;
DBG ( " <- %s:%d \n " , __func__ , __LINE__ ) ;
}
static void __init ps3_progress ( char * s , unsigned short hex )
{
printk ( " *** %04x : %s \n " , hex , s ? s : " " ) ;
}
static int __init ps3_probe ( void )
{
unsigned long htab_size ;
unsigned long dt_root ;
DBG ( " -> %s:%d \n " , __func__ , __LINE__ ) ;
dt_root = of_get_flat_dt_root ( ) ;
2007-06-16 08:06:51 +10:00
if ( ! of_flat_dt_is_compatible ( dt_root , " sony,ps3 " ) )
2006-11-23 00:46:51 +01:00
return 0 ;
2006-11-27 19:18:57 +01:00
powerpc_firmware_features | = FW_FEATURE_PS3_POSSIBLE ;
2006-11-23 00:46:51 +01:00
ps3_os_area_init ( ) ;
ps3_mm_init ( ) ;
ps3_mm_vas_create ( & htab_size ) ;
ps3_hpte_init ( htab_size ) ;
DBG ( " <- %s:%d \n " , __func__ , __LINE__ ) ;
return 1 ;
}
# if defined(CONFIG_KEXEC)
static void ps3_kexec_cpu_down ( int crash_shutdown , int secondary )
{
2007-06-16 07:19:32 +10:00
int cpu = smp_processor_id ( ) ;
2006-11-23 00:46:51 +01:00
2007-06-16 07:19:32 +10:00
DBG ( " -> %s:%d: (%d) \n " , __func__ , __LINE__ , cpu ) ;
2006-11-23 00:46:51 +01:00
2007-06-16 07:19:32 +10:00
ps3_smp_cleanup_cpu ( cpu ) ;
ps3_shutdown_IRQ ( cpu ) ;
2006-11-23 00:46:51 +01:00
DBG ( " <- %s:%d \n " , __func__ , __LINE__ ) ;
}
# endif
define_machine ( ps3 ) {
. name = " PS3 " ,
. probe = ps3_probe ,
. setup_arch = ps3_setup_arch ,
. init_IRQ = ps3_init_IRQ ,
. panic = ps3_panic ,
. get_boot_time = ps3_get_boot_time ,
. set_rtc_time = ps3_set_rtc_time ,
. get_rtc_time = ps3_get_rtc_time ,
2007-05-01 07:00:50 +10:00
. set_dabr = ps3_set_dabr ,
2006-11-23 00:46:51 +01:00
. calibrate_decr = ps3_calibrate_decr ,
. progress = ps3_progress ,
2007-02-07 12:20:01 -08:00
. restart = ps3_restart ,
. power_off = ps3_power_off ,
2006-11-23 00:46:51 +01:00
# if defined(CONFIG_KEXEC)
. kexec_cpu_down = ps3_kexec_cpu_down ,
2007-06-16 07:19:32 +10:00
. machine_kexec = default_machine_kexec ,
2006-11-23 00:46:51 +01:00
. machine_kexec_prepare = default_machine_kexec_prepare ,
. machine_crash_shutdown = default_machine_crash_shutdown ,
# endif
} ;