blob: 57db589b94f266e7057ca966a72b6487918cd186 [file] [log] [blame]
wdenkc0218802003-03-27 12:09:35 +00001/*
2 * Startup Code for MIPS32 CPU-core
3 *
4 * Copyright (c) 2003 Wolfgang Denk <wd@denx.de>
5 *
6 * See file CREDITS for list of people who contributed to this
7 * project.
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License as
11 * published by the Free Software Foundation; either version 2 of
12 * the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
22 * MA 02111-1307 USA
23 */
24
wdenkc0218802003-03-27 12:09:35 +000025#include <config.h>
wdenkc0218802003-03-27 12:09:35 +000026#include <asm/regdef.h>
27#include <asm/mipsregs.h>
28
Shinya Kuribayashidecaba62008-03-25 21:30:07 +090029 /*
30 * For the moment disable interrupts, mark the kernel mode and
31 * set ST0_KX so that the CPU does not spit fire when using
32 * 64-bit addresses.
33 */
34 .macro setup_c0_status set clr
35 .set push
36 mfc0 t0, CP0_STATUS
37 or t0, ST0_CU0 | \set | 0x1f | \clr
38 xor t0, 0x1f | \clr
39 mtc0 t0, CP0_STATUS
40 .set noreorder
41 sll zero, 3 # ehb
42 .set pop
43 .endm
44
45 .macro setup_c0_status_reset
46#ifdef CONFIG_64BIT
47 setup_c0_status ST0_KX 0
48#else
49 setup_c0_status 0 0
50#endif
51 .endm
52
wdenkc0218802003-03-27 12:09:35 +000053#define RVECENT(f,n) \
54 b f; nop
55#define XVECENT(f,bev) \
56 b f ; \
57 li k0,bev
58
59 .set noreorder
60
61 .globl _start
62 .text
63_start:
64 RVECENT(reset,0) /* U-boot entry point */
65 RVECENT(reset,1) /* software reboot */
wdenk3e386912003-04-05 00:53:31 +000066#if defined(CONFIG_INCA_IP)
67 .word INFINEON_EBU_BOOTCFG /* EBU init code, fetched during booting */
68 .word 0x00000000 /* phase of the flash */
69#elif defined(CONFIG_PURPLE)
70 .word INFINEON_EBU_BOOTCFG /* EBU init code, fetched during booting */
71 .word INFINEON_EBU_BOOTCFG /* EBU init code, fetched during booting */
wdenkc0218802003-03-27 12:09:35 +000072#else
73 RVECENT(romReserved,2)
74#endif
75 RVECENT(romReserved,3)
76 RVECENT(romReserved,4)
77 RVECENT(romReserved,5)
78 RVECENT(romReserved,6)
79 RVECENT(romReserved,7)
80 RVECENT(romReserved,8)
81 RVECENT(romReserved,9)
82 RVECENT(romReserved,10)
83 RVECENT(romReserved,11)
84 RVECENT(romReserved,12)
85 RVECENT(romReserved,13)
86 RVECENT(romReserved,14)
87 RVECENT(romReserved,15)
88 RVECENT(romReserved,16)
wdenk8bde7f72003-06-27 21:31:46 +000089 RVECENT(romReserved,17)
wdenkc0218802003-03-27 12:09:35 +000090 RVECENT(romReserved,18)
91 RVECENT(romReserved,19)
92 RVECENT(romReserved,20)
93 RVECENT(romReserved,21)
94 RVECENT(romReserved,22)
95 RVECENT(romReserved,23)
96 RVECENT(romReserved,24)
97 RVECENT(romReserved,25)
98 RVECENT(romReserved,26)
99 RVECENT(romReserved,27)
100 RVECENT(romReserved,28)
101 RVECENT(romReserved,29)
102 RVECENT(romReserved,30)
103 RVECENT(romReserved,31)
104 RVECENT(romReserved,32)
105 RVECENT(romReserved,33)
106 RVECENT(romReserved,34)
107 RVECENT(romReserved,35)
108 RVECENT(romReserved,36)
109 RVECENT(romReserved,37)
110 RVECENT(romReserved,38)
111 RVECENT(romReserved,39)
112 RVECENT(romReserved,40)
113 RVECENT(romReserved,41)
114 RVECENT(romReserved,42)
115 RVECENT(romReserved,43)
116 RVECENT(romReserved,44)
117 RVECENT(romReserved,45)
118 RVECENT(romReserved,46)
119 RVECENT(romReserved,47)
120 RVECENT(romReserved,48)
121 RVECENT(romReserved,49)
122 RVECENT(romReserved,50)
123 RVECENT(romReserved,51)
124 RVECENT(romReserved,52)
125 RVECENT(romReserved,53)
126 RVECENT(romReserved,54)
127 RVECENT(romReserved,55)
128 RVECENT(romReserved,56)
129 RVECENT(romReserved,57)
130 RVECENT(romReserved,58)
131 RVECENT(romReserved,59)
132 RVECENT(romReserved,60)
133 RVECENT(romReserved,61)
134 RVECENT(romReserved,62)
wdenk8bde7f72003-06-27 21:31:46 +0000135 RVECENT(romReserved,63)
wdenkc0218802003-03-27 12:09:35 +0000136 XVECENT(romExcHandle,0x200) /* bfc00200: R4000 tlbmiss vector */
137 RVECENT(romReserved,65)
138 RVECENT(romReserved,66)
139 RVECENT(romReserved,67)
140 RVECENT(romReserved,68)
141 RVECENT(romReserved,69)
142 RVECENT(romReserved,70)
143 RVECENT(romReserved,71)
144 RVECENT(romReserved,72)
145 RVECENT(romReserved,73)
146 RVECENT(romReserved,74)
147 RVECENT(romReserved,75)
148 RVECENT(romReserved,76)
149 RVECENT(romReserved,77)
150 RVECENT(romReserved,78)
wdenk8bde7f72003-06-27 21:31:46 +0000151 RVECENT(romReserved,79)
wdenkc0218802003-03-27 12:09:35 +0000152 XVECENT(romExcHandle,0x280) /* bfc00280: R4000 xtlbmiss vector */
153 RVECENT(romReserved,81)
154 RVECENT(romReserved,82)
155 RVECENT(romReserved,83)
156 RVECENT(romReserved,84)
157 RVECENT(romReserved,85)
158 RVECENT(romReserved,86)
159 RVECENT(romReserved,87)
160 RVECENT(romReserved,88)
161 RVECENT(romReserved,89)
162 RVECENT(romReserved,90)
163 RVECENT(romReserved,91)
164 RVECENT(romReserved,92)
165 RVECENT(romReserved,93)
166 RVECENT(romReserved,94)
wdenk8bde7f72003-06-27 21:31:46 +0000167 RVECENT(romReserved,95)
wdenkc0218802003-03-27 12:09:35 +0000168 XVECENT(romExcHandle,0x300) /* bfc00300: R4000 cache vector */
169 RVECENT(romReserved,97)
170 RVECENT(romReserved,98)
171 RVECENT(romReserved,99)
172 RVECENT(romReserved,100)
173 RVECENT(romReserved,101)
174 RVECENT(romReserved,102)
175 RVECENT(romReserved,103)
176 RVECENT(romReserved,104)
177 RVECENT(romReserved,105)
178 RVECENT(romReserved,106)
179 RVECENT(romReserved,107)
180 RVECENT(romReserved,108)
181 RVECENT(romReserved,109)
182 RVECENT(romReserved,110)
183 RVECENT(romReserved,111)
184 XVECENT(romExcHandle,0x380) /* bfc00380: R4000 general vector */
185 RVECENT(romReserved,113)
186 RVECENT(romReserved,114)
187 RVECENT(romReserved,115)
188 RVECENT(romReserved,116)
189 RVECENT(romReserved,116)
190 RVECENT(romReserved,118)
191 RVECENT(romReserved,119)
192 RVECENT(romReserved,120)
193 RVECENT(romReserved,121)
194 RVECENT(romReserved,122)
195 RVECENT(romReserved,123)
196 RVECENT(romReserved,124)
197 RVECENT(romReserved,125)
198 RVECENT(romReserved,126)
199 RVECENT(romReserved,127)
wdenk8bde7f72003-06-27 21:31:46 +0000200
wdenkc0218802003-03-27 12:09:35 +0000201 /* We hope there are no more reserved vectors!
202 * 128 * 8 == 1024 == 0x400
203 * so this is address R_VEC+0x400 == 0xbfc00400
204 */
wdenk3e386912003-04-05 00:53:31 +0000205#ifdef CONFIG_PURPLE
206/* 0xbfc00400 */
207 .word 0xdc870000
208 .word 0xfca70000
209 .word 0x20840008
210 .word 0x20a50008
211 .word 0x20c6ffff
212 .word 0x14c0fffa
213 .word 0x00000000
214 .word 0x03e00008
215 .word 0x00000000
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900216 .word 0x00000000
wdenk3e386912003-04-05 00:53:31 +0000217/* 0xbfc00428 */
218 .word 0xdc870000
219 .word 0xfca70000
220 .word 0x20840008
221 .word 0x20a50008
222 .word 0x20c6ffff
223 .word 0x14c0fffa
224 .word 0x00000000
225 .word 0x03e00008
226 .word 0x00000000
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900227 .word 0x00000000
wdenk3e386912003-04-05 00:53:31 +0000228#endif /* CONFIG_PURPLE */
wdenkc0218802003-03-27 12:09:35 +0000229 .align 4
230reset:
231
232 /* Clear watch registers.
233 */
234 mtc0 zero, CP0_WATCHLO
235 mtc0 zero, CP0_WATCHHI
236
Shinya Kuribayashid43d43e2008-03-25 21:30:07 +0900237 /* WP(Watch Pending), SW0/1 should be cleared. */
238 mtc0 zero, CP0_CAUSE
239
Shinya Kuribayashidecaba62008-03-25 21:30:07 +0900240 setup_c0_status_reset
wdenkc0218802003-03-27 12:09:35 +0000241
wdenkc0218802003-03-27 12:09:35 +0000242 /* Init Timer */
243 mtc0 zero, CP0_COUNT
244 mtc0 zero, CP0_COMPARE
245
Stefan Roese44174342008-11-18 16:36:22 +0100246#if !defined(CONFIG_SKIP_LOWLEVEL_INIT)
wdenkc0218802003-03-27 12:09:35 +0000247 /* CONFIG0 register */
248 li t0, CONF_CM_UNCACHED
249 mtc0 t0, CP0_CONFIG
Stefan Roese44174342008-11-18 16:36:22 +0100250#endif /* !CONFIG_SKIP_LOWLEVEL_INIT */
wdenkc0218802003-03-27 12:09:35 +0000251
Shinya Kuribayashi22069212007-10-21 10:55:36 +0900252 /* Initialize $gp.
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900253 */
254 bal 1f
Wolfgang Denkc75eba32005-12-01 02:15:07 +0100255 nop
Shinya Kuribayashi22069212007-10-21 10:55:36 +0900256 .word _gp
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +09002571:
Shinya Kuribayashi16664f72007-11-17 20:05:26 +0900258 lw gp, 0(ra)
Wolfgang Denkc75eba32005-12-01 02:15:07 +0100259
Stefan Roese44174342008-11-18 16:36:22 +0100260#if !defined(CONFIG_SKIP_LOWLEVEL_INIT)
wdenk8bde7f72003-06-27 21:31:46 +0000261 /* Initialize any external memory.
wdenkc0218802003-03-27 12:09:35 +0000262 */
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900263 la t9, lowlevel_init
264 jalr t9
wdenkc0218802003-03-27 12:09:35 +0000265 nop
266
267 /* Initialize caches...
268 */
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900269 la t9, mips_cache_reset
270 jalr t9
wdenkc0218802003-03-27 12:09:35 +0000271 nop
272
273 /* ... and enable them.
274 */
275 li t0, CONF_CM_CACHABLE_NONCOHERENT
276 mtc0 t0, CP0_CONFIG
Stefan Roese44174342008-11-18 16:36:22 +0100277#endif /* !CONFIG_SKIP_LOWLEVEL_INIT */
wdenkc0218802003-03-27 12:09:35 +0000278
wdenkc0218802003-03-27 12:09:35 +0000279 /* Set up temporary stack.
280 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200281#ifdef CONFIG_SYS_INIT_RAM_LOCK_MIPS
282 li a0, CONFIG_SYS_INIT_SP_OFFSET
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900283 la t9, mips_cache_lock
284 jalr t9
wdenkc0218802003-03-27 12:09:35 +0000285 nop
Shinya Kuribayashie1390802008-03-25 11:39:29 +0900286#endif
wdenkc0218802003-03-27 12:09:35 +0000287
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200288 li t0, CONFIG_SYS_SDRAM_BASE + CONFIG_SYS_INIT_SP_OFFSET
wdenkc0218802003-03-27 12:09:35 +0000289 la sp, 0(t0)
290
wdenkc0218802003-03-27 12:09:35 +0000291 la t9, board_init_f
Shinya Kuribayashi43c50922008-04-17 23:35:13 +0900292 jr t9
wdenkc0218802003-03-27 12:09:35 +0000293 nop
294
wdenkc0218802003-03-27 12:09:35 +0000295/*
296 * void relocate_code (addr_sp, gd, addr_moni)
297 *
298 * This "function" does not return, instead it continues in RAM
299 * after relocating the monitor code.
300 *
301 * a0 = addr_sp
302 * a1 = gd
303 * a2 = destination address
304 */
305 .globl relocate_code
306 .ent relocate_code
307relocate_code:
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900308 move sp, a0 /* Set new stack pointer */
wdenkc0218802003-03-27 12:09:35 +0000309
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200310 li t0, CONFIG_SYS_MONITOR_BASE
wdenk27b207f2003-07-24 23:38:38 +0000311 la t3, in_ram
312 lw t2, -12(t3) /* t2 <-- uboot_end_data */
313 move t1, a2
Stefan Roese71fa0712008-11-18 16:36:12 +0100314 move s2, a2 /* s2 <-- destination address */
wdenk27b207f2003-07-24 23:38:38 +0000315
wdenkc0218802003-03-27 12:09:35 +0000316 /*
Shinya Kuribayashi22069212007-10-21 10:55:36 +0900317 * Fix $gp:
wdenkc0218802003-03-27 12:09:35 +0000318 *
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200319 * New $gp = (Old $gp - CONFIG_SYS_MONITOR_BASE) + Destination Address
wdenkc0218802003-03-27 12:09:35 +0000320 */
321 move t6, gp
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200322 sub gp, CONFIG_SYS_MONITOR_BASE
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900323 add gp, a2 /* gp now adjusted */
Stefan Roese71fa0712008-11-18 16:36:12 +0100324 sub s1, gp, t6 /* s1 <-- relocation offset */
wdenk8bde7f72003-06-27 21:31:46 +0000325
wdenkc0218802003-03-27 12:09:35 +0000326 /*
327 * t0 = source address
328 * t1 = target address
329 * t2 = source end address
330 */
Stefan Roese71fa0712008-11-18 16:36:12 +0100331
332 /*
333 * Save destination address and size for later usage in flush_cache()
334 */
335 move s0, a1 /* save gd in s0 */
336 move a0, t1 /* a0 <-- destination addr */
337 sub a1, t2, t0 /* a1 <-- size */
338
wdenk3e386912003-04-05 00:53:31 +0000339 /* On the purple board we copy the code earlier in a special way
340 * in order to solve flash problems
341 */
342#ifndef CONFIG_PURPLE
wdenkc0218802003-03-27 12:09:35 +00003431:
344 lw t3, 0(t0)
345 sw t3, 0(t1)
346 addu t0, 4
347 ble t0, t2, 1b
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900348 addu t1, 4 /* delay slot */
wdenk3e386912003-04-05 00:53:31 +0000349#endif
wdenkc0218802003-03-27 12:09:35 +0000350
351 /* If caches were enabled, we would have to flush them here.
352 */
353
Stefan Roese71fa0712008-11-18 16:36:12 +0100354 /* a0 & a1 are already set up for flush_cache(start, size) */
355 la t9, flush_cache
356 jalr t9
357 nop
358
wdenkc0218802003-03-27 12:09:35 +0000359 /* Jump to where we've relocated ourselves.
360 */
Stefan Roese71fa0712008-11-18 16:36:12 +0100361 addi t0, s2, in_ram - _start
Shinya Kuribayashi43c50922008-04-17 23:35:13 +0900362 jr t0
wdenkc0218802003-03-27 12:09:35 +0000363 nop
364
Wolfgang Denk53677ef2008-05-20 16:00:29 +0200365 .word _gp
Vlad Lungu0f8c62a2008-05-05 14:04:00 +0300366 .word _GLOBAL_OFFSET_TABLE_
wdenkc0218802003-03-27 12:09:35 +0000367 .word uboot_end_data
368 .word uboot_end
369 .word num_got_entries
370
371in_ram:
Shinya Kuribayashi22069212007-10-21 10:55:36 +0900372 /*
373 * Now we want to update GOT.
374 *
375 * GOT[0] is reserved. GOT[1] is also reserved for the dynamic object
376 * generated by GNU ld. Skip these reserved entries from relocation.
wdenkc0218802003-03-27 12:09:35 +0000377 */
378 lw t3, -4(t0) /* t3 <-- num_got_entries */
Vlad Lungu0f8c62a2008-05-05 14:04:00 +0300379 lw t4, -16(t0) /* t4 <-- _GLOBAL_OFFSET_TABLE_ */
380 lw t5, -20(t0) /* t5 <-- _gp */
381 sub t4, t5 /* compute offset*/
382 add t4, t4, gp /* t4 now holds relocated _GLOBAL_OFFSET_TABLE_ */
Shinya Kuribayashi22069212007-10-21 10:55:36 +0900383 addi t4, t4, 8 /* Skipping first two entries. */
wdenkc0218802003-03-27 12:09:35 +0000384 li t2, 2
3851:
386 lw t1, 0(t4)
387 beqz t1, 2f
Stefan Roese71fa0712008-11-18 16:36:12 +0100388 add t1, s1
wdenkc0218802003-03-27 12:09:35 +0000389 sw t1, 0(t4)
3902:
391 addi t2, 1
392 blt t2, t3, 1b
393 addi t4, 4 /* delay slot */
394
395 /* Clear BSS.
396 */
397 lw t1, -12(t0) /* t1 <-- uboot_end_data */
398 lw t2, -8(t0) /* t2 <-- uboot_end */
Stefan Roese71fa0712008-11-18 16:36:12 +0100399 add t1, s1 /* adjust pointers */
400 add t2, s1
wdenkc0218802003-03-27 12:09:35 +0000401
402 sub t1, 4
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +09004031:
404 addi t1, 4
wdenkc0218802003-03-27 12:09:35 +0000405 bltl t1, t2, 1b
406 sw zero, 0(t1) /* delay slot */
wdenk8bde7f72003-06-27 21:31:46 +0000407
Stefan Roese71fa0712008-11-18 16:36:12 +0100408 move a0, s0 /* a0 <-- gd */
wdenkc0218802003-03-27 12:09:35 +0000409 la t9, board_init_r
Shinya Kuribayashi43c50922008-04-17 23:35:13 +0900410 jr t9
Stefan Roese71fa0712008-11-18 16:36:12 +0100411 move a1, s2 /* delay slot */
wdenkc0218802003-03-27 12:09:35 +0000412
413 .end relocate_code
wdenkc0218802003-03-27 12:09:35 +0000414
wdenkc0218802003-03-27 12:09:35 +0000415 /* Exception handlers.
416 */
417romReserved:
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900418 b romReserved
wdenkc0218802003-03-27 12:09:35 +0000419
420romExcHandle:
Shinya Kuribayashi03c031d2007-10-27 15:27:06 +0900421 b romExcHandle