STM32的DS18B20的驱动移植

发布者:平凡梦想最新更新时间:2017-04-04 来源: eefocus关键字:STM32  DS18B20  驱动移植 手机看文章 扫描二维码
随时随地手机看文章

折腾了一晚上,才把DS18B20的驱动移植到STM32上来。以前在51上使用过单个和多个连接的DS18B20,有现成的程序了,以为很快就能弄好,结果还是被卡住了,下面说下几个关键点吧:

   首先是延时的问题,STM32上若用软件延时的话不太好算时间,所以要么用定时器要么用SysTick这个定时器来完成延时的计算。相比之下用SysTick来的简单方便点。

   接着是STM32 IO脚的配置问题,因为51是双向的IO,所以作为输入输出都比较方便。STM32的IO是准双向的IO,网上查了下资料,说将STM32的IO配置成开漏输出,然后外接上拉即可实现双向IO。于是我也按规定做了,但调了老半天都不成功,是因为DS18B20没有响应的信号。在烦躁之际只有试下将接DQ的IO分别拉低和拉高看能不能读入正确的信号。结果果然是读入数据不对,原来我将IO配成开漏输出后相当然的以为读数据是用GPIO_ReadOutputDataBit(),这正是问题所在,后来将读入的函数改为GPIO_ReadInputDataBit()就OK了。现在温度是现实出来了,但跟我家里那台德胜收音机上显示的温度相差2度,都不知道是哪个准了,改天再找个温度计验证下。

   下面引用一段DS18B20的时序描述,写的很详细:

DS18B20的控制流程

   根据DS18B20的通信协议,DS18B20只能作为从机,而单片机系统作为主机,单片机控制DS18B20完成一次温度转换必须经过3个步骤:复位、发送ROM指令、发送RAM指令。每次对DS18B20的操作都要进行以上三个步骤。

    复位过程为:单片机将数据线拉低至少480uS,然后释放数据线,等待15-60uS让DS18B20接收信号,DS18B20接收到信号后,会把数据线拉低60-240uS,主机检测到数据线被拉低后标识复位成功;
    发送ROM指令:ROM指令表示主机对系统上所接的全部DS18B20进行寻址,以确定对那一个DS18B20进行操作,或者是读取某个DS18B20的ROM序列号。
    发送RAM指令:RAM指令用于单片机对DS18B20内部RAM进行操作,如读取寄存器的值,或者设置寄存器的值。
    具体的RAM和RAM指令请查阅DS18B20的数据手册。下面简单介绍:
      1、ROM操作命令:DS18B20采用一线通信接口。因为一线通信接口,必须在先完成ROM设定,否则记忆和控制功能将无法使用。一旦总线检测到从属器件的存在,它便可以发出器件ROM操作指令,所有ROM操作指令均为8位长度,主要提供以下功能命令:

1 )读ROM(指令码0X33H):当总线上只有一个节点(器件)时,读此节点的64位序列号。如果总线上存在多于一个的节点,则此指令不能使用。

 2 )ROM匹配(指令码0X55H):此命令后跟64位的ROM序列号,总线上只有与此序列号相同的DS18B20才会做出反应;该指令用于选中某个DS18B20,然后对该DS18B20进行读写操作。

3 )搜索ROM(指令码0XF0H):用于确定接在总线上DS18B20的个数和识别所有的64位ROM序列号。当系统开始工作,总线主机可能不知道总线上的器件个数或者不知道其64位ROM序列号,搜索命令用于识别所有连接于总线上的64位ROM序列号。

4 )跳过ROM(指令码0XCCH):此指令只适合于总线上只有一个节点;该命令通过允许总线主机不提供64位ROM序列号而直接访问RAM,以节省操作时间。

5 )报警检查(指令码0XECH):此指令与搜索ROM指令基本相同,差别在于只有温度超过设定的上限或者下限值的DS18B20才会作出响应。只要DS18B20一上电,告警条件就保持在设置状态,直到另一次温度测量显示出非告警值,或者改变TH或TL的设置使得测量值再一次位于允许的范围之内。储存在EEPROM内的触发器用于告警。

   

