找回密码
 立即注册

QQ登录

只需一步,快速开始

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

ad7705的单片机源程序

[复制链接]
跳转到指定楼层
楼主
ID:309907 发表于 2018-4-17 13:44 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
ad7705单片机源程序如下:
  1. ------------AD7705头文件开始-------------------
  2. #ifndef _AD7705_H
  3. #define _AD7705_H

  4. //通讯寄存器地址定义
  5. #define     WR_SETUP_REG        0x10   //选中写设置寄存器
  6. #define     RD_SETUP_REG        0x18   //选中写设置寄存器
  7. #define     WR_CLOCK_REG     0x20   //选中写时钟寄存器
  8. #define RD_DATA_REG         0x38   //选中数据寄存器读
  9. #define     WR_OFFSET_REG      0x60   //选中写offset寄存器
  10. #define     RD_OFFSET_REG      0x68   //选中读offset寄存器
  11. #define     WR_FULL_REG        0x70   //选中写full scale寄存器
  12. #define     RD_FULL_REG        0x78   //选中读full scale寄存器

  13. #define     SYS_ZERO_CALI     0x80   //系统零校准模式
  14. #define     SYS_FULL_CALI     0xC0   //系统满量程校准模式

  15. #define ZERO_CALIBRATION    0x00 //系统零校准
  16. #define FULL_CALIBRATION    0x01 //系统满量程校准

  17. //CLOCK寄存器设置,无分频,50HZ输出更新速率
  18. #define CLOCK_REG_SET    0X04

  19. //函数声明
  20. void reset_AD7705(void);

  21. unsigned char read_AD7705_byte(void);
  22. unsigned int   read_AD7705_word(void);
  23. unsigned long int read_AD7705_dword(void);

  24. void write_AD7705_byte(unsigned char data);
  25. void write_AD7705_word(unsigned int data);
  26. void write_AD7705_dword(unsigned long int data);

  27. void ReadData7705(unsigned int *const pdata);
  28. void AD7705_calibration(void);
  29. void start_AD7705(void);

  30. #endif

  31. -----------AD7705主文件开始--------------
  32. #include <util/delay.h>
  33. #include <avr/eeprom.h>
  34. #include "ad7705.h"
  35. #include "main.h"
  36. #include "crc16.h"
  37. #include "Usart.h"

  38. //针对四个量程的设置寄存器的设置内容
  39. //(1)对于单极性V级别输入0-5V、0-20mA、0-10V这三个量程,输入范围为0-2V,无极性,增益为1,缓冲模式--0-2V
  40. //(2)对于双极性V级别输入+-2.5V、+-5V这两个量程,输入范围为+-1V,双极性,增益为2,缓冲模式--+-2V
  41. //(3)对于双精度mV级别输入+-500mV,增益为4,双极性,缓冲模式--+-2V
  42. //(4)对双精度mV级别+-50mV,增益为32,双极性,缓冲模式--+-1.6V
  43. //----MD1(0)    MD0(0)    G2(0)    G1(0)    G0(0)    B/U(0)    BUF(0)    FSYNC(0)--------------------//
  44. const unsigned char text_of_setup[4]={0X06,0X0A,0X12,0X2A}; //缓冲模式,数字滤波同步

  45. extern volatile unsigned char   command[7];    //校准命令全局数组
  46. extern volatile unsigned char   scale;         //记录系统量程
  47. extern volatile unsigned char   NO_CALI_TYPE; //未校准类型
  48. extern volatile unsigned long   int ZS,GS;    //当前量程的校准系数
  49. extern volatile unsigned char   time_count;    //超时标志

  50. //----------------------------------------------------------------------------
  51. //函数:reset_AD7705
  52. //功能:AD7705串行接口失步后将其复位。复位后要延时500us再访问
  53. //参数:无
  54. //返回:无
  55. //变量:无
  56. //备注:无
  57. //----------------------------------------------------------------------------
  58. void reset_AD7705(void)
  59. {
  60.       unsigned char i;
  61.      
  62.       AD_DIN1;
  63.       for( i=0; i<36; i++ )
  64.       {
  65.           AD_CLK0;
  66.           asm("nop");
  67.           asm("nop");
  68.           asm("nop");         
  69.           AD_CLK1;
  70.           asm("nop");
  71.           asm("nop");
  72.           asm("nop");         
  73.       }     
  74.       _delay_us(30);
  75. }

  76. //------------------------------------------------------------------------------------------
  77. //函数:read_AD7705_byte
  78. //功能:从AD7705读一个字节的数据
  79. //参数:无
  80. //返回:读到的一字节数据
  81. //变量:无
  82. //备注:无
  83. //------------------------------------------------------------------------------------------
  84. unsigned char read_AD7705_byte(void)
  85. {
  86.       unsigned char data = 0;
  87.       unsigned char i = 0;
  88.      
  89.       for( i=0; i<8; i++ )
  90.       {
  91.           data <<= 1;
  92.           AD_CLK0;         
  93.           asm("nop");
  94.           asm("nop");
  95.           asm("nop");
  96.           if(AD_DOUT)
  97.           {
  98.               data++;
  99.           }
  100.           AD_CLK1;         
  101.           asm("nop");
  102.           asm("nop");
  103.           asm("nop");
  104.       }     
  105.       return data;   
  106. }

  107. //------------------------------------------------------------------------------------------
  108. //函数:read_AD7705_word
  109. //功能:从AD7705读一个字的数据,共16bit
  110. //参数:无
  111. //返回:读到的一字节数据
  112. //变量:无
  113. //备注:无
  114. //------------------------------------------------------------------------------------------
  115. unsigned int read_AD7705_word(void)
  116. {
  117.       unsigned int data = 0;
  118.       unsigned char i = 0;
  119.      
  120.       for( i=0; i<16; i++ )
  121.       {
  122.           data <<= 1;
  123.           AD_CLK0;         
  124.           asm("nop");
  125.           asm("nop");
  126.           asm("nop");
  127.           if(AD_DOUT)
  128.           {
  129.               data++;
  130.           }
  131.           AD_CLK1;         
  132.           asm("nop");
  133.           asm("nop");
  134.           asm("nop");
  135.       }     
  136.       return data;   
  137. }


  138. //------------------------------------------------------------------------------------------
  139. //函数:read_AD7705_dword
  140. //功能:从AD7705读一个24的数据
  141. //参数:无
  142. //返回:读到的一字节数据
  143. //变量:无
  144. //备注:AD7705是一个16位AD
  145. //------------------------------------------------------------------------------------------
  146. unsigned long int read_AD7705_dword(void)
  147. {
  148.       unsigned long data = 0;
  149.       unsigned char i = 0;
  150.      
  151.       for( i=0; i<24; i++ )
  152.       {
  153.           data <<= 1;
  154.           AD_CLK0;         
  155.           asm("nop");
  156.           asm("nop");
  157.           asm("nop");
  158.           if(AD_DOUT)
  159.           {
  160.                data++;
  161.           }
  162.           AD_CLK1;         
  163.           asm("nop");
  164.           asm("nop");
  165.           asm("nop");
  166.       }     
  167.       return data;   
  168. }


  169. //------------------------------------------------------------------------------------------
  170. //函数:write_AD7705_byte
  171. //功能:往AD7705写8位数据
  172. //参数:IN - uint8_t   data,要写入AD7705的数据
  173. //返回:无
  174. //变量:无
  175. //备注:无
  176. //------------------------------------------------------------------------------------------
  177. void write_AD7705_byte(unsigned char data)
  178. {
  179.       for(unsigned char i=0; i<8; i++)
  180.       {
  181.           AD_CLK0;
  182.           if(data&0x80)
  183.               AD_DIN1;
  184.           else
  185.               AD_DIN0;         
  186.           asm("nop");
  187.           asm("nop");
  188.           asm("nop");
  189.           AD_CLK1;         
  190.           asm("nop");
  191.           asm("nop");
  192.           asm("nop");
  193.           data <<= 1;
  194.       }     
  195.       AD_DIN1;     
  196. }

  197. //------------------------------------------------------------------------------------------
  198. //函数:write_AD7705_dword
  199. //功能:往AD7705写24位数据,因为AD7705是24位的器件
  200. //参数:IN - int32_t   data,要写入AD7705的数据
  201. //返回:无
  202. //变量:无
  203. //备注:无
  204. //------------------------------------------------------------------------------------------
  205. void write_AD7705_dword(unsigned long int data)
  206. {
  207.      for(unsigned char i = 0; i<24; i++)
  208.      {
  209.           AD_CLK0;
  210.           if(data&0x800000)
  211.               AD_DIN1;
  212.           else
  213.               AD_DIN0;         
  214.           asm("nop");
  215.           asm("nop");
  216.           asm("nop");
  217.           AD_CLK1;         
  218.           asm("nop");
  219.           asm("nop");
  220.           asm("nop");
  221.           data <<= 1;
  222.       }     
  223.       AD_DIN1;
  224. }

  225. //------------------------------------------------------------------------------------------
  226. //函数:AD7705_calibration
  227. //功能:根据cali_type的值对AD7705进行系统0校正或系统满量程校正,并将各校正值和校正标志存入
  228. //       EEPROM,数据保存为双备份。数据块格式为: 内部0校正值(4byte), 内部满量程校正值(4byte),
  229. //          系统0校正值(4byte), 系统满量程校正值(4byte), 系统0校正标志(1byte),系统满量程校正标志
  230. //       (1byte),CRC16校验值(2byte),共20byte。
  231. //参数:IN - uint8_t board, 0 - 对主板进行校正,1-对副板进行校正
  232. //         IN - uint8_t range, 需要校正的量程
  233. //       IN - uint8_t cali_type, 校正类型,ZERO_CALIBRATION- 0校正;
  234. //            FULL_CALIBRATION - 满量程校正
  235. //返回:返回-1表示校准失败,非0表示校正成功,并返回相应的索引值
  236. //变量:无
  237. //备注:做满量程校正前必须先做零校正
  238. //------------------------------------------------------------------------------------------
  239. //校准命令格式   
  240. //STX     Data Long     Command Code     Parameter     CheckSum     ETX
  241. //0x55     数据长度(2)     量程指示     00H/01H         CRC16(2)     0x0D
  242. //
  243. //校准过程中要用到Command[]的数据,所以校准之前要关掉串口接收中断

  244. void AD7705_calibration(void)
  245. {
  246.       //记录读取EEPROM的次数
  247.       unsigned char readtimes =0;
  248.      
  249.       //记录上位机发送的校准量程类型
  250.       unsigned char cali_scale =0;
  251.      
  252.       //读取24位校准系数的临时变量
  253.       unsigned long int temp =0;
  254.      
  255.       //临时的校准系数数组,存放格式ZSL、ZSM、ZSH;GSL、GSM、GSH;CRCL、CRCH
  256.       //并在校准结束时作为参数传递给TXOUT()函数,发送校准系数给上位机
  257.       unsigned char coefficient[8] ={0}; //test[8]={0};
  258.      
  259.       //16位校验和的临时变量
  260.       unsigned int crcvalue =0;
  261.      
  262.       AD_CS1;
  263.      
  264.       cali_scale = command[2]; //获取上位机发送的要校准的量程类型
  265.       
  266.       //读取EEPROM的第一份校准系数
  267.       eeprom_busy_wait();
  268.       eeprom_read_block( &coefficient[0], (void*)(ADDR_EEPROM_1+(cali_scale-1)*10), 8 );
  269.       crcvalue = checksum( &coefficient[0], 6 ); //将6个值调用CRC校验函数得到校验
  270.       if( (coefficient[7]*256+coefficient[6]) != crcvalue )
  271.       {
  272.           readtimes++;        
  273.       }
  274.      
  275.       //如果校准系数不可用则读取第二份
  276.       if( 1 == readtimes )
  277.       {
  278.           eeprom_busy_wait();
  279.           eeprom_read_block( &coefficient[0], (void*)(ADDR_EEPROM_2+(cali_scale-1)*10), 8 );
  280.           crcvalue = checksum( &coefficient[0], 6 ); //将6个值调用CRC校验函数得到校验
  281.           if( (coefficient[7]*256+coefficient[6]) != crcvalue )
  282.           {
  283.               readtimes++;            
  284.           }
  285.       }
  286.      
  287.       //如果校准系数不可用则读取第三份
  288.       if( 2 == readtimes )
  289.       {
  290.           eeprom_busy_wait();
  291.           eeprom_read_block( &coefficient[0], (void*)(ADDR_EEPROM_3+(cali_scale-1)*10), 8 );         
  292.       }
  293.      
  294.       ADDR409_MASK; //切换到第一通道进行校准
  295.      
  296.       AD_CS0;
  297.       _delay_us(5);
  298.       reset_AD7705();
  299.      
  300.       //CLOCK寄存器设置,无分频,50HZ输出更新速率
  301.       write_AD7705_byte( WR_CLOCK_REG );
  302.       write_AD7705_byte( CLOCK_REG_SET );
  303.      
  304.       if( ZERO_CALIBRATION == command[3] ) //校准命令为零校准
  305.       {
  306.           //写设置寄存器,选择零校准
  307.           write_AD7705_byte( WR_SETUP_REG );
  308.           write_AD7705_byte( text_of_setup[cali_scale-1] | SYS_ZERO_CALI );
  309.          
  310.           //等待校准完成,系统校准延时时间
  311.           start_timer0();
  312.           while( time_count < time_sys_cali );
  313.           stop_timer0();
  314.          
  315.           while( AD_DRDY );//若将滤波器同步位FSYNC置为1,AD_DRDY信号将不会变低,这里将一直是死循环
  316.          
  317.           //读OFFSET寄存器
  318.           write_AD7705_byte( RD_OFFSET_REG );
  319.           temp = read_AD7705_dword();
  320.          
  321.           if( cali_scale == scale)
  322.           {
  323.               ZS = temp; //如果是当前量程零校准还要更新ZS
  324.               //如果是当前量程的校准,还要将NO_CALI_TYPE赋值为1表示已经经过零校准
  325.               //更新上电没有校准时readEEPROM()函数的运行状态
  326.               NO_CALI_TYPE = NO_FULL_CALIBRATION;            
  327.           }     
  328.          
  329.           coefficient[0] = (unsigned char)( temp%256 );
  330.           coefficient[1] = (unsigned char)( (temp/256)%256 );
  331.           coefficient[2] = (unsigned char)( (temp/65536)%256 );
  332.       }
  333.       else if( FULL_CALIBRATION == command[3] )//系统满量程校准
  334.       {
  335.          
  336.           //计算ZS,一定要作强制类型转换,否则将出现错误
  337.           temp = (unsigned long int)(coefficient[0]) + (unsigned long int)(coefficient[1])*256
  338.                + (unsigned long int)(coefficient[2])*65536;         
  339.          
  340.           //将ZS写入到AD7705的OFFSET寄存器         
  341.           write_AD7705_byte( WR_OFFSET_REG );
  342.           write_AD7705_dword( temp );         
  343.          
  344.           //写设置寄存器,选择满量程校准
  345.           write_AD7705_byte( WR_SETUP_REG );
  346.           write_AD7705_byte( text_of_setup[cali_scale-1] | SYS_FULL_CALI );         
  347.          
  348.           //等待校准完成,系统校准延时时间
  349.           start_timer0();
  350.           while( time_count < time_sys_cali );
  351.           stop_timer0();
  352.          
  353.           while( AD_DRDY );//若将滤波器同步位FSYNC置为1,AD_DRDY信号将不会变低,这里将一直是死循环
  354.          
  355.           //读FULL寄存器
  356.           write_AD7705_byte( RD_FULL_REG );
  357.           temp = read_AD7705_dword();
  358.          
  359.           if( cali_scale == scale )
  360.           {
  361.               GS = temp; //如果是当前量程满量程校准还要更新GS
  362.               //如果是当前量程的校准,还要将NO_CALI_TYPE赋值为2表示已经经过零校准
  363.               //更新上电没有校准的情况,让readEEPROM()函数退出循环状态
  364.               NO_CALI_TYPE = ALREADY_CALIBRATION;            
  365.           }     
  366.          
  367.           coefficient[3] = (unsigned char)( temp%256 );
  368.           coefficient[4] = (unsigned char)( (temp/256)%256 );
  369.           coefficient[5] = (unsigned char)( (temp/65536)%256 );
  370.       }
  371.       else
  372.       {
  373.           AD_CS1;
  374.           _delay_us(5);
  375.           return;
  376.       }
  377.      
  378.       AD_CS1;
  379.       _delay_us(5);
  380.       
  381.       crcvalue = checksum(&coefficient[0],6); //将6个校准值调用CRC校验函数得到校验码     
  382.       coefficient[6] = (unsigned char)(crcvalue%256);//取校验值的高8位和低8位
  383.       coefficient[7] = (unsigned char)(crcvalue/256);
  384.      
  385.       //保存第一份校准系数
  386.       eeprom_busy_wait();
  387. ……………………

  388. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:
ad7705源程序.rar (3.5 KB, 下载次数: 27)


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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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