mirror of
git://gcc.gnu.org/git/gcc.git
synced 2025-04-04 23:51:47 +08:00
c++: More set_identifier_type_value fixing [PR 99116]
My recent change looked under template_parms in two places, but that was covering up a separate problem. We were attempting to set the identifier_type_value of a template_parm into the template_parm scope. The peeking stopped us doing that, but confused poplevel, leaving an identifier value lying around. This fixes the underlying problem in do_pushtag -- we only need to set the identifier_type_value directly when we're in a template_parm scope (a later pushdecl will push the actual template_decl). for non-class non-template-parm bindings do_pushdecl already ends up manipulating identifier_type_value correctly. PR c++/99116 gcc/cp/ * name-lookup.c (do_pushdecl): Don't peek under template_parm bindings here ... (set_identifier_type_value_with_scope): ... or here. (do_pushtag): Only set_identifier_type_value_with_scope at non-class template parm scope, and use parent scope. gcc/testsuite/ * g++.dg/lookup/pr99116-1.C: New. * g++.dg/lookup/pr99116-2.C: New.
This commit is contained in:
parent
d46c7e2c54
commit
24bf79f179
@ -3691,14 +3691,9 @@ do_pushdecl (tree decl, bool hiding)
|
||||
if (match == error_mark_node)
|
||||
;
|
||||
else if (TREE_CODE (match) == TYPE_DECL)
|
||||
{
|
||||
auto *l = level;
|
||||
while (l->kind == sk_template_parms)
|
||||
l = l->level_chain;
|
||||
gcc_checking_assert (REAL_IDENTIFIER_TYPE_VALUE (name)
|
||||
== (l->kind == sk_namespace
|
||||
? NULL_TREE : TREE_TYPE (match)));
|
||||
}
|
||||
gcc_checking_assert (REAL_IDENTIFIER_TYPE_VALUE (name)
|
||||
== (level->kind == sk_namespace
|
||||
? NULL_TREE : TREE_TYPE (match)));
|
||||
else if (iter.hidden_p () && !hiding)
|
||||
{
|
||||
/* Unhiding a previously hidden decl. */
|
||||
@ -4756,12 +4751,13 @@ print_binding_stack (void)
|
||||
static void
|
||||
set_identifier_type_value_with_scope (tree id, tree decl, cp_binding_level *b)
|
||||
{
|
||||
while (b->kind == sk_template_parms)
|
||||
b = b->level_chain;
|
||||
|
||||
if (b->kind == sk_namespace)
|
||||
/* At namespace scope we should not see an identifier type value. */
|
||||
gcc_checking_assert (!REAL_IDENTIFIER_TYPE_VALUE (id));
|
||||
gcc_checking_assert (!REAL_IDENTIFIER_TYPE_VALUE (id)
|
||||
/* We could be pushing a friend underneath a template
|
||||
parm (ill-formed). */
|
||||
|| (TEMPLATE_PARM_P
|
||||
(TYPE_NAME (REAL_IDENTIFIER_TYPE_VALUE (id)))));
|
||||
else
|
||||
{
|
||||
/* Push the current type value, so we can restore it later */
|
||||
@ -8257,10 +8253,8 @@ do_pushtag (tree name, tree type, TAG_how how)
|
||||
if (decl == error_mark_node)
|
||||
return decl;
|
||||
|
||||
bool in_class = false;
|
||||
if (b->kind == sk_class)
|
||||
{
|
||||
in_class = true;
|
||||
if (!TYPE_BEING_DEFINED (current_class_type))
|
||||
/* Don't push anywhere if the class is complete; a lambda in an
|
||||
NSDMI is not a member of the class. */
|
||||
@ -8275,7 +8269,12 @@ do_pushtag (tree name, tree type, TAG_how how)
|
||||
pushdecl_class_level (decl);
|
||||
}
|
||||
else if (b->kind == sk_template_parms)
|
||||
in_class = b->level_chain->kind == sk_class;
|
||||
{
|
||||
/* Do not push the tag here -- we'll want to push the
|
||||
TEMPLATE_DECL. */
|
||||
if (b->level_chain->kind != sk_class)
|
||||
set_identifier_type_value_with_scope (name, tdef, b->level_chain);
|
||||
}
|
||||
else
|
||||
{
|
||||
decl = do_pushdecl_with_scope
|
||||
@ -8293,9 +8292,6 @@ do_pushtag (tree name, tree type, TAG_how how)
|
||||
}
|
||||
}
|
||||
|
||||
if (!in_class)
|
||||
set_identifier_type_value_with_scope (name, tdef, b);
|
||||
|
||||
TYPE_CONTEXT (type) = DECL_CONTEXT (decl);
|
||||
|
||||
/* If this is a local class, keep track of it. We need this
|
||||
|
25
gcc/testsuite/g++.dg/lookup/pr99116-1.C
Normal file
25
gcc/testsuite/g++.dg/lookup/pr99116-1.C
Normal file
@ -0,0 +1,25 @@
|
||||
// PR 99116 sliding hidden friends under template parm scopes
|
||||
|
||||
template<int T> struct Z {
|
||||
|
||||
friend struct T; // { dg-error "shadows template parameter" }
|
||||
};
|
||||
|
||||
struct Y {
|
||||
|
||||
template<typename S> struct A {};
|
||||
|
||||
friend struct S;
|
||||
};
|
||||
|
||||
struct X
|
||||
{
|
||||
struct S2 {};
|
||||
|
||||
struct In
|
||||
{
|
||||
friend struct S2;
|
||||
};
|
||||
};
|
||||
|
||||
typedef int S2;
|
19
gcc/testsuite/g++.dg/lookup/pr99116-2.C
Normal file
19
gcc/testsuite/g++.dg/lookup/pr99116-2.C
Normal file
@ -0,0 +1,19 @@
|
||||
// PR 99116, we need to remove the namespace-scope meaning of
|
||||
// IDENTIFIER_TYPE_VALUE.
|
||||
|
||||
namespace __gnu_cxx
|
||||
{
|
||||
template<typename _CharT>
|
||||
struct char_traits
|
||||
{
|
||||
static void length();
|
||||
};
|
||||
|
||||
template<typename _T>
|
||||
void char_traits<_T>::length ()
|
||||
{
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
struct char_traits;
|
Loading…
x
Reference in New Issue
Block a user