基于STM32Cube MX开发的DS18B20驱动

本文档详细介绍了使用STM32F03VET6单片机的DS18B20温度传感器的HAL库驱动,包括初始化、延时、引脚配置、复位、检测、读写操作及温度测量,旨在学习HAL库并分享开发过程。

最近准备把以前搞得许多小硬件用HAL库重新写一遍,这样做的目的更多的是想学习一下HAL库以及记录一下自己的开发过程,同时在这样的过程中要是能够帮助到需要的人那也是相当不错的。

下面直接上代码:

/**
    @文 件 名:DS18B20.C
    @功能描述:温度传感器DS18B20初始化驱动代码
    @日    期:5/2/2022
    @作    者:自由eπ
    @注    释:本段程序是基于STM32F03VET6单片机和STM32Cube MX工具开发的
*/

#include "main.h"
#include "ds18b20.h"

/**
    @Brief :  void delay_us(uint8_t Time)
    @Param:   无
    @Function:us延时
    @Return:  无
    @Author:  自由eπ
  @Note:    无 
*/
void delay_us(uint32_t udelay)
{
  uint32_t startval,tickn,delays,wait;
 
  startval = SysTick->VAL;
  tickn = HAL_GetTick();
  //sysc = 72000;  //SystemCoreClock / (1000U / uwTickFreq);
  delays =udelay * 72; //sysc / 1000 * udelay;
  if(delays > startval)
    {
      while(HAL_GetTick() == tickn)
        {
 
        }
      wait = 72000 + startval - delays;
      while(wait < SysTick->VAL)
        {
 
        }
    }
  else
    {
      wait = startval - delays;
      while(wait < SysTick->VAL && HAL_GetTick() == tickn)
        {
 
        }
    }
}
/**
    @Brief :  static void Set_DS_Pin_In(void)
    @Param:   无
    @Function:对温度传感器的数据引脚配置为输入模式
    @Return:  无
    @Author:  自由eπ
  @Note:    无 
*/
static void Set_DS_Pin_In(void){
  GPIO_InitTypeDef GPIO_InitStruct = {0};
    
        /* GPIO Ports Clock Enable */
  __HAL_RCC_GPIOB_CLK_ENABLE();
    
  GPIO_InitStruct.Pin = DS_PORT_Pin;
    GPIO_InitStruct.Mode =GPIO_MODE_INPUT;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
    
  HAL_GPIO_Init(DS_PORT_GPIO_Port, &GPIO_InitStruct);
}
/**
    @Brief :  static void Set_DS_Pin_Out(void)
    @Param:   无
    @Function:对温度传感器的数据引脚配置为出模式
    @Return:  无
    @Author:  自由eπ
  @Note:    无 
*/
static void Set_DS_Pin_Out(void){
  GPIO_InitTypeDef GPIO_InitStruct = {0};
    
    /* GPIO Ports Clock Enable */
  __HAL_RCC_GPIOB_CLK_ENABLE();
    
  GPIO_InitStruct.Pin = DS_PORT_Pin;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    
  HAL_GPIO_Init(DS_PORT_GPIO_Port, &GPIO_InitStruct);
}
/**
    @Brief :  void DS_Reset(void)
    @Param:   无
    @Function:对温度传感器进行复位
    @Return:  无
    @Author:  自由eπ
  @Note:    无 
*/
void DS_Reset(void){
    Set_DS_Pin_Out();
    HAL_GPIO_WritePin (DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_RESET);//将DS引脚拉低
    delay_us (750);
    HAL_GPIO_WritePin(DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_SET);//将DS引脚拉高
    delay_us (15);
}

