https://github.com/spaits updated https://github.com/llvm/llvm-project/pull/94752
From 8e1f3aa676b891c9566344ad2be046898df34a3a Mon Sep 17 00:00:00 2001 From: Gabor Spaits <gaborspai...@gmail.com> Date: Fri, 7 Jun 2024 14:38:08 +0200 Subject: [PATCH 1/4] [CTAD] Allow user defined conversion --- clang/include/clang/Sema/Initialization.h | 2 +- clang/lib/Sema/SemaInit.cpp | 6 +- .../SemaCXX/ctad-copy-init-list-narrowing.cpp | 90 +++++++++++++++++++ 3 files changed, 93 insertions(+), 5 deletions(-) create mode 100644 clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp diff --git a/clang/include/clang/Sema/Initialization.h b/clang/include/clang/Sema/Initialization.h index f443e327eaf32..4b876db436b48 100644 --- a/clang/include/clang/Sema/Initialization.h +++ b/clang/include/clang/Sema/Initialization.h @@ -603,7 +603,7 @@ class InitializationKind { /// Normal context IC_Normal, - /// Normal context, but allows explicit conversion functionss + /// Normal context, but allows explicit conversion functions IC_ExplicitConvs, /// Implicit context (value initialization) diff --git a/clang/lib/Sema/SemaInit.cpp b/clang/lib/Sema/SemaInit.cpp index ed8b226a6b39f..211b6887befa3 100644 --- a/clang/lib/Sema/SemaInit.cpp +++ b/clang/lib/Sema/SemaInit.cpp @@ -10892,8 +10892,6 @@ QualType Sema::DeduceTemplateSpecializationFromInitializer( // FIXME: The "second phase of [over.match.list] case can also // theoretically happen here, but it's not clear whether we can // ever have a parameter of the right type. - bool SuppressUserConversions = Kind.isCopyInit(); - if (TD) { SmallVector<Expr *, 8> TmpInits; for (Expr *E : Inits) @@ -10903,12 +10901,12 @@ QualType Sema::DeduceTemplateSpecializationFromInitializer( TmpInits.push_back(E); AddTemplateOverloadCandidate( TD, FoundDecl, /*ExplicitArgs=*/nullptr, TmpInits, Candidates, - SuppressUserConversions, + /*SuppressUserConversions=*/false, /*PartialOverloading=*/false, AllowExplicit, ADLCallKind::NotADL, /*PO=*/{}, AllowAggregateDeductionCandidate); } else { AddOverloadCandidate(GD, FoundDecl, Inits, Candidates, - SuppressUserConversions, + /*SuppressUserConversions=*/false, /*PartialOverloading=*/false, AllowExplicit); } }; diff --git a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp new file mode 100644 index 0000000000000..21b1137158d5a --- /dev/null +++ b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp @@ -0,0 +1,90 @@ +// RUN: %clang_cc1 -fsyntax-only -verify -Wno-unused-value -std=c++20 %s +namespace std +{ + typedef long unsigned int size_t; +} + +namespace std +{ + template<class _E> + class initializer_list + { + public: + typedef _E value_type; + typedef const _E& reference; + typedef const _E& const_reference; + typedef size_t size_type; + typedef const _E* iterator; + typedef const _E* const_iterator; + + private: + iterator _M_array; + size_type _M_len; + + + constexpr initializer_list(const_iterator __a, size_type __l) + : _M_array(__a), _M_len(__l) { } + + public: + constexpr initializer_list() noexcept + : _M_array(0), _M_len(0) { } + + + constexpr size_type + size() const noexcept { return _M_len; } + + + constexpr const_iterator + begin() const noexcept { return _M_array; } + + + constexpr const_iterator + end() const noexcept { return begin() + size(); } + }; + + template<class _Tp> + constexpr const _Tp* + begin(initializer_list<_Tp> __ils) noexcept + { return __ils.begin(); } + + template<class _Tp> + constexpr const _Tp* + end(initializer_list<_Tp> __ils) noexcept + { return __ils.end(); } +} + +template<class T, class Y> +class pair{ + private: + T fst; + Y snd; + public: + pair(T f, Y s) : fst(f), snd(s) {} +}; + +template<class T, class Y> +class map { + public: + map(std::initializer_list<pair<T, Y>>, int a = 4, int b = 5) {} +}; + +template<class T, class Y> +class Contained { + public: + Contained(T, Y) {} +}; + +template<class T, class Y> +class A { + public: + A(std::initializer_list<Contained<T, Y> >, int) {} +}; + +int main() { + map mOk ={pair{5, 'a'}, {6, 'b'}, {7, 'c'}}; + map mNarrow ={pair{5, 'a'}, {6.0f, 'b'}, {7, 'c'}}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} + + A aOk = {{Contained{5, 'c'}, {5, 'c'}}, 5}; + A aNarrowNested = {{Contained{5, 'c'}, {5.0f, 'c'}}, 5}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} + A aNarrow = {{Contained{5, 'c'}, {5, 'c'}}, 5.0f}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} +} From 8e33ac051a04e5d5c6d73efdd4cf47868d5dc644 Mon Sep 17 00:00:00 2001 From: Gabor Spaits <gaborspai...@gmail.com> Date: Sun, 9 Jun 2024 14:22:42 +0200 Subject: [PATCH 2/4] Simplify boiler plate and re-organize tests --- .../SemaCXX/ctad-copy-init-list-narrowing.cpp | 95 +++++++------------ 1 file changed, 35 insertions(+), 60 deletions(-) diff --git a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp index 21b1137158d5a..a2de4cf8a2f90 100644 --- a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp +++ b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp @@ -1,65 +1,25 @@ // RUN: %clang_cc1 -fsyntax-only -verify -Wno-unused-value -std=c++20 %s -namespace std -{ - typedef long unsigned int size_t; -} - -namespace std -{ - template<class _E> - class initializer_list - { - public: - typedef _E value_type; - typedef const _E& reference; - typedef const _E& const_reference; - typedef size_t size_type; - typedef const _E* iterator; - typedef const _E* const_iterator; - - private: - iterator _M_array; - size_type _M_len; - - - constexpr initializer_list(const_iterator __a, size_type __l) - : _M_array(__a), _M_len(__l) { } - - public: - constexpr initializer_list() noexcept - : _M_array(0), _M_len(0) { } - - - constexpr size_type - size() const noexcept { return _M_len; } - - constexpr const_iterator - begin() const noexcept { return _M_array; } +namespace std { + typedef decltype(sizeof(int)) size_t; + template <typename E> + struct initializer_list + { + const E *p; + size_t n; + initializer_list(const E *p, size_t n) : p(p), n(n) {} + }; - constexpr const_iterator - end() const noexcept { return begin() + size(); } - }; - - template<class _Tp> - constexpr const _Tp* - begin(initializer_list<_Tp> __ils) noexcept - { return __ils.begin(); } - - template<class _Tp> - constexpr const _Tp* - end(initializer_list<_Tp> __ils) noexcept - { return __ils.end(); } -} + struct string { + string(const char *); + }; + // Classes to use to reproduce the exact scenario present in 62925. template<class T, class Y> class pair{ - private: - T fst; - Y snd; public: - pair(T f, Y s) : fst(f), snd(s) {} + pair(T f, Y s) {} }; template<class T, class Y> @@ -68,6 +28,10 @@ class map { map(std::initializer_list<pair<T, Y>>, int a = 4, int b = 5) {} }; +} // namespace std + + +// Classes to test different levels of nestings and conversions. template<class T, class Y> class Contained { public: @@ -80,11 +44,22 @@ class A { A(std::initializer_list<Contained<T, Y> >, int) {} }; -int main() { - map mOk ={pair{5, 'a'}, {6, 'b'}, {7, 'c'}}; - map mNarrow ={pair{5, 'a'}, {6.0f, 'b'}, {7, 'c'}}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} - A aOk = {{Contained{5, 'c'}, {5, 'c'}}, 5}; - A aNarrowNested = {{Contained{5, 'c'}, {5.0f, 'c'}}, 5}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} - A aNarrow = {{Contained{5, 'c'}, {5, 'c'}}, 5.0f}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} +// This is the almost the exact code that was in issue #62925. +void testOneLevelNesting() { + std::map mOk = {std::pair{5, 'a'}, {6, 'b'}, {7, 'c'}}; + + // Verify that narrowing conversion is disabled in the first level of nesting. + std::map mNarrow = {std::pair{5, 'a'}, {6.0f, 'b'}, {7, 'c'}}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} +} + +void testMultipleLevelNesting() { + A aOk = {{Contained{5, 'c'}, {5, 'c'}}, 5}; + + // Verify that narrowing conversion is disabled when it is not in a nested + // in another std::initializer_list, but it happens in the most outer one. + A aNarrowNested = {{Contained{5, 'c'}, {5.0f, 'c'}}, 5}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} + + // Verify that narrowing conversion is disabled in the first level of nesting. + A aNarrow = {{Contained{5, 'c'}, {5, 'c'}}, 5.0f}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} } From 307615536e00d6fd9f679e8d3cfc9050fc51bf87 Mon Sep 17 00:00:00 2001 From: Gabor Spaits <gaborspai...@gmail.com> Date: Tue, 18 Jun 2024 12:43:40 +0200 Subject: [PATCH 3/4] Simplify test --- .../SemaCXX/ctad-copy-init-list-narrowing.cpp | 30 ++++--------------- 1 file changed, 6 insertions(+), 24 deletions(-) diff --git a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp index a2de4cf8a2f90..c58a204aee254 100644 --- a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp +++ b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp @@ -11,11 +11,7 @@ namespace std { initializer_list(const E *p, size_t n) : p(p), n(n) {} }; - struct string { - string(const char *); - }; - - // Classes to use to reproduce the exact scenario present in 62925. +// Classes to use to reproduce the exact scenario present in #62925. template<class T, class Y> class pair{ public: @@ -25,26 +21,12 @@ class pair{ template<class T, class Y> class map { public: - map(std::initializer_list<pair<T, Y>>, int a = 4, int b = 5) {} + map(std::initializer_list<pair<T, Y>>) {} + map(std::initializer_list<pair<T, Y>>, int a) {} }; } // namespace std - -// Classes to test different levels of nestings and conversions. -template<class T, class Y> -class Contained { - public: - Contained(T, Y) {} -}; - -template<class T, class Y> -class A { - public: - A(std::initializer_list<Contained<T, Y> >, int) {} -}; - - // This is the almost the exact code that was in issue #62925. void testOneLevelNesting() { std::map mOk = {std::pair{5, 'a'}, {6, 'b'}, {7, 'c'}}; @@ -54,12 +36,12 @@ void testOneLevelNesting() { } void testMultipleLevelNesting() { - A aOk = {{Contained{5, 'c'}, {5, 'c'}}, 5}; + std::map aOk = {{std::pair{5, 'c'}, {5, 'c'}}, 5}; // Verify that narrowing conversion is disabled when it is not in a nested // in another std::initializer_list, but it happens in the most outer one. - A aNarrowNested = {{Contained{5, 'c'}, {5.0f, 'c'}}, 5}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} + std::map aNarrowNested = {{std::pair{5, 'c'}, {5.0f, 'c'}}, 5}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} // Verify that narrowing conversion is disabled in the first level of nesting. - A aNarrow = {{Contained{5, 'c'}, {5, 'c'}}, 5.0f}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} + std::map aNarrow = {{std::pair{5, 'c'}, {5, 'c'}}, 5.0f}; // expected-error {{type 'float' cannot be narrowed to 'int' in initializer list}} // expected-note {{insert an explicit cast to silence this issue}} } From 44b6f2c840f91e39b1f59f62446b0f762f30bf2f Mon Sep 17 00:00:00 2001 From: Gabor Spaits <gaborspai...@gmail.com> Date: Tue, 18 Jun 2024 12:46:13 +0200 Subject: [PATCH 4/4] Format by hand --- .../SemaCXX/ctad-copy-init-list-narrowing.cpp | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp index c58a204aee254..368857c83a88e 100644 --- a/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp +++ b/clang/test/SemaCXX/ctad-copy-init-list-narrowing.cpp @@ -1,19 +1,18 @@ // RUN: %clang_cc1 -fsyntax-only -verify -Wno-unused-value -std=c++20 %s namespace std { - typedef decltype(sizeof(int)) size_t; +typedef decltype(sizeof(int)) size_t; - template <typename E> - struct initializer_list - { - const E *p; - size_t n; - initializer_list(const E *p, size_t n) : p(p), n(n) {} - }; +template <typename E> +struct initializer_list { + const E *p; + size_t n; + initializer_list(const E *p, size_t n) : p(p), n(n) {} +}; // Classes to use to reproduce the exact scenario present in #62925. template<class T, class Y> -class pair{ +class pair { public: pair(T f, Y s) {} }; _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits