在 AVR 中断中保留 sreg

Posted

技术标签:

【中文标题】在 AVR 中断中保留 sreg【英文标题】:Preserving sreg in AVR interrupts 【发布时间】:2012-01-17 20:48:44 【问题描述】:

在 AVR 微控制器中保存状态寄存器 sreg 的机制是什么? RETI 表示这些位不在堆栈上。通用寄存器之一是否也是 sreg 或类似的东西?

【问题讨论】:

状态寄存器不是通用寄存器。它驻留在 I/O 寄存器文件中,可以使用 inout 指令访问。 【参考方案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 + 0x20STS SREG + 0x20, Rn 我很惊讶近五年没有人发现它!

以上是关于在 AVR 中断中保留 sreg的主要内容,如果未能解决你的问题,请参考以下文章

汇编中的 AVR 外部中断触发

AVR开发 Arduino方法 中断子系统

AVR的UART接受中断问题,中断老是不能正常触发

AVR_Interrupt

Arduino/AVR:中断串行/I2C 通信是不是安全

AVR - 高速中断驱动的 UART 代码不工作