mdk中,堆栈地址范围不像在gcc工程中那么容易看出来。过程被mdk隐藏了,单纯从代码层面不好看出来。但是基本的流程是这样的:先确定代码其他部分使用RAM的情况,然后紧跟着已使用的RAM地址,在剩下的RAM地址中分配Stack_Size大小来作为栈的范围使用(Stack_Size在启动文件中定义)。
1.启动文件中堆栈范围的指定
__initial_sp 这个参数是栈顶地址,因为栈的增长是向下增长,所以这个参数就是栈顶,我设置的栈尺寸是0x400,所以栈的地址范围是 __initial_sp ~ (__initial_sp - 0x400),__initial_sp 这个参数是keil编译代码之后计算出来的,有多种方法确定。
2.确定__initial_sp数值
2.1.方法:1:通过map文件
编译代码,编译完成之后,双击工程名打开map的文件,在里面搜索__initial_sp,即可看到__initial_sp的值:
2.2.方法2:mdk仿真查看内存空间
Debug执行代码,然后查看运行地址最开始的四个字节的数据:
2.3.方法3:根据RW-data + ZI-data
因为SRAM的基地址是0x20000000,加上基址,就是0x200007B0,结果和上面一致。
3.自定义栈顶地址__initial_sp
3.1.方法1:直接修改启动文件
Stack_Size EQU 0x00000400
;ADDR_STACK_TOP EQU 0x20020000
; 0x2001FC00 = ADDR_STACK_TOP - Stack_Size
AREA |.ARM.__AT_0x2001FC00|, DATA, NOINIT, READWRITE, ALIGN=3
Stack_Mem SPACE Stack_Size
__initial_sp
其中ARM.__AT_0x2001FC00
指定了栈区域的启始地址为:0x2001FC00,则__initial_sp的值为:0x2001FC00 + 0x00000400 = 0x20020000。
3.2.方法2:通过.sct文件
- 1.设置.sct文件:
;*************************************************************
; *** Scatter-Loading Description Filegenerated by uVision ***
; *************************************************************
LR_IROM1 0x08000000 0x00020000 { ; load region size_region
ER_IROM1 0x08000000 0x00020000 { ; load address = executionaddress
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}
RW_IRAM1 0x20000000 0x00005000 { ; RW data
.ANY (+RW +ZI)
}
RW_IRAM2 0x20004800 UNINIT 0x00000800 { ; STACK ADDRESS
startup_stm32f10x_md.o (STACK)
}
RW_IRAM3 0x20004600 UNINIT 0x00000200 { ; HEAP ADDRESS
startup_stm32f10x_md.o (HEAP)
}
}
其中:
RW_IRAM2 0x20004800 UNINIT 0x00000800 //设置栈起始地址为0x20004800,栈大小为0x00000800
RW_IRAM3 0x20004600 UNINIT 0x00000200 //设置堆起始地址为0x20004600,堆大小为0x00000200
如图:
-
2.在启动文件中设置堆栈大小
此处设置的堆栈大小要与第一步一致,否则将会以这次设置的大小为准,第一步设置的堆栈大小将失效。 -
3.设置存储区分配地址方式;
如下图所示设置,Scatter file选择第一步设置好的.sct文件: