[CC'ing Jason] On Wed, 25 Sep 2024, Patrick Palka wrote:
> On Wed, 31 Jul 2024, Giuseppe D'Angelo wrote: > > > On 29/07/2024 22:53, Giuseppe D'Angelo wrote: > > > Hi, > > > > > > The attached patch is a stab at adding the necessary compiler builtin to > > > support std::is_virtual_base_of (P2985R0, approved for C++26). The name > > > of the builtin matches the one just merged into clang: > > > > > > https://github.com/llvm/llvm-project/issues/98310 > > > > > > The next patch will add the libstdc++ bits. > > > > Hello, > > > > This is a new revision of the same patch, this time with a DCO. > > > > Thank you, > > -- > > Giuseppe D'Angelo > > > > From d4340076cffc9895336e2d7afe5c416acb9aac3f Mon Sep 17 00:00:00 2001 > > From: Giuseppe D'Angelo <giuseppe.dang...@kdab.com> > > Date: Mon, 29 Jul 2024 17:43:20 +0200 > > Subject: [PATCH 1/2] Introduce __builtin_is_virtual_base_of > > > > P2985R0 (C++26) introduces std::is_virtual_base_of; this is the compiler > > builtin that will back up the library trait (which strictly requires > > compiler support). > > > > The name has been chosen to match LLVM/MSVC's, as per the discussion > > here: > > > > https://github.com/llvm/llvm-project/issues/98310 > > > > The actual user-facing type trait in libstdc++ will be added later. > > > > gcc/cp/ChangeLog: > > > > * constraint.cc (diagnose_trait_expr): New diagnostic. > > * cp-trait.def (IS_VIRTUAL_BASE_OF): New built-in. > > * cp-tree.h (DERIVED_VIRTUALLY_FROM_P): New macro to search > > for virtual inheritance. > > (enum base_access_flags): Add a new flag to be able to > > request a search for a virtual base class. > > * cxx-pretty-print.cc (pp_cxx_userdef_literal): Update the > > list of GNU extensions to the grammar. > > * search.cc (struct lookup_base_data_s): Add a field to > > request searching for a virtual base class. > > (dfs_lookup_base): Add the ability to look for a virtual > > base class. > > (lookup_base): Forward the flag to dfs_lookup_base. > > * semantics.cc (trait_expr_value): Implement the builtin > > by using IS_VIRTUAL_BASE_OF. Unlike IS_BASE_OF, the same > > non-union class type is not a virtual base of itself. > > (finish_trait_expr): Handle the new builtin. > > > > gcc/ChangeLog: > > > > * doc/extend.texi: Document the new > > __builtin_is_virtual_base_of builtin. > > > > gcc/testsuite/ChangeLog: > > > > * g++.dg/ext/is_virtual_base_of.C: New test. > > * g++.dg/ext/is_virtual_base_of_diagnostic.C: New test. > > LGTM! (though I can't formally approve the patch) > > > > > Signed-off-by: Giuseppe D'Angelo <giuseppe.dang...@kdab.com> > > --- > > gcc/cp/constraint.cc | 3 + > > gcc/cp/cp-trait.def | 1 + > > gcc/cp/cp-tree.h | 8 +- > > gcc/cp/cxx-pretty-print.cc | 2 + > > gcc/cp/search.cc | 17 +- > > gcc/cp/semantics.cc | 11 ++ > > gcc/doc/extend.texi | 10 ++ > > gcc/testsuite/g++.dg/ext/is_virtual_base_of.C | 163 ++++++++++++++++++ > > .../ext/is_virtual_base_of_diagnostic.C | 15 ++ > > 9 files changed, 225 insertions(+), 5 deletions(-) > > create mode 100644 gcc/testsuite/g++.dg/ext/is_virtual_base_of.C > > create mode 100644 gcc/testsuite/g++.dg/ext/is_virtual_base_of_diagnostic.C > > > > diff --git a/gcc/cp/constraint.cc b/gcc/cp/constraint.cc > > index 853ea8c4b93..6d3a4a167aa 100644 > > --- a/gcc/cp/constraint.cc > > +++ b/gcc/cp/constraint.cc > > @@ -3600,6 +3600,9 @@ diagnose_trait_expr (tree expr, tree args) > > case CPTK_IS_UNION: > > inform (loc, " %qT is not a union", t1); > > break; > > + case CPTK_IS_VIRTUAL_BASE_OF: > > + inform (loc, " %qT is not a virtual base of %qT", t1, t2); > > + break; > > case CPTK_IS_VOLATILE: > > inform (loc, " %qT is not a volatile type", t1); > > break; > > diff --git a/gcc/cp/cp-trait.def b/gcc/cp/cp-trait.def > > index 0721ecdf3f0..6aaca13c5e9 100644 > > --- a/gcc/cp/cp-trait.def > > +++ b/gcc/cp/cp-trait.def > > @@ -100,6 +100,7 @@ DEFTRAIT_EXPR (IS_TRIVIALLY_CONSTRUCTIBLE, > > "__is_trivially_constructible", -1) > > DEFTRAIT_EXPR (IS_TRIVIALLY_COPYABLE, "__is_trivially_copyable", 1) > > DEFTRAIT_EXPR (IS_UNBOUNDED_ARRAY, "__is_unbounded_array", 1) > > DEFTRAIT_EXPR (IS_UNION, "__is_union", 1) > > +DEFTRAIT_EXPR (IS_VIRTUAL_BASE_OF, "__builtin_is_virtual_base_of", 2) > > DEFTRAIT_EXPR (IS_VOLATILE, "__is_volatile", 1) > > DEFTRAIT_EXPR (RANK, "__array_rank", 1) > > DEFTRAIT_EXPR (REF_CONSTRUCTS_FROM_TEMPORARY, > > "__reference_constructs_from_temporary", 2) > > diff --git a/gcc/cp/cp-tree.h b/gcc/cp/cp-tree.h > > index 238d786b067..6720be3ad7e 100644 > > --- a/gcc/cp/cp-tree.h > > +++ b/gcc/cp/cp-tree.h > > @@ -2330,6 +2330,11 @@ enum languages { lang_c, lang_cplusplus }; > > #define DERIVED_FROM_P(PARENT, TYPE) \ > > (lookup_base ((TYPE), (PARENT), ba_any, NULL, tf_none) != NULL_TREE) > > > > +/* Nonzero iff TYPE is derived virtually from PARENT. Ignores > > accessibility and > > + ambiguity issues. */ > > +#define DERIVED_VIRTUALLY_FROM_P(PARENT, TYPE) \ > > + (lookup_base ((TYPE), (PARENT), ba_require_virtual, NULL, tf_none) != > > NULL_TREE) > > + > > /* Gives the visibility specification for a class type. */ > > #define CLASSTYPE_VISIBILITY(TYPE) \ > > DECL_VISIBILITY (TYPE_MAIN_DECL (TYPE)) > > @@ -5727,7 +5732,8 @@ enum base_access_flags { > > ba_unique = 1 << 0, /* Must be a unique base. */ > > ba_check_bit = 1 << 1, /* Check access. */ > > ba_check = ba_unique | ba_check_bit, > > - ba_ignore_scope = 1 << 2 /* Ignore access allowed by local scope. */ > > + ba_ignore_scope = 1 << 2, /* Ignore access allowed by local scope. */ > > + ba_require_virtual = 1 << 3 /* Require a virtual base */ > > }; > > > > /* This type is used for parameters and variables which hold > > diff --git a/gcc/cp/cxx-pretty-print.cc b/gcc/cp/cxx-pretty-print.cc > > index e690354e08e..41e6bdfdda5 100644 > > --- a/gcc/cp/cxx-pretty-print.cc > > +++ b/gcc/cp/cxx-pretty-print.cc > > @@ -418,6 +418,8 @@ pp_cxx_userdef_literal (cxx_pretty_printer *pp, tree t) > > __builtin_offsetof ( type-id, offsetof-expression ) > > __builtin_addressof ( expression ) > > > > + __builtin_is_virtual_base_of ( type-id , type-id ) > > + > > __has_nothrow_assign ( type-id ) > > __has_nothrow_constructor ( type-id ) > > __has_nothrow_copy ( type-id ) > > diff --git a/gcc/cp/search.cc b/gcc/cp/search.cc > > index 827f48e8604..c391761482f 100644 > > --- a/gcc/cp/search.cc > > +++ b/gcc/cp/search.cc > > @@ -65,6 +65,7 @@ struct lookup_base_data_s > > bool repeated_base; /* Whether there are repeated bases in the > > hierarchy. */ > > bool want_any; /* Whether we want any matching binfo. */ > > + bool require_virtual; /* Whether we require a virtual path. */ > > }; > > > > /* Worker function for lookup_base. See if we've found the desired > > @@ -93,11 +94,18 @@ dfs_lookup_base (tree binfo, void *data_) > > > > if (SAME_BINFO_TYPE_P (BINFO_TYPE (binfo), data->base)) > > { > > + const bool via_virtual = > > + binfo_via_virtual (binfo, data->t) != NULL_TREE; > > + > > + if (data->require_virtual && !via_virtual) > > + /* Skip this result if we require virtual inheritance > > + and this is not a virtual base. */ > > + return dfs_skip_bases; > > + > > if (!data->binfo) > > { > > data->binfo = binfo; > > - data->via_virtual > > - = binfo_via_virtual (data->binfo, data->t) != NULL_TREE; > > + data->via_virtual = via_virtual; > > > > if (!data->repeated_base) > > /* If there are no repeated bases, we can stop now. */ > > @@ -124,7 +132,7 @@ dfs_lookup_base (tree binfo, void *data_) > > } > > > > /* Prefer one via a non-virtual path. */ > > - if (!binfo_via_virtual (binfo, data->t)) > > + if (!via_virtual) > > { > > data->binfo = binfo; > > data->via_virtual = false; > > @@ -265,8 +273,9 @@ lookup_base (tree t, tree base, base_access access, > > data.binfo = NULL_TREE; > > data.ambiguous = data.via_virtual = false; > > data.repeated_base = (offset == -1) && CLASSTYPE_REPEATED_BASE_P (t); > > - data.want_any = access == ba_any; > > + data.want_any = (access & ~ba_require_virtual) == ba_any; > > data.offset = offset; > > + data.require_virtual = (access & ba_require_virtual); > > > > dfs_walk_once (t_binfo, dfs_lookup_base, NULL, &data); > > binfo = data.binfo; > > diff --git a/gcc/cp/semantics.cc b/gcc/cp/semantics.cc > > index a9abf32e01f..e3a2168e297 100644 > > --- a/gcc/cp/semantics.cc > > +++ b/gcc/cp/semantics.cc > > @@ -12815,6 +12815,10 @@ trait_expr_value (cp_trait_kind kind, tree type1, > > tree type2) > > case CPTK_IS_UNION: > > return type_code1 == UNION_TYPE; > > > > + case CPTK_IS_VIRTUAL_BASE_OF: > > + return (NON_UNION_CLASS_TYPE_P (type1) && NON_UNION_CLASS_TYPE_P > > (type2) > > + && DERIVED_VIRTUALLY_FROM_P (type1, type2)); > > + > > case CPTK_IS_VOLATILE: > > return CP_TYPE_VOLATILE_P (type1); > > > > @@ -13037,6 +13041,13 @@ finish_trait_expr (location_t loc, cp_trait_kind > > kind, tree type1, tree type2) > > return error_mark_node; > > break; > > > > + case CPTK_IS_VIRTUAL_BASE_OF: > > + if (NON_UNION_CLASS_TYPE_P (type1) && NON_UNION_CLASS_TYPE_P (type2) > > + && !complete_type_or_else (type2, NULL_TREE)) > > + /* We already issued an error. */ > > + return error_mark_node; > > + break; > > + > > case CPTK_IS_ARRAY: > > case CPTK_IS_BOUNDED_ARRAY: > > case CPTK_IS_CLASS: > > diff --git a/gcc/doc/extend.texi b/gcc/doc/extend.texi > > index 927aa24ab63..4d3a99b51bf 100644 > > --- a/gcc/doc/extend.texi > > +++ b/gcc/doc/extend.texi > > @@ -29794,6 +29794,16 @@ type (disregarding cv-qualifiers), > > @var{derived_type} shall be a complete > > type. A diagnostic is produced if this requirement is not met. > > @enddefbuiltin > > > > +@defbuiltin{bool __builtin_is_virtual_base_of (@var{base_type}, > > @var{derived_type})} > > +If @var{base_type} is a virtual base class of @var{derived_type} > > +([class.derived]) then the trait is @code{true}, otherwise it is > > @code{false}. > > +Top-level cv-qualifications of @var{base_type} and > > +@var{derived_type} are ignored. > > +Requires: if @code{__is_class (base_type)} and @code{__is_class > > (derived_type)} > > +are @code{true}, @var{derived_type} shall be a complete > > +type. A diagnostic is produced if this requirement is not met. > > +@enddefbuiltin > > + > > @defbuiltin{bool __is_class (@var{type})} > > If @var{type} is a cv-qualified class type, and not a union type > > ([basic.compound]) the trait is @code{true}, else it is @code{false}. > > diff --git a/gcc/testsuite/g++.dg/ext/is_virtual_base_of.C > > b/gcc/testsuite/g++.dg/ext/is_virtual_base_of.C > > new file mode 100644 > > index 00000000000..775a3539388 > > --- /dev/null > > +++ b/gcc/testsuite/g++.dg/ext/is_virtual_base_of.C > > @@ -0,0 +1,163 @@ > > +// { dg-do run } > > +#include <cassert> > > + > > +class A1 > > +{ > > + double a; > > + double b; > > +}; > > + > > +class A2 > > +{ > > + double a; > > + double b; > > +}; > > + > > +class B > > +: private A1 { }; > > + > > +class C > > +: private A1, private A2 { }; > > + > > +class VD1 > > +: public virtual A1 { }; > > + > > +class VD2 > > +: protected virtual A1 { }; > > + > > +class VD3 > > +: private virtual A1 { }; > > + > > +class D > > +: public B, public VD1 { }; > > + > > +class VDMultiple1 > > +: public virtual A1, public A2 { }; > > + > > +class VDMultiple2 > > +: public virtual A1, public virtual A2 { }; > > + > > +template <typename T> > > +class VDTemplate : public virtual T { }; > > + > > +// example from from [class.mi] > > +namespace class_mi > > +{ > > +class B { int b; }; > > +class X : virtual public B { int x; }; > > +class Y : virtual public B { int y; }; > > +class Z : public B { int z; }; > > +class AA : public X, public Y, public Z { int aa; }; > > +} > > + > > +union U > > +{ > > + double a; > > + double b; > > +}; > > + > > +template<typename T, typename U> > > + bool > > + f() > > + { return __builtin_is_virtual_base_of(T, U); } > > + > > +template<typename T, typename U> > > + class My > > + { > > + public: > > + bool > > + f() > > + { return !!__builtin_is_virtual_base_of(T, U); } > > + }; > > + > > +template<typename T, typename U> > > + class My2 > > + { > > + public: > > + static const bool trait = __builtin_is_virtual_base_of(T, U); > > + }; > > + > > +template<typename T, typename U> > > + const bool My2<T, U>::trait; > > + > > +template<typename T, typename U, bool b = __builtin_is_virtual_base_of(T, > > U)> > > + struct My3_help > > + { static const bool trait = b; }; > > + > > +template<typename T, typename U, bool b> > > + const bool My3_help<T, U, b>::trait; > > + > > +template<typename T, typename U> > > + class My3 > > + { > > + public: > > + bool > > + f() > > + { return My3_help<T, U>::trait; } > > + }; > > + > > +#define PTEST(T, U) (__builtin_is_virtual_base_of(T, U) && f<T, U>() \ > > + && My<T, U>().f() && My2<T, U>::trait && My3<T, U>().f()) > > + > > +#define NTEST(T, U) (!__builtin_is_virtual_base_of(T, U) && !f<T, U>() \ > > + && !My<T, U>().f() && !My2<T, U>::trait && !My3<T, > > U>().f()) > > + > > +int main() > > +{ > > + assert (NTEST (int, A1)); > > + assert (NTEST (A1, void)); > > + assert (NTEST (A1, A1)); > > + assert (NTEST (A1*, A1*)); > > + assert (NTEST (A1&, A1&)); > > + assert (NTEST (A1, B)); > > + assert (NTEST (B, A1)); > > + assert (NTEST (A1, C)); > > + assert (NTEST (A2, C)); > > + assert (NTEST (C, A1)); > > + assert (NTEST (A1, const B)); > > + assert (NTEST (const B, A1)); > > + assert (NTEST (A1, volatile C)); > > + assert (NTEST (volatile A2, const C)); > > + assert (NTEST (const volatile C, A1)); > > + > > + assert (PTEST (A1, VD1)); > > + assert (PTEST (A1, VD2)); > > + assert (PTEST (A1, VD3)); > > + > > + assert (PTEST (A1, const VD1)); > > + assert (PTEST (A1, const VD2)); > > + assert (PTEST (A1, const VD3)); > > + > > + assert (PTEST (const A1, VD1)); > > + assert (PTEST (const A1, VD2)); > > + assert (PTEST (const A1, VD3)); > > + > > + assert (PTEST (const A1, const VD1)); > > + assert (PTEST (const A1, const VD2)); > > + assert (PTEST (const A1, const VD3)); > > + > > + assert (NTEST (A2, VD1)); > > + assert (NTEST (A2, VD2)); > > + assert (NTEST (A2, VD3)); > > + > > + assert (PTEST (A1, D)); > > + > > + assert (PTEST (A1, VDMultiple1)); > > + assert (PTEST (A1, VDMultiple2)); > > + assert (NTEST (A2, VDMultiple1)); > > + assert (PTEST (A2, VDMultiple2)); > > + > > + assert (PTEST (A1, VDTemplate<A1>)); > > + assert (NTEST (A2, VDTemplate<A1>)); > > + assert (NTEST (A1, VDTemplate<A2>)); > > + > > + assert (NTEST (class_mi::B, class_mi::B)); > > + assert (PTEST (class_mi::B, class_mi::X)); > > + assert (PTEST (class_mi::B, class_mi::Y)); > > + assert (NTEST (class_mi::B, class_mi::Z)); > > + assert (PTEST (class_mi::B, class_mi::AA)); > > + > > + assert (NTEST (U, U)); > > + > > + return 0; > > +} > > diff --git a/gcc/testsuite/g++.dg/ext/is_virtual_base_of_diagnostic.C > > b/gcc/testsuite/g++.dg/ext/is_virtual_base_of_diagnostic.C > > new file mode 100644 > > index 00000000000..a50459afdca > > --- /dev/null > > +++ b/gcc/testsuite/g++.dg/ext/is_virtual_base_of_diagnostic.C > > @@ -0,0 +1,15 @@ > > +class A > > +{ }; > > + > > +class B; // { dg-message "forward declaration" } > > + > > +union C > > +{ }; > > + > > +union D; > > + > > +void f() > > +{ > > + __builtin_is_virtual_base_of(A, B); // { dg-error "incomplete type" } > > + __builtin_is_virtual_base_of(C, D); > > +} > > -- > > 2.34.1 > > >