找回密码
 立即注册

QQ登录

只需一步,快速开始

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

分享一个用数码管显示PCF8591 AD值的小程序加Proteus仿真

[复制链接]
跳转到指定楼层
楼主
程序中,也有一些我自己写的注解,仿真图的画法不是那么标准,但是不影响仿真效果,亲测。
本人水平有限,如有错误的地方,希望大家指出,谢谢了

有图有真相:程序 图


仿真 图


单片机源程序如下:
  1. #include<reg51.h>
  2. #include<intrins.h>

  3. typedef unsigned char uint8;
  4. typedef unsigned int uint16;
  5. typedef unsigned long uint32;

  6. sbit ADDRC = P1^0;
  7. sbit ADDRB = P1^1;
  8. sbit ADDRA = P1^2;//控制138译码器的三个引脚

  9. sbit I2C_SDA = P2^1;
  10. sbit I2C_SCL = P2^0;//I2C通信的两个引脚

  11. bit flag_300ms = 0;//300ms时间标志位
  12. uint8 T0RH = 0;//定时器高8位初值
  13. uint8 T0RL = 0;//定时器低8位初值
  14. uint8 AD_value = 0;//AD值
  15. uint8 code led_char[] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
  16.                                                  0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//LED段码

  17. void config_timer0(uint8 ms);
  18. uint8 get_ADC_vaule(uint8 chn);

  19. void main()
  20. {
  21.         config_timer0(1);//初始化通用定时器0,定时1ms
  22.         while(1)
  23.         {
  24.                 if(flag_300ms == 1)        //判断300ms时间是否到了
  25.                 {
  26.                         flag_300ms = 0;//清除标志位
  27.                         AD_value = get_ADC_vaule(0);//读取通道0的AD值
  28.                 }                        
  29.         }
  30. }

  31. void I2C_delay()//I2C延时函数
  32. {
  33.         _nop_();
  34.         _nop_();
  35.         _nop_();
  36.         _nop_();
  37. }

  38. void I2C_start()//I2C起始信号
  39. {
  40.         I2C_SDA = 1;
  41.         I2C_SCL = 1;
  42.         I2C_delay();
  43.         I2C_SDA = 0;
  44.         I2C_delay();
  45.         I2C_SCL = 0;
  46.         I2C_delay();
  47. }

  48. void I2C_stop()//I2C停止信号
  49. {
  50.         I2C_SDA = 0;
  51.         I2C_SCL = 0;
  52.         I2C_delay();
  53.         I2C_SCL = 1;
  54.         I2C_delay();
  55.         I2C_SDA = 1;
  56.         I2C_delay();
  57. }

  58. bit I2C_write(uint8 dat)//I2C写一个字节
  59. {
  60.         bit ack = 0;
  61.         uint8 mask = 0;
  62.         for(mask=0x80;mask!=0;mask>>=1)
  63.         {
  64.                 if((mask&dat) == 0)
  65.                         I2C_SDA = 0;
  66.                 else
  67.                         I2C_SDA = 1;
  68.                 I2C_delay();
  69.                 I2C_SCL = 1;
  70.                 I2C_delay();
  71.                 I2C_SCL = 0;
  72.                 I2C_delay();
  73.         }
  74.         I2C_SDA = 1;
  75.         I2C_delay();
  76.         I2C_SCL = 1;
  77.         I2C_delay();
  78.         ack = I2C_SDA;
  79.         I2C_delay();
  80.         I2C_SCL = 0;
  81.         I2C_delay();
  82.         
  83.         return (~ack);        
  84. }

  85. uint8 I2C_read_ACK()//I2C读一个字节,并发送应答位
  86. {
  87.         uint8 dat = 0;
  88.         uint8 mask = 0;
  89.         I2C_SDA = 1;
  90.         for(mask=0x80;mask!=0;mask>>=1)
  91.         {
  92.                 if(I2C_SDA == 0)
  93.                         dat = dat & (~mask);
  94.                 else
  95.                         dat = dat | mask;
  96.                 I2C_delay();
  97.                 I2C_SCL = 1;
  98.                 I2C_delay();
  99.                 I2C_SCL = 0;
  100.                 I2C_delay();
  101.         }
  102.         I2C_SDA = 0;
  103.         I2C_delay();
  104.         I2C_SCL = 1;
  105.         I2C_delay();
  106.         I2C_SCL = 0;
  107.         I2C_delay();

  108.         return dat;
  109. }

  110. uint8 I2C_read_NACK()//I2C读一个字节,并发送非应答位
  111. {
  112.         uint8 dat = 0;
  113.         uint8 mask = 0;
  114.         I2C_SDA = 1;
  115.         for(mask=0x80;mask!=0;mask>>=1)
  116.         {
  117.                 if(I2C_SDA == 0)
  118.                         dat = dat & (~mask);
  119.                 else
  120.                         dat = dat | mask;
  121.                 I2C_delay();
  122.                 I2C_SCL = 1;
  123.                 I2C_delay();
  124.                 I2C_SCL = 0;
  125.                 I2C_delay();
  126.         }
  127.         I2C_SDA = 1;
  128.         I2C_delay();
  129.         I2C_SCL = 1;
  130.         I2C_delay();
  131.         I2C_SCL = 0;
  132.         I2C_delay();

  133.         return dat;
  134. }

  135. uint8 get_ADC_vaule(uint8 chn)//获取AD值
  136. {
  137.         uint8 value = 0;
  138.         I2C_start();//I2C起始信号
  139.         if(!I2C_write(0X90))//写入PCF8591地址及读写选择位为写
  140.         {
  141.                 I2C_stop();
  142.                 return 0;
  143.         }
  144. //        I2C_write(0X40 | chn);//写入PCF8591通道0
  145.         I2C_write(0x00 | chn);//写入PCF8591通道0
  146.         I2C_start();//I2C起始信号
  147.         I2C_write(0x48<<1 | 0x01);
  148.         I2C_read_ACK();//提供转换所需的时钟信号
  149.         value = I2C_read_NACK();//读取上一次转换的结果
  150.         I2C_stop();//I2C结束信号

  151.         return value;
  152. }

  153. void config_timer0(uint8 ms)//配置定时器0的通用函数
  154. {
  155.         uint32 temp =0 ;
  156.         temp = 11059200/12;
  157.         temp = (temp * ms)/1000;
  158.         temp = 65536 - (temp + 12);
  159.         T0RH = (uint8)(temp >> 8);
  160.         T0RL = (uint8)temp;

  161.         TMOD = TMOD & 0XF0;
  162.         TMOD = TMOD | 0X01;
  163.         TH0 = T0RH;
  164.         TL0 = T0RL;
  165.         EA = 1;
  166.         ET0 = 1;
  167.         TR0 = 1;
  168. }

  169. void led_scan()//LED扫描函数,每个1ms扫描一次
  170. {
  171.         static uint8 index = 0;

  172.         P0 = 0X00;
  173.         switch(index)
  174.         {
  175.                 case 0:ADDRC=1;ADDRB=1;ADDRA=1;P0=led_char[AD_value%10];break;
  176.                 case 1:ADDRC=1;ADDRB=1;ADDRA=0;P0=led_char[AD_value/10%10];break;
  177.                 case 2:ADDRC=1;ADDRB=0;ADDRA=1;P0=led_char[AD_value/100];break;
  178.                 default:break;
  179.         }
  180.         index++;
  181.         if(index >= 3)
  182.         {
  183.                 index = 0;
  184.         }
  185. }

  186. void interrupt_timer0() interrupt 1//定时器0的中断函数
  187. {
  188.         static uint16 cnt = 0;

  189.         TH0 = T0RH;
  190.         TL0 = T0RL;

  191.         led_scan();
  192.         cnt++;
  193.         if(cnt >= 300)
  194.         {
  195.                 cnt = 0;
  196.                 flag_300ms = 1;
  197.         }
  198. }
复制代码

所有资料51hei提供下载:
数码管显示AD值.rar (91.21 KB, 下载次数: 58)

评分

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

查看全部评分

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

使用道具 举报

无效楼层,该帖已经被删除
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

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

Powered by 单片机教程网

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