2017-02-10 23:20:24 +01:00
|
|
|
/* Provide SSL/TLS functions to ESP32 with Arduino IDE
|
|
|
|
*
|
|
|
|
* Adapted from the ssl_client1 example in mbedtls.
|
|
|
|
*
|
|
|
|
* Original Copyright (C) 2006-2015, ARM Limited, All Rights Reserved, Apache 2.0 License.
|
|
|
|
* Additions Copyright (C) 2017 Evandro Luis Copercini, Apache 2.0 License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Arduino.h"
|
|
|
|
#include <lwip/sockets.h>
|
|
|
|
#include <lwip/err.h>
|
|
|
|
#include <lwip/sockets.h>
|
|
|
|
#include <lwip/sys.h>
|
|
|
|
#include <lwip/netdb.h>
|
|
|
|
|
|
|
|
#include "ssl_client.h"
|
|
|
|
|
|
|
|
|
|
|
|
const char *pers = "esp32-tls";
|
|
|
|
|
2017-02-28 22:18:03 +01:00
|
|
|
#define DEBUG //Comment to supress debug messages
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
#define DEBUG_PRINT(...) printf( __VA_ARGS__ )
|
|
|
|
#else
|
|
|
|
#define DEBUG_PRINT(x)
|
|
|
|
#endif
|
|
|
|
|
2017-02-28 22:18:03 +01:00
|
|
|
#ifdef CONFIG_MBEDTLS_DEBUG
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
#define MBEDTLS_DEBUG_LEVEL 4
|
|
|
|
|
|
|
|
/* mbedtls debug function that translates mbedTLS debug output
|
|
|
|
to ESP_LOGx debug output.
|
|
|
|
|
|
|
|
MBEDTLS_DEBUG_LEVEL 4 means all mbedTLS debug output gets sent here,
|
|
|
|
and then filtered to the ESP logging mechanism.
|
|
|
|
*/
|
|
|
|
static void mbedtls_debug(void *ctx, int level,
|
|
|
|
const char *file, int line,
|
|
|
|
const char *str)
|
|
|
|
{
|
|
|
|
const char *MBTAG = "mbedtls";
|
|
|
|
char *file_sep;
|
|
|
|
|
|
|
|
/* Shorten 'file' from the whole file path to just the filename
|
|
|
|
|
|
|
|
This is a bit wasteful because the macros are compiled in with
|
|
|
|
the full _FILE_ path in each case.
|
|
|
|
*/
|
|
|
|
file_sep = rindex(file, '/');
|
|
|
|
if (file_sep) {
|
|
|
|
file = file_sep + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case 1:
|
|
|
|
printf( "%s:%d %s \n", file, line, str);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
case 3:
|
|
|
|
printf( "%s:%d %s \n", file, line, str);
|
|
|
|
case 4:
|
|
|
|
printf( "%s:%d %s \n", file, line, str);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf( "Unexpected log level %d: %s \n", level, str);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
static int handle_error(int err)
|
|
|
|
{
|
|
|
|
#ifdef MBEDTLS_ERROR_C
|
|
|
|
char error_buf[100];
|
|
|
|
|
|
|
|
mbedtls_strerror(err, error_buf, 100);
|
|
|
|
printf("\n%s\n", error_buf);
|
|
|
|
#endif
|
|
|
|
printf("\nMbedTLS message code: %d\n", err);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
|
|
|
|
void ssl_init(sslclient_context *ssl_client)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Initialize the RNG and the session data
|
|
|
|
*/
|
|
|
|
|
|
|
|
mbedtls_ssl_init(&ssl_client->ssl_ctx);
|
|
|
|
mbedtls_ssl_config_init(&ssl_client->ssl_conf);
|
|
|
|
|
|
|
|
mbedtls_ctr_drbg_init(&ssl_client->drbg_ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
int start_ssl_client(sslclient_context *ssl_client, uint32_t ipAddress, uint32_t port, const char *rootCABuff, const char *cli_cert, const char *cli_key)
|
2017-02-10 23:20:24 +01:00
|
|
|
{
|
|
|
|
char buf[512];
|
|
|
|
int ret, flags, len, timeout;
|
|
|
|
int enable = 1;
|
|
|
|
DEBUG_PRINT("Free heap before TLS %u\n", xPortGetFreeHeapSize());
|
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
ssl_client->socket = -1;
|
|
|
|
|
|
|
|
ssl_client->socket = lwip_socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
|
|
|
|
if (ssl_client->socket < 0) {
|
|
|
|
printf("\r\nERROR opening socket\r\n");
|
|
|
|
return ssl_client->socket;
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
struct sockaddr_in serv_addr;
|
|
|
|
memset(&serv_addr, 0, sizeof(serv_addr));
|
|
|
|
serv_addr.sin_family = AF_INET;
|
|
|
|
serv_addr.sin_addr.s_addr = ipAddress;
|
|
|
|
serv_addr.sin_port = htons(port);
|
|
|
|
|
|
|
|
if (lwip_connect(ssl_client->socket, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) == 0) {
|
|
|
|
timeout = 30000;
|
|
|
|
lwip_setsockopt(ssl_client->socket, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout));
|
|
|
|
lwip_setsockopt(ssl_client->socket, SOL_SOCKET, SO_SNDTIMEO, &timeout, sizeof(timeout));
|
|
|
|
lwip_setsockopt(ssl_client->socket, IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(enable));
|
|
|
|
lwip_setsockopt(ssl_client->socket, SOL_SOCKET, SO_KEEPALIVE, &enable, sizeof(enable));
|
|
|
|
} else {
|
|
|
|
printf("\r\nConnect to Server failed!\r\n");
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
fcntl( ssl_client->socket, F_SETFL, fcntl( ssl_client->socket, F_GETFL, 0 ) | O_NONBLOCK );
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Seeding the random number generator\n");
|
|
|
|
mbedtls_entropy_init(&ssl_client->entropy_ctx);
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
ret = mbedtls_ctr_drbg_seed(&ssl_client->drbg_ctx, mbedtls_entropy_func,
|
|
|
|
&ssl_client->entropy_ctx, (const unsigned char *) pers, strlen(pers));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
return handle_error(ret);
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
/* MBEDTLS_SSL_VERIFY_REQUIRED if a CA certificate is defined on Arduino IDE and
|
2017-02-10 23:20:24 +01:00
|
|
|
MBEDTLS_SSL_VERIFY_NONE if not.
|
|
|
|
*/
|
2017-03-10 15:52:50 +01:00
|
|
|
if (rootCABuff != NULL) {
|
|
|
|
DEBUG_PRINT( "Loading CA cert\n");
|
|
|
|
mbedtls_x509_crt_init(&ssl_client->ca_cert);
|
|
|
|
mbedtls_ssl_conf_authmode(&ssl_client->ssl_conf, MBEDTLS_SSL_VERIFY_REQUIRED);
|
|
|
|
ret = mbedtls_x509_crt_parse(&ssl_client->ca_cert, (const unsigned char *)rootCABuff, strlen(rootCABuff) + 1);
|
|
|
|
mbedtls_ssl_conf_ca_chain(&ssl_client->ssl_conf, &ssl_client->ca_cert, NULL);
|
|
|
|
//mbedtls_ssl_conf_verify(&ssl_client->ssl_ctx, my_verify, NULL );
|
|
|
|
if (ret < 0) {
|
|
|
|
return handle_error(ret);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
2017-03-10 15:52:50 +01:00
|
|
|
} else {
|
|
|
|
|
|
|
|
mbedtls_ssl_conf_authmode(&ssl_client->ssl_conf, MBEDTLS_SSL_VERIFY_NONE);
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if (cli_cert != NULL && cli_key != NULL) {
|
|
|
|
mbedtls_x509_crt_init(&ssl_client->client_cert);
|
|
|
|
mbedtls_pk_init(&ssl_client->client_key);
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Loading CRT cert\n");
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
ret = mbedtls_x509_crt_parse(&ssl_client->client_cert, (const unsigned char *)cli_cert, strlen(cli_cert) + 1);
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if (ret < 0) {
|
|
|
|
return handle_error(ret);
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Loading private key\n");
|
|
|
|
ret = mbedtls_pk_parse_key(&ssl_client->client_key, (const unsigned char *)cli_key, strlen(cli_key) + 1, NULL, 0);
|
|
|
|
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if (ret != 0) {
|
|
|
|
return handle_error(ret);
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
mbedtls_ssl_conf_own_cert(&ssl_client->ssl_conf, &ssl_client->client_cert, &ssl_client->client_key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2017-02-10 23:20:24 +01:00
|
|
|
// TODO: implement match CN verification
|
|
|
|
|
|
|
|
DEBUG_PRINT( "Setting hostname for TLS session...\n");
|
|
|
|
|
|
|
|
// Hostname set here should match CN in server certificate
|
|
|
|
if((ret = mbedtls_ssl_set_hostname(&ssl_client->ssl_ctx, host)) != 0)
|
|
|
|
{
|
2017-03-10 15:52:50 +01:00
|
|
|
return handle_error(ret);
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Setting up the SSL/TLS structure...\n");
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if ((ret = mbedtls_ssl_config_defaults(&ssl_client->ssl_conf,
|
|
|
|
MBEDTLS_SSL_IS_CLIENT,
|
|
|
|
MBEDTLS_SSL_TRANSPORT_STREAM,
|
|
|
|
MBEDTLS_SSL_PRESET_DEFAULT)) != 0) {
|
|
|
|
return handle_error(ret);
|
|
|
|
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
mbedtls_ssl_conf_rng(&ssl_client->ssl_conf, mbedtls_ctr_drbg_random, &ssl_client->drbg_ctx);
|
2017-02-28 22:18:03 +01:00
|
|
|
#ifdef CONFIG_MBEDTLS_DEBUG
|
2017-03-10 15:52:50 +01:00
|
|
|
mbedtls_debug_set_threshold(MBEDTLS_DEBUG_LEVEL);
|
|
|
|
mbedtls_ssl_conf_dbg(&ssl_client->ssl_conf, mbedtls_debug, NULL);
|
2017-02-10 23:20:24 +01:00
|
|
|
#endif
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if ((ret = mbedtls_ssl_setup(&ssl_client->ssl_ctx, &ssl_client->ssl_conf)) != 0) {
|
|
|
|
return handle_error(ret);
|
|
|
|
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
mbedtls_ssl_set_bio(&ssl_client->ssl_ctx, &ssl_client->socket, mbedtls_net_send, mbedtls_net_recv, NULL );
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Performing the SSL/TLS handshake...\n");
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
while ((ret = mbedtls_ssl_handshake(&ssl_client->ssl_ctx)) != 0) {
|
|
|
|
if (ret != MBEDTLS_ERR_SSL_WANT_READ && ret != MBEDTLS_ERR_SSL_WANT_WRITE && ret != -76) {
|
|
|
|
return handle_error(ret);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
2017-03-10 15:52:50 +01:00
|
|
|
delay(10);
|
|
|
|
vPortYield();
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if (cli_cert != NULL && cli_key != NULL) {
|
|
|
|
DEBUG_PRINT("Protocol is %s \nCiphersuite is %s\n", mbedtls_ssl_get_version(&ssl_client->ssl_ctx), mbedtls_ssl_get_ciphersuite(&ssl_client->ssl_ctx));
|
|
|
|
if ((ret = mbedtls_ssl_get_record_expansion(&ssl_client->ssl_ctx)) >= 0) {
|
|
|
|
DEBUG_PRINT("Record expansion is %d\n", ret);
|
|
|
|
} else {
|
|
|
|
DEBUG_PRINT("Record expansion is unknown (compression)\n");
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
2017-03-10 15:52:50 +01:00
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
DEBUG_PRINT( "Verifying peer X.509 certificate...\n");
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
if ((flags = mbedtls_ssl_get_verify_result(&ssl_client->ssl_ctx)) != 0) {
|
|
|
|
printf( "Failed to verify peer certificate!\n");
|
|
|
|
bzero(buf, sizeof(buf));
|
|
|
|
mbedtls_x509_crt_verify_info(buf, sizeof(buf), " ! ", flags);
|
|
|
|
printf( "verification info: %s\n", buf);
|
|
|
|
stop_ssl_socket(ssl_client, rootCABuff, cli_cert, cli_key); //It's not safe continue.
|
|
|
|
return handle_error(ret);
|
|
|
|
} else {
|
|
|
|
DEBUG_PRINT( "Certificate verified.\n");
|
|
|
|
}
|
2017-02-10 23:20:24 +01:00
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
|
|
|
|
DEBUG_PRINT("Free heap after TLS %u\n", xPortGetFreeHeapSize());
|
|
|
|
|
|
|
|
return ssl_client->socket;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-10 15:52:50 +01:00
|
|
|
void stop_ssl_socket(sslclient_context *ssl_client, const char *rootCABuff, const char *cli_cert, const char *cli_key)
|
2017-02-10 23:20:24 +01:00
|
|
|
{
|
|
|
|
DEBUG_PRINT( "\nCleaning SSL connection.\n");
|
2017-03-10 15:52:50 +01:00
|
|
|
|
|
|
|
if (ssl_client->socket >= 0) {
|
|
|
|
close(ssl_client->socket);
|
|
|
|
ssl_client->socket = -1;
|
|
|
|
}
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
mbedtls_ssl_free(&ssl_client->ssl_ctx);
|
|
|
|
mbedtls_ssl_config_free(&ssl_client->ssl_conf);
|
|
|
|
mbedtls_ctr_drbg_free(&ssl_client->drbg_ctx);
|
|
|
|
mbedtls_entropy_free(&ssl_client->entropy_ctx);
|
|
|
|
|
|
|
|
if (rootCABuff != NULL) {
|
|
|
|
mbedtls_x509_crt_free(&ssl_client->ca_cert);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cli_cert != NULL) {
|
|
|
|
mbedtls_x509_crt_free(&ssl_client->client_cert);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cli_key != NULL) {
|
|
|
|
mbedtls_pk_free(&ssl_client->client_key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int data_to_read(sslclient_context *ssl_client)
|
|
|
|
{
|
|
|
|
|
|
|
|
int ret, res;
|
|
|
|
ret = mbedtls_ssl_read(&ssl_client->ssl_ctx, NULL, 0);
|
|
|
|
//printf("RET: %i\n",ret); //for low level debug
|
|
|
|
res = mbedtls_ssl_get_bytes_avail(&ssl_client->ssl_ctx);
|
|
|
|
//printf("RES: %i\n",res);
|
2017-03-10 15:52:50 +01:00
|
|
|
if (ret != MBEDTLS_ERR_SSL_WANT_READ && ret != MBEDTLS_ERR_SSL_WANT_WRITE && ret < 0 && ret != -76) { //RC:76 sockets is not connected
|
|
|
|
return handle_error(ret);
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int send_ssl_data(sslclient_context *ssl_client, const uint8_t *data, uint16_t len)
|
|
|
|
{
|
|
|
|
//DEBUG_PRINT( "Writing HTTP request...\n"); //for low level debug
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
while ((ret = mbedtls_ssl_write(&ssl_client->ssl_ctx, data, len)) <= 0) {
|
2017-03-10 15:52:50 +01:00
|
|
|
if (ret != MBEDTLS_ERR_SSL_WANT_READ && ret != MBEDTLS_ERR_SSL_WANT_WRITE && ret != -76) { //RC:76 sockets is not connected
|
|
|
|
return handle_error(ret);
|
|
|
|
|
2017-02-10 23:20:24 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
len = ret;
|
|
|
|
//DEBUG_PRINT( "%d bytes written\n", len); //for low level debug
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int get_ssl_receive(sslclient_context *ssl_client, uint8_t *data, int length)
|
|
|
|
{
|
|
|
|
//DEBUG_PRINT( "Reading HTTP response...\n"); //for low level debug
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
ret = mbedtls_ssl_read(&ssl_client->ssl_ctx, data, length);
|
|
|
|
|
|
|
|
//DEBUG_PRINT( "%d bytes readed\n", ret); //for low level debug
|
|
|
|
return ret;
|
|
|
|
}
|