2012-05-27 07:12:19 +08:00
|
|
|
// RUN: %clang_cc1 -fms-extensions -emit-llvm %s -o - -cxx-abi microsoft -triple=i386-pc-win32 | FileCheck %s
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class Class {
|
|
|
|
public:
|
|
|
|
void method() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
class Typename { };
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class Nested { };
|
|
|
|
|
|
|
|
template<bool flag>
|
|
|
|
class BoolTemplate {
|
|
|
|
public:
|
|
|
|
BoolTemplate() {}
|
|
|
|
};
|
|
|
|
|
2012-05-28 08:43:56 +08:00
|
|
|
template<int param>
|
|
|
|
class IntTemplate {
|
|
|
|
public:
|
|
|
|
IntTemplate() {}
|
|
|
|
};
|
|
|
|
|
2012-06-23 08:27:49 +08:00
|
|
|
template<>
|
|
|
|
class BoolTemplate<true> {
|
|
|
|
public:
|
|
|
|
BoolTemplate() {}
|
|
|
|
template<class T> void Foo(T arg) {}
|
|
|
|
};
|
|
|
|
|
2012-05-27 07:12:19 +08:00
|
|
|
void template_mangling() {
|
|
|
|
Class<Typename> c1;
|
|
|
|
c1.method();
|
|
|
|
// CHECK: call {{.*}} @"\01?method@?$Class@VTypename@@@@QAEXXZ"
|
|
|
|
|
2013-04-09 20:47:38 +08:00
|
|
|
Class<const Typename> c1_const;
|
|
|
|
Class<volatile Typename> c1_volatile;
|
|
|
|
Class<const volatile Typename> c1_cv;
|
|
|
|
c1_const.method();
|
|
|
|
c1_volatile.method();
|
|
|
|
c1_cv.method();
|
|
|
|
// Types with qualifiers have an extra $$C escape when used as template
|
|
|
|
// arguments. Not sure why.
|
|
|
|
// CHECK: call {{.*}} @"\01?method@?$Class@$$CBVTypename@@@@QAEXXZ"
|
|
|
|
// CHECK: call {{.*}} @"\01?method@?$Class@$$CCVTypename@@@@QAEXXZ"
|
|
|
|
// CHECK: call {{.*}} @"\01?method@?$Class@$$CDVTypename@@@@QAEXXZ"
|
|
|
|
|
2012-05-27 07:12:19 +08:00
|
|
|
Class<Nested<Typename> > c2;
|
|
|
|
c2.method();
|
|
|
|
// CHECK: call {{.*}} @"\01?method@?$Class@V?$Nested@VTypename@@@@@@QAEXXZ"
|
|
|
|
|
|
|
|
BoolTemplate<false> _false;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$BoolTemplate@$0A@@@QAE@XZ"
|
|
|
|
|
|
|
|
BoolTemplate<true> _true;
|
2012-06-23 08:27:49 +08:00
|
|
|
// PR13158
|
|
|
|
_true.Foo(1);
|
2012-05-27 07:12:19 +08:00
|
|
|
// CHECK: call {{.*}} @"\01??0?$BoolTemplate@$00@@QAE@XZ"
|
2012-06-23 08:27:49 +08:00
|
|
|
// CHECK: call {{.*}} @"\01??$Foo@H@?$BoolTemplate@$00@@QAEXH@Z"
|
2012-05-28 08:43:56 +08:00
|
|
|
|
2012-11-09 07:38:59 +08:00
|
|
|
IntTemplate<0> zero;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0A@@@QAE@XZ"
|
|
|
|
|
2012-05-29 00:53:33 +08:00
|
|
|
IntTemplate<5> five;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$04@@QAE@XZ"
|
|
|
|
|
2012-05-28 08:43:56 +08:00
|
|
|
IntTemplate<11> eleven;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0L@@@QAE@XZ"
|
|
|
|
|
2012-10-03 21:39:49 +08:00
|
|
|
IntTemplate<256> _256;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0BAA@@@QAE@XZ"
|
|
|
|
|
|
|
|
IntTemplate<513> _513;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0CAB@@@QAE@XZ"
|
|
|
|
|
|
|
|
IntTemplate<1026> _1026;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0EAC@@@QAE@XZ"
|
|
|
|
|
2012-05-28 08:43:56 +08:00
|
|
|
IntTemplate<65535> ffff;
|
|
|
|
// CHECK: call {{.*}} @"\01??0?$IntTemplate@$0PPPP@@@QAE@XZ"
|
2012-05-27 07:12:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace space {
|
|
|
|
template<class T> const T& foo(const T& l) { return l; }
|
|
|
|
}
|
|
|
|
// CHECK: "\01??$foo@H@space@@YAABHABH@Z"
|
|
|
|
|
|
|
|
void use() {
|
|
|
|
space::foo(42);
|
|
|
|
}
|
2013-03-21 06:29:42 +08:00
|
|
|
|
|
|
|
// PR13455
|
|
|
|
typedef void (*FunctionPointer)(void);
|
|
|
|
|
|
|
|
template <FunctionPointer function>
|
|
|
|
void FunctionPointerTemplate() {
|
|
|
|
function();
|
|
|
|
}
|
|
|
|
|
|
|
|
void spam() {
|
|
|
|
FunctionPointerTemplate<spam>();
|
|
|
|
// CHECK: "\01??$FunctionPointerTemplate@$1?spam@@YAXXZ@@YAXXZ"
|
|
|
|
}
|