/* * This file is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This file is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . * * Author: Siddharth Bharat Purohit */ #include #if HAL_WITH_UAVCAN #include "AP_UAVCAN_Servers.h" #ifdef HAS_UAVCAN_SERVERS #include #include #include #include #include #include #include #include #if HAL_OS_POSIX_IO #include #include #include #include #include #include #include #include #include #include #if defined(__APPLE__) && defined(__MACH__) #include #include #else #include #endif #endif #if HAL_OS_FATFS_IO #include #endif #ifndef UAVCAN_NODE_DB_PATH #define UAVCAN_NODE_DB_PATH HAL_BOARD_STORAGE_DIRECTORY "/UAVCAN" #endif #include extern const AP_HAL::HAL& hal; #define debug_uavcan(fmt, args...) do { hal.console->printf(fmt, ##args); } while (0) class AP_UAVCAN_CentralizedServer : public uavcan::dynamic_node_id_server::CentralizedServer { public: AP_UAVCAN_CentralizedServer(uavcan::INode& node, uavcan::dynamic_node_id_server::IStorageBackend &storage_backend, uavcan::dynamic_node_id_server::IEventTracer &tracer) : uavcan::dynamic_node_id_server::CentralizedServer(node, storage_backend, tracer) {} }; class AP_UAVCAN_FileEventTracer : public uavcan::dynamic_node_id_server::IEventTracer { protected: virtual void onEvent(uavcan::dynamic_node_id_server::TraceCode code, uavcan::int64_t argument) { AP::logger().Write("UCEV", "TimeUS,code,arg", "s--", "F--", "Qhq", AP_HAL::micros64(), code, argument); } }; class AP_UAVCAN_RestartRequestHandler : public uavcan::IRestartRequestHandler { public: bool handleRestartRequest(uavcan::NodeID request_source) override { // swiped from reboot handling in GCS_Common.cpp AP_Notify *notify = AP_Notify::get_singleton(); if (notify) { AP_Notify::flags.firmware_update = 1; notify->update(); } // force safety on hal.rcout->force_safety_on(); hal.rcout->force_safety_no_wait(); // flush pending parameter writes AP_Param::flush(); hal.scheduler->delay(200); hal.scheduler->reboot(false); return true; } }; class AP_UAVCAN_FileStorageBackend : public uavcan::dynamic_node_id_server::IStorageBackend { /** * Maximum length of full path including / and key max */ enum { MaxPathLength = 128 }; enum { FilePermissions = 438 }; ///< 0o666 enum { MaxNumOpens = 100 }; /** * This type is used for the path */ typedef uavcan::MakeString::Type PathString; PathString base_path; static uint8_t num_opens; protected: virtual String get(const String& key) const { using namespace std; PathString path = base_path.c_str(); path += key; String value; //This is to deter frequent inflight opening and closing of files during an event //where the device is misbehaving if (num_opens >= MaxNumOpens) { return value; } num_opens++; int fd = open(path.c_str(), O_RDONLY); if (fd >= 0) { char buffer[MaxStringLength + 1]; (void)memset(buffer, 0, sizeof(buffer)); ssize_t remaining = MaxStringLength; ssize_t total_read = 0; ssize_t nread = 0; do { nread = ::read(fd, &buffer[total_read], remaining); if (nread > 0) { remaining -= nread, total_read += nread; } } while (nread > 0 && remaining > 0); (void)close(fd); if (total_read > 0) { for (int i = 0; i < total_read; i++) { if (buffer[i] == ' ' || buffer[i] == '\n' || buffer[i] == '\r') { buffer[i] = '\0'; break; } } value = buffer; } } return value; } virtual void set(const String& key, const String& value) { using namespace std; PathString path = base_path.c_str(); path += key; //This is to deter frequent inflight opening and closing of files during an event //where the device is misbehaving if (num_opens >= MaxNumOpens) { return; } num_opens++; #if HAL_OS_POSIX_IO int fd = open(path.c_str(), O_WRONLY | O_CREAT | O_TRUNC, FilePermissions); #else int fd = open(path.c_str(), O_WRONLY | O_CREAT | O_TRUNC); #endif if (fd >= 0) { ssize_t remaining = value.size(); ssize_t total_written = 0; ssize_t written = 0; do { written = write(fd, &value.c_str()[total_written], remaining); if (written > 0) { total_written += written; remaining -= written; } } while (written > 0 && remaining > 0); (void)fsync(fd); (void)close(fd); } } public: /** * Initializes the file based backend storage by passing a path to * the directory where the key named files will be stored. * The return value should be 0 on success. * If it is -ErrInvalidConfiguration then the the path name is too long to * accommodate the trailing slash and max key length. */ int init(const PathString& path) { using namespace std; int rv = -uavcan::ErrInvalidParam; if (path.size() > 0) { base_path = path.c_str(); if (base_path.back() == '/') { base_path.pop_back(); } rv = 0; struct stat sb; if (stat(base_path.c_str(), &sb) != 0 || !S_ISDIR(sb.st_mode)) { // coverity[toctou] rv = mkdir(base_path.c_str(), S_IRWXU | S_IRWXG | S_IRWXO); } if (rv >= 0) { base_path.push_back('/'); if ((base_path.size() + MaxStringLength) > MaxPathLength) { rv = -uavcan::ErrInvalidConfiguration; } } } return rv; } }; uint8_t AP_UAVCAN_FileStorageBackend::num_opens = 0; bool AP_UAVCAN_Servers::init(uavcan::Node<0> &node) { if (_server_instance != nullptr) { return true; } int ret = 0; _storage_backend = new AP_UAVCAN_FileStorageBackend; if (_storage_backend == nullptr) { debug_uavcan("UAVCAN_Servers: Failed to Allocate FileStorageBackend\n"); goto failed; } ret = _storage_backend->init(UAVCAN_NODE_DB_PATH); if (ret < 0) { debug_uavcan("UAVCAN_Servers: FileStorageBackend init: %d, errno: %d\n", ret, errno); goto failed; } _tracer = new AP_UAVCAN_FileEventTracer; if (_tracer == nullptr) { debug_uavcan("UAVCAN_Servers: Failed to Allocate FileEventTracer\n"); goto failed; } _server_instance = new AP_UAVCAN_CentralizedServer(node, *_storage_backend, *_tracer); if (_server_instance == nullptr) { debug_uavcan("UAVCAN_Servers: Failed to Allocate Server\n"); goto failed; } { uavcan::dynamic_node_id_server::centralized::Storage storage(*_storage_backend); if (storage.getNodeIDForUniqueID(node.getHardwareVersion().unique_id) != node.getNodeID()) { //Node ID was changed, reseting database reset(); } } if (_restart_request_handler == nullptr) { _restart_request_handler = new AP_UAVCAN_RestartRequestHandler(); if (_restart_request_handler == nullptr) { goto failed; } } node.setRestartRequestHandler(_restart_request_handler); //Start Dynamic Node Server ret = _server_instance->init(node.getHardwareVersion().unique_id); if (ret < 0) { debug_uavcan("UAVCAN_Server init: %d, errno: %d\n", ret, errno); goto failed; } return true; failed: delete _restart_request_handler; delete _storage_backend; delete _tracer; delete _server_instance; return false; } void AP_UAVCAN_Servers::reset() { debug_uavcan("UAVCAN_Servers: Resetting Server Database...\n"); DIR* dp; struct dirent* ep; dp = opendir(UAVCAN_NODE_DB_PATH); char abs_filename[100]; if (dp != NULL) { while((ep = readdir(dp))) { snprintf(abs_filename, 100, "%s/%s", UAVCAN_NODE_DB_PATH, ep->d_name); unlink(abs_filename); } } closedir(dp); } #endif #endif //HAL_WITH_UAVCAN