#include "options.h" #include #include "parameters.h" #include #include #include "romfs.h" #include "util.h" Parameters g; static nvs_handle handle; const Parameters::Param Parameters::params[] = { { "LOCK_LEVEL", Parameters::ParamType::UINT8, (const void*)&g.lock_level, 0, 0, 2 }, { "CAN_NODE", Parameters::ParamType::UINT8, (const void*)&g.can_node, 0, 0, 127 }, { "UAS_TYPE", Parameters::ParamType::UINT8, (const void*)&g.ua_type, 0, 0, 15 }, { "UAS_ID_TYPE", Parameters::ParamType::UINT8, (const void*)&g.id_type, 0, 0, 4 }, { "UAS_ID", Parameters::ParamType::CHAR20, (const void*)&g.uas_id[0], 0, 0, 0 }, { "BAUDRATE", Parameters::ParamType::UINT32, (const void*)&g.baudrate, 57600, 9600, 921600 }, { "WIFI_NAN_RATE", Parameters::ParamType::FLOAT, (const void*)&g.wifi_nan_rate, 0, 0, 5 }, { "WIFI_POWER", Parameters::ParamType::FLOAT, (const void*)&g.wifi_power, 20, 2, 20 }, { "BT4_RATE", Parameters::ParamType::FLOAT, (const void*)&g.bt4_rate, 1, 0, 5 }, { "BT4_POWER", Parameters::ParamType::FLOAT, (const void*)&g.bt4_power, 18, -27, 18 }, { "BT5_RATE", Parameters::ParamType::FLOAT, (const void*)&g.bt5_rate, 1, 0, 5 }, { "BT5_POWER", Parameters::ParamType::FLOAT, (const void*)&g.bt5_power, 18, -27, 18 }, { "WEBSERVER_ENABLE", Parameters::ParamType::UINT8, (const void*)&g.webserver_enable, 1, 0, 1 }, { "WIFI_SSID", Parameters::ParamType::CHAR20, (const void*)&g.wifi_ssid, }, { "WIFI_PASSWORD", Parameters::ParamType::CHAR20, (const void*)&g.wifi_password, 0, 0, 0, PARAM_FLAG_PASSWORD, 8 }, { "BCAST_POWERUP", Parameters::ParamType::UINT8, (const void*)&g.bcast_powerup, 1, 0, 1 }, { "PUBLIC_KEY1", Parameters::ParamType::CHAR64, (const void*)&g.public_keys[0], }, { "PUBLIC_KEY2", Parameters::ParamType::CHAR64, (const void*)&g.public_keys[1], }, { "PUBLIC_KEY3", Parameters::ParamType::CHAR64, (const void*)&g.public_keys[2], }, { "PUBLIC_KEY4", Parameters::ParamType::CHAR64, (const void*)&g.public_keys[3], }, { "PUBLIC_KEY5", Parameters::ParamType::CHAR64, (const void*)&g.public_keys[4], }, { "DONE_INIT", Parameters::ParamType::UINT8, (const void*)&g.done_init, 0, 0, 0, PARAM_FLAG_HIDDEN}, { "", Parameters::ParamType::NONE, nullptr, }, }; /* find by name */ const Parameters::Param *Parameters::find(const char *name) { for (const auto &p : params) { if (strcmp(name, p.name) == 0) { return &p; } } return nullptr; } /* find by index */ const Parameters::Param *Parameters::find_by_index(uint16_t index) { if (index >= (sizeof(params)/sizeof(params[0])-1)) { return nullptr; } return ¶ms[index]; } void Parameters::Param::set_uint8(uint8_t v) const { auto *p = (uint8_t *)ptr; *p = v; nvs_set_u8(handle, name, *p); } void Parameters::Param::set_uint32(uint32_t v) const { auto *p = (uint32_t *)ptr; *p = v; nvs_set_u32(handle, name, *p); } void Parameters::Param::set_float(float v) const { auto *p = (float *)ptr; *p = v; union { float f; uint32_t u32; } u; u.f = v; nvs_set_u32(handle, name, u.u32); } void Parameters::Param::set_char20(const char *v) const { if (min_len > 0 && strlen(v) < min_len) { return; } memset((void*)ptr, 0, 21); strncpy((char *)ptr, v, 20); nvs_set_str(handle, name, v); } void Parameters::Param::set_char64(const char *v) const { if (min_len > 0 && strlen(v) < min_len) { return; } memset((void*)ptr, 0, 65); strncpy((char *)ptr, v, 64); nvs_set_str(handle, name, v); } uint8_t Parameters::Param::get_uint8() const { const auto *p = (const uint8_t *)ptr; return *p; } uint32_t Parameters::Param::get_uint32() const { const auto *p = (const uint32_t *)ptr; return *p; } float Parameters::Param::get_float() const { const auto *p = (const float *)ptr; return *p; } const char *Parameters::Param::get_char20() const { const char *p = (const char *)ptr; return p; } const char *Parameters::Param::get_char64() const { const char *p = (const char *)ptr; return p; } /* load defaults from parameter table */ void Parameters::load_defaults(void) { for (const auto &p : params) { switch (p.ptype) { case ParamType::UINT8: *(uint8_t *)p.ptr = uint8_t(p.default_value); break; case ParamType::UINT32: *(uint32_t *)p.ptr = uint32_t(p.default_value); break; case ParamType::FLOAT: *(float *)p.ptr = p.default_value; break; } } } void Parameters::init(void) { load_defaults(); if (nvs_flash_init() != ESP_OK || nvs_open("storage", NVS_READWRITE, &handle) != ESP_OK) { Serial.printf("NVS init failed\n"); } // load values from NVS for (const auto &p : params) { switch (p.ptype) { case ParamType::UINT8: nvs_get_u8(handle, p.name, (uint8_t *)p.ptr); break; case ParamType::UINT32: nvs_get_u32(handle, p.name, (uint32_t *)p.ptr); break; case ParamType::FLOAT: nvs_get_u32(handle, p.name, (uint32_t *)p.ptr); break; case ParamType::CHAR20: { size_t len = 21; nvs_get_str(handle, p.name, (char *)p.ptr, &len); break; } case ParamType::CHAR64: { size_t len = 65; nvs_get_str(handle, p.name, (char *)p.ptr, &len); break; } } } if (strlen(g.wifi_ssid) == 0) { uint8_t mac[6] {}; esp_read_mac(mac, ESP_MAC_WIFI_STA); snprintf(wifi_ssid, 20, "RID_%02x%02x%02x%02x%02x%02x", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); } if (g.done_init == 0) { set_by_name_uint8("DONE_INIT", 1); // setup public keys set_by_name_char64("PUBLIC_KEY1", ROMFS::find_string("public_keys/ArduPilot_public_key1.dat")); set_by_name_char64("PUBLIC_KEY2", ROMFS::find_string("public_keys/ArduPilot_public_key2.dat")); set_by_name_char64("PUBLIC_KEY3", ROMFS::find_string("public_keys/ArduPilot_public_key2.dat")); } } /* check if BasicID info is filled in with parameters */ bool Parameters::have_basic_id_info(void) const { return strlen(g.uas_id) > 0 && g.id_type > 0 && g.ua_type > 0; } bool Parameters::set_by_name_uint8(const char *name, uint8_t v) { const auto *f = find(name); if (!f) { return false; } f->set_uint8(v); return true; } bool Parameters::set_by_name_char64(const char *name, const char *s) { const auto *f = find(name); if (!f) { return false; } f->set_char64(s); return true; } bool Parameters::set_by_name_string(const char *name, const char *s) { const auto *f = find(name); if (!f) { return false; } switch (f->ptype) { case ParamType::UINT8: f->set_uint8(uint8_t(strtoul(s, nullptr, 0))); return true; case ParamType::UINT32: f->set_uint32(strtoul(s, nullptr, 0)); return true; case ParamType::FLOAT: f->set_float(atof(s)); return true; case ParamType::CHAR20: f->set_char20(s); return true; case ParamType::CHAR64: f->set_char64(s); return true; } return false; } /* return a public key */ bool Parameters::get_public_key(uint8_t i, uint8_t key[32]) const { if (i >= MAX_PUBLIC_KEYS) { return false; } const char *b64_key = g.public_keys[i].b64_key; const char *ktype = "PUBLIC_KEYV1:"; if (strncmp(b64_key, ktype, strlen(ktype)) != 0) { return false; } b64_key += strlen(ktype); int32_t out_len = base64_decode(b64_key, key, 32); if (out_len != 32) { return false; } return true; } bool Parameters::no_public_keys(void) const { for (uint8_t i=0; i