2006-03-27 14:58:25 +01:00
/ *
* linux/ a r c h / a r m / k e r n e l / h e a d - c o m m o n . S
*
* Copyright ( C ) 1 9 9 4 - 2 0 0 2 R u s s e l l K i n g
* Copyright ( c ) 2 0 0 3 A R M L i m i t e d
* All R i g h t s R e s e r v e d
*
* 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
* it 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 v e r s i o n 2 a s
* published b y t h e F r e e S o f t w a r e F o u n d a t i o n .
*
* /
2008-03-05 06:50:07 +01:00
# define A T A G _ C O R E 0 x54 4 1 0 0 0 1
# define A T A G _ C O R E _ S I Z E ( ( 2 * 4 + 3 * 4 ) > > 2 )
2009-10-01 17:43:29 +01:00
# define A T A G _ C O R E _ S I Z E _ E M P T Y ( ( 2 * 4 ) > > 2 )
2008-03-05 06:50:07 +01:00
2011-04-28 14:27:20 -06:00
# ifdef C O N F I G _ C P U _ B I G _ E N D I A N
# define O F _ D T _ M A G I C 0 x d00 d f e e d
# else
# define O F _ D T _ M A G I C 0 x e d f e 0 d d0 / * 0 x d00 d f e e d i n b i g - e n d i a n * /
# endif
2006-03-27 14:58:25 +01:00
/ *
* Exception h a n d l i n g . S o m e t h i n g w e n t w r o n g a n d w e c a n ' t p r o c e e d . W e
* ought t o t e l l t h e u s e r , b u t s i n c e w e d o n ' t h a v e a n y g u a r a n t e e t h a t
* we' r e e v e n r u n n i n g o n t h e r i g h t a r c h i t e c t u r e , w e d o v i r t u a l l y n o t h i n g .
*
* If C O N F I G _ D E B U G _ L L i s s e t w e t r y t o p r i n t o u t s o m e t h i n g a b o u t t h e e r r o r
* and h o p e f o r t h e b e s t ( u s e f u l i f b o o t l o a d e r f a i l s t o p a s s a p r o p e r
* machine I D f o r e x a m p l e ) .
* /
2010-10-04 17:45:25 +01:00
_ _ HEAD
2007-05-31 22:02:22 +01:00
/ * Determine v a l i d i t y o f t h e r2 a t a g s p o i n t e r . T h e h e u r i s t i c r e q u i r e s
* that t h e p o i n t e r b e a l i g n e d , i n t h e f i r s t 1 6 k o f p h y s i c a l R A M a n d
2011-04-28 14:27:20 -06:00
* that t h e A T A G _ C O R E m a r k e r i s f i r s t a n d p r e s e n t . I f C O N F I G _ O F _ F L A T T R E E
* is s e l e c t e d , t h e n i t w i l l a l s o a c c e p t a d t b p o i n t e r . F u t u r e r e v i s i o n s
2007-05-31 22:02:22 +01:00
* of t h i s f u n c t i o n m a y b e m o r e l e n i e n t w i t h t h e p h y s i c a l a d d r e s s a n d
* may a l s o b e a b l e t o m o v e t h e A T A G S b l o c k i f n e c e s s a r y .
*
* Returns :
2011-04-28 14:27:20 -06:00
* r2 e i t h e r v a l i d a t a g s p o i n t e r , v a l i d d t b p o i n t e r , o r z e r o
2007-05-31 22:02:22 +01:00
* r5 , r6 c o r r u p t e d
* /
__vet_atags :
tst r2 , #0x3 @ aligned?
bne 1 f
2011-04-28 14:27:20 -06:00
ldr r5 , [ r2 , #0 ]
# ifdef C O N F I G _ O F _ F L A T T R E E
ldr r6 , =OF_DT_MAGIC @ is it a DTB?
cmp r5 , r6
beq 2 f
# endif
cmp r5 , #A T A G _ C O R E _ S I Z E @ i s f i r s t t a g A T A G _ C O R E ?
2009-10-01 17:43:29 +01:00
cmpne r5 , #A T A G _ C O R E _ S I Z E _ E M P T Y
2007-05-31 22:02:22 +01:00
bne 1 f
ldr r5 , [ r2 , #4 ]
ldr r6 , =ATAG_CORE
cmp r5 , r6
bne 1 f
2011-04-28 14:27:20 -06:00
2 : mov p c , l r @ atag/dtb pointer is ok
2007-05-31 22:02:22 +01:00
1 : mov r2 , #0
mov p c , l r
2008-08-28 11:22:32 +01:00
ENDPROC( _ _ v e t _ a t a g s )
2010-10-01 15:37:05 +01:00
2010-10-04 16:29:35 +01:00
/ *
* The f o l l o w i n g f r a g m e n t o f c o d e i s e x e c u t e d w i t h t h e M M U o n i n M M U m o d e ,
* and u s e s a b s o l u t e a d d r e s s e s ; this is not position independent.
*
* r0 = c p #15 c o n t r o l r e g i s t e r
* r1 = m a c h i n e I D
2011-04-28 14:27:20 -06:00
* r2 = a t a g s / d t b p o i n t e r
2010-10-04 16:29:35 +01:00
* r9 = p r o c e s s o r I D
* /
_ _ INIT
__mmap_switched :
adr r3 , _ _ m m a p _ s w i t c h e d _ d a t a
ldmia r3 ! , { r4 , r5 , r6 , r7 }
cmp r4 , r5 @ Copy data segment if needed
1 : cmpne r5 , r6
ldrne f p , [ r4 ] , #4
strne f p , [ r5 ] , #4
bne 1 b
mov f p , #0 @ Clear BSS (and zero fp)
1 : cmp r6 , r7
strcc f p , [ r6 ] ,#4
bcc 1 b
ARM( l d m i a r3 , { r4 , r5 , r6 , r7 , s p } )
THUMB( l d m i a r3 , { r4 , r5 , r6 , r7 } )
THUMB( l d r s p , [ r3 , #16 ] )
str r9 , [ r4 ] @ Save processor ID
str r1 , [ r5 ] @ Save machine type
str r2 , [ r6 ] @ Save atags pointer
2012-01-16 10:34:31 +01:00
cmp r7 , #0
bicne r4 , r0 , #C R _ A @ C l e a r ' A ' b i t
stmneia r7 , { r0 , r4 } @ Save control register values
2010-10-04 16:29:35 +01:00
b s t a r t _ k e r n e l
ENDPROC( _ _ m m a p _ s w i t c h e d )
.align 2
.type _ _ mmap_ s w i t c h e d _ d a t a , % o b j e c t
__mmap_switched_data :
.long __data_loc @ r4
2010-10-18 22:34:47 +01:00
.long _sdata @ r5
2010-10-04 16:29:35 +01:00
.long __bss_start @ r6
.long _end @ r7
.long processor_id @ r4
.long __machine_arch_type @ r5
.long __atags_pointer @ r6
2012-01-16 10:34:31 +01:00
# ifdef C O N F I G _ C P U _ C P 1 5
2010-10-04 16:29:35 +01:00
.long cr_alignment @ r7
2012-01-16 10:34:31 +01:00
# else
.long 0 @ r7
# endif
2010-10-04 16:29:35 +01:00
.long init_thread_union + THREAD_ S T A R T _ S P @ sp
.size _ _ mmap_ s w i t c h e d _ d a t a , . - _ _ m m a p _ s w i t c h e d _ d a t a
/ *
* This p r o v i d e s a C - A P I v e r s i o n o f _ _ l o o k u p _ p r o c e s s o r _ t y p e
* /
ENTRY( l o o k u p _ p r o c e s s o r _ t y p e )
stmfd s p ! , { r4 - r6 , r9 , l r }
mov r9 , r0
bl _ _ l o o k u p _ p r o c e s s o r _ t y p e
mov r0 , r5
ldmfd s p ! , { r4 - r6 , r9 , p c }
ENDPROC( l o o k u p _ p r o c e s s o r _ t y p e )
2013-07-05 22:59:42 +01:00
_ _ FINIT
.text
2010-10-01 15:37:05 +01:00
/ *
* Read p r o c e s s o r I D r e g i s t e r ( C P #15 , C R 0 ) , a n d l o o k u p i n t h e l i n k e r - b u i l t
* supported p r o c e s s o r l i s t . N o t e t h a t w e c a n ' t u s e t h e a b s o l u t e a d d r e s s e s
* for t h e _ _ p r o c _ i n f o l i s t s s i n c e w e a r e n ' t r u n n i n g w i t h t h e M M U o n
* ( and t h e r e f o r e , w e a r e n o t i n t h e c o r r e c t a d d r e s s s p a c e ) . W e h a v e t o
* calculate t h e o f f s e t .
*
* r9 = c p u i d
* Returns :
* r3 , r4 , r6 c o r r u p t e d
* r5 = p r o c _ i n f o p o i n t e r i n p h y s i c a l a d d r e s s s p a c e
* r9 = c p u i d ( p r e s e r v e d )
* /
__lookup_processor_type :
adr r3 , _ _ l o o k u p _ p r o c e s s o r _ t y p e _ d a t a
ldmia r3 , { r4 - r6 }
sub r3 , r3 , r4 @ get offset between virt&phys
add r5 , r5 , r3 @ convert virt addresses to
add r6 , r6 , r3 @ physical address space
1 : ldmia r5 , { r3 , r4 } @ value, mask
and r4 , r4 , r9 @ mask wanted bits
teq r3 , r4
beq 2 f
add r5 , r5 , #P R O C _ I N F O _ S Z @ s i z e o f ( p r o c _ i n f o _ l i s t )
cmp r5 , r6
blo 1 b
mov r5 , #0 @ unknown processor
2 : mov p c , l r
ENDPROC( _ _ l o o k u p _ p r o c e s s o r _ t y p e )
/ *
* Look i n < a s m / p r o c i n f o . h > f o r i n f o r m a t i o n a b o u t t h e _ _ p r o c _ i n f o s t r u c t u r e .
* /
.align 2
.type _ _ lookup_ p r o c e s s o r _ t y p e _ d a t a , % o b j e c t
__lookup_processor_type_data :
.long .
.long __proc_info_begin
.long __proc_info_end
.size _ _ lookup_ p r o c e s s o r _ t y p e _ d a t a , . - _ _ l o o k u p _ p r o c e s s o r _ t y p e _ d a t a
2010-10-04 17:39:20 +01:00
__error_p :
# ifdef C O N F I G _ D E B U G _ L L
adr r0 , s t r _ p1
bl p r i n t a s c i i
mov r0 , r9
bl p r i n t h e x8
adr r0 , s t r _ p2
bl p r i n t a s c i i
b _ _ e r r o r
str_p1 : .asciz " \nError: unrecognized/unsupported processor variant (0x "
str_p2 : .asciz " ).\n "
.align
# endif
ENDPROC( _ _ e r r o r _ p )
__error :
# ifdef C O N F I G _ A R C H _ R P C
/ *
* Turn t h e s c r e e n r e d o n a e r r o r - R i s c P C o n l y .
* /
mov r0 , #0x02000000
mov r3 , #0x11
orr r3 , r3 , r3 , l s l #8
orr r3 , r3 , r3 , l s l #16
str r3 , [ r0 ] , #4
str r3 , [ r0 ] , #4
str r3 , [ r0 ] , #4
str r3 , [ r0 ] , #4
# endif
1 : mov r0 , r0
b 1 b
ENDPROC( _ _ e r r o r )