码迷,mamicode.com
首页 > 其他好文 > 详细

STM32F207V 进行DS18B20处理

时间:2015-07-24 12:21:07      阅读:295      评论:0      收藏:0      [点我收藏+]

标签:

1、  DS18B20接口很简单,VCC、DQ、GND三个引脚,VCC采用外部供电3.3V,DQ需上拉电阻,当时按照参考资料上外接4.7K的上拉电阻,GPIO设置的OD无上拉,始终读不到ROM中的64位序列号,后面发送的转换指令更别想了,后来GPIO改成推挽输出时,诶,就能读到数据,推挽输出无非就是增大驱动能力,仔细看手册,其中有提到,DS18B20空闲处于低功耗状态,在执行温度转换或从高速暂存器向EPPROM传送数据时,工作电流可高达1.5mA,但并有说读ROM中的64位数据时,电流需要多少,也许在读ROM中的数据电流也需要要求。知道了加大驱动能力才能读到,加上之前写I2C和FLASH擦除时,数据线都是设置OD类型,通用性较好,好移植,所以还是改成OD类型,把4.7K的电阻换小点,换成1K电阻,就可以正确的读写操作。

2、操作步骤,在使用任何一条功能指令前,都必须先执行有复位、ROM指令。

(1)、复位信号就是双方握手,达成协议后才能相互通信。时序如下,一定要注意minimum和maximum,Master主机先发送低电平脉冲,低电平持续480us~960us(这里设置700us,延迟后记得释放数据线,也就是拉高),DS18B20等待15~60us后(用while语句检测低电平的到来),会发出低电平脉冲信号给Master,该低电平脉冲60~240us,Master接收到后,就说明握手成功,用while语句检测高电平结束。

技术分享

代码:

 1 void Ds18b20_rst(void)
 2 {
 3     DS18B20_DATA_H;
 4     DS18B20_DATA_L;
 5     Ds18b20_delay(T700US);         //700us
 6     DS18B20_DATA_H;                //释放数据线
 7     while(DS18B20_DATA_Read);      //检测低电平
 8     Ds18b20_delay(T700US);
 9     while(!DS18B20_DATA_Read);    //检测高电平
10     DS18B20_DATA_H;
11 }

(2)、发ROM指令和功能指令,就是1bit 1bit 的往外发,读也是1bit 1bit的往里读,发送顺序是先低后高。读写时有高有低,一定要按照时序严格进行操作:

  a)、写“0”  和“1”时序,先拉低15us,发送高电平持续45us,注意不低于60us,发送完1bit,释放数据线。

技术分享

代码:

 1 void Ds18b20_Send_0(void)
 2 {
 3     DS18B20_DATA_L;
 4     Ds18b20_delay(T15US);   //15us
 5     DS18B20_DATA_L;
 6     Ds18b20_delay(T45US);        //45us
 7     DS18B20_DATA_H;        //释放总线
 8 }
 9 
10 void Ds18b20_Send_1(void)
11 {
12     DS18B20_DATA_L;
13     Ds18b20_delay(T15US);   //15us
14     DS18B20_DATA_H;
15     Ds18b20_delay(T45US);
16     DS18B20_DATA_H;        //释放总线
17 }
18 
19 void Ds18b20_send_byte(uint8_t data)
20 {
21     uint8_t i = 8;
22     
23     while(i--)
24     {
25         if(data & 0x01)    //从低位开始发送
26             Ds18b20_Send_1();
27         else 
28             Ds18b20_Send_0();
29         
30         data >>= 1;
31     }
32 }

   b)、读“0”  和“1”时序,在MASTER SAMPLES就可以采样数据,但一般会大于15us后开始采集数据,读写前 和 读写完后,都要记得释放数据

技术分享

代码:

 1 uint8_t Ds18b20_read_byte(void)
 2 {
 3     uint8_t i = 8;
 4     uint8_t data_buffer = 0;
 5     
 6     DS18B20_DATA_H;        //释放数据线
 7     while(i--)
 8     {
 9         data_buffer >>= 1;
10         DS18B20_DATA_L;
11         Ds18b20_delay(T1US);        //1us
12         DS18B20_DATA_H;                //释放数据线
13         Ds18b20_delay(T15US); 
14         if(DS18B20_DATA_Read)    //接收低位开始
15         {
16             data_buffer |= 0x80;
17         }
18         Ds18b20_delay(T45US);
19         DS18B20_DATA_H;    
20     }
21     
22     return data_buffer;
23 }

    写好写字节函数 和 读字节函数,那么就可以发送ROM指令和功能指令了,具体指令参考数据手册。

一些具体操作指令代码:

