This chapter contains a description of various issues related to installation, setup, and usage of the NetX Duo TFTP component.
NetX Duo can be obtained from our public source code repository at https://github.com/eclipse-threadx/netxduo/. The package includes the following files:
nxd_tftp_client.h Header file for NetX Duo TFTP Client
nxd_tftp_client.c C Source file for NetX Duo TFTP Client
nxd_tftp_server.h Header file for NetX Duo TFTP Server
nxd_tftp_server.c C Source file for NetX Duo TFTP Server
filex_stub.h Stub file if FileX is not present
nxd_tftp.pdf PDF description of NetX Duo TFTP
demo_netxduo_tftp.c NetX Duo TFTP demonstration
To use NetX Duo TFTP, the entire distribution mentioned previously may be copied to the same directory where NetX Duo is installed. For example, if NetX Duo is installed in the directory “\threadx\arm7\green” then the nxd_tftp_client.h, nxd_tftp_client.c, nxd_tftp_server.h and nxd_tftp_server.c files could be copied into this directory.
To run a TFTP application, the application code must include nxd_tftp_client.h and/or nxd_tftp_server.h after it includes tx_api.h, fx_api.h, and nx_api.h, in order to use ThreadX, FileX, and NetX Duo, respectively. The application project must also include nxd_tftp_client.c and/or nxd_tftp_server.c in the build process. These files must be compiled in the same manner as other application files and its object form must be linked along with the files of the application. This is all that is required to use NetX Duo TFTP. Once the header file(s) is included, the application code is then able to use TFTP services.
Note: Since TFTP utilizes NetX Duo UDP services, UDP must be enabled with the nx_udp_enable call prior to using TFTP.
An example of how easy it is to use NetX Duo TFTP is described below. In this example, the TFTP include file nxd_tftp_client.h and nxd_tftp_server.h are brought in at line 19 and 20. Next, the TFTP Server is created in “tx_application_define” at line 179.
Note: The TFTP Server control block “server” was defined as a global variable at line 45 previously. This demo chooses to use IPv4 for its TFTP communication in line 14. After successful creation, the TFTP Server is started at line 304. At line 411 the TFTP Client is created. And finally, the Client writes the file at line 450 and reads the file back at line 485.
The TFTP server thread task is stopped and the TFTP Server is deleted on line 324. The application should call fx_media_close (line 331) to close all files and update file and directory data to the USB media.
Note: This example uses FileX for the TFTP Server handling of receiving and downloading TFTP Client file requests. However, if NX_TFTP_NO_FILEX is defined, the application can include file_stub.h instead of fx_api.h.
Also note that existing NetX TFTP client and server applications will work with NetX Duo TFTP. However, the application developer is encouraged to port their NetX TFTP applications to NetX Duo. The equivalent NetX Duo TFTP services are:
nxd_tftp_server_start
nxd_tftp_server_stop
nxd_tftp_client_file_read
nxd_tftp_client_file_write
nxd_tftp_client_file_open
/* This is a small demo of TFTP on the high-performance NetX TCP/IP stack. This demo
relies on ThreadX and NetX Duo, to show a simple file transfer from the client
and then back to the server. */
/* Indicate if using IPv6. */
#define USE_DUO
/* If the host application is using NetX Duo, determine which IP version to use.
Make sure IPv6 in NetX Duo is enabled if planning to use TFTP over IPv6 */
#ifdef USE_DUO
#define IP_TYPE 6
#endif /* USE_DUO */
#include "tx_api.h"
#include "nx_api.h"
#include "nxd_tftp_client.h"
#include "nxd_tftp_server.h"
#ifndef NX_TFTP_NO_FILEX
#include "fx_api.h"
#endif
#define DEMO_STACK_SIZE 4096
/* To use another file storage utility define this symbol:
#define NX_TFTP_NO_FILEX
*/
/* Define the ThreadX, NetX, and FileX object control blocks... */
TX_THREAD server_thread;
TX_THREAD client_thread;
NX_PACKET_POOL server_pool;
NX_IP server_ip;
NX_PACKET_POOL client_pool;
NX_IP client_ip;
FX_MEDIA ram_disk;
/* Define the NetX TFTP object control blocks. */
NX_TFTP_CLIENT client;
NX_TFTP_SERVER server;
/* Define the application global variables */
#define CLIENT_ADDRESS IP_ADDRESS(1, 2, 3, 5)
#define SERVER_ADDRESS IP_ADDRESS(1, 2, 3, 4)
NXD_ADDRESS server_ip_address;
NXD_ADDRESS client_ip_address;
UINT error_counter = 0;
/* Define buffer used in the demo application. */
UCHAR buffer[255];
ULONG data_length;
/* Define the memory area for the FileX RAM disk. */
#ifndef NX_TFTP_NO_FILEX
UCHAR ram_disk_memory[32000];
UCHAR ram_disk_sector_cache[512];
#endif
/* Define function prototypes. */
VOID _fx_ram_driver(FX_MEDIA *media_ptr);
VOID _nx_ram_network_driver(NX_IP_DRIVER *driver_req_ptr);
void client_thread_entry(ULONG thread_input);
void server_thread_entry(ULONG thread_input);
/* Define main entry point. */
int main()
{
/* Enter the ThreadX kernel. */
tx_kernel_enter();
}
/* Define what the initial system looks like. */
void tx_application_define(void *first_unused_memory)
{
UINT status;
UCHAR *pointer;
/* Setup the working pointer. */
pointer = (UCHAR *) first_unused_memory;
/* Create the main TFTP server thread. */
status = tx_thread_create(&server_thread, "TFTP Server Thread", server_thread_entry, 0,
pointer, DEMO_STACK_SIZE,
4,4, TX_NO_TIME_SLICE, TX_AUTO_START);
pointer += DEMO_STACK_SIZE ;
/* Check for errors. */
if (status)
error_counter++;
/* Create the main TFTP client thread at a slightly lower priority. */
status = tx_thread_create(&client_thread, "TFTP Client Thread", client_thread_entry, 0,
pointer, DEMO_STACK_SIZE,
5, 5, TX_NO_TIME_SLICE, TX_DONT_START);
pointer += DEMO_STACK_SIZE ;
/* Check for errors. */
if (status)
error_counter++;
/* Initialize the NetX system. */
nx_system_initialize();
/* Note: The data portion of a packet is exactly 512 bytes, but the packet payload size must
be at least 580 bytes. The remaining bytes are used for the UDP, IP, and Ethernet
headers and byte alignment requirements. */
status = nx_packet_pool_create(&server_pool, "TFTP Server Packet Pool", NX_TFTP_PACKET_SIZE, pointer, 8192);
pointer = pointer + 8192;
/* Check for errors. */
if (status)
error_counter++;
/* Create the IP instance for the TFTP Server. */
status = nx_ip_create(&server_ip, "NetX Server IP Instance", SERVER_ADDRESS, 0xFFFFFF00UL,
&server_pool, _nx_ram_network_driver, pointer, 2048, 1);
pointer = pointer + 2048;
/* Check for errors. */
if (status)
error_counter++;
/* Enable ARP and supply ARP cache memory for IP Instance 0. */
status = nx_arp_enable(&server_ip, (void *) pointer, 1024);
pointer = pointer + 1024;
/* Check for errors. */
if (status)
error_counter++;
/* Enable UDP. */
status = nx_udp_enable(&server_ip);
/* Check for errors. */
if (status)
error_counter++;
/* Create the TFTP server. */
#ifdef USE_DUO
#if (IP_TYPE == 6)
#ifdef FEATURE_NX_IPV6
/* Specify the tftp server global address. */
server_ip_address.nxd_ip_version = NX_IP_VERSION_V6;
server_ip_address.nxd_ip_address.v6[0] = 0x20010db1;
server_ip_address.nxd_ip_address.v6[1] = 0xf101;
server_ip_address.nxd_ip_address.v6[2] = 0;
server_ip_address.nxd_ip_address.v6[3] = 0x102;
#endif
#else
server_ip_address.nxd_ip_version = NX_IP_VERSION_V4;
server_ip_address.nxd_ip_address.v4 = SERVER_ADDRESS;
#endif
status = nxd_tftp_server_create(&server, "TFTP Server Instance", &server_ip, &ram_disk,
pointer, DEMO_STACK_SIZE, &server_pool);
#else
status = nx_tftp_server_create(&server, "TFTP Server Instance", &server_ip, &ram_disk,
pointer, DEMO_STACK_SIZE, &server_pool);
#endif
pointer = pointer + DEMO_STACK_SIZE;
/* Check for errors for the server. */
if (status)
error_counter++;
/* Create a packet pool for the TFTP client. */
/* Note: The data portion of a packet is exactly 512 bytes, but the packet payload size must
be at least 580 bytes. The remaining bytes are used for the UDP, IP, and Ethernet
headers and byte alignment requirements. */
status = nx_packet_pool_create(&client_pool, "TFTP Client Packet Pool", NX_TFTP_PACKET_SIZE, pointer, 8192);
pointer = pointer + 8192;
/* Create an IP instance for the TFTP client. */
status = nx_ip_create(&client_ip, "TFTP Client IP Instance", CLIENT_ADDRESS, 0xFFFFFF00UL,
&client_pool, _nx_ram_network_driver, pointer, 2048, 1);
pointer = pointer + 2048;
/* Enable ARP and supply ARP cache memory for IP Instance 1. */
status = nx_arp_enable(&client_ip, (void *) pointer, 1024);
pointer = pointer + 1024;
/* Enable UDP for client IP instance. */
status |= nx_udp_enable(&client_ip);
status |= nx_icmp_enable(&client_ip);
tx_thread_resume(&client_thread);
}
void server_thread_entry(ULONG thread_input)
{
UINT status, running;
#if (IP_TYPE == 6)
#ifdef FEATURE_NX_IPV6
UINT address_index;
UINT iface_index;
#endif
#endif
/* Allow time for the network driver and NetX to get initialized. */
tx_thread_sleep(100);
#ifndef NX_TFTP_NO_FILEX
/* Format the RAM disk - the memory for the RAM disk was defined above. */
status = fx_media_format(&ram_disk,
_fx_ram_driver, /* Driver entry */
ram_disk_memory, /* RAM disk memory pointer */
ram_disk_sector_cache, /* Media buffer pointer */
sizeof(ram_disk_sector_cache), /* Media buffer size */
"MY_RAM_DISK", /* Volume Name */
1, /* Number of FATs */
32, /* Directory Entries */
0, /* Hidden sectors */
256, /* Total sectors */
128, /* Sector size */
1, /* Sectors per cluster */
1, /* Heads */
1); /* Sectors per track */
/* Check for errors. */
if (status != FX_SUCCESS)
{
return;
}
/* Open the RAM disk. */
status = fx_media_open(&ram_disk, "RAM DISK", _fx_ram_driver, ram_disk_memory, ram_disk_sector_cache,
sizeof(ram_disk_sector_cache));
/* Check for errors. */
if (status != FX_SUCCESS)
{
return;
}
#endif /* NX_TFTP_NO_FILEX */
#if (IP_TYPE == 6)
#ifdef FEATURE_NX_IPV6
/* Enable ICMPv6 services. */
status |= nxd_icmp_enable(&server_ip);
if (status != NX_SUCCESS)
{
return;
}
/* Enable IPv6 services for the server. */
status = nxd_ipv6_enable(&server_ip);
if (status != NX_SUCCESS)
{
return;
}
/* This assumes the primary interface. See the NetX Duo
User Guide for more information on address configuration. */
iface_index = 0;
status = nxd_ipv6_address_set(&server_ip, iface_index, NX_NULL, 10, &address_index);
status += nxd_ipv6_address_set(&server_ip, iface_index, &server_ip_address, 64, &address_index);
if (status != NX_SUCCESS)
{
return;
}
/* Wait for DAD to validate the address. */
tx_thread_sleep(500);
#endif
#endif /* IP_TYPE == 6 */
/* Start the NetX TFTP server. */
#ifdef USE_DUO
status = nxd_tftp_server_start(&server);
#else
status = nx_tftp_server_start(&server);
#endif
/* Check for errors. */
if (status)
{
error_counter++;
return;
}
/* Run for a while */
running = NX_TRUE;
while(running)
tx_thread_sleep(200);
/* Stop and delete the TFTP server. */
#ifdef USE_DUO
nxd_tftp_server_delete(&server);
#else
nx_tftp_server_delete(&server);
#endif
/* Close all open files and ensure directory information is also written out to the media.
This will also flush the file data to USB media*/
status = fx_media_close(&ram_disk);
if (status)
{
error_counter++;
}
return;
}
/* Define the TFTP client thread. */
void client_thread_entry(ULONG thread_input)
{
NX_PACKET *my_packet;
UINT status;
UINT all_done = NX_FALSE;
#if (IP_TYPE == 6)
#ifdef FEATURE_NX_IPV6
UINT address_index;
UINT iface_index;
#endif
#endif
/* Allow time for the network driver and NetX to get initialized. */
tx_thread_sleep(100);
#if (IP_TYPE == 6)
#ifdef FEATURE_NX_IPV6
/* Enable ECMPv6 services for the client. */
status = nxd_icmp_enable(&client_ip);
if (status != NX_SUCCESS)
{
return;
}
/* Enable IPv6 services for the client. */
status = nxd_ipv6_enable(&client_ip);
if (status != NX_SUCCESS)
{
return;
}
/* Set the Client IPv6 address */
client_ip_address.nxd_ip_version = NX_IP_VERSION_V6;
client_ip_address.nxd_ip_address.v6[0] = 0x20010db1;
client_ip_address.nxd_ip_address.v6[1] = 0xf101;
client_ip_address.nxd_ip_address.v6[2] = 0;
client_ip_address.nxd_ip_address.v6[3] = 0x101;
/* This assumes the primary interface. See the NetX Duo
User Guide for more information on address configuration. */
iface_index = 0;
status = nxd_ipv6_address_set(&client_ip, iface_index, NX_NULL, 10, &address_index);
status += nxd_ipv6_address_set(&client_ip, iface_index, &client_ip_address, 64, &address_index);
if (status != NX_SUCCESS)
{
return;
}
/* Wait for the link local and global addresses to be validated. */
tx_thread_sleep(500);
#endif
#endif /*(IP_TYPE == 6) */
/* The TFTP services used below include the NetX equivalent service which will work with
NetX Duo TFTP. However, it is recommended for developers to port their applications
to the newer services that take the NXD_ADDRESS type and support both IPv4 and IPv6
communication.
*/
/* Create a TFTP client. */
#ifdef USE_DUO
status = nxd_tftp_client_create(&client, "TFTP Client", &client_ip, &client_pool, IP_TYPE);
#else
status = nx_tftp_client_create(&client, "TFTP Client", &client_ip, &client_pool);
#endif
/* Check status. */
if (status)
return;
/* Open a TFTP file for writing. */
#ifdef USE_DUO
status = nxd_tftp_client_file_open(&client, "test.txt", &server_ip_address, NX_TFTP_OPEN_FOR_WRITE, 100, IP_TYPE);
#else
status = nx_tftp_client_file_open(&client, "test.txt", SERVER_ADDRESS, NX_TFTP_OPEN_FOR_WRITE, 100);
#endif
/* Check status. */
if (status)
return;
/* Allocate a TFTP packet. */
#ifdef USE_DUO
status = nxd_tftp_client_packet_allocate(&client_pool, &my_packet, 100, IP_TYPE);
#else
status = nx_tftp_client_packet_allocate(&client_pool, &my_packet, 100);
#endif
/* Check status. */
if (status)
error_counter++;
/* Write ABCs into the packet payload! */
memcpy(my_packet -> nx_packet_prepend_ptr, "ABCDEFGHIJKLMNOPQRSTUVWXYZ ", 28);
/* Adjust the write pointer. */
my_packet -> nx_packet_length = 28;
my_packet -> nx_packet_append_ptr = my_packet -> nx_packet_prepend_ptr + 28;
/* Write this packet to the file via TFTP. */
#ifdef USE_DUO
status = nxd_tftp_client_file_write(&client, my_packet, 100, IP_TYPE);
#else
status = nx_tftp_client_file_write(&client, my_packet, 100);
#endif
/* Check status. */
if (status)
error_counter++;
/* Close this file. */
#ifdef USE_DUO
status = nxd_tftp_client_file_close(&client, IP_TYPE);
#else
status = nx_tftp_client_file_close(&client);
#endif
/* Check status. */
if (status)
error_counter++;
/* Open the same file for reading. */
#ifdef USE_DUO
status = nxd_tftp_client_file_open(&client, "test.txt", &server_ip_address, NX_TFTP_OPEN_FOR_READ, 100, IP_TYPE);
#else
status = nx_tftp_client_file_open(&client, "test.txt", SERVER_ADDRESS, NX_TFTP_OPEN_FOR_READ, 100);
#endif
/* Check status. */
if (status)
error_counter++;
do
{
/* Read the file back. */
#ifdef USE_DUO
status = nxd_tftp_client_file_read(&client, &my_packet, 100, IP_TYPE);
#else
status = nx_tftp_client_file_read(&client, &my_packet, 100);
#endif
/* Check for retransmission/dropped packet error. Benign. Try again... */
if (status == NX_TFTP_INVALID_BLOCK_NUMBER)
{
continue;
}
else if (status == NX_TFTP_END_OF_FILE)
{
/* All done. */
all_done = NX_TRUE;
}
else if (status != NX_SUCCESS)
{
/* Internal error, invalid packet or error on read. */
break;
}
/* Do something with the packet data and release when done. */
nx_packet_data_retrieve(my_packet, buffer, &data_length);
buffer[data_length] = 0;
printf("Receive data: %s\n", buffer);
printf("release packet in demo.\n");
nx_packet_release(my_packet);
} while (all_done == NX_FALSE);
/* Close the file again. */
#ifdef USE_DUO
status = nxd_tftp_client_file_close(&client, IP_TYPE);
#else
status = nx_tftp_client_file_close(&client);
#endif
/* Check status. */
if (status)
error_counter++;
/* Delete the client. */
#ifdef USE_DUO
status = nxd_tftp_client_delete(&client);
#else
status = nx_tftp_client_delete(&client);
#endif
/* Check status. */
if (status)
error_counter++;
return;
}
There are several configuration options for building NetX Duo TFTP. The following list describes each in detail. Unless otherwise specified, these options are found in nxd_tftp_client.h and nxd_tftp_server.h.
NX_DISABLE_ERROR_CHECKING Defined, this option removes the basic TFTP error checking. It is typically used after the application has been debugged.
NX_TFTP_SERVER_PRIORITY The priority of the TFTP server thread. By default, this value is defined as 16 to specify priority 16.
NX_TFTP_SERVER_TIME_SLICE The time slice for the TFTP Server to run before yielding to other threads of the same priority. The default value is 2.
NX_TFTP_MAX_CLIENTS The maximum number of clients the server can handle at one time. By default, this value is 10 to support 10 clients at once.
NX_TFTP_ERROR_STRING_MAX The maximum number of characters in the error string. By default, this value is 64.
NX_TFTP_NO_FILEX Defined, this option provides a stub for FileX dependencies. The TFTP Client will function without any change if this option is defined. The TFTP Server will need to either be modified or the user will have to create a handful of FileX services in order to function properly.
NX_TFTP_TYPE_OF_SERVICE Type of service required for the TFTP UDP requests. By default, this value is defined as NX_IP_NORMAL to indicate normal IP packet service.
NX_TFTP_FRAGMENT_OPTION Fragment enable for TFTP UDP requests. By default, this value is NX_DONT_FRAGMENT to disable TFTP UDP fragmenting.
NX_TFTP_TIME_TO_LIVE Specifies the number of routers this packet can pass before it is discarded. The default value is set to 0x80.
NX_TFTP_SOURCE_PORT This option allows a TFTP Client application to specify the TFTP Client UDP socket port. It is defaulted to NX_ANY_PORT.
NX_TFTP_SERVER_RETRANSMIT_ENABLE Enables the TFTP server’s timer to check each TFTP client session with for recent activity (either an ACK or data packet). When the session timeout expires after the maximum number of times, it is assumed the connection was lost. The Server clears the Client request, closes any open files and makes the connection request available for the next Client. The default setting is disabled.
NX_TFTP_SERVER_TIMEOUT_PERIOD Specifies the interval when the TFTP server timer entry function checks Client connections for receiving any packets. The default value is 20 (timer ticks).
NX_TFTP_SERVER_RETRANSMIT_TIMEOUT This is the timeout for receiving a valid ACK or data packet from the Client. The default value is 200 (timer ticks).
NX_TFTP_SERVER_MAX_RETRIES Specifies the maximum number of times the Client session retransmit timeout is renewed. Thereafter, the session is closed by the Server.
NX_TFTP_MAX_CLIENT_RETRANSMITS Specifies the maximum number of times the Server receives a duplicate ACK or data packet from the Client (which it drops) without sending an error message to the Client and closing the session. Has no effect if NX_TFTP_SERVER_RETRANSMIT_ENABLE is defined.