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 1980 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1991 } | 1991 } |
1992 | 1992 |
1993 | 1993 |
1994 void FullCodeGenerator::VisitYield(Yield* expr) { | 1994 void FullCodeGenerator::VisitYield(Yield* expr) { |
1995 Comment cmnt(masm_, "[ Yield"); | 1995 Comment cmnt(masm_, "[ Yield"); |
1996 // Evaluate yielded value first; the initial iterator definition depends on | 1996 // Evaluate yielded value first; the initial iterator definition depends on |
1997 // this. It stays on the stack while we update the iterator. | 1997 // this. It stays on the stack while we update the iterator. |
1998 VisitForStackValue(expr->expression()); | 1998 VisitForStackValue(expr->expression()); |
1999 | 1999 |
2000 switch (expr->yield_kind()) { | 2000 switch (expr->yield_kind()) { |
2001 case Yield::INITIAL: | 2001 case Yield::SUSPEND: |
2002 case Yield::SUSPEND: { | 2002 // Pop value from top-of-stack slot; box result into result register. |
| 2003 EmitCreateIteratorResult(false); |
| 2004 __ push(result_register()); |
| 2005 // Fall through. |
| 2006 case Yield::INITIAL: { |
2003 VisitForStackValue(expr->generator_object()); | 2007 VisitForStackValue(expr->generator_object()); |
2004 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); | 2008 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); |
2005 __ lw(context_register(), | 2009 __ lw(context_register(), |
2006 MemOperand(fp, StandardFrameConstants::kContextOffset)); | 2010 MemOperand(fp, StandardFrameConstants::kContextOffset)); |
2007 | 2011 |
2008 Label resume; | 2012 Label resume; |
2009 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); | 2013 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); |
2010 __ Branch(&resume, ne, result_register(), Operand(at)); | 2014 __ Branch(&resume, ne, result_register(), Operand(at)); |
2011 if (expr->yield_kind() == Yield::SUSPEND) { | 2015 __ pop(result_register()); |
2012 EmitReturnIteratorResult(false); | 2016 EmitReturnSequence(); |
2013 } else { | |
2014 __ pop(result_register()); | |
2015 EmitReturnSequence(); | |
2016 } | |
2017 | 2017 |
2018 __ bind(&resume); | 2018 __ bind(&resume); |
2019 context()->Plug(result_register()); | 2019 context()->Plug(result_register()); |
2020 break; | 2020 break; |
2021 } | 2021 } |
2022 | 2022 |
2023 case Yield::FINAL: { | 2023 case Yield::FINAL: { |
2024 VisitForAccumulatorValue(expr->generator_object()); | 2024 VisitForAccumulatorValue(expr->generator_object()); |
2025 __ li(a1, Operand(Smi::FromInt(JSGeneratorObject::kGeneratorClosed))); | 2025 __ li(a1, Operand(Smi::FromInt(JSGeneratorObject::kGeneratorClosed))); |
2026 __ sw(a1, FieldMemOperand(result_register(), | 2026 __ sw(a1, FieldMemOperand(result_register(), |
2027 JSGeneratorObject::kContinuationOffset)); | 2027 JSGeneratorObject::kContinuationOffset)); |
2028 EmitReturnIteratorResult(true); | 2028 // Pop value from top-of-stack slot, box result into result register. |
| 2029 EmitCreateIteratorResult(true); |
| 2030 EmitUnwindBeforeReturn(); |
| 2031 EmitReturnSequence(); |
2029 break; | 2032 break; |
2030 } | 2033 } |
2031 | 2034 |
2032 case Yield::DELEGATING: { | 2035 case Yield::DELEGATING: { |
2033 VisitForStackValue(expr->generator_object()); | 2036 VisitForStackValue(expr->generator_object()); |
2034 | 2037 |
2035 // Initial stack layout is as follows: | 2038 // Initial stack layout is as follows: |
2036 // [sp + 1 * kPointerSize] iter | 2039 // [sp + 1 * kPointerSize] iter |
2037 // [sp + 0 * kPointerSize] g | 2040 // [sp + 0 * kPointerSize] g |
2038 | 2041 |
(...skipping 11 matching lines...) Expand all Loading... |
2050 __ push(a0); // exception | 2053 __ push(a0); // exception |
2051 __ mov(a0, a3); // iter | 2054 __ mov(a0, a3); // iter |
2052 __ LoadRoot(a2, Heap::kthrow_stringRootIndex); // "throw" | 2055 __ LoadRoot(a2, Heap::kthrow_stringRootIndex); // "throw" |
2053 Handle<Code> throw_ic = isolate()->builtins()->LoadIC_Initialize(); | 2056 Handle<Code> throw_ic = isolate()->builtins()->LoadIC_Initialize(); |
2054 CallIC(throw_ic); // iter.throw in a0 | 2057 CallIC(throw_ic); // iter.throw in a0 |
2055 __ mov(a0, v0); | 2058 __ mov(a0, v0); |
2056 __ jmp(&l_call); | 2059 __ jmp(&l_call); |
2057 | 2060 |
2058 // try { received = yield result.value } | 2061 // try { received = yield result.value } |
2059 __ bind(&l_try); | 2062 __ bind(&l_try); |
2060 __ pop(a0); // result.value | 2063 EmitCreateIteratorResult(false); // pop and box to v0 |
2061 __ PushTryHandler(StackHandler::CATCH, expr->index()); | 2064 __ PushTryHandler(StackHandler::CATCH, expr->index()); |
2062 const int handler_size = StackHandlerConstants::kSize; | 2065 const int handler_size = StackHandlerConstants::kSize; |
2063 __ push(a0); // result.value | 2066 __ push(a0); // result |
2064 __ lw(a3, MemOperand(sp, (0 + 1) * kPointerSize + handler_size)); // g | 2067 __ lw(a3, MemOperand(sp, (0 + 1) * kPointerSize + handler_size)); // g |
2065 __ push(a3); // g | 2068 __ push(a3); // g |
2066 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); | 2069 __ CallRuntime(Runtime::kSuspendJSGeneratorObject, 1); |
2067 __ mov(a0, v0); | 2070 __ mov(a0, v0); |
2068 __ lw(context_register(), | 2071 __ lw(context_register(), |
2069 MemOperand(fp, StandardFrameConstants::kContextOffset)); | 2072 MemOperand(fp, StandardFrameConstants::kContextOffset)); |
2070 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); | 2073 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); |
2071 __ Branch(&l_resume, ne, a0, Operand(at)); | 2074 __ Branch(&l_resume, ne, a0, Operand(at)); |
2072 EmitReturnIteratorResult(false); | 2075 __ pop(v0); // result |
| 2076 EmitReturnSequence(); |
2073 __ mov(a0, v0); | 2077 __ mov(a0, v0); |
2074 __ bind(&l_resume); // received in a0 | 2078 __ bind(&l_resume); // received in a0 |
2075 __ PopTryHandler(); | 2079 __ PopTryHandler(); |
2076 | 2080 |
2077 // receiver = iter; f = iter.next; arg = received; | 2081 // receiver = iter; f = iter.next; arg = received; |
2078 __ bind(&l_next); | 2082 __ bind(&l_next); |
2079 __ lw(a3, MemOperand(sp, 1 * kPointerSize)); // iter | 2083 __ lw(a3, MemOperand(sp, 1 * kPointerSize)); // iter |
2080 __ push(a3); // iter | 2084 __ push(a3); // iter |
2081 __ push(a0); // received | 2085 __ push(a0); // received |
2082 __ mov(a0, a3); // iter | 2086 __ mov(a0, a3); // iter |
(...skipping 136 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2219 // Throw error if we attempt to operate on a running generator. | 2223 // Throw error if we attempt to operate on a running generator. |
2220 __ bind(&wrong_state); | 2224 __ bind(&wrong_state); |
2221 __ push(a1); | 2225 __ push(a1); |
2222 __ CallRuntime(Runtime::kThrowGeneratorStateError, 1); | 2226 __ CallRuntime(Runtime::kThrowGeneratorStateError, 1); |
2223 | 2227 |
2224 __ bind(&done); | 2228 __ bind(&done); |
2225 context()->Plug(result_register()); | 2229 context()->Plug(result_register()); |
2226 } | 2230 } |
2227 | 2231 |
2228 | 2232 |
2229 void FullCodeGenerator::EmitReturnIteratorResult(bool done) { | 2233 void FullCodeGenerator::EmitCreateIteratorResult(bool done) { |
2230 Label gc_required; | 2234 Label gc_required; |
2231 Label allocated; | 2235 Label allocated; |
2232 | 2236 |
2233 Handle<Map> map(isolate()->native_context()->generator_result_map()); | 2237 Handle<Map> map(isolate()->native_context()->generator_result_map()); |
2234 | 2238 |
2235 __ Allocate(map->instance_size(), a0, a2, a3, &gc_required, TAG_OBJECT); | 2239 __ Allocate(map->instance_size(), a0, a2, a3, &gc_required, TAG_OBJECT); |
| 2240 __ jmp(&allocated); |
| 2241 |
| 2242 __ bind(&gc_required); |
| 2243 __ Push(Smi::FromInt(map->instance_size())); |
| 2244 __ CallRuntime(Runtime::kAllocateInNewSpace, 1); |
| 2245 __ lw(context_register(), |
| 2246 MemOperand(fp, StandardFrameConstants::kContextOffset)); |
2236 | 2247 |
2237 __ bind(&allocated); | 2248 __ bind(&allocated); |
2238 __ li(a1, Operand(map)); | 2249 __ li(a1, Operand(map)); |
2239 __ pop(a2); | 2250 __ pop(a2); |
2240 __ li(a3, Operand(isolate()->factory()->ToBoolean(done))); | 2251 __ li(a3, Operand(isolate()->factory()->ToBoolean(done))); |
2241 __ li(t0, Operand(isolate()->factory()->empty_fixed_array())); | 2252 __ li(t0, Operand(isolate()->factory()->empty_fixed_array())); |
2242 ASSERT_EQ(map->instance_size(), 5 * kPointerSize); | 2253 ASSERT_EQ(map->instance_size(), 5 * kPointerSize); |
2243 __ sw(a1, FieldMemOperand(a0, HeapObject::kMapOffset)); | 2254 __ sw(a1, FieldMemOperand(a0, HeapObject::kMapOffset)); |
2244 __ sw(t0, FieldMemOperand(a0, JSObject::kPropertiesOffset)); | 2255 __ sw(t0, FieldMemOperand(a0, JSObject::kPropertiesOffset)); |
2245 __ sw(t0, FieldMemOperand(a0, JSObject::kElementsOffset)); | 2256 __ sw(t0, FieldMemOperand(a0, JSObject::kElementsOffset)); |
2246 __ sw(a2, | 2257 __ sw(a2, |
2247 FieldMemOperand(a0, JSGeneratorObject::kResultValuePropertyOffset)); | 2258 FieldMemOperand(a0, JSGeneratorObject::kResultValuePropertyOffset)); |
2248 __ sw(a3, | 2259 __ sw(a3, |
2249 FieldMemOperand(a0, JSGeneratorObject::kResultDonePropertyOffset)); | 2260 FieldMemOperand(a0, JSGeneratorObject::kResultDonePropertyOffset)); |
2250 | 2261 |
2251 // Only the value field needs a write barrier, as the other values are in the | 2262 // Only the value field needs a write barrier, as the other values are in the |
2252 // root set. | 2263 // root set. |
2253 __ RecordWriteField(a0, JSGeneratorObject::kResultValuePropertyOffset, | 2264 __ RecordWriteField(a0, JSGeneratorObject::kResultValuePropertyOffset, |
2254 a2, a3, kRAHasBeenSaved, kDontSaveFPRegs); | 2265 a2, a3, kRAHasBeenSaved, kDontSaveFPRegs); |
2255 | |
2256 if (done) { | |
2257 // Exit all nested statements. | |
2258 NestedStatement* current = nesting_stack_; | |
2259 int stack_depth = 0; | |
2260 int context_length = 0; | |
2261 while (current != NULL) { | |
2262 current = current->Exit(&stack_depth, &context_length); | |
2263 } | |
2264 __ Drop(stack_depth); | |
2265 } | |
2266 | |
2267 __ mov(result_register(), a0); | 2266 __ mov(result_register(), a0); |
2268 EmitReturnSequence(); | |
2269 | |
2270 __ bind(&gc_required); | |
2271 __ Push(Smi::FromInt(map->instance_size())); | |
2272 __ CallRuntime(Runtime::kAllocateInNewSpace, 1); | |
2273 __ lw(context_register(), | |
2274 MemOperand(fp, StandardFrameConstants::kContextOffset)); | |
2275 __ jmp(&allocated); | |
2276 } | 2267 } |
2277 | 2268 |
2278 | 2269 |
2279 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { | 2270 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { |
2280 SetSourcePosition(prop->position()); | 2271 SetSourcePosition(prop->position()); |
2281 Literal* key = prop->key()->AsLiteral(); | 2272 Literal* key = prop->key()->AsLiteral(); |
2282 __ mov(a0, result_register()); | 2273 __ mov(a0, result_register()); |
2283 __ li(a2, Operand(key->handle())); | 2274 __ li(a2, Operand(key->handle())); |
2284 // Call load IC. It has arguments receiver and property name a0 and a2. | 2275 // Call load IC. It has arguments receiver and property name a0 and a2. |
2285 Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize(); | 2276 Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize(); |
(...skipping 2676 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4962 *context_length = 0; | 4953 *context_length = 0; |
4963 return previous_; | 4954 return previous_; |
4964 } | 4955 } |
4965 | 4956 |
4966 | 4957 |
4967 #undef __ | 4958 #undef __ |
4968 | 4959 |
4969 } } // namespace v8::internal | 4960 } } // namespace v8::internal |
4970 | 4961 |
4971 #endif // V8_TARGET_ARCH_MIPS | 4962 #endif // V8_TARGET_ARCH_MIPS |
OLD | NEW |