2009-01-08 16:46:40 -08:00
/*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*
* Copyright ( C ) 2004 - 2007 Cavium Networks
2012-10-17 00:39:09 +02:00
* Copyright ( C ) 2008 , 2009 Wind River Systems
* written by Ralf Baechle < ralf @ linux - mips . org >
2009-01-08 16:46:40 -08:00
*/
2013-06-13 01:29:24 +02:00
# include <linux/compiler.h>
2013-07-25 20:26:48 +03:00
# include <linux/vmalloc.h>
2009-01-08 16:46:40 -08:00
# include <linux/init.h>
2012-10-17 00:39:09 +02:00
# include <linux/kernel.h>
2009-01-08 16:46:40 -08:00
# include <linux/console.h>
# include <linux/delay.h>
2012-02-28 19:24:46 -05:00
# include <linux/export.h>
2009-01-08 16:46:40 -08:00
# include <linux/interrupt.h>
# include <linux/io.h>
2020-10-09 14:14:46 +02:00
# include <linux/memblock.h>
2009-01-08 16:46:40 -08:00
# include <linux/serial.h>
2009-06-19 14:05:26 +01:00
# include <linux/smp.h>
2009-01-08 16:46:40 -08:00
# include <linux/types.h>
# include <linux/string.h> /* for memset */
# include <linux/tty.h>
# include <linux/time.h>
# include <linux/platform_device.h>
# include <linux/serial_core.h>
# include <linux/serial_8250.h>
2012-07-05 18:12:38 +02:00
# include <linux/of_fdt.h>
# include <linux/libfdt.h>
2012-10-25 16:23:31 +02:00
# include <linux/kexec.h>
2009-01-08 16:46:40 -08:00
# include <asm/processor.h>
# include <asm/reboot.h>
# include <asm/smp-ops.h>
# include <asm/irq_cpu.h>
# include <asm/mipsregs.h>
# include <asm/bootinfo.h>
# include <asm/sections.h>
2018-11-22 00:37:24 +02:00
# include <asm/fw/fw.h>
2018-07-13 17:51:56 +02:00
# include <asm/setup.h>
2018-11-22 00:37:25 +02:00
# include <asm/prom.h>
2009-01-08 16:46:40 -08:00
# include <asm/time.h>
# include <asm/octeon/octeon.h>
2010-08-04 14:53:57 -07:00
# include <asm/octeon/pci-octeon.h>
2015-01-15 16:11:17 +03:00
# include <asm/octeon/cvmx-rst-defs.h>
2009-01-08 16:46:40 -08:00
2016-07-08 21:45:01 -05:00
/*
* TRUE for devices having registers with little - endian byte
* order , FALSE for registers with native - endian byte order .
* PCI mandates little - endian , USB and SATA are configuraable ,
* but we chose little - endian for these .
*/
const bool octeon_should_swizzle_table [ 256 ] = {
[ 0x00 ] = true , /* bootbus/CF */
[ 0x1b ] = true , /* PCI mmio window */
[ 0x1c ] = true , /* PCI mmio window */
[ 0x1d ] = true , /* PCI mmio window */
[ 0x1e ] = true , /* PCI mmio window */
[ 0x68 ] = true , /* OCTEON III USB */
[ 0x69 ] = true , /* OCTEON III USB */
[ 0x6c ] = true , /* OCTEON III SATA */
[ 0x6f ] = true , /* OCTEON II USB */
} ;
EXPORT_SYMBOL ( octeon_should_swizzle_table ) ;
2009-01-08 16:46:40 -08:00
# ifdef CONFIG_PCI
extern void pci_console_init ( const char * arg ) ;
# endif
2016-05-24 17:09:31 +03:00
static unsigned long long max_memory = ULLONG_MAX ;
static unsigned long long reserve_low_mem ;
2009-01-08 16:46:40 -08:00
2023-03-29 12:14:42 +02:00
DEFINE_SEMAPHORE ( octeon_bootbus_sem , 1 ) ;
2015-03-05 17:31:29 +03:00
EXPORT_SYMBOL ( octeon_bootbus_sem ) ;
2018-11-22 00:37:23 +02:00
static struct octeon_boot_descriptor * octeon_boot_desc_ptr ;
2009-01-08 16:46:40 -08:00
struct cvmx_bootinfo * octeon_bootinfo ;
EXPORT_SYMBOL ( octeon_bootinfo ) ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
# ifdef CONFIG_SMP
/*
* Wait for relocation code is prepared and send
* secondary CPUs to spin until kernel is relocated .
*/
static void octeon_kexec_smp_down ( void * ignored )
{
int cpu = smp_processor_id ( ) ;
local_irq_disable ( ) ;
set_cpu_online ( cpu , false ) ;
while ( ! atomic_read ( & kexec_ready_to_reboot ) )
cpu_relax ( ) ;
asm volatile (
" sync \n "
" synci ($0) \n " ) ;
2019-01-12 20:37:28 +01:00
kexec_reboot ( ) ;
2012-10-25 16:23:31 +02:00
}
# endif
# define OCTEON_DDR0_BASE (0x0ULL)
# define OCTEON_DDR0_SIZE (0x010000000ULL)
# define OCTEON_DDR1_BASE (0x410000000ULL)
# define OCTEON_DDR1_SIZE (0x010000000ULL)
# define OCTEON_DDR2_BASE (0x020000000ULL)
# define OCTEON_DDR2_SIZE (0x3e0000000ULL)
# define OCTEON_MAX_PHY_MEM_SIZE (16*1024*1024*1024ULL)
static struct kimage * kimage_ptr ;
static void kexec_bootmem_init ( uint64_t mem_size , uint32_t low_reserved_bytes )
{
int64_t addr ;
struct cvmx_bootmem_desc * bootmem_desc ;
bootmem_desc = cvmx_bootmem_get_desc ( ) ;
if ( mem_size > OCTEON_MAX_PHY_MEM_SIZE ) {
mem_size = OCTEON_MAX_PHY_MEM_SIZE ;
pr_err ( " Error: requested memory too large, "
" truncating to maximum size \n " ) ;
}
bootmem_desc - > major_version = CVMX_BOOTMEM_DESC_MAJ_VER ;
bootmem_desc - > minor_version = CVMX_BOOTMEM_DESC_MIN_VER ;
2016-05-24 17:09:31 +03:00
addr = ( OCTEON_DDR0_BASE + reserve_low_mem + low_reserved_bytes ) ;
2012-10-25 16:23:31 +02:00
bootmem_desc - > head_addr = 0 ;
if ( mem_size < = OCTEON_DDR0_SIZE ) {
__cvmx_bootmem_phy_free ( addr ,
2016-05-24 17:09:31 +03:00
mem_size - reserve_low_mem -
2012-10-25 16:23:31 +02:00
low_reserved_bytes , 0 ) ;
return ;
}
__cvmx_bootmem_phy_free ( addr ,
2016-05-24 17:09:31 +03:00
OCTEON_DDR0_SIZE - reserve_low_mem -
2012-10-25 16:23:31 +02:00
low_reserved_bytes , 0 ) ;
mem_size - = OCTEON_DDR0_SIZE ;
if ( mem_size > OCTEON_DDR1_SIZE ) {
__cvmx_bootmem_phy_free ( OCTEON_DDR1_BASE , OCTEON_DDR1_SIZE , 0 ) ;
__cvmx_bootmem_phy_free ( OCTEON_DDR2_BASE ,
mem_size - OCTEON_DDR1_SIZE , 0 ) ;
} else
__cvmx_bootmem_phy_free ( OCTEON_DDR1_BASE , mem_size , 0 ) ;
}
static int octeon_kexec_prepare ( struct kimage * image )
{
int i ;
char * bootloader = " kexec " ;
octeon_boot_desc_ptr - > argc = 0 ;
for ( i = 0 ; i < image - > nr_segments ; i + + ) {
if ( ! strncmp ( bootloader , ( char * ) image - > segment [ i ] . buf ,
strlen ( bootloader ) ) ) {
/*
* convert command line string to array
* of parameters ( as bootloader does ) .
*/
int argc = 0 , offt ;
char * str = ( char * ) image - > segment [ i ] . buf ;
char * ptr = strchr ( str , ' ' ) ;
while ( ptr & & ( OCTEON_ARGV_MAX_ARGS > argc ) ) {
* ptr = ' \0 ' ;
if ( ptr [ 1 ] ! = ' ' ) {
offt = ( int ) ( ptr - str + 1 ) ;
octeon_boot_desc_ptr - > argv [ argc ] =
image - > segment [ i ] . mem + offt ;
argc + + ;
}
ptr = strchr ( ptr + 1 , ' ' ) ;
}
octeon_boot_desc_ptr - > argc = argc ;
break ;
}
}
/*
* Information about segments will be needed during pre - boot memory
* initialization .
*/
kimage_ptr = image ;
return 0 ;
}
static void octeon_generic_shutdown ( void )
{
2013-03-24 22:18:35 +00:00
int i ;
# ifdef CONFIG_SMP
int cpu ;
# endif
2012-10-25 16:23:31 +02:00
struct cvmx_bootmem_desc * bootmem_desc ;
void * named_block_array_ptr ;
bootmem_desc = cvmx_bootmem_get_desc ( ) ;
named_block_array_ptr =
cvmx_phys_to_ptr ( bootmem_desc - > named_block_array_addr ) ;
# ifdef CONFIG_SMP
/* disable watchdogs */
for_each_online_cpu ( cpu )
cvmx_write_csr ( CVMX_CIU_WDOGX ( cpu_logical_map ( cpu ) ) , 0 ) ;
# else
cvmx_write_csr ( CVMX_CIU_WDOGX ( cvmx_get_core_num ( ) ) , 0 ) ;
# endif
if ( kimage_ptr ! = kexec_crash_image ) {
memset ( named_block_array_ptr ,
0x0 ,
CVMX_BOOTMEM_NUM_NAMED_BLOCKS *
sizeof ( struct cvmx_bootmem_named_block_desc ) ) ;
/*
* Mark all memory ( except low 0x100000 bytes ) as free .
* It is the same thing that bootloader does .
*/
kexec_bootmem_init ( octeon_bootinfo - > dram_size * 1024ULL * 1024ULL ,
0x100000 ) ;
/*
* Allocate all segments to avoid their corruption during boot .
*/
for ( i = 0 ; i < kimage_ptr - > nr_segments ; i + + )
cvmx_bootmem_alloc_address (
kimage_ptr - > segment [ i ] . memsz + 2 * PAGE_SIZE ,
kimage_ptr - > segment [ i ] . mem - PAGE_SIZE ,
PAGE_SIZE ) ;
} else {
/*
* Do not mark all memory as free . Free only named sections
* leaving the rest of memory unchanged .
*/
struct cvmx_bootmem_named_block_desc * ptr =
( struct cvmx_bootmem_named_block_desc * )
named_block_array_ptr ;
for ( i = 0 ; i < bootmem_desc - > named_block_num_blocks ; i + + )
if ( ptr [ i ] . size )
cvmx_bootmem_free_named ( ptr [ i ] . name ) ;
}
kexec_args [ 2 ] = 1UL ; /* running on octeon_main_processor */
kexec_args [ 3 ] = ( unsigned long ) octeon_boot_desc_ptr ;
# ifdef CONFIG_SMP
secondary_kexec_args [ 2 ] = 0UL ; /* running on secondary cpu */
secondary_kexec_args [ 3 ] = ( unsigned long ) octeon_boot_desc_ptr ;
# endif
}
static void octeon_shutdown ( void )
{
octeon_generic_shutdown ( ) ;
# ifdef CONFIG_SMP
smp_call_function ( octeon_kexec_smp_down , NULL , 0 ) ;
smp_wmb ( ) ;
while ( num_online_cpus ( ) > 1 ) {
cpu_relax ( ) ;
mdelay ( 1 ) ;
}
# endif
}
static void octeon_crash_shutdown ( struct pt_regs * regs )
{
octeon_generic_shutdown ( ) ;
default_machine_crash_shutdown ( regs ) ;
}
2016-10-11 13:54:26 -07:00
# ifdef CONFIG_SMP
void octeon_crash_smp_send_stop ( void )
{
int cpu ;
/* disable watchdogs */
for_each_online_cpu ( cpu )
cvmx_write_csr ( CVMX_CIU_WDOGX ( cpu_logical_map ( cpu ) ) , 0 ) ;
}
# endif
2012-10-25 16:23:31 +02:00
# endif /* CONFIG_KEXEC */
2022-07-25 11:17:40 +02:00
uint64_t octeon_reserve32_memory ;
EXPORT_SYMBOL ( octeon_reserve32_memory ) ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
/* crashkernel cmdline parameter is parsed _after_ memory setup
* we also parse it here ( workaround for EHB5200 ) */
static uint64_t crashk_size , crashk_base ;
# endif
2009-01-08 16:46:40 -08:00
static int octeon_uart ;
extern asmlinkage void handle_int ( void ) ;
/**
2021-07-25 09:02:57 -07:00
* octeon_is_simulation - Return non - zero if we are currently running
* in the Octeon simulator
2009-01-08 16:46:40 -08:00
*
2021-07-25 09:02:57 -07:00
* Return : non - 0 if running in the Octeon simulator , 0 otherwise
2009-01-08 16:46:40 -08:00
*/
int octeon_is_simulation ( void )
{
return octeon_bootinfo - > board_type = = CVMX_BOARD_TYPE_SIM ;
}
EXPORT_SYMBOL ( octeon_is_simulation ) ;
/**
2021-07-25 09:02:57 -07:00
* octeon_is_pci_host - Return true if Octeon is in PCI Host mode . This means
2009-01-08 16:46:40 -08:00
* Linux can control the PCI bus .
*
2021-07-25 09:02:57 -07:00
* Return : Non - zero if Octeon is in host mode .
2009-01-08 16:46:40 -08:00
*/
int octeon_is_pci_host ( void )
{
# ifdef CONFIG_PCI
return octeon_bootinfo - > config_flags & CVMX_BOOTINFO_CFG_FLAG_PCI_HOST ;
# else
return 0 ;
# endif
}
/**
2021-07-25 09:02:57 -07:00
* octeon_get_clock_rate - Get the clock rate of Octeon
2009-01-08 16:46:40 -08:00
*
2021-07-25 09:02:57 -07:00
* Return : Clock rate in HZ
2009-01-08 16:46:40 -08:00
*/
uint64_t octeon_get_clock_rate ( void )
{
2010-10-07 16:03:49 -07:00
struct cvmx_sysinfo * sysinfo = cvmx_sysinfo_get ( ) ;
return sysinfo - > cpu_clock_hz ;
2009-01-08 16:46:40 -08:00
}
EXPORT_SYMBOL ( octeon_get_clock_rate ) ;
2010-10-07 16:03:49 -07:00
static u64 octeon_io_clock_rate ;
u64 octeon_get_io_clock_rate ( void )
{
return octeon_io_clock_rate ;
}
EXPORT_SYMBOL ( octeon_get_io_clock_rate ) ;
2009-01-08 16:46:40 -08:00
/**
2021-07-25 09:02:57 -07:00
* octeon_write_lcd - Write to the LCD display connected to the bootbus .
2013-01-22 12:59:30 +01:00
* @ s : String to write
2021-07-25 09:02:57 -07:00
*
* This display exists on most Cavium evaluation boards . If it doesn ' t exist ,
* then this function doesn ' t do anything .
2009-01-08 16:46:40 -08:00
*/
2018-11-22 00:37:23 +02:00
static void octeon_write_lcd ( const char * s )
2009-01-08 16:46:40 -08:00
{
if ( octeon_bootinfo - > led_display_base_addr ) {
void __iomem * lcd_address =
2020-01-06 09:43:50 +01:00
ioremap ( octeon_bootinfo - > led_display_base_addr ,
2009-01-08 16:46:40 -08:00
8 ) ;
int i ;
for ( i = 0 ; i < 8 ; i + + , s + + ) {
if ( * s )
iowrite8 ( * s , lcd_address + i ) ;
else
iowrite8 ( ' ' , lcd_address + i ) ;
}
iounmap ( lcd_address ) ;
}
}
/**
2021-07-25 09:02:57 -07:00
* octeon_get_boot_uart - Return the console uart passed by the bootloader
2009-01-08 16:46:40 -08:00
*
2021-07-25 09:02:57 -07:00
* Return : uart number ( 0 or 1 )
2009-01-08 16:46:40 -08:00
*/
2018-11-22 00:37:23 +02:00
static int octeon_get_boot_uart ( void )
2009-01-08 16:46:40 -08:00
{
2017-02-17 11:45:55 -08:00
return ( octeon_boot_desc_ptr - > flags & OCTEON_BL_FLAG_CONSOLE_UART1 ) ?
2009-01-08 16:46:40 -08:00
1 : 0 ;
}
/**
2021-07-25 09:02:57 -07:00
* octeon_get_boot_coremask - Get the coremask Linux was booted on .
2009-01-08 16:46:40 -08:00
*
2021-07-25 09:02:57 -07:00
* Return : Core mask
2009-01-08 16:46:40 -08:00
*/
int octeon_get_boot_coremask ( void )
{
return octeon_boot_desc_ptr - > core_mask ;
}
/**
2021-07-25 09:02:57 -07:00
* octeon_check_cpu_bist - Check the hardware BIST results for a CPU
2009-01-08 16:46:40 -08:00
*/
void octeon_check_cpu_bist ( void )
{
const int coreid = cvmx_get_core_num ( ) ;
unsigned long long mask ;
unsigned long long bist_val ;
/* Check BIST results for COP0 registers */
mask = 0x1f00000000ull ;
bist_val = read_octeon_c0_icacheerr ( ) ;
if ( bist_val & mask )
pr_err ( " Core%d BIST Failure: CacheErr(icache) = 0x%llx \n " ,
coreid , bist_val ) ;
bist_val = read_octeon_c0_dcacheerr ( ) ;
if ( bist_val & 1 )
pr_err ( " Core%d L1 Dcache parity error: "
" CacheErr(dcache) = 0x%llx \n " ,
coreid , bist_val ) ;
mask = 0xfc00000000000000ull ;
bist_val = read_c0_cvmmemctl ( ) ;
if ( bist_val & mask )
pr_err ( " Core%d BIST Failure: COP0_CVM_MEM_CTL = 0x%llx \n " ,
coreid , bist_val ) ;
write_octeon_c0_dcacheerr ( 0 ) ;
}
/**
2021-07-25 09:02:57 -07:00
* octeon_restart - Reboot Octeon
2009-01-08 16:46:40 -08:00
*
* @ command : Command to pass to the bootloader . Currently ignored .
*/
static void octeon_restart ( char * command )
{
/* Disable all watchdogs before soft reset. They don't get cleared */
# ifdef CONFIG_SMP
int cpu ;
for_each_online_cpu ( cpu )
cvmx_write_csr ( CVMX_CIU_WDOGX ( cpu_logical_map ( cpu ) ) , 0 ) ;
# else
cvmx_write_csr ( CVMX_CIU_WDOGX ( cvmx_get_core_num ( ) ) , 0 ) ;
# endif
mb ( ) ;
while ( 1 )
2015-03-05 18:06:11 +03:00
if ( OCTEON_IS_OCTEON3 ( ) )
cvmx_write_csr ( CVMX_RST_SOFT_RST , 1 ) ;
else
cvmx_write_csr ( CVMX_CIU_SOFT_RST , 1 ) ;
2009-01-08 16:46:40 -08:00
}
/**
2021-07-25 09:02:57 -07:00
* octeon_kill_core - Permanently stop a core .
2009-01-08 16:46:40 -08:00
*
* @ arg : Ignored .
*/
static void octeon_kill_core ( void * arg )
{
2013-05-24 16:23:02 +00:00
if ( octeon_is_simulation ( ) )
2009-01-08 16:46:40 -08:00
/* A break instruction causes the simulator stop a core */
2013-05-24 16:23:02 +00:00
asm volatile ( " break " : : : " memory " ) ;
local_irq_disable ( ) ;
/* Disable watchdog on this core. */
cvmx_write_csr ( CVMX_CIU_WDOGX ( cvmx_get_core_num ( ) ) , 0 ) ;
/* Spin in a low power mode. */
while ( true )
asm volatile ( " wait " : : : " memory " ) ;
2009-01-08 16:46:40 -08:00
}
/**
2021-07-25 09:02:57 -07:00
* octeon_halt - Halt the system
2009-01-08 16:46:40 -08:00
*/
static void octeon_halt ( void )
{
smp_call_function ( octeon_kill_core , NULL , 0 ) ;
switch ( octeon_bootinfo - > board_type ) {
case CVMX_BOARD_TYPE_NAO38 :
/* Driving a 1 to GPIO 12 shuts off this board */
cvmx_write_csr ( CVMX_GPIO_BIT_CFGX ( 12 ) , 1 ) ;
cvmx_write_csr ( CVMX_GPIO_TX_SET , 0x1000 ) ;
break ;
default :
octeon_write_lcd ( " PowerOff " ) ;
break ;
}
octeon_kill_core ( NULL ) ;
}
2014-07-22 14:51:08 +03:00
static char __read_mostly octeon_system_type [ 80 ] ;
2016-02-23 00:22:56 +02:00
static void __init init_octeon_system_type ( void )
2014-07-22 14:51:08 +03:00
{
2016-02-23 00:22:55 +02:00
char const * board_type ;
board_type = cvmx_board_type_to_string ( octeon_bootinfo - > board_type ) ;
2016-02-23 00:22:57 +02:00
if ( board_type = = NULL ) {
struct device_node * root ;
int ret ;
root = of_find_node_by_path ( " / " ) ;
ret = of_property_read_string ( root , " model " , & board_type ) ;
of_node_put ( root ) ;
if ( ret )
board_type = " Unsupported Board " ;
}
2016-02-23 00:22:55 +02:00
2014-07-22 14:51:08 +03:00
snprintf ( octeon_system_type , sizeof ( octeon_system_type ) , " %s (%s) " ,
2016-02-23 00:22:55 +02:00
board_type , octeon_model_get_string ( read_c0_prid ( ) ) ) ;
2014-07-22 14:51:08 +03:00
}
2009-01-08 16:46:40 -08:00
/**
2021-07-25 09:02:57 -07:00
* octeon_board_type_string - Return a string representing the system type
2009-01-08 16:46:40 -08:00
*
2021-07-25 09:02:57 -07:00
* Return : system type string
2009-01-08 16:46:40 -08:00
*/
const char * octeon_board_type_string ( void )
{
2014-07-22 14:51:08 +03:00
return octeon_system_type ;
2009-01-08 16:46:40 -08:00
}
const char * get_system_type ( void )
__attribute__ ( ( alias ( " octeon_board_type_string " ) ) ) ;
void octeon_user_io_init ( void )
{
union octeon_cvmemctl cvmmemctl ;
/* Get the current settings for CP0_CVMMEMCTL_REG */
cvmmemctl . u64 = read_c0_cvmmemctl ( ) ;
/* R/W If set, marked write-buffer entries time out the same
2022-08-27 11:46:40 -04:00
* as other entries ; if clear , marked write - buffer entries
2009-01-08 16:46:40 -08:00
* use the maximum timeout . */
cvmmemctl . s . dismarkwblongto = 1 ;
/* R/W If set, a merged store does not clear the write-buffer
* entry timeout state . */
cvmmemctl . s . dismrgclrwbto = 0 ;
/* R/W Two bits that are the MSBs of the resultant CVMSEG LM
* word location for an IOBDMA . The other 8 bits come from the
* SCRADDR field of the IOBDMA . */
cvmmemctl . s . iobdmascrmsb = 0 ;
/* R/W If set, SYNCWS and SYNCS only order marked stores; if
* clear , SYNCWS and SYNCS only order unmarked
* stores . SYNCWSMARKED has no effect when DISSYNCWS is
* set . */
cvmmemctl . s . syncwsmarked = 0 ;
/* R/W If set, SYNCWS acts as SYNCW and SYNCS acts as SYNC. */
cvmmemctl . s . dissyncws = 0 ;
/* R/W If set, no stall happens on write buffer full. */
if ( OCTEON_IS_MODEL ( OCTEON_CN38XX_PASS2 ) )
cvmmemctl . s . diswbfst = 1 ;
else
cvmmemctl . s . diswbfst = 0 ;
/* R/W If set (and SX set), supervisor-level loads/stores can
* use XKPHYS addresses with < 48 > = = 0 */
cvmmemctl . s . xkmemenas = 0 ;
/* R/W If set (and UX set), user-level loads/stores can use
* XKPHYS addresses with VA < 48 > = = 0 */
cvmmemctl . s . xkmemenau = 0 ;
/* R/W If set (and SX set), supervisor-level loads/stores can
* use XKPHYS addresses with VA < 48 > = = 1 */
cvmmemctl . s . xkioenas = 0 ;
/* R/W If set (and UX set), user-level loads/stores can use
* XKPHYS addresses with VA < 48 > = = 1 */
cvmmemctl . s . xkioenau = 0 ;
/* R/W If set, all stores act as SYNCW (NOMERGE must be set
* when this is set ) RW , reset to 0. */
cvmmemctl . s . allsyncw = 0 ;
/* R/W If set, no stores merge, and all stores reach the
* coherent bus in order . */
cvmmemctl . s . nomerge = 0 ;
/* R/W Selects the bit in the counter used for DID time-outs 0
* = 231 , 1 = 230 , 2 = 229 , 3 = 214. Actual time - out is
* between 1 x and 2 x this interval . For example , with
* DIDTTO = 3 , expiration interval is between 16 K and 32 K . */
cvmmemctl . s . didtto = 0 ;
/* R/W If set, the (mem) CSR clock never turns off. */
cvmmemctl . s . csrckalwys = 0 ;
/* R/W If set, mclk never turns off. */
cvmmemctl . s . mclkalwys = 0 ;
/* R/W Selects the bit in the counter used for write buffer
* flush time - outs ( WBFLT + 11 ) is the bit position in an
* internal counter used to determine expiration . The write
* buffer expires between 1 x and 2 x this interval . For
* example , with WBFLT = 0 , a write buffer expires between 2 K
* and 4 K cycles after the write buffer entry is allocated . */
cvmmemctl . s . wbfltime = 0 ;
/* R/W If set, do not put Istream in the L2 cache. */
cvmmemctl . s . istrnol2 = 0 ;
2010-10-07 16:03:53 -07:00
/*
* R / W The write buffer threshold . As per erratum Core - 14752
* for CN63XX , a sc / scd might fail if the write buffer is
* full . Lowering WBTHRESH greatly lowers the chances of the
* write buffer ever being full and triggering the erratum .
*/
if ( OCTEON_IS_MODEL ( OCTEON_CN63XX_PASS1_X ) )
cvmmemctl . s . wbthresh = 4 ;
else
cvmmemctl . s . wbthresh = 10 ;
2009-01-08 16:46:40 -08:00
/* R/W If set, CVMSEG is available for loads/stores in
* kernel / debug mode . */
# if CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE > 0
cvmmemctl . s . cvmsegenak = 1 ;
# else
cvmmemctl . s . cvmsegenak = 0 ;
# endif
/* R/W If set, CVMSEG is available for loads/stores in
* supervisor mode . */
cvmmemctl . s . cvmsegenas = 0 ;
/* R/W If set, CVMSEG is available for loads/stores in user
* mode . */
cvmmemctl . s . cvmsegenau = 0 ;
2010-10-07 16:03:53 -07:00
write_c0_cvmmemctl ( cvmmemctl . u64 ) ;
2009-01-08 16:46:40 -08:00
2015-01-15 16:11:15 +03:00
/* Setup of CVMSEG is done in kernel-entry-init.h */
2009-01-08 16:46:40 -08:00
if ( smp_processor_id ( ) = = 0 )
pr_notice ( " CVMSEG size: %d cache lines (%d bytes) \n " ,
CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE ,
CONFIG_CAVIUM_OCTEON_CVMSEG_SIZE * 128 ) ;
2016-02-09 11:00:10 -08:00
if ( octeon_has_feature ( OCTEON_FEATURE_FAU ) ) {
union cvmx_iob_fau_timeout fau_timeout ;
/* Set a default for the hardware timeouts */
fau_timeout . u64 = 0 ;
fau_timeout . s . tout_val = 0xfff ;
/* Disable tagwait FAU timeout */
fau_timeout . s . tout_enb = 0 ;
cvmx_write_csr ( CVMX_IOB_FAU_TIMEOUT , fau_timeout . u64 ) ;
}
if ( ( ! OCTEON_IS_MODEL ( OCTEON_CN68XX ) & &
! OCTEON_IS_MODEL ( OCTEON_CN7XXX ) ) | |
OCTEON_IS_MODEL ( OCTEON_CN70XX ) ) {
union cvmx_pow_nw_tim nm_tim ;
nm_tim . u64 = 0 ;
/* 4096 cycles */
nm_tim . s . nw_tim = 3 ;
cvmx_write_csr ( CVMX_POW_NW_TIM , nm_tim . u64 ) ;
}
2009-01-08 16:46:40 -08:00
write_octeon_c0_icacheerr ( 0 ) ;
write_c0_derraddr1 ( 0 ) ;
}
/**
2021-07-25 09:02:57 -07:00
* prom_init - Early entry point for arch setup
2009-01-08 16:46:40 -08:00
*/
void __init prom_init ( void )
{
struct cvmx_sysinfo * sysinfo ;
2012-10-25 16:23:31 +02:00
const char * arg ;
char * p ;
2009-01-08 16:46:40 -08:00
int i ;
2015-01-15 16:11:05 +03:00
u64 t ;
2009-01-08 16:46:40 -08:00
int argc ;
/*
* The bootloader passes a pointer to the boot descriptor in
* $ a3 , this is available as fw_arg3 .
*/
octeon_boot_desc_ptr = ( struct octeon_boot_descriptor * ) fw_arg3 ;
octeon_bootinfo =
cvmx_phys_to_ptr ( octeon_boot_desc_ptr - > cvmx_desc_vaddr ) ;
cvmx_bootmem_init ( cvmx_phys_to_ptr ( octeon_bootinfo - > phy_mem_desc_addr ) ) ;
2010-10-07 16:03:49 -07:00
sysinfo = cvmx_sysinfo_get ( ) ;
memset ( sysinfo , 0 , sizeof ( * sysinfo ) ) ;
sysinfo - > system_dram_size = octeon_bootinfo - > dram_size < < 20 ;
2016-02-01 17:46:54 -08:00
sysinfo - > phy_mem_desc_addr = ( u64 ) phys_to_virt ( octeon_bootinfo - > phy_mem_desc_addr ) ;
if ( ( octeon_bootinfo - > major_version > 1 ) | |
( octeon_bootinfo - > major_version = = 1 & &
octeon_bootinfo - > minor_version > = 4 ) )
cvmx_coremask_copy ( & sysinfo - > core_mask ,
& octeon_bootinfo - > ext_core_mask ) ;
else
cvmx_coremask_set64 ( & sysinfo - > core_mask ,
octeon_bootinfo - > core_mask ) ;
/* Some broken u-boot pass garbage in upper bits, clear them out */
if ( ! OCTEON_IS_MODEL ( OCTEON_CN78XX ) )
for ( i = 512 ; i < 1024 ; i + + )
cvmx_coremask_clear_core ( & sysinfo - > core_mask , i ) ;
2010-10-07 16:03:49 -07:00
sysinfo - > exception_base_addr = octeon_bootinfo - > exception_base_addr ;
sysinfo - > cpu_clock_hz = octeon_bootinfo - > eclock_hz ;
sysinfo - > dram_data_rate_hz = octeon_bootinfo - > dclock_hz * 2 ;
sysinfo - > board_type = octeon_bootinfo - > board_type ;
sysinfo - > board_rev_major = octeon_bootinfo - > board_rev_major ;
sysinfo - > board_rev_minor = octeon_bootinfo - > board_rev_minor ;
memcpy ( sysinfo - > mac_addr_base , octeon_bootinfo - > mac_addr_base ,
sizeof ( sysinfo - > mac_addr_base ) ) ;
sysinfo - > mac_addr_count = octeon_bootinfo - > mac_addr_count ;
memcpy ( sysinfo - > board_serial_number ,
octeon_bootinfo - > board_serial_number ,
sizeof ( sysinfo - > board_serial_number ) ) ;
sysinfo - > compact_flash_common_base_addr =
octeon_bootinfo - > compact_flash_common_base_addr ;
sysinfo - > compact_flash_attribute_base_addr =
octeon_bootinfo - > compact_flash_attribute_base_addr ;
sysinfo - > led_display_base_addr = octeon_bootinfo - > led_display_base_addr ;
sysinfo - > dfa_ref_clock_hz = octeon_bootinfo - > dfa_ref_clock_hz ;
sysinfo - > bootloader_config_flags = octeon_bootinfo - > config_flags ;
2015-01-15 16:11:17 +03:00
if ( OCTEON_IS_OCTEON2 ( ) ) {
2010-10-07 16:03:49 -07:00
/* I/O clock runs at a different rate than the CPU. */
union cvmx_mio_rst_boot rst_boot ;
rst_boot . u64 = cvmx_read_csr ( CVMX_MIO_RST_BOOT ) ;
octeon_io_clock_rate = 50000000 * rst_boot . s . pnr_mul ;
2015-01-15 16:11:17 +03:00
} else if ( OCTEON_IS_OCTEON3 ( ) ) {
/* I/O clock runs at a different rate than the CPU. */
union cvmx_rst_boot rst_boot ;
rst_boot . u64 = cvmx_read_csr ( CVMX_RST_BOOT ) ;
octeon_io_clock_rate = 50000000 * rst_boot . s . pnr_mul ;
2010-10-07 16:03:49 -07:00
} else {
octeon_io_clock_rate = sysinfo - > cpu_clock_hz ;
}
2015-01-15 16:11:05 +03:00
t = read_c0_cvmctl ( ) ;
if ( ( t & ( 1ull < < 27 ) ) = = 0 ) {
/*
* Setup the multiplier save / restore code if
* CvmCtl [ NOMUL ] clear .
*/
void * save ;
void * save_end ;
void * restore ;
void * restore_end ;
int save_len ;
int restore_len ;
int save_max = ( char * ) octeon_mult_save_end -
( char * ) octeon_mult_save ;
int restore_max = ( char * ) octeon_mult_restore_end -
( char * ) octeon_mult_restore ;
if ( current_cpu_data . cputype = = CPU_CAVIUM_OCTEON3 ) {
save = octeon_mult_save3 ;
save_end = octeon_mult_save3_end ;
restore = octeon_mult_restore3 ;
restore_end = octeon_mult_restore3_end ;
} else {
save = octeon_mult_save2 ;
save_end = octeon_mult_save2_end ;
restore = octeon_mult_restore2 ;
restore_end = octeon_mult_restore2_end ;
}
save_len = ( char * ) save_end - ( char * ) save ;
restore_len = ( char * ) restore_end - ( char * ) restore ;
if ( ! WARN_ON ( save_len > save_max | |
restore_len > restore_max ) ) {
memcpy ( octeon_mult_save , save , save_len ) ;
memcpy ( octeon_mult_restore , restore , restore_len ) ;
}
}
2009-01-08 16:46:40 -08:00
/*
* Only enable the LED controller if we ' re running on a CN38XX , CN58XX ,
* or CN56XX . The CN30XX and CN31XX don ' t have an LED controller .
*/
if ( ! octeon_is_simulation ( ) & &
octeon_has_feature ( OCTEON_FEATURE_LED_CONTROLLER ) ) {
cvmx_write_csr ( CVMX_LED_EN , 0 ) ;
cvmx_write_csr ( CVMX_LED_PRT , 0 ) ;
cvmx_write_csr ( CVMX_LED_DBG , 0 ) ;
cvmx_write_csr ( CVMX_LED_PRT_FMT , 0 ) ;
cvmx_write_csr ( CVMX_LED_UDD_CNTX ( 0 ) , 32 ) ;
cvmx_write_csr ( CVMX_LED_UDD_CNTX ( 1 ) , 32 ) ;
cvmx_write_csr ( CVMX_LED_UDD_DATX ( 0 ) , 0 ) ;
cvmx_write_csr ( CVMX_LED_UDD_DATX ( 1 ) , 0 ) ;
cvmx_write_csr ( CVMX_LED_EN , 1 ) ;
}
2022-09-06 10:32:39 +02:00
2022-07-25 11:17:40 +02:00
/*
* We need to temporarily allocate all memory in the reserve32
* region . This makes sure the kernel doesn ' t allocate this
* memory when it is getting memory from the
* bootloader . Later , after the memory allocations are
* complete , the reserve32 will be freed .
*
* Allocate memory for RESERVED32 aligned on 2 MB boundary . This
* is in case we later use hugetlb entries with it .
*/
2022-09-06 10:32:39 +02:00
if ( CONFIG_CAVIUM_RESERVE32 ) {
int64_t addr =
cvmx_bootmem_phy_named_block_alloc ( CONFIG_CAVIUM_RESERVE32 < < 20 ,
0 , 0 , 2 < < 20 ,
" CAVIUM_RESERVE32 " , 0 ) ;
if ( addr < 0 )
pr_err ( " Failed to allocate CAVIUM_RESERVE32 memory area \n " ) ;
else
octeon_reserve32_memory = addr ;
}
2009-01-08 16:46:40 -08:00
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2
if ( cvmx_read_csr ( CVMX_L2D_FUS3 ) & ( 3ull < < 34 ) ) {
pr_info ( " Skipping L2 locking due to reduced L2 cache size \n " ) ;
} else {
2013-06-13 01:29:24 +02:00
uint32_t __maybe_unused ebase = read_c0_ebase ( ) & 0x3ffff000 ;
2009-01-08 16:46:40 -08:00
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2_TLB
/* TLB refill */
cvmx_l2c_lock_mem_region ( ebase , 0x100 ) ;
# endif
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2_EXCEPTION
/* General exception */
cvmx_l2c_lock_mem_region ( ebase + 0x180 , 0x80 ) ;
# endif
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2_LOW_LEVEL_INTERRUPT
/* Interrupt handler */
cvmx_l2c_lock_mem_region ( ebase + 0x200 , 0x80 ) ;
# endif
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2_INTERRUPT
cvmx_l2c_lock_mem_region ( __pa_symbol ( handle_int ) , 0x100 ) ;
cvmx_l2c_lock_mem_region ( __pa_symbol ( plat_irq_dispatch ) , 0x80 ) ;
# endif
# ifdef CONFIG_CAVIUM_OCTEON_LOCK_L2_MEMCPY
cvmx_l2c_lock_mem_region ( __pa_symbol ( memcpy ) , 0x480 ) ;
# endif
}
# endif
octeon_check_cpu_bist ( ) ;
octeon_uart = octeon_get_boot_uart ( ) ;
# ifdef CONFIG_SMP
octeon_write_lcd ( " LinuxSMP " ) ;
# else
octeon_write_lcd ( " Linux " ) ;
# endif
2012-08-21 11:45:05 -07:00
octeon_setup_delays ( ) ;
2009-01-08 16:46:40 -08:00
/*
* BIST should always be enabled when doing a soft reset . L2
* Cache locking for instance is not cleared unless BIST is
* enabled . Unfortunately due to a chip errata G - 200 for
2019-09-02 11:55:06 +02:00
* Cn38XX and CN31XX , BIST must be disabled on these parts .
2009-01-08 16:46:40 -08:00
*/
if ( OCTEON_IS_MODEL ( OCTEON_CN38XX_PASS2 ) | |
OCTEON_IS_MODEL ( OCTEON_CN31XX ) )
cvmx_write_csr ( CVMX_CIU_SOFT_BIST , 0 ) ;
else
cvmx_write_csr ( CVMX_CIU_SOFT_BIST , 1 ) ;
/* Default to 64MB in the simulator to speed things up */
if ( octeon_is_simulation ( ) )
2016-05-24 17:09:31 +03:00
max_memory = 64ull < < 20 ;
2009-01-08 16:46:40 -08:00
2012-10-25 16:23:31 +02:00
arg = strstr ( arcs_cmdline , " mem= " ) ;
if ( arg ) {
2016-05-24 17:09:31 +03:00
max_memory = memparse ( arg + 4 , & p ) ;
if ( max_memory = = 0 )
max_memory = 32ull < < 30 ;
2012-10-25 16:23:31 +02:00
if ( * p = = ' @ ' )
2016-05-24 17:09:31 +03:00
reserve_low_mem = memparse ( p + 1 , & p ) ;
2012-10-25 16:23:31 +02:00
}
2009-01-08 16:46:40 -08:00
arcs_cmdline [ 0 ] = 0 ;
argc = octeon_boot_desc_ptr - > argc ;
for ( i = 0 ; i < argc ; i + + ) {
const char * arg =
cvmx_phys_to_ptr ( octeon_boot_desc_ptr - > argv [ i ] ) ;
if ( ( strncmp ( arg , " MEM= " , 4 ) = = 0 ) | |
( strncmp ( arg , " mem= " , 4 ) = = 0 ) ) {
2016-05-24 17:09:31 +03:00
max_memory = memparse ( arg + 4 , & p ) ;
if ( max_memory = = 0 )
max_memory = 32ull < < 30 ;
2012-10-25 16:23:31 +02:00
if ( * p = = ' @ ' )
2016-05-24 17:09:31 +03:00
reserve_low_mem = memparse ( p + 1 , & p ) ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
} else if ( strncmp ( arg , " crashkernel= " , 12 ) = = 0 ) {
crashk_size = memparse ( arg + 12 , & p ) ;
if ( * p = = ' @ ' )
crashk_base = memparse ( p + 1 , & p ) ;
strcat ( arcs_cmdline , " " ) ;
strcat ( arcs_cmdline , arg ) ;
/*
* To do : switch parsing to new style , something like :
* parse_crashkernel ( arg , sysinfo - > system_dram_size ,
2013-01-22 12:59:30 +01:00
* & crashk_size , & crashk_base ) ;
2012-10-25 16:23:31 +02:00
*/
2009-01-08 16:46:40 -08:00
# endif
} else if ( strlen ( arcs_cmdline ) + strlen ( arg ) + 1 <
sizeof ( arcs_cmdline ) - 1 ) {
strcat ( arcs_cmdline , " " ) ;
strcat ( arcs_cmdline , arg ) ;
}
}
if ( strstr ( arcs_cmdline , " console= " ) = = NULL ) {
if ( octeon_uart = = 1 )
strcat ( arcs_cmdline , " console=ttyS1,115200 " ) ;
else
strcat ( arcs_cmdline , " console=ttyS0,115200 " ) ;
}
mips_hpt_frequency = octeon_get_clock_rate ( ) ;
octeon_init_cvmcount ( ) ;
_machine_restart = octeon_restart ;
_machine_halt = octeon_halt ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
_machine_kexec_shutdown = octeon_shutdown ;
_machine_crash_shutdown = octeon_crash_shutdown ;
_machine_kexec_prepare = octeon_kexec_prepare ;
2016-10-11 13:54:26 -07:00
# ifdef CONFIG_SMP
_crash_smp_send_stop = octeon_crash_smp_send_stop ;
# endif
2012-10-25 16:23:31 +02:00
# endif
2009-01-08 16:46:40 -08:00
octeon_user_io_init ( ) ;
2016-02-09 11:00:12 -08:00
octeon_setup_smp ( ) ;
2009-01-08 16:46:40 -08:00
}
2010-08-04 14:53:57 -07:00
/* Exclude a single page from the regions obtained in plat_mem_setup. */
2012-10-25 16:23:31 +02:00
# ifndef CONFIG_CRASH_DUMP
2010-08-04 14:53:57 -07:00
static __init void memory_exclude_page ( u64 addr , u64 * mem , u64 * size )
{
if ( addr > * mem & & addr < * mem + * size ) {
u64 inc = addr - * mem ;
2020-10-09 14:14:46 +02:00
memblock_add ( * mem , inc ) ;
2010-08-04 14:53:57 -07:00
* mem + = inc ;
* size - = inc ;
}
if ( addr = = * mem & & * size > PAGE_SIZE ) {
* mem + = PAGE_SIZE ;
* size - = PAGE_SIZE ;
}
}
2012-10-25 16:23:31 +02:00
# endif /* CONFIG_CRASH_DUMP */
2010-08-04 14:53:57 -07:00
2016-11-22 13:44:10 -06:00
void __init fw_init_cmdline ( void )
{
int i ;
octeon_boot_desc_ptr = ( struct octeon_boot_descriptor * ) fw_arg3 ;
for ( i = 0 ; i < octeon_boot_desc_ptr - > argc ; i + + ) {
const char * arg =
cvmx_phys_to_ptr ( octeon_boot_desc_ptr - > argv [ i ] ) ;
if ( strlen ( arcs_cmdline ) + strlen ( arg ) + 1 <
sizeof ( arcs_cmdline ) - 1 ) {
strcat ( arcs_cmdline , " " ) ;
strcat ( arcs_cmdline , arg ) ;
}
}
}
2016-11-22 13:44:20 -06:00
void __init * plat_get_fdt ( void )
{
octeon_bootinfo =
cvmx_phys_to_ptr ( octeon_boot_desc_ptr - > cvmx_desc_vaddr ) ;
return phys_to_virt ( octeon_bootinfo - > fdt_addr ) ;
}
2009-01-08 16:46:40 -08:00
void __init plat_mem_setup ( void )
{
uint64_t mem_alloc_size ;
uint64_t total ;
2012-10-25 16:23:31 +02:00
uint64_t crashk_end ;
# ifndef CONFIG_CRASH_DUMP
2009-01-08 16:46:40 -08:00
int64_t memory ;
2012-10-25 16:23:31 +02:00
# endif
2009-01-08 16:46:40 -08:00
total = 0 ;
2012-10-25 16:23:31 +02:00
crashk_end = 0 ;
2009-01-08 16:46:40 -08:00
/*
* The Mips memory init uses the first memory location for
* some memory vectors . When SPARSEMEM is in use , it doesn ' t
* verify that the size is big enough for the final
* vectors . Making the smallest chuck 4 MB seems to be enough
2011-03-30 22:57:33 -03:00
* to consistently work .
2009-01-08 16:46:40 -08:00
*/
mem_alloc_size = 4 < < 20 ;
2016-05-24 17:09:31 +03:00
if ( mem_alloc_size > max_memory )
mem_alloc_size = max_memory ;
2009-01-08 16:46:40 -08:00
2012-10-25 16:23:31 +02:00
/* Crashkernel ignores bootmem list. It relies on mem=X@Y option */
# ifdef CONFIG_CRASH_DUMP
2020-10-09 14:14:46 +02:00
memblock_add ( reserve_low_mem , max_memory ) ;
2016-05-24 17:09:31 +03:00
total + = max_memory ;
2012-10-25 16:23:31 +02:00
# else
# ifdef CONFIG_KEXEC
if ( crashk_size > 0 ) {
2020-10-09 14:14:46 +02:00
memblock_add ( crashk_base , crashk_size ) ;
2012-10-25 16:23:31 +02:00
crashk_end = crashk_base + crashk_size ;
}
# endif
2009-01-08 16:46:40 -08:00
/*
2020-10-09 14:14:46 +02:00
* When allocating memory , we want incrementing addresses ,
* which is handled by memblock
2009-01-08 16:46:40 -08:00
*/
cvmx_bootmem_lock ( ) ;
2019-08-19 22:23:06 +08:00
while ( total < max_memory ) {
2009-01-08 16:46:40 -08:00
memory = cvmx_bootmem_phy_alloc ( mem_alloc_size ,
2015-09-30 17:40:42 +13:00
__pa_symbol ( & _end ) , - 1 ,
2009-01-08 16:46:40 -08:00
0x100000 ,
CVMX_BOOTMEM_FLAG_NO_LOCKING ) ;
if ( memory > = 0 ) {
2010-08-04 14:53:57 -07:00
u64 size = mem_alloc_size ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
uint64_t end ;
# endif
2010-08-04 14:53:57 -07:00
/*
* exclude a page at the beginning and end of
* the 256 MB PCIe ' hole ' so the kernel will not
* try to allocate multi - page buffers that
* span the discontinuity .
*/
memory_exclude_page ( CVMX_PCIE_BAR1_PHYS_BASE ,
& memory , & size ) ;
memory_exclude_page ( CVMX_PCIE_BAR1_PHYS_BASE +
CVMX_PCIE_BAR1_PHYS_SIZE ,
& memory , & size ) ;
2012-10-25 16:23:31 +02:00
# ifdef CONFIG_KEXEC
end = memory + mem_alloc_size ;
2010-08-04 14:53:57 -07:00
2009-01-08 16:46:40 -08:00
/*
2012-10-25 16:23:31 +02:00
* This function automatically merges address regions
* next to each other if they are received in
* incrementing order
2009-01-08 16:46:40 -08:00
*/
2012-10-25 16:23:31 +02:00
if ( memory < crashk_base & & end > crashk_end ) {
/* region is fully in */
2020-10-09 14:14:46 +02:00
memblock_add ( memory , crashk_base - memory ) ;
2012-10-25 16:23:31 +02:00
total + = crashk_base - memory ;
2020-10-09 14:14:46 +02:00
memblock_add ( crashk_end , end - crashk_end ) ;
2012-10-25 16:23:31 +02:00
total + = end - crashk_end ;
continue ;
}
if ( memory > = crashk_base & & end < = crashk_end )
/*
* Entire memory region is within the new
* kernel ' s memory , ignore it .
*/
continue ;
if ( memory > crashk_base & & memory < crashk_end & &
end > crashk_end ) {
/*
* Overlap with the beginning of the region ,
* reserve the beginning .
*/
mem_alloc_size - = crashk_end - memory ;
memory = crashk_end ;
} else if ( memory < crashk_base & & end > crashk_base & &
end < crashk_end )
/*
* Overlap with the beginning of the region ,
* chop of end .
*/
mem_alloc_size - = end - crashk_base ;
# endif
2020-10-09 14:14:46 +02:00
memblock_add ( memory , mem_alloc_size ) ;
2009-01-08 16:46:40 -08:00
total + = mem_alloc_size ;
2012-10-25 16:23:31 +02:00
/* Recovering mem_alloc_size */
mem_alloc_size = 4 < < 20 ;
2009-01-08 16:46:40 -08:00
} else {
break ;
}
}
cvmx_bootmem_unlock ( ) ;
2012-10-25 16:23:31 +02:00
# endif /* CONFIG_CRASH_DUMP */
2009-01-08 16:46:40 -08:00
2022-07-25 11:17:40 +02:00
/*
* Now that we ' ve allocated the kernel memory it is safe to
* free the reserved region . We free it here so that builtin
* drivers can use the memory .
*/
if ( octeon_reserve32_memory )
cvmx_bootmem_free_named ( " CAVIUM_RESERVE32 " ) ;
2009-01-08 16:46:40 -08:00
if ( total = = 0 )
panic ( " Unable to allocate memory from "
2013-09-18 16:05:26 +02:00
" cvmx_bootmem_phy_alloc " ) ;
2009-01-08 16:46:40 -08:00
}
2010-07-23 18:41:44 -07:00
/*
2013-01-22 12:59:30 +01:00
* Emit one character to the boot UART . Exported for use by the
2010-07-23 18:41:44 -07:00
* watchdog timer .
*/
2018-07-13 17:51:56 +02:00
void prom_putchar ( char c )
2009-01-08 16:46:40 -08:00
{
uint64_t lsrval ;
/* Spin until there is room */
do {
lsrval = cvmx_read_csr ( CVMX_MIO_UARTX_LSR ( octeon_uart ) ) ;
} while ( ( lsrval & 0x20 ) = = 0 ) ;
/* Write the byte */
2010-05-19 14:16:32 -07:00
cvmx_write_csr ( CVMX_MIO_UARTX_THR ( octeon_uart ) , c & 0xffull ) ;
2009-01-08 16:46:40 -08:00
}
2010-07-23 18:41:44 -07:00
EXPORT_SYMBOL ( prom_putchar ) ;
2009-01-08 16:46:40 -08:00
2015-02-26 01:31:04 +02:00
void __init prom_free_prom_memory ( void )
2009-01-08 16:46:40 -08:00
{
2020-08-24 18:32:53 +02:00
if ( OCTEON_IS_MODEL ( OCTEON_CN6XXX ) ) {
2010-10-07 16:03:53 -07:00
/* Check for presence of Core-14449 fix. */
u32 insn ;
u32 * foo ;
foo = & insn ;
asm volatile ( " # before " : : : " memory " ) ;
prefetch ( foo ) ;
asm volatile (
" .set push \n \t "
" .set noreorder \n \t "
" bal 1f \n \t "
" nop \n "
" 1: \t lw %0,-12($31) \n \t "
" .set pop \n \t "
: " =r " ( insn ) : : " $31 " , " memory " ) ;
if ( ( insn > > 26 ) ! = 0x33 )
2011-11-17 15:07:31 +00:00
panic ( " No PREF instruction at Core-14449 probe point. " ) ;
2010-10-07 16:03:53 -07:00
if ( ( ( insn > > 16 ) & 0x1f ) ! = 28 )
2015-01-15 16:11:13 +03:00
panic ( " OCTEON II DCache prefetch workaround not in place (%04x). \n "
" Please build kernel with proper options (CONFIG_CAVIUM_CN63XXP1). " ,
insn ) ;
2010-10-07 16:03:53 -07:00
}
2009-01-08 16:46:40 -08:00
}
2012-07-05 18:12:38 +02:00
2016-02-24 00:52:05 +02:00
void __init octeon_fill_mac_addresses ( void ) ;
2012-07-05 18:12:38 +02:00
void __init device_tree_init ( void )
{
2014-04-02 17:45:06 -05:00
const void * fdt ;
2012-07-05 18:12:38 +02:00
bool do_prune ;
2016-02-24 00:52:07 +02:00
bool fill_mac ;
2012-07-05 18:12:38 +02:00
2021-01-19 19:15:31 +08:00
# ifdef CONFIG_MIPS_ELF_APPENDED_DTB
if ( ! fdt_check_header ( & __appended_dtb ) ) {
fdt = & __appended_dtb ;
2015-09-11 17:46:15 +03:00
do_prune = false ;
2016-02-24 00:52:07 +02:00
fill_mac = true ;
2015-09-11 17:46:15 +03:00
pr_info ( " Using appended Device Tree. \n " ) ;
2021-01-19 19:15:31 +08:00
} else
# endif
if ( octeon_bootinfo - > minor_version > = 3 & & octeon_bootinfo - > fdt_addr ) {
2012-07-05 18:12:38 +02:00
fdt = phys_to_virt ( octeon_bootinfo - > fdt_addr ) ;
if ( fdt_check_header ( fdt ) )
panic ( " Corrupt Device Tree passed to kernel. " ) ;
do_prune = false ;
2016-02-24 00:52:07 +02:00
fill_mac = false ;
2015-09-11 17:46:15 +03:00
pr_info ( " Using passed Device Tree. \n " ) ;
2012-07-05 18:12:38 +02:00
} else if ( OCTEON_IS_MODEL ( OCTEON_CN68XX ) ) {
2014-04-02 17:45:06 -05:00
fdt = & __dtb_octeon_68xx_begin ;
2012-07-05 18:12:38 +02:00
do_prune = true ;
2016-02-24 00:52:07 +02:00
fill_mac = true ;
2012-07-05 18:12:38 +02:00
} else {
2014-04-02 17:45:06 -05:00
fdt = & __dtb_octeon_3xxx_begin ;
2012-07-05 18:12:38 +02:00
do_prune = true ;
2016-02-24 00:52:07 +02:00
fill_mac = true ;
2012-07-05 18:12:38 +02:00
}
2014-04-02 17:45:06 -05:00
initial_boot_params = ( void * ) fdt ;
2012-07-05 18:12:38 +02:00
if ( do_prune ) {
octeon_prune_device_tree ( ) ;
pr_info ( " Using internal Device Tree. \n " ) ;
}
2016-02-24 00:52:07 +02:00
if ( fill_mac )
octeon_fill_mac_addresses ( ) ;
2014-04-02 17:45:06 -05:00
unflatten_and_copy_device_tree ( ) ;
2016-02-23 00:22:56 +02:00
init_octeon_system_type ( ) ;
2012-07-05 18:12:38 +02:00
}
2012-10-17 00:39:09 +02:00
2012-11-15 13:58:59 -08:00
static int __initdata disable_octeon_edac_p ;
static int __init disable_octeon_edac ( char * str )
{
disable_octeon_edac_p = 1 ;
return 0 ;
}
early_param ( " disable_octeon_edac " , disable_octeon_edac ) ;
2012-10-17 00:39:09 +02:00
static char * edac_device_names [ ] = {
2012-11-15 13:58:59 -08:00
" octeon_l2c_edac " ,
" octeon_pc_edac " ,
2012-10-17 00:39:09 +02:00
} ;
static int __init edac_devinit ( void )
{
struct platform_device * dev ;
int i , err = 0 ;
2012-11-15 13:58:59 -08:00
int num_lmc ;
2012-10-17 00:39:09 +02:00
char * name ;
2012-11-15 13:58:59 -08:00
if ( disable_octeon_edac_p )
return 0 ;
2012-10-17 00:39:09 +02:00
for ( i = 0 ; i < ARRAY_SIZE ( edac_device_names ) ; i + + ) {
name = edac_device_names [ i ] ;
dev = platform_device_register_simple ( name , - 1 , NULL , 0 ) ;
if ( IS_ERR ( dev ) ) {
2014-11-05 22:26:48 +09:00
pr_err ( " Registration of %s failed! \n " , name ) ;
2012-10-17 00:39:09 +02:00
err = PTR_ERR ( dev ) ;
}
}
2012-11-15 13:58:59 -08:00
num_lmc = OCTEON_IS_MODEL ( OCTEON_CN68XX ) ? 4 :
( OCTEON_IS_MODEL ( OCTEON_CN56XX ) ? 2 : 1 ) ;
for ( i = 0 ; i < num_lmc ; i + + ) {
dev = platform_device_register_simple ( " octeon_lmc_edac " ,
i , NULL , 0 ) ;
if ( IS_ERR ( dev ) ) {
2014-11-05 22:26:48 +09:00
pr_err ( " Registration of octeon_lmc_edac %d failed! \n " , i ) ;
2012-11-15 13:58:59 -08:00
err = PTR_ERR ( dev ) ;
}
}
2012-10-17 00:39:09 +02:00
return err ;
}
device_initcall ( edac_devinit ) ;
2013-07-25 20:26:48 +03:00
static void __initdata * octeon_dummy_iospace ;
static int __init octeon_no_pci_init ( void )
{
/*
* Initially assume there is no PCI . The PCI / PCIe platform code will
* later re - initialize these to correct values if they are present .
*/
octeon_dummy_iospace = vzalloc ( IO_SPACE_LIMIT ) ;
set_io_port_base ( ( unsigned long ) octeon_dummy_iospace ) ;
ioport_resource . start = MAX_RESOURCE ;
ioport_resource . end = 0 ;
return 0 ;
}
core_initcall ( octeon_no_pci_init ) ;
static int __init octeon_no_pci_release ( void )
{
/*
* Release the allocated memory if a real IO space is there .
*/
if ( ( unsigned long ) octeon_dummy_iospace ! = mips_io_port_base )
vfree ( octeon_dummy_iospace ) ;
return 0 ;
}
late_initcall ( octeon_no_pci_release ) ;