| // RUN: %clang_cc1 -triple x86_64-apple-darwin10 -emit-llvm -std=c++11 -O1 -disable-llvm-passes %s -o - | FileCheck %s |
| |
| namespace test1 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test11fIZNS_1gEvE1SEEvT_( |
| template <typename T> void f(T) {} |
| inline void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test2 { |
| // CHECK-DAG: define internal void @_ZN5test21fIZNS_L1gEvE1SEEvT_( |
| template <typename T> void f(T) {} |
| static inline void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test3 { |
| // CHECK-DAG: define internal void @_ZN5test31fIZNS_1gEvE1SEEvT_( |
| template <typename T> void f(T) {} |
| void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test4 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test41fIZNS_1gILi1EEEPvvE1SEEvT_( |
| template <typename T> void f(T) {} |
| template <int N> inline void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| extern template void *g<1>(); |
| template void *g<1>(); |
| } |
| |
| namespace test5 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test51fIZNS_1gILi1EEEPvvE1SEEvT_( |
| template <typename T> void f(T) {} |
| template <int N> inline void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| extern template void *g<1>(); |
| void *h() { return g<1>(); } |
| } |
| |
| namespace test6 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test61fIZZNS_1gEvEN1S1hEvE1TEEvv( |
| template <typename T> void f() {} |
| |
| inline void *g() { |
| struct S { |
| void *h() { |
| struct T { |
| }; |
| return (void *)f<T>; |
| } |
| } s; |
| return s.h(); |
| } |
| |
| void *h() { return g(); } |
| } |
| |
| namespace test7 { |
| // CHECK-DAG: define internal void @_ZN5test71fIZZNS_1gEvEN1S1hEvE1TEEvv( |
| template <typename T> void f() {} |
| |
| void *g() { |
| struct S { |
| void *h() { |
| struct T { |
| }; |
| return (void *)f<T>; |
| } |
| } s; |
| return s.h(); |
| } |
| |
| void *h() { return g(); } |
| } |
| |
| namespace test8 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test81fIZNS_1gEvE1SEEvT_( |
| template <typename T> void f(T) {} |
| inline void *g() { |
| enum S { |
| }; |
| return reinterpret_cast<void *>(f<S>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test9 { |
| // CHECK-DAG: define linkonce_odr void @_ZN5test91fIPZNS_1gEvE1SEEvT_( |
| template <typename T> void f(T) {} |
| inline void *g() { |
| struct S { |
| } s; |
| return reinterpret_cast<void *>(f<S*>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test10 { |
| // CHECK-DAG: define linkonce_odr void @_ZN6test101fIPFZNS_1gEvE1SvEEEvT_( |
| template <typename T> void f(T) {} |
| inline void *g() { |
| struct S { |
| } s; |
| typedef S(*ftype)(); |
| return reinterpret_cast<void *>(f<ftype>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test11 { |
| // CHECK-DAG: define internal void @_ZN6test111fIPFZNS_1gEvE1SPNS_12_GLOBAL__N_11IEEEEvT_( |
| namespace { |
| struct I { |
| }; |
| } |
| |
| template <typename T> void f(T) {} |
| inline void *g() { |
| struct S { |
| }; |
| typedef S(*ftype)(I * x); |
| return reinterpret_cast<void *>(f<ftype>); |
| } |
| void *h() { return g(); } |
| } |
| |
| namespace test12 { |
| // CHECK-DAG: define linkonce_odr void @_ZN6test123fooIZNS_3barIZNS_3zedEvE2S2EEPvvE2S1EEvv |
| template <typename T> void foo() {} |
| template <typename T> inline void *bar() { |
| enum S1 { |
| }; |
| return reinterpret_cast<void *>(foo<S1>); |
| } |
| inline void *zed() { |
| enum S2 { |
| }; |
| return reinterpret_cast<void *>(bar<S2>); |
| } |
| void *h() { return zed(); } |
| } |
| |
| namespace test13 { |
| // CHECK-DAG: define linkonce_odr void @_ZZN6test133fooEvEN1S3barEv( |
| inline void *foo() { |
| struct S { |
| static void bar() {} |
| }; |
| return (void *)S::bar; |
| } |
| void *zed() { return foo(); } |
| } |
| |
| namespace test14 { |
| // CHECK-DAG: define linkonce_odr void @_ZN6test143fooIZNS_1fEvE1SE3barILPS1_0EEEvv( |
| template <typename T> struct foo { |
| template <T *P> static void bar() {} |
| static void *g() { return (void *)bar<nullptr>; } |
| }; |
| inline void *f() { |
| struct S { |
| }; |
| return foo<S>::g(); |
| } |
| void h() { f(); } |
| } |
| |
| namespace test15 { |
| // CHECK-DAG: define linkonce_odr void @_ZN6test153zedIZNS_3fooIiEEPvvE3barEEvv( |
| template <class T> void zed() {} |
| template <class T> void *foo() { |
| class bar { |
| }; |
| return reinterpret_cast<void *>(zed<bar>); |
| } |
| void test() { foo<int>(); } |
| } |
| |
| namespace test16 { |
| // CHECK-DAG: define linkonce_odr void @_ZN6test163zedIZNS_3fooIiE3barEvE1SEEvv( |
| template <class T> void zed() {} |
| template <class T> struct foo { |
| static void *bar(); |
| }; |
| template <class T> void *foo<T>::bar() { |
| class S { |
| }; |
| return reinterpret_cast<void *>(zed<S>); |
| } |
| void *test() { return foo<int>::bar(); } |
| } |
| |
| namespace test17 { |
| // CHECK-DAG: @_ZZN6test173fooILi42EEEPivE3bar = linkonce_odr |
| // CHECK-DAG: define weak_odr i32* @_ZN6test173fooILi42EEEPiv( |
| template<int I> |
| int *foo() { |
| static int bar; |
| return &bar; |
| } |
| template int *foo<42>(); |
| } |
| |
| // PR18408 |
| namespace test18 { |
| template<template<typename> class> struct A {}; |
| struct B { template<typename> struct C; }; |
| void f(A<B::C>) {} |
| // CHECK-DAG: define void @_ZN6test181fENS_1AINS_1B1CEEE( |
| } |