小鹏浇花套件单机版程序V1.0.0

本程序是小鹏物联网智能浇花套件的单机版程序(不连接物联网),供同学们参考。

需求

1. 通过土壤湿度传感器收集湿度数据;

2. 根据湿度数据决定是否浇水;

3. 通过OLED显示相关信息;

4. 可手动进行浇水(不考虑土壤湿度);

硬件

小鹏物联网智能浇花套件

思路

1.需要使用多任务来实现,本程序使用ESP32 FreeRTOS

2.在主循环内读取土壤湿度数据并放入队列(任务0)

3.在数据接收任务内接收队列数据,并判断是否要浇水(任务1)

4.使用一个任务来处理按键(任务2)

5.使用一个任务来处理OLED显示(任务3)

完整程序

/**********************************************************************
 **  小鹏浇花系统
 * 
 **  version: 1.0.0
 **  author: 老张
 **  homesite: www.xpstem.com/product/intelligent-watering/index
 **********************************************************************/
#include <Arduino.h>
#include <SSD1306Wire.h>    // 4.6.1 (ThingPulse)
#include <OneButton.h>      // 2.6.1
#include <ArrayList.h>      // 1.0.2

//************** 常数数据 开始 **********************
// Pins
int soil_moisture_sensor_pin = 27;      // 土壤湿度传感器引脚
int pump_control_relay_pin = 15;        // 水泵继电器引脚
int oled_display_pin_scl = 22;          // OLED显示屏I2C引脚
int oled_display_pin_sda = 21;          // OLED显示屏I2C引脚
int oled_display_i2c_addr = 61;         // OLED显示屏I2C地址
int manual_button_pin = 26;             // 手动按钮引脚

//************* 配置变量 结束 *********************

//************* 运行时变量 开始 ********************
OneButton manualBtn(manual_button_pin, true);

int stateCode = 0; //1:config, 2:work, 3:sleep, 9:error
String stateInfo = "";

int collectCount = 0;
int sensorVal = 0;
int Threshold = 50;  // 湿度阀值

ArrayList<int> valueList; // 湿度值列表

QueueHandle_t msgQueue = xQueueCreate(10, sizeof(int)); // 消息队列

//************* 运行时变量 结束 *********************

/***********************************************************
 ***  控制水泵继电器
 **********************************************************/ 
void main_watering(int relayPin, int seconds) {
  pinMode(relayPin, OUTPUT);
  
  digitalWrite(relayPin, HIGH);
  Serial.printf("打开水泵继电器,引脚:%d,保持 %d 秒。\n", relayPin, seconds);
  
  for (int i=0; i<seconds; i++) { 
    Serial.print("."); 
    delay(1000);
  }
  Serial.println("");
        
  digitalWrite(relayPin, LOW);
  Serial.println("关闭水泵继电器。");
}

/***********************************************************
 ***  UI任务
 **********************************************************/
void task_display(void *pvParam) {

  SSD1306Wire display(/* address */ oled_display_i2c_addr, 
    /* sda */ oled_display_pin_sda, 
    /* scl */ oled_display_pin_scl); 

  display.init();
  //display.flipScreenVertically();
  display.setFont(ArialMT_Plain_16);
  
  while(1) {
    display.clear();
    if (stateCode==1) {
      display.drawString(0, 0, "[config]");
      
    } else if (stateCode==2) {
      display.drawString(0, 0, "[work]" );
      display.drawString(0, 20, "value: " + (sensorVal==0 ? "" : String(sensorVal)) );
      if (stateInfo.length()>0) {
        display.drawString(0, 40, stateInfo );
      }
      
    } else if (stateCode==3) {
      display.drawString(0, 0, "Zzz..." );
      
    } else if (stateCode==9) {
      display.drawString(0, 0, "[error]");
      display.drawString(0, 20, stateInfo );
    }
    display.display();
    
    vTaskDelay(1000 / portTICK_PERIOD_MS);
  };
}

/******************************************************** 
 ***  Button任务
 ********************************************************/
void task_buttonTick(void *pvParam) {
  while(1) {
    manualBtn.tick();
    vTaskDelay(10 / portTICK_PERIOD_MS); //10ms
  }
}

/******************************************************** 
 ***  数据接收任务
 ********************************************************/
void task_dataReceive(void *pvParam) {
  while (1) {
    int received;
    // 从队列接收数据
    if (xQueueReceive(msgQueue, &received, 100/portTICK_PERIOD_MS) == pdTRUE) {
      valueList.add(received);

      int flag = 0;
      if (valueList.size() > 5) {
        // 检查是否全部低于阀值
        for (int i=0; i<valueList.size(); i++) {
          if (valueList.get(i) > Threshold) { // 超过阀值
            flag = 1;
            break;
          }
        }

        if (flag == 0) {
          // 全部低于阀值,则浇水一次
          main_watering(pump_control_relay_pin, 5);
        }
        
        valueList.remove(0);
      }

    }
  }
}

/***********************************************************
 ***  进入休眠状态
 **********************************************************/
