#ifdef ESP_PLATFORM #include "Tactility/hal/uart/UartEsp.h" #include #include #include #include #define TAG "uart" namespace tt::hal::uart { bool UartEsp::start() { auto lock = mutex.asScopedLock(); lock.lock(); if (started) { TT_LOG_E(TAG, "(%d) Starting: Already started", configuration.port); return false; } int intr_alloc_flags; #if CONFIG_UART_ISR_IN_IRAM intr_alloc_flags = ESP_INTR_FLAG_IRAM; #else intr_alloc_flags = 0; #endif esp_err_t result = uart_param_config(configuration.port, &configuration.config); if (result != ESP_OK) { TT_LOG_E(TAG, "(%d) Starting: Failed to configure: %s", configuration.port, esp_err_to_name(result)); return false; } result = uart_set_pin(configuration.port, configuration.txPin, configuration.rxPin, configuration.rtsPin, configuration.ctsPin); if (result != ESP_OK) { TT_LOG_E(TAG, "(%d) Starting: Failed set pins: %s", configuration.port, esp_err_to_name(result)); return false; } result = uart_driver_install(configuration.port, (int)configuration.rxBufferSize, (int)configuration.txBufferSize, 0, nullptr, intr_alloc_flags); if (result != ESP_OK) { TT_LOG_E(TAG, "(%d) Starting: Failed to install driver: %s", configuration.port, esp_err_to_name(result)); return false; } started = true; TT_LOG_I(TAG, "(%d) Started", configuration.port); return true; } bool UartEsp::stop() { auto lock = mutex.asScopedLock(); lock.lock(); if (!started) { TT_LOG_E(TAG, "(%d) Stopping: Not started", configuration.port); return false; } esp_err_t result = uart_driver_delete(configuration.port); if (result != ESP_OK) { TT_LOG_E(TAG, "(%d) Stopping: Failed to delete driver: %s", configuration.port, esp_err_to_name(result)); return false; } started = false; TT_LOG_I(TAG, "(%d) Stopped", configuration.port); return true; } bool UartEsp::isStarted() const { auto lock = mutex.asScopedLock(); lock.lock(); return started; } size_t UartEsp::readBytes(std::byte* buffer, size_t bufferSize, TickType_t timeout) { auto lock = mutex.asScopedLock(); if (!lock.lock(timeout)) { return false; } auto start_time = kernel::getTicks(); auto lock_time = kernel::getTicks() - start_time; auto remaining_timeout = std::max(timeout - lock_time, 0UL); auto result = uart_read_bytes(configuration.port, buffer, bufferSize, remaining_timeout); return result; } bool UartEsp::readByte(std::byte* output, TickType_t timeout) { return readBytes(output, 1, timeout) == 1; } size_t UartEsp::writeBytes(const std::byte* buffer, size_t bufferSize, TickType_t timeout) { if (!mutex.lock(timeout)) { return false; } return uart_write_bytes(configuration.port, buffer, bufferSize); } size_t UartEsp::available(TickType_t timeout) { auto lock = mutex.asScopedLock(); if (!lock.lock(timeout)) { return false; } size_t size = 0; uart_get_buffered_data_len(configuration.port, &size); return size; } void UartEsp::flushInput() { uart_flush_input(configuration.port); } uint32_t UartEsp::getBaudRate() { uint32_t baud_rate = 0; auto result = uart_get_baudrate(configuration.port, &baud_rate); ESP_ERROR_CHECK_WITHOUT_ABORT(result); return baud_rate; } bool UartEsp::setBaudRate(uint32_t baudRate, TickType_t timeout) { auto lock = mutex.asScopedLock(); if (!lock.lock(timeout)) { return false; } auto result = uart_set_baudrate(configuration.port, baudRate); ESP_ERROR_CHECK_WITHOUT_ABORT(result); return result == ESP_OK; } std::unique_ptr create(const Configuration& configuration) { return std::make_unique(configuration); } } // namespace tt::hal::uart #endif