mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 3477e71d53
			
		
	
	
		3477e71d53
		
	
	
	
	
		
			
			SPE exception handlers are now defined for 32-bit e500mc cores even though SPE unit is not present and CONFIG_SPE is undefined. Restrict SPE exception handlers to e200/e500 cores adding CONFIG_SPE_POSSIBLE and consequently guard __stup_ivors and __setup_cpu functions. Signed-off-by: Mihai Caraman <mihai.caraman@freescale.com> Acked-by: Scott Wood <scottwood@freescale.com> Signed-off-by: Alexander Graf <agraf@suse.de>
		
			
				
	
	
		
			236 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
			
		
		
	
	
			236 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			ArmAsm
		
	
	
	
	
	
| /*
 | |
|  * This file contains low level CPU setup functions.
 | |
|  * Kumar Gala <galak@kernel.crashing.org>
 | |
|  * Copyright 2009 Freescale Semiconductor, Inc.
 | |
|  *
 | |
|  * Based on cpu_setup_6xx code 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.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include <asm/processor.h>
 | |
| #include <asm/cputable.h>
 | |
| #include <asm/ppc_asm.h>
 | |
| #include <asm/mmu-book3e.h>
 | |
| #include <asm/asm-offsets.h>
 | |
| 
 | |
| _GLOBAL(__e500_icache_setup)
 | |
| 	mfspr	r0, SPRN_L1CSR1
 | |
| 	andi.	r3, r0, L1CSR1_ICE
 | |
| 	bnelr				/* Already enabled */
 | |
| 	oris	r0, r0, L1CSR1_CPE@h
 | |
| 	ori	r0, r0, (L1CSR1_ICFI | L1CSR1_ICLFR |  L1CSR1_ICE)
 | |
| 	mtspr	SPRN_L1CSR1, r0		/* Enable I-Cache */
 | |
| 	isync
 | |
| 	blr
 | |
| 
 | |
| _GLOBAL(__e500_dcache_setup)
 | |
| 	mfspr	r0, SPRN_L1CSR0
 | |
| 	andi.	r3, r0, L1CSR0_DCE
 | |
| 	bnelr				/* Already enabled */
 | |
| 	msync
 | |
| 	isync
 | |
| 	li	r0, 0
 | |
| 	mtspr	SPRN_L1CSR0, r0		/* Disable */
 | |
| 	msync
 | |
| 	isync
 | |
| 	li	r0, (L1CSR0_DCFI | L1CSR0_CLFC)
 | |
| 	mtspr	SPRN_L1CSR0, r0		/* Invalidate */
 | |
| 	isync
 | |
| 1:	mfspr	r0, SPRN_L1CSR0
 | |
| 	andi.	r3, r0, L1CSR0_CLFC
 | |
| 	bne+	1b			/* Wait for lock bits reset */
 | |
| 	oris	r0, r0, L1CSR0_CPE@h
 | |
| 	ori	r0, r0, L1CSR0_DCE
 | |
| 	msync
 | |
| 	isync
 | |
| 	mtspr	SPRN_L1CSR0, r0		/* Enable */
 | |
| 	isync
 | |
| 	blr
 | |
| 
 | |
| /*
 | |
|  * FIXME - we haven't yet done testing to determine a reasonable default
 | |
|  * value for PW20_WAIT_IDLE_BIT.
 | |
|  */
 | |
| #define PW20_WAIT_IDLE_BIT		50 /* 1ms, TB frequency is 41.66MHZ */
 | |
| _GLOBAL(setup_pw20_idle)
 | |
| 	mfspr	r3, SPRN_PWRMGTCR0
 | |
| 
 | |
| 	/* Set PW20_WAIT bit, enable pw20 state*/
 | |
| 	ori	r3, r3, PWRMGTCR0_PW20_WAIT
 | |
| 	li	r11, PW20_WAIT_IDLE_BIT
 | |
| 
 | |
| 	/* Set Automatic PW20 Core Idle Count */
 | |
| 	rlwimi	r3, r11, PWRMGTCR0_PW20_ENT_SHIFT, PWRMGTCR0_PW20_ENT
 | |
| 
 | |
| 	mtspr	SPRN_PWRMGTCR0, r3
 | |
| 
 | |
| 	blr
 | |
| 
 | |
