| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | #include "Arduino.h"
 | 
					
						
							|  |  |  | #include "AsyncUDP.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | #include "lwip/opt.h"
 | 
					
						
							|  |  |  | #include "lwip/inet.h"
 | 
					
						
							|  |  |  | #include "lwip/udp.h"
 | 
					
						
							|  |  |  | #include "lwip/igmp.h"
 | 
					
						
							|  |  |  | #include "lwip/ip_addr.h"
 | 
					
						
							|  |  |  | #include "lwip/mld6.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | #include "lwip/prot/ethernet.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | #include <esp_err.h>
 | 
					
						
							|  |  |  | #include <esp_wifi.h>
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "lwip/priv/tcpip_priv.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     struct tcpip_api_call_data call; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_pcb * pcb; | 
					
						
							|  |  |  |     const ip_addr_t *addr; | 
					
						
							|  |  |  |     uint16_t port; | 
					
						
							|  |  |  |     struct pbuf *pb; | 
					
						
							|  |  |  |     struct netif *netif; | 
					
						
							|  |  |  |     err_t err; | 
					
						
							|  |  |  | } udp_api_call_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_connect_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = udp_connect(msg->pcb, msg->addr, msg->port); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static err_t _udp_connect(struct udp_pcb *pcb, const ip_addr_t *addr, u16_t port){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							|  |  |  |     msg.addr = addr; | 
					
						
							|  |  |  |     msg.port = port; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_connect_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     return msg.err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_disconnect_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = 0; | 
					
						
							|  |  |  |     udp_disconnect(msg->pcb); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void  _udp_disconnect(struct udp_pcb *pcb){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_disconnect_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_remove_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = 0; | 
					
						
							|  |  |  |     udp_remove(msg->pcb); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void  _udp_remove(struct udp_pcb *pcb){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_remove_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_bind_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = udp_bind(msg->pcb, msg->addr, msg->port); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static err_t _udp_bind(struct udp_pcb *pcb, const ip_addr_t *addr, u16_t port){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							|  |  |  |     msg.addr = addr; | 
					
						
							|  |  |  |     msg.port = port; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_bind_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     return msg.err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_sendto_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = udp_sendto(msg->pcb, msg->pb, msg->addr, msg->port); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static err_t _udp_sendto(struct udp_pcb *pcb, struct pbuf *pb, const ip_addr_t *addr, u16_t port){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							|  |  |  |     msg.addr = addr; | 
					
						
							|  |  |  |     msg.port = port; | 
					
						
							|  |  |  |     msg.pb = pb; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_sendto_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     return msg.err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static err_t _udp_sendto_if_api(struct tcpip_api_call_data *api_call_msg){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_api_call_t * msg = (udp_api_call_t *)api_call_msg; | 
					
						
							|  |  |  |     msg->err = udp_sendto_if(msg->pcb, msg->pb, msg->addr, msg->port, msg->netif); | 
					
						
							|  |  |  |     return msg->err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static err_t _udp_sendto_if(struct udp_pcb *pcb, struct pbuf *pb, const ip_addr_t *addr, u16_t port, struct netif *netif){ | 
					
						
							|  |  |  |     udp_api_call_t msg; | 
					
						
							|  |  |  |     msg.pcb = pcb; | 
					
						
							|  |  |  |     msg.addr = addr; | 
					
						
							|  |  |  |     msg.port = port; | 
					
						
							|  |  |  |     msg.pb = pb; | 
					
						
							|  |  |  |     msg.netif = netif; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     tcpip_api_call(_udp_sendto_if_api, (struct tcpip_api_call_data*)&msg); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     return msg.err; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |         void *arg; | 
					
						
							|  |  |  |         udp_pcb *pcb; | 
					
						
							|  |  |  |         pbuf *pb; | 
					
						
							|  |  |  |         const ip_addr_t *addr; | 
					
						
							|  |  |  |         uint16_t port; | 
					
						
							|  |  |  |         struct netif * netif; | 
					
						
							|  |  |  | } lwip_event_packet_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static xQueueHandle _udp_queue; | 
					
						
							|  |  |  | static volatile TaskHandle_t _udp_task_handle = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void _udp_task(void *pvParameters){ | 
					
						
							|  |  |  |     lwip_event_packet_t * e = NULL; | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							|  |  |  |         if(xQueueReceive(_udp_queue, &e, portMAX_DELAY) == pdTRUE){ | 
					
						
							|  |  |  |             if(!e->pb){ | 
					
						
							|  |  |  |                 free((void*)(e)); | 
					
						
							| 
									
										
										
										
											2018-09-26 23:29:51 +02:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             AsyncUDP::_s_recv(e->arg, e->pcb, e->pb, e->addr, e->port, e->netif); | 
					
						
							|  |  |  |             free((void*)(e)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     _udp_task_handle = NULL; | 
					
						
							|  |  |  |     vTaskDelete(NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool _udp_task_start(){ | 
					
						
							|  |  |  |     if(!_udp_queue){ | 
					
						
							|  |  |  |         _udp_queue = xQueueCreate(32, sizeof(lwip_event_packet_t *)); | 
					
						
							|  |  |  |         if(!_udp_queue){ | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if(!_udp_task_handle){ | 
					
						
							|  |  |  |         xTaskCreate(_udp_task, "async_udp", 4096, NULL, 3, (TaskHandle_t*)&_udp_task_handle); | 
					
						
							|  |  |  |         if(!_udp_task_handle){ | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool _udp_task_post(void *arg, udp_pcb *pcb, pbuf *pb, const ip_addr_t *addr, uint16_t port, struct netif *netif) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_udp_task_handle || !_udp_queue){ | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     lwip_event_packet_t * e = (lwip_event_packet_t *)malloc(sizeof(lwip_event_packet_t)); | 
					
						
							|  |  |  |     if(!e){ | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     e->arg = arg; | 
					
						
							|  |  |  |     e->pcb = pcb; | 
					
						
							|  |  |  |     e->pb = pb; | 
					
						
							|  |  |  |     e->addr = addr; | 
					
						
							|  |  |  |     e->port = port; | 
					
						
							|  |  |  |     e->netif = netif; | 
					
						
							|  |  |  |     if (xQueueSend(_udp_queue, &e, portMAX_DELAY) != pdPASS) { | 
					
						
							|  |  |  |         free((void*)(e)); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void _udp_recv(void *arg, udp_pcb *pcb, pbuf *pb, const ip_addr_t *addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while(pb != NULL) { | 
					
						
							|  |  |  |         pbuf * this_pb = pb; | 
					
						
							|  |  |  |         pb = pb->next; | 
					
						
							|  |  |  |         this_pb->next = NULL; | 
					
						
							|  |  |  |         if(!_udp_task_post(arg, pcb, this_pb, addr, port, ip_current_input_netif())){ | 
					
						
							|  |  |  |             pbuf_free(this_pb); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  | static bool _udp_task_stop(){ | 
					
						
							|  |  |  |     if(!_udp_task_post(NULL, NULL, NULL, NULL, 0, NULL)){ | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     while(_udp_task_handle){ | 
					
						
							|  |  |  |         vTaskDelay(10); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     lwip_event_packet_t * e; | 
					
						
							|  |  |  |     while (xQueueReceive(_udp_queue, &e, 0) == pdTRUE) { | 
					
						
							|  |  |  |         if(e->pb){ | 
					
						
							|  |  |  |             pbuf_free(e->pb); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         free((void*)(e)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     vQueueDelete(_udp_queue); | 
					
						
							|  |  |  |     _udp_queue = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define UDP_MUTEX_LOCK()    //xSemaphoreTake(_lock, portMAX_DELAY)
 | 
					
						
							|  |  |  | #define UDP_MUTEX_UNLOCK()  //xSemaphoreGive(_lock)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDPMessage::AsyncUDPMessage(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _index = 0; | 
					
						
							|  |  |  |     if(size > CONFIG_TCP_MSS) { | 
					
						
							|  |  |  |         size = CONFIG_TCP_MSS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     _size = size; | 
					
						
							|  |  |  |     _buffer = (uint8_t *)malloc(size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDPMessage::~AsyncUDPMessage() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_buffer) { | 
					
						
							|  |  |  |         free(_buffer); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPMessage::write(const uint8_t *data, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_buffer == NULL) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     size_t s = space(); | 
					
						
							|  |  |  |     if(len > s) { | 
					
						
							|  |  |  |         len = s; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     memcpy(_buffer + _index, data, len); | 
					
						
							|  |  |  |     _index += len; | 
					
						
							|  |  |  |     return len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPMessage::write(uint8_t data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return write(&data, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPMessage::space() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_buffer == NULL) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return _size - _index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint8_t * AsyncUDPMessage::data() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _buffer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPMessage::length() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDPMessage::flush() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _index = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDPPacket::AsyncUDPPacket(AsyncUDP *udp, pbuf *pb, const ip_addr_t *raddr, uint16_t rport, struct netif * ntif) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _udp = udp; | 
					
						
							|  |  |  |     _pb = pb; | 
					
						
							|  |  |  |     _if = TCPIP_ADAPTER_IF_MAX; | 
					
						
							|  |  |  |     _data = (uint8_t*)(pb->payload); | 
					
						
							|  |  |  |     _len = pb->len; | 
					
						
							|  |  |  |     _index = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-11 20:46:54 +02:00
										 |  |  |     pbuf_ref(_pb); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     //memcpy(&_remoteIp, raddr, sizeof(ip_addr_t));
 | 
					
						
							|  |  |  |     _remoteIp.type = raddr->type; | 
					
						
							|  |  |  |     _localIp.type = _remoteIp.type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     eth_hdr* eth = NULL; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     udp_hdr* udphdr = reinterpret_cast<udp_hdr*>(_data - UDP_HLEN); | 
					
						
							|  |  |  |     _localPort = ntohs(udphdr->dest); | 
					
						
							|  |  |  |     _remotePort = ntohs(udphdr->src); | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     if (_remoteIp.type == IPADDR_TYPE_V4) { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         eth = (eth_hdr *)(((uint8_t *)(pb->payload)) - UDP_HLEN - IP_HLEN - SIZEOF_ETH_HDR); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         struct ip_hdr * iphdr = (struct ip_hdr *)(((uint8_t *)(pb->payload)) - UDP_HLEN - IP_HLEN); | 
					
						
							|  |  |  |         _localIp.u_addr.ip4.addr = iphdr->dest.addr; | 
					
						
							|  |  |  |         _remoteIp.u_addr.ip4.addr = iphdr->src.addr; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         eth = (eth_hdr *)(((uint8_t *)(pb->payload)) - UDP_HLEN - IP6_HLEN - SIZEOF_ETH_HDR); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         struct ip6_hdr * ip6hdr = (struct ip6_hdr *)(((uint8_t *)(pb->payload)) - UDP_HLEN - IP6_HLEN); | 
					
						
							|  |  |  |         memcpy(&_localIp.u_addr.ip6.addr, (uint8_t *)ip6hdr->dest.addr, 16); | 
					
						
							|  |  |  |         memcpy(&_remoteIp.u_addr.ip6.addr, (uint8_t *)ip6hdr->src.addr, 16); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     memcpy(_remoteMac, eth->src.addr, 6); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     struct netif * netif = NULL; | 
					
						
							|  |  |  |     void * nif = NULL; | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  |     for (i=0; i<TCPIP_ADAPTER_IF_MAX; i++) { | 
					
						
							|  |  |  |         tcpip_adapter_get_netif ((tcpip_adapter_if_t)i, &nif); | 
					
						
							|  |  |  |         netif = (struct netif *)nif; | 
					
						
							|  |  |  |         if (netif && netif == ntif) { | 
					
						
							|  |  |  |             _if = (tcpip_adapter_if_t)i; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDPPacket::~AsyncUDPPacket() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     pbuf_free(_pb); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint8_t * AsyncUDPPacket::data() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _data; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPPacket::length() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AsyncUDPPacket::available(){ | 
					
						
							|  |  |  |     return _len - _index; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPPacket::read(uint8_t *data, size_t len){ | 
					
						
							|  |  |  |     size_t i; | 
					
						
							|  |  |  |     size_t a = _len - _index; | 
					
						
							|  |  |  |     if(len > a){ | 
					
						
							|  |  |  |         len = a; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for(i=0;i<len;i++){ | 
					
						
							|  |  |  |         data[i] = read(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AsyncUDPPacket::read(){ | 
					
						
							|  |  |  |     if(_index < _len){ | 
					
						
							|  |  |  |         return _data[_index++]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int AsyncUDPPacket::peek(){ | 
					
						
							|  |  |  |     if(_index < _len){ | 
					
						
							|  |  |  |         return _data[_index]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDPPacket::flush(){ | 
					
						
							|  |  |  |     _index = _len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | tcpip_adapter_if_t AsyncUDPPacket::interface() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _if; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPAddress AsyncUDPPacket::localIP() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_localIp.type != IPADDR_TYPE_V4){ | 
					
						
							|  |  |  |         return IPAddress(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPAddress(_localIp.u_addr.ip4.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPv6Address AsyncUDPPacket::localIPv6() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_localIp.type != IPADDR_TYPE_V6){ | 
					
						
							|  |  |  |         return IPv6Address(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPv6Address(_localIp.u_addr.ip6.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint16_t AsyncUDPPacket::localPort() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _localPort; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPAddress AsyncUDPPacket::remoteIP() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_remoteIp.type != IPADDR_TYPE_V4){ | 
					
						
							|  |  |  |         return IPAddress(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPAddress(_remoteIp.u_addr.ip4.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPv6Address AsyncUDPPacket::remoteIPv6() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_remoteIp.type != IPADDR_TYPE_V6){ | 
					
						
							|  |  |  |         return IPv6Address(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPv6Address(_remoteIp.u_addr.ip6.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint16_t AsyncUDPPacket::remotePort() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _remotePort; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | void AsyncUDPPacket::remoteMac(uint8_t * mac) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     memcpy(mac, _remoteMac, 6); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | bool AsyncUDPPacket::isIPv6() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _localIp.type == IPADDR_TYPE_V6; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDPPacket::isBroadcast() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_localIp.type == IPADDR_TYPE_V6){ | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     uint32_t ip = _localIp.u_addr.ip4.addr; | 
					
						
							|  |  |  |     return ip == 0xFFFFFFFF || ip == 0 || (ip & 0xFF000000) == 0xFF000000; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDPPacket::isMulticast() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return ip_addr_ismulticast(&(_localIp)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPPacket::write(const uint8_t *data, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!data){ | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return _udp->writeTo(data, len, &_remoteIp, _remotePort, _if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPPacket::write(uint8_t data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return write(&data, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDPPacket::send(AsyncUDPMessage &message) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return write(message.data(), message.length()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | bool AsyncUDP::_init(){ | 
					
						
							|  |  |  |     if(_pcb){ | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     _pcb = udp_new(); | 
					
						
							|  |  |  |     if(!_pcb){ | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     //_lock = xSemaphoreCreateMutex();
 | 
					
						
							|  |  |  |     udp_recv(_pcb, &_udp_recv, (void *) this); | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDP::AsyncUDP() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _pcb = NULL; | 
					
						
							|  |  |  |     _connected = false; | 
					
						
							|  |  |  |     _handler = NULL; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDP::~AsyncUDP() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     close(); | 
					
						
							|  |  |  |     UDP_MUTEX_LOCK(); | 
					
						
							|  |  |  |     udp_recv(_pcb, NULL, NULL); | 
					
						
							|  |  |  |     _udp_remove(_pcb); | 
					
						
							|  |  |  |     _pcb = NULL; | 
					
						
							|  |  |  |     UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |     //vSemaphoreDelete(_lock);
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDP::close() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     UDP_MUTEX_LOCK(); | 
					
						
							|  |  |  |     if(_pcb != NULL) { | 
					
						
							|  |  |  |         if(_connected) { | 
					
						
							|  |  |  |             _udp_disconnect(_pcb); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         _connected = false; | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         //todo: unjoin multicast group
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::connect(const ip_addr_t *addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_udp_task_start()){ | 
					
						
							|  |  |  |         log_e("failed to start task"); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     if(!_init()) { | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     close(); | 
					
						
							|  |  |  |     UDP_MUTEX_LOCK(); | 
					
						
							|  |  |  |     err_t err = _udp_connect(_pcb, addr, port); | 
					
						
							|  |  |  |     if(err != ERR_OK) { | 
					
						
							|  |  |  |         UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     _connected = true; | 
					
						
							|  |  |  |     UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listen(const ip_addr_t *addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_udp_task_start()){ | 
					
						
							|  |  |  |         log_e("failed to start task"); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     if(!_init()) { | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     close(); | 
					
						
							|  |  |  |     if(addr){ | 
					
						
							|  |  |  |         IP_SET_TYPE_VAL(_pcb->local_ip,  addr->type); | 
					
						
							|  |  |  |         IP_SET_TYPE_VAL(_pcb->remote_ip, addr->type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     UDP_MUTEX_LOCK(); | 
					
						
							|  |  |  |     if(_udp_bind(_pcb, addr, port) != ERR_OK) { | 
					
						
							|  |  |  |         UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     _connected = true; | 
					
						
							|  |  |  |     UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | static esp_err_t joinMulticastGroup(const ip_addr_t *addr, bool join, tcpip_adapter_if_t tcpip_if=TCPIP_ADAPTER_IF_MAX) | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     struct netif * netif = NULL; | 
					
						
							|  |  |  |     if(tcpip_if < TCPIP_ADAPTER_IF_MAX){ | 
					
						
							|  |  |  |         void * nif = NULL; | 
					
						
							|  |  |  |         esp_err_t err = tcpip_adapter_get_netif(tcpip_if, &nif); | 
					
						
							|  |  |  |         if (err) { | 
					
						
							|  |  |  |             return ESP_ERR_INVALID_ARG; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         netif = (struct netif *)nif; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     if (addr->type == IPADDR_TYPE_V4) { | 
					
						
							|  |  |  |         if(join){ | 
					
						
							|  |  |  |             if (igmp_joingroup_netif(netif, (const ip4_addr *)&(addr->u_addr.ip4))) { | 
					
						
							|  |  |  |                 return ESP_ERR_INVALID_STATE; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |             if (igmp_leavegroup_netif(netif, (const ip4_addr *)&(addr->u_addr.ip4))) { | 
					
						
							|  |  |  |                 return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         if(join){ | 
					
						
							|  |  |  |             if (mld6_joingroup_netif(netif, &(addr->u_addr.ip6))) { | 
					
						
							|  |  |  |                 return ESP_ERR_INVALID_STATE; | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |             if (mld6_leavegroup_netif(netif, &(addr->u_addr.ip6))) { | 
					
						
							|  |  |  |                 return ESP_ERR_INVALID_STATE; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return ESP_OK; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  | bool AsyncUDP::listenMulticast(const ip_addr_t *addr, uint16_t port, uint8_t ttl, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!ip_addr_ismulticast(addr)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     if (joinMulticastGroup(addr, true, tcpip_if)!= ERR_OK) { | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     if(!listen(NULL, port)) { | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UDP_MUTEX_LOCK(); | 
					
						
							|  |  |  |     _pcb->mcast_ttl = ttl; | 
					
						
							|  |  |  |     _pcb->remote_port = port; | 
					
						
							|  |  |  |     ip_addr_copy(_pcb->remote_ip, *addr); | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |     //ip_addr_copy(_pcb->remote_ip, ip_addr_any_type);
 | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |     UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::writeTo(const uint8_t * data, size_t len, const ip_addr_t * addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_pcb) { | 
					
						
							|  |  |  |         UDP_MUTEX_LOCK(); | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         _pcb = udp_new(); | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |         UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |         if(_pcb == NULL) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if(len > CONFIG_TCP_MSS) { | 
					
						
							|  |  |  |         len = CONFIG_TCP_MSS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     err_t err = ERR_OK; | 
					
						
							|  |  |  |     pbuf* pbt = pbuf_alloc(PBUF_TRANSPORT, len, PBUF_RAM); | 
					
						
							|  |  |  |     if(pbt != NULL) { | 
					
						
							|  |  |  |         uint8_t* dst = reinterpret_cast<uint8_t*>(pbt->payload); | 
					
						
							|  |  |  |         memcpy(dst, data, len); | 
					
						
							|  |  |  |         UDP_MUTEX_LOCK(); | 
					
						
							| 
									
										
										
										
											2018-09-21 08:39:36 +02:00
										 |  |  |         if(tcpip_if < TCPIP_ADAPTER_IF_MAX){ | 
					
						
							| 
									
										
										
										
											2018-06-27 09:01:06 +02:00
										 |  |  |             void * nif = NULL; | 
					
						
							|  |  |  |             tcpip_adapter_get_netif((tcpip_adapter_if_t)tcpip_if, &nif); | 
					
						
							|  |  |  |             if(!nif){ | 
					
						
							|  |  |  |                 err = _udp_sendto(_pcb, pbt, addr, port); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 err = _udp_sendto_if(_pcb, pbt, addr, port, (struct netif *)nif); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             err = _udp_sendto(_pcb, pbt, addr, port); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         UDP_MUTEX_UNLOCK(); | 
					
						
							|  |  |  |         pbuf_free(pbt); | 
					
						
							|  |  |  |         if(err < ERR_OK) { | 
					
						
							|  |  |  |             return 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return len; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDP::_recv(udp_pcb *upcb, pbuf *pb, const ip_addr_t *addr, uint16_t port, struct netif * netif) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while(pb != NULL) { | 
					
						
							|  |  |  |         pbuf * this_pb = pb; | 
					
						
							|  |  |  |         pb = pb->next; | 
					
						
							|  |  |  |         this_pb->next = NULL; | 
					
						
							|  |  |  |         if(_handler) { | 
					
						
							|  |  |  |             AsyncUDPPacket packet(this, this_pb, addr, port, netif); | 
					
						
							|  |  |  |             _handler(packet); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             pbuf_free(this_pb); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDP::_s_recv(void *arg, udp_pcb *upcb, pbuf *p, const ip_addr_t *addr, uint16_t port, struct netif * netif) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     reinterpret_cast<AsyncUDP*>(arg)->_recv(upcb, p, addr, port, netif); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listen(uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return listen(IP_ANY_TYPE, port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listen(const IPAddress addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t laddr; | 
					
						
							|  |  |  |     laddr.type = IPADDR_TYPE_V4; | 
					
						
							|  |  |  |     laddr.u_addr.ip4.addr = addr; | 
					
						
							|  |  |  |     return listen(&laddr, port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listenMulticast(const IPAddress addr, uint16_t port, uint8_t ttl, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t laddr; | 
					
						
							|  |  |  |     laddr.type = IPADDR_TYPE_V4; | 
					
						
							|  |  |  |     laddr.u_addr.ip4.addr = addr; | 
					
						
							|  |  |  |     return listenMulticast(&laddr, port, ttl, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::connect(const IPAddress addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t daddr; | 
					
						
							|  |  |  |     daddr.type = IPADDR_TYPE_V4; | 
					
						
							|  |  |  |     daddr.u_addr.ip4.addr = addr; | 
					
						
							|  |  |  |     return connect(&daddr, port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::writeTo(const uint8_t *data, size_t len, const IPAddress addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t daddr; | 
					
						
							|  |  |  |     daddr.type = IPADDR_TYPE_V4; | 
					
						
							|  |  |  |     daddr.u_addr.ip4.addr = addr; | 
					
						
							|  |  |  |     return writeTo(data, len, &daddr, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPAddress AsyncUDP::listenIP() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_pcb || _pcb->remote_ip.type != IPADDR_TYPE_V4){ | 
					
						
							|  |  |  |         return IPAddress(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPAddress(_pcb->remote_ip.u_addr.ip4.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listen(const IPv6Address addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t laddr; | 
					
						
							|  |  |  |     laddr.type = IPADDR_TYPE_V6; | 
					
						
							|  |  |  |     memcpy((uint8_t*)(laddr.u_addr.ip6.addr), (const uint8_t*)addr, 16); | 
					
						
							|  |  |  |     return listen(&laddr, port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::listenMulticast(const IPv6Address addr, uint16_t port, uint8_t ttl, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t laddr; | 
					
						
							|  |  |  |     laddr.type = IPADDR_TYPE_V6; | 
					
						
							|  |  |  |     memcpy((uint8_t*)(laddr.u_addr.ip6.addr), (const uint8_t*)addr, 16); | 
					
						
							|  |  |  |     return listenMulticast(&laddr, port, ttl, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::connect(const IPv6Address addr, uint16_t port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t daddr; | 
					
						
							|  |  |  |     daddr.type = IPADDR_TYPE_V6; | 
					
						
							|  |  |  |     memcpy((uint8_t*)(daddr.u_addr.ip6.addr), (const uint8_t*)addr, 16); | 
					
						
							|  |  |  |     return connect(&daddr, port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::writeTo(const uint8_t *data, size_t len, const IPv6Address addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ip_addr_t daddr; | 
					
						
							|  |  |  |     daddr.type = IPADDR_TYPE_V6; | 
					
						
							|  |  |  |     memcpy((uint8_t*)(daddr.u_addr.ip6.addr), (const uint8_t*)addr, 16); | 
					
						
							|  |  |  |     return writeTo(data, len, &daddr, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | IPv6Address AsyncUDP::listenIPv6() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!_pcb || _pcb->remote_ip.type != IPADDR_TYPE_V6){ | 
					
						
							|  |  |  |         return IPv6Address(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return IPv6Address(_pcb->remote_ip.u_addr.ip6.addr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::write(const uint8_t *data, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return writeTo(data, len, &(_pcb->remote_ip), _pcb->remote_port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::write(uint8_t data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return write(&data, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcastTo(uint8_t *data, size_t len, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return writeTo(data, len, IP_ADDR_BROADCAST, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcastTo(const char * data, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return broadcastTo((uint8_t *)data, strlen(data), port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcast(uint8_t *data, size_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(_pcb->local_port != 0) { | 
					
						
							|  |  |  |         return broadcastTo(data, len, _pcb->local_port); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcast(const char * data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return broadcast((uint8_t *)data, strlen(data)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::sendTo(AsyncUDPMessage &message, const ip_addr_t *addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return writeTo(message.data(), message.length(), addr, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::sendTo(AsyncUDPMessage &message, const IPAddress addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return writeTo(message.data(), message.length(), addr, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::sendTo(AsyncUDPMessage &message, const IPv6Address addr, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return writeTo(message.data(), message.length(), addr, port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::send(AsyncUDPMessage &message) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return writeTo(message.data(), message.length(), &(_pcb->remote_ip), _pcb->remote_port); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcastTo(AsyncUDPMessage &message, uint16_t port, tcpip_adapter_if_t tcpip_if) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return broadcastTo(message.data(), message.length(), port, tcpip_if); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t AsyncUDP::broadcast(AsyncUDPMessage &message) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if(!message) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return broadcast(message.data(), message.length()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AsyncUDP::operator bool() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _connected; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool AsyncUDP::connected() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _connected; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDP::onPacket(AuPacketHandlerFunctionWithArg cb, void * arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     onPacket(std::bind(cb, arg, std::placeholders::_1)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void AsyncUDP::onPacket(AuPacketHandlerFunction cb) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _handler = cb; | 
					
						
							|  |  |  | } |