2021-02-26 18:32:15 +01:00
|
|
|
#ifndef SIMPLE_CXX_CLIENT_ESP_MODEM_DTE_HPP
|
|
|
|
#define SIMPLE_CXX_CLIENT_ESP_MODEM_DTE_HPP
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <functional>
|
|
|
|
#include <exception>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <utility>
|
|
|
|
#include "esp_err.h"
|
|
|
|
#include "terminal_objects.hpp"
|
2021-02-26 20:23:29 +01:00
|
|
|
#include "ppp_netif.hpp"
|
2021-02-26 18:32:15 +01:00
|
|
|
|
|
|
|
|
|
|
|
enum class terminal_error {
|
|
|
|
BUFFER_OVERFLOW,
|
|
|
|
CHECKSUM_ERROR,
|
|
|
|
UNEXPECTED_CONTROL_FLOW,
|
|
|
|
};
|
|
|
|
|
|
|
|
class terminal {
|
|
|
|
public:
|
|
|
|
virtual ~terminal() = default;
|
|
|
|
void set_data_cb(std::function<void(size_t len)> f) { on_data = std::move(f); }
|
|
|
|
void set_error_cb(std::function<void(terminal_error)> f) { on_error = std::move(f); }
|
|
|
|
virtual int write(uint8_t *data, size_t len) = 0;
|
|
|
|
virtual int read(uint8_t *data, size_t len) = 0;
|
|
|
|
virtual void start() = 0;
|
|
|
|
virtual void stop() = 0;
|
|
|
|
|
|
|
|
protected:
|
|
|
|
std::function<void(size_t len)> on_data;
|
|
|
|
std::function<void(terminal_error)> on_error;
|
|
|
|
};
|
|
|
|
|
|
|
|
class dte_adapter: public terminal {
|
|
|
|
public:
|
|
|
|
dte_adapter(std::unique_ptr<terminal> terminal):
|
|
|
|
original_dte(std::move(terminal)) {}
|
|
|
|
~dte_adapter() override = default;
|
|
|
|
int write(uint8_t *data, size_t len) override { return original_dte->write(data, len); }
|
|
|
|
int read(uint8_t *data, size_t len) override { return original_dte->read(data, len); }
|
|
|
|
private:
|
|
|
|
std::unique_ptr<terminal> original_dte;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
enum class dte_mode {
|
|
|
|
UNDEF,
|
|
|
|
COMMAND_MODE,
|
|
|
|
DATA_MODE
|
|
|
|
};
|
|
|
|
|
2021-03-03 20:35:08 +01:00
|
|
|
enum class command_result {
|
|
|
|
OK,
|
|
|
|
FAIL,
|
|
|
|
TIMEOUT
|
|
|
|
};
|
|
|
|
|
2021-02-26 18:32:15 +01:00
|
|
|
const int DTE_BUFFER_SIZE = 1024;
|
|
|
|
|
|
|
|
|
2021-03-03 20:35:08 +01:00
|
|
|
typedef std::function<command_result(uint8_t *data, size_t len)> got_line_cb;
|
2021-02-26 18:32:15 +01:00
|
|
|
|
2021-03-03 20:35:08 +01:00
|
|
|
class DTE {
|
2021-02-26 18:32:15 +01:00
|
|
|
public:
|
2021-03-03 20:35:08 +01:00
|
|
|
explicit DTE(std::unique_ptr<terminal> t);
|
|
|
|
~DTE() = default;
|
2021-02-26 18:32:15 +01:00
|
|
|
// void set_line_cb(got_line f) { on_line_cb = std::move(f); }
|
2021-02-26 20:23:29 +01:00
|
|
|
int write(uint8_t *data, size_t len) { return term->write(data, len); }
|
2021-02-27 09:32:14 +01:00
|
|
|
int read(uint8_t **d, size_t len) {
|
|
|
|
auto data_to_read = std::min(len, buffer_size);
|
|
|
|
auto data = buffer.get();
|
|
|
|
auto actual_len = term->read(data, data_to_read);
|
|
|
|
*d = data;
|
|
|
|
return actual_len;
|
|
|
|
}
|
|
|
|
void set_data_cb(std::function<void(size_t len)> f) { on_data = std::move(f); }
|
2021-02-26 18:32:15 +01:00
|
|
|
|
2021-02-26 20:23:29 +01:00
|
|
|
void start() { term->start(); }
|
|
|
|
void data_mode_closed() { term->stop(); }
|
2021-02-27 09:32:14 +01:00
|
|
|
void set_mode(dte_mode m) {
|
|
|
|
term->start(); mode = m;
|
|
|
|
if (m == dte_mode::DATA_MODE) {
|
|
|
|
term->set_data_cb(on_data);
|
|
|
|
}
|
|
|
|
}
|
2021-03-03 20:35:08 +01:00
|
|
|
command_result command(const std::string& command, got_line_cb got_line, uint32_t time_ms);
|
2021-02-26 18:32:15 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
const size_t GOT_LINE = BIT0;
|
|
|
|
size_t buffer_size;
|
|
|
|
size_t consumed;
|
|
|
|
std::unique_ptr<uint8_t[]> buffer;
|
|
|
|
std::unique_ptr<terminal> term;
|
|
|
|
got_line_cb on_line;
|
|
|
|
dte_mode mode;
|
|
|
|
signal_group signal;
|
2021-02-27 09:32:14 +01:00
|
|
|
std::function<void(size_t len)> on_data;
|
2021-02-26 18:32:15 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2021-03-03 20:35:08 +01:00
|
|
|
|
2021-02-26 20:23:29 +01:00
|
|
|
|
2021-02-26 18:32:15 +01:00
|
|
|
|
|
|
|
|
|
|
|
#endif //SIMPLE_CXX_CLIENT_ESP_MODEM_DTE_HPP
|