2011-09-19 17:44:52 +00:00
/*
* PowerNV setup code .
*
* Copyright 2011 IBM 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 ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# undef DEBUG
# include <linux/cpu.h>
# include <linux/errno.h>
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/tty.h>
# include <linux/reboot.h>
# include <linux/init.h>
# include <linux/console.h>
# include <linux/delay.h>
# include <linux/irq.h>
# include <linux/seq_file.h>
# include <linux/of.h>
2013-09-26 07:40:04 -05:00
# include <linux/of_fdt.h>
2011-09-19 17:44:52 +00:00
# include <linux/interrupt.h>
# include <linux/bug.h>
2014-02-11 11:32:38 +11:00
# include <linux/pci.h>
2014-03-11 17:01:19 +05:30
# include <linux/cpufreq.h>
2011-09-19 17:44:52 +00:00
# include <asm/machdep.h>
# include <asm/firmware.h>
# include <asm/xics.h>
2011-09-19 17:45:01 +00:00
# include <asm/rtas.h>
2011-09-19 17:44:59 +00:00
# include <asm/opal.h>
2013-08-21 13:03:20 +10:00
# include <asm/kexec.h>
2011-09-19 17:44:52 +00:00
# include "powernv.h"
static void __init pnv_setup_arch ( void )
{
/* Initialize SMP */
pnv_smp_init ( ) ;
2011-09-19 17:45:05 +00:00
/* Setup PCI */
pnv_pci_init ( ) ;
2011-09-19 17:44:52 +00:00
2011-09-19 17:45:01 +00:00
/* Setup RTC and NVRAM callbacks */
if ( firmware_has_feature ( FW_FEATURE_OPAL ) )
opal_nvram_init ( ) ;
2011-09-19 17:44:52 +00:00
/* Enable NAP mode */
powersave_nap = 1 ;
/* XXX PMCS */
}
static void __init pnv_init_early ( void )
{
2013-07-15 13:03:11 +10:00
/*
* Initialize the LPC bus now so that legacy serial
* ports can be found on it
*/
opal_lpc_init ( ) ;
2011-09-19 17:44:59 +00:00
# ifdef CONFIG_HVC_OPAL
if ( firmware_has_feature ( FW_FEATURE_OPAL ) )
hvc_opal_init_early ( ) ;
else
# endif
add_preferred_console ( " hvc " , 0 , NULL ) ;
2011-09-19 17:44:52 +00:00
}
static void __init pnv_init_IRQ ( void )
{
xics_init ( ) ;
WARN_ON ( ! ppc_md . get_irq ) ;
}
static void pnv_show_cpuinfo ( struct seq_file * m )
{
struct device_node * root ;
const char * model = " " ;
root = of_find_node_by_path ( " / " ) ;
if ( root )
model = of_get_property ( root , " model " , NULL ) ;
seq_printf ( m , " machine \t \t : PowerNV %s \n " , model ) ;
2013-05-14 15:10:02 +10:00
if ( firmware_has_feature ( FW_FEATURE_OPALv3 ) )
seq_printf ( m , " firmware \t : OPAL v3 \n " ) ;
else if ( firmware_has_feature ( FW_FEATURE_OPALv2 ) )
2011-09-19 17:44:57 +00:00
seq_printf ( m , " firmware \t : OPAL v2 \n " ) ;
else if ( firmware_has_feature ( FW_FEATURE_OPAL ) )
seq_printf ( m , " firmware \t : OPAL v1 \n " ) ;
else
seq_printf ( m , " firmware \t : BML \n " ) ;
2011-09-19 17:44:52 +00:00
of_node_put ( root ) ;
}
2014-04-09 22:48:55 +05:30
static void pnv_prepare_going_down ( void )
{
/*
* Disable all notifiers from OPAL , we can ' t
* service interrupts anymore anyway
*/
opal_notifier_disable ( ) ;
/* Soft disable interrupts */
local_irq_disable ( ) ;
/*
* Return secondary CPUs to firwmare if a flash update
* is pending otherwise we will get all sort of error
* messages about CPU being stuck etc . . This will also
* have the side effect of hard disabling interrupts so
* past this point , the kernel is effectively dead .
*/
opal_flash_term_callback ( ) ;
}
2011-09-19 18:28:03 +00:00
static void __noreturn pnv_restart ( char * cmd )
2011-09-19 17:44:52 +00:00
{
2011-09-19 18:28:03 +00:00
long rc = OPAL_BUSY ;
2014-04-09 22:48:55 +05:30
pnv_prepare_going_down ( ) ;
2013-06-20 18:13:23 +08:00
2011-09-19 18:28:03 +00:00
while ( rc = = OPAL_BUSY | | rc = = OPAL_BUSY_EVENT ) {
rc = opal_cec_reboot ( ) ;
if ( rc = = OPAL_BUSY_EVENT )
opal_poll_events ( NULL ) ;
else
mdelay ( 10 ) ;
}
for ( ; ; )
opal_poll_events ( NULL ) ;
2011-09-19 17:44:52 +00:00
}
2011-09-19 18:28:03 +00:00
static void __noreturn pnv_power_off ( void )
2011-09-19 17:44:52 +00:00
{
2011-09-19 18:28:03 +00:00
long rc = OPAL_BUSY ;
2014-04-09 22:48:55 +05:30
pnv_prepare_going_down ( ) ;
2013-06-20 18:13:23 +08:00
2011-09-19 18:28:03 +00:00
while ( rc = = OPAL_BUSY | | rc = = OPAL_BUSY_EVENT ) {
rc = opal_cec_power_down ( 0 ) ;
if ( rc = = OPAL_BUSY_EVENT )
opal_poll_events ( NULL ) ;
else
mdelay ( 10 ) ;
}
for ( ; ; )
opal_poll_events ( NULL ) ;
2011-09-19 17:44:52 +00:00
}
2011-09-19 18:28:03 +00:00
static void __noreturn pnv_halt ( void )
2011-09-19 17:44:52 +00:00
{
2011-09-19 18:28:03 +00:00
pnv_power_off ( ) ;
2011-09-19 17:44:52 +00:00
}
2011-09-19 17:45:01 +00:00
static void pnv_progress ( char * s , unsigned short hex )
2011-09-19 17:44:52 +00:00
{
}
2014-02-11 11:32:38 +11:00
static int pnv_dma_set_mask ( struct device * dev , u64 dma_mask )
{
if ( dev_is_pci ( dev ) )
return pnv_pci_dma_set_mask ( to_pci_dev ( dev ) , dma_mask ) ;
return __dma_set_mask ( dev , dma_mask ) ;
}
2013-05-10 16:59:18 +10:00
static void pnv_shutdown ( void )
{
/* Let the PCI code clear up IODA tables */
pnv_pci_shutdown ( ) ;
2014-01-15 17:02:04 +11:00
/*
* Stop OPAL activity : Unregister all OPAL interrupts so they
* don ' t fire up while we kexec and make sure all potentially
* DMA ' ing ops are complete ( such as dump retrieval ) .
2013-05-10 16:59:18 +10:00
*/
opal_shutdown ( ) ;
}
2011-09-19 17:45:01 +00:00
# ifdef CONFIG_KEXEC
2014-04-24 16:14:25 +10:00
static void pnv_kexec_wait_secondaries_down ( void )
{
int my_cpu , i , notified = - 1 ;
my_cpu = get_cpu ( ) ;
for_each_online_cpu ( i ) {
uint8_t status ;
int64_t rc ;
if ( i = = my_cpu )
continue ;
for ( ; ; ) {
rc = opal_query_cpu_status ( get_hard_smp_processor_id ( i ) ,
& status ) ;
if ( rc ! = OPAL_SUCCESS | | status ! = OPAL_THREAD_STARTED )
break ;
barrier ( ) ;
if ( i ! = notified ) {
printk ( KERN_INFO " kexec: waiting for cpu %d "
" (physical %d) to enter OPAL \n " ,
i , paca [ i ] . hw_cpu_id ) ;
notified = i ;
}
}
}
}
2011-09-19 17:45:01 +00:00
static void pnv_kexec_cpu_down ( int crash_shutdown , int secondary )
2011-09-19 17:44:52 +00:00
{
2011-09-19 17:45:01 +00:00
xics_kexec_teardown_cpu ( secondary ) ;
2013-08-21 13:03:20 +10:00
2014-04-24 16:14:25 +10:00
/* On OPAL v3, we return all CPUs to firmware */
if ( ! firmware_has_feature ( FW_FEATURE_OPALv3 ) )
return ;
if ( secondary ) {
/* Return secondary CPUs to firmware on OPAL v3 */
2013-08-21 13:03:20 +10:00
mb ( ) ;
get_paca ( ) - > kexec_state = KEXEC_STATE_REAL_MODE ;
mb ( ) ;
/* Return the CPU to OPAL */
opal_return_cpu ( ) ;
2014-04-24 16:14:25 +10:00
} else if ( crash_shutdown ) {
/*
* On crash , we don ' t wait for secondaries to go
* down as they might be unreachable or hung , so
* instead we just wait a bit and move on .
*/
mdelay ( 1 ) ;
} else {
/* Primary waits for the secondaries to have reached OPAL */
pnv_kexec_wait_secondaries_down ( ) ;
2013-08-21 13:03:20 +10:00
}
2011-09-19 17:44:52 +00:00
}
2011-09-19 17:45:01 +00:00
# endif /* CONFIG_KEXEC */
2011-09-19 17:44:52 +00:00
2014-06-04 17:52:42 +10:00
# ifdef CONFIG_MEMORY_HOTPLUG_SPARSE
static unsigned long pnv_memory_block_size ( void )
{
return 256UL * 1024 * 1024 ;
}
# endif
2011-09-19 17:45:01 +00:00
static void __init pnv_setup_machdep_opal ( void )
2011-09-19 17:44:52 +00:00
{
2011-09-19 17:45:01 +00:00
ppc_md . get_boot_time = opal_get_boot_time ;
ppc_md . get_rtc_time = opal_get_rtc_time ;
ppc_md . set_rtc_time = opal_set_rtc_time ;
ppc_md . restart = pnv_restart ;
ppc_md . power_off = pnv_power_off ;
ppc_md . halt = pnv_halt ;
2011-09-19 17:45:04 +00:00
ppc_md . machine_check_exception = opal_machine_check ;
2013-12-16 10:46:24 +05:30
ppc_md . mce_check_early_recovery = opal_mce_check_early_recovery ;
2011-09-19 17:44:52 +00:00
}
2011-09-19 17:45:01 +00:00
# ifdef CONFIG_PPC_POWERNV_RTAS
static void __init pnv_setup_machdep_rtas ( void )
2011-09-19 17:44:52 +00:00
{
2011-09-19 17:45:01 +00:00
if ( rtas_token ( " get-time-of-day " ) ! = RTAS_UNKNOWN_SERVICE ) {
ppc_md . get_boot_time = rtas_get_boot_time ;
ppc_md . get_rtc_time = rtas_get_rtc_time ;
ppc_md . set_rtc_time = rtas_set_rtc_time ;
}
ppc_md . restart = rtas_restart ;
ppc_md . power_off = rtas_power_off ;
ppc_md . halt = rtas_halt ;
2011-09-19 17:44:52 +00:00
}
2011-09-19 17:45:01 +00:00
# endif /* CONFIG_PPC_POWERNV_RTAS */
2011-09-19 17:44:52 +00:00
static int __init pnv_probe ( void )
{
unsigned long root = of_get_flat_dt_root ( ) ;
if ( ! of_flat_dt_is_compatible ( root , " ibm,powernv " ) )
return 0 ;
hpte_init_native ( ) ;
2011-09-19 17:45:01 +00:00
if ( firmware_has_feature ( FW_FEATURE_OPAL ) )
pnv_setup_machdep_opal ( ) ;
# ifdef CONFIG_PPC_POWERNV_RTAS
else if ( rtas . base )
pnv_setup_machdep_rtas ( ) ;
# endif /* CONFIG_PPC_POWERNV_RTAS */
2011-09-19 17:44:52 +00:00
pr_debug ( " PowerNV detected ! \n " ) ;
return 1 ;
}
2014-03-11 17:01:19 +05:30
/*
* Returns the cpu frequency for ' cpu ' in Hz . This is used by
* / proc / cpuinfo
*/
unsigned long pnv_get_proc_freq ( unsigned int cpu )
{
unsigned long ret_freq ;
ret_freq = cpufreq_quick_get ( cpu ) * 1000ul ;
/*
* If the backend cpufreq driver does not exist ,
* then fallback to old way of reporting the clockrate .
*/
if ( ! ret_freq )
ret_freq = ppc_proc_freq ;
return ret_freq ;
}
2011-09-19 17:44:52 +00:00
define_machine ( powernv ) {
. name = " PowerNV " ,
. probe = pnv_probe ,
. init_early = pnv_init_early ,
2011-09-19 17:45:01 +00:00
. setup_arch = pnv_setup_arch ,
2011-09-19 17:44:52 +00:00
. init_IRQ = pnv_init_IRQ ,
. show_cpuinfo = pnv_show_cpuinfo ,
2014-03-11 17:01:19 +05:30
. get_proc_freq = pnv_get_proc_freq ,
2011-09-19 17:44:52 +00:00
. progress = pnv_progress ,
2013-05-10 16:59:18 +10:00
. machine_shutdown = pnv_shutdown ,
2014-02-17 10:59:29 -05:00
. power_save = power7_idle ,
2011-09-19 17:44:52 +00:00
. calibrate_decr = generic_calibrate_decr ,
2014-02-11 11:32:38 +11:00
. dma_set_mask = pnv_dma_set_mask ,
2011-09-19 17:44:52 +00:00
# ifdef CONFIG_KEXEC
. kexec_cpu_down = pnv_kexec_cpu_down ,
# endif
2014-06-04 17:52:42 +10:00
# ifdef CONFIG_MEMORY_HOTPLUG_SPARSE
. memory_block_size = pnv_memory_block_size ,
# endif
2011-09-19 17:44:52 +00:00
} ;