| OLD | NEW | 
|---|
| 1 // Copyright 2009 the V8 project authors. All rights reserved. | 1 // Copyright 2009 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 2344 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 2355   __ CompareRoot(boilerplate.reg(), Heap::kUndefinedValueRootIndex); | 2355   __ CompareRoot(boilerplate.reg(), Heap::kUndefinedValueRootIndex); | 
| 2356   deferred->Branch(equal); | 2356   deferred->Branch(equal); | 
| 2357   deferred->BindExit(); | 2357   deferred->BindExit(); | 
| 2358   literals.Unuse(); | 2358   literals.Unuse(); | 
| 2359 | 2359 | 
| 2360   // Push the boilerplate object. | 2360   // Push the boilerplate object. | 
| 2361   frame_->Push(&boilerplate); | 2361   frame_->Push(&boilerplate); | 
| 2362 } | 2362 } | 
| 2363 | 2363 | 
| 2364 | 2364 | 
| 2365 // Materialize the object literal 'node' in the literals array |  | 
| 2366 // 'literals' of the function.  Leave the object boilerplate in |  | 
| 2367 // 'boilerplate'. |  | 
| 2368 class DeferredObjectLiteral: public DeferredCode { |  | 
| 2369  public: |  | 
| 2370   DeferredObjectLiteral(Register boilerplate, |  | 
| 2371                         Register literals, |  | 
| 2372                         ObjectLiteral* node) |  | 
| 2373       : boilerplate_(boilerplate), literals_(literals), node_(node) { |  | 
| 2374     set_comment("[ DeferredObjectLiteral"); |  | 
| 2375   } |  | 
| 2376 |  | 
| 2377   void Generate(); |  | 
| 2378 |  | 
| 2379  private: |  | 
| 2380   Register boilerplate_; |  | 
| 2381   Register literals_; |  | 
| 2382   ObjectLiteral* node_; |  | 
| 2383 }; |  | 
| 2384 |  | 
| 2385 |  | 
| 2386 void DeferredObjectLiteral::Generate() { |  | 
| 2387   // Since the entry is undefined we call the runtime system to |  | 
| 2388   // compute the literal. |  | 
| 2389   // Literal array (0). |  | 
| 2390   __ push(literals_); |  | 
| 2391   // Literal index (1). |  | 
| 2392   __ Push(Smi::FromInt(node_->literal_index())); |  | 
| 2393   // Constant properties (2). |  | 
| 2394   __ Push(node_->constant_properties()); |  | 
| 2395   __ CallRuntime(Runtime::kCreateObjectLiteralBoilerplate, 3); |  | 
| 2396   if (!boilerplate_.is(rax)) __ movq(boilerplate_, rax); |  | 
| 2397 } |  | 
| 2398 |  | 
| 2399 |  | 
| 2400 void CodeGenerator::VisitObjectLiteral(ObjectLiteral* node) { | 2365 void CodeGenerator::VisitObjectLiteral(ObjectLiteral* node) { | 
| 2401   Comment cmnt(masm_, "[ ObjectLiteral"); | 2366   Comment cmnt(masm_, "[ ObjectLiteral"); | 
| 2402 | 2367 | 
| 2403   // Retrieve the literals array and check the allocated entry.  Begin | 2368   // Load a writable copy of the function of this activation in a | 
| 2404   // with a writable copy of the function of this activation in a |  | 
| 2405   // register. | 2369   // register. | 
| 2406   frame_->PushFunction(); | 2370   frame_->PushFunction(); | 
| 2407   Result literals = frame_->Pop(); | 2371   Result literals = frame_->Pop(); | 
| 2408   literals.ToRegister(); | 2372   literals.ToRegister(); | 
| 2409   frame_->Spill(literals.reg()); | 2373   frame_->Spill(literals.reg()); | 
| 2410 | 2374 | 
| 2411   // Load the literals array of the function. | 2375   // Load the literals array of the function. | 
| 2412   __ movq(literals.reg(), | 2376   __ movq(literals.reg(), | 
| 2413           FieldOperand(literals.reg(), JSFunction::kLiteralsOffset)); | 2377           FieldOperand(literals.reg(), JSFunction::kLiteralsOffset)); | 
| 2414 | 2378   // Literal array. | 
| 2415   // Load the literal at the ast saved index. | 2379   frame_->Push(&literals); | 
| 2416   Result boilerplate = allocator_->Allocate(); | 2380   // Literal index. | 
| 2417   ASSERT(boilerplate.is_valid()); | 2381   frame_->Push(Smi::FromInt(node->literal_index())); | 
| 2418   int literal_offset = | 2382   // Constant properties. | 
| 2419       FixedArray::kHeaderSize + node->literal_index() * kPointerSize; | 2383   frame_->Push(node->constant_properties()); | 
| 2420   __ movq(boilerplate.reg(), FieldOperand(literals.reg(), literal_offset)); | 2384   Result clone; | 
| 2421 | 2385   if (node->depth() > 1) { | 
| 2422   // Check whether we need to materialize the object literal boilerplate. | 2386     clone = frame_->CallRuntime(Runtime::kCreateObjectLiteral, 3); | 
| 2423   // If so, jump to the deferred code passing the literals array. | 2387   } else { | 
| 2424   DeferredObjectLiteral* deferred = | 2388     clone = frame_->CallRuntime(Runtime::kCreateObjectLiteralShallow, 3); | 
| 2425       new DeferredObjectLiteral(boilerplate.reg(), literals.reg(), node); |  | 
| 2426   __ CompareRoot(boilerplate.reg(), Heap::kUndefinedValueRootIndex); |  | 
| 2427   deferred->Branch(equal); |  | 
| 2428   deferred->BindExit(); |  | 
| 2429   literals.Unuse(); |  | 
| 2430 |  | 
| 2431   // Push the boilerplate object. |  | 
| 2432   frame_->Push(&boilerplate); |  | 
| 2433   // Clone the boilerplate object. |  | 
| 2434   Runtime::FunctionId clone_function_id = Runtime::kCloneLiteralBoilerplate; |  | 
| 2435   if (node->depth() == 1) { |  | 
| 2436     clone_function_id = Runtime::kCloneShallowLiteralBoilerplate; |  | 
| 2437   } | 2389   } | 
| 2438   Result clone = frame_->CallRuntime(clone_function_id, 1); |  | 
| 2439   // Push the newly cloned literal object as the result. |  | 
| 2440   frame_->Push(&clone); | 2390   frame_->Push(&clone); | 
| 2441 | 2391 | 
| 2442   for (int i = 0; i < node->properties()->length(); i++) { | 2392   for (int i = 0; i < node->properties()->length(); i++) { | 
| 2443     ObjectLiteral::Property* property = node->properties()->at(i); | 2393     ObjectLiteral::Property* property = node->properties()->at(i); | 
| 2444     switch (property->kind()) { | 2394     switch (property->kind()) { | 
| 2445       case ObjectLiteral::Property::CONSTANT: | 2395       case ObjectLiteral::Property::CONSTANT: | 
| 2446         break; | 2396         break; | 
| 2447       case ObjectLiteral::Property::MATERIALIZED_LITERAL: | 2397       case ObjectLiteral::Property::MATERIALIZED_LITERAL: | 
| 2448         if (CompileTimeValue::IsCompileTimeValue(property->value())) break; | 2398         if (CompileTimeValue::IsCompileTimeValue(property->value())) break; | 
| 2449         // else fall through. | 2399         // else fall through. | 
| (...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 2489         Result ignored = frame_->CallRuntime(Runtime::kDefineAccessor, 4); | 2439         Result ignored = frame_->CallRuntime(Runtime::kDefineAccessor, 4); | 
| 2490         // Ignore the result. | 2440         // Ignore the result. | 
| 2491         break; | 2441         break; | 
| 2492       } | 2442       } | 
| 2493       default: UNREACHABLE(); | 2443       default: UNREACHABLE(); | 
| 2494     } | 2444     } | 
| 2495   } | 2445   } | 
| 2496 } | 2446 } | 
| 2497 | 2447 | 
| 2498 | 2448 | 
| 2499 // Materialize the array literal 'node' in the literals array 'literals' |  | 
| 2500 // of the function.  Leave the array boilerplate in 'boilerplate'. |  | 
| 2501 class DeferredArrayLiteral: public DeferredCode { |  | 
| 2502  public: |  | 
| 2503   DeferredArrayLiteral(Register boilerplate, |  | 
| 2504                        Register literals, |  | 
| 2505                        ArrayLiteral* node) |  | 
| 2506       : boilerplate_(boilerplate), literals_(literals), node_(node) { |  | 
| 2507     set_comment("[ DeferredArrayLiteral"); |  | 
| 2508   } |  | 
| 2509 |  | 
| 2510   void Generate(); |  | 
| 2511 |  | 
| 2512  private: |  | 
| 2513   Register boilerplate_; |  | 
| 2514   Register literals_; |  | 
| 2515   ArrayLiteral* node_; |  | 
| 2516 }; |  | 
| 2517 |  | 
| 2518 |  | 
| 2519 void DeferredArrayLiteral::Generate() { |  | 
| 2520   // Since the entry is undefined we call the runtime system to |  | 
| 2521   // compute the literal. |  | 
| 2522   // Literal array (0). |  | 
| 2523   __ push(literals_); |  | 
| 2524   // Literal index (1). |  | 
| 2525   __ Push(Smi::FromInt(node_->literal_index())); |  | 
| 2526   // Constant properties (2). |  | 
| 2527   __ Push(node_->literals()); |  | 
| 2528   __ CallRuntime(Runtime::kCreateArrayLiteralBoilerplate, 3); |  | 
| 2529   if (!boilerplate_.is(rax)) __ movq(boilerplate_, rax); |  | 
| 2530 } |  | 
| 2531 |  | 
| 2532 |  | 
| 2533 void CodeGenerator::VisitArrayLiteral(ArrayLiteral* node) { | 2449 void CodeGenerator::VisitArrayLiteral(ArrayLiteral* node) { | 
| 2534   Comment cmnt(masm_, "[ ArrayLiteral"); | 2450   Comment cmnt(masm_, "[ ArrayLiteral"); | 
| 2535 | 2451 | 
| 2536   // Retrieve the literals array and check the allocated entry.  Begin | 2452   // Load a writable copy of the function of this activation in a | 
| 2537   // with a writable copy of the function of this activation in a |  | 
| 2538   // register. | 2453   // register. | 
| 2539   frame_->PushFunction(); | 2454   frame_->PushFunction(); | 
| 2540   Result literals = frame_->Pop(); | 2455   Result literals = frame_->Pop(); | 
| 2541   literals.ToRegister(); | 2456   literals.ToRegister(); | 
| 2542   frame_->Spill(literals.reg()); | 2457   frame_->Spill(literals.reg()); | 
| 2543 | 2458 | 
| 2544   // Load the literals array of the function. | 2459   // Load the literals array of the function. | 
| 2545   __ movq(literals.reg(), | 2460   __ movq(literals.reg(), | 
| 2546           FieldOperand(literals.reg(), JSFunction::kLiteralsOffset)); | 2461           FieldOperand(literals.reg(), JSFunction::kLiteralsOffset)); | 
| 2547 | 2462   // Literal array. | 
| 2548   // Load the literal at the ast saved index. | 2463   frame_->Push(&literals); | 
| 2549   Result boilerplate = allocator_->Allocate(); | 2464   // Literal index. | 
| 2550   ASSERT(boilerplate.is_valid()); | 2465   frame_->Push(Smi::FromInt(node->literal_index())); | 
| 2551   int literal_offset = | 2466   // Constant elements. | 
| 2552       FixedArray::kHeaderSize + node->literal_index() * kPointerSize; | 2467   frame_->Push(node->constant_elements()); | 
| 2553   __ movq(boilerplate.reg(), FieldOperand(literals.reg(), literal_offset)); | 2468   Result clone; | 
| 2554 | 2469   if (node->depth() > 1) { | 
| 2555   // Check whether we need to materialize the object literal boilerplate. | 2470     clone = frame_->CallRuntime(Runtime::kCreateArrayLiteral, 3); | 
| 2556   // If so, jump to the deferred code passing the literals array. | 2471   } else { | 
| 2557   DeferredArrayLiteral* deferred = | 2472     clone = frame_->CallRuntime(Runtime::kCreateArrayLiteralShallow, 3); | 
| 2558       new DeferredArrayLiteral(boilerplate.reg(), literals.reg(), node); |  | 
| 2559   __ CompareRoot(boilerplate.reg(), Heap::kUndefinedValueRootIndex); |  | 
| 2560   deferred->Branch(equal); |  | 
| 2561   deferred->BindExit(); |  | 
| 2562   literals.Unuse(); |  | 
| 2563 |  | 
| 2564   // Push the resulting array literal boilerplate on the stack. |  | 
| 2565   frame_->Push(&boilerplate); |  | 
| 2566   // Clone the boilerplate object. |  | 
| 2567   Runtime::FunctionId clone_function_id = Runtime::kCloneLiteralBoilerplate; |  | 
| 2568   if (node->depth() == 1) { |  | 
| 2569     clone_function_id = Runtime::kCloneShallowLiteralBoilerplate; |  | 
| 2570   } | 2473   } | 
| 2571   Result clone = frame_->CallRuntime(clone_function_id, 1); |  | 
| 2572   // Push the newly cloned literal object as the result. |  | 
| 2573   frame_->Push(&clone); | 2474   frame_->Push(&clone); | 
| 2574 | 2475 | 
| 2575   // Generate code to set the elements in the array that are not | 2476   // Generate code to set the elements in the array that are not | 
| 2576   // literals. | 2477   // literals. | 
| 2577   for (int i = 0; i < node->values()->length(); i++) { | 2478   for (int i = 0; i < node->values()->length(); i++) { | 
| 2578     Expression* value = node->values()->at(i); | 2479     Expression* value = node->values()->at(i); | 
| 2579 | 2480 | 
| 2580     // If value is a literal the property value is already set in the | 2481     // If value is a literal the property value is already set in the | 
| 2581     // boilerplate object. | 2482     // boilerplate object. | 
| 2582     if (value->AsLiteral() != NULL) continue; | 2483     if (value->AsLiteral() != NULL) continue; | 
| (...skipping 5560 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 8143   masm.GetCode(&desc); | 8044   masm.GetCode(&desc); | 
| 8144   // Call the function from C++. | 8045   // Call the function from C++. | 
| 8145   return FUNCTION_CAST<ModuloFunction>(buffer); | 8046   return FUNCTION_CAST<ModuloFunction>(buffer); | 
| 8146 } | 8047 } | 
| 8147 | 8048 | 
| 8148 #endif | 8049 #endif | 
| 8149 | 8050 | 
| 8150 #undef __ | 8051 #undef __ | 
| 8151 | 8052 | 
| 8152 } }  // namespace v8::internal | 8053 } }  // namespace v8::internal | 
| OLD | NEW | 
|---|