技术分享
 1 void Ds18b20_Read_ROM(uint8_t *pBuff)
 2 {
 3     uint8_t i =0;
 4     Ds18b20_rst();
 5     Ds18b20_send_byte(0x33);            //读Rom指令
 6     for(i=0; i<8; i++)
 7     {
 8         pBuff[i] = Ds18b20_read_byte();
 9     }
10 }
11 
12 void Ds18b20_Convert_temprature(void)
13 {
14     Ds18b20_rst();
15     Ds18b20_send_byte(0xcc);                //忽略ROM指令
16     Ds18b20_send_byte(0x44);                //转换指令
17 }
18 
19 void Ds18b20_Read_register(uint8_t *pBuff)
20 {
21     uint8_t i = 0;
22     Ds18b20_Convert_temprature();
23     Ds18b20_rst();
24     Ds18b20_send_byte(0xcc);                //忽略ROM指令
25     Ds18b20_send_byte(0xbe);                //读取存储器指令
26     for(i=0; i<8; i++)
27     {
28         pBuff[i] = Ds18b20_read_byte();
29     }
30 }
31 
32 void Ds18b20_weite_threshold(uint8_t TH, uint8_t TL, uint8_t Resolution)
33 {
34     Ds18b20_rst();
35     Ds18b20_send_byte(0xcc);                //忽略ROM指令
36     Ds18b20_send_byte(0x4e);                //写暂存器指令
37     Ds18b20_send_byte(TH);
38     Ds18b20_send_byte(TL);
39     if(Resolution == 9)
40     {
41         Ds18b20_send_byte(0x1f);
42     }
43     else if(Resolution == 10)
44     {
45         Ds18b20_send_byte(0x3f);
46     }
47     else if(Resolution == 11)
48     {
49         Ds18b20_send_byte(0x5f);
50     }
51     else //if(Resolution == 12)    //其他任何都默认设置12bit分辨率
52     {
53         Ds18b20_send_byte(0x7f);
54     }
55 }
View Code

主函数:

技术分享
 1 int main(void)
 2 {
 3     u8 i = 0;
 4     u8 id[8];
 5     RCC_ClocksTypeDef rcc_clocks;
 6 
 7     Clock_Config();
 8     USART_Config();
 9 
10     Ds18b20_GPIO_Config();
11     Ds18b20_Read_ROM(id);
12     printf("DS18B20 64bit ROM Data:\n");
13     for(i=0; i<8; i++)
14     {
15         printf("0x%02x,",id[i]);
16     }
17     printf("\n");
18     printf("DS18B20 Memory Data:\n");
19     
20     Ds18b20_Read_register(id);
21     for(i=0; i<8; i++)
22     {
23         printf("0x%02x,",id[i]);
24     }
25     
26     printf("\n");
27     
28     Ds18b20_weite_threshold(0x3b,0x36,12);
29     Ds18b20_Read_register(id);
30     for(i=0; i<8; i++)
31     {
32         printf("0x%02x,",id[i]);
33     }
34     
35     printf("\n");
36     
37     while(1)
38     {    
39         Ds18b20_Read_register(id);
40         printf("temprature:%.4f\n",((id[1]<<8) | id[0]) * 6.25 /100);
41         Delay(0x1fffff0);   /* delay 1000ms */  
42     }
43 }
View Code

3、打印信息

(1)、读5个不同DS18B20中ROM数据:

    0x28,0x37,0x4f,0xca,0x06,0x00,0x00,0x03
    0x28,0x3b,0xf4,0xc9,0x06,0x00,0x00,0xc9
    0x28,0xee,0xa0,0xcc,0x06,0x00,0x00,0xed
    0x28,0xca,0x02,0xcb,0x06,0x00,0x00,0xb1
    0x28,0xf0,0xbf,0xcb,0x06,0x00,0x00,0x30

    最前面8位是单线系列编码固定28H,接着48位是唯一序列号,最后8位是CRC,有关CRC计算参考手册,这里也没有去验证CRC是否正确。

 (2)、

  DS18B20 64bit ROM Data:

  0x28,0xf0,0xbf,0xcb,0x06,0x00,0x00,0x30,   //64位中的序列号

  DS18B20 Memory Data:

  0xcd,0x01,0x4b,0x46,0x7f,0xff,0x03,0x10,  //读暂存器中的默认值

  0xcd,0x01,0x3b,0x36,0x7f,0xff,0x03,0x10, //改写后,读暂存器中的数据

  temprature:28.8125            //转换温度

printf("temprature:%.4f\n",((id[1]<<8) | id[0]) * 6.25 /100);

这里只考虑了正数温度,关于*6.25*100,参考数据手册:"温度传感器的精度为用户可编程的9、10、11或12位,分别以0.5°、0.25°、0.125°和0.0625°增量递增",程序选的是12bit分辨率,单位增量即为0.0625

STM32F207V 进行DS18B20处理

标签:

原文地址:http://www.cnblogs.com/wen2376/p/4672863.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!