左手阿飞 发表于 2024-10-22 15:30

【2024 DigiKey 创意大赛】会议声源追踪器 会议室环境检测(2)

本帖最后由 左手阿飞 于 2024-10-22 15:44 编辑

<p>这次打算做一个会议拾音系统,主要功能如下:</p>

<p>1、声源定位,识别声音方向;(麦克风阵列,控制器)</p>

<p>2、声源追踪,用步进电机控制麦克风方向,追踪声源,减少干扰噪声;(麦克风阵列,控制器)</p>

<p>3、检测会议室环境,上传云端,可以提醒工作人员提前开启空调等设备。(传感器、ESP32)。</p>

<p>&nbsp;</p>

<p>上一次发帖,通过DHT11进行温湿度检测,并上传到阿里云,通过手机可以查看。</p>

<p>本次增加了SGP30,进行TVOC和CO2检测并上传阿里云。</p>

<p>一、SGP30基本参数:</p>

<p>SGP30通过I2C通信,并且可以采集的数据有CO2和TVOC的含量。TVOC是&ldquo;Total Volatile Organic Compounds&rdquo;,意思是总挥发性有机化合物。</p>

<p>1、TVOC输出范围&nbsp; 0~&middot;60000,CO2输出范围400~60000,IIC地址0x58。</p>

<p>2、数据格式:6个字为一组,每3个字为一个数据:前两个字为数据内容,第三个字为CRC校验。</p>

<p></p>

<p> &nbsp; &nbsp;</p>

<p> &nbsp;</p>

<p>二、软件实现:</p>

<p>代码如下:</p>

<p>1、包含库函数:</p>

<p> &nbsp;</p>

<p>2、IIC引脚定义:</p>

<p>&nbsp;</p>

<p>3、声明一个类:</p>

<p> &nbsp;</p>

<p>4、初始化:</p>

<p> &nbsp;</p>

<p>5;检测TVOC和CO2:</p>

<p> &nbsp;</p>

<p>6、上传到阿里云:</p>

<p>&nbsp;</p>

<p>三、效果展示:</p>

<p>1、硬件连接:内部已经有上拉电阻,使用IIC连接时,不再外加上拉电阻,也可以正常工作。</p>

<p> &nbsp;</p>

<p> &nbsp;</p>

<p>2、阿里云后台数据显示:</p>

<p>(1)正常数据显示:</p>

<p></p>

<p>&nbsp;&nbsp;&nbsp; (2)吹气后,数据显示:</p>

<p>&nbsp; (3)手机APP数据显示:</p>

<p> &nbsp;</p>

<p>附代码:</p>

<pre>
<code class="language-cpp">#include &lt;Arduino.h&gt;
#include &lt;WiFi.h&gt;
#include "PubSubClient.h"
#include "DHT.h"
#include "ArduinoJson.h"
#include "Adafruit_SGP30.h"

#define DHTPIN 4
#define LED 38
#define DHTTYPE DHT11

#define SCL 20
#define SDA 21

DHT dht(DHTPIN,DHTTYPE) ;

#define WIFI_SSID         "HONOR 200 Pro"
#define WIFI_PASSWORD   "87654321"

#define PRODUCT_KEY       "a1H4kfnF05D"
#define DEVICE_NAME       "ESP32S3Test"
#define DEVICE_SECRET   "c0fcb349106f4730b6358b62320f4aab"
#define REGION_ID         "cn-shanghai"


#define MQTT_SERVER                  PRODUCT_KEY".iot-as-mqtt."REGION_ID ".aliyuncs.com"
#define MQTT_PORT                  1883
#define MQTT_USERNAME                DEVICE_NAME"&amp;"PRODUCT_KEY

#define CLIENT_ID                  "a1H4kfnF05D.ESP32S3Test|securemode=2,signmethod=hmacsha256,timestamp=1729067953966|"
#define MQTT_PASSWD                  "252b17f2417acf371e14d0d1f3ac85d070ea49960db96f001ff33b1eafbe1bc0"
#define ALINK_BODY_FORMAT            "{\"id\":\"ESP32S3Test\",\"version\":\"1.0\",\"method\":\"thing.event.property.post\",\"params\":%s}"
#define ALINK_TOPIC_PROP_POST      "/sys/" PRODUCT_KEY "/" DEVICE_NAME "/thing/event/property/post"

const char* topic="/sys/a1H4kfnF05D/ESP32S3Test/thing/service/property/set";

unsigned long lastMs=0;


Adafruit_SGP30 sgp;

/* return absolute humidity with approximation formula
* @param temperature [°C]
* @param humidity [%RH]
*/
uint32_t getAbsoluteHumidity(float temperature, float humidity) {
    // approximation formula from Sensirion SGP30 Driver Integration chapter 3.15
    const float absoluteHumidity = 216.7f * ((humidity / 100.0f) * 6.112f * exp((17.62f * temperature) / (243.12f + temperature)) / (273.15f + temperature)); //
    const uint32_t absoluteHumidityScaled = static_cast&lt;uint32_t&gt;(1000.0f * absoluteHumidity); //
    return absoluteHumidityScaled;
}

WiFiClient espClient;
PubSubClient client(espClient);

float soil_data;
float tep;

float TVOC_data;
uint32_t CO2_data;