/**
    @Brief :  uint8_t Check_DS(void)
    @Param:   无
    @Function:检查温度传感器DS18B20是否存在
    @Return:  0->存在;1->不存在
    @Author:  自由eπ
  @Note:    无 
*/
uint8_t Check_DS(void){
    uint8_t Pulse_Time=0;
    Set_DS_Pin_In();//Set DSPin Into Push Up Input
    //Waitting for the low pulse from DS18B20 and can keep 60-240us 
    while(HAL_GPIO_ReadPin (DS_PORT_GPIO_Port,DS_PORT_Pin)&&Pulse_Time<100){
        Pulse_Time++;
        delay_us (1);
    }
    if( Pulse_Time>=100)
        return 1;
    else
        Pulse_Time= 0;
    //The processing of out time
        while(!HAL_GPIO_ReadPin (DS_PORT_GPIO_Port,DS_PORT_Pin) && Pulse_Time<240){
        Pulse_Time++;
        delay_us (1);
    }
    if( Pulse_Time>=240)
        return 1;
    else
        return 0;
}
/**
    @Brief :  static uint8_t DS_Read_Bit(void)
    @Param:   无
    @Function:从温度传感器中获取一个比特位的数据
    @Return:  0->   ;1->    
    @Author:  自由eπ
  @Note:    无 
*/
static  uint8_t DS_Read_Bit(void){
    uint8_t data;
    Set_DS_Pin_Out();
    HAL_GPIO_WritePin (DS_PORT_GPIO_Port ,DS_PORT_Pin ,GPIO_PIN_RESET);//Set DS pin into reset
    delay_us (10);
    
    Set_DS_Pin_In();
    if(HAL_GPIO_ReadPin (DS_PORT_GPIO_Port ,DS_PORT_Pin) == SET)
        data=1;
    else 
        data=0;
    delay_us(45);
    return data ;
}
/**
    @Brief :  static uint8_t DS_Read_Byte(void)
    @Param:   无
    @Function:从温度传感器中获取一个字节的数据,低位先行
    @Return:  0->   ;1->    
    @Author:  自由eπ
  @Note:    无 
*/
static  uint8_t DS_Read_Byte(void){
    
    uint8_t i,j,data=0;
    for(i=0;i<8;i++){
        j=DS_Read_Bit();
        data=(data)|(j<<i);
    }
    
    return data ;
}
/**
    @Brief :  static  void  Write_DS_Byte(uint8_t data)
    @Param:   uint8_t data
    @Function:向温度传感器中写入一个字节的数据,低位先行
    @Return:  无 
    @Author:  自由eπ
  @Note:    无 
*/
static  void  DS_Write_Byte(uint8_t data){
    uint8_t Test_Bit;
    
    Set_DS_Pin_Out();
    for(Test_Bit=0x01; Test_Bit !=0; Test_Bit<<=1){
        //Write 0
        if((data&Test_Bit)==RESET){
            HAL_GPIO_WritePin (DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_RESET);//Set DS Pin into reset
            delay_us(70);
            HAL_GPIO_WritePin (DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_SET);
            delay_us(2);
        }
        //Write 1
        else {
            HAL_GPIO_WritePin (DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_RESET);//Set DS Pin into reset
            delay_us(9);
            HAL_GPIO_WritePin (DS_PORT_GPIO_Port,DS_PORT_Pin,GPIO_PIN_SET);
            delay_us (55);
        }
    }
}
/**
    @Brief :  uint8_t DS_Init(void)
    @Param:   无
    @Function:对温度传感器进行初始化
    @Return:  无
    @Author:  自由eπ
  @Note:    无
*/
uint8_t DS_Init(void){
    DS_Reset();
    return Check_DS() ;
}
/**
    @Brief :  float Get_DS_Tempture(void)
    @Param:   无
    @Function:获取温度
    @Return:  从DS18B20上读取到的温度值
    @Author:  自由eπ
  @Note:    此获取方法为跳过 ROM 读取,适合于总线上只有一个设备
*/
float Get_DS_Temperature(void){
    
    uint8_t tpmsb,tplsb;
    short s_tem;
    float f_tem;
    
    DS_Reset ();
    Check_DS ();
    DS_Write_Byte (SKIP_ROM );       //Skip ROM
    DS_Write_Byte (CONVERT_TEMP );   //Let transition temperature
    delay_us (350);
    DS_Reset ();
    Check_DS ();
    DS_Write_Byte (SKIP_ROM );       //Skip ROM
    DS_Write_Byte (READ_SCRATCHPAD); //Get temperature from DS18B20
    
    tplsb=DS_Read_Byte();
    tpmsb=DS_Read_Byte();
    
    s_tem = tpmsb<<8;
    s_tem = s_tem | tplsb;
    
    if( s_tem < 0 )        /* 负温度 */
        f_tem = (~s_tem+1) * 0.0625;    
    else
        f_tem = s_tem * 0.0625;
    
    return f_tem; 
}

