From e94db4695e236b42ae1be44b2605075161d5144f Mon Sep 17 00:00:00 2001 From: Dominick Allen Date: Fri, 18 Oct 2024 21:43:44 -0500 Subject: Add temporary work for formatting. --- .clang-format | 1 + .gitignore | 1 + include/fud_assert.hpp | 13 -- include/fud_format.hpp | 293 ++++++++++++++++++++++++++++++++++++++++++++ include/fud_span.hpp | 16 +++ include/fud_string_view.hpp | 16 ++- include/fud_utf8.hpp | 2 +- source/fud_assert.cpp | 55 +++------ test/CMakeLists.txt | 1 + test/test_format.cpp | 35 ++++++ 10 files changed, 377 insertions(+), 56 deletions(-) create mode 100644 include/fud_format.hpp create mode 100644 test/test_format.cpp diff --git a/.clang-format b/.clang-format index 50883a6..87aff92 100644 --- a/.clang-format +++ b/.clang-format @@ -30,6 +30,7 @@ AlignConsecutiveMacros: PadOperators: false AlignEscapedNewlines: Left AlignOperands: false +AlignTrailingComments: Never AllowAllArgumentsOnNextLine: false AllowAllParametersOfDeclarationOnNextLine: false AllowShortBlocksOnASingleLine: Never diff --git a/.gitignore b/.gitignore index 40a320a..a7abdc1 100644 --- a/.gitignore +++ b/.gitignore @@ -12,3 +12,4 @@ dist/ .#* fudenv/ CMakeLists.txt.user +*.pch diff --git a/include/fud_assert.hpp b/include/fud_assert.hpp index ecfaa2f..6b21fdc 100644 --- a/include/fud_assert.hpp +++ b/include/fud_assert.hpp @@ -22,22 +22,9 @@ // #include #include -#include "fud_span.hpp" namespace fud { -constexpr std::size_t MAX_FILE_CHARS = 256; -constexpr std::size_t MAX_FUNCTION_CHARS = 256; -constexpr std::size_t BITS_PER_OCTAL = 3; -constexpr auto MAX_LINE_CHARS = BITS_PER_OCTAL * sizeof(decltype(std::source_location{}.line())) + 3; -constexpr std::size_t MAX_ASSERT_CHARS = 512 - MAX_LINE_CHARS; -constexpr std::size_t ASSERT_MSG_SIZE = MAX_FILE_CHARS + MAX_LINE_CHARS + MAX_FUNCTION_CHARS + MAX_ASSERT_CHARS; - -void assertFormat( - const char* assertion, - std::source_location sourceLocation, - Span buffer); - [[noreturn]] void assertFail( const char* assertion, std::source_location sourceLocation = std::source_location::current()); diff --git a/include/fud_format.hpp b/include/fud_format.hpp new file mode 100644 index 0000000..8985faf --- /dev/null +++ b/include/fud_format.hpp @@ -0,0 +1,293 @@ +/* + * 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. + */ + +#ifndef FUD_FORMAT_HPP +#define FUD_FORMAT_HPP + +#include "fud_assert.hpp" +#include "fud_result.hpp" +#include "fud_span.hpp" +#include "fud_status.hpp" +#include "fud_string_view.hpp" + +#include // for std::format_string +#include +#include + +namespace fud { + +template +using CharSpan = Span; + +template +using FormatLiteral = std::format_string; + +template +Result format(CharSpan buffer, FormatLiteral formatLiteral, Args&&... args); + +enum class FormatAlign : uint8_t { + Left, + Right, + Center +}; + +struct FormatFill { + FormatAlign align; + char fill; +}; + +enum class FormatSign : uint8_t { + Plus, + Minus, + Space +}; + +struct FormatSpec { + std::optional fill; + std::optional formatSign; + uint32_t minWidth; +}; + +namespace detail { + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView, + Arg&& arg, + Args&&... args); + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView, + Arg&& arg); + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView); + +} // namespace detail + +template +Result format(CharSpan buffer, FormatLiteral formatLiteral, Args&&... args) +{ + static_assert(Size > 0); + + if (buffer.data() == nullptr) { + return FudStatus::NullPointer; + } + + StringView formatView{formatLiteral.get()}; + + if (formatView.length() == 0 || formatView.data()[0] == '\0') { + return 0U; + } + + size_t argCount = sizeof...(args); + static_cast(argCount); + + size_t formattedSize = 0; + + return detail::formatHelper(buffer, formattedSize, formatView, std::forward(args)...); +} + +namespace detail { + +#define FUDETAIL_ADVANCE_FORMAT(FORMAT_VIEW, ADVANCE_BY) \ + fudAssert(ADVANCE_BY <= FORMAT_VIEW.m_length); \ + FORMAT_VIEW.m_length -= ADVANCE_BY; \ + FORMAT_VIEW.m_data += ADVANCE_BY; \ + +constexpr bool findBracket(size_t& copyLength, StringView formatView) +{ + while (copyLength < formatView.m_length) { + if (formatView.m_data[copyLength] == '{') { + return true; + } + copyLength++; + } + + return false; +} + +template +size_t copyRemaining( + CharSpan buffer, + size_t formattedSize, + StringView& formatView, + size_t copyLength) +{ + fudAssert(copyLength <= formatView.length()); + if (copyLength + formattedSize > Size) { + copyLength = Size - formattedSize; + } + auto copyResult = copyMem( + buffer.data() + formattedSize, + Size - formattedSize, + formatView.m_data, + copyLength); + fudAssert(copyResult == FudStatus::Success); + FUDETAIL_ADVANCE_FORMAT(formatView, copyLength); + return formattedSize + copyLength; +} + +template +Result handleSpec( + CharSpan buffer, + size_t formattedSize, + StringView& formatView, + Arg&& arg, + bool& consumed) +{ + fudAssert(formattedSize < Size); + fudAssert(formatView.length() > 1); + + if (formatView.m_data[1] == '{') { + consumed = false; + buffer[formattedSize] = '{'; + FUDETAIL_ADVANCE_FORMAT(formatView, 2); + return formattedSize + 1; + } + + static_cast(arg); + buffer[formattedSize] = 'X'; + formattedSize += 1; + size_t index = 0; + for (; index < formatView.m_length; ++index) { + if (formatView.m_data[index] == '}') { + break; + } + } + FUDETAIL_ADVANCE_FORMAT(formatView, index + 1); + return formattedSize; +} + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView, + Arg&& arg, + Args&&... args) +{ + while (formattedSize < Size) { + size_t copyLength = 0; + auto found = findBracket(copyLength, formatView); + formattedSize = copyRemaining(buffer, formattedSize, formatView, copyLength); + fudAssert(formattedSize <= Size); + if (!found || formattedSize == Size) { + return formattedSize; + } + + bool consumed = false; + auto specResult = handleSpec(buffer, formattedSize, formatView, std::forward(arg), consumed); + formattedSize = M_TakeOrReturn(specResult); + fudAssert(formattedSize <= Size); + if (formattedSize == Size) { + return formattedSize; + } + + if (consumed) { + return formatHelper(buffer, formattedSize, formatView, std::forward(args)...); + } + } + + return formattedSize; +} + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView, + Arg&& arg) +{ + while (formattedSize < Size) { + size_t copyLength = 0; + auto found = findBracket(copyLength, formatView); + formattedSize = copyRemaining(buffer, formattedSize, formatView, copyLength); + fudAssert(formattedSize <= Size); + if (!found || formattedSize == Size) { + return formattedSize; + } + + bool consumed = false; + auto specResult = handleSpec(buffer, formattedSize, formatView, std::forward(arg), consumed); + formattedSize = M_TakeOrReturn(specResult); + if (consumed) { + return formatHelper(buffer, formattedSize, formatView); + } + } + return formattedSize; +} + +template +Result formatHelper( + CharSpan buffer, + size_t formattedSize, + StringView formatView) +{ + size_t index = 0; + while (formattedSize < Size && formatView.m_length > 0) { + while (index < formatView.m_length && formattedSize + index < Size) { + if (formatView.m_data[index] == '{') { + break; + } + index++; + } + bool isBracket{false}; + if (index + 1 < formatView.m_length && formattedSize + index + 1 < Size) { + if (formatView.m_data[index] == '{') { + isBracket = true; + index++; + } + } + auto copyResult = copyMem( + buffer.data() + formattedSize, + Size - formattedSize, + formatView.m_data, + index); + formattedSize += index; + formatView.m_length -= index; + formatView.m_data += index; + if (isBracket) { + index = 0; + if (formatView.m_length > 0) { + formatView.m_length--; + formatView.m_data++; + } + if (formattedSize < Size) { + buffer.data()[formattedSize] = 'X'; + formattedSize++; + } + } + } + return formattedSize; +} + +#undef FUDETAIL_ADVANCE_FORMAT + +} // namespace detail + +} // namespace fud + +#endif diff --git a/include/fud_span.hpp b/include/fud_span.hpp index e8c5704..cc693f8 100644 --- a/include/fud_span.hpp +++ b/include/fud_span.hpp @@ -43,6 +43,22 @@ struct Span { return output; } + template + static Span make(Array& array) { + static_assert(std::convertible_to); + Span output{}; + output.m_data = array.data(); + return output; + } + + template + static Span makeCStringBuffer(Array& array) { + static_assert(ArraySize > Size); + Span output{}; + output.m_data = array.data(); + return output; + } + T* m_data; [[nodiscard]] constexpr size_t size() const diff --git a/include/fud_string_view.hpp b/include/fud_string_view.hpp index 4796003..7b4925e 100644 --- a/include/fud_string_view.hpp +++ b/include/fud_string_view.hpp @@ -15,23 +15,27 @@ * limitations under the License. */ - #ifndef FUD_STRING_VIEW_HPP #define FUD_STRING_VIEW_HPP #include "fud_status.hpp" #include "fud_utf8.hpp" +#include + namespace fud { class String; -class StringView { - public: +struct StringView { constexpr StringView() noexcept = default; + constexpr StringView(const StringView& rhs) noexcept = default; + constexpr StringView(StringView&& rhs) noexcept = default; + constexpr ~StringView() noexcept = default; + constexpr StringView& operator=(const StringView& rhs) = default; constexpr StringView& operator=(StringView&& rhs) = default; @@ -40,11 +44,13 @@ class StringView { } StringView(size_t strLen, const char* strData) : - m_length(strLen), // line break + m_length(strLen), // line break m_data{reinterpret_cast(strData)} // NOLINT(cppcoreguidelines-pro-type-reinterpret-cast) { } + StringView(std::string_view rhs) noexcept : StringView(rhs.length(), rhs.data()) {} + explicit StringView(const String& fudString) noexcept; [[nodiscard]] constexpr size_t length() const @@ -85,8 +91,8 @@ class StringView { FudStatus toDouble(double& number, size_t& strLen) const; - private: size_t m_length{0}; + const utf8* m_data{nullptr}; }; diff --git a/include/fud_utf8.hpp b/include/fud_utf8.hpp index 058e4f9..539e0f4 100644 --- a/include/fud_utf8.hpp +++ b/include/fud_utf8.hpp @@ -31,7 +31,7 @@ namespace fud { using utf8 = unsigned char; class String; -class StringView; +struct StringView; constexpr utf8 ASCII_MASK = 0x7F; diff --git a/source/fud_assert.cpp b/source/fud_assert.cpp index 14bf5eb..98f17d0 100644 --- a/source/fud_assert.cpp +++ b/source/fud_assert.cpp @@ -1,8 +1,6 @@ #include "fud_assert.hpp" #include "fud_array.hpp" -#include "fud_span.hpp" -#include "fud_string_view.hpp" #include #include @@ -10,46 +8,29 @@ namespace fud { -void assertFormat( - const char* assertion, - const std::source_location sourceLocation, - Span buffer) -{ - auto lengthResult = cStringLength(sourceLocation.file_name()); - size_t filenameLength = 0; - auto badLength = lengthResult < 1 || lengthResult > SSIZE_MAX; - if (!badLength) { - filenameLength = static_cast(lengthResult); - } - const char* filename = nullptr; - if (badLength) { - constexpr const char invalidFile[] = "INVALID FILE"; - static_assert(sizeof(invalidFile) < MAX_FILE_CHARS); - filenameLength = sizeof(invalidFile); - filename = invalidFile; - } else if (filenameLength > MAX_LINE_CHARS) { - filename = sourceLocation.file_name() + filenameLength - MAX_LINE_CHARS; - filenameLength = MAX_FILE_CHARS; - } - - // clang-format off - static_cast(std::format_to_n( - buffer.data(), buffer.size() - 1U, - "{:.{}s}:{}: {:.{}s}: Assertion `{:.{}s}` failed\n", - filename, filenameLength, - sourceLocation.line(), - sourceLocation.function_name(), MAX_FUNCTION_CHARS, - assertion, MAX_ASSERT_CHARS)); - // clang-format on -} +constexpr std::size_t BITS_PER_OCTAL = 3; +constexpr auto MAX_LINE_CHARS = BITS_PER_OCTAL * sizeof(decltype(std::source_location{}.line())) + 3; [[noreturn]] void assertFail(const char* assertion, const std::source_location sourceLocation) { - Array buffer{}; - assertFormat(assertion, sourceLocation, Span::make(buffer)); + const char* file_name = sourceLocation.file_name(); + if (file_name == nullptr) { + fputs("Unknown file", stderr); + } else { + fputs(file_name, stderr); + } + + constexpr std::size_t assertMsgSize = MAX_LINE_CHARS + 3; + Array buffer{}; + static_cast(std::format_to_n(buffer.data(), buffer.size() - 1U, ":{}:", sourceLocation.line())); buffer[buffer.size() - 1] = '\0'; fputs(buffer.data(), stderr); - std::abort(); + + fputs(sourceLocation.function_name(), stderr); + fputs(": ", stderr); + fputs(assertion, stderr); + + std::terminate(); } } // namespace fud diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index a2fcf7b..0860c0d 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -60,6 +60,7 @@ fud_add_test(test_fud SOURCES test_fud.cpp) fud_add_test(test_allocator SOURCES test_allocator.cpp) fud_add_test(test_assert SOURCES test_assert.cpp) fud_add_test(test_directory SOURCES test_directory.cpp) +fud_add_test(test_format SOURCES test_format.cpp) fud_add_test(test_result SOURCES test_result.cpp) fud_add_test(test_span SOURCES test_span.cpp) fud_add_test(test_sqlite SOURCES test_sqlite.cpp) diff --git a/test/test_format.cpp b/test/test_format.cpp new file mode 100644 index 0000000..a373fec --- /dev/null +++ b/test/test_format.cpp @@ -0,0 +1,35 @@ +/* + * 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_array.hpp" +#include "fud_format.hpp" +#include "fud_span.hpp" + +#include "gtest/gtest.h" + +namespace fud { + +TEST(FormatTest, BasicTest) +{ + auto buffer{Array::constFill('\0')}; + auto span = Span::makeCStringBuffer(buffer); + + auto formatResult = format(span, "Hello, {}! {}", "world", 42); + printf("%s\n", buffer.data()); +} + +} // namespace fud -- cgit v1.2.3