2、RAM指令

    DS18B20有六条RAM命令:

  1)温度转换(指令码0X44H):启动DS18B20进行温度转换,结果存入内部RAM。

  2)读暂存器(指令码0XBEH):读暂存器9个字节内容,此指令从RAM的第1个字节(字节0)开始读取,直到九个字节(字节8,CRC值)被读出为止。如果不需要读出所有字节的内容,那么主机可以在任何时候发出复位信号以中止读操作。

  3)写暂存器(指令码0X4EH):将上下限温度报警值和配置数据写入到RAM的2、3、4字节,此命令后跟需要些入到这三个字节的数据。

  4)复制暂存器(指令码0X48H):把暂存器的2、3、4字节复制到EEPROM中,用以掉电保存。

  5)重新调E2RAM(指令码0XB8H):把EEROM中的温度上下限及配置字节恢复到RAM的2、3、4字节,用以上电后恢复以前保存的报警值及配置字节。

6)读电源供电方式(指令码0XB4H):启动DS18B20发送电源供电方式的信号给主CPU。对于在此命令送至DS18B20后所发出的第一次读出数据的时间片,器件都会给出其电源方式的信号。“0”表示寄生电源供电。“1”表示外部电源供电。

 

 

     下面是结合实际测试总结出来的DS18B20的操作流程:

1、DS18B20的初始化

  (1) 先将数据线置高电平“1”。

  (2) 延时(该时间要求的不是很严格,但是尽可能的短一点)。

  (3) 数据线拉到低电平“0”。

  (4) 延时490微秒(该时间的时间范围可以从480到960微秒)。

  (5) 数据线拉到高电平“1”。

  (6)延时等待(如果初始化成功则在15到60毫秒时间之内产生一个由DS18B20所返回的低电平“0”。据该状态可以来确定它的存在,但是应注意不能无限的进行等待,不然会使程序进入死循环,所以要进行超时控制)。

  (7)若CPU读到了数据线上的低电平“0”后,还要做延时,其延时的时间从发出的高电平算起(第(5)步的时间算起)最少要480微秒。

  (8) 将数据线再次拉高到高电平“1”后结束。  

  2、DS18B20的写操作

  (1) 数据线先置低电平“0”。

  (2) 延时确定的时间为2(小于15)微秒。

  (3) 按从低位到高位的顺序发送字节(一次只发送一位)。

  (4) 延时时间为62(大于60)微秒。

  (5) 将数据线拉到高电平,延时2(小于15)微秒。

  (6) 重复上(1)到(6)的操作直到所有的字节全部发送完为止。

  (7) 最后将数据线拉高。  

  3、 DS18B20的读操作

  (1)将数据线拉高“1”。

  (2)延时2微秒。

  (3)将数据线拉低“0”。

  (4)延时2(小于15)微秒。

  (5)将数据线拉高“1”,同时端口应为输入状态。

  (6)延时4(小于15)微秒。

  (7)读数据线的状态得到1个状态位,并进行数据处理。

  (8)延时62(大于60)微秒。

 

 

顺便把程序也贴上来吧,给大家参考下。

使用的方法:

只要调用一次 ds18b20_start() 来初始化DS18B20,然后每次读温度时直接调用 ds18b20_read()就可以了。如

ds18b20_start();
 while(1)
 {
   for(i=1000000;i>0;i--);
   val = ds18b20_read();
 }


view plaincopy to clipboardprint?
//========================================================  
 
//       DS18B20.C   By ligh  
 
//========================================================  
 
#include "STM32Lib\\stm32f10x.h"  
#include "DS18B20.h"  
 
   
 
#define EnableINT()    
#define DisableINT()   
 
#define DS_PORT  GPIOA  
#define DS_DQIO  GPIO_Pin_1  
#define DS_RCC_PORT RCC_APB2Periph_GPIOA  
#define DS_PRECISION 0x7f  //精度配置寄存器 1f=9位; 3f=10位; 5f=11位; 7f=12位;  
#define DS_AlarmTH 0x64  
#define DS_AlarmTL 0x8a  
#define DS_CONVERT_TICK 1000  
 
