2005-04-16 15:20:36 -07:00
/ *
* SA1 1 x0 A s s e m b l e r S l e e p / W a k e U p M a n a g e m e n t R o u t i n e s
*
* Copyright ( c ) 2 0 0 1 C l i f f B r a k e < c b r a k e @accelent.com>
*
* This p r o g r a m i s f r e e s o f t w a r e ; you can redistribute it and/or
* modify i t u n d e r t h e t e r m s o f t h e G N U G e n e r a l P u b l i c L i c e n s e .
*
* History :
*
* 2 0 0 1 - 0 2 - 06 : Cliff B r a k e I n i t i a l c o d e
*
* 2 0 0 1 - 0 8 - 29 : Nicolas P i t r e S i m p l i f i e d .
*
* 2 0 0 2 - 0 5 - 27 : Nicolas P i t r e R e v i s i t e d , m o r e c l e a n u p a n d s i m p l i f i c a t i o n .
* Storage i s o n t h e s t a c k n o w .
* /
# include < l i n u x / l i n k a g e . h >
# include < a s m / a s s e m b l e r . h >
2008-08-05 16:14:15 +01:00
# include < m a c h / h a r d w a r e . h >
2005-04-16 15:20:36 -07:00
.text
/ *
* sa1 1 0 0 _ c p u _ s u s p e n d ( )
*
* Causes s a11 x0 t o e n t e r s l e e p s t a t e
*
* /
ENTRY( s a11 0 0 _ c p u _ s u s p e n d )
stmfd s p ! , { r4 - r12 , l r } @ save registers on stack
@ get coprocessor registers
mrc p15 , 0 , r4 , c3 , c0 , 0 @ domain ID
mrc p15 , 0 , r5 , c2 , c0 , 0 @ translation table base addr
mrc p15 , 0 , r6 , c13 , c0 , 0 @ PID
mrc p15 , 0 , r7 , c1 , c0 , 0 @ control reg
@ store them plus current virtual stack ptr on stack
mov r8 , s p
stmfd s p ! , { r4 - r8 }
@ preserve phys address of stack
mov r0 , s p
bl s l e e p _ p h y s _ s p
ldr r1 , =sleep_save_sp
str r0 , [ r1 ]
@ clean data cache and invalidate WB
bl v4 w b _ f l u s h _ k e r n _ c a c h e _ a l l
@ disable clock switching
mcr p15 , 0 , r1 , c15 , c2 , 2
@ Adjust memory timing before lowering CPU clock
@ Clock speed adjustment without changing memory timing makes
@ CPU hang in some cases
ldr r0 , =MDREFR
ldr r1 , [ r0 ]
orr r1 , r1 , #M D R E F R _ K 1 D B 2
str r1 , [ r0 ]
@ delay 90us and set CPU PLL to lowest speed
@ fixes resume problem on high speed SA1110
mov r0 , #90
bl _ _ u d e l a y
ldr r0 , =PPCR
mov r1 , #0
str r1 , [ r0 ]
mov r0 , #90
bl _ _ u d e l a y
/ *
* SA1 1 1 0 S D R A M c o n t r o l l e r w o r k a r o u n d . r e g i s t e r v a l u e s :
*
* r0 = & M S C 0
* r1 = & M S C 1
* r2 = & M S C 2
* r3 = M S C 0 v a l u e
* r4 = M S C 1 v a l u e
* r5 = M S C 2 v a l u e
* r6 = & M D R E F R
* r7 = f i r s t M D R E F R v a l u e
* r8 = s e c o n d M D R E F R v a l u e
* r9 = & M D C N F G
* r1 0 = M D C N F G v a l u e
* r1 1 = t h i r d M D R E F R v a l u e
* r1 2 = & P M C R
* r1 3 = P M C R v a l u e ( 1 )
* /
ldr r0 , =MSC0
ldr r1 , =MSC1
ldr r2 , =MSC2
2008-11-26 20:58:43 +01:00
ldr r3 , [ r0 ]
bic r3 , r3 , #F M s k ( M S C _ R T )
bic r3 , r3 , #F M s k ( M S C _ R T ) < < 16
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r4 , [ r1 ]
bic r4 , r4 , #F M s k ( M S C _ R T )
bic r4 , r4 , #F M s k ( M S C _ R T ) < < 16
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r5 , [ r2 ]
bic r5 , r5 , #F M s k ( M S C _ R T )
bic r5 , r5 , #F M s k ( M S C _ R T ) < < 16
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r6 , =MDREFR
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r7 , [ r6 ]
bic r7 , r7 , #0x0000FF00
bic r7 , r7 , #0x000000F0
orr r8 , r7 , #M D R E F R _ S L F R S H
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r9 , =MDCNFG
ldr r10 , [ r9 ]
bic r10 , r10 , #( M D C N F G _ D E 0 + M D C N F G _ D E 1 )
bic r10 , r10 , #( M D C N F G _ D E 2 + M D C N F G _ D E 3 )
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
bic r11 , r8 , #M D R E F R _ S L F R S H
bic r11 , r11 , #M D R E F R _ E 1 P I N
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
ldr r12 , =PMCR
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
mov r13 , #P M C R _ S F
2005-04-16 15:20:36 -07:00
b s a11 1 0 _ s d r a m _ c o n t r o l l e r _ f i x
.align 5
sa1110_sdram_controller_fix :
@ Step 1 clear RT field of all MSCx registers
str r3 , [ r0 ]
str r4 , [ r1 ]
str r5 , [ r2 ]
@ Step 2 clear DRI field in MDREFR
str r7 , [ r6 ]
@ Step 3 set SLFRSH bit in MDREFR
str r8 , [ r6 ]
@ Step 4 clear DE bis in MDCNFG
str r10 , [ r9 ]
@ Step 5 clear DRAM refresh control register
str r11 , [ r6 ]
@ Wow, now the hardware suspend request pins can be used, that makes them functional for
@ about 7 ns out of the entire time that the CPU is running!
@ Step 6 set force sleep bit in PMCR
str r13 , [ r12 ]
20 : b 2 0 b @ loop waiting for sleep
/ *
* cpu_ s a11 0 0 _ r e s u m e ( )
*
* entry p o i n t f r o m b o o t l o a d e r i n t o k e r n e l d u r i n g r e s u m e
*
* Note : Yes, p a r t o f t h e f o l l o w i n g c o d e i s l o c a t e d i n t o t h e . d a t a s e c t i o n .
* This i s t o a l l o w s l e e p _ s a v e _ s p t o b e a c c e s s e d w i t h a r e l a t i v e l o a d
* while w e c a n ' t r e l y o n a n y M M U t r a n s l a t i o n . W e c o u l d h a v e p u t
* sleep_ s a v e _ s p i n t h e . t e x t s e c t i o n a s w e l l , b u t s o m e s e t u p s m i g h t
* insist o n i t t o b e t r u l y r e a d - o n l y .
* /
.data
.align 5
ENTRY( s a11 0 0 _ c p u _ r e s u m e )
2006-06-25 12:01:48 +01:00
mov r0 , #P S R _ F _ B I T | P S R _ I _ B I T | S V C _ M O D E
2005-04-16 15:20:36 -07:00
msr c p s r _ c , r0 @ set SVC, irqs off
ldr r0 , s l e e p _ s a v e _ s p @ stack phys addr
ldr r2 , =resume_after_mmu @ its absolute virtual address
ldmfd r0 , { r4 - r7 , s p } @ CP regs + virt stack ptr
mov r1 , #0
mcr p15 , 0 , r1 , c8 , c7 , 0 @ flush I+D TLBs
mcr p15 , 0 , r1 , c7 , c7 , 0 @ flush I&D cache
mcr p15 , 0 , r1 , c9 , c0 , 0 @ invalidate RB
2008-11-26 20:58:43 +01:00
mcr p15 , 0 , r1 , c9 , c0 , 5 @ allow user space to use RB
2005-04-16 15:20:36 -07:00
2008-11-26 20:58:43 +01:00
mcr p15 , 0 , r4 , c3 , c0 , 0 @ domain ID
mcr p15 , 0 , r5 , c2 , c0 , 0 @ translation table base addr
2005-04-16 15:20:36 -07:00
mcr p15 , 0 , r6 , c13 , c0 , 0 @ PID
b r e s u m e _ t u r n _ o n _ m m u @ cache align execution
.align 5
resume_turn_on_mmu :
mcr p15 , 0 , r7 , c1 , c0 , 0 @ turn on MMU, caches, etc.
nop
mov p c , r2 @ jump to virtual addr
nop
nop
nop
sleep_save_sp :
.word 0 @ preserve stack phys ptr here
.text
resume_after_mmu :
2008-11-26 20:58:43 +01:00
mcr p15 , 0 , r1 , c15 , c1 , 2 @ enable clock switching
2005-04-16 15:20:36 -07:00
ldmfd s p ! , { r4 - r12 , p c } @ return to caller