| /*
 | |
|  * FIXME - we haven't yet done testing to determine a reasonable default
 | |
|  * value for AV_WAIT_IDLE_BIT.
 | |
|  */
 | |
| #define AV_WAIT_IDLE_BIT		50 /* 1ms, TB frequency is 41.66MHZ */
 | |
| _GLOBAL(setup_altivec_idle)
 | |
| 	mfspr	r3, SPRN_PWRMGTCR0
 | |
| 
 | |
| 	/* Enable Altivec Idle */
 | |
| 	oris	r3, r3, PWRMGTCR0_AV_IDLE_PD_EN@h
 | |
| 	li	r11, AV_WAIT_IDLE_BIT
 | |
| 
 | |
| 	/* Set Automatic AltiVec Idle Count */
 | |
| 	rlwimi	r3, r11, PWRMGTCR0_AV_IDLE_CNT_SHIFT, PWRMGTCR0_AV_IDLE_CNT
 | |
| 
 | |
| 	mtspr	SPRN_PWRMGTCR0, r3
 | |
| 
 | |
| 	blr
 | |
| 
 | |
| #ifdef CONFIG_PPC_E500MC
 | |
| _GLOBAL(__setup_cpu_e6500)
 | |
| 	mflr	r6
 | |
| #ifdef CONFIG_PPC64
 | |
| 	bl	setup_altivec_ivors
 | |
| 	/* Touch IVOR42 only if the CPU supports E.HV category */
 | |
| 	mfspr	r10,SPRN_MMUCFG
 | |
| 	rlwinm.	r10,r10,0,MMUCFG_LPIDSIZE
 | |
| 	beq	1f
 | |
| 	bl	setup_lrat_ivor
 | |
| 1:
 | |
| #endif
 | |
| 	bl	setup_pw20_idle
 | |
| 	bl	setup_altivec_idle
 | |
| 	bl	__setup_cpu_e5500
 | |
| 	mtlr	r6
 | |
| 	blr
 | |
| #endif /* CONFIG_PPC_E500MC */
 | |
| 
 | |
| #ifdef CONFIG_PPC32
 | |
| #ifdef CONFIG_E200
 | |
| _GLOBAL(__setup_cpu_e200)
 | |
| 	/* enable dedicated debug exception handling resources (Debug APU) */
 | |
| 	mfspr	r3,SPRN_HID0
 | |
| 	ori	r3,r3,HID0_DAPUEN@l
 | |
| 	mtspr	SPRN_HID0,r3
 | |
| 	b	__setup_e200_ivors
 | |
| #endif /* CONFIG_E200 */
 | |
| 
 | |
| #ifdef CONFIG_E500
 | |
| #ifndef CONFIG_PPC_E500MC
 | |
| _GLOBAL(__setup_cpu_e500v1)
 | |
| _GLOBAL(__setup_cpu_e500v2)
 | |
| 	mflr	r4
 | |
| 	bl	__e500_icache_setup
 | |
| 	bl	__e500_dcache_setup
 | |
| 	bl	__setup_e500_ivors
 | |
| #if defined(CONFIG_FSL_RIO) || defined(CONFIG_FSL_PCI)
 | |
| 	/* Ensure that RFXE is set */
 | |
| 	mfspr	r3,SPRN_HID1
 | |
| 	oris	r3,r3,HID1_RFXE@h
 | |
| 	mtspr	SPRN_HID1,r3
 | |
| #endif
 | |
| 	mtlr	r4
 | |
| 	blr
 | |
| #else /* CONFIG_PPC_E500MC */
 | |
| _GLOBAL(__setup_cpu_e500mc)
 | |
| _GLOBAL(__setup_cpu_e5500)
 | |
| 	mflr	r5
 | |
| 	bl	__e500_icache_setup
 | |
| 	bl	__e500_dcache_setup
 | |
| 	bl	__setup_e500mc_ivors
 | |
| 	/*
 | |
| 	 * We only want to touch IVOR38-41 if we're running on hardware
 | |
| 	 * that supports category E.HV.  The architectural way to determine
 | |
| 	 * this is MMUCFG[LPIDSIZE].
 | |
| 	 */
 | |
| 	mfspr	r3, SPRN_MMUCFG
 | |
| 	rlwinm.	r3, r3, 0, MMUCFG_LPIDSIZE
 | |
