OLD | NEW |
1 // Copyright 2006-2008 the V8 project authors. All rights reserved. | 1 // Copyright 2006-2008 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 56 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
67 | 67 |
68 // Optimize the AST. | 68 // Optimize the AST. |
69 if (!Rewriter::Optimize(literal)) { | 69 if (!Rewriter::Optimize(literal)) { |
70 // Signal a stack overflow by returning a null handle. The stack | 70 // Signal a stack overflow by returning a null handle. The stack |
71 // overflow exception will be thrown by the caller. | 71 // overflow exception will be thrown by the caller. |
72 return Handle<Code>::null(); | 72 return Handle<Code>::null(); |
73 } | 73 } |
74 | 74 |
75 // Generate code and return it. | 75 // Generate code and return it. |
76 Handle<Code> result = CodeGenerator::MakeCode(literal, script, is_eval); | 76 Handle<Code> result = CodeGenerator::MakeCode(literal, script, is_eval); |
| 77 // Check for stack-overflow exception. |
| 78 if (result.is_null()) { |
| 79 Top::StackOverflow(); |
| 80 Top::ReportPendingMessages(); |
| 81 } |
77 return result; | 82 return result; |
78 } | 83 } |
79 | 84 |
80 | 85 |
81 static Handle<JSFunction> MakeFunction(bool is_global, | 86 static Handle<JSFunction> MakeFunction(bool is_global, |
82 bool is_eval, | 87 bool is_eval, |
83 Handle<Script> script, | 88 Handle<Script> script, |
84 v8::Extension* extension, | 89 v8::Extension* extension, |
85 ScriptDataImpl* pre_data) { | 90 ScriptDataImpl* pre_data) { |
86 ZoneScope zone_scope(DELETE_ON_EXIT); | 91 ZoneScope zone_scope(DELETE_ON_EXIT); |
87 | 92 |
88 // Make sure we have an initial stack limit. | 93 // Make sure we have an initial stack limit. |
89 StackGuard guard; | 94 StackGuard guard; |
90 PostponeInterruptsScope postpone; | 95 PostponeInterruptsScope postpone; |
91 | 96 |
92 // Notify debugger | 97 // Notify debugger |
93 Debugger::OnBeforeCompile(script); | 98 Debugger::OnBeforeCompile(script); |
94 | 99 |
95 // Only allow non-global compiles for eval. | 100 // Only allow non-global compiles for eval. |
96 ASSERT(is_eval || is_global); | 101 ASSERT(is_eval || is_global); |
97 | 102 |
98 // Build AST. | 103 // Build AST. |
99 FunctionLiteral* lit = MakeAST(is_global, script, extension, pre_data); | 104 FunctionLiteral* lit = MakeAST(is_global, script, extension, pre_data); |
100 | 105 |
101 // Check for parse errors. | 106 // Check for parse errors. |
102 if (lit == NULL) { | 107 if (lit == NULL) { |
103 ASSERT(Top::has_pending_exception()); | 108 ASSERT(Top::has_pending_exception()); |
| 109 Top::ReportPendingMessages(); |
104 return Handle<JSFunction>::null(); | 110 return Handle<JSFunction>::null(); |
105 } | 111 } |
106 | 112 |
107 // Measure how long it takes to do the compilation; only take the | 113 // Measure how long it takes to do the compilation; only take the |
108 // rest of the function into account to avoid overlap with the | 114 // rest of the function into account to avoid overlap with the |
109 // parsing statistics. | 115 // parsing statistics. |
110 StatsRate* rate = is_eval | 116 StatsRate* rate = is_eval |
111 ? &Counters::compile_eval | 117 ? &Counters::compile_eval |
112 : &Counters::compile; | 118 : &Counters::compile; |
113 StatsRateScope timer(rate); | 119 StatsRateScope timer(rate); |
114 | 120 |
115 // Compile the code. | 121 // Compile the code. |
116 Handle<Code> code = MakeCode(lit, script, is_eval); | 122 Handle<Code> code = MakeCode(lit, script, is_eval); |
117 | 123 |
118 // Check for stack-overflow exceptions. | 124 // Check for stack-overflow exceptions. |
119 if (code.is_null()) { | 125 if (code.is_null()) { |
120 Top::StackOverflow(); | |
121 return Handle<JSFunction>::null(); | 126 return Handle<JSFunction>::null(); |
122 } | 127 } |
123 | 128 |
124 if (script->name()->IsString()) { | 129 if (script->name()->IsString()) { |
125 SmartPointer<char> data = | 130 SmartPointer<char> data = |
126 String::cast(script->name())->ToCString(DISALLOW_NULLS); | 131 String::cast(script->name())->ToCString(DISALLOW_NULLS); |
127 LOG(CodeCreateEvent(is_eval ? "Eval" : "Script", *code, *data)); | 132 LOG(CodeCreateEvent(is_eval ? "Eval" : "Script", *code, *data)); |
128 } else { | 133 } else { |
129 LOG(CodeCreateEvent(is_eval ? "Eval" : "Script", *code, "")); | 134 LOG(CodeCreateEvent(is_eval ? "Eval" : "Script", *code, "")); |
130 } | 135 } |
(...skipping 68 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
199 if (extension == NULL && !result.is_null()) { | 204 if (extension == NULL && !result.is_null()) { |
200 CompilationCache::PutFunction(source, CompilationCache::SCRIPT, result); | 205 CompilationCache::PutFunction(source, CompilationCache::SCRIPT, result); |
201 } | 206 } |
202 | 207 |
203 // Get rid of the pre-parsing data (if necessary). | 208 // Get rid of the pre-parsing data (if necessary). |
204 if (input_pre_data == NULL && pre_data != NULL) { | 209 if (input_pre_data == NULL && pre_data != NULL) { |
205 delete pre_data; | 210 delete pre_data; |
206 } | 211 } |
207 } | 212 } |
208 | 213 |
| 214 if (result.is_null()) Top::ReportPendingMessages(); |
| 215 |
209 return result; | 216 return result; |
210 } | 217 } |
211 | 218 |
212 | 219 |
213 Handle<JSFunction> Compiler::CompileEval(Handle<String> source, | 220 Handle<JSFunction> Compiler::CompileEval(Handle<String> source, |
214 int line_offset, | 221 int line_offset, |
215 bool is_global) { | 222 bool is_global) { |
216 int source_length = source->length(); | 223 int source_length = source->length(); |
217 Counters::total_eval_size.Increment(source_length); | 224 Counters::total_eval_size.Increment(source_length); |
218 Counters::total_compile_size.Increment(source_length); | 225 Counters::total_compile_size.Increment(source_length); |
219 | 226 |
220 // The VM is in the COMPILER state until exiting this function. | 227 // The VM is in the COMPILER state until exiting this function. |
221 VMState state(COMPILER); | 228 VMState state(COMPILER); |
222 CompilationCache::Entry entry = is_global | 229 CompilationCache::Entry entry = is_global |
223 ? CompilationCache::EVAL_GLOBAL | 230 ? CompilationCache::EVAL_GLOBAL |
224 : CompilationCache::EVAL_CONTEXTUAL; | 231 : CompilationCache::EVAL_CONTEXTUAL; |
225 | 232 |
226 // Do a lookup in the compilation cache; if the entry is not there, | 233 // Do a lookup in the compilation cache; if the entry is not there, |
227 // invoke the compiler and add the result to the cache. | 234 // invoke the compiler and add the result to the cache. |
228 Handle<JSFunction> result = CompilationCache::LookupEval(source, entry); | 235 Handle<JSFunction> result = CompilationCache::LookupEval(source, entry); |
229 if (result.is_null()) { | 236 if (result.is_null()) { |
230 // Create a script object describing the script to be compiled. | 237 // Create a script object describing the script to be compiled. |
231 Handle<Script> script = Factory::NewScript(source); | 238 Handle<Script> script = Factory::NewScript(source); |
232 script->set_line_offset(Smi::FromInt(line_offset)); | 239 script->set_line_offset(Smi::FromInt(line_offset)); |
233 result = MakeFunction(is_global, true, script, NULL, NULL); | 240 result = MakeFunction(is_global, true, script, NULL, NULL); |
234 if (!result.is_null()) { | 241 if (!result.is_null()) { |
235 CompilationCache::PutFunction(source, entry, result); | 242 CompilationCache::PutFunction(source, entry, result); |
236 } | 243 } |
237 } | 244 } |
| 245 |
| 246 if (result.is_null()) Top::ReportPendingMessages(); |
| 247 |
238 return result; | 248 return result; |
239 } | 249 } |
240 | 250 |
241 | 251 |
242 bool Compiler::CompileLazy(Handle<SharedFunctionInfo> shared, | 252 bool Compiler::CompileLazy(Handle<SharedFunctionInfo> shared, |
243 int loop_nesting) { | 253 int loop_nesting) { |
244 ZoneScope zone_scope(DELETE_ON_EXIT); | 254 ZoneScope zone_scope(DELETE_ON_EXIT); |
245 | 255 |
246 // The VM is in the COMPILER state until exiting this function. | 256 // The VM is in the COMPILER state until exiting this function. |
247 VMState state(COMPILER); | 257 VMState state(COMPILER); |
(...skipping 14 matching lines...) Expand all Loading... |
262 // Generate the AST for the lazily compiled function. The AST may be | 272 // Generate the AST for the lazily compiled function. The AST may be |
263 // NULL in case of parser stack overflow. | 273 // NULL in case of parser stack overflow. |
264 FunctionLiteral* lit = MakeLazyAST(script, name, | 274 FunctionLiteral* lit = MakeLazyAST(script, name, |
265 start_position, | 275 start_position, |
266 end_position, | 276 end_position, |
267 is_expression); | 277 is_expression); |
268 | 278 |
269 // Check for parse errors. | 279 // Check for parse errors. |
270 if (lit == NULL) { | 280 if (lit == NULL) { |
271 ASSERT(Top::has_pending_exception()); | 281 ASSERT(Top::has_pending_exception()); |
| 282 Top::ReportPendingMessages(); |
272 return false; | 283 return false; |
273 } | 284 } |
274 | 285 |
275 // Update the loop nesting in the function literal. | 286 // Update the loop nesting in the function literal. |
276 lit->set_loop_nesting(loop_nesting); | 287 lit->set_loop_nesting(loop_nesting); |
277 | 288 |
278 // Measure how long it takes to do the lazy compilation; only take | 289 // Measure how long it takes to do the lazy compilation; only take |
279 // the rest of the function into account to avoid overlap with the | 290 // the rest of the function into account to avoid overlap with the |
280 // lazy parsing statistics. | 291 // lazy parsing statistics. |
281 StatsRateScope timer(&Counters::compile_lazy); | 292 StatsRateScope timer(&Counters::compile_lazy); |
282 | 293 |
283 // Compile the code. | 294 // Compile the code. |
284 Handle<Code> code = MakeCode(lit, script, false); | 295 Handle<Code> code = MakeCode(lit, script, false); |
285 | 296 |
286 // Check for stack-overflow exception. | |
287 if (code.is_null()) { | 297 if (code.is_null()) { |
288 Top::StackOverflow(); | |
289 return false; | 298 return false; |
290 } | 299 } |
291 | 300 |
292 // Generate the code, update the function info, and return the code. | 301 // Generate the code, update the function info, and return the code. |
293 LOG(CodeCreateEvent("LazyCompile", *code, *lit->name())); | 302 LOG(CodeCreateEvent("LazyCompile", *code, *lit->name())); |
294 | 303 |
295 // Update the shared function info with the compiled code. | 304 // Update the shared function info with the compiled code. |
296 shared->set_code(*code); | 305 shared->set_code(*code); |
297 | 306 |
298 // Set the expected number of properties for instances. | 307 // Set the expected number of properties for instances. |
299 SetExpectedNofPropertiesFromEstimate(shared, lit->expected_property_count()); | 308 SetExpectedNofPropertiesFromEstimate(shared, lit->expected_property_count()); |
300 | 309 |
301 // Check the function has compiled code. | 310 // Check the function has compiled code. |
302 ASSERT(shared->is_compiled()); | 311 ASSERT(shared->is_compiled()); |
303 return true; | 312 return true; |
304 } | 313 } |
305 | 314 |
306 | 315 |
307 } } // namespace v8::internal | 316 } } // namespace v8::internal |
OLD | NEW |