久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 1715|回復: 4
收起左側

I2C 為啥有的程序沒被編譯

[復制鏈接]
ID:611626 發表于 2023-10-8 16:56 | 顯示全部樓層 |閱讀模式

直接跳過,好像沒被編譯

直接跳過,好像沒被編譯
  1. #include <reg51.h>
  2. #include <intrins.h>
  3. #define uchar   unsigned char  
  4. #define uint    unsigned int
  5. sbit SDA=P3^6;     //模擬I2C數據傳送位SDA    SDA=P3^6;     
  6. sbit SCL=P3^7;     //模擬I2C時鐘控制位SCL  SCL=P3^7;
  7. sbit INT=P3^2;      
  8. sbit RESET=P3^3;
  9. sbit led0=P1^0;
  10. sbit led1=P1^1;
  11. sbit led2=P1^2;
  12. sbit led3=P1^3;
  13. sbit led4=P1^4;
  14. sbit led5=P1^5;
  15. sbit led6=P1^6;
  16. sbit led7=P1^7;
  17. bit  ack;          //應答標志位

  18. #define DS3231_WriteAddress 0xD0    //器件寫地址
  19. #define DS3231_ReadAddress  0xD1    //器件讀地址
  20. #define DS3231_SECOND       0x00    //秒
  21. #define DS3231_MINUTE       0x01    //分
  22. #define DS3231_HOUR         0x02    //時
  23. #define DS3231_WEEK         0x03    //星期
  24. #define DS3231_DAY          0x04    //日
  25. #define DS3231_MONTH        0x05    //月
  26. #define DS3231_YEAR         0x06    //年
  27. //鬧鈴1            
  28. #define DS3231_SALARM1ECOND 0x07    //秒
  29. #define DS3231_ALARM1MINUTE 0x08    //分
  30. #define DS3231_ALARM1HOUR   0x09    //時
  31. #define DS3231_ALARM1WEEK   0x0A    //星期/日
  32. //鬧鈴2
  33. #define DS3231_ALARM2MINUTE 0x0b    //分
  34. #define DS3231_ALARM2HOUR   0x0c    //時
  35. #define DS3231_ALARM2WEEK   0x0d    //星期/日
  36. #define DS3231_CONTROL      0x0e    //控制寄存器
  37. #define DS3231_STATUS       0x0f    //狀態寄存器
  38. #define BSY                 2       //忙
  39. #define OSF                 7       //振蕩器停止標志
  40. #define DS3231_XTAL         0x10    //晶體老化寄存器
  41. #define DS3231_TEMPERATUREH 0x11    //溫度寄存器高字節(8位)
  42. #define DS3231_TEMPERATUREL 0x12    //溫度寄存器低字節(高2位)


  43. uchar code dis_code[11]={0xc0,0xf9,0xa4,0xb0,   // 0,1,2,3共陽字碼表
  44.              0x99,0x92,0x82,0xf8,0x80,0x90, 0xff};       // 4,5,6,7,8,9,off
  45. //uchar code dis_code[11]={0x3f,0x06,0x5b,0x4f,   // 0,1,2,3
  46. //            0x66,0x6d,0x7d,0x07,0x7f,0x6f, 0x00};       // 4,5,6,7,8,9,off共陰
  47. uchar data dis_buf[8]; //8位數碼管
  48. uchar data dis_index;
  49. uchar data dis_digit;

  50. uchar BCD2HEX(uchar val)    //BCD轉換為Byte
  51. {
  52.      uchar temp;
  53.      temp=val&0x0f;
  54.      val>>=4;
  55.      val&=0x0f;
  56.      val*=10;
  57.      temp+=val;
  58.    
  59.      return temp;
  60. }
  61. uchar HEX2BCD(uchar val)    //B碼轉換為BCD碼
  62. {
  63.      uchar i,j,k;
  64.      i=val/10;
  65.     // j=val%10;   //上面代碼這個地方漏了取余,其它的代碼模塊都OK
  66.      k=j+(i<<4);
  67.      return k;
  68. }

  69. //延時
  70. void delayus(uint us)
  71. {
  72.      while (us--);
  73. }
  74. //啟動
  75. void Start_I2C()
  76. {
  77.      SDA=1;                  //發送起始條件的數據信號
  78.     delayus(1);
  79.      SCL=1;
  80.      delayus(5);             //起始條件建立時間大于4.7us,延時
  81.    
  82.      SDA=0;                  //發送起始信號
  83.     delayus(5);             // 起始條件鎖定時間大于4μs
  84.         
  85.      SCL=0;                  //鉗住I2C總線,準備發送或接收數據
  86.     delayus(2);
  87. }
  88. //停止
  89. void Stop_I2C()
  90. {
  91.      SDA=0;                  //發送結束條件的數據信號
  92.     delayus(1);             //發送結束條件的時鐘信號
  93.     SCL=1;                  //結束條件建立時間大于4us
  94.      delayus(5);
  95.    
  96.      SDA=1;                  //發送I2C總線結束信號
  97.     delayus(4);
  98. }
  99. //發送一個字節
  100. void SendByte(uchar c)
  101. {
  102.      uchar BitCnt;
  103.    
  104.      for(BitCnt=0;BitCnt<8;BitCnt++)         //要傳送的數據長度為8位
  105.     {
  106.          if((c<<BitCnt)&0x80)
  107.              SDA=1;                          //判斷發送位
  108.         else
  109.              SDA=0;               
  110.            delayus(1);
  111.            SCL=1;                            //置時鐘線為高,通知被控器開始接收數據位
  112.           delayus(5);                       //保證時鐘高電平周期大于4μs   
  113.            SCL=0;
  114.      }
  115.    
  116.      delayus(2);
  117.      SDA=1;                                  //8位發送完后釋放數據線,準備接收應答位
  118.     delayus(2);  
  119.      SCL=1;
  120.      delayus(3);
  121.      if(SDA==1)
  122.          ack=0;   
  123.      else
  124.          ack=1;                              //判斷是否接收到應答信號
  125.     SCL=0;
  126.      delayus(2);
  127. }
  128. //讀一個字節
  129. uchar RcvByte()
  130. {
  131.     uchar retc;
  132.     uchar BitCnt;
  133.   
  134.     retc=0;
  135.     SDA=1;                           //置數據線為輸入方式
  136.    for(BitCnt=0;BitCnt<8;BitCnt++)
  137.     {
  138.          delayus(1);  
  139.          SCL=0;                      //置時鐘線為低,準備接收數據位
  140.       
  141.          delayus(5);                 //時鐘低電平周期大于4.7μs
  142.       
  143.          SCL=1;                      //置時鐘線為高使數據線上數據有效
  144.         delayus(3);
  145.          retc=retc<<1;
  146.          if(SDA==1)
  147.              retc=retc+1;            //讀數據位,接收的數據位放入retc中
  148.         delayus(2);
  149.     }
  150.     SCL=0;
  151.     delayus(2);
  152.     return(retc);
  153. }

  154. //應答
  155. void Ack_I2C(bit a)
  156. {
  157.   
  158.      if(a==0)
  159.          SDA=0;              //在此發出應答或非應答信號
  160.     else
  161.          SDA=1;
  162.      delayus(3);     
  163.      SCL=1;
  164.    
  165.      delayus(5);             //時鐘低電平周期大于4μs
  166.    
  167.      SCL=0;                  //清時鐘線,鉗住I2C總線以便繼續接收
  168.     delayus(2);   
  169. }

  170. //寫入一個字節
  171. uchar write_byte(uchar addr, uchar write_data)
  172. {
  173.      Start_I2C();
  174.      SendByte(DS3231_WriteAddress);
  175.      if (ack == 0)
  176.          return 0;
  177.    
  178.      SendByte(addr);   
  179.      if (ack == 0)
  180.          return 0;
  181.    
  182.      SendByte(write_data);
  183.      if (ack == 0)
  184.          return 0;
  185.    
  186.      Stop_I2C();
  187.      delayus(10);      
  188.      return 1;
  189. }


  190. uchar read_current()
  191. {
  192.      uchar read_data;
  193.      Start_I2C();
  194.      SendByte(DS3231_ReadAddress);
  195.      if(ack==0)
  196.          return(0);
  197.    
  198.      read_data = RcvByte();
  199.      Ack_I2C(1);
  200.      Stop_I2C();
  201.      return read_data;
  202. }

  203. //讀取時間
  204. uchar read_random(uchar random_addr)
  205. {
  206.      Start_I2C();
  207.      SendByte(DS3231_WriteAddress);
  208.      if(ack==0)
  209.          return(0);
  210.    
  211.      SendByte(random_addr);
  212.      if(ack==0)
  213.          return(0);
  214.    
  215.      return(read_current());
  216. }

  217. //修改
  218. void ModifyTime(uchar yea,uchar mon,uchar da,uchar hou,uchar min,uchar sec)
  219. {
  220.      uchar temp=0;
  221.    
  222.      temp=HEX2BCD(yea);
  223.      write_byte(DS3231_YEAR,temp);   //修改年
  224.    
  225.      temp=HEX2BCD(mon);
  226.      write_byte(DS3231_MONTH,temp);  //修改月
  227.    
  228.      temp=HEX2BCD(da);
  229.      write_byte(DS3231_DAY,temp);    //修改日
  230.    
  231.      temp=HEX2BCD(hou);
  232.      write_byte(DS3231_HOUR,temp);   //修改時
  233.    
  234.      temp=HEX2BCD(min);
  235.      write_byte(DS3231_MINUTE,temp); //修改分
  236.    
  237.      temp=HEX2BCD(sec);
  238.      write_byte(DS3231_SECOND,temp); //修改秒
  239. }
  240. //提取時間段碼送顯示緩存

  241. void TimeDisplay(uchar Dhour,uchar Dmin,uchar Dsec)
  242. {
  243.     dis_buf[7]=dis_code[Dhour / 10];        //第八位顯示 時十位
  244.     dis_buf[6]=dis_code[Dhour % 10];        //第七位數碼顯示: 時個位
  245.     dis_buf[4]=dis_code[Dmin / 10];         // 分十位
  246.     dis_buf[3]=dis_code[Dmin % 10];         // 分個位
  247.     dis_buf[1]=dis_code[Dsec / 10];         // 秒十位
  248.     dis_buf[0]=dis_code[Dsec % 10];         // 秒個位
  249.     dis_buf[2]=0xbf;                        // 顯示"-"
  250.      dis_buf[5]=0xbf;
  251. }

  252. //取日期
  253. void DateDisplay(uchar Dyear,uchar Dmonth,uchar Dday)
  254. {
  255.      dis_buf[7]=dis_code[Dyear / 10];        // 年十位
  256.     dis_buf[6]=dis_code[Dyear % 10];        // 年個位
  257.     dis_buf[4]=dis_code[Dmonth / 10];       // 月十位
  258.     dis_buf[3]=dis_code[Dmonth % 10];       // 月個位
  259.     dis_buf[1]=dis_code[Dday / 10];         // 天十位
  260.     dis_buf[0]=dis_code[Dday % 10];         // 天個位
  261.     dis_buf[2]=0xbf;                        // 顯示"-"
  262.      dis_buf[5]=0xbf;
  263. }

  264. //讀取并顯示時間
  265. void get_show_time(void)
  266. {
  267.        
  268.      uchar Htemp1,Htemp2,Mtemp1,Mtemp2,Stemp1,Stemp2;
  269.   
  270.      Htemp1=read_random(DS3231_HOUR);    //時 24小時制
  271.      Htemp1&=0x3f;                  
  272.      Htemp2=BCD2HEX(Htemp1);
  273.        
  274.      Mtemp1=read_random(DS3231_MINUTE);  //分
  275.      Mtemp2=BCD2HEX(Mtemp1); //轉十進制顯示
  276.    
  277.      Stemp1=read_random(DS3231_SECOND);  //秒
  278.     Stemp2=BCD2HEX(Stemp1);

  279.    TimeDisplay(Htemp2,Mtemp2,Stemp2);  //調用 時 分 秒 提取個位 十位到顯示緩存
  280.        
  281. }

  282. //顯示日期
  283. void get_show_date(void)
  284. {
  285.      uchar Ytemp1,Ytemp2,Mtemp1,Mtemp2,Dtemp1,Dtemp2;
  286.    
  287.      Ytemp1=read_random(DS3231_YEAR);        //年
  288.     Ytemp2=BCD2HEX(Ytemp1);
  289.    
  290.      Mtemp1=read_random(DS3231_MONTH);       //月
  291.     Mtemp2=BCD2HEX(Mtemp1);
  292.    
  293.      Dtemp1=read_random(DS3231_DAY);         //日
  294.     Dtemp2=BCD2HEX(Dtemp1);
  295.    
  296.         DateDisplay(Ytemp2,Mtemp2,Dtemp2); //調用提取年月日個十位送顯示緩存
  297. }

  298. //顯示溫度
  299. void get_show_Temperature(void)
  300. {
  301.      uchar Ttemp1,Ttemp2,Ttemp3,Ttemp4;
  302.    
  303.      Ttemp1=read_random(DS3231_TEMPERATUREH);    //溫度 高字節
  304.     Ttemp2=BCD2HEX(Ttemp1);
  305.    
  306.      Ttemp3=read_random(DS3231_TEMPERATUREL);    //溫度低字節
  307.     Ttemp4=BCD2HEX(Ttemp3);
  308.      
  309.      DateDisplay(0,Ttemp2,Ttemp4); //
  310. }

  311. //T0中斷循環顯示數碼管
  312. void timer0() interrupt 1
  313. {
  314.      TH0=0xFC; //重新賦初值高位
  315.      TL0=0x17;
  316.    
  317.     P2=~0xff;   // 先關閉所有數碼管 P2=0xff;
  318.     P0=dis_buf[dis_index];  // 顯示代碼傳送到P0口段碼TimeDisplay
  319.     P2=dis_digit;  //位選fe=1111 1110
  320.    //如果最高位為1
  321.   
  322.      if (dis_digit & 0x80)  //0x80=1000 0000&1111 1110結果1000 0000 &兩位同時為“1”,結果才為“1”,否則為0
  323.          dis_digit=(dis_digit << 1) | 0x1; //結果0000 00001從最低位開始 |運算的兩個對象只要有一個為1,其值為1
  324.      else
  325.          dis_digit=(dis_digit << 1);  //否則左移移位
  326.    
  327.      dis_index++; //字碼緩存dis_buf[0]-dis_buf[7]+1
  328.      dis_index&=0x07; // 8個數碼管全部掃描完一遍之后,再回到第一個開始下一次掃描7=0000 0111

  329. }

  330. //主函數
  331. void main()
  332. {
  333.    //  uint ii = 0;
  334.    
  335.    
  336.   //   RESET=0x1;          //DS3231復位操作,正常操作下不需要每次都復位
  337.     delayus(5000);
  338.    
  339.      led0=0;
  340.      led1=0;
  341.      led2=0;
  342.      led3=0;
  343.      led4=0;
  344.    
  345.      P0=0xff;  // 先關閉所有數碼管段選
  346.      P2=0x00;  // 先關閉所有數碼管位選P2=0xff;
  347.    
  348.      dis_digit=0x01; //P2位選1111 1110  0xfe; 0x80; 共陰0xfe 共陽0x01個位第一位即P2.0
  349.      dis_index=0; //初始賦值 第 dis_buf[0]個緩存,秒,或者天個位   段碼0-7
  350.      TimeDisplay(12, 5, 18); //提取時間段碼送顯示緩存 時分秒

  351.    
  352.      TMOD=0x11;          // 定時器0, 1工作模式1, 16位定時方式
  353.     TH0=0xFC;
  354.      TL0=0x17;
  355.    
  356.      TCON=0x01;
  357.      IE=0x82;            // 使能timer0,1 中斷
  358.    
  359.      TR0=1;
  360.    

  361.      if (write_byte(DS3231_CONTROL, 0x1C) == 0)    //控制寄存器0x0e  
  362.          led0=1;
  363.    
  364.      if (write_byte(DS3231_STATUS, 0x00) == 0)  //狀態寄存器0x0F  
  365.          led1=1;
  366.    
  367.      ModifyTime(10,6,13,15,30,00);       //初始化時鐘,2010/6/13,15/30/00
  368.                                          //小時采用24小時制
  369.     while(1)
  370.      {
  371.        //  get_show_date();              //顯示日期
  372.      //    get_show_Temperature();       //顯示溫度
  373.        get_show_time();                //顯示時間
  374.                          
  375.         delayus(50000);
  376.      }
  377.   /*  */
  378. }
