#include "tx_api.h" #include "nx_api.h" #include "netxtestcontrol.h" extern void test_control_return(UINT); #if !defined(NX_DISABLE_IPV4) && defined(__PRODUCT_NETXDUO__) && !defined(NX_DISABLE_PACKET_CHAIN) #include "nx_rtp_sender.h" #define DEMO_STACK_SIZE 4096 #define NUM_PACKETS 10 #define PACKET_SIZE 1536 #define PACKET_POOL_SIZE (NUM_PACKETS * (PACKET_SIZE + sizeof(NX_PACKET))) #define RTP_SERVER_ADDRESS IP_ADDRESS(1,2,3,4) #define RTP_CLIENT_ADDRESS IP_ADDRESS(1,2,3,5) #define RTP_CLIENT_RTP_PORT 6002 #define RTP_CLIENT_RTCP_PORT 6003 #define RTP_PAYLOAD_TYPE 96 #define CNAME "AzureRTOS@microsoft.com" /* Define test data. */ #define TEST_TIMESTAMP 1234 #define TEST_MSW 123 #define TEST_LSW 456 /* Define the number of tests to do */ #define TEST_CYCLES 4 /* Define jpeg test data */ #define TEST_JPEG_Q_TABLE_SIZE 64 #define TEST_JPEG_Q_TABLE_START_POS (2 + 2 + 0x10 + 5) #define TEST_JPEG_IMAGE_DATA_START_POS (TEST_JPEG_Q_TABLE_START_POS + TEST_JPEG_Q_TABLE_SIZE + 2 + 0x11 + 2 + 0x0C) static UCHAR test_rtp_packet_data[] = { 0xFF, 0xD8, /* JPEG file header */ /* APP0 */ 0xFF, 0xE0, 0x00, 0x10, 0x4A, 0x46, 0x49, 0x46, 0x00, 0x01, 0x01, 0x00, 0x00, 0x48, 0x00, 0x48, 0x00, 0x00, /* Quntization tables */ 0xFF, 0xDB, 0x00, 0x43, 0x00, 0x07, 0x05, 0x05, 0x06, 0x05, 0x04, 0x07, 0x06, 0x06, 0x06, 0x08, 0x07, 0x07, 0x08, 0x0B, 0x12, 0x0B, 0x0B, 0x0A, 0x0A, 0x0B, 0x16, 0x0F, 0x10, 0x0D, 0x12, 0x1A, 0x16, 0x1B, 0x1A, 0x19, 0x16, 0x19, 0x18, 0x1C, 0x20, 0x28, 0x22, 0x1C, 0x1E, 0x26, 0x1E, 0x18, 0x19, 0x23, 0x30, 0x24, 0x26, 0x2A, 0x2B, 0x2D, 0x2E, 0x2D, 0x1B, 0x22, 0x32, 0x35, 0x31, 0x2C, 0x35, 0x28, 0x2C, 0x2D, 0x2C, /* Baseline information */ 0xFF, 0xC0, 0x00, 0x11, 0x08, 0x00, 0x84, 0x00, 0x84, 0x03, 0x01, 0x21, 0x00, 0x02, 0x11, 0x01, 0x03, 0x11, 0x01, 0xFF, 0xDA, /* JPEG image data begin marker */ 0x00, 0x0C, 0x03, 0x01, 0x00, 0x02, 0x11, 0x03, 0x11, 0x00, 0x3F, 0x00, /* Data header */ 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, /* Test data */ 0xFF, 0xD9 /* JPEG image data end marker*/ }; /* test_rtp_packet_data */ #define TEST_LONG_JPEG_Q_TABLE_1_START_POS (2 + 2 + 0x10 + 5) #define TEST_LONG_JPEG_Q_TABLE_2_START_POS (TEST_LONG_JPEG_Q_TABLE_1_START_POS + TEST_JPEG_Q_TABLE_SIZE + 5) #define TEST_LONG_JPEG_BASELINE_INFO_START_POS (TEST_LONG_JPEG_Q_TABLE_2_START_POS + TEST_JPEG_Q_TABLE_SIZE) #define TEST_LONG_JPEG_HUFFMAN_TABLE_START_POS (TEST_LONG_JPEG_BASELINE_INFO_START_POS + 2 + 0x11) #define TEST_LONG_JPEG_IMAGE_DATA_START_POS (TEST_LONG_JPEG_HUFFMAN_TABLE_START_POS + 2 + 0x1C + 2 + 0x3E + 2 + 0x19 + 2 + 0x23 + 2 + 0x0C) static UCHAR test_long_rtp_packet_data[] = { 0xFF, 0xD8, /* JPEG file header */ /* APP0 */ 0xFF, 0xE0, 0x00, 0x10, 0x4A, 0x46, 0x49, 0x46, 0x00, 0x01, 0x01, 0x00, 0x00, 0x48, 0x00, 0x48, 0x00, 0x00, /* Quntization tables */ 0xFF, 0xDB, 0x00, 0x43, 0x00, 0x07, 0x05, 0x05, 0x06, 0x05, 0x04, 0x07, 0x06, 0x06, 0x06, 0x08, 0x07, 0x07, 0x08, 0x0B, 0x12, 0x0B, 0x0B, 0x0A, 0x0A, 0x0B, 0x16, 0x0F, 0x10, 0x0D, 0x12, 0x1A, 0x16, 0x1B, 0x1A, 0x19, 0x16, 0x19, 0x18, 0x1C, 0x20, 0x28, 0x22, 0x1C, 0x1E, 0x26, 0x1E, 0x18, 0x19, 0x23, 0x30, 0x24, 0x26, 0x2A, 0x2B, 0x2D, 0x2E, 0x2D, 0x1B, 0x22, 0x32, 0x35, 0x31, 0x2C, 0x35, 0x28, 0x2C, 0x2D, 0x2C, 0xFF, 0xDB, 0x00, 0x43, 0x01, 0x07, 0x08, 0x08, 0x0B, 0x09, 0x0B, 0x15, 0x0B, 0x0B, 0x15, 0x2C, 0x1D, 0x19, 0x1D, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, 0x2C, /* Baseline information */ 0xFF, 0xC0, 0x00, 0x11, 0x08, 0x00, 0x84, 0x00, 0x84, 0x03, 0x01, 0x22, 0x00, 0x02, 0x11, 0x01, 0x03, 0x11, 0x01, /* Huffman tables */ 0xFF, 0xC4, 0x00, 0x1C, 0x00, 0x00, 0x01, 0x05, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x04, 0x05, 0x06, 0x07, 0x01, 0x02, 0x08, 0xFF, 0xC4, 0x00, 0x3E, 0x10, 0x00, 0x02, 0x01, 0x03, 0x02, 0x03, 0x05, 0x04, 0x06, 0x08, 0x06, 0x03, 0x00, 0x00, 0x00, 0x00, 0x01, 0x02, 0x03, 0x00, 0x04, 0x11, 0x05, 0x12, 0x06, 0x21, 0x31, 0x13, 0x41, 0x51, 0x61, 0x71, 0x07, 0x22, 0x32, 0x72, 0x14, 0x42, 0x81, 0x91, 0xA1, 0xC1, 0x23, 0x33, 0x43, 0x52, 0x62, 0xB1, 0xD1, 0xE1, 0x15, 0x24, 0x34, 0x92, 0xA2, 0xF0, 0x44, 0xB2, 0xF1, 0xFF, 0xC4, 0x00, 0x19, 0x01, 0x00, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x03, 0x01, 0x04, 0x05, 0xFF, 0xC4, 0x00, 0x23, 0x11, 0x00, 0x02, 0x02, 0x02, 0x01, 0x03, 0x05, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x02, 0x11, 0x03, 0x31, 0x21, 0x12, 0x22, 0x41, 0x04, 0x13, 0x32, 0x51, 0x61, 0xF0, 0xFF, 0xDA, /* JPEG image data begin marker */ 0x00, 0x0C, 0x03, 0x01, 0x00, 0x02, 0x11, 0x03, 0x11, 0x00, 0x3F, 0x00, /* Data header */ /* Test data */ 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0xFF, 0xD9 /* JPEG image data end marker*/ }; /* test_rtp_packet_data */ /* Define the ThreadX object control blocks... */ static TX_THREAD ntest_0; static TX_THREAD ntest_1; static NX_PACKET_POOL pool_0; static NX_IP ip_0; static NX_IP ip_1; static NX_UDP_SOCKET rtp_client_socket; static TX_SEMAPHORE semaphore_test_0_done; static TX_SEMAPHORE semaphore_test_1_done; /* Define rtp sender control block. */ static NX_RTP_SENDER rtp_0; static NX_RTP_SESSION rtp_session_0; static UINT rtp_port; static UINT rtcp_port; /* Define thread prototypes. */ static void ntest_0_entry(ULONG thread_input); static void ntest_1_entry(ULONG thread_input); extern void _nx_ram_network_driver(struct NX_IP_DRIVER_STRUCT *driver_req); extern VOID _nx_ram_network_driver_256(NX_IP_DRIVER *driver_req_ptr); extern void test_control_return(UINT status); #ifdef CTEST VOID test_application_define(void *first_unused_memory) #else void netx_rtp_session_jpeg_send_test_application_define(void *first_unused_memory) #endif { CHAR *pointer; UINT status; /* Print out test information banner. */ printf("NetX Test: RTP Session JPEG Send Test............................................"); /* Setup the working pointer. */ pointer = (CHAR *)first_unused_memory; /* Create the server thread. */ tx_thread_create(&ntest_0, "thread 0", ntest_0_entry, 0, pointer, DEMO_STACK_SIZE, 4, 4, TX_NO_TIME_SLICE, TX_AUTO_START); pointer = pointer + DEMO_STACK_SIZE; /* Create the client thread. */ tx_thread_create(&ntest_1, "thread 1", ntest_1_entry, 0, pointer, DEMO_STACK_SIZE, 3, 3, TX_NO_TIME_SLICE, TX_AUTO_START); pointer = pointer + DEMO_STACK_SIZE; /* Initialize the NetX system. */ nx_system_initialize(); /* Create a packet pool. */ status = nx_packet_pool_create(&pool_0, "NetX Main Packet Pool", PACKET_SIZE, pointer, PACKET_POOL_SIZE); pointer = pointer + PACKET_POOL_SIZE; CHECK_STATUS(0, status); /* Create server IP instance. */ status = nx_ip_create(&ip_0, "NetX IP Instance 0", RTP_SERVER_ADDRESS, 0xFFFFFF00UL, &pool_0, _nx_ram_network_driver_256, pointer, 2048, 1); pointer = pointer + 2048; CHECK_STATUS(0, status); /* Create client IP instance. */ status = nx_ip_create(&ip_1, "NetX IP Instance 1", RTP_CLIENT_ADDRESS, 0xFFFFFF00UL, &pool_0, _nx_ram_network_driver_256, pointer, 2048, 1); pointer = pointer + 2048; CHECK_STATUS(0, status); /* Enable ARP and supply ARP cache memory for IP Instance 0. */ status = nx_arp_enable(&ip_0, (void *) pointer, 1024); pointer = pointer + 1024; CHECK_STATUS(0, status); /* Enable ARP and supply ARP cache memory for IP Instance 1. */ status = nx_arp_enable(&ip_1, (void *) pointer, 1024); pointer = pointer + 1024; CHECK_STATUS(0, status); /* Enable UDP processing for both IP instances. */ status = nx_udp_enable(&ip_0); CHECK_STATUS(0, status); status = nx_udp_enable(&ip_1); CHECK_STATUS(0, status); /* Create semaphores for test done notification */ tx_semaphore_create(&semaphore_test_0_done, "semaphore test 0", 0); tx_semaphore_create(&semaphore_test_1_done, "semaphore test 1", 0); } static UINT validate_rtp_jpeg_data(UCHAR *data, UINT data_length) { UINT i; ULONG size_offset; UCHAR *data_ptr = data; /* The first byte is always 0 */ if (data_ptr[0] != 0x00) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Compute size_offset and compare with target offset */ size_offset = (ULONG)(data_ptr[0] << 16 | data_ptr[1] << 8 | data_ptr[2]); if (size_offset != 0) { return(NX_NOT_SUCCESSFUL); } data_ptr += 3; /* Check rtp/jpeg type, for YUV420 (0x21 in jpeg image), type shall be 0 */ if (data_ptr[0] != 0x00) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Check Q value */ if (data_ptr[0] != 0xFF) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Check jpeg image width and height */ if ((data_ptr[0] != (0x84 >> 3)) || (data_ptr[1] != (0x84 >> 3))) { return(NX_NOT_SUCCESSFUL); } data_ptr += 2; /* Check quantization table header */ if (data_ptr[0] != 0x00 || data_ptr[1] != 0x00 || data_ptr[2] != 0x00 || data_ptr[3] != 0x40) { return(NX_NOT_SUCCESSFUL); } data_ptr += 4; /* Check the quantization table */ for (i = 0; i < TEST_JPEG_Q_TABLE_SIZE; i++) { if (data_ptr[i] != test_rtp_packet_data[TEST_JPEG_Q_TABLE_START_POS + i]) { return(NX_NOT_SUCCESSFUL); } } data_ptr += TEST_JPEG_Q_TABLE_SIZE; /* Check JPEG image data */ i = 0; while (data_ptr < (data + data_length)) { if (*data_ptr != test_rtp_packet_data[TEST_JPEG_IMAGE_DATA_START_POS + i]) { return(NX_NOT_SUCCESSFUL); } i++; data_ptr++; } return(NX_SUCCESS); } static UINT validate_long_rtp_jpeg_data(UCHAR *data, UINT data_length) { UINT i; ULONG size_offset; UCHAR *data_ptr = data; static ULONG offset = 0; /* The first byte is always 0 */ if (data_ptr[0] != 0x00) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Compute size_offset and compare with target offset */ size_offset = (ULONG)(data_ptr[0] << 16 | data_ptr[1] << 8 | data_ptr[2]); if (size_offset != offset) { return(NX_NOT_SUCCESSFUL); } data_ptr += 3; /* Check rtp/jpeg type, for YUV422 (0x22 in jpeg image), type shall be 1 */ if (data_ptr[0] != 0x01) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Check Q value */ if (data_ptr[0] != 0xFF) { return(NX_NOT_SUCCESSFUL); } data_ptr++; /* Check jpeg image width and height */ if ((data_ptr[0] != (0x84 >> 3)) || (data_ptr[1] != (0x84 >> 3))) { return(NX_NOT_SUCCESSFUL); } data_ptr += 2; /* Only first fragmented packet contains quantization tables */ if (offset == 0) { /* Check quantization table header */ if (data_ptr[0] != 0x00 || data_ptr[1] != 0x00 || data_ptr[2] != 0x00 || data_ptr[3] != 0x80) { return(NX_NOT_SUCCESSFUL); } data_ptr += 4; /* Check both 2 quantization tables */ for (i = 0; i < TEST_JPEG_Q_TABLE_SIZE; i++) { if (data_ptr[i] != test_long_rtp_packet_data[TEST_LONG_JPEG_Q_TABLE_1_START_POS + i]) { return(NX_NOT_SUCCESSFUL); } if (data_ptr[TEST_JPEG_Q_TABLE_SIZE + i] != test_long_rtp_packet_data[TEST_LONG_JPEG_Q_TABLE_2_START_POS + i]) { return(NX_NOT_SUCCESSFUL); } } data_ptr += TEST_JPEG_Q_TABLE_SIZE * 2; } /* Check JPEG image data */ while (data_ptr < (data + data_length)) { if (*data_ptr != test_long_rtp_packet_data[TEST_LONG_JPEG_IMAGE_DATA_START_POS + offset]) { return(NX_NOT_SUCCESSFUL); } offset++; data_ptr++; } return(NX_SUCCESS); } /* Define server threads. */ static void ntest_0_entry(ULONG thread_input) { UINT status; NXD_ADDRESS client_ip_address; NX_PACKET *send_packet; UINT time_start; ULONG temp_socket_id; /* Create RTP sender. */ status = nx_rtp_sender_create(&rtp_0, &ip_0, &pool_0, CNAME, sizeof(CNAME) - 1); CHECK_STATUS(0, status); /* Get the udp port pair for rtp and rtcp */ status = nx_rtp_sender_port_get(&rtp_0, &rtp_port, &rtcp_port); CHECK_STATUS(0, status); /* Setup rtp sender session. */ client_ip_address.nxd_ip_version = NX_IP_VERSION_V4; client_ip_address.nxd_ip_address.v4 = RTP_CLIENT_ADDRESS; status = nx_rtp_sender_session_create(&rtp_0, &rtp_session_0, RTP_PAYLOAD_TYPE, 0, &client_ip_address, RTP_CLIENT_RTP_PORT, RTP_CLIENT_RTCP_PORT); CHECK_STATUS(0, status); /* If more than one rtp packet is sent during the first tick, rtcp packet will also be sent more than once. To make a stable test result, wait for a tick here to avoid this situation. */ tx_thread_sleep(1); /* ---- Test cycle 1 ---- */ /* Send jpeg data. */ status = nx_rtp_sender_session_jpeg_send(&rtp_session_0, test_rtp_packet_data, sizeof(test_rtp_packet_data), TEST_TIMESTAMP, TEST_MSW, TEST_LSW, 1); CHECK_STATUS(NX_SUCCESS, status); /* ---- Test cycle 2 ~ 4 ---- */ /* Send jpeg data. */ status = nx_rtp_sender_session_jpeg_send(&rtp_session_0, test_long_rtp_packet_data, sizeof(test_long_rtp_packet_data), TEST_TIMESTAMP, TEST_MSW, TEST_LSW, 1); CHECK_STATUS(NX_SUCCESS, status); /* Wait for the check in test thread 1 done. */ status = tx_semaphore_get(&semaphore_test_1_done, 5 * NX_IP_PERIODIC_RATE); CHECK_STATUS(0, status); /* Delete and release resources */ status = nx_rtp_sender_session_delete(&rtp_session_0); CHECK_STATUS(0, status); status = nx_rtp_sender_delete(&rtp_0); CHECK_STATUS(0, status); /* Put the semaphore to notify thread 1 it is fine to check resource leakage. */ tx_semaphore_put(&semaphore_test_0_done); } /* Define the client threads. */ static void ntest_1_entry(ULONG thread_input) { NX_PACKET *received_packet; UINT j; UINT status; UCHAR *data; UINT test_data_pos = 0; ULONG offset = 0; /* Create the rtp client socket. */ status = nx_udp_socket_create(&ip_1, &rtp_client_socket, "RTCP Client Socket", NX_IP_NORMAL, NX_FRAGMENT_OKAY, 0x80, 5); CHECK_STATUS(0, status); status = nx_udp_socket_bind(&rtp_client_socket, RTP_CLIENT_RTP_PORT, NX_IP_PERIODIC_RATE); CHECK_STATUS(0, status); for (UINT i = 0; i < TEST_CYCLES; i++) { /* Receive rtp data packet. */ status = nx_udp_socket_receive(&rtp_client_socket, &received_packet, 5 * TX_TIMER_TICKS_PER_SECOND); CHECK_STATUS(0, status); /* Validate RTP payload data */ data = received_packet -> nx_packet_prepend_ptr; /* Check RTP version byte */ CHECK_STATUS(0x80, *data); /* Move to check RTP data byte for payload type with marker */ data++; if (i == 1 || i == 2) { CHECK_STATUS((RTP_PAYLOAD_TYPE), *data); } else { CHECK_STATUS((NX_RTP_HEADER_MARKER_BIT | RTP_PAYLOAD_TYPE), *data); } /* Move to check RTP data bytes for sequence number */ data++; CHECK_STATUS((rtp_session_0.nx_rtp_session_sequence_number - 1), (data[0] << 8 | data[1])); /* Move to check RTP data bytes for time stamp */ data += 2; CHECK_STATUS(rtp_session_0.nx_rtp_session_rtp_timestamp, (ULONG)(data[0] << 24 | data[1] << 16 | data[2] << 8 | data[3])); /* Move to check RTP data bytes for ssrc */ data += 4; CHECK_STATUS(rtp_session_0.nx_rtp_session_ssrc, (ULONG)(data[0] << 24 | data[1] << 16 | data[2] << 8 | data[3])); /* Move to check RTP data bytes for data payload */ data += 4; if (i == 0) { status = validate_rtp_jpeg_data(data, received_packet -> nx_packet_length - 12); CHECK_STATUS(NX_SUCCESS, status); } else if (i >= 1 && i <= 3) { status = validate_long_rtp_jpeg_data(data, received_packet -> nx_packet_length - 12); if (status) CHECK_STATUS(NX_SUCCESS, status); } /* Release the receive packet when the check finishes. */ nx_packet_release(received_packet); } /* Set the flag to notify test thread 0 that the check finishes. */ tx_semaphore_put(&semaphore_test_1_done); /* Wait for the check in test thread 0 done. */ status = tx_semaphore_get(&semaphore_test_0_done, 5 * NX_IP_PERIODIC_RATE); CHECK_STATUS(0, status); /* Check if there is memory leak. */ CHECK_STATUS(pool_0.nx_packet_pool_total, pool_0.nx_packet_pool_available); /* Return the test result. */ printf("SUCCESS!\n"); test_control_return(0); } #else #ifdef CTEST VOID test_application_define(void *first_unused_memory) #else void netx_rtp_session_jpeg_send_test_application_define(void *first_unused_memory) #endif { /* Print out test information banner. */ printf("NetX Test: RTP Session JPEG Send Test............................................N/A\n"); test_control_return(3); } #endif