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

425 lines
13 KiB
C++
Raw Normal View History

2021-03-29 19:34:45 +02:00
// Copyright 2021 Espressif Systems (Shanghai) PTE LTD
2021-03-04 20:19:18 +01:00
//
2021-03-29 19:34:45 +02:00
// 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
2021-03-04 20:19:18 +01:00
//
2021-03-29 19:34:45 +02:00
// 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-04 20:19:18 +01:00
#include <cstring>
2021-04-04 22:15:46 +02:00
#include <unistd.h>
#include <cxx_include/esp_modem_cmux.hpp>
2021-03-04 20:19:18 +01:00
#include "cxx_include/esp_modem_dte.hpp"
#include "esp_log.h"
2021-03-29 19:34:45 +02:00
using namespace esp_modem;
2021-05-17 14:59:03 +02:00
/**
* @brief Define this to defragment partially received data of CMUX payload
* This is useful if upper layers expect the entire payload available
* for parsing.
*/
#define DEFRAGMENT_CMUX_PAYLOAD
2021-03-04 20:19:18 +01:00
#define EA 0x01 /* Extension bit */
#define CR 0x02 /* Command / Response */
#define PF 0x10 /* Poll / Final */
/* Frame types */
#define FT_RR 0x01 /* Receive Ready */
#define FT_UI 0x03 /* Unnumbered Information */
#define FT_RNR 0x05 /* Receive Not Ready */
#define FT_REJ 0x09 /* Reject */
#define FT_DM 0x0F /* Disconnected Mode */
#define FT_SABM 0x2F /* Set Asynchronous Balanced Mode */
#define FT_DISC 0x43 /* Disconnect */
#define FT_UA 0x63 /* Unnumbered Acknowledgement */
#define FT_UIH 0xEF /* Unnumbered Information with Header check */
/* Control channel commands */
#define CMD_NSC 0x08 /* Non Supported Command Response */
#define CMD_TEST 0x10 /* Test Command */
#define CMD_PSC 0x20 /* Power Saving Control */
#define CMD_RLS 0x28 /* Remote Line Status Command */
#define CMD_FCOFF 0x30 /* Flow Control Off Command */
#define CMD_PN 0x40 /* DLC parameter negotiation */
#define CMD_RPN 0x48 /* Remote Port Negotiation Command */
#define CMD_FCON 0x50 /* Flow Control On Command */
#define CMD_CLD 0x60 /* Multiplexer close down */
#define CMD_SNC 0x68 /* Service Negotiation Command */
#define CMD_MSC 0x70 /* Modem Status Command */
/* Flag sequence field between messages (start of frame) */
#define SOF_MARKER 0xF9
uint8_t CMux::fcs_crc(const uint8_t frame[6])
2021-03-04 20:19:18 +01:00
{
// #define FCS_GOOD_VALUE 0xCF
uint8_t crc = 0xFF; // FCS_INIT_VALUE
2021-03-04 20:19:18 +01:00
for (int i = 1; i < 4; i++) {
crc ^= frame[i];
2021-03-04 20:19:18 +01:00
for (int j = 0; j < 8; j++) {
if (crc & 0x01) {
crc = (crc >> 1) ^ 0xe0; // FCS_POLYNOMIAL
2021-03-04 20:19:18 +01:00
} else {
crc >>= 1;
2021-03-04 20:19:18 +01:00
}
}
}
return crc;
}
void CMux::close_down()
{
uint8_t frame[] = {
SOF_MARKER, 0x3, 0xFF, 0x5, 0xC3, 0x1, 0xE7, SOF_MARKER };
term->write(frame, 8);
}
void CMux::send_disc(size_t i)
{
uint8_t frame[6];
frame[0] = SOF_MARKER;
frame[1] = (i << 2) | 0x3;
frame[2] = FT_DISC | PF;
frame[3] = 1;
frame[4] = 0xFF - fcs_crc(frame);
frame[5] = SOF_MARKER;
term->write(frame, 6);
}
2021-05-17 14:59:03 +02:00
void CMux::send_sabm(size_t i)
2021-03-04 20:19:18 +01:00
{
uint8_t frame[6];
frame[0] = SOF_MARKER;
2021-05-17 14:59:03 +02:00
frame[1] = (i << 2) | 0x3;
2021-03-04 20:19:18 +01:00
frame[2] = FT_SABM | PF;
frame[3] = 1;
frame[4] = 0xFF - fcs_crc(frame);
2021-03-04 20:19:18 +01:00
frame[5] = SOF_MARKER;
term->write(frame, 6);
}
struct CMux::CMuxFrame {
uint8_t *ptr; /*!< pointer to the currently processing byte of the CMUX frame */
size_t len; /*!< length of available data in the current CMUX frame */
void advance(int size = 1)
{
ptr += size;
len -= size;
}
};
void CMux::data_available(uint8_t *data, size_t len)
2021-03-04 20:19:18 +01:00
{
if (data && (type&FT_UIH) == FT_UIH && len > 0 && dlci > 0) { // valid payload on a virtual term
int virtual_term = dlci - 1;
if (virtual_term < MAX_TERMINALS_NUM && read_cb[virtual_term]) {
2021-05-17 14:59:03 +02:00
// Post partial data (or defragment to post on CMUX footer)
#ifdef DEFRAGMENT_CMUX_PAYLOAD
if (payload_start == nullptr) {
payload_start = data;
total_payload_size = 0;
}
total_payload_size += len;
#else
read_cb[virtual_term](data, len);
2021-05-17 14:59:03 +02:00
#endif
}
} else if (data == nullptr && type == 0x73 && len == 0) { // notify the initial SABM command
2021-04-15 17:23:37 +02:00
Scoped<Lock> l(lock);
sabm_ack = dlci;
} else if (data == nullptr) {
int virtual_term = dlci - 1;
if (virtual_term < MAX_TERMINALS_NUM && read_cb[virtual_term]) {
2021-05-17 14:59:03 +02:00
#ifdef DEFRAGMENT_CMUX_PAYLOAD
read_cb[virtual_term](payload_start, total_payload_size);
#endif
}
}
}
bool CMux::on_init(CMuxFrame &frame)
{
if (frame.ptr[0] != SOF_MARKER) {
ESP_LOGW("CMUX", "Protocol mismatch: Missed leading SOF, recovering...");
state = cmux_state::RECOVER;
return true;
}
if (frame.len > 1 && frame.ptr[1] == SOF_MARKER) {
// empty frame
frame.advance();
return true;
}
state = cmux_state::HEADER;
frame.advance();
frame_header_offset = 1;
return true;
}
bool CMux::on_recovery(CMuxFrame &frame)
{
uint8_t *recover_ptr;
if (frame.ptr[0] == SOF_MARKER) {
// already init state
state = cmux_state::INIT;
return true;
}
recover_ptr = static_cast<uint8_t *>(memchr(frame.ptr, SOF_MARKER, frame.len));
if (recover_ptr && frame.len > recover_ptr - frame.ptr) {
frame.len -= (recover_ptr - frame.ptr);
frame.ptr = recover_ptr;
state = cmux_state::INIT;
ESP_LOGI("CMUX", "Protocol recovered");
if (frame.len > 1 && frame.ptr[1] == SOF_MARKER) {
// empty frame
frame.advance();
}
return true;
}
// marker not found, continue with recovery
return false;
}
bool CMux::on_header(CMuxFrame &frame)
{
if (frame.len > 0 && frame_header_offset == 1 && frame.ptr[0] == SOF_MARKER) {
// Previously trailing SOF interpreted as heading SOF, remove it and restart HEADER
frame.advance();
return true;
}
if (frame.len + frame_header_offset < 4) {
memcpy(frame_header + frame_header_offset, frame.ptr, frame.len);
frame_header_offset += frame.len;
return false; // need read more
}
size_t payload_offset = std::min(frame.len, 4 - frame_header_offset);
memcpy(frame_header + frame_header_offset, frame.ptr, payload_offset);
if ((frame_header[3] & 1) == 0) {
if (frame_header_offset + frame.len <= 4) {
frame_header_offset += frame.len;
return false; // need read more
}
payload_offset = std::min(frame.len, 5 - frame_header_offset);
memcpy(frame_header + frame_header_offset, frame.ptr, payload_offset);
payload_len = frame_header[4] << 7;
frame_header_offset += payload_offset - 1; // rewind frame_header back to hold only 6 bytes size
} else {
payload_len = 0;
frame_header_offset += payload_offset;
}
dlci = frame_header[1] >> 2;
type = frame_header[2];
payload_len += (frame_header[3] >> 1);
frame.advance(payload_offset);
state = cmux_state::PAYLOAD;
return true;
}
bool CMux::on_payload(CMuxFrame &frame)
{
ESP_LOGD("CMUX", "Payload frame: dlci:%02x type:%02x payload:%d available:%d", dlci, type, payload_len, frame.len);
if (frame.len < payload_len) { // payload
state = cmux_state::PAYLOAD;
data_available(frame.ptr, frame.len); // partial read
payload_len -= frame.len;
return false;
} else { // complete
if (payload_len > 0) {
data_available(&frame.ptr[0], payload_len); // rest read
}
frame.advance((payload_len));
state = cmux_state::FOOTER;
payload_len = 0;
}
return true;
}
bool CMux::on_footer(CMuxFrame &frame)
{
size_t footer_offset = 0;
if (frame.len + frame_header_offset < 6) {
memcpy(frame_header + frame_header_offset, frame.ptr, frame.len);
frame_header_offset += frame.len;
return false; // need read more
} else {
footer_offset = std::min(frame.len, 6 - frame_header_offset);
memcpy(frame_header + frame_header_offset, frame.ptr, footer_offset);
if (frame_header[5] != SOF_MARKER) {
ESP_LOGW("CMUX", "Protocol mismatch: Missed trailing SOF, recovering...");
payload_start = nullptr;
total_payload_size = 0;
state = cmux_state::RECOVER;
return true;
}
frame.advance(footer_offset);
state = cmux_state::INIT;
frame_header_offset = 0;
data_available(nullptr, 0);
payload_start = nullptr;
total_payload_size = 0;
2021-03-04 20:19:18 +01:00
}
return true;
2021-03-04 20:19:18 +01:00
}
bool CMux::on_cmux(uint8_t *data, size_t actual_len)
2021-03-04 20:19:18 +01:00
{
if (!data) {
2021-05-17 14:59:03 +02:00
#ifdef DEFRAGMENT_CMUX_PAYLOAD
auto data_to_read = buffer_size - 128; // keep 128 (max CMUX payload) backup buffer)
if (payload_start) {
data = payload_start + total_payload_size;
data_to_read = payload_len + 2;
} else {
data = buffer.get();
}
actual_len = term->read(data, data_to_read);
2021-05-17 14:59:03 +02:00
#else
data = buffer.get();
actual_len = term->read(data, buffer_size);
#endif
2021-05-17 14:59:03 +02:00
}
ESP_LOG_BUFFER_HEXDUMP("CMUX Received", data, actual_len, ESP_LOG_VERBOSE);
CMuxFrame frame = { .ptr = data, .len = actual_len };
while (frame.len > 0) {
2021-03-04 20:19:18 +01:00
switch (state) {
2021-06-01 10:21:51 +02:00
case cmux_state::RECOVER:
if (!on_recovery(frame)) {
return false;
2021-06-01 10:21:51 +02:00
}
break;
2021-06-01 10:21:51 +02:00
case cmux_state::INIT:
if (!on_init(frame)) {
return false;
2021-06-01 10:21:51 +02:00
}
break;
case cmux_state::HEADER:
if (!on_header(frame)) {
return false;
2021-06-01 10:21:51 +02:00
}
break;
case cmux_state::PAYLOAD:
if (!on_payload(frame)) {
2021-06-01 10:21:51 +02:00
return false;
}
break;
case cmux_state::FOOTER:
if (!on_footer(frame)) {
return false;
2021-06-01 10:21:51 +02:00
}
break;
2021-03-04 20:19:18 +01:00
}
}
return true;
}
2021-04-15 17:23:37 +02:00
bool CMux::exit_cmux_protocol()
{
sabm_ack = -1;
for (size_t i = 1; i < 3; i++) {
int timeout = 0;
send_disc(i);
while (true) {
usleep(10'000);
Scoped<Lock> l(lock);
if (sabm_ack == i) {
sabm_ack = -1;
break;
}
if (timeout++ > 100) {
return false;
}
}
}
close_down();
usleep(100'000);
term->set_read_cb(nullptr);
return true;
}
2021-04-15 17:23:37 +02:00
bool CMux::init()
2021-03-04 20:19:18 +01:00
{
frame_header_offset = 0;
state = cmux_state::INIT;
term->set_read_cb([this](uint8_t *data, size_t len) {
this->on_cmux(data, len);
2021-03-06 16:56:50 +01:00
return false;
});
2021-03-04 20:19:18 +01:00
2021-04-15 17:23:37 +02:00
sabm_ack = -1;
2021-06-01 10:21:51 +02:00
for (size_t i = 0; i < 3; i++) {
2021-04-15 17:23:37 +02:00
int timeout = 0;
2021-03-04 20:19:18 +01:00
send_sabm(i);
2021-05-17 14:59:03 +02:00
while (true) {
2021-04-15 17:23:37 +02:00
usleep(10'000);
Scoped<Lock> l(lock);
if (sabm_ack == i) {
sabm_ack = -1;
break;
}
2021-06-01 10:21:51 +02:00
if (timeout++ > 100) {
2021-04-15 17:23:37 +02:00
return false;
2021-06-01 10:21:51 +02:00
}
2021-04-15 17:23:37 +02:00
}
2021-03-04 20:19:18 +01:00
}
2021-04-15 17:23:37 +02:00
return true;
2021-03-04 20:19:18 +01:00
}
int CMux::write(int virtual_term, uint8_t *data, size_t len)
{
2021-04-15 14:59:30 +02:00
const size_t cmux_max_len = 127;
Scoped<Lock> l(lock);
int i = virtual_term + 1;
2021-04-15 14:59:30 +02:00
size_t need_write = len;
while (need_write > 0) {
size_t batch_len = need_write;
2021-06-01 10:21:51 +02:00
if (batch_len > cmux_max_len) {
2021-04-15 14:59:30 +02:00
batch_len = cmux_max_len;
2021-06-01 10:21:51 +02:00
}
2021-04-15 14:59:30 +02:00
uint8_t frame[6];
frame[0] = SOF_MARKER;
frame[1] = (i << 2) + 1;
frame[2] = FT_UIH;
frame[3] = (batch_len << 1) + 1;
frame[4] = 0xFF - fcs_crc(frame);
frame[5] = SOF_MARKER;
2021-03-05 21:18:38 +01:00
2021-04-15 14:59:30 +02:00
term->write(frame, 4);
term->write(data, batch_len);
term->write(frame + 4, 2);
ESP_LOG_BUFFER_HEXDUMP("Send", frame, 4, ESP_LOG_VERBOSE);
ESP_LOG_BUFFER_HEXDUMP("Send", data, batch_len, ESP_LOG_VERBOSE);
2021-06-01 10:21:51 +02:00
ESP_LOG_BUFFER_HEXDUMP("Send", frame + 4, 2, ESP_LOG_VERBOSE);
2021-04-15 14:59:30 +02:00
need_write -= batch_len;
data += batch_len;
}
return len;
}
2021-06-01 10:21:51 +02:00
void CMux::set_read_cb(int inst, std::function<bool(uint8_t *, size_t)> f)
{
if (inst < MAX_TERMINALS_NUM) {
read_cb[inst] = std::move(f);
2021-06-01 10:21:51 +02:00
}
}
std::tuple<std::unique_ptr<Terminal>, std::unique_ptr<uint8_t[]>, size_t> esp_modem::CMux::deinit_and_eject()
{
if (exit_cmux_protocol()) {
return std::make_tuple(std::move(term), std::move(buffer), buffer_size);
}
return std::tuple(nullptr, nullptr, 0);
}