Files
esp-protocols/components/esp_modem/src/esp_modem_uart.cpp

202 lines
5.7 KiB
C++
Raw Normal View History

2021-03-29 19:34:45 +02:00
// Copyright 2021 Espressif Systems (Shanghai) PTE LTD
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2021-03-03 20:35:08 +01:00
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/semphr.h"
#include "esp_log.h"
#include "driver/uart.h"
#include "uart_compat.h"
2021-03-08 15:10:15 +01:00
#include "esp_modem_config.h"
2021-03-16 21:36:13 +01:00
#include "exception_stub.hpp"
#include "cxx_include/esp_modem_dte.hpp"
#include "uart_resource.hpp"
2021-03-03 20:35:08 +01:00
static const char *TAG = "uart_terminal";
2021-03-29 19:34:45 +02:00
namespace esp_modem {
2021-03-03 20:35:08 +01:00
struct uart_task {
2021-03-29 19:34:45 +02:00
explicit uart_task(size_t stack_size, size_t priority, void *task_param, TaskFunction_t task_function) :
2021-06-01 10:21:51 +02:00
task_handle(nullptr)
{
2021-04-18 19:14:22 +02:00
BaseType_t ret = xTaskCreate(task_function, "uart_task", stack_size, task_param, priority, &task_handle);
ESP_MODEM_THROW_IF_FALSE(ret == pdTRUE, "create uart event task failed");
2021-03-03 20:35:08 +01:00
}
2021-03-29 19:34:45 +02:00
2021-06-01 10:21:51 +02:00
~uart_task()
{
if (task_handle) {
vTaskDelete(task_handle);
}
2021-03-03 20:35:08 +01:00
}
TaskHandle_t task_handle; /*!< UART event task handle */
};
2021-05-19 23:00:28 +08:00
class UartTerminal : public Terminal {
2021-03-03 20:35:08 +01:00
public:
2021-05-19 23:00:28 +08:00
explicit UartTerminal(const esp_modem_dte_config *config) :
2021-06-01 10:21:51 +02:00
event_queue(), uart(&config->uart_config, &event_queue, -1), signal(),
task_handle(config->task_stack_size, config->task_priority, this, s_task) {}
2021-03-03 20:35:08 +01:00
2021-05-19 23:00:28 +08:00
~UartTerminal() override = default;
2021-03-29 19:34:45 +02:00
2021-06-01 10:21:51 +02:00
void start() override
{
2021-03-03 20:35:08 +01:00
signal.set(TASK_START);
}
2021-03-29 19:34:45 +02:00
2021-06-01 10:21:51 +02:00
void stop() override
{
2021-03-03 20:35:08 +01:00
signal.set(TASK_STOP);
}
int write(uint8_t *data, size_t len) override;
2021-03-29 19:34:45 +02:00
2021-03-03 20:35:08 +01:00
int read(uint8_t *data, size_t len) override;
2021-03-29 19:34:45 +02:00
2021-06-01 10:21:51 +02:00
void set_read_cb(std::function<bool(uint8_t *data, size_t len)> f) override
{
on_read = std::move(f);
2021-03-04 20:19:18 +01:00
signal.set(TASK_PARAMS);
}
2021-03-29 19:34:45 +02:00
2021-03-03 20:35:08 +01:00
private:
2021-06-01 10:21:51 +02:00
static void s_task(void *task_param)
{
2021-05-19 23:00:28 +08:00
auto t = static_cast<UartTerminal *>(task_param);
2021-03-03 20:35:08 +01:00
t->task();
2021-05-17 14:59:03 +02:00
vTaskDelete(nullptr);
2021-03-03 20:35:08 +01:00
}
2021-03-29 19:34:45 +02:00
2021-03-03 20:35:08 +01:00
void task();
2021-06-01 10:21:51 +02:00
bool get_event(uart_event_t &event, uint32_t time_ms)
{
2021-05-17 14:59:03 +02:00
return xQueueReceive(event_queue, &event, pdMS_TO_TICKS(time_ms));
}
2021-06-01 10:21:51 +02:00
void reset_events()
{
2021-05-17 14:59:03 +02:00
uart_flush_input(uart.port);
xQueueReset(event_queue);
}
2021-03-03 20:35:08 +01:00
2021-03-04 20:19:18 +01:00
static const size_t TASK_INIT = BIT0;
static const size_t TASK_START = BIT1;
static const size_t TASK_STOP = BIT2;
static const size_t TASK_PARAMS = BIT3;
2021-03-03 20:35:08 +01:00
2021-05-17 14:59:03 +02:00
QueueHandle_t event_queue;
2021-03-03 20:35:08 +01:00
uart_resource uart;
SignalGroup signal;
2021-03-03 20:35:08 +01:00
uart_task task_handle;
};
2021-06-01 10:21:51 +02:00
std::unique_ptr<Terminal> create_uart_terminal(const esp_modem_dte_config *config)
{
2021-03-16 21:36:13 +01:00
TRY_CATCH_RET_NULL(
2021-06-01 10:21:51 +02:00
auto term = std::make_unique<UartTerminal>(config);
term->start();
return term;
2021-03-16 21:36:13 +01:00
)
2021-03-03 20:35:08 +01:00
}
2021-06-01 10:21:51 +02:00
void UartTerminal::task()
{
std::function<bool(uint8_t *data, size_t len)> on_read_priv = nullptr;
2021-03-03 20:35:08 +01:00
uart_event_t event;
size_t len;
signal.set(TASK_INIT);
signal.wait_any(TASK_START | TASK_STOP, portMAX_DELAY);
if (signal.is_any(TASK_STOP)) {
return; // exits to the static method where the task gets deleted
}
2021-03-29 19:34:45 +02:00
while (signal.is_any(TASK_START)) {
2021-05-17 14:59:03 +02:00
if (get_event(event, 100)) {
2021-03-04 20:19:18 +01:00
if (signal.is_any(TASK_PARAMS)) {
on_read_priv = on_read;
2021-03-04 20:19:18 +01:00
signal.clear(TASK_PARAMS);
}
2021-03-03 20:35:08 +01:00
switch (event.type) {
2021-06-01 10:21:51 +02:00
case UART_DATA:
uart_get_buffered_data_len(uart.port, &len);
if (len && on_read_priv) {
if (on_read_priv(nullptr, len)) {
on_read_priv = nullptr;
2021-03-03 20:35:08 +01:00
}
2021-06-01 10:21:51 +02:00
}
break;
case UART_FIFO_OVF:
ESP_LOGW(TAG, "HW FIFO Overflow");
if (on_error) {
on_error(terminal_error::BUFFER_OVERFLOW);
}
reset_events();
break;
case UART_BUFFER_FULL:
ESP_LOGW(TAG, "Ring Buffer Full");
if (on_error) {
on_error(terminal_error::BUFFER_OVERFLOW);
}
reset_events();
break;
case UART_BREAK:
ESP_LOGW(TAG, "Rx Break");
if (on_error) {
on_error(terminal_error::UNEXPECTED_CONTROL_FLOW);
}
break;
case UART_PARITY_ERR:
ESP_LOGE(TAG, "Parity Error");
if (on_error) {
on_error(terminal_error::CHECKSUM_ERROR);
}
break;
case UART_FRAME_ERR:
ESP_LOGE(TAG, "Frame Error");
if (on_error) {
on_error(terminal_error::UNEXPECTED_CONTROL_FLOW);
}
break;
default:
ESP_LOGW(TAG, "unknown uart event type: %d", event.type);
break;
2021-03-03 20:35:08 +01:00
}
}
}
}
int UartTerminal::read(uint8_t *data, size_t len)
{
2021-03-03 20:35:08 +01:00
size_t length = 0;
uart_get_buffered_data_len(uart.port, &length);
length = std::min(len, length);
2021-03-03 20:35:08 +01:00
if (length > 0) {
return uart_read_bytes(uart.port, data, length, portMAX_DELAY);
}
return 0;
}
int UartTerminal::write(uint8_t *data, size_t len)
{
return uart_write_bytes_compat(uart.port, data, len);
2021-03-03 20:35:08 +01:00
}
2021-03-29 19:34:45 +02:00
} // namespace esp_modem
2021-03-03 20:35:08 +01:00