Gitee仓库地址:https://gitee.com/QC8851/ds18-b20.git

### 使用 CubeIDE 开发 STM32 监测 DS18B20 温度并通过 OLED 显示 #### 一、环境搭建 在使用 CubeIDE 进行开发之前,需要配置好开发环境。安装 STMCubeMXCubeIDE 工具链,并确保能够生成初始化代码。对于本项目使用的 STM32F103C8T6 芯片,可以通过 CubeMX 配置 GPIO 口作为 DS18B20 的单总线接口以及 IIC 接口用于驱动 OLED 屏幕。 #### 二、硬件连接 根据引用中的描述[^2],DS18B20 应该连接到 STM32 的某个 GPIO 引脚上实现单总线通信;OLED 则通过 IIC 协议与 STM32 连接。具体连线如下: - **DS18B20**: 将其数据引脚接到 STM32 的任意一个 GPIO 上(例如 PA0),并加上一个 4.7kΩ 的上拉电阻。 - **OLED**: SDA 和 SCL 分别对应 STM32 的 PB9 (SDA) 和 PB8 (SCL),GND 和 VCC 按照正常供电方式接入。 #### 三、软件设计 ##### 初始化外设 利用 CubeMX 自动生成的 HAL 库函数来初始化必要的外设。主要包括: - **GPIO 初始化**:为 DS18B20 设置单总线模式下的 GPIO 输出/输入切换功能。 - **IIC 初始化**:配置 IIC 总线以便于控制 OLED 显示屏。 以下是部分初始化代码片段: ```c // 初始化 GPIO 和 IIC 外设 void MX_GPIO_Init(void) { __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitTypeDef GPIO_InitStruct = {0}; /* Configure the GPIO pin as output/input */ GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; // Open Drain mode for single-wire communication GPIO_InitStruct.Pull = GPIO_PULLUP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } void MX_I2C1_Init(void) { hi2c1.Instance = I2C1; hi2c1.Init.ClockSpeed = 100000; hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2; hi2c1.Init.OwnAddress1 = 0; hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE; hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE; hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE; if (HAL_I2C_Init(&hi2c1) != HAL_OK) { Error_Handler(); } } ``` ##### 实现温度读取逻辑 基于提供的 `DS18B20_Read_Bit` 函数[^4],扩展成完整的温度读取流程。这通常涉及发送复位脉冲、匹配 ROM 命令、启动转换命令等步骤。 示例代码展示如何调用这些子程序获取最终温度值: ```c float readTemperature() { uint8_t present = 0; int16_t temperature; DS18B20_Reset(); // Reset sensor bus. DS18B20_WriteByte(0xCC); // Skip ROM command to address all devices on a multi-drop line. DS18B20_WriteByte(0x44); // Start conversion. DS18B20_WaitForConversion(); // Wait until conversion is complete. DS18B20_Reset(); DS18B20_WriteByte(0xCC); // Repeat skip ROM sequence before reading scratchpad contents. DS18B20_WriteByte(0xBE); present = DS18B20_ReadByte(); temperature = DS18B20_ReadWord(); return ((temperature * 0.0625)); // Convert raw value into degrees Celsius with resolution of 0.0625°C per LSB. } ``` ##### 数据显示至 OLED 最后一步是在 OLED 屏幕上打印出所测量得到的温度数值。这里采用的是 SSD1306 控制器兼容的小型图形库来进行字符绘制操作[^3]: ```c void displayTemperature(float tempValue) { char buffer[16]; sprintf(buffer, "%.2f C", tempValue); OLED_Clear(); OLED_SetCursor(0, 0); OLED_DisplayString(buffer); OLED_UpdateScreen(); } ``` 整个主循环结构可参照上述方法构建起来,不断刷新最新的温感信息直至断电停止运行为止。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值