在MAXQ2000数据存储器中实现软堆栈

maxq2000微控制器与maxim的risc微控制器系列maxq器件一样,都是基于maxq20内核。基于maxq20的微控制器通常可实现一个16位宽硬堆栈,其深度固定不变(maxq2000为16),存储在与数据和程序空间分开的专用内部存储器中。通过子程序调用和中断,这一硬堆栈可用于保存和恢复微控制器的操作状态。
虽然非常适合紧凑型应用,但是在规模较大的汇编应用中,使用深度嵌套的子程序(或者使用了在堆栈中保存和恢复多个工作寄存器的子程序),硬堆栈会很快耗尽空间。利用数据存储器中的“软堆栈”,以c编程语言(使用iar的embedded workbench®等编译器)或者rowley associates的crossworks for maxq编写应用程序可避免这一问题。这一软堆栈存储子程序的调用/返回地址以及本地工作变量。然而,maxq20内核没有内置机制来定位数据存储器中的堆栈,这些数据存储器只用在汇编应用中。
本应用笔记介绍了汇编应用中在数据存储器中实现软堆栈的简单方法。应用笔记中的代码可以用于maxq2000和其他基于maxq20的微控制器。采用了max-ide的宏预处理特性,在maxim的maxq系列工程应用开发和调试环境中编写了实例代码。
可以免费下载max-ide环境最新的安装包和文档:
max-ide安装(zip)
maxq内核汇编指南(pdf)
开发工具指南(pdf)
maxq20内核中的硬堆栈工作
maxq20内核使用了两种不同类型的堆栈工作模式:
push工作(它包括操作代码push、lcall和scall)被用于在堆栈中存储数据。这些操作将堆栈指针sp预增1,然后,把数据存储在sp指针(@sp)指向的堆栈位置。
pop工作(它包括操作代码pop、popi、ret和reti)被用于从堆栈中恢复数据。这些操作从sp指向的堆栈位置恢复数据,然后,将堆栈指针减1。
硬堆栈的一项主要功能是在调用子程序时保存并恢复地址,因此,堆栈含有16位(字)位置。这一位宽支持在一次push或者pop操作中保存或者恢复16位指令指针(ip)寄存器。尽管可以使用push或者pop在堆栈中保存或者恢复8位寄存器(例如,ap),但每一堆栈操作总是使用整个16位字。
除了利用堆栈的各种操作代码之外,还有两种其他的情况,在这些情况下,微控制器自动使用硬堆栈:
当处理中断时,开始执行中断服务程序(中断矢量寄存器iv所指)之前,当前程序执行点被推入堆栈。
当调用某些调试命令(例如,读寄存器和写数据存储器)时,需要执行完程序rom中的代码,在控制被传送给程序rom之前,当前的执行点被推入堆栈。一旦执行完程序rom中的调试程序,执行点从堆栈中弹出,恢复处理器以前的状态。
找到中断服务程序或者执行调试器命令总是需要使用硬堆栈。由于这一行为嵌入在硬件中,因此,无法避开这一问题。然而,对于更常用的push/pop和call/ret指令对,可以采用软堆栈。
注意,当以下任一堆栈错误状态出现时,maxq2000 (和其他的基于maxq20内核的器件一样)并不支持任何错误探测或者报警:
堆栈上溢:将数值推入已经满的堆栈时发生。这一错误导致堆栈中最早的数值被覆写。
堆栈下溢:从空堆栈中弹出数值时发生。这一错误导致返回无效数据值。例如,如果堆栈空时执行ret,执行后将返回一个不正确的(有可能是随机的)地址。
在数据存储器中建立一个软堆栈
在数据存储器中建立一个软堆栈的第一步是定义要使用数据存储器的哪一部分。然后,必须定义跟踪堆栈顶当前位置的数据存储器指针(dp[0]、dp[1]或者bp[offs])。注意:应用软件不会出于其他目的使用堆栈专用数据存储器(例如,变量或者缓冲)。
定义并初始化这类软堆栈的一种简单方法涉及到bp[offs]寄存器对和一个公式。
ss_base equ 0100hss_init: move dpc, #1ch ; set all pointers to word mode move bp, #ss_base ; set base pointer to stack base location move offs, #0 ; set stack to start ret 如果基指针(bp)寄存器被设置在ss_base位置,那么,可以采用offs寄存器指向堆栈的当前顶部。由于offs寄存器只有8位宽,硬件会自动把堆栈限制在数据存储器的范围内(bp .. (bp+255))。如果当offs等于255 (上溢)时出现推入,或者offs等于0 (下溢)时出现弹出,那么,offs寄存器只会限制在堆栈的另一端。这一行为模仿硬堆栈工作方式,支持简单的软堆栈实现;它不探测下溢和上溢状态。
在使用软堆栈前,主程序必须调用ss_init例程。它将bp[offs]指针设置为字模式,由于软堆栈以16位宽堆栈实现,因此,必须完成这一步。它还将bp[offs]寄存器对指向堆栈开始。
软堆栈工作
不能重新定义使用堆栈的内置操作代码(push、pop、call、ret等)以使用软堆栈,这是因为这些代码意味着maxq汇编器的硬线连接。而且,还有可能需要在应用程序中的某些部分使用标准硬堆栈,例如,中断服务例程等。出于这些原因,由复制标准操作代码的宏来访问软堆栈。
mpush macro reg move @bp[++offs], reg ; push value to soft stackendmmpop macro reg move reg, @bp[offs--] ; pop value from soft stackendmmcall macro addrlocal return move @bp[++offs], #return ; push return destination to soft stack jump addrreturn:endmmret macro jump @bp[offs--] ; jump to popped destination from soft stackendm 下面将讨论这些宏是怎样工作的。
mpush
该宏的使用方式和push操作代码相同。它支持将8位或者16位寄存器,或者将立即值推入堆栈。
mpush a[0] ; save the value of the a[0] register mpush a[1] ; save a[1] mpush a[2] ; save a[2] ... ; code which destroys a[0]-a[2] mpop a[2] ; restore the value of a[2] (pop in reverse order) mpop a[1] ; restore a[1] mpop a[0] ; restore a[0] mpop
该宏的使用方式和pop操作代码相同。它支持从堆栈中装入8位或者16位寄存器,如上所示。注意,如果推入了一个16位值,该值弹入到一个8位寄存器中,那么,只有低位字节被存储在寄存器中。高位字节丢失。这与内置硬堆栈的方式一致。
subroutine: mpush a[0] ; save the current value of a[0] ... ; code which destroys a[0] mpop a[1] ; restore a[0] mret mcall
mret
mcall宏的使用方式和call操作代码执行子程序相同。一旦执行完成,子程序必须使用mret宏(而不是标准ret操作代码)才能返回。
mcall mysub ...mysub: mpush a[0] ; save a[0] mpush a[1] ; save a[1] ... ; perform calculations, etc. mpop a[1] mpop a[0] mret 扩展软堆栈的大小
某些应用程序需要大于256级的软堆栈。使用其他数据指针(dp[0]或者dp[1])之一可以实现任意大小的堆栈(最大达到数据存储器的上限)。
ss_base equ 0000hss_top equ 01ffhss_init: move dpc, #1ch ; set all data pointers to word mode move dp[0], #ss_base ; set pointer to stack base location ret 上面的代码存储在数据存储器的0000h至01ffh,因此,产生的堆栈可以保持511级。(没有使用堆栈存储器空间中的某一位置;这样可以更有效,以更短的代码实现
只改变数据指针,而代码中的其他部分保持不变可以扩展堆栈。虽然如此,由于dp[0]并没有限制在数据存储器的一定范围内,因此,没有措施来防止推入或者弹出操作使dp[0]增加或者减小导致超出所指定的软堆栈边界。为避免这一点,可以加入简单的下溢/上溢检查。
加入下溢和上溢检查
为保持宏(每次使用时扩展为代码)尽量短,在子程序中完成下溢和上溢检查,使用硬堆栈,由宏调用子程序。
mpush macro reg call ss_check_over ; check for possible overflow move @++dp[0], reg ; push value to soft stackendmmpop macro reg call ss_check_under ; check for possible underflow move reg, @dp[0]-- ; pop value from soft stackendmmcall macro addrlocal return call ss_check_over ; check for possible overflow move @++dp[0], #return ; push return destination to soft stack jump addrreturn:endmmret macro call ss_check_under ; check for possible underflow jump @dp[0]-- ; jump to popped destination from soft stackendmss_check_under: push a[0] push ap push apc push psf move apc, #80h ; set acc to a[0], standard mode, no auto inc/dec move acc, dp[0] ; get current value of stack pointer cmp #ss_base jump ne, ss_check_under_ok nop ; ss_check_under_ok: pop psf pop apc pop ap pop a[0] retss_check_over: push a[0] push ap push apc push psf move apc, #80h ; set acc to a[0], standard mode, no auto inc/dec move acc, dp[0] ; get current value of stack pointer cmp #ss_top jump ne, ss_check_over_ok nop ; ss_check_over_ok: pop psf pop apc pop ap pop a[0] ret 上面的代码导致在推入或者弹出(或者call,或者ret)操作之前检查当前的堆栈位置。如果探测到下溢或者上溢错误,不同的应用程序有不同的响应。一般情况下,这类错误只会出现在应用程序开发阶段;如果代码编写正确,不会出现这类错误。如果的确出现了错误,一般将其考虑为重大错误,就像使用硬堆栈时出现下溢/上溢错误。对这类错误可能的响应包括暂停,发送错误消息,或者闪烁led。在开发阶段,一个好的方法是两个程序中的每一个都在max-ide中设置断点(例如,在“error handler should be implemented here”行中),如果出现下溢或者上溢,立即发出反馈。
但是,应用程序有时候也可以从堆栈下溢或者上溢中恢复(例如,重新装入和从头重新启动应用程序子任务)。在这种情况下,需要简单地设置一个标志,指示出现了堆栈错误。可选的这类标志包括位于psf寄存器中的两个通用标志(gpf0和gpf1)。由于有两个位标志,因此,其中一个可以用于指示上溢,另一个用于指示下溢。
结论
max-ide强大的宏预处理功能实现了maxq2000以及其他maxq20微控制器数据存储器软堆栈的直接替换。这一软堆栈使子程序更加模块化,可重复使用,有助于大规模汇编应用程序的开发。堆栈还支持堆栈错误的探测。


