mch2022-talktopics/main/main.c

433 lines
16 KiB
C
Raw Normal View History

#include <stdio.h>
2022-01-01 04:51:12 +00:00
#include <string.h>
#include <sdkconfig.h>
#include <freertos/FreeRTOS.h>
#include <freertos/task.h>
2022-04-08 21:15:57 +00:00
#include <freertos/queue.h>
#include <esp_system.h>
#include <esp_err.h>
#include <esp_log.h>
#include <nvs_flash.h>
#include <nvs.h>
#include "hardware.h"
#include "managed_i2c.h"
#include "pax_gfx.h"
#include "sdcard.h"
#include "appfs.h"
2022-01-11 23:53:38 +00:00
#include "driver_framebuffer.h"
2022-01-24 21:32:33 +00:00
#include "rp2040.h"
2022-04-08 21:15:57 +00:00
#include "fpga_test.h"
#include "menu.h"
2022-04-15 23:16:36 +00:00
#include "system_wrapper.h"
#include "graphics_wrapper.h"
#include "appfs_wrapper.h"
2022-04-16 00:14:28 +00:00
#include "settings.h"
2022-04-18 00:10:44 +00:00
#include "wifi_connection.h"
2022-02-04 00:04:51 +00:00
2022-04-23 02:00:55 +00:00
#include "ws2812.h"
static const char *TAG = "main";
typedef enum action {
ACTION_NONE,
2022-04-08 21:15:57 +00:00
ACTION_APPFS,
2022-02-04 00:04:51 +00:00
ACTION_INSTALLER,
2022-04-15 23:16:36 +00:00
ACTION_SETTINGS,
ACTION_OTA,
2022-04-16 00:14:28 +00:00
ACTION_FPGA,
ACTION_WIFI_SCAN,
ACTION_WIFI_MANUAL,
ACTION_WIFI_LIST,
ACTION_BACK
} menu_action_t;
2022-04-08 21:15:57 +00:00
typedef struct _menu_args {
appfs_handle_t fd;
menu_action_t action;
2022-04-08 21:15:57 +00:00
} menu_args_t;
2022-01-23 00:27:22 +00:00
2022-04-15 23:16:36 +00:00
void menu_launcher(xQueueHandle buttonQueue, pax_buf_t* pax_buffer, ILI9341* ili9341, uint8_t* framebuffer, menu_action_t* menu_action, appfs_handle_t* appfs_fd) {
menu_t* menu = menu_alloc("Main menu");
*appfs_fd = APPFS_INVALID_FD;
*menu_action = ACTION_NONE;
while (1) {
*appfs_fd = appfsNextEntry(*appfs_fd);
if (*appfs_fd == APPFS_INVALID_FD) break;
const char* name = NULL;
appfsEntryInfo(*appfs_fd, &name, NULL);
menu_args_t* args = malloc(sizeof(menu_args_t));
args->fd = *appfs_fd;
args->action = ACTION_APPFS;
menu_insert_item(menu, name, NULL, (void*) args, -1);
}
2022-04-15 23:16:36 +00:00
*appfs_fd = APPFS_INVALID_FD;
2022-04-15 23:16:36 +00:00
menu_args_t* install_args = malloc(sizeof(menu_args_t));
install_args->action = ACTION_INSTALLER;
menu_insert_item(menu, "Hatchery", NULL, install_args, -1);
2022-04-15 23:16:36 +00:00
menu_args_t* settings_args = malloc(sizeof(menu_args_t));
settings_args->action = ACTION_SETTINGS;
2022-04-16 00:14:28 +00:00
menu_insert_item(menu, "WiFi settings", NULL, settings_args, -1);
2022-04-15 23:16:36 +00:00
menu_args_t* ota_args = malloc(sizeof(menu_args_t));
ota_args->action = ACTION_OTA;
menu_insert_item(menu, "Firmware update", NULL, ota_args, -1);
2022-04-15 23:16:36 +00:00
menu_args_t* fpga_args = malloc(sizeof(menu_args_t));
fpga_args->action = ACTION_FPGA;
2022-04-16 00:14:28 +00:00
menu_insert_item(menu, "FPGA test", NULL, fpga_args, -1);
2022-04-16 00:14:28 +00:00
bool render = true;
menu_args_t* menuArgs = NULL;
while (1) {
2022-04-23 02:00:55 +00:00
rp2040_input_message_t buttonMessage = {0};
2022-04-16 00:14:28 +00:00
if (xQueueReceive(buttonQueue, &buttonMessage, 16 / portTICK_PERIOD_MS) == pdTRUE) {
2022-04-23 02:00:55 +00:00
uint8_t pin = buttonMessage.input;
2022-04-16 00:14:28 +00:00
bool value = buttonMessage.state;
switch(pin) {
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_JOYSTICK_DOWN:
2022-04-16 00:14:28 +00:00
if (value) {
menu_navigate_next(menu);
render = true;
}
break;
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_JOYSTICK_UP:
2022-04-16 00:14:28 +00:00
if (value) {
menu_navigate_previous(menu);
render = true;
}
break;
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_BUTTON_ACCEPT:
2022-04-16 00:14:28 +00:00
if (value) {
menuArgs = menu_get_callback_args(menu, menu_get_position(menu));
}
break;
default:
break;
}
}
if (render) {
graphics_task(pax_buffer, ili9341, framebuffer, menu, NULL);
render = false;
}
if (menuArgs != NULL) {
*appfs_fd = menuArgs->fd;
*menu_action = menuArgs->action;
break;
}
2022-04-15 23:16:36 +00:00
}
2022-04-16 00:14:28 +00:00
for (size_t index = 0; index < menu_get_length(menu); index++) {
free(menu_get_callback_args(menu, index));
}
menu_free(menu);
}
void menu_wifi_settings(xQueueHandle buttonQueue, pax_buf_t* pax_buffer, ILI9341* ili9341, uint8_t* framebuffer, menu_action_t* menu_action) {
menu_t* menu = menu_alloc("WiFi settings");
*menu_action = ACTION_NONE;
menu_args_t* wifi_scan_args = malloc(sizeof(menu_args_t));
wifi_scan_args->action = ACTION_WIFI_SCAN;
menu_insert_item(menu, "Add by scan...", NULL, wifi_scan_args, -1);
menu_args_t* wifi_manual_args = malloc(sizeof(menu_args_t));
wifi_manual_args->action = ACTION_WIFI_MANUAL;
menu_insert_item(menu, "Add manually...", NULL, wifi_manual_args, -1);
menu_args_t* wifi_list_args = malloc(sizeof(menu_args_t));
wifi_list_args->action = ACTION_WIFI_LIST;
menu_insert_item(menu, "List known networks", NULL, wifi_list_args, -1);
menu_args_t* back_args = malloc(sizeof(menu_args_t));
back_args->action = ACTION_BACK;
menu_insert_item(menu, "< Back", NULL, back_args, -1);
2022-01-24 21:32:33 +00:00
2022-04-15 23:16:36 +00:00
bool render = true;
menu_args_t* menuArgs = NULL;
2022-04-15 23:16:36 +00:00
while (1) {
2022-04-23 02:00:55 +00:00
rp2040_input_message_t buttonMessage = {0};
2022-04-15 23:16:36 +00:00
if (xQueueReceive(buttonQueue, &buttonMessage, 16 / portTICK_PERIOD_MS) == pdTRUE) {
2022-04-23 02:00:55 +00:00
uint8_t pin = buttonMessage.input;
2022-04-15 23:16:36 +00:00
bool value = buttonMessage.state;
switch(pin) {
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_JOYSTICK_DOWN:
2022-04-15 23:16:36 +00:00
if (value) {
menu_navigate_next(menu);
render = true;
}
break;
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_JOYSTICK_UP:
2022-04-15 23:16:36 +00:00
if (value) {
menu_navigate_previous(menu);
render = true;
}
break;
2022-04-23 02:00:55 +00:00
case RP2040_INPUT_BUTTON_ACCEPT:
2022-04-15 23:16:36 +00:00
if (value) {
menuArgs = menu_get_callback_args(menu, menu_get_position(menu));
}
break;
default:
break;
}
}
2022-04-15 23:16:36 +00:00
if (render) {
graphics_task(pax_buffer, ili9341, framebuffer, menu, NULL);
render = false;
}
if (menuArgs != NULL) {
*menu_action = menuArgs->action;
break;
}
2022-01-06 00:12:28 +00:00
}
2022-01-23 00:27:22 +00:00
2022-04-15 23:16:36 +00:00
for (size_t index = 0; index < menu_get_length(menu); index++) {
free(menu_get_callback_args(menu, index));
2022-01-06 00:12:28 +00:00
}
2022-04-15 23:16:36 +00:00
menu_free(menu);
}
void app_main(void) {
esp_err_t res;
2022-04-15 23:16:36 +00:00
/* Initialize memory */
uint8_t* framebuffer = heap_caps_malloc(ILI9341_BUFFER_SIZE, MALLOC_CAP_8BIT);
if (framebuffer == NULL) {
ESP_LOGE(TAG, "Failed to allocate framebuffer");
restart();
}
memset(framebuffer, 0, ILI9341_BUFFER_SIZE);
2022-04-15 23:16:36 +00:00
pax_buf_t* pax_buffer = malloc(sizeof(pax_buf_t));
2022-02-04 00:04:51 +00:00
if (framebuffer == NULL) {
ESP_LOGE(TAG, "Failed to allocate pax buffer");
restart();
}
memset(pax_buffer, 0, sizeof(pax_buf_t));
pax_buf_init(pax_buffer, framebuffer, ILI9341_WIDTH, ILI9341_HEIGHT, PAX_BUF_16_565RGB);
2022-01-11 23:53:38 +00:00
driver_framebuffer_init(framebuffer);
2022-04-15 23:16:36 +00:00
/* Initialize hardware */
2022-04-23 02:00:55 +00:00
bool lcdReady = false;
res = board_init(&lcdReady);
2022-01-23 00:27:22 +00:00
if (res != ESP_OK) {
2022-04-23 02:00:55 +00:00
if (lcdReady) {
ILI9341* ili9341 = get_ili9341();
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "HARDWARE ERROR");
}
2022-01-23 00:27:22 +00:00
printf("Failed to initialize hardware!\n");
restart();
}
2022-04-15 23:16:36 +00:00
ILI9341* ili9341 = get_ili9341();
ICE40* ice40 = get_ice40();
BNO055* bno055 = get_bno055();
RP2040* rp2040 = get_rp2040();
2022-04-08 21:15:57 +00:00
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "AppFS init...");
2022-01-06 00:12:28 +00:00
res = appfs_init();
if (res != ESP_OK) {
ESP_LOGE(TAG, "AppFS init failed: %d", res);
2022-04-16 00:14:28 +00:00
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "AppFS init failed!");
2022-01-06 00:12:28 +00:00
return;
}
ESP_LOGI(TAG, "AppFS initialized");
2022-04-16 00:14:28 +00:00
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "NVS init...");
res = nvs_init();
if (res != ESP_OK) {
ESP_LOGE(TAG, "NVS init failed: %d", res);
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "NVS init failed!");
return;
}
ESP_LOGI(TAG, "NVS initialized");
2022-04-08 21:15:57 +00:00
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "Mount SD card...");
2022-01-06 00:12:28 +00:00
res = mount_sd(SD_CMD, SD_CLK, SD_D0, SD_PWR, "/sd", false, 5);
bool sdcard_ready = (res == ESP_OK);
2022-04-08 21:15:57 +00:00
if (sdcard_ready) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "SD card mounted");
}
2022-04-23 02:00:55 +00:00
ws2812_init(GPIO_LED_DATA);
2022-04-29 03:24:17 +00:00
uint8_t ledBuffer[15] = {50, 0, 0, 50, 0, 0, 50, 0, 0, 50, 0, 0, 50, 0, 0};
2022-04-23 02:00:55 +00:00
ws2812_send_data(ledBuffer, sizeof(ledBuffer));
2022-04-29 03:24:17 +00:00
uint8_t fw_version;
if (rp2040_get_firmware_version(rp2040, &fw_version) != ESP_OK) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "RP2040 FW VERSION READ FAILED");
restart();
}
if (fw_version == 0xFF) {
// RP2040 is in bootloader mode
char buffer[255] = {0};
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "RP2040 BL");
uint8_t bl_version;
if (rp2040_get_bootloader_version(rp2040, &bl_version) != ESP_OK) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "RP2040 BL VERSION READ FAILED");
restart();
}
while (true) {
uint8_t bl_state;
if (rp2040_get_bootloader_state(rp2040, &bl_state) != ESP_OK) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "RP2040 BL STATE READ FAILED");
restart();
}
snprintf(buffer, sizeof(buffer), "RP2040 BL (%02X): %02X", bl_version, bl_state);
graphics_task(pax_buffer, ili9341, framebuffer, NULL, buffer);
vTaskDelay(200 / portTICK_PERIOD_MS);
if (bl_state == 0xB0) {
printf("SYNC");
}
}
return;
}
/*while (true) {
fpga_test(ili9341, ice40, rp2040->queue);
vTaskDelay(200 / portTICK_PERIOD_MS);
}*/
2022-04-23 02:00:55 +00:00
/*while (true) {
uint16_t state;
rp2040_read_buttons(rp2040, &state);
printf("Button state: %04X\n", state);
vTaskDelay(100 / portTICK_PERIOD_MS);
ledBuffer[1] = 255;
ws2812_send_data(ledBuffer, sizeof(ledBuffer));
vTaskDelay(200 / portTICK_PERIOD_MS);
ledBuffer[1] = 0;
ledBuffer[0] = 255;
ws2812_send_data(ledBuffer, sizeof(ledBuffer));
fpga_test(ili9341, ice40, rp2040->queue);
ledBuffer[0] = 0;
ws2812_send_data(ledBuffer, sizeof(ledBuffer));
}*/
2022-04-15 23:16:36 +00:00
while (true) {
menu_action_t menu_action;
appfs_handle_t appfs_fd;
2022-04-23 02:00:55 +00:00
menu_launcher(rp2040->queue, pax_buffer, ili9341, framebuffer, &menu_action, &appfs_fd);
2022-04-15 23:16:36 +00:00
if (menu_action == ACTION_APPFS) {
appfs_boot_app(appfs_fd);
} else if (menu_action == ACTION_FPGA) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "FPGA TEST");
2022-04-29 03:24:17 +00:00
//fpga_test(ili9341, ice40, rp2040->queue);
rp2040_reboot_to_bootloader(rp2040);
restart();
2022-04-15 23:16:36 +00:00
} else if (menu_action == ACTION_INSTALLER) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "INSTALLER");
//appfs_store_app();
2022-04-18 00:10:44 +00:00
nvs_handle_t handle;
nvs_open("system", NVS_READWRITE, &handle);
char ssid[33];
char password[33];
size_t requiredSize;
esp_err_t res = nvs_get_str(handle, "wifi.ssid", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(ssid, "");
} else if (requiredSize < sizeof(ssid)) {
res = nvs_get_str(handle, "wifi.ssid", ssid, &requiredSize);
if (res != ESP_OK) strcpy(ssid, "");
res = nvs_get_str(handle, "wifi.password", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(password, "");
} else if (requiredSize < sizeof(password)) {
res = nvs_get_str(handle, "wifi.password", password, &requiredSize);
if (res != ESP_OK) strcpy(password, "");
}
}
nvs_close(&handle);
wifi_init(ssid, password, WIFI_AUTH_WPA2_PSK, 3);
2022-04-16 00:14:28 +00:00
} else if (menu_action == ACTION_OTA) {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "Firmware update...");
} else if (menu_action == ACTION_SETTINGS) {
2022-04-17 18:22:54 +00:00
while (true) {
2022-04-23 02:00:55 +00:00
menu_wifi_settings(rp2040->queue, pax_buffer, ili9341, framebuffer, &menu_action);
2022-04-17 18:22:54 +00:00
if (menu_action == ACTION_WIFI_MANUAL) {
nvs_handle_t handle;
nvs_open("system", NVS_READWRITE, &handle);
char ssid[33];
char password[33];
size_t requiredSize;
esp_err_t res = nvs_get_str(handle, "wifi.ssid", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(ssid, "");
strcpy(password, "");
} else if (requiredSize < sizeof(ssid)) {
res = nvs_get_str(handle, "wifi.ssid", ssid, &requiredSize);
if (res != ESP_OK) strcpy(ssid, "");
res = nvs_get_str(handle, "wifi.password", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(password, "");
} else if (requiredSize < sizeof(password)) {
res = nvs_get_str(handle, "wifi.password", password, &requiredSize);
if (res != ESP_OK) strcpy(password, "");
}
}
2022-04-23 02:00:55 +00:00
bool accepted = keyboard(rp2040->queue, pax_buffer, ili9341, framebuffer, 30, 30, pax_buffer->width - 60, pax_buffer->height - 60, "WiFi SSID", "Press HOME to exit", ssid, sizeof(ssid));
2022-04-17 18:22:54 +00:00
if (accepted) {
2022-04-23 02:00:55 +00:00
accepted = keyboard(rp2040->queue, pax_buffer, ili9341, framebuffer, 30, 30, pax_buffer->width - 60, pax_buffer->height - 60, "WiFi password", "Press HOME to exit", password, sizeof(password));
2022-04-17 18:22:54 +00:00
}
if (accepted) {
nvs_set_str(handle, "wifi.ssid", ssid);
nvs_set_str(handle, "wifi.password", password);
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "WiFi settings stored");
} else {
graphics_task(pax_buffer, ili9341, framebuffer, NULL, "Canceled");
}
nvs_close(&handle);
} else if (menu_action == ACTION_WIFI_LIST) {
nvs_handle_t handle;
nvs_open("system", NVS_READWRITE, &handle);
char ssid[33];
char password[33];
size_t requiredSize;
esp_err_t res = nvs_get_str(handle, "wifi.ssid", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(ssid, "");
} else if (requiredSize < sizeof(ssid)) {
res = nvs_get_str(handle, "wifi.ssid", ssid, &requiredSize);
if (res != ESP_OK) strcpy(ssid, "");
res = nvs_get_str(handle, "wifi.password", NULL, &requiredSize);
if (res != ESP_OK) {
strcpy(password, "");
} else if (requiredSize < sizeof(password)) {
res = nvs_get_str(handle, "wifi.password", password, &requiredSize);
if (res != ESP_OK) strcpy(password, "");
}
}
nvs_close(&handle);
char buffer[300];
snprintf(buffer, sizeof(buffer), "SSID is %s\nPassword is %s", ssid, password);
graphics_task(pax_buffer, ili9341, framebuffer, NULL, buffer);
} else {
break;
}
2022-04-16 00:14:28 +00:00
}
2022-01-01 06:13:50 +00:00
}
}
2022-04-08 21:15:57 +00:00
free(framebuffer);
}