void TVOC_CO2()
{
int counter = 0;
if (! sgp.IAQmeasure()) {
    Serial.println("Measurement failed");
    return;
}
Serial.print("TVOC "); Serial.print(sgp.TVOC); Serial.print(" ppb\t");
Serial.print("eCO2 "); Serial.print(sgp.eCO2); Serial.println(" ppm");

if (! sgp.IAQmeasureRaw()) {
    Serial.println("Raw Measurement failed");
    return;
}
Serial.print("Raw H2 "); Serial.print(sgp.rawH2); Serial.print(" \t");
Serial.print("Raw Ethanol "); Serial.print(sgp.rawEthanol); Serial.println("");

delay(1000);

counter++;
if (counter == 30) {
    counter = 0;

    uint16_t TVOC_base, eCO2_base;
    if (! sgp.getIAQBaseline(&amp;eCO2_base, &amp;TVOC_base)) {
      Serial.println("Failed to get baseline readings");
      return;
    }
    Serial.print("****Baseline values: eCO2: 0x"); Serial.print(eCO2_base, HEX);
    Serial.print(" &amp; TVOC: 0x"); Serial.println(TVOC_base, HEX);
}
}



void wifiInit()
{
WiFi.begin(WIFI_SSID,WIFI_PASSWORD);
while(WiFi.status()!=WL_CONNECTED)
{
    delay(500);
    Serial.println("WiFi not Connect");
}


}


void mqttCheckConnect()
{
while(!client.connected())
{
    Serial.println("Connect to MQTT Server...");
    if(client.connect(CLIENT_ID,MQTT_USERNAME,MQTT_PASSWD))
    {
      Serial.println("MQTT Connected!");
    }
    else{
      Serial.print("MQTT Connect err..");
      Serial.println(client.state());
      delay(5000);
    }


}


}

void mqttIntervalPost()
{
    char param;
    char jsonBuf;
   
    //upload humidity
    soil_data = dht.readHumidity();   
    sprintf(param, "{\"humidity\":%2f}", soil_data);
    sprintf(jsonBuf, ALINK_BODY_FORMAT, param);
    Serial.println(jsonBuf);
    boolean b = client.publish(ALINK_TOPIC_PROP_POST, jsonBuf);
    if(b){
      Serial.println("publish Humidity success");
    }else{
      Serial.println("publish Humidity fail");
    }

    // Upload temperature
    tep =dht.readTemperature();
    sprintf(param, "{\"temperature\":%2f}",tep);
    sprintf(jsonBuf, ALINK_BODY_FORMAT, param);
    Serial.println(jsonBuf);
    boolean c = client.publish(ALINK_TOPIC_PROP_POST, jsonBuf);
    if(c){
      Serial.println("publish Temperature success");
    }else{
      Serial.println("publish Temperature fail");
    }

    // Upload TVOC
    TVOC_data =sgp.TVOC;
    sprintf(param, "{\"tvocCurrent\":%2f}",TVOC_data);
    sprintf(jsonBuf, ALINK_BODY_FORMAT, param);
    Serial.println(jsonBuf);
    boolean d = client.publish(ALINK_TOPIC_PROP_POST, jsonBuf);
    if(d){
      Serial.println("publish tvocCurrent success");
    }else{
      Serial.println("publish tvocCurrent fail");
    }

    // Upload CO2
    CO2_data =sgp.eCO2;
    sprintf(param, "{\"coCurrent\":%2d}",CO2_data);
    sprintf(jsonBuf, ALINK_BODY_FORMAT, param);
    Serial.println(jsonBuf);
    boolean e = client.publish(ALINK_TOPIC_PROP_POST, jsonBuf);
    if(e){
      Serial.println("publish coCurrent success");
    }else{
      Serial.println("publish coCurrent fail");
    }
}

//回调函数
void callback(char* topic, byte* payload, unsigned int length) {
Serial.print("Message arrived [");
Serial.print(topic);   // 打印主题信息
Serial.print("] ");
for (int i = 0; i &lt; length; i++) {
    Serial.print((char)payload); // 打印主题内容
}
Serial.println();

DynamicJsonDocument doc(1024);//创建了一个名为 doc 的动态 JSON 文档
deserializeJson(doc, String((char *)payload));//从一个名为 payload 的数据中解析 JSON 数据并将其填充到 doc 中

// DynamicJsonDocument params = doc["params"];

if(doc["params"].containsKey("deng"))
{
Serial.println("GOT DENG CMD");
digitalWrite(LED, doc["params"]["deng"]);
}

}

void setup() {

pinMode(LED,OUTPUT);

Serial.begin(115200);
dht.begin();
wifiInit();
Wire.begin(SDA,SCL);

client.setServer(MQTT_SERVER, MQTT_PORT);   /* 连接MQTT服务器 */
client.setCallback(callback);

digitalWrite(LED,LOW);

if (! sgp.begin())
{
    while (1);
}
Serial.print("Found SGP30 serial #");
Serial.print(sgp.serialnumber, HEX);
Serial.print(sgp.serialnumber, HEX);
Serial.println(sgp.serialnumber, HEX);



}

void loop() {
TVOC_CO2();
    if (millis() - lastMs &gt;= 5000)
    {
      lastMs = millis();
      mqttCheckConnect();
      /* 上报 */
      mqttIntervalPost();
    }
    client.loop();
    delay(2000);


}</code></pre>

<p>&nbsp;</p>

Jacktang 发表于 2024-10-23 07:29

<p>这次测试增加了SGP30,进行TVOC和CO2检测,看起来是成功的</p>

左手阿飞 发表于 2024-10-23 17:15

Jacktang 发表于 2024-10-23 07:29
这次测试增加了SGP30,进行TVOC和CO2检测,看起来是成功的

<p>重点在声源定位,能找到的例程不多,还需要努力。</p>
页: [1]
查看完整版本: 【2024 DigiKey 创意大赛】会议声源追踪器 会议室环境检测(2)