123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314 |
- #include "ua_services.h"
- #include "ua_statuscodes.h"
- #include "ua_nodestoreExample.h"
- #include "ua_namespace_0.h"
- #include "ua_util.h"
- #include "ua_namespace_manager.h"
- UA_Int32 Service_Browse_getReferenceDescription(UA_NodeStoreExample *ns, UA_ReferenceNode *reference,
- UA_UInt32 nodeClassMask, UA_UInt32 resultMask,
- UA_ReferenceDescription *referenceDescription) {
- const UA_Node *foundNode;
- if(UA_NodeStoreExample_get(ns, &reference->targetId.nodeId, &foundNode) != UA_STATUSCODE_GOOD)
- return UA_STATUSCODE_BADINTERNALERROR;
- UA_NodeId_copy(&foundNode->nodeId, &referenceDescription->nodeId.nodeId);
- //TODO ExpandedNodeId is a mockup
- referenceDescription->nodeId.serverIndex = 0;
- referenceDescription->nodeId.namespaceUri.length = -1;
- /* UA_UInt32 mask = 0; */
- /* for(mask = 0x01;mask <= 0x40;mask *= 2) { */
- /* switch(mask & (resultMask)) { */
- if(resultMask & UA_BROWSERESULTMASK_REFERENCETYPEID)
- UA_NodeId_copy(&reference->referenceTypeId, &referenceDescription->referenceTypeId);
- if(resultMask & UA_BROWSERESULTMASK_ISFORWARD)
- referenceDescription->isForward = !reference->isInverse;
- if(resultMask & UA_BROWSERESULTMASK_NODECLASS)
- UA_NodeClass_copy(&foundNode->nodeClass, &referenceDescription->nodeClass);
- if(resultMask & UA_BROWSERESULTMASK_BROWSENAME)
- UA_QualifiedName_copy(&foundNode->browseName, &referenceDescription->browseName);
- if(resultMask & UA_BROWSERESULTMASK_DISPLAYNAME)
- UA_LocalizedText_copy(&foundNode->displayName, &referenceDescription->displayName);
- if(resultMask & UA_BROWSERESULTMASK_TYPEDEFINITION) {
- if(foundNode->nodeClass != UA_NODECLASS_OBJECT &&
- foundNode->nodeClass != UA_NODECLASS_VARIABLE)
- goto end;
-
- for(UA_Int32 i = 0;i < foundNode->referencesSize;i++) {
- UA_ReferenceNode *ref = &foundNode->references[i];
- if(ref->referenceTypeId.identifier.numeric == 40 /* hastypedefinition */) {
- UA_ExpandedNodeId_copy(&ref->targetId, &referenceDescription->typeDefinition);
- goto end;
- }
- }
- }
- end:
- UA_NodeStoreExample_releaseManagedNode(foundNode);
- return UA_STATUSCODE_GOOD;
- }
- /* singly-linked list */
- struct SubRefTypeId {
- UA_NodeId id;
- SLIST_ENTRY(SubRefTypeId) next;
- };
- SLIST_HEAD(SubRefTypeIdList, SubRefTypeId);
- static UA_UInt32 walkReferenceTree(UA_NodeStoreExample *ns, const UA_ReferenceTypeNode *current,
- struct SubRefTypeIdList *list) {
- // insert the current referencetype
- struct SubRefTypeId *element = UA_alloc(sizeof(struct SubRefTypeId));
- element->id = current->nodeId;
- SLIST_INSERT_HEAD(list, element, next);
- UA_UInt32 count = 1; // the current element
- // walk the tree
- for(UA_Int32 i = 0;i < current->referencesSize;i++) {
- if(current->references[i].referenceTypeId.identifier.numeric == 45 /* HasSubtype */ &&
- current->references[i].isInverse == UA_FALSE) {
- const UA_Node *node;
- if(UA_NodeStoreExample_get(ns, ¤t->references[i].targetId.nodeId, &node) == UA_STATUSCODE_GOOD
- && node->nodeClass == UA_NODECLASS_REFERENCETYPE) {
- count += walkReferenceTree(ns, (UA_ReferenceTypeNode *)node, list);
- UA_NodeStoreExample_releaseManagedNode(node);
- }
- }
- }
- return count;
- }
- /* We do not search across namespaces so far. The id of the father-referencetype is returned in the array also. */
- static UA_Int32 findSubReferenceTypes(UA_NodeStoreExample *ns, UA_NodeId *rootReferenceType,
- UA_NodeId **ids, UA_UInt32 *idcount) {
- struct SubRefTypeIdList list;
- UA_UInt32 count;
- SLIST_INIT(&list);
- // walk the tree
- const UA_ReferenceTypeNode *root;
- if(UA_NodeStoreExample_get(ns, rootReferenceType, (const UA_Node **)&root) != UA_STATUSCODE_GOOD ||
- root->nodeClass != UA_NODECLASS_REFERENCETYPE)
- return UA_STATUSCODE_BADINTERNALERROR;
- count = walkReferenceTree(ns, root, &list);
- UA_NodeStoreExample_releaseManagedNode((const UA_Node *)root);
- // copy results into an array
- *ids = UA_alloc(sizeof(UA_NodeId)*count);
- for(UA_UInt32 i = 0;i < count;i++) {
- struct SubRefTypeId *element = SLIST_FIRST(&list);
- UA_NodeId_copy(&element->id, &(*ids)[i]);
- SLIST_REMOVE_HEAD(&list, next);
- UA_free(element);
- }
- *idcount = count;
- return UA_STATUSCODE_GOOD;
- }
- /* is this a relevant reference? */
- static INLINE UA_Boolean Service_Browse_returnReference(UA_BrowseDescription *browseDescription,
- UA_ReferenceNode *reference,
- UA_NodeId *relevantRefTypes,
- UA_UInt32 relevantRefTypesCount) {
- if(reference->isInverse == UA_TRUE &&
- browseDescription->browseDirection == UA_BROWSEDIRECTION_FORWARD)
- return UA_FALSE;
- else if(reference->isInverse == UA_FALSE &&
- browseDescription->browseDirection == UA_BROWSEDIRECTION_INVERSE)
- return UA_FALSE;
- for(UA_UInt32 i = 0;i < relevantRefTypesCount;i++) {
- if(UA_NodeId_equal(&browseDescription->referenceTypeId, &relevantRefTypes[i]) == UA_EQUAL)
- return UA_TRUE;
- }
- return UA_FALSE;
- }
- /* Return results to a single browsedescription. */
- static void Service_Browse_getBrowseResult(UA_NodeStoreExample *ns,
- UA_BrowseDescription *browseDescription,
- UA_UInt32 maxReferences,
- UA_BrowseResult *browseResult) {
- const UA_Node *node;
- UA_NodeId *relevantReferenceTypes = UA_NULL;
- UA_UInt32 relevantReferenceTypesCount = 0;
- if(UA_NodeStoreExample_get(ns, &browseDescription->nodeId, &node) != UA_STATUSCODE_GOOD) {
- browseResult->statusCode = UA_STATUSCODE_BADNODEIDUNKNOWN;
- return;
- }
- // 0 => unlimited references
- if(maxReferences == 0)
- maxReferences = node->referencesSize;
- // discover the relevant subtypes
- if(!browseDescription->includeSubtypes ||
- findSubReferenceTypes(ns, &browseDescription->referenceTypeId, &relevantReferenceTypes,
- &relevantReferenceTypesCount) != UA_STATUSCODE_GOOD) {
- if(!(relevantReferenceTypes = UA_alloc(sizeof(UA_NodeId)))) {
- return;
- }
- UA_NodeId_copy(&browseDescription->referenceTypeId, relevantReferenceTypes);
- relevantReferenceTypesCount = 1;
- }
- /* We do not use a linked list but traverse the nodes references list twice
- * (once for counting, once for generating the referencedescriptions). That
- * is much faster than using a linked list, since the references are
- * allocated in a continuous blob and RAM access is predictible/does not
- * miss cache lines so often. TODO: measure with some huge objects! */
- UA_UInt32 refs = 0;
- for(UA_Int32 i = 0;i < node->referencesSize && refs <= maxReferences;i++) {
- if(Service_Browse_returnReference(browseDescription, &node->references[i], relevantReferenceTypes,
- relevantReferenceTypesCount))
- refs++;
- }
- // can we return all relevant references at once?
- UA_Boolean finished = UA_TRUE;
- if(refs > maxReferences) {
- refs--;
- finished = UA_FALSE;
- }
- browseResult->referencesSize = refs;
- UA_Array_new((void **)&browseResult->references, refs, &UA_[UA_REFERENCEDESCRIPTION]);
- for(UA_UInt32 i = 0, j = 0;j < refs;i++) {
- if(!Service_Browse_returnReference(browseDescription, &node->references[i], relevantReferenceTypes,
- relevantReferenceTypesCount))
- continue;
- if(Service_Browse_getReferenceDescription(ns, &node->references[i], browseDescription->nodeClassMask,
- browseDescription->resultMask, &browseResult->references[j]) != UA_STATUSCODE_GOOD)
- browseResult->statusCode = UA_STATUSCODE_UNCERTAINNOTALLNODESAVAILABLE;
- j++;
- }
- if(!finished) {
- // Todo. Set the Statuscode and the continuation point.
- }
- UA_NodeStoreExample_releaseManagedNode(node);
- UA_Array_delete(relevantReferenceTypes, relevantReferenceTypesCount, &UA_[UA_NODEID]);
- }
- void Service_Browse(UA_Server *server, UA_Session *session,
- const UA_BrowseRequest *request, UA_BrowseResponse *response) {
- UA_assert(server != UA_NULL && session != UA_NULL && request != UA_NULL && response != UA_NULL);
- if(request->nodesToBrowseSize <= 0) {
- response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
- return;
- }
- if(UA_Array_new((void **)&(response->results), request->nodesToBrowseSize, &UA_[UA_BROWSERESULT])
- != UA_STATUSCODE_GOOD) {
- response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
- return;
- }
- response->resultsSize = request->nodesToBrowseSize;
- UA_Int32 *numberOfFoundIndices;
- UA_UInt16 *associatedIndices;
- UA_UInt32 differentNamespaceIndexCount = 0;
- if(UA_Array_new((void **)&numberOfFoundIndices,request->nodesToBrowseSize,&UA_[UA_UINT32]) != UA_STATUSCODE_GOOD){
- response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
- return ;
- }
- if(UA_Array_new((void **)&associatedIndices,request->nodesToBrowseSize,&UA_[UA_UINT16]) != UA_STATUSCODE_GOOD){
- response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
- return ;
- }
- // find out count of different namespace indices
- for(UA_Int32 i = 0; i<request->nodesToBrowseSize; i++){
- for(UA_UInt32 j = 0; j <= differentNamespaceIndexCount; j++){
- if(associatedIndices[j] == request->nodesToBrowse[i].nodeId.namespaceIndex){
- if(j==0){
- differentNamespaceIndexCount++;
- }
- numberOfFoundIndices[j]++;
- break;
- }
- else if(j == (differentNamespaceIndexCount - 1)){
- associatedIndices[j] = request->nodesToBrowse[i].nodeId.namespaceIndex;
- associatedIndices[j] = 1;
- differentNamespaceIndexCount++;
- }
- }
- }
- UA_UInt32 *browseDescriptionIndices;
- if(UA_Array_new((void **)&browseDescriptionIndices,request->nodesToBrowseSize,&UA_[UA_UINT32]) != UA_STATUSCODE_GOOD){
- response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
- return ;
- }
- for(UA_UInt32 i = 0; i < differentNamespaceIndexCount; i++){
- UA_Namespace *tmpNamespace;
- UA_NamespaceManager_getNamespace(server->namespaceManager,associatedIndices[i],&tmpNamespace);
- if(tmpNamespace != UA_NULL){
- //build up index array for each read operation onto a different namespace
- UA_UInt32 n = 0;
- for(UA_Int32 j = 0; j < request->nodesToBrowseSize; j++){
- if(request->nodesToBrowse[j].nodeId.namespaceIndex == associatedIndices[i]){
- browseDescriptionIndices[n] = j;
- }
- }
- //call read for every namespace
- tmpNamespace->nodeStore->browseNodes(
- request->nodesToBrowse,
- browseDescriptionIndices,
- numberOfFoundIndices[i],
- request->requestedMaxReferencesPerNode,
- response->results,
- response->diagnosticInfos);
- // response->results[i] = service_read_node(server, &request->nodesToRead[i]);
- }
- }
- UA_free(browseDescriptionIndices);
- UA_free(numberOfFoundIndices);
- UA_free(associatedIndices);
- // for(UA_Int32 i = 0;i < request->nodesToBrowseSize;i++)
- // Service_Browse_getBrowseResult(server->nodestore, &request->nodesToBrowse[i],
- // request->requestedMaxReferencesPerNode, &response->results[i]);
- }
- void Service_TranslateBrowsePathsToNodeIds(UA_Server *server, UA_Session *session,
- const UA_TranslateBrowsePathsToNodeIdsRequest *request,
- UA_TranslateBrowsePathsToNodeIdsResponse *response) {
- UA_assert(server != UA_NULL && session != UA_NULL && request != UA_NULL && response != UA_NULL);
- if(request->browsePathsSize <= 0) {
- response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
- return;
- }
- // Allocate space for a correct answer
- response->resultsSize = request->browsePathsSize;
- // _init of the elements is done in Array_new
- if(UA_Array_new((void **)&response->results, request->browsePathsSize, &UA_[UA_BROWSEPATHRESULT])
- != UA_STATUSCODE_GOOD) {
- response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
- return;
- }
- for(UA_Int32 i = 0;i < request->browsePathsSize;i++)
- response->results[i].statusCode = UA_STATUSCODE_BADNOMATCH; //FIXME: implement
- }
|