sleep.S
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:5k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * SA11x0 Assembler Sleep/WakeUp Management Routines
  3.  *
  4.  * Copyright (c) 2001 Cliff Brake <cbrake@accelent.com>
  5.  *
  6.  * This program is free software; you can redistribute it and/or
  7.  * modify it under the terms of the GNU General Public License.
  8.  *
  9.  * History:
  10.  *
  11.  * 2001-02-06: Cliff Brake         Initial code
  12.  *
  13.  * 2001-08-29: Nicolas Pitre Simplified.
  14.  */
  15. #include <linux/linkage.h>
  16. #include <asm/assembler.h>
  17. #include <asm/hardware.h>
  18. #include "sleep.h"
  19. /*
  20.  * sa1100_cpu_suspend()
  21.  *
  22.  * Causes sa11x0 to enter sleep state
  23.  *
  24.  */
  25. .text
  26. ENTRY(sleep_save) .word 0 @ virtual address of parameter array
  27. ENTRY(sleep_save_p) .word 0 @ physical address of parameter array
  28. ENTRY(sa1100_cpu_suspend)
  29. @ save registers on stack
  30. stmfd sp!, {r4 - r12, lr}
  31. @ load virtual address for sleep_save array
  32. ldr r4, sleep_save
  33. @ save stack pointer
  34. str sp, [r4, #(SLEEP_SAVE_SP*4)]
  35. @ save coprocessor registers
  36. mrc  p15, 0, r1, c1, c0, 0
  37. str r1, [r4, #(SLEEP_SAVE_CP15_R1*4)]
  38. mrc  p15, 0, r1, c2, c0, 0
  39. str r1, [r4, #(SLEEP_SAVE_CP15_R2*4)]
  40. mrc  p15, 0, r1, c3, c0, 0
  41. str r1, [r4, #(SLEEP_SAVE_CP15_R3*4)]
  42. mrc  p15, 0, r1, c5, c0, 0
  43. str r1, [r4, #(SLEEP_SAVE_CP15_R5*4)]
  44. mrc  p15, 0, r1, c6, c0, 0
  45. str r1, [r4, #(SLEEP_SAVE_CP15_R6*4)]
  46. mrc  p15, 0, r1, c13, c0, 0
  47. str r1, [r4, #(SLEEP_SAVE_CP15_R13*4)]
  48. @ clean data cache and invalidate WB
  49. bl cpu_sa1100_cache_clean_invalidate_all
  50. @ disable clock switching
  51. mcr p15, 0, r1, c15, c2, 2
  52.         @ Adjust memory timing before lowering CPU clock
  53. @ Clock speed adjustment without changing memory timing makes
  54. @ CPU hang in some cases
  55.         ldr     r0, =MDREFR
  56.         ldr     r1, [r0]
  57.         orr     r1, r1, #MDREFR_K1DB2
  58.         str     r1, [r0]
  59. @ delay 90us and set CPU PLL to lowest speed
  60. @ fixes resume problem on high speed SA1110
  61. mov r0, #90
  62. bl SYMBOL_NAME(udelay)
  63. ldr r0, =PPCR
  64. mov r1, #0
  65. str r1, [r0]
  66. mov r0, #90
  67. bl SYMBOL_NAME(udelay)
  68. /* setup up register contents for jump to page containing SA1110 SDRAM controller bug fix suspend code
  69.  *
  70.  * r0 points to MSC0 register
  71.  * r1 points to MSC1 register
  72.  * r2 points to MSC2 register
  73.  * r3 is MSC0 value
  74.  * r4 is MSC1 value
  75.  * r5 is MSC2 value
  76.  * r6 points to MDREFR register
  77.  * r7 is first MDREFR value
  78.  * r8 is second MDREFR value
  79.  * r9 is pointer to MDCNFG register
  80.  * r10 is MDCNFG value
  81.  * r11 is third MDREFR value
  82.  * r12 is pointer to PMCR register
  83.  * r13 is PMCR value (1)
  84.  *
  85.  */
  86. ldr r0, =MSC0
  87. ldr r1, =MSC1
  88. ldr r2, =MSC2
  89.         ldr     r3, [r0]
  90.         bic     r3, r3, #FMsk(MSC_RT)
  91.         bic     r3, r3, #FMsk(MSC_RT)<<16
  92.         ldr     r4, [r1]
  93.         bic     r4, r4, #FMsk(MSC_RT)
  94.         bic     r4, r4, #FMsk(MSC_RT)<<16
  95.         ldr     r5, [r2]
  96.         bic     r5, r5, #FMsk(MSC_RT)
  97.         bic     r5, r5, #FMsk(MSC_RT)<<16
  98.         ldr     r6, =MDREFR
  99.         ldr     r7, [r6]
  100.         bic     r7, r7, #0x0000FF00
  101.         bic     r7, r7, #0x000000F0
  102.         orr     r8, r7, #MDREFR_SLFRSH
  103.         ldr     r9, =MDCNFG
  104.         ldr     r10, [r9]
  105.         bic     r10, r10, #(MDCNFG_DE0+MDCNFG_DE1)
  106.         bic     r10, r10, #(MDCNFG_DE2+MDCNFG_DE3)
  107.         bic     r11, r8, #MDREFR_SLFRSH
  108.         bic     r11, r11, #MDREFR_E1PIN
  109.         ldr     r12, =PMCR
  110.         mov     r13, #PMCR_SF
  111. b sa1110_sdram_controller_fix
  112. .align 5
  113. sa1110_sdram_controller_fix:
  114. @ Step 1 clear RT field of all MSCx registers
  115. str  r3, [r0]
  116. str r4, [r1]
  117. str r5, [r2]
  118. @ Step 2 clear DRI field in MDREFR
  119. str r7, [r6]
  120. @ Step 3 set SLFRSH bit in MDREFR
  121. str r8, [r6]
  122. @ Step 4 clear DE bis in MDCNFG
  123. str r10, [r9]
  124. @ Step 5 clear DRAM refresh control register
  125. str r11, [r6]
  126. @ Wow, now the hardware suspend request pins can be used, that makes them functional for
  127. @ about 7 ns out of the entire time that the CPU is running!
  128. @ Step 6 set force sleep bit in PMCR
  129. str r13, [r12]
  130. 20: b 20b @ loop waiting for sleep
  131. /*
  132.  * cpu_sa1100_resume()
  133.  *
  134.  * entry point from bootloader into kernel during resume
  135.  *
  136.  */
  137. .align 5
  138. ENTRY(sa1100_cpu_resume)
  139. @ set SVC, irqs off
  140. mov r0, #I_BIT | MODE_SVC
  141. msr cpsr_c, r0
  142. @ load physical address of sleep_save
  143. ldr r4, sleep_save_p
  144. @ restore cp15_r3, domain id
  145. ldr r1, [r4, #(SLEEP_SAVE_CP15_R3*4)]
  146. mcr  p15, 0, r1, c3, c0 ,0
  147. @ restore cp15_r2, translation table base address
  148. ldr r1, [r4, #(SLEEP_SAVE_CP15_R2*4)]
  149. mcr  p15, 0, r1, c2, c0 ,0
  150. mov r1, #0
  151. mcr p15, 0, r1, c8, c7, 0    @ flush I+D TLBs
  152. mcr p15, 0, r1, c7, c7, 0 @ flush I&D cache
  153. @ get saved cp15 r1 (control register)
  154. ldr r1, [r4, #(SLEEP_SAVE_CP15_R1*4)]
  155. @ get address to jump to after turning on MMU
  156. ldr r2, =resume_after_mmu
  157. cmp r2, #0
  158. b resume_turn_on_mmu
  159. .align 5
  160. resume_turn_on_mmu:
  161. @ turn on mmu
  162. mcr  p15, 0, r1, c1, c0 ,0
  163. @ jump to resume_after_mmu
  164. mov pc, r2
  165. nop
  166. nop
  167. .align 5
  168. resume_after_mmu:
  169. @ load virtual address for sleep_save array
  170. ldr r4, sleep_save
  171. @ restore the rest of CPU state
  172. ldr r1, [r4, #(SLEEP_SAVE_CP15_R13*4)]
  173. mcr p15, 0, r1, c13, c0, 0
  174. ldr r1, [r4, #(SLEEP_SAVE_CP15_R5*4)]
  175. mcr  p15, 0, r1, c5, c0 ,0
  176. ldr r1, [r4, #(SLEEP_SAVE_CP15_R6*4)]
  177. mcr  p15, 0, r1, c6, c0 ,0
  178. @ restore stack pointer
  179. ldr sp, [r4, #(SLEEP_SAVE_SP*4)]
  180. @ return to caller
  181. ldmfd sp!, {r4 - r12, pc}