Netseye

  • 2024-12-28
  • 发表了主题帖: 【Follow me第二季第4期】任务汇总

    本帖最后由 Netseye 于 2025-1-11 23:01 编辑 【Follow me第二季第4期】任务汇总 本次一共选购了三件商品.为了方便自己增加额外的一个编码器模块和一个speaker模块 如图: Arduino_Nano_Connect板子开箱 https://bbs.eeworld.com.cn/thread-1300050-1-1.html 任务一:搭建环境并开启第一步Blink三色LEDand串口打印 https://bbs.eeworld.com.cn/thread-1301096-1-1.html 任务二 学习IMU基础知识,通过串口打印六轴原始数据 https://bbs.eeworld.com.cn/thread-1301100-1-1.html 任务三 学调试PDM麦克风,通过串口打印收音数据和音频波形 https://bbs.eeworld.com.cn/thread-1301103-1-1.html 任务四 RGB LED亮度显示PDM麦克风收到的声音大小 https://bbs.eeworld.com.cn/thread-1301110-1-1.html     总结帖是在之前的基础上吧所有代码整合到一个项目中.并增加了一些oled的展示和任务五的展示. 项目流程图:     数据采集流程图   使用mlc   图片展示   任务四:通过IMU数据结合机器学习算法,识别运动状态,并通过串口打印。 做了一些大概的了解和尝试做一下总结: 首先介绍一下LSM6DSOX 传感器自带机器学习核心(MLC,Machine Learning Core)。这是由意法半导体(STMicroelectronics)开发的一项创新功能,使 LSM6DSOX 能够通过机器学习算法对传感器数据进行分类和模式识别。(好像STM家所有以X 结尾的传感器都会带MLC) 实现方法: 大致的实现方案就是可以基于LSM6DSOX (MLC) 或者MCU Tinyml 实现,整体实现思路是完全一致的只是.使用的工具不同 开发流程: 数据采集: 使用 LSM6DSOX 采集运动数据。 模型训练: 借助 ST 提供的 Unico GUI 工具或者Edge Impulse,使用采集的数据进行模型训练。 模型导出: 将训练好的模型转化下载到项目中 传感器部署: 将模型上传到 LSM6DSOX,并通过寄存器配置启用 MLC。或者使用MCU 中. 运行与分类: 传感器在运行时实时分类并输出结果。   工作的大头是在数据采集上.由于我比较懒提供下方案和代码就不做对应的实践了. LSM6DSOX (MLC)  参考: https://github.com/STMicroelectronics/STMems_Machine_Learning_Core/tree/master/configuration_examples 数据采集格式 C++ A_X [mg]          A_Y [mg]          A_Z [mg]         G_X [dps]          G_Y [dps]          G_Z [dps] 80        931        -354        0.665        -5.075        -5.01375 65        931        -352        -0.02625        -7.72625        -8.575 4        949        -344        -1.40875        -12.8713        -11.515 50        960        -343        -4.66375        -23.8613        -15.54 174        933        -341        -7.81375        -25.025        -11.7075 实现(按下按钮进行动作录制) C++ #include <Arduino_LSM6DSOX.h> // IMU 库 #include <OneButton.h> const int PIN_INPUT = D6;     // D6 引脚连接按钮 const int ledPin = LED_BUILTIN; // 板载LED OneButton button; float Ax, Ay, Az; float Gx, Gy, Gz; bool isRecording = false; unsigned long startTime = 0; void setup() {   Serial.begin(115200);   while (!Serial);     if (!IMU.begin()) {     Serial.println("Failed to initialize IMU!");     while (1);   }   button.setup(PIN_INPUT, INPUT_PULLUP);   button.attachPress(singleClick); // 设置单击事件 } void loop() {   button.tick(); // 检测按钮事件   if (isRecording) {     if (IMU.accelerationAvailable() && IMU.gyroscopeAvailable()) {       IMU.readAcceleration(Ax, Ay, Az);       IMU.readGyroscope(Gx, Gy, Gz);       Serial.print(Ax * 1000.0f);       Serial.print("\t");       Serial.print(Ay * 1000.0f);       Serial.print("\t");       Serial.print(Az * 1000.0f);       Serial.print("\t");       Serial.print(Gx);       Serial.print("\t");       Serial.print(Gy);       Serial.print("\t");       Serial.println(Gz);     }   } } void singleClick() {   isRecording = !isRecording;   digitalWrite(ledPin, isRecording);     if (isRecording) {         Serial.println("开始录制...");         startTime = millis();         Serial.println("A_X [mg]\tA_Y [mg]\tA_Z [mg]\tG_X [dps]\tG_Y [dps]\tG_Z [dps]");       } else {         Serial.println("停止录制...");         Serial.print("录制时长:");         Serial.print(millis()-startTime);         Serial.println("ms");       } } 使用 Edge Impulse实现 参考 https://wiki.seeedstudio.com/cn/XIAO-RP2040-EI/ 创建Edge Impulse账号项目和安装Edge Impulse cli我就不再赘述了. 数据格式可以完全使用上面的代码输出格式.可以使用同一份数据 执行采集命令 C++ edge-impulse-data-forwarder --clean                                                                                                                                                                                                 ⏎ Edge Impulse data forwarder v1.30.1 ? What is your user name or e-mail address (edgeimpulse.com)? netseye@gmail.com ? What is your password? [hidden] Endpoints:     Websocket: wss://remote-mgmt.edgeimpulse.com     API:       https://studio.edgeimpulse.com     Ingestion: https://ingestion.edgeimpulse.com [SER] Connecting to /dev/tty.usbmodem11101 [SER] Serial is connected (50:24:B0:93:82:61:7D:18) [WS ] Connecting to wss://remote-mgmt.edgeimpulse.com [WS ] Connected to wss://remote-mgmt.edgeimpulse.com [SER] Detecting data frequency... [SER] Detected data frequency: 10Hz ? 6 sensor axes detected (example values: [-0.04,-0.15,0.99,0.12,0.06,-0.37]). What do you want to call them? Separate the names with ',':  A_X,A_Y,A_Z,G_X,G_Y,G_Z ? What name do you want to give this device? Nano RP2040 Connect [WS ] Device "Nano RP2040 Connect" is now connected to project "nano rp2040". To connect to another project, run `edge-impulse-data-forwarder --clean`. 训练: STMems_Machine_Learning_Core 中看起来有4种可以使用的方式:Unico GUI   Weka MATLAB Scikit-learn (Python) Edge Impulse:直接在web平台上进行训练:   例子体验: 本次直接使用了Head gestures 相关代码进行演示,只需要下载lsm6dsox_head_gestures.h 带自己项目中. C++ #include "../../bsp/config.h" #include "../../bsp/display.hpp" #include "../App.h" #include "LSM6DSOXSensor.h" // #include "lsm6dsox_activity_recognition_for_mobile.h" #include "lsm6dsox_head_gestures.h" // 活动状态映射表 const char *activityNames[] = {     "点头", // 0     "摇动", // 1     "静止", // 2     "摆动", // 3     "步行"  // 4 }; void drawAccelBar(LGFX_Sprite *canvas, const char *label, float accel,                   int yPos) {   int centerX = 64;   int barMaxWidth = 60;   int barHeight = 10;   int barY = yPos + 10;   // 将加速度值映射到 [-barMaxWidth, barMaxWidth]   // 使用浮点数计算,最后再转换为整数   float barWidthFloat = accel * barMaxWidth;   int barWidth = constrain(int(barWidthFloat), -barMaxWidth, barMaxWidth);   canvas->fillRect(centerX - barMaxWidth - 1, barY - 1, barMaxWidth * 2 + 2,                    barHeight + 2, TFT_WHITE); // 清除区域略微扩大   // 绘制中心线(在清除之后绘制,确保可见)   canvas->drawLine(centerX, barY, centerX, barY + barHeight,                    TFT_BLACK); // 使用白色绘制中心线   if (barWidth < 0) {     canvas->fillRect(centerX + barWidth, barY, abs(barWidth), barHeight,                      TFT_BLACK);   } else if (barWidth > 0) {     canvas->fillRect(centerX, barY, barWidth, barHeight, TFT_BLACK);   } } // Interrupts. volatile int mems_event = 0; // Components LSM6DSOXSensor AccGyr(&Wire, LSM6DSOX_I2C_ADD_L); // MLC ucf_line_t *ProgramPointer; int32_t LineCounter; int32_t TotalNumberOfLine; void INT1Event_cb(); void printMLCStatus(uint8_t status); void App::_imu_mlc_init() {   uint8_t mlc_out[8];   // Led.   pinMode(LED_BUILTIN, OUTPUT);   // Force INT1 of LSM6DSOX low in order to enable I2C   pinMode(INT_IMU, OUTPUT);   digitalWrite(INT_IMU, LOW);   delay(200);   // Initialize I2C bus.   Wire.begin();   AccGyr.begin();   AccGyr.Enable_X();   AccGyr.Enable_G();   /* Feed the program to Machine Learning Core */   /* Activity Recognition Default program */   ProgramPointer = (ucf_line_t *)lsm6dsox_head_gestures;   TotalNumberOfLine = sizeof(lsm6dsox_head_gestures) / sizeof(ucf_line_t);   Serial.println("Activity Recognition for LSM6DSOX MLC");   Serial.print("UCF Number Line=");   Serial.println(TotalNumberOfLine);   for (LineCounter = 0; LineCounter < TotalNumberOfLine; LineCounter++) {     if (AccGyr.Write_Reg(ProgramPointer[LineCounter].address,                          ProgramPointer[LineCounter].data)) {       Serial.print("Error loading the Program to LSM6DSOX at line: ");       Serial.println(LineCounter);       while (1) {         // Led blinking.         digitalWrite(LED_BUILTIN, HIGH);         delay(250);         digitalWrite(LED_BUILTIN, LOW);         delay(250);       }     }   }   Serial.println("Program loaded inside the LSM6DSOX MLC");   // Interrupts.   pinMode(INT_IMU, INPUT);   attachInterrupt(INT_IMU, INT1Event_cb, RISING);   /* We need to wait for a time window before having the first MLC status */   delay(3000);   AccGyr.Get_MLC_Output(mlc_out);   Serial.println("MLC Status: ");   if (mlc_out[0] <= 4) {     const char *activity = activityNames[mlc_out[0]];     Serial.print("Activity: ");     Serial.println(activity);   } } void App::_imu_mlc_show() {   _canvas->setFont(&fonts::efontCN_14);   while (1) {     _canvas->fillScreen(TFT_WHITE);     _canvas->setTextColor(TFT_BLACK);     _canvas->drawString("-IMU Motion", 8, 0);     if (mems_event) {       mems_event = 0;       LSM6DSOX_MLC_Status_t status;       AccGyr.Get_MLC_Status(&status);       Serial.println("MLC Status: ");       if (status.is_mlc1) {         uint8_t mlc_out[8];         AccGyr.Get_MLC_Output(mlc_out);         Serial.print("MLC1: ");         // Serial.println(mlc_out[0]);         const char *activity =             (mlc_out[0] <= 4) ? activityNames[mlc_out[0]] : "Unknown";         Serial.println(activity);         _canvas->drawCenterString(String("Activity: ") + activity,                                   _canvas->width() / 2, 30);         _canvas_update();       }     }     if (_check_next())       break;   } } void INT1Event_cb() { mems_event = 1; } void App::_imu_x_show() {   _canvas->setFont(&fonts::efontCN_14);   while (1) {     _canvas->fillScreen(TFT_WHITE);     _canvas->setTextColor(TFT_BLACK);     _canvas->drawString("-IMU", 8, 0);     float ax, ay, az; // 加速度     float gx, gy, gz; // 角速度     // 读取加速度值     uint8_t acceleroStatus;     AccGyr.Get_X_DRDY_Status(&acceleroStatus);     if (acceleroStatus == 1) { // 新数据可用       int32_t acceleration[3];       AccGyr.Get_X_Axes(acceleration);       // 将 mg 转换为 m/s²(1g ≈ 9.80665 m/s², 1000mg = 1g)       ax = acceleration[0] * 9.80665f / 1000.0f;       ay = acceleration[1] * 9.80665f / 1000.0f;       az = acceleration[2] * 9.80665f / 1000.0f;       Serial.print("加速度 (m/s^2): X=");       Serial.print(ax);       Serial.print(" Y=");       Serial.print(ay);       Serial.print(" Z=");       Serial.println(az);       // 绘制加速度条形图       drawAccelBar(_canvas, "Accel X:", ax, 10);       drawAccelBar(_canvas, "Accel Y:", ay, 25);       drawAccelBar(_canvas, "Accel Z:", az, 40);     }     // 读取角速度值     uint8_t gyroStatus;     AccGyr.Get_G_DRDY_Status(&gyroStatus);     if (gyroStatus == 1) { // 新数据可用       int32_t rotation[3];       AccGyr.Get_G_Axes(rotation);       // 将 mdps (milli degrees per second) 转换为 rad/s       // 1度 = π / 180 弧度,1000 mdps = 1 dps       gx = rotation[0] * (PI / 180.0f) / 1000.0f;       gy = rotation[1] * (PI / 180.0f) / 1000.0f;       gz = rotation[2] * (PI / 180.0f) / 1000.0f;       Serial.print("陀螺仪 (rad/s): X=");       Serial.print(gx);       Serial.print(" Y=");       Serial.print(gy);       Serial.print(" Z=");       Serial.println(gz);     }     _canvas_update();     if (_check_next())       break;   } } 由于本次吧所有任务集成的一个项目中,并做了相关的菜单切换和展示.所以代码也做了一些调整 之前的任务中的imu直接使用的#include <Arduino_LSM6DSOX.h> 这次直接为了使用mlc试了的STM的库 看起来rp2040在aruduino环境中没有rtos可用所有blink这个任务也做了相关代码的调整.好像有一个TimeAlarms 库可以做一些异步处理. 不过我选择了直接用计时的方式来实现[localvideo]1a60b39f9f4edf6e5f0a1a32f6ca794b[/localvideo] 心得体会:        感谢得捷和电子工程世界提供了一次学习机会.不进体验了aruduino和树莓派rp2040的强大生态还接触到了自带mlc的传感器.希望follow me活动越办越好.

  • 2024-12-26
  • 加入了学习《 【Follow me第二季第4期】任务汇总》,观看 FM4

  • 2024-12-11
  • 加入了学习《followme3——基于lvgl多功能摆件的设计》,观看 followme3完整介绍内容

  • 2024-12-08
  • 回复了主题帖: 【Follow me第二季第4期】任务四 RGB LED亮度显示PDM麦克风收到的声音大小

    想着视频总结的时候发一下好了.

  • 回复了主题帖: 【Follow me第二季第4期】任务一 搭建环境&Blink三色LED / 串口打印

    我视频上不是演示了么.

  • 2024-12-06
  • 加入了学习《【Follow me第二季第4期】任务汇报》,观看 【Follow me第二季第4期】任务汇报

  • 发表了主题帖: 【Follow me第二季第4期】任务四 RGB LED亮度显示PDM麦克风收到的声音大小

    本帖最后由 Netseye 于 2024-12-6 16:11 编辑 要在 Arduino Nano RP2040 Connect 上通过 RGB LED 显示 PDM 麦克风 收到的声音大小,可以通过读取麦克风的音量数据,并根据音量调整 RGB LED 的颜色和亮度。 主要步骤: 读取 PDM 麦克风音频数据。 计算音频信号的强度(音量)。 调整 RGB LED 的颜色和亮度,根据音量大小显示不同的效果。 实现 计算声音的大小(音量或振幅)通常需要对采样数据进行处理。在数字音频处理中,声音的大小可以通过计算振幅的平均值或最大值来近似表示。以下是几种常见的计算方法: 1. 峰值振幅法(Peak Amplitude) 计算采样数据中的最大或最小振幅,取绝对值的最大值。 int16_t getMaxAmplitude(short *buffer, int length) { int16_t maxAmplitude = 0; for (int i = 0; i < length; i++) { int16_t amplitude = abs(buffer[i]); if (amplitude > maxAmplitude) { maxAmplitude = amplitude; } } return maxAmplitude; // 返回最大振幅 } 优点:简单、直观。 缺点:容易受短时间内的尖锐噪声影响。 2. 均方根法(RMS,Root Mean Square) RMS 是衡量声音能量的常用方法,它计算采样值的平方平均再开方,能更稳定地反映声音的大小。 float getRMSAmplitude(short *buffer, int length) { long sum = 0; for (int i = 0; i < length; i++) { sum += (long)buffer[i] * buffer[i]; // 计算平方和 } float rms = sqrt(sum / (float)length); // 计算平方平均值并开方 return rms; } 优点:更稳定,能更准确地反映整体音量。 缺点:计算稍微复杂一些。 3. 平均绝对振幅法(Mean Absolute Amplitude) 计算所有采样值的绝对值平均。 float getAverageAmplitude(short *buffer, int length) { long sum = 0; for (int i = 0; i < length; i++) { sum += abs(buffer[i]); // 计算绝对值和 } return sum / (float)length; // 返回平均值 } 优点:简单且能消除正负波形的抵消影响。 缺点:没有 RMS 稳定,但比峰值法好。   能够更灵敏一些我们采用峰值振幅法(Peak Amplitude) 来计算音量   您可以根据 音量 值的大小划分不同的等级。例如,设置 4 个等级: 低音量:音量 值在 0 到 300 范围内 中低音量:音量 值在 300 到 500 范围内 中高音量:音量 值在 500 到 3000 范围内 高音量:音量 值大于 3000 将 音量 值映射到 LED 亮度或颜色:   根据音量等级,设置 LED 的颜色或亮度。例如: 低音量:红色 LED 亮起 中低音量:绿色 LED 亮起 中高音量:蓝色 LED 亮起 高音量:RGB 全亮 代码示例: 以下代码实现了通过 RGB LED 显示声音的强度,根据声音的大小变化调整 LED 的亮度和颜色。 #include <PDM.h> #include <WiFiNINA.h> // 默认输出通道数 static const char channels = 1; // 默认 PCM 输出频率 static const int frequency = 20000; // 用于读取样本的缓冲区,每个样本是 16 位 short sampleBuffer[512]; // 已读取的音频样本数量 volatile int samplesRead; void setup() { Serial.begin(9600); // 初始化串口通信 while (!Serial); // 等待串口连接 pinMode(LEDR, OUTPUT); // 设置红色 LED 引脚为输出 pinMode(LEDG, OUTPUT); // 设置绿色 LED 引脚为输出 pinMode(LEDB, OUTPUT); // 设置蓝色 LED 引脚为输出 // 配置数据接收回调函数 PDM.onReceive(onPDMdata); // 初始化 PDM 麦克风: // - 一个通道(单声道模式) // - 20 kHz 采样率 if (!PDM.begin(channels, frequency)) { Serial.println("无法启动 PDM!"); while (1); // 启动失败时停止程序 } } void loop() { // 等待读取样本 if (samplesRead) { float rms = getMaxAmplitude(sampleBuffer, samplesRead); controlLEDs(rms); Serial.println(rms); // 清空已读取的样本数量 samplesRead = 0; } } void onPDMdata() { // 查询可用字节数 int bytesAvailable = PDM.available(); // 从 PDM 麦克风读取数据到样本缓冲区 PDM.read(sampleBuffer, bytesAvailable); // 16 位样本,每个样本占 2 字节 samplesRead = bytesAvailable / 2; } int16_t getMaxAmplitude(short *buffer, int length) { int16_t maxAmplitude = 0; for (int i = 0; i < length; i++) { int16_t amplitude = abs(buffer[i]); if (amplitude > maxAmplitude) { maxAmplitude = amplitude; } } return maxAmplitude; // 返回最大振幅 } void controlLEDs(float rms) { // 按 RMS 值进行分级 if (rms < 300) { digitalWrite(LEDR, HIGH); digitalWrite(LEDG, LOW); digitalWrite(LEDB, LOW); } else if (rms < 500) { digitalWrite(LEDR, LOW); digitalWrite(LEDG, HIGH); digitalWrite(LEDB, LOW); } else if (rms < 3000) { digitalWrite(LEDR, LOW); digitalWrite(LEDG, LOW); digitalWrite(LEDB, HIGH); } else { digitalWrite(LEDR, HIGH); digitalWrite(LEDG, HIGH); digitalWrite(LEDB, HIGH); } } 代码说明: 初始化 PDM 麦克风: PDM.begin(1, 16000) 启动单声道,采样率为 16kHz。 设置回调函数 PDM.onReceive(onPDMdata) 处理采样数据。 计算最大振幅: getMaxAmplitude() 函数计算采样数据中的最大振幅,反映音频信号的强度。 RGB LED 控制: setLEDColor() 函数将音量值映射到 LED 的亮度。map() 函数将音量值(0 到 255)映射到 RGB LED 的亮度值。 使用红色和绿色 LED 显示不同的颜色,蓝色 LED 不随音量变化。 音量和颜色映射: 随着音量的增大,红色和绿色 LED 的亮度增亮,从而实现声音大小与 RGB LED 亮度和颜色的动态显示。 调试: 上传代码:将代码上传到 Arduino Nano RP2040 Connect 板子。 串口监视器:打开串口监视器查看音量数据。 LED 反馈:观察 RGB LED 的颜色和亮度变化,根据麦克风拾取到的音频信号显示不同的颜色和亮度。  

  • 发表了主题帖: 【Follow me第二季第4期】任务三 学调试PDM麦克风,通过串口打印收音数据和音频波形。

    Arduino Nano RP2040 Connect 板载了 MP34DT06 PDM 数字麦克风,它是一种基于脉冲密度调制(PDM)的麦克风。通过采样 PDM 信号可以获取音频数据,适用于语音识别、音频采集等场景。 PDM 麦克风基础知识 PDM(Pulse Density Modulation)是一种以高频率表示声音强度的调制方式。 相比于传统的模拟麦克风,PDM 麦克风输出的是数字信号,适合直接处理数字音频。 2. 示例代码 下面的代码演示了如何初始化 PDM 麦克风并通过串口打印音频采样数据: C++ #include <PDM.h> // 默认输出通道数 static const char channels = 1; // 默认 PCM 输出频率 static const int frequency = 16000; // 用于读取样本的缓冲区,每个样本是 16 位 short sampleBuffer[512]; // 已读取的音频样本数量 volatile int samplesRead; void setup() { Serial.begin(9600); // 初始化串口通信 while (!Serial); // 等待串口连接 // 配置数据接收回调函数 PDM.onReceive(onPDMdata); // 初始化 PDM 麦克风: // - 一个通道(单声道模式) // - 20 kHz 采样率 if (!PDM.begin(channels, frequency)) { Serial.println("无法启动 PDM!"); while (1); // 启动失败时停止程序 } } void loop() { // 等待读取样本 if (samplesRead) { // 将样本数据打印到串口监视器或绘图工具中 for (int i = 0; i < samplesRead; i++) { Serial.println(sampleBuffer); // 打印当前样本数据 } // 清空已读取的样本数量 samplesRead = 0; } } void onPDMdata() { // 查询可用字节数 int bytesAvailable = PDM.available(); // 从 PDM 麦克风读取数据到样本缓冲区 PDM.read(sampleBuffer, bytesAvailable); // 16 位样本,每个样本占 2 字节 samplesRead = bytesAvailable / 2; } 3. 代码说明 PDM.begin(1, 16000):启动 PDM 麦克风,单声道,采样率为 16kHz。 PDM.onReceive(onPDMdata):注册一个回调函数,当有数据时调用。 sampleBuffer[]:用于存储音频样本数据。 Serial.println(sampleBuffer):通过串口打印采样数据。 4. 查看波形数据 将代码上传到 Arduino Nano RP2040 Connect。 打开 串口监视器,设置波特率为 9600。 使用 Arduino IDE 串口绘图器(Serial Plotter),观察音频数据的波形变化。 5. 调试建议 调整采样率:通过修改 PDM.begin() 的第二个参数来提高或降低采样率。 滤波处理:可在 loop() 中增加滤波或降噪处理,平滑音频波形。

  • 发表了主题帖: 【Follow me第二季第4期】任务二 学习IMU基础知识,通过串口打印六轴原始数据

    Arduino Nano RP2040 Connect 板子上集成了 LSM6DSOX IMU 传感器,它可以测量三轴加速度和三轴角速度。以下是学习和调试 LSM6DSOX 的详细步骤。 硬件概述: LSM6DSOX 是一款 6 轴 IMU 传感器,包含: 三轴加速度计(单位:m/s²) 三轴陀螺仪(单位:°/s 或 rad/s) Accelerometer Gyroscope 2. 安装所需库: 打开 Arduino IDE。 进入 工具 > 库管理器。 搜索并安装 Arduino_LSM6DSOX 库。 3. 示例代码: 下面是一个简单的示例,演示如何读取加速度和陀螺仪数据,并通过串口打印输出: C++ #include <Arduino_LSM6DSOX.h> void setup() { Serial.begin(115200); // 初始化串口 while (!Serial); // 等待串口连接 if (!IMU.begin()) { Serial.println("无法初始化 LSM6DSOX IMU 传感器!"); while (1); } Serial.println("LSM6DSOX IMU 传感器已初始化"); } void loop() { float ax, ay, az; // 加速度 float gx, gy, gz; // 角速度 // 读取加速度值 if (IMU.accelerationAvailable()) { IMU.readAcceleration(ax, ay, az); Serial.print("加速度 (m/s^2): X="); Serial.print(ax); Serial.print(" Y="); Serial.print(ay); Serial.print(" Z="); Serial.println(az); } // 读取角速度值 if (IMU.gyroscopeAvailable()) { IMU.readGyroscope(gx, gy, gz); Serial.print("陀螺仪 (rad/s): X="); Serial.print(gx); Serial.print(" Y="); Serial.print(gy); Serial.print(" Z="); Serial.println(gz); } delay(500); // 延迟 500 毫秒 } 代码说明: IMU.begin():初始化 IMU 传感器。 IMU.readAcceleration():读取 X、Y、Z 三轴加速度值。 IMU.readGyroscope():读取 X、Y、Z 三轴角速度值。 5. 上传并查看数据: 将代码上传到 Arduino Nano RP2040 Connect。 打开 串口监视器,设置波特率为 9600。 观察串口输出的六轴原始数据。 6. 调试建议: 如果数据不稳定,可以调整延迟时间或尝试增加滤波算法。 使用滤波技术(如卡尔曼滤波)来平滑数据。

  • 发表了主题帖: 【Follow me第二季第4期】任务一 搭建环境&Blink三色LED / 串口打印

    本帖最后由 Netseye 于 2024-12-6 15:34 编辑   搭建开发环境: 下载并安装 Arduino IDE(确保使用最新版本)。 打开 Arduino IDE,进入 工具 > 板子 > 开发板管理器,搜索并安装 Arduino Mbed OS RP2040 或 Arduino Nano RP2040 Connect 的支持包。 连接 Arduino Nano RP2040 Connect 到电脑,通过 工具 > 端口,选择正确的端口。 2. Blink 三色 LED: How can I use the embedded RGB LED? RGB: The RGB LEDs are connected through the Wi-Fi module, so it is required to include the WiFiNINA library to use it. 使用板载的 RGB LED 控制三种颜色的闪烁(红、绿、蓝)。板子上还有一个LED_BUILTIN的橘色led 也一同点亮 C++ #include <WiFiNINA.h> void setup() { // initialize digital pin LED_BUILTIN as an output. pinMode(LED_BUILTIN, OUTPUT); pinMode(LEDR, OUTPUT); // 红色 LED pinMode(LEDG, OUTPUT); // 绿色 LED pinMode(LEDB, OUTPUT); // 蓝色 LED } // the loop function runs over and over again forever void loop() { digitalWrite(LED_BUILTIN, !digitalRead(LED_BUILTIN)); delay(500); digitalWrite(LEDR, HIGH); // 红色亮起 delay(500); digitalWrite(LEDR, LOW); digitalWrite(LEDG, HIGH); // 绿色亮起 delay(500); digitalWrite(LEDG, LOW); digitalWrite(LEDB, HIGH); // 蓝色亮起 delay(500); digitalWrite(LEDB, LOW); }   [localvideo]c5ec6e394b334757fbcb1e03a2ec2da0[/localvideo]   3. 串口打印“Hello DigiKey & EEWorld!”: 结合 Blink 程序,在串口输出文本。 C++ #include <WiFiNINA.h> void setup() { Serial.begin(9600); // 初始化串口通信 while (!Serial) {} // 等待串口连接 // initialize digital pin LED_BUILTIN as an output. pinMode(LED_BUILTIN, OUTPUT); pinMode(LEDR, OUTPUT); // 红色 LED pinMode(LEDG, OUTPUT); // 绿色 LED pinMode(LEDB, OUTPUT); // 蓝色 LED } // the loop function runs over and over again forever void loop() { digitalWrite(LED_BUILTIN, !digitalRead(LED_BUILTIN)); delay(500); digitalWrite(LEDR, HIGH); // 红色亮起 delay(500); digitalWrite(LEDR, LOW); digitalWrite(LEDG, HIGH); // 绿色亮起 delay(500); digitalWrite(LEDG, LOW); digitalWrite(LEDB, HIGH); // 蓝色亮起 delay(500); digitalWrite(LEDB, LOW); Serial.println("Hello DigiKey & EEWorld!"); // 打印信息 } 4. 上传代码并查看效果: 点击 上传 按钮,将代码上传到开发板。 打开 串口监视器,选择波特率 9600,观察输出 "Hello DigiKey & EEWorld!"。

  • 加入了学习《基于ESP32-S3-LCD-EV-Board的物联网多功能平台》,观看 基于ESP32-S3-LCD-EV-Board的物联网多功能平台

  • 2024-11-28
  • 回复了主题帖: 【2024 DigiKey 创意大赛】基于ESP32-S3-LCD-EV-Board的物联网多功能平台

    优秀作品

  • 2024-11-26
  • 加入了学习《基于Arduino玩转pico RP2040》,观看 基于Arduino玩转pico RP2040

  • 2024-11-25
  • 发表了主题帖: 【Follow me第二季第4期】Arduino_Nano_Connect板子开箱

    本帖最后由 Netseye 于 2024-11-25 12:49 编辑     感谢DigiKey联合EEWorld组织的Follow me第二季第4期 ,本次选择了一下元件参与活动. 商品列表 ARDUINO NANO RP2040 CONNECT 文档: https://content.arduino.cc/assets/ABX00053-datasheet.pdf ARDUINO NANO GROVE SHIELD 文档:https://mm.digikey.com/Volume0/opasdata/d220001/medias/docus/317/103100124_Web.pdf YELLOW&BLUE DISPLAY GROVE OLED 文档: https://wiki.seeedstudio.com/cn/Grove-OLED-Yellow&Blue-Display-0.96-SSD1315_V1.0/ 商品展示 上手体验 点亮屏幕 pdm音频展示 视频展示 [localvideo]dd0fa4129c64074a110da6fd7cee56e6[/localvideo] [localvideo]b4555d6872e8a94238ade0d8fb183495[/localvideo]   ok大概这么多.后续详在详细介绍.

  • 2024-10-31
  • 加入了学习《【2024 DigiKey创意大赛】- 基于毫米波雷达的生命体征检测及健康监护系统》,观看 【2024 DigiKey创意大赛】- 基于毫米波雷达的生命体征检测及健康监护系统-作品提交

  • 2024-10-28
  • 发表了主题帖: 【2024 DigiKey创意大赛】智能家居控制中心

    智能家居控制中心 作者:Netseye   一、作品简介 本次作品使用ESP32-S3-LCD-EV-BOARD 和STEMMA QT BME680 SENSOR BOARD BME680 来实现智能家居相关的一些尝试. 主要实现功能基于ESP32-S3-LCD-EV-BOARD实现esp-rainmaker 二维码配网.  通过bme680来获取室内气体,湿度,压力,温度 数据显示在屏幕上.并定时同步到ESP RAINMAKER.bme680温度超过30 会通过esp-rainker发送高温预警通知.     并通过esp-rainker实现数据上报. bme680温度超过30 会通过esp-rainker发送高温预警通知.   实现通过网络获取室外温度显示在屏幕上.并且可以通过语音指令进行开关灯控制.和室内温湿度等循环   二、系统框图、     文件目录 3185 ◯ tree -L 2 . ├── CMakeLists.txt ├── README.md ├── bsp │ └── bsp_extra // esp32 bsp extra ├── components │ ├── app_insights // ESP Insights (Beta) │ ├── app_network // 网络配置相关库 │ ├── app_reset // 重置相关库提供给esp-rmaker使用 │ ├── bme68x_lib // BME68X传感器驱动 │ ├── gpio_button // 按键驱动 │ ├── i2c_bus // I2C总线驱动 │ ├── nimble_central_utils // NimBLE工具库 │ └── qrcode // 二维码库 ├── dependencies.lock ├── esp_tts_voice_data_xiaoxin.dat // tts语音数据 ├── main // 项目主目录 │ ├── CMakeLists.txt │ ├── Kconfig │ ├── app // 项目应用层逻辑代码 │ ├── gui // 项目GUI代码 │ ├── idf_component.yml // 项目组件依赖配置 │ ├── main.c // 项目入口 │ ├── main.h │ ├── rmaker // esp-rmaker相关代码 │ ├── settings.c // 项目配置 │ └── settings.h ├── partitions.csv // 分区表 ├── sdkconfig ├── sdkconfig.defaults ├── sdkconfig.defaults.psram_120m_ddr ├── sdkconfig.old └── spiffs // spiffs文件系统存音频文件和图片 ├── echo_cn_end.wav ├── echo_cn_ok.wav ├── echo_cn_wake.wav ├── echo_en_end.wav ├── echo_en_ok.wav ├── echo_en_wake.wav ├── gpt_avatar.gif └── mp3 18 directories, 23 files 核心代码 3193 ◯ tree -L 2 . ├── CMakeLists.txt ├── Kconfig ├── app │ ├── app_audio.c │ ├── app_audio.h │ ├── app_ble.c │ ├── app_ble.h │ ├── app_chat.c │ ├── app_chat.h │ ├── app_sr.c │ ├── app_sr.h │ ├── app_sr_handler.c │ ├── app_sr_handler.h │ ├── app_sr_tts.c │ ├── app_sr_tts.h │ ├── app_weather.c │ ├── app_weather.h │ └── blecent.h ├── gui │ ├── assert │ ├── font │ ├── lv_example_image.h │ ├── lv_example_pub.c │ ├── lv_example_pub.h │ ├── lv_example_top_menu.c │ ├── lv_schedule_basic.c │ ├── lv_schedule_basic.h │ ├── ui_chat.c │ ├── ui_chat.h │ ├── ui_rmaker.c │ ├── ui_rmaker.h │ ├── ui_sr.c │ └── ui_sr.h ├── idf_component.yml ├── main.c ├── main.h ├── rmaker │ ├── app_bme680.c │ ├── app_bme680.h │ ├── app_driver.c │ ├── app_priv.h │ ├── app_rmaker.c │ └── app_rmaker.h ├── settings.c └── settings.h 三、各部分功能说明 Main void app_main(void) { /* Initialize NVS. */ esp_err_t err = nvs_flash_init(); if (err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND) { ESP_ERROR_CHECK(nvs_flash_erase()); err = nvs_flash_init(); } ESP_ERROR_CHECK(err); ESP_ERROR_CHECK(settings_read_parameter_from_nvs()); bsp_spiffs_mount(); bsp_i2c_init(); bsp_display_start(); bsp_extra_led_init(); bsp_extra_codec_init(); app_bme680_init(); ESP_LOGI(TAG, "Display LVGL"); bsp_display_lock(0); lv_style_pre_init(); lv_create_home(&rmaker_Layer); bsp_display_unlock(); ESP_LOGI(TAG, "Initializing RainMaker"); app_rmaker_start(); app_weather_start(); ESP_LOGI(TAG, "speech recognition Enable"); app_sr_start(false); bsp_audio_poweramp_enable(true); ESP_LOGI(TAG, "TTS Enable"); app_tts_init(); }   BME680数据采集 // 初始化i2c bme680 void app_bme680_init(void) { // 初始化 I2C 总线 esp_err_t ret = i2c_bus_init( &i2c_bus, BSP_I2C_NUM, BSP_I2C_SDA_R16, BSP_I2C_SCL_R16, GPIO_PULLUP_DISABLE, GPIO_PULLUP_DISABLE, CONFIG_BSP_I2C_CLK_SPEED_HZ); // 初始化 bme68x_lib_t 实例 esp_err_t result = bme68x_lib_init(&bme68x_instance, &i2c_bus, intf); if (result != ESP_OK) { ESP_LOGE(TAG, "BME680 initialization failed"); } else { ESP_LOGI(TAG, "BME680 initialized successfully"); } bme68x_lib_set_filter(&bme68x_instance, BME68X_FILTER_OFF); bme68x_lib_set_tph(&bme68x_instance, BME68X_OS_2X, BME68X_OS_1X, BME68X_OS_16X); bme68x_lib_set_heater_prof_for(&bme68x_instance, 300, 100); bme68x_lib_set_op_mode(&bme68x_instance, ope_mode); } // 获取bme680信息 esp_err_t app_bme680_get_current_info(bme68x_data_t *info) { if (bme68x_lib_intf_error(&bme68x_instance) == 0) { bme68x_lib_set_op_mode(&bme68x_instance, ope_mode); bme_delay = bme68x_lib_get_meas_dur(&bme68x_instance, ope_mode); (bme68x_instance.bme6).delay_us(bme_delay, (bme68x_instance.bme6).intf_ptr); bme68x_lib_fetch_data(&bme68x_instance); bme68x_data_t *data = bme68x_lib_get_all_data(&bme68x_instance); info->temperature = data->temperature; info->pressure = data->pressure; info->humidity = data->humidity; info->gas_resistance = data->gas_resistance; info->status = data->status; return ESP_OK; } else { return ESP_FAIL; } }      ESP RainMaker /* Create a Switch device and add the relevant parameters to it */ switch_device = esp_rmaker_switch_device_create("开关", NULL, DEFAULT_SWITCH_POWER); esp_rmaker_device_add_cb(switch_device, write_cb, NULL); esp_rmaker_node_add_device(node, switch_device); /* Create a Light device and add the relevant parameters to it */ light_device = esp_rmaker_lightbulb_device_create("灯", NULL, DEFAULT_LIGHT_POWER); esp_rmaker_device_add_cb(light_device, write_cb, NULL); esp_rmaker_device_add_param(light_device, esp_rmaker_brightness_param_create( ESP_RMAKER_DEF_BRIGHTNESS_NAME, DEFAULT_LIGHT_BRIGHTNESS)); esp_rmaker_param_t *color_param = esp_rmaker_param_create("color", NULL, esp_rmaker_int(0), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_add_ui_type(color_param, ESP_RMAKER_UI_HUE_SLIDER); esp_rmaker_device_add_param(light_device, color_param); esp_rmaker_device_add_attribute(light_device, "Desc", "2024 DigiKey"); esp_rmaker_node_add_device(node, light_device); /* Create a Fan device and add the relevant parameters to it */ fan_device = esp_rmaker_fan_device_create("风扇", NULL, DEFAULT_FAN_POWER); esp_rmaker_device_add_cb(fan_device, write_cb, NULL); esp_rmaker_device_add_param(fan_device, esp_rmaker_speed_param_create(ESP_RMAKER_DEF_SPEED_NAME, DEFAULT_FAN_SPEED)); esp_rmaker_node_add_device(node, fan_device); //2 节点上: 创建一个BME680设备。 temp_sensor_device = esp_rmaker_temp_sensor_device_create("BME680", NULL, 0); // // 接下来的几行代码创建了几个参数,如“地区”、“天气”、“风力”和“湿度”,并设置了它们的属性和初始值。 esp_rmaker_param_t *temperature_param = esp_rmaker_param_create("温度", "bme680.param.temperature", esp_rmaker_float(app_get_current_temperature()), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_t *humidity_param = esp_rmaker_param_create("湿度", "bme680.param.humidity", esp_rmaker_float(app_get_current_humidity()), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_t *pressure_param = esp_rmaker_param_create("气压", "bme680.param.pressure", esp_rmaker_float(app_get_current_pressure()), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_t *altitude_param = esp_rmaker_param_create("海拔", "bme680.param.altitude", esp_rmaker_float(app_get_current_altitude()), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_t *gas_param = esp_rmaker_param_create("GAS", "bme680.param.gas", esp_rmaker_float(app_get_current_gas()), PROP_FLAG_READ | PROP_FLAG_WRITE); esp_rmaker_param_t *iaq_param = esp_rmaker_param_create("IAQ", "bme680.param.iaq", esp_rmaker_float(app_get_current_iaq()), PROP_FLAG_READ | PROP_FLAG_WRITE); // // 这些参数随后被添加到温度传感器虚拟设备中。 esp_rmaker_device_add_param(temp_sensor_device, temperature_param); esp_rmaker_device_add_param(temp_sensor_device, humidity_param); esp_rmaker_device_add_param(temp_sensor_device, pressure_param); esp_rmaker_device_add_param(temp_sensor_device, altitude_param); esp_rmaker_device_add_param(temp_sensor_device, gas_param); esp_rmaker_device_add_param(temp_sensor_device, iaq_param); // // 设置主要参数 app上首先显示 esp_rmaker_device_assign_primary_param(temp_sensor_device, temperature_param); // esp_rmaker_param_t *get_param = esp_rmaker_param_create("get_weather", NULL, esp_rmaker_bool(false), PROP_FLAG_READ | PROP_FLAG_WRITE); // esp_rmaker_param_add_ui_type(get_param, ESP_RMAKER_UI_PUSHBUTTON); // esp_rmaker_device_add_param(temp_sensor_device, get_param); esp_rmaker_node_add_device(node, temp_sensor_device); Esp-sr 命令列表 static const sr_cmd_t g_default_cmd_info[] = { // Chinese {SR_CMD_TEMP_DEC, SR_LANG_CN, 0, "室内温度", "shi nei wen du", {NULL}}, {SR_CMD_PRESSURE_DEC, SR_LANG_CN, 0, "室内气压", "shi nei qi ya", {NULL}}, {SR_CMD_HUMIDITY_DEC, SR_LANG_CN, 0, "室内湿度", "shi nei shi du", {NULL}}, {SR_CMD_LIGHT_ON, SR_LANG_CN, 0, "打开电灯", "da kai dian deng", {NULL}}, {SR_CMD_LIGHT_OFF, SR_LANG_CN, 0, "关闭电灯", "guan bi dian deng", {NULL}}, }; Esp-sr 初始化 BaseType_t ret_val; models = esp_srmodel_init("model"); afe_handle = (esp_afe_sr_iface_t *)&ESP_AFE_SR_HANDLE; afe_config_t afe_config = AFE_CONFIG_DEFAULT(); afe_config.wakenet_model_name = esp_srmodel_filter(models, ESP_WN_PREFIX, NULL); afe_config.aec_init = false; esp_afe_sr_data_t *afe_data = afe_handle->create_from_config(&afe_config); g_sr_data->afe_handle = afe_handle; g_sr_data->afe_data = afe_data; sys_param_t *param = settings_get_parameter(); g_sr_data->lang = SR_LANG_MAX; ret = app_sr_set_language(param->sr_lang); esp-sr handler void sr_handler_task(void *pvParam) { static lv_event_info_t event; event.event = LV_EVENT_EXIT_CLOCK; while (true) { sr_result_t result; app_sr_get_result(&result, portMAX_DELAY); if (true == result.beep_enable) { sr_echo_play(AUDIO_PRESS); continue; } if (ESP_MN_STATE_TIMEOUT == result.state) { #if !SR_RUN_TEST event.event = LV_EVENT_I_AM_LEAVING; app_lvgl_send_event(&event); sr_echo_play(AUDIO_END); #endif continue; } if (WAKENET_DETECTED == result.wakenet_mode) { event.event = LV_EVENT_I_AM_HERE; app_lvgl_send_event(&event); #if !SR_RUN_TEST sr_echo_play(AUDIO_WAKE); #endif continue; } if (ESP_MN_STATE_DETECTED & result.state) { const sr_cmd_t *cmd = app_sr_get_cmd_from_id(result.command_id); if (NULL == cmd) { continue; } ESP_LOGI(TAG, "command:%s, act:%d", cmd->str, cmd->cmd); event.event_data = (void *)cmd->str; switch (cmd->cmd) { case SR_CMD_SET_RED: event.event = LV_EVENT_LIGHT_RGB_SET; app_lvgl_send_event(&event); break; case SR_CMD_SET_WHITE: event.event = LV_EVENT_LIGHT_RGB_SET; app_lvgl_send_event(&event); break; case SR_CMD_SET_YELLOW: event.event = LV_EVENT_LIGHT_RGB_SET; app_lvgl_send_event(&event); break; case SR_CMD_SET_BLUE: event.event = LV_EVENT_LIGHT_RGB_SET; app_lvgl_send_event(&event); break; case SR_CMD_LIGHT_ON: event.event = LV_EVENT_LIGHT_ON; app_lvgl_send_event(&event); break; case SR_CMD_LIGHT_OFF: event.event = LV_EVENT_LIGHT_OFF; app_lvgl_send_event(&event); break; case SR_CMD_TEMP_DEC: char temp_str[64]; snprintf(temp_str, sizeof(temp_str), "室内当前 %.2f 摄氏度 ", info.temperature); app_tts_play(temp_str); break; case SR_CMD_PRESSURE_DEC: char pressure_str[64]; snprintf(pressure_str, sizeof(pressure_str), "室内当前 %.2f 千帕 ", info.pressure / 100.0); app_tts_play(pressure_str); break; case SR_CMD_HUMIDITY_DEC: char humidity_str[64]; snprintf(humidity_str, sizeof(humidity_str), "室内当前湿度 %.2f ", info.humidity); app_tts_play(humidity_str); break; default: ESP_LOGE(TAG, "Unknow cmd"); break; } #if !SR_RUN_TEST // sr_echo_play(AUDIO_OK); #endif } } vTaskDelete(NULL); } Esp-tts esp_err_t app_tts_init(void) { /* Initial voice set from separate voice data partition */ const esp_partition_t* part = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_ANY, "voice_data"); if (part==NULL) { ESP_LOGE(TAG, "Couldn't find voice data partition!"); return 0; } else { ESP_LOGI(TAG, "voice_data paration size:%ld", part->size); } const void* voicedata; esp_partition_mmap_handle_t mmap; esp_err_t err = esp_partition_mmap(part, 0, part->size, ESP_PARTITION_MMAP_DATA, &voicedata, &mmap); if (err != ESP_OK) { ESP_LOGE(TAG, "Couldn't map voice data partition!"); return ESP_FAIL; } esp_tts_voice_t *voice = esp_tts_voice_set_init(&esp_tts_voice_template, (int16_t*)voicedata); tts_handle = esp_tts_create(voice); if (tts_handle == NULL) { ESP_LOGE(TAG, "Failed to create TTS handle!"); return ESP_FAIL; } spk_codec_dev = bsp_audio_codec_speaker_init(); assert(spk_codec_dev); esp_codec_dev_set_out_vol(spk_codec_dev, DEFAULT_VOLUME); return ESP_OK; } esp_err_t app_tts_play(const char *prompt1) { esp_codec_dev_sample_info_t fs = { .sample_rate = SAMPLE_RATE, .channel = EXAMPLE_CHANNEL, .bits_per_sample = EXAMPLE_BITS, }; esp_codec_dev_open(spk_codec_dev, &fs); /* Play prompt text */ ESP_LOGI(TAG, "prompt1: %s", prompt1); if (esp_tts_parse_chinese(tts_handle, prompt1)) { int len[1]={0}; do { short *pcm_data=esp_tts_stream_play(tts_handle, len, 1); int error_code = esp_codec_dev_write(spk_codec_dev, pcm_data, len[0]*2); // ESP_LOGI(TAG, "Write result: %d ", error_code); // ESP_LOGI(TAG, "data: %d", len[0]); } while(len[0]>0); } esp_codec_dev_close(spk_codec_dev); esp_tts_stream_reset(tts_handle); return ESP_OK; } 四、作品源码 iot-box.zip 五、作品功能演示视频 [localvideo]b9870e2701dad886393917ba283ef82e[/localvideo]   六、项目总结       https://bbs.eeworld.com.cn/thread-1291239-1-1.html       https://bbs.eeworld.com.cn/thread-1292908-1-1.html       https://bbs.eeworld.com.cn/thread-1296188-1-1.html ESP32-S3-LCD-EV-BOARD 和 bme680 是非常优秀的开发板和传感器.本次项目抛砖引玉,仅仅是为了完整在乐鑫生态下的iot探索,并未能完全展示其在IOT上的应用.本来计划是有对应的llm大模型对话和ble链接米家蓝牙设备的.尝试过程中存在一些性能和冲突问题.暂时未能解决.调试过程较为痛苦.完整刷一次固件要好几分钟...   七、其他        后续会对代码进行优化和整理.解决大模型对话和ble接入米家设备的集成问题.使项目更加完善.

  • 上传了资料: 【2024 DigiKey创意大赛】智能家居控制中心

  • 2024-10-20
  • 加入了学习《得捷电子专区》,观看 【2024 DigiKey 创意大赛】红外温度检测及火灾报警器

  • 2024-10-15
  • 回复了主题帖: 【2024 DigiKey创意大赛】【智能家居控制中心】【EP03】ESP32-S3-LCD-EV + BME680 BLE

    后记 关于如何发现米家蓝牙设备地址和关于idf中的数据大小端问题做个总结: 如何发现米家设备: https://github.com/rytilahti/python-miio 通过手机ble app 找名字LYWSD03MMC或者自己写一个工具 https://lywsd02mmc.bilaldurrani.com/类似的一些web端的ble工具 其他...就不列举了 大小端转换问题: 我这边写了一个小工具来实现   #include <stdint.h> #include <stdio.h> #include <string.h> void hexStringToByteArray(const char *hexString, uint8_t *byteArray) { size_t len = strlen(hexString); for (size_t i = 0; i < len; i += 2) { sscanf(hexString + i, "%2hhx", &byteArray[i / 2]); } } void printByteArray(const uint8_t *byteArray) { printf("BLE_UUID128_DECLARE("); for (size_t i = 15; i < 16; i--) { printf("0x%02x", byteArray[i]); if (i > 0) { printf(", "); } } printf(")\n"); } int main() { const char *hexStrings[] = { "ebe0ccb07a0a4b0c8a1a6ff2997da3a6", "ebe0ccc17a0a4b0c8a1a6ff2997da3a6" }; uint8_t byteArray[16]; // 32 hex digits = 16 bytes for (size_t j = 0; j < sizeof(hexStrings) / sizeof(hexStrings[0]); j++) { hexStringToByteArray(hexStrings[j], byteArray); printf("Little-endian Byte array for %s: ", hexStrings[j]); printByteArray(byteArray); } return 0; }  

  • 发表了主题帖: 【2024 DigiKey创意大赛】【智能家居控制中心】【EP03】ESP32-S3-LCD-EV + BME680 BLE

    本次实现通过ESP32 BLE 来实现链接米家的温湿度计.esp32 存在2个蓝牙协议栈. Bluedroid - Dual-mode和 NimBLE - BLE only二选一. 最早使用了Bluedroid来实现这个功能.集成过程中发现esp-rainmarker 使用的是NimBLE,又不是一波令人窒息的操作... NimBLE: https://docs.espressif.com/projects/esp-idf/en/stable/esp32/api-reference/bluetooth/nimble/index.html Bluedroid - Dual-mode: https://docs.espressif.com/projects/esp-idf/en/stable/esp32/api-reference/bluetooth/bt_le.html 概念 Gap GAP 的广播工作流程       GAP(Generic Access Profile,通用访问配置文件)是蓝牙协议栈中的一部分,负责定义蓝牙设备之间如何发现、连接和进行交互。GAP提供了设备连接和通信的基本框架。它主要包含以下几个方面:   1. 广播和扫描 广播(Advertising):设备通过广播信道发送数据,其他设备可以扫描到这些广播数据。广播包含了设备的基本信息,例如名称、服务UUID等。 扫描(Scanning):设备监听广播信道以发现周围的蓝牙设备和其广播数据。   2. 连接 发起连接(Connection Initiation):扫描设备可以发起与广播设备的连接,建立一条点对点的连接。 连接参数(Connection Parameters):GAP定义了在连接建立时如何设置连接参数,如连接间隔、超时时间等。   3. 角色   GAP规定了蓝牙设备可以扮演的四种角色: 广播者(Broadcaster):只广播数据,不接收数据或发起连接。 观察者(Observer):只扫描广播数据,不发起连接。 主设备(Central):扫描并发起连接,通常是主控设备(如手机)。 从设备(Peripheral):广播并接受连接请求,通常是外围设备(如传感器)。   4. 配对与安全 配对(Pairing):GAP定义了蓝牙设备如何通过加密进行配对,确保通信的安全性。 密钥分发(Key Distribution):配对后,设备之间可以交换密钥用于加密通信。 安全等级(Security Levels):GAP支持不同的安全等级,从没有加密到加密和认证。   5. 设备名称和可见性 设备名称(Device Name):GAP允许设备广播其名称,以便用户识别。 可见性(Visibility):设备可以配置为可见或不可见,影响它是否能够被其他设备发现。   6. 连接管理 连接建立与终止:GAP定义了设备如何建立和终止蓝牙连接。 连接参数更新(Connection Parameter Update):在连接过程中,设备可以协商更新连接参数以优化性能。   7. 扩展功能 扩展广告(Extended Advertising):如你提到的,GAP标准中的一部分允许广播更多的数据,支持长时间的广播周期。 周期性广告(Periodic Advertising):用于周期性发送广播数据,适用于广播数据不频繁更新的场景。   GAP是设备发现、连接建立、连接管理、安全配对的基础层,支持BLE应用的各种交互方式。 GATT (GATT服务端)和中心设备(GATT客户端)之间的数据交流流程,   GATT(Generic Attribute Profile,通用属性配置文件)是蓝牙低功耗(BLE)协议栈中的一个重要部分,负责设备之间数据的组织、交换和传输。它基于属性(Attribute)的形式进行通信,定义了如何在已连接的设备之间通过服务和特征(Characteristics)进行数据传输。GATT的设计使得BLE设备能够以结构化的方式进行数据交互,主要包括以下内容:   1. 属性(Attribute)   属性是GATT通信的核心,表示设备内部的某一项数据,通常由一个16位或128位的UUID标识。属性包含三个关键元素: 句柄(Handle):属性的唯一标识符,通常是一个16位的值,表示属性在设备上的位置。 类型(Type):属性的类型,通过UUID进行标识。例如,某属性可能是一个服务或特征。 值(Value):属性的实际数据,可以是任意数据类型。   2. 服务(Service)   服务是由一组相关的特征和属性组成的,代表一个逻辑功能。例如,心率服务可以包含多个特征来传输心率数据。 服务类型: 主要服务(Primary Service):表示设备的主要功能服务。 次要服务(Secondary Service):依附于主要服务的辅助功能。   3. 特征(Characteristic)   特征是GATT的核心数据单元,每个特征由一个值和可选的描述符组成。特征用于读写或通知数据传输。 特征值(Characteristic Value):特征的具体数据,可以被读取、写入或订阅。 特征属性(Characteristic Properties):定义特征支持的操作类型,如读取、写入、通知等。   4. 描述符(Descriptor)   描述符是特征的附加信息,描述特征的特定细节。例如,描述符可以定义特征值的单位或数据格式。 常见的描述符包括: 客户端特征配置描述符(Client Characteristic Configuration Descriptor, CCCD):用于配置通知或指示。 特征用户描述符(Characteristic User Description, CUD):对特征值的简要说明。   5. GATT 操作   GATT定义了设备如何通过服务和特征来交互数据,包括以下操作: 发现操作(Discovery Operations):设备可以发现远程设备支持的服务、特征和描述符。这包括发现所有服务、发现特定特征、发现特征的描述符等。 读取操作(Read Operations):读取某个特征或描述符的值,通常主设备(Central)向从设备(Peripheral)发起。 写入操作(Write Operations):写入某个特征或描述符的值,可以是有响应的写入(Write with Response)或无响应的写入(Write without Response)。 通知(Notification)和指示(Indication):从设备可以主动向主设备发送特征值的变化。通知不需要确认,指示则需要主设备确认接收。   6. GATT 服务协议栈   GATT作为一个协议栈分层,通常与其他蓝牙层次协同工作: ATT(Attribute Protocol):GATT基于ATT协议构建,ATT提供了对设备属性进行读写的机制。GATT扩展了ATT的功能,加入了服务和特征的结构化数据模型。 L2CAP(Logical Link Control and Adaptation Protocol):ATT和GATT协议通过L2CAP层进行数据封装和传输。   7. GATT 角色   GATT协议中,设备可以扮演以下两种角色: GATT 服务器(Server):保存数据并响应客户端的请求,通常是外围设备(如传感器)。 GATT 客户端(Client):发起请求来读取或写入数据,通常是主设备(如手机或平板)。   8. 标准化服务和特征   蓝牙组织定义了一系列标准化的服务和特征,用于特定应用场景。例如: 心率服务(Heart Rate Service):用于传输心率数据。 电池服务(Battery Service):报告设备的电池电量。 设备信息服务(Device Information Service, DIS):提供设备的制造商信息、硬件版本等。   9. 长特征值   GATT支持通过分块操作(Read Blob/Write Blob)来传输超出单个ATT帧限制的长特征值(通常大于20字节)。   10. GATT 协议操作示例 连接后发现服务:在连接建立后,客户端通常会首先发现服务器支持的服务。 读取心率数据:客户端可以读取“心率特征”值,获取实时心率。 写入控制命令:客户端可以通过写入特定特征来控制设备的行为。 GATT 结构 GATT事务是建立在嵌套的Profiles,Services和Characteristics之上的,如下如所示:   总结来说,GATT负责组织和传输设备之间的属性数据,通过服务和特征的形式,使BLE设备能够以标准化的方式进行数据交互。   实践 首先我们需要通过GAP协议进行扫描和链接,在ble 5.0以上存在着Advertising 和 Extended Advertising两种不同的方式.可以通过 idf.py menuconfig 进行修改Enable extended advertising esp32芯片的板子就不要尝试了,用不了. (Top) → Component config → Bluetooth → NimBLE Options → Enable BLE 5 feature Espressif IoT Development Framework Configuration [*] Enable 2M Phy [*] Enable coded Phy [ ] Enable extended advertising (0) Maximum number of periodic advertising syncs [ ] Enable GATT caching ---- 开启蓝牙 void app_ble_init(void) { int rc; /* Initialize NVS — it is used to store PHY calibration data */ esp_err_t ret = nvs_flash_init(); if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) { ESP_ERROR_CHECK(nvs_flash_erase()); ret = nvs_flash_init(); } ESP_ERROR_CHECK(ret); ret = nimble_port_init(); if (ret != ESP_OK) { ESP_LOGE(tag, "Failed to init nimble %d ", ret); return; } ESP_LOGI(tag, "init nimble success"); /* Configure the host. */ ble_hs_cfg.reset_cb = blecent_on_reset; ble_hs_cfg.sync_cb = blecent_on_sync; ble_hs_cfg.store_status_cb = ble_store_util_status_rr; /* Initialize data structures to track connected peers. */ rc = peer_init(MYNEWT_VAL(BLE_MAX_CONNECTIONS), 64, 64, 64); assert(rc == 0); /* Set the default device name. */ rc = ble_svc_gap_device_name_set("nimble-blecent"); assert(rc == 0); /* XXX Need to have template for store */ ble_store_config_init(); nimble_port_freertos_init(blecent_host_task); } 实现cb_event和开启扫描   /** * Initiates the GAP general discovery procedure. */ static void blecent_scan(void) { uint8_t own_addr_type; struct ble_gap_disc_params disc_params; int rc; /* Figure out address to use while advertising (no privacy for now) */ rc = ble_hs_id_infer_auto(0, &own_addr_type); if (rc != 0) { MODLOG_DFLT(ERROR, "error determining address type; rc=%d\n", rc); return; } /* Tell the controller to filter duplicates; we don't want to process * repeated advertisements from the same device. */ disc_params.filter_duplicates = 1; /** * Perform a passive scan. I.e., don't send follow-up scan requests to * each advertiser. */ disc_params.passive = 0; /* Use defaults for the rest of the parameters. */ disc_params.itvl = 0; disc_params.window = 0; disc_params.filter_policy = 0; disc_params.limited = 0; rc = ble_gap_disc(own_addr_type, BLE_HS_FOREVER, &disc_params, blecent_gap_event, NULL); if (rc != 0) { MODLOG_DFLT(ERROR, "Error initiating GAP discovery procedure; rc=%d\n", rc); } } static void blecent_on_reset(int reason) { MODLOG_DFLT(ERROR, "Resetting state; reason=%d\n", reason); } static void blecent_on_sync(void) { int rc; /* Make sure we have proper identity address set (public preferred) */ rc = ble_hs_util_ensure_addr(0); assert(rc == 0); /* Begin scanning for a peripheral to connect to. */ blecent_scan(); } void blecent_host_task(void *param) { ESP_LOGI(tag, "BLE Host Task Started"); /* This function will return only when nimble_port_stop() is executed */ nimble_port_run(); nimble_port_freertos_deinit(); } /* This function showcases stack init and deinit procedure. */ static void stack_init_deinit(void) { int rc; while (1) { vTaskDelay(1000); ESP_LOGI(tag, "Deinit host"); rc = nimble_port_stop(); if (rc == 0) { nimble_port_deinit(); } else { ESP_LOGI(tag, "Nimble port stop failed, rc = %d", rc); break; } vTaskDelay(1000); ESP_LOGI(tag, "Init host"); rc = nimble_port_init(); if (rc != ESP_OK) { ESP_LOGI(tag, "Failed to init nimble %d ", rc); break; } nimble_port_freertos_init(blecent_host_task); ESP_LOGI(tag, "Waiting for 1 second"); } } 链接米家温湿度计 温度计定义   /*** The UUID of the service containing the subscribable characterstic ***/ static const ble_uuid_t *remote_svc_uuid = BLE_UUID128_DECLARE(0xa6, 0xa3, 0x7d, 0x99, 0xf2, 0x6f, 0x1a, 0x8a, 0x0c, 0x4b, 0x0a, 0x7a, 0xb0, 0xcc, 0xe0, 0xeb); /*** The UUID of the subscribable chatacteristic ***/ static const ble_uuid_t *remote_chr_uuid = BLE_UUID128_DECLARE(0xa6, 0xa3, 0x7d, 0x99, 0xf2, 0x6f, 0x1a, 0x8a, 0x0c, 0x4b, 0x0a, 0x7a, 0xc1, 0xcc, 0xe0, 0xeb); static uint8_t target_addr[6] = {0xE1, 0x0A, 0x04, 0x38, 0xC1, 0xA4}; typedef struct { float temperature; float humidity; float voltage; float battery; } Result; Result processBuffer(const uint8_t *buff) { Result result = {0, 0, 0, 0}; // Temperature conversion int16_t temp = (int16_t)(buff[0] | (buff[1] << 8)); // Little-endian result.temperature = temp / 100.0f; // Humidity conversion result.humidity = (float)buff[2]; // Voltage conversion int16_t voltageRaw = (int16_t)(buff[3] | (buff[4] << 8)); // Little-endian result.voltage = voltageRaw / 1000.0f; // Battery percentage calculation result.battery = ((result.voltage - 2) / (3.261 - 2)) * 100; result.battery = (result.battery < 0) ? 0 : (result.battery > 100) ? 100 : result.battery; // Clamp between 0 and 100 return result; } BLE读取信息   static int blecent_on_read(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg) { MODLOG_DFLT(INFO, "Read complete; status=%d conn_handle=%d", error->status, conn_handle); if (error->status == 0) { MODLOG_DFLT(INFO, " attr_handle=%d value=", attr->handle); print_mbuf(attr->om); Result result = processBuffer(attr->om->om_data); printf("Temperature: %.2f\n", result.temperature); printf("Humidity: %.2f\n", result.humidity); printf("Voltage: %.3f\n", result.voltage); printf("Battery: %.2f%%\n", result.battery); } MODLOG_DFLT(INFO, "\n"); return 0; } static void blecent_read_write_subscribe(const struct peer *peer) { const struct peer_chr *chr; int rc; /* Read the supported-new-alert-category characteristic. */ chr = peer_chr_find_uuid( peer, remote_svc_uuid, remote_chr_uuid); if (chr == NULL) { MODLOG_DFLT(ERROR, "Error \n"); goto err; } rc = ble_gattc_read(peer->conn_handle, chr->chr.val_handle, blecent_on_read, NULL); if (rc != 0) { MODLOG_DFLT(ERROR, "Error: Failed to read characteristic; rc=%d\n", rc); goto err; } return; err: /* Terminate the connection. */ ble_gap_terminate(peer->conn_handle, BLE_ERR_REM_USER_CONN_TERM); } static void blecent_on_disc_complete(const struct peer *peer, int status, void *arg) { if (status != 0) { /* Service discovery failed. Terminate the connection. */ MODLOG_DFLT(ERROR, "Error: Service discovery failed; status=%d " "conn_handle=%d\n", status, peer->conn_handle); ble_gap_terminate(peer->conn_handle, BLE_ERR_REM_USER_CONN_TERM); return; } /* Service discovery has completed successfully. Now we have a complete * list of services, characteristics, and descriptors that the peer * supports. */ MODLOG_DFLT(INFO, "Service discovery complete; status=%d " "conn_handle=%d\n", status, peer->conn_handle); /* Now perform three GATT procedures against the peer: read, * write, and subscribe to notifications for the ANS service. */ blecent_read_write_subscribe(peer); } static int blecent_gap_event(struct ble_gap_event *event, void *arg) { struct ble_gap_conn_desc desc; struct ble_hs_adv_fields fields; #if MYNEWT_VAL(BLE_HCI_VS) #if MYNEWT_VAL(BLE_POWER_CONTROL) struct ble_gap_set_auto_pcl_params params; #endif #endif int rc; switch (event->type) { case BLE_GAP_EVENT_DISC: rc = ble_hs_adv_parse_fields(&fields, event->disc.data, event->disc.length_data); if (rc != 0) { return 0; } /* An advertisment report was received during GAP discovery. */ print_adv_fields(&fields); /* Try to connect to the advertiser if it looks interesting. */ blecent_connect_if_interesting(&event->disc); return 0; case BLE_GAP_EVENT_EXT_DISC: /* An advertisment report was received during GAP discovery. */ ext_print_adv_report(&event->disc); blecent_connect_if_interesting(&event->disc); return 0; default: return 0; } } 结果: 优化 目前提供的是代码核心部分和简单的测试逻辑.项目中会做成可以手动控制扫码设备.通过触摸屏点击对应的蓝牙设备进行连接.并支持多个设备连接.  

最近访客

< 1/2 >

统计信息

已有15人来访过

  • 芯积分:90
  • 好友:--
  • 主题:10
  • 回复:9

留言

你需要登录后才可以留言 登录 | 注册


现在还没有留言