// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -emit-llvm-only %s
// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fdelayed-template-parsing %s -DDELAYED_TEMPLATE_PARSING
// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fms-extensions %s -DMS_EXTENSIONS
// RUN: %clang_cc1 -std=c++1z -verify -fsyntax-only -fblocks -fdelayed-template-parsing -fms-extensions %s -DMS_EXTENSIONS -DDELAYED_TEMPLATE_PARSING
template<class, class> constexpr bool is_same = false;
template<class T> constexpr bool is_same<T, T> = true;
namespace test_star_this {
namespace ns1 {
class A {
int x = 345;
auto foo() {
(void) [*this, this] { }; //expected-error{{'this' can appear only once}}
(void) [this] { ++x; };
(void) [*this] { ++x; }; //expected-error{{read-only variable}}
(void) [*this] () mutable { ++x; };
(void) [=] { return x; };
(void) [&, this] { return x; };
(void) [=, *this] { return x; };
(void) [&, *this] { return x; };
}
};
} // end ns1
namespace ns2 {
class B {
B(const B&) = delete; //expected-note{{deleted here}}
int *x = (int *) 456;
void foo() {
(void)[this] { return x; };
(void)[*this] { return x; }; //expected-error{{call to deleted}}
}
};
} // end ns2
namespace ns3 {
class B {
B(const B&) = delete; //expected-note2{{deleted here}}
int *x = (int *) 456;
public:
template<class T = int>
void foo() {
(void)[this] { return x; };
(void)[*this] { return x; }; //expected-error2{{call to deleted}}
}
B() = default;
} b;
B *c = (b.foo(), nullptr); //expected-note{{in instantiation}}
} // end ns3
namespace ns4 {
template<class U>
class B {
B(const B&) = delete; //expected-note{{deleted here}}
double d = 3.14;
public:
template<class T = int>
auto foo() {
const auto &L = [*this] (auto a) mutable { //expected-error{{call to deleted}}
d += a;
return [this] (auto b) { return d +=b; };
};
}
B() = default;
};
void main() {
B<int*> b;
b.foo(); //expected-note{{in instantiation}}
} // end main
} // end ns4
namespace ns5 {
struct X {
double d = 3.14;
X(const volatile X&);
void foo() {
}
void foo() const { //expected-note{{const}}
auto L = [*this] () mutable {
static_assert(is_same<decltype(this), X*>);
++d;
auto M = [this] {
static_assert(is_same<decltype(this), X*>);
++d;
auto N = [] {
static_assert(is_same<decltype(this), X*>);
};
};
};
auto L1 = [*this] {
static_assert(is_same<decltype(this), const X*>);
auto M = [this] () mutable {
static_assert(is_same<decltype(this), const X*>);
auto N = [] {
static_assert(is_same<decltype(this), const X*>);
};
};
auto M2 = [*this] () mutable {
static_assert(is_same<decltype(this), X*>);
auto N = [] {
static_assert(is_same<decltype(this), X*>);
};
};
};
auto GL1 = [*this] (auto a) {
static_assert(is_same<decltype(this), const X*>);
auto M = [this] (auto b) mutable {
static_assert(is_same<decltype(this), const X*>);
auto N = [] (auto c) {
static_assert(is_same<decltype(this), const X*>);
};
return N;
};
auto M2 = [*this] (auto a) mutable {
static_assert(is_same<decltype(this), X*>);
auto N = [] (auto b) {
static_assert(is_same<decltype(this), X*>);
};
return N;
};
return [=](auto a) mutable { M(a)(a); M2(a)(a); };
};
GL1("abc")("abc");
auto L2 = [this] () mutable {
static_assert(is_same<decltype(this), const X*>);
++d; //expected-error{{cannot assign}}
};
auto GL = [*this] (auto a) mutable {
static_assert(is_same<decltype(this), X*>);
++d;
auto M = [this] (auto b) {
static_assert(is_same<decltype(this), X*>);
++d;
auto N = [] (auto c) {
static_assert(is_same<decltype(this), X*>);
};
N(3.14);
};
M("abc");
};
GL(3.14);
}
void foo() volatile const {
auto L = [this] () {
static_assert(is_same<decltype(this), const volatile X*>);
auto M = [*this] () mutable {
static_assert(is_same<decltype(this), X*>);
auto N = [this] {
static_assert(is_same<decltype(this), X*>);
auto M = [] {
static_assert(is_same<decltype(this), X*>);
};
};
auto N2 = [*this] {
static_assert(is_same<decltype(this), const X*>);
};
};
auto M2 = [*this] () {
static_assert(is_same<decltype(this), const X*>);
auto N = [this] {
static_assert(is_same<decltype(this), const X*>);
};
};
};
}
};
} //end ns5
namespace ns6 {
struct X {
double d;
auto foo() const {
auto L = [*this] () mutable {
auto M = [=] (auto a) {
auto N = [this] {
++d;
static_assert(is_same<decltype(this), X*>);
auto O = [*this] {
static_assert(is_same<decltype(this), const X*>);
};
};
N();
static_assert(is_same<decltype(this), X*>);
};
return M;
};
return L;
}
};
int main() {
auto L = X{}.foo();
auto M = L();
M(3.14);
}
} // end ns6
namespace ns7 {
struct X {
double d;
X();
X(const X&);
X(X&) = delete;
auto foo() const {
//OK - the object used to initialize our capture is a const object and so prefers the non-deleted ctor.
const auto &&L = [*this] { };
}
};
int main() {
X x;
x.foo();
}
} // end ns7
} //end ns test_star_this