start_gcc.S 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530
  1. /*
  2. * Copyright (c) 2006-2019, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2010-05-17 swkyer first version
  9. * 2010-09-04 bernard porting to JZ47xx
  10. * 2019-07-19 Zhou Yanjie clean up code
  11. */
  12. #ifndef __ASSEMBLY__
  13. #define __ASSEMBLY__
  14. #endif
  15. #include "../common/mips_def.h"
  16. #include "../common/stackframe.h"
  17. #include "sdram_cfg.h"
  18. #include "cache.h"
  19. #include "rtconfig.h"
  20. #define SR_BOOT_EXC_VEC 0x00400000
  21. /* config pll div for cpu and sdram */
  22. #define PLL_MULT (0x54) // 晶振为24Mhz时,PLL=504Mhz
  23. #define SDRAM_DIV (0) // SDRAM为CPU的2分频
  24. #define CPU_DIV (2) // CPU为PLL的2分频
  25. // 配置内存大小
  26. #define MEM_SIZE (0x02000000) // 32MByte
  27. /* Delay macro */
  28. #define DELAY(count) \
  29. li v0, count; \
  30. 99: \
  31. bnez v0, 99b;\
  32. addiu v0, -1
  33. #define msize s2
  34. #define output_en s3
  35. .section ".start", "ax"
  36. .set noreorder
  37. /* the program entry */
  38. .globl _start
  39. _start:
  40. .set noreorder
  41. la ra, _start
  42. #if !defined(RT_USING_SELF_BOOT)
  43. /* disable interrupt */
  44. mfc0 t0, CP0_STATUS
  45. and t0, 0xfffffffe # By default it will be disabled.
  46. mtc0 t0, CP0_STATUS # Set CPU to disable interrupt.
  47. nop
  48. /* disable cache */
  49. mfc0 t0, CP0_CONFIG
  50. and t0, 0xfffffff8
  51. or t0, 0x2 # disable,!default value is not it!
  52. mtc0 t0, CP0_CONFIG # Set CPU to disable cache.
  53. nop
  54. /* setup stack pointer */
  55. li sp, SYSTEM_STACK
  56. la gp, _gp
  57. /* clear bss */
  58. la t0, __bss_start
  59. la t1, __bss_end
  60. _clr_bss_loop:
  61. sw zero, 0(t0)
  62. bne t0, t1, _clr_bss_loop
  63. addiu t0, t0, 4
  64. /* jump to RT-Thread RTOS */
  65. jal rtthread_startup
  66. nop
  67. /* restart, never die */
  68. j _start
  69. nop
  70. #else
  71. mtc0 zero, CP0_STATUS // 清零cp0 status寄存器
  72. mtc0 zero, CP0_CAUSE // 清零cp0 cause寄存器
  73. /*
  74. 设置启动异常向量入口地址为ROM地址(0xbfc00000)
  75. 将寄存器cp0 status的BEV置1,使CPU采用ROM(kseg1)空间的异常入口点
  76. */
  77. li t0, SR_BOOT_EXC_VEC /* Exception to Boostrap Location */
  78. mtc0 t0, CP0_STATUS
  79. /* setup stack pointer */
  80. li sp, SYSTEM_STACK
  81. la gp, _gp
  82. /* initialize spi */
  83. li t0, 0xbfe80000 //地址0xbfe80000为SPI0的寄存器基地址
  84. li t1, 0x17 // div 4, fast_read + burst_en + memory_en double I/O 模式 部分SPI flash可能不支持
  85. sb t1, 0x4(t0) // 设置寄存器sfc_param
  86. li t1, 0x05
  87. sb t1, 0x6(t0) // 设置寄存器sfc_timing
  88. /* 设置sdram cs1复用关系,开发板使用ejtag_sel gpio_0引脚(第五复用)作为第二片sdram的片选
  89. 注意sw2拨码开关的设置,使用ejtag烧录pmon时需要调整拨码开关,烧录完再调整回来 */
  90. li a0, 0xbfd011c0
  91. lw a1, 0x40(a0)
  92. ori a1, 0x01
  93. sw a1, 0x40(a0)
  94. bal locate
  95. nop
  96. /* restart, never die */
  97. j _start
  98. nop
  99. #endif
  100. .set reorder
  101. .globl cp0_get_cause
  102. cp0_get_cause:
  103. mfc0 v0, CP0_CAUSE
  104. jr ra
  105. nop
  106. .globl cp0_get_status
  107. cp0_get_status:
  108. mfc0 v0, CP0_STATUS
  109. jr ra
  110. nop
  111. .globl cp0_get_hi
  112. cp0_get_hi:
  113. mfhi v0
  114. jr ra
  115. nop
  116. .globl cp0_get_lo
  117. cp0_get_lo:
  118. mflo v0
  119. jr ra
  120. nop
  121. #if defined(RT_USING_SELF_BOOT)
  122. /****************************************LOCATE*********************************/
  123. /*
  124. * We get here from executing a bal to get the PC value of the current execute
  125. * location into ra. Check to see if we run from ROM or if this is ramloaded.
  126. * 寄存器ra内保持着函数的返回地址,根据ra的值来判断当前是从ROM冷启动,还是从RAM热复位的
  127. * ROM冷启动由通电引起,RAM热复位为各种异常引起,比如看门狗引起的复位等,
  128. * 也就是RAM热复位之前CPU已经开始运行了
  129. * 如果是从ROM冷启动,则寄存器ra的值为指令"bal locate"所在位置加8字节,大概在0xBFC00000附近
  130. * 如果是从RAM热复位,则集成器ra的值为0x80xxxxxx
  131. */
  132. locate:
  133. // la s0, uncached
  134. // subu s0, ra, s0
  135. /*
  136. * start.s的这段汇编程序在ROM(入口点为0xBFC00000)中运行
  137. * 而编译链接时指定的起始地址是0x80100000,所以需要修正一下地址
  138. * s0中保存着ra与start的差值,在后续的代码中可以起到修正地址的作用
  139. * 在看看文件开始的时候,对寄存器s0用途的描述是“ link versus load offset, used to relocate absolute adresses”
  140. * 除了修正地址外,还通过s0的值来判断是从ROM冷启动,还是从RAM热启动
  141. */
  142. la s0, _start // s0 = _start, 其中start的地址为编译链接时,指定的0x80010000
  143. subu s0, ra, s0 // s0 = ra - s0,其中ra的值在ROM入口地址0xBFC00000附近
  144. and s0, 0xffff0000 // s0 = s0 & 0xffff0000
  145. /*
  146. * 初始化cp0的status寄存器和cause寄存器
  147. * 在异常引起的(从RAM)热复位后,需要重新初始化cp0的status和cause,
  148. * 如果是从ROM冷启动的,那么前面已经初始化了,这里是再次重复初始化,没有影响的
  149. */
  150. li t0, SR_BOOT_EXC_VEC
  151. mtc0 t0, CP0_CONFIG // 重新初始化cp0的status寄存器
  152. mtc0 zero, CP0_CAUSE // 重新清零cp0的cause寄存器
  153. .set noreorder
  154. li t0, 0xbfe78030 // 地址0xbfe78030为PLL/SDRAM频率配置寄存器的地址
  155. /* 设置PLL倍频 及SDRAM分频 */
  156. li t2, (0x80000008 | (PLL_MULT << 8) | (0x3 << 2) | SDRAM_DIV)
  157. /* 设置CPU分频 */
  158. li t3, (0x00008003 | (CPU_DIV << 8))
  159. /* 注意:首先需要把分频使能位清零 */
  160. li t1, 0x2
  161. sw t1, 0x4(t0) // 清零CPU_DIV_VALID,即disable
  162. sw t2, 0x0(t0) // 写寄存器START_FREQ
  163. sw t3, 0x4(t0) // 写寄存器CLK_DIV_PARAM
  164. DELAY(2000)
  165. /* 芯片上电默认使用gpio(输入模式)但大多时候是使用模块的功能,如lcd i2c spi ac97等
  166. 所以这里把gpio都关闭,方便使用模块功能。如果上电后需要gpio输出一个确定电平,
  167. 如继电器、LDE等,可以修改这里的代码。*/
  168. /* disable all gpio */
  169. li a0,0xbfd00000
  170. sw zero,0x10c0(a0) /* disable gpio 0-31 */
  171. sw zero,0x10c4(a0) /* disable gpio 32-63 */
  172. sw zero,0x10c8(a0) /* disable gpio 64-95 */
  173. sw zero,0x10cc(a0)
  174. li t0, 0xffffffff
  175. sw t0, 0x10d0(a0)
  176. sw t0, 0x10d4(a0)
  177. sw t0, 0x10d8(a0)
  178. sw t0, 0x10dc(a0)
  179. sw t0, 0x10f0(a0)
  180. sw t0, 0x10f4(a0)
  181. sw t0, 0x10f8(a0)
  182. sw t0, 0x10fc(a0)
  183. /* lcd soft_reset and panel config & timing */
  184. #ifdef DC_FB0
  185. /* li a0, 0xbc301240
  186. li a1, 0x00100103
  187. sw a1, 0x0(a0)
  188. li a1, 0x00000103
  189. sw a1, 0x0(a0) //soft_reset
  190. li a1, 0x00100103
  191. sw a1, 0x0(a0)
  192. li a1, 0x80001111
  193. sw a1, 0x180(a0) //panel config
  194. li a1, 0x33333333
  195. sw a1, 0x1a0(a0)*/
  196. #endif
  197. li output_en, 0x1
  198. #ifdef FAST_STARTUP
  199. li a1, 0x03000000
  200. sw a1, 0x10c4(a0)
  201. sw a1, 0x10d4(a0)
  202. lw a2, 0x10e4(a0)
  203. and a2, a1
  204. beq a2, a1, get_pin_val_finish
  205. nop
  206. li output_en, 0x1
  207. get_pin_val_finish:
  208. #endif
  209. /* Initializing. Standby... */
  210. /*
  211. * 根据s0的值判断是否为ROM冷启动
  212. * 如果s0不等于0,则是ROM冷启动;如果等于0,则是RAM热复位
  213. * 冷启动,则需要初始化内存,cache,加载代码到内存等
  214. */
  215. bnez s0, 1f // 如果寄存器s0不等于0,则说明是ROM冷启动,则跳转到下一个标号1处进行彻底初始化
  216. nop
  217. li a0, 128
  218. jal rtthread_startup // 热复位,则直接跳转到函数main
  219. nop
  220. 1:
  221. /* use only 8wins */
  222. #define CPU_WIN_BASE 0xbfd00000
  223. #define CPU_WIN_MASK 0xbfd00040
  224. #define CPU_WIN_MMAP 0xbfd00080
  225. #define set_cpu_window(id, base, mask, mmap) \
  226. li t0, CPU_WIN_BASE ; \
  227. sw $0, 0x80+id*8(t0) ; \
  228. li t1, base ; \
  229. sw t1, 0x00+id*8(t0) ; \
  230. sw $0, 0x04+id*8(t0) ; \
  231. li t1, mask ; \
  232. sw t1, 0x40+id*8(t0) ; \
  233. sw $0, 0x44+id*8(t0) ; \
  234. li t1, mmap ; \
  235. sw t1, 0x80+id*8(t0) ; \
  236. sw $0, 0x84+id*8(t0)
  237. /* fixup cpu window */
  238. cpu_win_fixup:
  239. //
  240. // hit = (paddr & mask) == (mmap & mask)
  241. // mapped_addr = paddr &~mask | mmap & mask
  242. //
  243. // mmap[7] -> enable
  244. // mmap[5] -> block trans enable
  245. // mmap[4] -> cachable
  246. // mmap[1:0] -> destination
  247. //
  248. // NOTE: the address windows has priority, win0 > win1 > ... > win7
  249. /* set_cpu_window(0, 0x1c280000, 0xfff80000, 0x1c280083) // camera 512K
  250. set_cpu_window(1, 0x1c300000, 0xfff00000, 0x1c300081) // dc 1M
  251. set_cpu_window(2, 0x1fe10000, 0xffffe000, 0x1fe10082) // gmac0 8K
  252. set_cpu_window(3, 0x1fe10000, 0xffff0000, 0x1fe100d0) // gmac0 64K
  253. set_cpu_window(4, 0x1f000000, 0xff000000, 0x1f000082) // AXIMUX 16M
  254. set_cpu_window(5, 0x00000000, 0x00000000, 0x000000f0) // ddr 0
  255. set_cpu_window(6, 0x00000000, 0x00000000, 0x000000f0) // ddr 0
  256. set_cpu_window(7, 0x00000000, 0x00000000, 0x000000f0) // ddr 0*/
  257. /* set_cpu_window(0, 0x1c280000, 0xfff80000, 0x1c2800d3) // camera
  258. // set_cpu_window(1, 0x1fc00000, 0xfff00000, 0x1fc000f2) //
  259. set_cpu_window(2, 0x1c300000, 0xfff00000, 0x1c3000d1) // dc 1M
  260. // set_cpu_window(3, 0x1f000000, 0xff000000, 0x1f0000d2) //
  261. set_cpu_window(4, 0x00000000, 0x00000000, 0x000000f0)
  262. set_cpu_window(5, 0x00000000, 0x00000000, 0x000000f0)
  263. set_cpu_window(6, 0x00000000, 0x00000000, 0x000000f0) // ddr 0
  264. set_cpu_window(7, 0x00000000, 0x00000000, 0x000000f0) // ddr 0*/
  265. // after this fixup, the kernel code should be compiled with
  266. // uncached instruction fetch patch
  267. /* 配置内存 */
  268. li msize, MEM_SIZE
  269. #if !defined(NAND_BOOT_EN)
  270. /*
  271. 手册建议,先写寄存器SD_CONFIG[31:0],然后再写寄存器的SD_CONFIG[63:32],
  272. 即先写低32位,再写高32位。
  273. 写三次寄存器,最后一次将最高位置一,即使能
  274. */
  275. // 写第一次
  276. li t1, 0xbfd00410 // 寄存器SD_CONFIG[31:0]的地址为0xbfd00410
  277. li a1, SD_PARA0 // 宏SD_PARA0在sdram_cfg.S中定义的
  278. sw a1, 0x0(t1) // 将宏SD_PARA0的值写入寄存器SD_CONFIG[31:0]
  279. li a1, SD_PARA1
  280. sw a1, 0x4(t1) // 同理,将宏SD_PARA1的值写入寄存器SD_CONFIG[63:32]
  281. // 写第二次
  282. li a1, SD_PARA0
  283. sw a1, 0x0(t1)
  284. li a1, SD_PARA1
  285. sw a1, 0x4(t1)
  286. // 写第三次
  287. li a1, SD_PARA0
  288. sw a1, 0x0(t1)
  289. li a1, SD_PARA1_EN // 使能
  290. sw a1, 0x4(t1)
  291. // DELAY(100)
  292. #endif
  293. /**************************************CACHE*****************************/
  294. #define CF_7_SE (1 << 3) /* Secondary cache enable */
  295. #define CF_7_SC (1 << 31) /* Secondary cache not present */
  296. #define CF_7_TE (1 << 12) /* Tertiary cache enable */
  297. #define CF_7_TC (1 << 17) /* Tertiary cache not present */
  298. #define CF_7_TS (3 << 20) /* Tertiary cache size */
  299. #define CF_7_TS_AL 20 /* Shift to align */
  300. #define NOP8 nop;nop;nop;nop;nop;nop;nop;nop
  301. do_caches:
  302. /* Init caches... */
  303. li s7, 0 /* no L2 cache */
  304. li s8, 0 /* no L3 cache */
  305. bal cache_init // 调用汇编函数cache_init
  306. nop
  307. mfc0 a0, CP0_CONFIG // 将协处理器0的config寄存器的值加载到寄存器a0
  308. and a0, a0, ~((1<<12) | 7) // a0 = a0 & ~((1<<12) | 7)
  309. or a0, a0, 2 // a0 |= 2
  310. mtc0 a0, CP0_CONFIG // 将寄存器a0的值写入协处理器0的config寄存器
  311. /***********************MEMORY DEBUGGING AND COPY SELF TO RAM***********************/
  312. //#include "newtest.32/mydebug.S"
  313. bootnow:
  314. /* copy program to sdram to make copy fast */
  315. /* 先将执行拷贝pmon到内存任务的代码,拷贝到内存0xa0000000 */
  316. /* 先确定需要拷贝的代码段为标号121到标号122之间的代码
  317. * 由于链接时指定的起始地址是0x80010000,
  318. * 而目前正在ROM(SPI NOR FLASH,起始地址为0xBFC00000)运行
  319. * 所以需要用寄存器s0来修正一下地址
  320. */
  321. la t0, 121f // 将下一个标号121所在地址,加载到寄存器t0
  322. addu t0, s0 // 使用寄存器s0修正t0中的(标号121的)地址
  323. la t1, 122f // 将下一个标号122所在地址,加载到寄存器t1
  324. addu t1, s0 // 使用寄存器s0修正t1中的(标号122的)地址
  325. li t2, 0xa0000000 // 将立即数0xa0000000(起始地址)加载到寄存器t2
  326. 1:
  327. lw v0, (t0) // 将寄存器t0所指的内存地址开始4字节的数据加载到寄存器v0
  328. sw v0, (t2) // 将寄存器v0的内容保存到寄存器t2所指的内存中
  329. addu t0, 4 // 寄存器t0向后移4字节
  330. addu t2, 4 // 寄存器t2向后移4字节
  331. ble t0, t1, 1b // 如果t0 <= t1,则跳转到上一个标号1处,继续拷贝后面的4字节
  332. nop
  333. li t0, 0xa0000000 // 将立即数0xa0000000加载到寄存器t0
  334. jr t0 // 跳转到起始地址0xa0000000处开始执行(拷贝任务)
  335. nop
  336. 121:
  337. /* Copy PMON to execute location... */
  338. /* 将固件拷贝到起始地址为0xa0010000的内存空间
  339. 由于kseg0(0x8000 0000 - 0x9FFF FFFF)和kseg1(0xA000 0000 - 0xBFFF FFFF)是映射到物理内存的相同区域
  340. 即拷贝到0xA000 0000开始的kseg1,就相当于拷贝到0x8000 0000开始的kseg0
  341. 这就是为什么链接时,指定的地址是0x8001 0000,而拷贝的目标起始地址是0xA001 0000
  342. */
  343. la a0, _start // 加载符号start所在地址0x80010000加载到寄存器a0中
  344. addu a1, a0, s0 // 使用寄存器s0修正寄存器a0中的地址,a1=0xBFC00000
  345. la a2, __bss_start // 加载_edata(链接脚本中的一个符号)到寄存器a2
  346. or a0, 0xa0000000 // a0 = a0 | 0xa0000000 = 0xa0010000
  347. or a2, 0xa0000000 // a2 = a2 | 0xa0000000,修正地址_edata
  348. subu t1, a2, a0 // t1 = a2 - a0,即计算从start到_edata之间的长度(字节数)
  349. srl t1, t1, 2 // t1 >>= 2,即t1除以4。(和前面类似,每次拷贝4字节,所以除以4)
  350. // 似乎t1计算结果没有被使用,马上就被后面的覆盖了
  351. move t0, a0 // t0 = a0 = 0xa0010000 (目标起始地址)
  352. move t1, a1 // t1 = a1 = 0xBFC00000 (start在ROM中的地址,源起始地址)
  353. move t2, a2 // t2 = a2 (_edata在ROM中的地址,源结束地址)
  354. /* copy text section */
  355. 1: and t3, t0, 0x0000ffff // t3 = t0 & 0x0000ffff,取低16位
  356. bnez t3, 2f // 如果t3不等于0,则跳转到下一个标号2处继续执行,t3的计算结果似乎没被使用,就被后面的覆盖了
  357. nop
  358. 2: lw t3, 0(t1) // 从源地址t1处加载4字节到寄存器t3中
  359. nop
  360. sw t3, 0(t0) // 将寄存器t3中的4字节数据保存到目标地址t0处
  361. addu t0, 4 // 目标地址t0后移4字节
  362. addu t1, 4 // 源地址t1 后移4字节
  363. bne t2, t0, 1b // 如果t2不等于t0,则跳到上一个标号1处继续拷贝,总的来说就是判断拷贝是否结束
  364. nop
  365. /* copy text section done. */
  366. /* clear bss */
  367. la t0, __bss_start
  368. la t1, __bss_end
  369. _clr_bss_loop:
  370. sw zero, 0(t0)
  371. bne t0, t1, _clr_bss_loop
  372. addiu t0, t0, 4
  373. /* disable interrupt */
  374. mfc0 t0, CP0_STATUS
  375. and t0, 0xfffffffe # By default it will be disabled.
  376. mtc0 t0, CP0_STATUS # Set CPU to disable interrupt.
  377. nop
  378. /* disable cache */
  379. mfc0 t0, CP0_CONFIG
  380. and t0, 0xfffffff8
  381. or t0, 0x2 # disable,!default value is not it!
  382. mtc0 t0, CP0_CONFIG # Set CPU to disable cache.
  383. nop
  384. /* jump to RT-Thread RTOS */
  385. jal rtthread_startup
  386. nop
  387. /* restart, never die */
  388. j _start
  389. nop
  390. 122:
  391. stuck:
  392. b stuck
  393. nop
  394. #endif
  395. .extern tlb_refill_handler
  396. .extern cache_error_handler
  397. /* Exception Handler */
  398. /* 0x0 - TLB refill handler */
  399. .section .vectors.1, "ax", %progbits
  400. .global tlb_refill_exception
  401. .type tlb_refill_exception,@function
  402. tlb_refill_exception:
  403. j tlb_refill_handler
  404. nop
  405. /* 0x100 - Cache error handler */
  406. .section .vectors.2, "ax", %progbits
  407. j cache_error_handler
  408. nop
  409. /* 0x180 - Exception/Interrupt handler */
  410. .section .vectors.3, "ax", %progbits
  411. .global general_exception
  412. .type general_exception,@function
  413. general_exception:
  414. j _general_exception_handler
  415. nop
  416. /* 0x200 - Special Exception Interrupt handler (when IV is set in CP0_CAUSE) */
  417. .section .vectors.4, "ax", %progbits
  418. .global irq_exception
  419. .type irq_exception,@function
  420. irq_exception:
  421. j _irq_handler
  422. nop
  423. .section .vectors, "ax", %progbits
  424. .extern mips_irq_handle
  425. /* general exception handler */
  426. _general_exception_handler:
  427. .set noreorder
  428. la k0, mips_irq_handle
  429. jr k0
  430. nop
  431. .set reorder
  432. /* interrupt handler */
  433. _irq_handler:
  434. .set noreorder
  435. la k0, mips_irq_handle
  436. jr k0
  437. nop
  438. .set reorder