name_generator.hpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. // Boost name_generator.hpp header file ----------------------------------------------//
  2. // Copyright 2010 Andy Tompkins.
  3. // Distributed under the Boost Software License, Version 1.0. (See
  4. // accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. #ifndef BOOST_UUID_NAME_GENERATOR_HPP
  7. #define BOOST_UUID_NAME_GENERATOR_HPP
  8. #include <boost/uuid/uuid.hpp>
  9. #include <boost/uuid/sha1.hpp>
  10. #include <boost/assert.hpp>
  11. #include <string>
  12. #include <cstring> // for strlen, wcslen
  13. #ifdef BOOST_NO_STDC_NAMESPACE
  14. namespace std {
  15. using ::strlen;
  16. using ::wcslen;
  17. } //namespace std
  18. #endif //BOOST_NO_STDC_NAMESPACE
  19. namespace boost {
  20. namespace uuids {
  21. // generate a name-based uuid
  22. // TODO: add in common namesspace uuids
  23. class name_generator {
  24. public:
  25. typedef uuid result_type;
  26. explicit name_generator(uuid const& namespace_uuid_)
  27. : namespace_uuid(namespace_uuid_)
  28. {}
  29. uuid operator()(const char* name) {
  30. reset();
  31. process_characters(name, std::strlen(name));
  32. return sha_to_uuid();
  33. }
  34. uuid operator()(const wchar_t* name) {
  35. reset();
  36. process_characters(name, std::wcslen(name));
  37. return sha_to_uuid();
  38. }
  39. template <typename ch, typename char_traits, typename alloc>
  40. uuid operator()(std::basic_string<ch, char_traits, alloc> const& name) {
  41. reset();
  42. process_characters(name.c_str(), name.length());
  43. return sha_to_uuid();
  44. }
  45. uuid operator()(void const* buffer, std::size_t byte_count) {
  46. reset();
  47. sha.process_bytes(buffer, byte_count);
  48. return sha_to_uuid();
  49. };
  50. private:
  51. // we convert all characters to uint32_t so that each
  52. // character is 4 bytes reguardless of sizeof(char) or
  53. // sizeof(wchar_t). We want the name string on any
  54. // platform / compiler to generate the same uuid
  55. // except for char
  56. template <typename char_type>
  57. void process_characters(char_type const*const characters, size_t count) {
  58. BOOST_ASSERT(sizeof(uint32_t) >= sizeof(char_type));
  59. for (size_t i=0; i<count; i++) {
  60. uint32_t c = characters[i];
  61. sha.process_byte(static_cast<unsigned char>((c >> 0) & 0xFF));
  62. sha.process_byte(static_cast<unsigned char>((c >> 8) & 0xFF));
  63. sha.process_byte(static_cast<unsigned char>((c >> 16) & 0xFF));
  64. sha.process_byte(static_cast<unsigned char>((c >> 24) & 0xFF));
  65. }
  66. }
  67. void process_characters(char const*const characters, size_t count) {
  68. sha.process_bytes(characters, count);
  69. }
  70. void reset()
  71. {
  72. sha.reset();
  73. sha.process_bytes(namespace_uuid.begin(), namespace_uuid.size());
  74. }
  75. uuid sha_to_uuid()
  76. {
  77. unsigned int digest[5];
  78. sha.get_digest(digest);
  79. uuid u;
  80. for (int i=0; i<4; ++i) {
  81. *(u.begin() + i*4+0) = static_cast<uint8_t>((digest[i] >> 24) & 0xFF);
  82. *(u.begin() + i*4+1) = static_cast<uint8_t>((digest[i] >> 16) & 0xFF);
  83. *(u.begin() + i*4+2) = static_cast<uint8_t>((digest[i] >> 8) & 0xFF);
  84. *(u.begin() + i*4+3) = static_cast<uint8_t>((digest[i] >> 0) & 0xFF);
  85. }
  86. // set variant
  87. // must be 0b10xxxxxx
  88. *(u.begin()+8) &= 0xBF;
  89. *(u.begin()+8) |= 0x80;
  90. // set version
  91. // must be 0b0101xxxx
  92. *(u.begin()+6) &= 0x5F; //0b01011111
  93. *(u.begin()+6) |= 0x50; //0b01010000
  94. return u;
  95. }
  96. private:
  97. uuid namespace_uuid;
  98. detail::sha1 sha;
  99. };
  100. }} // namespace boost::uuids
  101. #endif // BOOST_UUID_NAME_GENERATOR_HPP