找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5616|回复: 2
收起左侧

一种智能频率计的设计与制作(AVR) 带仿真与源程序和原理图

[复制链接]
ID:137190 发表于 2016-9-8 21:13 | 显示全部楼层 |阅读模式
0.png
基于avr单片机的频率计的仿真原理图

0.png

俺毕业设计的小玩意.也许有人可以拿去玩.

当时刚出毕业选题就赶紧的选了这个简单的.免得以后为毕业设计费神.
这是最初的,后来给学校也是交的这个. 花了三天时间.

//*********************11月29日开始毕业选题
读了一点频率计原理,高频记数,低频测周期 规划了一下大体,坚决不用老师指定的51,
//*********************11月30日开始写代码
凑合着开始了,偏差很大,很无聊,晚上跟123JJ在51hei灌水,灌了阵找到思维了,继续写,基本完成
//*********************12月1日调试
写了个串口上位机对数据进行分析和记录,方便多了.仔细的调了下.不管他效率和规范了,出结果了就要得,马虎的收工了.
16MHz晶振,范围1Hz--5MHz,软件仿真测周期误差<1/8us.计数误差最大可能1hz. 当然液晶显示误差就大多了.显示位数有限.随便忽悠一下, 可以给老师交差了.前端信号处理和pcb以后再弄.

后来前端处理参照冬瓜哥的精华帖和建议,把CD4069作运放用.
感谢冬瓜哥给我打板,搭顺路车.

后来在寒假发神经又琢磨了一下,修改下代码,加上些别的东西,反正资源还足够,有些作了测试,有些只是空想,规划和写代码,没有测试.脉宽和占空比,电压表,示波器,欧姆表,信号发生器,温度计,电容表,电感表,逻辑仪,红外分析,弄了5,6天又停了,人太懒.再后来又忘干净了,只记得电容表是测充电时间.
再后来,写毕业论文,网上下了n多资料和论文用来copy,弄的电脑很乱,写到一半,实在太乱,文件整理中,大把的删文件夹,一不小心把软件都删了,还好在qq群共享里留下了这个最初版本.好险.差点作无用功.

0.png

0.png


下面是智能频率计的原理图是用ad画的.
0.png

下面是主程序的源程序:
  1. #ifndef main_c   //test ok
  2.   #define main_c
  3. #endif

  4. #include "main.h"
  5. //************************全局变量

  6. volatile uint t0_cont;

  7. volatile uint t1_cont;
  8. volatile uchar flg1; //
  9. //-------------------------------------------------------------------------
  10. //     测试函数 用来初始化
  11. //-------------------------------------------------------------------------
  12. void test()
  13. {
  14.   send_str( "pinlvji" ); //完成后把所有串口去掉
  15.   showstr( "cymometer", 1, 1, 9 ); //液晶显示
  16. //  showstr( "1KHz", 12, 1, 4 );
  17.   showstr( "Hz", 14, 2, 2 );
  18. }

  19. //-------------------------------------------------------------------------
  20. //mcu初始化 端口 定时器 中断 外中断 变量
  21. //-------------------------------------------------------------------------
  22. void mcu_init()
  23. {
  24. //******************端口
  25.   data_direc = out; //输出
  26.   DDRB = out; //输出
  27.   DDRC = out; //输出
  28.   PORTC = 0X00; //输出0

  29.   DDRB=0XE0;   //输入口      
  30.   PORTB=0XE9;
  31. //********************定时器
  32.   TIMSK |= 0x12; //定时中断允许位 只允许两个匹配
  33. }

  34. //-------------------------------------------------------------------------
  35. //       主函数
  36. //-------------------------------------------------------------------------
  37. int main()
  38. {
  39.   uchar f_t; //用测频还是测周
  40. //******************
  41.   mcu_init();                 //初始化
  42.   usart_init( 57600 );        //初始化
  43.   send_str( "com ok" ); //完成后把所有串口去掉
  44.   yj_set( 0x38, 0x0f, 0x06 ); //初始化
  45.   clearall(); //初始化
  46.   mzd_on();   //开中断

  47.   #if debug
  48.     test();
  49.   #endif

  50.   while ( 1 )
  51.   {
  52.     if ( !( PINB& ( 1 << 3 )))
  53.     {
  54.       refish();       //刷新所有的东西
  55.       f_t = hz_try(); //估计范围 1k
  56.       if ( f_t )
  57.       {
  58.         cl_f(); //频率法
  59.       }
  60.       else
  61.       {
  62.         cl_t(); //周期法
  63.       }
  64.     }
  65.   }
  66.   return 1;
  67. }
