mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
324 lines
8.8 KiB
C
324 lines
8.8 KiB
C
// Copyright 2015-2018 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.
|
|
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <esp_tls.h>
|
|
|
|
#include "sys/queue.h"
|
|
#include "esp_log.h"
|
|
|
|
#include "esp_transport.h"
|
|
#include "esp_transport_utils.h"
|
|
|
|
static const char *TAG = "TRANSPORT";
|
|
|
|
/**
|
|
* Transport layer structure, which will provide functions, basic properties for transport types
|
|
*/
|
|
struct esp_transport_item_t {
|
|
int port;
|
|
int socket; /*!< Socket to use in this transport */
|
|
char *scheme; /*!< Tag name */
|
|
void *context; /*!< Context data */
|
|
void *data; /*!< Additional transport data */
|
|
connect_func _connect; /*!< Connect function of this transport */
|
|
io_read_func _read; /*!< Read */
|
|
io_func _write; /*!< Write */
|
|
trans_func _close; /*!< Close */
|
|
poll_func _poll_read; /*!< Poll and read */
|
|
poll_func _poll_write; /*!< Poll and write */
|
|
trans_func _destroy; /*!< Destroy and free transport */
|
|
connect_async_func _connect_async; /*!< non-blocking connect function of this transport */
|
|
payload_transfer_func _parent_transfer; /*!< Function returning underlying transport layer */
|
|
esp_tls_error_handle_t error_handle; /*!< Pointer to esp-tls error handle */
|
|
esp_transport_keep_alive_t *keep_alive_cfg; /*!< TCP keep-alive config */
|
|
STAILQ_ENTRY(esp_transport_item_t) next;
|
|
};
|
|
|
|
|
|
/**
|
|
* This list will hold all transport available
|
|
*/
|
|
STAILQ_HEAD(esp_transport_list_t, esp_transport_item_t);
|
|
|
|
/**
|
|
* Internal transport structure holding list of transports and other data common to all transports
|
|
*/
|
|
typedef struct esp_transport_internal {
|
|
struct esp_transport_list_t list; /*!< List of transports */
|
|
esp_tls_error_handle_t error_handle; /*!< Pointer to the error tracker if enabled */
|
|
} esp_transport_internal_t;
|
|
|
|
static esp_transport_handle_t esp_transport_get_default_parent(esp_transport_handle_t t)
|
|
{
|
|
/*
|
|
* By default, the underlying transport layer handle is the handle itself
|
|
*/
|
|
return t;
|
|
}
|
|
|
|
esp_transport_list_handle_t esp_transport_list_init(void)
|
|
{
|
|
esp_transport_list_handle_t transport = calloc(1, sizeof(esp_transport_internal_t));
|
|
ESP_TRANSPORT_MEM_CHECK(TAG, transport, return NULL);
|
|
STAILQ_INIT(&transport->list);
|
|
transport->error_handle = calloc(1, sizeof(esp_tls_last_error_t));
|
|
return transport;
|
|
}
|
|
|
|
esp_err_t esp_transport_list_add(esp_transport_list_handle_t h, esp_transport_handle_t t, const char *scheme)
|
|
{
|
|
if (h == NULL || t == NULL) {
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
t->scheme = calloc(1, strlen(scheme) + 1);
|
|
ESP_TRANSPORT_MEM_CHECK(TAG, t->scheme, return ESP_ERR_NO_MEM);
|
|
strcpy(t->scheme, scheme);
|
|
STAILQ_INSERT_TAIL(&h->list, t, next);
|
|
// Each transport in a list to share the same error tracker
|
|
t->error_handle = h->error_handle;
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_transport_handle_t esp_transport_list_get_transport(esp_transport_list_handle_t h, const char *scheme)
|
|
{
|
|
if (!h) {
|
|
return NULL;
|
|
}
|
|
if (scheme == NULL) {
|
|
return STAILQ_FIRST(&h->list);
|
|
}
|
|
esp_transport_handle_t item;
|
|
STAILQ_FOREACH(item, &h->list, next) {
|
|
if (strcasecmp(item->scheme, scheme) == 0) {
|
|
return item;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
esp_err_t esp_transport_list_destroy(esp_transport_list_handle_t h)
|
|
{
|
|
esp_transport_list_clean(h);
|
|
free(h->error_handle);
|
|
free(h);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_transport_list_clean(esp_transport_list_handle_t h)
|
|
{
|
|
esp_transport_handle_t item = STAILQ_FIRST(&h->list);
|
|
esp_transport_handle_t tmp;
|
|
while (item != NULL) {
|
|
tmp = STAILQ_NEXT(item, next);
|
|
esp_transport_destroy(item);
|
|
item = tmp;
|
|
}
|
|
STAILQ_INIT(&h->list);
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_transport_handle_t esp_transport_init(void)
|
|
{
|
|
esp_transport_handle_t t = calloc(1, sizeof(struct esp_transport_item_t));
|
|
ESP_TRANSPORT_MEM_CHECK(TAG, t, return NULL);
|
|
return t;
|
|
}
|
|
|
|
esp_transport_handle_t esp_transport_get_payload_transport_handle(esp_transport_handle_t t)
|
|
{
|
|
if (t && t->_read) {
|
|
return t->_parent_transfer(t);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
esp_err_t esp_transport_destroy(esp_transport_handle_t t)
|
|
{
|
|
if (t->_destroy) {
|
|
t->_destroy(t);
|
|
}
|
|
if (t->scheme) {
|
|
free(t->scheme);
|
|
}
|
|
free(t);
|
|
return ESP_OK;
|
|
}
|
|
|
|
int esp_transport_connect(esp_transport_handle_t t, const char *host, int port, int timeout_ms)
|
|
{
|
|
int ret = -1;
|
|
if (t && t->_connect) {
|
|
return t->_connect(t, host, port, timeout_ms);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int esp_transport_connect_async(esp_transport_handle_t t, const char *host, int port, int timeout_ms)
|
|
{
|
|
int ret = -1;
|
|
if (t && t->_connect_async) {
|
|
return t->_connect_async(t, host, port, timeout_ms);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int esp_transport_read(esp_transport_handle_t t, char *buffer, int len, int timeout_ms)
|
|
{
|
|
if (t && t->_read) {
|
|
return t->_read(t, buffer, len, timeout_ms);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int esp_transport_write(esp_transport_handle_t t, const char *buffer, int len, int timeout_ms)
|
|
{
|
|
if (t && t->_write) {
|
|
return t->_write(t, buffer, len, timeout_ms);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int esp_transport_poll_read(esp_transport_handle_t t, int timeout_ms)
|
|
{
|
|
if (t && t->_poll_read) {
|
|
return t->_poll_read(t, timeout_ms);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int esp_transport_poll_write(esp_transport_handle_t t, int timeout_ms)
|
|
{
|
|
if (t && t->_poll_write) {
|
|
return t->_poll_write(t, timeout_ms);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int esp_transport_close(esp_transport_handle_t t)
|
|
{
|
|
if (t && t->_close) {
|
|
return t->_close(t);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void *esp_transport_get_context_data(esp_transport_handle_t t)
|
|
{
|
|
if (t) {
|
|
return t->data;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
esp_err_t esp_transport_set_context_data(esp_transport_handle_t t, void *data)
|
|
{
|
|
if (t) {
|
|
t->data = data;
|
|
return ESP_OK;
|
|
}
|
|
return ESP_FAIL;
|
|
}
|
|
|
|
esp_err_t esp_transport_set_func(esp_transport_handle_t t,
|
|
connect_func _connect,
|
|
io_read_func _read,
|
|
io_func _write,
|
|
trans_func _close,
|
|
poll_func _poll_read,
|
|
poll_func _poll_write,
|
|
trans_func _destroy)
|
|
{
|
|
if (t == NULL) {
|
|
return ESP_FAIL;
|
|
}
|
|
t->_connect = _connect;
|
|
t->_read = _read;
|
|
t->_write = _write;
|
|
t->_close = _close;
|
|
t->_poll_read = _poll_read;
|
|
t->_poll_write = _poll_write;
|
|
t->_destroy = _destroy;
|
|
t->_connect_async = NULL;
|
|
t->_parent_transfer = esp_transport_get_default_parent;
|
|
return ESP_OK;
|
|
}
|
|
|
|
int esp_transport_get_default_port(esp_transport_handle_t t)
|
|
{
|
|
if (t == NULL) {
|
|
return -1;
|
|
}
|
|
return t->port;
|
|
}
|
|
|
|
esp_err_t esp_transport_set_default_port(esp_transport_handle_t t, int port)
|
|
{
|
|
if (t == NULL) {
|
|
return ESP_FAIL;
|
|
}
|
|
t->port = port;
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_transport_set_async_connect_func(esp_transport_handle_t t, connect_async_func _connect_async_func)
|
|
{
|
|
if (t == NULL) {
|
|
return ESP_FAIL;
|
|
}
|
|
t->_connect_async = _connect_async_func;
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_err_t esp_transport_set_parent_transport_func(esp_transport_handle_t t, payload_transfer_func _parent_transport)
|
|
{
|
|
if (t == NULL) {
|
|
return ESP_FAIL;
|
|
}
|
|
t->_parent_transfer = _parent_transport;
|
|
return ESP_OK;
|
|
}
|
|
|
|
esp_tls_error_handle_t esp_transport_get_error_handle(esp_transport_handle_t t)
|
|
{
|
|
if (t) {
|
|
return t->error_handle;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void esp_transport_set_errors(esp_transport_handle_t t, const esp_tls_error_handle_t error_handle)
|
|
{
|
|
if (t) {
|
|
memcpy(t->error_handle, error_handle, sizeof(esp_tls_last_error_t));
|
|
}
|
|
}
|
|
|
|
void esp_transport_set_keep_alive(esp_transport_handle_t t, esp_transport_keep_alive_t *keep_alive_cfg)
|
|
{
|
|
if (t && keep_alive_cfg) {
|
|
t->keep_alive_cfg = keep_alive_cfg;
|
|
}
|
|
}
|
|
|
|
void *esp_transport_get_keep_alive(esp_transport_handle_t t)
|
|
{
|
|
if (t) {
|
|
return t->keep_alive_cfg;
|
|
}
|
|
return NULL;
|
|
}
|