| OLD | NEW | 
|---|
| 1 // Copyright 2011 the V8 project authors. All rights reserved. | 1 // Copyright 2011 the V8 project authors. All rights reserved. | 
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without | 
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are | 
| 4 // met: | 4 // met: | 
| 5 // | 5 // | 
| 6 //     * Redistributions of source code must retain the above copyright | 6 //     * Redistributions of source code must retain the above copyright | 
| 7 //       notice, this list of conditions and the following disclaimer. | 7 //       notice, this list of conditions and the following disclaimer. | 
| 8 //     * Redistributions in binary form must reproduce the above | 8 //     * Redistributions in binary form must reproduce the above | 
| 9 //       copyright notice, this list of conditions and the following | 9 //       copyright notice, this list of conditions and the following | 
| 10 //       disclaimer in the documentation and/or other materials provided | 10 //       disclaimer in the documentation and/or other materials provided | 
| (...skipping 268 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 279     ASSERT(!isolate->has_pending_exception()); | 279     ASSERT(!isolate->has_pending_exception()); | 
| 280     return false; | 280     return false; | 
| 281   } | 281   } | 
| 282   unsigned table_offset = cgen.EmitStackCheckTable(); | 282   unsigned table_offset = cgen.EmitStackCheckTable(); | 
| 283 | 283 | 
| 284   Code::Flags flags = Code::ComputeFlags(Code::FUNCTION); | 284   Code::Flags flags = Code::ComputeFlags(Code::FUNCTION); | 
| 285   Handle<Code> code = CodeGenerator::MakeCodeEpilogue(&masm, flags, info); | 285   Handle<Code> code = CodeGenerator::MakeCodeEpilogue(&masm, flags, info); | 
| 286   code->set_optimizable(info->IsOptimizable()); | 286   code->set_optimizable(info->IsOptimizable()); | 
| 287   cgen.PopulateDeoptimizationData(code); | 287   cgen.PopulateDeoptimizationData(code); | 
| 288   code->set_has_deoptimization_support(info->HasDeoptimizationSupport()); | 288   code->set_has_deoptimization_support(info->HasDeoptimizationSupport()); | 
|  | 289   code->set_handler_table(*cgen.handler_table()); | 
| 289 #ifdef ENABLE_DEBUGGER_SUPPORT | 290 #ifdef ENABLE_DEBUGGER_SUPPORT | 
| 290   code->set_has_debug_break_slots( | 291   code->set_has_debug_break_slots( | 
| 291       info->isolate()->debugger()->IsDebuggerActive()); | 292       info->isolate()->debugger()->IsDebuggerActive()); | 
| 292   code->set_compiled_optimizable(info->IsOptimizable()); | 293   code->set_compiled_optimizable(info->IsOptimizable()); | 
| 293 #endif  // ENABLE_DEBUGGER_SUPPORT | 294 #endif  // ENABLE_DEBUGGER_SUPPORT | 
| 294   code->set_allow_osr_at_loop_nesting_level(0); | 295   code->set_allow_osr_at_loop_nesting_level(0); | 
| 295   code->set_stack_check_table_offset(table_offset); | 296   code->set_stack_check_table_offset(table_offset); | 
| 296   CodeGenerator::PrintCode(code, info); | 297   CodeGenerator::PrintCode(code, info); | 
| 297   info->SetCode(code);  // May be an empty handle. | 298   info->SetCode(code);  // May be an empty handle. | 
| 298 #ifdef ENABLE_GDB_JIT_INTERFACE | 299 #ifdef ENABLE_GDB_JIT_INTERFACE | 
| (...skipping 780 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 1079 | 1080 | 
| 1080   PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); | 1081   PrepareForBailoutForId(stmt->ExitId(), NO_REGISTERS); | 
| 1081   __ bind(loop_statement.break_label()); | 1082   __ bind(loop_statement.break_label()); | 
| 1082   decrement_loop_depth(); | 1083   decrement_loop_depth(); | 
| 1083 } | 1084 } | 
| 1084 | 1085 | 
| 1085 | 1086 | 
| 1086 void FullCodeGenerator::VisitTryCatchStatement(TryCatchStatement* stmt) { | 1087 void FullCodeGenerator::VisitTryCatchStatement(TryCatchStatement* stmt) { | 
| 1087   Comment cmnt(masm_, "[ TryCatchStatement"); | 1088   Comment cmnt(masm_, "[ TryCatchStatement"); | 
| 1088   SetStatementPosition(stmt); | 1089   SetStatementPosition(stmt); | 
| 1089   // The try block adds a handler to the exception handler chain | 1090   // The try block adds a handler to the exception handler chain before | 
| 1090   // before entering, and removes it again when exiting normally. | 1091   // entering, and removes it again when exiting normally.  If an exception | 
| 1091   // If an exception is thrown during execution of the try block, | 1092   // is thrown during execution of the try block, the handler is consumed | 
| 1092   // control is passed to the handler, which also consumes the handler. | 1093   // and control is passed to the catch block with the exception in the | 
| 1093   // At this point, the exception is in a register, and store it in | 1094   // result register. | 
| 1094   // the temporary local variable (prints as ".catch-var") before |  | 
| 1095   // executing the catch block. The catch block has been rewritten |  | 
| 1096   // to introduce a new scope to bind the catch variable and to remove |  | 
| 1097   // that scope again afterwards. |  | 
| 1098 | 1095 | 
| 1099   Label try_handler_setup, done; | 1096   Label try_entry, handler_entry, exit; | 
| 1100   __ Call(&try_handler_setup); | 1097   __ jmp(&try_entry); | 
| 1101   // Try handler code, exception in result register. | 1098   __ bind(&handler_entry); | 
| 1102 | 1099   handler_table()->set(stmt->index(), Smi::FromInt(handler_entry.pos())); | 
|  | 1100   // Exception handler code, the exception is in the result register. | 
| 1103   // Extend the context before executing the catch block. | 1101   // Extend the context before executing the catch block. | 
| 1104   { Comment cmnt(masm_, "[ Extend catch context"); | 1102   { Comment cmnt(masm_, "[ Extend catch context"); | 
| 1105     __ Push(stmt->variable()->name()); | 1103     __ Push(stmt->variable()->name()); | 
| 1106     __ push(result_register()); | 1104     __ push(result_register()); | 
| 1107     PushFunctionArgumentForContextAllocation(); | 1105     PushFunctionArgumentForContextAllocation(); | 
| 1108     __ CallRuntime(Runtime::kPushCatchContext, 3); | 1106     __ CallRuntime(Runtime::kPushCatchContext, 3); | 
| 1109     StoreToFrameField(StandardFrameConstants::kContextOffset, | 1107     StoreToFrameField(StandardFrameConstants::kContextOffset, | 
| 1110                       context_register()); | 1108                       context_register()); | 
| 1111   } | 1109   } | 
| 1112 | 1110 | 
| 1113   Scope* saved_scope = scope(); | 1111   Scope* saved_scope = scope(); | 
| 1114   scope_ = stmt->scope(); | 1112   scope_ = stmt->scope(); | 
| 1115   ASSERT(scope_->declarations()->is_empty()); | 1113   ASSERT(scope_->declarations()->is_empty()); | 
| 1116   { WithOrCatch body(this); | 1114   { WithOrCatch catch_body(this); | 
| 1117     Visit(stmt->catch_block()); | 1115     Visit(stmt->catch_block()); | 
| 1118   } | 1116   } | 
| 1119   // Restore the context. | 1117   // Restore the context. | 
| 1120   LoadContextField(context_register(), Context::PREVIOUS_INDEX); | 1118   LoadContextField(context_register(), Context::PREVIOUS_INDEX); | 
| 1121   StoreToFrameField(StandardFrameConstants::kContextOffset, context_register()); | 1119   StoreToFrameField(StandardFrameConstants::kContextOffset, context_register()); | 
| 1122   scope_ = saved_scope; | 1120   scope_ = saved_scope; | 
| 1123   __ jmp(&done); | 1121   __ jmp(&exit); | 
| 1124 | 1122 | 
| 1125   // Try block code. Sets up the exception handler chain. | 1123   // Try block code. Sets up the exception handler chain. | 
| 1126   __ bind(&try_handler_setup); | 1124   __ bind(&try_entry); | 
| 1127   { | 1125   __ PushTryHandler(IN_JAVASCRIPT, TRY_CATCH_HANDLER, stmt->index()); | 
| 1128     TryCatch try_block(this); | 1126   { TryCatch try_body(this); | 
| 1129     __ PushTryHandler(IN_JAVASCRIPT, TRY_CATCH_HANDLER); |  | 
| 1130     Visit(stmt->try_block()); | 1127     Visit(stmt->try_block()); | 
| 1131     __ PopTryHandler(); |  | 
| 1132   } | 1128   } | 
| 1133   __ bind(&done); | 1129   __ PopTryHandler(); | 
|  | 1130   __ bind(&exit); | 
| 1134 } | 1131 } | 
| 1135 | 1132 | 
| 1136 | 1133 | 
| 1137 void FullCodeGenerator::VisitTryFinallyStatement(TryFinallyStatement* stmt) { | 1134 void FullCodeGenerator::VisitTryFinallyStatement(TryFinallyStatement* stmt) { | 
| 1138   Comment cmnt(masm_, "[ TryFinallyStatement"); | 1135   Comment cmnt(masm_, "[ TryFinallyStatement"); | 
| 1139   SetStatementPosition(stmt); | 1136   SetStatementPosition(stmt); | 
| 1140   // Try finally is compiled by setting up a try-handler on the stack while | 1137   // Try finally is compiled by setting up a try-handler on the stack while | 
| 1141   // executing the try body, and removing it again afterwards. | 1138   // executing the try body, and removing it again afterwards. | 
| 1142   // | 1139   // | 
| 1143   // The try-finally construct can enter the finally block in three ways: | 1140   // The try-finally construct can enter the finally block in three ways: | 
| 1144   // 1. By exiting the try-block normally. This removes the try-handler and | 1141   // 1. By exiting the try-block normally. This removes the try-handler and | 
| 1145   //      calls the finally block code before continuing. | 1142   //    calls the finally block code before continuing. | 
| 1146   // 2. By exiting the try-block with a function-local control flow transfer | 1143   // 2. By exiting the try-block with a function-local control flow transfer | 
| 1147   //    (break/continue/return). The site of the, e.g., break removes the | 1144   //    (break/continue/return). The site of the, e.g., break removes the | 
| 1148   //    try handler and calls the finally block code before continuing | 1145   //    try handler and calls the finally block code before continuing | 
| 1149   //    its outward control transfer. | 1146   //    its outward control transfer. | 
| 1150   // 3. by exiting the try-block with a thrown exception. | 1147   // 3. By exiting the try-block with a thrown exception. | 
| 1151   //    This can happen in nested function calls. It traverses the try-handler | 1148   //    This can happen in nested function calls. It traverses the try-handler | 
| 1152   //    chain and consumes the try-handler entry before jumping to the | 1149   //    chain and consumes the try-handler entry before jumping to the | 
| 1153   //    handler code. The handler code then calls the finally-block before | 1150   //    handler code. The handler code then calls the finally-block before | 
| 1154   //    rethrowing the exception. | 1151   //    rethrowing the exception. | 
| 1155   // | 1152   // | 
| 1156   // The finally block must assume a return address on top of the stack | 1153   // The finally block must assume a return address on top of the stack | 
| 1157   // (or in the link register on ARM chips) and a value (return value or | 1154   // (or in the link register on ARM chips) and a value (return value or | 
| 1158   // exception) in the result register (rax/eax/r0), both of which must | 1155   // exception) in the result register (rax/eax/r0), both of which must | 
| 1159   // be preserved. The return address isn't GC-safe, so it should be | 1156   // be preserved. The return address isn't GC-safe, so it should be | 
| 1160   // cooked before GC. | 1157   // cooked before GC. | 
| 1161   Label finally_entry; | 1158   Label try_entry, handler_entry, finally_entry; | 
| 1162   Label try_handler_setup; |  | 
| 1163 | 1159 | 
| 1164   // Setup the try-handler chain. Use a call to | 1160   // Jump to try-handler setup and try-block code. | 
| 1165   // Jump to try-handler setup and try-block code. Use call to put try-handler | 1161   __ jmp(&try_entry); | 
| 1166   // address on stack. | 1162   __ bind(&handler_entry); | 
| 1167   __ Call(&try_handler_setup); | 1163   handler_table()->set(stmt->index(), Smi::FromInt(handler_entry.pos())); | 
| 1168   // Try handler code. Return address of call is pushed on handler stack. | 1164   // Exception handler code.  This code is only executed when an exception | 
| 1169   { | 1165   // is thrown.  The exception is in the result register, and must be | 
| 1170     // This code is only executed during stack-handler traversal when an | 1166   // preserved by the finally block.  Call the finally block and then | 
| 1171     // exception is thrown. The exception is in the result register, which | 1167   // rethrow the exception if it returns. | 
| 1172     // is retained by the finally block. | 1168   __ Call(&finally_entry); | 
| 1173     // Call the finally block and then rethrow the exception if it returns. | 1169   __ push(result_register()); | 
| 1174     __ Call(&finally_entry); | 1170   __ CallRuntime(Runtime::kReThrow, 1); | 
| 1175     __ push(result_register()); | 1171 | 
| 1176     __ CallRuntime(Runtime::kReThrow, 1); | 1172   // Finally block implementation. | 
|  | 1173   __ bind(&finally_entry); | 
|  | 1174   EnterFinallyBlock(); | 
|  | 1175   { Finally finally_body(this); | 
|  | 1176     Visit(stmt->finally_block()); | 
| 1177   } | 1177   } | 
|  | 1178   ExitFinallyBlock();  // Return to the calling code. | 
| 1178 | 1179 | 
| 1179   __ bind(&finally_entry); | 1180   // Setup try handler. | 
| 1180   { | 1181   __ bind(&try_entry); | 
| 1181     // Finally block implementation. | 1182   __ PushTryHandler(IN_JAVASCRIPT, TRY_FINALLY_HANDLER, stmt->index()); | 
| 1182     Finally finally_block(this); | 1183   { TryFinally try_body(this, &finally_entry); | 
| 1183     EnterFinallyBlock(); | 1184     Visit(stmt->try_block()); | 
| 1184     Visit(stmt->finally_block()); |  | 
| 1185     ExitFinallyBlock();  // Return to the calling code. |  | 
| 1186   } | 1185   } | 
| 1187 | 1186   __ PopTryHandler(); | 
| 1188   __ bind(&try_handler_setup); |  | 
| 1189   { |  | 
| 1190     // Setup try handler (stack pointer registers). |  | 
| 1191     TryFinally try_block(this, &finally_entry); |  | 
| 1192     __ PushTryHandler(IN_JAVASCRIPT, TRY_FINALLY_HANDLER); |  | 
| 1193     Visit(stmt->try_block()); |  | 
| 1194     __ PopTryHandler(); |  | 
| 1195   } |  | 
| 1196   // Execute the finally block on the way out.  Clobber the unpredictable | 1187   // Execute the finally block on the way out.  Clobber the unpredictable | 
| 1197   // value in the accumulator with one that's safe for GC.  The finally | 1188   // value in the result register with one that's safe for GC because the | 
| 1198   // block will unconditionally preserve the accumulator on the stack. | 1189   // finally block will unconditionally preserve the result register on the | 
|  | 1190   // stack. | 
| 1199   ClearAccumulator(); | 1191   ClearAccumulator(); | 
| 1200   __ Call(&finally_entry); | 1192   __ Call(&finally_entry); | 
| 1201 } | 1193 } | 
| 1202 | 1194 | 
| 1203 | 1195 | 
| 1204 void FullCodeGenerator::VisitDebuggerStatement(DebuggerStatement* stmt) { | 1196 void FullCodeGenerator::VisitDebuggerStatement(DebuggerStatement* stmt) { | 
| 1205 #ifdef ENABLE_DEBUGGER_SUPPORT | 1197 #ifdef ENABLE_DEBUGGER_SUPPORT | 
| 1206   Comment cmnt(masm_, "[ DebuggerStatement"); | 1198   Comment cmnt(masm_, "[ DebuggerStatement"); | 
| 1207   SetStatementPosition(stmt); | 1199   SetStatementPosition(stmt); | 
| 1208 | 1200 | 
| (...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 1309   } | 1301   } | 
| 1310 | 1302 | 
| 1311   return false; | 1303   return false; | 
| 1312 } | 1304 } | 
| 1313 | 1305 | 
| 1314 | 1306 | 
| 1315 #undef __ | 1307 #undef __ | 
| 1316 | 1308 | 
| 1317 | 1309 | 
| 1318 } }  // namespace v8::internal | 1310 } }  // namespace v8::internal | 
| OLD | NEW | 
|---|