This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

MSP430F5529捕获频率问题求解

Other Parts Discussed in Thread: MSP430F5529

在网上看到一篇高精度测频模块论文,我改了一下 ,在闸门时间内,用F5529TA2CCR0捕获测频,可是好像有很大问题,一是捕获值会出现负值,不应该是在0-65535内吗?二是频率误差非常大,不知道为什么?这是我修改的代码,希望大家探讨指点。#include <msp430f5529.h>
#include <stdint.h>
#include <stdio.h>
#include "Dogs102x6.h"
#define CPUF ((double)1048576)
#define delay_us(x) __delay_cycles((long)(CPUF*(double)x/1000000))
#define delay_ms(x) __delay_cycles((long)(CPUF*(double)x/1000))
#define delay_s(x) __delay_cycles(CPUF*x)
#define ulint unsigned long int

/*
* main.c
*/
ulint captar=0,pulse=0,time=0,capfirst,caplast,flag=0,f;
void main(void)
{
  WDTCTL = WDTPW + WDTHOLD;                      // 关闭看门狗
  lcdinit();
  P2DIR &= ~BIT3;
  P2SEL |=BIT3;
  TA2CCTL0=CM_1+SCS+CCIS_0+CAP+CCIE;
  TA2CTL=TASSEL_1+MC_2+TAIE;
  _EINT();
  while(1){
          if(flag==1){
          f=pulse*32768/time;
          display(f);//显示程序

          flag=0;

          }
  }
  }
#pragma vector = TIMER2_A0_VECTOR
__interrupt void TIMER2_A0 (void)
{
        if(captar==0)//第一次捕获上升沿
                {
                        capfirst=TA2CCR0;
                        captar++;
                }
        else{
                        caplast=TA2CCR0;
                        captar++;
                }
}
#pragma vector = TIMER2_A1_VECTOR
__interrupt void TIMER2_A1 (void)
{

        //TA2CTL&=~TAIE;
        switch(TA2IV){
        case 14:{//开启闸门时间2s;
                if(captar==0)
                        pulse=0;
                else{
                        //TA2CTL&= ~TAIE;
                        pulse=captar-1;待测信号个数
                        time=caplast-capfirst;标准信号个数
                        captar=0;
                        flag=1;//置位标志
                }
                break;
        }
        default:break;
}
        //TA2CTL |= TAIE;
}

