2021-03-29 19:34:45 +02:00
|
|
|
/*
|
2023-09-20 12:06:47 +02:00
|
|
|
* SPDX-FileCopyrightText: 2021-2023 Espressif Systems (Shanghai) CO LTD
|
2022-10-11 16:31:57 +02:00
|
|
|
*
|
2021-03-29 19:34:45 +02:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2021-03-30 08:25:16 +02:00
|
|
|
#include <cstring>
|
2021-02-26 18:32:15 +01:00
|
|
|
#include "esp_log.h"
|
2021-04-18 19:14:22 +02:00
|
|
|
#include "cxx_include/esp_modem_dte.hpp"
|
2022-06-10 18:04:10 +02:00
|
|
|
#include "cxx_include/esp_modem_cmux.hpp"
|
2021-04-18 19:14:22 +02:00
|
|
|
#include "esp_modem_config.h"
|
2021-02-26 18:32:15 +01:00
|
|
|
|
2021-03-29 19:34:45 +02:00
|
|
|
using namespace esp_modem;
|
|
|
|
|
2021-04-19 11:28:53 +02:00
|
|
|
static const size_t dte_default_buffer_size = 1000;
|
|
|
|
|
2021-04-18 19:14:22 +02:00
|
|
|
DTE::DTE(const esp_modem_dte_config *config, std::unique_ptr<Terminal> terminal):
|
2022-10-25 16:50:34 +02:00
|
|
|
buffer(config->dte_buffer_size),
|
|
|
|
cmux_term(nullptr), primary_term(std::move(terminal)), secondary_term(primary_term),
|
2023-06-23 19:35:49 +02:00
|
|
|
mode(modem_mode::UNDEF)
|
|
|
|
{
|
|
|
|
set_command_callbacks();
|
|
|
|
}
|
2021-02-26 18:32:15 +01:00
|
|
|
|
2021-04-19 11:28:53 +02:00
|
|
|
DTE::DTE(std::unique_ptr<Terminal> terminal):
|
2022-10-25 16:50:34 +02:00
|
|
|
buffer(dte_default_buffer_size),
|
|
|
|
cmux_term(nullptr), primary_term(std::move(terminal)), secondary_term(primary_term),
|
2023-06-23 19:35:49 +02:00
|
|
|
mode(modem_mode::UNDEF)
|
|
|
|
{
|
|
|
|
set_command_callbacks();
|
|
|
|
}
|
2021-04-19 11:28:53 +02:00
|
|
|
|
2023-03-07 13:35:34 +01:00
|
|
|
DTE::DTE(const esp_modem_dte_config *config, std::unique_ptr<Terminal> t, std::unique_ptr<Terminal> s):
|
|
|
|
buffer(config->dte_buffer_size),
|
|
|
|
cmux_term(nullptr), primary_term(std::move(t)), secondary_term(std::move(s)),
|
2023-06-23 19:35:49 +02:00
|
|
|
mode(modem_mode::UNDEF)
|
|
|
|
{
|
|
|
|
set_command_callbacks();
|
|
|
|
}
|
2023-03-07 13:35:34 +01:00
|
|
|
|
|
|
|
DTE::DTE(std::unique_ptr<Terminal> t, std::unique_ptr<Terminal> s):
|
|
|
|
buffer(dte_default_buffer_size),
|
|
|
|
cmux_term(nullptr), primary_term(std::move(t)), secondary_term(std::move(s)),
|
2023-06-23 19:35:49 +02:00
|
|
|
mode(modem_mode::UNDEF)
|
|
|
|
{
|
|
|
|
set_command_callbacks();
|
|
|
|
}
|
2023-03-07 13:35:34 +01:00
|
|
|
|
2023-06-23 19:35:49 +02:00
|
|
|
void DTE::set_command_callbacks()
|
2021-02-26 18:32:15 +01:00
|
|
|
{
|
2023-06-23 19:35:49 +02:00
|
|
|
primary_term->set_read_cb([this](uint8_t *data, size_t len) {
|
|
|
|
Scoped<Lock> l(command_cb.line_lock);
|
|
|
|
if (command_cb.got_line == nullptr) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (data) {
|
|
|
|
// For terminals which post data directly with the callback (CMUX)
|
|
|
|
// we cannot defragment unless we allocate, but
|
|
|
|
// we'll try to process the data on the actual buffer
|
|
|
|
#ifdef CONFIG_ESP_MODEM_USE_INFLATABLE_BUFFER_IF_NEEDED
|
|
|
|
if (inflatable.consumed != 0) {
|
|
|
|
inflatable.grow(inflatable.consumed + len);
|
|
|
|
std::memcpy(inflatable.current(), data, len);
|
|
|
|
data = inflatable.begin();
|
|
|
|
}
|
|
|
|
if (command_cb.process_line(data, inflatable.consumed, len)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// at this point we're sure that the data processing hasn't finished,
|
|
|
|
// and we have to grow the inflatable buffer (if enabled) or give up
|
|
|
|
if (inflatable.consumed == 0) {
|
|
|
|
inflatable.grow(len);
|
|
|
|
std::memcpy(inflatable.begin(), data, len);
|
|
|
|
}
|
|
|
|
inflatable.consumed += len;
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
if (command_cb.process_line(data, 0, len)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// cannot inflate and the processing hasn't finishes in the first iteration -> report a failure
|
|
|
|
command_cb.give_up();
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
// data == nullptr: Terminals which request users to read current data
|
|
|
|
// we're able to use DTE's buffer to defragment it; as long as we consume less that the buffer size
|
|
|
|
if (buffer.size > buffer.consumed) {
|
2021-05-17 14:59:03 +02:00
|
|
|
data = buffer.get();
|
2022-10-25 16:50:34 +02:00
|
|
|
len = primary_term->read(data + buffer.consumed, buffer.size - buffer.consumed);
|
2023-06-23 19:35:49 +02:00
|
|
|
if (command_cb.process_line(data, buffer.consumed, len)) {
|
2021-03-04 20:19:18 +01:00
|
|
|
return true;
|
2021-02-26 18:32:15 +01:00
|
|
|
}
|
2023-06-23 19:35:49 +02:00
|
|
|
buffer.consumed += len;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// we have used the entire DTE's buffer, need to use the inflatable buffer to continue
|
|
|
|
#ifdef CONFIG_ESP_MODEM_USE_INFLATABLE_BUFFER_IF_NEEDED
|
|
|
|
if (inflatable.consumed == 0) {
|
|
|
|
inflatable.grow(buffer.size + len);
|
|
|
|
std::memcpy(inflatable.begin(), buffer.get(), buffer.size);
|
|
|
|
inflatable.consumed = buffer.size;
|
|
|
|
} else {
|
|
|
|
inflatable.grow(inflatable.consumed + len);
|
|
|
|
}
|
|
|
|
len = primary_term->read(inflatable.current(), len);
|
|
|
|
if (command_cb.process_line(inflatable.begin(), inflatable.consumed, len)) {
|
|
|
|
return true;
|
2021-02-26 18:32:15 +01:00
|
|
|
}
|
2023-06-23 19:35:49 +02:00
|
|
|
inflatable.consumed += len;
|
2021-03-04 20:19:18 +01:00
|
|
|
return false;
|
2023-06-23 19:35:49 +02:00
|
|
|
#else
|
|
|
|
// cannot inflate -> report a failure
|
|
|
|
command_cb.give_up();
|
|
|
|
return true;
|
|
|
|
#endif
|
2021-02-26 18:32:15 +01:00
|
|
|
});
|
2023-09-20 12:06:47 +02:00
|
|
|
primary_term->set_error_cb([this](terminal_error err) {
|
|
|
|
if (user_error_cb) {
|
|
|
|
user_error_cb(err);
|
|
|
|
}
|
|
|
|
handle_error(err);
|
|
|
|
});
|
|
|
|
secondary_term->set_error_cb([this](terminal_error err) {
|
|
|
|
if (user_error_cb) {
|
|
|
|
user_error_cb(err);
|
|
|
|
}
|
|
|
|
handle_error(err);
|
|
|
|
});
|
|
|
|
|
2023-06-23 19:35:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
command_result DTE::command(const std::string &command, got_line_cb got_line, uint32_t time_ms, const char separator)
|
|
|
|
{
|
|
|
|
Scoped<Lock> l1(internal_lock);
|
|
|
|
command_cb.set(got_line, separator);
|
2022-10-25 16:50:34 +02:00
|
|
|
primary_term->write((uint8_t *)command.c_str(), command.length());
|
2023-06-23 19:35:49 +02:00
|
|
|
command_cb.wait_for_line(time_ms);
|
|
|
|
command_cb.set(nullptr);
|
2022-06-10 18:04:10 +02:00
|
|
|
buffer.consumed = 0;
|
2023-06-23 19:35:49 +02:00
|
|
|
#ifdef CONFIG_ESP_MODEM_USE_INFLATABLE_BUFFER_IF_NEEDED
|
|
|
|
inflatable.deflate();
|
|
|
|
#endif
|
|
|
|
return command_cb.result;
|
2021-04-06 08:33:40 +02:00
|
|
|
}
|
|
|
|
|
2021-04-18 19:14:22 +02:00
|
|
|
command_result DTE::command(const std::string &cmd, got_line_cb got_line, uint32_t time_ms)
|
|
|
|
{
|
|
|
|
return command(cmd, got_line, time_ms, '\n');
|
|
|
|
}
|
|
|
|
|
2022-05-26 17:31:22 +02:00
|
|
|
bool DTE::exit_cmux()
|
|
|
|
{
|
2022-06-10 18:04:10 +02:00
|
|
|
if (!cmux_term->deinit()) {
|
2022-05-26 17:31:22 +02:00
|
|
|
return false;
|
|
|
|
}
|
2023-08-18 12:44:37 +03:00
|
|
|
exit_cmux_internal();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DTE::exit_cmux_internal()
|
|
|
|
{
|
|
|
|
if (!cmux_term) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-06-10 18:04:10 +02:00
|
|
|
auto ejected = cmux_term->detach();
|
|
|
|
// return the ejected terminal and buffer back to this DTE
|
2022-10-25 16:50:34 +02:00
|
|
|
primary_term = std::move(ejected.first);
|
2022-06-10 18:04:10 +02:00
|
|
|
buffer = std::move(ejected.second);
|
2022-10-25 16:50:34 +02:00
|
|
|
secondary_term = primary_term;
|
2023-06-23 19:35:49 +02:00
|
|
|
set_command_callbacks();
|
2022-05-26 17:31:22 +02:00
|
|
|
}
|
|
|
|
|
2021-04-15 17:23:37 +02:00
|
|
|
bool DTE::setup_cmux()
|
2021-04-06 08:33:40 +02:00
|
|
|
{
|
2022-10-25 16:50:34 +02:00
|
|
|
cmux_term = std::make_shared<CMux>(primary_term, std::move(buffer));
|
2021-06-01 10:21:51 +02:00
|
|
|
if (cmux_term == nullptr) {
|
2021-04-15 17:23:37 +02:00
|
|
|
return false;
|
2021-06-01 10:21:51 +02:00
|
|
|
}
|
2023-08-18 12:44:37 +03:00
|
|
|
|
2021-06-01 10:21:51 +02:00
|
|
|
if (!cmux_term->init()) {
|
2023-08-18 12:44:37 +03:00
|
|
|
exit_cmux_internal();
|
|
|
|
cmux_term = nullptr;
|
2021-04-15 17:23:37 +02:00
|
|
|
return false;
|
2021-06-01 10:21:51 +02:00
|
|
|
}
|
2023-08-18 12:44:37 +03:00
|
|
|
|
|
|
|
primary_term = std::make_unique<CMuxInstance>(cmux_term, 0);
|
|
|
|
secondary_term = std::make_unique<CMuxInstance>(cmux_term, 1);
|
|
|
|
if (primary_term == nullptr || secondary_term == nullptr) {
|
|
|
|
exit_cmux_internal();
|
|
|
|
cmux_term = nullptr;
|
2021-04-15 17:23:37 +02:00
|
|
|
return false;
|
2021-06-01 10:21:51 +02:00
|
|
|
}
|
2023-06-23 19:35:49 +02:00
|
|
|
set_command_callbacks();
|
2021-04-15 17:23:37 +02:00
|
|
|
return true;
|
2021-04-15 09:46:28 +02:00
|
|
|
}
|
2021-04-18 19:34:58 +02:00
|
|
|
|
|
|
|
bool DTE::set_mode(modem_mode m)
|
|
|
|
{
|
2022-06-08 17:16:15 +02:00
|
|
|
// transitions (COMMAND|UNDEF) -> CMUX
|
|
|
|
if (m == modem_mode::CMUX_MODE) {
|
|
|
|
if (mode == modem_mode::UNDEF || mode == modem_mode::COMMAND_MODE) {
|
|
|
|
if (setup_cmux()) {
|
|
|
|
mode = m;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
mode = modem_mode::UNDEF;
|
|
|
|
return false;
|
2022-05-26 17:31:22 +02:00
|
|
|
}
|
|
|
|
}
|
2023-03-07 13:35:34 +01:00
|
|
|
// transitions (COMMAND|DUAL|CMUX|UNDEF) -> DATA
|
2021-04-18 19:34:58 +02:00
|
|
|
if (m == modem_mode::DATA_MODE) {
|
2023-03-07 13:35:34 +01:00
|
|
|
if (mode == modem_mode::CMUX_MODE || mode == modem_mode::CMUX_MANUAL_MODE || mode == modem_mode::DUAL_MODE) {
|
2022-10-25 15:41:48 +02:00
|
|
|
// mode stays the same, but need to swap terminals (as command has been switched)
|
2022-10-25 16:50:34 +02:00
|
|
|
secondary_term.swap(primary_term);
|
2023-06-23 19:35:49 +02:00
|
|
|
set_command_callbacks();
|
2022-06-08 17:16:15 +02:00
|
|
|
} else {
|
|
|
|
mode = m;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2023-03-07 13:35:34 +01:00
|
|
|
// transitions (DATA|DUAL|CMUX|UNDEF) -> COMMAND
|
2022-06-08 17:16:15 +02:00
|
|
|
if (m == modem_mode::COMMAND_MODE) {
|
|
|
|
if (mode == modem_mode::CMUX_MODE) {
|
|
|
|
if (exit_cmux()) {
|
|
|
|
mode = m;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
mode = modem_mode::UNDEF;
|
|
|
|
return false;
|
2023-03-07 13:35:34 +01:00
|
|
|
} if (mode == modem_mode::CMUX_MANUAL_MODE || mode == modem_mode::DUAL_MODE) {
|
2022-10-25 15:41:48 +02:00
|
|
|
return true;
|
2022-06-08 17:16:15 +02:00
|
|
|
} else {
|
|
|
|
mode = m;
|
|
|
|
return true;
|
2021-04-18 19:34:58 +02:00
|
|
|
}
|
|
|
|
}
|
2022-10-25 15:41:48 +02:00
|
|
|
// manual CMUX transitions: Enter CMUX
|
|
|
|
if (m == modem_mode::CMUX_MANUAL_MODE) {
|
|
|
|
if (setup_cmux()) {
|
|
|
|
mode = m;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
mode = modem_mode::UNDEF;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// manual CMUX transitions: Exit CMUX
|
|
|
|
if (m == modem_mode::CMUX_MANUAL_EXIT && mode == modem_mode::CMUX_MANUAL_MODE) {
|
|
|
|
if (exit_cmux()) {
|
|
|
|
mode = modem_mode::COMMAND_MODE;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
mode = modem_mode::UNDEF;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// manual CMUX transitions: Swap terminals
|
|
|
|
if (m == modem_mode::CMUX_MANUAL_SWAP && mode == modem_mode::CMUX_MANUAL_MODE) {
|
2022-10-25 16:50:34 +02:00
|
|
|
secondary_term.swap(primary_term);
|
2023-06-23 19:35:49 +02:00
|
|
|
set_command_callbacks();
|
2022-10-25 15:41:48 +02:00
|
|
|
return true;
|
|
|
|
}
|
2022-10-25 16:50:34 +02:00
|
|
|
mode = modem_mode::UNDEF;
|
|
|
|
return false;
|
2021-04-18 19:34:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DTE::set_read_cb(std::function<bool(uint8_t *, size_t)> f)
|
|
|
|
{
|
2023-06-23 19:35:49 +02:00
|
|
|
if (f == nullptr) {
|
|
|
|
set_command_callbacks();
|
|
|
|
return;
|
|
|
|
}
|
2021-04-18 19:34:58 +02:00
|
|
|
on_data = std::move(f);
|
2022-10-25 16:50:34 +02:00
|
|
|
secondary_term->set_read_cb([this](uint8_t *data, size_t len) {
|
2021-04-18 19:34:58 +02:00
|
|
|
if (!data) { // if no data available from terminal callback -> need to explicitly read some
|
|
|
|
data = buffer.get();
|
2022-10-25 16:50:34 +02:00
|
|
|
len = secondary_term->read(buffer.get(), buffer.size);
|
2021-04-18 19:34:58 +02:00
|
|
|
}
|
2021-06-01 10:21:51 +02:00
|
|
|
if (on_data) {
|
2021-04-18 19:34:58 +02:00
|
|
|
return on_data(data, len);
|
2021-06-01 10:21:51 +02:00
|
|
|
}
|
2021-04-18 19:34:58 +02:00
|
|
|
return false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-09-02 09:14:38 +02:00
|
|
|
void DTE::set_error_cb(std::function<void(terminal_error err)> f)
|
|
|
|
{
|
2023-09-20 12:06:47 +02:00
|
|
|
user_error_cb = std::move(f);
|
|
|
|
set_command_callbacks();
|
2022-09-02 09:14:38 +02:00
|
|
|
}
|
|
|
|
|
2021-04-18 19:34:58 +02:00
|
|
|
int DTE::read(uint8_t **d, size_t len)
|
|
|
|
{
|
2022-06-10 18:04:10 +02:00
|
|
|
auto data_to_read = std::min(len, buffer.size);
|
2021-04-18 19:34:58 +02:00
|
|
|
auto data = buffer.get();
|
2022-10-25 16:50:34 +02:00
|
|
|
auto actual_len = secondary_term->read(data, data_to_read);
|
2021-04-18 19:34:58 +02:00
|
|
|
*d = data;
|
|
|
|
return actual_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DTE::write(uint8_t *data, size_t len)
|
|
|
|
{
|
2022-10-25 16:50:34 +02:00
|
|
|
return secondary_term->write(data, len);
|
2022-06-10 18:04:10 +02:00
|
|
|
}
|
|
|
|
|
2023-03-16 19:17:53 +01:00
|
|
|
int DTE::write(DTE_Command command)
|
|
|
|
{
|
|
|
|
return primary_term->write(command.data, command.len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DTE::on_read(got_line_cb on_read_cb)
|
|
|
|
{
|
|
|
|
if (on_read_cb == nullptr) {
|
|
|
|
primary_term->set_read_cb(nullptr);
|
|
|
|
internal_lock.unlock();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
internal_lock.lock();
|
|
|
|
primary_term->set_read_cb([this, on_read_cb](uint8_t *data, size_t len) {
|
|
|
|
if (!data) {
|
|
|
|
data = buffer.get();
|
|
|
|
len = primary_term->read(data, buffer.size);
|
|
|
|
}
|
|
|
|
auto res = on_read_cb(data, len);
|
|
|
|
if (res == command_result::OK || res == command_result::FAIL) {
|
|
|
|
primary_term->set_read_cb(nullptr);
|
|
|
|
internal_lock.unlock();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-06-23 19:35:49 +02:00
|
|
|
bool DTE::command_cb::process_line(uint8_t *data, size_t consumed, size_t len)
|
|
|
|
{
|
|
|
|
if (memchr(data + consumed, separator, len)) {
|
|
|
|
result = got_line(data, consumed + len);
|
|
|
|
if (result == command_result::OK || result == command_result::FAIL) {
|
|
|
|
signal.set(GOT_LINE);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-20 12:06:47 +02:00
|
|
|
bool DTE::recover()
|
|
|
|
{
|
|
|
|
if (mode == modem_mode::CMUX_MODE || mode == modem_mode::CMUX_MANUAL_MODE || mode == modem_mode::DUAL_MODE) {
|
|
|
|
return cmux_term->recover();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DTE::handle_error(terminal_error err)
|
2023-06-23 19:35:49 +02:00
|
|
|
{
|
2023-09-20 12:06:47 +02:00
|
|
|
if (err == terminal_error::BUFFER_OVERFLOW ||
|
|
|
|
err == terminal_error::CHECKSUM_ERROR ||
|
|
|
|
err == terminal_error::UNEXPECTED_CONTROL_FLOW) {
|
|
|
|
recover();
|
2023-06-23 19:35:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_ESP_MODEM_USE_INFLATABLE_BUFFER_IF_NEEDED
|
|
|
|
void DTE::extra_buffer::grow(size_t need_size)
|
|
|
|
{
|
|
|
|
if (need_size == 0) {
|
|
|
|
delete buffer;
|
|
|
|
buffer = nullptr;
|
|
|
|
} else if (buffer == nullptr) {
|
|
|
|
buffer = new std::vector<uint8_t>(need_size);
|
|
|
|
} else {
|
|
|
|
buffer->resize(need_size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-06-10 18:04:10 +02:00
|
|
|
/**
|
|
|
|
* Implemented here to keep all headers C++11 compliant
|
|
|
|
*/
|
|
|
|
unique_buffer::unique_buffer(size_t size):
|
|
|
|
data(std::make_unique<uint8_t[]>(size)), size(size), consumed(0) {}
|