Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(460)

Unified Diff: src/ia32/full-codegen-ia32.cc

Issue 13870007: Generators return boxed values (Closed) Base URL: git://github.com/v8/v8.git@master
Patch Set: Rebase; add needed write barrier Created 7 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« src/contexts.h ('K') | « src/full-codegen.h ('k') | src/objects.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: src/ia32/full-codegen-ia32.cc
diff --git a/src/ia32/full-codegen-ia32.cc b/src/ia32/full-codegen-ia32.cc
index f71a76dd092b7a310ae7dfe360d436e11c0ecc7c..67497864d575848bfaeb39f4598ef902181cf8ff 100644
--- a/src/ia32/full-codegen-ia32.cc
+++ b/src/ia32/full-codegen-ia32.cc
@@ -1900,11 +1900,12 @@ void FullCodeGenerator::VisitYield(Yield* expr) {
Label resume;
__ CompareRoot(result_register(), Heap::kTheHoleValueRootIndex);
__ j(not_equal, &resume);
- __ pop(result_register());
if (expr->yield_kind() == Yield::SUSPEND) {
- // TODO(wingo): Box into { value: VALUE, done: false }.
+ EmitReturnIteratorResult(false);
+ } else {
+ __ pop(result_register());
+ EmitReturnSequence();
}
- EmitReturnSequence();
__ bind(&resume);
context()->Plug(result_register());
@@ -1916,18 +1917,7 @@ void FullCodeGenerator::VisitYield(Yield* expr) {
__ mov(FieldOperand(result_register(),
JSGeneratorObject::kContinuationOffset),
Immediate(Smi::FromInt(JSGeneratorObject::kGeneratorClosed)));
- __ pop(result_register());
- // TODO(wingo): Box into { value: VALUE, done: true }.
-
- // Exit all nested statements.
- NestedStatement* current = nesting_stack_;
- int stack_depth = 0;
- int context_length = 0;
- while (current != NULL) {
- current = current->Exit(&stack_depth, &context_length);
- }
- __ Drop(stack_depth);
- EmitReturnSequence();
+ EmitReturnIteratorResult(true);
break;
}
@@ -2033,6 +2023,63 @@ void FullCodeGenerator::EmitGeneratorResume(Expression *generator,
}
+void FullCodeGenerator::EmitReturnIteratorResult(bool done) {
+ Label gc_required;
+ Label allocated;
+
+ STATIC_ASSERT(HeapObject::kMapOffset == 0);
+ STATIC_ASSERT(JSObject::kPropertiesOffset == kPointerSize);
+ STATIC_ASSERT(JSObject::kElementsOffset == 2 * kPointerSize);
+ STATIC_ASSERT(JSObject::kHeaderSize == 3 * kPointerSize);
+ STATIC_ASSERT(JSGeneratorObject::kResultValuePropertyIndex == 0);
+ STATIC_ASSERT(JSGeneratorObject::kResultDonePropertyIndex == 1);
+
+ const int size = 5 * kPointerSize;
Michael Starzinger 2013/05/06 11:22:32 The size should be determined by isolate()->native
+ __ Allocate(size, eax, ecx, edx, &gc_required, TAG_OBJECT);
+
+ __ bind(&allocated);
+ // The object is now in eax. Initialize its map, value, and done fields.
+ Handle<Map> map(isolate()->native_context()->iterator_result_map());
+ __ mov(ebx, map);
+ __ pop(ecx);
+ __ mov(edx,
+ done
Michael Starzinger 2013/05/06 11:22:32 Use factory->ToBoolean(done) here.
+ ? isolate()->factory()->true_value()
+ : isolate()->factory()->false_value());
+ __ mov(FieldOperand(eax, 0 * kPointerSize), ebx); // map
Michael Starzinger 2013/05/06 11:22:32 Same comment as in bootstrapper.cc applies. Can we
+ __ mov(FieldOperand(eax, 1 * kPointerSize), // properties
+ isolate()->factory()->empty_fixed_array());
+ __ mov(FieldOperand(eax, 2 * kPointerSize), // elements
+ isolate()->factory()->empty_fixed_array());
+ __ mov(FieldOperand(eax, 3 * kPointerSize), ecx); // value
+ __ mov(FieldOperand(eax, 4 * kPointerSize), edx); // done?
+
+ // Only the value field needs a write barrier, as the other values are in the
+ // root set.
+ __ RecordWriteField(eax, 3 * kPointerSize, ecx, edx, kDontSaveFPRegs);
+
+ if (done) {
+ // Exit all nested statements.
+ NestedStatement* current = nesting_stack_;
+ int stack_depth = 0;
+ int context_length = 0;
+ while (current != NULL) {
+ current = current->Exit(&stack_depth, &context_length);
+ }
+ __ Drop(stack_depth);
+ }
+
+ EmitReturnSequence();
+
+ __ bind(&gc_required);
+ __ Push(Smi::FromInt(size));
+ __ CallRuntime(Runtime::kAllocateInNewSpace, 1);
+ __ mov(context_register(),
+ Operand(ebp, StandardFrameConstants::kContextOffset));
+ __ jmp(&allocated);
+}
+
+
void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) {
SetSourcePosition(prop->position());
Literal* key = prop->key()->AsLiteral();
« src/contexts.h ('K') | « src/full-codegen.h ('k') | src/objects.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698