一般常见的STM32的关于DS18B20的例程都是检测一个传感器,代码一般都是跳过ROM检测,直接获取温度值。这种写法并不适用于单总线上挂载多个DS18B20的情况,Sandeepin的这个代码就是针对这种情况完善的单总线挂多个DS18B20检测,实现获取每个DS18B20的ID和温度。

  主要的DS18B20时序代码没变,增加了搜索ROM函数,获取温度时先匹配ID。

  核心代码如下:

  DS18B20.c文件代码:

#include "DS18B20.h"
#include "Delay.h"
#include "stdio.h" // printf用 #define DS18B20_GPIO_NUM GPIO_Pin_5
#define DS18B20_GPIO_X GPIOC
#define RCC_APB2Periph_DS18B20_GPIO_X RCC_APB2Periph_GPIOC #define DS18B20_DQ_OUT_Low GPIO_ResetBits(DS18B20_GPIO_X,DS18B20_GPIO_NUM)
#define DS18B20_DQ_OUT_High GPIO_SetBits(DS18B20_GPIO_X,DS18B20_GPIO_NUM)
#define DS18B20_DQ_IN GPIO_ReadInputDataBit(DS18B20_GPIO_X,DS18B20_GPIO_NUM) #define MaxSensorNum 8
unsigned char DS18B20_ID[MaxSensorNum][8]; // 存检测到的传感器DS18B20_ID的数组,前面的维数代表单根线传感器数量上限
unsigned char DS18B20_SensorNum; // 检测到的传感器数量(从1开始,例如显示1代表1个,8代表8个) // 配置DS18B20用到的I/O口
void DS18B20_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_DS18B20_GPIO_X, ENABLE);
GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_NUM;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(DS18B20_GPIO_X, &GPIO_InitStructure);
GPIO_SetBits(DS18B20_GPIO_X, DS18B20_GPIO_NUM);
} // 引脚输入
void DS18B20_Mode_IPU(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_NUM;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_Init(DS18B20_GPIO_X, &GPIO_InitStructure);
} // 引脚输出
void DS18B20_Mode_Out(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = DS18B20_GPIO_NUM;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(DS18B20_GPIO_X, &GPIO_InitStructure); } // 复位,主机给从机发送复位脉冲
void DS18B20_Rst(void)
{
DS18B20_Mode_Out();
DS18B20_DQ_OUT_Low; // 产生至少480us的低电平复位信号
Delay_us(480);
DS18B20_DQ_OUT_High; // 在产生复位信号后,需将总线拉高
Delay_us(15);
} // 检测从机给主机返回的应答脉冲。从机接收到主机的复位信号后,会在15~60us后给主机发一个应答脉冲
u8 DS18B20_Answer_Check(void)
{
u8 delay = 0;
DS18B20_Mode_IPU(); // 主机设置为上拉输入
// 等待应答脉冲(一个60~240us的低电平信号 )的到来
// 如果100us内,没有应答脉冲,退出函数,注意:从机接收到主机的复位信号后,会在15~60us后给主机发一个存在脉冲
while (DS18B20_DQ_IN&&delay < 100)
{
delay++;
Delay_us(1);
}
// 经过100us后,如果没有应答脉冲,退出函数
if (delay >= 100)//Hu200
return 1;
else
delay = 0;
// 有应答脉冲,且存在时间不超过240us
while (!DS18B20_DQ_IN&&delay < 240)
{
delay++;
Delay_us(1);
}
if (delay >= 240)
return 1;
return 0;
} // 从DS18B20读取1个位
u8 DS18B20_Read_Bit(void)
{
u8 data;
DS18B20_Mode_Out();
DS18B20_DQ_OUT_Low; // 读时间的起始:必须由主机产生 >1us <15us 的低电平信号
Delay_us(2);
DS18B20_DQ_OUT_High;
Delay_us(12);
DS18B20_Mode_IPU();// 设置成输入,释放总线,由外部上拉电阻将总线拉高
if (DS18B20_DQ_IN)
data = 1;
else
data = 0;
Delay_us(50);
return data;
} // 从DS18B20读取2个位
u8 DS18B20_Read_2Bit(void)//读二位 子程序
{
u8 i;
u8 dat = 0;
for (i = 2; i > 0; i--)
{
dat = dat << 1;
DS18B20_Mode_Out();
DS18B20_DQ_OUT_Low;
Delay_us(2);
DS18B20_DQ_OUT_High;
DS18B20_Mode_IPU();
Delay_us(12);
if (DS18B20_DQ_IN) dat |= 0x01;
Delay_us(50);
}
return dat;
} // 从DS18B20读取1个字节
u8 DS18B20_Read_Byte(void) // read one byte
{
u8 i, j, dat;
dat = 0;
for (i = 0; i < 8; i++)
{
j = DS18B20_Read_Bit();
dat = (dat) | (j << i);
}
return dat;
} // 写1位到DS18B20
void DS18B20_Write_Bit(u8 dat)
{
DS18B20_Mode_Out();
if (dat)
{
DS18B20_DQ_OUT_Low;// Write 1
Delay_us(2);
DS18B20_DQ_OUT_High;
Delay_us(60);
}
else
{
DS18B20_DQ_OUT_Low;// Write 0
Delay_us(60);
DS18B20_DQ_OUT_High;
Delay_us(2);
}
} // 写1字节到DS18B20
void DS18B20_Write_Byte(u8 dat)
{
u8 j;
u8 testb;
DS18B20_Mode_Out();
for (j = 1; j <= 8; j++)
{
testb = dat & 0x01;
dat = dat >> 1;
if (testb)
{
DS18B20_DQ_OUT_Low;// 写1
Delay_us(10);
DS18B20_DQ_OUT_High;
Delay_us(50);
}
else
{
DS18B20_DQ_OUT_Low;// 写0
Delay_us(60);
DS18B20_DQ_OUT_High;// 释放总线
Delay_us(2);
}
}
} //初始化DS18B20的IO口,同时检测DS的存在
u8 DS18B20_Init(void)
{
DS18B20_GPIO_Config();
DS18B20_Rst();
return DS18B20_Answer_Check();
} // 从ds18b20得到温度值,精度:0.1C,返回温度值(-550~1250),Temperature1返回浮点实际温度
float DS18B20_Get_Temp(u8 i)
{
//u8 flag;
u8 j;//匹配的字节
u8 TL, TH;
short Temperature;
float Temperature1;
DS18B20_Rst();
DS18B20_Answer_Check();
DS18B20_Write_Byte(0xcc);// skip rom
DS18B20_Write_Byte(0x44);// convert
DS18B20_Rst();
DS18B20_Answer_Check(); // DS18B20_Write_Byte(0xcc);// skip rom
//匹配ID,i为形参
DS18B20_Write_Byte(0x55);
for (j = 0; j < 8; j++)
{
DS18B20_Write_Byte(DS18B20_ID[i][j]);
} DS18B20_Write_Byte(0xbe);// convert
TL = DS18B20_Read_Byte(); // LSB
TH = DS18B20_Read_Byte(); // MSB
if (TH & 0xfc)
{
//flag=1;
Temperature = (TH << 8) | TL;
Temperature1 = (~Temperature) + 1;
Temperature1 *= 0.0625;
}
else
{
//flag=0;
Temperature1 = ((TH << 8) | TL)*0.0625;
}
return Temperature1;
} // 自动搜索ROM
void DS18B20_Search_Rom(void)
{
u8 k, l, chongtuwei, m, n, num;
u8 zhan[5];
u8 ss[64];
u8 tempp;
l = 0;
num = 0;
do
{
DS18B20_Rst(); //注意:复位的延时不够
Delay_us(480); //480、720
DS18B20_Write_Byte(0xf0);
for (m = 0; m < 8; m++)
{
u8 s = 0;
for (n = 0; n < 8; n++)
{
k = DS18B20_Read_2Bit();//读两位数据 k = k & 0x03;
s >>= 1;
if (k == 0x01)//01读到的数据为0 写0 此位为0的器件响应
{
DS18B20_Write_Bit(0);
ss[(m * 8 + n)] = 0;
}
else if (k == 0x02)//读到的数据为1 写1 此位为1的器件响应
{
s = s | 0x80;
DS18B20_Write_Bit(1);
ss[(m * 8 + n)] = 1;
}
else if (k == 0x00)//读到的数据为00 有冲突位 判断冲突位
{
//如果冲突位大于栈顶写0 小于栈顶写以前数据 等于栈顶写1
chongtuwei = m * 8 + n + 1;
if (chongtuwei > zhan[l])
{
DS18B20_Write_Bit(0);
ss[(m * 8 + n)] = 0;
zhan[++l] = chongtuwei;
}
else if (chongtuwei < zhan[l])
{
s = s | ((ss[(m * 8 + n)] & 0x01) << 7);
DS18B20_Write_Bit(ss[(m * 8 + n)]);
}
else if (chongtuwei == zhan[l])
{
s = s | 0x80;
DS18B20_Write_Bit(1);
ss[(m * 8 + n)] = 1;
l = l - 1;
}
}
else
{
//没有搜索到
}
}
tempp = s;
DS18B20_ID[num][m] = tempp; // 保存搜索到的ID
}
num = num + 1;// 保存搜索到的个数
} while (zhan[l] != 0 && (num < MaxSensorNum));
DS18B20_SensorNum = num;
//printf("DS18B20_SensorNum=%d\r\n",DS18B20_SensorNum);
}

  DS18B20.h文件代码:

