From: Sebastian Redl Date: Mon, 17 Nov 2008 23:24:37 +0000 (+0000) Subject: Implement effects of 'mutable', and a few comments from Chris on its parsing. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=a11f42f4bca694b9be91350d0a74815f119e3fbf;p=clang Implement effects of 'mutable', and a few comments from Chris on its parsing. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@59470 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/clang/AST/DeclCXX.h b/include/clang/AST/DeclCXX.h index 86cd68ac8e..77eb8fade4 100644 --- a/include/clang/AST/DeclCXX.h +++ b/include/clang/AST/DeclCXX.h @@ -112,17 +112,20 @@ protected: /// CXXFieldDecl - Represents an instance field of a C++ struct/union/class. class CXXFieldDecl : public FieldDecl { CXXRecordDecl *Parent; + bool Mutable; CXXFieldDecl(CXXRecordDecl *RD, SourceLocation L, IdentifierInfo *Id, - QualType T, Expr *BW = NULL) - : FieldDecl(CXXField, L, Id, T, BW), Parent(RD) {} + QualType T, bool Mut, Expr *BW = NULL) + : FieldDecl(CXXField, L, Id, T, BW), Parent(RD), Mutable(Mut) {} public: static CXXFieldDecl *Create(ASTContext &C, CXXRecordDecl *RD,SourceLocation L, - IdentifierInfo *Id, QualType T, Expr *BW = NULL); + IdentifierInfo *Id, QualType T, bool Mut, + Expr *BW = NULL); void setAccess(AccessSpecifier AS) { Access = AS; } AccessSpecifier getAccess() const { return AccessSpecifier(Access); } CXXRecordDecl *getParent() const { return Parent; } + bool isMutable() const { return Mutable; } // Implement isa/cast/dyncast/etc. static bool classof(const Decl *D) { return D->getKind() == CXXField; } diff --git a/lib/AST/DeclCXX.cpp b/lib/AST/DeclCXX.cpp index 4037404124..6be9eaf1a3 100644 --- a/lib/AST/DeclCXX.cpp +++ b/lib/AST/DeclCXX.cpp @@ -22,9 +22,9 @@ using namespace clang; CXXFieldDecl *CXXFieldDecl::Create(ASTContext &C, CXXRecordDecl *RD, SourceLocation L, IdentifierInfo *Id, - QualType T, Expr *BW) { + QualType T, bool Mut, Expr *BW) { void *Mem = C.getAllocator().Allocate(); - return new (Mem) CXXFieldDecl(RD, L, Id, T, BW); + return new (Mem) CXXFieldDecl(RD, L, Id, T, Mut, BW); } CXXRecordDecl::CXXRecordDecl(TagKind TK, DeclContext *DC, diff --git a/lib/Sema/SemaDecl.cpp b/lib/Sema/SemaDecl.cpp index 9c0a40e74a..fdc4aff031 100644 --- a/lib/Sema/SemaDecl.cpp +++ b/lib/Sema/SemaDecl.cpp @@ -1149,6 +1149,7 @@ Sema::ActOnDeclarator(Scope *S, Declarator &D, DeclTy *lastDecl) { // an error here Diag(D.getIdentifierLoc(), diag::err_mutable_nonmember); InvalidDecl = true; + break; } IdentifierInfo *II = Name.getAsIdentifierInfo(); @@ -2603,7 +2604,9 @@ Sema::DeclTy *Sema::ActOnField(Scope *S, if (getLangOptions().CPlusPlus) { // FIXME: Replace CXXFieldDecls with FieldDecls for simple structs. NewFD = CXXFieldDecl::Create(Context, cast(CurContext), - Loc, II, T, BitWidth); + Loc, II, T, + D.getDeclSpec().getStorageClassSpec() == + DeclSpec::SCS_mutable, BitWidth); if (II) PushOnScopeChains(NewFD, S); } diff --git a/lib/Sema/SemaDeclCXX.cpp b/lib/Sema/SemaDeclCXX.cpp index f601fe4dc6..68dcdc8d2b 100644 --- a/lib/Sema/SemaDeclCXX.cpp +++ b/lib/Sema/SemaDeclCXX.cpp @@ -447,6 +447,8 @@ Sema::ActOnCXXMemberDeclarator(Scope *S, AccessSpecifier AS, Declarator &D, else Diag(DS.getThreadSpecLoc(), diag::err_mutable_function); + // FIXME: It would be nicer if the keyword was ignored only for this + // declarator. Otherwise we could get follow-up errors. D.getMutableDeclSpec().ClearStorageClassSpecs(); } else { QualType T = GetTypeForDeclarator(D, S); @@ -460,6 +462,8 @@ Sema::ActOnCXXMemberDeclarator(Scope *S, AccessSpecifier AS, Declarator &D, Diag(DS.getStorageClassSpecLoc(), err); else Diag(DS.getThreadSpecLoc(), err); + // FIXME: It would be nicer if the keyword was ignored only for this + // declarator. Otherwise we could get follow-up errors. D.getMutableDeclSpec().ClearStorageClassSpecs(); } } diff --git a/lib/Sema/SemaExpr.cpp b/lib/Sema/SemaExpr.cpp index 0506b4076e..2b621f2a2e 100644 --- a/lib/Sema/SemaExpr.cpp +++ b/lib/Sema/SemaExpr.cpp @@ -951,6 +951,10 @@ ActOnMemberReferenceExpr(ExprTy *Base, SourceLocation OpLoc, QualType MemberType = MemberDecl->getType(); unsigned combinedQualifiers = MemberType.getCVRQualifiers() | BaseType.getCVRQualifiers(); + if (CXXFieldDecl *CXXMember = dyn_cast(MemberDecl)) { + if (CXXMember->isMutable()) + combinedQualifiers &= ~QualType::Const; + } MemberType = MemberType.getQualifiedType(combinedQualifiers); return new MemberExpr(BaseExpr, OpKind == tok::arrow, MemberDecl, diff --git a/test/SemaCXX/class.cpp b/test/SemaCXX/class.cpp index ada508ac32..b7d1ca34dc 100644 --- a/test/SemaCXX/class.cpp +++ b/test/SemaCXX/class.cpp @@ -82,6 +82,18 @@ class C2 { } }; +struct C3 { + int i; + mutable int j; +}; +void f() +{ + const C3 c3 = { 1, 2 }; + (void)static_cast(&c3.i); // expected-error {{static_cast from 'int const *' to 'int *' is not allowed}} + // but no error here + (void)static_cast(&c3.j); +} + // Play with mutable a bit more, to make sure it doesn't crash anything. mutable int gi; // expected-error {{error: 'mutable' can only be applied to member variables}} mutable void gfn(); // expected-error {{illegal storage class on function}}