123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214 |
- /*
- * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are met:
- *
- * 1. Redistributions of source code must retain the above copyright notice, this
- * list of conditions and the following disclaimer.
- *
- * 2. Redistributions in binary form must reproduce the above copyright notice,
- * this list of conditions and the following disclaimer in the documentation
- * and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
- * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
- * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
- * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include <AK/Assertions.h>
- #include <AK/LogStream.h>
- #include <AK/Utf8View.h>
- namespace AK {
- Utf8View::Utf8View(const String& string)
- : m_string(string)
- {
- }
- Utf8View::Utf8View(const StringView& string)
- : m_string(string)
- {
- }
- Utf8View::Utf8View(const char* string)
- : m_string(string)
- {
- }
- const unsigned char* Utf8View::begin_ptr() const
- {
- return (const unsigned char*)m_string.characters_without_null_termination();
- }
- const unsigned char* Utf8View::end_ptr() const
- {
- return begin_ptr() + m_string.length();
- }
- Utf8CodepointIterator Utf8View::begin() const
- {
- return { begin_ptr(), m_string.length() };
- }
- Utf8CodepointIterator Utf8View::end() const
- {
- return { end_ptr(), 0 };
- }
- size_t Utf8View::byte_offset_of(const Utf8CodepointIterator& it) const
- {
- VERIFY(it.m_ptr >= begin_ptr());
- VERIFY(it.m_ptr <= end_ptr());
- return it.m_ptr - begin_ptr();
- }
- Utf8View Utf8View::substring_view(int byte_offset, int byte_length) const
- {
- StringView string = m_string.substring_view(byte_offset, byte_length);
- return Utf8View { string };
- }
- static inline bool decode_first_byte(
- unsigned char byte,
- size_t& out_code_point_length_in_bytes,
- u32& out_value)
- {
- if ((byte & 128) == 0) {
- out_value = byte;
- out_code_point_length_in_bytes = 1;
- return true;
- }
- if ((byte & 64) == 0) {
- return false;
- }
- if ((byte & 32) == 0) {
- out_value = byte & 31;
- out_code_point_length_in_bytes = 2;
- return true;
- }
- if ((byte & 16) == 0) {
- out_value = byte & 15;
- out_code_point_length_in_bytes = 3;
- return true;
- }
- if ((byte & 8) == 0) {
- out_value = byte & 7;
- out_code_point_length_in_bytes = 4;
- return true;
- }
- return false;
- }
- bool Utf8View::validate(size_t& valid_bytes) const
- {
- valid_bytes = 0;
- for (auto ptr = begin_ptr(); ptr < end_ptr(); ptr++) {
- size_t code_point_length_in_bytes;
- u32 value;
- bool first_byte_makes_sense = decode_first_byte(*ptr, code_point_length_in_bytes, value);
- if (!first_byte_makes_sense)
- return false;
- for (size_t i = 1; i < code_point_length_in_bytes; i++) {
- ptr++;
- if (ptr >= end_ptr())
- return false;
- if (*ptr >> 6 != 2)
- return false;
- }
- valid_bytes += code_point_length_in_bytes;
- }
- return true;
- }
- size_t Utf8View::calculate_length() const
- {
- size_t length = 0;
- for ([[maybe_unused]] auto code_point : *this) {
- ++length;
- }
- return length;
- }
- Utf8CodepointIterator::Utf8CodepointIterator(const unsigned char* ptr, size_t length)
- : m_ptr(ptr)
- , m_length(length)
- {
- }
- bool Utf8CodepointIterator::operator==(const Utf8CodepointIterator& other) const
- {
- return m_ptr == other.m_ptr && m_length == other.m_length;
- }
- bool Utf8CodepointIterator::operator!=(const Utf8CodepointIterator& other) const
- {
- return !(*this == other);
- }
- Utf8CodepointIterator& Utf8CodepointIterator::operator++()
- {
- VERIFY(m_length > 0);
- size_t code_point_length_in_bytes = 0;
- u32 value;
- bool first_byte_makes_sense = decode_first_byte(*m_ptr, code_point_length_in_bytes, value);
- VERIFY(first_byte_makes_sense);
- VERIFY(code_point_length_in_bytes <= m_length);
- m_ptr += code_point_length_in_bytes;
- m_length -= code_point_length_in_bytes;
- return *this;
- }
- size_t Utf8CodepointIterator::code_point_length_in_bytes() const
- {
- VERIFY(m_length > 0);
- size_t code_point_length_in_bytes = 0;
- u32 value;
- bool first_byte_makes_sense = decode_first_byte(*m_ptr, code_point_length_in_bytes, value);
- VERIFY(first_byte_makes_sense);
- return code_point_length_in_bytes;
- }
- u32 Utf8CodepointIterator::operator*() const
- {
- VERIFY(m_length > 0);
- u32 code_point_value_so_far = 0;
- size_t code_point_length_in_bytes = 0;
- bool first_byte_makes_sense = decode_first_byte(m_ptr[0], code_point_length_in_bytes, code_point_value_so_far);
- if (!first_byte_makes_sense)
- dbgln("First byte doesn't make sense, bytes: {}", StringView { (const char*)m_ptr, m_length });
- VERIFY(first_byte_makes_sense);
- if (code_point_length_in_bytes > m_length)
- dbgln("Not enough bytes (need {}, have {}), first byte is: {:#02x}, '{}'", code_point_length_in_bytes, m_length, m_ptr[0], (const char*)m_ptr);
- VERIFY(code_point_length_in_bytes <= m_length);
- for (size_t offset = 1; offset < code_point_length_in_bytes; offset++) {
- VERIFY(m_ptr[offset] >> 6 == 2);
- code_point_value_so_far <<= 6;
- code_point_value_so_far |= m_ptr[offset] & 63;
- }
- return code_point_value_so_far;
- }
- }
|