2016-10-28 11:57:21 +01:00
/* Copyright (c) 2013 Nordic Semiconductor. All Rights Reserved.
*
* The information contained herein is property of Nordic Semiconductor ASA .
* Terms and conditions of usage are described in detail in NORDIC
* SEMICONDUCTOR STANDARD SOFTWARE LICENSE AGREEMENT .
*
* Licensees are granted free , non - transferable use of the information . NO
* WARRANTY of ANY KIND is provided . This heading must NOT be removed from
* the file .
*
*/
/* clang-format off */
/** @file
*
* @ defgroup iot_sdk_tcp_server main . c
* @ {
* @ ingroup iot_sdk_app_lwip
* @ brief This file contains the source code for LwIP TCP Server sample application .
*
*/
# include <stdbool.h>
# include <stdint.h>
# include "boards.h"
# include "app_timer_appsh.h"
# include "app_scheduler.h"
# include "app_button.h"
# include "nordic_common.h"
# include "softdevice_handler_appsh.h"
# include "ble_advdata.h"
# include "ble_srv_common.h"
# include "ble_ipsp.h"
# include "ble_6lowpan.h"
# include "mem_manager.h"
# include "app_trace.h"
2016-11-04 18:54:40 +00:00
/*
* arm - none - eabi - gcc has BYTE_ORDER already defined , so in order to avoid
* warnings in lwip , we have to undefine it
*
* TODO : Check if in the future versions of nRF5 SDK that changes .
* Current version of nRF51 SDK : 0.8 .0
* nRF5 SDK : 0.9 .0
*/
# undef BYTE_ORDER
2016-10-28 11:57:21 +01:00
# include "lwip/init.h"
# include "lwip/inet6.h"
# include "lwip/ip6.h"
# include "lwip/ip6_addr.h"
# include "lwip/netif.h"
/*lint -save -e607 */
# include "lwip/tcp.h"
/*lint -restore */
# include "lwip/timers.h"
# include "nrf_platform_port.h"
# include "app_util_platform.h"
# define DEVICE_NAME "LwIPTCPServer" /**< Device name used in BLE undirected advertisement. */
# define APP_TIMER_PRESCALER NRF51_DRIVER_TIMER_PRESCALER /**< Value of the RTC1 PRESCALER register. */
# define APP_TIMER_MAX_TIMERS 1 /**< Maximum number of simultaneously created timers. */
# define APP_TIMER_OP_QUEUE_SIZE 1
# define LWIP_SYS_TIMER_INTERVAL APP_TIMER_TICKS(250, APP_TIMER_PRESCALER) /**< Interval for timer used as trigger to send. */
# define SCHED_MAX_EVENT_DATA_SIZE 128 /**< Maximum size of scheduler events. */
# define SCHED_QUEUE_SIZE 12 /**< Maximum number of events in the scheduler queue. */
# define ADVERTISING_LED BSP_LED_0_MASK /**< Is on when device is advertising. */
# define CONNECTED_LED BSP_LED_1_MASK /**< Is on when device is connected. */
# define TCP_CONNECTED_LED BSP_LED_2_MASK /**< Is on when device is connected. */
# define DISPLAY_LED_0 BSP_LED_0_MASK /**< LED used for displaying mod 4 of data payload received on UDP port. */
# define DISPLAY_LED_1 BSP_LED_1_MASK /**< LED used for displaying mod 4 of data payload received on UDP port. */
# define DISPLAY_LED_2 BSP_LED_2_MASK /**< LED used for displaying mod 4 of data payload received on UDP port. */
# define DISPLAY_LED_3 BSP_LED_3_MASK /**< LED used for displaying mod 4 of data payload received on UDP port. */
# define ALL_APP_LED (BSP_LED_0_MASK | BSP_LED_1_MASK | \
BSP_LED_2_MASK | BSP_LED_3_MASK ) /**< Define used for simultaneous operation of all application LEDs. */
# define APP_ADV_TIMEOUT 0 /**< Time for which the device must be advertising in non-connectable mode (in seconds). 0 disables timeout. */
# define APP_ADV_ADV_INTERVAL MSEC_TO_UNITS(100, UNIT_0_625_MS) /**< The advertising interval. This value can vary between 100ms to 10.24s). */
# define DEAD_BEEF 0xDEADBEEF /**< Value used as error code on stack dump, can be used to identify stack location on stack unwind. */
# define APPL_LOG app_trace_log /**< Macro for logging application messages on UART, in case ENABLE_DEBUG_LOG_SUPPORT is not defined, no logging occurs. */
# define APPL_DUMP app_trace_dump /**< Macro for dumping application data on UART, in case ENABLE_DEBUG_LOG_SUPPORT is not defined, no logging occurs. */
# define TCP_SERVER_PORT 9000 /**< TCP server listen port number. */
# define TCP_DATA_SIZE 8 /**< UDP Data size sent on remote. */
typedef enum
{
TCP_STATE_IDLE ,
TCP_STATE_REQUEST_CONNECTION ,
TCP_STATE_CONNECTED ,
TCP_STATE_DATA_TX_IN_PROGRESS ,
TCP_STATE_DISCONNECTED
} tcp_state_t ;
eui64_t eui64_local_iid ; /**< Local EUI64 value that is used as the IID for*/
static ble_gap_adv_params_t m_adv_params ; /**< Parameters to be passed to the stack when starting advertising. */
static app_timer_id_t m_sys_timer_id ; /**< System Timer used to service LwIP timers periodically. */
static struct tcp_pcb * mp_tcp_port ; /**< TCP Port to listen on. */
static tcp_state_t m_tcp_state ; /**< TCP State information. */
/**@brief Function for error handling, which is called when an error has occurred.
*
* @ warning This handler is an example only and does not fit a final product . You need to analyse
* how your product is supposed to react in case of error .
*
* @ param [ in ] error_code Error code supplied to the handler .
* @ param [ in ] line_num Line number where the handler is called .
* @ param [ in ] p_file_name Pointer to the file name .
*/
void app_error_handler ( uint32_t error_code , uint32_t line_num , const uint8_t * p_file_name )
{
//Halt the application and notify of error using the LEDs.
APPL_LOG ( " [** ASSERT **]: Error 0x%08lX, Line %ld, File %s \r \n " , error_code , line_num , p_file_name ) ;
LEDS_ON ( ALL_APP_LED ) ;
for ( ; ; )
{
}
// @note: In case on assert, it is desired to only recover and reset, uncomment the line below.
//NVIC_SystemReset();
}
/**@brief Callback function for asserts in the SoftDevice.
*
* @ details This function will be called in case of an assert in the SoftDevice .
*
* @ warning This handler is an example only and does not fit a final product . You need to analyse
* how your product is supposed to react in case of Assert .
* @ warning On assert from the SoftDevice , the system can only recover on reset .
*
* @ param [ in ] line_num Line number of the failing ASSERT call .
* @ param [ in ] file_name File name of the failing ASSERT call .
*/
void assert_nrf_callback ( uint16_t line_num , const uint8_t * p_file_name )
{
app_error_handler ( DEAD_BEEF , line_num , p_file_name ) ;
}
/**@brief Function for the LEDs initialization.
*
* @ details Initializes all LEDs used by this application .
*/
static void leds_init ( void )
{
// Configure application LED pins.
LEDS_CONFIGURE ( ALL_APP_LED ) ;
// Turn off all LED on initialization.
LEDS_OFF ( ALL_APP_LED ) ;
}
/**@brief Function for initializing the Advertising functionality.
*
* @ details Encodes the required advertising data and passes it to the stack .
* Also builds a structure to be passed to the stack when starting advertising .
*/
static void advertising_init ( void )
{
uint32_t err_code ;
ble_advdata_t advdata ;
uint8_t flags = BLE_GAP_ADV_FLAG_BR_EDR_NOT_SUPPORTED ;
ble_gap_conn_sec_mode_t sec_mode ;
ble_gap_addr_t my_addr ;
BLE_GAP_CONN_SEC_MODE_SET_OPEN ( & sec_mode ) ;
err_code = sd_ble_gap_device_name_set ( & sec_mode ,
( const uint8_t * ) DEVICE_NAME ,
strlen ( DEVICE_NAME ) ) ;
APP_ERROR_CHECK ( err_code ) ;
err_code = sd_ble_gap_address_get ( & my_addr ) ;
APP_ERROR_CHECK ( err_code ) ;
my_addr . addr [ 5 ] = 0x00 ;
my_addr . addr_type = BLE_GAP_ADDR_TYPE_PUBLIC ;
err_code = sd_ble_gap_address_set ( & my_addr ) ;
APP_ERROR_CHECK ( err_code ) ;
IPV6_EUI64_CREATE_FROM_EUI48 ( eui64_local_iid . identifier ,
my_addr . addr ,
my_addr . addr_type ) ;
ble_uuid_t adv_uuids [ ] =
{
{ BLE_UUID_IPSP_SERVICE , BLE_UUID_TYPE_BLE }
} ;
//Build and set advertising data.
memset ( & advdata , 0 , sizeof ( advdata ) ) ;
advdata . name_type = BLE_ADVDATA_FULL_NAME ;
advdata . flags = flags ;
advdata . uuids_complete . uuid_cnt = sizeof ( adv_uuids ) / sizeof ( adv_uuids [ 0 ] ) ;
advdata . uuids_complete . p_uuids = adv_uuids ;
err_code = ble_advdata_set ( & advdata , NULL ) ;
APP_ERROR_CHECK ( err_code ) ;
//Initialize advertising parameters (used when starting advertising).
memset ( & m_adv_params , 0 , sizeof ( m_adv_params ) ) ;
m_adv_params . type = BLE_GAP_ADV_TYPE_ADV_IND ;
m_adv_params . p_peer_addr = NULL ; // Undirected advertisement.
m_adv_params . fp = BLE_GAP_ADV_FP_ANY ;
m_adv_params . interval = APP_ADV_ADV_INTERVAL ;
m_adv_params . timeout = APP_ADV_TIMEOUT ;
}
/**@brief Function for starting advertising.
*/
static void advertising_start ( void )
{
uint32_t err_code ;
err_code = sd_ble_gap_adv_start ( & m_adv_params ) ;
APP_ERROR_CHECK ( err_code ) ;
LEDS_ON ( ADVERTISING_LED ) ;
}
/**@brief Function for handling the Application's BLE Stack events.
*
* @ param [ in ] p_ble_evt Bluetooth stack event .
*/
static void on_ble_evt ( ble_evt_t * p_ble_evt )
{
switch ( p_ble_evt - > header . evt_id )
{
case BLE_GAP_EVT_CONNECTED :
APPL_LOG ( " [APPL]: Connected. \r \n " ) ;
break ;
case BLE_GAP_EVT_DISCONNECTED :
APPL_LOG ( " [APPL]: Disconnected. \r \n " ) ;
advertising_start ( ) ;
break ;
default :
break ;
}
}
/**@brief Function for dispatching a BLE stack event to all modules with a BLE stack event handler.
*
* @ details This function is called from the BLE Stack event interrupt handler after a BLE stack
* event has been received .
*
* @ param [ in ] p_ble_evt Bluetooth stack event .
*/
static void ble_evt_dispatch ( ble_evt_t * p_ble_evt )
{
ble_ipsp_evt_handler ( p_ble_evt ) ;
on_ble_evt ( p_ble_evt ) ;
}
/**@brief Function for initializing the BLE stack.
*
* @ details Initializes the SoftDevice and the BLE event interrupt .
*/
static void ble_stack_init ( void )
{
uint32_t err_code ;
// Initialize the SoftDevice handler module.
SOFTDEVICE_HANDLER_APPSH_INIT ( NRF_CLOCK_LFCLKSRC_XTAL_20_PPM , true ) ;
// Register with the SoftDevice handler module for BLE events.
err_code = softdevice_ble_evt_handler_set ( ble_evt_dispatch ) ;
APP_ERROR_CHECK ( err_code ) ;
}
/**@brief Function for the Event Scheduler initialization.
*/
static void scheduler_init ( void )
{
APP_SCHED_INIT ( SCHED_MAX_EVENT_DATA_SIZE , SCHED_QUEUE_SIZE ) ;
}
/**@brief Function to TCP port.
*
* @ details Function to close the TCP port and reset any information on the port .
*/
static void tcp_port_close ( struct tcp_pcb * p_pcb )
{
m_tcp_state = TCP_STATE_REQUEST_CONNECTION ;
//Reset all information set on and/or callback registered for the port.
tcp_arg ( p_pcb , NULL ) ;
tcp_sent ( p_pcb , NULL ) ;
tcp_recv ( p_pcb , NULL ) ;
tcp_err ( p_pcb , NULL ) ;
tcp_poll ( p_pcb , NULL , 0 ) ;
UNUSED_VARIABLE ( tcp_close ( p_pcb ) ) ;
LEDS_OFF ( ( DISPLAY_LED_0 | DISPLAY_LED_1 | DISPLAY_LED_2 | DISPLAY_LED_3 ) ) ;
LEDS_ON ( CONNECTED_LED ) ;
}
/**@brief TCP Port Write complete callback.
*
* @ details Calbback registered to be notified of the write complete event on the TCP port .
* In case write complete is notified with ' zero ' length , port is closed .
*
* @ param [ in ] p_arg Receive argument set on the port .
* @ param [ in ] p_pcb PCB identifier of the port .
* @ param [ in ] len Length of data written successfully .
*
* @ retval ERR_OK .
*/
static err_t tcp_write_complete ( void * p_arg ,
struct tcp_pcb * p_pcb ,
u16_t len )
{
UNUSED_PARAMETER ( p_arg ) ;
UNUSED_PARAMETER ( p_pcb ) ;
if ( len ! = 0 )
{
//Write complete, reset the state to connected from transmit pending.
m_tcp_state = TCP_STATE_CONNECTED ;
}
else
{
//Something is not right on the port, close the port.
tcp_port_close ( mp_tcp_port ) ;
}
return ERR_OK ;
}
/**@brief Send test data on the port.
*
* @ details Sends TCP data in Request of size 8 in format described in description above .
*
* @ param [ in ] p_pcb PCB identifier of the port .
*/
static void tcp_send_data ( struct tcp_pcb * p_pcb , uint32_t sequence_number )
{
err_t err = ERR_OK ;
if ( m_tcp_state ! = TCP_STATE_DATA_TX_IN_PROGRESS )
{
//Register callback to get notification of data reception is complete.
tcp_sent ( p_pcb , tcp_write_complete ) ;
uint8_t tcp_data [ TCP_DATA_SIZE ] ;
tcp_data [ 0 ] = ( uint8_t ) ( ( sequence_number > > 24 ) & 0x000000FF ) ;
tcp_data [ 1 ] = ( uint8_t ) ( ( sequence_number > > 16 ) & 0x000000FF ) ;
tcp_data [ 2 ] = ( uint8_t ) ( ( sequence_number > > 8 ) & 0x000000FF ) ;
tcp_data [ 3 ] = ( uint8_t ) ( sequence_number & 0x000000FF ) ;
tcp_data [ 4 ] = ' P ' ;
tcp_data [ 5 ] = ' o ' ;
tcp_data [ 6 ] = ' n ' ;
tcp_data [ 7 ] = ' g ' ;
//Enqueue data for transmission.
err = tcp_write ( p_pcb , tcp_data , TCP_DATA_SIZE , 1 ) ;
if ( err ! = ERR_OK )
{
APPL_LOG ( " [APPL]: Failed to send TCP packet, reason %d \r \n " , err ) ;
}
else
{
m_tcp_state = TCP_STATE_DATA_TX_IN_PROGRESS ;
}
}
else
{
//Wait for tx to be complete.
}
}
/**@brief Callback registered for receiving data on the TCP Port.
*
* @ param [ in ] p_arg Receive argument set on the TCP port .
* @ param [ in ] p_pcb TCP PCB on which data is received .
* @ param [ in ] p_buffer Buffer with received data .
* @ param [ in ] err Event result indicating error associated with the receive ,
* if any , else ERR_OK .
*/
err_t tcp_recv_data_handler ( void * p_arg ,
struct tcp_pcb * p_pcb ,
struct pbuf * p_buffer ,
err_t err )
{
APPL_LOG ( " [APPL]: >> TCP Data. \r \n " ) ;
//Check event result before proceeding.
if ( err = = ERR_OK )
{
uint8_t * p_data = p_buffer - > payload ;
if ( p_buffer - > len = = TCP_DATA_SIZE )
{
uint32_t sequence_number = 0 ;
sequence_number = ( ( p_data [ 0 ] < < 24 ) & 0xFF000000 ) ;
sequence_number | = ( ( p_data [ 1 ] < < 16 ) & 0x00FF0000 ) ;
sequence_number | = ( ( p_data [ 2 ] < < 8 ) & 0x0000FF00 ) ;
sequence_number | = ( p_data [ 3 ] & 0x000000FF ) ;
LEDS_OFF ( ALL_APP_LED ) ;
if ( sequence_number & 0x00000001 )
{
LEDS_ON ( DISPLAY_LED_0 ) ;
}
if ( sequence_number & 0x00000002 )
{
LEDS_ON ( DISPLAY_LED_1 ) ;
}
if ( sequence_number & 0x00000004 )
{
LEDS_ON ( DISPLAY_LED_2 ) ;
}
if ( sequence_number & 0x00000008 )
{
LEDS_ON ( DISPLAY_LED_3 ) ;
}
//Send Response
tcp_send_data ( p_pcb , sequence_number ) ;
}
else
{
APPL_LOG ( " [APPL]: TCP data received in incorrect format. \r \n " ) ;
}
// All is good with the data received, process it.
tcp_recved ( p_pcb , p_buffer - > tot_len ) ;
UNUSED_VARIABLE ( pbuf_free ( p_buffer ) ) ;
}
else
{
//Free the buffer in case its not NULL.
if ( p_buffer ! = NULL )
{
UNUSED_VARIABLE ( pbuf_free ( p_buffer ) ) ;
}
//Something is not right with the port, close the port.
tcp_port_close ( mp_tcp_port ) ;
}
return ERR_OK ;
}
/**@brief Function for initializing IP stack.
*
* @ details Initialize the IP Stack and its driver .
*/
static void ip_stack_init ( void )
{
uint32_t err_code = nrf51_sdk_mem_init ( ) ;
APP_ERROR_CHECK ( err_code ) ;
//Initialize LwIP stack.
lwip_init ( ) ;
//Initialize LwIP stack driver.
err_code = nrf51_driver_init ( ) ;
APP_ERROR_CHECK ( err_code ) ;
}
/**@brief Timer callback used for periodic servicing of LwIP protocol timers.
*
* @ details Timer callback used for periodic servicing of LwIP protocol timers .
*
* @ param [ in ] p_context Pointer used for passing context . No context used in this application .
*/
static void system_timer_callback ( void * p_context )
{
UNUSED_VARIABLE ( p_context ) ;
sys_check_timeouts ( ) ;
}
/**@brief Function for the Timer initialization.
*
* @ details Initializes the timer module . This creates and starts application timers .
*/
static void timers_init ( void )
{
uint32_t err_code ;
// Initialize timer module.
APP_TIMER_APPSH_INIT ( APP_TIMER_PRESCALER , APP_TIMER_MAX_TIMERS , APP_TIMER_OP_QUEUE_SIZE , true ) ;
// Create timers.
err_code = app_timer_create ( & m_sys_timer_id ,
APP_TIMER_MODE_REPEATED ,
system_timer_callback ) ;
APP_ERROR_CHECK ( err_code ) ;
}
/**@brief Function to handle interface up event. */
void nrf51_driver_interface_up ( void )
{
uint32_t err_code ;
APPL_LOG ( " [APPL]: IPv6 interface up. \r \n " ) ;
sys_check_timeouts ( ) ;
m_tcp_state = TCP_STATE_REQUEST_CONNECTION ;
err_code = app_timer_start ( m_sys_timer_id , LWIP_SYS_TIMER_INTERVAL , NULL ) ;
APP_ERROR_CHECK ( err_code ) ;
LEDS_OFF ( ADVERTISING_LED ) ;
LEDS_ON ( CONNECTED_LED ) ;
}
/**@brief Function to handle interface down event. */
void nrf51_driver_interface_down ( void )
{
uint32_t err_code ;
APPL_LOG ( " [APPL]: IPv6 interface down. \r \n " ) ;
err_code = app_timer_stop ( m_sys_timer_id ) ;
APP_ERROR_CHECK ( err_code ) ;
LEDS_OFF ( ( DISPLAY_LED_0 | DISPLAY_LED_1 | DISPLAY_LED_2 | DISPLAY_LED_3 ) ) ;
LEDS_ON ( ADVERTISING_LED ) ;
m_tcp_state = TCP_STATE_DISCONNECTED ;
}
void bleconfig_init ( void ) {
//Initialize.
app_trace_init ( ) ;
leds_init ( ) ;
timers_init ( ) ;
ble_stack_init ( ) ;
advertising_init ( ) ;
ip_stack_init ( ) ;
scheduler_init ( ) ;
APPL_LOG ( " \r \n " ) ;
APPL_LOG ( " [APPL]: Init done. \r \n " ) ;
//Start execution.
advertising_start ( ) ;
}
void bleconfig_poll ( void ) {
//Execute event schedule.
app_sched_execute ( ) ;
}