Files
ESP-Nodes/ESP-IDF_Robot_RC/main/espnow_example_main.c

150 lines
5.0 KiB
C
Executable File

/* ESP-NOW Remote Controller & Receiver
by: Alexander Bobkov
Jan 4, 2025
Program that sends values saved in struct from controller device to the receiver using ESP-NOW communication protocol.
*/
#include <stdlib.h>
#include <time.h>
#include <string.h>
#include <assert.h>
#include "freertos/FreeRTOS.h"
#include "freertos/semphr.h"
#include "freertos/timers.h"
#include "nvs_flash.h"
#include "esp_random.h"
#include "esp_event.h"
#include "esp_netif.h"
#include "esp_wifi.h"
#include "esp_log.h"
#include "esp_mac.h"
#include "esp_now.h"
#include "esp_crc.h"
//#define ESPNOW_MAXDELAY 512
// Struct holding sensors values
typedef struct {
uint16_t crc; // CRC16 value of ESPNOW data
uint8_t x_axis; // Joystick x-position
uint8_t y_axis; // Joystick y-position
bool nav_bttn; // Joystick push button
uint8_t motor1_rpm_pcm; // PCMs for 4 motors
uint8_t motor2_rpm_pcm;
uint8_t motor3_rpm_pcm;
uint8_t motor4_rpm_pcm;
} __attribute__((packed)) sensors_data_t;
// MAC address of receiver. For one-to-many broadcast, change MAC address to FF:FF...:FF
static uint8_t receiver_mac[ESP_NOW_ETH_ALEN] = {0xE4, 0xB0, 0x63, 0x17, 0x9E, 0x45};
static esp_now_peer_info_t peerInfo; // ESP-NOW pointer holding info about devices (peers)
static sensors_data_t *buf; // Pointer to the struct buffer holding data being sent.
static const char *TAG = "Remote Controller";
/* WiFi is required to run ESPNOW */
static void wifi_init(void)
{
ESP_ERROR_CHECK(esp_netif_init());
ESP_ERROR_CHECK(esp_event_loop_create_default());
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
ESP_ERROR_CHECK( esp_wifi_init(&cfg) );
ESP_ERROR_CHECK( esp_wifi_set_storage(WIFI_STORAGE_RAM) ); // Keep configurations in RAM
ESP_ERROR_CHECK( esp_wifi_set_mode(WIFI_MODE_STA)); // Do not change WiFi device mode
ESP_ERROR_CHECK( esp_wifi_start());
ESP_ERROR_CHECK( esp_wifi_set_channel(CONFIG_ESPNOW_CHANNEL, WIFI_SECOND_CHAN_NONE)); // Both sender & receiver must be on the same channel
}
// Call-back for the event when data is being received
void onDataReceived (uint8_t *mac_addr, uint8_t *data, uint8_t data_len) {
//memcpy(buf, data, data_len);
buf = (sensors_data_t*)data;
ESP_LOGW(TAG, "Data was received");
ESP_LOGI(TAG, "x-axis: 0x%04x", buf->x_axis);
ESP_LOGI(TAG, "x-axis: 0x%04x", buf->y_axis);
ESP_LOGI(TAG, "PCM 1: 0x%04x", buf->motor1_rpm_pcm);
}
void onDataSent (uint8_t *mac_addr, esp_now_send_status_t status) {
//status == ESP_NOW_SEND_SUCCESS ? "Delivery Success" : "Delivery Fail");
ESP_LOGW(TAG, "Packet send status: 0x%04X", status);
}
void deletePeer (void) {
uint8_t delStatus = esp_now_del_peer(receiver_mac);
if (delStatus != 0) {
ESP_LOGE("ESP-NOW", "Could not delete peer");
}
}
void sendData (void) {
// Send data, specify receiver MAC address, pointer to the data being sent, and length of data being sent.
sensors_data_t buffer;
buffer.type = 1;
buffer.crc = 0;
buffer.x_axis = 240;
buffer.y_axis = 256;
buffer.nav_bttn = 0;
buffer.motor1_rpm_pcm = 10;
buffer.motor2_rpm_pcm = 0;
buffer.motor3_rpm_pcm = 0;
buffer.motor4_rpm_pcm = 0;
ESP_LOGI(TAG, "Joystick (x,y) position ( 0x%04X, 0x%04X )", (uint8_t)buffer.x_axis, (uint8_t)buffer.y_axis);
ESP_LOGI(TAG, "pcm 1, pcm 2 [ 0x%04X, 0x%04X ]", (uint8_t)buffer.motor1_rpm_pcm, (uint8_t)buffer.motor2_rpm_pcm);
ESP_LOGI(TAG, "pcm 3, pcm 4 [ 0x%04X, 0x%04X ]", (uint8_t)buffer.motor3_rpm_pcm, (uint8_t)buffer.motor4_rpm_pcm);
//uint8_t result = esp_now_send(receiver_mac, &flagToSend, sizeof(flagToSend));
uint8_t result = esp_now_send(receiver_mac, &buffer, sizeof(buffer));
if (result != 0) {
ESP_LOGE("ESP-NOW", "Error sending data! Error code: 0x%04X", result);
deletePeer();
}
else
ESP_LOGW("ESP-NOW", "Data was sent.");
}
static void rc_send_data_task (void *arg) {
while (true) {
if (esp_now_is_peer_exist(receiver_mac)) {
sendData();
}
vTaskDelay (1000 / portTICK_PERIOD_MS);
}
}
void app_main(void)
{
// Initialize NVS to store Wi-Fi configurations
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 );
wifi_init();
esp_now_init();
esp_now_register_recv_cb(onDataReceived);
esp_now_register_send_cb(onDataSent);
memcpy (peerInfo.peer_addr, receiver_mac, 6);
peerInfo.channel = 1;
peerInfo.encrypt = false;
esp_now_add_peer(&peerInfo);
/*if (esp_now_is_peer_exist(receiver_mac)) {
ESP_LOGI("ESP-NOW", "Receiver exists.");
sendData();
}
else
ESP_LOGE("ESP-NOW", "Receiver does not exists.");*/
xTaskCreate (rc_send_data_task, "RC", 2048, NULL, 15, NULL);
}