| 	beq	1f
 | |
| 	bl	__setup_ehv_ivors
 | |
| 	b	2f
 | |
| 1:
 | |
| 	lwz	r3, CPU_SPEC_FEATURES(r4)
 | |
| 	/* We need this check as cpu_setup is also called for
 | |
| 	 * the secondary cores. So, if we have already cleared
 | |
| 	 * the feature on the primary core, avoid doing it on the
 | |
| 	 * secondary core.
 | |
| 	 */
 | |
| 	andis.	r6, r3, CPU_FTR_EMB_HV@h
 | |
| 	beq	2f
 | |
| 	rlwinm	r3, r3, 0, ~CPU_FTR_EMB_HV
 | |
| 	stw	r3, CPU_SPEC_FEATURES(r4)
 | |
| 2:
 | |
| 	mtlr	r5
 | |
| 	blr
 | |
| #endif /* CONFIG_PPC_E500MC */
 | |
| #endif /* CONFIG_E500 */
 | |
| #endif /* CONFIG_PPC32 */
 | |
| 
 | |
| #ifdef CONFIG_PPC_BOOK3E_64
 | |
| _GLOBAL(__restore_cpu_e6500)
 | |
| 	mflr	r5
 | |
| 	bl	setup_altivec_ivors
 | |
| 	/* Touch IVOR42 only if the CPU supports E.HV category */
 | |
| 	mfspr	r10,SPRN_MMUCFG
 | |
| 	rlwinm.	r10,r10,0,MMUCFG_LPIDSIZE
 | |
| 	beq	1f
 | |
| 	bl	setup_lrat_ivor
 | |
| 1:
 | |
| 	bl	setup_pw20_idle
 | |
| 	bl	setup_altivec_idle
 | |
| 	bl	__restore_cpu_e5500
 | |
| 	mtlr	r5
 | |
| 	blr
 | |
| 
 | |
| _GLOBAL(__restore_cpu_e5500)
 | |
| 	mflr	r4
 | |
| 	bl	__e500_icache_setup
 | |
| 	bl	__e500_dcache_setup
 | |
| 	bl	__setup_base_ivors
 | |
| 	bl	setup_perfmon_ivor
 | |
| 	bl	setup_doorbell_ivors
 | |
| 	/*
 | |
| 	 * We only want to touch IVOR38-41 if we're running on hardware
 | |
| 	 * that supports category E.HV.  The architectural way to determine
 | |
| 	 * this is MMUCFG[LPIDSIZE].
 | |
| 	 */
 | |
| 	mfspr	r10,SPRN_MMUCFG
 | |
| 	rlwinm.	r10,r10,0,MMUCFG_LPIDSIZE
 | |
| 	beq	1f
 | |
| 	bl	setup_ehv_ivors
 | |
| 1:
 | |
| 	mtlr	r4
 | |
| 	blr
 | |
| 
 | |
| _GLOBAL(__setup_cpu_e5500)
 | |
| 	mflr	r5
 | |
| 	bl	__e500_icache_setup
 | |
| 	bl	__e500_dcache_setup
 | |
| 	bl	__setup_base_ivors
 | |
| 	bl	setup_perfmon_ivor
 | |
| 	bl	setup_doorbell_ivors
 | |
| 	/*
 | |
| 	 * We only want to touch IVOR38-41 if we're running on hardware
 | |
| 	 * that supports category E.HV.  The architectural way to determine
 | |
| 	 * this is MMUCFG[LPIDSIZE].
 | |
| 	 */
 | |
| 	mfspr	r10,SPRN_MMUCFG
 | |
| 	rlwinm.	r10,r10,0,MMUCFG_LPIDSIZE
 | |
| 	beq	1f
 | |
| 	bl	setup_ehv_ivors
 | |
| 	b	2f
 | |
| 1:
 | |
| 	ld	r10,CPU_SPEC_FEATURES(r4)
 | |
| 	LOAD_REG_IMMEDIATE(r9,CPU_FTR_EMB_HV)
 | |
| 	andc	r10,r10,r9
 | |
| 	std	r10,CPU_SPEC_FEATURES(r4)
 | |
| 2:
 | |
| 	mtlr	r5
 | |
| 	blr
 | |
| #endif
 |