2005-09-26 10:04:21 +04:00
/*
2005-10-22 10:02:39 +04:00
* Powermac setup and early boot code plus other random bits .
2005-09-26 10:04:21 +04:00
*
* PowerPC version
* Copyright ( C ) 1995 - 1996 Gary Thomas ( gdt @ linuxppc . org )
*
* Adapted for Power Macintosh by Paul Mackerras
2005-10-22 10:02:39 +04:00
* Copyright ( C ) 1996 Paul Mackerras ( paulus @ samba . org )
2005-09-26 10:04:21 +04:00
*
* Derived from " arch/alpha/kernel/setup.c "
* Copyright ( C ) 1995 Linus Torvalds
*
* Maintained by Benjamin Herrenschmidt ( benh @ kernel . crashing . org )
*
* 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 .
*
*/
/*
* bootup setup stuff . .
*/
# include <linux/init.h>
# include <linux/errno.h>
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/stddef.h>
# include <linux/unistd.h>
# include <linux/ptrace.h>
# include <linux/slab.h>
# include <linux/user.h>
# include <linux/a.out.h>
# include <linux/tty.h>
# include <linux/string.h>
# include <linux/delay.h>
# include <linux/ioport.h>
# include <linux/major.h>
# include <linux/initrd.h>
# include <linux/vt_kern.h>
# include <linux/console.h>
# include <linux/pci.h>
# include <linux/adb.h>
# include <linux/cuda.h>
# include <linux/pmu.h>
# include <linux/irq.h>
# include <linux/seq_file.h>
# include <linux/root_dev.h>
# include <linux/bitops.h>
# include <linux/suspend.h>
# include <asm/reg.h>
# include <asm/sections.h>
# include <asm/prom.h>
# include <asm/system.h>
# include <asm/pgtable.h>
# include <asm/io.h>
[PATCH] powerpc: Merge kexec
This patch merges, to some extent, the PPC32 and PPC64 kexec implementations.
We adopt the PPC32 approach of having ppc_md callbacks for the kexec functions.
The current PPC64 implementation becomes the "default" implementation for PPC64
which platforms can select if they need no special treatment.
I've added these default callbacks to pseries/maple/cell/powermac, this means
iSeries no longer supports kexec - but it never worked anyway.
I've renamed PPC32's machine_kexec_simple to default_machine_kexec, inline with
PPC64. Judging by the comments it might be better named machine_kexec_non_of,
or something, but at the moment it's the only implementation for PPC32 so it's
the "default".
Kexec requires machine_shutdown(), which is in machine_kexec.c on PPC32, but we
already have in setup-common.c on powerpc. All this does is call
ppc_md.nvram_sync, which only powermac implements, so instead make
machine_shutdown a ppc_md member and have it call core99_nvram_sync directly
on powermac.
I've also stuck relocate_kernel.S into misc_32.S for powerpc.
Built for ARCH=ppc, and 32 & 64 bit ARCH=powerpc, with KEXEC=y/n. Booted on
P5 LPAR and successfully kexec'ed.
Should apply on top of 493f25ef4087395891c99fcfe2c72e62e293e89f.
Signed-off-by: Michael Ellerman <michael@ellerman.id.au>
Signed-off-by: Paul Mackerras <paulus@samba.org>
2005-11-14 15:35:00 +03:00
# include <asm/kexec.h>
2005-09-26 10:04:21 +04:00
# include <asm/pci-bridge.h>
# include <asm/ohare.h>
# include <asm/mediabay.h>
# include <asm/machdep.h>
# include <asm/dma.h>
# include <asm/cputable.h>
# include <asm/btext.h>
# include <asm/pmac_feature.h>
# include <asm/time.h>
# include <asm/of_device.h>
2006-11-11 09:24:59 +03:00
# include <asm/of_platform.h>
2005-09-26 10:04:21 +04:00
# include <asm/mmu_context.h>
2005-10-22 10:02:39 +04:00
# include <asm/iommu.h>
# include <asm/smu.h>
# include <asm/pmc.h>
2005-11-02 08:07:22 +03:00
# include <asm/lmb.h>
2005-11-23 09:57:25 +03:00
# include <asm/udbg.h>
2005-09-26 10:04:21 +04:00
2005-10-10 16:58:41 +04:00
# include "pmac.h"
2005-09-26 10:04:21 +04:00
# undef SHOW_GATWICK_IRQS
int ppc_override_l2cr = 0 ;
int ppc_override_l2cr_value ;
int has_l2cache = 0 ;
2006-01-23 00:19:02 +03:00
int pmac_newworld ;
2005-10-06 06:06:20 +04:00
2005-09-26 10:04:21 +04:00
static int current_root_goodness = - 1 ;
2005-10-22 10:02:39 +04:00
extern struct machdep_calls pmac_md ;
2005-09-26 10:04:21 +04:00
# define DEFAULT_ROOT_DEVICE Root_SDA1 /* sda1 - slightly silly choice */
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC64
# include <asm/udbg.h>
int sccdbg ;
2005-09-26 10:04:21 +04:00
# endif
2005-10-22 10:02:39 +04:00
extern void zs_kgdb_hook ( int tty_num ) ;
2005-09-26 10:04:21 +04:00
sys_ctrler_t sys_ctrler = SYS_CTRLER_UNKNOWN ;
2005-10-22 10:02:39 +04:00
EXPORT_SYMBOL ( sys_ctrler ) ;
# ifdef CONFIG_PMAC_SMU
unsigned long smu_cmdbuf_abs ;
EXPORT_SYMBOL ( smu_cmdbuf_abs ) ;
# endif
2005-09-26 10:04:21 +04:00
# ifdef CONFIG_SMP
extern struct smp_ops_t psurge_smp_ops ;
extern struct smp_ops_t core99_smp_ops ;
# endif /* CONFIG_SMP */
2005-10-20 14:48:19 +04:00
static void pmac_show_cpuinfo ( struct seq_file * m )
2005-09-26 10:04:21 +04:00
{
struct device_node * np ;
2006-07-12 09:40:29 +04:00
const char * pp ;
2005-09-26 10:04:21 +04:00
int plen ;
2005-10-20 14:48:19 +04:00
int mbmodel ;
unsigned int mbflags ;
2005-09-26 10:04:21 +04:00
char * mbname ;
2005-10-20 14:48:19 +04:00
mbmodel = pmac_call_feature ( PMAC_FTR_GET_MB_INFO , NULL ,
PMAC_MB_INFO_MODEL , 0 ) ;
mbflags = pmac_call_feature ( PMAC_FTR_GET_MB_INFO , NULL ,
PMAC_MB_INFO_FLAGS , 0 ) ;
if ( pmac_call_feature ( PMAC_FTR_GET_MB_INFO , NULL , PMAC_MB_INFO_NAME ,
( long ) & mbname ) ! = 0 )
2005-09-26 10:04:21 +04:00
mbname = " Unknown " ;
/* find motherboard type */
seq_printf ( m , " machine \t \t : " ) ;
2005-10-20 14:48:19 +04:00
np = of_find_node_by_path ( " / " ) ;
2005-09-26 10:04:21 +04:00
if ( np ! = NULL ) {
2007-04-03 16:26:41 +04:00
pp = of_get_property ( np , " model " , NULL ) ;
2005-09-26 10:04:21 +04:00
if ( pp ! = NULL )
seq_printf ( m , " %s \n " , pp ) ;
else
seq_printf ( m , " PowerMac \n " ) ;
2007-04-03 16:26:41 +04:00
pp = of_get_property ( np , " compatible " , & plen ) ;
2005-09-26 10:04:21 +04:00
if ( pp ! = NULL ) {
seq_printf ( m , " motherboard \t : " ) ;
while ( plen > 0 ) {
int l = strlen ( pp ) + 1 ;
seq_printf ( m , " %s " , pp ) ;
plen - = l ;
pp + = l ;
}
seq_printf ( m , " \n " ) ;
}
2005-10-20 14:48:19 +04:00
of_node_put ( np ) ;
2005-09-26 10:04:21 +04:00
} else
seq_printf ( m , " PowerMac \n " ) ;
/* print parsed model */
seq_printf ( m , " detected as \t : %d (%s) \n " , mbmodel , mbname ) ;
seq_printf ( m , " pmac flags \t : %08x \n " , mbflags ) ;
/* find l2 cache info */
2005-10-20 14:48:19 +04:00
np = of_find_node_by_name ( NULL , " l2-cache " ) ;
if ( np = = NULL )
np = of_find_node_by_type ( NULL , " cache " ) ;
if ( np ! = NULL ) {
2007-04-03 16:26:41 +04:00
const unsigned int * ic =
of_get_property ( np , " i-cache-size " , NULL ) ;
const unsigned int * dc =
of_get_property ( np , " d-cache-size " , NULL ) ;
2005-09-26 10:04:21 +04:00
seq_printf ( m , " L2 cache \t : " ) ;
has_l2cache = 1 ;
2007-04-03 16:26:41 +04:00
if ( of_get_property ( np , " cache-unified " , NULL ) ! = 0 & & dc ) {
2005-09-26 10:04:21 +04:00
seq_printf ( m , " %dK unified " , * dc / 1024 ) ;
} else {
if ( ic )
seq_printf ( m , " %dK instruction " , * ic / 1024 ) ;
if ( dc )
seq_printf ( m , " %s %dK data " ,
( ic ? " + " : " " ) , * dc / 1024 ) ;
}
2007-04-03 16:26:41 +04:00
pp = of_get_property ( np , " ram-type " , NULL ) ;
2005-09-26 10:04:21 +04:00
if ( pp )
seq_printf ( m , " %s " , pp ) ;
seq_printf ( m , " \n " ) ;
2005-10-20 14:48:19 +04:00
of_node_put ( np ) ;
2005-09-26 10:04:21 +04:00
}
/* Indicate newworld/oldworld */
seq_printf ( m , " pmac-generation \t : %s \n " ,
pmac_newworld ? " NewWorld " : " OldWorld " ) ;
}
2005-10-22 10:02:39 +04:00
# ifndef CONFIG_ADB_CUDA
int find_via_cuda ( void )
{
if ( ! find_devices ( " via-cuda " ) )
return 0 ;
printk ( " WARNING ! Your machine is CUDA-based but your kernel \n " ) ;
printk ( " wasn't compiled with CONFIG_ADB_CUDA option ! \n " ) ;
return 0 ;
}
# endif
2005-09-26 10:04:21 +04:00
2005-10-22 10:02:39 +04:00
# ifndef CONFIG_ADB_PMU
int find_via_pmu ( void )
2005-09-26 10:04:21 +04:00
{
2005-10-22 10:02:39 +04:00
if ( ! find_devices ( " via-pmu " ) )
return 0 ;
printk ( " WARNING ! Your machine is PMU-based but your kernel \n " ) ;
printk ( " wasn't compiled with CONFIG_ADB_PMU option ! \n " ) ;
2005-10-23 11:23:21 +04:00
return 0 ;
2005-10-22 10:02:39 +04:00
}
# endif
2005-09-26 10:04:21 +04:00
2005-10-22 10:02:39 +04:00
# ifndef CONFIG_PMAC_SMU
int smu_init ( void )
{
/* should check and warn if SMU is present */
return 0 ;
}
# endif
2005-09-26 10:04:21 +04:00
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC32
static volatile u32 * sysctrl_regs ;
2005-09-26 10:04:21 +04:00
2005-10-22 10:02:39 +04:00
static void __init ohare_init ( void )
{
2005-09-26 10:04:21 +04:00
/* this area has the CPU identification register
and some registers used by smp boards */
sysctrl_regs = ( volatile u32 * ) ioremap ( 0xf8000000 , 0x1000 ) ;
2005-10-22 10:02:39 +04:00
/*
* Turn on the L2 cache .
* We assume that we have a PSX memory controller iff
* we have an ohare I / O controller .
*/
if ( find_devices ( " ohare " ) ! = NULL ) {
if ( ( ( sysctrl_regs [ 2 ] > > 24 ) & 0xf ) > = 3 ) {
if ( sysctrl_regs [ 4 ] & 0x10 )
sysctrl_regs [ 4 ] | = 0x04000020 ;
else
sysctrl_regs [ 4 ] | = 0x04000000 ;
if ( has_l2cache )
printk ( KERN_INFO " Level 2 cache enabled \n " ) ;
}
}
}
2005-09-26 10:04:21 +04:00
2005-10-22 10:02:39 +04:00
static void __init l2cr_init ( void )
{
2005-09-26 10:04:21 +04:00
/* Checks "l2cr-value" property in the registry */
if ( cpu_has_feature ( CPU_FTR_L2CR ) ) {
struct device_node * np = find_devices ( " cpus " ) ;
if ( np = = 0 )
np = find_type_devices ( " cpu " ) ;
if ( np ! = 0 ) {
2006-07-12 09:40:29 +04:00
const unsigned int * l2cr =
2007-04-03 16:26:41 +04:00
of_get_property ( np , " l2cr-value " , NULL ) ;
2005-09-26 10:04:21 +04:00
if ( l2cr ! = 0 ) {
ppc_override_l2cr = 1 ;
ppc_override_l2cr_value = * l2cr ;
_set_L2CR ( 0 ) ;
_set_L2CR ( ppc_override_l2cr_value ) ;
}
}
}
if ( ppc_override_l2cr )
2005-10-22 10:02:39 +04:00
printk ( KERN_INFO " L2CR overridden (0x%x), "
" backside cache is %s \n " ,
ppc_override_l2cr_value ,
( ppc_override_l2cr_value & 0x80000000 )
2005-09-26 10:04:21 +04:00
? " enabled " : " disabled " ) ;
2005-10-22 10:02:39 +04:00
}
# endif
2006-01-11 03:00:03 +03:00
static void __init pmac_setup_arch ( void )
2005-10-22 10:02:39 +04:00
{
2005-10-23 11:23:21 +04:00
struct device_node * cpu , * ic ;
2006-07-12 09:40:29 +04:00
const int * fp ;
2005-10-22 10:02:39 +04:00
unsigned long pvr ;
pvr = PVR_VER ( mfspr ( SPRN_PVR ) ) ;
/* Set loops_per_jiffy to a half-way reasonable value,
for use until calibrate_delay gets called . */
loops_per_jiffy = 50000000 / HZ ;
cpu = of_find_node_by_type ( NULL , " cpu " ) ;
if ( cpu ! = NULL ) {
2007-04-03 16:26:41 +04:00
fp = of_get_property ( cpu , " clock-frequency " , NULL ) ;
2005-10-22 10:02:39 +04:00
if ( fp ! = NULL ) {
if ( pvr > = 0x30 & & pvr < 0x80 )
/* PPC970 etc. */
loops_per_jiffy = * fp / ( 3 * HZ ) ;
else if ( pvr = = 4 | | pvr > = 8 )
/* 604, G3, G4 etc. */
loops_per_jiffy = * fp / HZ ;
else
/* 601, 603, etc. */
loops_per_jiffy = * fp / ( 2 * HZ ) ;
}
of_node_put ( cpu ) ;
}
2005-10-23 11:23:21 +04:00
/* See if newworld or oldworld */
2005-10-27 16:45:33 +04:00
for ( ic = NULL ; ( ic = of_find_all_nodes ( ic ) ) ! = NULL ; )
2007-04-03 16:26:41 +04:00
if ( of_get_property ( ic , " interrupt-controller " , NULL ) )
2005-10-27 16:45:33 +04:00
break ;
2006-01-23 00:19:02 +03:00
if ( ic ) {
pmac_newworld = 1 ;
2005-10-23 11:23:21 +04:00
of_node_put ( ic ) ;
2006-01-23 00:19:02 +03:00
}
2005-10-23 11:23:21 +04:00
2005-10-22 10:02:39 +04:00
/* Lookup PCI hosts */
pmac_pci_init ( ) ;
# ifdef CONFIG_PPC32
ohare_init ( ) ;
l2cr_init ( ) ;
# endif /* CONFIG_PPC32 */
2005-09-26 10:04:21 +04:00
# ifdef CONFIG_KGDB
zs_kgdb_hook ( 0 ) ;
# endif
find_via_cuda ( ) ;
find_via_pmu ( ) ;
2005-10-22 10:02:39 +04:00
smu_init ( ) ;
2005-11-02 19:54:46 +03:00
# if defined(CONFIG_NVRAM) || defined(CONFIG_PPC64)
2005-09-26 10:04:21 +04:00
pmac_nvram_init ( ) ;
# endif
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC32
2005-09-26 10:04:21 +04:00
# ifdef CONFIG_BLK_DEV_INITRD
if ( initrd_start )
ROOT_DEV = Root_RAM0 ;
else
# endif
ROOT_DEV = DEFAULT_ROOT_DEVICE ;
2005-10-22 10:02:39 +04:00
# endif
2005-09-26 10:04:21 +04:00
# ifdef CONFIG_SMP
/* Check for Core99 */
2005-12-14 05:10:10 +03:00
if ( find_devices ( " uni-n " ) | | find_devices ( " u3 " ) | | find_devices ( " u4 " ) )
2005-10-19 15:44:51 +04:00
smp_ops = & core99_smp_ops ;
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC32
2005-09-26 10:04:21 +04:00
else
2005-10-19 15:44:51 +04:00
smp_ops = & psurge_smp_ops ;
2005-10-22 10:02:39 +04:00
# endif
2005-09-26 10:04:21 +04:00
# endif /* CONFIG_SMP */
2006-03-28 16:15:54 +04:00
# ifdef CONFIG_ADB
if ( strstr ( cmd_line , " adb_sync " ) ) {
extern int __adb_probe_sync ;
__adb_probe_sync = 1 ;
}
# endif /* CONFIG_ADB */
2005-09-26 10:04:21 +04:00
}
2005-10-06 06:06:20 +04:00
char * bootpath ;
char * bootdevice ;
2005-09-26 10:04:21 +04:00
void * boot_host ;
int boot_target ;
int boot_part ;
2006-11-02 15:56:10 +03:00
static dev_t boot_dev ;
2005-09-26 10:04:21 +04:00
# ifdef CONFIG_SCSI
2005-10-22 10:02:39 +04:00
void __init note_scsi_host ( struct device_node * node , void * host )
2005-09-26 10:04:21 +04:00
{
int l ;
char * p ;
l = strlen ( node - > full_name ) ;
if ( bootpath ! = NULL & & bootdevice ! = NULL
& & strncmp ( node - > full_name , bootdevice , l ) = = 0
& & ( bootdevice [ l ] = = ' / ' | | bootdevice [ l ] = = 0 ) ) {
boot_host = host ;
/*
* There ' s a bug in OF 1.0 .5 . ( Why am I not surprised . )
* If you pass a path like scsi / sd @ 1 : 0 to canon , it returns
* something like / bandit @ F2000000 / gc @ 10 / 53 c94 @ 10000 / sd @ 0 , 0
* That is , the scsi target number doesn ' t get preserved .
* So we pick the target number out of bootpath and use that .
*/
p = strstr ( bootpath , " /sd@ " ) ;
if ( p ! = NULL ) {
p + = 4 ;
boot_target = simple_strtoul ( p , NULL , 10 ) ;
p = strchr ( p , ' : ' ) ;
if ( p ! = NULL )
boot_part = simple_strtoul ( p + 1 , NULL , 10 ) ;
}
}
}
2005-10-06 06:06:20 +04:00
EXPORT_SYMBOL ( note_scsi_host ) ;
2005-09-26 10:04:21 +04:00
# endif
# if defined(CONFIG_BLK_DEV_IDE) && defined(CONFIG_BLK_DEV_IDE_PMAC)
2005-10-22 10:02:39 +04:00
static dev_t __init find_ide_boot ( void )
2005-09-26 10:04:21 +04:00
{
char * p ;
int n ;
dev_t __init pmac_find_ide_boot ( char * bootdevice , int n ) ;
if ( bootdevice = = NULL )
return 0 ;
p = strrchr ( bootdevice , ' / ' ) ;
if ( p = = NULL )
return 0 ;
n = p - bootdevice ;
return pmac_find_ide_boot ( bootdevice , n ) ;
}
# endif /* CONFIG_BLK_DEV_IDE && CONFIG_BLK_DEV_IDE_PMAC */
2005-10-22 10:02:39 +04:00
static void __init find_boot_device ( void )
2005-09-26 10:04:21 +04:00
{
# if defined(CONFIG_BLK_DEV_IDE) && defined(CONFIG_BLK_DEV_IDE_PMAC)
boot_dev = find_ide_boot ( ) ;
# endif
}
/* TODO: Merge the suspend-to-ram with the common code !!!
* currently , this is a stub implementation for suspend - to - disk
* only
*/
# ifdef CONFIG_SOFTWARE_SUSPEND
static int pmac_pm_prepare ( suspend_state_t state )
{
printk ( KERN_DEBUG " %s(%d) \n " , __FUNCTION__ , state ) ;
return 0 ;
}
static int pmac_pm_enter ( suspend_state_t state )
{
printk ( KERN_DEBUG " %s(%d) \n " , __FUNCTION__ , state ) ;
/* Giveup the lazy FPU & vec so we don't have to back them
* up from the low level code
*/
enable_kernel_fp ( ) ;
# ifdef CONFIG_ALTIVEC
2005-09-28 00:13:12 +04:00
if ( cur_cpu_spec - > cpu_features & CPU_FTR_ALTIVEC )
2005-09-26 10:04:21 +04:00
enable_kernel_altivec ( ) ;
# endif /* CONFIG_ALTIVEC */
return 0 ;
}
static int pmac_pm_finish ( suspend_state_t state )
{
printk ( KERN_DEBUG " %s(%d) \n " , __FUNCTION__ , state ) ;
/* Restore userland MMU context */
2006-06-11 08:15:17 +04:00
set_context ( current - > active_mm - > context . id , current - > active_mm - > pgd ) ;
2005-09-26 10:04:21 +04:00
return 0 ;
}
2006-05-26 05:44:24 +04:00
static int pmac_pm_valid ( suspend_state_t state )
{
switch ( state ) {
case PM_SUSPEND_DISK :
return 1 ;
/* can't do any other states via generic mechanism yet */
default :
return 0 ;
}
}
2005-09-26 10:04:21 +04:00
static struct pm_ops pmac_pm_ops = {
. pm_disk_mode = PM_DISK_SHUTDOWN ,
. prepare = pmac_pm_prepare ,
. enter = pmac_pm_enter ,
. finish = pmac_pm_finish ,
2006-05-26 05:44:24 +04:00
. valid = pmac_pm_valid ,
2005-09-26 10:04:21 +04:00
} ;
# endif /* CONFIG_SOFTWARE_SUSPEND */
2005-10-22 10:02:39 +04:00
static int initializing = 1 ;
2005-09-26 10:04:21 +04:00
static int pmac_late_init ( void )
{
initializing = 0 ;
# ifdef CONFIG_SOFTWARE_SUSPEND
pm_set_ops ( & pmac_pm_ops ) ;
# endif /* CONFIG_SOFTWARE_SUSPEND */
return 0 ;
}
late_initcall ( pmac_late_init ) ;
/* can't be __init - can be called whenever a disk is first accessed */
2005-10-22 10:02:39 +04:00
void note_bootable_part ( dev_t dev , int part , int goodness )
2005-09-26 10:04:21 +04:00
{
static int found_boot = 0 ;
char * p ;
if ( ! initializing )
return ;
if ( ( goodness < = current_root_goodness ) & &
ROOT_DEV ! = DEFAULT_ROOT_DEVICE )
return ;
2007-02-12 11:54:17 +03:00
p = strstr ( boot_command_line , " root= " ) ;
if ( p ! = NULL & & ( p = = boot_command_line | | p [ - 1 ] = = ' ' ) )
2005-09-26 10:04:21 +04:00
return ;
if ( ! found_boot ) {
find_boot_device ( ) ;
found_boot = 1 ;
}
if ( ! boot_dev | | dev = = boot_dev ) {
ROOT_DEV = dev + part ;
boot_dev = 0 ;
current_root_goodness = goodness ;
}
}
# ifdef CONFIG_ADB_CUDA
2005-10-22 10:02:39 +04:00
static void cuda_restart ( void )
{
2005-09-26 10:04:21 +04:00
struct adb_request req ;
2005-10-22 10:02:39 +04:00
cuda_request ( & req , NULL , 2 , CUDA_PACKET , CUDA_RESET_SYSTEM ) ;
for ( ; ; )
cuda_poll ( ) ;
}
static void cuda_shutdown ( void )
{
struct adb_request req ;
cuda_request ( & req , NULL , 2 , CUDA_PACKET , CUDA_POWERDOWN ) ;
for ( ; ; )
cuda_poll ( ) ;
}
# else
# define cuda_restart()
# define cuda_shutdown()
# endif
# ifndef CONFIG_ADB_PMU
# define pmu_restart()
# define pmu_shutdown()
# endif
# ifndef CONFIG_PMAC_SMU
# define smu_restart()
# define smu_shutdown()
# endif
static void pmac_restart ( char * cmd )
{
2005-09-26 10:04:21 +04:00
switch ( sys_ctrler ) {
case SYS_CTRLER_CUDA :
2005-10-22 10:02:39 +04:00
cuda_restart ( ) ;
2005-09-26 10:04:21 +04:00
break ;
case SYS_CTRLER_PMU :
pmu_restart ( ) ;
break ;
2005-10-22 10:02:39 +04:00
case SYS_CTRLER_SMU :
smu_restart ( ) ;
break ;
2005-09-26 10:04:21 +04:00
default : ;
}
}
2005-10-22 10:02:39 +04:00
static void pmac_power_off ( void )
2005-09-26 10:04:21 +04:00
{
switch ( sys_ctrler ) {
case SYS_CTRLER_CUDA :
2005-10-22 10:02:39 +04:00
cuda_shutdown ( ) ;
2005-09-26 10:04:21 +04:00
break ;
case SYS_CTRLER_PMU :
pmu_shutdown ( ) ;
break ;
2005-10-22 10:02:39 +04:00
case SYS_CTRLER_SMU :
smu_shutdown ( ) ;
break ;
2005-09-26 10:04:21 +04:00
default : ;
}
}
static void
pmac_halt ( void )
{
pmac_power_off ( ) ;
}
2005-10-22 10:02:39 +04:00
/*
* Early initialization .
*/
static void __init pmac_init_early ( void )
{
2005-11-23 09:57:25 +03:00
/* Enable early btext debug if requested */
if ( strstr ( cmd_line , " btextdbg " ) ) {
udbg_adb_init_early ( ) ;
register_early_udbg_console ( ) ;
2005-10-22 10:02:39 +04:00
}
2005-11-23 09:57:25 +03:00
/* Probe motherboard chipset */
pmac_feature_init ( ) ;
/* Initialize debug stuff */
udbg_scc_init ( ! ! strstr ( cmd_line , " sccdbg " ) ) ;
udbg_adb_init ( ! ! strstr ( cmd_line , " btextdbg " ) ) ;
# ifdef CONFIG_PPC64
2005-12-14 05:10:10 +03:00
iommu_init_early_dart ( ) ;
2005-10-22 10:02:39 +04:00
# endif
}
/*
* pmac has no legacy IO , anything calling this function has to
* fail or bad things will happen
*/
static int pmac_check_legacy_ioport ( unsigned int baseport )
{
return - ENODEV ;
}
static int __init pmac_declare_of_platform_devices ( void )
2005-09-26 10:04:21 +04:00
{
2006-01-07 03:35:26 +03:00
struct device_node * np ;
2005-09-26 10:04:21 +04:00
2006-03-28 16:15:54 +04:00
if ( machine_is ( chrp ) )
return - 1 ;
if ( ! machine_is ( powermac ) )
return 0 ;
2006-01-07 03:30:44 +03:00
np = of_find_node_by_name ( NULL , " valkyrie " ) ;
2005-10-22 10:02:39 +04:00
if ( np )
of_platform_device_create ( np , " valkyrie " , NULL ) ;
2006-01-07 03:30:44 +03:00
np = of_find_node_by_name ( NULL , " platinum " ) ;
2005-10-22 10:02:39 +04:00
if ( np )
of_platform_device_create ( np , " platinum " , NULL ) ;
np = of_find_node_by_type ( NULL , " smu " ) ;
if ( np ) {
of_platform_device_create ( np , " smu " , NULL ) ;
of_node_put ( np ) ;
}
2005-09-26 10:04:21 +04:00
return 0 ;
}
device_initcall ( pmac_declare_of_platform_devices ) ;
2005-10-22 10:02:39 +04:00
/*
* Called very early , MMU is off , device - tree isn ' t unflattened
*/
2006-03-28 16:15:54 +04:00
static int __init pmac_probe ( void )
2005-10-22 10:02:39 +04:00
{
2006-03-28 16:15:54 +04:00
unsigned long root = of_get_flat_dt_root ( ) ;
if ( ! of_flat_dt_is_compatible ( root , " Power Macintosh " ) & &
! of_flat_dt_is_compatible ( root , " MacRISC " ) )
2005-10-22 10:02:39 +04:00
return 0 ;
2006-03-28 16:15:54 +04:00
# ifdef CONFIG_PPC64
2005-10-22 10:02:39 +04:00
/*
* On U3 , the DART ( iommu ) must be allocated now since it
* has an impact on htab_initialize ( due to the large page it
* occupies having to be broken up so the DART itself is not
* part of the cacheable linar mapping
*/
2005-12-14 05:10:10 +03:00
alloc_dart_table ( ) ;
2006-06-23 12:16:38 +04:00
hpte_init_native ( ) ;
2005-10-22 10:02:39 +04:00
# endif
2006-03-28 16:15:54 +04:00
# ifdef CONFIG_PPC32
/* isa_io_base gets set in pmac_pci_init */
ISA_DMA_THRESHOLD = ~ 0L ;
DMA_MODE_READ = 1 ;
DMA_MODE_WRITE = 2 ;
# if defined(CONFIG_BLK_DEV_IDE) || defined(CONFIG_BLK_DEV_IDE_MODULE)
# ifdef CONFIG_BLK_DEV_IDE_PMAC
ppc_ide_md . ide_init_hwif = pmac_ide_init_hwif_ports ;
ppc_ide_md . default_io_base = pmac_ide_get_base ;
# endif /* CONFIG_BLK_DEV_IDE_PMAC */
# endif /* defined(CONFIG_BLK_DEV_IDE) || defined(CONFIG_BLK_DEV_IDE_MODULE) */
# endif /* CONFIG_PPC32 */
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PMAC_SMU
/*
* SMU based G5s need some memory below 2 Gb , at least the current
* driver needs that . We have to allocate it now . We allocate 4 k
* ( 1 small page ) for now .
*/
smu_cmdbuf_abs = lmb_alloc_base ( 4096 , 4096 , 0x80000000UL ) ;
# endif /* CONFIG_PMAC_SMU */
return 1 ;
}
# ifdef CONFIG_PPC64
2005-11-23 09:57:25 +03:00
/* Move that to pci.c */
static int pmac_pci_probe_mode ( struct pci_bus * bus )
2005-10-22 10:02:39 +04:00
{
struct device_node * node = bus - > sysdata ;
/* We need to use normal PCI probing for the AGP bus,
2005-12-14 05:10:10 +03:00
* since the device for the AGP bridge isn ' t in the tree .
*/
if ( bus - > self = = NULL & & ( device_is_compatible ( node , " u3-agp " ) | |
device_is_compatible ( node , " u4-pcie " ) ) )
2005-10-22 10:02:39 +04:00
return PCI_PROBE_NORMAL ;
return PCI_PROBE_DEVTREE ;
}
# endif
2006-03-28 16:15:54 +04:00
define_machine ( powermac ) {
. name = " PowerMac " ,
2005-10-22 10:02:39 +04:00
. probe = pmac_probe ,
. setup_arch = pmac_setup_arch ,
. init_early = pmac_init_early ,
. show_cpuinfo = pmac_show_cpuinfo ,
. init_IRQ = pmac_pic_init ,
2005-12-13 10:01:21 +03:00
. get_irq = NULL , /* changed later */
2006-11-11 09:24:51 +03:00
. pci_irq_fixup = pmac_pci_irq_fixup ,
2005-10-22 10:02:39 +04:00
. restart = pmac_restart ,
. power_off = pmac_power_off ,
. halt = pmac_halt ,
. time_init = pmac_time_init ,
. get_boot_time = pmac_get_boot_time ,
. set_rtc_time = pmac_set_rtc_time ,
. get_rtc_time = pmac_get_rtc_time ,
. calibrate_decr = pmac_calibrate_decr ,
. feature_call = pmac_do_feature_call ,
. check_legacy_ioport = pmac_check_legacy_ioport ,
2005-12-21 01:37:07 +03:00
. progress = udbg_progress ,
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC64
2005-11-23 09:57:25 +03:00
. pci_probe_mode = pmac_pci_probe_mode ,
2006-03-27 08:03:03 +04:00
. power_save = power4_idle ,
2005-10-22 10:02:39 +04:00
. enable_pmcs = power4_enable_pmcs ,
[PATCH] powerpc: Merge kexec
This patch merges, to some extent, the PPC32 and PPC64 kexec implementations.
We adopt the PPC32 approach of having ppc_md callbacks for the kexec functions.
The current PPC64 implementation becomes the "default" implementation for PPC64
which platforms can select if they need no special treatment.
I've added these default callbacks to pseries/maple/cell/powermac, this means
iSeries no longer supports kexec - but it never worked anyway.
I've renamed PPC32's machine_kexec_simple to default_machine_kexec, inline with
PPC64. Judging by the comments it might be better named machine_kexec_non_of,
or something, but at the moment it's the only implementation for PPC32 so it's
the "default".
Kexec requires machine_shutdown(), which is in machine_kexec.c on PPC32, but we
already have in setup-common.c on powerpc. All this does is call
ppc_md.nvram_sync, which only powermac implements, so instead make
machine_shutdown a ppc_md member and have it call core99_nvram_sync directly
on powermac.
I've also stuck relocate_kernel.S into misc_32.S for powerpc.
Built for ARCH=ppc, and 32 & 64 bit ARCH=powerpc, with KEXEC=y/n. Booted on
P5 LPAR and successfully kexec'ed.
Should apply on top of 493f25ef4087395891c99fcfe2c72e62e293e89f.
Signed-off-by: Michael Ellerman <michael@ellerman.id.au>
Signed-off-by: Paul Mackerras <paulus@samba.org>
2005-11-14 15:35:00 +03:00
# ifdef CONFIG_KEXEC
. machine_kexec = default_machine_kexec ,
. machine_kexec_prepare = default_machine_kexec_prepare ,
2005-12-04 10:39:43 +03:00
. machine_crash_shutdown = default_machine_crash_shutdown ,
2005-10-22 10:02:39 +04:00
# endif
[PATCH] powerpc: Merge kexec
This patch merges, to some extent, the PPC32 and PPC64 kexec implementations.
We adopt the PPC32 approach of having ppc_md callbacks for the kexec functions.
The current PPC64 implementation becomes the "default" implementation for PPC64
which platforms can select if they need no special treatment.
I've added these default callbacks to pseries/maple/cell/powermac, this means
iSeries no longer supports kexec - but it never worked anyway.
I've renamed PPC32's machine_kexec_simple to default_machine_kexec, inline with
PPC64. Judging by the comments it might be better named machine_kexec_non_of,
or something, but at the moment it's the only implementation for PPC32 so it's
the "default".
Kexec requires machine_shutdown(), which is in machine_kexec.c on PPC32, but we
already have in setup-common.c on powerpc. All this does is call
ppc_md.nvram_sync, which only powermac implements, so instead make
machine_shutdown a ppc_md member and have it call core99_nvram_sync directly
on powermac.
I've also stuck relocate_kernel.S into misc_32.S for powerpc.
Built for ARCH=ppc, and 32 & 64 bit ARCH=powerpc, with KEXEC=y/n. Booted on
P5 LPAR and successfully kexec'ed.
Should apply on top of 493f25ef4087395891c99fcfe2c72e62e293e89f.
Signed-off-by: Michael Ellerman <michael@ellerman.id.au>
Signed-off-by: Paul Mackerras <paulus@samba.org>
2005-11-14 15:35:00 +03:00
# endif /* CONFIG_PPC64 */
2005-10-22 10:02:39 +04:00
# ifdef CONFIG_PPC32
. pcibios_enable_device_hook = pmac_pci_enable_device_hook ,
. pcibios_after_init = pmac_pcibios_after_init ,
. phys_mem_access_prot = pci_phys_mem_access_prot ,
# endif
2006-03-28 16:15:54 +04:00
# if defined(CONFIG_HOTPLUG_CPU) && defined(CONFIG_PPC64)
. cpu_die = generic_mach_cpu_die ,
# endif
2005-10-22 10:02:39 +04:00
} ;