123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206 |
- /* This work is licensed under a Creative Commons CCZero 1.0 Universal License.
- * See http://creativecommons.org/publicdomain/zero/1.0/ for more information. */
- #include <open62541/plugin/log_stdout.h>
- #include <open62541/plugin/pubsub_udp.h>
- #include <open62541/server.h>
- #include <open62541/server_config_default.h>
- #include <signal.h>
- #include <stdlib.h>
- #include <open62541/server_pubsub.h>
- /* possible options: PUBSUB_CONFIG_FASTPATH_NONE, PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS, PUBSUB_CONFIG_FASTPATH_STATIC_VALUES */
- #define PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS
- #define PUBSUB_CONFIG_PUBLISH_CYCLE_MS 100
- #define PUBSUB_CONFIG_PUBLISH_CYCLES 100
- #define PUBSUB_CONFIG_FIELD_COUNT 10
- /**
- * The PubSub RT level example points out the configuration of different PubSub RT-levels. These levels will be
- * used together with an RT capable OS for deterministic message generation. The main target is to reduce the time
- * spread and effort during the publish cycle. Most of the RT levels are based on pre-generated and buffered
- * DataSetMesseges and NetworkMessages. Since changes in the PubSub-configuration will invalidate the buffered
- * frames, the PubSub configuration must be frozen after the configuration phase.
- *
- * This example can be configured to compare and measure the different PubSub options by the following define options:
- * PUBSUB_CONFIG_FASTPATH_NONE -> The published fields are added to the information model and published in the default non-RT mode
- * PUBSUB_CONFIG_FASTPATH_STATIC_VALUES -> The published fields are not visible in the information model. The publish cycle is improved
- * by prevent the value lookup within the information model.
- * PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS -> The published fields are not visible in the information model. After the PubSub-configuration
- * freeze, the NetworkMessages and DataSetMessages will be calculated and buffered. During the publish cycle these buffers will only be updated.
- */
- UA_NodeId publishedDataSetIdent, dataSetFieldIdent, writerGroupIdent, connectionIdentifier;
- UA_UInt32 *valueStore[PUBSUB_CONFIG_FIELD_COUNT];
- UA_Boolean running = true;
- static void stopHandler(int sign) {
- UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "received ctrl-c");
- running = false;
- }
- /* The following PubSub configuration does not differ from the 'normal' configuration */
- static void
- addMinimalPubSubConfiguration(UA_Server * server){
- /* Add one PubSubConnection */
- UA_PubSubConnectionConfig connectionConfig;
- memset(&connectionConfig, 0, sizeof(connectionConfig));
- connectionConfig.name = UA_STRING("UDP-UADP Connection 1");
- connectionConfig.transportProfileUri = UA_STRING("http://opcfoundation.org/UA-Profile/Transport/pubsub-udp-uadp");
- connectionConfig.enabled = UA_TRUE;
- UA_NetworkAddressUrlDataType networkAddressUrl = {UA_STRING_NULL , UA_STRING("opc.udp://224.0.0.22:4840/")};
- UA_Variant_setScalar(&connectionConfig.address, &networkAddressUrl, &UA_TYPES[UA_TYPES_NETWORKADDRESSURLDATATYPE]);
- connectionConfig.publisherId.numeric = UA_UInt32_random();
- UA_Server_addPubSubConnection(server, &connectionConfig, &connectionIdentifier);
- /* Add one PublishedDataSet */
- UA_PublishedDataSetConfig publishedDataSetConfig;
- memset(&publishedDataSetConfig, 0, sizeof(UA_PublishedDataSetConfig));
- publishedDataSetConfig.publishedDataSetType = UA_PUBSUB_DATASET_PUBLISHEDITEMS;
- publishedDataSetConfig.name = UA_STRING("Demo PDS");
- /* Add one DataSetField to the PDS */
- UA_Server_addPublishedDataSet(server, &publishedDataSetConfig, &publishedDataSetIdent);
- }
- static void
- valueUpdateCallback(UA_Server *server, void *data) {
- #if defined PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS || defined PUBSUB_CONFIG_FASTPATH_STATIC_VALUES
- for (int i = 0; i < PUBSUB_CONFIG_FIELD_COUNT; ++i)
- *valueStore[i] = *valueStore[i] + 1;
- if(*valueStore[0] > PUBSUB_CONFIG_PUBLISH_CYCLES)
- running = false;
- #endif
- #if defined PUBSUB_CONFIG_FASTPATH_NONE
- for(size_t i = 0; i < PUBSUB_CONFIG_FIELD_COUNT; i++){
- UA_Variant value;
- UA_Variant_init(&value);
- if(UA_Server_readValue(server, UA_NODEID_NUMERIC(1, 1000 + (UA_UInt32) i), &value) != UA_STATUSCODE_GOOD) {
- UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "Failed to read publish value. Node number: %zu", i);
- continue;
- }
- UA_UInt32 *intValue = (UA_UInt32 *) value.data;
- *intValue = *intValue + 1;
- if(UA_Server_writeValue(server, UA_NODEID_NUMERIC(1, 1000 + (UA_UInt32) i), value) != UA_STATUSCODE_GOOD){
- UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "Failed to update publish value. Node number: %zu", i);
- continue;
- }
- if(i == 0 && *intValue > PUBSUB_CONFIG_PUBLISH_CYCLES)
- running = false;
- UA_Variant_deleteMembers(&value);
- }
- #endif
- }
- int main(void) {
- signal(SIGINT, stopHandler);
- signal(SIGTERM, stopHandler);
- UA_Server *server = UA_Server_new();
- UA_ServerConfig *config = UA_Server_getConfig(server);
- UA_ServerConfig_setDefault(config);
- config->pubsubTransportLayers = (UA_PubSubTransportLayer *) UA_malloc(sizeof(UA_PubSubTransportLayer));
- if(!config->pubsubTransportLayers) {
- UA_Server_delete(server);
- return -1;
- }
- config->pubsubTransportLayers[0] = UA_PubSubTransportLayerUDPMP();
- config->pubsubTransportLayersSize++;
- /*Add standard PubSub configuration (no difference to the std. configuration)*/
- addMinimalPubSubConfiguration(server);
- UA_WriterGroupConfig writerGroupConfig;
- memset(&writerGroupConfig, 0, sizeof(UA_WriterGroupConfig));
- writerGroupConfig.name = UA_STRING("Demo WriterGroup");
- writerGroupConfig.publishingInterval = PUBSUB_CONFIG_PUBLISH_CYCLE_MS;
- writerGroupConfig.enabled = UA_FALSE;
- writerGroupConfig.writerGroupId = 100;
- writerGroupConfig.encodingMimeType = UA_PUBSUB_ENCODING_UADP;
- writerGroupConfig.messageSettings.encoding = UA_EXTENSIONOBJECT_DECODED;
- writerGroupConfig.messageSettings.content.decoded.type = &UA_TYPES[UA_TYPES_UADPWRITERGROUPMESSAGEDATATYPE];
- UA_UadpWriterGroupMessageDataType writerGroupMessage;
- UA_UadpWriterGroupMessageDataType_init(&writerGroupMessage);
- /* Change message settings of writerGroup to send PublisherId,
- * WriterGroupId in GroupHeader and DataSetWriterId in PayloadHeader
- * of NetworkMessage */
- writerGroupMessage.networkMessageContentMask = (UA_UadpNetworkMessageContentMask) ((UA_UadpNetworkMessageContentMask) UA_UADPNETWORKMESSAGECONTENTMASK_PUBLISHERID |
- (UA_UadpNetworkMessageContentMask) UA_UADPNETWORKMESSAGECONTENTMASK_GROUPHEADER |
- (UA_UadpNetworkMessageContentMask) UA_UADPNETWORKMESSAGECONTENTMASK_WRITERGROUPID |
- (UA_UadpNetworkMessageContentMask) UA_UADPNETWORKMESSAGECONTENTMASK_PAYLOADHEADER);
- writerGroupConfig.messageSettings.content.decoded.data = &writerGroupMessage;
- #ifdef PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS
- writerGroupConfig.rtLevel = UA_PUBSUB_RT_FIXED_SIZE;
- #elif defined PUBSUB_CONFIG_FASTPATH_STATIC_VALUES
- writerGroupConfig.rtLevel = UA_PUBSUB_RT_DIRECT_VALUE_ACCESS;
- #endif
- UA_Server_addWriterGroup(server, connectionIdentifier, &writerGroupConfig, &writerGroupIdent);
- /* Add one DataSetWriter */
- UA_NodeId dataSetWriterIdent;
- UA_DataSetWriterConfig dataSetWriterConfig;
- memset(&dataSetWriterConfig, 0, sizeof(UA_DataSetWriterConfig));
- dataSetWriterConfig.name = UA_STRING("Demo DataSetWriter");
- dataSetWriterConfig.dataSetWriterId = 62541;
- dataSetWriterConfig.keyFrameCount = 10;
- UA_Server_addDataSetWriter(server, writerGroupIdent, publishedDataSetIdent, &dataSetWriterConfig, &dataSetWriterIdent);
- #if defined PUBSUB_CONFIG_FASTPATH_FIXED_OFFSETS || defined PUBSUB_CONFIG_FASTPATH_STATIC_VALUES
- /* Add one DataSetField with static value source to PDS */
- UA_DataSetFieldConfig dsfConfig;
- for(size_t i = 0; i < PUBSUB_CONFIG_FIELD_COUNT; i++){
- memset(&dsfConfig, 0, sizeof(UA_DataSetFieldConfig));
- /* Create Variant and configure as DataSetField source */
- UA_UInt32 *intValue = UA_UInt32_new();
- *intValue = (UA_UInt32) i * 1000;
- valueStore[i] = intValue;
- UA_Variant variant;
- memset(&variant, 0, sizeof(UA_Variant));
- UA_Variant_setScalar(&variant, intValue, &UA_TYPES[UA_TYPES_UINT32]);
- UA_DataValue staticValueSource;
- memset(&staticValueSource, 0, sizeof(staticValueSource));
- staticValueSource.value = variant;
- dsfConfig.field.variable.staticValueSourceEnabled = UA_TRUE;
- dsfConfig.field.variable.staticValueSource.value = variant;
- UA_Server_addDataSetField(server, publishedDataSetIdent, &dsfConfig, &dataSetFieldIdent);
- }
- #endif
- #if defined PUBSUB_CONFIG_FASTPATH_NONE
- UA_DataSetFieldConfig dsfConfig;
- memset(&dsfConfig, 0, sizeof(UA_DataSetFieldConfig));
- for(size_t i = 0; i < PUBSUB_CONFIG_FIELD_COUNT; i++){
- UA_VariableAttributes attributes = UA_VariableAttributes_default;
- UA_Variant value;
- UA_Variant_init(&value);
- UA_UInt32 intValue = (UA_UInt32) i * 1000;
- UA_Variant_setScalar(&value, &intValue, &UA_TYPES[UA_TYPES_UINT32]);
- attributes.value = value;
- if(UA_Server_addVariableNode(server, UA_NODEID_NUMERIC(1, 1000 + (UA_UInt32) i),
- UA_NODEID_NUMERIC(0, UA_NS0ID_OBJECTSFOLDER), UA_NODEID_NUMERIC(0, UA_NS0ID_ORGANIZES),
- UA_QUALIFIEDNAME(1, "variable"), UA_NODEID_NUMERIC(0, UA_NS0ID_BASEDATAVARIABLETYPE),
- attributes, NULL, NULL) != UA_STATUSCODE_GOOD){
- UA_LOG_INFO(UA_Log_Stdout, UA_LOGCATEGORY_SERVER, "Failed to add Publish-Node to the server. Node number: %zu", i);
- continue;
- }
- dsfConfig.field.variable.publishParameters.publishedVariable = UA_NODEID_NUMERIC(1, 1000 + (UA_UInt32) i);
- dsfConfig.field.variable.publishParameters.attributeId = UA_ATTRIBUTEID_VALUE;
- UA_Server_addDataSetField(server, publishedDataSetIdent, &dsfConfig, &dataSetFieldIdent);
- }
- #endif
- /* The PubSub configuration is currently editable and the publish callback is not running */
- writerGroupConfig.publishingInterval = PUBSUB_CONFIG_PUBLISH_CYCLE_MS;
- UA_Server_updateWriterGroupConfig(server, writerGroupIdent, &writerGroupConfig);
- /* Freeze the PubSub configuration (and start implicitly the publish callback) */
- UA_Server_freezeWriterGroupConfiguration(server, writerGroupIdent);
- UA_Server_setWriterGroupOperational(server, writerGroupIdent);
- UA_UInt64 callbackId;
- UA_Server_addRepeatedCallback(server, valueUpdateCallback, NULL, PUBSUB_CONFIG_PUBLISH_CYCLE_MS, &callbackId);
- UA_StatusCode retval = UA_STATUSCODE_GOOD;
- retval |= UA_Server_run(server, &running);
- UA_Server_delete(server);
- return retval == UA_STATUSCODE_GOOD ? EXIT_SUCCESS : EXIT_FAILURE;
- }
|