From da2f1ed6373ed72ff141ac491a27c4a075042f90 Mon Sep 17 00:00:00 2001 From: Mark de Wever Date: Sun, 4 Feb 2024 18:52:19 +0100 Subject: [PATCH] [libc++][test] Fixes constexpr nasty_char_traits. This was discovered by @StephanTLavavej who provided the solution they use in MSVC STL. This solution is based on that example. The same issue affects the constexpr_char_traits which was discovered in https://github.com/llvm/llvm-project/pull/88389. This uses the same fix. Fixes: https://github.com/llvm/llvm-project/issues/74221 --- libcxx/test/support/constexpr_char_traits.h | 57 +++++++++++++++------ libcxx/test/support/nasty_string.h | 12 +++-- 2 files changed, 48 insertions(+), 21 deletions(-) diff --git a/libcxx/test/support/constexpr_char_traits.h b/libcxx/test/support/constexpr_char_traits.h index 75380d5a7ffbb3..7c487c504af133 100644 --- a/libcxx/test/support/constexpr_char_traits.h +++ b/libcxx/test/support/constexpr_char_traits.h @@ -16,6 +16,31 @@ #include "test_macros.h" +// Tests whether the pointer p is in the range [first, last). +// +// Precondition: The range [first, last) is a valid range. +// +// Typically the pointers are compared with less than. This is not allowed when +// the pointers belong to different ranges, which is UB. Typically, this is +// benign at run-time, however since UB is not allowed during constant +// evaluation this does not compile. This function does the validation without +// UB. +// +// When p is in the range [first, last) the data can be copied from the +// beginning to the end. Otherwise it needs to be copied from the end to the +// beginning. +template +TEST_CONSTEXPR_CXX14 bool is_pointer_in_range(const CharT* first, const CharT* last, const CharT* p) { + if (first == p) // Needed when n == 0 + return true; + + for (; first != last; ++first) + if (first == p) + return true; + + return false; +} + template struct constexpr_char_traits { @@ -98,23 +123,21 @@ constexpr_char_traits::find(const char_type* s, std::size_t n, const char } template -TEST_CONSTEXPR_CXX14 CharT* -constexpr_char_traits::move(char_type* s1, const char_type* s2, std::size_t n) -{ - char_type* r = s1; - if (s1 < s2) - { - for (; n; --n, ++s1, ++s2) - assign(*s1, *s2); - } - else if (s2 < s1) - { - s1 += n; - s2 += n; - for (; n; --n) - assign(*--s1, *--s2); - } - return r; +TEST_CONSTEXPR_CXX14 CharT* constexpr_char_traits::move(char_type* s1, const char_type* s2, std::size_t n) { + if (s1 == s2) + return s1; + + char_type* r = s1; + if (is_pointer_in_range(s1, s1 + n, s2)) { + for (; n; --n) + assign(*s1++, *s2++); + } else { + s1 += n; + s2 += n; + for (; n; --n) + assign(*--s1, *--s2); + } + return r; } template diff --git a/libcxx/test/support/nasty_string.h b/libcxx/test/support/nasty_string.h index 672c3cb4ed9ea5..ea9d83ccf282aa 100644 --- a/libcxx/test/support/nasty_string.h +++ b/libcxx/test/support/nasty_string.h @@ -16,6 +16,7 @@ #include "make_string.h" #include "test_macros.h" +#include "constexpr_char_traits.h" // is_pointer_in_range // This defines a nasty_string similar to nasty_containers. This string's // value_type does operator hijacking, which allows us to ensure that the @@ -118,11 +119,14 @@ constexpr const nasty_char* nasty_char_traits::find(const nasty_char* s, std::si } constexpr nasty_char* nasty_char_traits::move(nasty_char* s1, const nasty_char* s2, std::size_t n) { + if (s1 == s2) + return s1; + nasty_char* r = s1; - if (s1 < s2) { - for (; n; --n, ++s1, ++s2) - assign(*s1, *s2); - } else if (s2 < s1) { + if (is_pointer_in_range(s1, s1 + n, s2)) { + for (; n; --n) + assign(*s1++, *s2++); + } else { s1 += n; s2 += n; for (; n; --n)