单片机下电,如果电压由正常的工作电压3.3V降到0V的过程时长在500ms左右(使用主频16M),这样会不会在电压到下降的这个过程中导致单片机程序跑飞,如果单片机程序中有擦写Flash的函数(函数有编译),如果频繁下电的测试操作,那跑飞有可能执行到擦写Flash的函数是吧?近期下电测试就有遇到这些问题。论坛里,有人遇到这样的问题吗?
gaoyang9992006:
所以你要设置欠压复位啊。
gaoyang9992006:
所以单片机都提供了欠压复位的功能,就是防止出现这种问题。而这款单片机提供了这个功能,你不用怎么可以,如果应用环境需要这种频繁的下电上电操作。肯定要启动欠压复位功能,另外也可以提供电容稳压电路,防止这种极其短暂的掉电。
HG:
当然会有这样的风险,而且到不了0V,16M的主频到2.2V估计就差不多了,具体你可以看一下2955的规格书。电压不足以支撑主频的时候会有未知的风险,BOR也是不能完全防得住的,毕竟BOR的电压要低一些。
一般情况下,可以加大电容,让电压降的慢一下,或者减小主频,让安全电压下限更低。会有所改善。
user4633009:
回复 gaoyang9992006:
看了MSP430G2955的寄存器,这单片机没有欠压复位的功能,没有相关的寄存器可以操作
user4633009:
回复 HG:
加了电容电压下降的不是更缓慢,如果程序跑飞,那就有更多的时间去运行跑飞后的程序?应该让电压降的更快一些?
可能降主频会更好些,正打算测试一下直接运行低主频的(1M),再连续测试下电看看效果
Gary Gao:
回复 user4633009:
你好
我有一个办法你可以测试一下:你可以使用ADC去检测电压电压(不需要外部电路)ADC的采集通道可直接采用内部参考电压(一般为1.5V,具体请查看技术手册)进行采样,且ADC的参考电压是可以配置为电源电压。(这种电源电压检测的方法有官方例程的)。然后当检测到电压低时可以进入低功耗模式关闭CPU。
user4633009:
回复 Gary Gao:
关闭CPU可以防止程序跑飞吗?ADC采样的方法也想过,但是没法实时采样,因为程序还要跑其他任务,还是可能出现还没采样就出问题了
Gary Gao:
回复 user4633009:
ADC可以设置成定时器或RTC触发然后用窗口比较器比较整个过程不需要CPU干预的,CPU可以做其它的事情。你可以先不用做这么复杂,先用CPU进行判断测试一下这个方法。
user4633009:
回复 Gary Gao:
CPU干预的ADC可以测试看看,但用比较器的这个要怎么用?
Gary Gao:
回复 user4633009:
给你一个参考代码,但这个是在MSP430FR4133上测试的,到G2你要再稍微修改一下
#include <msp430.h>
#define High_Threshold 568
int main(void){ WDTCTL = WDTPW | WDTHOLD; // Stop WDT P1REN = 0xff; P2REN = 0xff; P3REN = 0xff; P4REN = 0xf9; P5REN = 0xff; P6REN = 0xff; P7REN = 0xff; P8REN = 0x0f; P1OUT =0; P2OUT =0; P3OUT =0; P4OUT =0; P5OUT =0; P6OUT =0; P7OUT =0; P8OUT =0; P8DIR=BIT0;//P8.0 SMCLK P8SEL0 |= BIT0; // SYSCFG2 |= ADCPCTL1; P4SEL0 |= BIT1 | BIT2; // set XT1 pin as second function // Disable the GPIO power-on default high-impedance mode to activate // previously configured port settings PM5CTL0 &= ~LOCKLPM5;
do { CSCTL7 &= ~(XT1OFFG | DCOFFG); // Clear XT1 and DCO fault flag SFRIFG1 &= ~OFIFG; } while (SFRIFG1 & OFIFG); // Test oscillator fault flag
__bis_SR_register(SCG0); // disable FLL CSCTL3 |= SELREF__XT1CLK; // Set XT1CLK as FLL reference source __delay_cycles(3); __bic_SR_register(SCG0); // enable FLL while(CSCTL7 & (FLLUNLOCK0 | FLLUNLOCK1)); // Poll until FLL is locked
CSCTL4 = SELMS__DCOCLKDIV | SELA__XT1CLK; // Set ACLK = XT1CLK = 32768Hz
// Configure ADC10 ADCCTL0 &= ~ADCENC; // Disable ADC ADCCTL0 = ADCSHT_2 | ADCON; // ADCON, S&H=16 ADC clks // ADCCTL1 = ADCSHP|ADCSSEL_1|ADCSHS_1|ADCCONSEQ_2; // sample_timer,ACLK,rtc,repeat_single ADCCTL1 = ADCSHP|ADCSHS_1|ADCCONSEQ_2; // sample_timer,MODCLK,rtc,repeat_single ADCHI = High_Threshold; // Window Comparator Hi-threshold ADCMCTL0 = ADCINCH_13 | ADCSREF_0; ADCIE = ADCHIIE; ADCCTL2 = ADCRES|ADCSR;
// Configure reference module located in the PMM PMMCTL0_H = PMMPW_H; // Unlock the PMM registers PMMCTL2 |= INTREFEN; // Enable internal reference while(!(PMMCTL2 & REFGENRDY)); // Poll till internal reference settles __delay_cycles(400); // Delay for reference settling
RTCMOD = 0x7; // Initialize RTC RTCCTL = RTCSS_3 | RTCSR | RTCPS__1000 ;//VLOCLK, ADCCTL0 |= ADCENC; // Sampling and conversion start while(1) { // __delay_cycles(90000); __bis_SR_register(LPM3_bits | GIE); // Enter LPM0, ADC_ISR will force exit }}
// ADC interrupt service routine#if defined(__TI_COMPILER_VERSION__) || defined(__IAR_SYSTEMS_ICC__)#pragma vector=ADC_VECTOR__interrupt void ADC_ISR(void)#elif defined(__GNUC__)void __attribute__ ((interrupt(ADC_VECTOR))) ADC_ISR (void)#else#error Compiler not supported!#endif{ switch(__even_in_range(ADCIV, ADCIV_ADCIFG)) { case ADCIV_NONE: break; case ADCIV_ADCOVIFG: break; case ADCIV_ADCTOVIFG: break; case ADCIV_ADCHIIFG: __delay_cycles(2);//alarming __bic_SR_register_on_exit(LPM3_bits);// Exit from LPM break; case ADCIV_ADCLOIFG: break; case ADCIV_ADCINIFG: break; case ADCIV_ADCIFG: // adcResult = ADCMEM0; // Read ADC memory // __bic_SR_register_on_exit(LPM0_bits);// Exit from LPM break; default: break; }}