| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "src/v8.h" | 5 #include "src/v8.h" |
| 6 | 6 |
| 7 #include "src/ast.h" | 7 #include "src/ast.h" |
| 8 #include "src/ast-numbering.h" | 8 #include "src/ast-numbering.h" |
| 9 #include "src/code-factory.h" | 9 #include "src/code-factory.h" |
| 10 #include "src/codegen.h" | 10 #include "src/codegen.h" |
| (...skipping 118 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 129 } | 129 } |
| 130 | 130 |
| 131 | 131 |
| 132 void BreakableStatementChecker::VisitWhileStatement(WhileStatement* stmt) { | 132 void BreakableStatementChecker::VisitWhileStatement(WhileStatement* stmt) { |
| 133 // Mark while statements breakable if the condition expression is. | 133 // Mark while statements breakable if the condition expression is. |
| 134 Visit(stmt->cond()); | 134 Visit(stmt->cond()); |
| 135 } | 135 } |
| 136 | 136 |
| 137 | 137 |
| 138 void BreakableStatementChecker::VisitForStatement(ForStatement* stmt) { | 138 void BreakableStatementChecker::VisitForStatement(ForStatement* stmt) { |
| 139 // Mark for statements breakable if the condition expression is. | 139 // We set positions for both init and condition, if they exist. |
| 140 if (stmt->cond() != NULL) { | 140 if (stmt->cond() != NULL || stmt->init() != NULL) is_breakable_ = true; |
| 141 Visit(stmt->cond()); | |
| 142 } | |
| 143 } | 141 } |
| 144 | 142 |
| 145 | 143 |
| 146 void BreakableStatementChecker::VisitForInStatement(ForInStatement* stmt) { | 144 void BreakableStatementChecker::VisitForInStatement(ForInStatement* stmt) { |
| 147 // Mark for in statements breakable if the enumerable expression is. | 145 // For-in is breakable because we set the position for the enumerable. |
| 148 Visit(stmt->enumerable()); | 146 is_breakable_ = true; |
| 149 } | 147 } |
| 150 | 148 |
| 151 | 149 |
| 152 void BreakableStatementChecker::VisitForOfStatement(ForOfStatement* stmt) { | 150 void BreakableStatementChecker::VisitForOfStatement(ForOfStatement* stmt) { |
| 153 // For-of is breakable because of the next() call. | 151 // For-of is breakable because we set the position for the next() call. |
| 154 is_breakable_ = true; | 152 is_breakable_ = true; |
| 155 } | 153 } |
| 156 | 154 |
| 157 | 155 |
| 158 void BreakableStatementChecker::VisitTryCatchStatement( | 156 void BreakableStatementChecker::VisitTryCatchStatement( |
| 159 TryCatchStatement* stmt) { | 157 TryCatchStatement* stmt) { |
| 160 // Mark try catch as breakable to avoid adding a break slot in front of it. | 158 // Mark try catch as breakable to avoid adding a break slot in front of it. |
| 161 is_breakable_ = true; | 159 is_breakable_ = true; |
| 162 } | 160 } |
| 163 | 161 |
| (...skipping 1174 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1338 void FullCodeGenerator::VisitForStatement(ForStatement* stmt) { | 1336 void FullCodeGenerator::VisitForStatement(ForStatement* stmt) { |
| 1339 Comment cmnt(masm_, "[ ForStatement"); | 1337 Comment cmnt(masm_, "[ ForStatement"); |
| 1340 Label test, body; | 1338 Label test, body; |
| 1341 | 1339 |
| 1342 Iteration loop_statement(this, stmt); | 1340 Iteration loop_statement(this, stmt); |
| 1343 | 1341 |
| 1344 // Set statement position for a break slot before entering the for-body. | 1342 // Set statement position for a break slot before entering the for-body. |
| 1345 SetStatementPosition(stmt); | 1343 SetStatementPosition(stmt); |
| 1346 | 1344 |
| 1347 if (stmt->init() != NULL) { | 1345 if (stmt->init() != NULL) { |
| 1346 SetStatementPosition(stmt->init()); |
| 1348 Visit(stmt->init()); | 1347 Visit(stmt->init()); |
| 1349 } | 1348 } |
| 1350 | 1349 |
| 1351 increment_loop_depth(); | 1350 increment_loop_depth(); |
| 1352 // Emit the test at the bottom of the loop (even if empty). | 1351 // Emit the test at the bottom of the loop (even if empty). |
| 1353 __ jmp(&test); | 1352 __ jmp(&test); |
| 1354 | 1353 |
| 1355 PrepareForBailoutForId(stmt->BodyId(), NO_REGISTERS); | 1354 PrepareForBailoutForId(stmt->BodyId(), NO_REGISTERS); |
| 1356 __ bind(&body); | 1355 __ bind(&body); |
| 1357 Visit(stmt->body()); | 1356 Visit(stmt->body()); |
| 1358 | 1357 |
| 1359 PrepareForBailoutForId(stmt->ContinueId(), NO_REGISTERS); | 1358 PrepareForBailoutForId(stmt->ContinueId(), NO_REGISTERS); |
| 1360 __ bind(loop_statement.continue_label()); | 1359 __ bind(loop_statement.continue_label()); |
| 1361 if (stmt->next() != NULL) { | 1360 if (stmt->next() != NULL) { |
| 1361 SetStatementPosition(stmt->next()); |
| 1362 Visit(stmt->next()); | 1362 Visit(stmt->next()); |
| 1363 } | 1363 } |
| 1364 | 1364 |
| 1365 // Emit the statement position here as this is where the for | 1365 // Emit the statement position here as this is where the for |
| 1366 // statement code starts. | 1366 // statement code starts. |
| 1367 SetStatementPosition(stmt); | 1367 SetStatementPosition(stmt); |
| 1368 | 1368 |
| 1369 // Check stack before looping. | 1369 // Check stack before looping. |
| 1370 EmitBackEdgeBookkeeping(stmt, &body); | 1370 EmitBackEdgeBookkeeping(stmt, &body); |
| 1371 | 1371 |
| 1372 __ bind(&test); | 1372 __ bind(&test); |
| 1373 if (stmt->cond() != NULL) { | 1373 if (stmt->cond() != NULL) { |
| 1374 SetExpressionPosition(stmt->cond()); |
| 1374 VisitForControl(stmt->cond(), | 1375 VisitForControl(stmt->cond(), |
| 1375 &body, | 1376 &body, |
| 1376 loop_statement.break_label(), | 1377 loop_statement.break_label(), |
| 1377 loop_statement.break_label()); | 1378 loop_statement.break_label()); |
| 1378 } else { | 1379 } else { |
| 1379 __ jmp(&body); | 1380 __ jmp(&body); |
| 1380 } | 1381 } |
| 1381 | 1382 |
| 1382 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); | 1383 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); |
| 1383 __ bind(loop_statement.break_label()); | 1384 __ bind(loop_statement.break_label()); |
| 1384 decrement_loop_depth(); | 1385 decrement_loop_depth(); |
| 1385 } | 1386 } |
| 1386 | 1387 |
| 1387 | 1388 |
| 1389 void FullCodeGenerator::VisitForOfStatement(ForOfStatement* stmt) { |
| 1390 Comment cmnt(masm_, "[ ForOfStatement"); |
| 1391 SetStatementPosition(stmt); |
| 1392 |
| 1393 Iteration loop_statement(this, stmt); |
| 1394 increment_loop_depth(); |
| 1395 |
| 1396 // var iterator = iterable[Symbol.iterator](); |
| 1397 VisitForEffect(stmt->assign_iterator()); |
| 1398 |
| 1399 // Loop entry. |
| 1400 __ bind(loop_statement.continue_label()); |
| 1401 |
| 1402 // result = iterator.next() |
| 1403 SetExpressionPosition(stmt->next_result()); |
| 1404 VisitForEffect(stmt->next_result()); |
| 1405 |
| 1406 // if (result.done) break; |
| 1407 Label result_not_done; |
| 1408 VisitForControl(stmt->result_done(), loop_statement.break_label(), |
| 1409 &result_not_done, &result_not_done); |
| 1410 __ bind(&result_not_done); |
| 1411 |
| 1412 // each = result.value |
| 1413 VisitForEffect(stmt->assign_each()); |
| 1414 |
| 1415 // Generate code for the body of the loop. |
| 1416 Visit(stmt->body()); |
| 1417 |
| 1418 // Check stack before looping. |
| 1419 PrepareForBailoutForId(stmt->BackEdgeId(), NO_REGISTERS); |
| 1420 EmitBackEdgeBookkeeping(stmt, loop_statement.continue_label()); |
| 1421 __ jmp(loop_statement.continue_label()); |
| 1422 |
| 1423 // Exit and decrement the loop depth. |
| 1424 PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); |
| 1425 __ bind(loop_statement.break_label()); |
| 1426 decrement_loop_depth(); |
| 1427 } |
| 1428 |
| 1429 |
| 1388 void FullCodeGenerator::VisitTryCatchStatement(TryCatchStatement* stmt) { | 1430 void FullCodeGenerator::VisitTryCatchStatement(TryCatchStatement* stmt) { |
| 1389 Comment cmnt(masm_, "[ TryCatchStatement"); | 1431 Comment cmnt(masm_, "[ TryCatchStatement"); |
| 1390 SetStatementPosition(stmt); | 1432 SetStatementPosition(stmt); |
| 1391 // The try block adds a handler to the exception handler chain before | 1433 // The try block adds a handler to the exception handler chain before |
| 1392 // entering, and removes it again when exiting normally. If an exception | 1434 // entering, and removes it again when exiting normally. If an exception |
| 1393 // is thrown during execution of the try block, the handler is consumed | 1435 // is thrown during execution of the try block, the handler is consumed |
| 1394 // and control is passed to the catch block with the exception in the | 1436 // and control is passed to the catch block with the exception in the |
| 1395 // result register. | 1437 // result register. |
| 1396 | 1438 |
| 1397 Label try_entry, handler_entry, exit; | 1439 Label try_entry, handler_entry, exit; |
| (...skipping 353 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1751 } | 1793 } |
| 1752 return true; | 1794 return true; |
| 1753 } | 1795 } |
| 1754 #endif // DEBUG | 1796 #endif // DEBUG |
| 1755 | 1797 |
| 1756 | 1798 |
| 1757 #undef __ | 1799 #undef __ |
| 1758 | 1800 |
| 1759 | 1801 |
| 1760 } } // namespace v8::internal | 1802 } } // namespace v8::internal |
| OLD | NEW |