在 AVR 中断中保留 sreg
Posted
技术标签:
【中文标题】在 AVR 中断中保留 sreg【英文标题】:Preserving sreg in AVR interrupts 【发布时间】:2012-01-17 20:48:44 【问题描述】:在 AVR 微控制器中保存状态寄存器 sreg 的机制是什么? RETI
表示这些位不在堆栈上。通用寄存器之一是否也是 sreg 或类似的东西?
【问题讨论】:
状态寄存器不是通用寄存器。它驻留在 I/O 寄存器文件中,可以使用in
和 out
指令访问。
【参考方案1】:
这在每个 AVR 数据表中都有说明。例如,ATtiny2313 datasheet 第 8 页上写着:
状态寄存器在进入中断程序时不会自动存储,并在从中断返回时恢复。这必须由软件处理。
您可以通过将其存储在临时寄存器中来实现:
interrupt:
in r16, SREG ; save SREG
...
out SREG, r16 ; restore SREG
reti
另外请注意,如果您访问的寄存器并非专门用于此中断例程,您也需要保存这些寄存器。此外,如果您的寄存器不足,您可以将 SREG 的值推送到堆栈:
interrupt:
push r16 ; save global registers on stack
push r17
push r18
in r16, SREG ; save SREG
push r16 ; do this if you want to use r16 in your interrupt routine
...
pop r16 ; do this if you pushed SREG above
out SREG, r16 ; restore SREG
pop r18 ; restore global registers
pop r17
pop r16
reti
更多信息请看here。
【讨论】:
【参考方案2】:或者
PUSH Rn
LDS Rn, SREG
PUSH Rn
和
POP Rn
STS SREG, Rn
POP Rn
似乎是有效的。
【讨论】:
关闭!但这并不能保存 SREG——我认为它保存了 EECR 寄存器。但至少我已经验证上述不会保存/恢复 SREG。 ...IN 和 OUT 助记符对数据寄存器 0 到 0x3F 进行操作。当您使用 LDS 或 STS 访问相同的位置时,您需要为其添加 0x20。在这种情况下,您将使用LDS Rn, SREG + 0x20
和 STS SREG + 0x20, Rn
我很惊讶近五年没有人发现它!以上是关于在 AVR 中断中保留 sreg的主要内容,如果未能解决你的问题,请参考以下文章