#define ResetDQ() GPIO_ResetBits(DS_PORT,DS_DQIO)  
#define SetDQ() GPIO_SetBits(DS_PORT,DS_DQIO)  
#define GetDQ() GPIO_ReadInputDataBit(DS_PORT,DS_DQIO)  
   
   
static unsigned char TempX_TAB[16]={0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};  
 
 
void Delay_us(u32 Nus)   
{    
 SysTick->LOAD=Nus*9;         //时间加载         
 SysTick->CTRL|=0x01;            //开始倒数       
 while(!(SysTick->CTRL&(1<<16))); //等待时间到达    
 SysTick->CTRL=0X00000000;       //关闭计数器   
 SysTick->VAL=0X00000000;        //清空计数器        
}    
 
   
 
unsigned char ResetDS18B20(void)  
{  
 unsigned char resport;  
 SetDQ();  
 Delay_us(50);  
   
 ResetDQ();  
 Delay_us(500); //500us (该时间的时间范围可以从480到960微秒)  
 SetDQ();  
 Delay_us(40); //40us  
 //resport = GetDQ();  
 while(GetDQ());  
 Delay_us(500); //500us  
 SetDQ();  
 return resport;  
}  
 
void DS18B20WriteByte(unsigned char Dat)  
{  
 unsigned char i;  
 for(i=8;i>0;i--)  
 {  
  ResetDQ();    //在15u内送数到数据线上,DS18B20在15-60u读数  
 Delay_us(5);   //5us  
 if(Dat & 0x01)  
  SetDQ();  
 else 
  ResetDQ();  
 Delay_us(65);   //65us  
 SetDQ();  
 Delay_us(2);   //连续两位间应大于1us  
 Dat >>= 1;   
 }   
}  
 
 
unsigned char DS18B20ReadByte(void)  
{  
 unsigned char i,Dat;  
 SetDQ();  
 Delay_us(5);  
 for(i=8;i>0;i--)  
 {  
  Dat >>= 1;  
   ResetDQ();    //从读时序开始到采样信号线必须在15u内,且采样尽量安排在15u的最后  
 Delay_us(5);  //5us  
 SetDQ();  
 Delay_us(5);  //5us  
 if(GetDQ())  
   Dat|=0x80;  
 else 
  Dat&=0x7f;    
 Delay_us(65);  //65us  
 SetDQ();  
 }  
 return Dat;  
}  
 
 
void ReadRom(unsigned char *Read_Addr)  
{  
 unsigned char i;  
 
 DS18B20WriteByte(ReadROM);  
    
 for(i=8;i>0;i--)  
 {  
 *Read_Addr=DS18B20ReadByte();  
 Read_Addr++;  
 }  
}  
 
 
void DS18B20Init(unsigned char Precision,unsigned char AlarmTH,unsigned char AlarmTL)  
{  
 DisableINT();  
 ResetDS18B20();  
 DS18B20WriteByte(SkipROM);   
 DS18B20WriteByte(WriteScratchpad);  
 DS18B20WriteByte(AlarmTL);  
 DS18B20WriteByte(AlarmTH);  
 DS18B20WriteByte(Precision);  
 
 ResetDS18B20();  
 DS18B20WriteByte(SkipROM);   
 DS18B20WriteByte(CopyScratchpad);  
 EnableINT();  
 
 while(!GetDQ()); //等待复制完成 ///////////  
}  
 
 
void DS18B20StartConvert(void)  
{  
 DisableINT();  
 ResetDS18B20();  
 DS18B20WriteByte(SkipROM);   
 DS18B20WriteByte(StartConvert);   
 EnableINT();  
}  
 
void DS18B20_Configuration(void)  
{  
 GPIO_InitTypeDef GPIO_InitStructure;  
   
 RCC_APB2PeriphClockCmd(DS_RCC_PORT, ENABLE);  
 
 GPIO_InitStructure.GPIO_Pin = DS_DQIO;  
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; //开漏输出  
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //2M时钟速度  
 GPIO_Init(DS_PORT, &GPIO_InitStructure);  
}  
 
 
void ds18b20_start(void)  
{  
 DS18B20_Configuration();  
 DS18B20Init(DS_PRECISION, DS_AlarmTH, DS_AlarmTL);  
 DS18B20StartConvert();  
}  
 
 
unsigned short ds18b20_read(void)  
{  
 unsigned char TemperatureL,TemperatureH;  
 unsigned int Temperature;  
 
 DisableINT();  
 ResetDS18B20();  
 DS18B20WriteByte(SkipROM);   
 DS18B20WriteByte(ReadScratchpad);  
 TemperatureL=DS18B20ReadByte();  
 TemperatureH=DS18B20ReadByte();   
 ResetDS18B20();  
 EnableINT();  
 
 if(TemperatureH & 0x80)  
 {  
 TemperatureH=(~TemperatureH) | 0x08;  
 TemperatureL=~TemperatureL+1;  
 if(TemperatureL==0)  
  TemperatureH+=1;  
 }  
 
 TemperatureH=(TemperatureH<<4)+((TemperatureL&0xf0)>>4);  
 TemperatureL=TempX_TAB[TemperatureL&0x0f];  
 
 //bit0-bit7为小数位,bit8-bit14为整数位,bit15为正负位  
 Temperature=TemperatureH;  
 Temperature=(Temperature<<8) | TemperatureL;   
 
 DS18B20StartConvert();  
 
 return Temperature;  
}  
 
   
 
   
 
