找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机读卡系统仿真程序 IIC通信

[复制链接]
跳转到指定楼层
楼主
用STM32或者51单片机设计一个读卡系统,对外接的I2C接口电路或者SPI接口电路进行读写操作,读写内容可以通过数码管或者其他监视器件查看,并用示波器观看通信线路波形。完成主机跟两个从机之间的I2C通信,可以用数码管等显示通信的内容,也可以用示波器监视通信线路的数据信息。


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


  3. typedef unsigned int u16;
  4. typedef unsigned char u8;
  5. sbit SDA=P1^1;
  6. sbit SCL=P1^0;
  7. void At24CO2Write(u8 add,u8 dat,u8 addr);
  8. u8 At24CO2Read(u8 add,u8 addr);

  9. u8 numer[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
  10. u8 display[4];
  11. sbit A1=P2^2;
  12. sbit B1=P2^3;
  13. sbit C1=P2^4;
  14. #define SHUMA P0
  15. sbit K1=P3^0;//保存
  16. sbit K2=P3^1;//读取
  17. sbit K3=P3^2;
  18. sbit K4=P3^3;
  19. sbit K5=P3^4;//num++
  20. sbit K6=P3^5;//清零

  21. sbit led=P2^0;
  22. u8 num;
  23. void delay(u16 i)
  24. {
  25. while(i--);
  26. }

  27. void delay10us(void)   
  28. {
  29.     unsigned char a,b;
  30.     for(b=1;b>0;b--)
  31.         for(a=2;a>0;a--);
  32. }

  33. void shumaguan(u8 word[])
  34. {         u8 i;
  35.   for(i=0;i<4;i++)
  36. {
  37.   switch(i)
  38.   {
  39.   case 0:C1=0;B1=0;A1=0;break;
  40.   case 1:C1=0;B1=0;A1=1;break;
  41.   case 2:C1=0;B1=1;A1=0;break;
  42.   case 3:C1=0;B1=1;A1=1;break;
  43.   }
  44.   SHUMA=word[i];
  45.   delay(100);
  46.   SHUMA=0x00;
  47. }
  48. }
  49. void Key(u8 dat)
  50. {
  51.         
  52. if(K1==0)
  53. {
  54.   delay(1000);
  55.   if(K1==0)
  56.         {        
  57.                 At24CO2Write(0xa0,dat,1);
  58.         }
  59.   while(!K1);
  60. }
  61. if(K2==0)
  62. {
  63.   delay(1000);
  64.   if(K2==0)
  65.         {
  66.                 num=At24CO2Read(0xa0,1);
  67.         }
  68.   while(!K2);
  69. }
  70.   if(K3==0)
  71. {
  72.   delay(1000);
  73.   if(K3==0)
  74.         {        
  75.                 At24CO2Write(0xa2,dat,1);
  76.         }
  77.   while(!K3);
  78. }
  79. if(K4==0)
  80. {
  81.   delay(1000);
  82.   if(K4==0)
  83.         {
  84.                 num=At24CO2Read(0xa2,1);
  85.         }
  86.   while(!K4);
  87. }
  88. if(K5==0)
  89. {
  90.   delay(1000);
  91.   if(K5==0)
  92.   num++;
  93.   while(!K5);
  94. }
  95. if(K6==0)
  96. {
  97.   delay(1000);
  98.   if(K6==0)
  99.   num=0;
  100.   while(!K6);
  101. }
  102. }
  103. void change()
  104. {
  105.    display[0]=numer[num/1000];
  106.          display[1]=numer[(num%1000)/100];
  107.          display[2]=numer[(num%100)/10];
  108.          display[3]=numer[num%10];
  109. }
  110. void main()
  111. {
  112. //        SDA=1;
  113. //        delay10us();
  114. //        SCL=0;
  115. //        delay10us();
  116. //// u8        addr = 0x66;
  117. //                uchar addr1=0x66, addr2=0x88, dat1, dat2;
  118.                
  119. //                write(0xa2, addr2, 0);
  120.          while(1)
  121.          {
  122.                  Key(num);
  123.                  change();
  124.                  shumaguan(display);
  125.          }
  126. }


  127. void I2cStart() //开始函数,时钟信号始终为高,当数据线由高变低时开始
  128. {
  129.   SDA=1;
  130.   delay10us();
  131.   SCL=1;
  132.   delay10us();
  133.   SDA=0;
  134.   delay10us();
  135.   SCL=0;
  136.   delay10us();
  137. }
  138. void I2cStop()        //停止函数,时钟信号始终为高,当数据线由低变高时开结束
  139. {
  140.   SDA=0;
  141.   delay10us();
  142.   SCL=1;
  143.   delay10us();
  144.   SDA=1;
  145.   delay10us();
  146. }
  147. void I2cAck()
  148. {

  149.          SDA=0;
  150.          delay10us();
  151.          SCL=1;
  152.          delay10us();

  153.          SCL=0;
  154.          delay10us();
  155. }

  156. void nack()
  157. {
  158.         SDA = 1;        //拉高SDA,发送非应答信号
  159.         Delay();        
  160.         SCL = 1;        //拉高SCL
  161.         Delay();        
  162.         SCL = 0;        //再拉低完成非应答位,并保持住总线
  163. }

  164. u8 I2cSend(u8 dat)
  165. {
  166.         u8 i;
  167.         for(i=0x80; i!=0; i>>=1)        //从高位到低位依次进行
  168.         {
  169.                 if((dat&i) == 0)        //该位的值依次输出到SDA上
  170.                         SDA = 0;
  171.                 else
  172.                         SDA = 1;
  173.                 delay10us();
  174.                 SCL = 1;        //拉高SCL
  175.                 delay10us();
  176.                 SCL = 0;        //再拉低SCL完成一个位周期
  177.         }

  178. }
  179. u8 I2cRead()
  180. {
  181.          u8 i, dat;
  182.         
  183.         SDA = 1;        //确保主机释放SDA
  184.         for(i=0x80; i!=0; i>>=1)        //从高位到低位依次进行
  185.         {
  186.                 delay10us();
  187.                 SCL = 1;        //拉高SCL
  188.                 if(SDA == 0)        //读取SDA的值
  189.                         dat &= ~i;        //为0时,dat中对应位清零
  190.                 else
  191.                         dat |= i;        //为1时,dat中对应位置1
  192.                 delay10us();
  193.                 SCL = 0;        //再拉低SCL使从机发送出下一位
  194.         }
  195.         return dat;

  196. }
  197. void At24CO2Write(u8 add,u8 dat,u8 addr)
  198. {
  199.    I2cStart();
  200.    I2cSend(add);
  201.          I2cAck();
  202.    I2cSend(addr);
  203.          I2cAck();
  204.    I2cSend(dat);
  205.          I2cAck();
  206.    I2cStop();
  207. }
  208. u8 At24CO2Read(u8 add,u8 addr)
  209. {
  210.    u8 num;
  211.    I2cStart();
  212.    I2cSend(add);
  213.          I2cAck();
  214.    I2cSend(addr);
  215.          I2cAck();
  216.    I2cStart();
  217.    I2cSend(add+1);
  218.          I2cAck();
  219.    num=I2cRead();
  220. //         nack();
  221.    I2cStop();
  222.    return num;
  223. }



复制代码
Keil代码与Proteus仿真下载:
实验40.rar (98.02 KB, 下载次数: 12)



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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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