]> git.kernelconcepts.de Git - karo-tx-uboot.git/blob - arch/mips/cpu/mips64/start.S
drivers/i2c: Update fti2c010.[ch], i2c_core.c to use SPDX identifiers
[karo-tx-uboot.git] / arch / mips / cpu / mips64 / start.S
1 /*
2  *  Startup Code for MIPS64 CPU-core
3  *
4  *  Copyright (c) 2003  Wolfgang Denk <wd@denx.de>
5  *
6   * SPDX-License-Identifier:    GPL-2.0+
7  */
8
9 #include <asm-offsets.h>
10 #include <config.h>
11 #include <asm/regdef.h>
12 #include <asm/mipsregs.h>
13
14 #ifndef CONFIG_SYS_MIPS_CACHE_MODE
15 #define CONFIG_SYS_MIPS_CACHE_MODE CONF_CM_CACHABLE_NONCOHERENT
16 #endif
17
18 #ifdef CONFIG_SYS_LITTLE_ENDIAN
19 #define MIPS64_R_INFO(ssym, r_type3, r_type2, r_type) \
20         (((r_type) << 24) | ((r_type2) << 16) | ((r_type3) << 8) | (ssym))
21 #else
22 #define MIPS64_R_INFO(ssym, r_type3, r_type2, r_type) \
23         ((r_type) | ((r_type2) << 8) | ((r_type3) << 16) | (ssym) << 24)
24 #endif
25
26         /*
27          * For the moment disable interrupts, mark the kernel mode and
28          * set ST0_KX so that the CPU does not spit fire when using
29          * 64-bit addresses.
30          */
31         .macro  setup_c0_status set clr
32         .set    push
33         mfc0    t0, CP0_STATUS
34         or      t0, ST0_CU0 | \set | 0x1f | \clr
35         xor     t0, 0x1f | \clr
36         mtc0    t0, CP0_STATUS
37         .set    noreorder
38         sll     zero, 3                         # ehb
39         .set    pop
40         .endm
41
42         .set noreorder
43
44         .globl _start
45         .text
46 _start:
47         /* U-boot entry point */
48         b       reset
49          nop
50
51         .org 0x200
52         /* TLB refill, 32 bit task */
53 1:      b       1b
54          nop
55
56         .org 0x280
57         /* XTLB refill, 64 bit task */
58 1:      b       1b
59          nop
60
61         .org 0x300
62         /* Cache error exception */
63 1:      b       1b
64          nop
65
66         .org 0x380
67         /* General exception */
68 1:      b       1b
69          nop
70
71         .org 0x400
72         /* Catch interrupt exceptions */
73 1:      b       1b
74          nop
75
76         .org 0x480
77         /* EJTAG debug exception */
78 1:      b       1b
79          nop
80
81         .align 4
82 reset:
83
84         /* Clear watch registers */
85         dmtc0   zero, CP0_WATCHLO
86         dmtc0   zero, CP0_WATCHHI
87
88         /* WP(Watch Pending), SW0/1 should be cleared */
89         mtc0    zero, CP0_CAUSE
90
91         setup_c0_status ST0_KX 0
92
93         /* Init Timer */
94         mtc0    zero, CP0_COUNT
95         mtc0    zero, CP0_COMPARE
96
97 #ifndef CONFIG_SKIP_LOWLEVEL_INIT
98         /* CONFIG0 register */
99         dli     t0, CONF_CM_UNCACHED
100         mtc0    t0, CP0_CONFIG
101 #endif
102
103         /*
104          * Initialize $gp, force 8 byte alignment of bal instruction to forbid
105          * the compiler to put nop's between bal and _gp. This is required to
106          * keep _gp and ra aligned to 8 byte.
107          */
108         .align  3
109         bal     1f
110          nop
111         .dword  _gp
112 1:
113         ld      gp, 0(ra)
114
115 #ifndef CONFIG_SKIP_LOWLEVEL_INIT
116         /* Initialize any external memory */
117         dla     t9, lowlevel_init
118         jalr    t9
119          nop
120
121         /* Initialize caches... */
122         dla     t9, mips_cache_reset
123         jalr    t9
124          nop
125
126         /* ... and enable them */
127         dli     t0, CONFIG_SYS_MIPS_CACHE_MODE
128         mtc0    t0, CP0_CONFIG
129 #endif
130
131         /* Set up temporary stack */
132         dli     sp, CONFIG_SYS_SDRAM_BASE + CONFIG_SYS_INIT_SP_OFFSET
133
134         dla     t9, board_init_f
135         jr      t9
136          nop
137
138 /*
139  * void relocate_code (addr_sp, gd, addr_moni)
140  *
141  * This "function" does not return, instead it continues in RAM
142  * after relocating the monitor code.
143  *
144  * a0 = addr_sp
145  * a1 = gd
146  * a2 = destination address
147  */
148         .globl  relocate_code
149         .ent    relocate_code
150 relocate_code:
151         move    sp, a0                  # set new stack pointer
152
153         move    s0, a1                  # save gd in s0
154         move    s2, a2                  # save destination address in s2
155
156         dli     t0, CONFIG_SYS_MONITOR_BASE
157         dsub    s1, s2, t0              # s1 <-- relocation offset
158
159         dla     t3, in_ram
160         ld      t2, -24(t3)             # t2 <-- __image_copy_end
161         move    t1, a2
162
163         dadd    gp, s1                  # adjust gp
164
165         /*
166          * t0 = source address
167          * t1 = target address
168          * t2 = source end address
169          */
170 1:
171         lw      t3, 0(t0)
172         sw      t3, 0(t1)
173         daddu   t0, 4
174         blt     t0, t2, 1b
175          daddu  t1, 4
176
177         /* If caches were enabled, we would have to flush them here. */
178         dsub    a1, t1, s2              # a1 <-- size
179         dla     t9, flush_cache
180         jalr    t9
181          move   a0, s2                  # a0 <-- destination address
182
183         /* Jump to where we've relocated ourselves */
184         daddi   t0, s2, in_ram - _start
185         jr      t0
186          nop
187
188         .dword  __rel_dyn_end
189         .dword  __rel_dyn_start
190         .dword  __image_copy_end
191         .dword  _GLOBAL_OFFSET_TABLE_
192         .dword  num_got_entries
193
194 in_ram:
195         /*
196          * Now we want to update GOT.
197          *
198          * GOT[0] is reserved. GOT[1] is also reserved for the dynamic object
199          * generated by GNU ld. Skip these reserved entries from relocation.
200          */
201         ld      t3, -8(t0)              # t3 <-- num_got_entries
202         ld      t8, -16(t0)             # t8 <-- _GLOBAL_OFFSET_TABLE_
203         dadd    t8, s1                  # t8 now holds relocated _G_O_T_
204         daddi   t8, t8, 16              # skipping first two entries
205         dli     t2, 2
206 1:
207         ld      t1, 0(t8)
208         beqz    t1, 2f
209          dadd   t1, s1
210         sd      t1, 0(t8)
211 2:
212         daddi   t2, 1
213         blt     t2, t3, 1b
214          daddi  t8, 8
215
216         /* Update dynamic relocations */
217         ld      t1, -32(t0)             # t1 <-- __rel_dyn_start
218         ld      t2, -40(t0)             # t2 <-- __rel_dyn_end
219
220         b       2f                      # skip first reserved entry
221          daddi  t1, 16
222
223 1:
224         lw      t8, -4(t1)              # t8 <-- relocation info
225
226         dli     t3, MIPS64_R_INFO(0x00, 0x00, 0x12, 0x03)
227         bne     t8, t3, 2f              # skip non R_MIPS_REL32 entries
228          nop
229
230         ld      t3, -16(t1)             # t3 <-- location to fix up in FLASH
231
232         ld      t8, 0(t3)               # t8 <-- original pointer
233         dadd    t8, s1                  # t8 <-- adjusted pointer
234
235         dadd    t3, s1                  # t3 <-- location to fix up in RAM
236         sd      t8, 0(t3)
237
238 2:
239         blt     t1, t2, 1b
240          daddi  t1, 16                  # each rel.dyn entry is 16 bytes
241
242         /*
243          * Clear BSS
244          *
245          * GOT is now relocated. Thus __bss_start and __bss_end can be
246          * accessed directly via $gp.
247          */
248         dla     t1, __bss_start         # t1 <-- __bss_start
249         dla     t2, __bss_end           # t2 <-- __bss_end
250
251 1:
252         sd      zero, 0(t1)
253         blt     t1, t2, 1b
254          daddi  t1, 8
255
256         move    a0, s0                  # a0 <-- gd
257         dla     t9, board_init_r
258         jr      t9
259          move   a1, s2
260
261         .end    relocate_code