Base64.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. /*
  2. * Author: Patrick-Christopher Mattulat
  3. * Company: Lynar Studios
  4. * E-Mail: webmaster@lynarstudios.com
  5. * Created: 2022-01-03
  6. * Changed: 2022-05-20
  7. *
  8. * */
  9. #include <ls_std/encoding/Base64.hpp>
  10. #include <bitset>
  11. ::std::string ls::std::encoding::Base64::encode(const ::std::string &_sequence)
  12. {
  13. ::std::string encodedString{};
  14. for(size_t index = 0 ; index < _sequence.size() ; index += 3)
  15. {
  16. ::std::string byteTriple = ls::std::encoding::Base64::_getNextByteTriple(_sequence, index);
  17. encodedString += ls::std::encoding::Base64::_encodeByteTriple(byteTriple);
  18. }
  19. return ls::std::encoding::Base64::_applyEndingRule(encodedString, _sequence.size());
  20. }
  21. ::std::string ls::std::encoding::Base64::decode(const ::std::string &_sequence)
  22. {
  23. ::std::string decodedString{};
  24. for (int index{} ; index < _sequence.size() ; index += 4)
  25. {
  26. ::std::string quadruple = ls::std::encoding::Base64::_getNextByteQuadruple(_sequence, index);
  27. decodedString += ls::std::encoding::Base64::_decodeByteQuadruple(quadruple);
  28. }
  29. return decodedString;
  30. }
  31. ::std::string ls::std::encoding::Base64::_applyEndingRule(::std::string _encodedString, size_t _sequenceSize)
  32. {
  33. size_t size = _encodedString.size();
  34. if (_sequenceSize % 3 == 1)
  35. {
  36. _encodedString[size - 2] = '=';
  37. _encodedString[size - 1] = '=';
  38. }
  39. if (_sequenceSize % 3 == 2)
  40. {
  41. _encodedString[size - 1] = '=';
  42. }
  43. return _encodedString;
  44. }
  45. ::std::string ls::std::encoding::Base64::_decodeByteQuadruple(const ::std::string& _quadruple)
  46. {
  47. ::std::string decodedText{};
  48. uint8_t shiftValue = 16;
  49. uint32_t bitStorage = ls::std::encoding::Base64::_toDecodingBitStorage(_quadruple);
  50. for (uint8_t index = 0; index < ((uint8_t) _quadruple.size() - 1); index++)
  51. {
  52. uint32_t bitMask = ls::std::encoding::Base64::_generateBitMask(255, shiftValue);
  53. uint32_t bitSequence = ls::std::encoding::Base64::_extractBitSequence(bitMask, bitStorage);
  54. bitSequence = bitSequence >> shiftValue;
  55. decodedText += (char) bitSequence;
  56. shiftValue -= 8;
  57. }
  58. return decodedText;
  59. }
  60. ::std::string ls::std::encoding::Base64::_encodeByteTriple(const ::std::string& _byteTriple)
  61. {
  62. ::std::string encodedText{};
  63. uint32_t bitStorage = ls::std::encoding::Base64::_toEncodingBitStorage(_byteTriple);
  64. static ::std::vector<uint32_t> bitMaskStorage = {16515072, 258048, 4032, 63};
  65. static ::std::unordered_map<uint8_t, char> encodingMap = ls::std::encoding::Base64::_getEncodingMap();
  66. uint8_t shiftValue = 18;
  67. for (uint8_t bitMaskIndex = 0 ; bitMaskIndex < 4 ; bitMaskIndex++)
  68. {
  69. uint32_t extractedBitSequence = ls::std::encoding::Base64::_extractBitSequence(bitMaskStorage[bitMaskIndex], bitStorage);
  70. extractedBitSequence = extractedBitSequence >> shiftValue;
  71. encodedText += encodingMap[(uint8_t) extractedBitSequence];
  72. shiftValue -= 6;
  73. }
  74. return encodedText;
  75. }
  76. uint32_t ls::std::encoding::Base64::_extractBitSequence(uint32_t _bitMask, uint32_t _bitStorage)
  77. {
  78. return _bitStorage & _bitMask;
  79. }
  80. uint32_t ls::std::encoding::Base64::_generateBitMask(uint32_t _maskValue, uint8_t _shiftValue)
  81. {
  82. if(_shiftValue == 0)
  83. {
  84. return _maskValue;
  85. }
  86. if(_shiftValue < 0)
  87. {
  88. return _maskValue >> _shiftValue;
  89. }
  90. return _maskValue << _shiftValue;
  91. }
  92. ::std::unordered_map<char, uint8_t> ls::std::encoding::Base64::_getDecodingMap()
  93. {
  94. static ::std::unordered_map<char, uint8_t> decodingMap =
  95. {
  96. {'A', 0}, {'B', 1}, {'C', 2}, {'D', 3}, {'E', 4}, {'F', 5}, {'G', 6}, {'H', 7},
  97. {'I', 8}, {'J', 9}, {'K', 10}, {'L', 11}, {'M', 12}, {'N', 13}, {'O', 14}, {'P', 15},
  98. {'Q', 16}, {'R', 17}, {'S', 18}, {'T', 19}, {'U', 20}, {'V', 21}, {'W', 22}, {'X', 23},
  99. {'Y', 24}, {'Z', 25}, {'a', 26}, {'b', 27}, {'c', 28}, {'d', 29}, {'e', 30}, {'f', 31},
  100. {'g', 32}, {'h', 33}, {'i', 34}, {'j', 35}, {'k', 36}, {'l', 37}, {'m', 38}, {'n', 39},
  101. {'o', 40}, {'p', 41}, {'q', 42}, {'r', 43}, {'s', 44}, {'t', 45}, {'u', 46}, {'v', 47},
  102. {'w', 48}, {'x', 49}, {'y', 50}, {'z', 51}, {'0', 52}, {'1', 53}, {'2', 54}, {'3', 55},
  103. {'4', 56}, {'5', 57}, {'6', 58}, {'7', 59}, {'8', 60}, {'9', 61}, {'+', 62}, {'/', 63}
  104. };
  105. return decodingMap;
  106. }
  107. ::std::unordered_map<uint8_t, char> ls::std::encoding::Base64::_getEncodingMap()
  108. {
  109. static ::std::unordered_map<uint8_t, char> encodingMap =
  110. {
  111. {0, 'A'}, {1, 'B'}, {2, 'C'}, {3, 'D'}, {4, 'E'}, {5, 'F'}, {6, 'G'}, {7, 'H'},
  112. {8, 'I'}, {9, 'J'}, {10, 'K'}, {11, 'L'}, {12, 'M'}, {13, 'N'}, {14, 'O'}, {15, 'P'},
  113. {16, 'Q'}, {17, 'R'}, {18, 'S'}, {19, 'T'}, {20, 'U'}, {21, 'V'}, {22, 'W'}, {23, 'X'},
  114. {24, 'Y'}, {25, 'Z'}, {26, 'a'}, {27, 'b'}, {28, 'c'}, {29, 'd'}, {30, 'e'}, {31, 'f'},
  115. {32, 'g'}, {33, 'h'}, {34, 'i'}, {35, 'j'}, {36, 'k'}, {37, 'l'}, {38, 'm'}, {39, 'n'},
  116. {40, 'o'}, {41, 'p'}, {42, 'q'}, {43, 'r'}, {44, 's'}, {45, 't'}, {46, 'u'}, {47, 'v'},
  117. {48, 'w'}, {49, 'x'}, {50, 'y'}, {51, 'z'}, {52, '0'}, {53, '1'}, {54, '2'}, {55, '3'},
  118. {56, '4'}, {57, '5'}, {58, '6'}, {59, '7'}, {60, '8'}, {61, '9'}, {62, '+'}, {63, '/'},
  119. };
  120. return encodingMap;
  121. }
  122. ::std::string ls::std::encoding::Base64::_getNextByteQuadruple(const ::std::string &_sequence, size_t _index)
  123. {
  124. return _sequence.substr(_index, 4);
  125. }
  126. ::std::string ls::std::encoding::Base64::_getNextByteTriple(const ::std::string &_sequence, size_t _index)
  127. {
  128. return _sequence.substr(_index, 3);
  129. }
  130. void ls::std::encoding::Base64::_mergeBitSequence(uint32_t &_bitStorage, const uint32_t &_bitMask)
  131. {
  132. _bitStorage = _bitStorage | _bitMask;
  133. }
  134. uint32_t ls::std::encoding::Base64::_toDecodingBitStorage(const ::std::string &_quadruple)
  135. {
  136. uint32_t bitStorage{};
  137. uint8_t letterCounter = 1;
  138. ::std::unordered_map<char, uint8_t> decodingMap = ls::std::encoding::Base64::_getDecodingMap();
  139. for(char letter : _quadruple)
  140. {
  141. uint32_t bitMask = ls::std::encoding::Base64::_generateBitMask(decodingMap[(char) letter], (4 - letterCounter) * 6); // must be hardcoded - even in case of less than 4 characters, so that conversion is correct
  142. ls::std::encoding::Base64::_mergeBitSequence(bitStorage, bitMask);
  143. ++letterCounter;
  144. }
  145. return bitStorage;
  146. }
  147. uint32_t ls::std::encoding::Base64::_toEncodingBitStorage(const ::std::string &_triple)
  148. {
  149. uint32_t bitStorage{};
  150. uint8_t shiftValue = 16;
  151. for(char letter : _triple)
  152. {
  153. uint32_t bitMask = ls::std::encoding::Base64::_generateBitMask((uint8_t) letter, shiftValue);
  154. ls::std::encoding::Base64::_mergeBitSequence(bitStorage, bitMask);
  155. shiftValue -= 8;
  156. }
  157. return bitStorage;
  158. }