《AVR单片机原理与GCC编程实践》P117讲到AVR代码段时,提到了.bss段中包含了一个很有意思的.noinit段。
.noinit段的定义方法:int bar __attribute__ ((section (".noinit")));
不能指定初值,因为这些变量是在 .bss段。也就是说、如果我在.noinit段定义一个变量,按下开发板的复位键(单片机不掉电复位),这个变量的值不会被清0。
下面是测试代码,我在.noinit段定义了变量bar,初值未知。
8bit的bar正好可以用连接在PORTB的8个LED灯来显示它的数值。
每次进入程序后、bar++,所以每按一次复位键,bar对应的8个LED灯的二进制数值就加1(二进制计数器)。
#include
uint8_t bar __attribute__ ((section(".noinit")));
int main(void)
{
bar++;
DDRB = 0xFF;
PORTB = 0xFF;
PORTB = bar;
while(1) { }
return 0;
}
我的8个LED是低推的,所以点亮的LED对应的引脚逻辑值是0,熄灭的LED对应的引脚逻辑值是1。
于是得到下面的结果:
程序下载到开发板后,这一次bar的初值是00000001(这次比较巧),每按一次复位键,bar的值都加1了。
重新下载程序后,bar的初值也会自己加1,也相当于单片机不掉电复位。
------------------------------------------------------------------------------------------------------------------------
2016.05.30
今天发现.noinit的特性还可以测试BOD掉电。
依然是使用上面的代码中的.noinit,加上PA0输出脉冲,同时查看PORTB的二进制计数器来得知复位次数。
int main(void)
{
// PA0为输出口
DDRA = (IO_OUTPUT << DDA0);
// 记录复位次数
_noinit_using();
while(1)
{
// PA0拉高
PORTA = (1 << PA0);
// 维持0.5秒
delay_ms(500);
// PA0拉低
PORTA &= ~(1 << PA0);
// 维持0.5秒
delay_ms(500);
}
return 0;
}
其中的_noinit_using()如下:
volatile uint8_t v_noinit __attribute__ ((section(".noinit")));
// ==========================================================================================================
// 通过观察PORTB上的8位LED的状态得知复位次数
// ==========================================================================================================
void _noinit_using(void)
{
v_noinit++;
// 8位LED,低推
DDRB = 0xFF;
PORTB = 0xFF;
PORTB = v_noinit;
}
1、当前芯片BOD监测的电压是4.0V
2、使用.noinit可以测试复位次数(二进制计数器)
3、每次将电压源电压低到<3.7V,就会引发复位,而PORTB的二进制计数器就会加1
4、而且低到1V以下,复位后依然可以记住上一次的.noinit变量的值,大概是没有完全掉电的缘故
5、正常工作时可以看到PA0的脉冲,复位时中断。
------------------------------------------------------------------------------------------------------------------------
在另一篇测试WDT效果的文章中,还是用这个.noinit+二进制计数器的方法测试了WDT的复位:
http://blog.csdn.net/manon_des_source/article/details/51540217