超声波测距模块HC-SR04提供2cm-400cm非接触测量功能,测距精度可达3mm。这些模块包括超声波发射器、接收器和控制电路。工作的基本原理:
虽然所有这些听起来很简单,但由于涉及到精确计时,我没有找到很多关于在 Zephyr 上实现它的最佳方法。所以我决定自己做一个。
#include
#include
#include
#include
#include
#include
#include
#define GPIO_OUT_PIN 1
#define GPIO_INT_PIN 3
#define PORT "GPIOA"
void main(void)
{
uint32_t cycles_spent;
uint32_t nanseconds_spent;
uint32_t val;
uint32_t cm;
uint32_t stop_time;
uint32_t start_time;
struct device *dev;
dev = device_get_binding(PORT);
gpio_pin_configure(dev, GPIO_OUT_PIN, GPIO_DIR_OUT);
gpio_pin_configure(dev, GPIO_INT_PIN, (GPIO_DIR_IN | GPIO_INT_EDGE| GPIO_INT_ACTIVE_HIGH | GPIO_INT_DEBOUNCE));
while (1) {
gpio_pin_write(dev, GPIO_OUT_PIN, 1);
k_sleep(K_MSEC(10));
gpio_pin_write(dev, GPIO_OUT_PIN, 0);
do {
gpio_pin_read(dev, GPIO_INT_PIN, &val);
} while (val == 0);
start_time = k_cycle_get_32();
do {
gpio_pin_read(dev, GPIO_INT_PIN, &val);
stop_time = k_cycle_get_32();
cycles_spent = stop_time - start_time;
if (cycles_spent > 1266720) //260cm for 84MHz (((MAX_RANGE * 58000) / 1000000000) * (CLOCK * 1000000))
{
break;
}
} while (val == 1);
nanseconds_spent = SYS_CLOCK_HW_CYCLES_TO_NS(cycles_spent);
cm = nanseconds_spent / 58000;
printk("%d\n", cm);
k_sleep(100);
}
}
头文件
#include
#include
#include
#include
#include
#include
#include
管脚定义
#define GPIO_OUT_PIN 1
#define GPIO_INT_PIN 3
#define PORT "GPIOA"
变量初始化
void main(void)
{
uint32_t cycles_spent;
uint32_t nanseconds_spent;
uint32_t val;
uint32_t cm;
uint32_t stop_time;
uint32_t start_time;
初始化引脚并设置 IO 方向
struct device *dev;
dev = device_get_binding(PORT);
gpio_pin_configure(dev, GPIO_OUT_PIN, GPIO_DIR_OUT);
gpio_pin_configure(dev, GPIO_INT_PIN, (GPIO_DIR_IN | GPIO_INT_EDGE| GPIO_INT_ACTIVE_HIGH | GPIO_INT_DEBOUNCE));
首先,我们将触发引脚设置为高电平 10 微秒
gpio_pin_write(dev, GPIO_OUT_PIN, 1);
k_sleep(K_MSEC(10));
gpio_pin_write(dev, GPIO_OUT_PIN, 0);
接下来我们的第一个 do while 循环运行直到 Echo Pin 为低电平,并在它变高时立即中断,然后我们开始计算周期
do {
gpio_pin_read(dev, GPIO_INT_PIN, &val);
} while (val == 0);
start_time = k_cycle_get_32();
我们的下一个 do while 循环运行直到 ECHO 引脚保持高电平:在这个 do while 块中,我们计算花费的周期数并不断检查if
用于限制最大范围的值。我们使用公式 (((MAX_RANGE * 58000) / 1000000000) * (CLOCK * 1000000)) 并将 MAX_RANGE 替换为以厘米为单位的所需值,并将 CLOCK 替换为微处理器的时钟速度。
do {
gpio_pin_read(dev, GPIO_INT_PIN, &val);
stop_time = k_cycle_get_32();
cycles_spent = stop_time - start_time;
if (cycles_spent > 1266720) //260cm for 84MHz (((MAX_RANGE * 58000) / 1000000000) * (CLOCK * 1000000))
{
break;
}
} while (val == 1);
最后我们使用该SYS_CLOCK_HW_CYCLES_TO_NS
函数计算 ECHO 引脚为高电平时的持续时间,它以纳秒为单位返回结果。然后将值乘以 58000 得到以厘米为单位的结果,请记住空气中的声速是 340 米/秒
nanseconds_spent = SYS_CLOCK_HW_CYCLES_TO_NS(cycles_spent);
cm = nanseconds_spent / 58000;
printk("%d\n", cm);
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
全部0条评论
快来发表一下你的评论吧 !