標(biāo)題: 51單片機(jī)+PCF8591仿真不正確,但是燒錄到單片機(jī)中是可以的,I2C有問題? [打印本頁]

作者: chuangyu1897    時(shí)間: 2022-5-22 01:09
標(biāo)題: 51單片機(jī)+PCF8591仿真不正確,但是燒錄到單片機(jī)中是可以的,I2C有問題?
如題,使用PCF8591的AD模式時(shí),仿真結(jié)果是正確的,能夠讀到正確的輸出,但是使用DA功能時(shí)就不對了,使用DA時(shí),設(shè)置輸出2.5V以上能夠正常輸出,設(shè)置輸出2.5V以下就不能正常輸出。
另外在使用AD模式時(shí),控制字不能設(shè)置為0x00,講道理我用AD的時(shí)候模擬輸出端口使能我應(yīng)該是可以置零的啊,但是置零就不對。。。
貼出原理圖和部分代碼,文件在附件中,求各位大佬解惑
PS.同樣的問題也出現(xiàn)在EEPROM ATC02中,也是仿真不正確燒錄到單片機(jī)中正確。。。
這兩個(gè)芯片都是I2C通信,難道是I2C的問題?

PCF8591.zip

84.92 KB, 下載次數(shù): 9

代碼及仿真文件


作者: chuangyu1897    時(shí)間: 2022-5-22 01:13
帖子長度有限制,現(xiàn)將部分單片機(jī)代碼貼上
  1. #include <REGX52.H>
  2. #include "delay.h"

  3. //這里的延時(shí)都可以去掉,是用來調(diào)試代碼的時(shí)候方便分析的

  4. /*********************
  5. /*@brief        該文件定義了I2C的6個(gè)時(shí)序,可以直接調(diào)用,
  6. /*@param        6個(gè)時(shí)序分別為起始、發(fā)送字節(jié)、接收應(yīng)答、接收字節(jié)、發(fā)送應(yīng)答、終止
  7. /*@retval
  8. *********************/

  9. //單片機(jī)實(shí)際連接引腳
  10. sbit scl = P3^4;
  11. sbit sda = P3^5;

  12. /*********************
  13. /*@brief        I2C起始條件,SCL高電平期間,SDA從高電平切換到低電平
  14. /*@param        無
  15. /*@retval        無
  16. *********************/
  17. void I2C_start() {
  18.     sda =1;
  19.     scl = 1;
  20.     delay10us();
  21.     sda =0;
  22.     delay10us();
  23.     scl =0;
  24. }

  25. /*********************
  26. /*@brief        I2C終止條件,SCL高電平期間,SDA從低電平切換到高電平
  27. /*@param        無
  28. /*@retval        無
  29. *********************/
  30. void I2C_stop() {
  31.     sda = 0;
  32.     scl = 1;
  33.     delay10us();
  34.     sda = 1;
  35. }

  36. /*********************
  37. /*@brief        I2C發(fā)送一個(gè)字節(jié),SCL低電平期間,主機(jī)將數(shù)據(jù)位依次放到SDA線上(高位在前),然后拉高SCL,
  38.                                         從機(jī)將在SCL高電平期間讀取數(shù)據(jù)位,SCL高電平期間SDA不允許數(shù)據(jù)變化,
  39. /*@param        要發(fā)送的字節(jié)
  40. /*@retval        無
  41. *********************/
  42. void I2C_sendByte(unsigned char Byte) {
  43.     int i;
  44.     for(i=0; i<8; i++) {
  45.         sda = Byte&(0x80>>i);
  46.         delay10us();
  47.         scl = 1;
  48.         delay10us();//這里可能比較重要?
  49.         scl = 0;
  50.     }
  51. }

  52. /*********************
  53. /*@brief        I2C接收一個(gè)字節(jié),主機(jī)在接收之前需要釋放SDA
  54.                                         SCL低電平期間,從機(jī)將數(shù)據(jù)位依次放到SDA線上(高位在前),然后拉高SCL,
  55.                                         主機(jī)將在SCL高電平期間讀取數(shù)據(jù)位,所以SCL高電平期間SDA不允許數(shù)據(jù)變化
  56. /*@param        無
  57. /*@retval        接收到的數(shù)據(jù)
  58. *********************/
  59. unsigned char I2C_receiveByte() {
  60.     unsigned char Byte=0x00;
  61.     int i;
  62.     sda =1;
  63.     for(i=0; i<8; i++) {
  64.         scl=1;
  65.         if(sda)(Byte|=(0x80>>i));
  66.         delay10us();
  67.         scl=0;
  68.         delay10us();
  69.     }
  70.     return Byte;
  71. }

  72. /*********************
  73. /*@brief        I2C發(fā)送應(yīng)答
  74. /*@param        數(shù)據(jù)0表示應(yīng)答,數(shù)據(jù)1表示非應(yīng)答
  75. /*@retval        無
  76. *********************/
  77. void I2C_sendACK(bit ack) {
  78.     sda = ack;
  79.     delay10us();
  80.     scl = 1;
  81.     scl =0;
  82. }

  83. /*********************
  84. /*@brief        I2C接收應(yīng)答,主機(jī)在接收之前,需要釋放SDA
  85. /*@param        數(shù)據(jù)0表示應(yīng)答,數(shù)據(jù)1表示非應(yīng)答
  86. /*@retval        接收到的應(yīng)答標(biāo)志
  87. *********************/
  88. bit I2C_receiveACK() {
  89.     bit ack;
  90.     sda =1;
  91.     delay10us();
  92.     scl =1;
  93.     ack = sda;
  94.     delay10us();
  95.     scl =0;
  96.     return ack;
  97. }
復(fù)制代碼





歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1