Files
SerreklimaatSensor/main/main.c

397 lines
12 KiB
C

#include <stdio.h>
#include <inttypes.h>
#include <locale.h>
#include <time.h>
#include <string.h>
#include <sys/time.h>
#include "esp_spiffs.h"
#include "sdkconfig.h"
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
#include "freertos/semphr.h"
#include "esp_chip_info.h"
#include "esp_flash.h"
#include "esp_system.h"
#include "esp_timer.h"
#include "driver/gpio.h"
#define PRINT_DEBUG false
#define DHT11_GPIO 0
#define BUTTON_PIN 9 // BOOT button on ESP32-C3 SuperMini
#define DATA_BUFFER_SIZE 100
#define MAX_MEASUREMENTS_PER_DAY 96 // 96 * 15 minutes = 24 hours
static int measurement_count = 0;
QueueHandle_t sensor_data_queue;
SemaphoreHandle_t xMutex;
typedef struct {
char timestamp[64];
float humidity;
float temperature;
} SensorData;
TaskHandle_t file_handler_task_handle;
// ############### Base Setup ###############
void set_system_time_from_compiler() {
// Parse __DATE__: "Mmm dd yyyy"
char date[] = __DATE__;
char time[] = __TIME__;
struct tm tm = {0};
char month[4];
int day, year;
int hour, min, sec;
// Parse date
sscanf(date, "%3s %d %d", month, &day, &year);
// Parse time
sscanf(time, "%d:%d:%d", &hour, &min, &sec);
// Convert month abbreviation to number
const char *months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun",
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
for (int i = 0; i < 12; i++) {
if (strcmp(month, months[i]) == 0) {
tm.tm_mon = i;
break;
}
}
tm.tm_mday = day;
tm.tm_year = year - 1900; // Years since 1900
tm.tm_hour = hour;
tm.tm_min = min;
tm.tm_sec = sec;
tm.tm_isdst = -1; // Let system determine daylight saving time
// Set system time
struct timeval tv = { .tv_sec = mktime(&tm), .tv_usec = 0 };
settimeofday(&tv, NULL);
}
void print_chip_info(){
/* Print chip information */
#if PRINT_DEBUG
esp_chip_info_t chip_info;
uint32_t flash_size;
esp_chip_info(&chip_info);
printf("This is %s chip with %d CPU core(s), %s%s%s%s, ",
CONFIG_IDF_TARGET,
chip_info.cores,
(chip_info.features & CHIP_FEATURE_WIFI_BGN) ? "WiFi/" : "",
(chip_info.features & CHIP_FEATURE_BT) ? "BT" : "",
(chip_info.features & CHIP_FEATURE_BLE) ? "BLE" : "",
(chip_info.features & CHIP_FEATURE_IEEE802154) ? ", 802.15.4 (Zigbee/Thread)" : "");
unsigned major_rev = chip_info.revision / 100;
unsigned minor_rev = chip_info.revision % 100;
printf("silicon revision v%d.%d, ", major_rev, minor_rev);
if(esp_flash_get_size(NULL, &flash_size) != ESP_OK) {
printf("Get flash size failed");
return;
}
printf("%" PRIu32 "MB %s flash\n", flash_size / (uint32_t)(1024 * 1024),
(chip_info.features & CHIP_FEATURE_EMB_FLASH) ? "embedded" : "external");
printf("Minimum free heap size: %" PRIu32 " bytes\n", esp_get_minimum_free_heap_size());
#endif
}
// ############### Measurement task ###############
void get_current_time(char* time_string, size_t size) {
time_t now;
struct tm timeinfo;
time(&now);
localtime_r(&now, &timeinfo);
strftime(time_string, size, "%Y-%m-%d %H:%M:%S", &timeinfo);
}
void read_dht11(float *humidity, float *temperature) {
uint8_t data[5] = {0};
// Send start signal: pull the line low for 20ms
gpio_set_direction(DHT11_GPIO, GPIO_MODE_OUTPUT);
gpio_set_level(DHT11_GPIO, 0);
vTaskDelay(pdMS_TO_TICKS(20));
// Release the line and wait for sensor response
gpio_set_level(DHT11_GPIO, 1);
gpio_set_direction(DHT11_GPIO, GPIO_MODE_INPUT);
// Wait for the sensor to pull the line low (~20-40us)
while (gpio_get_level(DHT11_GPIO) == 1) {
vTaskDelay(pdMS_TO_TICKS(0.1));
}
// Wait for the sensor to pull the line high (~80us)
while (gpio_get_level(DHT11_GPIO) == 0) {
vTaskDelay(pdMS_TO_TICKS(0.1));
}
// Read 40 bits of data
for (int i = 0; i < 40; i++) {
// Wait for the start of the bit (low for ~50us)
while (gpio_get_level(DHT11_GPIO) == 1) {
vTaskDelay(pdMS_TO_TICKS(0.1));
}
while (gpio_get_level(DHT11_GPIO) == 0) {
vTaskDelay(pdMS_TO_TICKS(0.1));
}
// Measure the length of the high pulse
uint32_t pulse_start = esp_timer_get_time();
while (gpio_get_level(DHT11_GPIO) == 1) {
vTaskDelay(pdMS_TO_TICKS(0.1));
}
uint32_t pulse_end = esp_timer_get_time();
// Determine if the bit is 0 or 1
uint8_t bit = (pulse_end - pulse_start) > 50 ? 1 : 0;
data[i / 8] <<= 1;
data[i / 8] |= bit;
}
// Parse the data
uint8_t humidity_int = data[0];
uint8_t humidity_dec = data[1]; // Decimal part (0-9)
int8_t temperature_int = (data[2]); // signed value
uint8_t temperature_dec = data[3]; // Decimal part (0-9)
uint8_t checksum = data[4];
#if PRINT_DEBUG
printf("\nReadout values: [hi: %d, hd: %d, ti %d, td %d, c: %d]\n", humidity_int, humidity_dec, temperature_int, temperature_dec, checksum);
#endif
// Validate checksum
uint8_t sum = humidity_int + humidity_dec + temperature_int + temperature_dec;
if (sum != checksum) {
#if PRINT_DEBUG
printf("Checksum failed! Retrying...\n");
#endif
vTaskDelay(pdMS_TO_TICKS(1000));
read_dht11(humidity, temperature);
}
#if PRINT_DEBUG
printf("Checksum checks out!\n");
#endif
// Calculate humidity and temperature
*humidity = (float)humidity_int + (float)humidity_dec / 10.0f;
*temperature = (float)temperature_int + (float)temperature_dec / 10.0f;
}
void measure_task(void *pvParameters){
while (1) {
float humidity, temperature;
read_dht11(&humidity, &temperature);
char time_string[64] = {0};
get_current_time(time_string, sizeof(time_string));
#if PRINT_DEBUG
printf("Measurment: [%s, %.1f%%, %.1f°C]\n", time_string, humidity, temperature);
#endif
SensorData data;
strncpy(data.timestamp, time_string, sizeof(data.timestamp));
data.humidity = humidity;
data.temperature = temperature;
// Send data to the queue
if (xQueueSend(sensor_data_queue, &data, pdMS_TO_TICKS(100)) != pdPASS) {
#if PRINT_DEBUG
printf("Failed to add measurement to queue!\n");
#endif
} else {
measurement_count++;
if (measurement_count >= MAX_MEASUREMENTS_PER_DAY) {
// Signal the file handler to write and reset
xTaskNotifyGive(file_handler_task_handle);
}
}
// Wait 15 minute before next read
vTaskDelay(pdMS_TO_TICKS(15 * 60 * 1000));
}
}
// ############### Filehandling task ###############
void init_filesystem() {
esp_vfs_spiffs_conf_t conf = {
.base_path = "/spiffs",
.partition_label = "storage",
.max_files = 5,
.format_if_mount_failed = true
};
esp_err_t ret = esp_vfs_spiffs_register(&conf);
if (ret != ESP_OK) {
#if PRINT_DEBUG
if (ret == ESP_FAIL) {
printf("Failed to mount or format filesystem\n");
} else if (ret == ESP_ERR_NOT_FOUND) {
printf("Failed to find SPIFFS partition\n");
} else {
printf("Failed to initialize SPIFFS (%s)\n", esp_err_to_name(ret));
}
#endif
return;
}
#if PRINT_DEBUG
printf("SPIFFS initialized successfully\n");
#endif
}
void file_handler_task(void *pvParameters) {
while (1) {
// Wait for notification from measurement task
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
SensorData data;
FILE *f = NULL;
xSemaphoreTake(xMutex, portMAX_DELAY);
f = fopen("/spiffs/measurements.csv", "a");
if (f == NULL) {
#if PRINT_DEBUG
printf("Failed to open CSV file!\n");
#endif
xSemaphoreGive(xMutex);
continue;
}
// Write header if file is empty
fseek(f, 0, SEEK_END);
if (ftell(f) == 0) {
fprintf(f, "Timestamp,Humidity,Temperature\n");
}
// Read all data from the queue and write to CSV
while (xQueueReceive(sensor_data_queue, &data, 0) == pdPASS) {
fprintf(f, "%s,%.1f,%.1f\n", data.timestamp, data.humidity, data.temperature);
#if PRINT_DEBUG
printf("%s,%.1f,%.1f\n", data.timestamp, data.humidity, data.temperature);
#endif
}
fclose(f);
measurement_count = 0;
#if PRINT_DEBUG
printf("CSV file updated!\n");
#endif
xSemaphoreGive(xMutex);
}
}
// ############### Button task ###############
void button_task(void *pvParameters) {
// Configure BOOT button (GPIO9, active low with internal pullup)
gpio_set_direction(BUTTON_PIN, GPIO_MODE_INPUT);
gpio_set_pull_mode(BUTTON_PIN, GPIO_PULLUP_ONLY);
uint32_t last_press_tick = 0;
bool waiting_for_double = false;
while (1) {
if (gpio_get_level(BUTTON_PIN) == 0) { // Button pressed (active low)
vTaskDelay(pdMS_TO_TICKS(50)); // Debounce
if (gpio_get_level(BUTTON_PIN) == 0) {
uint32_t press_start = xTaskGetTickCount();
// Wait for release
while (gpio_get_level(BUTTON_PIN) == 0) {
vTaskDelay(pdMS_TO_TICKS(10));
}
vTaskDelay(pdMS_TO_TICKS(50)); // Debounce release
uint32_t press_duration = xTaskGetTickCount() - press_start;
// Only count short presses (< 500ms)
if (press_duration < pdMS_TO_TICKS(500)) {
if (waiting_for_double) {
// ===== DOUBLE PRESS: Reset CSV =====
xSemaphoreTake(xMutex, portMAX_DELAY);
FILE *f = fopen("/spiffs/measurements.csv", "w");
if (f) {
fprintf(f, "Timestamp,Humidity,Temperature\n");
fclose(f);
}
xSemaphoreGive(xMutex);
waiting_for_double = false;
} else {
// ===== FIRST PRESS: Wait for possible double press =====
waiting_for_double = true;
last_press_tick = press_start;
}
}
}
} else {
// Check if double-press window expired
if (waiting_for_double && (xTaskGetTickCount() - last_press_tick) > pdMS_TO_TICKS(500)) {
xTaskNotifyGive(file_handler_task_handle);
// ===== SINGLE PRESS: Print CSV contents (via USB CDC) =====
xSemaphoreTake(xMutex, portMAX_DELAY);
FILE *f = fopen("/spiffs/measurements.csv", "r");
if (f) {
char line[256];
while (fgets(line, sizeof(line), f)) {
printf("%s", line); // Goes to USB CDC (/dev/ttyACM0)
}
fclose(f);
}
xSemaphoreGive(xMutex);
waiting_for_double = false;
}
}
vTaskDelay(pdMS_TO_TICKS(10));
}
}
// ############### Shared buffer ###############
void init_shared_buffer() {
sensor_data_queue = xQueueCreate(DATA_BUFFER_SIZE, sizeof(SensorData));
xMutex = xSemaphoreCreateMutex();
if (sensor_data_queue == NULL || xMutex == NULL) {
#if PRINT_DEBUG
printf("Failed to create queue or mutex!\n");
#endif
}
}
// ############### Main thread ###############
void app_main(void)
{
// Setup
setlocale(LC_ALL, "C"); // Use ASCII for text
set_system_time_from_compiler();
init_shared_buffer();
init_filesystem();
// Intro
#if PRINT_DEBUG
printf("DHT11 temperature and humidity sensor\n");
char start_time[64] = {0};
get_current_time(start_time, sizeof(start_time));
printf("Program started at: %s\n", start_time);
print_chip_info();
printf("\n\n=============\n\n");
#endif
// Run tasks
xTaskCreate(measure_task, "MeasureTask", 4096, NULL, 3, NULL);
xTaskCreate(file_handler_task, "FileHandlerTask", 4096, NULL, 4, &file_handler_task_handle);
xTaskCreate(button_task, "ButtonTask", 2048, NULL, 2, NULL);
}