xref: /openbmc/linux/arch/powerpc/kernel/misc_64.S (revision e55174e911637f0093bc9dac137b568e8a4a1f29)
19994a338SPaul Mackerras/*
29994a338SPaul Mackerras * This file contains miscellaneous low-level functions.
39994a338SPaul Mackerras *    Copyright (C) 1995-1996 Gary Thomas (gdt@linuxppc.org)
49994a338SPaul Mackerras *
59994a338SPaul Mackerras * Largely rewritten by Cort Dougan (cort@cs.nmt.edu)
69994a338SPaul Mackerras * and Paul Mackerras.
79994a338SPaul Mackerras * Adapted for iSeries by Mike Corrigan (mikejc@us.ibm.com)
89994a338SPaul Mackerras * PPC64 updates by Dave Engebretsen (engebret@us.ibm.com)
99994a338SPaul Mackerras *
109994a338SPaul Mackerras * This program is free software; you can redistribute it and/or
119994a338SPaul Mackerras * modify it under the terms of the GNU General Public License
129994a338SPaul Mackerras * as published by the Free Software Foundation; either version
139994a338SPaul Mackerras * 2 of the License, or (at your option) any later version.
149994a338SPaul Mackerras *
159994a338SPaul Mackerras */
169994a338SPaul Mackerras
179994a338SPaul Mackerras#include <linux/sys.h>
189994a338SPaul Mackerras#include <asm/unistd.h>
199994a338SPaul Mackerras#include <asm/errno.h>
209994a338SPaul Mackerras#include <asm/processor.h>
219994a338SPaul Mackerras#include <asm/page.h>
229994a338SPaul Mackerras#include <asm/cache.h>
239994a338SPaul Mackerras#include <asm/ppc_asm.h>
249994a338SPaul Mackerras#include <asm/asm-offsets.h>
259994a338SPaul Mackerras#include <asm/cputable.h>
266cb7bfebSDavid Gibson#include <asm/thread_info.h>
271fc711f7SMichael Neuling#include <asm/kexec.h>
2846f52210SStephen Rothwell#include <asm/ptrace.h>
299994a338SPaul Mackerras
309994a338SPaul Mackerras	.text
319994a338SPaul Mackerras
329994a338SPaul Mackerras_GLOBAL(call_do_softirq)
339994a338SPaul Mackerras	mflr	r0
349994a338SPaul Mackerras	std	r0,16(r1)
354ae2dcb6SKumar Gala	stdu	r1,THREAD_SIZE-STACK_FRAME_OVERHEAD(r3)
369994a338SPaul Mackerras	mr	r1,r3
379994a338SPaul Mackerras	bl	.__do_softirq
389994a338SPaul Mackerras	ld	r1,0(r1)
399994a338SPaul Mackerras	ld	r0,16(r1)
409994a338SPaul Mackerras	mtlr	r0
419994a338SPaul Mackerras	blr
429994a338SPaul Mackerras
43b9e5b4e6SBenjamin Herrenschmidt_GLOBAL(call_handle_irq)
447d12e780SDavid Howells	ld	r8,0(r6)
459994a338SPaul Mackerras	mflr	r0
469994a338SPaul Mackerras	std	r0,16(r1)
47b9e5b4e6SBenjamin Herrenschmidt	mtctr	r8
484ae2dcb6SKumar Gala	stdu	r1,THREAD_SIZE-STACK_FRAME_OVERHEAD(r5)
497d12e780SDavid Howells	mr	r1,r5
50b9e5b4e6SBenjamin Herrenschmidt	bctrl
519994a338SPaul Mackerras	ld	r1,0(r1)
529994a338SPaul Mackerras	ld	r0,16(r1)
539994a338SPaul Mackerras	mtlr	r0
549994a338SPaul Mackerras	blr
559994a338SPaul Mackerras
569994a338SPaul Mackerras	.section	".toc","aw"
579994a338SPaul MackerrasPPC64_CACHES:
589994a338SPaul Mackerras	.tc		ppc64_caches[TC],ppc64_caches
599994a338SPaul Mackerras	.section	".text"
609994a338SPaul Mackerras
619994a338SPaul Mackerras/*
629994a338SPaul Mackerras * Write any modified data cache blocks out to memory
639994a338SPaul Mackerras * and invalidate the corresponding instruction cache blocks.
649994a338SPaul Mackerras *
659994a338SPaul Mackerras * flush_icache_range(unsigned long start, unsigned long stop)
669994a338SPaul Mackerras *
679994a338SPaul Mackerras *   flush all bytes from start through stop-1 inclusive
689994a338SPaul Mackerras */
699994a338SPaul Mackerras
709994a338SPaul Mackerras_KPROBE(__flush_icache_range)
719994a338SPaul Mackerras
729994a338SPaul Mackerras/*
739994a338SPaul Mackerras * Flush the data cache to memory
749994a338SPaul Mackerras *
759994a338SPaul Mackerras * Different systems have different cache line sizes
769994a338SPaul Mackerras * and in some cases i-cache and d-cache line sizes differ from
779994a338SPaul Mackerras * each other.
789994a338SPaul Mackerras */
799994a338SPaul Mackerras 	ld	r10,PPC64_CACHES@toc(r2)
809994a338SPaul Mackerras	lwz	r7,DCACHEL1LINESIZE(r10)/* Get cache line size */
819994a338SPaul Mackerras	addi	r5,r7,-1
829994a338SPaul Mackerras	andc	r6,r3,r5		/* round low to line bdy */
839994a338SPaul Mackerras	subf	r8,r6,r4		/* compute length */
849994a338SPaul Mackerras	add	r8,r8,r5		/* ensure we get enough */
859994a338SPaul Mackerras	lwz	r9,DCACHEL1LOGLINESIZE(r10)	/* Get log-2 of cache line size */
869994a338SPaul Mackerras	srw.	r8,r8,r9		/* compute line count */
879994a338SPaul Mackerras	beqlr				/* nothing to do? */
889994a338SPaul Mackerras	mtctr	r8
899994a338SPaul Mackerras1:	dcbst	0,r6
909994a338SPaul Mackerras	add	r6,r6,r7
919994a338SPaul Mackerras	bdnz	1b
929994a338SPaul Mackerras	sync
939994a338SPaul Mackerras
949994a338SPaul Mackerras/* Now invalidate the instruction cache */
959994a338SPaul Mackerras
969994a338SPaul Mackerras	lwz	r7,ICACHEL1LINESIZE(r10)	/* Get Icache line size */
979994a338SPaul Mackerras	addi	r5,r7,-1
989994a338SPaul Mackerras	andc	r6,r3,r5		/* round low to line bdy */
999994a338SPaul Mackerras	subf	r8,r6,r4		/* compute length */
1009994a338SPaul Mackerras	add	r8,r8,r5
1019994a338SPaul Mackerras	lwz	r9,ICACHEL1LOGLINESIZE(r10)	/* Get log-2 of Icache line size */
1029994a338SPaul Mackerras	srw.	r8,r8,r9		/* compute line count */
1039994a338SPaul Mackerras	beqlr				/* nothing to do? */
1049994a338SPaul Mackerras	mtctr	r8
1059994a338SPaul Mackerras2:	icbi	0,r6
1069994a338SPaul Mackerras	add	r6,r6,r7
1079994a338SPaul Mackerras	bdnz	2b
1089994a338SPaul Mackerras	isync
1099994a338SPaul Mackerras	blr
1109994a338SPaul Mackerras	.previous .text
1119994a338SPaul Mackerras/*
1129994a338SPaul Mackerras * Like above, but only do the D-cache.
1139994a338SPaul Mackerras *
1149994a338SPaul Mackerras * flush_dcache_range(unsigned long start, unsigned long stop)
1159994a338SPaul Mackerras *
1169994a338SPaul Mackerras *    flush all bytes from start to stop-1 inclusive
1179994a338SPaul Mackerras */
1189994a338SPaul Mackerras_GLOBAL(flush_dcache_range)
1199994a338SPaul Mackerras
1209994a338SPaul Mackerras/*
1219994a338SPaul Mackerras * Flush the data cache to memory
1229994a338SPaul Mackerras *
1239994a338SPaul Mackerras * Different systems have different cache line sizes
1249994a338SPaul Mackerras */
1259994a338SPaul Mackerras 	ld	r10,PPC64_CACHES@toc(r2)
1269994a338SPaul Mackerras	lwz	r7,DCACHEL1LINESIZE(r10)	/* Get dcache line size */
1279994a338SPaul Mackerras	addi	r5,r7,-1
1289994a338SPaul Mackerras	andc	r6,r3,r5		/* round low to line bdy */
1299994a338SPaul Mackerras	subf	r8,r6,r4		/* compute length */
1309994a338SPaul Mackerras	add	r8,r8,r5		/* ensure we get enough */
1319994a338SPaul Mackerras	lwz	r9,DCACHEL1LOGLINESIZE(r10)	/* Get log-2 of dcache line size */
1329994a338SPaul Mackerras	srw.	r8,r8,r9		/* compute line count */
1339994a338SPaul Mackerras	beqlr				/* nothing to do? */
1349994a338SPaul Mackerras	mtctr	r8
1359994a338SPaul Mackerras0:	dcbst	0,r6
1369994a338SPaul Mackerras	add	r6,r6,r7
1379994a338SPaul Mackerras	bdnz	0b
1389994a338SPaul Mackerras	sync
1399994a338SPaul Mackerras	blr
1409994a338SPaul Mackerras
1419994a338SPaul Mackerras/*
1429994a338SPaul Mackerras * Like above, but works on non-mapped physical addresses.
1439994a338SPaul Mackerras * Use only for non-LPAR setups ! It also assumes real mode
1449994a338SPaul Mackerras * is cacheable. Used for flushing out the DART before using
1459994a338SPaul Mackerras * it as uncacheable memory
1469994a338SPaul Mackerras *
1479994a338SPaul Mackerras * flush_dcache_phys_range(unsigned long start, unsigned long stop)
1489994a338SPaul Mackerras *
1499994a338SPaul Mackerras *    flush all bytes from start to stop-1 inclusive
1509994a338SPaul Mackerras */
1519994a338SPaul Mackerras_GLOBAL(flush_dcache_phys_range)
1529994a338SPaul Mackerras 	ld	r10,PPC64_CACHES@toc(r2)
1539994a338SPaul Mackerras	lwz	r7,DCACHEL1LINESIZE(r10)	/* Get dcache line size */
1549994a338SPaul Mackerras	addi	r5,r7,-1
1559994a338SPaul Mackerras	andc	r6,r3,r5		/* round low to line bdy */
1569994a338SPaul Mackerras	subf	r8,r6,r4		/* compute length */
1579994a338SPaul Mackerras	add	r8,r8,r5		/* ensure we get enough */
1589994a338SPaul Mackerras	lwz	r9,DCACHEL1LOGLINESIZE(r10)	/* Get log-2 of dcache line size */
1599994a338SPaul Mackerras	srw.	r8,r8,r9		/* compute line count */
1609994a338SPaul Mackerras	beqlr				/* nothing to do? */
1619994a338SPaul Mackerras	mfmsr	r5			/* Disable MMU Data Relocation */
1629994a338SPaul Mackerras	ori	r0,r5,MSR_DR
1639994a338SPaul Mackerras	xori	r0,r0,MSR_DR
1649994a338SPaul Mackerras	sync
1659994a338SPaul Mackerras	mtmsr	r0
1669994a338SPaul Mackerras	sync
1679994a338SPaul Mackerras	isync
1689994a338SPaul Mackerras	mtctr	r8
1699994a338SPaul Mackerras0:	dcbst	0,r6
1709994a338SPaul Mackerras	add	r6,r6,r7
1719994a338SPaul Mackerras	bdnz	0b
1729994a338SPaul Mackerras	sync
1739994a338SPaul Mackerras	isync
1749994a338SPaul Mackerras	mtmsr	r5			/* Re-enable MMU Data Relocation */
1759994a338SPaul Mackerras	sync
1769994a338SPaul Mackerras	isync
1779994a338SPaul Mackerras	blr
1789994a338SPaul Mackerras
1799994a338SPaul Mackerras_GLOBAL(flush_inval_dcache_range)
1809994a338SPaul Mackerras 	ld	r10,PPC64_CACHES@toc(r2)
1819994a338SPaul Mackerras	lwz	r7,DCACHEL1LINESIZE(r10)	/* Get dcache line size */
1829994a338SPaul Mackerras	addi	r5,r7,-1
1839994a338SPaul Mackerras	andc	r6,r3,r5		/* round low to line bdy */
1849994a338SPaul Mackerras	subf	r8,r6,r4		/* compute length */
1859994a338SPaul Mackerras	add	r8,r8,r5		/* ensure we get enough */
1869994a338SPaul Mackerras	lwz	r9,DCACHEL1LOGLINESIZE(r10)/* Get log-2 of dcache line size */
1879994a338SPaul Mackerras	srw.	r8,r8,r9		/* compute line count */
1889994a338SPaul Mackerras	beqlr				/* nothing to do? */
1899994a338SPaul Mackerras	sync
1909994a338SPaul Mackerras	isync
1919994a338SPaul Mackerras	mtctr	r8
1929994a338SPaul Mackerras0:	dcbf	0,r6
1939994a338SPaul Mackerras	add	r6,r6,r7
1949994a338SPaul Mackerras	bdnz	0b
1959994a338SPaul Mackerras	sync
1969994a338SPaul Mackerras	isync
1979994a338SPaul Mackerras	blr
1989994a338SPaul Mackerras
1999994a338SPaul Mackerras
2009994a338SPaul Mackerras/*
2019994a338SPaul Mackerras * Flush a particular page from the data cache to RAM.
2029994a338SPaul Mackerras * Note: this is necessary because the instruction cache does *not*
2039994a338SPaul Mackerras * snoop from the data cache.
2049994a338SPaul Mackerras *
2059994a338SPaul Mackerras *	void __flush_dcache_icache(void *page)
2069994a338SPaul Mackerras */
2079994a338SPaul Mackerras_GLOBAL(__flush_dcache_icache)
2089994a338SPaul Mackerras/*
2099994a338SPaul Mackerras * Flush the data cache to memory
2109994a338SPaul Mackerras *
2119994a338SPaul Mackerras * Different systems have different cache line sizes
2129994a338SPaul Mackerras */
2139994a338SPaul Mackerras
2149994a338SPaul Mackerras/* Flush the dcache */
2159994a338SPaul Mackerras 	ld	r7,PPC64_CACHES@toc(r2)
2169994a338SPaul Mackerras	clrrdi	r3,r3,PAGE_SHIFT           	    /* Page align */
2179994a338SPaul Mackerras	lwz	r4,DCACHEL1LINESPERPAGE(r7)	/* Get # dcache lines per page */
2189994a338SPaul Mackerras	lwz	r5,DCACHEL1LINESIZE(r7)		/* Get dcache line size */
2199994a338SPaul Mackerras	mr	r6,r3
2209994a338SPaul Mackerras	mtctr	r4
2219994a338SPaul Mackerras0:	dcbst	0,r6
2229994a338SPaul Mackerras	add	r6,r6,r5
2239994a338SPaul Mackerras	bdnz	0b
2249994a338SPaul Mackerras	sync
2259994a338SPaul Mackerras
2269994a338SPaul Mackerras/* Now invalidate the icache */
2279994a338SPaul Mackerras
2289994a338SPaul Mackerras	lwz	r4,ICACHEL1LINESPERPAGE(r7)	/* Get # icache lines per page */
2299994a338SPaul Mackerras	lwz	r5,ICACHEL1LINESIZE(r7)		/* Get icache line size */
2309994a338SPaul Mackerras	mtctr	r4
2319994a338SPaul Mackerras1:	icbi	0,r3
2329994a338SPaul Mackerras	add	r3,r3,r5
2339994a338SPaul Mackerras	bdnz	1b
2349994a338SPaul Mackerras	isync
2359994a338SPaul Mackerras	blr
2369994a338SPaul Mackerras
2373f639ee8SStephen Rothwell
2389994a338SPaul Mackerras#if defined(CONFIG_PPC_PMAC) || defined(CONFIG_PPC_MAPLE)
2399994a338SPaul Mackerras/*
2409994a338SPaul Mackerras * Do an IO access in real mode
2419994a338SPaul Mackerras */
2429994a338SPaul Mackerras_GLOBAL(real_readb)
2439994a338SPaul Mackerras	mfmsr	r7
2449994a338SPaul Mackerras	ori	r0,r7,MSR_DR
2459994a338SPaul Mackerras	xori	r0,r0,MSR_DR
2469994a338SPaul Mackerras	sync
2479994a338SPaul Mackerras	mtmsrd	r0
2489994a338SPaul Mackerras	sync
2499994a338SPaul Mackerras	isync
2509994a338SPaul Mackerras	mfspr	r6,SPRN_HID4
2519994a338SPaul Mackerras	rldicl	r5,r6,32,0
2529994a338SPaul Mackerras	ori	r5,r5,0x100
2539994a338SPaul Mackerras	rldicl	r5,r5,32,0
2549994a338SPaul Mackerras	sync
2559994a338SPaul Mackerras	mtspr	SPRN_HID4,r5
2569994a338SPaul Mackerras	isync
2579994a338SPaul Mackerras	slbia
2589994a338SPaul Mackerras	isync
2599994a338SPaul Mackerras	lbz	r3,0(r3)
2609994a338SPaul Mackerras	sync
2619994a338SPaul Mackerras	mtspr	SPRN_HID4,r6
2629994a338SPaul Mackerras	isync
2639994a338SPaul Mackerras	slbia
2649994a338SPaul Mackerras	isync
2659994a338SPaul Mackerras	mtmsrd	r7
2669994a338SPaul Mackerras	sync
2679994a338SPaul Mackerras	isync
2689994a338SPaul Mackerras	blr
2699994a338SPaul Mackerras
2709994a338SPaul Mackerras	/*
2719994a338SPaul Mackerras * Do an IO access in real mode
2729994a338SPaul Mackerras */
2739994a338SPaul Mackerras_GLOBAL(real_writeb)
2749994a338SPaul Mackerras	mfmsr	r7
2759994a338SPaul Mackerras	ori	r0,r7,MSR_DR
2769994a338SPaul Mackerras	xori	r0,r0,MSR_DR
2779994a338SPaul Mackerras	sync
2789994a338SPaul Mackerras	mtmsrd	r0
2799994a338SPaul Mackerras	sync
2809994a338SPaul Mackerras	isync
2819994a338SPaul Mackerras	mfspr	r6,SPRN_HID4
2829994a338SPaul Mackerras	rldicl	r5,r6,32,0
2839994a338SPaul Mackerras	ori	r5,r5,0x100
2849994a338SPaul Mackerras	rldicl	r5,r5,32,0
2859994a338SPaul Mackerras	sync
2869994a338SPaul Mackerras	mtspr	SPRN_HID4,r5
2879994a338SPaul Mackerras	isync
2889994a338SPaul Mackerras	slbia
2899994a338SPaul Mackerras	isync
2909994a338SPaul Mackerras	stb	r3,0(r4)
2919994a338SPaul Mackerras	sync
2929994a338SPaul Mackerras	mtspr	SPRN_HID4,r6
2939994a338SPaul Mackerras	isync
2949994a338SPaul Mackerras	slbia
2959994a338SPaul Mackerras	isync
2969994a338SPaul Mackerras	mtmsrd	r7
2979994a338SPaul Mackerras	sync
2989994a338SPaul Mackerras	isync
2999994a338SPaul Mackerras	blr
3009994a338SPaul Mackerras#endif /* defined(CONFIG_PPC_PMAC) || defined(CONFIG_PPC_MAPLE) */
3019994a338SPaul Mackerras
30239c870d5SOlof Johansson#ifdef CONFIG_PPC_PASEMI
30339c870d5SOlof Johansson
30439c870d5SOlof Johansson_GLOBAL(real_205_readb)
30539c870d5SOlof Johansson	mfmsr	r7
30639c870d5SOlof Johansson	ori	r0,r7,MSR_DR
30739c870d5SOlof Johansson	xori	r0,r0,MSR_DR
30839c870d5SOlof Johansson	sync
30939c870d5SOlof Johansson	mtmsrd	r0
31039c870d5SOlof Johansson	sync
31139c870d5SOlof Johansson	isync
312*e55174e9SMichael Neuling	LBZCIX(R3,R0,R3)
31339c870d5SOlof Johansson	isync
31439c870d5SOlof Johansson	mtmsrd	r7
31539c870d5SOlof Johansson	sync
31639c870d5SOlof Johansson	isync
31739c870d5SOlof Johansson	blr
31839c870d5SOlof Johansson
31939c870d5SOlof Johansson_GLOBAL(real_205_writeb)
32039c870d5SOlof Johansson	mfmsr	r7
32139c870d5SOlof Johansson	ori	r0,r7,MSR_DR
32239c870d5SOlof Johansson	xori	r0,r0,MSR_DR
32339c870d5SOlof Johansson	sync
32439c870d5SOlof Johansson	mtmsrd	r0
32539c870d5SOlof Johansson	sync
32639c870d5SOlof Johansson	isync
327*e55174e9SMichael Neuling	STBCIX(R3,R0,R4)
32839c870d5SOlof Johansson	isync
32939c870d5SOlof Johansson	mtmsrd	r7
33039c870d5SOlof Johansson	sync
33139c870d5SOlof Johansson	isync
33239c870d5SOlof Johansson	blr
33339c870d5SOlof Johansson
33439c870d5SOlof Johansson#endif /* CONFIG_PPC_PASEMI */
33539c870d5SOlof Johansson
33639c870d5SOlof Johansson
337e48f7eb2SDmitry Eremin-Solenikov#if defined(CONFIG_CPU_FREQ_PMAC64) || defined(CONFIG_CPU_FREQ_MAPLE)
3389994a338SPaul Mackerras/*
3394350147aSBenjamin Herrenschmidt * SCOM access functions for 970 (FX only for now)
3404350147aSBenjamin Herrenschmidt *
3414350147aSBenjamin Herrenschmidt * unsigned long scom970_read(unsigned int address);
3424350147aSBenjamin Herrenschmidt * void scom970_write(unsigned int address, unsigned long value);
3434350147aSBenjamin Herrenschmidt *
3444350147aSBenjamin Herrenschmidt * The address passed in is the 24 bits register address. This code
3454350147aSBenjamin Herrenschmidt * is 970 specific and will not check the status bits, so you should
3464350147aSBenjamin Herrenschmidt * know what you are doing.
3474350147aSBenjamin Herrenschmidt */
3484350147aSBenjamin Herrenschmidt_GLOBAL(scom970_read)
3494350147aSBenjamin Herrenschmidt	/* interrupts off */
3504350147aSBenjamin Herrenschmidt	mfmsr	r4
3514350147aSBenjamin Herrenschmidt	ori	r0,r4,MSR_EE
3524350147aSBenjamin Herrenschmidt	xori	r0,r0,MSR_EE
3534350147aSBenjamin Herrenschmidt	mtmsrd	r0,1
3544350147aSBenjamin Herrenschmidt
3554350147aSBenjamin Herrenschmidt	/* rotate 24 bits SCOM address 8 bits left and mask out it's low 8 bits
3564350147aSBenjamin Herrenschmidt	 * (including parity). On current CPUs they must be 0'd,
3574350147aSBenjamin Herrenschmidt	 * and finally or in RW bit
3584350147aSBenjamin Herrenschmidt	 */
3594350147aSBenjamin Herrenschmidt	rlwinm	r3,r3,8,0,15
3604350147aSBenjamin Herrenschmidt	ori	r3,r3,0x8000
3614350147aSBenjamin Herrenschmidt
3624350147aSBenjamin Herrenschmidt	/* do the actual scom read */
3634350147aSBenjamin Herrenschmidt	sync
3644350147aSBenjamin Herrenschmidt	mtspr	SPRN_SCOMC,r3
3654350147aSBenjamin Herrenschmidt	isync
3664350147aSBenjamin Herrenschmidt	mfspr	r3,SPRN_SCOMD
3674350147aSBenjamin Herrenschmidt	isync
3684350147aSBenjamin Herrenschmidt	mfspr	r0,SPRN_SCOMC
3694350147aSBenjamin Herrenschmidt	isync
3704350147aSBenjamin Herrenschmidt
3714350147aSBenjamin Herrenschmidt	/* XXX:	fixup result on some buggy 970's (ouch ! we lost a bit, bah
3724350147aSBenjamin Herrenschmidt	 * that's the best we can do). Not implemented yet as we don't use
3734350147aSBenjamin Herrenschmidt	 * the scom on any of the bogus CPUs yet, but may have to be done
3744350147aSBenjamin Herrenschmidt	 * ultimately
3754350147aSBenjamin Herrenschmidt	 */
3764350147aSBenjamin Herrenschmidt
3774350147aSBenjamin Herrenschmidt	/* restore interrupts */
3784350147aSBenjamin Herrenschmidt	mtmsrd	r4,1
3794350147aSBenjamin Herrenschmidt	blr
3804350147aSBenjamin Herrenschmidt
3814350147aSBenjamin Herrenschmidt
3824350147aSBenjamin Herrenschmidt_GLOBAL(scom970_write)
3834350147aSBenjamin Herrenschmidt	/* interrupts off */
3844350147aSBenjamin Herrenschmidt	mfmsr	r5
3854350147aSBenjamin Herrenschmidt	ori	r0,r5,MSR_EE
3864350147aSBenjamin Herrenschmidt	xori	r0,r0,MSR_EE
3874350147aSBenjamin Herrenschmidt	mtmsrd	r0,1
3884350147aSBenjamin Herrenschmidt
3894350147aSBenjamin Herrenschmidt	/* rotate 24 bits SCOM address 8 bits left and mask out it's low 8 bits
3904350147aSBenjamin Herrenschmidt	 * (including parity). On current CPUs they must be 0'd.
3914350147aSBenjamin Herrenschmidt	 */
3924350147aSBenjamin Herrenschmidt
3934350147aSBenjamin Herrenschmidt	rlwinm	r3,r3,8,0,15
3944350147aSBenjamin Herrenschmidt
3954350147aSBenjamin Herrenschmidt	sync
3964350147aSBenjamin Herrenschmidt	mtspr	SPRN_SCOMD,r4      /* write data */
3974350147aSBenjamin Herrenschmidt	isync
3984350147aSBenjamin Herrenschmidt	mtspr	SPRN_SCOMC,r3      /* write command */
3994350147aSBenjamin Herrenschmidt	isync
4004350147aSBenjamin Herrenschmidt	mfspr	3,SPRN_SCOMC
4014350147aSBenjamin Herrenschmidt	isync
4024350147aSBenjamin Herrenschmidt
4034350147aSBenjamin Herrenschmidt	/* restore interrupts */
4044350147aSBenjamin Herrenschmidt	mtmsrd	r5,1
4054350147aSBenjamin Herrenschmidt	blr
406e48f7eb2SDmitry Eremin-Solenikov#endif /* CONFIG_CPU_FREQ_PMAC64 || CONFIG_CPU_FREQ_MAPLE */
4074350147aSBenjamin Herrenschmidt
4084350147aSBenjamin Herrenschmidt
4094350147aSBenjamin Herrenschmidt/*
4109994a338SPaul Mackerras * Create a kernel thread
4119994a338SPaul Mackerras *   kernel_thread(fn, arg, flags)
4129994a338SPaul Mackerras */
4139994a338SPaul Mackerras_GLOBAL(kernel_thread)
4149994a338SPaul Mackerras	std	r29,-24(r1)
4159994a338SPaul Mackerras	std	r30,-16(r1)
4169994a338SPaul Mackerras	stdu	r1,-STACK_FRAME_OVERHEAD(r1)
4179994a338SPaul Mackerras	mr	r29,r3
4189994a338SPaul Mackerras	mr	r30,r4
4199994a338SPaul Mackerras	ori	r3,r5,CLONE_VM	/* flags */
4209994a338SPaul Mackerras	oris	r3,r3,(CLONE_UNTRACED>>16)
4219994a338SPaul Mackerras	li	r4,0		/* new sp (unused) */
4229994a338SPaul Mackerras	li	r0,__NR_clone
4239994a338SPaul Mackerras	sc
42441c2e949SJosh Poimboeuf	bns+	1f		/* did system call indicate error? */
42541c2e949SJosh Poimboeuf	neg	r3,r3		/* if so, make return code negative */
42641c2e949SJosh Poimboeuf1:	cmpdi	0,r3,0		/* parent or child? */
42741c2e949SJosh Poimboeuf	bne	2f		/* return if parent */
4289994a338SPaul Mackerras	li	r0,0
4299994a338SPaul Mackerras	stdu	r0,-STACK_FRAME_OVERHEAD(r1)
4309994a338SPaul Mackerras	ld	r2,8(r29)
4319994a338SPaul Mackerras	ld	r29,0(r29)
4329994a338SPaul Mackerras	mtlr	r29              /* fn addr in lr */
4339994a338SPaul Mackerras	mr	r3,r30	        /* load arg and call fn */
4349994a338SPaul Mackerras	blrl
4359994a338SPaul Mackerras	li	r0,__NR_exit	/* exit after child exits */
4369994a338SPaul Mackerras        li	r3,0
4379994a338SPaul Mackerras	sc
43841c2e949SJosh Poimboeuf2:	addi	r1,r1,STACK_FRAME_OVERHEAD
4399994a338SPaul Mackerras	ld	r29,-24(r1)
4409994a338SPaul Mackerras	ld	r30,-16(r1)
4419994a338SPaul Mackerras	blr
4429994a338SPaul Mackerras
4439994a338SPaul Mackerras/*
4449994a338SPaul Mackerras * disable_kernel_fp()
4459994a338SPaul Mackerras * Disable the FPU.
4469994a338SPaul Mackerras */
4479994a338SPaul Mackerras_GLOBAL(disable_kernel_fp)
4489994a338SPaul Mackerras	mfmsr	r3
4499994a338SPaul Mackerras	rldicl	r0,r3,(63-MSR_FP_LG),1
4509994a338SPaul Mackerras	rldicl	r3,r0,(MSR_FP_LG+1),0
4519994a338SPaul Mackerras	mtmsrd	r3			/* disable use of fpu now */
4529994a338SPaul Mackerras	isync
4539994a338SPaul Mackerras	blr
4549994a338SPaul Mackerras
4559994a338SPaul Mackerras/* kexec_wait(phys_cpu)
4569994a338SPaul Mackerras *
4579994a338SPaul Mackerras * wait for the flag to change, indicating this kernel is going away but
4589994a338SPaul Mackerras * the slave code for the next one is at addresses 0 to 100.
4599994a338SPaul Mackerras *
4603d2cea73SMilton Miller * This is used by all slaves, even those that did not find a matching
4613d2cea73SMilton Miller * paca in the secondary startup code.
4629994a338SPaul Mackerras *
4639994a338SPaul Mackerras * Physical (hardware) cpu id should be in r3.
4649994a338SPaul Mackerras */
4659994a338SPaul Mackerras_GLOBAL(kexec_wait)
4669994a338SPaul Mackerras	bl	1f
4679994a338SPaul Mackerras1:	mflr	r5
4689994a338SPaul Mackerras	addi	r5,r5,kexec_flag-1b
4699994a338SPaul Mackerras
4709994a338SPaul Mackerras99:	HMT_LOW
4719994a338SPaul Mackerras#ifdef CONFIG_KEXEC		/* use no memory without kexec */
4729994a338SPaul Mackerras	lwz	r4,0(r5)
4739994a338SPaul Mackerras	cmpwi	0,r4,0
4749994a338SPaul Mackerras	bnea	0x60
4759994a338SPaul Mackerras#endif
4769994a338SPaul Mackerras	b	99b
4779994a338SPaul Mackerras
4789994a338SPaul Mackerras/* this can be in text because we won't change it until we are
4799994a338SPaul Mackerras * running in real anyways
4809994a338SPaul Mackerras */
4819994a338SPaul Mackerraskexec_flag:
4829994a338SPaul Mackerras	.long	0
4839994a338SPaul Mackerras
4849994a338SPaul Mackerras
4859994a338SPaul Mackerras#ifdef CONFIG_KEXEC
4869994a338SPaul Mackerras
4879994a338SPaul Mackerras/* kexec_smp_wait(void)
4889994a338SPaul Mackerras *
4899994a338SPaul Mackerras * call with interrupts off
4909994a338SPaul Mackerras * note: this is a terminal routine, it does not save lr
4919994a338SPaul Mackerras *
4929994a338SPaul Mackerras * get phys id from paca
4939994a338SPaul Mackerras * switch to real mode
4943d2cea73SMilton Miller * mark the paca as no longer used
4959994a338SPaul Mackerras * join other cpus in kexec_wait(phys_id)
4969994a338SPaul Mackerras */
4979994a338SPaul Mackerras_GLOBAL(kexec_smp_wait)
4989994a338SPaul Mackerras	lhz	r3,PACAHWCPUID(r13)
4999994a338SPaul Mackerras	bl	real_mode
5003d2cea73SMilton Miller
5013d2cea73SMilton Miller	li	r4,KEXEC_STATE_REAL_MODE
5023d2cea73SMilton Miller	stb	r4,PACAKEXECSTATE(r13)
5033d2cea73SMilton Miller	SYNC
5043d2cea73SMilton Miller
5059994a338SPaul Mackerras	b	.kexec_wait
5069994a338SPaul Mackerras
5079994a338SPaul Mackerras/*
5089994a338SPaul Mackerras * switch to real mode (turn mmu off)
5099994a338SPaul Mackerras * we use the early kernel trick that the hardware ignores bits
5109994a338SPaul Mackerras * 0 and 1 (big endian) of the effective address in real mode
5119994a338SPaul Mackerras *
5129994a338SPaul Mackerras * don't overwrite r3 here, it is live for kexec_wait above.
5139994a338SPaul Mackerras */
5149994a338SPaul Mackerrasreal_mode:	/* assume normal blr return */
5159994a338SPaul Mackerras1:	li	r9,MSR_RI
5169994a338SPaul Mackerras	li	r10,MSR_DR|MSR_IR
5179994a338SPaul Mackerras	mflr	r11		/* return address to SRR0 */
5189994a338SPaul Mackerras	mfmsr	r12
5199994a338SPaul Mackerras	andc	r9,r12,r9
5209994a338SPaul Mackerras	andc	r10,r12,r10
5219994a338SPaul Mackerras
5229994a338SPaul Mackerras	mtmsrd	r9,1
5239994a338SPaul Mackerras	mtspr	SPRN_SRR1,r10
5249994a338SPaul Mackerras	mtspr	SPRN_SRR0,r11
5259994a338SPaul Mackerras	rfid
5269994a338SPaul Mackerras
5279994a338SPaul Mackerras
5289994a338SPaul Mackerras/*
5291767c8f3SMilton Miller * kexec_sequence(newstack, start, image, control, clear_all())
5309994a338SPaul Mackerras *
5319994a338SPaul Mackerras * does the grungy work with stack switching and real mode switches
5329994a338SPaul Mackerras * also does simple calls to other code
5339994a338SPaul Mackerras */
5349994a338SPaul Mackerras
5359994a338SPaul Mackerras_GLOBAL(kexec_sequence)
5369994a338SPaul Mackerras	mflr	r0
5379994a338SPaul Mackerras	std	r0,16(r1)
5389994a338SPaul Mackerras
5399994a338SPaul Mackerras	/* switch stacks to newstack -- &kexec_stack.stack */
5404ae2dcb6SKumar Gala	stdu	r1,THREAD_SIZE-STACK_FRAME_OVERHEAD(r3)
5419994a338SPaul Mackerras	mr	r1,r3
5429994a338SPaul Mackerras
5439994a338SPaul Mackerras	li	r0,0
5449994a338SPaul Mackerras	std	r0,16(r1)
5459994a338SPaul Mackerras
5469994a338SPaul Mackerras	/* save regs for local vars on new stack.
5479994a338SPaul Mackerras	 * yes, we won't go back, but ...
5489994a338SPaul Mackerras	 */
5499994a338SPaul Mackerras	std	r31,-8(r1)
5509994a338SPaul Mackerras	std	r30,-16(r1)
5519994a338SPaul Mackerras	std	r29,-24(r1)
5529994a338SPaul Mackerras	std	r28,-32(r1)
5539994a338SPaul Mackerras	std	r27,-40(r1)
5549994a338SPaul Mackerras	std	r26,-48(r1)
5559994a338SPaul Mackerras	std	r25,-56(r1)
5569994a338SPaul Mackerras
5574ae2dcb6SKumar Gala	stdu	r1,-STACK_FRAME_OVERHEAD-64(r1)
5589994a338SPaul Mackerras
5599994a338SPaul Mackerras	/* save args into preserved regs */
5609994a338SPaul Mackerras	mr	r31,r3			/* newstack (both) */
5619994a338SPaul Mackerras	mr	r30,r4			/* start (real) */
5629994a338SPaul Mackerras	mr	r29,r5			/* image (virt) */
5639994a338SPaul Mackerras	mr	r28,r6			/* control, unused */
5649994a338SPaul Mackerras	mr	r27,r7			/* clear_all() fn desc */
5651767c8f3SMilton Miller	mr	r26,r8			/* spare */
5669994a338SPaul Mackerras	lhz	r25,PACAHWCPUID(r13)	/* get our phys cpu from paca */
5679994a338SPaul Mackerras
5689994a338SPaul Mackerras	/* disable interrupts, we are overwriting kernel data next */
5699994a338SPaul Mackerras	mfmsr	r3
5709994a338SPaul Mackerras	rlwinm	r3,r3,0,17,15
5719994a338SPaul Mackerras	mtmsrd	r3,1
5729994a338SPaul Mackerras
5739994a338SPaul Mackerras	/* copy dest pages, flush whole dest image */
5749994a338SPaul Mackerras	mr	r3,r29
5759994a338SPaul Mackerras	bl	.kexec_copy_flush	/* (image) */
5769994a338SPaul Mackerras
5779994a338SPaul Mackerras	/* turn off mmu */
5789994a338SPaul Mackerras	bl	real_mode
5799994a338SPaul Mackerras
580ee46a90bSMilton Miller	/* copy  0x100 bytes starting at start to 0 */
581ee46a90bSMilton Miller	li	r3,0
582ee46a90bSMilton Miller	mr	r4,r30		/* start, aka phys mem offset */
583ee46a90bSMilton Miller	li	r5,0x100
584ee46a90bSMilton Miller	li	r6,0
585ee46a90bSMilton Miller	bl	.copy_and_flush	/* (dest, src, copy limit, start offset) */
586ee46a90bSMilton Miller1:	/* assume normal blr return */
587ee46a90bSMilton Miller
588ee46a90bSMilton Miller	/* release other cpus to the new kernel secondary start at 0x60 */
589ee46a90bSMilton Miller	mflr	r5
590ee46a90bSMilton Miller	li	r6,1
591ee46a90bSMilton Miller	stw	r6,kexec_flag-1b(5)
592ee46a90bSMilton Miller
5939994a338SPaul Mackerras	/* clear out hardware hash page table and tlb */
5949994a338SPaul Mackerras	ld	r5,0(r27)		/* deref function descriptor */
5959994a338SPaul Mackerras	mtctr	r5
5968d950cb8SGeoff Levand	bctrl				/* ppc_md.hpte_clear_all(void); */
5979994a338SPaul Mackerras
5989994a338SPaul Mackerras/*
5999994a338SPaul Mackerras *   kexec image calling is:
6009994a338SPaul Mackerras *      the first 0x100 bytes of the entry point are copied to 0
6019994a338SPaul Mackerras *
6029994a338SPaul Mackerras *      all slaves branch to slave = 0x60 (absolute)
6039994a338SPaul Mackerras *              slave(phys_cpu_id);
6049994a338SPaul Mackerras *
6059994a338SPaul Mackerras *      master goes to start = entry point
6069994a338SPaul Mackerras *              start(phys_cpu_id, start, 0);
6079994a338SPaul Mackerras *
6089994a338SPaul Mackerras *
6099994a338SPaul Mackerras *   a wrapper is needed to call existing kernels, here is an approximate
6109994a338SPaul Mackerras *   description of one method:
6119994a338SPaul Mackerras *
6129994a338SPaul Mackerras * v2: (2.6.10)
6139994a338SPaul Mackerras *   start will be near the boot_block (maybe 0x100 bytes before it?)
6149994a338SPaul Mackerras *   it will have a 0x60, which will b to boot_block, where it will wait
6159994a338SPaul Mackerras *   and 0 will store phys into struct boot-block and load r3 from there,
6169994a338SPaul Mackerras *   copy kernel 0-0x100 and tell slaves to back down to 0x60 again
6179994a338SPaul Mackerras *
6189994a338SPaul Mackerras * v1: (2.6.9)
6199994a338SPaul Mackerras *    boot block will have all cpus scanning device tree to see if they
6209994a338SPaul Mackerras *    are the boot cpu ?????
6219994a338SPaul Mackerras *    other device tree differences (prop sizes, va vs pa, etc)...
6229994a338SPaul Mackerras */
6239994a338SPaul Mackerras	mr	r3,r25	# my phys cpu
6249994a338SPaul Mackerras	mr	r4,r30	# start, aka phys mem offset
6259994a338SPaul Mackerras	mtlr	4
6269994a338SPaul Mackerras	li	r5,0
6271767c8f3SMilton Miller	blr	/* image->start(physid, image->start, 0); */
6289994a338SPaul Mackerras#endif /* CONFIG_KEXEC */
629