|
@@ -0,0 +1,448 @@
|
|
|
|
+/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
+ *
|
|
|
|
+ * Copyright 2018 (c) Kontron Europe GmbH (Author: Rudolf Hoyler)
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+#include "ua_plugin_network.h"
|
|
|
|
+#include "ua_log_stdout.h"
|
|
|
|
+#include "ua_util.h"
|
|
|
|
+
|
|
|
|
+#include <netinet/ether.h>
|
|
|
|
+#include <linux/if_packet.h>
|
|
|
|
+
|
|
|
|
+#include "ua_network_pubsub_ethernet.h"
|
|
|
|
+#include "../src/ua_util_internal.h"
|
|
|
|
+
|
|
|
|
+#ifndef ETHERTYPE_UADP
|
|
|
|
+#define ETHERTYPE_UADP 0xb62c
|
|
|
|
+#endif
|
|
|
|
+
|
|
|
|
+/* Ethernet network layer specific internal data */
|
|
|
|
+typedef struct {
|
|
|
|
+ int ifindex;
|
|
|
|
+ UA_UInt16 vid;
|
|
|
|
+ UA_Byte prio;
|
|
|
|
+ UA_Byte ifAddress[ETH_ALEN];
|
|
|
|
+ UA_Byte targetAddress[ETH_ALEN];
|
|
|
|
+} UA_PubSubChannelDataEthernet;
|
|
|
|
+
|
|
|
|
+/*
|
|
|
|
+ * OPC-UA specification Part 14:
|
|
|
|
+ *
|
|
|
|
+ * "The target is a MAC address, an IP address or a registered name like a
|
|
|
|
+ * hostname. The format of a MAC address is six groups of hexadecimal digits,
|
|
|
|
+ * separated by hyphens (e.g. 01-23-45-67-89-ab). A system may also accept
|
|
|
|
+ * hostnames and/or IP addresses if it provides means to resolve it to a MAC
|
|
|
|
+ * address (e.g. DNS and Reverse-ARP)."
|
|
|
|
+ *
|
|
|
|
+ * We do not support currently IP addresses or hostnames.
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_parseHardwareAddress(UA_String* target, UA_Byte* destinationMac) {
|
|
|
|
+ size_t curr = 0, idx = 0;
|
|
|
|
+ for(; idx < ETH_ALEN; idx++) {
|
|
|
|
+ UA_UInt32 value;
|
|
|
|
+ size_t progress =
|
|
|
|
+ UA_readNumberWithBase(&target->data[curr],
|
|
|
|
+ target->length - curr, &value, 16);
|
|
|
|
+ if(progress == 0 || value > (long)0xff)
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+
|
|
|
|
+ destinationMac[idx] = (UA_Byte) value;
|
|
|
|
+
|
|
|
|
+ curr += progress;
|
|
|
|
+ if(curr == target->length)
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ if(target->data[curr] != '-')
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+
|
|
|
|
+ curr++; /* skip '-' */
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if(idx != (ETH_ALEN-1))
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Open communication socket based on the connectionConfig.
|
|
|
|
+ *
|
|
|
|
+ * @return ref to created channel, NULL on error
|
|
|
|
+ */
|
|
|
|
+static UA_PubSubChannel *
|
|
|
|
+UA_PubSubChannelEthernet_open(const UA_PubSubConnectionConfig *connectionConfig) {
|
|
|
|
+
|
|
|
|
+ UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "Open PubSub ethernet connection.");
|
|
|
|
+
|
|
|
|
+ /* allocate and init memory for the ethernet specific internal data */
|
|
|
|
+ UA_PubSubChannelDataEthernet* channelDataEthernet =
|
|
|
|
+ (UA_PubSubChannelDataEthernet*) UA_calloc(1, sizeof(*channelDataEthernet));
|
|
|
|
+ if(!channelDataEthernet) {
|
|
|
|
+ UA_LOG_ERROR (UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub Connection creation failed. Out of memory.");
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* handle specified network address */
|
|
|
|
+ UA_NetworkAddressUrlDataType *address;
|
|
|
|
+ if(UA_Variant_hasScalarType(&connectionConfig->address,
|
|
|
|
+ &UA_TYPES[UA_TYPES_NETWORKADDRESSURLDATATYPE])) {
|
|
|
|
+ address = (UA_NetworkAddressUrlDataType *) connectionConfig->address.data;
|
|
|
|
+ } else {
|
|
|
|
+ UA_LOG_ERROR (UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub Connection creation failed. Invalid Address.");
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+ UA_LOG_DEBUG(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "Specified Interface Name = %.*s",
|
|
|
|
+ (int) address->networkInterface.length, address->networkInterface.data);
|
|
|
|
+ UA_LOG_DEBUG(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "Specified Network Url = %.*s",
|
|
|
|
+ (int)address->url.length, address->url.data);
|
|
|
|
+
|
|
|
|
+ UA_String target;
|
|
|
|
+ /* encode the URL and store information in internal structure */
|
|
|
|
+ if(UA_parseEndpointUrlEthernet(&address->url, &target, &channelDataEthernet->vid,
|
|
|
|
+ &channelDataEthernet->prio)) {
|
|
|
|
+ UA_LOG_ERROR (UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub Connection creation failed. Invalid Address URL.");
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Get a valid MAC address from target definition */
|
|
|
|
+ if(UA_parseHardwareAddress(&target, channelDataEthernet->targetAddress) != UA_STATUSCODE_GOOD) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub Connection creation failed. Invalid destination MAC address.");
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* generate a new Pub/Sub channel and open a related socket */
|
|
|
|
+ UA_PubSubChannel *newChannel = (UA_PubSubChannel*)UA_calloc(1, sizeof(UA_PubSubChannel));
|
|
|
|
+ if(!newChannel) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub Connection creation failed. Out of memory.");
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Open a packet socket */
|
|
|
|
+ int sockFd = UA_socket(PF_PACKET, SOCK_RAW, 0);
|
|
|
|
+ if(sockFd < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection creation failed. Cannot create socket.");
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ UA_free(newChannel);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+ newChannel->sockfd = sockFd;
|
|
|
|
+
|
|
|
|
+ /* allow the socket to be reused */
|
|
|
|
+ int opt = 1;
|
|
|
|
+ if(UA_setsockopt(sockFd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection creation failed. Cannot set socket reuse.");
|
|
|
|
+ UA_close(sockFd);
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ UA_free(newChannel);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* get interface index */
|
|
|
|
+ struct ifreq ifreq = { 0 };
|
|
|
|
+ strncpy(ifreq.ifr_name, (char*)address->networkInterface.data,
|
|
|
|
+ UA_MIN(address->networkInterface.length, sizeof(ifreq.ifr_name)-1));
|
|
|
|
+
|
|
|
|
+ if(ioctl(sockFd, SIOCGIFINDEX, &ifreq) < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection creation failed. Cannot get interface index.");
|
|
|
|
+ UA_close(sockFd);
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ UA_free(newChannel);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+ channelDataEthernet->ifindex = ifreq.ifr_ifindex;
|
|
|
|
+
|
|
|
|
+ /* determine own MAC address (source address for send) */
|
|
|
|
+ if(ioctl(sockFd, SIOCGIFHWADDR, &ifreq) < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection creation failed. Cannot determine own MAC address.");
|
|
|
|
+ UA_close(sockFd);
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ UA_free(newChannel);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+ memcpy(channelDataEthernet->ifAddress, &ifreq.ifr_hwaddr.sa_data, ETH_ALEN);
|
|
|
|
+
|
|
|
|
+ /* bind the socket to interface and ethertype */
|
|
|
|
+ struct sockaddr_ll sll = { 0 };
|
|
|
|
+ sll.sll_family = AF_PACKET;
|
|
|
|
+ sll.sll_ifindex = channelDataEthernet->ifindex;
|
|
|
|
+ sll.sll_protocol = htons(ETHERTYPE_UADP);
|
|
|
|
+
|
|
|
|
+ if(UA_bind(sockFd, (struct sockaddr*)&sll, sizeof(sll)) < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection creation failed. Cannot bind socket.");
|
|
|
|
+ UA_close(sockFd);
|
|
|
|
+ UA_free(channelDataEthernet);
|
|
|
|
+ UA_free(newChannel);
|
|
|
|
+ return NULL;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ newChannel->handle = channelDataEthernet;
|
|
|
|
+ newChannel->state = UA_PUBSUB_CHANNEL_PUB;
|
|
|
|
+
|
|
|
|
+ return newChannel;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static UA_Boolean
|
|
|
|
+is_multicast_address(const UA_Byte *address) {
|
|
|
|
+ /* check if it is a unicast address */
|
|
|
|
+ if((address[0] & 1) == 0) {
|
|
|
|
+ return UA_FALSE;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* and exclude broadcast addresses */
|
|
|
|
+ for(size_t i = 0; i < ETH_ALEN; i++) {
|
|
|
|
+ if(address[i] != 0xff)
|
|
|
|
+ return UA_TRUE;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* reaching this point, we know it has to be a broadcast address */
|
|
|
|
+ return UA_FALSE;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Subscribe to a given address.
|
|
|
|
+ *
|
|
|
|
+ * @return UA_STATUSCODE_GOOD on success
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_PubSubChannelEthernet_regist(UA_PubSubChannel *channel,
|
|
|
|
+ UA_ExtensionObject *transportSettings) {
|
|
|
|
+ UA_PubSubChannelDataEthernet *channelDataEthernet =
|
|
|
|
+ (UA_PubSubChannelDataEthernet *) channel->handle;
|
|
|
|
+
|
|
|
|
+ if(!is_multicast_address(channelDataEthernet->targetAddress))
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+
|
|
|
|
+ struct packet_mreq mreq;
|
|
|
|
+ mreq.mr_ifindex = channelDataEthernet->ifindex;
|
|
|
|
+ mreq.mr_type = PACKET_MR_MULTICAST;
|
|
|
|
+ mreq.mr_alen = ETH_ALEN;
|
|
|
|
+ memcpy(mreq.mr_address, channelDataEthernet->targetAddress, ETH_ALEN);
|
|
|
|
+
|
|
|
|
+ if(UA_setsockopt(channel->sockfd, SOL_PACKET, PACKET_ADD_MEMBERSHIP, (char*) &mreq, sizeof(mreq)) < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "PubSub Connection regist failed. %s", strerror(errno));
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Remove current subscription.
|
|
|
|
+ *
|
|
|
|
+ * @return UA_STATUSCODE_GOOD on success
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_PubSubChannelEthernet_unregist(UA_PubSubChannel *channel,
|
|
|
|
+ UA_ExtensionObject *transportSettings) {
|
|
|
|
+ UA_PubSubChannelDataEthernet *channelDataEthernet =
|
|
|
|
+ (UA_PubSubChannelDataEthernet *) channel->handle;
|
|
|
|
+
|
|
|
|
+ if(!is_multicast_address(channelDataEthernet->targetAddress)) {
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ struct packet_mreq mreq;
|
|
|
|
+ mreq.mr_ifindex = channelDataEthernet->ifindex;
|
|
|
|
+ mreq.mr_type = PACKET_MR_MULTICAST;
|
|
|
|
+ mreq.mr_alen = ETH_ALEN;
|
|
|
|
+ memcpy(mreq.mr_address, channelDataEthernet->targetAddress, ETH_ALEN);
|
|
|
|
+
|
|
|
|
+ if(UA_setsockopt(channel->sockfd, SOL_PACKET, PACKET_DROP_MEMBERSHIP, (char*) &mreq, sizeof(mreq) < 0)) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "PubSub Connection regist failed.");
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Send messages to the connection defined address
|
|
|
|
+ *
|
|
|
|
+ * @return UA_STATUSCODE_GOOD if success
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_PubSubChannelEthernet_send(UA_PubSubChannel *channel,
|
|
|
|
+ UA_ExtensionObject *transportSettings,
|
|
|
|
+ const UA_ByteString *buf) {
|
|
|
|
+ UA_PubSubChannelDataEthernet *channelDataEthernet =
|
|
|
|
+ (UA_PubSubChannelDataEthernet *) channel->handle;
|
|
|
|
+
|
|
|
|
+ /* Allocate a buffer for the ethernet data which contains the ethernet
|
|
|
|
+ * header (without VLAN tag), the VLAN tag and the OPC-UA/Ethernet data. */
|
|
|
|
+ char *bufSend, *ptrCur;
|
|
|
|
+ size_t lenBuf;
|
|
|
|
+ struct ether_header* ethHdr;
|
|
|
|
+
|
|
|
|
+ lenBuf = sizeof(*ethHdr) + 4 + buf->length;
|
|
|
|
+ bufSend = (char*) UA_malloc(lenBuf);
|
|
|
|
+ ethHdr = (struct ether_header*) bufSend;
|
|
|
|
+
|
|
|
|
+ /* Set (own) source MAC address */
|
|
|
|
+ memcpy(ethHdr->ether_shost, channelDataEthernet->ifAddress, ETH_ALEN);
|
|
|
|
+
|
|
|
|
+ /* Set destination MAC address */
|
|
|
|
+ memcpy(ethHdr->ether_dhost, channelDataEthernet->targetAddress, ETH_ALEN);
|
|
|
|
+
|
|
|
|
+ /* Set ethertype */
|
|
|
|
+ /* Either VLAN or Ethernet */
|
|
|
|
+ ptrCur = bufSend + sizeof(*ethHdr);
|
|
|
|
+ if(channelDataEthernet->vid == 0) {
|
|
|
|
+ ethHdr->ether_type = htons(ETHERTYPE_UADP);
|
|
|
|
+ lenBuf -= 4; /* no VLAN tag */
|
|
|
|
+ } else {
|
|
|
|
+ ethHdr->ether_type = htons(ETHERTYPE_VLAN);
|
|
|
|
+ /* set VLAN ID */
|
|
|
|
+ UA_UInt16 vlanTag;
|
|
|
|
+ vlanTag = (UA_UInt16) (channelDataEthernet->vid + (channelDataEthernet->prio << 13));
|
|
|
|
+ *((UA_UInt16 *) ptrCur) = htons(vlanTag);
|
|
|
|
+ ptrCur += sizeof(UA_UInt16);
|
|
|
|
+ /* set Ethernet */
|
|
|
|
+ *((UA_UInt16 *) ptrCur) = htons(ETHERTYPE_UADP);
|
|
|
|
+ ptrCur += sizeof(UA_UInt16);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* copy payload of ethernet message */
|
|
|
|
+ memcpy(ptrCur, buf->data, buf->length);
|
|
|
|
+
|
|
|
|
+ ssize_t rc;
|
|
|
|
+ rc = UA_send(channel->sockfd, bufSend, lenBuf, 0);
|
|
|
|
+ if(rc < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection send failed. Send message failed.");
|
|
|
|
+ UA_free(bufSend);
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+ UA_free(bufSend);
|
|
|
|
+
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Receive messages.
|
|
|
|
+ *
|
|
|
|
+ * @param timeout in usec -> not used
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_PubSubChannelEthernet_receive(UA_PubSubChannel *channel, UA_ByteString *message,
|
|
|
|
+ UA_ExtensionObject *transportSettings, UA_UInt32 timeout) {
|
|
|
|
+ UA_PubSubChannelDataEthernet *channelDataEthernet =
|
|
|
|
+ (UA_PubSubChannelDataEthernet *) channel->handle;
|
|
|
|
+
|
|
|
|
+ struct ether_header eth_hdr;
|
|
|
|
+ struct msghdr msg;
|
|
|
|
+ struct iovec iov[2];
|
|
|
|
+
|
|
|
|
+ iov[0].iov_base = ð_hdr;
|
|
|
|
+ iov[0].iov_len = sizeof(eth_hdr);
|
|
|
|
+ iov[1].iov_base = message->data;
|
|
|
|
+ iov[1].iov_len = message->length;
|
|
|
|
+ msg.msg_namelen = 0;
|
|
|
|
+ msg.msg_iov = iov;
|
|
|
|
+ msg.msg_iovlen = 2;
|
|
|
|
+ msg.msg_controllen = 0;
|
|
|
|
+
|
|
|
|
+ /* Sleep in a select call if a timeout was set */
|
|
|
|
+ if(timeout > 0) {
|
|
|
|
+ fd_set fdset;
|
|
|
|
+ FD_ZERO(&fdset);
|
|
|
|
+ UA_fd_set(channel->sockfd, &fdset);
|
|
|
|
+ struct timeval tmptv = {(long int)(timeout / 1000000),
|
|
|
|
+ (long int)(timeout % 1000000)};
|
|
|
|
+ int resultsize = UA_select(channel->sockfd+1, &fdset, NULL, NULL, &tmptv);
|
|
|
|
+ if(resultsize == 0) {
|
|
|
|
+ message->length = 0;
|
|
|
|
+ return UA_STATUSCODE_GOODNONCRITICALTIMEOUT;
|
|
|
|
+ }
|
|
|
|
+ if(resultsize == -1) {
|
|
|
|
+ message->length = 0;
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Read the current packet on the socket */
|
|
|
|
+ ssize_t dataLen = recvmsg(channel->sockfd, &msg, 0);
|
|
|
|
+ if(dataLen < 0) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection receive failed. Receive message failed.");
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+ if((size_t)dataLen < sizeof(eth_hdr)) {
|
|
|
|
+ UA_LOG_ERROR(UA_Log_Stdout, UA_LOGCATEGORY_SERVER,
|
|
|
|
+ "PubSub connection receive failed. Packet too small.");
|
|
|
|
+ return UA_STATUSCODE_BADINTERNALERROR;
|
|
|
|
+ }
|
|
|
|
+ if(dataLen == 0)
|
|
|
|
+ return UA_STATUSCODE_GOODNODATA;
|
|
|
|
+
|
|
|
|
+ /* Make sure we match our target */
|
|
|
|
+ if(memcmp(eth_hdr.ether_dhost, channelDataEthernet->targetAddress, ETH_ALEN) != 0)
|
|
|
|
+ return UA_STATUSCODE_GOODNODATA;
|
|
|
|
+
|
|
|
|
+ /* Set the message length */
|
|
|
|
+ message->length = (size_t)dataLen - sizeof(eth_hdr);
|
|
|
|
+
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Close channel and free the channel data.
|
|
|
|
+ *
|
|
|
|
+ * @return UA_STATUSCODE_GOOD if success
|
|
|
|
+ */
|
|
|
|
+static UA_StatusCode
|
|
|
|
+UA_PubSubChannelEthernet_close(UA_PubSubChannel *channel) {
|
|
|
|
+ UA_close(channel->sockfd);
|
|
|
|
+ UA_free(channel->handle);
|
|
|
|
+ UA_free(channel);
|
|
|
|
+ return UA_STATUSCODE_GOOD;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Generate a new channel. based on the given configuration.
|
|
|
|
+ *
|
|
|
|
+ * @param connectionConfig connection configuration
|
|
|
|
+ * @return ref to created channel, NULL on error
|
|
|
|
+ */
|
|
|
|
+static UA_PubSubChannel *
|
|
|
|
+TransportLayerEthernet_addChannel(UA_PubSubConnectionConfig *connectionConfig) {
|
|
|
|
+ UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_USERLAND, "PubSub channel requested");
|
|
|
|
+ UA_PubSubChannel * pubSubChannel = UA_PubSubChannelEthernet_open(connectionConfig);
|
|
|
|
+ if(pubSubChannel) {
|
|
|
|
+ pubSubChannel->regist = UA_PubSubChannelEthernet_regist;
|
|
|
|
+ pubSubChannel->unregist = UA_PubSubChannelEthernet_unregist;
|
|
|
|
+ pubSubChannel->send = UA_PubSubChannelEthernet_send;
|
|
|
|
+ pubSubChannel->receive = UA_PubSubChannelEthernet_receive;
|
|
|
|
+ pubSubChannel->close = UA_PubSubChannelEthernet_close;
|
|
|
|
+ pubSubChannel->connectionConfig = connectionConfig;
|
|
|
|
+ }
|
|
|
|
+ return pubSubChannel;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+UA_PubSubTransportLayer
|
|
|
|
+UA_PubSubTransportLayerEthernet() {
|
|
|
|
+ UA_PubSubTransportLayer pubSubTransportLayer;
|
|
|
|
+ pubSubTransportLayer.transportProfileUri =
|
|
|
|
+ UA_STRING("http://opcfoundation.org/UA-Profile/Transport/pubsub-eth-uadp");
|
|
|
|
+ pubSubTransportLayer.createPubSubChannel = &TransportLayerEthernet_addChannel;
|
|
|
|
+ return pubSubTransportLayer;
|
|
|
|
+}
|