example and dhcp debug (#447)

This commit is contained in:
Zaltora 2017-10-08 08:47:23 +02:00 committed by Ruslan V. Uss
parent 546cc47121
commit 68cc1451b2
3 changed files with 244 additions and 23 deletions

View file

@ -0,0 +1,5 @@
# Makefile for tcp_non_blocking example
PROGRAM=tcp_non_blocking
EXTRA_COMPONENTS=extras/dhcpserver
include ../../common.mk

View file

@ -0,0 +1,203 @@
/*
The ESP in the example runs a echo server on 172.16.0.1 (port 50 and 100 ) that
outputs information about your ip/port then echo all text you write.
It is manage multiple connection and multiple port with one task.
This example code is in the public domain.
*/
#include <string.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <espressif/esp_common.h>
#include <esp8266.h>
#include <esp/uart.h>
#include <FreeRTOS.h>
#include <task.h>
#include <queue.h>
#include <dhcpserver.h>
#include <lwip/api.h>
#define AP_SSID "esp-open-rtos AP"
#define AP_PSK "esp-open-rtos"
#define ECHO_PORT_1 50
#define ECHO_PORT_2 100
#define EVENTS_QUEUE_SIZE 10
#ifdef CALLBACK_DEBUG
#define debug(s, ...) printf("%s: " s "\n", "Cb:", ## __VA_ARGS__)
#else
#define debug(s, ...)
#endif
QueueHandle_t xQueue_events;
typedef struct {
struct netconn *nc ;
uint8_t type ;
} netconn_events;
/*
* This function will be call in Lwip in each event on netconn
*/
static void netCallback(struct netconn *conn, enum netconn_evt evt, uint16_t length)
{
//Show some callback information (debug)
debug("sock:%u\tsta:%u\tevt:%u\tlen:%u\ttyp:%u\tfla:%02X\terr:%d", \
(uint32_t)conn,conn->state,evt,length,conn->type,conn->flags,conn->last_err);
netconn_events events ;
//If netconn got error, it is close or deleted, dont do treatments on it.
if (conn->pending_err)
{
return;
}
//Treatments only on rcv events.
switch (evt) {
case NETCONN_EVT_RCVPLUS:
events.nc = conn ;
events.type = evt ;
break;
default:
return;
break;
}
//Send the event to the queue
xQueueSend(xQueue_events, &events, 1000);
}
/*
* Initialize a server netconn and listen port
*/
static void set_tcp_server_netconn(struct netconn **nc, uint16_t port, netconn_callback callback)
{
if(nc == NULL)
{
printf("%s: netconn missing .\n",__FUNCTION__);
return;
}
*nc = netconn_new_with_callback(NETCONN_TCP, netCallback);
if(!*nc) {
printf("Status monitor: Failed to allocate netconn.\n");
return;
}
netconn_set_nonblocking(*nc,NETCONN_FLAG_NON_BLOCKING);
//netconn_set_recvtimeout(*nc, 10);
netconn_bind(*nc, IP_ADDR_ANY, port);
netconn_listen(*nc);
}
/*
* Close and delete a socket properly
*/
static void close_tcp_netconn(struct netconn *nc)
{
nc->pending_err=ERR_CLSD; //It is hacky way to be sure than callback will don't do treatment on a netconn closed and deleted
netconn_close(nc);
netconn_delete(nc);
}
/*
* This task manage each netconn connection without block anything
*/
static void nonBlockingTCP(void *pvParameters)
{
struct netconn *nc = NULL; // To create servers
set_tcp_server_netconn(&nc, ECHO_PORT_1, netCallback);
printf("Server netconn %u ready on port %u.\n",(uint32_t)nc, ECHO_PORT_1);
set_tcp_server_netconn(&nc, ECHO_PORT_2, netCallback);
printf("Server netconn %u ready on port %u.\n",(uint32_t)nc, ECHO_PORT_2);
struct netbuf *netbuf = NULL; // To store incoming Data
struct netconn *nc_in = NULL; // To accept incoming netconn
//
char buf[50];
char* buffer;
uint16_t len_buf;
while(1) {
netconn_events events;
xQueueReceive(xQueue_events, &events, portMAX_DELAY); // Wait here an event on netconn
if (events.nc->state == NETCONN_LISTEN) // If netconn is a server and receive incoming event on it
{
printf("Client incoming on server %u.\n", (uint32_t)events.nc);
int err = netconn_accept(events.nc, &nc_in);
if (err != ERR_OK)
{
if(nc_in)
netconn_delete(nc_in);
}
printf("New client is %u.\n",(uint32_t)nc_in);
ip_addr_t client_addr; //Address port
uint16_t client_port; //Client port
netconn_peer(nc_in, &client_addr, &client_port);
snprintf(buf, sizeof(buf), "Your address is %d.%d.%d.%d:%u.\r\n",
ip4_addr1(&client_addr), ip4_addr2(&client_addr),
ip4_addr3(&client_addr), ip4_addr4(&client_addr),
client_port);
netconn_write(nc_in, buf, strlen(buf), NETCONN_COPY);
}
else if(events.nc->state != NETCONN_LISTEN) // If netconn is the client and receive data
{
if ((netconn_recv(events.nc, &netbuf)) == ERR_OK) // data incoming ?
{
do
{
netbuf_data(netbuf, (void*)&buffer, &len_buf);
netconn_write(events.nc, buffer, strlen(buffer), NETCONN_COPY);
printf("Client %u send: %s\n",(uint32_t)events.nc,buffer);
}
while (netbuf_next(netbuf) >= 0);
netbuf_delete(netbuf);
}
else
{
close_tcp_netconn(events.nc);
printf("Error read netconn %u, close it \n",(uint32_t)events.nc);
}
}
}
}
void user_init(void)
{
uart_set_baud(0, 115200);
sdk_os_delay_us(500); // Wait UART
printf("SDK version:%s\n", sdk_system_get_sdk_version());
sdk_wifi_set_opmode(SOFTAP_MODE);
struct ip_info ap_ip;
IP4_ADDR(&ap_ip.ip, 172, 16, 0, 1);
IP4_ADDR(&ap_ip.gw, 0, 0, 0, 0);
IP4_ADDR(&ap_ip.netmask, 255, 255, 0, 0);
sdk_wifi_set_ip_info(1, &ap_ip);
struct sdk_softap_config ap_config = {
.ssid = AP_SSID,
.ssid_hidden = 0,
.channel = 3,
.ssid_len = strlen(AP_SSID),
.authmode = AUTH_WPA_WPA2_PSK,
.password = AP_PSK,
.max_connection = 3,
.beacon_interval = 100,
};
sdk_wifi_softap_set_config(&ap_config);
ip_addr_t first_client_ip;
IP4_ADDR(&first_client_ip, 172, 16, 0, 2);
dhcpserver_start(&first_client_ip, 4);
printf("DHCP started\n");
//Create a queue to store events on netconns
xQueue_events = xQueueCreate( EVENTS_QUEUE_SIZE, sizeof(netconn_events));
xTaskCreate(nonBlockingTCP, "lwiptest_noblock", 512, NULL, 2, NULL);
}

View file

@ -22,6 +22,18 @@
#include <lwip/api.h> #include <lwip/api.h>
#include "esplibs/libmain.h" #include "esplibs/libmain.h"
#if (DHCP_DEBUG == LWIP_DBG_ON)
#define debug(s, ...) printf("%s: " s "\n", "DHCP", ## __VA_ARGS__)
#else
#define debug(s, ...)
#endif
#if (DHCP_DEBUG == LWIP_DBG_ON)
#define debug(s, ...) printf("%s: " s "\n", "DHCP", ## __VA_ARGS__)
#else
#define debug(s, ...)
#endif
/* Grow the size of the lwip dhcp_msg struct's options field, as LWIP /* Grow the size of the lwip dhcp_msg struct's options field, as LWIP
defaults to a 68 octet options field for its DHCP client, and most defaults to a 68 octet options field for its DHCP client, and most
full-sized clients send us more than this. */ full-sized clients send us more than this. */
@ -131,8 +143,8 @@ static void dhcpserver_task(void *pxParameter)
state->server_if = netif_list; /* TODO: Make this configurable */ state->server_if = netif_list; /* TODO: Make this configurable */
state->nc = netconn_new (NETCONN_UDP); state->nc = netconn_new (NETCONN_UDP);
if (!state->nc) { if(!state->nc) {
printf("DHCP Server Error: Failed to allocate socket.\r\n"); debug("DHCP Server Error: Failed to allocate socket.");
return; return;
} }
@ -146,8 +158,8 @@ static void dhcpserver_task(void *pxParameter)
/* Receive a DHCP packet */ /* Receive a DHCP packet */
err_t err = netconn_recv(state->nc, &netbuf); err_t err = netconn_recv(state->nc, &netbuf);
if (err != ERR_OK) { if(err != ERR_OK) {
printf("DHCP Server Error: Failed to receive DHCP packet. err=%d\r\n", err); debug("DHCP Server Error: Failed to receive DHCP packet. err=%d", err);
continue; continue;
} }
@ -171,8 +183,8 @@ static void dhcpserver_task(void *pxParameter)
netbuf_delete(netbuf); netbuf_delete(netbuf);
continue; continue;
} }
if (netbuf_len(netbuf) >= sizeof(struct dhcp_msg)) { if(netbuf_len(netbuf) >= sizeof(struct dhcp_msg)) {
printf("DHCP Server Warning: Client sent more options than we know how to parse. len=%d\r\n", netbuf_len(netbuf)); debug("DHCP Server Warning: Client sent more options than we know how to parse. len=%d", netbuf_len(netbuf));
} }
netbuf_copy(netbuf, &received, sizeof(struct dhcp_msg)); netbuf_copy(netbuf, &received, sizeof(struct dhcp_msg));
@ -180,19 +192,20 @@ static void dhcpserver_task(void *pxParameter)
uint8_t *message_type = find_dhcp_option(&received, DHCP_OPTION_MESSAGE_TYPE, uint8_t *message_type = find_dhcp_option(&received, DHCP_OPTION_MESSAGE_TYPE,
DHCP_OPTION_MESSAGE_TYPE_LEN, NULL); DHCP_OPTION_MESSAGE_TYPE_LEN, NULL);
if (!message_type) { if(!message_type) {
printf("DHCP Server Error: No message type field found"); debug("DHCP Server Error: No message type field found");
continue; continue;
} }
printf("State dump. Message type %d\n", *message_type); #if (DHCP_DEBUG == LWIP_DBG_ON)
for (int i = 0; i < state->max_leases; i++) { debug("State dump. Message type %d", *message_type);
for(int i = 0; i < state->max_leases; i++) {
dhcp_lease_t *lease = &state->leases[i]; dhcp_lease_t *lease = &state->leases[i];
printf("lease slot %d active %d expiry %d hwaddr %02x:%02x:%02x:%02x:%02x:%02x\r\n", i, debug("lease slot %d expiry %d hwaddr %02x:%02x:%02x:%02x:%02x:%02x", i, lease->expires, lease->hwaddr[0],
lease->active, lease->expires - now, lease->hwaddr[1], lease->hwaddr[2], lease->hwaddr[3], lease->hwaddr[4],
lease->hwaddr[0], lease->hwaddr[1], lease->hwaddr[2], lease->hwaddr[5]);
lease->hwaddr[3], lease->hwaddr[4], lease->hwaddr[5]);
} }
#endif
switch(*message_type) { switch(*message_type) {
case DHCP_DISCOVER: case DHCP_DISCOVER:
@ -204,7 +217,7 @@ static void dhcpserver_task(void *pxParameter)
case DHCP_RELEASE: case DHCP_RELEASE:
handle_dhcp_release(&received); handle_dhcp_release(&received);
default: default:
printf("DHCP Server Error: Unsupported message type %d\r\n", *message_type); debug("DHCP Server Error: Unsupported message type %d", *message_type);
break; break;
} }
} }
@ -218,8 +231,8 @@ static void handle_dhcp_discover(struct dhcp_msg *dhcpmsg)
return; return;
dhcp_lease_t *freelease = find_lease_slot(dhcpmsg->chaddr); dhcp_lease_t *freelease = find_lease_slot(dhcpmsg->chaddr);
if (!freelease) { if(!freelease) {
printf("DHCP Server: All leases taken.\r\n"); debug("DHCP Server: All leases taken.");
return; /* Nothing available, so do nothing */ return; /* Nothing available, so do nothing */
} }
@ -264,7 +277,7 @@ static void handle_dhcp_request(struct dhcp_msg *dhcpmsg)
} else if (ip4_addr_cmp(&requested_ip, IP4_ADDR_ANY4)) { } else if (ip4_addr_cmp(&requested_ip, IP4_ADDR_ANY4)) {
ip4_addr_copy(requested_ip, dhcpmsg->ciaddr); ip4_addr_copy(requested_ip, dhcpmsg->ciaddr);
} else { } else {
printf("DHCP Server Error: No requested IP\r\n"); debug("DHCP Server Error: No requested IP");
send_dhcp_nak(dhcpmsg); send_dhcp_nak(dhcpmsg);
return; return;
} }
@ -274,14 +287,14 @@ static void handle_dhcp_request(struct dhcp_msg *dhcpmsg)
|| ip4_addr2(&requested_ip) != ip4_addr2(&state->first_client_addr) || ip4_addr2(&requested_ip) != ip4_addr2(&state->first_client_addr)
|| ip4_addr3(&requested_ip) != ip4_addr3(&state->first_client_addr)) { || ip4_addr3(&requested_ip) != ip4_addr3(&state->first_client_addr)) {
sprintf_ipaddr(&requested_ip, ipbuf); sprintf_ipaddr(&requested_ip, ipbuf);
printf("DHCP Server Error: %s not an allowed IP\r\n", ipbuf); debug("DHCP Server Error: %s not an allowed IP", ipbuf);
send_dhcp_nak(dhcpmsg); send_dhcp_nak(dhcpmsg);
return; return;
} }
/* Test the last octet is in the MAXCLIENTS range */ /* Test the last octet is in the MAXCLIENTS range */
int16_t octet_offs = ip4_addr4(&requested_ip) - ip4_addr4(&state->first_client_addr); int16_t octet_offs = ip4_addr4(&requested_ip) - ip4_addr4(&state->first_client_addr);
if (octet_offs < 0 || octet_offs >= state->max_leases) { if(octet_offs < 0 || octet_offs >= state->max_leases) {
printf("DHCP Server Error: Address out of range\r\n"); debug("DHCP Server Error: Address out of range");
send_dhcp_nak(dhcpmsg); send_dhcp_nak(dhcpmsg);
return; return;
} }
@ -289,14 +302,14 @@ static void handle_dhcp_request(struct dhcp_msg *dhcpmsg)
dhcp_lease_t *requested_lease = state->leases + octet_offs; dhcp_lease_t *requested_lease = state->leases + octet_offs;
if (requested_lease->active && memcmp(requested_lease->hwaddr, dhcpmsg->chaddr,dhcpmsg->hlen)) if (requested_lease->active && memcmp(requested_lease->hwaddr, dhcpmsg->chaddr,dhcpmsg->hlen))
{ {
printf("DHCP Server Error: Lease for address already taken\r\n"); debug("DHCP Server Error: Lease for address already taken");
send_dhcp_nak(dhcpmsg); send_dhcp_nak(dhcpmsg);
return; return;
} }
memcpy(requested_lease->hwaddr, dhcpmsg->chaddr, dhcpmsg->hlen); memcpy(requested_lease->hwaddr, dhcpmsg->chaddr, dhcpmsg->hlen);
sprintf_ipaddr(&requested_ip, ipbuf); sprintf_ipaddr(&requested_ip, ipbuf);
printf("DHCP lease addr %s assigned to MAC %02x:%02x:%02x:%02x:%02x:%02x\r\n", ipbuf, requested_lease->hwaddr[0], debug("DHCP lease addr %s assigned to MAC %02x:%02x:%02x:%02x:%02x:%02x", ipbuf, requested_lease->hwaddr[0],
requested_lease->hwaddr[1], requested_lease->hwaddr[2], requested_lease->hwaddr[3], requested_lease->hwaddr[4], requested_lease->hwaddr[1], requested_lease->hwaddr[2], requested_lease->hwaddr[3], requested_lease->hwaddr[4],
requested_lease->hwaddr[5]); requested_lease->hwaddr[5]);
uint32_t now = xTaskGetTickCount(); uint32_t now = xTaskGetTickCount();