| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 1 | //===--- Expr.cpp - Expression AST Node Implementation --------------------===// | 
|  | 2 | // | 
|  | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
| Chris Lattner | 5b12ab8 | 2007-12-29 19:59:25 +0000 | [diff] [blame] | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 | // | 
|  | 10 | // This file implements the Expr class and subclasses. | 
|  | 11 | // | 
|  | 12 | //===----------------------------------------------------------------------===// | 
|  | 13 |  | 
| Daniel Dunbar | 6e8aa53 | 2008-08-11 05:35:13 +0000 | [diff] [blame] | 14 | #include "clang/AST/Expr.h" | 
| Chris Lattner | 86ee286 | 2008-10-06 06:40:35 +0000 | [diff] [blame] | 15 | #include "clang/AST/APValue.h" | 
| Chris Lattner | 5c4664e | 2007-07-15 23:32:58 +0000 | [diff] [blame] | 16 | #include "clang/AST/ASTContext.h" | 
| Chris Lattner | 86ee286 | 2008-10-06 06:40:35 +0000 | [diff] [blame] | 17 | #include "clang/AST/DeclObjC.h" | 
| Douglas Gregor | 9a65793 | 2008-10-21 23:43:52 +0000 | [diff] [blame] | 18 | #include "clang/AST/DeclCXX.h" | 
| Douglas Gregor | ded2d7b | 2009-02-04 19:02:06 +0000 | [diff] [blame] | 19 | #include "clang/AST/DeclTemplate.h" | 
| Anders Carlsson | 15b73de | 2009-07-18 19:43:29 +0000 | [diff] [blame^] | 20 | #include "clang/AST/RecordLayout.h" | 
| Chris Lattner | 5e9a878 | 2006-11-04 06:21:51 +0000 | [diff] [blame] | 21 | #include "clang/AST/StmtVisitor.h" | 
| Chris Lattner | 15ba949 | 2009-06-14 01:54:56 +0000 | [diff] [blame] | 22 | #include "clang/Basic/Builtins.h" | 
| Chris Lattner | a7944d8 | 2007-11-27 18:22:04 +0000 | [diff] [blame] | 23 | #include "clang/Basic/TargetInfo.h" | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 24 | #include <algorithm> | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 25 | using namespace clang; | 
|  | 26 |  | 
| Chris Lattner | 0eedafe | 2006-08-24 04:56:27 +0000 | [diff] [blame] | 27 | //===----------------------------------------------------------------------===// | 
|  | 28 | // Primary Expressions. | 
|  | 29 | //===----------------------------------------------------------------------===// | 
|  | 30 |  | 
| Sebastian Redl | e769ecf | 2009-05-16 18:50:46 +0000 | [diff] [blame] | 31 | PredefinedExpr* PredefinedExpr::Clone(ASTContext &C) const { | 
|  | 32 | return new (C) PredefinedExpr(Loc, getType(), Type); | 
|  | 33 | } | 
|  | 34 |  | 
| Anders Carlsson | a390581 | 2009-03-15 18:34:13 +0000 | [diff] [blame] | 35 | IntegerLiteral* IntegerLiteral::Clone(ASTContext &C) const { | 
|  | 36 | return new (C) IntegerLiteral(Value, getType(), Loc); | 
|  | 37 | } | 
|  | 38 |  | 
| Sebastian Redl | e769ecf | 2009-05-16 18:50:46 +0000 | [diff] [blame] | 39 | CharacterLiteral* CharacterLiteral::Clone(ASTContext &C) const { | 
|  | 40 | return new (C) CharacterLiteral(Value, IsWide, getType(), Loc); | 
|  | 41 | } | 
|  | 42 |  | 
|  | 43 | FloatingLiteral* FloatingLiteral::Clone(ASTContext &C) const { | 
| Chris Lattner | e4edb8e | 2009-06-29 17:34:55 +0000 | [diff] [blame] | 44 | return new (C) FloatingLiteral(Value, IsExact, getType(), Loc); | 
| Sebastian Redl | e769ecf | 2009-05-16 18:50:46 +0000 | [diff] [blame] | 45 | } | 
|  | 46 |  | 
| Douglas Gregor | 38e7f8b | 2009-05-18 22:38:38 +0000 | [diff] [blame] | 47 | ImaginaryLiteral* ImaginaryLiteral::Clone(ASTContext &C) const { | 
|  | 48 | // FIXME: Use virtual Clone(), once it is available | 
|  | 49 | Expr *ClonedVal = 0; | 
|  | 50 | if (const IntegerLiteral *IntLit = dyn_cast<IntegerLiteral>(Val)) | 
|  | 51 | ClonedVal = IntLit->Clone(C); | 
|  | 52 | else | 
|  | 53 | ClonedVal = cast<FloatingLiteral>(Val)->Clone(C); | 
|  | 54 | return new (C) ImaginaryLiteral(ClonedVal, getType()); | 
|  | 55 | } | 
|  | 56 |  | 
| Sebastian Redl | e769ecf | 2009-05-16 18:50:46 +0000 | [diff] [blame] | 57 | GNUNullExpr* GNUNullExpr::Clone(ASTContext &C) const { | 
|  | 58 | return new (C) GNUNullExpr(getType(), TokenLoc); | 
|  | 59 | } | 
|  | 60 |  | 
| Chris Lattner | a017313 | 2008-06-07 22:13:43 +0000 | [diff] [blame] | 61 | /// getValueAsApproximateDouble - This returns the value as an inaccurate | 
|  | 62 | /// double.  Note that this may cause loss of precision, but is useful for | 
|  | 63 | /// debugging dumps, etc. | 
|  | 64 | double FloatingLiteral::getValueAsApproximateDouble() const { | 
|  | 65 | llvm::APFloat V = getValue(); | 
| Dale Johannesen | c48814b | 2008-10-09 23:02:32 +0000 | [diff] [blame] | 66 | bool ignored; | 
|  | 67 | V.convert(llvm::APFloat::IEEEdouble, llvm::APFloat::rmNearestTiesToEven, | 
|  | 68 | &ignored); | 
| Chris Lattner | a017313 | 2008-06-07 22:13:43 +0000 | [diff] [blame] | 69 | return V.convertToDouble(); | 
|  | 70 | } | 
|  | 71 |  | 
| Chris Lattner | f83b5af | 2009-02-18 06:40:38 +0000 | [diff] [blame] | 72 | StringLiteral *StringLiteral::Create(ASTContext &C, const char *StrData, | 
|  | 73 | unsigned ByteLength, bool Wide, | 
|  | 74 | QualType Ty, | 
| Anders Carlsson | a390581 | 2009-03-15 18:34:13 +0000 | [diff] [blame] | 75 | const SourceLocation *Loc, | 
|  | 76 | unsigned NumStrs) { | 
| Chris Lattner | f83b5af | 2009-02-18 06:40:38 +0000 | [diff] [blame] | 77 | // Allocate enough space for the StringLiteral plus an array of locations for | 
|  | 78 | // any concatenated string tokens. | 
|  | 79 | void *Mem = C.Allocate(sizeof(StringLiteral)+ | 
|  | 80 | sizeof(SourceLocation)*(NumStrs-1), | 
|  | 81 | llvm::alignof<StringLiteral>()); | 
|  | 82 | StringLiteral *SL = new (Mem) StringLiteral(Ty); | 
|  | 83 |  | 
| Steve Naroff | df7855b | 2007-02-21 23:46:25 +0000 | [diff] [blame] | 84 | // OPTIMIZE: could allocate this appended to the StringLiteral. | 
| Chris Lattner | f83b5af | 2009-02-18 06:40:38 +0000 | [diff] [blame] | 85 | char *AStrData = new (C, 1) char[ByteLength]; | 
|  | 86 | memcpy(AStrData, StrData, ByteLength); | 
|  | 87 | SL->StrData = AStrData; | 
|  | 88 | SL->ByteLength = ByteLength; | 
|  | 89 | SL->IsWide = Wide; | 
|  | 90 | SL->TokLocs[0] = Loc[0]; | 
|  | 91 | SL->NumConcatenated = NumStrs; | 
| Chris Lattner | d3e9895 | 2006-10-06 05:22:26 +0000 | [diff] [blame] | 92 |  | 
| Chris Lattner | 630970d | 2009-02-18 05:49:11 +0000 | [diff] [blame] | 93 | if (NumStrs != 1) | 
| Chris Lattner | f83b5af | 2009-02-18 06:40:38 +0000 | [diff] [blame] | 94 | memcpy(&SL->TokLocs[1], Loc+1, sizeof(SourceLocation)*(NumStrs-1)); | 
|  | 95 | return SL; | 
| Chris Lattner | 630970d | 2009-02-18 05:49:11 +0000 | [diff] [blame] | 96 | } | 
|  | 97 |  | 
| Douglas Gregor | 958dfc9 | 2009-04-15 16:35:07 +0000 | [diff] [blame] | 98 | StringLiteral *StringLiteral::CreateEmpty(ASTContext &C, unsigned NumStrs) { | 
|  | 99 | void *Mem = C.Allocate(sizeof(StringLiteral)+ | 
|  | 100 | sizeof(SourceLocation)*(NumStrs-1), | 
|  | 101 | llvm::alignof<StringLiteral>()); | 
|  | 102 | StringLiteral *SL = new (Mem) StringLiteral(QualType()); | 
|  | 103 | SL->StrData = 0; | 
|  | 104 | SL->ByteLength = 0; | 
|  | 105 | SL->NumConcatenated = NumStrs; | 
|  | 106 | return SL; | 
|  | 107 | } | 
|  | 108 |  | 
| Anders Carlsson | a390581 | 2009-03-15 18:34:13 +0000 | [diff] [blame] | 109 | StringLiteral* StringLiteral::Clone(ASTContext &C) const { | 
|  | 110 | return Create(C, StrData, ByteLength, IsWide, getType(), | 
|  | 111 | TokLocs, NumConcatenated); | 
|  | 112 | } | 
| Chris Lattner | 630970d | 2009-02-18 05:49:11 +0000 | [diff] [blame] | 113 |  | 
| Ted Kremenek | 6b7ecf6 | 2009-02-06 19:55:15 +0000 | [diff] [blame] | 114 | void StringLiteral::Destroy(ASTContext &C) { | 
| Ted Kremenek | 5a20195 | 2009-02-07 01:47:29 +0000 | [diff] [blame] | 115 | C.Deallocate(const_cast<char*>(StrData)); | 
| Ted Kremenek | e2559b2 | 2009-02-09 17:10:09 +0000 | [diff] [blame] | 116 | this->~StringLiteral(); | 
|  | 117 | C.Deallocate(this); | 
| Chris Lattner | d3e9895 | 2006-10-06 05:22:26 +0000 | [diff] [blame] | 118 | } | 
|  | 119 |  | 
| Douglas Gregor | 958dfc9 | 2009-04-15 16:35:07 +0000 | [diff] [blame] | 120 | void StringLiteral::setStrData(ASTContext &C, const char *Str, unsigned Len) { | 
|  | 121 | if (StrData) | 
|  | 122 | C.Deallocate(const_cast<char*>(StrData)); | 
|  | 123 |  | 
|  | 124 | char *AStrData = new (C, 1) char[Len]; | 
|  | 125 | memcpy(AStrData, Str, Len); | 
|  | 126 | StrData = AStrData; | 
|  | 127 | ByteLength = Len; | 
|  | 128 | } | 
|  | 129 |  | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 130 | /// getOpcodeStr - Turn an Opcode enum value into the punctuation char it | 
|  | 131 | /// corresponds to, e.g. "sizeof" or "[pre]++". | 
|  | 132 | const char *UnaryOperator::getOpcodeStr(Opcode Op) { | 
|  | 133 | switch (Op) { | 
| Chris Lattner | c52b118 | 2006-10-25 05:45:55 +0000 | [diff] [blame] | 134 | default: assert(0 && "Unknown unary operator"); | 
| Chris Lattner | 1576870 | 2006-11-05 23:54:51 +0000 | [diff] [blame] | 135 | case PostInc: return "++"; | 
|  | 136 | case PostDec: return "--"; | 
|  | 137 | case PreInc:  return "++"; | 
|  | 138 | case PreDec:  return "--"; | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 139 | case AddrOf:  return "&"; | 
|  | 140 | case Deref:   return "*"; | 
|  | 141 | case Plus:    return "+"; | 
|  | 142 | case Minus:   return "-"; | 
|  | 143 | case Not:     return "~"; | 
|  | 144 | case LNot:    return "!"; | 
|  | 145 | case Real:    return "__real"; | 
|  | 146 | case Imag:    return "__imag"; | 
| Chris Lattner | c52b118 | 2006-10-25 05:45:55 +0000 | [diff] [blame] | 147 | case Extension: return "__extension__"; | 
| Chris Lattner | f17bd42 | 2007-08-30 17:45:32 +0000 | [diff] [blame] | 148 | case OffsetOf: return "__builtin_offsetof"; | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 149 | } | 
|  | 150 | } | 
|  | 151 |  | 
| Douglas Gregor | 084d855 | 2009-03-13 23:49:33 +0000 | [diff] [blame] | 152 | UnaryOperator::Opcode | 
|  | 153 | UnaryOperator::getOverloadedOpcode(OverloadedOperatorKind OO, bool Postfix) { | 
|  | 154 | switch (OO) { | 
| Douglas Gregor | 084d855 | 2009-03-13 23:49:33 +0000 | [diff] [blame] | 155 | default: assert(false && "No unary operator for overloaded function"); | 
| Chris Lattner | 17556b2 | 2009-03-22 00:10:22 +0000 | [diff] [blame] | 156 | case OO_PlusPlus:   return Postfix ? PostInc : PreInc; | 
|  | 157 | case OO_MinusMinus: return Postfix ? PostDec : PreDec; | 
|  | 158 | case OO_Amp:        return AddrOf; | 
|  | 159 | case OO_Star:       return Deref; | 
|  | 160 | case OO_Plus:       return Plus; | 
|  | 161 | case OO_Minus:      return Minus; | 
|  | 162 | case OO_Tilde:      return Not; | 
|  | 163 | case OO_Exclaim:    return LNot; | 
| Douglas Gregor | 084d855 | 2009-03-13 23:49:33 +0000 | [diff] [blame] | 164 | } | 
|  | 165 | } | 
|  | 166 |  | 
|  | 167 | OverloadedOperatorKind UnaryOperator::getOverloadedOperator(Opcode Opc) { | 
|  | 168 | switch (Opc) { | 
|  | 169 | case PostInc: case PreInc: return OO_PlusPlus; | 
|  | 170 | case PostDec: case PreDec: return OO_MinusMinus; | 
|  | 171 | case AddrOf: return OO_Amp; | 
|  | 172 | case Deref: return OO_Star; | 
|  | 173 | case Plus: return OO_Plus; | 
|  | 174 | case Minus: return OO_Minus; | 
|  | 175 | case Not: return OO_Tilde; | 
|  | 176 | case LNot: return OO_Exclaim; | 
|  | 177 | default: return OO_None; | 
|  | 178 | } | 
|  | 179 | } | 
|  | 180 |  | 
|  | 181 |  | 
| Chris Lattner | 0eedafe | 2006-08-24 04:56:27 +0000 | [diff] [blame] | 182 | //===----------------------------------------------------------------------===// | 
|  | 183 | // Postfix Operators. | 
|  | 184 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | e165d94 | 2006-08-24 04:40:38 +0000 | [diff] [blame] | 185 |  | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 186 | CallExpr::CallExpr(ASTContext& C, StmtClass SC, Expr *fn, Expr **args, | 
| Ted Kremenek | 5a20195 | 2009-02-07 01:47:29 +0000 | [diff] [blame] | 187 | unsigned numargs, QualType t, SourceLocation rparenloc) | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 188 | : Expr(SC, t, | 
|  | 189 | fn->isTypeDependent() || hasAnyTypeDependentArguments(args, numargs), | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 190 | fn->isValueDependent() || hasAnyValueDependentArguments(args,numargs)), | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 191 | NumArgs(numargs) { | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 192 |  | 
|  | 193 | SubExprs = new (C) Stmt*[numargs+1]; | 
| Douglas Gregor | 993603d | 2008-11-14 16:09:21 +0000 | [diff] [blame] | 194 | SubExprs[FN] = fn; | 
|  | 195 | for (unsigned i = 0; i != numargs; ++i) | 
|  | 196 | SubExprs[i+ARGS_START] = args[i]; | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 197 |  | 
| Douglas Gregor | 993603d | 2008-11-14 16:09:21 +0000 | [diff] [blame] | 198 | RParenLoc = rparenloc; | 
|  | 199 | } | 
| Nate Begeman | 1e36a85 | 2008-01-17 17:46:27 +0000 | [diff] [blame] | 200 |  | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 201 | CallExpr::CallExpr(ASTContext& C, Expr *fn, Expr **args, unsigned numargs, | 
|  | 202 | QualType t, SourceLocation rparenloc) | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 203 | : Expr(CallExprClass, t, | 
|  | 204 | fn->isTypeDependent() || hasAnyTypeDependentArguments(args, numargs), | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 205 | fn->isValueDependent() || hasAnyValueDependentArguments(args,numargs)), | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 206 | NumArgs(numargs) { | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 207 |  | 
|  | 208 | SubExprs = new (C) Stmt*[numargs+1]; | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 209 | SubExprs[FN] = fn; | 
| Chris Lattner | e165d94 | 2006-08-24 04:40:38 +0000 | [diff] [blame] | 210 | for (unsigned i = 0; i != numargs; ++i) | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 211 | SubExprs[i+ARGS_START] = args[i]; | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 212 |  | 
| Chris Lattner | 9b3b9a1 | 2007-06-27 06:08:24 +0000 | [diff] [blame] | 213 | RParenLoc = rparenloc; | 
| Chris Lattner | e165d94 | 2006-08-24 04:40:38 +0000 | [diff] [blame] | 214 | } | 
|  | 215 |  | 
| Argyrios Kyrtzidis | eeaaead | 2009-07-14 03:19:21 +0000 | [diff] [blame] | 216 | CallExpr::CallExpr(ASTContext &C, StmtClass SC, EmptyShell Empty) | 
|  | 217 | : Expr(SC, Empty), SubExprs(0), NumArgs(0) { | 
| Douglas Gregor | e20a2e5 | 2009-04-15 17:43:59 +0000 | [diff] [blame] | 218 | SubExprs = new (C) Stmt*[1]; | 
|  | 219 | } | 
|  | 220 |  | 
| Ted Kremenek | d7b4f40 | 2009-02-09 20:51:47 +0000 | [diff] [blame] | 221 | void CallExpr::Destroy(ASTContext& C) { | 
|  | 222 | DestroyChildren(C); | 
|  | 223 | if (SubExprs) C.Deallocate(SubExprs); | 
|  | 224 | this->~CallExpr(); | 
|  | 225 | C.Deallocate(this); | 
|  | 226 | } | 
|  | 227 |  | 
| Zhongxing Xu | 3c8fa97 | 2009-07-17 07:29:51 +0000 | [diff] [blame] | 228 | FunctionDecl *CallExpr::getDirectCallee() { | 
|  | 229 | Expr *CEE = getCallee()->IgnoreParenCasts(); | 
| Chris Lattner | 5230191 | 2009-07-17 15:46:27 +0000 | [diff] [blame] | 230 | if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(CEE)) | 
| Zhongxing Xu | 3c8fa97 | 2009-07-17 07:29:51 +0000 | [diff] [blame] | 231 | return dyn_cast<FunctionDecl>(DRE->getDecl()); | 
| Zhongxing Xu | 3c8fa97 | 2009-07-17 07:29:51 +0000 | [diff] [blame] | 232 |  | 
|  | 233 | return 0; | 
|  | 234 | } | 
|  | 235 |  | 
| Chris Lattner | e4407ed | 2007-12-28 05:25:02 +0000 | [diff] [blame] | 236 | /// setNumArgs - This changes the number of arguments present in this call. | 
|  | 237 | /// Any orphaned expressions are deleted by this, and any new operands are set | 
|  | 238 | /// to null. | 
| Ted Kremenek | 5a20195 | 2009-02-07 01:47:29 +0000 | [diff] [blame] | 239 | void CallExpr::setNumArgs(ASTContext& C, unsigned NumArgs) { | 
| Chris Lattner | e4407ed | 2007-12-28 05:25:02 +0000 | [diff] [blame] | 240 | // No change, just return. | 
|  | 241 | if (NumArgs == getNumArgs()) return; | 
|  | 242 |  | 
|  | 243 | // If shrinking # arguments, just delete the extras and forgot them. | 
|  | 244 | if (NumArgs < getNumArgs()) { | 
|  | 245 | for (unsigned i = NumArgs, e = getNumArgs(); i != e; ++i) | 
| Ted Kremenek | 5a20195 | 2009-02-07 01:47:29 +0000 | [diff] [blame] | 246 | getArg(i)->Destroy(C); | 
| Chris Lattner | e4407ed | 2007-12-28 05:25:02 +0000 | [diff] [blame] | 247 | this->NumArgs = NumArgs; | 
|  | 248 | return; | 
|  | 249 | } | 
|  | 250 |  | 
|  | 251 | // Otherwise, we are growing the # arguments.  New an bigger argument array. | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 252 | Stmt **NewSubExprs = new Stmt*[NumArgs+1]; | 
| Chris Lattner | e4407ed | 2007-12-28 05:25:02 +0000 | [diff] [blame] | 253 | // Copy over args. | 
|  | 254 | for (unsigned i = 0; i != getNumArgs()+ARGS_START; ++i) | 
|  | 255 | NewSubExprs[i] = SubExprs[i]; | 
|  | 256 | // Null out new args. | 
|  | 257 | for (unsigned i = getNumArgs()+ARGS_START; i != NumArgs+ARGS_START; ++i) | 
|  | 258 | NewSubExprs[i] = 0; | 
|  | 259 |  | 
| Douglas Gregor | ba6e557 | 2009-04-17 21:46:47 +0000 | [diff] [blame] | 260 | if (SubExprs) C.Deallocate(SubExprs); | 
| Chris Lattner | e4407ed | 2007-12-28 05:25:02 +0000 | [diff] [blame] | 261 | SubExprs = NewSubExprs; | 
|  | 262 | this->NumArgs = NumArgs; | 
|  | 263 | } | 
|  | 264 |  | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 265 | /// isBuiltinCall - If this is a call to a builtin, return the builtin ID.  If | 
|  | 266 | /// not, return 0. | 
| Douglas Gregor | e711f70 | 2009-02-14 18:57:46 +0000 | [diff] [blame] | 267 | unsigned CallExpr::isBuiltinCall(ASTContext &Context) const { | 
| Steve Naroff | f6e3b329 | 2008-01-31 01:07:12 +0000 | [diff] [blame] | 268 | // All simple function calls (e.g. func()) are implicitly cast to pointer to | 
|  | 269 | // function. As a result, we try and obtain the DeclRefExpr from the | 
|  | 270 | // ImplicitCastExpr. | 
|  | 271 | const ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(getCallee()); | 
|  | 272 | if (!ICE) // FIXME: deal with more complex calls (e.g. (func)(), (*func)()). | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 273 | return 0; | 
|  | 274 |  | 
| Steve Naroff | f6e3b329 | 2008-01-31 01:07:12 +0000 | [diff] [blame] | 275 | const DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(ICE->getSubExpr()); | 
|  | 276 | if (!DRE) | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 277 | return 0; | 
|  | 278 |  | 
| Anders Carlsson | fbcf676 | 2008-01-31 02:13:57 +0000 | [diff] [blame] | 279 | const FunctionDecl *FDecl = dyn_cast<FunctionDecl>(DRE->getDecl()); | 
|  | 280 | if (!FDecl) | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 281 | return 0; | 
|  | 282 |  | 
| Douglas Gregor | 9eb16ea | 2008-11-21 15:30:19 +0000 | [diff] [blame] | 283 | if (!FDecl->getIdentifier()) | 
|  | 284 | return 0; | 
|  | 285 |  | 
| Douglas Gregor | e711f70 | 2009-02-14 18:57:46 +0000 | [diff] [blame] | 286 | return FDecl->getBuiltinID(Context); | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 287 | } | 
| Anders Carlsson | fbcf676 | 2008-01-31 02:13:57 +0000 | [diff] [blame] | 288 |  | 
| Anders Carlsson | 00a2759 | 2009-05-26 04:57:27 +0000 | [diff] [blame] | 289 | QualType CallExpr::getCallReturnType() const { | 
|  | 290 | QualType CalleeType = getCallee()->getType(); | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 291 | if (const PointerType *FnTypePtr = CalleeType->getAsPointerType()) | 
| Anders Carlsson | 00a2759 | 2009-05-26 04:57:27 +0000 | [diff] [blame] | 292 | CalleeType = FnTypePtr->getPointeeType(); | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 293 | else if (const BlockPointerType *BPT = CalleeType->getAsBlockPointerType()) | 
| Anders Carlsson | 00a2759 | 2009-05-26 04:57:27 +0000 | [diff] [blame] | 294 | CalleeType = BPT->getPointeeType(); | 
|  | 295 |  | 
|  | 296 | const FunctionType *FnType = CalleeType->getAsFunctionType(); | 
|  | 297 | return FnType->getResultType(); | 
|  | 298 | } | 
| Chris Lattner | 01ff98a | 2008-10-06 05:00:53 +0000 | [diff] [blame] | 299 |  | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 300 | /// getOpcodeStr - Turn an Opcode enum value into the punctuation char it | 
|  | 301 | /// corresponds to, e.g. "<<=". | 
|  | 302 | const char *BinaryOperator::getOpcodeStr(Opcode Op) { | 
|  | 303 | switch (Op) { | 
| Douglas Gregor | 0f60e9a | 2009-03-12 22:51:37 +0000 | [diff] [blame] | 304 | case PtrMemD:   return ".*"; | 
|  | 305 | case PtrMemI:   return "->*"; | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 306 | case Mul:       return "*"; | 
|  | 307 | case Div:       return "/"; | 
|  | 308 | case Rem:       return "%"; | 
|  | 309 | case Add:       return "+"; | 
|  | 310 | case Sub:       return "-"; | 
|  | 311 | case Shl:       return "<<"; | 
|  | 312 | case Shr:       return ">>"; | 
|  | 313 | case LT:        return "<"; | 
|  | 314 | case GT:        return ">"; | 
|  | 315 | case LE:        return "<="; | 
|  | 316 | case GE:        return ">="; | 
|  | 317 | case EQ:        return "=="; | 
|  | 318 | case NE:        return "!="; | 
|  | 319 | case And:       return "&"; | 
|  | 320 | case Xor:       return "^"; | 
|  | 321 | case Or:        return "|"; | 
|  | 322 | case LAnd:      return "&&"; | 
|  | 323 | case LOr:       return "||"; | 
|  | 324 | case Assign:    return "="; | 
|  | 325 | case MulAssign: return "*="; | 
|  | 326 | case DivAssign: return "/="; | 
|  | 327 | case RemAssign: return "%="; | 
|  | 328 | case AddAssign: return "+="; | 
|  | 329 | case SubAssign: return "-="; | 
|  | 330 | case ShlAssign: return "<<="; | 
|  | 331 | case ShrAssign: return ">>="; | 
|  | 332 | case AndAssign: return "&="; | 
|  | 333 | case XorAssign: return "^="; | 
|  | 334 | case OrAssign:  return "|="; | 
|  | 335 | case Comma:     return ","; | 
|  | 336 | } | 
| Douglas Gregor | 0f60e9a | 2009-03-12 22:51:37 +0000 | [diff] [blame] | 337 |  | 
|  | 338 | return ""; | 
| Chris Lattner | 1b92649 | 2006-08-23 06:42:10 +0000 | [diff] [blame] | 339 | } | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 340 |  | 
| Douglas Gregor | 1baf54e | 2009-03-13 18:40:31 +0000 | [diff] [blame] | 341 | BinaryOperator::Opcode | 
|  | 342 | BinaryOperator::getOverloadedOpcode(OverloadedOperatorKind OO) { | 
|  | 343 | switch (OO) { | 
| Chris Lattner | 17556b2 | 2009-03-22 00:10:22 +0000 | [diff] [blame] | 344 | default: assert(false && "Not an overloadable binary operator"); | 
| Douglas Gregor | 1baf54e | 2009-03-13 18:40:31 +0000 | [diff] [blame] | 345 | case OO_Plus: return Add; | 
|  | 346 | case OO_Minus: return Sub; | 
|  | 347 | case OO_Star: return Mul; | 
|  | 348 | case OO_Slash: return Div; | 
|  | 349 | case OO_Percent: return Rem; | 
|  | 350 | case OO_Caret: return Xor; | 
|  | 351 | case OO_Amp: return And; | 
|  | 352 | case OO_Pipe: return Or; | 
|  | 353 | case OO_Equal: return Assign; | 
|  | 354 | case OO_Less: return LT; | 
|  | 355 | case OO_Greater: return GT; | 
|  | 356 | case OO_PlusEqual: return AddAssign; | 
|  | 357 | case OO_MinusEqual: return SubAssign; | 
|  | 358 | case OO_StarEqual: return MulAssign; | 
|  | 359 | case OO_SlashEqual: return DivAssign; | 
|  | 360 | case OO_PercentEqual: return RemAssign; | 
|  | 361 | case OO_CaretEqual: return XorAssign; | 
|  | 362 | case OO_AmpEqual: return AndAssign; | 
|  | 363 | case OO_PipeEqual: return OrAssign; | 
|  | 364 | case OO_LessLess: return Shl; | 
|  | 365 | case OO_GreaterGreater: return Shr; | 
|  | 366 | case OO_LessLessEqual: return ShlAssign; | 
|  | 367 | case OO_GreaterGreaterEqual: return ShrAssign; | 
|  | 368 | case OO_EqualEqual: return EQ; | 
|  | 369 | case OO_ExclaimEqual: return NE; | 
|  | 370 | case OO_LessEqual: return LE; | 
|  | 371 | case OO_GreaterEqual: return GE; | 
|  | 372 | case OO_AmpAmp: return LAnd; | 
|  | 373 | case OO_PipePipe: return LOr; | 
|  | 374 | case OO_Comma: return Comma; | 
|  | 375 | case OO_ArrowStar: return PtrMemI; | 
| Douglas Gregor | 1baf54e | 2009-03-13 18:40:31 +0000 | [diff] [blame] | 376 | } | 
|  | 377 | } | 
|  | 378 |  | 
|  | 379 | OverloadedOperatorKind BinaryOperator::getOverloadedOperator(Opcode Opc) { | 
|  | 380 | static const OverloadedOperatorKind OverOps[] = { | 
|  | 381 | /* .* Cannot be overloaded */OO_None, OO_ArrowStar, | 
|  | 382 | OO_Star, OO_Slash, OO_Percent, | 
|  | 383 | OO_Plus, OO_Minus, | 
|  | 384 | OO_LessLess, OO_GreaterGreater, | 
|  | 385 | OO_Less, OO_Greater, OO_LessEqual, OO_GreaterEqual, | 
|  | 386 | OO_EqualEqual, OO_ExclaimEqual, | 
|  | 387 | OO_Amp, | 
|  | 388 | OO_Caret, | 
|  | 389 | OO_Pipe, | 
|  | 390 | OO_AmpAmp, | 
|  | 391 | OO_PipePipe, | 
|  | 392 | OO_Equal, OO_StarEqual, | 
|  | 393 | OO_SlashEqual, OO_PercentEqual, | 
|  | 394 | OO_PlusEqual, OO_MinusEqual, | 
|  | 395 | OO_LessLessEqual, OO_GreaterGreaterEqual, | 
|  | 396 | OO_AmpEqual, OO_CaretEqual, | 
|  | 397 | OO_PipeEqual, | 
|  | 398 | OO_Comma | 
|  | 399 | }; | 
|  | 400 | return OverOps[Opc]; | 
|  | 401 | } | 
|  | 402 |  | 
| Anders Carlsson | 4692db0 | 2007-08-31 04:56:16 +0000 | [diff] [blame] | 403 | InitListExpr::InitListExpr(SourceLocation lbraceloc, | 
| Chris Lattner | 07d754a | 2008-10-26 23:43:26 +0000 | [diff] [blame] | 404 | Expr **initExprs, unsigned numInits, | 
| Douglas Gregor | 347f7ea | 2009-01-28 21:54:33 +0000 | [diff] [blame] | 405 | SourceLocation rbraceloc) | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 406 | : Expr(InitListExprClass, QualType(), | 
|  | 407 | hasAnyTypeDependentArguments(initExprs, numInits), | 
|  | 408 | hasAnyValueDependentArguments(initExprs, numInits)), | 
| Douglas Gregor | 5169570 | 2009-01-29 16:53:55 +0000 | [diff] [blame] | 409 | LBraceLoc(lbraceloc), RBraceLoc(rbraceloc), SyntacticForm(0), | 
| Douglas Gregor | bf7207a | 2009-01-29 19:42:23 +0000 | [diff] [blame] | 410 | UnionFieldInit(0), HadArrayRangeDesignator(false) { | 
| Chris Lattner | 07d754a | 2008-10-26 23:43:26 +0000 | [diff] [blame] | 411 |  | 
|  | 412 | InitExprs.insert(InitExprs.end(), initExprs, initExprs+numInits); | 
| Anders Carlsson | 4692db0 | 2007-08-31 04:56:16 +0000 | [diff] [blame] | 413 | } | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 414 |  | 
| Douglas Gregor | 6d00c99 | 2009-03-20 23:58:33 +0000 | [diff] [blame] | 415 | void InitListExpr::reserveInits(unsigned NumInits) { | 
|  | 416 | if (NumInits > InitExprs.size()) | 
|  | 417 | InitExprs.reserve(NumInits); | 
|  | 418 | } | 
|  | 419 |  | 
| Douglas Gregor | 347f7ea | 2009-01-28 21:54:33 +0000 | [diff] [blame] | 420 | void InitListExpr::resizeInits(ASTContext &Context, unsigned NumInits) { | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 421 | for (unsigned Idx = NumInits, LastIdx = InitExprs.size(); | 
| Daniel Dunbar | 45a2a20 | 2009-02-16 22:42:44 +0000 | [diff] [blame] | 422 | Idx < LastIdx; ++Idx) | 
| Douglas Gregor | 52a47e9 | 2009-03-20 23:38:03 +0000 | [diff] [blame] | 423 | InitExprs[Idx]->Destroy(Context); | 
| Douglas Gregor | 347f7ea | 2009-01-28 21:54:33 +0000 | [diff] [blame] | 424 | InitExprs.resize(NumInits, 0); | 
|  | 425 | } | 
|  | 426 |  | 
|  | 427 | Expr *InitListExpr::updateInit(unsigned Init, Expr *expr) { | 
|  | 428 | if (Init >= InitExprs.size()) { | 
|  | 429 | InitExprs.insert(InitExprs.end(), Init - InitExprs.size() + 1, 0); | 
|  | 430 | InitExprs.back() = expr; | 
|  | 431 | return 0; | 
|  | 432 | } | 
|  | 433 |  | 
|  | 434 | Expr *Result = cast_or_null<Expr>(InitExprs[Init]); | 
|  | 435 | InitExprs[Init] = expr; | 
|  | 436 | return Result; | 
|  | 437 | } | 
|  | 438 |  | 
| Steve Naroff | 991e99d | 2008-09-04 15:31:07 +0000 | [diff] [blame] | 439 | /// getFunctionType - Return the underlying function type for this block. | 
| Steve Naroff | c540d66 | 2008-09-03 18:15:37 +0000 | [diff] [blame] | 440 | /// | 
|  | 441 | const FunctionType *BlockExpr::getFunctionType() const { | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 442 | return getType()->getAsBlockPointerType()-> | 
| Steve Naroff | c540d66 | 2008-09-03 18:15:37 +0000 | [diff] [blame] | 443 | getPointeeType()->getAsFunctionType(); | 
|  | 444 | } | 
|  | 445 |  | 
| Steve Naroff | 415d3d5 | 2008-10-08 17:01:13 +0000 | [diff] [blame] | 446 | SourceLocation BlockExpr::getCaretLocation() const { | 
|  | 447 | return TheBlock->getCaretLocation(); | 
|  | 448 | } | 
| Douglas Gregor | e3dcb2d | 2009-04-18 00:02:19 +0000 | [diff] [blame] | 449 | const Stmt *BlockExpr::getBody() const { | 
|  | 450 | return TheBlock->getBody(); | 
|  | 451 | } | 
|  | 452 | Stmt *BlockExpr::getBody() { | 
|  | 453 | return TheBlock->getBody(); | 
|  | 454 | } | 
| Steve Naroff | 415d3d5 | 2008-10-08 17:01:13 +0000 | [diff] [blame] | 455 |  | 
|  | 456 |  | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 457 | //===----------------------------------------------------------------------===// | 
|  | 458 | // Generic Expression Routines | 
|  | 459 | //===----------------------------------------------------------------------===// | 
|  | 460 |  | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 461 | /// isUnusedResultAWarning - Return true if this immediate expression should | 
|  | 462 | /// be warned about if the result is unused.  If so, fill in Loc and Ranges | 
|  | 463 | /// with location to warn on and the source range[s] to report with the | 
|  | 464 | /// warning. | 
|  | 465 | bool Expr::isUnusedResultAWarning(SourceLocation &Loc, SourceRange &R1, | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 466 | SourceRange &R2) const { | 
| Anders Carlsson | 789e2cc | 2009-05-15 23:10:19 +0000 | [diff] [blame] | 467 | // Don't warn if the expr is type dependent. The type could end up | 
|  | 468 | // instantiating to void. | 
|  | 469 | if (isTypeDependent()) | 
|  | 470 | return false; | 
|  | 471 |  | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 472 | switch (getStmtClass()) { | 
|  | 473 | default: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 474 | Loc = getExprLoc(); | 
|  | 475 | R1 = getSourceRange(); | 
|  | 476 | return true; | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 477 | case ParenExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 478 | return cast<ParenExpr>(this)->getSubExpr()-> | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 479 | isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 480 | case UnaryOperatorClass: { | 
|  | 481 | const UnaryOperator *UO = cast<UnaryOperator>(this); | 
|  | 482 |  | 
|  | 483 | switch (UO->getOpcode()) { | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 484 | default: break; | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 485 | case UnaryOperator::PostInc: | 
|  | 486 | case UnaryOperator::PostDec: | 
|  | 487 | case UnaryOperator::PreInc: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 488 | case UnaryOperator::PreDec:                 // ++/-- | 
|  | 489 | return false;  // Not a warning. | 
| Chris Lattner | a44d116 | 2007-06-27 05:58:59 +0000 | [diff] [blame] | 490 | case UnaryOperator::Deref: | 
|  | 491 | // Dereferencing a volatile pointer is a side-effect. | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 492 | if (getType().isVolatileQualified()) | 
|  | 493 | return false; | 
|  | 494 | break; | 
| Chris Lattner | a44d116 | 2007-06-27 05:58:59 +0000 | [diff] [blame] | 495 | case UnaryOperator::Real: | 
|  | 496 | case UnaryOperator::Imag: | 
|  | 497 | // accessing a piece of a volatile complex is a side-effect. | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 498 | if (UO->getSubExpr()->getType().isVolatileQualified()) | 
|  | 499 | return false; | 
|  | 500 | break; | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 501 | case UnaryOperator::Extension: | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 502 | return UO->getSubExpr()->isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 503 | } | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 504 | Loc = UO->getOperatorLoc(); | 
|  | 505 | R1 = UO->getSubExpr()->getSourceRange(); | 
|  | 506 | return true; | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 507 | } | 
| Chris Lattner | ae7a834 | 2007-12-01 06:07:34 +0000 | [diff] [blame] | 508 | case BinaryOperatorClass: { | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 509 | const BinaryOperator *BO = cast<BinaryOperator>(this); | 
|  | 510 | // Consider comma to have side effects if the LHS or RHS does. | 
|  | 511 | if (BO->getOpcode() == BinaryOperator::Comma) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 512 | return BO->getRHS()->isUnusedResultAWarning(Loc, R1, R2) || | 
|  | 513 | BO->getLHS()->isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | ae7a834 | 2007-12-01 06:07:34 +0000 | [diff] [blame] | 514 |  | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 515 | if (BO->isAssignmentOp()) | 
|  | 516 | return false; | 
|  | 517 | Loc = BO->getOperatorLoc(); | 
|  | 518 | R1 = BO->getLHS()->getSourceRange(); | 
|  | 519 | R2 = BO->getRHS()->getSourceRange(); | 
|  | 520 | return true; | 
| Chris Lattner | ae7a834 | 2007-12-01 06:07:34 +0000 | [diff] [blame] | 521 | } | 
| Chris Lattner | 8692811 | 2007-08-25 02:00:02 +0000 | [diff] [blame] | 522 | case CompoundAssignOperatorClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 523 | return false; | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 524 |  | 
| Fariborz Jahanian | 9fac54d | 2007-12-01 19:58:28 +0000 | [diff] [blame] | 525 | case ConditionalOperatorClass: { | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 526 | // The condition must be evaluated, but if either the LHS or RHS is a | 
|  | 527 | // warning, warn about them. | 
| Fariborz Jahanian | 9fac54d | 2007-12-01 19:58:28 +0000 | [diff] [blame] | 528 | const ConditionalOperator *Exp = cast<ConditionalOperator>(this); | 
| Douglas Gregor | 78bd61f | 2009-06-18 16:11:24 +0000 | [diff] [blame] | 529 | if (Exp->getLHS() && | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 530 | Exp->getLHS()->isUnusedResultAWarning(Loc, R1, R2)) | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 531 | return true; | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 532 | return Exp->getRHS()->isUnusedResultAWarning(Loc, R1, R2); | 
| Fariborz Jahanian | 9fac54d | 2007-12-01 19:58:28 +0000 | [diff] [blame] | 533 | } | 
|  | 534 |  | 
| Chris Lattner | a44d116 | 2007-06-27 05:58:59 +0000 | [diff] [blame] | 535 | case MemberExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 536 | // If the base pointer or element is to a volatile pointer/field, accessing | 
|  | 537 | // it is a side effect. | 
|  | 538 | if (getType().isVolatileQualified()) | 
|  | 539 | return false; | 
|  | 540 | Loc = cast<MemberExpr>(this)->getMemberLoc(); | 
|  | 541 | R1 = SourceRange(Loc, Loc); | 
|  | 542 | R2 = cast<MemberExpr>(this)->getBase()->getSourceRange(); | 
|  | 543 | return true; | 
|  | 544 |  | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 545 | case ArraySubscriptExprClass: | 
| Chris Lattner | a44d116 | 2007-06-27 05:58:59 +0000 | [diff] [blame] | 546 | // If the base pointer or element is to a volatile pointer/field, accessing | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 547 | // it is a side effect. | 
|  | 548 | if (getType().isVolatileQualified()) | 
|  | 549 | return false; | 
|  | 550 | Loc = cast<ArraySubscriptExpr>(this)->getRBracketLoc(); | 
|  | 551 | R1 = cast<ArraySubscriptExpr>(this)->getLHS()->getSourceRange(); | 
|  | 552 | R2 = cast<ArraySubscriptExpr>(this)->getRHS()->getSourceRange(); | 
|  | 553 | return true; | 
| Eli Friedman | 824f8c1 | 2008-05-27 15:24:04 +0000 | [diff] [blame] | 554 |  | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 555 | case CallExprClass: | 
| Eli Friedman | debdc1d | 2009-04-29 16:35:53 +0000 | [diff] [blame] | 556 | case CXXOperatorCallExprClass: | 
|  | 557 | case CXXMemberCallExprClass: { | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 558 | // If this is a direct call, get the callee. | 
|  | 559 | const CallExpr *CE = cast<CallExpr>(this); | 
|  | 560 | const Expr *CalleeExpr = CE->getCallee()->IgnoreParenCasts(); | 
|  | 561 | if (const DeclRefExpr *CalleeDRE = dyn_cast<DeclRefExpr>(CalleeExpr)) { | 
|  | 562 | // If the callee has attribute pure, const, or warn_unused_result, warn | 
|  | 563 | // about it. void foo() { strlen("bar"); } should warn. | 
|  | 564 | if (const FunctionDecl *FD = dyn_cast<FunctionDecl>(CalleeDRE->getDecl())) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 565 | if (FD->getAttr<WarnUnusedResultAttr>() || | 
|  | 566 | FD->getAttr<PureAttr>() || FD->getAttr<ConstAttr>()) { | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 567 | Loc = CE->getCallee()->getLocStart(); | 
|  | 568 | R1 = CE->getCallee()->getSourceRange(); | 
|  | 569 |  | 
|  | 570 | if (unsigned NumArgs = CE->getNumArgs()) | 
|  | 571 | R2 = SourceRange(CE->getArg(0)->getLocStart(), | 
|  | 572 | CE->getArg(NumArgs-1)->getLocEnd()); | 
|  | 573 | return true; | 
|  | 574 | } | 
|  | 575 | } | 
|  | 576 | return false; | 
|  | 577 | } | 
| Chris Lattner | e6d9ca5 | 2007-09-26 22:06:30 +0000 | [diff] [blame] | 578 | case ObjCMessageExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 579 | return false; | 
| Chris Lattner | 944d306 | 2008-07-26 19:51:01 +0000 | [diff] [blame] | 580 | case StmtExprClass: { | 
|  | 581 | // Statement exprs don't logically have side effects themselves, but are | 
|  | 582 | // sometimes used in macros in ways that give them a type that is unused. | 
|  | 583 | // For example ({ blah; foo(); }) will end up with a type if foo has a type. | 
|  | 584 | // however, if the result of the stmt expr is dead, we don't want to emit a | 
|  | 585 | // warning. | 
|  | 586 | const CompoundStmt *CS = cast<StmtExpr>(this)->getSubStmt(); | 
|  | 587 | if (!CS->body_empty()) | 
|  | 588 | if (const Expr *E = dyn_cast<Expr>(CS->body_back())) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 589 | return E->isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 590 |  | 
|  | 591 | Loc = cast<StmtExpr>(this)->getLParenLoc(); | 
|  | 592 | R1 = getSourceRange(); | 
|  | 593 | return true; | 
| Chris Lattner | 944d306 | 2008-07-26 19:51:01 +0000 | [diff] [blame] | 594 | } | 
| Douglas Gregor | f19b231 | 2008-10-28 15:36:24 +0000 | [diff] [blame] | 595 | case CStyleCastExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 596 | // If this is a cast to void, check the operand.  Otherwise, the result of | 
|  | 597 | // the cast is unused. | 
|  | 598 | if (getType()->isVoidType()) | 
| Douglas Gregor | 78bd61f | 2009-06-18 16:11:24 +0000 | [diff] [blame] | 599 | return cast<CastExpr>(this)->getSubExpr() | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 600 | ->isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 601 | Loc = cast<CStyleCastExpr>(this)->getLParenLoc(); | 
|  | 602 | R1 = cast<CStyleCastExpr>(this)->getSubExpr()->getSourceRange(); | 
|  | 603 | return true; | 
| Argyrios Kyrtzidis | 857fcc2 | 2008-08-22 15:38:55 +0000 | [diff] [blame] | 604 | case CXXFunctionalCastExprClass: | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 605 | // If this is a cast to void, check the operand.  Otherwise, the result of | 
|  | 606 | // the cast is unused. | 
|  | 607 | if (getType()->isVoidType()) | 
| Douglas Gregor | 78bd61f | 2009-06-18 16:11:24 +0000 | [diff] [blame] | 608 | return cast<CastExpr>(this)->getSubExpr() | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 609 | ->isUnusedResultAWarning(Loc, R1, R2); | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 610 | Loc = cast<CXXFunctionalCastExpr>(this)->getTypeBeginLoc(); | 
|  | 611 | R1 = cast<CXXFunctionalCastExpr>(this)->getSubExpr()->getSourceRange(); | 
|  | 612 | return true; | 
|  | 613 |  | 
| Eli Friedman | ca8da1d | 2008-05-19 21:24:43 +0000 | [diff] [blame] | 614 | case ImplicitCastExprClass: | 
|  | 615 | // Check the operand, since implicit casts are inserted by Sema | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 616 | return cast<ImplicitCastExpr>(this) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 617 | ->getSubExpr()->isUnusedResultAWarning(Loc, R1, R2); | 
| Eli Friedman | ca8da1d | 2008-05-19 21:24:43 +0000 | [diff] [blame] | 618 |  | 
| Chris Lattner | aa9c7ae | 2008-04-08 04:40:51 +0000 | [diff] [blame] | 619 | case CXXDefaultArgExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 620 | return cast<CXXDefaultArgExpr>(this) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 621 | ->getExpr()->isUnusedResultAWarning(Loc, R1, R2); | 
| Sebastian Redl | bd150f4 | 2008-11-21 19:14:01 +0000 | [diff] [blame] | 622 |  | 
|  | 623 | case CXXNewExprClass: | 
|  | 624 | // FIXME: In theory, there might be new expressions that don't have side | 
|  | 625 | // effects (e.g. a placement new with an uninitialized POD). | 
|  | 626 | case CXXDeleteExprClass: | 
| Chris Lattner | 237f275 | 2009-02-14 07:37:35 +0000 | [diff] [blame] | 627 | return false; | 
| Anders Carlsson | 24824e5 | 2009-05-17 21:11:30 +0000 | [diff] [blame] | 628 | case CXXExprWithTemporariesClass: | 
|  | 629 | return cast<CXXExprWithTemporaries>(this) | 
| Argyrios Kyrtzidis | b4b64ca | 2009-06-30 02:34:44 +0000 | [diff] [blame] | 630 | ->getSubExpr()->isUnusedResultAWarning(Loc, R1, R2); | 
| Sebastian Redl | bd150f4 | 2008-11-21 19:14:01 +0000 | [diff] [blame] | 631 | } | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 632 | } | 
|  | 633 |  | 
| Douglas Gregor | 4b62ec6 | 2008-10-22 15:04:37 +0000 | [diff] [blame] | 634 | /// DeclCanBeLvalue - Determine whether the given declaration can be | 
|  | 635 | /// an lvalue. This is a helper routine for isLvalue. | 
|  | 636 | static bool DeclCanBeLvalue(const NamedDecl *Decl, ASTContext &Ctx) { | 
| Douglas Gregor | 5101c24 | 2008-12-05 18:15:24 +0000 | [diff] [blame] | 637 | // C++ [temp.param]p6: | 
|  | 638 | //   A non-type non-reference template-parameter is not an lvalue. | 
|  | 639 | if (const NonTypeTemplateParmDecl *NTTParm | 
|  | 640 | = dyn_cast<NonTypeTemplateParmDecl>(Decl)) | 
|  | 641 | return NTTParm->getType()->isReferenceType(); | 
|  | 642 |  | 
| Douglas Gregor | 91f8421 | 2008-12-11 16:49:14 +0000 | [diff] [blame] | 643 | return isa<VarDecl>(Decl) || isa<FieldDecl>(Decl) || | 
| Douglas Gregor | 4b62ec6 | 2008-10-22 15:04:37 +0000 | [diff] [blame] | 644 | // C++ 3.10p2: An lvalue refers to an object or function. | 
|  | 645 | (Ctx.getLangOptions().CPlusPlus && | 
| Douglas Gregor | 9b14658 | 2009-07-08 20:55:45 +0000 | [diff] [blame] | 646 | (isa<FunctionDecl>(Decl) || isa<OverloadedFunctionDecl>(Decl) || | 
|  | 647 | isa<FunctionTemplateDecl>(Decl))); | 
| Douglas Gregor | 4b62ec6 | 2008-10-22 15:04:37 +0000 | [diff] [blame] | 648 | } | 
|  | 649 |  | 
| Steve Naroff | 475cca0 | 2007-05-14 17:19:29 +0000 | [diff] [blame] | 650 | /// isLvalue - C99 6.3.2.1: an lvalue is an expression with an object type or an | 
|  | 651 | /// incomplete type other than void. Nonarray expressions that can be lvalues: | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 652 | ///  - name, where name must be a variable | 
|  | 653 | ///  - e[i] | 
|  | 654 | ///  - (e), where e must be an lvalue | 
|  | 655 | ///  - e.name, where e must be an lvalue | 
|  | 656 | ///  - e->name | 
| Steve Naroff | 35d8515 | 2007-05-07 00:24:15 +0000 | [diff] [blame] | 657 | ///  - *e, the type of e cannot be a function type | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 658 | ///  - string-constant | 
| Chris Lattner | 595db86 | 2007-10-30 22:53:42 +0000 | [diff] [blame] | 659 | ///  - (__real__ e) and (__imag__ e) where e is an lvalue  [GNU extension] | 
| Bill Wendling | dfc8107 | 2007-07-17 03:52:31 +0000 | [diff] [blame] | 660 | ///  - reference type [C++ [expr]] | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 661 | /// | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 662 | Expr::isLvalueResult Expr::isLvalue(ASTContext &Ctx) const { | 
| Eli Friedman | b8c4fd8 | 2009-05-03 22:36:05 +0000 | [diff] [blame] | 663 | assert(!TR->isReferenceType() && "Expressions can't have reference type."); | 
|  | 664 |  | 
|  | 665 | isLvalueResult Res = isLvalueInternal(Ctx); | 
|  | 666 | if (Res != LV_Valid || Ctx.getLangOptions().CPlusPlus) | 
|  | 667 | return Res; | 
|  | 668 |  | 
| Douglas Gregor | 9a65793 | 2008-10-21 23:43:52 +0000 | [diff] [blame] | 669 | // first, check the type (C99 6.3.2.1). Expressions with function | 
|  | 670 | // type in C are not lvalues, but they can be lvalues in C++. | 
| Douglas Gregor | 9b14658 | 2009-07-08 20:55:45 +0000 | [diff] [blame] | 671 | if (TR->isFunctionType() || TR == Ctx.OverloadTy) | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 672 | return LV_NotObjectType; | 
| Steve Naroff | e728ba3 | 2007-07-10 22:20:04 +0000 | [diff] [blame] | 673 |  | 
| Steve Naroff | 1018ea3 | 2008-02-10 01:39:04 +0000 | [diff] [blame] | 674 | // Allow qualified void which is an incomplete type other than void (yuck). | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 675 | if (TR->isVoidType() && !Ctx.getCanonicalType(TR).getCVRQualifiers()) | 
| Steve Naroff | 1018ea3 | 2008-02-10 01:39:04 +0000 | [diff] [blame] | 676 | return LV_IncompleteVoidType; | 
|  | 677 |  | 
| Eli Friedman | b8c4fd8 | 2009-05-03 22:36:05 +0000 | [diff] [blame] | 678 | return LV_Valid; | 
|  | 679 | } | 
| Bill Wendling | dfc8107 | 2007-07-17 03:52:31 +0000 | [diff] [blame] | 680 |  | 
| Eli Friedman | b8c4fd8 | 2009-05-03 22:36:05 +0000 | [diff] [blame] | 681 | // Check whether the expression can be sanely treated like an l-value | 
|  | 682 | Expr::isLvalueResult Expr::isLvalueInternal(ASTContext &Ctx) const { | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 683 | switch (getStmtClass()) { | 
| Chris Lattner | d7e7b8e | 2009-02-24 22:18:39 +0000 | [diff] [blame] | 684 | case StringLiteralClass:  // C99 6.5.1p4 | 
|  | 685 | case ObjCEncodeExprClass: // @encode behaves like its string in every way. | 
| Anders Carlsson | 7a9a38a | 2007-11-30 22:47:59 +0000 | [diff] [blame] | 686 | return LV_Valid; | 
| Steve Naroff | 5dd642e | 2007-05-14 18:14:51 +0000 | [diff] [blame] | 687 | case ArraySubscriptExprClass: // C99 6.5.3p4 (e1[e2] == (*((e1)+(e2)))) | 
| Steve Naroff | e728ba3 | 2007-07-10 22:20:04 +0000 | [diff] [blame] | 688 | // For vectors, make sure base is an lvalue (i.e. not a function call). | 
|  | 689 | if (cast<ArraySubscriptExpr>(this)->getBase()->getType()->isVectorType()) | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 690 | return cast<ArraySubscriptExpr>(this)->getBase()->isLvalue(Ctx); | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 691 | return LV_Valid; | 
| Douglas Gregor | c7acfdf | 2009-01-06 05:10:23 +0000 | [diff] [blame] | 692 | case DeclRefExprClass: | 
|  | 693 | case QualifiedDeclRefExprClass: { // C99 6.5.1p2 | 
| Douglas Gregor | 4b62ec6 | 2008-10-22 15:04:37 +0000 | [diff] [blame] | 694 | const NamedDecl *RefdDecl = cast<DeclRefExpr>(this)->getDecl(); | 
|  | 695 | if (DeclCanBeLvalue(RefdDecl, Ctx)) | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 696 | return LV_Valid; | 
|  | 697 | break; | 
| Chris Lattner | 5696e7b | 2008-06-17 18:05:57 +0000 | [diff] [blame] | 698 | } | 
| Steve Naroff | 8de9c3a | 2008-09-05 22:11:13 +0000 | [diff] [blame] | 699 | case BlockDeclRefExprClass: { | 
|  | 700 | const BlockDeclRefExpr *BDR = cast<BlockDeclRefExpr>(this); | 
| Steve Naroff | ba756cb | 2008-09-26 14:41:28 +0000 | [diff] [blame] | 701 | if (isa<VarDecl>(BDR->getDecl())) | 
| Steve Naroff | 8de9c3a | 2008-09-05 22:11:13 +0000 | [diff] [blame] | 702 | return LV_Valid; | 
|  | 703 | break; | 
|  | 704 | } | 
| Douglas Gregor | 2eedc3a | 2008-12-20 23:49:58 +0000 | [diff] [blame] | 705 | case MemberExprClass: { | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 706 | const MemberExpr *m = cast<MemberExpr>(this); | 
| Douglas Gregor | 2eedc3a | 2008-12-20 23:49:58 +0000 | [diff] [blame] | 707 | if (Ctx.getLangOptions().CPlusPlus) { // C++ [expr.ref]p4: | 
|  | 708 | NamedDecl *Member = m->getMemberDecl(); | 
|  | 709 | // C++ [expr.ref]p4: | 
|  | 710 | //   If E2 is declared to have type "reference to T", then E1.E2 | 
|  | 711 | //   is an lvalue. | 
|  | 712 | if (ValueDecl *Value = dyn_cast<ValueDecl>(Member)) | 
|  | 713 | if (Value->getType()->isReferenceType()) | 
|  | 714 | return LV_Valid; | 
|  | 715 |  | 
|  | 716 | //   -- If E2 is a static data member [...] then E1.E2 is an lvalue. | 
| Douglas Gregor | 212cab3 | 2009-03-11 20:22:50 +0000 | [diff] [blame] | 717 | if (isa<VarDecl>(Member) && Member->getDeclContext()->isRecord()) | 
| Douglas Gregor | 2eedc3a | 2008-12-20 23:49:58 +0000 | [diff] [blame] | 718 | return LV_Valid; | 
|  | 719 |  | 
|  | 720 | //   -- If E2 is a non-static data member [...]. If E1 is an | 
|  | 721 | //      lvalue, then E1.E2 is an lvalue. | 
|  | 722 | if (isa<FieldDecl>(Member)) | 
|  | 723 | return m->isArrow() ? LV_Valid : m->getBase()->isLvalue(Ctx); | 
|  | 724 |  | 
|  | 725 | //   -- If it refers to a static member function [...], then | 
|  | 726 | //      E1.E2 is an lvalue. | 
|  | 727 | //   -- Otherwise, if E1.E2 refers to a non-static member | 
|  | 728 | //      function [...], then E1.E2 is not an lvalue. | 
|  | 729 | if (CXXMethodDecl *Method = dyn_cast<CXXMethodDecl>(Member)) | 
|  | 730 | return Method->isStatic()? LV_Valid : LV_MemberFunction; | 
|  | 731 |  | 
|  | 732 | //   -- If E2 is a member enumerator [...], the expression E1.E2 | 
|  | 733 | //      is not an lvalue. | 
|  | 734 | if (isa<EnumConstantDecl>(Member)) | 
|  | 735 | return LV_InvalidExpression; | 
|  | 736 |  | 
|  | 737 | // Not an lvalue. | 
|  | 738 | return LV_InvalidExpression; | 
|  | 739 | } | 
|  | 740 |  | 
|  | 741 | // C99 6.5.2.3p4 | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 742 | return m->isArrow() ? LV_Valid : m->getBase()->isLvalue(Ctx); | 
| Anton Korobeynikov | b76cda0 | 2007-07-12 15:26:50 +0000 | [diff] [blame] | 743 | } | 
| Chris Lattner | 595db86 | 2007-10-30 22:53:42 +0000 | [diff] [blame] | 744 | case UnaryOperatorClass: | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 745 | if (cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::Deref) | 
| Chris Lattner | 595db86 | 2007-10-30 22:53:42 +0000 | [diff] [blame] | 746 | return LV_Valid; // C99 6.5.3p4 | 
|  | 747 |  | 
|  | 748 | if (cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::Real || | 
| Chris Lattner | ec8996d | 2008-07-25 18:07:19 +0000 | [diff] [blame] | 749 | cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::Imag || | 
|  | 750 | cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::Extension) | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 751 | return cast<UnaryOperator>(this)->getSubExpr()->isLvalue(Ctx);  // GNU. | 
| Douglas Gregor | d08452f | 2008-11-19 15:42:04 +0000 | [diff] [blame] | 752 |  | 
|  | 753 | if (Ctx.getLangOptions().CPlusPlus && // C++ [expr.pre.incr]p1 | 
|  | 754 | (cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::PreInc || | 
|  | 755 | cast<UnaryOperator>(this)->getOpcode() == UnaryOperator::PreDec)) | 
|  | 756 | return LV_Valid; | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 757 | break; | 
| Douglas Gregor | a11693b | 2008-11-12 17:17:38 +0000 | [diff] [blame] | 758 | case ImplicitCastExprClass: | 
|  | 759 | return cast<ImplicitCastExpr>(this)->isLvalueCast()? LV_Valid | 
|  | 760 | : LV_InvalidExpression; | 
| Steve Naroff | 475cca0 | 2007-05-14 17:19:29 +0000 | [diff] [blame] | 761 | case ParenExprClass: // C99 6.5.1p5 | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 762 | return cast<ParenExpr>(this)->getSubExpr()->isLvalue(Ctx); | 
| Douglas Gregor | a11693b | 2008-11-12 17:17:38 +0000 | [diff] [blame] | 763 | case BinaryOperatorClass: | 
|  | 764 | case CompoundAssignOperatorClass: { | 
|  | 765 | const BinaryOperator *BinOp = cast<BinaryOperator>(this); | 
| Douglas Gregor | 40412ac | 2008-11-19 17:17:41 +0000 | [diff] [blame] | 766 |  | 
|  | 767 | if (Ctx.getLangOptions().CPlusPlus && // C++ [expr.comma]p1 | 
|  | 768 | BinOp->getOpcode() == BinaryOperator::Comma) | 
|  | 769 | return BinOp->getRHS()->isLvalue(Ctx); | 
|  | 770 |  | 
| Sebastian Redl | 112a9766 | 2009-02-07 00:15:38 +0000 | [diff] [blame] | 771 | // C++ [expr.mptr.oper]p6 | 
|  | 772 | if ((BinOp->getOpcode() == BinaryOperator::PtrMemD || | 
|  | 773 | BinOp->getOpcode() == BinaryOperator::PtrMemI) && | 
|  | 774 | !BinOp->getType()->isFunctionType()) | 
|  | 775 | return BinOp->getLHS()->isLvalue(Ctx); | 
|  | 776 |  | 
| Douglas Gregor | 58e008d | 2008-11-13 20:12:29 +0000 | [diff] [blame] | 777 | if (!BinOp->isAssignmentOp()) | 
| Douglas Gregor | a11693b | 2008-11-12 17:17:38 +0000 | [diff] [blame] | 778 | return LV_InvalidExpression; | 
|  | 779 |  | 
| Douglas Gregor | 58e008d | 2008-11-13 20:12:29 +0000 | [diff] [blame] | 780 | if (Ctx.getLangOptions().CPlusPlus) | 
|  | 781 | // C++ [expr.ass]p1: | 
|  | 782 | //   The result of an assignment operation [...] is an lvalue. | 
|  | 783 | return LV_Valid; | 
|  | 784 |  | 
|  | 785 |  | 
|  | 786 | // C99 6.5.16: | 
|  | 787 | //   An assignment expression [...] is not an lvalue. | 
|  | 788 | return LV_InvalidExpression; | 
| Douglas Gregor | a11693b | 2008-11-12 17:17:38 +0000 | [diff] [blame] | 789 | } | 
| Douglas Gregor | 993603d | 2008-11-14 16:09:21 +0000 | [diff] [blame] | 790 | case CallExprClass: | 
| Douglas Gregor | 97fd6e2 | 2008-12-22 05:46:06 +0000 | [diff] [blame] | 791 | case CXXOperatorCallExprClass: | 
|  | 792 | case CXXMemberCallExprClass: { | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 793 | // C++0x [expr.call]p10 | 
| Douglas Gregor | 6b75484 | 2008-10-28 00:22:11 +0000 | [diff] [blame] | 794 | //   A function call is an lvalue if and only if the result type | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 795 | //   is an lvalue reference. | 
| Anders Carlsson | 00a2759 | 2009-05-26 04:57:27 +0000 | [diff] [blame] | 796 | QualType ReturnType = cast<CallExpr>(this)->getCallReturnType(); | 
|  | 797 | if (ReturnType->isLValueReferenceType()) | 
|  | 798 | return LV_Valid; | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 799 |  | 
| Douglas Gregor | 6b75484 | 2008-10-28 00:22:11 +0000 | [diff] [blame] | 800 | break; | 
|  | 801 | } | 
| Steve Naroff | 2644aaf | 2007-12-05 04:00:10 +0000 | [diff] [blame] | 802 | case CompoundLiteralExprClass: // C99 6.5.2.5p5 | 
|  | 803 | return LV_Valid; | 
| Chris Lattner | 053441f | 2008-12-12 05:35:08 +0000 | [diff] [blame] | 804 | case ChooseExprClass: | 
|  | 805 | // __builtin_choose_expr is an lvalue if the selected operand is. | 
| Eli Friedman | e0a5b8b | 2009-03-04 05:52:32 +0000 | [diff] [blame] | 806 | return cast<ChooseExpr>(this)->getChosenSubExpr(Ctx)->isLvalue(Ctx); | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 807 | case ExtVectorElementExprClass: | 
|  | 808 | if (cast<ExtVectorElementExpr>(this)->containsDuplicateElements()) | 
| Steve Naroff | 0d595ca | 2007-07-30 03:29:09 +0000 | [diff] [blame] | 809 | return LV_DuplicateVectorComponents; | 
|  | 810 | return LV_Valid; | 
| Steve Naroff | b342361 | 2007-11-12 14:34:27 +0000 | [diff] [blame] | 811 | case ObjCIvarRefExprClass: // ObjC instance variables are lvalues. | 
|  | 812 | return LV_Valid; | 
| Steve Naroff | 6600228 | 2008-05-30 23:23:16 +0000 | [diff] [blame] | 813 | case ObjCPropertyRefExprClass: // FIXME: check if read-only property. | 
|  | 814 | return LV_Valid; | 
| Fariborz Jahanian | 8a1810f | 2008-11-22 18:39:36 +0000 | [diff] [blame] | 815 | case ObjCKVCRefExprClass: // FIXME: check if read-only property. | 
| Chris Lattner | 053441f | 2008-12-12 05:35:08 +0000 | [diff] [blame] | 816 | return LV_Valid; | 
| Chris Lattner | 6307f19 | 2008-08-10 01:53:14 +0000 | [diff] [blame] | 817 | case PredefinedExprClass: | 
| Douglas Gregor | 97a9c81 | 2008-11-04 14:32:21 +0000 | [diff] [blame] | 818 | return LV_Valid; | 
| Chris Lattner | aa9c7ae | 2008-04-08 04:40:51 +0000 | [diff] [blame] | 819 | case CXXDefaultArgExprClass: | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 820 | return cast<CXXDefaultArgExpr>(this)->getExpr()->isLvalue(Ctx); | 
| Argyrios Kyrtzidis | 0fdbd6c | 2008-09-11 04:22:26 +0000 | [diff] [blame] | 821 | case CXXConditionDeclExprClass: | 
|  | 822 | return LV_Valid; | 
| Douglas Gregor | f19b231 | 2008-10-28 15:36:24 +0000 | [diff] [blame] | 823 | case CStyleCastExprClass: | 
| Douglas Gregor | 6b75484 | 2008-10-28 00:22:11 +0000 | [diff] [blame] | 824 | case CXXFunctionalCastExprClass: | 
|  | 825 | case CXXStaticCastExprClass: | 
|  | 826 | case CXXDynamicCastExprClass: | 
|  | 827 | case CXXReinterpretCastExprClass: | 
|  | 828 | case CXXConstCastExprClass: | 
|  | 829 | // The result of an explicit cast is an lvalue if the type we are | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 830 | // casting to is an lvalue reference type. See C++ [expr.cast]p1, | 
| Douglas Gregor | 6b75484 | 2008-10-28 00:22:11 +0000 | [diff] [blame] | 831 | // C++ [expr.static.cast]p2, C++ [expr.dynamic.cast]p2, | 
|  | 832 | // C++ [expr.reinterpret.cast]p1, C++ [expr.const.cast]p1. | 
| Sebastian Redl | 0f8b23f | 2009-03-16 23:22:08 +0000 | [diff] [blame] | 833 | if (cast<ExplicitCastExpr>(this)->getTypeAsWritten()-> | 
|  | 834 | isLValueReferenceType()) | 
| Douglas Gregor | 6b75484 | 2008-10-28 00:22:11 +0000 | [diff] [blame] | 835 | return LV_Valid; | 
|  | 836 | break; | 
| Sebastian Redl | c470476 | 2008-11-11 11:37:55 +0000 | [diff] [blame] | 837 | case CXXTypeidExprClass: | 
|  | 838 | // C++ 5.2.8p1: The result of a typeid expression is an lvalue of ... | 
|  | 839 | return LV_Valid; | 
| Sebastian Redl | 5775af1a | 2009-04-17 16:30:52 +0000 | [diff] [blame] | 840 | case ConditionalOperatorClass: { | 
|  | 841 | // Complicated handling is only for C++. | 
|  | 842 | if (!Ctx.getLangOptions().CPlusPlus) | 
|  | 843 | return LV_InvalidExpression; | 
|  | 844 |  | 
|  | 845 | // Sema should have taken care to ensure that a CXXTemporaryObjectExpr is | 
|  | 846 | // everywhere there's an object converted to an rvalue. Also, any other | 
|  | 847 | // casts should be wrapped by ImplicitCastExprs. There's just the special | 
|  | 848 | // case involving throws to work out. | 
|  | 849 | const ConditionalOperator *Cond = cast<ConditionalOperator>(this); | 
| Douglas Gregor | 115652d | 2009-05-19 20:13:50 +0000 | [diff] [blame] | 850 | Expr *True = Cond->getTrueExpr(); | 
|  | 851 | Expr *False = Cond->getFalseExpr(); | 
| Sebastian Redl | 5775af1a | 2009-04-17 16:30:52 +0000 | [diff] [blame] | 852 | // C++0x 5.16p2 | 
|  | 853 | //   If either the second or the third operand has type (cv) void, [...] | 
|  | 854 | //   the result [...] is an rvalue. | 
| Douglas Gregor | 115652d | 2009-05-19 20:13:50 +0000 | [diff] [blame] | 855 | if (True->getType()->isVoidType() || False->getType()->isVoidType()) | 
| Sebastian Redl | 5775af1a | 2009-04-17 16:30:52 +0000 | [diff] [blame] | 856 | return LV_InvalidExpression; | 
|  | 857 |  | 
|  | 858 | // Both sides must be lvalues for the result to be an lvalue. | 
| Douglas Gregor | 115652d | 2009-05-19 20:13:50 +0000 | [diff] [blame] | 859 | if (True->isLvalue(Ctx) != LV_Valid || False->isLvalue(Ctx) != LV_Valid) | 
| Sebastian Redl | 5775af1a | 2009-04-17 16:30:52 +0000 | [diff] [blame] | 860 | return LV_InvalidExpression; | 
|  | 861 |  | 
|  | 862 | // That's it. | 
|  | 863 | return LV_Valid; | 
|  | 864 | } | 
|  | 865 |  | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 866 | default: | 
|  | 867 | break; | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 868 | } | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 869 | return LV_InvalidExpression; | 
| Steve Naroff | 4750051 | 2007-04-19 23:00:49 +0000 | [diff] [blame] | 870 | } | 
| Steve Naroff | 218bc2b | 2007-05-04 21:54:46 +0000 | [diff] [blame] | 871 |  | 
| Steve Naroff | 475cca0 | 2007-05-14 17:19:29 +0000 | [diff] [blame] | 872 | /// isModifiableLvalue - C99 6.3.2.1: an lvalue that does not have array type, | 
|  | 873 | /// does not have an incomplete type, does not have a const-qualified type, and | 
|  | 874 | /// if it is a structure or union, does not have any member (including, | 
|  | 875 | /// recursively, any member or element of all contained aggregates or unions) | 
|  | 876 | /// with a const-qualified type. | 
| Daniel Dunbar | c2223ab | 2009-04-15 00:08:05 +0000 | [diff] [blame] | 877 | Expr::isModifiableLvalueResult | 
|  | 878 | Expr::isModifiableLvalue(ASTContext &Ctx, SourceLocation *Loc) const { | 
| Chris Lattner | 6731544 | 2008-07-26 21:30:36 +0000 | [diff] [blame] | 879 | isLvalueResult lvalResult = isLvalue(Ctx); | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 880 |  | 
|  | 881 | switch (lvalResult) { | 
| Douglas Gregor | 293a3c6 | 2008-10-22 00:03:08 +0000 | [diff] [blame] | 882 | case LV_Valid: | 
|  | 883 | // C++ 3.10p11: Functions cannot be modified, but pointers to | 
|  | 884 | // functions can be modifiable. | 
|  | 885 | if (Ctx.getLangOptions().CPlusPlus && TR->isFunctionType()) | 
|  | 886 | return MLV_NotObjectType; | 
|  | 887 | break; | 
|  | 888 |  | 
| Chris Lattner | 1ec5f56 | 2007-06-27 05:38:08 +0000 | [diff] [blame] | 889 | case LV_NotObjectType: return MLV_NotObjectType; | 
|  | 890 | case LV_IncompleteVoidType: return MLV_IncompleteVoidType; | 
| Steve Naroff | 0d595ca | 2007-07-30 03:29:09 +0000 | [diff] [blame] | 891 | case LV_DuplicateVectorComponents: return MLV_DuplicateVectorComponents; | 
| Chris Lattner | 9b3bbe9 | 2008-11-17 19:51:54 +0000 | [diff] [blame] | 892 | case LV_InvalidExpression: | 
|  | 893 | // If the top level is a C-style cast, and the subexpression is a valid | 
|  | 894 | // lvalue, then this is probably a use of the old-school "cast as lvalue" | 
|  | 895 | // GCC extension.  We don't support it, but we want to produce good | 
|  | 896 | // diagnostics when it happens so that the user knows why. | 
| Daniel Dunbar | c2223ab | 2009-04-15 00:08:05 +0000 | [diff] [blame] | 897 | if (const CStyleCastExpr *CE = dyn_cast<CStyleCastExpr>(IgnoreParens())) { | 
|  | 898 | if (CE->getSubExpr()->isLvalue(Ctx) == LV_Valid) { | 
|  | 899 | if (Loc) | 
|  | 900 | *Loc = CE->getLParenLoc(); | 
| Chris Lattner | 9b3bbe9 | 2008-11-17 19:51:54 +0000 | [diff] [blame] | 901 | return MLV_LValueCast; | 
| Daniel Dunbar | c2223ab | 2009-04-15 00:08:05 +0000 | [diff] [blame] | 902 | } | 
|  | 903 | } | 
| Chris Lattner | 9b3bbe9 | 2008-11-17 19:51:54 +0000 | [diff] [blame] | 904 | return MLV_InvalidExpression; | 
| Douglas Gregor | 2eedc3a | 2008-12-20 23:49:58 +0000 | [diff] [blame] | 905 | case LV_MemberFunction: return MLV_MemberFunction; | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 906 | } | 
| Eli Friedman | e8dd7b3 | 2009-03-22 23:26:56 +0000 | [diff] [blame] | 907 |  | 
|  | 908 | // The following is illegal: | 
|  | 909 | //   void takeclosure(void (^C)(void)); | 
|  | 910 | //   void func() { int x = 1; takeclosure(^{ x = 7; }); } | 
|  | 911 | // | 
| Chris Lattner | 656711a | 2009-03-23 17:57:53 +0000 | [diff] [blame] | 912 | if (isa<BlockDeclRefExpr>(this)) { | 
| Eli Friedman | e8dd7b3 | 2009-03-22 23:26:56 +0000 | [diff] [blame] | 913 | const BlockDeclRefExpr *BDR = cast<BlockDeclRefExpr>(this); | 
|  | 914 | if (!BDR->isByRef() && isa<VarDecl>(BDR->getDecl())) | 
|  | 915 | return MLV_NotBlockQualified; | 
|  | 916 | } | 
|  | 917 |  | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 918 | QualType CT = Ctx.getCanonicalType(getType()); | 
|  | 919 |  | 
|  | 920 | if (CT.isConstQualified()) | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 921 | return MLV_ConstQualified; | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 922 | if (CT->isArrayType()) | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 923 | return MLV_ArrayType; | 
| Chris Lattner | 7adf076 | 2008-08-04 07:31:14 +0000 | [diff] [blame] | 924 | if (CT->isIncompleteType()) | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 925 | return MLV_IncompleteType; | 
|  | 926 |  | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 927 | if (const RecordType *r = CT->getAsRecordType()) { | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 928 | if (r->hasConstFields()) | 
|  | 929 | return MLV_ConstQualified; | 
|  | 930 | } | 
| Fariborz Jahanian | 8e1555c | 2009-01-12 19:55:42 +0000 | [diff] [blame] | 931 |  | 
| Fariborz Jahanian | 5118c41 | 2008-11-22 20:25:50 +0000 | [diff] [blame] | 932 | // Assigning to an 'implicit' property? | 
| Chris Lattner | 656711a | 2009-03-23 17:57:53 +0000 | [diff] [blame] | 933 | else if (isa<ObjCKVCRefExpr>(this)) { | 
| Fariborz Jahanian | 5118c41 | 2008-11-22 20:25:50 +0000 | [diff] [blame] | 934 | const ObjCKVCRefExpr* KVCExpr = cast<ObjCKVCRefExpr>(this); | 
|  | 935 | if (KVCExpr->getSetterMethod() == 0) | 
|  | 936 | return MLV_NoSetterProperty; | 
|  | 937 | } | 
| Steve Naroff | 9358c71 | 2007-05-27 23:58:33 +0000 | [diff] [blame] | 938 | return MLV_Valid; | 
| Steve Naroff | 475cca0 | 2007-05-14 17:19:29 +0000 | [diff] [blame] | 939 | } | 
|  | 940 |  | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 941 | /// hasGlobalStorage - Return true if this expression has static storage | 
| Chris Lattner | da22eec | 2007-11-27 21:35:27 +0000 | [diff] [blame] | 942 | /// duration.  This means that the address of this expression is a link-time | 
|  | 943 | /// constant. | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 944 | bool Expr::hasGlobalStorage() const { | 
| Chris Lattner | 2f72c42 | 2007-11-13 18:05:45 +0000 | [diff] [blame] | 945 | switch (getStmtClass()) { | 
|  | 946 | default: | 
|  | 947 | return false; | 
| Steve Naroff | a0c3270 | 2009-04-16 19:02:57 +0000 | [diff] [blame] | 948 | case BlockExprClass: | 
|  | 949 | return true; | 
| Chris Lattner | da22eec | 2007-11-27 21:35:27 +0000 | [diff] [blame] | 950 | case ParenExprClass: | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 951 | return cast<ParenExpr>(this)->getSubExpr()->hasGlobalStorage(); | 
| Chris Lattner | da22eec | 2007-11-27 21:35:27 +0000 | [diff] [blame] | 952 | case ImplicitCastExprClass: | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 953 | return cast<ImplicitCastExpr>(this)->getSubExpr()->hasGlobalStorage(); | 
| Steve Naroff | d32419d | 2008-01-14 18:19:28 +0000 | [diff] [blame] | 954 | case CompoundLiteralExprClass: | 
|  | 955 | return cast<CompoundLiteralExpr>(this)->isFileScope(); | 
| Douglas Gregor | c7acfdf | 2009-01-06 05:10:23 +0000 | [diff] [blame] | 956 | case DeclRefExprClass: | 
|  | 957 | case QualifiedDeclRefExprClass: { | 
| Chris Lattner | 2f72c42 | 2007-11-13 18:05:45 +0000 | [diff] [blame] | 958 | const Decl *D = cast<DeclRefExpr>(this)->getDecl(); | 
|  | 959 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 960 | return VD->hasGlobalStorage(); | 
| Seo Sanghyeon | 055c94f | 2008-04-04 09:45:30 +0000 | [diff] [blame] | 961 | if (isa<FunctionDecl>(D)) | 
|  | 962 | return true; | 
| Chris Lattner | 2f72c42 | 2007-11-13 18:05:45 +0000 | [diff] [blame] | 963 | return false; | 
|  | 964 | } | 
| Chris Lattner | d0d8d4e | 2007-11-28 04:30:09 +0000 | [diff] [blame] | 965 | case MemberExprClass: { | 
| Chris Lattner | 2f72c42 | 2007-11-13 18:05:45 +0000 | [diff] [blame] | 966 | const MemberExpr *M = cast<MemberExpr>(this); | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 967 | return !M->isArrow() && M->getBase()->hasGlobalStorage(); | 
| Chris Lattner | d0d8d4e | 2007-11-28 04:30:09 +0000 | [diff] [blame] | 968 | } | 
| Chris Lattner | da22eec | 2007-11-27 21:35:27 +0000 | [diff] [blame] | 969 | case ArraySubscriptExprClass: | 
| Ted Kremenek | 41dd9d0 | 2008-02-27 18:39:48 +0000 | [diff] [blame] | 970 | return cast<ArraySubscriptExpr>(this)->getBase()->hasGlobalStorage(); | 
| Chris Lattner | 6307f19 | 2008-08-10 01:53:14 +0000 | [diff] [blame] | 971 | case PredefinedExprClass: | 
| Chris Lattner | a81a027 | 2008-01-12 08:14:25 +0000 | [diff] [blame] | 972 | return true; | 
| Chris Lattner | aa9c7ae | 2008-04-08 04:40:51 +0000 | [diff] [blame] | 973 | case CXXDefaultArgExprClass: | 
|  | 974 | return cast<CXXDefaultArgExpr>(this)->getExpr()->hasGlobalStorage(); | 
| Chris Lattner | 2f72c42 | 2007-11-13 18:05:45 +0000 | [diff] [blame] | 975 | } | 
|  | 976 | } | 
|  | 977 |  | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 978 | /// isOBJCGCCandidate - Check if an expression is objc gc'able. | 
|  | 979 | /// | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 980 | bool Expr::isOBJCGCCandidate(ASTContext &Ctx) const { | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 981 | switch (getStmtClass()) { | 
|  | 982 | default: | 
|  | 983 | return false; | 
|  | 984 | case ObjCIvarRefExprClass: | 
|  | 985 | return true; | 
| Fariborz Jahanian | 392124c | 2009-02-23 18:59:50 +0000 | [diff] [blame] | 986 | case Expr::UnaryOperatorClass: | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 987 | return cast<UnaryOperator>(this)->getSubExpr()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 988 | case ParenExprClass: | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 989 | return cast<ParenExpr>(this)->getSubExpr()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 990 | case ImplicitCastExprClass: | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 991 | return cast<ImplicitCastExpr>(this)->getSubExpr()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | a16904b | 2009-05-05 23:28:21 +0000 | [diff] [blame] | 992 | case CStyleCastExprClass: | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 993 | return cast<CStyleCastExpr>(this)->getSubExpr()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 994 | case DeclRefExprClass: | 
|  | 995 | case QualifiedDeclRefExprClass: { | 
|  | 996 | const Decl *D = cast<DeclRefExpr>(this)->getDecl(); | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 997 | if (const VarDecl *VD = dyn_cast<VarDecl>(D)) { | 
|  | 998 | if (VD->hasGlobalStorage()) | 
|  | 999 | return true; | 
|  | 1000 | QualType T = VD->getType(); | 
|  | 1001 | // dereferencing to an object pointer is always a gc'able candidate | 
|  | 1002 | if (T->isPointerType() && | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 1003 | T->getAsPointerType()->getPointeeType()->isObjCObjectPointerType()) | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 1004 | return true; | 
|  | 1005 |  | 
|  | 1006 | } | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 1007 | return false; | 
|  | 1008 | } | 
|  | 1009 | case MemberExprClass: { | 
|  | 1010 | const MemberExpr *M = cast<MemberExpr>(this); | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 1011 | return M->getBase()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 1012 | } | 
|  | 1013 | case ArraySubscriptExprClass: | 
| Fariborz Jahanian | c6d9800 | 2009-06-01 21:29:32 +0000 | [diff] [blame] | 1014 | return cast<ArraySubscriptExpr>(this)->getBase()->isOBJCGCCandidate(Ctx); | 
| Fariborz Jahanian | 0773533 | 2009-02-22 18:40:18 +0000 | [diff] [blame] | 1015 | } | 
|  | 1016 | } | 
| Ted Kremenek | fff7096 | 2008-01-17 16:57:34 +0000 | [diff] [blame] | 1017 | Expr* Expr::IgnoreParens() { | 
|  | 1018 | Expr* E = this; | 
|  | 1019 | while (ParenExpr* P = dyn_cast<ParenExpr>(E)) | 
|  | 1020 | E = P->getSubExpr(); | 
|  | 1021 |  | 
|  | 1022 | return E; | 
|  | 1023 | } | 
|  | 1024 |  | 
| Chris Lattner | f266096 | 2008-02-13 01:02:39 +0000 | [diff] [blame] | 1025 | /// IgnoreParenCasts - Ignore parentheses and casts.  Strip off any ParenExpr | 
|  | 1026 | /// or CastExprs or ImplicitCastExprs, returning their operand. | 
|  | 1027 | Expr *Expr::IgnoreParenCasts() { | 
|  | 1028 | Expr *E = this; | 
|  | 1029 | while (true) { | 
|  | 1030 | if (ParenExpr *P = dyn_cast<ParenExpr>(E)) | 
|  | 1031 | E = P->getSubExpr(); | 
|  | 1032 | else if (CastExpr *P = dyn_cast<CastExpr>(E)) | 
|  | 1033 | E = P->getSubExpr(); | 
| Chris Lattner | f266096 | 2008-02-13 01:02:39 +0000 | [diff] [blame] | 1034 | else | 
|  | 1035 | return E; | 
|  | 1036 | } | 
|  | 1037 | } | 
|  | 1038 |  | 
| Chris Lattner | ef26c77 | 2009-03-13 17:28:01 +0000 | [diff] [blame] | 1039 | /// IgnoreParenNoopCasts - Ignore parentheses and casts that do not change the | 
|  | 1040 | /// value (including ptr->int casts of the same size).  Strip off any | 
|  | 1041 | /// ParenExpr or CastExprs, returning their operand. | 
|  | 1042 | Expr *Expr::IgnoreParenNoopCasts(ASTContext &Ctx) { | 
|  | 1043 | Expr *E = this; | 
|  | 1044 | while (true) { | 
|  | 1045 | if (ParenExpr *P = dyn_cast<ParenExpr>(E)) { | 
|  | 1046 | E = P->getSubExpr(); | 
|  | 1047 | continue; | 
|  | 1048 | } | 
|  | 1049 |  | 
|  | 1050 | if (CastExpr *P = dyn_cast<CastExpr>(E)) { | 
|  | 1051 | // We ignore integer <-> casts that are of the same width, ptr<->ptr and | 
|  | 1052 | // ptr<->int casts of the same width.  We also ignore all identify casts. | 
|  | 1053 | Expr *SE = P->getSubExpr(); | 
|  | 1054 |  | 
|  | 1055 | if (Ctx.hasSameUnqualifiedType(E->getType(), SE->getType())) { | 
|  | 1056 | E = SE; | 
|  | 1057 | continue; | 
|  | 1058 | } | 
|  | 1059 |  | 
|  | 1060 | if ((E->getType()->isPointerType() || E->getType()->isIntegralType()) && | 
|  | 1061 | (SE->getType()->isPointerType() || SE->getType()->isIntegralType()) && | 
|  | 1062 | Ctx.getTypeSize(E->getType()) == Ctx.getTypeSize(SE->getType())) { | 
|  | 1063 | E = SE; | 
|  | 1064 | continue; | 
|  | 1065 | } | 
|  | 1066 | } | 
|  | 1067 |  | 
|  | 1068 | return E; | 
|  | 1069 | } | 
|  | 1070 | } | 
|  | 1071 |  | 
|  | 1072 |  | 
| Douglas Gregor | 4619e43 | 2008-12-05 23:32:09 +0000 | [diff] [blame] | 1073 | /// hasAnyTypeDependentArguments - Determines if any of the expressions | 
|  | 1074 | /// in Exprs is type-dependent. | 
|  | 1075 | bool Expr::hasAnyTypeDependentArguments(Expr** Exprs, unsigned NumExprs) { | 
|  | 1076 | for (unsigned I = 0; I < NumExprs; ++I) | 
|  | 1077 | if (Exprs[I]->isTypeDependent()) | 
|  | 1078 | return true; | 
|  | 1079 |  | 
|  | 1080 | return false; | 
|  | 1081 | } | 
|  | 1082 |  | 
|  | 1083 | /// hasAnyValueDependentArguments - Determines if any of the expressions | 
|  | 1084 | /// in Exprs is value-dependent. | 
|  | 1085 | bool Expr::hasAnyValueDependentArguments(Expr** Exprs, unsigned NumExprs) { | 
|  | 1086 | for (unsigned I = 0; I < NumExprs; ++I) | 
|  | 1087 | if (Exprs[I]->isValueDependent()) | 
|  | 1088 | return true; | 
|  | 1089 |  | 
|  | 1090 | return false; | 
|  | 1091 | } | 
|  | 1092 |  | 
| Eli Friedman | 7139af4 | 2009-01-25 02:32:41 +0000 | [diff] [blame] | 1093 | bool Expr::isConstantInitializer(ASTContext &Ctx) const { | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1094 | // This function is attempting whether an expression is an initializer | 
|  | 1095 | // which can be evaluated at compile-time.  isEvaluatable handles most | 
|  | 1096 | // of the cases, but it can't deal with some initializer-specific | 
|  | 1097 | // expressions, and it can't deal with aggregates; we deal with those here, | 
|  | 1098 | // and fall back to isEvaluatable for the other cases. | 
|  | 1099 |  | 
| Eli Friedman | cf7cbe7 | 2009-02-20 02:36:22 +0000 | [diff] [blame] | 1100 | // FIXME: This function assumes the variable being assigned to | 
|  | 1101 | // isn't a reference type! | 
|  | 1102 |  | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1103 | switch (getStmtClass()) { | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1104 | default: break; | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1105 | case StringLiteralClass: | 
| Steve Naroff | 7cae42b | 2009-07-10 23:34:53 +0000 | [diff] [blame] | 1106 | case ObjCStringLiteralClass: | 
| Chris Lattner | d7e7b8e | 2009-02-24 22:18:39 +0000 | [diff] [blame] | 1107 | case ObjCEncodeExprClass: | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1108 | return true; | 
| Nate Begeman | 2f2bdeb | 2009-01-18 03:20:47 +0000 | [diff] [blame] | 1109 | case CompoundLiteralExprClass: { | 
| Eli Friedman | cf7cbe7 | 2009-02-20 02:36:22 +0000 | [diff] [blame] | 1110 | // This handles gcc's extension that allows global initializers like | 
|  | 1111 | // "struct x {int x;} x = (struct x) {};". | 
|  | 1112 | // FIXME: This accepts other cases it shouldn't! | 
| Nate Begeman | 2f2bdeb | 2009-01-18 03:20:47 +0000 | [diff] [blame] | 1113 | const Expr *Exp = cast<CompoundLiteralExpr>(this)->getInitializer(); | 
| Eli Friedman | 7139af4 | 2009-01-25 02:32:41 +0000 | [diff] [blame] | 1114 | return Exp->isConstantInitializer(Ctx); | 
| Nate Begeman | 2f2bdeb | 2009-01-18 03:20:47 +0000 | [diff] [blame] | 1115 | } | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1116 | case InitListExprClass: { | 
| Eli Friedman | cf7cbe7 | 2009-02-20 02:36:22 +0000 | [diff] [blame] | 1117 | // FIXME: This doesn't deal with fields with reference types correctly. | 
|  | 1118 | // FIXME: This incorrectly allows pointers cast to integers to be assigned | 
|  | 1119 | // to bitfields. | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1120 | const InitListExpr *Exp = cast<InitListExpr>(this); | 
|  | 1121 | unsigned numInits = Exp->getNumInits(); | 
|  | 1122 | for (unsigned i = 0; i < numInits; i++) { | 
| Eli Friedman | 7139af4 | 2009-01-25 02:32:41 +0000 | [diff] [blame] | 1123 | if (!Exp->getInit(i)->isConstantInitializer(Ctx)) | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1124 | return false; | 
|  | 1125 | } | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1126 | return true; | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1127 | } | 
| Douglas Gregor | 0202cb4 | 2009-01-29 17:44:32 +0000 | [diff] [blame] | 1128 | case ImplicitValueInitExprClass: | 
|  | 1129 | return true; | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1130 | case ParenExprClass: { | 
|  | 1131 | return cast<ParenExpr>(this)->getSubExpr()->isConstantInitializer(Ctx); | 
|  | 1132 | } | 
|  | 1133 | case UnaryOperatorClass: { | 
|  | 1134 | const UnaryOperator* Exp = cast<UnaryOperator>(this); | 
|  | 1135 | if (Exp->getOpcode() == UnaryOperator::Extension) | 
|  | 1136 | return Exp->getSubExpr()->isConstantInitializer(Ctx); | 
|  | 1137 | break; | 
|  | 1138 | } | 
| Chris Lattner | 1f02e05 | 2009-04-21 05:19:11 +0000 | [diff] [blame] | 1139 | case ImplicitCastExprClass: | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1140 | case CStyleCastExprClass: | 
|  | 1141 | // Handle casts with a destination that's a struct or union; this | 
|  | 1142 | // deals with both the gcc no-op struct cast extension and the | 
|  | 1143 | // cast-to-union extension. | 
|  | 1144 | if (getType()->isRecordType()) | 
|  | 1145 | return cast<CastExpr>(this)->getSubExpr()->isConstantInitializer(Ctx); | 
|  | 1146 | break; | 
| Anders Carlsson | a7c5eb7 | 2008-11-24 05:23:59 +0000 | [diff] [blame] | 1147 | } | 
| Eli Friedman | 384da27 | 2009-01-25 03:12:18 +0000 | [diff] [blame] | 1148 | return isEvaluatable(Ctx); | 
| Steve Naroff | b03f594 | 2007-09-02 20:30:18 +0000 | [diff] [blame] | 1149 | } | 
|  | 1150 |  | 
| Chris Lattner | 1f4479e | 2007-06-05 04:15:44 +0000 | [diff] [blame] | 1151 | /// isIntegerConstantExpr - this recursive routine will test if an expression is | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1152 | /// an integer constant expression. | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1153 |  | 
|  | 1154 | /// FIXME: Pass up a reason why! Invalid operation in i-c-e, division by zero, | 
|  | 1155 | /// comma, etc | 
| Chris Lattner | 4ef4001 | 2007-06-11 01:28:17 +0000 | [diff] [blame] | 1156 | /// | 
| Chris Lattner | d7372ba | 2007-07-18 05:21:20 +0000 | [diff] [blame] | 1157 | /// FIXME: Handle offsetof.  Two things to do:  Handle GCC's __builtin_offsetof | 
|  | 1158 | /// to support gcc 4.0+  and handle the idiom GCC recognizes with a null pointer | 
|  | 1159 | /// cast+dereference. | 
| Daniel Dunbar | 4750e63 | 2009-02-18 00:47:45 +0000 | [diff] [blame] | 1160 |  | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1161 | // CheckICE - This function does the fundamental ICE checking: the returned | 
|  | 1162 | // ICEDiag contains a Val of 0, 1, or 2, and a possibly null SourceLocation. | 
|  | 1163 | // Note that to reduce code duplication, this helper does no evaluation | 
|  | 1164 | // itself; the caller checks whether the expression is evaluatable, and | 
|  | 1165 | // in the rare cases where CheckICE actually cares about the evaluated | 
|  | 1166 | // value, it calls into Evalute. | 
|  | 1167 | // | 
|  | 1168 | // Meanings of Val: | 
|  | 1169 | // 0: This expression is an ICE if it can be evaluated by Evaluate. | 
|  | 1170 | // 1: This expression is not an ICE, but if it isn't evaluated, it's | 
|  | 1171 | //    a legal subexpression for an ICE. This return value is used to handle | 
|  | 1172 | //    the comma operator in C99 mode. | 
|  | 1173 | // 2: This expression is not an ICE, and is not a legal subexpression for one. | 
|  | 1174 |  | 
|  | 1175 | struct ICEDiag { | 
|  | 1176 | unsigned Val; | 
|  | 1177 | SourceLocation Loc; | 
|  | 1178 |  | 
|  | 1179 | public: | 
|  | 1180 | ICEDiag(unsigned v, SourceLocation l) : Val(v), Loc(l) {} | 
|  | 1181 | ICEDiag() : Val(0) {} | 
|  | 1182 | }; | 
|  | 1183 |  | 
|  | 1184 | ICEDiag NoDiag() { return ICEDiag(); } | 
|  | 1185 |  | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1186 | static ICEDiag CheckEvalInICE(const Expr* E, ASTContext &Ctx) { | 
|  | 1187 | Expr::EvalResult EVResult; | 
|  | 1188 | if (!E->Evaluate(EVResult, Ctx) || EVResult.HasSideEffects || | 
|  | 1189 | !EVResult.Val.isInt()) { | 
|  | 1190 | return ICEDiag(2, E->getLocStart()); | 
|  | 1191 | } | 
|  | 1192 | return NoDiag(); | 
|  | 1193 | } | 
|  | 1194 |  | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1195 | static ICEDiag CheckICE(const Expr* E, ASTContext &Ctx) { | 
| Anders Carlsson | 54b2698 | 2009-03-14 00:33:21 +0000 | [diff] [blame] | 1196 | assert(!E->isValueDependent() && "Should not see value dependent exprs!"); | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1197 | if (!E->getType()->isIntegralType()) { | 
|  | 1198 | return ICEDiag(2, E->getLocStart()); | 
| Eli Friedman | 5a332ea | 2008-11-13 06:09:17 +0000 | [diff] [blame] | 1199 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1200 |  | 
|  | 1201 | switch (E->getStmtClass()) { | 
| Steve Naroff | 72cada0 | 2007-05-18 00:18:52 +0000 | [diff] [blame] | 1202 | default: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1203 | return ICEDiag(2, E->getLocStart()); | 
|  | 1204 | case Expr::ParenExprClass: | 
|  | 1205 | return CheckICE(cast<ParenExpr>(E)->getSubExpr(), Ctx); | 
|  | 1206 | case Expr::IntegerLiteralClass: | 
|  | 1207 | case Expr::CharacterLiteralClass: | 
|  | 1208 | case Expr::CXXBoolLiteralExprClass: | 
|  | 1209 | case Expr::CXXZeroInitValueExprClass: | 
|  | 1210 | case Expr::TypesCompatibleExprClass: | 
|  | 1211 | case Expr::UnaryTypeTraitExprClass: | 
|  | 1212 | return NoDiag(); | 
|  | 1213 | case Expr::CallExprClass: | 
|  | 1214 | case Expr::CXXOperatorCallExprClass: { | 
|  | 1215 | const CallExpr *CE = cast<CallExpr>(E); | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1216 | if (CE->isBuiltinCall(Ctx)) | 
|  | 1217 | return CheckEvalInICE(E, Ctx); | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1218 | return ICEDiag(2, E->getLocStart()); | 
| Chris Lattner | 5c4664e | 2007-07-15 23:32:58 +0000 | [diff] [blame] | 1219 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1220 | case Expr::DeclRefExprClass: | 
|  | 1221 | case Expr::QualifiedDeclRefExprClass: | 
|  | 1222 | if (isa<EnumConstantDecl>(cast<DeclRefExpr>(E)->getDecl())) | 
|  | 1223 | return NoDiag(); | 
| Sebastian Redl | f3b5e27 | 2009-02-07 13:06:23 +0000 | [diff] [blame] | 1224 | if (Ctx.getLangOptions().CPlusPlus && | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1225 | E->getType().getCVRQualifiers() == QualType::Const) { | 
| Sebastian Redl | f3b5e27 | 2009-02-07 13:06:23 +0000 | [diff] [blame] | 1226 | // C++ 7.1.5.1p2 | 
|  | 1227 | //   A variable of non-volatile const-qualified integral or enumeration | 
|  | 1228 | //   type initialized by an ICE can be used in ICEs. | 
|  | 1229 | if (const VarDecl *Dcl = | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1230 | dyn_cast<VarDecl>(cast<DeclRefExpr>(E)->getDecl())) { | 
| Douglas Gregor | 31cf12c | 2009-05-26 18:54:04 +0000 | [diff] [blame] | 1231 | if (Dcl->isInitKnownICE()) { | 
|  | 1232 | // We have already checked whether this subexpression is an | 
|  | 1233 | // integral constant expression. | 
|  | 1234 | if (Dcl->isInitICE()) | 
|  | 1235 | return NoDiag(); | 
|  | 1236 | else | 
|  | 1237 | return ICEDiag(2, E->getLocStart()); | 
|  | 1238 | } | 
|  | 1239 |  | 
|  | 1240 | if (const Expr *Init = Dcl->getInit()) { | 
|  | 1241 | ICEDiag Result = CheckICE(Init, Ctx); | 
|  | 1242 | // Cache the result of the ICE test. | 
|  | 1243 | Dcl->setInitKnownICE(Ctx, Result.Val == 0); | 
|  | 1244 | return Result; | 
|  | 1245 | } | 
| Sebastian Redl | f3b5e27 | 2009-02-07 13:06:23 +0000 | [diff] [blame] | 1246 | } | 
|  | 1247 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1248 | return ICEDiag(2, E->getLocStart()); | 
|  | 1249 | case Expr::UnaryOperatorClass: { | 
|  | 1250 | const UnaryOperator *Exp = cast<UnaryOperator>(E); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1251 | switch (Exp->getOpcode()) { | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1252 | default: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1253 | return ICEDiag(2, E->getLocStart()); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1254 | case UnaryOperator::Extension: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1255 | case UnaryOperator::LNot: | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1256 | case UnaryOperator::Plus: | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1257 | case UnaryOperator::Minus: | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1258 | case UnaryOperator::Not: | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1259 | case UnaryOperator::Real: | 
|  | 1260 | case UnaryOperator::Imag: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1261 | return CheckICE(Exp->getSubExpr(), Ctx); | 
| Anders Carlsson | a8dc3e6 | 2008-01-29 15:56:48 +0000 | [diff] [blame] | 1262 | case UnaryOperator::OffsetOf: | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1263 | // Note that per C99, offsetof must be an ICE. And AFAIK, using | 
|  | 1264 | // Evaluate matches the proposed gcc behavior for cases like | 
|  | 1265 | // "offsetof(struct s{int x[4];}, x[!.0])".  This doesn't affect | 
|  | 1266 | // compliance: we should warn earlier for offsetof expressions with | 
|  | 1267 | // array subscripts that aren't ICEs, and if the array subscripts | 
|  | 1268 | // are ICEs, the value of the offsetof must be an integer constant. | 
|  | 1269 | return CheckEvalInICE(E, Ctx); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1270 | } | 
| Steve Naroff | 8eeeb13 | 2007-05-08 21:09:37 +0000 | [diff] [blame] | 1271 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1272 | case Expr::SizeOfAlignOfExprClass: { | 
|  | 1273 | const SizeOfAlignOfExpr *Exp = cast<SizeOfAlignOfExpr>(E); | 
|  | 1274 | if (Exp->isSizeOf() && Exp->getTypeOfArgument()->isVariableArrayType()) | 
|  | 1275 | return ICEDiag(2, E->getLocStart()); | 
|  | 1276 | return NoDiag(); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1277 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1278 | case Expr::BinaryOperatorClass: { | 
|  | 1279 | const BinaryOperator *Exp = cast<BinaryOperator>(E); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1280 | switch (Exp->getOpcode()) { | 
|  | 1281 | default: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1282 | return ICEDiag(2, E->getLocStart()); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1283 | case BinaryOperator::Mul: | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1284 | case BinaryOperator::Div: | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1285 | case BinaryOperator::Rem: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1286 | case BinaryOperator::Add: | 
|  | 1287 | case BinaryOperator::Sub: | 
| Chris Lattner | 901ae1f | 2007-06-08 21:54:26 +0000 | [diff] [blame] | 1288 | case BinaryOperator::Shl: | 
| Chris Lattner | 901ae1f | 2007-06-08 21:54:26 +0000 | [diff] [blame] | 1289 | case BinaryOperator::Shr: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1290 | case BinaryOperator::LT: | 
|  | 1291 | case BinaryOperator::GT: | 
|  | 1292 | case BinaryOperator::LE: | 
|  | 1293 | case BinaryOperator::GE: | 
|  | 1294 | case BinaryOperator::EQ: | 
|  | 1295 | case BinaryOperator::NE: | 
|  | 1296 | case BinaryOperator::And: | 
|  | 1297 | case BinaryOperator::Xor: | 
|  | 1298 | case BinaryOperator::Or: | 
|  | 1299 | case BinaryOperator::Comma: { | 
|  | 1300 | ICEDiag LHSResult = CheckICE(Exp->getLHS(), Ctx); | 
|  | 1301 | ICEDiag RHSResult = CheckICE(Exp->getRHS(), Ctx); | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1302 | if (Exp->getOpcode() == BinaryOperator::Div || | 
|  | 1303 | Exp->getOpcode() == BinaryOperator::Rem) { | 
|  | 1304 | // Evaluate gives an error for undefined Div/Rem, so make sure | 
|  | 1305 | // we don't evaluate one. | 
|  | 1306 | if (LHSResult.Val != 2 && RHSResult.Val != 2) { | 
|  | 1307 | llvm::APSInt REval = Exp->getRHS()->EvaluateAsInt(Ctx); | 
|  | 1308 | if (REval == 0) | 
|  | 1309 | return ICEDiag(1, E->getLocStart()); | 
|  | 1310 | if (REval.isSigned() && REval.isAllOnesValue()) { | 
|  | 1311 | llvm::APSInt LEval = Exp->getLHS()->EvaluateAsInt(Ctx); | 
|  | 1312 | if (LEval.isMinSignedValue()) | 
|  | 1313 | return ICEDiag(1, E->getLocStart()); | 
|  | 1314 | } | 
|  | 1315 | } | 
|  | 1316 | } | 
|  | 1317 | if (Exp->getOpcode() == BinaryOperator::Comma) { | 
|  | 1318 | if (Ctx.getLangOptions().C99) { | 
|  | 1319 | // C99 6.6p3 introduces a strange edge case: comma can be in an ICE | 
|  | 1320 | // if it isn't evaluated. | 
|  | 1321 | if (LHSResult.Val == 0 && RHSResult.Val == 0) | 
|  | 1322 | return ICEDiag(1, E->getLocStart()); | 
|  | 1323 | } else { | 
|  | 1324 | // In both C89 and C++, commas in ICEs are illegal. | 
|  | 1325 | return ICEDiag(2, E->getLocStart()); | 
|  | 1326 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1327 | } | 
|  | 1328 | if (LHSResult.Val >= RHSResult.Val) | 
|  | 1329 | return LHSResult; | 
|  | 1330 | return RHSResult; | 
|  | 1331 | } | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1332 | case BinaryOperator::LAnd: | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1333 | case BinaryOperator::LOr: { | 
|  | 1334 | ICEDiag LHSResult = CheckICE(Exp->getLHS(), Ctx); | 
|  | 1335 | ICEDiag RHSResult = CheckICE(Exp->getRHS(), Ctx); | 
|  | 1336 | if (LHSResult.Val == 0 && RHSResult.Val == 1) { | 
|  | 1337 | // Rare case where the RHS has a comma "side-effect"; we need | 
|  | 1338 | // to actually check the condition to see whether the side | 
|  | 1339 | // with the comma is evaluated. | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1340 | if ((Exp->getOpcode() == BinaryOperator::LAnd) != | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1341 | (Exp->getLHS()->EvaluateAsInt(Ctx) == 0)) | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1342 | return RHSResult; | 
|  | 1343 | return NoDiag(); | 
| Eli Friedman | 8553a98 | 2008-11-13 02:13:11 +0000 | [diff] [blame] | 1344 | } | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1345 |  | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1346 | if (LHSResult.Val >= RHSResult.Val) | 
|  | 1347 | return LHSResult; | 
|  | 1348 | return RHSResult; | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1349 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1350 | } | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1351 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1352 | case Expr::ImplicitCastExprClass: | 
|  | 1353 | case Expr::CStyleCastExprClass: | 
|  | 1354 | case Expr::CXXFunctionalCastExprClass: { | 
|  | 1355 | const Expr *SubExpr = cast<CastExpr>(E)->getSubExpr(); | 
|  | 1356 | if (SubExpr->getType()->isIntegralType()) | 
|  | 1357 | return CheckICE(SubExpr, Ctx); | 
|  | 1358 | if (isa<FloatingLiteral>(SubExpr->IgnoreParens())) | 
|  | 1359 | return NoDiag(); | 
|  | 1360 | return ICEDiag(2, E->getLocStart()); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1361 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1362 | case Expr::ConditionalOperatorClass: { | 
|  | 1363 | const ConditionalOperator *Exp = cast<ConditionalOperator>(E); | 
| Chris Lattner | 85b25bc | 2008-12-12 06:55:44 +0000 | [diff] [blame] | 1364 | // If the condition (ignoring parens) is a __builtin_constant_p call, | 
|  | 1365 | // then only the true side is actually considered in an integer constant | 
| Chris Lattner | 0439735 | 2008-12-12 18:00:51 +0000 | [diff] [blame] | 1366 | // expression, and it is fully evaluated.  This is an important GNU | 
|  | 1367 | // extension.  See GCC PR38377 for discussion. | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1368 | if (const CallExpr *CallCE = dyn_cast<CallExpr>(Exp->getCond()->IgnoreParenCasts())) | 
| Douglas Gregor | e711f70 | 2009-02-14 18:57:46 +0000 | [diff] [blame] | 1369 | if (CallCE->isBuiltinCall(Ctx) == Builtin::BI__builtin_constant_p) { | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1370 | Expr::EvalResult EVResult; | 
|  | 1371 | if (!E->Evaluate(EVResult, Ctx) || EVResult.HasSideEffects || | 
|  | 1372 | !EVResult.Val.isInt()) { | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1373 | return ICEDiag(2, E->getLocStart()); | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1374 | } | 
|  | 1375 | return NoDiag(); | 
| Chris Lattner | 0439735 | 2008-12-12 18:00:51 +0000 | [diff] [blame] | 1376 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1377 | ICEDiag CondResult = CheckICE(Exp->getCond(), Ctx); | 
|  | 1378 | ICEDiag TrueResult = CheckICE(Exp->getTrueExpr(), Ctx); | 
|  | 1379 | ICEDiag FalseResult = CheckICE(Exp->getFalseExpr(), Ctx); | 
|  | 1380 | if (CondResult.Val == 2) | 
|  | 1381 | return CondResult; | 
|  | 1382 | if (TrueResult.Val == 2) | 
|  | 1383 | return TrueResult; | 
|  | 1384 | if (FalseResult.Val == 2) | 
|  | 1385 | return FalseResult; | 
|  | 1386 | if (CondResult.Val == 1) | 
|  | 1387 | return CondResult; | 
|  | 1388 | if (TrueResult.Val == 0 && FalseResult.Val == 0) | 
|  | 1389 | return NoDiag(); | 
|  | 1390 | // Rare case where the diagnostics depend on which side is evaluated | 
|  | 1391 | // Note that if we get here, CondResult is 0, and at least one of | 
|  | 1392 | // TrueResult and FalseResult is non-zero. | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1393 | if (Exp->getCond()->EvaluateAsInt(Ctx) == 0) { | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1394 | return FalseResult; | 
|  | 1395 | } | 
|  | 1396 | return TrueResult; | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1397 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1398 | case Expr::CXXDefaultArgExprClass: | 
|  | 1399 | return CheckICE(cast<CXXDefaultArgExpr>(E)->getExpr(), Ctx); | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1400 | case Expr::ChooseExprClass: { | 
| Eli Friedman | e0a5b8b | 2009-03-04 05:52:32 +0000 | [diff] [blame] | 1401 | return CheckICE(cast<ChooseExpr>(E)->getChosenSubExpr(Ctx), Ctx); | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1402 | } | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1403 | } | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1404 | } | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1405 |  | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1406 | bool Expr::isIntegerConstantExpr(llvm::APSInt &Result, ASTContext &Ctx, | 
|  | 1407 | SourceLocation *Loc, bool isEvaluated) const { | 
|  | 1408 | ICEDiag d = CheckICE(this, Ctx); | 
|  | 1409 | if (d.Val != 0) { | 
|  | 1410 | if (Loc) *Loc = d.Loc; | 
|  | 1411 | return false; | 
|  | 1412 | } | 
|  | 1413 | EvalResult EvalResult; | 
| Eli Friedman | 90afd3d | 2009-02-27 04:07:58 +0000 | [diff] [blame] | 1414 | if (!Evaluate(EvalResult, Ctx)) | 
|  | 1415 | assert(0 && "ICE cannot be evaluated!"); | 
|  | 1416 | assert(!EvalResult.HasSideEffects && "ICE with side effects!"); | 
|  | 1417 | assert(EvalResult.Val.isInt() && "ICE that isn't integer!"); | 
| Eli Friedman | 98c56a4 | 2009-02-26 09:29:13 +0000 | [diff] [blame] | 1418 | Result = EvalResult.Val.getInt(); | 
| Chris Lattner | e0da5dc | 2007-06-05 05:58:31 +0000 | [diff] [blame] | 1419 | return true; | 
| Steve Naroff | 8eeeb13 | 2007-05-08 21:09:37 +0000 | [diff] [blame] | 1420 | } | 
|  | 1421 |  | 
| Chris Lattner | 7eef919 | 2007-05-24 01:23:49 +0000 | [diff] [blame] | 1422 | /// isNullPointerConstant - C99 6.3.2.3p3 -  Return true if this is either an | 
|  | 1423 | /// integer constant expression with the value zero, or if this is one that is | 
|  | 1424 | /// cast to void*. | 
| Anders Carlsson | 4bb9113 | 2008-12-01 02:13:57 +0000 | [diff] [blame] | 1425 | bool Expr::isNullPointerConstant(ASTContext &Ctx) const | 
|  | 1426 | { | 
| Sebastian Redl | 72b8aef | 2008-10-31 14:43:28 +0000 | [diff] [blame] | 1427 | // Strip off a cast to void*, if it exists. Except in C++. | 
| Argyrios Kyrtzidis | 3bab3d2 | 2008-08-18 23:01:59 +0000 | [diff] [blame] | 1428 | if (const ExplicitCastExpr *CE = dyn_cast<ExplicitCastExpr>(this)) { | 
| Sebastian Redl | 273ce56 | 2008-11-04 11:45:54 +0000 | [diff] [blame] | 1429 | if (!Ctx.getLangOptions().CPlusPlus) { | 
| Sebastian Redl | 72b8aef | 2008-10-31 14:43:28 +0000 | [diff] [blame] | 1430 | // Check that it is a cast to void*. | 
| Ted Kremenek | 8a286fb | 2009-07-17 17:50:17 +0000 | [diff] [blame] | 1431 | if (const PointerType *PT = CE->getType()->getAsPointerType()) { | 
| Sebastian Redl | 72b8aef | 2008-10-31 14:43:28 +0000 | [diff] [blame] | 1432 | QualType Pointee = PT->getPointeeType(); | 
|  | 1433 | if (Pointee.getCVRQualifiers() == 0 && | 
|  | 1434 | Pointee->isVoidType() &&                              // to void* | 
|  | 1435 | CE->getSubExpr()->getType()->isIntegerType())         // from int. | 
| Anders Carlsson | eade3ad | 2008-12-01 06:28:23 +0000 | [diff] [blame] | 1436 | return CE->getSubExpr()->isNullPointerConstant(Ctx); | 
| Sebastian Redl | 72b8aef | 2008-10-31 14:43:28 +0000 | [diff] [blame] | 1437 | } | 
| Steve Naroff | ada7d42 | 2007-05-20 17:54:12 +0000 | [diff] [blame] | 1438 | } | 
| Steve Naroff | 4871fe0 | 2008-01-14 16:10:57 +0000 | [diff] [blame] | 1439 | } else if (const ImplicitCastExpr *ICE = dyn_cast<ImplicitCastExpr>(this)) { | 
|  | 1440 | // Ignore the ImplicitCastExpr type entirely. | 
| Anders Carlsson | eade3ad | 2008-12-01 06:28:23 +0000 | [diff] [blame] | 1441 | return ICE->getSubExpr()->isNullPointerConstant(Ctx); | 
| Steve Naroff | 4871fe0 | 2008-01-14 16:10:57 +0000 | [diff] [blame] | 1442 | } else if (const ParenExpr *PE = dyn_cast<ParenExpr>(this)) { | 
|  | 1443 | // Accept ((void*)0) as a null pointer constant, as many other | 
|  | 1444 | // implementations do. | 
| Anders Carlsson | eade3ad | 2008-12-01 06:28:23 +0000 | [diff] [blame] | 1445 | return PE->getSubExpr()->isNullPointerConstant(Ctx); | 
| Chris Lattner | 5825824 | 2008-04-10 02:22:51 +0000 | [diff] [blame] | 1446 | } else if (const CXXDefaultArgExpr *DefaultArg | 
|  | 1447 | = dyn_cast<CXXDefaultArgExpr>(this)) { | 
| Chris Lattner | aa9c7ae | 2008-04-08 04:40:51 +0000 | [diff] [blame] | 1448 | // See through default argument expressions | 
| Anders Carlsson | eade3ad | 2008-12-01 06:28:23 +0000 | [diff] [blame] | 1449 | return DefaultArg->getExpr()->isNullPointerConstant(Ctx); | 
| Douglas Gregor | 3be4b12 | 2008-11-29 04:51:27 +0000 | [diff] [blame] | 1450 | } else if (isa<GNUNullExpr>(this)) { | 
|  | 1451 | // The GNU __null extension is always a null pointer constant. | 
|  | 1452 | return true; | 
| Steve Naroff | 0903531 | 2008-01-14 02:53:34 +0000 | [diff] [blame] | 1453 | } | 
| Douglas Gregor | 3be4b12 | 2008-11-29 04:51:27 +0000 | [diff] [blame] | 1454 |  | 
| Sebastian Redl | 576fd42 | 2009-05-10 18:38:11 +0000 | [diff] [blame] | 1455 | // C++0x nullptr_t is always a null pointer constant. | 
|  | 1456 | if (getType()->isNullPtrType()) | 
|  | 1457 | return true; | 
|  | 1458 |  | 
| Steve Naroff | 4871fe0 | 2008-01-14 16:10:57 +0000 | [diff] [blame] | 1459 | // This expression must be an integer type. | 
|  | 1460 | if (!getType()->isIntegerType()) | 
|  | 1461 | return false; | 
|  | 1462 |  | 
| Chris Lattner | 1abbd41 | 2007-06-08 17:58:43 +0000 | [diff] [blame] | 1463 | // If we have an integer constant expression, we need to *evaluate* it and | 
|  | 1464 | // test for the value 0. | 
| Eli Friedman | 7524de1 | 2009-04-25 22:37:12 +0000 | [diff] [blame] | 1465 | llvm::APSInt Result; | 
|  | 1466 | return isIntegerConstantExpr(Result, Ctx) && Result == 0; | 
| Steve Naroff | 218bc2b | 2007-05-04 21:54:46 +0000 | [diff] [blame] | 1467 | } | 
| Steve Naroff | f7a5da1 | 2007-07-28 23:10:27 +0000 | [diff] [blame] | 1468 |  | 
| Douglas Gregor | 71235ec | 2009-05-02 02:18:30 +0000 | [diff] [blame] | 1469 | FieldDecl *Expr::getBitField() { | 
| Douglas Gregor | 19623dc | 2009-07-06 15:38:40 +0000 | [diff] [blame] | 1470 | Expr *E = this->IgnoreParens(); | 
| Douglas Gregor | 71235ec | 2009-05-02 02:18:30 +0000 | [diff] [blame] | 1471 |  | 
| Douglas Gregor | 8e1cf60 | 2008-10-29 00:13:59 +0000 | [diff] [blame] | 1472 | if (MemberExpr *MemRef = dyn_cast<MemberExpr>(E)) | 
| Douglas Gregor | 2eedc3a | 2008-12-20 23:49:58 +0000 | [diff] [blame] | 1473 | if (FieldDecl *Field = dyn_cast<FieldDecl>(MemRef->getMemberDecl())) | 
| Douglas Gregor | 71235ec | 2009-05-02 02:18:30 +0000 | [diff] [blame] | 1474 | if (Field->isBitField()) | 
|  | 1475 | return Field; | 
|  | 1476 |  | 
|  | 1477 | if (BinaryOperator *BinOp = dyn_cast<BinaryOperator>(E)) | 
|  | 1478 | if (BinOp->isAssignmentOp() && BinOp->getLHS()) | 
|  | 1479 | return BinOp->getLHS()->getBitField(); | 
|  | 1480 |  | 
|  | 1481 | return 0; | 
| Douglas Gregor | 8e1cf60 | 2008-10-29 00:13:59 +0000 | [diff] [blame] | 1482 | } | 
|  | 1483 |  | 
| Chris Lattner | b8211f6 | 2009-02-16 22:14:05 +0000 | [diff] [blame] | 1484 | /// isArrow - Return true if the base expression is a pointer to vector, | 
|  | 1485 | /// return false if the base expression is a vector. | 
|  | 1486 | bool ExtVectorElementExpr::isArrow() const { | 
|  | 1487 | return getBase()->getType()->isPointerType(); | 
|  | 1488 | } | 
|  | 1489 |  | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 1490 | unsigned ExtVectorElementExpr::getNumElements() const { | 
| Nate Begeman | f322eab | 2008-05-09 06:41:27 +0000 | [diff] [blame] | 1491 | if (const VectorType *VT = getType()->getAsVectorType()) | 
|  | 1492 | return VT->getNumElements(); | 
|  | 1493 | return 1; | 
| Chris Lattner | 177bd45 | 2007-08-03 16:00:20 +0000 | [diff] [blame] | 1494 | } | 
|  | 1495 |  | 
| Nate Begeman | f322eab | 2008-05-09 06:41:27 +0000 | [diff] [blame] | 1496 | /// containsDuplicateElements - Return true if any element access is repeated. | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 1497 | bool ExtVectorElementExpr::containsDuplicateElements() const { | 
| Douglas Gregor | 2ebf8717 | 2009-04-15 23:02:49 +0000 | [diff] [blame] | 1498 | const char *compStr = Accessor->getName(); | 
|  | 1499 | unsigned length = Accessor->getLength(); | 
| Nate Begeman | 7e5185b | 2009-01-18 02:01:21 +0000 | [diff] [blame] | 1500 |  | 
|  | 1501 | // Halving swizzles do not contain duplicate elements. | 
|  | 1502 | if (!strcmp(compStr, "hi") || !strcmp(compStr, "lo") || | 
|  | 1503 | !strcmp(compStr, "even") || !strcmp(compStr, "odd")) | 
|  | 1504 | return false; | 
|  | 1505 |  | 
|  | 1506 | // Advance past s-char prefix on hex swizzles. | 
| Nate Begeman | 0359e12 | 2009-06-25 21:06:09 +0000 | [diff] [blame] | 1507 | if (*compStr == 's' || *compStr == 'S') { | 
| Nate Begeman | 7e5185b | 2009-01-18 02:01:21 +0000 | [diff] [blame] | 1508 | compStr++; | 
|  | 1509 | length--; | 
|  | 1510 | } | 
| Steve Naroff | 0d595ca | 2007-07-30 03:29:09 +0000 | [diff] [blame] | 1511 |  | 
| Chris Lattner | 6ef2bc4 | 2008-11-19 07:55:04 +0000 | [diff] [blame] | 1512 | for (unsigned i = 0; i != length-1; i++) { | 
| Steve Naroff | 0d595ca | 2007-07-30 03:29:09 +0000 | [diff] [blame] | 1513 | const char *s = compStr+i; | 
|  | 1514 | for (const char c = *s++; *s; s++) | 
|  | 1515 | if (c == *s) | 
|  | 1516 | return true; | 
|  | 1517 | } | 
|  | 1518 | return false; | 
|  | 1519 | } | 
| Chris Lattner | 885b495 | 2007-08-02 23:36:59 +0000 | [diff] [blame] | 1520 |  | 
| Nate Begeman | f322eab | 2008-05-09 06:41:27 +0000 | [diff] [blame] | 1521 | /// getEncodedElementAccess - We encode the fields as a llvm ConstantArray. | 
| Nate Begeman | d386215 | 2008-05-13 21:03:02 +0000 | [diff] [blame] | 1522 | void ExtVectorElementExpr::getEncodedElementAccess( | 
|  | 1523 | llvm::SmallVectorImpl<unsigned> &Elts) const { | 
| Douglas Gregor | 2ebf8717 | 2009-04-15 23:02:49 +0000 | [diff] [blame] | 1524 | const char *compStr = Accessor->getName(); | 
| Nate Begeman | 0359e12 | 2009-06-25 21:06:09 +0000 | [diff] [blame] | 1525 | if (*compStr == 's' || *compStr == 'S') | 
| Nate Begeman | bb70bf6 | 2009-01-18 01:47:54 +0000 | [diff] [blame] | 1526 | compStr++; | 
|  | 1527 |  | 
|  | 1528 | bool isHi =   !strcmp(compStr, "hi"); | 
|  | 1529 | bool isLo =   !strcmp(compStr, "lo"); | 
|  | 1530 | bool isEven = !strcmp(compStr, "even"); | 
|  | 1531 | bool isOdd  = !strcmp(compStr, "odd"); | 
|  | 1532 |  | 
| Nate Begeman | f322eab | 2008-05-09 06:41:27 +0000 | [diff] [blame] | 1533 | for (unsigned i = 0, e = getNumElements(); i != e; ++i) { | 
|  | 1534 | uint64_t Index; | 
|  | 1535 |  | 
|  | 1536 | if (isHi) | 
|  | 1537 | Index = e + i; | 
|  | 1538 | else if (isLo) | 
|  | 1539 | Index = i; | 
|  | 1540 | else if (isEven) | 
|  | 1541 | Index = 2 * i; | 
|  | 1542 | else if (isOdd) | 
|  | 1543 | Index = 2 * i + 1; | 
|  | 1544 | else | 
|  | 1545 | Index = ExtVectorType::getAccessorIdx(compStr[i]); | 
| Chris Lattner | 885b495 | 2007-08-02 23:36:59 +0000 | [diff] [blame] | 1546 |  | 
| Nate Begeman | d386215 | 2008-05-13 21:03:02 +0000 | [diff] [blame] | 1547 | Elts.push_back(Index); | 
| Chris Lattner | 885b495 | 2007-08-02 23:36:59 +0000 | [diff] [blame] | 1548 | } | 
| Nate Begeman | f322eab | 2008-05-09 06:41:27 +0000 | [diff] [blame] | 1549 | } | 
|  | 1550 |  | 
| Steve Naroff | f73590d | 2007-09-27 14:38:14 +0000 | [diff] [blame] | 1551 | // constructor for instance messages. | 
| Steve Naroff | 8017506 | 2007-09-28 22:22:11 +0000 | [diff] [blame] | 1552 | ObjCMessageExpr::ObjCMessageExpr(Expr *receiver, Selector selInfo, | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 1553 | QualType retType, ObjCMethodDecl *mproto, | 
| Steve Naroff | 66697aa | 2007-11-03 16:37:59 +0000 | [diff] [blame] | 1554 | SourceLocation LBrac, SourceLocation RBrac, | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1555 | Expr **ArgExprs, unsigned nargs) | 
| Steve Naroff | 66697aa | 2007-11-03 16:37:59 +0000 | [diff] [blame] | 1556 | : Expr(ObjCMessageExprClass, retType), SelName(selInfo), | 
| Ted Kremenek | b8861a6 | 2008-05-01 17:26:20 +0000 | [diff] [blame] | 1557 | MethodProto(mproto) { | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1558 | NumArgs = nargs; | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1559 | SubExprs = new Stmt*[NumArgs+1]; | 
| Steve Naroff | f73590d | 2007-09-27 14:38:14 +0000 | [diff] [blame] | 1560 | SubExprs[RECEIVER] = receiver; | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1561 | if (NumArgs) { | 
|  | 1562 | for (unsigned i = 0; i != NumArgs; ++i) | 
| Steve Naroff | f73590d | 2007-09-27 14:38:14 +0000 | [diff] [blame] | 1563 | SubExprs[i+ARGS_START] = static_cast<Expr *>(ArgExprs[i]); | 
|  | 1564 | } | 
| Steve Naroff | d54978b | 2007-09-18 23:55:05 +0000 | [diff] [blame] | 1565 | LBracloc = LBrac; | 
|  | 1566 | RBracloc = RBrac; | 
|  | 1567 | } | 
|  | 1568 |  | 
| Anders Carlsson | 0880e9f | 2009-06-07 19:51:47 +0000 | [diff] [blame] | 1569 | ObjCStringLiteral* ObjCStringLiteral::Clone(ASTContext &C) const { | 
|  | 1570 | // Clone the string literal. | 
|  | 1571 | StringLiteral *NewString = | 
|  | 1572 | String ? cast<StringLiteral>(String)->Clone(C) : 0; | 
|  | 1573 |  | 
|  | 1574 | return new (C) ObjCStringLiteral(NewString, getType(), AtLoc); | 
|  | 1575 | } | 
|  | 1576 |  | 
|  | 1577 | ObjCSelectorExpr *ObjCSelectorExpr::Clone(ASTContext &C) const { | 
|  | 1578 | return new (C) ObjCSelectorExpr(getType(), SelName, AtLoc, RParenLoc); | 
|  | 1579 | } | 
|  | 1580 |  | 
|  | 1581 | ObjCProtocolExpr *ObjCProtocolExpr::Clone(ASTContext &C) const { | 
| Fariborz Jahanian | 99b57fb | 2009-06-21 18:26:03 +0000 | [diff] [blame] | 1582 | return new (C) ObjCProtocolExpr(getType(), TheProtocol, AtLoc, RParenLoc); | 
| Anders Carlsson | 0880e9f | 2009-06-07 19:51:47 +0000 | [diff] [blame] | 1583 | } | 
|  | 1584 |  | 
| Steve Naroff | f73590d | 2007-09-27 14:38:14 +0000 | [diff] [blame] | 1585 | // constructor for class messages. | 
|  | 1586 | // FIXME: clsName should be typed to ObjCInterfaceType | 
| Steve Naroff | 8017506 | 2007-09-28 22:22:11 +0000 | [diff] [blame] | 1587 | ObjCMessageExpr::ObjCMessageExpr(IdentifierInfo *clsName, Selector selInfo, | 
| Ted Kremenek | 1b0ea82 | 2008-01-07 19:49:32 +0000 | [diff] [blame] | 1588 | QualType retType, ObjCMethodDecl *mproto, | 
| Steve Naroff | 66697aa | 2007-11-03 16:37:59 +0000 | [diff] [blame] | 1589 | SourceLocation LBrac, SourceLocation RBrac, | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1590 | Expr **ArgExprs, unsigned nargs) | 
| Steve Naroff | 66697aa | 2007-11-03 16:37:59 +0000 | [diff] [blame] | 1591 | : Expr(ObjCMessageExprClass, retType), SelName(selInfo), | 
| Ted Kremenek | b8861a6 | 2008-05-01 17:26:20 +0000 | [diff] [blame] | 1592 | MethodProto(mproto) { | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1593 | NumArgs = nargs; | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1594 | SubExprs = new Stmt*[NumArgs+1]; | 
| Ted Kremenek | a3a37ae | 2008-06-24 15:50:53 +0000 | [diff] [blame] | 1595 | SubExprs[RECEIVER] = (Expr*) ((uintptr_t) clsName | IsClsMethDeclUnknown); | 
| Steve Naroff | e3ffc2f | 2007-11-15 13:05:42 +0000 | [diff] [blame] | 1596 | if (NumArgs) { | 
|  | 1597 | for (unsigned i = 0; i != NumArgs; ++i) | 
| Steve Naroff | f73590d | 2007-09-27 14:38:14 +0000 | [diff] [blame] | 1598 | SubExprs[i+ARGS_START] = static_cast<Expr *>(ArgExprs[i]); | 
|  | 1599 | } | 
| Steve Naroff | d54978b | 2007-09-18 23:55:05 +0000 | [diff] [blame] | 1600 | LBracloc = LBrac; | 
|  | 1601 | RBracloc = RBrac; | 
|  | 1602 | } | 
|  | 1603 |  | 
| Ted Kremenek | a3a37ae | 2008-06-24 15:50:53 +0000 | [diff] [blame] | 1604 | // constructor for class messages. | 
|  | 1605 | ObjCMessageExpr::ObjCMessageExpr(ObjCInterfaceDecl *cls, Selector selInfo, | 
|  | 1606 | QualType retType, ObjCMethodDecl *mproto, | 
|  | 1607 | SourceLocation LBrac, SourceLocation RBrac, | 
|  | 1608 | Expr **ArgExprs, unsigned nargs) | 
|  | 1609 | : Expr(ObjCMessageExprClass, retType), SelName(selInfo), | 
|  | 1610 | MethodProto(mproto) { | 
|  | 1611 | NumArgs = nargs; | 
|  | 1612 | SubExprs = new Stmt*[NumArgs+1]; | 
|  | 1613 | SubExprs[RECEIVER] = (Expr*) ((uintptr_t) cls | IsClsMethDeclKnown); | 
|  | 1614 | if (NumArgs) { | 
|  | 1615 | for (unsigned i = 0; i != NumArgs; ++i) | 
|  | 1616 | SubExprs[i+ARGS_START] = static_cast<Expr *>(ArgExprs[i]); | 
|  | 1617 | } | 
|  | 1618 | LBracloc = LBrac; | 
|  | 1619 | RBracloc = RBrac; | 
|  | 1620 | } | 
|  | 1621 |  | 
|  | 1622 | ObjCMessageExpr::ClassInfo ObjCMessageExpr::getClassInfo() const { | 
|  | 1623 | uintptr_t x = (uintptr_t) SubExprs[RECEIVER]; | 
|  | 1624 | switch (x & Flags) { | 
|  | 1625 | default: | 
|  | 1626 | assert(false && "Invalid ObjCMessageExpr."); | 
|  | 1627 | case IsInstMeth: | 
|  | 1628 | return ClassInfo(0, 0); | 
|  | 1629 | case IsClsMethDeclUnknown: | 
|  | 1630 | return ClassInfo(0, (IdentifierInfo*) (x & ~Flags)); | 
|  | 1631 | case IsClsMethDeclKnown: { | 
|  | 1632 | ObjCInterfaceDecl* D = (ObjCInterfaceDecl*) (x & ~Flags); | 
|  | 1633 | return ClassInfo(D, D->getIdentifier()); | 
|  | 1634 | } | 
|  | 1635 | } | 
|  | 1636 | } | 
|  | 1637 |  | 
| Chris Lattner | 7ec71da | 2009-04-26 00:44:05 +0000 | [diff] [blame] | 1638 | void ObjCMessageExpr::setClassInfo(const ObjCMessageExpr::ClassInfo &CI) { | 
|  | 1639 | if (CI.first == 0 && CI.second == 0) | 
|  | 1640 | SubExprs[RECEIVER] = (Expr*)((uintptr_t)0 | IsInstMeth); | 
|  | 1641 | else if (CI.first == 0) | 
|  | 1642 | SubExprs[RECEIVER] = (Expr*)((uintptr_t)CI.second | IsClsMethDeclUnknown); | 
|  | 1643 | else | 
|  | 1644 | SubExprs[RECEIVER] = (Expr*)((uintptr_t)CI.first | IsClsMethDeclKnown); | 
|  | 1645 | } | 
|  | 1646 |  | 
|  | 1647 |  | 
| Chris Lattner | 35e564e | 2007-10-25 00:29:32 +0000 | [diff] [blame] | 1648 | bool ChooseExpr::isConditionTrue(ASTContext &C) const { | 
| Eli Friedman | 1c4a175 | 2009-04-26 19:19:15 +0000 | [diff] [blame] | 1649 | return getCond()->EvaluateAsInt(C) != 0; | 
| Chris Lattner | 35e564e | 2007-10-25 00:29:32 +0000 | [diff] [blame] | 1650 | } | 
|  | 1651 |  | 
| Douglas Gregor | a3c5590 | 2009-04-16 00:01:45 +0000 | [diff] [blame] | 1652 | void ShuffleVectorExpr::setExprs(Expr ** Exprs, unsigned NumExprs) { | 
|  | 1653 | if (NumExprs) | 
|  | 1654 | delete [] SubExprs; | 
|  | 1655 |  | 
|  | 1656 | SubExprs = new Stmt* [NumExprs]; | 
|  | 1657 | this->NumExprs = NumExprs; | 
|  | 1658 | memcpy(SubExprs, Exprs, sizeof(Expr *) * NumExprs); | 
|  | 1659 | } | 
|  | 1660 |  | 
| Sebastian Redl | 6f28289 | 2008-11-11 17:56:53 +0000 | [diff] [blame] | 1661 | void SizeOfAlignOfExpr::Destroy(ASTContext& C) { | 
|  | 1662 | // Override default behavior of traversing children. If this has a type | 
|  | 1663 | // operand and the type is a variable-length array, the child iteration | 
|  | 1664 | // will iterate over the size expression. However, this expression belongs | 
|  | 1665 | // to the type, not to this, so we don't want to delete it. | 
|  | 1666 | // We still want to delete this expression. | 
| Ted Kremenek | 5a20195 | 2009-02-07 01:47:29 +0000 | [diff] [blame] | 1667 | if (isArgumentType()) { | 
|  | 1668 | this->~SizeOfAlignOfExpr(); | 
|  | 1669 | C.Deallocate(this); | 
|  | 1670 | } | 
| Sebastian Redl | 6f28289 | 2008-11-11 17:56:53 +0000 | [diff] [blame] | 1671 | else | 
|  | 1672 | Expr::Destroy(C); | 
| Daniel Dunbar | 3e1888e | 2008-08-28 18:02:04 +0000 | [diff] [blame] | 1673 | } | 
|  | 1674 |  | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1675 | //===----------------------------------------------------------------------===// | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1676 | //  DesignatedInitExpr | 
|  | 1677 | //===----------------------------------------------------------------------===// | 
|  | 1678 |  | 
|  | 1679 | IdentifierInfo *DesignatedInitExpr::Designator::getFieldName() { | 
|  | 1680 | assert(Kind == FieldDesignator && "Only valid on a field designator"); | 
|  | 1681 | if (Field.NameOrField & 0x01) | 
|  | 1682 | return reinterpret_cast<IdentifierInfo *>(Field.NameOrField&~0x01); | 
|  | 1683 | else | 
|  | 1684 | return getField()->getIdentifier(); | 
|  | 1685 | } | 
|  | 1686 |  | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1687 | DesignatedInitExpr::DesignatedInitExpr(QualType Ty, unsigned NumDesignators, | 
|  | 1688 | const Designator *Designators, | 
|  | 1689 | SourceLocation EqualOrColonLoc, | 
|  | 1690 | bool GNUSyntax, | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1691 | Expr **IndexExprs, | 
|  | 1692 | unsigned NumIndexExprs, | 
|  | 1693 | Expr *Init) | 
|  | 1694 | : Expr(DesignatedInitExprClass, Ty, | 
|  | 1695 | Init->isTypeDependent(), Init->isValueDependent()), | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1696 | EqualOrColonLoc(EqualOrColonLoc), GNUSyntax(GNUSyntax), | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1697 | NumDesignators(NumDesignators), NumSubExprs(NumIndexExprs + 1) { | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1698 | this->Designators = new Designator[NumDesignators]; | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1699 |  | 
|  | 1700 | // Record the initializer itself. | 
|  | 1701 | child_iterator Child = child_begin(); | 
|  | 1702 | *Child++ = Init; | 
|  | 1703 |  | 
|  | 1704 | // Copy the designators and their subexpressions, computing | 
|  | 1705 | // value-dependence along the way. | 
|  | 1706 | unsigned IndexIdx = 0; | 
|  | 1707 | for (unsigned I = 0; I != NumDesignators; ++I) { | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1708 | this->Designators[I] = Designators[I]; | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1709 |  | 
|  | 1710 | if (this->Designators[I].isArrayDesignator()) { | 
|  | 1711 | // Compute type- and value-dependence. | 
|  | 1712 | Expr *Index = IndexExprs[IndexIdx]; | 
|  | 1713 | ValueDependent = ValueDependent || | 
|  | 1714 | Index->isTypeDependent() || Index->isValueDependent(); | 
|  | 1715 |  | 
|  | 1716 | // Copy the index expressions into permanent storage. | 
|  | 1717 | *Child++ = IndexExprs[IndexIdx++]; | 
|  | 1718 | } else if (this->Designators[I].isArrayRangeDesignator()) { | 
|  | 1719 | // Compute type- and value-dependence. | 
|  | 1720 | Expr *Start = IndexExprs[IndexIdx]; | 
|  | 1721 | Expr *End = IndexExprs[IndexIdx + 1]; | 
|  | 1722 | ValueDependent = ValueDependent || | 
|  | 1723 | Start->isTypeDependent() || Start->isValueDependent() || | 
|  | 1724 | End->isTypeDependent() || End->isValueDependent(); | 
|  | 1725 |  | 
|  | 1726 | // Copy the start/end expressions into permanent storage. | 
|  | 1727 | *Child++ = IndexExprs[IndexIdx++]; | 
|  | 1728 | *Child++ = IndexExprs[IndexIdx++]; | 
|  | 1729 | } | 
|  | 1730 | } | 
|  | 1731 |  | 
|  | 1732 | assert(IndexIdx == NumIndexExprs && "Wrong number of index expressions"); | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1733 | } | 
|  | 1734 |  | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1735 | DesignatedInitExpr * | 
|  | 1736 | DesignatedInitExpr::Create(ASTContext &C, Designator *Designators, | 
|  | 1737 | unsigned NumDesignators, | 
|  | 1738 | Expr **IndexExprs, unsigned NumIndexExprs, | 
|  | 1739 | SourceLocation ColonOrEqualLoc, | 
|  | 1740 | bool UsesColonSyntax, Expr *Init) { | 
| Steve Naroff | 99c0cdf | 2009-01-27 23:20:32 +0000 | [diff] [blame] | 1741 | void *Mem = C.Allocate(sizeof(DesignatedInitExpr) + | 
| Steve Naroff | 99c0cdf | 2009-01-27 23:20:32 +0000 | [diff] [blame] | 1742 | sizeof(Stmt *) * (NumIndexExprs + 1), 8); | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1743 | return new (Mem) DesignatedInitExpr(C.VoidTy, NumDesignators, Designators, | 
|  | 1744 | ColonOrEqualLoc, UsesColonSyntax, | 
|  | 1745 | IndexExprs, NumIndexExprs, Init); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1746 | } | 
|  | 1747 |  | 
| Douglas Gregor | 38676d5 | 2009-04-16 00:55:48 +0000 | [diff] [blame] | 1748 | DesignatedInitExpr *DesignatedInitExpr::CreateEmpty(ASTContext &C, | 
|  | 1749 | unsigned NumIndexExprs) { | 
|  | 1750 | void *Mem = C.Allocate(sizeof(DesignatedInitExpr) + | 
|  | 1751 | sizeof(Stmt *) * (NumIndexExprs + 1), 8); | 
|  | 1752 | return new (Mem) DesignatedInitExpr(NumIndexExprs + 1); | 
|  | 1753 | } | 
|  | 1754 |  | 
|  | 1755 | void DesignatedInitExpr::setDesignators(const Designator *Desigs, | 
|  | 1756 | unsigned NumDesigs) { | 
|  | 1757 | if (Designators) | 
|  | 1758 | delete [] Designators; | 
|  | 1759 |  | 
|  | 1760 | Designators = new Designator[NumDesigs]; | 
|  | 1761 | NumDesignators = NumDesigs; | 
|  | 1762 | for (unsigned I = 0; I != NumDesigs; ++I) | 
|  | 1763 | Designators[I] = Desigs[I]; | 
|  | 1764 | } | 
|  | 1765 |  | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1766 | SourceRange DesignatedInitExpr::getSourceRange() const { | 
|  | 1767 | SourceLocation StartLoc; | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 1768 | Designator &First = | 
|  | 1769 | *const_cast<DesignatedInitExpr*>(this)->designators_begin(); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1770 | if (First.isFieldDesignator()) { | 
| Douglas Gregor | 5c7c9cb | 2009-03-28 00:41:23 +0000 | [diff] [blame] | 1771 | if (GNUSyntax) | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1772 | StartLoc = SourceLocation::getFromRawEncoding(First.Field.FieldLoc); | 
|  | 1773 | else | 
|  | 1774 | StartLoc = SourceLocation::getFromRawEncoding(First.Field.DotLoc); | 
|  | 1775 | } else | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 1776 | StartLoc = | 
|  | 1777 | SourceLocation::getFromRawEncoding(First.ArrayOrRange.LBracketLoc); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1778 | return SourceRange(StartLoc, getInit()->getSourceRange().getEnd()); | 
|  | 1779 | } | 
|  | 1780 |  | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1781 | Expr *DesignatedInitExpr::getArrayIndex(const Designator& D) { | 
|  | 1782 | assert(D.Kind == Designator::ArrayDesignator && "Requires array designator"); | 
|  | 1783 | char* Ptr = static_cast<char*>(static_cast<void *>(this)); | 
|  | 1784 | Ptr += sizeof(DesignatedInitExpr); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1785 | Stmt **SubExprs = reinterpret_cast<Stmt**>(reinterpret_cast<void**>(Ptr)); | 
|  | 1786 | return cast<Expr>(*(SubExprs + D.ArrayOrRange.Index + 1)); | 
|  | 1787 | } | 
|  | 1788 |  | 
|  | 1789 | Expr *DesignatedInitExpr::getArrayRangeStart(const Designator& D) { | 
|  | 1790 | assert(D.Kind == Designator::ArrayRangeDesignator && | 
|  | 1791 | "Requires array range designator"); | 
|  | 1792 | char* Ptr = static_cast<char*>(static_cast<void *>(this)); | 
|  | 1793 | Ptr += sizeof(DesignatedInitExpr); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1794 | Stmt **SubExprs = reinterpret_cast<Stmt**>(reinterpret_cast<void**>(Ptr)); | 
|  | 1795 | return cast<Expr>(*(SubExprs + D.ArrayOrRange.Index + 1)); | 
|  | 1796 | } | 
|  | 1797 |  | 
|  | 1798 | Expr *DesignatedInitExpr::getArrayRangeEnd(const Designator& D) { | 
|  | 1799 | assert(D.Kind == Designator::ArrayRangeDesignator && | 
|  | 1800 | "Requires array range designator"); | 
|  | 1801 | char* Ptr = static_cast<char*>(static_cast<void *>(this)); | 
|  | 1802 | Ptr += sizeof(DesignatedInitExpr); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1803 | Stmt **SubExprs = reinterpret_cast<Stmt**>(reinterpret_cast<void**>(Ptr)); | 
|  | 1804 | return cast<Expr>(*(SubExprs + D.ArrayOrRange.Index + 2)); | 
|  | 1805 | } | 
|  | 1806 |  | 
| Douglas Gregor | d5846a1 | 2009-04-15 06:41:24 +0000 | [diff] [blame] | 1807 | /// \brief Replaces the designator at index @p Idx with the series | 
|  | 1808 | /// of designators in [First, Last). | 
|  | 1809 | void DesignatedInitExpr::ExpandDesignator(unsigned Idx, | 
|  | 1810 | const Designator *First, | 
|  | 1811 | const Designator *Last) { | 
|  | 1812 | unsigned NumNewDesignators = Last - First; | 
|  | 1813 | if (NumNewDesignators == 0) { | 
|  | 1814 | std::copy_backward(Designators + Idx + 1, | 
|  | 1815 | Designators + NumDesignators, | 
|  | 1816 | Designators + Idx); | 
|  | 1817 | --NumNewDesignators; | 
|  | 1818 | return; | 
|  | 1819 | } else if (NumNewDesignators == 1) { | 
|  | 1820 | Designators[Idx] = *First; | 
|  | 1821 | return; | 
|  | 1822 | } | 
|  | 1823 |  | 
|  | 1824 | Designator *NewDesignators | 
|  | 1825 | = new Designator[NumDesignators - 1 + NumNewDesignators]; | 
|  | 1826 | std::copy(Designators, Designators + Idx, NewDesignators); | 
|  | 1827 | std::copy(First, Last, NewDesignators + Idx); | 
|  | 1828 | std::copy(Designators + Idx + 1, Designators + NumDesignators, | 
|  | 1829 | NewDesignators + Idx + NumNewDesignators); | 
|  | 1830 | delete [] Designators; | 
|  | 1831 | Designators = NewDesignators; | 
|  | 1832 | NumDesignators = NumDesignators - 1 + NumNewDesignators; | 
|  | 1833 | } | 
|  | 1834 |  | 
|  | 1835 | void DesignatedInitExpr::Destroy(ASTContext &C) { | 
|  | 1836 | delete [] Designators; | 
|  | 1837 | Expr::Destroy(C); | 
|  | 1838 | } | 
|  | 1839 |  | 
| Douglas Gregor | ca1aeec | 2009-05-21 23:17:49 +0000 | [diff] [blame] | 1840 | ImplicitValueInitExpr *ImplicitValueInitExpr::Clone(ASTContext &C) const { | 
|  | 1841 | return new (C) ImplicitValueInitExpr(getType()); | 
|  | 1842 | } | 
|  | 1843 |  | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 1844 | //===----------------------------------------------------------------------===// | 
| Ted Kremenek | 5778acf | 2008-10-27 18:40:21 +0000 | [diff] [blame] | 1845 | //  ExprIterator. | 
|  | 1846 | //===----------------------------------------------------------------------===// | 
|  | 1847 |  | 
|  | 1848 | Expr* ExprIterator::operator[](size_t idx) { return cast<Expr>(I[idx]); } | 
|  | 1849 | Expr* ExprIterator::operator*() const { return cast<Expr>(*I); } | 
|  | 1850 | Expr* ExprIterator::operator->() const { return cast<Expr>(*I); } | 
|  | 1851 | const Expr* ConstExprIterator::operator[](size_t idx) const { | 
|  | 1852 | return cast<Expr>(I[idx]); | 
|  | 1853 | } | 
|  | 1854 | const Expr* ConstExprIterator::operator*() const { return cast<Expr>(*I); } | 
|  | 1855 | const Expr* ConstExprIterator::operator->() const { return cast<Expr>(*I); } | 
|  | 1856 |  | 
|  | 1857 | //===----------------------------------------------------------------------===// | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1858 | //  Child Iterators for iterating over subexpressions/substatements | 
|  | 1859 | //===----------------------------------------------------------------------===// | 
|  | 1860 |  | 
|  | 1861 | // DeclRefExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1862 | Stmt::child_iterator DeclRefExpr::child_begin() { return child_iterator(); } | 
|  | 1863 | Stmt::child_iterator DeclRefExpr::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1864 |  | 
| Steve Naroff | e46504b | 2007-11-12 14:29:37 +0000 | [diff] [blame] | 1865 | // ObjCIvarRefExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1866 | Stmt::child_iterator ObjCIvarRefExpr::child_begin() { return &Base; } | 
|  | 1867 | Stmt::child_iterator ObjCIvarRefExpr::child_end() { return &Base+1; } | 
| Steve Naroff | e46504b | 2007-11-12 14:29:37 +0000 | [diff] [blame] | 1868 |  | 
| Steve Naroff | ebf4cb4 | 2008-06-02 23:03:37 +0000 | [diff] [blame] | 1869 | // ObjCPropertyRefExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1870 | Stmt::child_iterator ObjCPropertyRefExpr::child_begin() { return &Base; } | 
|  | 1871 | Stmt::child_iterator ObjCPropertyRefExpr::child_end() { return &Base+1; } | 
| Steve Naroff | ec94403 | 2008-05-30 00:40:33 +0000 | [diff] [blame] | 1872 |  | 
| Fariborz Jahanian | 8a1810f | 2008-11-22 18:39:36 +0000 | [diff] [blame] | 1873 | // ObjCKVCRefExpr | 
|  | 1874 | Stmt::child_iterator ObjCKVCRefExpr::child_begin() { return &Base; } | 
|  | 1875 | Stmt::child_iterator ObjCKVCRefExpr::child_end() { return &Base+1; } | 
|  | 1876 |  | 
| Douglas Gregor | 8ea1f53 | 2008-11-04 14:56:14 +0000 | [diff] [blame] | 1877 | // ObjCSuperExpr | 
|  | 1878 | Stmt::child_iterator ObjCSuperExpr::child_begin() { return child_iterator(); } | 
|  | 1879 | Stmt::child_iterator ObjCSuperExpr::child_end() { return child_iterator(); } | 
|  | 1880 |  | 
| Chris Lattner | 6307f19 | 2008-08-10 01:53:14 +0000 | [diff] [blame] | 1881 | // PredefinedExpr | 
|  | 1882 | Stmt::child_iterator PredefinedExpr::child_begin() { return child_iterator(); } | 
|  | 1883 | Stmt::child_iterator PredefinedExpr::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1884 |  | 
|  | 1885 | // IntegerLiteral | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1886 | Stmt::child_iterator IntegerLiteral::child_begin() { return child_iterator(); } | 
|  | 1887 | Stmt::child_iterator IntegerLiteral::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1888 |  | 
|  | 1889 | // CharacterLiteral | 
| Chris Lattner | 8ba2247 | 2009-02-16 22:33:34 +0000 | [diff] [blame] | 1890 | Stmt::child_iterator CharacterLiteral::child_begin() { return child_iterator();} | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1891 | Stmt::child_iterator CharacterLiteral::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1892 |  | 
|  | 1893 | // FloatingLiteral | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1894 | Stmt::child_iterator FloatingLiteral::child_begin() { return child_iterator(); } | 
|  | 1895 | Stmt::child_iterator FloatingLiteral::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1896 |  | 
| Chris Lattner | 1c20a17 | 2007-08-26 03:42:43 +0000 | [diff] [blame] | 1897 | // ImaginaryLiteral | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1898 | Stmt::child_iterator ImaginaryLiteral::child_begin() { return &Val; } | 
|  | 1899 | Stmt::child_iterator ImaginaryLiteral::child_end() { return &Val+1; } | 
| Chris Lattner | 1c20a17 | 2007-08-26 03:42:43 +0000 | [diff] [blame] | 1900 |  | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1901 | // StringLiteral | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1902 | Stmt::child_iterator StringLiteral::child_begin() { return child_iterator(); } | 
|  | 1903 | Stmt::child_iterator StringLiteral::child_end() { return child_iterator(); } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1904 |  | 
|  | 1905 | // ParenExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1906 | Stmt::child_iterator ParenExpr::child_begin() { return &Val; } | 
|  | 1907 | Stmt::child_iterator ParenExpr::child_end() { return &Val+1; } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1908 |  | 
|  | 1909 | // UnaryOperator | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1910 | Stmt::child_iterator UnaryOperator::child_begin() { return &Val; } | 
|  | 1911 | Stmt::child_iterator UnaryOperator::child_end() { return &Val+1; } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1912 |  | 
| Sebastian Redl | 6f28289 | 2008-11-11 17:56:53 +0000 | [diff] [blame] | 1913 | // SizeOfAlignOfExpr | 
|  | 1914 | Stmt::child_iterator SizeOfAlignOfExpr::child_begin() { | 
|  | 1915 | // If this is of a type and the type is a VLA type (and not a typedef), the | 
|  | 1916 | // size expression of the VLA needs to be treated as an executable expression. | 
|  | 1917 | // Why isn't this weirdness documented better in StmtIterator? | 
|  | 1918 | if (isArgumentType()) { | 
|  | 1919 | if (VariableArrayType* T = dyn_cast<VariableArrayType>( | 
|  | 1920 | getArgumentType().getTypePtr())) | 
|  | 1921 | return child_iterator(T); | 
|  | 1922 | return child_iterator(); | 
|  | 1923 | } | 
| Sebastian Redl | ba3fdfc | 2008-12-03 23:17:54 +0000 | [diff] [blame] | 1924 | return child_iterator(&Argument.Ex); | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1925 | } | 
| Sebastian Redl | 6f28289 | 2008-11-11 17:56:53 +0000 | [diff] [blame] | 1926 | Stmt::child_iterator SizeOfAlignOfExpr::child_end() { | 
|  | 1927 | if (isArgumentType()) | 
|  | 1928 | return child_iterator(); | 
| Sebastian Redl | ba3fdfc | 2008-12-03 23:17:54 +0000 | [diff] [blame] | 1929 | return child_iterator(&Argument.Ex + 1); | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1930 | } | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1931 |  | 
|  | 1932 | // ArraySubscriptExpr | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1933 | Stmt::child_iterator ArraySubscriptExpr::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1934 | return &SubExprs[0]; | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1935 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1936 | Stmt::child_iterator ArraySubscriptExpr::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1937 | return &SubExprs[0]+END_EXPR; | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1938 | } | 
|  | 1939 |  | 
|  | 1940 | // CallExpr | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1941 | Stmt::child_iterator CallExpr::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1942 | return &SubExprs[0]; | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1943 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1944 | Stmt::child_iterator CallExpr::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1945 | return &SubExprs[0]+NumArgs+ARGS_START; | 
| Ted Kremenek | 85e92ec | 2007-08-24 18:13:47 +0000 | [diff] [blame] | 1946 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1947 |  | 
|  | 1948 | // MemberExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1949 | Stmt::child_iterator MemberExpr::child_begin() { return &Base; } | 
|  | 1950 | Stmt::child_iterator MemberExpr::child_end() { return &Base+1; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1951 |  | 
| Nate Begeman | ce4d7fc | 2008-04-18 23:10:10 +0000 | [diff] [blame] | 1952 | // ExtVectorElementExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1953 | Stmt::child_iterator ExtVectorElementExpr::child_begin() { return &Base; } | 
|  | 1954 | Stmt::child_iterator ExtVectorElementExpr::child_end() { return &Base+1; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1955 |  | 
|  | 1956 | // CompoundLiteralExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1957 | Stmt::child_iterator CompoundLiteralExpr::child_begin() { return &Init; } | 
|  | 1958 | Stmt::child_iterator CompoundLiteralExpr::child_end() { return &Init+1; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1959 |  | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1960 | // CastExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1961 | Stmt::child_iterator CastExpr::child_begin() { return &Op; } | 
|  | 1962 | Stmt::child_iterator CastExpr::child_end() { return &Op+1; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1963 |  | 
|  | 1964 | // BinaryOperator | 
|  | 1965 | Stmt::child_iterator BinaryOperator::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1966 | return &SubExprs[0]; | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1967 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1968 | Stmt::child_iterator BinaryOperator::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1969 | return &SubExprs[0]+END_EXPR; | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1970 | } | 
|  | 1971 |  | 
|  | 1972 | // ConditionalOperator | 
|  | 1973 | Stmt::child_iterator ConditionalOperator::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1974 | return &SubExprs[0]; | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1975 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1976 | Stmt::child_iterator ConditionalOperator::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1977 | return &SubExprs[0]+END_EXPR; | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1978 | } | 
|  | 1979 |  | 
|  | 1980 | // AddrLabelExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1981 | Stmt::child_iterator AddrLabelExpr::child_begin() { return child_iterator(); } | 
|  | 1982 | Stmt::child_iterator AddrLabelExpr::child_end() { return child_iterator(); } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1983 |  | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1984 | // StmtExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1985 | Stmt::child_iterator StmtExpr::child_begin() { return &SubStmt; } | 
|  | 1986 | Stmt::child_iterator StmtExpr::child_end() { return &SubStmt+1; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1987 |  | 
|  | 1988 | // TypesCompatibleExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 1989 | Stmt::child_iterator TypesCompatibleExpr::child_begin() { | 
|  | 1990 | return child_iterator(); | 
|  | 1991 | } | 
|  | 1992 |  | 
|  | 1993 | Stmt::child_iterator TypesCompatibleExpr::child_end() { | 
|  | 1994 | return child_iterator(); | 
|  | 1995 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 1996 |  | 
|  | 1997 | // ChooseExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 1998 | Stmt::child_iterator ChooseExpr::child_begin() { return &SubExprs[0]; } | 
|  | 1999 | Stmt::child_iterator ChooseExpr::child_end() { return &SubExprs[0]+END_EXPR; } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 2000 |  | 
| Douglas Gregor | 3be4b12 | 2008-11-29 04:51:27 +0000 | [diff] [blame] | 2001 | // GNUNullExpr | 
|  | 2002 | Stmt::child_iterator GNUNullExpr::child_begin() { return child_iterator(); } | 
|  | 2003 | Stmt::child_iterator GNUNullExpr::child_end() { return child_iterator(); } | 
|  | 2004 |  | 
| Eli Friedman | a1b4ed8 | 2008-05-14 19:38:39 +0000 | [diff] [blame] | 2005 | // ShuffleVectorExpr | 
|  | 2006 | Stmt::child_iterator ShuffleVectorExpr::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2007 | return &SubExprs[0]; | 
| Eli Friedman | a1b4ed8 | 2008-05-14 19:38:39 +0000 | [diff] [blame] | 2008 | } | 
|  | 2009 | Stmt::child_iterator ShuffleVectorExpr::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2010 | return &SubExprs[0]+NumExprs; | 
| Eli Friedman | a1b4ed8 | 2008-05-14 19:38:39 +0000 | [diff] [blame] | 2011 | } | 
|  | 2012 |  | 
| Anders Carlsson | 7e13ab8 | 2007-10-15 20:28:48 +0000 | [diff] [blame] | 2013 | // VAArgExpr | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2014 | Stmt::child_iterator VAArgExpr::child_begin() { return &Val; } | 
|  | 2015 | Stmt::child_iterator VAArgExpr::child_end() { return &Val+1; } | 
| Anders Carlsson | 7e13ab8 | 2007-10-15 20:28:48 +0000 | [diff] [blame] | 2016 |  | 
| Anders Carlsson | 4692db0 | 2007-08-31 04:56:16 +0000 | [diff] [blame] | 2017 | // InitListExpr | 
|  | 2018 | Stmt::child_iterator InitListExpr::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2019 | return InitExprs.size() ? &InitExprs[0] : 0; | 
| Anders Carlsson | 4692db0 | 2007-08-31 04:56:16 +0000 | [diff] [blame] | 2020 | } | 
|  | 2021 | Stmt::child_iterator InitListExpr::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2022 | return InitExprs.size() ? &InitExprs[0] + InitExprs.size() : 0; | 
| Anders Carlsson | 4692db0 | 2007-08-31 04:56:16 +0000 | [diff] [blame] | 2023 | } | 
|  | 2024 |  | 
| Douglas Gregor | 0202cb4 | 2009-01-29 17:44:32 +0000 | [diff] [blame] | 2025 | // DesignatedInitExpr | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 2026 | Stmt::child_iterator DesignatedInitExpr::child_begin() { | 
|  | 2027 | char* Ptr = static_cast<char*>(static_cast<void *>(this)); | 
|  | 2028 | Ptr += sizeof(DesignatedInitExpr); | 
| Douglas Gregor | e4a0bb7 | 2009-01-22 00:58:24 +0000 | [diff] [blame] | 2029 | return reinterpret_cast<Stmt**>(reinterpret_cast<void**>(Ptr)); | 
|  | 2030 | } | 
|  | 2031 | Stmt::child_iterator DesignatedInitExpr::child_end() { | 
|  | 2032 | return child_iterator(&*child_begin() + NumSubExprs); | 
|  | 2033 | } | 
|  | 2034 |  | 
| Douglas Gregor | 0202cb4 | 2009-01-29 17:44:32 +0000 | [diff] [blame] | 2035 | // ImplicitValueInitExpr | 
|  | 2036 | Stmt::child_iterator ImplicitValueInitExpr::child_begin() { | 
|  | 2037 | return child_iterator(); | 
|  | 2038 | } | 
|  | 2039 |  | 
|  | 2040 | Stmt::child_iterator ImplicitValueInitExpr::child_end() { | 
|  | 2041 | return child_iterator(); | 
|  | 2042 | } | 
|  | 2043 |  | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 2044 | // ObjCStringLiteral | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2045 | Stmt::child_iterator ObjCStringLiteral::child_begin() { | 
| Chris Lattner | 112c2a9 | 2009-02-18 06:53:08 +0000 | [diff] [blame] | 2046 | return &String; | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2047 | } | 
|  | 2048 | Stmt::child_iterator ObjCStringLiteral::child_end() { | 
| Chris Lattner | 112c2a9 | 2009-02-18 06:53:08 +0000 | [diff] [blame] | 2049 | return &String+1; | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2050 | } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 2051 |  | 
|  | 2052 | // ObjCEncodeExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2053 | Stmt::child_iterator ObjCEncodeExpr::child_begin() { return child_iterator(); } | 
|  | 2054 | Stmt::child_iterator ObjCEncodeExpr::child_end() { return child_iterator(); } | 
| Ted Kremenek | 23702b6 | 2007-08-24 20:06:47 +0000 | [diff] [blame] | 2055 |  | 
| Fariborz Jahanian | 4bef462 | 2007-10-16 20:40:23 +0000 | [diff] [blame] | 2056 | // ObjCSelectorExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2057 | Stmt::child_iterator ObjCSelectorExpr::child_begin() { | 
|  | 2058 | return child_iterator(); | 
|  | 2059 | } | 
|  | 2060 | Stmt::child_iterator ObjCSelectorExpr::child_end() { | 
|  | 2061 | return child_iterator(); | 
|  | 2062 | } | 
| Fariborz Jahanian | 4bef462 | 2007-10-16 20:40:23 +0000 | [diff] [blame] | 2063 |  | 
| Fariborz Jahanian | a32aaef | 2007-10-17 16:58:11 +0000 | [diff] [blame] | 2064 | // ObjCProtocolExpr | 
| Ted Kremenek | 04746ce | 2007-10-18 23:28:49 +0000 | [diff] [blame] | 2065 | Stmt::child_iterator ObjCProtocolExpr::child_begin() { | 
|  | 2066 | return child_iterator(); | 
|  | 2067 | } | 
|  | 2068 | Stmt::child_iterator ObjCProtocolExpr::child_end() { | 
|  | 2069 | return child_iterator(); | 
|  | 2070 | } | 
| Fariborz Jahanian | a32aaef | 2007-10-17 16:58:11 +0000 | [diff] [blame] | 2071 |  | 
| Steve Naroff | d54978b | 2007-09-18 23:55:05 +0000 | [diff] [blame] | 2072 | // ObjCMessageExpr | 
| Ted Kremenek | b8861a6 | 2008-05-01 17:26:20 +0000 | [diff] [blame] | 2073 | Stmt::child_iterator ObjCMessageExpr::child_begin() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2074 | return getReceiver() ? &SubExprs[0] : &SubExprs[0] + ARGS_START; | 
| Steve Naroff | d54978b | 2007-09-18 23:55:05 +0000 | [diff] [blame] | 2075 | } | 
|  | 2076 | Stmt::child_iterator ObjCMessageExpr::child_end() { | 
| Ted Kremenek | 08e1711 | 2008-06-17 02:43:46 +0000 | [diff] [blame] | 2077 | return &SubExprs[0]+ARGS_START+getNumArgs(); | 
| Steve Naroff | d54978b | 2007-09-18 23:55:05 +0000 | [diff] [blame] | 2078 | } | 
|  | 2079 |  | 
| Steve Naroff | c540d66 | 2008-09-03 18:15:37 +0000 | [diff] [blame] | 2080 | // Blocks | 
| Steve Naroff | 415d3d5 | 2008-10-08 17:01:13 +0000 | [diff] [blame] | 2081 | Stmt::child_iterator BlockExpr::child_begin() { return child_iterator(); } | 
|  | 2082 | Stmt::child_iterator BlockExpr::child_end() { return child_iterator(); } | 
| Steve Naroff | c540d66 | 2008-09-03 18:15:37 +0000 | [diff] [blame] | 2083 |  | 
| Ted Kremenek | 8bafa2c | 2008-09-26 23:24:14 +0000 | [diff] [blame] | 2084 | Stmt::child_iterator BlockDeclRefExpr::child_begin() { return child_iterator();} | 
|  | 2085 | Stmt::child_iterator BlockDeclRefExpr::child_end() { return child_iterator(); } |