#ifndef __DS18B20_H
#define __DS18B20_H #include "stm32f10x.h" u8 DS18B20_Init(void);
u8 DS18B20_Read_Byte(void);
u8 DS18B20_Read_Bit(void);
u8 DS18B20_Answer_Check(void);
void DS18B20_GPIO_Config(void);
void DS18B20_Mode_IPU(void);
void DS18B20_Mode_Out(void);
void DS18B20_Rst(void);
void DS18B20_Search_Rom(void);
void DS18B20_Write_Byte(u8 dat);
float DS18B20_Get_Temp(u8 i); #endif

  main.c文件代码:

#include "stm32f10x.h"
#include "stdio.h"
#include "string.h"//strlen、memset用到
#include "USART.h"
#include "Delay.h"
#include "DS18B20.h" extern unsigned char DS18B20_ID[8][8];//检测到的传感器ID存数组
extern unsigned char DS18B20_SensorNum; int main(void)
{
u8 num=0;
USART1_init(9600);
while(DS18B20_Init())//初始化DS18B20,兼检测18B20
{
printf("DS18B20 Check Failed!\r\n");
}
printf("DS18B20 Ready!\r\n");
while(1)
{
DS18B20_Search_Rom();
printf("DS18B20_SensorNum:%d\r\n",DS18B20_SensorNum);
for(num=0;num<DS18B20_SensorNum;num++)
{
printf("ID:%02x%02x%02x%02x%02x%02x%02x%02x TM:%.2f\r\n",DS18B20_ID[num][0],DS18B20_ID[num][1],DS18B20_ID[num][2],DS18B20_ID[num][3],DS18B20_ID[num][4],DS18B20_ID[num][5],DS18B20_ID[num][6],DS18B20_ID[num][7],DS18B20_Get_Temp(num));
}
printf("\r\n");
Delay_s(2);
}
}

  运行结果如图:

  帮严博士出本科题的时候,出了一个DS18B20的分布式温度检测系统,要求肯定不仅仅是这篇文章的简略例子了。不仅单总线,一块单片机还要挂多总线,实现更多传感器数据采集,最好还配上上位机,反正把自己能想到的东西都加进来了,把一个简单的DS18B20包装得高大上。

