ua_nodestore_concurrent.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. #define _LGPL_SOURCE
  2. #include <urcu.h>
  3. #include <urcu/compiler.h> // for caa_container_of
  4. #include <urcu/uatomic.h>
  5. #include <urcu/rculfhash.h>
  6. #include "ua_nodestore.h"
  7. #include "ua_util.h"
  8. #define ALIVE_BIT (1 << 15) /* Alive bit in the refcount */
  9. struct nodeEntry {
  10. struct cds_lfht_node htn; /* contains next-ptr for urcu-hashmap */
  11. struct rcu_head rcu_head; /* For call-rcu */
  12. UA_UInt16 refcount; /* Counts the amount of readers on it [alive-bit, 15 counter-bits] */
  13. UA_Node node; /* Might be cast from any _bigger_ UA_Node* type. Allocate enough memory! */
  14. };
  15. struct UA_NodeStore {
  16. struct cds_lfht *ht; /* Hash table */
  17. };
  18. #include "ua_nodestore_hash.inc"
  19. static void node_deleteMembers(UA_Node *node) {
  20. switch(node->nodeClass) {
  21. case UA_NODECLASS_OBJECT:
  22. UA_ObjectNode_deleteMembers((UA_ObjectNode *)node);
  23. break;
  24. case UA_NODECLASS_VARIABLE:
  25. UA_VariableNode_deleteMembers((UA_VariableNode *)node);
  26. break;
  27. case UA_NODECLASS_METHOD:
  28. UA_MethodNode_deleteMembers((UA_MethodNode *)node);
  29. break;
  30. case UA_NODECLASS_OBJECTTYPE:
  31. UA_ObjectTypeNode_deleteMembers((UA_ObjectTypeNode *)node);
  32. break;
  33. case UA_NODECLASS_VARIABLETYPE:
  34. UA_VariableTypeNode_deleteMembers((UA_VariableTypeNode *)node);
  35. break;
  36. case UA_NODECLASS_REFERENCETYPE:
  37. UA_ReferenceTypeNode_deleteMembers((UA_ReferenceTypeNode *)node);
  38. break;
  39. case UA_NODECLASS_DATATYPE:
  40. UA_DataTypeNode_deleteMembers((UA_DataTypeNode *)node);
  41. break;
  42. case UA_NODECLASS_VIEW:
  43. UA_ViewNode_deleteMembers((UA_ViewNode *)node);
  44. break;
  45. default:
  46. UA_assert(UA_FALSE);
  47. break;
  48. }
  49. }
  50. /* We are in a rcu_read lock. So the node will not be freed under our feet. */
  51. static int compare(struct cds_lfht_node *htn, const void *orig) {
  52. const UA_NodeId *origid = (const UA_NodeId *)orig;
  53. const UA_NodeId *newid = &((struct nodeEntry *)htn)->node.nodeId; /* The htn is first in the entry structure. */
  54. return UA_NodeId_equal(newid, origid);
  55. }
  56. /* The entry was removed from the hashtable. No more readers can get it. Since
  57. all readers using the node for a longer time (outside the rcu critical
  58. section) increased the refcount, we only need to wait for the refcount
  59. to reach zero. */
  60. static void markDead(struct rcu_head *head) {
  61. #if defined(__GNUC__) || defined(__clang__)
  62. #pragma GCC diagnostic push
  63. #pragma GCC diagnostic ignored "-Wcast-align"
  64. #endif
  65. struct nodeEntry *entry = caa_container_of(head, struct nodeEntry, rcu_head);
  66. #if defined(__GNUC__) || defined(__clang__)
  67. #pragma GCC diagnostic pop
  68. #endif
  69. uatomic_and(&entry->refcount, ~ALIVE_BIT); // set the alive bit to zero
  70. if(uatomic_read(&entry->refcount) > 0)
  71. return;
  72. node_deleteMembers(&entry->node);
  73. UA_free(entry);
  74. }
  75. /* Free the entry if it is dead and nobody uses it anymore */
  76. void UA_NodeStore_release(const UA_Node *managed) {
  77. #if defined(__GNUC__) || defined(__clang__)
  78. #pragma GCC diagnostic push
  79. #pragma GCC diagnostic ignored "-Wcast-align"
  80. #endif
  81. struct nodeEntry *entry = caa_container_of(managed, struct nodeEntry, node); // pointer to the first entry
  82. #if defined(__GNUC__) || defined(__clang__)
  83. #pragma GCC diagnostic pop
  84. #endif
  85. if(uatomic_add_return(&entry->refcount, -1) > 0)
  86. return;
  87. node_deleteMembers(&entry->node);
  88. UA_free(entry);
  89. }
  90. UA_NodeStore * UA_NodeStore_new() {
  91. UA_NodeStore *ns;
  92. if(!(ns = UA_malloc(sizeof(UA_NodeStore))))
  93. return UA_NULL;
  94. /* 32 is the minimum size for the hashtable. */
  95. ns->ht = cds_lfht_new(32, 32, 0, CDS_LFHT_AUTO_RESIZE, NULL);
  96. if(!ns->ht) {
  97. UA_free(ns);
  98. return UA_NULL;
  99. }
  100. return ns;
  101. }
  102. void UA_NodeStore_delete(UA_NodeStore *ns) {
  103. struct cds_lfht *ht = ns->ht;
  104. struct cds_lfht_iter iter;
  105. rcu_read_lock();
  106. cds_lfht_first(ht, &iter);
  107. while(iter.node) {
  108. if(!cds_lfht_del(ht, iter.node)) {
  109. #if defined(__GNUC__) || defined(__clang__)
  110. #pragma GCC diagnostic push
  111. #pragma GCC diagnostic ignored "-Wcast-align"
  112. #endif
  113. struct nodeEntry *entry = caa_container_of(iter.node, struct nodeEntry, htn);
  114. #if defined(__GNUC__) || defined(__clang__)
  115. #pragma GCC diagnostic pop
  116. #endif
  117. call_rcu(&entry->rcu_head, markDead);
  118. }
  119. cds_lfht_next(ht, &iter);
  120. }
  121. rcu_read_unlock();
  122. cds_lfht_destroy(ht, UA_NULL);
  123. UA_free(ns);
  124. }
  125. UA_StatusCode UA_NodeStore_insert(UA_NodeStore *ns, const UA_Node **node, UA_Boolean getManaged) {
  126. UA_UInt32 nodesize;
  127. /* Copy the node into the entry. Then reset the original node. It shall no longer be used. */
  128. switch((*node)->nodeClass) {
  129. case UA_NODECLASS_OBJECT:
  130. nodesize = sizeof(UA_ObjectNode);
  131. break;
  132. case UA_NODECLASS_VARIABLE:
  133. nodesize = sizeof(UA_VariableNode);
  134. break;
  135. case UA_NODECLASS_METHOD:
  136. nodesize = sizeof(UA_MethodNode);
  137. break;
  138. case UA_NODECLASS_OBJECTTYPE:
  139. nodesize = sizeof(UA_ObjectTypeNode);
  140. break;
  141. case UA_NODECLASS_VARIABLETYPE:
  142. nodesize = sizeof(UA_VariableTypeNode);
  143. break;
  144. case UA_NODECLASS_REFERENCETYPE:
  145. nodesize = sizeof(UA_ReferenceTypeNode);
  146. break;
  147. case UA_NODECLASS_DATATYPE:
  148. nodesize = sizeof(UA_DataTypeNode);
  149. break;
  150. case UA_NODECLASS_VIEW:
  151. nodesize = sizeof(UA_ViewNode);
  152. break;
  153. default:
  154. return UA_STATUSCODE_BADINTERNALERROR;
  155. }
  156. struct nodeEntry *entry;
  157. if(!(entry = UA_malloc(sizeof(struct nodeEntry) - sizeof(UA_Node) + nodesize)))
  158. return UA_STATUSCODE_BADOUTOFMEMORY;
  159. UA_memcpy((void*)&entry->node, *node, nodesize);
  160. cds_lfht_node_init(&entry->htn);
  161. entry->refcount = ALIVE_BIT;
  162. if(getManaged) // increase the counter before adding the node
  163. entry->refcount++;
  164. struct cds_lfht_node *result;
  165. if(!UA_NodeId_isNull(&(*node)->nodeId)) {
  166. hash_t h = hash(&(*node)->nodeId);
  167. rcu_read_lock();
  168. result = cds_lfht_add_unique(ns->ht, h, compare, &entry->node.nodeId, &entry->htn);
  169. rcu_read_unlock();
  170. /* If the nodeid exists already */
  171. if(result != &entry->htn) {
  172. UA_free(entry);
  173. return UA_STATUSCODE_BADNODEIDEXISTS;
  174. }
  175. } else {
  176. /* create a unique nodeid */
  177. ((UA_Node *)&entry->node)->nodeId.identifierType = UA_NODEIDTYPE_NUMERIC;
  178. ((UA_Node *)&entry->node)->nodeId.namespaceIndex = 1; // namespace 1 is always in the local nodestore
  179. unsigned long identifier;
  180. long before, after;
  181. rcu_read_lock();
  182. cds_lfht_count_nodes(ns->ht, &before, &identifier, &after); // current amount of nodes stored
  183. identifier++;
  184. ((UA_Node *)&entry->node)->nodeId.identifier.numeric = identifier;
  185. while(UA_TRUE) {
  186. hash_t nhash = hash(&entry->node.nodeId);
  187. result = cds_lfht_add_unique(ns->ht, nhash, compare, &entry->node.nodeId, &entry->htn);
  188. if(result == &entry->htn)
  189. break;
  190. ((UA_Node *)&entry->node)->nodeId.identifier.numeric += (identifier * 2654435761);
  191. }
  192. rcu_read_unlock();
  193. }
  194. #if defined(__GNUC__) || defined(__clang__)
  195. #pragma GCC diagnostic push
  196. #pragma GCC diagnostic ignored "-Wcast-qual"
  197. #endif
  198. UA_free((UA_Node *)*node); /* The old node is replaced by a managed node. */
  199. #if defined(__GNUC__) || defined(__clang__)
  200. #pragma GCC diagnostic pop
  201. #endif
  202. if(getManaged)
  203. *node = &entry->node;
  204. else
  205. *node = UA_NULL;
  206. return UA_STATUSCODE_GOOD;
  207. }
  208. UA_StatusCode UA_NodeStore_replace(UA_NodeStore *ns, const UA_Node *oldNode,
  209. const UA_Node **node, UA_Boolean getManaged) {
  210. UA_UInt32 nodesize;
  211. /* Copy the node into the entry. Then reset the original node. It shall no longer be used. */
  212. switch((*node)->nodeClass) {
  213. case UA_NODECLASS_OBJECT:
  214. nodesize = sizeof(UA_ObjectNode);
  215. break;
  216. case UA_NODECLASS_VARIABLE:
  217. nodesize = sizeof(UA_VariableNode);
  218. break;
  219. case UA_NODECLASS_METHOD:
  220. nodesize = sizeof(UA_MethodNode);
  221. break;
  222. case UA_NODECLASS_OBJECTTYPE:
  223. nodesize = sizeof(UA_ObjectTypeNode);
  224. break;
  225. case UA_NODECLASS_VARIABLETYPE:
  226. nodesize = sizeof(UA_VariableTypeNode);
  227. break;
  228. case UA_NODECLASS_REFERENCETYPE:
  229. nodesize = sizeof(UA_ReferenceTypeNode);
  230. break;
  231. case UA_NODECLASS_DATATYPE:
  232. nodesize = sizeof(UA_DataTypeNode);
  233. break;
  234. case UA_NODECLASS_VIEW:
  235. nodesize = sizeof(UA_ViewNode);
  236. break;
  237. default:
  238. return UA_STATUSCODE_BADINTERNALERROR;
  239. }
  240. struct nodeEntry *newEntry;
  241. if(!(newEntry = UA_malloc(sizeof(struct nodeEntry) - sizeof(UA_Node) + nodesize)))
  242. return UA_STATUSCODE_BADOUTOFMEMORY;
  243. UA_memcpy((void*)&newEntry->node, *node, nodesize);
  244. cds_lfht_node_init(&newEntry->htn);
  245. newEntry->refcount = ALIVE_BIT;
  246. if(getManaged) // increase the counter before adding the node
  247. newEntry->refcount++;
  248. hash_t h = hash(&(*node)->nodeId);
  249. struct cds_lfht_iter iter;
  250. rcu_read_lock();
  251. cds_lfht_lookup(ns->ht, h, compare, &(*node)->nodeId, &iter);
  252. /* No node found that can be replaced */
  253. if(!iter.node) {
  254. rcu_read_unlock();
  255. UA_free(newEntry);
  256. return UA_STATUSCODE_BADNODEIDUNKNOWN;
  257. }
  258. #if defined(__GNUC__) || defined(__clang__)
  259. #pragma GCC diagnostic push
  260. #pragma GCC diagnostic ignored "-Wcast-align"
  261. #endif
  262. struct nodeEntry *oldEntry = caa_container_of(iter.node, struct nodeEntry, htn);
  263. #if defined(__GNUC__) || defined(__clang__)
  264. #pragma GCC diagnostic pop
  265. #endif
  266. /* The node we found is obsolete*/
  267. if(&oldEntry->node != oldNode) {
  268. rcu_read_unlock();
  269. UA_free(newEntry);
  270. return UA_STATUSCODE_BADINTERNALERROR;
  271. }
  272. /* The old node is replaced by a managed node. */
  273. if(cds_lfht_replace(ns->ht, &iter, h, compare, &(*node)->nodeId, &newEntry->htn) != 0) {
  274. /* Replacing failed. Maybe the node got replaced just before this thread tried to.*/
  275. rcu_read_unlock();
  276. UA_free(newEntry);
  277. return UA_STATUSCODE_BADINTERNALERROR;
  278. }
  279. /* If an entry got replaced, mark it as dead. */
  280. call_rcu(&oldEntry->rcu_head, markDead);
  281. rcu_read_unlock();
  282. #if defined(__GNUC__) || defined(__clang__)
  283. #pragma GCC diagnostic push
  284. #pragma GCC diagnostic ignored "-Wcast-qual"
  285. #endif
  286. UA_free((UA_Node *)*node); // was copied to newEntry and is obsolete
  287. #if defined(__GNUC__) || defined(__clang__)
  288. #pragma GCC diagnostic pop
  289. #endif
  290. if(getManaged)
  291. *node = &newEntry->node;
  292. else
  293. *node = UA_NULL;
  294. return UA_STATUSCODE_GOOD;
  295. }
  296. UA_StatusCode UA_NodeStore_remove(UA_NodeStore *ns, const UA_NodeId *nodeid) {
  297. hash_t nhash = hash(nodeid);
  298. struct cds_lfht_iter iter;
  299. rcu_read_lock();
  300. /* If this fails, then the node has already been removed. */
  301. cds_lfht_lookup(ns->ht, nhash, compare, &nodeid, &iter);
  302. if(!iter.node || cds_lfht_del(ns->ht, iter.node) != 0) {
  303. rcu_read_unlock();
  304. return UA_STATUSCODE_BADNODEIDUNKNOWN;
  305. }
  306. #if defined(__GNUC__) || defined(__clang__)
  307. #pragma GCC diagnostic push
  308. #pragma GCC diagnostic ignored "-Wcast-align"
  309. #endif
  310. struct nodeEntry *entry = caa_container_of(iter.node, struct nodeEntry, htn);
  311. #if defined(__GNUC__) || defined(__clang__)
  312. #pragma GCC diagnostic pop
  313. #endif
  314. call_rcu(&entry->rcu_head, markDead);
  315. rcu_read_unlock();
  316. return UA_STATUSCODE_GOOD;
  317. }
  318. const UA_Node * UA_NodeStore_get(const UA_NodeStore *ns, const UA_NodeId *nodeid) {
  319. hash_t nhash = hash(nodeid);
  320. struct cds_lfht_iter iter;
  321. rcu_read_lock();
  322. cds_lfht_lookup(ns->ht, nhash, compare, nodeid, &iter);
  323. struct nodeEntry *found_entry = (struct nodeEntry *)cds_lfht_iter_get_node(&iter);
  324. if(!found_entry) {
  325. rcu_read_unlock();
  326. return UA_NULL;
  327. }
  328. /* This is done within a read-lock. The node will not be marked dead within a read-lock. */
  329. uatomic_inc(&found_entry->refcount);
  330. rcu_read_unlock();
  331. return &found_entry->node;
  332. }
  333. void UA_NodeStore_iterate(const UA_NodeStore *ns, UA_NodeStore_nodeVisitor visitor) {
  334. struct cds_lfht *ht = ns->ht;
  335. struct cds_lfht_iter iter;
  336. rcu_read_lock();
  337. cds_lfht_first(ht, &iter);
  338. while(iter.node != UA_NULL) {
  339. struct nodeEntry *found_entry = (struct nodeEntry *)cds_lfht_iter_get_node(&iter);
  340. uatomic_inc(&found_entry->refcount);
  341. const UA_Node *node = &found_entry->node;
  342. rcu_read_unlock();
  343. visitor(node);
  344. UA_NodeStore_release((const UA_Node *)node);
  345. rcu_read_lock();
  346. cds_lfht_next(ht, &iter);
  347. }
  348. rcu_read_unlock();
  349. }