Chromium Code Reviews| 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 1919 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1930 EmitReturnSequence(); | 1930 EmitReturnSequence(); |
| 1931 break; | 1931 break; |
| 1932 } | 1932 } |
| 1933 | 1933 |
| 1934 case Yield::DELEGATING: | 1934 case Yield::DELEGATING: |
| 1935 UNIMPLEMENTED(); | 1935 UNIMPLEMENTED(); |
| 1936 } | 1936 } |
| 1937 } | 1937 } |
| 1938 | 1938 |
| 1939 | 1939 |
| 1940 void FullCodeGenerator::EmitResume(Expression *generator_expr, | |
| 1941 Expression *value_expr, | |
| 1942 JSGeneratorObject::ResumeMode resume_mode) { | |
| 1943 // ebx will hold the generator object until the activation has been resumed. | |
| 1944 VisitForAccumulatorValue(generator_expr); | |
| 1945 __ mov(ebx, result_register()); | |
|
Michael Starzinger
2013/04/21 22:45:21
The ebx register might get clobbered while evaluat
wingo
2013/04/23 13:51:04
Good catch! Fixed.
| |
| 1946 | |
| 1947 // The value stays in eax, and is ultimately read by the resumed generator, as | |
| 1948 // if the CallRuntime(Runtime::kSuspendJSGeneratorObject) returned it. | |
| 1949 VisitForAccumulatorValue(value_expr); | |
| 1950 | |
| 1951 // Check generator state. | |
| 1952 Label wrong_state, done; | |
| 1953 __ mov(edx, FieldOperand(ebx, JSGeneratorObject::kContinuationOffset)); | |
| 1954 __ SmiUntag(edx); | |
|
Michael Starzinger
2013/04/21 22:45:21
No need to untag and compare, we can compare right
wingo
2013/04/23 13:51:04
Done.
| |
| 1955 __ cmp(edx, Immediate(0)); | |
| 1956 __ j(less_equal, &wrong_state); | |
| 1957 | |
| 1958 // Load suspended function and context. | |
| 1959 __ mov(esi, FieldOperand(ebx, JSGeneratorObject::kContextOffset)); | |
| 1960 __ mov(edi, FieldOperand(ebx, JSGeneratorObject::kFunctionOffset)); | |
| 1961 | |
| 1962 // Push holes for arguments to generator function. | |
| 1963 __ mov(edx, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); | |
| 1964 __ mov(edx, | |
| 1965 FieldOperand(edx, SharedFunctionInfo::kFormalParameterCountOffset)); | |
| 1966 __ mov(ecx, isolate()->factory()->the_hole_value()); | |
| 1967 Label push_argument_holes; | |
| 1968 __ bind(&push_argument_holes); | |
| 1969 __ push(ecx); | |
| 1970 __ sub(edx, Immediate(1)); | |
| 1971 __ j(not_carry, &push_argument_holes); | |
| 1972 | |
| 1973 // Enter a new JavaScript frame, and initialize its slots as they were when | |
| 1974 // the generator was suspended. | |
| 1975 Label push_frame, resume_frame; | |
| 1976 __ bind(&push_frame); | |
| 1977 __ call(&resume_frame); | |
| 1978 __ jmp(&done); | |
| 1979 __ bind(&resume_frame); | |
| 1980 __ push(ebp); // Caller's frame pointer. | |
| 1981 __ mov(ebp, esp); | |
| 1982 __ push(esi); // Callee's context. | |
| 1983 __ push(edi); // Callee's JS Function. | |
| 1984 | |
| 1985 // Load the operand stack size. | |
| 1986 __ mov(edx, FieldOperand(ebx, JSGeneratorObject::kOperandStackOffset)); | |
| 1987 __ mov(edx, FieldOperand(edx, FixedArray::kLengthOffset)); | |
| 1988 __ SmiUntag(edx); | |
| 1989 | |
| 1990 // If we are sending a value and there is no operand stack, we can jump back | |
| 1991 // in directly. | |
| 1992 if (resume_mode == JSGeneratorObject::kSend) { | |
| 1993 Label slow_resume; | |
| 1994 __ cmp(edx, Immediate(0)); | |
| 1995 __ j(not_zero, &slow_resume); | |
| 1996 __ mov(edx, FieldOperand(edi, JSFunction::kCodeEntryOffset)); | |
| 1997 __ mov(ecx, FieldOperand(ebx, JSGeneratorObject::kContinuationOffset)); | |
| 1998 __ SmiUntag(ecx); | |
| 1999 __ add(edx, ecx); | |
| 2000 __ mov(FieldOperand(ebx, JSGeneratorObject::kContinuationOffset), | |
| 2001 Immediate(Smi::FromInt(JSGeneratorObject::kGeneratorExecuting))); | |
| 2002 __ jmp(edx); | |
| 2003 __ bind(&slow_resume); | |
| 2004 } | |
| 2005 | |
| 2006 // Otherwise, we push holes for the operand stack and call the runtime to fix | |
| 2007 // up the stack and the handlers. | |
| 2008 Label push_operand_holes, call_resume; | |
| 2009 __ bind(&push_operand_holes); | |
| 2010 __ sub(edx, Immediate(1)); | |
| 2011 __ j(carry, &call_resume); | |
| 2012 __ push(ecx); | |
| 2013 __ jmp(&push_operand_holes); | |
| 2014 __ bind(&call_resume); | |
| 2015 __ push(ebx); | |
| 2016 __ push(result_register()); | |
| 2017 __ Push(Smi::FromInt(resume_mode)); | |
| 2018 __ CallRuntime(Runtime::kResumeJSGeneratorObject, 3); | |
| 2019 // Not reached: the runtime call returns elsewhere. | |
| 2020 __ int3(); | |
| 2021 | |
| 2022 // Throw error if we attempt to operate on a running generator. | |
| 2023 __ bind(&wrong_state); | |
| 2024 __ push(ebx); | |
| 2025 __ CallRuntime(Runtime::kThrowGeneratorStateError, 1); | |
| 2026 | |
| 2027 __ bind(&done); | |
| 2028 context()->Plug(result_register()); | |
| 2029 } | |
| 2030 | |
| 2031 | |
| 1940 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { | 2032 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { |
| 1941 SetSourcePosition(prop->position()); | 2033 SetSourcePosition(prop->position()); |
| 1942 Literal* key = prop->key()->AsLiteral(); | 2034 Literal* key = prop->key()->AsLiteral(); |
| 1943 ASSERT(!key->handle()->IsSmi()); | 2035 ASSERT(!key->handle()->IsSmi()); |
| 1944 __ mov(ecx, Immediate(key->handle())); | 2036 __ mov(ecx, Immediate(key->handle())); |
| 1945 Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize(); | 2037 Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize(); |
| 1946 CallIC(ic, RelocInfo::CODE_TARGET, prop->PropertyFeedbackId()); | 2038 CallIC(ic, RelocInfo::CODE_TARGET, prop->PropertyFeedbackId()); |
| 1947 } | 2039 } |
| 1948 | 2040 |
| 1949 | 2041 |
| (...skipping 2650 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 4600 *stack_depth = 0; | 4692 *stack_depth = 0; |
| 4601 *context_length = 0; | 4693 *context_length = 0; |
| 4602 return previous_; | 4694 return previous_; |
| 4603 } | 4695 } |
| 4604 | 4696 |
| 4605 #undef __ | 4697 #undef __ |
| 4606 | 4698 |
| 4607 } } // namespace v8::internal | 4699 } } // namespace v8::internal |
| 4608 | 4700 |
| 4609 #endif // V8_TARGET_ARCH_IA32 | 4701 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |