obfuscate.hpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. /**
  2. ******************************************************************************
  3. * @file : obfuscate.hpp
  4. * @author : Dream
  5. * @brief : None
  6. * @attention : None
  7. * @date : 2024/12/4
  8. ******************************************************************************
  9. */
  10. /*
  11. 原文作者:Adam Yaxley
  12. */
  13. #pragma once
  14. #if __cplusplus >= 202002L
  15. #define AY_CONSTEVAL consteval
  16. #else
  17. #define AY_CONSTEVAL constexpr
  18. #endif
  19. #ifdef _MSC_VER
  20. #define AY_CAT(X,Y) AY_CAT2(X,Y)
  21. #define AY_CAT2(X,Y) X##Y
  22. #define AY_LINE int(AY_CAT(__LINE__,U))
  23. #else
  24. #define AY_LINE __LINE__
  25. #endif
  26. #ifndef AY_OBFUSCATE_DEFAULT_KEY
  27. //用于混淆字符串的默认 64 位密钥
  28. //这可以通过在之前定义 AY_OBFUSCATE_DEFAULT_KEY 来由用户指定
  29. //包括 obfuscate.h
  30. #define AY_OBFUSCATE_DEFAULT_KEY ay::generate_key(AY_LINE)
  31. #endif
  32. namespace ay
  33. {
  34. using size_type = unsigned long long;
  35. using key_type = unsigned long long;
  36. template <typename T>
  37. struct remove_const_ref {
  38. using type = T;
  39. };
  40. template <typename T>
  41. struct remove_const_ref<T&> {
  42. using type = T;
  43. };
  44. template <typename T>
  45. struct remove_const_ref<const T> {
  46. using type = T;
  47. };
  48. template <typename T>
  49. struct remove_const_ref<const T&> {
  50. using type = T;
  51. };
  52. template <typename T>
  53. using char_type = typename remove_const_ref<T>::type;
  54. //生成一个跨越所有 8 个字节的伪随机键
  55. AY_CONSTEVAL key_type generate_key(key_type seed)
  56. {
  57. // 使用 MurmurHash3 64 位终结器对种子进行哈希处理
  58. key_type key = seed;
  59. key ^= (key >> 33);
  60. key *= 0xff51afd7ed558ccd;
  61. key ^= (key >> 33);
  62. key *= 0xc4ceb9fe1a85ec53;
  63. key ^= (key >> 33);
  64. // 确保设置每个字节中的位
  65. key |= 0x0101010101010101ull;
  66. return key;
  67. }
  68. // 使用 key 对数据进行混淆或反混淆
  69. template <typename CHAR_TYPE>
  70. constexpr void cipher(CHAR_TYPE* data, size_type size, key_type key)
  71. {
  72. // 使用基于密钥的简单 XOR 密码进行混淆
  73. for (size_type i = 0; i < size; i++)
  74. {
  75. data[i] ^= CHAR_TYPE((key >> ((i % 8) * 8)) & 0xFF);
  76. }
  77. }
  78. // 在编译时对字符串进行模糊处理
  79. template <size_type N, key_type KEY, typename CHAR_TYPE = char>
  80. class obfuscator
  81. {
  82. public:
  83. // 在构造时混淆字符串 'data'
  84. AY_CONSTEVAL obfuscator(const CHAR_TYPE* data)
  85. {
  86. // Copy data
  87. for (size_type i = 0; i < N; i++)
  88. {
  89. m_data[i] = data[i];
  90. }
  91. // On construction each of the characters in the string is
  92. // obfuscated with an XOR cipher based on key
  93. cipher(m_data, N, KEY);
  94. }
  95. constexpr const CHAR_TYPE* data() const
  96. {
  97. return &m_data[0];
  98. }
  99. AY_CONSTEVAL size_type size() const
  100. {
  101. return N;
  102. }
  103. AY_CONSTEVAL key_type key() const
  104. {
  105. return KEY;
  106. }
  107. private:
  108. CHAR_TYPE m_data[N]{};
  109. };
  110. // Handles decryption and re-encryption of an encrypted string at runtime
  111. template <size_type N, key_type KEY, typename CHAR_TYPE = char>
  112. class obfuscated_data
  113. {
  114. public:
  115. obfuscated_data(const obfuscator<N, KEY, CHAR_TYPE>& obfuscator)
  116. {
  117. // Copy obfuscated data
  118. for (size_type i = 0; i < N; i++)
  119. {
  120. m_data[i] = obfuscator.data()[i];
  121. }
  122. }
  123. ~obfuscated_data()
  124. {
  125. // Zero m_data to remove it from memory
  126. for (size_type i = 0; i < N; i++)
  127. {
  128. m_data[i] = 0;
  129. }
  130. }
  131. // Returns a pointer to the plain text string, decrypting it if
  132. // necessary
  133. operator CHAR_TYPE* ()
  134. {
  135. decrypt();
  136. return m_data;
  137. }
  138. // Manually decrypt the string
  139. void decrypt()
  140. {
  141. if (m_encrypted)
  142. {
  143. cipher(m_data, N, KEY);
  144. m_encrypted = false;
  145. }
  146. }
  147. // Manually re-encrypt the string
  148. void encrypt()
  149. {
  150. if (!m_encrypted)
  151. {
  152. cipher(m_data, N, KEY);
  153. m_encrypted = true;
  154. }
  155. }
  156. // Returns true if this string is currently encrypted, false otherwise.
  157. bool is_encrypted() const
  158. {
  159. return m_encrypted;
  160. }
  161. private:
  162. // Local storage for the string. Call is_encrypted() to check whether or
  163. // not the string is currently obfuscated.
  164. CHAR_TYPE m_data[N];
  165. // Whether data is currently encrypted
  166. bool m_encrypted{ true };
  167. };
  168. // This function exists purely to extract the number of elements 'N' in the
  169. // array 'data'
  170. template <size_type N, key_type KEY = AY_OBFUSCATE_DEFAULT_KEY, typename CHAR_TYPE = char>
  171. AY_CONSTEVAL auto make_obfuscator(const CHAR_TYPE(&data)[N])
  172. {
  173. return obfuscator<N, KEY, CHAR_TYPE>(data);
  174. }
  175. }
  176. #define AY_OBFUSCATE(data) AY_OBFUSCATE_KEY(data, AY_OBFUSCATE_DEFAULT_KEY)
  177. #define AY_OBFUSCATE_KEY(data, key) \
  178. []() -> ay::obfuscated_data<sizeof(data)/sizeof(data[0]), key, ay::char_type<decltype(*data)>>& { \
  179. static_assert(sizeof(decltype(key)) == sizeof(ay::key_type), "key must be a 64 bit unsigned integer"); \
  180. static_assert((key) >= (1ull << 56), "key must span all 8 bytes"); \
  181. using char_type = ay::char_type<decltype(*data)>; \
  182. constexpr auto n = sizeof(data)/sizeof(data[0]); \
  183. constexpr auto obfuscator = ay::make_obfuscator<n, key, char_type>(data); \
  184. thread_local auto obfuscated_data = ay::obfuscated_data<n, key, char_type>(obfuscator); \
  185. return obfuscated_data; \
  186. }()