OLD | NEW |
---|---|
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 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 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
118 CallRuntimePassFunction(masm, Runtime::kTryInstallOptimizedCode); | 118 CallRuntimePassFunction(masm, Runtime::kTryInstallOptimizedCode); |
119 GenerateTailCallToReturnedCode(masm); | 119 GenerateTailCallToReturnedCode(masm); |
120 | 120 |
121 __ bind(&ok); | 121 __ bind(&ok); |
122 GenerateTailCallToSharedCode(masm); | 122 GenerateTailCallToSharedCode(masm); |
123 } | 123 } |
124 | 124 |
125 | 125 |
126 static void Generate_JSConstructStubHelper(MacroAssembler* masm, | 126 static void Generate_JSConstructStubHelper(MacroAssembler* masm, |
127 bool is_api_function, | 127 bool is_api_function, |
128 bool count_constructions) { | 128 bool count_constructions, |
129 bool create_memento) { | |
129 // ----------- S t a t e ------------- | 130 // ----------- S t a t e ------------- |
130 // -- eax: number of arguments | 131 // -- eax: number of arguments |
131 // -- edi: constructor function | 132 // -- edi: constructor function |
133 // -- ebx: allocation site or undefined | |
132 // ----------------------------------- | 134 // ----------------------------------- |
133 | 135 |
134 // Should never count constructions for api objects. | 136 // Should never count constructions for api objects. |
135 ASSERT(!is_api_function || !count_constructions); | 137 ASSERT(!is_api_function || !count_constructions); |
136 | 138 |
139 // Should never create mementos for api functions. (true?) | |
Hannes Payer (out of office)
2014/02/11 15:51:23
probably
mvstanton
2014/02/17 15:53:08
Done.
| |
140 ASSERT(!is_api_function || !create_memento); | |
141 | |
137 // Enter a construct frame. | 142 // Enter a construct frame. |
138 { | 143 { |
139 FrameScope scope(masm, StackFrame::CONSTRUCT); | 144 FrameScope scope(masm, StackFrame::CONSTRUCT); |
140 | 145 |
146 if (create_memento) { | |
147 __ AssertUndefinedOrAllocationSite(ebx); | |
148 __ push(ebx); | |
149 } | |
150 | |
141 // Store a smi-tagged arguments count on the stack. | 151 // Store a smi-tagged arguments count on the stack. |
142 __ SmiTag(eax); | 152 __ SmiTag(eax); |
143 __ push(eax); | 153 __ push(eax); |
144 | 154 |
145 // Push the function to invoke on the stack. | 155 // Push the function to invoke on the stack. |
146 __ push(edi); | 156 __ push(edi); |
147 | 157 |
148 // Try to allocate the object without transitioning into C code. If any of | 158 // Try to allocate the object without transitioning into C code. If any of |
149 // the preconditions is not met, the code bails out to the runtime call. | 159 // the preconditions is not met, the code bails out to the runtime call. |
150 Label rt_call, allocated; | 160 Label rt_call, allocated; |
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
195 __ pop(eax); | 205 __ pop(eax); |
196 | 206 |
197 __ bind(&allocate); | 207 __ bind(&allocate); |
198 } | 208 } |
199 | 209 |
200 // Now allocate the JSObject on the heap. | 210 // Now allocate the JSObject on the heap. |
201 // edi: constructor | 211 // edi: constructor |
202 // eax: initial map | 212 // eax: initial map |
203 __ movzx_b(edi, FieldOperand(eax, Map::kInstanceSizeOffset)); | 213 __ movzx_b(edi, FieldOperand(eax, Map::kInstanceSizeOffset)); |
204 __ shl(edi, kPointerSizeLog2); | 214 __ shl(edi, kPointerSizeLog2); |
215 if (create_memento) { | |
216 __ add(edi, Immediate(AllocationMemento::kSize)); | |
217 } | |
205 __ Allocate(edi, ebx, edi, no_reg, &rt_call, NO_ALLOCATION_FLAGS); | 218 __ Allocate(edi, ebx, edi, no_reg, &rt_call, NO_ALLOCATION_FLAGS); |
219 Factory* factory = masm->isolate()->factory(); | |
220 if (create_memento) { | |
221 __ sub(edi, Immediate(AllocationMemento::kSize)); | |
222 Handle<Map> allocation_memento_map = factory->allocation_memento_map(); | |
223 __ mov(Operand(edi, AllocationMemento::kMapOffset), | |
224 allocation_memento_map); | |
225 // Get the cell or undefined. | |
226 Label do_store; | |
227 __ mov(ecx, Operand(esp, kPointerSize*2)); | |
228 __ cmp(ecx, masm->isolate()->factory()->undefined_value()); | |
229 __ j(equal, &do_store); | |
230 // ecx is an AllocationSite. We are creating a memento from it, so we | |
231 // need to increment the memento create count. | |
232 __ add(Operand(ecx, AllocationSite::kPretenureCreateCountOffset), | |
233 Immediate(Smi::FromInt(1))); | |
234 __ bind(&do_store); | |
Hannes Payer (out of office)
2014/02/11 15:51:23
We could also undo the Memento allocation. WDYT?
mvstanton
2014/02/17 15:53:08
Good point, I'll go ahead and fix it to only incre
| |
235 __ mov(Operand(edi, AllocationMemento::kAllocationSiteOffset), | |
236 ecx); | |
237 } | |
206 // Allocated the JSObject, now initialize the fields. | 238 // Allocated the JSObject, now initialize the fields. |
207 // eax: initial map | 239 // eax: initial map |
208 // ebx: JSObject | 240 // ebx: JSObject |
209 // edi: start of next object | 241 // edi: start of next object (or memento if create_memento) |
210 __ mov(Operand(ebx, JSObject::kMapOffset), eax); | 242 __ mov(Operand(ebx, JSObject::kMapOffset), eax); |
211 Factory* factory = masm->isolate()->factory(); | |
212 __ mov(ecx, factory->empty_fixed_array()); | 243 __ mov(ecx, factory->empty_fixed_array()); |
213 __ mov(Operand(ebx, JSObject::kPropertiesOffset), ecx); | 244 __ mov(Operand(ebx, JSObject::kPropertiesOffset), ecx); |
214 __ mov(Operand(ebx, JSObject::kElementsOffset), ecx); | 245 __ mov(Operand(ebx, JSObject::kElementsOffset), ecx); |
215 // Set extra fields in the newly allocated object. | 246 // Set extra fields in the newly allocated object. |
216 // eax: initial map | 247 // eax: initial map |
217 // ebx: JSObject | 248 // ebx: JSObject |
218 // edi: start of next object | 249 // edi: start of next object (or memento if create_memento) |
219 __ lea(ecx, Operand(ebx, JSObject::kHeaderSize)); | 250 __ lea(ecx, Operand(ebx, JSObject::kHeaderSize)); |
220 __ mov(edx, factory->undefined_value()); | 251 __ mov(edx, factory->undefined_value()); |
221 if (count_constructions) { | 252 if (count_constructions) { |
222 __ movzx_b(esi, | 253 __ movzx_b(esi, |
223 FieldOperand(eax, Map::kPreAllocatedPropertyFieldsOffset)); | 254 FieldOperand(eax, Map::kPreAllocatedPropertyFieldsOffset)); |
224 __ lea(esi, | 255 __ lea(esi, |
225 Operand(ebx, esi, times_pointer_size, JSObject::kHeaderSize)); | 256 Operand(ebx, esi, times_pointer_size, JSObject::kHeaderSize)); |
226 // esi: offset of first field after pre-allocated fields | 257 // esi: offset of first field after pre-allocated fields |
227 if (FLAG_debug_code) { | 258 if (FLAG_debug_code) { |
228 __ cmp(esi, edi); | 259 __ cmp(esi, edi); |
229 __ Assert(less_equal, | 260 __ Assert(less_equal, |
230 kUnexpectedNumberOfPreAllocatedPropertyFields); | 261 kUnexpectedNumberOfPreAllocatedPropertyFields); |
231 } | 262 } |
232 __ InitializeFieldsWithFiller(ecx, esi, edx); | 263 __ InitializeFieldsWithFiller(ecx, esi, edx); |
233 __ mov(edx, factory->one_pointer_filler_map()); | 264 __ mov(edx, factory->one_pointer_filler_map()); |
234 } | 265 } |
235 __ InitializeFieldsWithFiller(ecx, edi, edx); | 266 __ InitializeFieldsWithFiller(ecx, edi, edx); |
267 if (create_memento) { | |
268 __ add(edi, Immediate(AllocationMemento::kSize)); | |
269 } | |
236 | 270 |
237 // Add the object tag to make the JSObject real, so that we can continue | 271 // Add the object tag to make the JSObject real, so that we can continue |
238 // and jump into the continuation code at any time from now on. Any | 272 // and jump into the continuation code at any time from now on. Any |
239 // failures need to undo the allocation, so that the heap is in a | 273 // failures need to undo the allocation, so that the heap is in a |
240 // consistent state and verifiable. | 274 // consistent state and verifiable. |
241 // eax: initial map | 275 // eax: initial map |
242 // ebx: JSObject | 276 // ebx: JSObject |
243 // edi: start of next object | 277 // edi: start of next object |
244 __ or_(ebx, Immediate(kHeapObjectTag)); | 278 __ or_(ebx, Immediate(kHeapObjectTag)); |
245 | 279 |
(...skipping 70 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
316 // Undo the setting of the new top so that the heap is verifiable. For | 350 // Undo the setting of the new top so that the heap is verifiable. For |
317 // example, the map's unused properties potentially do not match the | 351 // example, the map's unused properties potentially do not match the |
318 // allocated objects unused properties. | 352 // allocated objects unused properties. |
319 // ebx: JSObject (previous new top) | 353 // ebx: JSObject (previous new top) |
320 __ bind(&undo_allocation); | 354 __ bind(&undo_allocation); |
321 __ UndoAllocationInNewSpace(ebx); | 355 __ UndoAllocationInNewSpace(ebx); |
322 } | 356 } |
323 | 357 |
324 // Allocate the new receiver object using the runtime call. | 358 // Allocate the new receiver object using the runtime call. |
325 __ bind(&rt_call); | 359 __ bind(&rt_call); |
360 int offset = 0; | |
361 if (create_memento) { | |
362 // Get the cell or allocation site. | |
363 __ mov(edi, Operand(esp, kPointerSize*2)); | |
364 __ push(edi); | |
365 offset = kPointerSize; | |
366 } | |
367 | |
326 // Must restore edi (constructor) before calling runtime. | 368 // Must restore edi (constructor) before calling runtime. |
327 __ mov(edi, Operand(esp, 0)); | 369 __ mov(edi, Operand(esp, offset)); |
328 // edi: function (constructor) | 370 // edi: function (constructor) |
329 __ push(edi); | 371 __ push(edi); |
330 __ CallRuntime(Runtime::kNewObject, 1); | 372 if (create_memento) { |
373 __ CallRuntime(Runtime::kNewObjectWithAllocationSite, 2); | |
374 } else { | |
375 __ CallRuntime(Runtime::kNewObject, 1); | |
376 } | |
331 __ mov(ebx, eax); // store result in ebx | 377 __ mov(ebx, eax); // store result in ebx |
332 | 378 |
333 // New object allocated. | 379 // New object allocated. |
334 // ebx: newly allocated object | 380 // ebx: newly allocated object |
335 __ bind(&allocated); | 381 __ bind(&allocated); |
336 // Retrieve the function from the stack. | 382 // Retrieve the function from the stack. |
337 __ pop(edi); | 383 __ pop(edi); |
338 | 384 |
339 // Retrieve smi-tagged arguments count from the stack. | 385 // Retrieve smi-tagged arguments count from the stack. |
340 __ mov(eax, Operand(esp, 0)); | 386 __ mov(eax, Operand(esp, 0)); |
(...skipping 67 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
408 STATIC_ASSERT(kSmiTagSize == 1 && kSmiTag == 0); | 454 STATIC_ASSERT(kSmiTagSize == 1 && kSmiTag == 0); |
409 __ pop(ecx); | 455 __ pop(ecx); |
410 __ lea(esp, Operand(esp, ebx, times_2, 1 * kPointerSize)); // 1 ~ receiver | 456 __ lea(esp, Operand(esp, ebx, times_2, 1 * kPointerSize)); // 1 ~ receiver |
411 __ push(ecx); | 457 __ push(ecx); |
412 __ IncrementCounter(masm->isolate()->counters()->constructed_objects(), 1); | 458 __ IncrementCounter(masm->isolate()->counters()->constructed_objects(), 1); |
413 __ ret(0); | 459 __ ret(0); |
414 } | 460 } |
415 | 461 |
416 | 462 |
417 void Builtins::Generate_JSConstructStubCountdown(MacroAssembler* masm) { | 463 void Builtins::Generate_JSConstructStubCountdown(MacroAssembler* masm) { |
418 Generate_JSConstructStubHelper(masm, false, true); | 464 Generate_JSConstructStubHelper(masm, false, true, true); |
419 } | 465 } |
420 | 466 |
421 | 467 |
422 void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { | 468 void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { |
423 Generate_JSConstructStubHelper(masm, false, false); | 469 Generate_JSConstructStubHelper(masm, false, false, true); |
424 } | 470 } |
425 | 471 |
426 | 472 |
427 void Builtins::Generate_JSConstructStubApi(MacroAssembler* masm) { | 473 void Builtins::Generate_JSConstructStubApi(MacroAssembler* masm) { |
428 Generate_JSConstructStubHelper(masm, true, false); | 474 Generate_JSConstructStubHelper(masm, true, false, false); |
429 } | 475 } |
430 | 476 |
431 | 477 |
432 static void Generate_JSEntryTrampolineHelper(MacroAssembler* masm, | 478 static void Generate_JSEntryTrampolineHelper(MacroAssembler* masm, |
433 bool is_construct) { | 479 bool is_construct) { |
434 ProfileEntryHookStub::MaybeCallEntryHook(masm); | 480 ProfileEntryHookStub::MaybeCallEntryHook(masm); |
435 | 481 |
436 // Clear the context before we push it when entering the internal frame. | 482 // Clear the context before we push it when entering the internal frame. |
437 __ Set(esi, Immediate(0)); | 483 __ Set(esi, Immediate(0)); |
438 | 484 |
(...skipping 926 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1365 | 1411 |
1366 __ bind(&ok); | 1412 __ bind(&ok); |
1367 __ ret(0); | 1413 __ ret(0); |
1368 } | 1414 } |
1369 | 1415 |
1370 #undef __ | 1416 #undef __ |
1371 } | 1417 } |
1372 } // namespace v8::internal | 1418 } // namespace v8::internal |
1373 | 1419 |
1374 #endif // V8_TARGET_ARCH_IA32 | 1420 #endif // V8_TARGET_ARCH_IA32 |
OLD | NEW |