2009-07-02 23:22:36 +08:00
/*
* Based on Ocelot Linux port , which is
* Copyright 2001 MontaVista Software Inc .
* Author : jsun @ mvista . com or jsun @ junsun . net
*
* Copyright 2003 ICT CAS
* Author : Michael Guo < guoyi @ ict . ac . cn >
*
* Copyright ( C ) 2007 Lemote Inc . & Insititute of Computing Technology
* Author : Fuxin Zhang , zhangfx @ lemote . com
*
2010-01-04 17:16:51 +08:00
* Copyright ( C ) 2009 Lemote Inc .
* Author : Wu Zhangjin , wuzhangjin @ gmail . com
2009-07-02 23:22:36 +08:00
*
2013-01-22 12:59:30 +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
2009-07-02 23:22:36 +08:00
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*/
2009-11-17 01:32:59 +08:00
# include <linux/module.h>
2009-07-02 23:22:36 +08:00
# include <asm/bootinfo.h>
2009-07-02 23:23:03 +08:00
# include <loongson.h>
2014-03-21 18:44:02 +08:00
# include <boot_param.h>
2009-07-02 23:23:03 +08:00
2014-03-21 18:44:02 +08:00
u32 cpu_clock_freq ;
2009-11-17 01:32:59 +08:00
EXPORT_SYMBOL ( cpu_clock_freq ) ;
2014-03-21 18:44:02 +08:00
struct efi_memory_map_loongson * loongson_memmap ;
struct loongson_system_configuration loongson_sysconf ;
2009-07-02 23:22:36 +08:00
# define parse_even_earlier(res, option, p) \
do { \
2011-03-29 12:32:55 +02:00
unsigned int tmp __maybe_unused ; \
\
2009-07-02 23:22:36 +08:00
if ( strncmp ( option , ( char * ) p , strlen ( option ) ) = = 0 ) \
2014-03-21 18:44:02 +08:00
tmp = kstrtou32 ( ( char * ) p + strlen ( option " = " ) , 10 , & res ) ; \
2009-07-02 23:22:36 +08:00
} while ( 0 )
void __init prom_init_env ( void )
{
2010-01-04 17:16:48 +08:00
/* pmon passes arguments in 32bit pointers */
unsigned int processor_id ;
2014-03-21 18:44:02 +08:00
# ifndef CONFIG_LEFI_FIRMWARE_INTERFACE
int * _prom_envp ;
2009-07-02 23:22:36 +08:00
long l ;
/* firmware arguments are initialized in head.S */
_prom_envp = ( int * ) fw_arg2 ;
l = ( long ) * _prom_envp ;
while ( l ! = 0 ) {
parse_even_earlier ( cpu_clock_freq , " cpuclock " , l ) ;
parse_even_earlier ( memsize , " memsize " , l ) ;
parse_even_earlier ( highmemsize , " highmemsize " , l ) ;
_prom_envp + + ;
l = ( long ) * _prom_envp ;
}
if ( memsize = = 0 )
memsize = 256 ;
2014-03-21 18:44:02 +08:00
pr_info ( " memsize=%u, highmemsize=%u \n " , memsize , highmemsize ) ;
# else
struct boot_params * boot_p ;
struct loongson_params * loongson_p ;
struct efi_cpuinfo_loongson * ecpu ;
struct irq_source_routing_table * eirq_source ;
/* firmware arguments are initialized in head.S */
boot_p = ( struct boot_params * ) fw_arg2 ;
loongson_p = & ( boot_p - > efi . smbios . lp ) ;
ecpu = ( struct efi_cpuinfo_loongson * )
( ( u64 ) loongson_p + loongson_p - > cpu_offset ) ;
eirq_source = ( struct irq_source_routing_table * )
( ( u64 ) loongson_p + loongson_p - > irq_offset ) ;
loongson_memmap = ( struct efi_memory_map_loongson * )
( ( u64 ) loongson_p + loongson_p - > memory_offset ) ;
cpu_clock_freq = ecpu - > cpu_clock_freq ;
loongson_sysconf . cputype = ecpu - > cputype ;
loongson_sysconf . nr_cpus = ecpu - > nr_cpus ;
if ( ecpu - > nr_cpus > NR_CPUS | | ecpu - > nr_cpus = = 0 )
loongson_sysconf . nr_cpus = NR_CPUS ;
loongson_sysconf . pci_mem_start_addr = eirq_source - > pci_mem_start_addr ;
loongson_sysconf . pci_mem_end_addr = eirq_source - > pci_mem_end_addr ;
loongson_sysconf . pci_io_base = eirq_source - > pci_io_start_addr ;
loongson_sysconf . dma_mask_bits = eirq_source - > dma_mask_bits ;
if ( loongson_sysconf . dma_mask_bits < 32 | |
loongson_sysconf . dma_mask_bits > 64 )
loongson_sysconf . dma_mask_bits = 32 ;
loongson_sysconf . restart_addr = boot_p - > reset_system . ResetWarm ;
loongson_sysconf . poweroff_addr = boot_p - > reset_system . Shutdown ;
loongson_sysconf . suspend_addr = boot_p - > reset_system . DoSuspend ;
loongson_sysconf . ht_control_base = 0x90000EFDFB000000 ;
loongson_sysconf . vgabios_addr = boot_p - > efi . smbios . vga_bios ;
pr_debug ( " Shutdown Addr: %llx, Restart Addr: %llx, VBIOS Addr: %llx \n " ,
loongson_sysconf . poweroff_addr , loongson_sysconf . restart_addr ,
loongson_sysconf . vgabios_addr ) ;
# endif
2010-01-04 17:16:48 +08:00
if ( cpu_clock_freq = = 0 ) {
processor_id = ( & current_cpu_data ) - > processor_id ;
switch ( processor_id & PRID_REV_MASK ) {
case PRID_REV_LOONGSON2E :
cpu_clock_freq = 533080000 ;
break ;
case PRID_REV_LOONGSON2F :
cpu_clock_freq = 797000000 ;
break ;
2014-03-21 18:44:02 +08:00
case PRID_REV_LOONGSON3A :
cpu_clock_freq = 900000000 ;
break ;
2010-01-04 17:16:48 +08:00
default :
cpu_clock_freq = 100000000 ;
break ;
}
}
2014-03-21 18:44:02 +08:00
pr_info ( " CpuClock = %u \n " , cpu_clock_freq ) ;
2009-07-02 23:22:36 +08:00
}