完美实现STM32单总线挂多个DS18B20的更多相关文章

  1. DS18B20 for STM32 源代码 【worldsing笔记】

    DS18B20是DALLAS公司生产的一线式数字温度传感器,具有3引脚TO-92小体积封装形式:温度测量范围为-55℃-+125℃,可编程为9位-12位A/D转换精度,测温分辨率可达0.0625℃.主 ...

  2. 「雕爷学编程」Arduino动手做(39)——DS18B20温度传感器

    37款传感器与模块的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止37种的.鉴于本人手头积累了一些传感器和模块,依照实践出真知(一定要动手做)的理念,以学习和交流为目的,这里 ...

  3. STM32驱动DS18B20

    DS18B20 是由 DALLAS 半导体公司推出的一种的“一线总线”接口的温度传感器.与传 统的热敏电阻等测温元件相比,它是一种新型的体积小.适用电压宽.与微处理器接口简单的 数字化温度传感器.一线 ...

  4. 1-wire单总线DS18B20

    要想实现单总线通信,每一个挂在总线上的从机必须拥有开路或3态输出.单总线DS18B20的DQ引脚用内部电路实现了开漏输出,其等效电路如下图: 当单片机IO引脚配置为 mcu IO引脚 电流流向 DS1 ...

  5. 【蓝桥杯单片机11】单总线温度传感器DS18B20的基本操作

    [蓝桥杯单片机11]单总线温度传感器DS18B20的基本操作 广东职业技术学院 欧浩源 单总线数字温度传感器DS18B20几乎成了各类单片机甚至ARM实验板的标配模块来,在蓝桥杯的往届省赛和国赛中,这 ...

  6. 玩转X-CTR100 l STM32F4 l DS18B20单总线温度传感器

    我造轮子,你造车,创客一起造起来!塔克创新资讯[塔克社区 www.xtark.cn ][塔克博客 www.cnblogs.com/xtark/ ]      本文介绍X-CTR100控制器 扩展DS1 ...

  7. stm32 ds18b20 温度传感器

    相关文章:http://blog.csdn.net/zhangxuechao_/article/details/74991985 举例 void DS18B20_in() { GPIO_InitTyp ...

  8. TCP/IP协议学习(三) STM32中ETH驱动配置注意事项

    1.MII/RMII/SMI接口连接和配置 SMI又称站点管理接口,用于cpu与外置PHY芯片通讯,配置相关参数,包含MDC和MDIO两个管脚(CPU上有对应引脚,当然用普通GPIO口模拟SMI管理也 ...

  9. 基于ARM-LINUX的温度传感器驱动-DS18B20

    转载:http://blog.csdn.net/ayangke/article/details/6883244 作者:冯建,华清远见嵌入式学院讲师. DS18B20数字温度传感器接线方便,封装成后可应 ...

随机推荐

  1. css3 移动端旋转动画暂停

    音乐播放图片旋转动画 ios不支持暂停: animation-play-state: paused; @-webkit-keyframes rotate{ 100% { transform: rota ...

  2. [luogu1908]逆序对(upper_bound)

    对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对 用upper_bound法求逆序对,Code很棒 据说有用树状数组和线段树写逆序对的,这里用upper_bound水一 ...

  3. 大数据vs计算机

    大数据有两个方向,一个是偏计算机的,另一个是偏经济的.你学过Java,所以你可以偏将计算机 基础1. 读书<Introduction to Data Mining>,这本书很浅显易懂,没有 ...

  4. javascript 闭包的理解(二)

    // 定义一个User构造函数 function User(properties){ //遍历对象属性,确保它作用域正确 for(var i in properties){ (function(whi ...

  5. ES的索引查询和删除

    postman 1.查看es状态 get http://127.0.0.1:9200/_cat/health 红色表示数据不可用,黄色表示数据可用,部分副本没有分配,绿色表示一切正常 2.查看所有索引 ...

  6. 第二阶段:4.商业需求文档MRD:6.PRD-其他需求

  7. C# 中的IComparable和IComparer

    前言 在开发过程中经常会遇到比较排序的问题,比如说对集合数组的排序等情况,基本类型都提供了默认的比较算法,如string提供了按字母进行排序,而int整数则是根据整数大小进行排序.但是在引用类型中(具 ...

  8. win10开启我的第一个32位汇编程序

    遥想当年,上学期间,汇编程序,从未成功.今又试之,终成功,遂记录. Hello.asm文件如下: . .model flat,stdcall option casemap:none include w ...

  9. 「洛谷P3469」[POI2008]BLO-Blockade 解题报告

    P3469[POI2008]LO-Blockade 题意翻译 在Byteotia有n个城镇. 一些城镇之间由无向边连接. 在城镇外没有十字路口,尽管可能有桥,隧道或者高架公路(反正不考虑这些).每两个 ...

  10. (三)unittest断言方法的介绍

    断言如同在测试用例上,类似于预期结果与实际结果是否一致,如果一致则表示测试通过,Assert断言很好的用于测试结果判断上,更灵活的对预期结果和实际结果进行对比,下面简单的介绍一下unittest的As ...