void main_deepSleep(int sleepSeconds) {
  stateCode = 3;
  delay(3000); //3s for ui update
  
  //digitalWrite(power_led_pin, LOW);

  Serial.printf("进入深度休眠状态(%d%s)。\n", 
    sleepSeconds<3600 ? sleepSeconds/60 : sleepSeconds/3600,
    sleepSeconds<3600 ? "分钟" : "小时");
  ESP.deepSleep(sleepSeconds * 1e6);
}

/***********************************************************
 ***  收集传感器数据
 **********************************************************/
void main_collectData(int sensorPin) {
  Serial.printf("获取土壤湿度数据,引脚:%d\n", sensorPin);
  int originVal = analogRead(sensorPin);
  if (originVal > 0) {
    sensorVal = map(originVal, 1, 4095, 100, 1);
    // 加入队列
    xQueueSend(msgQueue, &sensorVal, portMAX_DELAY);
    
  } else { 
    // 数据无效,不做处理。
  }
}

/***********************************************************
 ***  初始化设置
 **********************************************************/
void setup() {
  Serial.begin(115200);
  Serial.println();
  Serial.println("-------------------------------");
  Serial.println("--------- 小鹏浇花系统 ----------");
  Serial.println("-------------------------------");

  Serial.println("启动UI任务程序。");
  // UI任务
  xTaskCreate(
    task_display,     // 任务函数
    "Display_Task",  // 任务名称
    4096,       // 堆栈大小(字节)
    NULL,       // 参数
    1,          // 优先级
    NULL        // 任务句柄
  );

  Serial.println("启动主程序。");
  
  stateCode = 2;

  // 等传感器稳定。
  Serial.print("等待传感器初始化(3sec).");
  for (int i=0;i<6;i++) {
    Serial.print(".");
    delay(500);
  }
  Serial.println("");

  // 手动按钮
  manualBtn.attachDoubleClick([]() {
    main_watering(pump_control_relay_pin, 5);
  });
  
  xTaskCreate(task_buttonTick, "ButtonTick_Task", 2048, NULL, 1, NULL);
  
  xTaskCreate(task_dataReceive, "DataReceive_Task", 2048, NULL, 1, NULL);
}

/***********************************************************
 ***  主循环
 ***  1.获取传感器数据并发送
 **********************************************************/
void loop() {

  // 采集数据
  main_collectData(soil_moisture_sensor_pin);
  collectCount++;
  
  vTaskDelay(120000 / portTICK_PERIOD_MS); //120s

  if (collectCount > 30) {
    main_deepSleep(23 * 3600); //23小时
  }
  
} 
- 本文为本站原创文章,转载请保留出处。
- 文章链接:https://www.xpstem.com/article/2000406

08-22   阅读(1)   评论(0)
 标签: 创客 Arduino ESP32 FreeRTOS

涨知识
二进制

二进制(binary),发现者莱布尼茨,是在数学和数字电路中以2为基数的记数系统,是以2为基数代表系统的二进位制。这一系统中,通常用两个不同的符号0(代表零)和1(代表一)来表示 。

评论:
相关文章
Arduino 数据结构: ArrayList

ArrayList 类是一个 C++ 模板类,它提供了 ArrayList 的实现,以便轻松存储任何指定类型的值。它允许使用索引进行高效存储和检索,支持排序操作。


ESP32 的中断机制和处理

本文介绍ESP32中的中断机制,以及如何通过GPIO中断实现按钮控制。重点讲解了如何设置中断服务例程、处理中断抖动问题,并提供了消除中断抖动的示例代码。


在ESP32上实现WEB交互界面

本文主要介绍在未联网(AP热点)情况下实现WEB交互界面的CSS和javascript库。


Arduino ESP32获取芯片、RAM信息

本文介绍如何使用Arduino-ESP32库中的API函数获取ESP32的芯片、RAM信息等,并提供了一个示例程序代码。


ESP32 FreeRTOS 双核使用

ESP32系列(包括ESP32-S3)搭载Xtensa双核处理器,默认情况下Arduino框架仅使用单核运行用户代码,通过多核编程,可以充分利用硬件资源来提升系统响应和性能。


ESP32 GPIO 矩阵和引脚多路复用

ESP32 芯片有34个物理GPIO管脚。每个GPIO管脚都可用作一个通用IO,或连接一个内部的外设信号。IO_MUX ¹、RTC IO MUX 和GPIO交换矩阵用于将信号从外设传输至GPIO管脚。


理解 MSBFIRST(最高有效位)和 LSBFIRST(最低有效位)

在本文中,先解释 MSB(最高有效位)和 LSB(最低有效位)的概念,以及 MSBFIRST 和 LSBFIRST。然后展示了 MSBFIRST 和 LSBFIRST 的使用如何影响移位寄存器的输出。


ESP32Encoder:高效的ESP32旋转编码器库

ESP32Encoder库是一个利用ESP32脉冲计数器硬件外设实现高效旋转编码器读取的软件库。


适合学习物联网的几款盒子

本文对比了几款适合物联网开发的盒子硬件参数,供大家参考。


乐动掌控

乐动掌控采用掌控板作为主控,塑胶一体式外壳,侧面和底面开具多个乐高扩展孔位,兼容乐高积木,可完成多种创意应用。

搜索
小鹏STEM教研服务

专属教研服务系统,助您构建STEM课程体系,打造一站式教学环境。