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

229 lines
8.2 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-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::send_sabm(size_t dlci)
2021-03-04 20:19:18 +01:00
{
uint8_t frame[6];
frame[0] = SOF_MARKER;
frame[1] = (dlci << 2) | 0x3;
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);
}
void CMux::data_available(uint8_t *data, size_t len)
2021-03-04 20:19:18 +01:00
{
if (type == 0xFF && len > 0 && dlci > 0) {
int virtual_term = dlci - 1;
if (virtual_term < max_terms && read_cb[virtual_term])
read_cb[virtual_term](data, len);
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) {
auto data_to_read = std::min(actual_len, buffer_size);
data = buffer.get();
actual_len = term->read(data, data_to_read);
2021-03-04 20:19:18 +01:00
}
ESP_LOG_BUFFER_HEXDUMP("Received", data, actual_len, ESP_LOG_DEBUG);
2021-03-04 20:19:18 +01:00
uint8_t* frame = data;
auto available_len = actual_len;
2021-03-04 20:19:18 +01:00
size_t payload_offset = 0;
size_t footer_offset = 0;
while (available_len > 0) {
switch (state) {
2021-03-06 16:56:50 +01:00
case cmux_state::RECOVER:
// TODO: Implement recovery, looking for SOF_MARKER's
break;
2021-03-04 20:19:18 +01:00
case cmux_state::INIT:
if (frame[0] != SOF_MARKER) {
ESP_LOGW("CMUX", "TODO: Recover!");
return true;
}
if (available_len > 1 && frame[1] == SOF_MARKER) {
ESP_LOGI("CMUX", "Empty frame");
// empty frame
available_len -= 2;
frame += 2;
break;
}
2021-03-04 20:19:18 +01:00
state = cmux_state::HEADER;
available_len--;
frame_header_offset = 1;
frame++;
break;
case cmux_state::HEADER:
if (available_len + frame_header_offset < 4) {
memcpy(frame_header + frame_header_offset, frame, available_len);
frame_header_offset += available_len;
return false; // need read more
}
payload_offset = std::min(available_len, 4 - frame_header_offset);
memcpy(frame_header + frame_header_offset, frame, payload_offset);
frame_header_offset += payload_offset;
dlci = frame_header[1] >> 2;
type = frame_header[2];
payload_len = (frame_header[3] >> 1);
ESP_LOGI("CMUX", "CMUX FR: A:%02x T:%02x L:%d", dlci, type, payload_len);
2021-03-04 20:19:18 +01:00
frame += payload_offset;
available_len -= payload_offset;
state = cmux_state::PAYLOAD;
break;
case cmux_state::PAYLOAD:
if (available_len < payload_len) { // payload
state = cmux_state::PAYLOAD;
data_available(frame, available_len); // partial read
2021-03-04 20:19:18 +01:00
payload_len -= available_len;
return false;
} else { // complete
if (payload_len > 0) {
data_available(&frame[0], payload_len); // rest read
2021-03-04 20:19:18 +01:00
}
available_len -= payload_len;
frame += payload_len;
state = cmux_state::FOOTER;
}
break;
case cmux_state::FOOTER:
if (available_len + frame_header_offset < 6) {
memcpy(frame_header + frame_header_offset, frame, available_len);
frame_header_offset += available_len;
return false; // need read more
} else {
footer_offset = std::min(available_len, 6 - frame_header_offset);
memcpy(frame_header + frame_header_offset, frame, footer_offset);
if (frame_header[5] != SOF_MARKER) {
ESP_LOGW("CMUX", "TODO: Recover!");
return true;
}
if (payload_len == 0) {
data_available(frame_header, 0); // Null payload
2021-03-04 20:19:18 +01:00
}
frame += footer_offset;
available_len -= footer_offset;
state = cmux_state::INIT;
frame_header_offset = 0;
}
break;
}
}
return true;
}
void 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
for (size_t i = 0; i < 3; i++)
{
send_sabm(i);
2021-04-04 22:15:46 +02:00
usleep(100'000);
2021-03-04 20:19:18 +01:00
}
}
int CMux::write(int virtual_term, uint8_t *data, size_t len)
{
Scoped<Lock> l(lock);
int i = virtual_term + 1;
2021-03-05 21:18:38 +01:00
uint8_t frame[6];
frame[0] = SOF_MARKER;
frame[1] = (i << 2) + 1;
frame[2] = FT_UIH;
frame[3] = (len << 1) + 1;
frame[4] = 0xFF - fcs_crc(frame);
2021-03-05 21:18:38 +01:00
frame[5] = SOF_MARKER;
term->write(frame, 4);
term->write(data, len);
term->write(frame + 4, 2);
ESP_LOG_BUFFER_HEXDUMP("Send", frame, 4, ESP_LOG_VERBOSE);
ESP_LOG_BUFFER_HEXDUMP("Send", data, len, ESP_LOG_VERBOSE);
ESP_LOG_BUFFER_HEXDUMP("Send", frame+4, 2, ESP_LOG_VERBOSE);
return len;
}
void CMux::set_read_cb(int inst, std::function<bool(uint8_t *, size_t)> f) {
if (inst < max_terms)
read_cb[inst] = std::move(f);
}