Justin Lebar | ba122ab | 2016-03-30 23:30:21 +0000 | [diff] [blame^] | 1 | // RUN: %clang_cc1 -std=c++11 -fsyntax-only -verify -isystem %S/Inputs %s |
| 2 | // RUN: %clang_cc1 -std=c++11 -fsyntax-only -verify -isystem %S/Inputs %s -fcuda-is-device |
| 3 | |
| 4 | #include "Inputs/cuda.h" |
| 5 | |
| 6 | // Declares one function and pulls it into namespace ns: |
| 7 | // |
| 8 | // __device__ int OverloadMe(); |
| 9 | // namespace ns { using ::OverloadMe; } |
| 10 | // |
| 11 | // Clang cares that this is done in a system header. |
| 12 | #include <overload.h> |
| 13 | |
| 14 | // Opaque type used to determine which overload we're invoking. |
| 15 | struct HostReturnTy {}; |
| 16 | |
| 17 | // These shouldn't become host+device because they already have attributes. |
| 18 | __host__ constexpr int HostOnly() { return 0; } |
| 19 | // expected-note@-1 0+ {{not viable}} |
| 20 | __device__ constexpr int DeviceOnly() { return 0; } |
| 21 | // expected-note@-1 0+ {{not viable}} |
| 22 | |
| 23 | constexpr int HostDevice() { return 0; } |
| 24 | |
| 25 | // This should be a host-only function, because there's a previous __device__ |
| 26 | // overload in <overload.h>. |
| 27 | constexpr HostReturnTy OverloadMe() { return HostReturnTy(); } |
| 28 | |
| 29 | namespace ns { |
| 30 | // The "using" statement in overload.h should prevent OverloadMe from being |
| 31 | // implicitly host+device. |
| 32 | constexpr HostReturnTy OverloadMe() { return HostReturnTy(); } |
| 33 | } // namespace ns |
| 34 | |
| 35 | // This is an error, because NonSysHdrOverload was not defined in a system |
| 36 | // header. |
| 37 | __device__ int NonSysHdrOverload() { return 0; } |
| 38 | // expected-note@-1 {{conflicting __device__ function declared here}} |
| 39 | constexpr int NonSysHdrOverload() { return 0; } |
| 40 | // expected-error@-1 {{constexpr function 'NonSysHdrOverload' without __host__ or __device__ attributes}} |
| 41 | |
| 42 | // Variadic device functions are not allowed, so this is just treated as |
| 43 | // host-only. |
| 44 | constexpr void Variadic(const char*, ...); |
| 45 | // expected-note@-1 {{call to __host__ function from __device__ function}} |
| 46 | |
| 47 | __host__ void HostFn() { |
| 48 | HostOnly(); |
| 49 | DeviceOnly(); // expected-error {{no matching function}} |
| 50 | HostReturnTy x = OverloadMe(); |
| 51 | HostReturnTy y = ns::OverloadMe(); |
| 52 | Variadic("abc", 42); |
| 53 | } |
| 54 | |
| 55 | __device__ void DeviceFn() { |
| 56 | HostOnly(); // expected-error {{no matching function}} |
| 57 | DeviceOnly(); |
| 58 | int x = OverloadMe(); |
| 59 | int y = ns::OverloadMe(); |
| 60 | Variadic("abc", 42); // expected-error {{no matching function}} |
| 61 | } |
| 62 | |
| 63 | __host__ __device__ void HostDeviceFn() { |
| 64 | #ifdef __CUDA_ARCH__ |
| 65 | int y = OverloadMe(); |
| 66 | #else |
| 67 | constexpr HostReturnTy y = OverloadMe(); |
| 68 | #endif |
| 69 | } |