复制代码
cale.c
  1. #ifndef cale_c
  2.   #define cale_c
  3.   #include "main.h"
  4. //-------------------------------------------------------------------------
  5. // //周期法   先笨方法
  6. //-------------------------------------------------------------------------
  7.   void cale_t( uchar* back )
  8.   {
  9.     ulint count_all; //存放周期 max 16m min 16k
  10.     uint z_sh;      
  11.     ulint y_sh;     
  12.     uchar test;      
  13. //*******************整数小数分离  
  14.     count_all = t1_cont * 50000+TCNT1;
  15.         if(count_all>max_t)
  16.         count_all=max_t;
  17.     z_sh = max_t / count_all;         
  18.         #if pc
  19.     y_sh=max_t-count_all*z_sh;
  20.         y_sh*=10;
  21.     count_all/=100;
  22. #else
  23.     y_sh = ( max_t % count_all )* 1000;
  24. #endif
  25. //*******************整数计算部分
  26.     *back++ = z_sh / 100;
  27.     z_sh = z_sh % 100;
  28.     *back++ = z_sh / 10;
  29.     *back++ = z_sh % 10;
  30. //***************小数计算
  31.     z_sh = y_sh / count_all;
  32.     *back++ = z_sh / 100;
  33.     z_sh = z_sh % 100;

  34.     *back = z_sh / 10;
  35.     z_sh = z_sh % 10;
  36. //************做四舍五入计算
  37.     test=0;
  38.     if ( z_sh >= 5 )
  39.     {
  40.         send_char(z_sh);
  41.       *back += 1; //
  42.       if ( *back == 10 )
  43.       {
  44.             test++;
  45.         *back-- = 0;
  46.         *back += 1;
  47.         if ( *back == 10 )
  48.         {
  49.             test++;
  50.           *back-- = 0;
  51.           *back += 1; //
  52.           if ( *back == 10 )
  53.           {
  54.             test++;
  55.             *back-- = 0;
  56.             *back += 1; //
  57.             if ( *back == 10 )
  58.             {
  59.             test++;
  60.               *back-- = 0;
  61.               *back += 1;
  62.             }
  63.           }
  64.         }
  65.       }
  66.     }
  67.     back-=(4-test);

  68. //****************转到ascii码
  69.     for ( test = 5; test > 0; test-- )
  70.     {
  71.       *back++ = * back + 0x30;
  72.     }
  73. back-=5;
  74. //****************插进小数点
  75.     back[5] = back[4];
  76.     back[4] = back[3];
  77.     back[3] = '.';

  78. //****************去掉不要的0

  79.     for ( test = 3; test > 0; test-- )
  80.     {
  81.       if ( *back == 0x30 )
  82.       {
  83.         *back++ = * back - 0x30;
  84.       }
  85.       else
  86.       {
  87.         break;
  88.       }
  89.     }
  90.   }

  91. //-------------------------------------------------------------------------
  92. //       //记数法
  93. //-------------------------------------------------------------------------
  94.   void cale_f( uchar* back )
  95.   {
  96.     ulint count_all;  
  97.     uchar test;      

  98.     count_all = t1_cont * 50000+TCNT1; //原始值
  99. //*****************计算 笨方法计算
  100.     *back++   = count_all / 1000000; //最高位
  101.     count_all = count_all % 1000000; //去掉最高位
  102.     *back++   = count_all / 100000; //100k位
  103.     count_all = count_all % 100000; //去掉
  104.     *back++   = count_all / 10000;
  105.     count_all = count_all % 10000;
  106.     *back++   = count_all / 1000;
  107.     count_all = count_all % 1000;
  108.     *back++   = count_all / 100;
  109.     count_all = count_all % 100;
  110.     *back++   = count_all / 10;   //十位
  111.     *back     = count_all % 10;   //个位

  112.     back = back - 6;

  113. //转到ascii字符
  114.     for ( test = 7; test > 0; test-- )
  115.     {
  116.       *back++ = * back + 0x30;
  117.     }
  118.     back = back - 7; //指针退回
  119. //处理前面的0
  120.     while ( *back == 0x30 )
  121.     {
  122.       *back++ = * back - 0x30;
  123.     }
  124.   }
  125. #endif
复制代码


0.png

所有资料下载:
一种智能频率计的设计与制作(AVR).rar (132.94 KB, 下载次数: 59)
回复

使用道具 举报

ID:82939 发表于 2016-10-28 14:45 | 显示全部楼层
学了一段时间了,正好给自己一个练习的机会
回复

使用道具 举报

ID:28656 发表于 2017-2-13 17:13 | 显示全部楼层
学习学习
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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