Utf8View.cpp 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. /*
  2. * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
  3. * Copyright (c) 2021, Max Wipfli <mail@maxwipfli.ch>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/Assertions.h>
  8. #include <AK/Format.h>
  9. #include <AK/Utf8View.h>
  10. namespace AK {
  11. Utf8View::Utf8View(const String& string)
  12. : m_string(string)
  13. {
  14. }
  15. Utf8View::Utf8View(const StringView& string)
  16. : m_string(string)
  17. {
  18. }
  19. Utf8View::Utf8View(const char* string)
  20. : m_string(string)
  21. {
  22. }
  23. const unsigned char* Utf8View::begin_ptr() const
  24. {
  25. return (const unsigned char*)m_string.characters_without_null_termination();
  26. }
  27. const unsigned char* Utf8View::end_ptr() const
  28. {
  29. return begin_ptr() + m_string.length();
  30. }
  31. Utf8CodePointIterator Utf8View::begin() const
  32. {
  33. return { begin_ptr(), m_string.length() };
  34. }
  35. Utf8CodePointIterator Utf8View::end() const
  36. {
  37. return { end_ptr(), 0 };
  38. }
  39. Utf8CodePointIterator Utf8View::iterator_at_byte_offset(size_t byte_offset) const
  40. {
  41. size_t current_offset = 0;
  42. for (auto iterator = begin(); !iterator.done(); ++iterator) {
  43. if (current_offset >= byte_offset)
  44. return iterator;
  45. current_offset += iterator.code_point_length_in_bytes();
  46. }
  47. return end();
  48. }
  49. size_t Utf8View::byte_offset_of(const Utf8CodePointIterator& it) const
  50. {
  51. VERIFY(it.m_ptr >= begin_ptr());
  52. VERIFY(it.m_ptr <= end_ptr());
  53. return it.m_ptr - begin_ptr();
  54. }
  55. Utf8View Utf8View::substring_view(size_t byte_offset, size_t byte_length) const
  56. {
  57. StringView string = m_string.substring_view(byte_offset, byte_length);
  58. return Utf8View { string };
  59. }
  60. Utf8View Utf8View::unicode_substring_view(size_t code_point_offset, size_t code_point_length) const
  61. {
  62. if (code_point_length == 0)
  63. return {};
  64. size_t code_point_index = 0, offset_in_bytes = 0;
  65. for (auto iterator = begin(); !iterator.done(); ++iterator) {
  66. if (code_point_index == code_point_offset)
  67. offset_in_bytes = byte_offset_of(iterator);
  68. if (code_point_index == code_point_offset + code_point_length - 1) {
  69. size_t length_in_bytes = byte_offset_of(++iterator) - offset_in_bytes;
  70. return substring_view(offset_in_bytes, length_in_bytes);
  71. }
  72. ++code_point_index;
  73. }
  74. VERIFY_NOT_REACHED();
  75. }
  76. static inline bool decode_first_byte(
  77. unsigned char byte,
  78. size_t& out_code_point_length_in_bytes,
  79. u32& out_value)
  80. {
  81. if ((byte & 128) == 0) {
  82. out_value = byte;
  83. out_code_point_length_in_bytes = 1;
  84. return true;
  85. }
  86. if ((byte & 64) == 0) {
  87. return false;
  88. }
  89. if ((byte & 32) == 0) {
  90. out_value = byte & 31;
  91. out_code_point_length_in_bytes = 2;
  92. return true;
  93. }
  94. if ((byte & 16) == 0) {
  95. out_value = byte & 15;
  96. out_code_point_length_in_bytes = 3;
  97. return true;
  98. }
  99. if ((byte & 8) == 0) {
  100. out_value = byte & 7;
  101. out_code_point_length_in_bytes = 4;
  102. return true;
  103. }
  104. return false;
  105. }
  106. bool Utf8View::validate(size_t& valid_bytes) const
  107. {
  108. valid_bytes = 0;
  109. for (auto ptr = begin_ptr(); ptr < end_ptr(); ptr++) {
  110. size_t code_point_length_in_bytes;
  111. u32 value;
  112. bool first_byte_makes_sense = decode_first_byte(*ptr, code_point_length_in_bytes, value);
  113. if (!first_byte_makes_sense)
  114. return false;
  115. for (size_t i = 1; i < code_point_length_in_bytes; i++) {
  116. ptr++;
  117. if (ptr >= end_ptr())
  118. return false;
  119. if (*ptr >> 6 != 2)
  120. return false;
  121. }
  122. valid_bytes += code_point_length_in_bytes;
  123. }
  124. return true;
  125. }
  126. size_t Utf8View::calculate_length() const
  127. {
  128. size_t length = 0;
  129. for ([[maybe_unused]] auto code_point : *this) {
  130. ++length;
  131. }
  132. return length;
  133. }
  134. bool Utf8View::starts_with(const Utf8View& start) const
  135. {
  136. if (start.is_empty())
  137. return true;
  138. if (is_empty())
  139. return false;
  140. if (start.length() > length())
  141. return false;
  142. if (begin_ptr() == start.begin_ptr())
  143. return true;
  144. for (auto k = begin(), l = start.begin(); l != start.end(); ++k, ++l) {
  145. if (*k != *l)
  146. return false;
  147. }
  148. return true;
  149. }
  150. Utf8CodePointIterator::Utf8CodePointIterator(const unsigned char* ptr, size_t length)
  151. : m_ptr(ptr)
  152. , m_length(length)
  153. {
  154. }
  155. bool Utf8CodePointIterator::operator==(const Utf8CodePointIterator& other) const
  156. {
  157. return m_ptr == other.m_ptr && m_length == other.m_length;
  158. }
  159. bool Utf8CodePointIterator::operator!=(const Utf8CodePointIterator& other) const
  160. {
  161. return !(*this == other);
  162. }
  163. Utf8CodePointIterator& Utf8CodePointIterator::operator++()
  164. {
  165. VERIFY(m_length > 0);
  166. size_t code_point_length_in_bytes = 0;
  167. u32 value;
  168. bool first_byte_makes_sense = decode_first_byte(*m_ptr, code_point_length_in_bytes, value);
  169. VERIFY(first_byte_makes_sense);
  170. VERIFY(code_point_length_in_bytes <= m_length);
  171. m_ptr += code_point_length_in_bytes;
  172. m_length -= code_point_length_in_bytes;
  173. return *this;
  174. }
  175. size_t Utf8CodePointIterator::code_point_length_in_bytes() const
  176. {
  177. VERIFY(m_length > 0);
  178. size_t code_point_length_in_bytes = 0;
  179. u32 value;
  180. bool first_byte_makes_sense = decode_first_byte(*m_ptr, code_point_length_in_bytes, value);
  181. VERIFY(first_byte_makes_sense);
  182. return code_point_length_in_bytes;
  183. }
  184. u32 Utf8CodePointIterator::operator*() const
  185. {
  186. VERIFY(m_length > 0);
  187. u32 code_point_value_so_far = 0;
  188. size_t code_point_length_in_bytes = 0;
  189. bool first_byte_makes_sense = decode_first_byte(m_ptr[0], code_point_length_in_bytes, code_point_value_so_far);
  190. if (!first_byte_makes_sense)
  191. dbgln("First byte doesn't make sense, bytes: {}", StringView { (const char*)m_ptr, m_length });
  192. VERIFY(first_byte_makes_sense);
  193. if (code_point_length_in_bytes > m_length)
  194. dbgln("Not enough bytes (need {}, have {}), first byte is: {:#02x}, '{}'", code_point_length_in_bytes, m_length, m_ptr[0], (const char*)m_ptr);
  195. VERIFY(code_point_length_in_bytes <= m_length);
  196. for (size_t offset = 1; offset < code_point_length_in_bytes; offset++) {
  197. VERIFY(m_ptr[offset] >> 6 == 2);
  198. code_point_value_so_far <<= 6;
  199. code_point_value_so_far |= m_ptr[offset] & 63;
  200. }
  201. return code_point_value_so_far;
  202. }
  203. Optional<u32> Utf8CodePointIterator::peek(size_t offset) const
  204. {
  205. if (offset == 0) {
  206. if (this->done())
  207. return {};
  208. return this->operator*();
  209. }
  210. auto new_iterator = *this;
  211. for (size_t index = 0; index < offset; ++index) {
  212. ++new_iterator;
  213. if (new_iterator.done())
  214. return {};
  215. }
  216. return *new_iterator;
  217. }
  218. }