//============================================  
 
//     DS18B20.H  
 
//============================================  
 
#ifndef __DS18B20_H  
#define __DS18B20_H  
 
#define SkipROM   0xCC    //跳过ROM  
#define SearchROM 0xF0 //搜索ROM  
#define ReadROM   0x33 //读ROM  
#define MatchROM  0x55 //匹配ROM  
#define AlarmROM  0xEC //告警ROM  
 
#define StartConvert   0x44 //开始温度转换,在温度转换期间总线上输出0,转换结束后输出1  
#define ReadScratchpad 0xBE //读暂存器的9个字节  
#define WriteScratchpad 0x4E //写暂存器的温度告警TH和TL  
#define CopyScratchpad 0x48 //将暂存器的温度告警复制到EEPROM,在复制期间总线上输出0,复制完后输出1  
#define RecallEEPROM   0xB8   //将EEPROM的温度告警复制到暂存器中,复制期间输出0,复制完成后输出1  
#define ReadPower      0xB4   //读电源的供电方式:0为寄生电源供电;1为外部电源供电  
 
 
void ds18b20_start(void);  
unsigned short ds18b20_read(void);  
 
 
#endif 


关键字:STM32  DS18B20  驱动移植 引用地址:STM32的DS18B20的驱动移植

上一篇:SPI 调试-74HC595 数码管控制实验
下一篇:ARM7单片机(学习ing)—(八)、IIC接口—01

推荐阅读最新更新时间:2024-03-16 15:34

