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

Side by Side Diff: src/ia32/macro-assembler-ia32.cc

Issue 1463803002: [debugger] flood function for stepping before calling it. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: rebase, ports, deoptimize builtins Created 5 years 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 unified diff | Download patch
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #if V8_TARGET_ARCH_IA32 5 #if V8_TARGET_ARCH_IA32
6 6
7 #include "src/base/bits.h" 7 #include "src/base/bits.h"
8 #include "src/base/division-by-constant.h" 8 #include "src/base/division-by-constant.h"
9 #include "src/bootstrapper.h" 9 #include "src/bootstrapper.h"
10 #include "src/codegen.h" 10 #include "src/codegen.h"
(...skipping 1968 matching lines...) Expand 10 before | Expand all | Expand 10 after
1979 jmp(done, done_near); 1979 jmp(done, done_near);
1980 } 1980 }
1981 } else { 1981 } else {
1982 jmp(adaptor, RelocInfo::CODE_TARGET); 1982 jmp(adaptor, RelocInfo::CODE_TARGET);
1983 } 1983 }
1984 bind(&invoke); 1984 bind(&invoke);
1985 } 1985 }
1986 } 1986 }
1987 1987
1988 1988
1989 void MacroAssembler::InvokeCode(const Operand& code, 1989 void MacroAssembler::FloodFunctionIfStepping(Register fun, Register new_target,
1990 Register new_target, 1990 const ParameterCount& expected,
1991 const ParameterCount& expected, 1991 const ParameterCount& actual) {
1992 const ParameterCount& actual, 1992 Label skip_flooding;
1993 InvokeFlag flag, 1993 ExternalReference debug_step_action =
1994 const CallWrapper& call_wrapper) { 1994 ExternalReference::debug_last_step_action_address(isolate());
1995 cmpb(Operand::StaticVariable(debug_step_action), StepIn);
1996 j(not_equal, &skip_flooding);
1997 {
1998 FrameScope frame(this,
1999 has_frame() ? StackFrame::NONE : StackFrame::INTERNAL);
2000 if (expected.is_reg()) {
2001 SmiTag(expected.reg());
2002 Push(expected.reg());
2003 }
2004 if (actual.is_reg()) {
2005 SmiTag(actual.reg());
2006 Push(actual.reg());
2007 }
2008 if (new_target.is_valid()) {
2009 Push(new_target);
2010 }
2011 Push(fun);
2012 Push(fun);
2013 CallRuntime(Runtime::kDebugPrepareStepInIfStepping, 1);
2014 Pop(fun);
2015 if (new_target.is_valid()) {
2016 Pop(new_target);
2017 }
2018 if (actual.is_reg()) {
2019 Pop(actual.reg());
2020 SmiUntag(actual.reg());
2021 }
2022 if (expected.is_reg()) {
2023 Pop(expected.reg());
2024 SmiUntag(expected.reg());
2025 }
2026 }
2027 bind(&skip_flooding);
2028 }
2029
2030
2031 void MacroAssembler::InvokeFunctionCode(Register function, Register new_target,
2032 const ParameterCount& expected,
2033 const ParameterCount& actual,
2034 InvokeFlag flag,
2035 const CallWrapper& call_wrapper) {
1995 // You can't call a function without a valid frame. 2036 // You can't call a function without a valid frame.
1996 DCHECK(flag == JUMP_FUNCTION || has_frame()); 2037 DCHECK(flag == JUMP_FUNCTION || has_frame());
2038 DCHECK(function.is(edi));
2039 DCHECK_IMPLIES(new_target.is_valid(), new_target.is(edx));
1997 2040
1998 // Ensure new target is passed in the correct register. Otherwise clear the 2041 if (call_wrapper.NeedsDebugStepCheck()) {
1999 // appropriate register in case new target is not given. 2042 FloodFunctionIfStepping(function, new_target, expected, actual);
2000 DCHECK_IMPLIES(new_target.is_valid(), new_target.is(edx)); 2043 }
2044
2045 // Clear the new.target register if not given.
2001 if (!new_target.is_valid()) { 2046 if (!new_target.is_valid()) {
2002 mov(edx, isolate()->factory()->undefined_value()); 2047 mov(edx, isolate()->factory()->undefined_value());
2003 } 2048 }
2004 2049
2005 Label done; 2050 Label done;
2006 bool definitely_mismatches = false; 2051 bool definitely_mismatches = false;
2007 InvokePrologue(expected, actual, &done, &definitely_mismatches, flag, 2052 InvokePrologue(expected, actual, &done, &definitely_mismatches, flag,
2008 Label::kNear, call_wrapper); 2053 Label::kNear, call_wrapper);
2009 if (!definitely_mismatches) { 2054 if (!definitely_mismatches) {
2055 // We call indirectly through the code field in the function to
2056 // allow recompilation to take effect without changing any of the
2057 // call sites.
2058 Operand code = FieldOperand(function, JSFunction::kCodeEntryOffset);
2010 if (flag == CALL_FUNCTION) { 2059 if (flag == CALL_FUNCTION) {
2011 call_wrapper.BeforeCall(CallSize(code)); 2060 call_wrapper.BeforeCall(CallSize(code));
2012 call(code); 2061 call(code);
2013 call_wrapper.AfterCall(); 2062 call_wrapper.AfterCall();
2014 } else { 2063 } else {
2015 DCHECK(flag == JUMP_FUNCTION); 2064 DCHECK(flag == JUMP_FUNCTION);
2016 jmp(code); 2065 jmp(code);
2017 } 2066 }
2018 bind(&done); 2067 bind(&done);
2019 } 2068 }
2020 } 2069 }
2021 2070
2022 2071
2023 void MacroAssembler::InvokeFunction(Register fun, 2072 void MacroAssembler::InvokeFunction(Register fun,
2024 Register new_target, 2073 Register new_target,
2025 const ParameterCount& actual, 2074 const ParameterCount& actual,
2026 InvokeFlag flag, 2075 InvokeFlag flag,
2027 const CallWrapper& call_wrapper) { 2076 const CallWrapper& call_wrapper) {
2028 // You can't call a function without a valid frame. 2077 // You can't call a function without a valid frame.
2029 DCHECK(flag == JUMP_FUNCTION || has_frame()); 2078 DCHECK(flag == JUMP_FUNCTION || has_frame());
2030 2079
2031 DCHECK(fun.is(edi)); 2080 DCHECK(fun.is(edi));
2032 mov(ebx, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); 2081 mov(ebx, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset));
2033 mov(esi, FieldOperand(edi, JSFunction::kContextOffset)); 2082 mov(esi, FieldOperand(edi, JSFunction::kContextOffset));
2034 mov(ebx, FieldOperand(ebx, SharedFunctionInfo::kFormalParameterCountOffset)); 2083 mov(ebx, FieldOperand(ebx, SharedFunctionInfo::kFormalParameterCountOffset));
2035 SmiUntag(ebx); 2084 SmiUntag(ebx);
2036 2085
2037 ParameterCount expected(ebx); 2086 ParameterCount expected(ebx);
2038 InvokeCode(FieldOperand(edi, JSFunction::kCodeEntryOffset), new_target, 2087 InvokeFunctionCode(edi, new_target, expected, actual, flag, call_wrapper);
2039 expected, actual, flag, call_wrapper);
2040 } 2088 }
2041 2089
2042 2090
2043 void MacroAssembler::InvokeFunction(Register fun, 2091 void MacroAssembler::InvokeFunction(Register fun,
2044 const ParameterCount& expected, 2092 const ParameterCount& expected,
2045 const ParameterCount& actual, 2093 const ParameterCount& actual,
2046 InvokeFlag flag, 2094 InvokeFlag flag,
2047 const CallWrapper& call_wrapper) { 2095 const CallWrapper& call_wrapper) {
2048 // You can't call a function without a valid frame. 2096 // You can't call a function without a valid frame.
2049 DCHECK(flag == JUMP_FUNCTION || has_frame()); 2097 DCHECK(flag == JUMP_FUNCTION || has_frame());
2050 2098
2051 DCHECK(fun.is(edi)); 2099 DCHECK(fun.is(edi));
2052 mov(esi, FieldOperand(edi, JSFunction::kContextOffset)); 2100 mov(esi, FieldOperand(edi, JSFunction::kContextOffset));
2053 2101
2054 InvokeCode(FieldOperand(edi, JSFunction::kCodeEntryOffset), no_reg, 2102 InvokeFunctionCode(edi, no_reg, expected, actual, flag, call_wrapper);
2055 expected, actual, flag, call_wrapper);
2056 } 2103 }
2057 2104
2058 2105
2059 void MacroAssembler::InvokeFunction(Handle<JSFunction> function, 2106 void MacroAssembler::InvokeFunction(Handle<JSFunction> function,
2060 const ParameterCount& expected, 2107 const ParameterCount& expected,
2061 const ParameterCount& actual, 2108 const ParameterCount& actual,
2062 InvokeFlag flag, 2109 InvokeFlag flag,
2063 const CallWrapper& call_wrapper) { 2110 const CallWrapper& call_wrapper) {
2064 LoadHeapObject(edi, function); 2111 LoadHeapObject(edi, function);
2065 InvokeFunction(edi, expected, actual, flag, call_wrapper); 2112 InvokeFunction(edi, expected, actual, flag, call_wrapper);
2066 } 2113 }
2067 2114
2068 2115
2069 void MacroAssembler::InvokeBuiltin(int native_context_index, InvokeFlag flag, 2116 void MacroAssembler::InvokeBuiltin(int native_context_index, InvokeFlag flag,
2070 const CallWrapper& call_wrapper) { 2117 const CallWrapper& call_wrapper) {
2071 // You can't call a builtin without a valid frame. 2118 // You can't call a builtin without a valid frame.
2072 DCHECK(flag == JUMP_FUNCTION || has_frame()); 2119 DCHECK(flag == JUMP_FUNCTION || has_frame());
2073 2120
2074 // Rely on the assertion to check that the number of provided 2121 // Rely on the assertion to check that the number of provided
2075 // arguments match the expected number of arguments. Fake a 2122 // arguments match the expected number of arguments. Fake a
2076 // parameter count to avoid emitting code to do the check. 2123 // parameter count to avoid emitting code to do the check.
2077 ParameterCount expected(0); 2124 ParameterCount expected(0);
2078 GetBuiltinFunction(edi, native_context_index); 2125 GetBuiltinFunction(edi, native_context_index);
2079 InvokeCode(FieldOperand(edi, JSFunction::kCodeEntryOffset), no_reg, 2126 InvokeFunctionCode(edi, no_reg, expected, expected, flag, call_wrapper);
2080 expected, expected, flag, call_wrapper);
2081 } 2127 }
2082 2128
2083 2129
2084 void MacroAssembler::GetBuiltinFunction(Register target, 2130 void MacroAssembler::GetBuiltinFunction(Register target,
2085 int native_context_index) { 2131 int native_context_index) {
2086 // Load the JavaScript builtin function from the builtins object. 2132 // Load the JavaScript builtin function from the builtins object.
2087 mov(target, GlobalObjectOperand()); 2133 mov(target, GlobalObjectOperand());
2088 mov(target, FieldOperand(target, JSGlobalObject::kNativeContextOffset)); 2134 mov(target, FieldOperand(target, JSGlobalObject::kNativeContextOffset));
2089 mov(target, ContextOperand(target, native_context_index)); 2135 mov(target, ContextOperand(target, native_context_index));
2090 } 2136 }
2091 2137
2092 2138
2093 void MacroAssembler::GetBuiltinEntry(Register target,
2094 int native_context_index) {
2095 DCHECK(!target.is(edi));
2096 // Load the JavaScript builtin function from the builtins object.
2097 GetBuiltinFunction(edi, native_context_index);
2098 // Load the code entry point from the function into the target register.
2099 mov(target, FieldOperand(edi, JSFunction::kCodeEntryOffset));
2100 }
2101
2102
2103 void MacroAssembler::LoadContext(Register dst, int context_chain_length) { 2139 void MacroAssembler::LoadContext(Register dst, int context_chain_length) {
2104 if (context_chain_length > 0) { 2140 if (context_chain_length > 0) {
2105 // Move up the chain of contexts to the context containing the slot. 2141 // Move up the chain of contexts to the context containing the slot.
2106 mov(dst, Operand(esi, Context::SlotOffset(Context::PREVIOUS_INDEX))); 2142 mov(dst, Operand(esi, Context::SlotOffset(Context::PREVIOUS_INDEX)));
2107 for (int i = 1; i < context_chain_length; i++) { 2143 for (int i = 1; i < context_chain_length; i++) {
2108 mov(dst, Operand(dst, Context::SlotOffset(Context::PREVIOUS_INDEX))); 2144 mov(dst, Operand(dst, Context::SlotOffset(Context::PREVIOUS_INDEX)));
2109 } 2145 }
2110 } else { 2146 } else {
2111 // Slot is in the current function context. Move it into the 2147 // Slot is in the current function context. Move it into the
2112 // destination register in case we store into it (the write barrier 2148 // destination register in case we store into it (the write barrier
(...skipping 1033 matching lines...) Expand 10 before | Expand all | Expand 10 after
3146 mov(eax, dividend); 3182 mov(eax, dividend);
3147 shr(eax, 31); 3183 shr(eax, 31);
3148 add(edx, eax); 3184 add(edx, eax);
3149 } 3185 }
3150 3186
3151 3187
3152 } // namespace internal 3188 } // namespace internal
3153 } // namespace v8 3189 } // namespace v8
3154 3190
3155 #endif // V8_TARGET_ARCH_IA32 3191 #endif // V8_TARGET_ARCH_IA32
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698