/* +----------------------------------------------------------------------+ | HipHop for PHP | +----------------------------------------------------------------------+ | Copyright (c) 2010- Facebook, Inc. (http://www.facebook.com) | +----------------------------------------------------------------------+ | This source file is subject to version 3.01 of the PHP license, | | that is bundled with this package in the file LICENSE, and is | | available through the world-wide-web at the following url: | | http://www.php.net/license/3_01.txt | | If you did not receive a copy of the PHP license and are unable to | | obtain it through the world-wide-web, please send a note to | | license@php.net so we can mail you a copy immediately. | +----------------------------------------------------------------------+ */ #include #include #include #include using namespace HPHP; /////////////////////////////////////////////////////////////////////////////// // constructors/destructors static void setListKind(ExpressionPtr e) { if (e && e->is(Expression::KindOfExpressionList)) { ExpressionListPtr list = static_pointer_cast(e); list->setListKind(ExpressionList::ListKindComma); } } ForStatement::ForStatement (STATEMENT_CONSTRUCTOR_PARAMETERS, ExpressionPtr exp1, ExpressionPtr exp2, ExpressionPtr exp3, StatementPtr stmt) : LoopStatement(STATEMENT_CONSTRUCTOR_PARAMETER_VALUES(ForStatement)), m_exp1(exp1), m_exp2(exp2), m_exp3(exp3), m_stmt(stmt) { setListKind(m_exp1); setListKind(m_exp2); setListKind(m_exp3); } StatementPtr ForStatement::clone() { ForStatementPtr stmt(new ForStatement(*this)); stmt->m_exp1 = Clone(m_exp1); stmt->m_exp2 = Clone(m_exp2); stmt->m_exp3 = Clone(m_exp3); stmt->m_stmt = Clone(m_stmt); return stmt; } /////////////////////////////////////////////////////////////////////////////// // parser functions /////////////////////////////////////////////////////////////////////////////// // static analysis functions void ForStatement::analyzeProgram(AnalysisResultPtr ar) { if (m_exp1) m_exp1->analyzeProgram(ar); if (m_exp2) m_exp2->analyzeProgram(ar); if (m_exp3) m_exp3->analyzeProgram(ar); if (m_stmt) m_stmt->analyzeProgram(ar); } ConstructPtr ForStatement::getNthKid(int n) const { switch (n) { case 0: return m_exp1; case 1: return m_exp2; case 2: return m_stmt; case 3: return m_exp3; default: assert(false); break; } return ConstructPtr(); } int ForStatement::getKidCount() const { return 4; } void ForStatement::setNthKid(int n, ConstructPtr cp) { switch (n) { case 0: m_exp1 = boost::dynamic_pointer_cast(cp); break; case 1: m_exp2 = boost::dynamic_pointer_cast(cp); break; case 2: m_stmt = boost::dynamic_pointer_cast(cp); break; case 3: m_exp3 = boost::dynamic_pointer_cast(cp); break; default: assert(false); break; } } void ForStatement::inferTypes(AnalysisResultPtr ar) { if (m_exp1) m_exp1->inferAndCheck(ar, Type::Any, false); if (m_exp2) m_exp2->inferAndCheck(ar, Type::Boolean, false); if (m_stmt) { getScope()->incLoopNestedLevel(); m_stmt->inferTypes(ar); if (m_exp3) m_exp3->inferAndCheck(ar, Type::Any, false); getScope()->decLoopNestedLevel(); } else { if (m_exp3) m_exp3->inferAndCheck(ar, Type::Any, false); } } /////////////////////////////////////////////////////////////////////////////// // code generation functions void ForStatement::outputPHP(CodeGenerator &cg, AnalysisResultPtr ar) { cg_printf("for ("); if (m_exp1) m_exp1->outputPHP(cg, ar); cg_printf("; "); if (m_exp2) m_exp2->outputPHP(cg, ar); cg_printf("; "); if (m_exp3) m_exp3->outputPHP(cg, ar); cg_printf(") "); if (m_stmt) { m_stmt->outputPHP(cg, ar); } else { cg_printf("{}\n"); } }