使用Arduino Uno和超声波传感器制作智能盲杆
iQOO Pro曝光采用了竖排三摄设计搭载骁龙855 Plus会拥有5G版本
韩亚航空计划从4月底开始将A350宽体客机投入到多个航线使用
iPhone8什么时候上市,iPhone8预售时间,iPhone8价格,据说前期2重磅新功能不能用
微型RGB视频多路复用器以100MHz切换像素
在MAXQ2000数据存储器中实现软堆栈
车载显示屏的发展朝向——大屏化、多屏化、高清化
iphone8什么时候上市?最新消息:iPhone8开启刷脸时代,三款新机来袭,价格震撼!
国产充电宝什么牌子好用?国产好用的充电宝推荐
【行业洞察】巨头纷纷入场,智慧教育硬件市场吸引力有多大
5G开启的物联网时代谁能跑赢
华阳集团旗下再添两家控股子公司入选“广东省智能制造生态合作伙伴名单”
儿童智能手表实测:娃丢了,还能找回来吗?
酷爽EDC-voopoocaliber&rex评测 值不值得买
美图M8和M6S哪个好?美图手机不仅仅只是高颜值摄像而已
苹果iOS10.3.2 Beat1更新了什么?又是修复BUG
英美两国欲将在人工智能领域建立合作伙伴关系?
零元购机 三星巧布中国5G市场局
用户首选X86 ARM Win10笔记本很难做
智原推出14纳米ASIC整合设计服务迈向人工智能新时代