下面是一段直接进入状态切换的例程:
;从ARM状态开始
CODE32 ;表明一下是ARM指令
ADR R0,Into_Thumb+1 ;得到目标地址,末位置1,表示转移到Thumb
BX R0 ;转向Thumb
…
CODE16 ;表明以下是Thumb指令Into_Thumb
…
ADR R5,Back_to_ARM ;得到目标地址,末位缺省为0 ,转移到ARM
BX R5 ;转向ARM
…
CODE32
Back_to_ARM ;ARM代码段起始地址
举例:从ARM态进入thumb态
① 将在Makefile中将.c文件编译指定为thumb态
参考:https://www.cnblogs.com/QuLory/archive/2012/10/23/2735226.html
%.o:%.c
arm-linux-gcc -mthumb -c -o $@ $^ -fno-builtin
%.o:%.S
arm-linux-gcc -c -o $@ $^
.S文件直接在文件中修改来实现态的转化
② 修改.S文件从arm转化到thumb态
.text
.global _start
.code 32
_start:
/* 怎样从arm_state -> thumb_state */
adr r0,thumb_func
add r0,r0,#1
bx r0
.code 16
thumb_func:
bl sdram_init
//bl sdram_init2 //用到有初始值的数组,不是位置无关码
bl copy2sdram
bl clean_bss
//bl main //使用BL命令相对跳转,程序仍然在Nor/sram执行
ldr r0,=main //绝对跳转,跳到SDRAM
mov pc,r0
halt:
b halt
-----------------------------------------------------------------------------------------------------------------------------------------------------
问题:
①:
start.S: Assembler messages:
start.S:60: Error: lo register required -- `ldr pc,=main'
Makefile:11: recipe for target 'start.o' failed
thumb不像arm态那样,不能直接将标号赋值给pc,需要通过寄存器转化:
ldr r0,=main
mov pc,r0
②:
init.o(.text+0x58): In function `sdram_init2':
: undefined reference to `memcpy'
thumb中自动给我们将数组中的值通过memcpy复制到了数组,我们可以通过static来解决这个问题:
参考Getting GCC to compile without inserting call to memcpy:
https://stackoverflow.com/questions/6410595/getting-gcc-to-compile-without-inserting-call-to-memcpy
unsigned int arr[] = {
0x02000000, //BWSCON
0x00000700, //BANKCON0
0x00000700, //BANKCON1
0x00000700, //BANKCON2
0x00000700, //BANKCON3
0x00000700, //BANKCON4
0x00000700, //BANKCON5
0x00018001, //BANKCON6
0x00018005, //BANKCON7
0x008404f5, //REFRESH
0x000000b1, //BANKSIZE
0x00000020, //MRSRB6
0x00000020, //MRSRB7
};//把那些值全部放在数组里面
->>>>>>
const static unsigned int arr[] = {
0x02000000, //BWSCON
0x00000700, //BANKCON0
0x00000700, //BANKCON1
0x00000700, //BANKCON2
0x00000700, //BANKCON3
0x00000700, //BANKCON4
0x00000700, //BANKCON5
0x00018001, //BANKCON6
0x00018005, //BANKCON7
0x008404f5, //REFRESH
0x000000b1, //BANKSIZE
0x00000020, //MRSRB6
0x00000020, //MRSRB7
};//把那些值全部放在数组里面
warning: conflicting types for built-in function
解决:在编译的时候加上-fno-builtin不使用内建函数
arm-linux-gcc -mthumb -c -o $@ $^ -fno-builtin