2012-09-25 15:32:39 +08:00
|
|
|
// RUN: %clang_cc1 %s -fsyntax-only -verify -fms-extensions -Wno-microsoft -std=c++11
|
|
|
|
|
|
|
|
__interface I1 {
|
|
|
|
// expected-error@+1 {{user-declared constructor is not permitted within an interface type}}
|
|
|
|
I1();
|
|
|
|
// expected-error@+1 {{user-declared destructor is not permitted within an interface type}}
|
|
|
|
~I1();
|
|
|
|
virtual void fn1() const;
|
|
|
|
// expected-error@+1 {{operator 'operator!' is not permitted within an interface type}}
|
|
|
|
bool operator!();
|
|
|
|
// expected-error@+1 {{operator 'operator int' is not permitted within an interface type}}
|
|
|
|
operator int();
|
2014-04-02 13:58:29 +08:00
|
|
|
// expected-error@+1 {{nested class I1::(anonymous) is not permitted within an interface type}}
|
2012-09-25 15:32:39 +08:00
|
|
|
struct { int a; };
|
2014-10-03 17:02:53 +08:00
|
|
|
void fn2() {
|
2012-09-25 15:32:39 +08:00
|
|
|
struct A { }; // should be ignored: not a nested class
|
|
|
|
}
|
|
|
|
protected: // expected-error {{interface types cannot specify 'protected' access}}
|
|
|
|
typedef void void_t;
|
|
|
|
using int_t = int;
|
|
|
|
private: // expected-error {{interface types cannot specify 'private' access}}
|
|
|
|
static_assert(true, "oops");
|
|
|
|
};
|
|
|
|
|
|
|
|
__interface I2 {
|
|
|
|
// expected-error@+1 {{data member 'i' is not permitted within an interface type}}
|
|
|
|
int i;
|
|
|
|
// expected-error@+1 {{static member function 'fn1' is not permitted within an interface type}}
|
|
|
|
static int fn1();
|
|
|
|
private: // expected-error {{interface types cannot specify 'private' access}}
|
|
|
|
// expected-error@+1 {{non-public member function 'fn2' is not permitted within an interface type}}
|
|
|
|
void fn2();
|
|
|
|
protected: // expected-error {{interface types cannot specify 'protected' access}}
|
|
|
|
// expected-error@+1 {{non-public member function 'fn3' is not permitted within an interface type}}
|
|
|
|
void fn3();
|
|
|
|
public:
|
|
|
|
void fn4();
|
|
|
|
};
|
|
|
|
|
|
|
|
// expected-error@+1 {{'final' keyword not permitted with interface types}}
|
|
|
|
__interface I3 final {
|
|
|
|
};
|
|
|
|
|
|
|
|
__interface I4 : I1, I2 {
|
|
|
|
void fn1() const override;
|
|
|
|
// expected-error@+1 {{'final' keyword not permitted with interface types}}
|
2014-10-03 17:02:53 +08:00
|
|
|
void fn2() final;
|
2012-09-25 15:32:39 +08:00
|
|
|
};
|
|
|
|
|
2018-03-28 12:16:13 +08:00
|
|
|
// expected-error@+1 {{interface type cannot inherit from non-public interface 'I1'}}
|
2012-09-25 15:32:39 +08:00
|
|
|
__interface I5 : private I1 {
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename X>
|
|
|
|
__interface I6 : X {
|
|
|
|
};
|
|
|
|
|
|
|
|
struct S { };
|
|
|
|
class C { };
|
|
|
|
__interface I { };
|
2015-11-16 14:58:51 +08:00
|
|
|
union U;
|
2012-09-25 15:32:39 +08:00
|
|
|
|
2012-09-25 15:32:49 +08:00
|
|
|
static_assert(!__is_interface_class(S), "oops");
|
|
|
|
static_assert(!__is_interface_class(C), "oops");
|
2015-11-16 14:58:51 +08:00
|
|
|
static_assert(!__is_interface_class(I), "oops");
|
|
|
|
static_assert(!__is_interface_class(U), "oops");
|
2012-09-25 15:32:49 +08:00
|
|
|
|
2018-03-28 12:16:13 +08:00
|
|
|
// expected-error@55 {{interface type cannot inherit from struct 'S'}}
|
2012-09-25 15:32:39 +08:00
|
|
|
// expected-note@+1 {{in instantiation of template class 'I6<S>' requested here}}
|
|
|
|
struct S1 : I6<S> {
|
|
|
|
};
|
|
|
|
|
2018-03-28 12:16:13 +08:00
|
|
|
// expected-error@55 {{interface type cannot inherit from class 'C'}}
|
2012-09-25 15:32:39 +08:00
|
|
|
// expected-note@+1 {{in instantiation of template class 'I6<C>' requested here}}
|
|
|
|
class C1 : I6<C> {
|
|
|
|
};
|
|
|
|
|
|
|
|
class C2 : I6<I> {
|
|
|
|
};
|
2017-09-21 06:28:24 +08:00
|
|
|
|
|
|
|
|
|
|
|
// MSVC makes a special case in that an interface is allowed to have a data
|
|
|
|
// member if it is a property.
|
|
|
|
__interface HasProp {
|
|
|
|
__declspec(property(get = Get, put = Put)) int data;
|
|
|
|
int Get(void);
|
|
|
|
void Put(int);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct __declspec(uuid("00000000-0000-0000-C000-000000000046"))
|
|
|
|
IUnknown {
|
|
|
|
void foo();
|
|
|
|
__declspec(property(get = Get, put = Put), deprecated) int data;
|
|
|
|
int Get(void);
|
|
|
|
void Put(int);
|
|
|
|
};
|
|
|
|
|
|
|
|
struct IFaceStruct : IUnknown {
|
|
|
|
__declspec(property(get = Get2, put = Put2), deprecated) int data2;
|
|
|
|
int Get2(void);
|
|
|
|
void Put2(int);
|
|
|
|
};
|
|
|
|
|
|
|
|
__interface IFaceInheritsStruct : IFaceStruct {};
|
|
|
|
static_assert(!__is_interface_class(HasProp), "oops");
|
|
|
|
static_assert(!__is_interface_class(IUnknown), "oops");
|
|
|
|
static_assert(!__is_interface_class(IFaceStruct), "oops");
|
|
|
|
static_assert(!__is_interface_class(IFaceInheritsStruct), "oops");
|
2021-11-09 18:10:11 +08:00
|
|
|
|
|
|
|
template<typename>
|
|
|
|
class TemplateContext {
|
|
|
|
class Base;
|
|
|
|
// Should not crash on an incomplete-type and dependent base specifier.
|
|
|
|
__interface Foo : Base {};
|
|
|
|
};
|