復制代碼


DS3231共陽.rar

32.96 KB, 下載次數: 2

回復

使用道具 舉報

ID:161164 發表于 2023-10-8 18:19 | 顯示全部樓層
被優化掉,加試試volatile
  1. //延時
  2. void delayus(volatile uint us)
  3. {
  4.     while (us--);
復制代碼
回復

使用道具 舉報

ID:611626 發表于 2023-10-8 18:40 | 顯示全部樓層
lkc8210 發表于 2023-10-8 18:19
被優化掉,加試試volatile

還是讀取不到數據
回復

使用道具 舉報

ID:883242 發表于 2023-10-8 19:21 | 顯示全部樓層
lkc8210 發表于 2023-10-8 18:19
被優化掉,加試試volatile

明顯是Keil C51,這個編譯器假設用戶都不懂volatile這個關鍵字,寫不寫都一樣。
回復

使用道具 舉報

ID:611626 發表于 2023-10-9 08:43 | 顯示全部樓層
  1.   delayus(2);
  2. //請教122行,這里SDA賦值1
  3.      SDA=1;                                  //8位發送完后釋放數據線,準備接收應答位
  4.     delayus(2);  
  5.      SCL=1;
  6.      delayus(3);
  7. //這里豈不是每次都讀取失敗,ACK返回0
  8.      if(SDA==1)
  9.          ack=0;   
  10. //這個函數豈不是執行不到底
  11. uchar read_current()
  12. {
  13.      uchar read_data;
  14.      Start_I2C();
  15.      SendByte(DS3231_ReadAddress);
  16.      if(ack==0)
  17.          return(0);
復制代碼
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国产乱一区二区三区视频 | 羞羞视频在线观看 | 成人免费观看视频 | 日韩一区二区在线视频 | 男女精品网站 | 亚洲在线看| 色爱综合网 | 国产一区二区三区免费观看在线 | 国产精品一区在线观看 | av在线黄 | 亚洲图片视频一区 | 色又黄又爽网站www久久 | 黄色精品视频网站 | 91色站| 国产在线观看不卡一区二区三区 | 国产精品视频在线观看 | av在线免费观看网址 | 人人做人人澡人人爽欧美 | 精品国产欧美一区二区三区不卡 | 午夜视频一区 | 91视频一区二区 | 视频精品一区二区三区 | 欧美日韩美女 | 亚洲精品一区中文字幕乱码 | 人人做人人澡人人爽欧美 | 日韩精品一区二区三区中文字幕 | 成人综合视频在线 | 亚洲美女av网站 | 99久久精品免费看国产四区 | 成人日b视频| 国产精品日日做人人爱 | 欧美精品a∨在线观看不卡 欧美日韩中文字幕在线播放 | 亚洲欧洲在线观看视频 | 成人精品国产 | 日本中文在线 | 国产成人在线视频免费观看 | 亚洲一区二区在线播放 | 天天干天天操 | 一区二区成人在线 | 亚洲成网 | 永久网站|