From: Douglas Gregor Date: Thu, 13 Jan 2011 16:39:34 +0000 (+0000) Subject: Add tests for name mangling of variadic templates. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=4fc4866945de9b0f1f77a17557060f0ff959b0b1;p=clang Add tests for name mangling of variadic templates. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@123378 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/CodeGen/Mangle.cpp b/lib/CodeGen/Mangle.cpp index ff0e9223c8..0dfa0520af 100644 --- a/lib/CodeGen/Mangle.cpp +++ b/lib/CodeGen/Mangle.cpp @@ -1476,7 +1476,7 @@ void CXXNameMangler::mangleType(const DependentSizedExtVectorType *T) { } void CXXNameMangler::mangleType(const PackExpansionType *T) { - // FIXME: We may need to push this mangling into the callers + // ::= Dp # pack expansion (C++0x) Out << "sp"; mangleType(T->getPattern()); } @@ -1626,6 +1626,7 @@ void CXXNameMangler::mangleExpression(const Expr *E, unsigned Arity) { // ::= sr # dependent name // ::= sr # dependent template-id // ::= sZ # size of a parameter pack + // ::= sZ # size of a function parameter pack // ::= // ::= L E # integer literal // ::= L E # floating literal @@ -2050,7 +2051,6 @@ void CXXNameMangler::mangleExpression(const Expr *E, unsigned Arity) { break; case Expr::SizeOfPackExprClass: { - // FIXME: Variadic templates missing mangling for function parameter packs? Out << "sZ"; const NamedDecl *Pack = cast(E)->getPack(); if (const TemplateTypeParmDecl *TTP = dyn_cast(Pack)) @@ -2062,7 +2062,8 @@ void CXXNameMangler::mangleExpression(const Expr *E, unsigned Arity) { = dyn_cast(Pack)) mangleTemplateParameter(TempTP->getIndex()); else { - // FIXME: This case isn't handled by the Itanium C++ ABI + // Note: proposed by Mike Herrick on 11/30/10 + // ::= sZ # size of function parameter pack Diagnostic &Diags = Context.getDiags(); unsigned DiagID = Diags.getCustomDiagID(Diagnostic::Error, "cannot mangle sizeof...(function parameter pack)"); @@ -2156,7 +2157,7 @@ void CXXNameMangler::mangleTemplateArg(const NamedDecl *P, // ::= # type or template // ::= X E # expression // ::= # simple expressions - // ::= I * E # argument pack + // ::= J * E # argument pack // ::= sp # pack expansion of (C++0x) switch (A.getKind()) { case TemplateArgument::Null: @@ -2170,7 +2171,7 @@ void CXXNameMangler::mangleTemplateArg(const NamedDecl *P, mangleType(A.getAsTemplate()); break; case TemplateArgument::TemplateExpansion: - // This is mangled as Dp . + // ::= Dp # pack expansion (C++0x) Out << "Dp"; mangleType(A.getAsTemplateOrTemplatePattern()); break; diff --git a/test/CodeGenCXX/mangle-variadic-templates.cpp b/test/CodeGenCXX/mangle-variadic-templates.cpp index e078f3e3ba..0167726b0d 100644 --- a/test/CodeGenCXX/mangle-variadic-templates.cpp +++ b/test/CodeGenCXX/mangle-variadic-templates.cpp @@ -1,8 +1,14 @@ -// RUN: %clang_cc1 -std=c++0x -emit-llvm -o - %s | FileCheck %s +// RUN: %clang_cc1 -std=c++0x -emit-llvm -triple=x86_64-apple-darwin9 -o - %s | FileCheck %s template struct X { }; +template struct identity { }; +template struct add_reference; +template struct tuple { }; +template struct int_tuple { }; +template class ...Templates> struct template_tuple { }; + // CHECK: define weak_odr void @_Z2f0IJEEv1XIXsZT_EJspRT_EE template void f0(X) { } @@ -11,3 +17,51 @@ template void f0(X<0>); // CHECK: define weak_odr void @_Z2f0IJifdEEv1XIXsZT_EJspRT_EE template void f0(X<3, int&, float&, double&>); + +// Mangling for template argument packs +template void f1() {} +// CHECK: define weak_odr void @_Z2f1IJEEvv +template void f1<>(); +// CHECK: define weak_odr void @_Z2f1IJiEEvv +template void f1(); +// CHECK: define weak_odr void @_Z2f1IJifEEvv +template void f1(); + +// Mangling function parameter packs +template void f2(Types...) {} +// CHECK: define weak_odr void @_Z2f2IJEEvspT_ +template void f2<>(); +// CHECK: define weak_odr void @_Z2f2IJiEEvspT_ +template void f2(int); +// CHECK: define weak_odr void @_Z2f2IJifEEvspT_ +template void f2(int, float); + +// Mangling non-trivial function parameter packs +template void f3(const Types *...) {} +// CHECK: define weak_odr void @_Z2f3IJEEvspPKT_ +template void f3<>(); +// CHECK: define weak_odr void @_Z2f3IJiEEvspPKT_ +template void f3(const int*); +// CHECK: define weak_odr void @_Z2f3IJifEEvspPKT_ +template void f3(const int*, const float*); + +// Mangling of type pack expansions in a template argument +template tuple f4() {} +// CHECK: define weak_odr void @_Z2f4IJifdEE5tupleIJspT_EEv +template tuple f4(); + +// Mangling of type pack expansions in a function type +template identity f5() {} +// CHECK: define weak_odr void @_Z2f5IiJifdEE8identityIFT_spT0_EEv +template identity f5(); + +// Mangling of non-type template argument expansions +template int_tuple f6() {} +// CHECK: define weak_odr void @_Z2f6IJLi1ELi2ELi3EEE9int_tupleIJXspT_EEEv +template int_tuple<1, 2, 3> f6(); + +// Mangling of template template argument expansions +template class ...Templates> +template_tuple f7() {} +// CHECK: define weak_odr void @_Z2f7IJ8identity13add_referenceEE14template_tupleIJDpT_EEv +template template_tuple f7();