找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2077|回复: 1
打印 上一主题 下一主题
收起左侧

基于msp430F149单片机的频测量+最高频率12MH

[复制链接]
跳转到指定楼层
楼主
ID:896677 发表于 2021-12-12 10:37 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
最高可采集到12MHZ,经过测试精度极高,可以精确到小数点后三位

单片机源程序如下:
  1. /*********************************************************************
  2. 程 序 名:基于msp430F149单片机的频率计
  3. 版   本 :PLj
  4. 时   间 :2015.11.17~2015.11.24
  5. 班   级        :电子1302班
  6. 功   能 :频率计                                                                       
  7. 使用说明:被测信号输入p1.0口;LCD12864显示;频率测量范围:1hz~12.5MHZ;
  8.           1hz~10MHZ 误差万分之一以内;
  9. *********************************************************************/
  10. #include <msp430x14x.h>
  11. #include "set_clock.h"
  12. #include "lcd12864.h"
  13. #include "delay.h"

  14. #define uchar unsigned char
  15. #define uint  unsigned int

  16. uchar xianshi1[]; //  定义显示数组
  17. long int pinlvzhi;  //  频率计数常量
  18. int number;         //  中断计数常量
  19. uint old_pinlvzhi,new_pinlvzhi;// 计数器原始值、计数器更新值

  20. void set_TimerA_TACTL(void);  //设置定时器A的寄存器TACTL,TAR清零,开启溢出中断,使用外部引脚TACLK输入。
  21. void fenchai();               //把频率的数值分拆放入显示数和去掉高位为0的数值

  22. /*********************************************************************
  23. 函数名:void main()
  24. 功能:  主函数;                                                                               
  25. *********************************************************************/
  26. void main()
  27. {
  28.     WDTCTL=WDTPW+WDTHOLD;
  29.     set_clock();                 //设置MSP430工作时钟;
  30.     lcd12864_init();             //LCD12864初始化函数;
  31.     set_TimerA_TACTL();          //设置定时器A的寄存器TACTL,TAR清零,开启溢出中断,使用外部引脚TACLK输入。
  32.     number=0;                    //中计数常量初始化;
  33.     P1DIR=0x00;                  //P1 I/O模式为输入;
  34.     P1SEL=0xff;                  //P1 I/O选择第二功能;外部引脚输入;
  35.     IE1 |= WDTIE;                //开启看门狗定时器中断;
  36.     _EINT();                     // 总中断开
  37.     lcd12864_display(1,3,"测频率");   
  38.     TACTL|=MC_2;                //连续计数模式 ;
  39.     Delay_ms(1);                //给MCU反应时间;
  40.     WDTCTL=0x5a1c;              //看门狗定时器定时一秒(WDTPW+WDTTMSEL+WDTCNTCL+WDTSSEL);
  41.     old_pinlvzhi=TAR;           //读取TAR计数器原始值;
  42.     while(1)
  43.     {
  44.         fenchai();
  45.         lcd12864_display(2,1,xianshi1);
  46.     }
  47. }

  48. /*********************************************************************
  49. *函数名:void set_TimerA_TACTL(void)
  50. *功  能:设置Timer_A的TACTL控制寄存器、
  51. **********************************************************************/
  52. void set_TimerA_TACTL(void)
  53. {  
  54.   TACTL|=TACLR;           //Timer_A计数器清零位
  55. /********************  选择Timer_A的输入时钟源  *********************/
  56.    TACTL|=TASSEL_0;        //Timer_A时钟源选择:TACLK(使用外部引脚输入)
  57.     TACTL|=TAIE;            //Timer_A溢出中断使能允许位
  58. }
  59. /*********************************************************************/


  60. /*********************************************************************
  61. *函数名:void fenchai()
  62. *功  能:把频率的数值分拆放入显示数和去掉高位为0的数值
  63. **********************************************************************/
  64. void fenchai()
  65. {          xianshi1[0]=pinlvzhi/10000000+'0';  //频率值高位
  66.           xianshi1[1]=pinlvzhi/1000000%10+'0';
  67.           xianshi1[2]=pinlvzhi/100000%10+'0';
  68.           xianshi1[3]=pinlvzhi/10000%10+'0';
  69.           xianshi1[4]=pinlvzhi/1000%10+'0';
  70.           xianshi1[5]=pinlvzhi/100%10+'0';
  71.           xianshi1[6]=pinlvzhi/10%10+'0';
  72.           xianshi1[7]=pinlvzhi%10+'0';
  73.           xianshi1[8]='H';                    //频率值低位
  74.           xianshi1[9]='z';                    //频率值低位
  75.           xianshi1[10]=0x20;  
  76.           xianshi1[11]=0x20;         
  77.          
  78.           if(xianshi1[0]=='0')                  //去掉最高位的0
  79.         {
  80.                 xianshi1[0]=0x20;
  81.                 if(xianshi1[1]=='0')
  82.                 {
  83.                         xianshi1[1]=0x20;
  84.                         if(xianshi1[2]=='0')
  85.                         {
  86.                                 xianshi1[2]=0x20;
  87.                                 if(xianshi1[3]=='0')
  88.                                 {
  89.                                         xianshi1[3]=0x20;
  90.                                         if(xianshi1[4]=='0')
  91.                                         {
  92.                                                 xianshi1[4]=0x20;
  93.                                                 if(xianshi1[5]=='0')
  94.                                                 {
  95.                                                         xianshi1[5]=0x20;
  96.                                                         if(xianshi1[6]=='0')
  97.                                                        {
  98.                                                             xianshi1[6]=0x20;
  99.                                                         
  100.                                                        }
  101.                                                 }
  102.                                         }
  103.                                 }
  104.                         }
  105.                 }
  106.         }  
  107. }
  108. /*********************************************************************/

  109. /********************************************************************
  110. *函数名:void set_TimerA_TACTL(void)
  111. *功  能:设置Timer_A的TACTL控制寄存器、
  112. ***************  TimerA CCR2的捕获中断  ****************************/
  113. #pragma vector=TIMERA1_VECTOR
  114. __interrupt void TimerA_I(void)
  115. {
  116.     switch(TAIV)
  117.         {
  118.             case 2:
  119.                     break;
  120.             case 4:
  121.                     break;
  122.             case 10:
  123.                     number++;//中断计数常量溢出加一
  124.                     break;
  125.         }
  126. }
  127. /********************************************************************/


  128. /********************************************************************
  129. 看门狗中断服务子程序
  130. ********************************************************************/      
  131. #pragma vector = WDT_VECTOR
  132. __interrupt void WDT_IRQ(void)
  133. {
  134.     new_pinlvzhi=TAR;//得到计数器计数后的值
  135.     if(number==0)////中断计数常量溢小于1
  136.     {
  137.        pinlvzhi=new_pinlvzhi-old_pinlvzhi;//得到频率值=计数器现在的值-计数器原始值
  138.     }
  139.     else
  140.     {
  141.       pinlvzhi=new_pinlvzhi+number*65536-old_pinlvzhi;//得到频率值;
  142.     }   
  143.     number=0; //清零;
  144.     old_pinlvzhi=new_pinlvzhi;//把上一次计数器的计数值作为下一次的计数初值;
  145. }
  146. /*************************************END*******************************/
复制代码

代码下载: 代码.7z (92.24 KB, 下载次数: 18)

评分

参与人数 1黑币 +10 收起 理由
admin + 10 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩1
回复

使用道具 举报

沙发
ID:1027664 发表于 2022-5-18 20:47 | 只看该作者
牛哇,写的很整齐。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表