ua_nodestore_hash.inc 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879
  1. typedef UA_UInt32 hash_t;
  2. static hash_t mod(hash_t h, hash_t size) { return h % size; }
  3. static hash_t mod2(hash_t h, hash_t size) { return 1 + (h % (size - 2)); }
  4. /* Based on Murmur-Hash 3 by Austin Appleby (public domain, freely usable) */
  5. static hash_t hash_array(const UA_Byte *data, UA_UInt32 len, UA_UInt32 seed) {
  6. if(data == UA_NULL)
  7. return 0;
  8. const int32_t nblocks = len / 4;
  9. const uint32_t *blocks;
  10. static const uint32_t c1 = 0xcc9e2d51;
  11. static const uint32_t c2 = 0x1b873593;
  12. static const uint32_t r1 = 15;
  13. static const uint32_t r2 = 13;
  14. static const uint32_t m = 5;
  15. static const uint32_t n = 0xe6546b64;
  16. hash_t hash = seed;
  17. /* Somce compilers emit a warning when casting from a byte array to ints. */
  18. #if defined(__GNUC__) || defined(__clang__)
  19. #pragma GCC diagnostic push
  20. #pragma GCC diagnostic ignored "-Wcast-align"
  21. #endif
  22. blocks = (const uint32_t *)data;
  23. #if defined(__GNUC__) || defined(__clang__)
  24. #pragma GCC diagnostic pop
  25. #endif
  26. for(int32_t i = 0;i < nblocks;i++) {
  27. uint32_t k = blocks[i];
  28. k *= c1;
  29. k = (k << r1) | (k >> (32 - r1));
  30. k *= c2;
  31. hash ^= k;
  32. hash = ((hash << r2) | (hash >> (32 - r2))) * m + n;
  33. }
  34. const uint8_t *tail = (const uint8_t *)(data + nblocks * 4);
  35. uint32_t k1 = 0;
  36. switch(len & 3) {
  37. case 3:
  38. k1 ^= tail[2] << 16;
  39. case 2:
  40. k1 ^= tail[1] << 8;
  41. case 1:
  42. k1 ^= tail[0];
  43. k1 *= c1;
  44. k1 = (k1 << r1) | (k1 >> (32 - r1));
  45. k1 *= c2;
  46. hash ^= k1;
  47. }
  48. hash ^= len;
  49. hash ^= (hash >> 16);
  50. hash *= 0x85ebca6b;
  51. hash ^= (hash >> 13);
  52. hash *= 0xc2b2ae35;
  53. hash ^= (hash >> 16);
  54. return hash;
  55. }
  56. static hash_t hash(const UA_NodeId *n) {
  57. switch(n->identifierType) {
  58. case UA_NODEIDTYPE_NUMERIC:
  59. /* Knuth's multiplicative hashing */
  60. return (n->identifier.numeric + n->namespaceIndex) * 2654435761; // mod(2^32) is implicit
  61. case UA_NODEIDTYPE_STRING:
  62. return hash_array(n->identifier.string.data, n->identifier.string.length, n->namespaceIndex);
  63. case UA_NODEIDTYPE_GUID:
  64. return hash_array((const UA_Byte *)&(n->identifier.guid), sizeof(UA_Guid), n->namespaceIndex);
  65. case UA_NODEIDTYPE_BYTESTRING:
  66. return hash_array((const UA_Byte *)n->identifier.byteString.data, n->identifier.byteString.length, n->namespaceIndex);
  67. default:
  68. UA_assert(UA_FALSE);
  69. return 0;
  70. }
  71. }