STM32下DS18B20的驱动
下面是结合实际测试总结出来的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
STM32DS18B2 相关文章:
- STM32 DS18B20代码详解(11-13)
- Windows CE 进程、线程和内存管理(11-09)
- RedHatLinux新手入门教程(5)(11-12)
- uClinux介绍(11-09)
- openwebmailV1.60安装教学(11-12)
- Linux嵌入式系统开发平台选型探讨(11-09)