18B20的时序是对的,为何读不出数据呢?

2020-02-06 10:26发布

  1. uchar read_bit(void)  //读一位(bit)
  2. {
  3.         unsigned char i,j;
  4.         TRISC&=0xfd;//配置RC1为输出
  5.         RC1=0;       //将RC1拉低开始读时间隙
  6.         //RC1=1;  //then return high
  7.         for(i=0;i<1;i++);  // 延时
  8.         TRISC|=0x02;//配置RC1为输入
  9.         for(i=0;i<1;i++);
  10.         return(RC1);  // 返回RC1线上的电平值
  11. }

  12. uchar ReadOneChar(void) //读一个字节
  13. {
  14.         unsigned char i = 0;
  15.         unsigned char dat = 0;
  16.         for (i=0;i<8;i++)  
  17.         {   
  18.                 if(read_bit())
  19.                 dat|=(0x01<<i);    // 然后将其左移
  20.             Delay(4);
  21.         }
  22.     return (dat);
  23. }

  24. void write_bit(char bitval)        //写一位
  25. {
  26.         RC1=0;         // 将RC1拉低开始写时间隙
  27.         if(bitval==1)
  28.         RC1=1;   // 如果写1,RC1返回高电平
  29.         Delay(5);        // 在时间隙内保持电平值,
  30.         RC1=1;           // Delay函数每次循环延时16μs,因此delay(5) = 104μs
  31. }

  32. void WriteOneChar(unsigned char dat) //写一个字节
  33. {
  34.     unsigned char i = 0;
  35.     unsigned char temp;
  36.     for (i=0; i<8; i++)  // 写入字节, 每次写入一位  
  37.     {
  38.             temp=(dat>>i);  
  39.                    temp&=0x01;  
  40.                    write_bit(temp);
  41.     }
  42.     Delay(5);
  43. }
复制代码

在网上的找的51程序,移植到PIC的单片机上,主程序就这2句
WriteOneChar(0x33);       
a=ReadOneChar();
TXREG=a;
把18B20的序列号的第一位28h发生到串口;
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
28条回答
R88
2020-02-09 03:44
codeman 发表于 2014-10-24 10:15
曾做过 10来米线上IO口取电并30来个,通讯稳定可靠。
数据手册上写的很详细,记得大约步骤如下
先复位总线 ...
  1. uchar RomCode[8] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00};

  2. void Delay(uint num)
  3. {
  4.   while( --num );
  5. }

  6. uchar Init_DS18B20(void)  //初始化ds1820,RC1作为单总线通信口
  7. {   
  8.            uchar presence=1;
  9.            TRISC|=0x02;//配置RC1为输入
  10.     Delay(8);    //稍做延时
  11.     TRISC&=0xfd;//配置RC1为输出
  12.     RC1=0;      //将RC1拉低
  13.     Delay(150);   //精确延时 大于 480us
  14.     TRISC|=0x02;//配置RC1为输入
  15.     Delay(20);
  16.     presence=RC1;    //读取存在信号
  17.     Delay(8);  
  18.     return(presence); //返回信号,0=presence,1= no presence
  19. }

  20. bit read_bit(void)  //读一位(bit)
  21. {
  22.         unsigned char i;
  23.         TRISC&=0xfd;//配置RC1为输出
  24.         RC1=0;       //将RC1拉低开始读时间隙
  25.         TRISC|=0x02;//配置RC1为输入
  26.         return(RC1);  // 返回RC1线上的电平值
  27. }

  28. uchar ReadOneChar(void) //读一个字节
  29. {
  30.         unsigned char i = 0;
  31.         unsigned char dat = 0;
  32.         for (i=0;i<8;i++)  
  33.         {   
  34.                 if(read_bit())
  35.                 dat|=(0x01<<i);    // 然后将其左移
  36.             Delay(4);
  37.         }
  38.     return (dat);
  39. }

  40. void write_bit(uchar bitval)        //写一位
  41. {
  42.         TRISC&=0xfd;//配置RC1为输出
  43.         RC1=0;         // 将RC1拉低开始写时间隙
  44.         Delay(1);
  45.         if(bitval==1)
  46.         TRISC|=0x02;//配置RC1为输入   // 如果写1,RC1返回高电平
  47.         Delay(10);        // 在时间隙内保持电平值,
  48.         TRISC|=0x02;
  49. }

  50. void WriteOneChar(unsigned char dat) //写一个字节
  51. {
  52.     unsigned char i = 0;
  53.     unsigned char temp;
  54.     for (i=0; i<8; i++)  // 写入字节, 每次写入一位  
  55.     {
  56.             temp=(dat>>i);  
  57.                    temp&=0x01;  
  58.                    write_bit(temp);
  59.     }
  60.     Delay(5);
  61. }

  62. uchar Read_RomCord(void) //读取64位序列码
  63. {
  64.     unsigned char j;
  65.     Init_DS18B20();
  66.     WriteOneChar(0x33);  // 读序列码的操作
  67.     for (j = 0; j < 8; j++)
  68.         {
  69.                   RomCode[j] = ReadOneChar() ;  
  70.         }
  71.         return (RomCode[0]);
  72. }
  73.        
复制代码

while(1)里面就这三句:       
a=Read_RomCord();
TXREG=a;
delay_1ms();
我看过时序,波形完全对的,但是读得都是FF。
下面是时序图,完全跟Read_RomCord的时序一样:

1111111.JPG (50.86 KB, 下载次数: 0)

下载附件

2014-10-24 11:01 上传

一周热门 更多>