forked from OSchip/llvm-project
Fix some minor bugs and add a lot more test cases for defaulted
constructors, including two more FIXMEs (one of which I don't actually understand). llvm-svn: 131487
This commit is contained in:
parent
fa7afe2a05
commit
604aeb3849
|
@ -3644,6 +3644,9 @@ def err_defaulted_copy_assign_params : Error<
|
|||
def err_defaulted_copy_assign_return_type : Error<
|
||||
"an explicitly-defaulted copy assignment operator must return an unqualified "
|
||||
"lvalue reference to its class type">;
|
||||
def err_defaulted_copy_assign_not_ref : Error<
|
||||
"the parameter for an explicitly-defaulted copy assignment operator must be an "
|
||||
"lvalue reference type">;
|
||||
def err_defaulted_copy_assign_volatile_param : Error<
|
||||
"the parameter for an explicitly-defaulted copy assignment operator may not "
|
||||
"be volatile">;
|
||||
|
|
|
@ -3187,17 +3187,21 @@ void Sema::CheckExplicitlyDefaultedCopyAssignment(CXXMethodDecl *MD) {
|
|||
*ExceptionType = Context.getFunctionType(
|
||||
Context.VoidTy, 0, 0, EPI)->getAs<FunctionProtoType>();
|
||||
|
||||
// Check for parameter type matching.
|
||||
// This is a copy ctor so we know it's a cv-qualified reference to T.
|
||||
QualType ArgType = OperType->getArgType(0);
|
||||
if (ArgType->getPointeeType().isVolatileQualified()) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_volatile_param);
|
||||
HadError = true;
|
||||
}
|
||||
if (ArgType->getPointeeType().isConstQualified() && !Const) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_const_param);
|
||||
if (!ArgType->isReferenceType()) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_not_ref);
|
||||
HadError = true;
|
||||
} else {
|
||||
if (ArgType->getPointeeType().isVolatileQualified()) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_volatile_param);
|
||||
HadError = true;
|
||||
}
|
||||
if (ArgType->getPointeeType().isConstQualified() && !Const) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_const_param);
|
||||
HadError = true;
|
||||
}
|
||||
}
|
||||
|
||||
if (OperType->getTypeQuals()) {
|
||||
Diag(MD->getLocation(), diag::err_defaulted_copy_assign_quals);
|
||||
HadError = true;
|
||||
|
|
|
@ -0,0 +1,69 @@
|
|||
// RUN: %clang_cc1 -std=c++0x -fsyntax-only -verify %s
|
||||
|
||||
struct non_copiable {
|
||||
non_copiable(const non_copiable&) = delete; // expected-note {{marked deleted here}}
|
||||
non_copiable& operator = (const non_copiable&) = delete; // expected-note {{explicitly deleted}}
|
||||
non_copiable() = default;
|
||||
};
|
||||
|
||||
struct non_const_copy {
|
||||
non_const_copy(non_const_copy&) = default; // expected-note {{not viable}}
|
||||
non_const_copy& operator = (non_const_copy&) & = default; // expected-note {{not viable}}
|
||||
non_const_copy& operator = (non_const_copy&) && = default; // expected-note {{not viable}}
|
||||
non_const_copy() = default; // expected-note {{not viable}}
|
||||
};
|
||||
|
||||
void fn1 () {
|
||||
non_copiable nc;
|
||||
non_copiable nc2 = nc; // expected-error {{deleted constructor}}
|
||||
nc = nc; // expected-error {{deleted operator}}
|
||||
|
||||
non_const_copy ncc;
|
||||
non_const_copy ncc2 = ncc;
|
||||
ncc = ncc2;
|
||||
const non_const_copy cncc;
|
||||
non_const_copy ncc3 = cncc; // expected-error {{no matching}}
|
||||
ncc = cncc; // expected-error {{no viable overloaded}}
|
||||
};
|
||||
|
||||
struct non_const_derived : non_const_copy {
|
||||
non_const_derived(const non_const_derived&) = default; // expected-error {{requires it to be non-const}}
|
||||
non_const_derived& operator =(non_const_derived&) = default;
|
||||
};
|
||||
|
||||
struct bad_decls {
|
||||
bad_decls(volatile bad_decls&) = default; // expected-error {{may not be volatile}}
|
||||
bad_decls&& operator = (bad_decls) = default; // expected-error 2{{lvalue reference}}
|
||||
bad_decls& operator = (volatile bad_decls&) = default; // expected-error {{may not be volatile}}
|
||||
bad_decls& operator = (const bad_decls&) const = default; // expected-error {{may not have 'const' or 'volatile' qualifiers}}
|
||||
};
|
||||
|
||||
struct A {}; struct B {};
|
||||
|
||||
struct except_spec_a {
|
||||
virtual ~except_spec_a() throw(A);
|
||||
except_spec_a() throw(A);
|
||||
};
|
||||
struct except_spec_b {
|
||||
virtual ~except_spec_b() throw(B);
|
||||
except_spec_b() throw(B);
|
||||
};
|
||||
|
||||
struct except_spec_d_good : except_spec_a, except_spec_b {
|
||||
~except_spec_d_good();
|
||||
};
|
||||
except_spec_d_good::~except_spec_d_good() = default;
|
||||
// FIXME: This should error in the virtual override check.
|
||||
// It doesn't because we generate the implicit specification later than
|
||||
// appropriate.
|
||||
struct except_spec_d_bad : except_spec_a, except_spec_b {
|
||||
~except_spec_d_bad() = default;
|
||||
};
|
||||
|
||||
// FIXME: This should error because the exceptions spec doesn't match.
|
||||
struct except_spec_d_mismatch : except_spec_a, except_spec_b {
|
||||
except_spec_d_mismatch() throw(A) = default;
|
||||
};
|
||||
struct except_spec_d_match : except_spec_a, except_spec_b {
|
||||
except_spec_d_match() throw(A, B) = default;
|
||||
};
|
Loading…
Reference in New Issue