// RUN: %clang_cc1 -emit-llvm %s -o - -triple=i386-pc-win32 | FileCheck %s template class A {}; template class B {}; template class C {}; void foo_abbb(A, B, B >) {} // CHECK: "?foo_abbb@@YAXV?$A@V?$B@D@@V1@V1@@@@Z" void foo_abb(A, B >) {} // CHECK: "?foo_abb@@YAXV?$A@DV?$B@D@@V1@@@@Z" void foo_abc(A, C >) {} // CHECK: "?foo_abc@@YAXV?$A@DV?$B@D@@V?$C@D@@@@@Z" void foo_bt(bool a, B b) {} // CHECK: "?foo_bt@@YAX_NV?$B@$$A6A_N_N@Z@@@Z" namespace N { template class A {}; template class B {}; template class C {}; template class D {}; class Z {}; } void foo_abbb(N::A, N::B, N::B >) {} // CHECK: "?foo_abbb@@YAXV?$A@V?$B@D@N@@V12@V12@@N@@@Z" void foo_abb(N::A, N::B >) {} // CHECK: "?foo_abb@@YAXV?$A@DV?$B@D@N@@V12@@N@@@Z" void foo_abc(N::A, N::C >) {} // CHECK: "?foo_abc@@YAXV?$A@DV?$B@D@N@@V?$C@D@2@@N@@@Z" N::A, N::C > abc_foo() { // CHECK: ?abc_foo@@YA?AV?$A@DV?$B@D@N@@V?$C@D@2@@N@@XZ return N::A, N::C >(); } N::Z z_foo(N::Z arg) { // CHECK: ?z_foo@@YA?AVZ@N@@V12@@Z return arg; } N::B b_foo(N::B arg) { // CHECK: ?b_foo@@YA?AV?$B@D@N@@V12@@Z return arg; } N::D d_foo(N::D arg) { // CHECK: ?d_foo@@YA?AV?$D@DD@N@@V12@@Z return arg; } N::A, N::C > abc_foo_abc(N::A, N::C >) { // CHECK: ?abc_foo_abc@@YA?AV?$A@DV?$B@D@N@@V?$C@D@2@@N@@V12@@Z return N::A, N::C >(); } namespace NA { class X {}; template class Y {}; } namespace NB { class X {}; template class Y {}; } void foo5(NA::Y > > > arg) {} // CHECK: "?foo5@@YAXV?$Y@V?$Y@V?$Y@V?$Y@VX@NA@@@NB@@@NA@@@NB@@@NA@@@Z" void foo11(NA::Y, NB::Y) {} // CHECK: "?foo11@@YAXV?$Y@VX@NA@@@NA@@V1NB@@@Z" void foo112(NA::Y, NB::Y) {} // CHECK: "?foo112@@YAXV?$Y@VX@NA@@@NA@@V?$Y@VX@NB@@@NB@@@Z" void foo22(NA::Y >, NB::Y >) {} // CHECK: "?foo22@@YAXV?$Y@V?$Y@VX@NA@@@NB@@@NA@@V?$Y@V?$Y@VX@NA@@@NA@@@NB@@@Z" namespace PR13207 { class A {}; class B {}; class C {}; template class F {}; template class I {}; template class J {}; template class K {}; class L { public: void foo(I x) {} }; // CHECK: "?foo@L@PR13207@@QAEXV?$I@VA@PR13207@@@2@@Z" void call_l_foo(L* l) { l->foo(I()); } void foo(I x) {} // CHECK: "?foo@PR13207@@YAXV?$I@VA@PR13207@@@1@@Z" void foo2(I x, I y) { } // CHECK: "?foo2@PR13207@@YAXV?$I@VA@PR13207@@@1@0@Z" void bar(J x) {} // CHECK: "?bar@PR13207@@YAXV?$J@VA@PR13207@@VB@2@@1@@Z" void spam(K x) {} // CHECK: "?spam@PR13207@@YAXV?$K@VA@PR13207@@VB@2@VC@2@@1@@Z" void baz(K, I >) {} // CHECK: "?baz@PR13207@@YAXV?$K@DV?$F@D@PR13207@@V?$I@D@2@@1@@Z" void qux(K, I >) {} // CHECK: "?qux@PR13207@@YAXV?$K@DV?$I@D@PR13207@@V12@@1@@Z" namespace NA { class X {}; template class Y {}; void foo(Y x) {} // CHECK: "?foo@NA@PR13207@@YAXV?$Y@VX@NA@PR13207@@@12@@Z" void foofoo(Y > x) {} // CHECK: "?foofoo@NA@PR13207@@YAXV?$Y@V?$Y@VX@NA@PR13207@@@NA@PR13207@@@12@@Z" } namespace NB { class X {}; template class Y {}; void foo(Y x) {} // CHECK: "?foo@NB@PR13207@@YAXV?$Y@VX@NA@PR13207@@@12@@Z" void bar(NA::Y x) {} // CHECK: "?bar@NB@PR13207@@YAXV?$Y@VX@NB@PR13207@@@NA@2@@Z" void spam(NA::Y x) {} // CHECK: "?spam@NB@PR13207@@YAXV?$Y@VX@NA@PR13207@@@NA@2@@Z" void foobar(NA::Y > a, Y >) {} // CHECK: "?foobar@NB@PR13207@@YAXV?$Y@V?$Y@VX@NB@PR13207@@@NB@PR13207@@@NA@2@V312@@Z" void foobarspam(Y a, NA::Y > b, Y >) {} // CHECK: "?foobarspam@NB@PR13207@@YAXV?$Y@VX@NB@PR13207@@@12@V?$Y@V?$Y@VX@NB@PR13207@@@NB@PR13207@@@NA@2@V412@@Z" void foobarbaz(Y a, NA::Y > b, Y >, Y > c) {} // CHECK: "?foobarbaz@NB@PR13207@@YAXV?$Y@VX@NB@PR13207@@@12@V?$Y@V?$Y@VX@NB@PR13207@@@NB@PR13207@@@NA@2@V412@2@Z" void foobarbazqux(Y a, NA::Y > b, Y >, Y > c , NA::Y > > d) {} // CHECK: "?foobarbazqux@NB@PR13207@@YAXV?$Y@VX@NB@PR13207@@@12@V?$Y@V?$Y@VX@NB@PR13207@@@NB@PR13207@@@NA@2@V412@2V?$Y@V?$Y@V?$Y@VX@NB@PR13207@@@NB@PR13207@@@NB@PR13207@@@52@@Z" } namespace NC { class X {}; template class Y {}; void foo(Y x) {} // CHECK: "?foo@NC@PR13207@@YAXV?$Y@VX@NB@PR13207@@@12@@Z" void foobar(NC::Y > > x) {} // CHECK: "?foobar@NC@PR13207@@YAXV?$Y@V?$Y@V?$Y@VX@NA@PR13207@@@NA@PR13207@@@NB@PR13207@@@12@@Z" } } // Function template names are not considered for backreferencing, but normal // function names are. namespace fn_space { struct RetVal { int hash; }; template RetVal fun_tmpl(const T &t) { return RetVal(); } RetVal fun_normal(int t) { return RetVal(); } void fun_instantiate() { fun_normal(1); fun_tmpl(1); } // CHECK: "?fun_normal@fn_space@@YA?AURetVal@1@H@Z" // CHECK: "??$fun_tmpl@H@fn_space@@YA?AURetVal@0@ABH@Z" template RetVal fun_tmpl_recurse(T t) { if (!t) return RetVal(); return F(t - 1); } RetVal ident(int x) { return RetVal(); } void fun_instantiate2() { fun_tmpl_recurse >(10); } // CHECK: "??$fun_tmpl_recurse@H$1??$fun_tmpl_recurse@H$1?ident@fn_space@@YA?AURetVal@2@H@Z@fn_space@@YA?AURetVal@1@H@Z@fn_space@@YA?AURetVal@0@H@Z" // CHECK: "??$fun_tmpl_recurse@H$1?ident@fn_space@@YA?AURetVal@2@H@Z@fn_space@@YA?AURetVal@0@H@Z" }