2009-04-03 05:49:22 +04:00
/*
2005-04-17 02:20:36 +04:00
* Initial setup - routines for HP 9000 based hardware .
*
* Copyright ( C ) 1991 , 1992 , 1995 Linus Torvalds
2008-12-31 06:12:10 +03:00
* Modifications for PA - RISC ( C ) 1999 - 2008 Helge Deller < deller @ gmx . de >
2005-04-17 02:20:36 +04:00
* Modifications copyright 1999 SuSE GmbH ( Philipp Rumpf )
* Modifications copyright 2000 Martin K . Petersen < mkp @ mkp . net >
* Modifications copyright 2000 Philipp Rumpf < prumpf @ tux . org >
* Modifications copyright 2001 Ryan Bradetich < rbradetich @ uswest . net >
*
* Initial PA - RISC Version : 04 - 23 - 1999 by Helge Deller
*
* 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 , or ( at your option )
* any later version .
*
* 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 . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*
*/
# include <linux/delay.h>
# include <linux/init.h>
# include <linux/mm.h>
# include <linux/module.h>
# include <linux/seq_file.h>
2017-08-10 20:20:47 +03:00
# include <linux/random.h>
2005-04-17 02:20:36 +04:00
# include <linux/slab.h>
# include <linux/cpu.h>
Detach sched.h from mm.h
First thing mm.h does is including sched.h solely for can_do_mlock() inline
function which has "current" dereference inside. By dealing with can_do_mlock()
mm.h can be detached from sched.h which is good. See below, why.
This patch
a) removes unconditional inclusion of sched.h from mm.h
b) makes can_do_mlock() normal function in mm/mlock.c
c) exports can_do_mlock() to not break compilation
d) adds sched.h inclusions back to files that were getting it indirectly.
e) adds less bloated headers to some files (asm/signal.h, jiffies.h) that were
getting them indirectly
Net result is:
a) mm.h users would get less code to open, read, preprocess, parse, ... if
they don't need sched.h
b) sched.h stops being dependency for significant number of files:
on x86_64 allmodconfig touching sched.h results in recompile of 4083 files,
after patch it's only 3744 (-8.3%).
Cross-compile tested on
all arm defconfigs, all mips defconfigs, all powerpc defconfigs,
alpha alpha-up
arm
i386 i386-up i386-defconfig i386-allnoconfig
ia64 ia64-up
m68k
mips
parisc parisc-up
powerpc powerpc-up
s390 s390-up
sparc sparc-up
sparc64 sparc64-up
um-x86_64
x86_64 x86_64-up x86_64-defconfig x86_64-allnoconfig
as well as my two usual configs.
Signed-off-by: Alexey Dobriyan <adobriyan@gmail.com>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2007-05-21 01:22:52 +04:00
# include <asm/param.h>
2005-04-17 02:20:36 +04:00
# include <asm/cache.h>
# include <asm/hardware.h> /* for register_parisc_driver() stuff */
# include <asm/processor.h>
# include <asm/page.h>
# include <asm/pdc.h>
# include <asm/pdcpat.h>
# include <asm/irq.h> /* for struct irq_region */
# include <asm/parisc-device.h>
2006-01-11 04:35:03 +03:00
struct system_cpuinfo_parisc boot_cpu_data __read_mostly ;
2005-04-17 02:20:36 +04:00
EXPORT_SYMBOL ( boot_cpu_data ) ;
2015-12-12 20:22:03 +03:00
# ifdef CONFIG_PA8X00
int _parisc_requires_coherency __read_mostly ;
EXPORT_SYMBOL ( _parisc_requires_coherency ) ;
# endif
2005-04-17 02:20:36 +04:00
2008-12-31 06:12:10 +03:00
DEFINE_PER_CPU ( struct cpuinfo_parisc , cpu_data ) ;
2005-04-17 02:20:36 +04:00
/*
* * PARISC CPU driver - claim " device " and initialize CPU data structures .
* *
* * Consolidate per CPU initialization into ( mostly ) one module .
* * Monarch CPU will initialize boot_cpu_data which shouldn ' t
* * change once the system has booted .
* *
* * The callback * should * do per - instance initialization of
* * everything including the monarch . " Per CPU " init code in
* * setup . c : start_parisc ( ) has migrated here and start_parisc ( )
* * will call register_parisc_driver ( & cpu_driver ) before calling do_inventory ( ) .
* *
* * The goal of consolidating CPU initialization into one place is
2007-05-11 23:42:34 +04:00
* * to make sure all CPUs get initialized the same way .
2005-04-17 02:20:36 +04:00
* * The code path not shared is how PDC hands control of the CPU to the OS .
* * The initialization of OS data structures is the same ( done below ) .
*/
2008-12-31 06:12:10 +03:00
/**
* init_cpu_profiler - enable / setup per cpu profiling hooks .
* @ cpunum : The processor instance .
*
* FIXME : doesn ' t do much yet . . .
*/
2013-06-17 23:43:14 +04:00
static void
2008-12-31 06:12:10 +03:00
init_percpu_prof ( unsigned long cpunum )
{
}
2005-04-17 02:20:36 +04:00
/**
* processor_probe - Determine if processor driver should claim this device .
* @ dev : The device which has been found .
*
* Determine if processor driver should claim this chip ( return 0 ) or not
* ( return 1 ) . If so , initialize the chip and tell other partners in crime
* they have work to do .
*/
2017-08-21 23:09:30 +03:00
static int __init processor_probe ( struct parisc_device * dev )
2005-04-17 02:20:36 +04:00
{
unsigned long txn_addr ;
unsigned long cpuid ;
struct cpuinfo_parisc * p ;
2017-01-08 13:01:11 +03:00
struct pdc_pat_cpu_num cpu_info = { } ;
2005-04-17 02:20:36 +04:00
2007-10-18 11:03:50 +04:00
# ifdef CONFIG_SMP
2009-03-16 06:49:37 +03:00
if ( num_online_cpus ( ) > = nr_cpu_ids ) {
printk ( KERN_INFO " num_online_cpus() >= nr_cpu_ids \n " ) ;
2007-10-18 11:03:50 +04:00
return 1 ;
}
# else
2005-04-17 02:20:36 +04:00
if ( boot_cpu_data . cpu_count > 0 ) {
printk ( KERN_INFO " CONFIG_SMP=n ignoring additional CPUs \n " ) ;
return 1 ;
}
# endif
/* logical CPU ID and update global counter
* May get overwritten by PAT code .
*/
cpuid = boot_cpu_data . cpu_count ;
2005-10-22 06:36:40 +04:00
txn_addr = dev - > hpa . start ; /* for legacy PDC */
2017-01-08 13:01:11 +03:00
cpu_info . cpu_num = cpu_info . cpu_loc = cpuid ;
2005-04-17 02:20:36 +04:00
2007-01-28 16:58:52 +03:00
# ifdef CONFIG_64BIT
2005-04-17 02:20:36 +04:00
if ( is_pdc_pat ( ) ) {
ulong status ;
unsigned long bytecnt ;
2009-06-23 21:10:03 +04:00
pdc_pat_cell_mod_maddr_block_t * pa_pdc_cell ;
2005-04-17 02:20:36 +04:00
2009-06-23 21:10:03 +04:00
pa_pdc_cell = kmalloc ( sizeof ( * pa_pdc_cell ) , GFP_KERNEL ) ;
if ( ! pa_pdc_cell )
panic ( " couldn't allocate memory for PDC_PAT_CELL! " ) ;
2005-04-17 02:20:36 +04:00
status = pdc_pat_cell_module ( & bytecnt , dev - > pcell_loc ,
2009-06-23 21:10:03 +04:00
dev - > mod_index , PA_VIEW , pa_pdc_cell ) ;
2005-04-17 02:20:36 +04:00
BUG_ON ( PDC_OK ! = status ) ;
/* verify it's the same as what do_pat_inventory() found */
2009-06-23 21:10:03 +04:00
BUG_ON ( dev - > mod_info ! = pa_pdc_cell - > mod_info ) ;
BUG_ON ( dev - > pmod_loc ! = pa_pdc_cell - > mod_location ) ;
txn_addr = pa_pdc_cell - > mod [ 0 ] ; /* id_eid for IO sapic */
2005-04-17 02:20:36 +04:00
2009-06-23 21:10:03 +04:00
kfree ( pa_pdc_cell ) ;
2005-04-17 02:20:36 +04:00
2016-11-17 23:27:50 +03:00
/* get the cpu number */
status = pdc_pat_cpu_get_number ( & cpu_info , dev - > hpa . start ) ;
BUG_ON ( PDC_OK ! = status ) ;
pr_info ( " Logical CPU #%lu is physical cpu #%lu at location "
" 0x%lx with hpa %pa \n " ,
cpuid , cpu_info . cpu_num , cpu_info . cpu_loc ,
& dev - > hpa . start ) ;
# undef USE_PAT_CPUID
2005-04-17 02:20:36 +04:00
# ifdef USE_PAT_CPUID
/* We need contiguous numbers for cpuid. Firmware's notion
* of cpuid is for physical CPUs and we just don ' t care yet .
* We ' ll care when we need to query PAT PDC about a CPU * after *
* boot time ( ie shutdown a CPU from an OS perspective ) .
*/
if ( cpu_info . cpu_num > = NR_CPUS ) {
2016-11-17 23:27:50 +03:00
printk ( KERN_WARNING " IGNORING CPU at %pa, "
2005-04-17 02:20:36 +04:00
" cpu_slot_id > NR_CPUS "
" (%ld > %d) \n " ,
2016-11-17 23:27:50 +03:00
& dev - > hpa . start , cpu_info . cpu_num , NR_CPUS ) ;
2005-04-17 02:20:36 +04:00
/* Ignore CPU since it will only crash */
boot_cpu_data . cpu_count - - ;
return 1 ;
} else {
cpuid = cpu_info . cpu_num ;
}
# endif
}
# endif
2008-12-31 06:12:10 +03:00
p = & per_cpu ( cpu_data , cpuid ) ;
2005-04-17 02:20:36 +04:00
boot_cpu_data . cpu_count + + ;
2006-09-02 01:18:39 +04:00
/* initialize counters - CPU 0 gets it_value set in time_init() */
if ( cpuid )
memset ( p , 0 , sizeof ( struct cpuinfo_parisc ) ) ;
2005-04-17 02:20:36 +04:00
p - > loops_per_jiffy = loops_per_jiffy ;
p - > dev = dev ; /* Save IODC data in case we need it */
2005-10-22 06:36:40 +04:00
p - > hpa = dev - > hpa . start ; /* save CPU hpa */
2005-04-17 02:20:36 +04:00
p - > cpuid = cpuid ; /* save CPU id */
p - > txn_addr = txn_addr ; /* save CPU IRQ address */
2017-01-08 13:01:11 +03:00
p - > cpu_num = cpu_info . cpu_num ;
p - > cpu_loc = cpu_info . cpu_loc ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
/*
* * FIXME : review if any other initialization is clobbered
2008-12-31 06:12:10 +03:00
* * for boot_cpu by the above memset ( ) .
2005-04-17 02:20:36 +04:00
*/
2008-12-31 06:12:10 +03:00
init_percpu_prof ( cpuid ) ;
2005-04-17 02:20:36 +04:00
# endif
/*
2007-05-11 23:42:34 +04:00
* * CONFIG_SMP : init_smp_config ( ) will attempt to get CPUs into
2005-04-17 02:20:36 +04:00
* * OS control . RENDEZVOUS is the default state - see mem_set above .
* * p - > state = STATE_RENDEZVOUS ;
*/
#if 0
/* CPU 0 IRQ table is statically allocated/initialized */
if ( cpuid ) {
struct irqaction actions [ ] ;
/*
* * itimer and ipi IRQ handlers are statically initialized in
* * arch / parisc / kernel / irq . c . ie Don ' t need to register them .
*/
actions = kmalloc ( sizeof ( struct irqaction ) * MAX_CPU_IRQ , GFP_ATOMIC ) ;
if ( ! actions ) {
/* not getting it's own table, share with monarch */
actions = cpu_irq_actions [ 0 ] ;
}
cpu_irq_actions [ cpuid ] = actions ;
}
# endif
/*
* Bring this CPU up now ! ( ignore bootstrap cpuid = = 0 )
*/
# ifdef CONFIG_SMP
if ( cpuid ) {
2009-03-16 06:49:38 +03:00
set_cpu_present ( cpuid , true ) ;
2005-04-17 02:20:36 +04:00
cpu_up ( cpuid ) ;
}
# endif
return 0 ;
}
/**
* collect_boot_cpu_data - Fill the boot_cpu_data structure .
*
* This function collects and stores the generic processor information
* in the boot_cpu_data structure .
*/
void __init collect_boot_cpu_data ( void )
{
2017-08-10 20:20:47 +03:00
unsigned long cr16_seed ;
2005-04-17 02:20:36 +04:00
memset ( & boot_cpu_data , 0 , sizeof ( boot_cpu_data ) ) ;
2017-08-10 20:20:47 +03:00
cr16_seed = get_cycles ( ) ;
add_device_randomness ( & cr16_seed , sizeof ( cr16_seed ) ) ;
2005-04-17 02:20:36 +04:00
boot_cpu_data . cpu_hz = 100 * PAGE0 - > mem_10msec ; /* Hz of this PARISC */
/* get CPU-Model Information... */
# define p ((unsigned long *)&boot_cpu_data.pdc.model)
2017-08-10 20:20:47 +03:00
if ( pdc_model_info ( & boot_cpu_data . pdc . model ) = = PDC_OK ) {
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO
" model %08lx %08lx %08lx %08lx %08lx %08lx %08lx %08lx %08lx \n " ,
p [ 0 ] , p [ 1 ] , p [ 2 ] , p [ 3 ] , p [ 4 ] , p [ 5 ] , p [ 6 ] , p [ 7 ] , p [ 8 ] ) ;
2017-08-10 20:20:47 +03:00
add_device_randomness ( & boot_cpu_data . pdc . model ,
sizeof ( boot_cpu_data . pdc . model ) ) ;
}
2005-04-17 02:20:36 +04:00
# undef p
2017-08-10 20:20:47 +03:00
if ( pdc_model_versions ( & boot_cpu_data . pdc . versions , 0 ) = = PDC_OK ) {
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO " vers %08lx \n " ,
boot_cpu_data . pdc . versions ) ;
2017-08-10 20:20:47 +03:00
add_device_randomness ( & boot_cpu_data . pdc . versions ,
sizeof ( boot_cpu_data . pdc . versions ) ) ;
}
if ( pdc_model_cpuid ( & boot_cpu_data . pdc . cpuid ) = = PDC_OK ) {
2005-04-17 02:20:36 +04:00
printk ( KERN_INFO " CPUID vers %ld rev %ld (0x%08lx) \n " ,
( boot_cpu_data . pdc . cpuid > > 5 ) & 127 ,
boot_cpu_data . pdc . cpuid & 31 ,
boot_cpu_data . pdc . cpuid ) ;
2017-08-10 20:20:47 +03:00
add_device_randomness ( & boot_cpu_data . pdc . cpuid ,
sizeof ( boot_cpu_data . pdc . cpuid ) ) ;
}
2005-04-17 02:20:36 +04:00
if ( pdc_model_capabilities ( & boot_cpu_data . pdc . capabilities ) = = PDC_OK )
printk ( KERN_INFO " capabilities 0x%lx \n " ,
boot_cpu_data . pdc . capabilities ) ;
if ( pdc_model_sysmodel ( boot_cpu_data . pdc . sys_model_name ) = = PDC_OK )
printk ( KERN_INFO " model %s \n " ,
boot_cpu_data . pdc . sys_model_name ) ;
boot_cpu_data . hversion = boot_cpu_data . pdc . model . hversion ;
boot_cpu_data . sversion = boot_cpu_data . pdc . model . sversion ;
boot_cpu_data . cpu_type = parisc_get_cpu_type ( boot_cpu_data . hversion ) ;
boot_cpu_data . cpu_name = cpu_name_version [ boot_cpu_data . cpu_type ] [ 0 ] ;
boot_cpu_data . family_name = cpu_name_version [ boot_cpu_data . cpu_type ] [ 1 ] ;
2015-12-12 20:22:03 +03:00
# ifdef CONFIG_PA8X00
_parisc_requires_coherency = ( boot_cpu_data . cpu_type = = mako ) | |
( boot_cpu_data . cpu_type = = mako2 ) ;
# endif
}
2005-04-17 02:20:36 +04:00
/**
* init_per_cpu - Handle individual processor initializations .
* @ cpunum : logical processor number .
*
* This function handles initialization for * every * CPU
* in the system :
*
* o Set " default " CPU width for trap handlers
*
* o Enable FP coprocessor
* REVISIT : this could be done in the " code 22 " trap handler .
* ( frowands idea - that way we know which processes need FP
* registers saved on the interrupt stack . )
* NEWS FLASH : wide kernels need FP coprocessor enabled to handle
* formatted printing of % lx for example ( double divides I think )
*
* o Enable CPU profiling hooks .
*/
2017-09-21 22:22:27 +03:00
int __init init_per_cpu ( int cpunum )
2005-04-17 02:20:36 +04:00
{
int ret ;
struct pdc_coproc_cfg coproc_cfg ;
set_firmware_width ( ) ;
ret = pdc_coproc_cfg ( & coproc_cfg ) ;
if ( ret > = 0 & & coproc_cfg . ccr_functional ) {
mtctl ( coproc_cfg . ccr_functional , 10 ) ; /* 10 == Coprocessor Control Reg */
/* FWIW, FP rev/model is a more accurate way to determine
* * CPU type . CPU rev / model has some ambiguous cases .
*/
2008-12-31 06:12:10 +03:00
per_cpu ( cpu_data , cpunum ) . fp_rev = coproc_cfg . revision ;
per_cpu ( cpu_data , cpunum ) . fp_model = coproc_cfg . model ;
2005-04-17 02:20:36 +04:00
2016-06-03 20:22:31 +03:00
if ( cpunum = = 0 )
printk ( KERN_INFO " FP[%d] enabled: Rev %ld Model %ld \n " ,
cpunum , coproc_cfg . revision , coproc_cfg . model ) ;
2005-04-17 02:20:36 +04:00
/*
* * store status register to stack ( hopefully aligned )
* * and clear the T - bit .
*/
asm volatile ( " fstd %fr0,8(%sp) " ) ;
} else {
printk ( KERN_WARNING " WARNING: No FP CoProcessor?! "
" (coproc_cfg.ccr_functional == 0x%lx, expected 0xc0) \n "
2007-01-28 16:58:52 +03:00
# ifdef CONFIG_64BIT
2005-04-17 02:20:36 +04:00
" Halting Machine - FP required \n "
# endif
, coproc_cfg . ccr_functional ) ;
2007-01-28 16:58:52 +03:00
# ifdef CONFIG_64BIT
2005-04-17 02:20:36 +04:00
mdelay ( 100 ) ; /* previous chars get pushed to console */
panic ( " FP CoProc not reported " ) ;
# endif
}
/* FUTURE: Enable Performance Monitor : ccr bit 0x20 */
init_percpu_prof ( cpunum ) ;
return ret ;
}
/*
2007-05-11 23:42:34 +04:00
* Display CPU info for all CPUs .
2005-04-17 02:20:36 +04:00
*/
int
show_cpuinfo ( struct seq_file * m , void * v )
{
2008-12-31 06:12:10 +03:00
unsigned long cpu ;
2005-04-17 02:20:36 +04:00
2008-12-31 06:12:10 +03:00
for_each_online_cpu ( cpu ) {
const struct cpuinfo_parisc * cpuinfo = & per_cpu ( cpu_data , cpu ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_SMP
2008-12-31 06:12:10 +03:00
if ( 0 = = cpuinfo - > hpa )
2005-04-17 02:20:36 +04:00
continue ;
# endif
2008-12-31 06:12:10 +03:00
seq_printf ( m , " processor \t : %lu \n "
2005-04-17 02:20:36 +04:00
" cpu family \t : PA-RISC %s \n " ,
2008-12-31 06:12:10 +03:00
cpu , boot_cpu_data . family_name ) ;
2005-04-17 02:20:36 +04:00
seq_printf ( m , " cpu \t \t : %s \n " , boot_cpu_data . cpu_name ) ;
/* cpu MHz */
seq_printf ( m , " cpu MHz \t \t : %d.%06d \n " ,
boot_cpu_data . cpu_hz / 1000000 ,
boot_cpu_data . cpu_hz % 1000000 ) ;
2009-01-30 04:03:50 +03:00
seq_printf ( m , " capabilities \t : " ) ;
if ( boot_cpu_data . pdc . capabilities & PDC_MODEL_OS32 )
2013-06-22 01:32:44 +04:00
seq_puts ( m , " os32 " ) ;
2009-01-30 04:03:50 +03:00
if ( boot_cpu_data . pdc . capabilities & PDC_MODEL_OS64 )
2013-06-22 01:32:44 +04:00
seq_puts ( m , " os64 " ) ;
if ( boot_cpu_data . pdc . capabilities & PDC_MODEL_IOPDIR_FDC )
seq_puts ( m , " iopdir_fdc " ) ;
switch ( boot_cpu_data . pdc . capabilities & PDC_MODEL_NVA_MASK ) {
case PDC_MODEL_NVA_SUPPORTED :
seq_puts ( m , " nva_supported " ) ;
break ;
case PDC_MODEL_NVA_SLOW :
seq_puts ( m , " nva_slow " ) ;
break ;
case PDC_MODEL_NVA_UNSUPPORTED :
seq_puts ( m , " needs_equivalent_aliasing " ) ;
break ;
}
seq_printf ( m , " (0x%02lx) \n " , boot_cpu_data . pdc . capabilities ) ;
2009-01-30 04:03:50 +03:00
2005-04-17 02:20:36 +04:00
seq_printf ( m , " model \t \t : %s \n "
" model name \t : %s \n " ,
boot_cpu_data . pdc . sys_model_name ,
2008-12-31 06:12:10 +03:00
cpuinfo - > dev ?
cpuinfo - > dev - > name : " Unknown " ) ;
2005-04-17 02:20:36 +04:00
seq_printf ( m , " hversion \t : 0x%08x \n "
" sversion \t : 0x%08x \n " ,
boot_cpu_data . hversion ,
boot_cpu_data . sversion ) ;
/* print cachesize info */
show_cache_info ( m ) ;
seq_printf ( m , " bogomips \t : %lu.%02lu \n " ,
2008-12-31 06:12:10 +03:00
cpuinfo - > loops_per_jiffy / ( 500000 / HZ ) ,
( cpuinfo - > loops_per_jiffy / ( 5000 / HZ ) ) % 100 ) ;
2005-04-17 02:20:36 +04:00
seq_printf ( m , " software id \t : %ld \n \n " ,
boot_cpu_data . pdc . model . sw_id ) ;
}
return 0 ;
}
2017-08-21 23:09:30 +03:00
static const struct parisc_device_id processor_tbl [ ] __initconst = {
2005-04-17 02:20:36 +04:00
{ HPHW_NPROC , HVERSION_REV_ANY_ID , HVERSION_ANY_ID , SVERSION_ANY_ID } ,
{ 0 , }
} ;
2017-08-21 23:09:30 +03:00
static struct parisc_driver cpu_driver __refdata = {
2005-04-17 02:20:36 +04:00
. name = " CPU " ,
. id_table = processor_tbl ,
. probe = processor_probe
} ;
/**
2007-05-11 23:42:34 +04:00
* processor_init - Processor initialization procedure .
2005-04-17 02:20:36 +04:00
*
* Register this driver .
*/
void __init processor_init ( void )
{
register_parisc_driver ( & cpu_driver ) ;
}