2007-07-11 21:01:13 +04:00
|
|
|
//===--- ExprCXX.h - Classes for representing expressions -------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-29 22:59:25 +03:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-07-11 21:01:13 +04:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines the Expr interface and subclasses for C++ expressions.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_CLANG_AST_EXPRCXX_H
|
|
|
|
#define LLVM_CLANG_AST_EXPRCXX_H
|
|
|
|
|
|
|
|
#include "clang/AST/Expr.h"
|
2008-08-11 09:35:13 +04:00
|
|
|
#include "clang/AST/Decl.h"
|
2007-07-11 21:01:13 +04:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
|
2008-11-21 22:14:01 +03:00
|
|
|
class CXXConstructorDecl;
|
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
// C++ Expressions.
|
|
|
|
//===--------------------------------------------------------------------===//
|
|
|
|
|
2008-11-14 19:09:21 +03:00
|
|
|
/// CXXOperatorCallExpr - Represents a call to an overloaded operator
|
|
|
|
/// written using operator syntax, e.g., "x + y" or "*p". While
|
|
|
|
/// semantically equivalent to a normal call, this AST node provides
|
|
|
|
/// better information about the syntactic representation of the call.
|
|
|
|
class CXXOperatorCallExpr : public CallExpr {
|
|
|
|
public:
|
|
|
|
CXXOperatorCallExpr(Expr *fn, Expr **args, unsigned numargs, QualType t,
|
|
|
|
SourceLocation operatorloc)
|
|
|
|
: CallExpr(CXXOperatorCallExprClass, fn, args, numargs, t, operatorloc) { }
|
|
|
|
|
|
|
|
/// getOperator - Returns the kind of overloaded operator that this
|
|
|
|
/// expression refers to.
|
|
|
|
OverloadedOperatorKind getOperator() const;
|
|
|
|
|
|
|
|
/// getOperatorLoc - Returns the location of the operator symbol in
|
|
|
|
/// the expression. When @c getOperator()==OO_Call, this is the
|
|
|
|
/// location of the right parentheses; when @c
|
|
|
|
/// getOperator()==OO_Subscript, this is the location of the right
|
|
|
|
/// bracket.
|
|
|
|
SourceLocation getOperatorLoc() const { return getRParenLoc(); }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const;
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXOperatorCallExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXOperatorCallExpr *) { return true; }
|
|
|
|
};
|
|
|
|
|
2008-10-27 22:41:14 +03:00
|
|
|
/// CXXNamedCastExpr - Abstract class common to all of the C++ "named"
|
|
|
|
/// casts, @c static_cast, @c dynamic_cast, @c reinterpret_cast, or @c
|
|
|
|
/// const_cast.
|
|
|
|
///
|
|
|
|
/// This abstract class is inherited by all of the classes
|
|
|
|
/// representing "named" casts, e.g., CXXStaticCastExpr,
|
|
|
|
/// CXXDynamicCastExpr, CXXReinterpretCastExpr, and CXXConstCastExpr.
|
|
|
|
class CXXNamedCastExpr : public ExplicitCastExpr {
|
2008-06-17 07:11:08 +04:00
|
|
|
private:
|
|
|
|
SourceLocation Loc; // the location of the casting op
|
2008-10-27 22:41:14 +03:00
|
|
|
|
|
|
|
protected:
|
|
|
|
CXXNamedCastExpr(StmtClass SC, QualType ty, Expr *op, QualType writtenTy,
|
|
|
|
SourceLocation l)
|
|
|
|
: ExplicitCastExpr(SC, ty, op, writtenTy), Loc(l) {}
|
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
public:
|
2008-10-27 22:41:14 +03:00
|
|
|
const char *getCastName() const;
|
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(Loc, getSubExpr()->getSourceRange().getEnd());
|
|
|
|
}
|
|
|
|
static bool classof(const Stmt *T) {
|
2008-10-27 22:41:14 +03:00
|
|
|
switch (T->getStmtClass()) {
|
|
|
|
case CXXNamedCastExprClass:
|
|
|
|
case CXXStaticCastExprClass:
|
|
|
|
case CXXDynamicCastExprClass:
|
|
|
|
case CXXReinterpretCastExprClass:
|
|
|
|
case CXXConstCastExprClass:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
2008-06-17 07:11:08 +04:00
|
|
|
}
|
2008-10-27 22:41:14 +03:00
|
|
|
static bool classof(const CXXNamedCastExpr *) { return true; }
|
2008-06-17 07:11:08 +04:00
|
|
|
|
2008-10-27 22:41:14 +03:00
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXNamedCastExpr *CreateImpl(llvm::Deserializer& D, ASTContext& C,
|
|
|
|
StmtClass SC);
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXStaticCastExpr - A C++ @c static_cast expression (C++ [expr.static.cast]).
|
|
|
|
///
|
|
|
|
/// This expression node represents a C++ static cast, e.g.,
|
|
|
|
/// @c static_cast<int>(1.0).
|
|
|
|
class CXXStaticCastExpr : public CXXNamedCastExpr {
|
|
|
|
public:
|
|
|
|
CXXStaticCastExpr(QualType ty, Expr *op, QualType writtenTy, SourceLocation l)
|
|
|
|
: CXXNamedCastExpr(CXXStaticCastExprClass, ty, op, writtenTy, l) {}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXStaticCastExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXStaticCastExpr *) { return true; }
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXDynamicCastExpr - A C++ @c dynamic_cast expression
|
|
|
|
/// (C++ [expr.dynamic.cast]), which may perform a run-time check to
|
|
|
|
/// determine how to perform the type cast.
|
|
|
|
///
|
|
|
|
/// This expression node represents a dynamic cast, e.g.,
|
|
|
|
/// @c dynamic_cast<Derived*>(BasePtr).
|
|
|
|
class CXXDynamicCastExpr : public CXXNamedCastExpr {
|
|
|
|
public:
|
|
|
|
CXXDynamicCastExpr(QualType ty, Expr *op, QualType writtenTy, SourceLocation l)
|
|
|
|
: CXXNamedCastExpr(CXXDynamicCastExprClass, ty, op, writtenTy, l) {}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXDynamicCastExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXDynamicCastExpr *) { return true; }
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXReinterpretCastExpr - A C++ @c reinterpret_cast expression (C++
|
|
|
|
/// [expr.reinterpret.cast]), which provides a differently-typed view
|
|
|
|
/// of a value but performs no actual work at run time.
|
|
|
|
///
|
|
|
|
/// This expression node represents a reinterpret cast, e.g.,
|
|
|
|
/// @c reinterpret_cast<int>(VoidPtr).
|
|
|
|
class CXXReinterpretCastExpr : public CXXNamedCastExpr {
|
|
|
|
public:
|
|
|
|
CXXReinterpretCastExpr(QualType ty, Expr *op, QualType writtenTy,
|
|
|
|
SourceLocation l)
|
|
|
|
: CXXNamedCastExpr(CXXReinterpretCastExprClass, ty, op, writtenTy, l) {}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXReinterpretCastExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXReinterpretCastExpr *) { return true; }
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXConstCastExpr - A C++ @c const_cast expression (C++ [expr.const.cast]),
|
|
|
|
/// which can remove type qualifiers but does not change the underlying value.
|
|
|
|
///
|
|
|
|
/// This expression node represents a const cast, e.g.,
|
|
|
|
/// @c const_cast<char*>(PtrToConstChar).
|
|
|
|
class CXXConstCastExpr : public CXXNamedCastExpr {
|
|
|
|
public:
|
|
|
|
CXXConstCastExpr(QualType ty, Expr *op, QualType writtenTy,
|
|
|
|
SourceLocation l)
|
|
|
|
: CXXNamedCastExpr(CXXConstCastExprClass, ty, op, writtenTy, l) {}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXConstCastExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXConstCastExpr *) { return true; }
|
2008-06-17 07:11:08 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXBoolLiteralExpr - [C++ 2.13.5] C++ Boolean Literal.
|
|
|
|
///
|
|
|
|
class CXXBoolLiteralExpr : public Expr {
|
|
|
|
bool Value;
|
|
|
|
SourceLocation Loc;
|
|
|
|
public:
|
|
|
|
CXXBoolLiteralExpr(bool val, QualType Ty, SourceLocation l) :
|
|
|
|
Expr(CXXBoolLiteralExprClass, Ty), Value(val), Loc(l) {}
|
|
|
|
|
|
|
|
bool getValue() const { return Value; }
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
virtual SourceRange getSourceRange() const { return SourceRange(Loc); }
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXBoolLiteralExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXBoolLiteralExpr *) { return true; }
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
};
|
|
|
|
|
2008-11-11 14:37:55 +03:00
|
|
|
/// CXXTypeidExpr - A C++ @c typeid expression (C++ [expr.typeid]), which gets
|
|
|
|
/// the type_info that corresponds to the supplied type, or the (possibly
|
|
|
|
/// dynamic) type of the supplied expression.
|
|
|
|
///
|
|
|
|
/// This represents code like @c typeid(int) or @c typeid(*objPtr)
|
|
|
|
class CXXTypeidExpr : public Expr {
|
|
|
|
private:
|
|
|
|
bool isTypeOp : 1;
|
|
|
|
void *Operand;
|
|
|
|
SourceRange Range;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CXXTypeidExpr(bool isTypeOp, void *op, QualType Ty, const SourceRange r) :
|
|
|
|
Expr(CXXTypeidExprClass, Ty), isTypeOp(isTypeOp), Operand(op), Range(r) {}
|
|
|
|
|
|
|
|
bool isTypeOperand() const { return isTypeOp; }
|
|
|
|
QualType getTypeOperand() const {
|
|
|
|
assert(isTypeOperand() && "Cannot call getTypeOperand for typeid(expr)");
|
|
|
|
return QualType::getFromOpaquePtr(Operand);
|
|
|
|
}
|
|
|
|
Expr* getExprOperand() const {
|
|
|
|
assert(!isTypeOperand() && "Cannot call getExprOperand for typeid(type)");
|
|
|
|
return static_cast<Expr*>(Operand);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return Range;
|
|
|
|
}
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXTypeidExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXTypeidExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXTypeidExpr* CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
2008-11-04 17:32:21 +03:00
|
|
|
/// CXXThisExpr - Represents the "this" expression in C++, which is a
|
|
|
|
/// pointer to the object on which the current member function is
|
|
|
|
/// executing (C++ [expr.prim]p3). Example:
|
|
|
|
///
|
|
|
|
/// @code
|
|
|
|
/// class Foo {
|
|
|
|
/// public:
|
|
|
|
/// void bar();
|
|
|
|
/// void test() { this->bar(); }
|
|
|
|
/// };
|
|
|
|
/// @endcode
|
|
|
|
class CXXThisExpr : public Expr {
|
|
|
|
SourceLocation Loc;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CXXThisExpr(SourceLocation L, QualType Type)
|
|
|
|
: Expr(CXXThisExprClass, Type), Loc(L) { }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const { return SourceRange(Loc); }
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXThisExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXThisExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXThisExpr* CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
2008-06-17 07:11:08 +04:00
|
|
|
/// CXXThrowExpr - [C++ 15] C++ Throw Expression. This handles
|
|
|
|
/// 'throw' and 'throw' assignment-expression. When
|
|
|
|
/// assignment-expression isn't present, Op will be null.
|
|
|
|
///
|
|
|
|
class CXXThrowExpr : public Expr {
|
|
|
|
Stmt *Op;
|
|
|
|
SourceLocation ThrowLoc;
|
|
|
|
public:
|
|
|
|
// Ty is the void type which is used as the result type of the
|
|
|
|
// exepression. The l is the location of the throw keyword. expr
|
|
|
|
// can by null, if the optional expression to throw isn't present.
|
|
|
|
CXXThrowExpr(Expr *expr, QualType Ty, SourceLocation l) :
|
|
|
|
Expr(CXXThrowExprClass, Ty), Op(expr), ThrowLoc(l) {}
|
|
|
|
const Expr *getSubExpr() const { return cast_or_null<Expr>(Op); }
|
|
|
|
Expr *getSubExpr() { return cast_or_null<Expr>(Op); }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
if (getSubExpr() == 0)
|
|
|
|
return SourceRange(ThrowLoc, ThrowLoc);
|
|
|
|
return SourceRange(ThrowLoc, getSubExpr()->getSourceRange().getEnd());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXThrowExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXThrowExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXDefaultArgExpr - C++ [dcl.fct.default]. This wraps up a
|
|
|
|
/// function call argument that was created from the corresponding
|
|
|
|
/// parameter's default argument, when the call did not explicitly
|
|
|
|
/// supply arguments for all of the parameters.
|
|
|
|
class CXXDefaultArgExpr : public Expr {
|
|
|
|
ParmVarDecl *Param;
|
|
|
|
public:
|
|
|
|
// Param is the parameter whose default argument is used by this
|
|
|
|
// expression.
|
|
|
|
explicit CXXDefaultArgExpr(ParmVarDecl *param)
|
|
|
|
: Expr(CXXDefaultArgExprClass, param->getDefaultArg()->getType()),
|
|
|
|
Param(param) { }
|
|
|
|
|
|
|
|
// Retrieve the parameter that the argument was created from.
|
|
|
|
const ParmVarDecl *getParam() const { return Param; }
|
|
|
|
ParmVarDecl *getParam() { return Param; }
|
|
|
|
|
|
|
|
// Retrieve the actual argument to the function call.
|
|
|
|
const Expr *getExpr() const { return Param->getDefaultArg(); }
|
|
|
|
Expr *getExpr() { return Param->getDefaultArg(); }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
// Default argument expressions have no representation in the
|
|
|
|
// source, so they have an empty source range.
|
|
|
|
return SourceRange();
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXDefaultArgExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXDefaultArgExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
// Serialization
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXDefaultArgExpr* CreateImpl(llvm::Deserializer& D,
|
|
|
|
ASTContext& C);
|
|
|
|
};
|
2008-08-22 19:38:55 +04:00
|
|
|
|
2008-10-27 22:41:14 +03:00
|
|
|
/// CXXFunctionalCastExpr - Represents an explicit C++ type conversion
|
|
|
|
/// that uses "functional" notion (C++ [expr.type.conv]). Example: @c
|
|
|
|
/// x = int(0.5);
|
|
|
|
class CXXFunctionalCastExpr : public ExplicitCastExpr {
|
2008-08-22 19:38:55 +04:00
|
|
|
SourceLocation TyBeginLoc;
|
|
|
|
SourceLocation RParenLoc;
|
|
|
|
public:
|
2008-10-27 22:41:14 +03:00
|
|
|
CXXFunctionalCastExpr(QualType ty, QualType writtenTy,
|
|
|
|
SourceLocation tyBeginLoc, Expr *castExpr,
|
2008-08-22 19:38:55 +04:00
|
|
|
SourceLocation rParenLoc) :
|
2008-10-27 22:41:14 +03:00
|
|
|
ExplicitCastExpr(CXXFunctionalCastExprClass, ty, castExpr, writtenTy),
|
2008-08-22 19:38:55 +04:00
|
|
|
TyBeginLoc(tyBeginLoc), RParenLoc(rParenLoc) {}
|
|
|
|
|
|
|
|
SourceLocation getTypeBeginLoc() const { return TyBeginLoc; }
|
|
|
|
SourceLocation getRParenLoc() const { return RParenLoc; }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(TyBeginLoc, RParenLoc);
|
|
|
|
}
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXFunctionalCastExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXFunctionalCastExpr *) { return true; }
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXFunctionalCastExpr *
|
|
|
|
CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXZeroInitValueExpr - [C++ 5.2.3p2]
|
|
|
|
/// Expression "T()" which creates a value-initialized Rvalue of non-class
|
|
|
|
/// type T.
|
|
|
|
///
|
|
|
|
class CXXZeroInitValueExpr : public Expr {
|
|
|
|
SourceLocation TyBeginLoc;
|
|
|
|
SourceLocation RParenLoc;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CXXZeroInitValueExpr(QualType ty, SourceLocation tyBeginLoc,
|
|
|
|
SourceLocation rParenLoc ) :
|
|
|
|
Expr(CXXZeroInitValueExprClass, ty),
|
|
|
|
TyBeginLoc(tyBeginLoc), RParenLoc(rParenLoc) {}
|
|
|
|
|
|
|
|
SourceLocation getTypeBeginLoc() const { return TyBeginLoc; }
|
|
|
|
SourceLocation getRParenLoc() const { return RParenLoc; }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(TyBeginLoc, RParenLoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXZeroInitValueExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXZeroInitValueExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXZeroInitValueExpr *
|
|
|
|
CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
2008-09-10 03:47:53 +04:00
|
|
|
/// CXXConditionDeclExpr - Condition declaration of a if/switch/while/for
|
|
|
|
/// statement, e.g: "if (int x = f()) {...}".
|
|
|
|
/// The main difference with DeclRefExpr is that CXXConditionDeclExpr owns the
|
|
|
|
/// decl that it references.
|
|
|
|
///
|
|
|
|
class CXXConditionDeclExpr : public DeclRefExpr {
|
|
|
|
public:
|
|
|
|
CXXConditionDeclExpr(SourceLocation startLoc,
|
|
|
|
SourceLocation eqLoc, VarDecl *var)
|
2008-10-28 03:22:11 +03:00
|
|
|
: DeclRefExpr(CXXConditionDeclExprClass, var,
|
|
|
|
var->getType().getNonReferenceType(), startLoc) {}
|
2008-09-10 03:47:53 +04:00
|
|
|
|
|
|
|
virtual void Destroy(ASTContext& Ctx);
|
|
|
|
|
|
|
|
SourceLocation getStartLoc() const { return getLocation(); }
|
|
|
|
|
|
|
|
VarDecl *getVarDecl() { return cast<VarDecl>(getDecl()); }
|
|
|
|
const VarDecl *getVarDecl() const { return cast<VarDecl>(getDecl()); }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(getStartLoc(), getVarDecl()->getInit()->getLocEnd());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXConditionDeclExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXConditionDeclExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
// FIXME: Implement these.
|
|
|
|
//virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
//static CXXConditionDeclExpr *
|
|
|
|
// CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
2008-11-21 22:14:01 +03:00
|
|
|
/// CXXNewExpr - A new expression for memory allocation and constructor calls,
|
|
|
|
/// e.g: "new CXXNewExpr(foo)".
|
|
|
|
class CXXNewExpr : public Expr {
|
|
|
|
// Was the usage ::new, i.e. is the global new to be used?
|
|
|
|
bool GlobalNew : 1;
|
|
|
|
// Was the form (type-id) used? Otherwise, it was new-type-id.
|
|
|
|
bool ParenTypeId : 1;
|
|
|
|
// Is there an initializer? If not, built-ins are uninitialized, else they're
|
|
|
|
// value-initialized.
|
|
|
|
bool Initializer : 1;
|
|
|
|
// The number of placement new arguments.
|
|
|
|
unsigned NumPlacementArgs : 14;
|
|
|
|
// The number of constructor arguments. This may be 1 even for non-class
|
|
|
|
// types; use the pseudo copy constructor.
|
|
|
|
unsigned NumConstructorArgs : 15;
|
|
|
|
// Contains any number of optional placement arguments, and any number of
|
|
|
|
// optional constructor arguments, in that order.
|
|
|
|
Stmt **SubExprs;
|
|
|
|
// Points to the allocation function used.
|
|
|
|
FunctionDecl *OperatorNew;
|
|
|
|
// Points to the deallocation function used in case of error. May be null.
|
|
|
|
FunctionDecl *OperatorDelete;
|
|
|
|
// Points to the constructor used. Cannot be null if AllocType is a record;
|
|
|
|
// it would still point at the default constructor (even an implicit one).
|
|
|
|
// Must be null for all other types.
|
|
|
|
CXXConstructorDecl *Constructor;
|
|
|
|
// The type to be allocated. Is either *ty or a VLA of that type.
|
|
|
|
QualType AllocType;
|
|
|
|
|
|
|
|
SourceLocation StartLoc;
|
|
|
|
SourceLocation EndLoc;
|
|
|
|
|
|
|
|
// Deserialization constructor
|
|
|
|
CXXNewExpr(QualType ty, QualType alloc, bool globalNew, bool parenTypeId,
|
|
|
|
bool initializer, unsigned numPlacementArgs,
|
|
|
|
unsigned numConstructorArgs, Stmt **subExprs,
|
|
|
|
FunctionDecl *operatorNew, FunctionDecl *operatorDelete,
|
|
|
|
CXXConstructorDecl *constructor, SourceLocation startLoc,
|
|
|
|
SourceLocation endLoc)
|
|
|
|
: Expr(CXXNewExprClass, ty), GlobalNew(globalNew), ParenTypeId(parenTypeId),
|
|
|
|
Initializer(initializer), NumPlacementArgs(numPlacementArgs),
|
|
|
|
NumConstructorArgs(numConstructorArgs), SubExprs(subExprs),
|
|
|
|
OperatorNew(operatorNew), OperatorDelete(operatorDelete),
|
|
|
|
Constructor(constructor), AllocType(alloc),
|
|
|
|
StartLoc(startLoc), EndLoc(endLoc)
|
|
|
|
{ }
|
|
|
|
public:
|
|
|
|
CXXNewExpr(bool globalNew, FunctionDecl *operatorNew, Expr **placementArgs,
|
|
|
|
unsigned numPlaceArgs, bool ParenTypeId, QualType alloc,
|
|
|
|
CXXConstructorDecl *constructor, bool initializer,
|
|
|
|
Expr **constructorArgs, unsigned numConsArgs,
|
|
|
|
FunctionDecl *operatorDelete, QualType ty,
|
|
|
|
SourceLocation startLoc, SourceLocation endLoc);
|
|
|
|
~CXXNewExpr() {
|
|
|
|
delete[] SubExprs;
|
|
|
|
}
|
|
|
|
|
|
|
|
QualType getAllocatedType() const { return AllocType; }
|
|
|
|
|
|
|
|
FunctionDecl *getOperatorNew() const { return OperatorNew; }
|
|
|
|
FunctionDecl *getOperatorDelete() const { return OperatorDelete; }
|
|
|
|
CXXConstructorDecl *getConstructor() const { return Constructor; }
|
|
|
|
|
|
|
|
unsigned getNumPlacementArgs() const { return NumPlacementArgs; }
|
|
|
|
Expr *getPlacementArg(unsigned i) {
|
|
|
|
assert(i < NumPlacementArgs && "Index out of range");
|
|
|
|
return cast<Expr>(SubExprs[i]);
|
|
|
|
}
|
|
|
|
const Expr *getPlacementArg(unsigned i) const {
|
|
|
|
assert(i < NumPlacementArgs && "Index out of range");
|
|
|
|
return cast<Expr>(SubExprs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isGlobalNew() const { return GlobalNew; }
|
|
|
|
bool isParenTypeId() const { return ParenTypeId; }
|
|
|
|
bool hasInitializer() const { return Initializer; }
|
|
|
|
|
|
|
|
unsigned getNumConstructorArgs() const { return NumConstructorArgs; }
|
|
|
|
Expr *getConstructorArg(unsigned i) {
|
|
|
|
assert(i < NumConstructorArgs && "Index out of range");
|
|
|
|
return cast<Expr>(SubExprs[NumPlacementArgs + i]);
|
|
|
|
}
|
|
|
|
const Expr *getConstructorArg(unsigned i) const {
|
|
|
|
assert(i < NumConstructorArgs && "Index out of range");
|
|
|
|
return cast<Expr>(SubExprs[NumPlacementArgs + i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef ExprIterator arg_iterator;
|
|
|
|
typedef ConstExprIterator const_arg_iterator;
|
|
|
|
|
|
|
|
arg_iterator placement_arg_begin() {
|
|
|
|
return SubExprs;
|
|
|
|
}
|
|
|
|
arg_iterator placement_arg_end() {
|
|
|
|
return SubExprs + getNumPlacementArgs();
|
|
|
|
}
|
|
|
|
const_arg_iterator placement_arg_begin() const {
|
|
|
|
return SubExprs;
|
|
|
|
}
|
|
|
|
const_arg_iterator placement_arg_end() const {
|
|
|
|
return SubExprs + getNumPlacementArgs();
|
|
|
|
}
|
|
|
|
|
|
|
|
arg_iterator constructor_arg_begin() {
|
|
|
|
return SubExprs + getNumPlacementArgs();
|
|
|
|
}
|
|
|
|
arg_iterator constructor_arg_end() {
|
|
|
|
return SubExprs + getNumPlacementArgs() + getNumConstructorArgs();
|
|
|
|
}
|
|
|
|
const_arg_iterator constructor_arg_begin() const {
|
|
|
|
return SubExprs + getNumPlacementArgs();
|
|
|
|
}
|
|
|
|
const_arg_iterator constructor_arg_end() const {
|
|
|
|
return SubExprs + getNumPlacementArgs() + getNumConstructorArgs();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(StartLoc, EndLoc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXNewExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXNewExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXNewExpr* CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
|
|
|
/// CXXDeleteExpr - A delete expression for memory deallocation and destructor
|
|
|
|
/// calls, e.g. "delete[] pArray".
|
|
|
|
class CXXDeleteExpr : public Expr {
|
|
|
|
// Is this a forced global delete, i.e. "::delete"?
|
|
|
|
bool GlobalDelete : 1;
|
|
|
|
// Is this the array form of delete, i.e. "delete[]"?
|
|
|
|
bool ArrayForm : 1;
|
|
|
|
// Points to the operator delete overload that is used. Could be a member.
|
|
|
|
FunctionDecl *OperatorDelete;
|
|
|
|
// The pointer expression to be deleted.
|
|
|
|
Stmt *Argument;
|
|
|
|
// Location of the expression.
|
|
|
|
SourceLocation Loc;
|
|
|
|
public:
|
|
|
|
CXXDeleteExpr(QualType ty, bool globalDelete, bool arrayForm,
|
|
|
|
FunctionDecl *operatorDelete, Expr *arg, SourceLocation loc)
|
|
|
|
: Expr(CXXDeleteExprClass, ty), GlobalDelete(globalDelete),
|
|
|
|
ArrayForm(arrayForm), OperatorDelete(operatorDelete), Argument(arg),
|
|
|
|
Loc(loc) { }
|
|
|
|
|
|
|
|
bool isGlobalDelete() const { return GlobalDelete; }
|
|
|
|
bool isArrayForm() const { return ArrayForm; }
|
|
|
|
|
|
|
|
FunctionDecl *getOperatorDelete() const { return OperatorDelete; }
|
|
|
|
|
|
|
|
Expr *getArgument() { return cast<Expr>(Argument); }
|
|
|
|
const Expr *getArgument() const { return cast<Expr>(Argument); }
|
|
|
|
|
|
|
|
virtual SourceRange getSourceRange() const {
|
|
|
|
return SourceRange(Loc, Argument->getLocEnd());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool classof(const Stmt *T) {
|
|
|
|
return T->getStmtClass() == CXXDeleteExprClass;
|
|
|
|
}
|
|
|
|
static bool classof(const CXXDeleteExpr *) { return true; }
|
|
|
|
|
|
|
|
// Iterators
|
|
|
|
virtual child_iterator child_begin();
|
|
|
|
virtual child_iterator child_end();
|
|
|
|
|
|
|
|
virtual void EmitImpl(llvm::Serializer& S) const;
|
|
|
|
static CXXDeleteExpr * CreateImpl(llvm::Deserializer& D, ASTContext& C);
|
|
|
|
};
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
} // end namespace clang
|
|
|
|
|
|
|
|
#endif
|