STM32下单只DS18B20的驱动

发布者:CreativeMind最新更新时间:2020-12-01 来源: 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();

 }


//========================================================

 

//        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的驱动

上一篇:在STM32上移植FreeModbus RTU的一点经验总结
下一篇:经典_STM32_ADC多通道采样的例子

推荐阅读最新更新时间:2024-11-10 19:19

直流电机驱动使用时的注意事项
直流电机驱动在驱动电机时,保证H桥上两个同侧的三极管不会同时导通非常重要。如果三极管Q1和Q2同时导通,那么电流就会从正极穿过两个三极管直接回到负极。此时,电路中除了三极管外没有其他任何负载,因此电路上的电流就可能达到最大值(该电流仅受电源性能限制),甚至烧坏三极管。 直流电机驱动后面三极管和电阻,稳压管组成的电路进一步放大信号,驱动场效应管的栅极并利用场效应管本身的栅极电容(大约1000pF)进行延时,防止H桥上下两臂的场效应管同时导通(“共态导通”)造成电源短路。 当运放输出端为低电平(约为1V至2V,不能完全达到零)时,下面的三极管截止,场效应管导通。上面的三极管导通,场效应管截止,输出为高电平。当
[嵌入式]
基于FPGA和STM32的FSMC通信
1、FSMC简介:FSMC即灵活的静态存储控制器,FSMC管理1GB空间,拥有4个Bank连接外部存储器,每个Bank有独立的片选信号和独立的时序配置;支持的存储器类型有SRAM、PSRAM、NOR/ONENAND、ROM、LCD接口(支持8080和6800模式)、NANDFlash和16位的PCCard。 2、在设计中将FPGA当做SRAM来驱动,使用库函数来实现FSMC的初始化配置代码如下: //初始化外部SRAM void FSMC_SRAM_Init(void) { FSMC_NORSRAMInitTypeDef FSMC_NORSRAMInitStructure; //定义FSMC初始化的结构体变量 FS
[单片机]
MLX90614非接触式红外体温计的单片机驱动程序
单片机源程序如下: #include reg52.h #include MLX906.h #include LCD1602.h #include EEPROM.h #include I2C.h #include WT558D.h #include USART.h sbit red =P3^3; sbit green =P3^4; sbit blue =P3^5; sbit key =P3^6; uchar tab = Temp: ; int main() { uchar i; uint wendu; //存储读取的值 uint xiaoshu,zhengshu;
[单片机]
STM32踩坑记录(1):编写代码时遇到STM32无法进入外部中断的原因及解决方法
@STM32踩坑记录(1) 使用的软件及硬件: 硬件平台:RoboMaster C板(STM32F407IGT6)、Jlink仿真调试器 软件平台:keil5 使用的是标准库+freertos。 ##遇到问题时的场景 在编写RoboMaster C板上的IMU(BMI088)温度闭环控制程序时 ,需要配置外部中断引脚 INT1_ACCEL_Pin(PC4),然后在中断服务函数中给温度控制任务发通知(也就是信号量),温度控制任务就会读取BMI088数据寄存器中的陀螺仪、加速度、温度的值,将温度的值传递给PID控制器,进行PID闭环控温程序。 ##:遇到的问题: 因为板载的BMI088上有两个外部中断引脚,分别为INT1
[单片机]
<font color='red'>STM32</font>踩坑记录(1):编写代码时遇到<font color='red'>STM32</font>无法进入外部中断的原因及解决方法
STM32 USB 设备音频数据流
音频数据流的应用 音频数据流是没有任何压缩的音频数据,我们可以直接通过I2S传输到数字功放,也可以对这些数字音频进行处理,比如EQ音量控制、音质补偿等等。 典型应用举例 首先是PC,然后PC通过USB数据线连接到MCU,MCU再通过I2S输出到功放,最后连接到喇叭或耳机。 当然,还有GUI的部分显示可以控制,同时还可以通过麦克风把音频传输到PC进行刻录或保存。 同步问题 USB外设时钟、I2C外设时钟和外部功放时钟是同步的,因为它们共用一个时钟域,但不能同步PC的时钟域。 这些不同的时钟域将会造成音频的不同步,出现断音或丢失部分音频。 主要问题 1. USB的参考时钟(SOF)并不跟系统时钟同步,主要表现在三个方面
[单片机]
STM32驱动LCD12864显示屏
我们做一个电子产品,往往需要实现人机交互的功能。那么人机交互的方式除了输出到上位机通过电脑去显示,显示器也是一个很不错的方式,可用于一些不能使用电脑的场合。LCD12864显示器中的一种,具有价格低廉,操作简单的优点。今天就为大家带来一个STM32驱动12864的例程,使用SPI串行通信,仅仅需要三根数据线就可以完成通信。废话不多说,进入正题。 接线: RS----PB15 RW----PB14 EN----PB13 PSB---GND 1.初始化IO口以及显示屏 void Lcd_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2Per
[单片机]
STM32的IIC应用详解2
IIC简单介绍 小编能力有限,写的不对处还望诸位大侠指正哈! 平时所说的IIC通信指的是用单片机的两个I/O端口模拟出来的IIC,正真的IIC实际上是一块硬件电路,那是飞利浦公司的专利,要想用那就拿钱来买。有大牛既想用又不想花钱,就用两个端口模拟出了IIC通信协议,因为方便(51上的IIC改一下端口配置就可以在STM32F103上使用)所以被广泛使用。啰嗦了这么多,下面进入正题,嘿嘿。 首先IIC通信由两根线组成: 时钟线SCL:在通信过程起到控制作用。 数据线SDA:用来一位一位的传送数据。 其次IIC通信过程由开始、结束、发送、接收四个函数构成,接下来小编通过介绍这四
[单片机]
利用电荷泵降低白光LED背光驱动器的成本和体积
  在手机和其他移动设备中,白光LED能为小尺寸彩屏提供完美的背光效果。但大部分手机使用单节锂电池供电,而单节锂电池很难直接驱动白光LED。通常锂电池的工作电压范围为3~4.2V,而白光LED的导通压降是 3.5~4.2V(20mA)。因此,锂电池电压降低后将无法直接驱动白光LED。   为了给白光LED提供足够的正向压降,可以使用基于电容的 电荷泵 或基于电感的升压电路。考虑到效率和电池寿命,基于电感的转换器可能是最好的选择,但是额外的电感会增加系统成本。而且,由于EMI和RF干扰,电感型升压电路需要仔细的设计和布板。与之相比,电荷泵解决方案具有价格便宜、易使用等优势,但效率较低,缩短了电池使用寿命。   随着电荷泵设计
[电源管理]
利用电荷泵降低白光LED背光<font color='red'>驱动</font>器的成本和体积
小广播
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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