ua_services_view.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. #include "ua_services.h"
  2. #include "ua_statuscodes.h"
  3. #include "ua_nodestoreExample.h"
  4. #include "ua_namespace_0.h"
  5. #include "ua_util.h"
  6. #include "ua_namespace_manager.h"
  7. UA_Int32 Service_Browse_getReferenceDescription(UA_NodeStoreExample *ns, UA_ReferenceNode *reference,
  8. UA_UInt32 nodeClassMask, UA_UInt32 resultMask,
  9. UA_ReferenceDescription *referenceDescription) {
  10. const UA_Node *foundNode;
  11. if(UA_NodeStoreExample_get(ns, &reference->targetId.nodeId, &foundNode) != UA_STATUSCODE_GOOD)
  12. return UA_STATUSCODE_BADINTERNALERROR;
  13. UA_NodeId_copy(&foundNode->nodeId, &referenceDescription->nodeId.nodeId);
  14. //TODO ExpandedNodeId is a mockup
  15. referenceDescription->nodeId.serverIndex = 0;
  16. referenceDescription->nodeId.namespaceUri.length = -1;
  17. /* UA_UInt32 mask = 0; */
  18. /* for(mask = 0x01;mask <= 0x40;mask *= 2) { */
  19. /* switch(mask & (resultMask)) { */
  20. if(resultMask & UA_BROWSERESULTMASK_REFERENCETYPEID)
  21. UA_NodeId_copy(&reference->referenceTypeId, &referenceDescription->referenceTypeId);
  22. if(resultMask & UA_BROWSERESULTMASK_ISFORWARD)
  23. referenceDescription->isForward = !reference->isInverse;
  24. if(resultMask & UA_BROWSERESULTMASK_NODECLASS)
  25. UA_NodeClass_copy(&foundNode->nodeClass, &referenceDescription->nodeClass);
  26. if(resultMask & UA_BROWSERESULTMASK_BROWSENAME)
  27. UA_QualifiedName_copy(&foundNode->browseName, &referenceDescription->browseName);
  28. if(resultMask & UA_BROWSERESULTMASK_DISPLAYNAME)
  29. UA_LocalizedText_copy(&foundNode->displayName, &referenceDescription->displayName);
  30. if(resultMask & UA_BROWSERESULTMASK_TYPEDEFINITION) {
  31. if(foundNode->nodeClass != UA_NODECLASS_OBJECT &&
  32. foundNode->nodeClass != UA_NODECLASS_VARIABLE)
  33. goto end;
  34. for(UA_Int32 i = 0;i < foundNode->referencesSize;i++) {
  35. UA_ReferenceNode *ref = &foundNode->references[i];
  36. if(ref->referenceTypeId.identifier.numeric == 40 /* hastypedefinition */) {
  37. UA_ExpandedNodeId_copy(&ref->targetId, &referenceDescription->typeDefinition);
  38. goto end;
  39. }
  40. }
  41. }
  42. end:
  43. UA_NodeStoreExample_releaseManagedNode(foundNode);
  44. return UA_STATUSCODE_GOOD;
  45. }
  46. /* singly-linked list */
  47. struct SubRefTypeId {
  48. UA_NodeId id;
  49. SLIST_ENTRY(SubRefTypeId) next;
  50. };
  51. SLIST_HEAD(SubRefTypeIdList, SubRefTypeId);
  52. static UA_UInt32 walkReferenceTree(UA_NodeStoreExample *ns, const UA_ReferenceTypeNode *current,
  53. struct SubRefTypeIdList *list) {
  54. // insert the current referencetype
  55. struct SubRefTypeId *element = UA_alloc(sizeof(struct SubRefTypeId));
  56. element->id = current->nodeId;
  57. SLIST_INSERT_HEAD(list, element, next);
  58. UA_UInt32 count = 1; // the current element
  59. // walk the tree
  60. for(UA_Int32 i = 0;i < current->referencesSize;i++) {
  61. if(current->references[i].referenceTypeId.identifier.numeric == 45 /* HasSubtype */ &&
  62. current->references[i].isInverse == UA_FALSE) {
  63. const UA_Node *node;
  64. if(UA_NodeStoreExample_get(ns, &current->references[i].targetId.nodeId, &node) == UA_STATUSCODE_GOOD
  65. && node->nodeClass == UA_NODECLASS_REFERENCETYPE) {
  66. count += walkReferenceTree(ns, (UA_ReferenceTypeNode *)node, list);
  67. UA_NodeStoreExample_releaseManagedNode(node);
  68. }
  69. }
  70. }
  71. return count;
  72. }
  73. /* We do not search across namespaces so far. The id of the father-referencetype is returned in the array also. */
  74. static UA_Int32 findSubReferenceTypes(UA_NodeStoreExample *ns, UA_NodeId *rootReferenceType,
  75. UA_NodeId **ids, UA_UInt32 *idcount) {
  76. struct SubRefTypeIdList list;
  77. UA_UInt32 count;
  78. SLIST_INIT(&list);
  79. // walk the tree
  80. const UA_ReferenceTypeNode *root;
  81. if(UA_NodeStoreExample_get(ns, rootReferenceType, (const UA_Node **)&root) != UA_STATUSCODE_GOOD ||
  82. root->nodeClass != UA_NODECLASS_REFERENCETYPE)
  83. return UA_STATUSCODE_BADINTERNALERROR;
  84. count = walkReferenceTree(ns, root, &list);
  85. UA_NodeStoreExample_releaseManagedNode((const UA_Node *)root);
  86. // copy results into an array
  87. *ids = UA_alloc(sizeof(UA_NodeId)*count);
  88. for(UA_UInt32 i = 0;i < count;i++) {
  89. struct SubRefTypeId *element = SLIST_FIRST(&list);
  90. UA_NodeId_copy(&element->id, &(*ids)[i]);
  91. SLIST_REMOVE_HEAD(&list, next);
  92. UA_free(element);
  93. }
  94. *idcount = count;
  95. return UA_STATUSCODE_GOOD;
  96. }
  97. /* is this a relevant reference? */
  98. static INLINE UA_Boolean Service_Browse_returnReference(UA_BrowseDescription *browseDescription,
  99. UA_ReferenceNode *reference,
  100. UA_NodeId *relevantRefTypes,
  101. UA_UInt32 relevantRefTypesCount) {
  102. if(reference->isInverse == UA_TRUE &&
  103. browseDescription->browseDirection == UA_BROWSEDIRECTION_FORWARD)
  104. return UA_FALSE;
  105. else if(reference->isInverse == UA_FALSE &&
  106. browseDescription->browseDirection == UA_BROWSEDIRECTION_INVERSE)
  107. return UA_FALSE;
  108. for(UA_UInt32 i = 0;i < relevantRefTypesCount;i++) {
  109. if(UA_NodeId_equal(&browseDescription->referenceTypeId, &relevantRefTypes[i]) == UA_EQUAL)
  110. return UA_TRUE;
  111. }
  112. return UA_FALSE;
  113. }
  114. /* Return results to a single browsedescription. */
  115. static void Service_Browse_getBrowseResult(UA_NodeStoreExample *ns,
  116. UA_BrowseDescription *browseDescription,
  117. UA_UInt32 maxReferences,
  118. UA_BrowseResult *browseResult) {
  119. const UA_Node *node;
  120. UA_NodeId *relevantReferenceTypes = UA_NULL;
  121. UA_UInt32 relevantReferenceTypesCount = 0;
  122. if(UA_NodeStoreExample_get(ns, &browseDescription->nodeId, &node) != UA_STATUSCODE_GOOD) {
  123. browseResult->statusCode = UA_STATUSCODE_BADNODEIDUNKNOWN;
  124. return;
  125. }
  126. // 0 => unlimited references
  127. if(maxReferences == 0)
  128. maxReferences = node->referencesSize;
  129. // discover the relevant subtypes
  130. if(!browseDescription->includeSubtypes ||
  131. findSubReferenceTypes(ns, &browseDescription->referenceTypeId, &relevantReferenceTypes,
  132. &relevantReferenceTypesCount) != UA_STATUSCODE_GOOD) {
  133. if(!(relevantReferenceTypes = UA_alloc(sizeof(UA_NodeId)))) {
  134. return;
  135. }
  136. UA_NodeId_copy(&browseDescription->referenceTypeId, relevantReferenceTypes);
  137. relevantReferenceTypesCount = 1;
  138. }
  139. /* We do not use a linked list but traverse the nodes references list twice
  140. * (once for counting, once for generating the referencedescriptions). That
  141. * is much faster than using a linked list, since the references are
  142. * allocated in a continuous blob and RAM access is predictible/does not
  143. * miss cache lines so often. TODO: measure with some huge objects! */
  144. UA_UInt32 refs = 0;
  145. for(UA_Int32 i = 0;i < node->referencesSize && refs <= maxReferences;i++) {
  146. if(Service_Browse_returnReference(browseDescription, &node->references[i], relevantReferenceTypes,
  147. relevantReferenceTypesCount))
  148. refs++;
  149. }
  150. // can we return all relevant references at once?
  151. UA_Boolean finished = UA_TRUE;
  152. if(refs > maxReferences) {
  153. refs--;
  154. finished = UA_FALSE;
  155. }
  156. browseResult->referencesSize = refs;
  157. UA_Array_new((void **)&browseResult->references, refs, &UA_[UA_REFERENCEDESCRIPTION]);
  158. for(UA_UInt32 i = 0, j = 0;j < refs;i++) {
  159. if(!Service_Browse_returnReference(browseDescription, &node->references[i], relevantReferenceTypes,
  160. relevantReferenceTypesCount))
  161. continue;
  162. if(Service_Browse_getReferenceDescription(ns, &node->references[i], browseDescription->nodeClassMask,
  163. browseDescription->resultMask, &browseResult->references[j]) != UA_STATUSCODE_GOOD)
  164. browseResult->statusCode = UA_STATUSCODE_UNCERTAINNOTALLNODESAVAILABLE;
  165. j++;
  166. }
  167. if(!finished) {
  168. // Todo. Set the Statuscode and the continuation point.
  169. }
  170. UA_NodeStoreExample_releaseManagedNode(node);
  171. UA_Array_delete(relevantReferenceTypes, relevantReferenceTypesCount, &UA_[UA_NODEID]);
  172. }
  173. void Service_Browse(UA_Server *server, UA_Session *session,
  174. const UA_BrowseRequest *request, UA_BrowseResponse *response) {
  175. UA_assert(server != UA_NULL && session != UA_NULL && request != UA_NULL && response != UA_NULL);
  176. if(request->nodesToBrowseSize <= 0) {
  177. response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
  178. return;
  179. }
  180. if(UA_Array_new((void **)&(response->results), request->nodesToBrowseSize, &UA_[UA_BROWSERESULT])
  181. != UA_STATUSCODE_GOOD) {
  182. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  183. return;
  184. }
  185. response->resultsSize = request->nodesToBrowseSize;
  186. UA_Int32 *numberOfFoundIndices;
  187. UA_UInt16 *associatedIndices;
  188. UA_UInt32 differentNamespaceIndexCount = 0;
  189. if(UA_Array_new((void **)&numberOfFoundIndices,request->nodesToBrowseSize,&UA_[UA_UINT32]) != UA_STATUSCODE_GOOD){
  190. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  191. return ;
  192. }
  193. if(UA_Array_new((void **)&associatedIndices,request->nodesToBrowseSize,&UA_[UA_UINT16]) != UA_STATUSCODE_GOOD){
  194. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  195. return ;
  196. }
  197. // find out count of different namespace indices
  198. for(UA_Int32 i = 0; i<request->nodesToBrowseSize; i++){
  199. for(UA_UInt32 j = 0; j <= differentNamespaceIndexCount; j++){
  200. if(associatedIndices[j] == request->nodesToBrowse[i].nodeId.namespaceIndex){
  201. if(j==0){
  202. differentNamespaceIndexCount++;
  203. }
  204. numberOfFoundIndices[j]++;
  205. break;
  206. }
  207. else if(j == (differentNamespaceIndexCount - 1)){
  208. associatedIndices[j] = request->nodesToBrowse[i].nodeId.namespaceIndex;
  209. associatedIndices[j] = 1;
  210. differentNamespaceIndexCount++;
  211. }
  212. }
  213. }
  214. UA_UInt32 *browseDescriptionIndices;
  215. if(UA_Array_new((void **)&browseDescriptionIndices,request->nodesToBrowseSize,&UA_[UA_UINT32]) != UA_STATUSCODE_GOOD){
  216. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  217. return ;
  218. }
  219. for(UA_UInt32 i = 0; i < differentNamespaceIndexCount; i++){
  220. UA_Namespace *tmpNamespace;
  221. UA_NamespaceManager_getNamespace(server->namespaceManager,associatedIndices[i],&tmpNamespace);
  222. if(tmpNamespace != UA_NULL){
  223. //build up index array for each read operation onto a different namespace
  224. UA_UInt32 n = 0;
  225. for(UA_Int32 j = 0; j < request->nodesToBrowseSize; j++){
  226. if(request->nodesToBrowse[j].nodeId.namespaceIndex == associatedIndices[i]){
  227. browseDescriptionIndices[n] = j;
  228. }
  229. }
  230. //call read for every namespace
  231. tmpNamespace->nodeStore->browseNodes(
  232. request->nodesToBrowse,
  233. browseDescriptionIndices,
  234. numberOfFoundIndices[i],
  235. request->requestedMaxReferencesPerNode,
  236. response->results,
  237. response->diagnosticInfos);
  238. // response->results[i] = service_read_node(server, &request->nodesToRead[i]);
  239. }
  240. }
  241. UA_free(browseDescriptionIndices);
  242. UA_free(numberOfFoundIndices);
  243. UA_free(associatedIndices);
  244. // for(UA_Int32 i = 0;i < request->nodesToBrowseSize;i++)
  245. // Service_Browse_getBrowseResult(server->nodestore, &request->nodesToBrowse[i],
  246. // request->requestedMaxReferencesPerNode, &response->results[i]);
  247. }
  248. void Service_TranslateBrowsePathsToNodeIds(UA_Server *server, UA_Session *session,
  249. const UA_TranslateBrowsePathsToNodeIdsRequest *request,
  250. UA_TranslateBrowsePathsToNodeIdsResponse *response) {
  251. UA_assert(server != UA_NULL && session != UA_NULL && request != UA_NULL && response != UA_NULL);
  252. if(request->browsePathsSize <= 0) {
  253. response->responseHeader.serviceResult = UA_STATUSCODE_BADNOTHINGTODO;
  254. return;
  255. }
  256. // Allocate space for a correct answer
  257. response->resultsSize = request->browsePathsSize;
  258. // _init of the elements is done in Array_new
  259. if(UA_Array_new((void **)&response->results, request->browsePathsSize, &UA_[UA_BROWSEPATHRESULT])
  260. != UA_STATUSCODE_GOOD) {
  261. response->responseHeader.serviceResult = UA_STATUSCODE_BADOUTOFMEMORY;
  262. return;
  263. }
  264. for(UA_Int32 i = 0;i < request->browsePathsSize;i++)
  265. response->results[i].statusCode = UA_STATUSCODE_BADNOMATCH; //FIXME: implement
  266. }