Greg Kroah-Hartman | b244131 | 2017-11-01 15:07:57 +0100 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0 */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 2 | /*****************************************************************************/ |
| 3 | |
| 4 | /* |
| 5 | * head.S -- common startup code for ColdFire CPUs. |
| 6 | * |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 7 | * (C) Copyright 1999-2011, Greg Ungerer <gerg@snapgear.com>. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 8 | */ |
| 9 | |
| 10 | /*****************************************************************************/ |
| 11 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 12 | #include <linux/linkage.h> |
Greg Ungerer | 9b0e741 | 2008-07-11 15:29:36 +1000 | [diff] [blame] | 13 | #include <linux/init.h> |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 14 | #include <asm/asm-offsets.h> |
| 15 | #include <asm/coldfire.h> |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 16 | #include <asm/mcfsim.h> |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 17 | #include <asm/mcfmmu.h> |
David Howells | df9ee29 | 2010-10-07 14:08:55 +0100 | [diff] [blame] | 18 | #include <asm/thread_info.h> |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 19 | |
| 20 | /*****************************************************************************/ |
| 21 | |
| 22 | /* |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 23 | * If we don't have a fixed memory size, then lets build in code |
Lucas De Marchi | 25985ed | 2011-03-30 22:57:33 -0300 | [diff] [blame] | 24 | * to auto detect the DRAM size. Obviously this is the preferred |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 25 | * method, and should work for most boards. It won't work for those |
| 26 | * that do not have their RAM starting at address 0, and it only |
| 27 | * works on SDRAM (not boards fitted with SRAM). |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 28 | */ |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 29 | #if CONFIG_RAMSIZE != 0 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 30 | .macro GET_MEM_SIZE |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 31 | movel #CONFIG_RAMSIZE,%d0 /* hard coded memory size */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 32 | .endm |
| 33 | |
| 34 | #elif defined(CONFIG_M5206) || defined(CONFIG_M5206e) || \ |
Steven King | 04e037a | 2012-06-05 08:23:08 -0700 | [diff] [blame] | 35 | defined(CONFIG_M5249) || defined(CONFIG_M525x) || \ |
| 36 | defined(CONFIG_M527x) || defined(CONFIG_M528x) || \ |
| 37 | defined(CONFIG_M5307) || defined(CONFIG_M5407) |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 38 | /* |
| 39 | * Not all these devices have exactly the same DRAM controller, |
| 40 | * but the DCMR register is virtually identical - give or take |
| 41 | * a couple of bits. The only exception is the 5272 devices, their |
| 42 | * DRAM controller is quite different. |
| 43 | */ |
| 44 | .macro GET_MEM_SIZE |
Greg Ungerer | 6a92e19 | 2011-03-06 23:01:46 +1000 | [diff] [blame] | 45 | movel MCFSIM_DMR0,%d0 /* get mask for 1st bank */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 46 | btst #0,%d0 /* check if region enabled */ |
| 47 | beq 1f |
| 48 | andl #0xfffc0000,%d0 |
| 49 | beq 1f |
| 50 | addl #0x00040000,%d0 /* convert mask to size */ |
| 51 | 1: |
Greg Ungerer | 6a92e19 | 2011-03-06 23:01:46 +1000 | [diff] [blame] | 52 | movel MCFSIM_DMR1,%d1 /* get mask for 2nd bank */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 53 | btst #0,%d1 /* check if region enabled */ |
| 54 | beq 2f |
Greg Ungerer | 6a92e19 | 2011-03-06 23:01:46 +1000 | [diff] [blame] | 55 | andl #0xfffc0000,%d1 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 56 | beq 2f |
| 57 | addl #0x00040000,%d1 |
| 58 | addl %d1,%d0 /* total mem size in d0 */ |
| 59 | 2: |
| 60 | .endm |
| 61 | |
| 62 | #elif defined(CONFIG_M5272) |
| 63 | .macro GET_MEM_SIZE |
Greg Ungerer | 1419ea3 | 2012-09-14 15:36:02 +1000 | [diff] [blame] | 64 | movel MCFSIM_CSOR7,%d0 /* get SDRAM address mask */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 65 | andil #0xfffff000,%d0 /* mask out chip select options */ |
| 66 | negl %d0 /* negate bits */ |
| 67 | .endm |
| 68 | |
Michael Broughton | 5529840 | 2006-12-04 17:27:29 +1000 | [diff] [blame] | 69 | #elif defined(CONFIG_M520x) |
| 70 | .macro GET_MEM_SIZE |
| 71 | clrl %d0 |
Greg Ungerer | 571f060 | 2011-03-05 23:50:37 +1000 | [diff] [blame] | 72 | movel MCFSIM_SDCS0, %d2 /* Get SDRAM chip select 0 config */ |
Michael Broughton | 5529840 | 2006-12-04 17:27:29 +1000 | [diff] [blame] | 73 | andl #0x1f, %d2 /* Get only the chip select size */ |
| 74 | beq 3f /* Check if it is enabled */ |
| 75 | addql #1, %d2 /* Form exponent */ |
| 76 | moveql #1, %d0 |
| 77 | lsll %d2, %d0 /* 2 ^ exponent */ |
| 78 | 3: |
Greg Ungerer | 571f060 | 2011-03-05 23:50:37 +1000 | [diff] [blame] | 79 | movel MCFSIM_SDCS1, %d2 /* Get SDRAM chip select 1 config */ |
Michael Broughton | 5529840 | 2006-12-04 17:27:29 +1000 | [diff] [blame] | 80 | andl #0x1f, %d2 /* Get only the chip select size */ |
| 81 | beq 4f /* Check if it is enabled */ |
| 82 | addql #1, %d2 /* Form exponent */ |
| 83 | moveql #1, %d1 |
| 84 | lsll %d2, %d1 /* 2 ^ exponent */ |
| 85 | addl %d1, %d0 /* Total size of SDRAM in d0 */ |
| 86 | 4: |
| 87 | .endm |
| 88 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 89 | #else |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 90 | #error "ERROR: I don't know how to probe your boards memory size?" |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 91 | #endif |
| 92 | |
| 93 | /*****************************************************************************/ |
| 94 | |
| 95 | /* |
| 96 | * Boards and platforms can do specific early hardware setup if |
| 97 | * they need to. Most don't need this, define away if not required. |
| 98 | */ |
| 99 | #ifndef PLATFORM_SETUP |
| 100 | #define PLATFORM_SETUP |
| 101 | #endif |
| 102 | |
| 103 | /*****************************************************************************/ |
| 104 | |
| 105 | .global _start |
| 106 | .global _rambase |
| 107 | .global _ramvec |
| 108 | .global _ramstart |
| 109 | .global _ramend |
Lennart Sorensen | 588baea | 2009-09-18 13:49:36 -0400 | [diff] [blame] | 110 | #if defined(CONFIG_UBOOT) |
| 111 | .global _init_sp |
| 112 | #endif |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 113 | |
| 114 | /*****************************************************************************/ |
| 115 | |
| 116 | .data |
| 117 | |
| 118 | /* |
| 119 | * During startup we store away the RAM setup. These are not in the |
| 120 | * bss, since their values are determined and written before the bss |
| 121 | * has been cleared. |
| 122 | */ |
| 123 | _rambase: |
| 124 | .long 0 |
| 125 | _ramvec: |
| 126 | .long 0 |
| 127 | _ramstart: |
| 128 | .long 0 |
| 129 | _ramend: |
| 130 | .long 0 |
Lennart Sorensen | 588baea | 2009-09-18 13:49:36 -0400 | [diff] [blame] | 131 | #if defined(CONFIG_UBOOT) |
| 132 | _init_sp: |
| 133 | .long 0 |
| 134 | #endif |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 135 | |
| 136 | /*****************************************************************************/ |
| 137 | |
Greg Ungerer | 9b0e741 | 2008-07-11 15:29:36 +1000 | [diff] [blame] | 138 | __HEAD |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 139 | |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 140 | #ifdef CONFIG_MMU |
| 141 | _start0: |
| 142 | jmp _start |
| 143 | .global kernel_pg_dir |
| 144 | .equ kernel_pg_dir,_start0 |
| 145 | .equ .,_start0+0x1000 |
| 146 | #endif |
| 147 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 148 | /* |
| 149 | * This is the codes first entry point. This is where it all |
| 150 | * begins... |
| 151 | */ |
| 152 | |
| 153 | _start: |
| 154 | nop /* filler */ |
| 155 | movew #0x2700, %sr /* no interrupts */ |
Greg Ungerer | ec84118 | 2011-11-15 15:26:28 +1000 | [diff] [blame] | 156 | movel #CACHE_INIT,%d0 /* disable cache */ |
| 157 | movec %d0,%CACR |
| 158 | nop |
Lennart Sorensen | 588baea | 2009-09-18 13:49:36 -0400 | [diff] [blame] | 159 | #if defined(CONFIG_UBOOT) |
| 160 | movel %sp,_init_sp /* save initial stack pointer */ |
| 161 | #endif |
Greg Ungerer | 10cb54d | 2011-11-23 14:46:36 +1000 | [diff] [blame] | 162 | #ifdef CONFIG_MBAR |
| 163 | movel #CONFIG_MBAR+1,%d0 /* configured MBAR address */ |
| 164 | movec %d0,%MBAR /* set it */ |
| 165 | #endif |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 166 | |
| 167 | /* |
| 168 | * Do any platform or board specific setup now. Most boards |
| 169 | * don't need anything. Those exceptions are define this in |
| 170 | * their board specific includes. |
| 171 | */ |
| 172 | PLATFORM_SETUP |
| 173 | |
| 174 | /* |
| 175 | * Create basic memory configuration. Set VBR accordingly, |
| 176 | * and size memory. |
| 177 | */ |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 178 | movel #CONFIG_VECTORBASE,%a7 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 179 | movec %a7,%VBR /* set vectors addr */ |
| 180 | movel %a7,_ramvec |
| 181 | |
Greg Ungerer | 12ddae3 | 2006-06-26 16:33:09 +1000 | [diff] [blame] | 182 | movel #CONFIG_RAMBASE,%a7 /* mark the base of RAM */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 183 | movel %a7,_rambase |
| 184 | |
| 185 | GET_MEM_SIZE /* macro code determines size */ |
Greg Ungerer | 029fc13 | 2005-09-02 10:42:52 +1000 | [diff] [blame] | 186 | addl %a7,%d0 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 187 | movel %d0,_ramend /* set end ram addr */ |
| 188 | |
| 189 | /* |
| 190 | * Now that we know what the memory is, lets enable cache |
Greg Ungerer | 8ce877a | 2010-11-09 13:35:55 +1000 | [diff] [blame] | 191 | * and get things moving. This is Coldfire CPU specific. Not |
| 192 | * all version cores have identical cache register setup. But |
| 193 | * it is very similar. Define the exact settings in the headers |
| 194 | * then the code here is the same for all. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 195 | */ |
Greg Ungerer | 8ce877a | 2010-11-09 13:35:55 +1000 | [diff] [blame] | 196 | movel #ACR0_MODE,%d0 /* set RAM region for caching */ |
| 197 | movec %d0,%ACR0 |
| 198 | movel #ACR1_MODE,%d0 /* anything else to cache? */ |
| 199 | movec %d0,%ACR1 |
| 200 | #ifdef ACR2_MODE |
| 201 | movel #ACR2_MODE,%d0 |
| 202 | movec %d0,%ACR2 |
| 203 | movel #ACR3_MODE,%d0 |
| 204 | movec %d0,%ACR3 |
| 205 | #endif |
| 206 | movel #CACHE_MODE,%d0 /* enable cache */ |
| 207 | movec %d0,%CACR |
| 208 | nop |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 209 | |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 210 | #ifdef CONFIG_MMU |
| 211 | /* |
| 212 | * Identity mapping for the kernel region. |
| 213 | */ |
| 214 | movel #(MMUBASE+1),%d0 /* enable MMUBAR registers */ |
| 215 | movec %d0,%MMUBAR |
| 216 | movel #MMUOR_CA,%d0 /* clear TLB entries */ |
| 217 | movel %d0,MMUOR |
| 218 | movel #0,%d0 /* set ASID to 0 */ |
| 219 | movec %d0,%asid |
| 220 | |
| 221 | movel #MMUCR_EN,%d0 /* Enable the identity map */ |
| 222 | movel %d0,MMUCR |
| 223 | nop /* sync i-pipeline */ |
| 224 | |
| 225 | movel #_vstart,%a0 /* jump to "virtual" space */ |
| 226 | jmp %a0@ |
| 227 | _vstart: |
| 228 | #endif /* CONFIG_MMU */ |
| 229 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 230 | #ifdef CONFIG_ROMFS_FS |
| 231 | /* |
| 232 | * Move ROM filesystem above bss :-) |
| 233 | */ |
Geert Uytterhoeven | dc06105 | 2012-05-31 21:46:03 +0200 | [diff] [blame] | 234 | lea __bss_start,%a0 /* get start of bss */ |
| 235 | lea __bss_stop,%a1 /* set up destination */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 236 | movel %a0,%a2 /* copy of bss start */ |
| 237 | |
| 238 | movel 8(%a0),%d0 /* get size of ROMFS */ |
| 239 | addql #8,%d0 /* allow for rounding */ |
| 240 | andl #0xfffffffc, %d0 /* whole words */ |
| 241 | |
| 242 | addl %d0,%a0 /* copy from end */ |
| 243 | addl %d0,%a1 /* copy from end */ |
| 244 | movel %a1,_ramstart /* set start of ram */ |
| 245 | |
| 246 | _copy_romfs: |
| 247 | movel -(%a0),%d0 /* copy dword */ |
| 248 | movel %d0,-(%a1) |
| 249 | cmpl %a0,%a2 /* check if at end */ |
| 250 | bne _copy_romfs |
| 251 | |
| 252 | #else /* CONFIG_ROMFS_FS */ |
Geert Uytterhoeven | dc06105 | 2012-05-31 21:46:03 +0200 | [diff] [blame] | 253 | lea __bss_stop,%a1 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 254 | movel %a1,_ramstart |
| 255 | #endif /* CONFIG_ROMFS_FS */ |
| 256 | |
| 257 | |
| 258 | /* |
| 259 | * Zero out the bss region. |
| 260 | */ |
Geert Uytterhoeven | dc06105 | 2012-05-31 21:46:03 +0200 | [diff] [blame] | 261 | lea __bss_start,%a0 /* get start of bss */ |
| 262 | lea __bss_stop,%a1 /* get end of bss */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 263 | clrl %d0 /* set value */ |
| 264 | _clear_bss: |
| 265 | movel %d0,(%a0)+ /* clear each word */ |
| 266 | cmpl %a0,%a1 /* check if at end */ |
| 267 | bne _clear_bss |
| 268 | |
| 269 | /* |
| 270 | * Load the current task pointer and stack. |
| 271 | */ |
| 272 | lea init_thread_union,%a0 |
| 273 | lea THREAD_SIZE(%a0),%sp |
| 274 | |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 275 | #ifdef CONFIG_MMU |
| 276 | .global m68k_cputype |
| 277 | .global m68k_mmutype |
| 278 | .global m68k_fputype |
| 279 | .global m68k_machtype |
| 280 | movel #CPU_COLDFIRE,%d0 |
| 281 | movel %d0,m68k_cputype /* Mark us as a ColdFire */ |
| 282 | movel #MMU_COLDFIRE,%d0 |
| 283 | movel %d0,m68k_mmutype |
Greg Ungerer | cbd5b98 | 2016-08-25 23:12:22 +1000 | [diff] [blame] | 284 | movel #FPUTYPE,%d0 |
| 285 | movel %d0,m68k_fputype /* Mark FPU type */ |
Greg Ungerer | 81d3335 | 2016-08-25 23:10:59 +1000 | [diff] [blame] | 286 | movel #MACHINE,%d0 |
| 287 | movel %d0,m68k_machtype /* Mark machine type */ |
Greg Ungerer | 0079fe7 | 2011-10-14 15:35:37 +1000 | [diff] [blame] | 288 | lea init_task,%a2 /* Set "current" init task */ |
| 289 | #endif |
| 290 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 291 | /* |
Andrea Gelmini | 86a8280 | 2016-05-21 13:57:20 +0200 | [diff] [blame] | 292 | * Assembler start up done, start code proper. |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 293 | */ |
| 294 | jsr start_kernel /* start Linux kernel */ |
| 295 | |
| 296 | _exit: |
| 297 | jmp _exit /* should never get here */ |
| 298 | |
| 299 | /*****************************************************************************/ |