STM32 SysTick秒用
这里针对的是无操作系统的情况下的使用。之前一直想利用systick既实现记录系统运行时间又能够精确实现微秒延时的功能,如果将SysTick的定时器的定时中断时间设置为1us,这在有些情况下会导致死机的问题。这样的话就无法利用systick来实现us延时函数了。 利用SysTick实现1ms定时中断,us延时函数可以利用SysTick的寄存器来运算得到精确的延时函数,具体实现如下: 头文件: #ifndef __SYSTICK_H #define __SYSTICK_H #ifdef __cplusplus extern C { #endif /* Includes -----------------------
[单片机]
单片机DS18b20 温度检测液晶显示
/*----------------------------------------------- 名称:DS18b20 温度检测液晶显示 论坛:www.doflye.net 编写:shifang 日期:2009.5 修改:无 内容: ------------------------------------------------*/ #include reg52.h //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义 #include stdio.h #include 18b20.h #include 1602.h #include delay.h #defi
[单片机]
STM32电机方波】记录3——TIM1时基初始化配置
定时器分类 : STM32F1x 系列中,除了互联型的产品,共有 8 个定时器,分为基本定时器,通用定时器和高级定时器。 基本定时器 TIM6 和 TIM7 是一个 16 位的只能向上计数的定时器,只能定时,没有外部 IO。 通用定时器 TIM2/3/4/5 是一个 16 位的可以向上/下计数的定时器,可以定时,可以输出比较,可以输入捕捉,每个定时器有四个外部 IO。 高级定时器 TIM1/8是一个 16 位的可以向上/下计数的定时器,可以定时,可以输出比较,可以输入捕捉,还可以有三相电机互补输出信号,每个定时器有 8 个外部 IO。 高级定时器TIM1的库函数: 定时器TIM1时基结构体缺省值: { TI
[单片机]
【<font color='red'>STM32</font>电机方波】记录3——TIM1时基初始化配置
一款基于STM32的便携式二氧化碳监测仪设计
CO2浓度的检测方法大致分化学方法和物理方法。CO2浓度检测方法有滴定法、热催化法、气敏法、电化学法,这些属于化学方法,这些方法普遍存在价格贵,普适性差等问题,且测量精度较低。而物理的方法有超声波法、气相色谱法以及众多借助于光学来实现检测的方法。也有像光声光谱法这种化学和物理结合的方法。吸收光谱法的依据是不同化学结构的气体分子对不同波长的辐射的吸收程度不同,CO2气体分子对特定波长的红外光有强烈的吸收。   目前各种检测用的CO2传感器主要有固体电解质式、钛酸钡复合氧化物电容式、电导变化型厚膜式等,这些传感器存在对气体的选择性差、易出现误报、需要频繁校准、使用寿命较短等不足。而红外吸收型CO2传感器具有测量范围宽、灵敏度高、响应
[医疗电子]
一款基于<font color='red'>STM32</font>的便携式二氧化碳监测仪设计
STM32之关于通用定时器的输出比较方式
1.简单介绍 对于STM32中通用定时器的应用,定时器可以测量输入信号的脉冲长度(输入采集)或者产生输出波形(输出比较和PWM)。 如果小伙伴对于STM32的PWM不满意,因为相位无法控制,只能改变占空比。所以如果想改变PWM的相位的话,我们就可以用到输出比较方式了。 2.知识的架构 1)输出比较:打开一个TIMx计数器,再打开TIMx的一路或几路输出比较器(共4路),都配置好以后,计数器开始计数,当计数器里的值和比较寄存器里的值相等时,产生输出比较中断,在中断中将计数器中的值读出,与翻转周期相加再写道比较寄存器中,使得和下一个事件有相同的翻转周期。 大致意思为打开计数器后,计数值不断增加,到增加到比较寄存器的值时,电平翻转,也会
[单片机]
STM32 FSMC总线深入研究
由于CPU与FPGA通信的需要,以及对8080总线的熟悉,首选采用了STM32的FSMC总线,作为片间通信接口。FSMC能达到16MHz的写入速度,理论上能写20fps的1024*768的图片哈哈。(当然实际上是不可能的,就算是DMA传输,数据源也跟不上,实际上刷模拟的图片每秒10fps,刷的很high)当然这不是本篇的要点,这里主要研究STM32的FSMC接口,将速度提升到极限。 1. FSMC协议分析 如下为ILI9325的8080接口的协议 CS(片选信号):低电平片选有效,高电平失能(默认为高:失能) RS(数据寄存器):低电平写寄存器,高电平写数据(默认为高:写数据)FSMC默认为低。。。。 RD(读信号)
[单片机]
<font color='red'>STM32</font> FSMC总线深入研究
STM32串口一直进中断解决方法
今天在使用USART模块,遇到了一些问题并解决了,于是发贴共享。 问题描述: 在使用USART做串口通讯时,我只把接收中断打开,并设置抢占优先级为最低一个级别,而接收中断上一个优先级处理事情比较多,可能占用了2ms时间。当我使用9600波特率往下位机发送数据,速度非常快,就是一直按回车发!问题就出来,不到1分钟时间,通讯没有反应了。USART配置代码如下: void uart_config(void) { USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = UART_GetBaud(BaudRate);
[单片机]
<font color='red'>STM32</font>串口一直进中断解决方法
STM32电源管理系统浅谈
电源对电子设备的重要性不言而喻,它是保证系统稳定运行的基础,而保证系统能稳定运行后,又有低功耗的要求。在很多应用场合中都对电子设备的功耗要求非常苛刻,如某些传感器信息采集设备,仅靠小型的电池提供电源,要求工作长达数年之久,且期间不需要任何维护;由于智慧穿戴设备的小型化要求,电池体积不能太大导致容量也比较小,所以也很有必要从控制功耗入手,提高设备的续行时间。 STM32的电源管理系统主要分为: 1、备份域 2、调压器供电电路 3、ADC电源电路 备份域电路 STM32的备份域包括LSE振荡器、RTC、备份寄存器及备份SRAM这些器件,这部分的电路可以通过STM32的VBAT引脚获取供电电源,在实际应用中一般会使用3V的钮扣电
[单片机]
<font color='red'>STM32</font>电源管理系统浅谈
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved