/* * LibFud * Copyright 2024 Dominick Allen * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "fud_string.hpp" #include "fud_assert.hpp" namespace fud { ssize_t cStringLength(const char* str) { constexpr auto maxLength = SSIZE_MAX - 1; return cStringLength(str, maxLength); } ssize_t cStringLength(const char* str, size_t maxLength) { if (str == nullptr || maxLength > (SSIZE_MAX - 1)) { return -1; } ssize_t size = 0; while (str[size] != 0 && static_cast(size) < maxLength) { size++; } if (str[size] != 0 && static_cast(size) == maxLength) { return static_cast(maxLength) + 1; } return size; } // NOLINTBEGIN(cppcoreguidelines-pro-type-reinterpret-cast) String::String(const utf8* cString) : String(reinterpret_cast(cString)) { } // NOLINTEND(cppcoreguidelines-pro-type-reinterpret-cast) String::String(const char* cString) { auto lenResult = cStringLength(cString); if (lenResult < 0 || lenResult >= SSIZE_MAX) { m_length = 1; m_capacity = 0; } else if (static_cast(lenResult) < SSO_BUF_SIZE) { m_length = static_cast(lenResult); fudAssert(copyMem(m_buffer.data(), m_buffer.size(), cString, m_length) == FudStatus::Success); fudAssert(nullTerminate() == FudStatus::Success); } else { m_length = static_cast(lenResult); m_capacity = m_length + 1; m_data = static_cast(fudAlloc(m_capacity)); fudAssert(m_data != nullptr); fudAssert(copyMem(m_data, m_capacity, cString, m_length) == FudStatus::Success); fudAssert(nullTerminate() == FudStatus::Success); } } String::String(const String& rhs) : m_length{rhs.m_length}, m_capacity{rhs.m_capacity} { if (rhs.valid()) { if (isLarge()) { m_data = static_cast(fudAlloc(m_capacity)); fudAssert(m_data != nullptr); } fudAssert(copyMem(data(), m_capacity, rhs.data(), m_length) == FudStatus::Success); fudAssert(nullTerminate() == FudStatus::Success); } } String::String(String&& rhs) noexcept : m_length{rhs.m_length}, m_capacity{rhs.m_capacity} { if (isLarge()) { cleanup(); } if (rhs.isLarge()) { m_data = rhs.m_data; rhs.m_data = nullptr; } else { m_buffer = rhs.m_buffer; fudAssert(nullTerminate() == FudStatus::Success); } } String::~String() { cleanup(); } String& String::operator=(const String& rhs) { if (this == &rhs) { return *this; } cleanup(); m_length = rhs.m_length; m_capacity = rhs.m_capacity; if (rhs.valid()) { if (isLarge()) { m_data = static_cast(fudAlloc(m_capacity)); fudAssert(m_data != nullptr); } fudAssert(copyMem(data(), m_capacity, rhs.data(), m_length) == FudStatus::Success); fudAssert(nullTerminate() == FudStatus::Success); } return *this; } String& String::operator=(String&& rhs) noexcept { cleanup(); m_length = rhs.m_length; m_capacity = rhs.m_capacity; if (rhs.isLarge()) { m_data = rhs.m_data; rhs.m_data = nullptr; } else { m_buffer = rhs.m_buffer; fudAssert(nullTerminate() == FudStatus::Success); } return *this; } void String::cleanup() { if (isLarge() && m_data != nullptr) { fudFree(m_data); m_data = nullptr; } } FudStatus String::resize(size_t newCapacity) { if (!valid()) { return FudStatus::StringInvalid; } if (m_length >= newCapacity) { return FudStatus::OperationInvalid; } if (!isLarge() && newCapacity <= SSO_BUF_SIZE) { m_capacity = SSO_BUF_SIZE; return FudStatus::Success; } if (newCapacity <= SSO_BUF_SIZE) { BufType temp{BufType::constFill(0)}; static_cast(copyMem(data(), temp.size(), temp.data(), length())); m_capacity = SSO_BUF_SIZE; fudFree(m_data); m_data = nullptr; copyMem(m_buffer, temp); data()[m_length] = '\0'; return FudStatus::Success; } utf8* newData = nullptr; if (isLarge()) { newData = static_cast(fudRealloc(m_data, newCapacity)); if (newData == nullptr) { return FudStatus::AllocFailure; } } else { newData = static_cast(fudAlloc(newCapacity)); if (newData == nullptr) { return FudStatus::AllocFailure; } static_cast(copyMem(newData, newCapacity, m_buffer.data(), length())); } m_capacity = newCapacity; m_data = newData; data()[m_length] = '\0'; fudAssert(valid()); return FudStatus::Success; } bool String::nullTerminated() const { return data() != nullptr && m_length < m_capacity && data()[m_length] == '\0'; } bool String::valid() const { return nullTerminated(); } bool String::utf8Valid() const { if (!valid()) { return false; } StringView view{*this}; return view.utf8Valid(); } FudStatus String::nullTerminate() { if (m_length < m_capacity) { data()[m_length] = '\0'; return FudStatus::Success; } return FudStatus::StringInvalid; } FudStatus String::reserve(size_t newCapacity) { if (!valid()) { return FudStatus::StringInvalid; } if (newCapacity < m_capacity) { return FudStatus::Success; } return resize(newCapacity); } [[nodiscard]] std::optional String::back() { if (!valid()) { return std::nullopt; } utf8 backChar = data()[m_length - 1]; if (Ascii::valid(backChar)) { return backChar; } return std::nullopt; } std::optional String::pop() { if (!valid()) { return std::nullopt; } if (m_length < 1) { return std::nullopt; } m_length--; auto letter = data()[m_length]; data()[m_length] = '\0'; return letter; } FudStatus String::pushBack(char letter) { return pushBack(static_cast(letter)); } FudStatus String::pushBack(utf8 letter) { if (!valid()) { return FudStatus::StringInvalid; } if (remainingLength() < 1) { auto newCapacity = m_capacity < SIZE_MAX / 2 ? m_capacity * 2 : SIZE_MAX; const auto resizeStatus = resize(newCapacity); if (resizeStatus != FudStatus::Success) { return resizeStatus; } fudAssert(m_capacity == newCapacity); } data()[m_length] = letter; m_length++; data()[m_length] = '\0'; return FudStatus::Success; } FudStatus String::pushBack(const FudUtf8& letter) { if (!valid()) { return FudStatus::StringInvalid; } if (!letter.valid()) { return FudStatus::InvalidInput; } const auto* letterData = letter.data(); if (letterData == nullptr) { return FudStatus::InvalidInput; } auto letterSize = letter.size(); if (remainingLength() < letterSize) { auto newCapacity = m_capacity < SIZE_MAX / 2 ? m_capacity * 2 : SIZE_MAX; if ((newCapacity - m_capacity) < (letterSize + 1)) { return FudStatus::OperationInvalid; } const auto resizeStatus = resize(newCapacity); if (resizeStatus != FudStatus::Success) { return resizeStatus; } fudAssert(m_capacity == newCapacity); } auto copyStatus = copyMem(m_data + m_length, remainingLength(), letterData, letterSize); if (copyStatus != FudStatus::Success) { return copyStatus; } m_length += letterSize; data()[m_length] = '\0'; return FudStatus::Success; } FudStatus String::append(const char* source) { auto lenResult = cStringLength(source); if (lenResult < 0 || lenResult >= SSIZE_MAX) { return FudStatus::InvalidInput; } return this->append(StringView{static_cast(lenResult), source}); } FudStatus String::append(const String& source) { return append(source.asView()); } FudStatus String::append(StringView source) { if (!valid()) { return FudStatus::StringInvalid; } if (source.data() == m_data) { return FudStatus::Aliased; } const size_t newLength = length() + source.length(); if (newLength < length()) { return FudStatus::OperationInvalid; } const size_t newSize = newLength + 1; // cppcheck-suppress knownConditionTrueFalse if (newSize < newLength) { // cppcheck-suppress knownConditionTrueFalse return FudStatus::OperationInvalid; } if (newSize >= m_capacity) { auto newCapacity = newSize < SIZE_MAX / 2 ? newSize * 2 : SIZE_MAX; const auto resizeStatus = resize(newCapacity); if (resizeStatus != FudStatus::Success) { return resizeStatus; } fudAssert(m_capacity == newCapacity); } auto* destPtr = data() + length(); auto status = copyMem(destPtr, m_capacity, source.data(), source.length()); fudAssert(status == FudStatus::Success); m_length += source.length(); status = nullTerminate(); return status; } String String::catenate(const char* rhs) const { String rhsString{rhs}; return catenate(rhsString); } String String::catenate(const String& rhs) const { String output{}; output.m_length = 1; output.m_capacity = 0; if (!valid() || !rhs.valid()) { return output; } output.m_length = m_length + rhs.length(); output.m_capacity = output.m_length + 1; if (output.m_capacity < SSO_BUF_SIZE) { output.m_capacity = SSO_BUF_SIZE; } if (output.isLarge()) { output.m_data = static_cast(fudAlloc(output.m_capacity)); } auto* destPtr = output.data(); auto status = copyMem(destPtr, m_capacity, data(), length()); fudAssert(status == FudStatus::Success); status = copyMem(destPtr + length(), output.m_capacity - length(), rhs.data(), rhs.length()); fudAssert(status == FudStatus::Success); static_cast(status); fudAssert(output.nullTerminate() == FudStatus::Success); return output; } bool String::compare(const String& rhs) const { if (!valid() || !rhs.valid()) { return false; } if (length() != rhs.length()) { return false; } if (isLarge() && data() == rhs.data()) { return true; } auto diffResult = compareMem(data(), length(), rhs.data(), rhs.length()); if (diffResult.isError()) { return false; } return diffResult.getOkay() == 0; } const utf8* String::begin() const { return data(); } const utf8* String::end() const { return data() + size(); } } // namespace fud