一、模块来源
采购链接:
HC-SR04 超声波测距模块 宽电压3-5.5V 工业级 传感器
资料下载链接:
https://pan.baidu.com/s/1sSah9PvLBrmbA7So-6YcSw
资料提取码:qq35
二、规格参数
工作电压:3-5.5V
工作电流:5.3MA
感应角度:小于15度
探测距离:2CM-600CM
探测精度:0.1CM+1%
输出方式: GPIO
管脚数量:4 Pin
以上信息见厂家资料文件
三、移植过程
我们的目标是将例程移植至开发板上【能够判断前方障碍物距离的功能】。首先要获取资料,查看数据手册应如何实现读取数据,再移植至我们的工程。
1、查看资料
只需要在 Trig 管脚(触发信号)输入一个 10US 以上的高电平,系统便可发出 8 个 40KHZ 的超声波脉冲,然后检测回波信号。当检测到回波信号后,通过 Echo 管脚输出。根据 Echo 管脚输出高电平的持续时间可以计算距离值。即距离值为:(高电平时间*340m/s)/2。
当测量距离超过 HC-SR04 的测量范围时,仍会通过 Echo管脚输出高电平的信号,高电平的宽度约为 66ms。如图所示:
测量周期:当接收到 HC-SR04 通过 Echo 管脚输出的高电平脉冲后,便可进行下一次测量,所以测量周期取决于测量距离,当距离被测物体很近时,Echo 返回的脉冲宽度较窄,测量周期 就很短;当距离被测物体比较远时,Echo 返回的脉冲宽度较宽,测量周期也就相应的变长。最坏情况下,被测物体超出超声波模块的测量范围,此时 返回的脉冲宽度最长,约为 66ms,所以最坏情况下的测量周期稍大于 66ms 即可(取 70ms 足够)。
2、引脚选择
3、移植至工程
移植步骤中的导入.c和.h文件与传感器章节的【DHT11温湿度传感器】相同,只是将.c和.h文件更改为bsp_ultrasonic.c与bsp_ultrasonic.h。这里不再过多讲述,移植完成后面修改相关代码。
在文件bsp_ultrasonic.c中,编写如下代码。
/*
* 立创开发板软硬件资料与相关扩展板软硬件资料官网全部开源
* 开发板官网:www.lckfb.com
* 技术支持常驻论坛,任何技术问题欢迎随时交流学习
* 立创论坛:https://oshwhub.com/forum
* 关注bilibili账号:【立创开发板】,掌握我们的最新动态!
* 不靠卖板赚钱,以培养中国工程师为己任
* Change Logs:
* Date Author Notes
* 2024-03-29 LCKFB-LP first version
*/
#include "bsp_ultrasonic.h"
#include "board.h"
unsigned char msHcCount = 0;//ms计数
float distance = 0;
/******************************************************************
* 函 数 名 称:bsp_ultrasonic
* 函 数 说 明:超声波初始化
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:TRIG引脚负责发送超声波脉冲串
******************************************************************/
void Ultrasonic_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_SR04, ENABLE);
RCC_APB1PeriphClockCmd(RCC_TIMER, ENABLE);
GPIO_InitTypeDef GPIO_InitStructure;
// GPIO配置参数
GPIO_InitStructure.GPIO_Pin = GPIO_TRIG;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(PORT_SR04, &GPIO_InitStructure);
GPIO_ResetBits(PORT_SR04,GPIO_TRIG);
GPIO_InitStructure.GPIO_Pin = GPIO_ECHO;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; // 上拉输入
GPIO_Init(PORT_SR04, &GPIO_InitStructure);
GPIO_ResetBits(PORT_SR04,GPIO_ECHO);
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
TIM_DeInit(PORT_TIMER);
/* 配置定时器参数 */
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseStructure.TIM_Period = 1000 - 1; // 1ms进入一次中断
TIM_TimeBaseStructure.TIM_Prescaler = 72 - 1; // TIMER3时钟频率72MHz
TIM_TimeBaseInit(PORT_TIMER, &TIM_TimeBaseStructure);
TIM_SetCounter(PORT_TIMER, 0); // 清除定时器计数
TIM_ClearFlag(PORT_TIMER, TIM_FLAG_Update); // 清除中断,避免一初始化立即产生中断
TIM_ITConfig(PORT_TIMER, TIM_IT_Update, ENABLE); // 打开定时器中断
NVIC_InitTypeDef NVIC_InitStructure;
// 配置中断分组
NVIC_InitStructure.NVIC_IRQChannel = TIMER_IRQ;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x03;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02;
NVIC_Init(&NVIC_InitStructure);
TIM_Cmd(PORT_TIMER, ENABLE); // 使能定时器
}
/******************************************************************
* 函 数 名 称:Open_Timer
* 函 数 说 明:打开定时器
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:
******************************************************************/
void Open_Timer(void)
{
TIM_SetCounter(PORT_TIMER, 0); // 清除定时器计数
msHcCount = 0;
TIM_Cmd(PORT_TIMER, ENABLE); // 使能定时器
}
/******************************************************************
* 函 数 名 称:Get_TIMER_Count
* 函 数 说 明:获取定时器定时时间
* 函 数 形 参:无
* 函 数 返 回:数据
* 作 者:LC
* 备 注:
******************************************************************/
uint32_t Get_TIMER_Count(void)
{
uint32_t time = 0;
time = msHcCount*1000; // 得到us
time += TIM_GetCounter(PORT_TIMER); // 得到ms
TIM_SetCounter(PORT_TIMER, 0); // 清除定时器计数
delay_ms(10);
return time ;
}
/******************************************************************
* 函 数 名 称:Close_Timer
* 函 数 说 明:关闭定时器
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:
******************************************************************/
void Close_Timer(void)
{
TIM_Cmd(PORT_TIMER, DISABLE); // 关闭定时器
}
/******************************************************************
* 函 数 名 称:TIMER_IRQHandler
* 函 数 说 明:定时器中断服务函数
* 函 数 形 参:无
* 函 数 返 回:无
* 作 者:LC
* 备 注:1ms进入一次
******************************************************************/
void TIMER_IRQHandler(void)
{
if (TIM_GetITStatus(PORT_TIMER, TIM_IT_Update) == SET) // 检查定时器中断发生
{
msHcCount++;
TIM_ClearITPendingBit(PORT_TIMER, TIM_IT_Update); // 清除中断标志
}
}
/******************************************************************
* 函 数 名 称:Hcsr04GetLength
* 函 数 说 明:获取测量距离
* 函 数 形 参:无
* 函 数 返 回:测量距离
* 作 者:LC
* 备 注:无
******************************************************************/
float Hcsr04GetLength(void)
{
/*测5次数据计算一次平均值*/
float length = 0;
float t = 0;
float sum = 0;
unsigned int i = 0;
while(i != 10)
{
SR04_TRIG(1);//trig拉高信号,发出高电平
delay_1us(20);//持续时间超过10us
SR04_TRIG(0);//trig拉低信号,发出低电平
/*Echo发出信号 等待回响信号*/
/*输入方波后,模块会自动发射8个40KHz的声波,与此同时回波引脚(echo)端的电平会由0变为1;
(此时应该启动定时器计时);当超声波返回被模块接收到时,回波引 脚端的电平会由1变为0;
(此时应该停止定时器计数),定时器记下的这个时间即为
超声波由发射到返回的总时长;*/
while(SR04_ECHO() == Bit_RESET);//echo等待回响
Open_Timer(); //打开定时器
i++;
while(SR04_ECHO() == Bit_SET);
Close_Timer(); // 关闭定时器
t = Get_TIMER_Count(); // 获取时间,分辨率为1us
length = (float)t / 58.0f; // cm
sum += length;
}
length = sum/10;//五次平均值
distance = length;
return length;
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
在文件bsp_ultrasonic.h中,编写如下代码。
/*
* 立创开发板软硬件资料与相关扩展板软硬件资料官网全部开源
* 开发板官网:www.lckfb.com
* 技术支持常驻论坛,任何技术问题欢迎随时交流学习
* 立创论坛:https://oshwhub.com/forum
* 关注bilibili账号:【立创开发板】,掌握我们的最新动态!
* 不靠卖板赚钱,以培养中国工程师为己任
* Change Logs:
* Date Author Notes
* 2024-03-29 LCKFB-LP first version
*/
#ifndef _BSP_ULTRASONIC_H_
#define _BSP_ULTRASONIC_H_
#include "stm32f10x.h"
#define RCC_SR04 RCC_APB2Periph_GPIOA
#define PORT_SR04 GPIOA
#define GPIO_TRIG GPIO_Pin_1
#define GPIO_ECHO GPIO_Pin_2
#define RCC_TIMER RCC_APB1Periph_TIM3
#define PORT_TIMER TIM3
#define TIMER_IRQ TIM3_IRQn
#define TIMER_IRQHandler TIM3_IRQHandler
#define SR04_TRIG(x) GPIO_WriteBit( PORT_SR04, GPIO_TRIG, x?Bit_SET:Bit_RESET)
#define SR04_ECHO() GPIO_ReadInputDataBit( PORT_SR04, GPIO_ECHO )
void Ultrasonic_Init(void);//超声波初始化
float Hcsr04GetLength(void );//获取超声波测距的距离
#endif
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
四、移植验证
在main.c中输入代码如下:
/*
* 立创开发板软硬件资料与相关扩展板软硬件资料官网全部开源
* 开发板官网:www.lckfb.com
* 技术支持常驻论坛,任何技术问题欢迎随时交流学习
* 立创论坛:https://oshwhub.com/forum
* 关注bilibili账号:【立创开发板】,掌握我们的最新动态!
* 不靠卖板赚钱,以培养中国工程师为己任
* Change Logs:
* Date Author Notes
* 2024-03-29 LCKFB-LP first version
*/
#include "stm32f10x.h"
#include "board.h"
#include "bsp_uart.h"
#include "stdio.h"
#include "bsp_ultrasonic.h"
int main(void)
{
board_init();
uart1_init(115200);
Ultrasonic_Init();
printf("Start.......\r\n");
while(1)
{
printf((const char *)"距离为 = %.2f\r\n",Hcsr04GetLength() );
delay_ms(1000);
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
上电效果:
移植成功案例代码下载链接: