xref: /freebsd/sys/arm/arm/locore.S (revision 262e143bd46171a6415a5b28af260a5efa2a3db8)
1/*	$NetBSD: locore.S,v 1.14 2003/04/20 16:21:40 thorpej Exp $	*/
2
3/*-
4 * Copyright (C) 1994-1997 Mark Brinicombe
5 * Copyright (C) 1994 Brini
6 * All rights reserved.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 *    notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 *    notice, this list of conditions and the following disclaimer in the
15 *    documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 *    must display the following acknowledgement:
18 *	This product includes software developed by Brini.
19 * 4. The name of Brini may not be used to endorse or promote products
20 *    derived from this software without specific prior written permission.
21 *
22 * THIS SOFTWARE IS PROVIDED BY BRINI ``AS IS'' AND ANY EXPRESS OR
23 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
24 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
25 * IN NO EVENT SHALL BRINI BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
27 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
28 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
29 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
30 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
31 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 *
33 */
34
35#include "assym.s"
36#include <sys/syscall.h>
37#include <machine/asm.h>
38#include <machine/armreg.h>
39#include <machine/pte.h>
40__FBSDID("$FreeBSD$");
41
42/* What size should this really be ? It is only used by init_arm() */
43#define INIT_ARM_STACK_SIZE	2048
44
45/*
46 * This is for kvm_mkdb, and should be the address of the beginning
47 * of the kernel text segment (not necessarily the same as kernbase).
48 */
49
50
51#define	CPWAIT_BRANCH							 \
52	sub	pc, pc, #4
53
54#define	CPWAIT(tmp)							 \
55	mrc	p15, 0, tmp, c2, c0, 0	/* arbitrary read of CP15 */	;\
56	mov	tmp, tmp		/* wait for it to complete */	;\
57	CPWAIT_BRANCH			/* branch to next insn */
58
59	.text
60	.align	0
61.globl kernbase
62.set kernbase,KERNBASE
63.globl physaddr
64.set physaddr,PHYSADDR
65
66ENTRY_NP(btext)
67
68ASENTRY_NP(_start)
69
70
71	/* Check if we are running on RAM, if not move ourself to RAM */
72#if 0
73	cmp	pc, #PHYSADDR
74	bhi	start_inram /* XXX: This is wrong */
75#endif
76	b	start_inram /*
77			     * XXX: this is even more wrong, but RedBoot
78			     * use 0x00000000-0x100000000 as virtual
79			     * addresses for the RAM.
80			     */
81
82	/* move me to RAM
83	 * XXX: we can use memcpy if it is PIC
84	 */
85	ldr r1, Lcopy_size
86	adr r0, _C_LABEL(_start)
87	add r1, r1, #3
88	mov r1, r1, LSR #2
89	mov r2, #PHYSADDR
90	add r2, r2, #0x00200000
91	mov r4, r2
92
935:	ldr r3,[r0],#4
94	str r3,[r2],#4
95	subs r1,r1,#1
96	bhi 5b
97
98	/* Jump to RAM */
99	ldr r0, Lstart_off
100	add pc, r4, r0
101
102Lcopy_size:	.word _edata-_C_LABEL(_start)
103Lstart_off:	.word start_inram-_C_LABEL(_start)
104start_inram:
105	adr	r7, Lunmapped
106	bic	r7, r7, #0xff000000
107	orr	r7, r7, #PHYSADDR
108
109	/* Disable MMU for a while */
110	mrc     p15, 0, r2, c1, c0, 0
111	bic     r2, r2, #CPU_CONTROL_MMU_ENABLE
112	mcr     p15, 0, r2, c1, c0, 0
113
114	nop
115	nop
116	nop
117	mov	pc, r7
118Lunmapped:
119
120#ifdef STARTUP_PAGETABLE_ADDR
121	/* build page table from scratch */
122	ldr	r0, Lstartup_pagetable
123	adr	r4, mmu_init_table
124	b	3f
125
1262:
127	str	r3, [r0, r2]
128	add	r2, r2, #4
129	add	r3, r3, #(L1_S_SIZE)
130	adds	r1, r1, #-1
131	bhi	2b
1323:
133	ldmia	r4!, {r1,r2,r3}   /* # of sections, PA|attr, VA */
134	cmp	r1, #0
135	adrne	r5, 2b
136	bicne	r5, r5, #0xff000000
137	orrne	r5, r5, #PHYSADDR
138	movne	pc, r5
139
140	mcr	p15, 0, r0, c2, c0, 0	/* Set TTB */
141	mcr	p15, 0, r0, c8, c7, 0	/* Flush TLB */
142
143	/* Set the Domain Access register.  Very important! */
144	mov     r0, #((DOMAIN_CLIENT << (PMAP_DOMAIN_KERNEL*2)) | DOMAIN_CLIENT)
145	mcr	p15, 0, r0, c3, c0, 0
146
147	/* Enable MMU */
148	mrc	p15, 0, r0, c1, c0, 0
149	orr	r0, r0, #CPU_CONTROL_MMU_ENABLE
150	mcr	p15, 0, r0, c1, c0, 0
151	CPWAIT(r0)
152
153	bl	mmu_done
154mmu_done:
155#endif
156	adr	r1, .Lstart
157	ldmia	r1, {r1, r2, sp}	/* Set initial stack and */
158	sub	r2, r2, r1		/* get zero init data */
159	mov	r3, #0
160
161.L1:
162	str	r3, [r1], #0x0004	/* get zero init data */
163	subs	r2, r2, #4
164	bgt	.L1
165
166	ldr	r4, =KERNVIRTADDR
167	cmp	pc, r4
168#if	KERNVIRTADDR > KERNPHYSADDR
169	bgt	virt_done
170	ldr	r4, =KERNVIRTADDR
171	ldr	r5, =KERNPHYSADDR
172	sub	r4, r4, r5
173	add	pc, pc, r4
174#else
175	blt	virt_done
176	ldr	r4, =KERNPHYSADDR
177	ldr	r5, =KERNVIRTADDR
178	sub	r4, r4, r5
179	sub	pc, pc, r4
180#endif
181virt_done:
182	nop
183	mov	fp, #0		/* trace back starts here */
184	bl	_C_LABEL(initarm)	/* Off we go */
185
186	/* init arm will return the new stack pointer. */
187	mov	sp, r0
188
189	bl	_C_LABEL(mi_startup)		/* call mi_startup()! */
190
191	adr	r0, .Lmainreturned
192	b	_C_LABEL(panic)
193	/* NOTEACHED */
194#ifdef STARTUP_PAGETABLE_ADDR
195#define MMU_INIT(va,pa,n_sec,attr) \
196	.word	n_sec					    ; \
197	.word	4*((va)>>L1_S_SHIFT)			    ; \
198	.word	(pa)|(attr)				    ;
199
200Lvirtaddr:
201	.word	KERNVIRTADDR
202Lphysaddr:
203	.word	KERNPHYSADDR
204Lstartup_pagetable:
205	.word	STARTUP_PAGETABLE_ADDR
206mmu_init_table:
207	/* fill all table VA==PA */
208	/* map SDRAM VA==PA, WT cacheable */
209	MMU_INIT(PHYSADDR, PHYSADDR , 64, L1_TYPE_S|L1_S_C|L1_S_AP(AP_KRW))
210	/* map VA 0xc0000000..0xc3ffffff to PA */
211	MMU_INIT(KERNBASE, PHYSADDR, 64, L1_TYPE_S|L1_S_C|L1_S_AP(AP_KRW))
212
213	.word 0	/* end of table */
214#endif
215.Lstart:
216	.word	_edata
217	.word	_end
218	.word	svcstk + INIT_ARM_STACK_SIZE
219
220.Lmainreturned:
221	.asciz	"main() returned"
222	.align	0
223
224	.bss
225svcstk:
226	.space	INIT_ARM_STACK_SIZE
227
228	.text
229	.align	0
230
231#ifndef OFW
232	/* OFW based systems will used OF_boot() */
233
234.Lcpufuncs:
235	.word	_C_LABEL(cpufuncs)
236
237ENTRY_NP(cpu_halt)
238	mrs     r2, cpsr
239	bic	r2, r2, #(PSR_MODE)
240	orr     r2, r2, #(PSR_SVC32_MODE)
241	orr	r2, r2, #(I32_bit | F32_bit)
242	msr     cpsr_all, r2
243
244	ldr	r4, .Lcpu_reset_address
245	ldr	r4, [r4]
246
247	ldr	r0, .Lcpufuncs
248	mov	lr, pc
249	ldr	pc, [r0, #CF_IDCACHE_WBINV_ALL]
250
251	/*
252	 * Load the cpu_reset_needs_v4_MMU_disable flag to determine if it's
253	 * necessary.
254	 */
255
256	ldr	r1, .Lcpu_reset_needs_v4_MMU_disable
257	ldr	r1, [r1]
258	cmp	r1, #0
259	mov	r2, #0
260
261	/*
262 	 * MMU & IDC off, 32 bit program & data space
263	 * Hurl ourselves into the ROM
264	 */
265	mov	r0, #(CPU_CONTROL_32BP_ENABLE | CPU_CONTROL_32BD_ENABLE)
266	mcr     15, 0, r0, c1, c0, 0
267	mcrne   15, 0, r2, c8, c7, 0 	/* nail I+D TLB on ARMv4 and greater */
268	mov     pc, r4
269
270	/*
271	 * _cpu_reset_address contains the address to branch to, to complete
272	 * the cpu reset after turning the MMU off
273	 * This variable is provided by the hardware specific code
274	 */
275.Lcpu_reset_address:
276	.word	_C_LABEL(cpu_reset_address)
277
278	/*
279	 * cpu_reset_needs_v4_MMU_disable contains a flag that signals if the
280	 * v4 MMU disable instruction needs executing... it is an illegal instruction
281	 * on f.e. ARM6/7 that locks up the computer in an endless illegal
282	 * instruction / data-abort / reset loop.
283	 */
284.Lcpu_reset_needs_v4_MMU_disable:
285	.word	_C_LABEL(cpu_reset_needs_v4_MMU_disable)
286
287#endif	/* OFW */
288
289#ifdef IPKDB
290/*
291 * Execute(inst, psr, args, sp)
292 *
293 * Execute INSTruction with PSR and ARGS[0] - ARGS[3] making
294 * available stack at SP for next undefined instruction trap.
295 *
296 * Move the instruction onto the stack and jump to it.
297 */
298ENTRY_NP(Execute)
299	mov	ip, sp
300	stmfd	sp!, {r2, r4-r7, fp, ip, lr, pc}
301	sub	fp, ip, #4
302	mov	ip, r3
303	ldr	r7, .Lreturn
304	stmfd	sp!, {r0, r7}
305	adr	r7, #.LExec
306	mov	r5, r1
307	mrs	r4, cpsr
308	ldmia	r2, {r0-r3}
309	mov	r6, sp
310	mov	sp, ip
311	msr	cpsr_all, r5
312	mov	pc, r6
313.LExec:
314	mrs	r5, cpsr
315/* XXX Cannot switch thus easily back from user mode */
316	msr	cpsr_all, r4
317	add	sp, r6, #8
318	ldmfd	sp!, {r6}
319	stmia	r6, {r0-r3}
320	mov	r0, r5
321	ldmdb	fp, {r4-r7, fp, sp, pc}
322.Lreturn:
323	mov	pc, r7
324#endif
325
326/*
327 * setjump + longjmp
328 */
329ENTRY(setjmp)
330	stmia	r0, {r4-r14}
331	mov	r0, #0x00000000
332	RET
333
334ENTRY(longjmp)
335	ldmia	r0, {r4-r14}
336	mov	r0, #0x00000001
337	RET
338
339	.data
340	.global _C_LABEL(esym)
341_C_LABEL(esym):	.word	_C_LABEL(end)
342
343ENTRY_NP(abort)
344	b	_C_LABEL(abort)
345
346ENTRY_NP(sigcode)
347	mov	r0, sp
348	swi	SYS_sigreturn
349
350	/* Well if that failed we better exit quick ! */
351
352	swi	SYS_exit
353	b	. - 8
354
355	.align	0
356	.global _C_LABEL(esigcode)
357		_C_LABEL(esigcode):
358
359	.data
360	.global szsigcode
361szsigcode:
362	.long esigcode-sigcode
363/* End of locore.S */
364