这是我参考的网上资料wenku.baidu.com/link

  •     if(captar==0)//第一次捕获上升沿
                    {
                            capfirst=TA2CCR0;
                            captar++;
                    }
            else{
                            caplast=TA2CCR0;
                            captar++;
                    }

    这个代码应该是有问题的,记录第一次的TA2count值和最后一次TA2count值,如果中间出现TIMER溢出你没有记录,所以会出现负值。

  • 你发布了重复帖子,没必要发多次,查看你最后一个帖子。如果问题没有解决,你在原帖上提出就好。谢谢!

  • 这个问题被人问起,查看论坛中其它帖子。我们正在尝试解决。

  •  或者换个思路也可以

  •  case 14:{//开启闸门时间2s;
                    if(captar==0)
                            pulse=0;
                    else{
                            //TA2CTL&= ~TAIE;
                            pulse=captar-1;待测信号个数
                            time=caplast-capfirst;标准信号个数
                            captar=0;
                            flag=1;//置位标志


    case 14就是处理当主计数器的值大于65535产生溢出问题,也就是相当于2S一个闸门。

  •   在两秒之内capfirst 和caplast 的值,就应该在65535之内,只要溢出TAIFG(case14)置位,然后就会复位,得到此时标准脉冲信号被测信号个数,进行计算。

  • 不放在ISR里把溢出中断显示出来,实际验证下是不是真的是因为溢出产生的问题。

  • 你的误差很大是什么概念?可能会有一到两个数值的误差,不应该很大

  • 为什么用下面程序测频时,感觉不是很对,误差很大。

    ulint captar=0,capfirst,caplast,time=0,pulse=0,f;
    void main(void)
    {
    WDTCTL = WDTPW + WDTHOLD; // 关闭看门狗
    lcdinit();
    P2DIR &=~BIT0;
    P2SEL |=BIT0;
    TA1CCTL1=CM_1+SCS+CCIS_0+CAP+CCIE;
    TA1CTL=TASSEL_1+MC_2+TACLR+TAIE;
    _EINT();
    while(1){
    f=pulse*32768/time;
    display(f);
    TA1CCTL1 |= CCIE;
    TA1CTL |= TAIE;
    }
    }
    #pragma vector = TIMER0_A1_VECTOR
    __interrupt void TIMER0_A1 (void)
    {
    switch(TA1IV){
    case 2:{
    if(captar==0)
    {
    capfirst=TA1CCR1;
    captar++;
    }
    else{
    caplast=TA1CCR1;
    captar++;
    }
    TA1CCTL1 &=~CCIFG;
    break;
    }
    case 14:{
    if(captar==0)
    pulse=0;
    else{
    pulse=captar-1;
    time=caplast-capfirst;
    captar=0;
    }
    TA1CCTL1&=~CCIE;
    TA1CTL=~TAIE;
    break;
    }
    default:break;
    }
    }

  • 频率测量,我用TA2CCR0捕获上升沿,为什么捕获值为负值,范围不应该是0—65535吗??这是我的程序。

    #include <msp430f5529.h>
    #include <stdint.h>
    #include <stdio.h>
    #include "Dogs102x6.h"
    #define CPUF ((double)1048576)
    #define delay_us(x) __delay_cycles((long)(CPUF*(double)x/1000000))
    #define delay_ms(x) __delay_cycles((long)(CPUF*(double)x/1000))
    #define delay_s(x) __delay_cycles(CPUF*x)
    #define ulint unsigned long int

    /*
    * main.c
    */
    ulint captar=0,pulse=0,time=0,tete;
    long capfirst,caplast;
    ulint t,f,flag=0;

    void main(void)
    {
    WDTCTL = WDTPW + WDTHOLD; // 关闭看门狗
    lcdinit();
    P2DIR &= ~BIT3;
    P2SEL |=BIT3;
    //TA2CCTL2=CCIE; //捕获比较器2开启中断 TA0CCR2=26214;
    //TA2CCR2=26214;
    TA2CCTL0=CM_1+SCS+CCIS_0+CAP+CCIE;
    TA2CTL=TASSEL_1+MC_2+TAIE;
    _EINT();
    while(1){
    if(flag==1){
    t=pulse;
    //t=f;
    //t=1234/55367*43768;
    //f=(int)(t);
    display(t);
    flag=0;
    Dogs102x6_stringDraw(5,5,"1",0);
    //TA2CTL |= TAIE;
    //delay_s(0.5);
    TA2CTL |= TAIE;
    }
    }
    }
    #pragma vector = TIMER2_A0_VECTOR
    __interrupt void TIMER2_A0 (void)
    {
    if(captar==0)
    {
    capfirst=TA2CCR0;
    captar++;
    }
    else{
    caplast=TA2CCR0;
    captar++;
    }
    }
    #pragma vector = TIMER2_A1_VECTOR
    __interrupt void TIMER2_A1 (void)
    {

    //TA2CTL&=~TAIE;
    switch(TA2IV){
    case 14:{
    if(captar==0)
    pulse=0;
    else{
    //TA2CTL&= ~TAIE;
    pulse=captar-1;
    time=caplast-capfirst;
    tete=capfirst-caplast;
    //Dogs102x6_numDisplay(4,58,time,0);
    //Dogs102x6_numDisplay(5,58,tete,0);
    //Dogs102x6_numDisplay(6,58,capfirst,0);
    //Dogs102x6_numDisplay(6,58,caplast,0);显示程序
    captar=0;
    flag=1;
    }
    break;
    }
    default:break;
    }
    //TA2CTL |= TAIE;
    }