diff --git a/doc/multiple_ports.rst b/doc/multiple_ports.rst index ed3a4fe8..8612e942 100644 --- a/doc/multiple_ports.rst +++ b/doc/multiple_ports.rst @@ -2,6 +2,7 @@ Multiple ports ============== This section describes how to configure the p-net stack, sample application and system for multiple network interfaces or ports. +So far, multiple port has only been tested using the linux port. Terminology @@ -92,12 +93,40 @@ Run ``ifconfig`` to check that the bridge is up and its network interfaces are a TX packets 0 bytes 0 (0.0 B) TX errors 0 dropped 0 overruns 0 carrier 0 collisions 0 -Configuration of p-net stack and sample application (TBD) +Configuration of p-net stack and sample application --------------------------------------------------------- To run p-net and the sample application with multiple ports a couple of things need to be done: -* Reconfigure setting ``PNET_MAX_PORT`` to the actual number of physical ports available in the system. - For this example ``PNET_MAX_PORT`` shall be set to 2. - -* TBD - update this document when multiport support is implemented +Reconfigure setting ``PNET_MAX_PORT`` to the actual number of physical ports available in the system. + For this example ``PNET_MAX_PORT`` shall be set to 2. + +Example of initial log when starting the demo application with a multi port configuration:: + + pi@pndevice-pi:~/profinet/build $ sudo ./pn_dev -v + ** Starting Profinet demo application ** + Number of slots: 5 (incl slot for DAP module) + P-net log level: 0 (DEBUG=0, FATAL=4) + App verbosity level: 1 + Nbr of ports: 2 + Network interfaces: br0,eth0,eth1 + Button1 file: + Button2 file: + Station name: rt-labs-dev + Management port: br0 C2:38:F3:A6:0A:66 + Physical port [1]: eth0 B8:27:EB:67:14:8A + Physical port [2]: eth1 58:EF:68:B5:11:0F + Current hostname: pndevice-pi + Current IP address: 192.168.0.50 + Current Netmask: 255.255.255.0 + Current Gateway: 192.168.0.1 + Storage directory: /home/pi/profinet/build + +Update gsdml file +----------------- +The sample app gsdml file contains a commented out block that defines +a second physical port. In the sample application gsdml file, search for "IDS_P2" +and enable commented out lines as described in the gsdml file. + +Note that you will have to the reload gsdml file in all tools you are using and +also the Automated RT tester any time the file is changed. \ No newline at end of file diff --git a/sample_app/GSDML-V2.4-rtlabs-IODevice-20201211.xml b/sample_app/GSDML-V2.4-rtlabs-IODevice-20201211.xml index f970a544..7eba1dda 100644 --- a/sample_app/GSDML-V2.4-rtlabs-IODevice-20201211.xml +++ b/sample_app/GSDML-V2.4-rtlabs-IODevice-20201211.xml @@ -59,6 +59,15 @@ + + + @@ -195,6 +204,7 @@ + diff --git a/sample_app/sampleapp_common.c b/sample_app/sampleapp_common.c index 257698b2..a00f5643 100644 --- a/sample_app/sampleapp_common.c +++ b/sample_app/sampleapp_common.c @@ -60,13 +60,7 @@ static void ip_to_string (pnal_ipaddr_t ip, char * outputstring) (uint8_t) (ip & 0xFF)); } -/** - * Convert MAC address to string - * @param mac In: MAC address - * @param outputstring Out: Resulting string buffer. Should have size - * PNAL_ETH_ADDRSTR_SIZE. - */ -static void mac_to_string (pnal_ethaddr_t mac, char * outputstring) +void app_mac_to_string (pnal_ethaddr_t mac, char * outputstring) { snprintf ( outputstring, @@ -189,7 +183,6 @@ static const char * ioxs_to_string (pnet_ioxs_values_t ioxs) } void app_print_network_details ( - pnal_ethaddr_t * p_macbuffer, pnal_ipaddr_t ip, pnal_ipaddr_t netmask, pnal_ipaddr_t gateway) @@ -197,16 +190,13 @@ void app_print_network_details ( char ip_string[PNAL_INET_ADDRSTR_SIZE]; /* Terminated string */ char netmask_string[PNAL_INET_ADDRSTR_SIZE]; /* Terminated string */ char gateway_string[PNAL_INET_ADDRSTR_SIZE]; /* Terminated string */ - char mac_string[PNAL_ETH_ADDRSTR_SIZE]; /* Terminated string */ char hostname_string[PNAL_HOSTNAME_MAX_SIZE]; /* Terminated string */ - mac_to_string (*p_macbuffer, mac_string); ip_to_string (ip, ip_string); ip_to_string (netmask, netmask_string); ip_to_string (gateway, gateway_string); pnal_get_hostname (hostname_string); - printf ("MAC address: %s\n", mac_string); printf ("Current hostname: %s\n", hostname_string); printf ("Current IP address: %s\n", ip_string); printf ("Current Netmask: %s\n", netmask_string); @@ -1281,8 +1271,9 @@ void app_plug_dap (pnet_t * net, void * arg) /************ Configuration of product ID, software version etc **************/ -int app_adjust_stack_configuration (pnet_cfg_t * stack_config) +int app_pnet_cfg_init_default (pnet_cfg_t * stack_config) { + uint16_t i; memset (stack_config, 0, sizeof (pnet_cfg_t)); stack_config->tick_us = APP_TICK_INTERVAL_US; @@ -1357,9 +1348,16 @@ int app_adjust_stack_configuration (pnet_cfg_t * stack_config) stack_config->min_device_interval = 32; /* Corresponds to 1 ms */ /* LLDP settings */ - strcpy (stack_config->if_cfg.ports[0].port_id, "port-001"); - stack_config->if_cfg.ports[0].rtclass_2_status = 0; - stack_config->if_cfg.ports[0].rtclass_3_status = 0; + for (i = 0; i < PNET_MAX_PORT; i++) + { + snprintf ( + stack_config->if_cfg.ports[i].port_id, + PNET_LLDP_PORT_ID_MAX_SIZE, + "port-%03d", + i + 1); + stack_config->if_cfg.ports[i].rtclass_2_status = 0; + stack_config->if_cfg.ports[i].rtclass_3_status = 0; + } /* Network configuration */ stack_config->send_hello = true; @@ -1373,6 +1371,190 @@ int app_adjust_stack_configuration (pnet_cfg_t * stack_config) return 0; } +/** + * Initialize a network interface configuration from a network + * interface name. + * This includes reading the Ethernet MAC address. + * If the network interface can not be found or the operation + * fails to read the mac address from the network interface, + * error is returned. + * @param if_name In: Network interface name + * @param p_netif Out: Network interface configuration + * @param verbosity In: Verbosity + * @return 0 on success + * -1 on error + */ +int app_port_cfg_init ( + const char * netif_name, + pnet_netif_t * p_netif, + int verbosity) +{ + pnal_ethaddr_t mac; + char mac_string[PNAL_ETH_ADDRSTR_SIZE]; /* Terminated string */ + + int ret = pnal_get_macaddress (netif_name, &mac); + if (ret != 0) + { + printf ( + "Error: The given network interface does not exist: \"%s\"\n", + netif_name); + return -1; + } + else + { + if (verbosity > 0) + { + app_mac_to_string (mac, mac_string); + printf ("%-10s %s\n", netif_name, mac_string); + } + } + + strncpy (p_netif->if_name, netif_name, PNET_INTERFACE_NAME_MAX_SIZE); + memcpy (p_netif->eth_addr.addr, mac.addr, sizeof (pnal_ethaddr_t)); + + return 0; +} + +int app_get_netif_namelist ( + const char * arg_str, + app_netif_namelist_t * p_if_list, + int max_port) +{ + int ret = 0; + uint16_t i = 0; + uint16_t j = 0; + uint16_t if_index = 0; + uint16_t if_list_size = max_port + 1; /* NELEMENTS (p_if_list->netif) dynamic + for test. */ + char c; + + memset (p_if_list, 0, sizeof (*p_if_list)); + c = arg_str[i++]; + while (c != '\0' && if_index < if_list_size) + { + if (c != ',') + { + p_if_list->netif[if_index].name[j++] = c; + } + else + { + p_if_list->netif[if_index].name[j++] = '\0'; + j = 0; + if_index++; + } + c = arg_str[i++]; + } + + if (max_port == 1) + { + if (if_index >= max_port) + { + printf ("Error: Only 1 network interface expected.\n"); + ret = -1; + } + + if (strlen (p_if_list->netif[0].name) == 0) + { + printf ("Error: Zero length network interface name.\n"); + ret = -1; + } + p_if_list->netif[1] = p_if_list->netif[0]; + } + + if (max_port > 1) + { + if (if_index != max_port) + { + printf ("Error: %u network interfaces expected.\n", max_port + 1); + ret = -1; + } + + for (i = 0; i <= max_port; i++) + { + if (strlen (p_if_list->netif[i].name) == 0) + { + printf ("Error: Zero length network interface name (%d).\n", i); + ret = -1; + } + } + } + + return ret; +} + +int app_pnet_cfg_init_netifs ( + const char * netif_list_str, + pnet_cfg_t * p_cfg, + int verbosity) +{ + int ret = 0; + int i = 0; + pnal_ipaddr_t ip; + pnal_ipaddr_t netmask; + pnal_ipaddr_t gateway; + app_netif_namelist_t if_list; + + ret = app_get_netif_namelist (netif_list_str, &if_list, PNET_MAX_PORT); + if (ret != 0) + { + return ret; + } + + if (verbosity > 0) + { + printf ("Management port: "); + } + if ( + app_port_cfg_init ( + if_list.netif[0].name, + &p_cfg->if_cfg.main_port, + verbosity) != 0) + { + return -1; + } + + for (i = 1; i <= PNET_MAX_PORT; i++) + { + if (verbosity > 0) + { + printf ("Physical port [%u]: ", i); + } + + if ( + app_port_cfg_init ( + if_list.netif[i].name, + &p_cfg->if_cfg.ports[i - 1].phy_port, + verbosity) != 0) + { + return -1; + } + } + + /* Read IP, netmask, gateway from operating system */ + ip = pnal_get_ip_address (if_list.netif[0].name); + netmask = pnal_get_netmask (if_list.netif[0].name); + gateway = pnal_get_gateway (if_list.netif[0].name); + + if (gateway == IP_INVALID) + { + printf ( + "Error: Invalid gateway IP address for Ethernet interface: %s\n", + if_list.netif[0].name); + return -1; + } + + if (verbosity > 0) + { + app_print_network_details (ip, netmask, gateway); + } + + app_copy_ip_to_struct (&p_cfg->if_cfg.ip_cfg.ip_addr, ip); + app_copy_ip_to_struct (&p_cfg->if_cfg.ip_cfg.ip_gateway, gateway); + app_copy_ip_to_struct (&p_cfg->if_cfg.ip_cfg.ip_mask, netmask); + + return ret; +} + /*************************** Helper functions ********************************/ void app_copy_ip_to_struct ( @@ -1566,7 +1748,8 @@ static void app_handle_cyclic_data ( if (outputdata_length != APP_DATASIZE_OUTPUT) { printf ("Wrong outputdata length: %u\n", outputdata_length); - app_set_outputs_default_value (p_appdata->arguments.verbosity); + app_set_outputs_default_value ( + p_appdata->arguments.verbosity); } else if (outputdata_iops == PNET_IOXS_GOOD) { @@ -1583,7 +1766,8 @@ static void app_handle_cyclic_data ( { printf ("Wrong IOPS: %u\n", outputdata_iops); } - app_set_outputs_default_value (p_appdata->arguments.verbosity); + app_set_outputs_default_value ( + p_appdata->arguments.verbosity); } } } diff --git a/sample_app/sampleapp_common.h b/sample_app/sampleapp_common.h index d061fbbe..0160fcbd 100644 --- a/sample_app/sampleapp_common.h +++ b/sample_app/sampleapp_common.h @@ -71,7 +71,7 @@ extern "C" { #define APP_LOGBOOK_ERROR_CODE 0x20 /* Manufacturer specific */ #define APP_LOGBOOK_ERROR_DECODE 0x82 /* Manufacturer specific */ #define APP_LOGBOOK_ERROR_CODE_1 PNET_ERROR_CODE_1_FSPM -#define APP_LOGBOOK_ERROR_CODE_2 0x00 /* Manufacturer specific */ +#define APP_LOGBOOK_ERROR_CODE_2 0x00 /* Manufacturer specific */ #define APP_LOGBOOK_ENTRY_DETAIL 0xFEE1DEAD /* Manufacturer specific */ /* @@ -162,14 +162,26 @@ struct cmd_args char path_button1[PNET_MAX_FILE_FULLPATH_SIZE]; /** Terminated string */ char path_button2[PNET_MAX_FILE_FULLPATH_SIZE]; /** Terminated string */ char path_storage_directory[PNET_MAX_DIRECTORYPATH_SIZE]; /** Terminated */ - char station_name[PNET_STATION_NAME_MAX_SIZE]; /** Terminated string */ - char eth_interface[PNET_INTERFACE_NAME_MAX_SIZE]; /** Terminated string */ + char station_name[PNET_STATION_NAME_MAX_SIZE]; /** Terminated string */ + char eth_interfaces + [PNET_INTERFACE_NAME_MAX_SIZE * (PNET_MAX_PORT + 1) + + PNET_MAX_PORT]; /** Terminated string */ int verbosity; int show; bool factory_reset; bool remove_files; }; +typedef struct app_netif_name +{ + char name[PNET_INTERFACE_NAME_MAX_SIZE]; +} app_netif_name_t; + +typedef struct app_netif_namelist +{ + app_netif_name_t netif[PNET_MAX_PORT + 1]; +} app_netif_namelist_t; + typedef struct app_subslot { bool used; @@ -233,9 +245,16 @@ typedef enum app_demo_state /********************* Helper function declarations ***************************/ /** - * Print out current IP address, MAC address etc. + * Convert MAC address to string + * @param mac In: MAC address + * @param outputstring Out: Resulting string buffer. Should have size + * PNAL_ETH_ADDRSTR_SIZE. + */ +void app_mac_to_string (pnal_ethaddr_t mac, char * outputstring); + +/** + * Print out current IP address, netmask and default gateway. * - * @param p_macbuffer In: MAC address * @param ip In: IP address * @param netmask In: Netmask * @param gateway In: Gateway @@ -243,7 +262,6 @@ typedef enum app_demo_state * -1 if an error occurred. */ void app_print_network_details ( - pnal_ethaddr_t * p_macbuffer, pnal_ipaddr_t ip, pnal_ipaddr_t netmask, pnal_ipaddr_t gateway); @@ -255,7 +273,44 @@ void app_print_network_details ( * @return 0 if the operation succeeded. * -1 if an error occurred. */ -int app_adjust_stack_configuration (pnet_cfg_t * stack_config); +int app_pnet_cfg_init_default (pnet_cfg_t * stack_config); + +/** + * Initialize interface configuration from argument string. + * + * @param netif_list_str In: Comma separated list of network interfaces. + * Terminated string. + * @param p_cfg InOut: p-net configuration + * @param verbosity In: Verbosity + * @return 0 on success + * -1 on error + */ +int app_pnet_cfg_init_netifs ( + const char * netif_list_str, + pnet_cfg_t * p_cfg, + int verbosity); + +/** + * Parse a comma separated list of network interfaces and check + * that the number of interfaces match the PNET_MAX_PORT configuration. + * Examples: + * PNET_MAX_PORT arg_str status + * 1 "eth0" ok + * 1 "eth0,eth1" err + * 2 "br0,eth0,eth1" ok + * 2 "br0,eth0," err + * + * @param arg_str In: Network interface list as comma separated, terminated + * string. For example "eth0" or "br0,eth0,eth1 + * @param p_if_list Out: List of network interfaces + * @param max_port In: PNET_MAX_PORT, passed as argument to allow test + * @return 0 if network interface list matches configuration + * -1 on error + */ +int app_get_netif_namelist ( + const char * arg_str, + app_netif_namelist_t * p_if_list, + int max_port); /** * Plug DAP (sub-)modules. This operation shall be called after p-net diff --git a/src/common/pf_eth.c b/src/common/pf_eth.c index 426c2ed3..58df20eb 100644 --- a/src/common/pf_eth.c +++ b/src/common/pf_eth.c @@ -41,6 +41,70 @@ int pf_eth_init (pnet_t * net) return ret; } +int pf_eth_init_network_interfaces (pnet_t * net, const pnet_cfg_t * p_cfg) +{ + int port; + pf_port_iterator_t port_iterator; + pf_port_t * p_port_data = NULL; + const pnet_port_cfg_t * p_port_cfg; + pnal_ethertype_t main_port_receive_type = PNAL_ETHTYPE_ALL; + +#if (PNET_MAX_PORT > 1) + main_port_receive_type = PNAL_ETHTYPE_PROFINET; +#endif + + /* Init management port */ + net->eth_handle = pnal_eth_init ( + p_cfg->if_cfg.main_port.if_name, + main_port_receive_type, + pf_eth_recv, + (void *)net); + if (net->eth_handle == NULL) + { + LOG_ERROR ( + PNET_LOG, + "Failed to init \"%s\"\n", + p_cfg->if_cfg.main_port.if_name); + return -1; + } + + /* Init physical ports */ + pf_port_init_iterator_over_ports (net, &port_iterator); + port = pf_port_get_next (&port_iterator); + while (port != 0) + { + p_port_data = pf_port_get_state (net, port); + p_port_cfg = pf_port_get_config (net, port); + +#if PNET_MAX_PORT > 1 + p_port_data->eth_handle = pnal_eth_init ( + p_port_cfg->phy_port.if_name, + PNAL_ETHTYPE_LLDP, + pf_eth_recv, + (void *)net); +#else + /* In single port configuration managed port is also physical port 1 */ + p_port_data->eth_handle = net->eth_handle; +#endif + + if (p_port_data->eth_handle != NULL) + { + pnal_eth_set_tag (p_port_data->eth_handle, port); + } + else + { + LOG_ERROR ( + PNET_LOG, + "Failed to init \"%s\"\n", + p_port_cfg->phy_port.if_name); + return -1; + } + + port = pf_port_get_next (&port_iterator); + } + return 0; +} + int pf_eth_send (pnet_t * net, pnal_eth_handle_t * handle, pnal_buf_t * buf) { int sent_len = 0; @@ -56,7 +120,7 @@ int pf_eth_send (pnet_t * net, pnal_eth_handle_t * handle, pnal_buf_t * buf) return sent_len; } -int pf_eth_recv (void * arg, pnal_buf_t * p_buf) +int pf_eth_recv (pnal_eth_handle_t * eth_handle, void * arg, pnal_buf_t * p_buf) { int ret = 0; /* Means: "Not handled" */ uint16_t eth_type_pos = 2 * sizeof (pnet_ethaddr_t); @@ -106,7 +170,7 @@ int pf_eth_recv (void * arg, pnal_buf_t * p_buf) } break; case PNAL_ETHTYPE_LLDP: - ret = pf_lldp_recv (net, p_buf, frame_pos); + ret = pf_lldp_recv (net, eth_handle, p_buf, frame_pos); break; case PNAL_ETHTYPE_IP: /* IP-packets (UDP) are also received via the UDP sockets. Do not count diff --git a/src/common/pf_eth.h b/src/common/pf_eth.h index fce1123b..122a67b1 100644 --- a/src/common/pf_eth.h +++ b/src/common/pf_eth.h @@ -28,6 +28,19 @@ extern "C" { */ int pf_eth_init (pnet_t * net); +/** + * Initialize network interfaces according to configuration + * If device is configured with one network interface (PNET_MAX_PORT==1), + * the management port and physical port 1 refers to same network interface. + * In a multi port configuration, the management port and physical ports are + * different network interfaces. + * @param net InOut: The p-net stack instance + * @param p_cfg In: Configuration + * @return 0 on success + * -1 on error + */ +int pf_eth_init_network_interfaces (pnet_t * net, const pnet_cfg_t * p_cfg); + /** * Send raw Ethernet data. * @@ -74,12 +87,13 @@ void pf_eth_frame_id_map_remove (pnet_t * net, uint16_t frame_id); * Note that this itself is a callback, and its arguments should fulfill the * prototype os_raw_frame_handler_t * + * @param eth_handle In: Network interface handle * @param arg InOut: User argument, will be converted to pnet_t * @param p_buf In: The Ethernet frame. Might be freed. * @return 0 If the frame was NOT handled by this function. * 1 If the frame was handled and the buffer freed. */ -int pf_eth_recv (void * arg, pnal_buf_t * p_buf); +int pf_eth_recv (pnal_eth_handle_t * eth_handle, void * arg, pnal_buf_t * p_buf); #ifdef __cplusplus } diff --git a/src/common/pf_lldp.c b/src/common/pf_lldp.c index 4cdcd3d3..106ffea2 100644 --- a/src/common/pf_lldp.c +++ b/src/common/pf_lldp.c @@ -835,10 +835,11 @@ size_t pf_lldp_construct_frame (pnet_t * net, int loc_port_num, uint8_t buf[]) } LOG_DEBUG ( PF_LLDP_LOG, - "LLDP(%d): Sending LLDP frame. MAC " + "LLDP(%d): Send port %u MAC: " "%02X:%02X:%02X:%02X:%02X:%02X " - "IP: %s Chassis ID: \"%s\" Port number: %u Port ID: \"%s\"\n", + "IP: %s Chassis ID: \"%s\" Port ID: \"%s\"\n", __LINE__, + loc_port_num, device_mac_address->addr[0], device_mac_address->addr[1], device_mac_address->addr[2], @@ -847,7 +848,6 @@ size_t pf_lldp_construct_frame (pnet_t * net, int loc_port_num, uint8_t buf[]) device_mac_address->addr[5], ip_string, chassis_id_description, - loc_port_num, p_port_cfg->port_id); #endif @@ -887,6 +887,7 @@ static void pf_lldp_send (pnet_t * net, int loc_port_num) { /* FIXME: Buffer size should include Ethernet header (14 bytes) */ pnal_buf_t * p_buffer = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); + pf_port_t * p_port_data = NULL; if (p_buffer != NULL) { @@ -895,7 +896,9 @@ static void pf_lldp_send (pnet_t * net, int loc_port_num) p_buffer->len = pf_lldp_construct_frame (net, loc_port_num, p_buffer->payload); - (void)pf_eth_send (net, net->eth_handle, p_buffer); + p_port_data = pf_port_get_state (net, loc_port_num); + + (void)pf_eth_send (net, p_port_data->eth_handle, p_buffer); } pnal_buf_free (p_buffer); @@ -1709,37 +1712,54 @@ void pf_lldp_update_peer ( pf_lldp_store_peer_info (net, loc_port_num, lldp_peer_info); } -int pf_lldp_recv (pnet_t * net, pnal_buf_t * p_frame_buf, uint16_t offset) +int pf_lldp_recv ( + pnet_t * net, + pnal_eth_handle_t * eth_handle, + pnal_buf_t * p_frame_buf, + uint16_t offset) { uint8_t * buf = p_frame_buf->payload + offset; uint16_t buf_len = p_frame_buf->len - offset; pf_lldp_peer_info_t peer_data; + int loc_port_num = pnal_eth_get_tag (eth_handle); int err = 0; - err = pf_lldp_parse_packet (buf, buf_len, &peer_data); - - if (!err) - { - LOG_DEBUG ( - PF_LLDP_LOG, - "LLDP(%d): Received LLDP packet from %02X:%02X:%02X:%02X:%02X:%02X " - "Len: %d Chassis ID: %s Port ID: %s\n", - __LINE__, - peer_data.mac_address.addr[0], - peer_data.mac_address.addr[1], - peer_data.mac_address.addr[2], - peer_data.mac_address.addr[3], - peer_data.mac_address.addr[4], - peer_data.mac_address.addr[5], - p_frame_buf->len, - peer_data.chassis_id.string, - peer_data.port_id.string); - - pf_lldp_update_peer (net, PNET_PORT_1, &peer_data); - } - if (p_frame_buf != NULL) { + err = pf_lldp_parse_packet (buf, buf_len, &peer_data); + + if (!err) + { + LOG_DEBUG ( + PF_LLDP_LOG, + "LLDP(%d): Receive port %u MAC: %02X:%02X:%02X:%02X:%02X:%02X " + "Len: %d Chassis ID: %s Port ID: %s\n", + __LINE__, + loc_port_num, + peer_data.mac_address.addr[0], + peer_data.mac_address.addr[1], + peer_data.mac_address.addr[2], + peer_data.mac_address.addr[3], + peer_data.mac_address.addr[4], + peer_data.mac_address.addr[5], + p_frame_buf->len, + peer_data.chassis_id.string, + peer_data.port_id.string); + + if (pf_port_is_valid (net, loc_port_num)) + { + pf_lldp_update_peer (net, loc_port_num, &peer_data); + } + else + { + LOG_ERROR ( + PF_LLDP_LOG, + "LLDP(%d): Unexpected local port %u, frame discarded\n", + __LINE__, + loc_port_num); + } + } + pnal_buf_free (p_frame_buf); } diff --git a/src/common/pf_lldp.h b/src/common/pf_lldp.h index 1433e573..41ff2c00 100644 --- a/src/common/pf_lldp.h +++ b/src/common/pf_lldp.h @@ -341,12 +341,17 @@ void pf_lldp_send_disable (pnet_t * net, int loc_port_num); * Parse LLDP tlv format and store selected information. * Trigger alarms if needed. * @param net InOut: The p-net stack instance. + @param eth_handle In: Network interface handle * @param p_frame_buf InOut: The Ethernet frame. * @param offset In: The offset to start of LLDP data. * @return 0 If the frame was NOT handled by this function. * 1 If the frame was handled and the buffer freed. */ -int pf_lldp_recv (pnet_t * net, pnal_buf_t * p_frame_buf, uint16_t offset); +int pf_lldp_recv ( + pnet_t * net, + pnal_eth_handle_t * eth_handle, + pnal_buf_t * p_frame_buf, + uint16_t offset); /************ Internal functions, made available for unit testing ************/ diff --git a/src/device/pf_port.c b/src/device/pf_port.c index eb08c881..7c2558de 100644 --- a/src/device/pf_port.c +++ b/src/device/pf_port.c @@ -97,3 +97,8 @@ const pnet_port_cfg_t * pf_port_get_config (pnet_t * net, int loc_port_num) CC_ASSERT (loc_port_num > 0 && loc_port_num <= PNET_MAX_PORT); return &net->fspm_cfg.if_cfg.ports[loc_port_num - 1]; } + +bool pf_port_is_valid (pnet_t * net, int loc_port_num) +{ + return (loc_port_num > 0 && loc_port_num <= PNET_MAX_PORT); +} diff --git a/src/device/pf_port.h b/src/device/pf_port.h index f9ec6e00..d8b9f158 100644 --- a/src/device/pf_port.h +++ b/src/device/pf_port.h @@ -99,6 +99,17 @@ pf_port_t * pf_port_get_state (pnet_t * net, int loc_port_num); */ const pnet_port_cfg_t * pf_port_get_config (pnet_t * net, int loc_port_num); +/** + * Check if a number is a valid local port number. + * + * @param net In: The p-net stack instance + * @param loc_port_num In: Local port number. + * Valid range: 1 .. PNET_MAX_PORT + * @return true if valid local port number + * false if not + */ +bool pf_port_is_valid (pnet_t * net, int loc_port_num); + #ifdef __cplusplus } #endif diff --git a/src/device/pnet_api.c b/src/device/pnet_api.c index c971dd01..9aeead41 100644 --- a/src/device/pnet_api.c +++ b/src/device/pnet_api.c @@ -24,14 +24,12 @@ #include "pf_includes.h" #include "pf_block_reader.h" -int pnet_init_only ( - pnet_t * net, - const pnet_cfg_t * p_cfg) +int pnet_init_only (pnet_t * net, const pnet_cfg_t * p_cfg) { memset (net, 0, sizeof (*net)); - net->cmdev_initialized = false; /* TODO How to handle that pf_cmdev_exit() is - used before pf_cmdev_init()? */ + net->cmdev_initialized = false; /* TODO How to handle that pf_cmdev_exit() + is used before pf_cmdev_init()? */ pf_cmsu_init (net); pf_cmwrr_init (net); @@ -45,12 +43,9 @@ int pnet_init_only ( return -1; } - /* Initialize everything (and the DCP protocol) */ - /* First initialize the network interface */ - net->eth_handle = - pnal_eth_init (p_cfg->if_cfg.main_port.if_name, pf_eth_recv, (void *)net); - if (net->eth_handle == NULL) + if (pf_eth_init_network_interfaces (net, p_cfg) != 0) { + LOG_ERROR (PNET_LOG, "Failed initialize network interfaces\n"); return -1; } @@ -81,8 +76,7 @@ int pnet_init_only ( return 0; } -pnet_t * pnet_init ( - const pnet_cfg_t * p_cfg) +pnet_t * pnet_init (const pnet_cfg_t * p_cfg) { pnet_t * net = NULL; @@ -484,7 +478,7 @@ int pnet_alarm_send_ack ( return ret; } -/************************** Low-level diagnosis functions*********************/ +/************************** Low-level diagnosis functions ******************/ int pnet_diag_add ( pnet_t * net, @@ -545,7 +539,7 @@ int pnet_diag_remove ( usi); } -/************************** Diagnosis in standard format *********************/ +/************************** Diagnosis in standard format *******************/ int pnet_diag_std_add ( pnet_t * net, @@ -596,7 +590,7 @@ int pnet_diag_std_remove ( ext_ch_error_type); } -/************************** Diagnosis in USI format **************************/ +/************************** Diagnosis in USI format ************************/ int pnet_diag_usi_add ( pnet_t * net, diff --git a/src/pf_types.h b/src/pf_types.h index 70eaf3f9..476a5bc8 100644 --- a/src/pf_types.h +++ b/src/pf_types.h @@ -2695,6 +2695,7 @@ typedef struct pf_lldp_port */ typedef struct pf_port { + pnal_eth_handle_t * eth_handle; uint8_t port_num; pf_pdport_t pdport; pf_lldp_port_t lldp; @@ -2716,7 +2717,7 @@ struct pnet uint32_t dcp_sam_timeout; /* Handle to the SAM timeout instance */ uint32_t dcp_identresp_timeout; /* Handle to the DCP identify timeout instance */ - pnal_eth_handle_t * eth_handle; + pnal_eth_handle_t * eth_handle; /* Management port */ pf_eth_frame_id_map_t eth_id_map[PF_ETH_MAX_MAP]; volatile pf_scheduler_timeouts_t scheduler_timeouts[PF_MAX_TIMEOUTS]; volatile uint32_t scheduler_timeout_first; diff --git a/src/pnal.h b/src/pnal.h index 94bfe1ea..dfe0beb5 100644 --- a/src/pnal.h +++ b/src/pnal.h @@ -108,7 +108,7 @@ typedef struct pnal_port_stats #define PNAL_IP4_ADDR_TO_U32(ipaddr, a, b, c, d) \ ipaddr = PNAL_MAKEU32 (a, b, c, d) -enum pnal_eth_type +typedef enum pnal_ethertype { PNAL_ETHTYPE_IP = 0x0800U, PNAL_ETHTYPE_ARP = 0x0806U, @@ -116,7 +116,8 @@ enum pnal_eth_type PNAL_ETHTYPE_PROFINET = 0x8892U, PNAL_ETHTYPE_ETHERCAT = 0x88A4U, PNAL_ETHTYPE_LLDP = 0x88CCU, -}; + PNAL_ETHTYPE_ALL = 0xFFFFU, +} pnal_ethertype_t; /* 255.255.255.255 */ #ifndef PNAL_IPADDR_NONE @@ -262,6 +263,26 @@ pnal_buf_t * pnal_buf_alloc (uint16_t length); void pnal_buf_free (pnal_buf_t * p); uint8_t pnal_buf_header (pnal_buf_t * p, int16_t header_size_increment); +/** + * Network interface handle, forward declaration. + */ +typedef struct pnal_eth_handle_t pnal_eth_handle_t; + +/** + * The prototype of raw Ethernet reception call-back functions. + * + * @param eth_handle In. Network interface handle + * @param arg In: User-defined (may be NULL). + * @param p_buf In: The incoming Ethernet frame + * + * @return 0 If the frame was NOT handled by this function. + * 1 If the frame was handled and the buffer freed. + */ +typedef int (pnal_eth_callback_t) ( + pnal_eth_handle_t * eth_handle, + void * arg, + pnal_buf_t * p_buf); + /** * Get status of Ethernet link on specified port * @@ -316,6 +337,8 @@ int pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf); * Initialize receiving of raw Ethernet frames (in separate thread) * * @param if_name In: Ethernet interface name + * @param receive_type In: Ethernet frame types that shall be received + * by the network interface / port. * @param callback In: Callback for received raw Ethernet frames * @param arg InOut: User argument passed to the callback * @@ -323,9 +346,25 @@ int pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf); */ pnal_eth_handle_t * pnal_eth_init ( const char * if_name, + pnal_ethertype_t receive_type, pnal_eth_callback_t * callback, void * arg); +/** + * Set a tag on a network interface. + * @param handle InOut: Ethernet handle + * @param tag In: Tag to apply + */ +void pnal_eth_set_tag (pnal_eth_handle_t * handle, int tag); + +/** + * Get the tag for a network interface. + * Note that the tag is set using pnal_eth_set_tag. + * @param handle In: Ethernet handle + * @return tag, previously set with pnal_eth_set_tag + */ +int pnal_eth_get_tag (pnal_eth_handle_t * handle); + /** * Open an UDP socket * diff --git a/src/ports/linux/pnal_eth.c b/src/ports/linux/pnal_eth.c index 6879859b..d4f470b7 100644 --- a/src/ports/linux/pnal_eth.c +++ b/src/ports/linux/pnal_eth.c @@ -28,6 +28,15 @@ #include #include +typedef struct pnal_eth_handle_t +{ + int tag; + pnal_eth_callback_t * callback; + void * arg; + int socket; + os_thread_t * thread; +} pnal_eth_handle_t; + /** * @internal * Run a thread that listens to incoming raw Ethernet sockets. @@ -56,7 +65,7 @@ static void os_eth_task (void * thread_arg) if (eth_handle->callback != NULL) { - handled = eth_handle->callback (eth_handle->arg, p); + handled = eth_handle->callback (eth_handle, eth_handle->arg, p); } else { @@ -73,6 +82,7 @@ static void os_eth_task (void * thread_arg) pnal_eth_handle_t * pnal_eth_init ( const char * if_name, + pnal_ethertype_t receive_type, pnal_eth_callback_t * callback, void * arg) { @@ -82,6 +92,8 @@ pnal_eth_handle_t * pnal_eth_init ( struct sockaddr_ll sll; int ifindex; struct timeval timeout; + const uint16_t linux_receive_type = + (receive_type == PNAL_ETHTYPE_ALL) ? ETH_P_ALL : receive_type; handle = malloc (sizeof (pnal_eth_handle_t)); if (handle == NULL) @@ -91,7 +103,7 @@ pnal_eth_handle_t * pnal_eth_init ( handle->arg = arg; handle->callback = callback; - handle->socket = socket (PF_PACKET, SOCK_RAW, htons (ETH_P_ALL)); + handle->socket = socket (PF_PACKET, SOCK_RAW, htons (linux_receive_type)); /* Adjust send timeout */ timeout.tv_sec = 0; @@ -123,7 +135,7 @@ pnal_eth_handle_t * pnal_eth_init ( /* Bind socket to all protocols */ sll.sll_family = AF_PACKET; sll.sll_ifindex = ifindex; - sll.sll_protocol = htons (ETH_P_ALL); + sll.sll_protocol = htons (linux_receive_type); bind (handle->socket, (struct sockaddr *)&sll, sizeof (sll)); if (handle->socket > -1) @@ -139,9 +151,18 @@ pnal_eth_handle_t * pnal_eth_init ( } } +void pnal_eth_set_tag (pnal_eth_handle_t * handle, int tag) +{ + handle->tag = tag; +} + +int pnal_eth_get_tag (pnal_eth_handle_t * handle) +{ + return handle->tag; +} + int pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf) { int ret = send (handle->socket, buf->payload, buf->len, 0); - return ret; } diff --git a/src/ports/linux/pnal_sys.h b/src/ports/linux/pnal_sys.h index e79123c8..e38d1a51 100644 --- a/src/ports/linux/pnal_sys.h +++ b/src/ports/linux/pnal_sys.h @@ -32,25 +32,6 @@ typedef struct os_buf uint16_t len; } pnal_buf_t; -/** - * The prototype of raw Ethernet reception call-back functions. - * * - * @param arg In: User-defined (may be NULL). - * @param p_buf In: The incoming Ethernet frame - * - * @return 0 If the frame was NOT handled by this function. - * 1 If the frame was handled and the buffer freed. - */ -typedef int (pnal_eth_callback_t) (void * arg, pnal_buf_t * p_buf); - -typedef struct os_eth_handle -{ - pnal_eth_callback_t * callback; - void * arg; - int socket; - os_thread_t * thread; -} pnal_eth_handle_t; - #ifdef __cplusplus } #endif diff --git a/src/ports/linux/sampleapp_main.c b/src/ports/linux/sampleapp_main.c index e66b96cb..d5cef3e6 100644 --- a/src/ports/linux/sampleapp_main.c +++ b/src/ports/linux/sampleapp_main.c @@ -34,10 +34,19 @@ #include #include +#if PNET_MAX_PORT == 1 #define APP_DEFAULT_ETHERNET_INTERFACE "eth0" -#define APP_THREAD_PRIORITY 15 -#define APP_THREAD_STACKSIZE 4096 /* bytes */ -#define APP_MAIN_SLEEPTIME_US 5000 * 1000 +#elif PNET_MAX_PORT == 2 +#define APP_DEFAULT_ETHERNET_INTERFACE "br0,eth0,eth1" +#elif PNET_MAX_PORT == 3 +#define APP_DEFAULT_ETHERNET_INTERFACE "br0,eth0,eth1,eth2" +#else +#define APP_DEFAULT_ETHERNET_INTERFACE "" +#endif + +#define APP_THREAD_PRIORITY 15 +#define APP_THREAD_STACKSIZE 4096 /* bytes */ +#define APP_MAIN_SLEEPTIME_US 5000 * 1000 /************************* Utilities ******************************************/ @@ -125,7 +134,7 @@ struct cmd_args parse_commandline_arguments (int argc, char * argv[]) strcpy (output_arguments.path_button2, ""); strcpy (output_arguments.path_storage_directory, ""); strcpy (output_arguments.station_name, APP_DEFAULT_STATION_NAME); - strcpy (output_arguments.eth_interface, APP_DEFAULT_ETHERNET_INTERFACE); + strcpy (output_arguments.eth_interfaces, APP_DEFAULT_ETHERNET_INTERFACE); output_arguments.verbosity = 0; output_arguments.show = 0; output_arguments.factory_reset = false; @@ -149,7 +158,12 @@ struct cmd_args parse_commandline_arguments (int argc, char * argv[]) output_arguments.remove_files = true; break; case 'i': - strcpy (output_arguments.eth_interface, optarg); + if ((strlen (optarg) + 1) > sizeof (output_arguments.eth_interfaces)) + { + printf ("Error: The argument to -i is too long.\n"); + exit (EXIT_FAILURE); + } + strcpy (output_arguments.eth_interfaces, optarg); break; case 's': strcpy (output_arguments.station_name, optarg); @@ -299,6 +313,48 @@ void pn_main_thread (void * arg) printf ("Quitting the sample application\n\n"); } +int app_pnet_cfg_init_storage (pnet_cfg_t * p_cfg, struct cmd_args * p_args) +{ + strcpy (p_cfg->file_directory, p_args->path_storage_directory); + + if (p_args->verbosity > 0) + { + printf ("Storage directory: %s\n\n", p_cfg->file_directory); + } + + /* Validate paths */ + if (!pnal_does_file_exist (p_cfg->file_directory)) + { + printf ( + "Error: The given storage directory does not exist: %s\n", + p_cfg->file_directory); + return -1; + } + + if (p_args->path_button1[0] != '\0') + { + if (!pnal_does_file_exist (p_args->path_button1)) + { + printf ( + "Error: The given input file for Button1 does not exist: %s\n", + p_args->path_button1); + return -1; + } + } + + if (p_args->path_button2[0] != '\0') + { + if (!pnal_does_file_exist (p_args->path_button2)) + { + printf ( + "Error: The given input file for Button2 does not exist: %s\n", + p_args->path_button2); + return -1; + } + } + return 0; +} + /****************************** Main ******************************************/ int main (int argc, char * argv[]) @@ -307,10 +363,6 @@ int main (int argc, char * argv[]) pnet_cfg_t pnet_default_cfg; app_data_and_stack_t appdata_and_stack; app_data_t appdata; - pnal_ethaddr_t macbuffer; - pnal_ipaddr_t ip; - pnal_ipaddr_t netmask; - pnal_ipaddr_t gateway; int ret = 0; memset (&appdata, 0, sizeof (appdata)); @@ -331,101 +383,31 @@ int main (int argc, char * argv[]) PNET_MAX_SLOTS); printf ("P-net log level: %u (DEBUG=0, FATAL=4)\n", LOG_LEVEL); printf ("App verbosity level: %u\n", appdata.arguments.verbosity); - printf ("Ethernet interface: %s\n", appdata.arguments.eth_interface); + printf ("Number of ports: %u\n", PNET_MAX_PORT); + printf ("Network interfaces: %s\n", appdata.arguments.eth_interfaces); printf ("Button1 file: %s\n", appdata.arguments.path_button1); printf ("Button2 file: %s\n", appdata.arguments.path_button2); - printf ("Default station name: %s\n", appdata.arguments.station_name); - } - - /* Read IP, netmask, gateway and MAC address from operating system */ - ret = pnal_get_macaddress (appdata.arguments.eth_interface, &macbuffer); - if (ret != 0) - { - printf ( - "Error: The given Ethernet interface does not exist: %s\n", - appdata.arguments.eth_interface); - exit (EXIT_FAILURE); - } - - ip = pnal_get_ip_address (appdata.arguments.eth_interface); - netmask = pnal_get_netmask (appdata.arguments.eth_interface); - gateway = pnal_get_gateway (appdata.arguments.eth_interface); - if (gateway == IP_INVALID) - { - printf ( - "Error: Invalid gateway IP address for Ethernet interface: %s\n", - appdata.arguments.eth_interface); - exit (EXIT_FAILURE); - } - - if (appdata.arguments.verbosity > 0) - { - app_print_network_details (&macbuffer, ip, netmask, gateway); + printf ("Station name: %s\n", appdata.arguments.station_name); } - /* Prepare stack config with IP address, gateway, station name etc */ - app_adjust_stack_configuration (&pnet_default_cfg); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_addr, ip); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_gateway, gateway); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_mask, netmask); - strcpy (pnet_default_cfg.station_name, appdata.arguments.station_name); - strcpy ( - pnet_default_cfg.file_directory, - appdata.arguments.path_storage_directory); - - strncpy ( - pnet_default_cfg.if_cfg.main_port.if_name, - appdata.arguments.eth_interface, - PNET_INTERFACE_NAME_MAX_SIZE); - memcpy ( - pnet_default_cfg.if_cfg.main_port.eth_addr.addr, - macbuffer.addr, - sizeof (pnal_ethaddr_t)); - - strncpy ( - pnet_default_cfg.if_cfg.ports[0].phy_port.if_name, - appdata.arguments.eth_interface, - PNET_INTERFACE_NAME_MAX_SIZE); - memcpy ( - pnet_default_cfg.if_cfg.ports[0].phy_port.eth_addr.addr, - macbuffer.addr, - sizeof (pnal_ethaddr_t)); - + app_pnet_cfg_init_default (&pnet_default_cfg); pnet_default_cfg.cb_arg = (void *)&appdata; + strcpy (pnet_default_cfg.station_name, appdata.arguments.station_name); - if (appdata.arguments.verbosity > 0) + ret = app_pnet_cfg_init_netifs ( + appdata.arguments.eth_interfaces, + &pnet_default_cfg, + appdata.arguments.verbosity); + if (ret != 0) { - printf ("Storage directory: %s\n\n", pnet_default_cfg.file_directory); + exit (EXIT_FAILURE); } - /* Validate paths */ - if (!pnal_does_file_exist (pnet_default_cfg.file_directory)) + ret = app_pnet_cfg_init_storage (&pnet_default_cfg, &appdata.arguments); + if (ret != 0) { - printf ( - "Error: The given storage directory does not exist: %s\n", - pnet_default_cfg.file_directory); exit (EXIT_FAILURE); } - if (appdata.arguments.path_button1[0] != '\0') - { - if (!pnal_does_file_exist (appdata.arguments.path_button1)) - { - printf ( - "Error: The given input file for Button1 does not exist: %s\n", - appdata.arguments.path_button1); - exit (EXIT_FAILURE); - } - } - if (appdata.arguments.path_button2[0] != '\0') - { - if (!pnal_does_file_exist (appdata.arguments.path_button2)) - { - printf ( - "Error: The given input file for Button2 does not exist: %s\n", - appdata.arguments.path_button2); - exit (EXIT_FAILURE); - } - } /* Remove files and exit */ if (appdata.arguments.remove_files == true) diff --git a/src/ports/rt-kernel/dwmac1000.c b/src/ports/rt-kernel/dwmac1000.c index 0c8a3fc0..938c45fe 100644 --- a/src/ports/rt-kernel/dwmac1000.c +++ b/src/ports/rt-kernel/dwmac1000.c @@ -240,7 +240,7 @@ typedef struct dwmac1000 uint32_t cr; } dwmac1000_t; -static pnal_eth_callback_t * input_rx_hook = NULL; +static pnal_eth_sys_recv_callback_t * input_rx_hook = NULL; static void * input_rx_arg = NULL; #ifdef DEBUG_DATA @@ -613,7 +613,7 @@ static void dwmac1000_input (dwmac1000_t * dwmac1000, struct netif * netif) /* Pass pbuf to rx hook if set */ if (input_rx_hook != NULL) { - handled = input_rx_hook (input_rx_arg, p); + handled = input_rx_hook (netif, input_rx_arg, p); if (handled != 0) { return; @@ -719,7 +719,7 @@ int eth_ioctl (drv_t * drv, void * arg, int req, void * param) if (req == IOCTL_NET_SET_RX_HOOK) { - input_rx_hook = (pnal_eth_callback_t *)param; + input_rx_hook = (pnal_eth_sys_recv_callback_t *)param; input_rx_arg = arg; return 0; } diff --git a/src/ports/rt-kernel/pnal_eth.c b/src/ports/rt-kernel/pnal_eth.c index adc1311f..988ae98d 100644 --- a/src/ports/rt-kernel/pnal_eth.c +++ b/src/ports/rt-kernel/pnal_eth.c @@ -28,42 +28,60 @@ #define MAX_NUMBER_OF_IF 1 #define NET_DRIVER_NAME "/net" -static struct netif * interface[MAX_NUMBER_OF_IF]; +typedef struct pnal_eth_handle_t +{ + int loc_port_num; + struct netif * netif; + pnal_eth_callback_t * eth_rx_callback; + void * arg; + int if_id; +} pnal_eth_handle_t; + +static pnal_eth_handle_t interface[MAX_NUMBER_OF_IF]; static int nic_index = 0; +static int pnal_eth_sys_recv ( + struct netif * netif, + void * arg, + pnal_buf_t * p_buf) +{ + int ret = -1; + pnal_eth_handle_t * handle; + if (arg != NULL && p_buf != NULL) + { + handle = (pnal_eth_handle_t *)arg; + ret = handle->eth_rx_callback (handle, handle->arg, p_buf); + } + return ret; +} + pnal_eth_handle_t * pnal_eth_init ( const char * if_name, + pnal_ethertype_t receive_type, pnal_eth_callback_t * callback, void * arg) { - pnal_eth_handle_t * handle; + pnal_eth_handle_t * handle = NULL; struct netif * found_if; drv_t * drv; - handle = malloc (sizeof (pnal_eth_handle_t)); - UASSERT (handle != NULL, EMEM); - - handle->arg = arg; - handle->callback = callback; - handle->if_id = -1; + (void)receive_type; /* Ignore, for now all frames will be received. */ if (nic_index < MAX_NUMBER_OF_IF) { drv = dev_find_driver (NET_DRIVER_NAME); - if (drv != NULL) + found_if = netif_find (if_name); + if (drv != NULL && found_if != NULL) { - eth_ioctl (drv, arg, IOCTL_NET_SET_RX_HOOK, callback); - - found_if = netif_find (if_name); - if (found_if == NULL) - { - interface[nic_index] = netif_default; - } - else - { - interface[nic_index] = found_if; - } - handle->if_id = nic_index++; + handle = &interface[nic_index]; + handle->if_id = nic_index; + nic_index++; + + handle->arg = arg; + handle->eth_rx_callback = callback; + handle->netif = found_if; + + eth_ioctl (drv, handle, IOCTL_NET_SET_RX_HOOK, pnal_eth_sys_recv); } else { @@ -71,15 +89,17 @@ pnal_eth_handle_t * pnal_eth_init ( } } - if (handle->if_id > -1) - { - return handle; - } - else - { - free (handle); - return NULL; - } + return handle; +} + +void pnal_eth_set_tag (pnal_eth_handle_t * handle, int loc_port_num) +{ + handle->loc_port_num = loc_port_num; +} + +int pnal_eth_get_tag (pnal_eth_handle_t * handle) +{ + return handle->loc_port_num; } int pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf) @@ -91,9 +111,9 @@ int pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf) /* TODO: remove tot_len from os_buff */ buf->tot_len = buf->len; - if (interface[handle->if_id]->linkoutput != NULL) + if (handle->netif->linkoutput != NULL) { - interface[handle->if_id]->linkoutput (interface[handle->if_id], buf); + handle->netif->linkoutput (handle->netif, buf); ret = buf->len; } } diff --git a/src/ports/rt-kernel/pnal_sys.h b/src/ports/rt-kernel/pnal_sys.h index 30a3997e..50e51fb0 100644 --- a/src/ports/rt-kernel/pnal_sys.h +++ b/src/ports/rt-kernel/pnal_sys.h @@ -34,25 +34,12 @@ typedef struct pbuf pnal_buf_t; */ #define IOCTL_NET_SET_RX_HOOK 0x601 -int eth_ioctl (drv_t * drv, void * arg, int req, void * param); +typedef int (pnal_eth_sys_recv_callback_t) ( + struct netif * netif, + void * arg, + pnal_buf_t * p_buf); -/** - * The prototype of raw Ethernet reception call-back functions. - * * - * @param arg In: User-defined (may be NULL). - * @param p_buf In: The incoming Ethernet frame - * - * @return 0 If the frame was NOT handled by this function. - * 1 If the frame was handled and the buffer freed. - */ -typedef int (pnal_eth_callback_t) (void * arg, pnal_buf_t * p_buf); - -typedef struct os_eth_handle -{ - pnal_eth_callback_t * callback; - void * arg; - int if_id; -} pnal_eth_handle_t; +int eth_ioctl (drv_t * drv, void * arg, int req, void * param); #ifdef __cplusplus } diff --git a/src/ports/rt-kernel/sampleapp_main.c b/src/ports/rt-kernel/sampleapp_main.c index 182fd015..7d81f63f 100644 --- a/src/ports/rt-kernel/sampleapp_main.c +++ b/src/ports/rt-kernel/sampleapp_main.c @@ -163,29 +163,22 @@ SHELL_CMD (cmd_pnio_remove_files); int main (void) { int ret = -1; - struct cmd_args cmdline_arguments; app_data_t appdata; - pnal_ethaddr_t macbuffer; - pnal_ipaddr_t ip; - pnal_ipaddr_t netmask; - pnal_ipaddr_t gateway; - g_net = NULL; + gp_appdata = &appdata; /* Prepare appdata */ - memset (&cmdline_arguments, 0, sizeof (cmdline_arguments)); - strcpy (cmdline_arguments.eth_interface, APP_DEFAULT_ETHERNET_INTERFACE); - strcpy (cmdline_arguments.station_name, APP_DEFAULT_STATION_NAME); - cmdline_arguments.verbosity = (LOG_LEVEL <= LOG_LEVEL_WARNING) ? 1 : 0; memset (&appdata, 0, sizeof (appdata)); appdata.alarm_allowed = true; - appdata.arguments = cmdline_arguments; + strcpy (appdata.arguments.eth_interfaces, APP_DEFAULT_ETHERNET_INTERFACE); + strcpy (appdata.arguments.station_name, APP_DEFAULT_STATION_NAME); + appdata.arguments.verbosity = (LOG_LEVEL <= LOG_LEVEL_WARNING) ? 1 : 0; + appdata.main_events = os_event_create(); appdata.main_timer = os_timer_create (APP_TICK_INTERVAL_US, main_timer_tick, NULL, false); - gp_appdata = &appdata; - printf ("\n** Starting Profinet sample application " PNET_VERSION " **\n"); + printf ("\n** Profinet sample application **\n"); if (appdata.arguments.verbosity > 0) { printf ( @@ -193,68 +186,27 @@ int main (void) PNET_MAX_SLOTS); printf ("P-net log level: %u (DEBUG=0, FATAL=4)\n", LOG_LEVEL); printf ("App verbosity level: %u\n", appdata.arguments.verbosity); - printf ("Ethernet interface: %s\n", appdata.arguments.eth_interface); + printf ("Number of ports: %u\n", PNET_MAX_PORT); + printf ("Network interfaces: %s\n", appdata.arguments.eth_interfaces); printf ("Default station name: %s\n", appdata.arguments.station_name); } - /* Read IP, netmask, gateway and MAC address from operating system */ - ret = pnal_get_macaddress (appdata.arguments.eth_interface, &macbuffer); - if (ret != 0) - { - printf ( - "Error: The given Ethernet interface does not exist: %s\n", - appdata.arguments.eth_interface); - return -1; - } + app_pnet_cfg_init_default (&pnet_default_cfg); + pnet_default_cfg.cb_arg = (void *)&appdata; + strcpy (pnet_default_cfg.station_name, appdata.arguments.station_name); + strcpy (pnet_default_cfg.file_directory, APP_DEFAULT_FILE_DIRECTORY); - ip = pnal_get_ip_address (appdata.arguments.eth_interface); - netmask = pnal_get_netmask (appdata.arguments.eth_interface); - gateway = pnal_get_gateway (appdata.arguments.eth_interface); - if (gateway == IP_INVALID) + ret = app_pnet_cfg_init_netifs ( + appdata.arguments.eth_interfaces, + &pnet_default_cfg, + appdata.arguments.verbosity); + if (ret != 0) { - printf ( - "Error: Invalid gateway IP address for Ethernet interface: %s\n", - appdata.arguments.eth_interface); return -1; } - if (appdata.arguments.verbosity > 0) - { - app_print_network_details (&macbuffer, ip, netmask, gateway); - } - - /* Prepare stack config with IP address, gateway, station name etc */ - app_adjust_stack_configuration (&pnet_default_cfg); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_addr, ip); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_gateway, gateway); - app_copy_ip_to_struct (&pnet_default_cfg.if_cfg.ip_cfg.ip_mask, netmask); - - strcpy (pnet_default_cfg.file_directory, APP_DEFAULT_FILE_DIRECTORY); - strcpy (pnet_default_cfg.station_name, gp_appdata->arguments.station_name); - - strncpy ( - pnet_default_cfg.if_cfg.main_port.if_name, - appdata.arguments.eth_interface, - PNET_INTERFACE_NAME_MAX_SIZE); - memcpy ( - pnet_default_cfg.if_cfg.main_port.eth_addr.addr, - macbuffer.addr, - sizeof (pnet_ethaddr_t)); - - strncpy ( - pnet_default_cfg.if_cfg.ports[0].phy_port.if_name, - appdata.arguments.eth_interface, - PNET_INTERFACE_NAME_MAX_SIZE); - memcpy ( - pnet_default_cfg.if_cfg.ports[0].phy_port.eth_addr.addr, - macbuffer.addr, - sizeof (pnet_ethaddr_t)); - - pnet_default_cfg.cb_arg = (void *)gp_appdata; - - /* Initialize Profinet stack */ - g_net = pnet_init ( - &pnet_default_cfg); + /* Initialize profinet stack */ + g_net = pnet_init (&pnet_default_cfg); if (g_net == NULL) { printf ("Failed to initialize p-net application.\n"); diff --git a/test/mocks.cpp b/test/mocks.cpp index 927fa2a9..6ed2c684 100644 --- a/test/mocks.cpp +++ b/test/mocks.cpp @@ -23,6 +23,7 @@ uint8_t pnet_log_level; os_mutex_t * mock_mutex; mock_os_data_t mock_os_data; mock_lldp_data_t mock_lldp_data; +pnal_eth_handle_t * mock_pnal_eth_if; void mock_clear (void) { @@ -53,8 +54,9 @@ pnal_eth_handle_t * mock_pnal_eth_init ( { pnal_eth_handle_t * handle; - handle = (pnal_eth_handle_t *)calloc (1, sizeof (pnal_eth_handle_t)); + handle = (pnal_eth_handle_t *)calloc (1, 10 /*sizeof (pnal_eth_handle_t)*/); + mock_pnal_eth_if = handle; return handle; } @@ -99,6 +101,11 @@ int mock_pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * p_buf) return p_buf->len; } +int mock_pnal_eth_recv (void * arg, pnal_buf_t * buf) +{ + return pf_eth_recv (mock_pnal_eth_if, arg, buf); +} + int mock_pnal_udp_open (pnal_ipaddr_t addr, pnal_ipport_t port) { int ret = 2; diff --git a/test/mocks.h b/test/mocks.h index 659157d5..78413589 100644 --- a/test/mocks.h +++ b/test/mocks.h @@ -84,6 +84,7 @@ pnal_eth_handle_t * mock_pnal_eth_init ( pnal_eth_callback_t * callback, void * arg); int mock_pnal_eth_send (pnal_eth_handle_t * handle, pnal_buf_t * buf); +int mock_pnal_eth_recv (void * arg, pnal_buf_t * buf); int mock_pnal_eth_get_status ( const char * interface_name, pnal_eth_status_t * status); diff --git a/test/test_cmrpc.cpp b/test/test_cmrpc.cpp index 3a9b432d..db15bed2 100644 --- a/test/test_cmrpc.cpp +++ b/test/test_cmrpc.cpp @@ -659,7 +659,7 @@ TEST_F (CmrpcTest, CmrpcConnectReleaseIOSAR_DA) EXPECT_EQ (appdata.call_counters.state_calls, 1); EXPECT_EQ (appdata.cmdev_state, PNET_EVENT_STARTUP); EXPECT_EQ (appdata.call_counters.connect_calls, 1); - EXPECT_EQ (mock_os_data.eth_send_count, 1); + EXPECT_EQ (mock_os_data.eth_send_count, PNET_MAX_PORT); TEST_TRACE ("\nGenerating mock release request IOSAR_DA\n"); mock_set_pnal_udp_recvfrom_buffer ( diff --git a/test/test_dcp.cpp b/test/test_dcp.cpp index 3509ace7..f68d2313 100644 --- a/test/test_dcp.cpp +++ b/test/test_dcp.cpp @@ -101,10 +101,10 @@ TEST_F (DcpTest, DcpHelloTest) p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, get_name_req, sizeof (get_name_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); - EXPECT_EQ (mock_os_data.eth_send_count, 2); + EXPECT_EQ (mock_os_data.eth_send_count, PNET_MAX_PORT + 1); EXPECT_EQ (appdata.call_counters.led_off_calls, 1); EXPECT_EQ (appdata.call_counters.led_on_calls, 0); @@ -125,36 +125,36 @@ TEST_F (DcpTest, DcpRunTest) TEST_TRACE ("\nGenerating mock set name request\n"); p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, set_name_req, sizeof (set_name_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); TEST_TRACE ("\nGenerating mock set IP request\n"); p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, set_ip_req, sizeof (set_ip_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); TEST_TRACE ("\nGenerating mock set ident request\n"); p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, ident_req, sizeof (ident_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); TEST_TRACE ("\nGenerating mock factory reset request\n"); p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, factory_reset_req, sizeof (factory_reset_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); TEST_TRACE ("\nGenerating mock flash LED request\n"); p_buf = pnal_buf_alloc (PF_FRAME_BUFFER_SIZE); memcpy (p_buf->payload, signal_req, sizeof (signal_req)); - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); /* Wait for LED to flash three times at 1 Hz */ run_stack (4 * 1000 * 1000); - EXPECT_EQ (mock_os_data.eth_send_count, 9); + EXPECT_EQ (mock_os_data.eth_send_count, 9 + (PNET_MAX_PORT - 1) * 4); EXPECT_EQ (mock_os_data.set_ip_suite_count, 2); EXPECT_EQ (appdata.call_counters.led_on_calls, 3); diff --git a/test/test_lldp.cpp b/test/test_lldp.cpp index 36236190..66463c89 100644 --- a/test/test_lldp.cpp +++ b/test/test_lldp.cpp @@ -373,7 +373,7 @@ TEST_F (LldpTest, LldpInitTest) EXPECT_EQ (appdata.call_counters.ccontrol_calls, 0); EXPECT_EQ (appdata.call_counters.read_calls, 0); EXPECT_EQ (appdata.call_counters.write_calls, 0); - EXPECT_EQ (mock_os_data.eth_send_count, 1); + EXPECT_EQ (mock_os_data.eth_send_count, PNET_MAX_PORT); } TEST_F (LldpTest, LldpGenerateAliasName) diff --git a/test/utils_for_testing.cpp b/test/utils_for_testing.cpp index 2b07e0f9..98ee167e 100644 --- a/test/utils_for_testing.cpp +++ b/test/utils_for_testing.cpp @@ -621,7 +621,7 @@ void PnetIntegrationTestBase::send_data (uint8_t * data_packet, uint16_t len) *(p_ctr + 1) = appdata.data_cycle_ctr & 0xff; p_buf->len = len; - ret = pf_eth_recv (net, p_buf); + ret = mock_pnal_eth_recv (net, p_buf); EXPECT_EQ (ret, 1); if (ret == 0) {