2021-03-04 03:00:55 +08:00
|
|
|
//RUN: %clang_cc1 %s -pedantic -ast-dump -verify | FileCheck %s
|
2019-05-02 22:40:40 +08:00
|
|
|
|
|
|
|
//expected-no-diagnostics
|
|
|
|
|
2019-06-05 22:50:01 +08:00
|
|
|
//CHECK: |-VarDecl {{.*}} foo 'const __global int'
|
2019-05-02 22:40:40 +08:00
|
|
|
constexpr int foo = 0;
|
|
|
|
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: |-VarDecl {{.*}} foo1 'T' cinit
|
|
|
|
//CHECK: `-VarTemplateSpecializationDecl {{.*}} used foo1 '__global long':'__global long' cinit
|
|
|
|
template <typename T>
|
|
|
|
T foo1 = 0;
|
|
|
|
|
2019-05-02 22:40:40 +08:00
|
|
|
class c {
|
|
|
|
public:
|
2019-06-05 22:50:01 +08:00
|
|
|
//CHECK: `-VarDecl {{.*}} foo2 'const __global int'
|
2019-05-02 22:40:40 +08:00
|
|
|
static constexpr int foo2 = 0;
|
|
|
|
};
|
2019-06-05 22:50:01 +08:00
|
|
|
|
|
|
|
struct c1 {};
|
|
|
|
|
|
|
|
// We only deduce addr space in type alias in pointer types.
|
|
|
|
//CHECK: TypeAliasDecl {{.*}} alias_c1 'c1'
|
|
|
|
using alias_c1 = c1;
|
|
|
|
//CHECK: TypeAliasDecl {{.*}} alias_c1_ptr '__generic c1 *'
|
|
|
|
using alias_c1_ptr = c1 *;
|
|
|
|
|
|
|
|
struct c2 {
|
|
|
|
alias_c1 y;
|
|
|
|
alias_c1_ptr ptr = &y;
|
|
|
|
};
|
|
|
|
|
2019-07-15 21:02:21 +08:00
|
|
|
|
|
|
|
// Addr spaces for pointee of dependent types are not deduced
|
|
|
|
// during parsing but during template instantiation instead.
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
struct x1 {
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: -CXXMethodDecl {{.*}} operator= 'x1<T> &(const x1<T> &){{( __attribute__.*)?}} __generic'
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: -CXXMethodDecl {{.*}} operator= '__generic x1<int> &(const __generic x1<int> &__private){{( __attribute__.*)?}} __generic'
|
2019-07-15 21:02:21 +08:00
|
|
|
x1<T>& operator=(const x1<T>& xx) {
|
|
|
|
y = xx.y;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
int y;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
struct x2 {
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: -CXXMethodDecl {{.*}} foo 'void (x1<T> *){{( __attribute__.*)?}} __generic'
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: -CXXMethodDecl {{.*}} foo 'void (__generic x1<int> *__private){{( __attribute__.*)?}} __generic'
|
2019-07-15 21:02:21 +08:00
|
|
|
void foo(x1<T>* xx) {
|
|
|
|
m[0] = *xx;
|
|
|
|
}
|
2021-10-15 05:52:47 +08:00
|
|
|
//CHECK: -FieldDecl {{.*}} m 'x1<int>[2]'
|
2019-07-15 21:02:21 +08:00
|
|
|
x1<T> m[2];
|
|
|
|
};
|
|
|
|
|
|
|
|
void bar(__global x1<int> *xx, __global x2<int> *bar) {
|
|
|
|
bar->foo(xx);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class x3 : public T {
|
|
|
|
public:
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: -CXXConstructorDecl {{.*}} x3<T> 'void (const x3<T> &){{( __attribute__.*)?}} __generic'
|
2019-07-15 21:02:21 +08:00
|
|
|
x3(const x3 &t);
|
|
|
|
};
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: -CXXConstructorDecl {{.*}} x3<T> 'void (const x3<T> &){{( __attribute__.*)?}} __generic'
|
2019-07-15 21:02:21 +08:00
|
|
|
template <typename T>
|
|
|
|
x3<T>::x3(const x3<T> &t) {}
|
2019-07-18 17:12:49 +08:00
|
|
|
|
|
|
|
template <class T>
|
2019-11-27 19:03:11 +08:00
|
|
|
T xxx(T *in1, T in2) {
|
2019-07-18 17:12:49 +08:00
|
|
|
// This pointer can't be deduced to generic because addr space
|
|
|
|
// will be taken from the template argument.
|
2020-07-13 18:30:13 +08:00
|
|
|
//CHECK: `-VarDecl {{.*}} 'T *' cinit
|
|
|
|
//CHECK: `-VarDecl {{.*}} i '__private int *__private' cinit
|
2019-11-27 19:03:11 +08:00
|
|
|
T *i = in1;
|
2019-07-18 17:12:49 +08:00
|
|
|
T ii;
|
2019-11-27 19:03:11 +08:00
|
|
|
__private T *ptr = ⅈ
|
|
|
|
ptr = &in2;
|
2019-07-18 17:12:49 +08:00
|
|
|
return *i;
|
|
|
|
}
|
|
|
|
|
|
|
|
__kernel void test() {
|
|
|
|
int foo[10];
|
2019-11-27 19:03:11 +08:00
|
|
|
xxx<__private int>(&foo[0], foo[0]);
|
|
|
|
// FIXME: Template param deduction fails here because
|
|
|
|
// temporaries are not in the __private address space.
|
|
|
|
// It is probably reasonable to put them in __private
|
|
|
|
// considering that stack and function params are
|
|
|
|
// implicitly in __private.
|
|
|
|
// However, if temporaries are left in default addr
|
|
|
|
// space we should at least pretty print the __private
|
|
|
|
// addr space. Otherwise diagnostic apprears to be
|
|
|
|
// confusing.
|
|
|
|
//xxx(&foo[0], foo[0]);
|
2019-07-18 17:12:49 +08:00
|
|
|
}
|
2019-08-19 19:43:16 +08:00
|
|
|
|
|
|
|
// Addr space for pointer/reference to an array
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: FunctionDecl {{.*}} t1 'void (const float (__generic &__private)[2])'
|
2019-08-19 19:43:16 +08:00
|
|
|
void t1(const float (&fYZ)[2]);
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: FunctionDecl {{.*}} t2 'void (const float (__generic *__private)[2])'
|
2019-08-19 19:43:16 +08:00
|
|
|
void t2(const float (*fYZ)[2]);
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: FunctionDecl {{.*}} t3 'void (float (((__generic *__private)))[2])'
|
2019-08-19 19:43:16 +08:00
|
|
|
void t3(float(((*fYZ)))[2]);
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: FunctionDecl {{.*}} t4 'void (float (((__generic *__generic *__private)))[2])'
|
2019-08-19 19:43:16 +08:00
|
|
|
void t4(float(((**fYZ)))[2]);
|
2019-12-27 21:38:48 +08:00
|
|
|
//CHECK: FunctionDecl {{.*}} t5 'void (float (__generic *(__generic *__private))[2])'
|
2019-08-19 19:43:16 +08:00
|
|
|
void t5(float (*(*fYZ))[2]);
|
|
|
|
|
|
|
|
__kernel void k() {
|
|
|
|
__local float x[2];
|
2020-02-06 19:56:21 +08:00
|
|
|
float(*p)[2];
|
2019-08-19 19:43:16 +08:00
|
|
|
t1(x);
|
|
|
|
t2(&x);
|
|
|
|
t3(&x);
|
|
|
|
t4(&p);
|
|
|
|
t5(&p);
|
2020-07-13 18:30:13 +08:00
|
|
|
long f1 = foo1<long>;
|
2019-08-19 19:43:16 +08:00
|
|
|
}
|