From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from us-smtp-delivery-124.mimecast.com (us-smtp-delivery-124.mimecast.com [170.10.129.124]) by sourceware.org (Postfix) with ESMTPS id C303C3858425 for ; Tue, 17 Oct 2023 17:05:03 +0000 (GMT) DMARC-Filter: OpenDMARC Filter v1.4.2 sourceware.org C303C3858425 Authentication-Results: sourceware.org; dmarc=pass (p=none dis=none) header.from=redhat.com Authentication-Results: sourceware.org; spf=pass smtp.mailfrom=redhat.com ARC-Filter: OpenARC Filter v1.0.0 sourceware.org C303C3858425 Authentication-Results: server2.sourceware.org; arc=none smtp.remote-ip=170.10.129.124 ARC-Seal: i=1; a=rsa-sha256; d=sourceware.org; s=key; t=1697562306; cv=none; b=upxHj1AyeZ8HK+4ihHIaKQ1/peXSM94qCjKMKpEDP8yfHi5tuJFzEVBslPURx0yeHCwqkqVL7W9yGLdRZ5aqYSzR5izh7+JCK7RJahdVczCLjr1gDCFQpQV2YhC3XTZH4HZjmAa1dTOE9TaOas+B0ywt/1bxxBo2dw6HxhZgiws= ARC-Message-Signature: i=1; a=rsa-sha256; d=sourceware.org; s=key; t=1697562306; c=relaxed/simple; bh=USGVJIxcEHdN+KEJuri1AQnSqpqaVVahZ3ZgxgF7CbA=; h=DKIM-Signature:From:Date:To:Subject:Message-ID:MIME-Version; b=EQAZ2TovTDf3Yb4nRnqAhOVE7y8ch4H115v71EavacDAyT5851ShYo0JaJp16YEx4sLSffL/I4zkpJoHpOhadFA/LAnV2ecpG6UJH1VeMDayZWq2xdXc9KziscxuhEu08CjFqMzUtfDxC/8wEQySapD/h/jQU+iMS/G50MBWpmg= ARC-Authentication-Results: i=1; server2.sourceware.org DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=redhat.com; s=mimecast20190719; t=1697562303; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version:content-type:content-type: in-reply-to:in-reply-to:references:references; bh=L1rSCWc0yZePzTi7U+xHau0vbHTMtqEwLffvRMOLlIY=; b=gfVz5F2VeQoA0GeFp4vGcZwnERytPvDGvBD/94EPuQgJTjtIICuFgMfzrcxaYJ3KQjthcH tO6eq66VP5vp8aUVj2FQNzU82UG/YM7onkS46E9RpmkjiZKnPr24hCbWemRs8J/bUvC3j/ tY7BXNQ1yn1yRIFcsFkXKVjc0xUuwik= Received: from mail-qk1-f199.google.com (mail-qk1-f199.google.com [209.85.222.199]) by relay.mimecast.com with ESMTP with STARTTLS (version=TLSv1.3, cipher=TLS_AES_256_GCM_SHA384) id us-mta-169-01NONNBNP-C0MhWRkOXKbQ-1; Tue, 17 Oct 2023 13:04:41 -0400 X-MC-Unique: 01NONNBNP-C0MhWRkOXKbQ-1 Received: by mail-qk1-f199.google.com with SMTP id af79cd13be357-775751c3437so770782485a.3 for ; Tue, 17 Oct 2023 10:04:41 -0700 (PDT) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1697562281; x=1698167081; h=mime-version:references:message-id:in-reply-to:subject:cc:to:date :from:x-gm-message-state:from:to:cc:subject:date:message-id:reply-to; bh=L1rSCWc0yZePzTi7U+xHau0vbHTMtqEwLffvRMOLlIY=; b=pGyW96nJdT6svf2Y8JDrU4TGtfK32xi+5aTt5L5bDvyOmP9KSvPro6zB5P+/EYEg2d 606ktBwiDohBW55j2L7PyJSo4Lwj3mTef/6zPr72w14TmXHrObK1EeTL/gFBnRvEuPun Xgddb7O4Zh5bK4YETKLlH5KErRnUrT0wB12aKRku+1al5eWHn+gRlKkoh/PW1R2P7mAZ OYn23mo547Y9G8vCsvBuzIrp1/6B821Li3rCbkRWdXJyaxu6cVNPUce9qCjCwJVb9KAk 8wOfTS0lXrQ1Jai89kzJwo8nHYc97f5Uc5ufD6F0RL4wATo+VYx65rmJSU5fzNYO7kur N52Q== X-Gm-Message-State: AOJu0YxMzr0qGpWc3TGVJUCbhW3yifmmSSUj5d2lTF2+b4TfqSf+M0WH Pl8CVAXX8PzLLGYECDb/1ZzEq6cxo6gGFpQ8Jb3DavCKiBRtlQ/yMY/4j/9ooUbPAep3yzUzpvH JhiG2/QFvkMwXE5q+cw== X-Received: by 2002:a05:620a:1a08:b0:76e:fba8:7565 with SMTP id bk8-20020a05620a1a0800b0076efba87565mr3061200qkb.62.1697562281153; Tue, 17 Oct 2023 10:04:41 -0700 (PDT) X-Google-Smtp-Source: AGHT+IGXy1cNu4uB9/JPMs1MxMnOkkVJQUKce3l2feFWUklJaTi8CZO1Fmypj9dOrTLrOL780AZhrQ== X-Received: by 2002:a05:620a:1a08:b0:76e:fba8:7565 with SMTP id bk8-20020a05620a1a0800b0076efba87565mr3061154qkb.62.1697562280630; Tue, 17 Oct 2023 10:04:40 -0700 (PDT) Received: from [192.168.1.130] (ool-457670bb.dyn.optonline.net. [69.118.112.187]) by smtp.gmail.com with ESMTPSA id h5-20020a05620a400500b0077413b342e9sm800768qko.128.2023.10.17.10.04.40 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 17 Oct 2023 10:04:40 -0700 (PDT) From: Patrick Palka X-Google-Original-From: Patrick Palka Date: Tue, 17 Oct 2023 13:04:39 -0400 (EDT) To: Ken Matsui cc: gcc-patches@gcc.gnu.org, libstdc++@gcc.gnu.org, Patrick Palka Subject: Re: [PATCH v22 02/31] c-family, c++: Look up built-in traits via identifier node In-Reply-To: <20231017113822.677344-3-kmatsui@gcc.gnu.org> Message-ID: <7649bb7d-7573-717c-abd9-1d8a16c3305c@idea> References: <20231017113242.664523-1-kmatsui@gcc.gnu.org> <20231017113822.677344-1-kmatsui@gcc.gnu.org> <20231017113822.677344-3-kmatsui@gcc.gnu.org> MIME-Version: 1.0 X-Mimecast-Spam-Score: 0 X-Mimecast-Originator: redhat.com Content-Type: text/plain; charset=US-ASCII X-Spam-Status: No, score=-13.0 required=5.0 tests=BAYES_00,DKIMWL_WL_HIGH,DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,GIT_PATCH_0,RCVD_IN_DNSWL_NONE,RCVD_IN_MSPIKE_H4,RCVD_IN_MSPIKE_WL,RCVD_IN_SORBS_WEB,SPF_HELO_NONE,SPF_NONE,TXREP autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on server2.sourceware.org List-Id: On Tue, 17 Oct 2023, Ken Matsui wrote: > Since RID_MAX soon reaches 255 and all built-in traits are used approximately > once in a C++ translation unit, this patch removes all RID values for built-in > traits and uses the identifier node to look up the specific trait. Rather > than holding traits as keywords, we set all trait identifiers as cik_trait, > which is a new cp_identifier_kind. As cik_reserved_for_udlit was unused and > cp_identifier_kind is 3 bits, we replaced the unused field with the new > cik_trait. Also, the later patch handles a subsequent token to the built-in > identifier so that we accept the use of non-function-like built-in trait > identifiers. > > gcc/c-family/ChangeLog: > > * c-common.cc (c_common_reswords): Remove all mappings of > built-in traits. > * c-common.h (enum rid): Remove all RID values for built-in traits. > > gcc/cp/ChangeLog: > > * cp-objcp-common.cc (names_builtin_p): Remove all RID value > cases for built-in traits. Check for built-in traits via > the new cik_trait kind. > * cp-tree.h (enum cp_trait_kind): Set its underlying type to > addr_space_t. > (struct cp_trait): New struct to hold trait information. > (cp_traits): New array to hold a mapping to all traits. > (num_cp_traits): New variable to hold the size of cp_traits. > (cik_reserved_for_udlit): Rename to ... > (cik_trait): ... this. > (IDENTIFIER_ANY_OP_P): Exclude cik_trait. > (IDENTIFIER_TRAIT_P): New macro to detect cik_trait. > * lex.cc (init_cp_traits): New function to set cik_trait and > IDENTIFIER_CP_INDEX for all built-in trait identifiers. > (cxx_init): Call init_cp_traits function. > * parser.cc (cp_traits): Define its values, declared in cp-tree.h. > (num_cp_traits): Define its value, declared in cp-tree.h. > (cp_lexer_lookup_trait): New function to look up a > built-in trait by IDENTIFIER_CP_INDEX. > (cp_lexer_lookup_trait_expr): Likewise, look up an > expression-yielding built-in trait. > (cp_lexer_lookup_trait_type): Likewise, look up a type-yielding > built-in trait. > (cp_keyword_starts_decl_specifier_p): Remove all RID value cases > for built-in traits. > (cp_lexer_next_token_is_decl_specifier_keyword): Handle > type-yielding built-in traits. > (cp_parser_primary_expression): Remove all RID value cases for > built-in traits. Handle expression-yielding built-in traits. > (cp_parser_trait): Handle cp_trait instead of enum rid. > (cp_parser_simple_type_specifier): Remove all RID value cases > for built-in traits. Handle type-yielding built-in traits. > > Co-authored-by: Patrick Palka > Signed-off-by: Ken Matsui > --- > gcc/c-family/c-common.cc | 7 --- > gcc/c-family/c-common.h | 5 -- > gcc/cp/cp-objcp-common.cc | 8 +-- > gcc/cp/cp-tree.h | 33 ++++++++--- > gcc/cp/lex.cc | 21 +++++++ > gcc/cp/parser.cc | 120 +++++++++++++++++++++++++------------- > 6 files changed, 129 insertions(+), 65 deletions(-) > > diff --git a/gcc/c-family/c-common.cc b/gcc/c-family/c-common.cc > index f044db5b797..21fd333ef57 100644 > --- a/gcc/c-family/c-common.cc > +++ b/gcc/c-family/c-common.cc > @@ -508,13 +508,6 @@ const struct c_common_resword c_common_reswords[] = > { "wchar_t", RID_WCHAR, D_CXXONLY }, > { "while", RID_WHILE, 0 }, > > -#define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > - { NAME, RID_##CODE, D_CXXONLY }, > -#include "cp/cp-trait.def" > -#undef DEFTRAIT > - /* An alias for __is_same. */ > - { "__is_same_as", RID_IS_SAME, D_CXXONLY }, > - > /* C++ transactional memory. */ > { "synchronized", RID_SYNCHRONIZED, D_CXX_OBJC | D_TRANSMEM }, > { "atomic_noexcept", RID_ATOMIC_NOEXCEPT, D_CXXONLY | D_TRANSMEM }, > diff --git a/gcc/c-family/c-common.h b/gcc/c-family/c-common.h > index 1fdba7ef3ea..051a442e0f4 100644 > --- a/gcc/c-family/c-common.h > +++ b/gcc/c-family/c-common.h > @@ -168,11 +168,6 @@ enum rid > RID_BUILTIN_LAUNDER, > RID_BUILTIN_BIT_CAST, > > -#define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > - RID_##CODE, > -#include "cp/cp-trait.def" > -#undef DEFTRAIT > - > /* C++11 */ > RID_CONSTEXPR, RID_DECLTYPE, RID_NOEXCEPT, RID_NULLPTR, RID_STATIC_ASSERT, > > diff --git a/gcc/cp/cp-objcp-common.cc b/gcc/cp/cp-objcp-common.cc > index 93b027b80ce..b1adacfec07 100644 > --- a/gcc/cp/cp-objcp-common.cc > +++ b/gcc/cp/cp-objcp-common.cc > @@ -421,6 +421,10 @@ names_builtin_p (const char *name) > } > } > > + /* Check for built-in traits. */ > + if (IDENTIFIER_TRAIT_P (id)) > + return true; > + > /* Also detect common reserved C++ words that aren't strictly built-in > functions. */ > switch (C_RID_CODE (id)) > @@ -434,10 +438,6 @@ names_builtin_p (const char *name) > case RID_BUILTIN_ASSOC_BARRIER: > case RID_BUILTIN_BIT_CAST: > case RID_OFFSETOF: > -#define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > - case RID_##CODE: > -#include "cp-trait.def" > -#undef DEFTRAIT > return true; > default: > break; > diff --git a/gcc/cp/cp-tree.h b/gcc/cp/cp-tree.h > index efcd2de54e5..81a5c06a574 100644 > --- a/gcc/cp/cp-tree.h > +++ b/gcc/cp/cp-tree.h > @@ -1226,7 +1226,7 @@ enum cp_identifier_kind { > cik_simple_op = 4, /* Non-assignment operator name. */ > cik_assign_op = 5, /* An assignment operator name. */ > cik_conv_op = 6, /* Conversion operator name. */ > - cik_reserved_for_udlit = 7, /* Not yet in use */ > + cik_trait = 7, /* Built-in trait name. */ > cik_max > }; > > @@ -1271,9 +1271,9 @@ enum cp_identifier_kind { > & IDENTIFIER_KIND_BIT_0 (NODE)) > > /* True if this identifier is for any operator name (including > - conversions). Value 4, 5, 6 or 7. */ > + conversions). Value 4, 5, or 6. */ > #define IDENTIFIER_ANY_OP_P(NODE) \ > - (IDENTIFIER_KIND_BIT_2 (NODE)) > + (IDENTIFIER_KIND_BIT_2 (NODE) && !IDENTIFIER_TRAIT_P (NODE)) > > /* True if this identifier is for an overloaded operator. Values 4, 5. */ > #define IDENTIFIER_OVL_OP_P(NODE) \ > @@ -1286,12 +1286,18 @@ enum cp_identifier_kind { > & IDENTIFIER_KIND_BIT_0 (NODE)) > > /* True if this identifier is the name of a type-conversion > - operator. Value 7. */ > + operator. Value 6. */ > #define IDENTIFIER_CONV_OP_P(NODE) \ > (IDENTIFIER_ANY_OP_P (NODE) \ > & IDENTIFIER_KIND_BIT_1 (NODE) \ > & (!IDENTIFIER_KIND_BIT_0 (NODE))) > > +/* True if this identifier is the name of a built-in trait. */ > +#define IDENTIFIER_TRAIT_P(NODE) \ > + (IDENTIFIER_KIND_BIT_0 (NODE) \ > + && IDENTIFIER_KIND_BIT_1 (NODE) \ > + && IDENTIFIER_KIND_BIT_2 (NODE)) > + > /* True if this identifier is a new or delete operator. */ > #define IDENTIFIER_NEWDEL_OP_P(NODE) \ > (IDENTIFIER_OVL_OP_P (NODE) \ > @@ -1375,16 +1381,27 @@ struct GTY (()) tree_argument_pack_select { > int index; > }; > > -/* The different kinds of traits that we encounter. */ > - > -enum cp_trait_kind > -{ > +/* The different kinds of traits that we encounter. The size is limited to > + addr_space_t since a trait is looked up by IDENTIFIER_CP_INDEX. */ > +enum cp_trait_kind : addr_space_t { > #define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > CPTK_##CODE, > #include "cp-trait.def" > #undef DEFTRAIT > }; > > +/* The trait type. */ > +struct cp_trait { > + const char *name; > + cp_trait_kind kind; > + short arity; > + bool type; > +}; > + > +/* The trait table. */ Let's mention that this table is indexed by cp_trait_kind. > +extern const struct cp_trait cp_traits[]; > +extern const addr_space_t num_cp_traits; > + > /* The types that we are processing. */ > #define TRAIT_EXPR_TYPE1(NODE) \ > (((struct tree_trait_expr *)TRAIT_EXPR_CHECK (NODE))->type1) > diff --git a/gcc/cp/lex.cc b/gcc/cp/lex.cc > index 64bcfb18196..872ca970545 100644 > --- a/gcc/cp/lex.cc > +++ b/gcc/cp/lex.cc > @@ -35,6 +35,7 @@ along with GCC; see the file COPYING3. If not see > #include "langhooks.h" > > static int interface_strcmp (const char *); > +static void init_cp_traits (void); > static void init_cp_pragma (void); > > static tree parse_strconst_pragma (const char *, int); > @@ -283,6 +284,25 @@ init_reswords (void) > } > } > > +/* Initialize the C++ traits. */ > +static void > +init_cp_traits (void) > +{ > + tree id; > + > + for (unsigned int i = 0; i < num_cp_traits; ++i) > + { > + id = get_identifier (cp_traits[i].name); > + IDENTIFIER_CP_INDEX (id) = cp_traits[i].kind; > + set_identifier_kind (id, cik_trait); > + } > + > + /* An alias for __is_same. */ > + id = get_identifier ("__is_same_as"); > + IDENTIFIER_CP_INDEX (id) = CPTK_IS_SAME; > + set_identifier_kind (id, cik_trait); > +} > + > static void > init_cp_pragma (void) > { > @@ -324,6 +344,7 @@ cxx_init (void) > input_location = BUILTINS_LOCATION; > > init_reswords (); > + init_cp_traits (); > init_tree (); > init_cp_semantics (); > init_operators (); > diff --git a/gcc/cp/parser.cc b/gcc/cp/parser.cc > index 59b9852895e..ece238c2072 100644 > --- a/gcc/cp/parser.cc > +++ b/gcc/cp/parser.cc > @@ -246,6 +246,12 @@ static void cp_lexer_start_debugging > (cp_lexer *) ATTRIBUTE_UNUSED; > static void cp_lexer_stop_debugging > (cp_lexer *) ATTRIBUTE_UNUSED; > +static const cp_trait *cp_lexer_lookup_trait > + (const cp_token *); > +static const cp_trait *cp_lexer_lookup_trait_expr > + (const cp_token *); > +static const cp_trait *cp_lexer_lookup_trait_type > + (const cp_token *); > > static cp_token_cache *cp_token_cache_new > (cp_token *, cp_token *); > @@ -279,6 +285,21 @@ static FILE *cp_lexer_debug_stream; > sizeof, typeof, or alignof. */ > int cp_unevaluated_operand; > > +/* The trait table, declared in cp-tree.h. */ > +const cp_trait cp_traits[] = > +{ > +#define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > + { NAME, CPTK_##CODE, ARITY, (TCC == tcc_type) }, > +#include "cp-trait.def" > +#undef DEFTRAIT > +}; > +const addr_space_t num_cp_traits = ARRAY_SIZE (cp_traits); Let's move the definition of cp_traits to lex.cc near the definition of the similar ovl_op_info table. I think we can in turn get rid of num_cp_traits since the size of the array will be statically known in lex.cc where the loop over it resides. Otherwise this patch series looks good to me. (Since only this patch needs changes, feel free to send the new version as a reply to this thread via git send-email's --in-reply-to flag instead of resending the entire series.) > + > +/* The trait table cannot have more than 255 (addr_space_t) entries since > + the index is retrieved through IDENTIFIER_CP_INDEX. */ > +static_assert(num_cp_traits <= 255, > + "cp_traits array cannot have more than 255 entries"); > + > /* Dump up to NUM tokens in BUFFER to FILE starting with token > START_TOKEN. If START_TOKEN is NULL, the dump starts with the > first token in BUFFER. If NUM is 0, dump all the tokens. If > @@ -1167,12 +1188,6 @@ cp_keyword_starts_decl_specifier_p (enum rid keyword) > case RID_CONSTEVAL: > return true; > > -#define DEFTRAIT_TYPE(CODE, NAME, ARITY) \ > - case RID_##CODE: > -#include "cp-trait.def" > -#undef DEFTRAIT_TYPE > - return true; > - > default: > if (keyword >= RID_FIRST_INT_N > && keyword < RID_FIRST_INT_N + NUM_INT_N_ENTS > @@ -1182,6 +1197,44 @@ cp_keyword_starts_decl_specifier_p (enum rid keyword) > } > } > > +/* Look ups the corresponding built-in trait if a given token is > + a built-in trait. Otherwise, returns nullptr. */ > + > +static const cp_trait * > +cp_lexer_lookup_trait (const cp_token *token) > +{ > + if (token->type == CPP_NAME && IDENTIFIER_TRAIT_P (token->u.value)) > + return &cp_traits[IDENTIFIER_CP_INDEX (token->u.value)]; > + > + return nullptr; > +} > + > +/* Similarly, but only if the token is an expression-yielding > + built-in trait. */ > + > +static const cp_trait * > +cp_lexer_lookup_trait_expr (const cp_token *token) > +{ > + const cp_trait *trait = cp_lexer_lookup_trait (token); > + if (trait && !trait->type) > + return trait; > + > + return nullptr; > +} > + > +/* Similarly, but only if the token is a type-yielding > + built-in trait. */ > + > +static const cp_trait * > +cp_lexer_lookup_trait_type (const cp_token *token) > +{ > + const cp_trait *trait = cp_lexer_lookup_trait (token); > + if (trait && trait->type) > + return trait; > + > + return nullptr; > +} > + > /* Return true if the next token is a keyword for a decl-specifier. */ > > static bool > @@ -1190,6 +1243,8 @@ cp_lexer_next_token_is_decl_specifier_keyword (cp_lexer *lexer) > cp_token *token; > > token = cp_lexer_peek_token (lexer); > + if (cp_lexer_lookup_trait_type (token)) > + return true; > return cp_keyword_starts_decl_specifier_p (token->keyword); > } > > @@ -2854,7 +2909,7 @@ static void cp_parser_late_parsing_default_args > static tree cp_parser_sizeof_operand > (cp_parser *, enum rid); > static cp_expr cp_parser_trait > - (cp_parser *, enum rid); > + (cp_parser *, const cp_trait *); > static bool cp_parser_declares_only_class_p > (cp_parser *); > static void cp_parser_set_storage_class > @@ -6029,12 +6084,6 @@ cp_parser_primary_expression (cp_parser *parser, > case RID_OFFSETOF: > return cp_parser_builtin_offsetof (parser); > > -#define DEFTRAIT_EXPR(CODE, NAME, ARITY) \ > - case RID_##CODE: > -#include "cp-trait.def" > -#undef DEFTRAIT_EXPR > - return cp_parser_trait (parser, token->keyword); > - > // C++ concepts > case RID_REQUIRES: > return cp_parser_requires_expression (parser); > @@ -6073,6 +6122,9 @@ cp_parser_primary_expression (cp_parser *parser, > `::' as the beginning of a qualified-id, or the "operator" > keyword. */ > case CPP_NAME: > + if (const cp_trait* trait = cp_lexer_lookup_trait_expr (token)) > + return cp_parser_trait (parser, trait); > + /* FALLTHRU */ > case CPP_SCOPE: > case CPP_TEMPLATE_ID: > case CPP_NESTED_NAME_SPECIFIER: > @@ -11041,28 +11093,13 @@ cp_parser_builtin_offsetof (cp_parser *parser) > /* Parse a builtin trait expression or type. */ > > static cp_expr > -cp_parser_trait (cp_parser* parser, enum rid keyword) > +cp_parser_trait (cp_parser* parser, const cp_trait* trait) > { > - cp_trait_kind kind; > + const cp_trait_kind kind = trait->kind; > tree type1, type2 = NULL_TREE; > - bool binary = false; > - bool variadic = false; > - bool type = false; > - > - switch (keyword) > - { > -#define DEFTRAIT(TCC, CODE, NAME, ARITY) \ > - case RID_##CODE: \ > - kind = CPTK_##CODE; \ > - binary = (ARITY == 2); \ > - variadic = (ARITY == -1); \ > - type = (TCC == tcc_type); \ > - break; > -#include "cp-trait.def" > -#undef DEFTRAIT > - default: > - gcc_unreachable (); > - } > + const bool binary = (trait->arity == 2); > + const bool variadic = (trait->arity == -1); > + const bool type = trait->type; > > /* Get location of initial token. */ > location_t start_loc = cp_lexer_peek_token (parser->lexer)->location; > @@ -20089,20 +20126,21 @@ cp_parser_simple_type_specifier (cp_parser* parser, > > return type; > > -#define DEFTRAIT_TYPE(CODE, NAME, ARITY) \ > - case RID_##CODE: > -#include "cp-trait.def" > -#undef DEFTRAIT_TYPE > - type = cp_parser_trait (parser, token->keyword); > + default: > + break; > + } > + > + /* If token is a type-yielding built-in traits, parse it. */ > + const cp_trait* trait = cp_lexer_lookup_trait_type (token); > + if (trait) > + { > + type = cp_parser_trait (parser, trait); > if (decl_specs) > cp_parser_set_decl_spec_type (decl_specs, type, > token, > /*type_definition_p=*/false); > > return type; > - > - default: > - break; > } > > /* If token is an already-parsed decltype not followed by ::, > -- > 2.42.0 > >