ChuanqiXu created this revision. ChuanqiXu added reviewers: erichkeane, rsmith, rjmccall, aaron.ballman, urnathan. ChuanqiXu added a project: clang. ChuanqiXu requested review of this revision. Herald added a subscriber: cfe-commits.
The compiler would crash if we lookup for name in transparent decl context. The tests in this revision are two examples. godbolt example is here: https://llvm.godbolt.org/z/W497zGW5T (I didn't attach module examples in godbolt since I don't know how to do). I think this should make sense since the member declared in transparent DeclContext are semantically defined in the enclosing (non-transparent) DeclContext, this is the definition for transparent DeclContext. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D116792 Files: clang/lib/AST/DeclBase.cpp clang/test/Modules/Inputs/template_name_lookup/foo.cppm clang/test/Modules/template_name_lookup.cpp clang/test/SemaCXX/lookup-template-name-extern-CXX.cpp Index: clang/test/SemaCXX/lookup-template-name-extern-CXX.cpp =================================================================== --- /dev/null +++ clang/test/SemaCXX/lookup-template-name-extern-CXX.cpp @@ -0,0 +1,12 @@ +// Tests that the lookup in transparent declaration context +// (linkage specifiaction context) wouldn't cause compiler crash. +// RUN: %clang_cc1 -std=c++20 %s -fsyntax-only -verify +extern "C++" { +template <class T> +class X {}; // expected-note {{candidate template ignored: couldn't infer template argument 'T'}} + // expected-note@-1 {{candidate function template not viable: requires 1 argument, but 0 were provided}} +} + +void foo() { + X x; // expected-error {{no viable constructor or deduction guide for deduction of template arguments of 'X'}} +} Index: clang/test/Modules/template_name_lookup.cpp =================================================================== --- /dev/null +++ clang/test/Modules/template_name_lookup.cpp @@ -0,0 +1,11 @@ +// RUN: rm -rf %t +// RUN: mkdir -p %t +// RUN: %clang_cc1 -std=c++20 %S/Inputs/template_name_lookup/foo.cppm -emit-module-interface -o %t/foo.pcm +// RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t %s -fsyntax-only -verify + +import foo; +void use() { + X x; // expected-error {{no viable constructor or deduction guide for deduction of template arguments of 'X'}} + // expected-note@Inputs/template_name_lookup/foo.cppm:3 {{candidate template ignored: couldn't infer template argument 'T'}} + // expected-note@Inputs/template_name_lookup/foo.cppm:3 {{candidate function template not viable: requires 1 argument, but 0 were provided}} +} Index: clang/test/Modules/Inputs/template_name_lookup/foo.cppm =================================================================== --- /dev/null +++ clang/test/Modules/Inputs/template_name_lookup/foo.cppm @@ -0,0 +1,3 @@ +export module foo; +export template <typename T> +class X {}; Index: clang/lib/AST/DeclBase.cpp =================================================================== --- clang/lib/AST/DeclBase.cpp +++ clang/lib/AST/DeclBase.cpp @@ -1643,9 +1643,9 @@ DeclContext::lookup_result DeclContext::lookup(DeclarationName Name) const { - assert(getDeclKind() != Decl::LinkageSpec && - getDeclKind() != Decl::Export && - "should not perform lookups into transparent contexts"); + // For transparent DeclContext, we should lookup in their enclosing context. + if (getDeclKind() == Decl::LinkageSpec || getDeclKind() == Decl::Export) + return getParent()->lookup(Name); const DeclContext *PrimaryContext = getPrimaryContext(); if (PrimaryContext != this)
Index: clang/test/SemaCXX/lookup-template-name-extern-CXX.cpp =================================================================== --- /dev/null +++ clang/test/SemaCXX/lookup-template-name-extern-CXX.cpp @@ -0,0 +1,12 @@ +// Tests that the lookup in transparent declaration context +// (linkage specifiaction context) wouldn't cause compiler crash. +// RUN: %clang_cc1 -std=c++20 %s -fsyntax-only -verify +extern "C++" { +template <class T> +class X {}; // expected-note {{candidate template ignored: couldn't infer template argument 'T'}} + // expected-note@-1 {{candidate function template not viable: requires 1 argument, but 0 were provided}} +} + +void foo() { + X x; // expected-error {{no viable constructor or deduction guide for deduction of template arguments of 'X'}} +} Index: clang/test/Modules/template_name_lookup.cpp =================================================================== --- /dev/null +++ clang/test/Modules/template_name_lookup.cpp @@ -0,0 +1,11 @@ +// RUN: rm -rf %t +// RUN: mkdir -p %t +// RUN: %clang_cc1 -std=c++20 %S/Inputs/template_name_lookup/foo.cppm -emit-module-interface -o %t/foo.pcm +// RUN: %clang_cc1 -std=c++20 -fprebuilt-module-path=%t %s -fsyntax-only -verify + +import foo; +void use() { + X x; // expected-error {{no viable constructor or deduction guide for deduction of template arguments of 'X'}} + // expected-note@Inputs/template_name_lookup/foo.cppm:3 {{candidate template ignored: couldn't infer template argument 'T'}} + // expected-note@Inputs/template_name_lookup/foo.cppm:3 {{candidate function template not viable: requires 1 argument, but 0 were provided}} +} Index: clang/test/Modules/Inputs/template_name_lookup/foo.cppm =================================================================== --- /dev/null +++ clang/test/Modules/Inputs/template_name_lookup/foo.cppm @@ -0,0 +1,3 @@ +export module foo; +export template <typename T> +class X {}; Index: clang/lib/AST/DeclBase.cpp =================================================================== --- clang/lib/AST/DeclBase.cpp +++ clang/lib/AST/DeclBase.cpp @@ -1643,9 +1643,9 @@ DeclContext::lookup_result DeclContext::lookup(DeclarationName Name) const { - assert(getDeclKind() != Decl::LinkageSpec && - getDeclKind() != Decl::Export && - "should not perform lookups into transparent contexts"); + // For transparent DeclContext, we should lookup in their enclosing context. + if (getDeclKind() == Decl::LinkageSpec || getDeclKind() == Decl::Export) + return getParent()->lookup(Name); const DeclContext *PrimaryContext = getPrimaryContext(); if (PrimaryContext != this)
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits