| OLD | NEW |
| 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 #include "src/debug/debug.h" | 5 #include "src/debug/debug.h" |
| 6 | 6 |
| 7 #include "src/api.h" | 7 #include "src/api.h" |
| 8 #include "src/arguments.h" | 8 #include "src/arguments.h" |
| 9 #include "src/bootstrapper.h" | 9 #include "src/bootstrapper.h" |
| 10 #include "src/code-stubs.h" | 10 #include "src/code-stubs.h" |
| 11 #include "src/codegen.h" | 11 #include "src/codegen.h" |
| 12 #include "src/compilation-cache.h" | 12 #include "src/compilation-cache.h" |
| 13 #include "src/compiler.h" | 13 #include "src/compiler.h" |
| 14 #include "src/deoptimizer.h" | 14 #include "src/deoptimizer.h" |
| 15 #include "src/execution.h" | 15 #include "src/execution.h" |
| 16 #include "src/frames-inl.h" | 16 #include "src/frames-inl.h" |
| 17 #include "src/full-codegen/full-codegen.h" | 17 #include "src/full-codegen/full-codegen.h" |
| 18 #include "src/global-handles.h" | 18 #include "src/global-handles.h" |
| 19 #include "src/interpreter/bytecodes.h" |
| 19 #include "src/isolate-inl.h" | 20 #include "src/isolate-inl.h" |
| 20 #include "src/list.h" | 21 #include "src/list.h" |
| 21 #include "src/log.h" | 22 #include "src/log.h" |
| 22 #include "src/messages.h" | 23 #include "src/messages.h" |
| 23 #include "src/snapshot/natives.h" | 24 #include "src/snapshot/natives.h" |
| 24 | 25 |
| 25 #include "include/v8-debug.h" | 26 #include "include/v8-debug.h" |
| 26 | 27 |
| 27 namespace v8 { | 28 namespace v8 { |
| 28 namespace internal { | 29 namespace internal { |
| (...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 62 int code_offset, int position, | 63 int code_offset, int position, |
| 63 int statement_position) | 64 int statement_position) |
| 64 : debug_info_(debug_info), | 65 : debug_info_(debug_info), |
| 65 code_offset_(code_offset), | 66 code_offset_(code_offset), |
| 66 type_(type), | 67 type_(type), |
| 67 position_(position), | 68 position_(position), |
| 68 statement_position_(statement_position) {} | 69 statement_position_(statement_position) {} |
| 69 | 70 |
| 70 BreakLocation::Iterator* BreakLocation::GetIterator( | 71 BreakLocation::Iterator* BreakLocation::GetIterator( |
| 71 Handle<DebugInfo> debug_info, BreakLocatorType type) { | 72 Handle<DebugInfo> debug_info, BreakLocatorType type) { |
| 72 if (debug_info->shared()->HasBytecodeArray()) { | 73 if (debug_info->abstract_code()->IsBytecodeArray()) { |
| 73 UNIMPLEMENTED(); | 74 return new BytecodeArrayIterator(debug_info, type); |
| 74 return nullptr; | 75 } else { |
| 76 return new CodeIterator(debug_info, type); |
| 75 } | 77 } |
| 76 return new CodeIterator(debug_info, type); | |
| 77 } | 78 } |
| 78 | 79 |
| 79 BreakLocation::Iterator::Iterator(Handle<DebugInfo> debug_info, | 80 BreakLocation::Iterator::Iterator(Handle<DebugInfo> debug_info) |
| 80 BreakLocatorType type) | |
| 81 : debug_info_(debug_info), | 81 : debug_info_(debug_info), |
| 82 break_index_(-1), | 82 break_index_(-1), |
| 83 position_(1), | 83 position_(1), |
| 84 statement_position_(1) { | 84 statement_position_(1) {} |
| 85 } | |
| 86 | 85 |
| 87 BreakLocation::CodeIterator::CodeIterator(Handle<DebugInfo> debug_info, | 86 BreakLocation::CodeIterator::CodeIterator(Handle<DebugInfo> debug_info, |
| 88 BreakLocatorType type) | 87 BreakLocatorType type) |
| 89 : Iterator(debug_info, type), | 88 : Iterator(debug_info), |
| 90 reloc_iterator_(debug_info->abstract_code()->GetCode(), | 89 reloc_iterator_(debug_info->abstract_code()->GetCode(), |
| 91 GetModeMask(type)) { | 90 GetModeMask(type)) { |
| 92 if (!Done()) Next(); | 91 if (!Done()) Next(); |
| 93 } | 92 } |
| 94 | 93 |
| 95 int BreakLocation::CodeIterator::GetModeMask(BreakLocatorType type) { | 94 int BreakLocation::CodeIterator::GetModeMask(BreakLocatorType type) { |
| 96 int mask = 0; | 95 int mask = 0; |
| 97 mask |= RelocInfo::ModeMask(RelocInfo::POSITION); | 96 mask |= RelocInfo::ModeMask(RelocInfo::POSITION); |
| 98 mask |= RelocInfo::ModeMask(RelocInfo::STATEMENT_POSITION); | 97 mask |= RelocInfo::ModeMask(RelocInfo::STATEMENT_POSITION); |
| 99 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_RETURN); | 98 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_RETURN); |
| 100 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_CALL); | 99 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_CALL); |
| 101 if (type == ALL_BREAK_LOCATIONS) { | 100 if (type == ALL_BREAK_LOCATIONS) { |
| 102 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_POSITION); | 101 mask |= RelocInfo::ModeMask(RelocInfo::DEBUG_BREAK_SLOT_AT_POSITION); |
| 103 mask |= RelocInfo::ModeMask(RelocInfo::DEBUGGER_STATEMENT); | 102 mask |= RelocInfo::ModeMask(RelocInfo::DEBUGGER_STATEMENT); |
| 104 } | 103 } |
| 105 return mask; | 104 return mask; |
| 106 } | 105 } |
| 107 | 106 |
| 108 void BreakLocation::CodeIterator::Next() { | 107 void BreakLocation::CodeIterator::Next() { |
| 109 DisallowHeapAllocation no_gc; | 108 DisallowHeapAllocation no_gc; |
| 110 DCHECK(!Done()); | 109 DCHECK(!Done()); |
| 111 | 110 |
| 112 // Iterate through reloc info for code and original code stopping at each | 111 // Iterate through reloc info stopping at each breakable code target. |
| 113 // breakable code target. | |
| 114 bool first = break_index_ == -1; | 112 bool first = break_index_ == -1; |
| 115 while (!Done()) { | 113 while (!Done()) { |
| 116 if (!first) reloc_iterator_.next(); | 114 if (!first) reloc_iterator_.next(); |
| 117 first = false; | 115 first = false; |
| 118 if (Done()) return; | 116 if (Done()) return; |
| 119 | 117 |
| 120 // Whenever a statement position or (plain) position is passed update the | 118 // Whenever a statement position or (plain) position is passed update the |
| 121 // current value of these. | 119 // current value of these. |
| 122 if (RelocInfo::IsPosition(rmode())) { | 120 if (RelocInfo::IsPosition(rmode())) { |
| 123 if (RelocInfo::IsStatementPosition(rmode())) { | 121 if (RelocInfo::IsStatementPosition(rmode())) { |
| (...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 162 type = DEBUGGER_STATEMENT; | 160 type = DEBUGGER_STATEMENT; |
| 163 } else if (RelocInfo::IsDebugBreakSlot(rmode())) { | 161 } else if (RelocInfo::IsDebugBreakSlot(rmode())) { |
| 164 type = DEBUG_BREAK_SLOT; | 162 type = DEBUG_BREAK_SLOT; |
| 165 } else { | 163 } else { |
| 166 type = NOT_DEBUG_BREAK; | 164 type = NOT_DEBUG_BREAK; |
| 167 } | 165 } |
| 168 return BreakLocation(debug_info_, type, code_offset(), position(), | 166 return BreakLocation(debug_info_, type, code_offset(), position(), |
| 169 statement_position()); | 167 statement_position()); |
| 170 } | 168 } |
| 171 | 169 |
| 170 BreakLocation::BytecodeArrayIterator::BytecodeArrayIterator( |
| 171 Handle<DebugInfo> debug_info, BreakLocatorType type) |
| 172 : Iterator(debug_info), |
| 173 source_position_iterator_( |
| 174 debug_info->abstract_code()->GetBytecodeArray()), |
| 175 break_locator_type_(type), |
| 176 start_position_(debug_info->shared()->start_position()) { |
| 177 if (!Done()) Next(); |
| 178 } |
| 179 |
| 180 void BreakLocation::BytecodeArrayIterator::Next() { |
| 181 DisallowHeapAllocation no_gc; |
| 182 DCHECK(!Done()); |
| 183 bool first = break_index_ == -1; |
| 184 while (!Done()) { |
| 185 if (!first) source_position_iterator_.Advance(); |
| 186 first = false; |
| 187 if (Done()) return; |
| 188 position_ = source_position_iterator_.source_position() - start_position_; |
| 189 if (source_position_iterator_.is_statement()) { |
| 190 statement_position_ = position_; |
| 191 } |
| 192 DCHECK(position_ >= 0); |
| 193 DCHECK(statement_position_ >= 0); |
| 194 break_index_++; |
| 195 |
| 196 if (break_locator_type_ == ALL_BREAK_LOCATIONS) break; |
| 197 |
| 198 DCHECK_EQ(CALLS_AND_RETURNS, break_locator_type_); |
| 199 enum DebugBreakType type = GetDebugBreakType(); |
| 200 if (type == DEBUG_BREAK_SLOT_AT_CALL || |
| 201 type == DEBUG_BREAK_SLOT_AT_RETURN) { |
| 202 break; |
| 203 } |
| 204 } |
| 205 } |
| 206 |
| 207 BreakLocation::DebugBreakType |
| 208 BreakLocation::BytecodeArrayIterator::GetDebugBreakType() { |
| 209 BytecodeArray* bytecode_array = |
| 210 debug_info_->abstract_code()->GetBytecodeArray(); |
| 211 interpreter::Bytecode bytecode = |
| 212 static_cast<interpreter::Bytecode>(bytecode_array->get(code_offset())); |
| 213 |
| 214 if (bytecode == interpreter::Bytecode::kDebugger) { |
| 215 return DEBUGGER_STATEMENT; |
| 216 } else if (bytecode == interpreter::Bytecode::kReturn) { |
| 217 return DEBUG_BREAK_SLOT_AT_RETURN; |
| 218 } else if (interpreter::Bytecodes::IsCallOrNew(bytecode)) { |
| 219 return DEBUG_BREAK_SLOT_AT_CALL; |
| 220 } else if (source_position_iterator_.is_statement()) { |
| 221 return DEBUG_BREAK_SLOT; |
| 222 } else { |
| 223 return NOT_DEBUG_BREAK; |
| 224 } |
| 225 } |
| 226 |
| 227 BreakLocation BreakLocation::BytecodeArrayIterator::GetBreakLocation() { |
| 228 return BreakLocation(debug_info_, GetDebugBreakType(), code_offset(), |
| 229 position(), statement_position()); |
| 230 } |
| 231 |
| 172 // Find the break point at the supplied address, or the closest one before | 232 // Find the break point at the supplied address, or the closest one before |
| 173 // the address. | 233 // the address. |
| 174 BreakLocation BreakLocation::FromCodeOffset(Handle<DebugInfo> debug_info, | 234 BreakLocation BreakLocation::FromCodeOffset(Handle<DebugInfo> debug_info, |
| 175 int offset) { | 235 int offset) { |
| 176 base::SmartPointer<Iterator> it(GetIterator(debug_info)); | 236 base::SmartPointer<Iterator> it(GetIterator(debug_info)); |
| 177 it->SkipTo(BreakIndexFromCodeOffset(debug_info, offset)); | 237 it->SkipTo(BreakIndexFromCodeOffset(debug_info, offset)); |
| 178 return it->GetBreakLocation(); | 238 return it->GetBreakLocation(); |
| 179 } | 239 } |
| 180 | 240 |
| 181 // Move GetFirstFrameSummary Definition to here as FromFrame use it. | 241 // Move GetFirstFrameSummary Definition to here as FromFrame use it. |
| (...skipping 153 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 335 if (abstract_code()->IsCode()) { | 395 if (abstract_code()->IsCode()) { |
| 336 Code* code = abstract_code()->GetCode(); | 396 Code* code = abstract_code()->GetCode(); |
| 337 DCHECK(code->kind() == Code::FUNCTION); | 397 DCHECK(code->kind() == Code::FUNCTION); |
| 338 Builtins* builtins = isolate()->builtins(); | 398 Builtins* builtins = isolate()->builtins(); |
| 339 Handle<Code> target = IsReturn() ? builtins->Return_DebugBreak() | 399 Handle<Code> target = IsReturn() ? builtins->Return_DebugBreak() |
| 340 : builtins->Slot_DebugBreak(); | 400 : builtins->Slot_DebugBreak(); |
| 341 Address pc = code->instruction_start() + code_offset(); | 401 Address pc = code->instruction_start() + code_offset(); |
| 342 DebugCodegen::PatchDebugBreakSlot(isolate(), pc, target); | 402 DebugCodegen::PatchDebugBreakSlot(isolate(), pc, target); |
| 343 DCHECK(IsDebugBreak()); | 403 DCHECK(IsDebugBreak()); |
| 344 } else { | 404 } else { |
| 345 UNIMPLEMENTED(); | 405 // TODO(yangguo): implement this once we have a way to record break points. |
| 346 } | 406 } |
| 347 } | 407 } |
| 348 | 408 |
| 349 | 409 |
| 350 void BreakLocation::ClearDebugBreak() { | 410 void BreakLocation::ClearDebugBreak() { |
| 351 // Debugger statement always calls debugger. No need to modify it. | 411 // Debugger statement always calls debugger. No need to modify it. |
| 352 if (IsDebuggerStatement()) return; | 412 if (IsDebuggerStatement()) return; |
| 353 | 413 |
| 354 DCHECK(IsDebugBreakSlot()); | 414 DCHECK(IsDebugBreakSlot()); |
| 355 if (abstract_code()->IsCode()) { | 415 if (abstract_code()->IsCode()) { |
| 356 Code* code = abstract_code()->GetCode(); | 416 Code* code = abstract_code()->GetCode(); |
| 357 DCHECK(code->kind() == Code::FUNCTION); | 417 DCHECK(code->kind() == Code::FUNCTION); |
| 358 Address pc = code->instruction_start() + code_offset(); | 418 Address pc = code->instruction_start() + code_offset(); |
| 359 DebugCodegen::ClearDebugBreakSlot(isolate(), pc); | 419 DebugCodegen::ClearDebugBreakSlot(isolate(), pc); |
| 360 DCHECK(!IsDebugBreak()); | 420 DCHECK(!IsDebugBreak()); |
| 361 } else { | 421 } else { |
| 362 UNIMPLEMENTED(); | 422 // TODO(yangguo): implement this once we have a way to record break points. |
| 363 } | 423 } |
| 364 } | 424 } |
| 365 | 425 |
| 366 | 426 |
| 367 bool BreakLocation::IsDebugBreak() const { | 427 bool BreakLocation::IsDebugBreak() const { |
| 368 if (IsDebuggerStatement()) return false; | 428 if (IsDebuggerStatement()) return false; |
| 369 DCHECK(IsDebugBreakSlot()); | 429 DCHECK(IsDebugBreakSlot()); |
| 370 if (abstract_code()->IsCode()) { | 430 if (abstract_code()->IsCode()) { |
| 371 Code* code = abstract_code()->GetCode(); | 431 Code* code = abstract_code()->GetCode(); |
| 372 DCHECK(code->kind() == Code::FUNCTION); | 432 DCHECK(code->kind() == Code::FUNCTION); |
| 373 Address pc = code->instruction_start() + code_offset(); | 433 Address pc = code->instruction_start() + code_offset(); |
| 374 return DebugCodegen::DebugBreakSlotIsPatched(pc); | 434 return DebugCodegen::DebugBreakSlotIsPatched(pc); |
| 375 } else { | 435 } else { |
| 376 UNIMPLEMENTED(); | 436 // TODO(yangguo): implement this once we have a way to record break points. |
| 377 return false; | 437 return false; |
| 378 } | 438 } |
| 379 } | 439 } |
| 380 | 440 |
| 381 | 441 |
| 382 Handle<Object> BreakLocation::BreakPointObjects() const { | 442 Handle<Object> BreakLocation::BreakPointObjects() const { |
| 383 return debug_info_->GetBreakPointObjects(code_offset_); | 443 return debug_info_->GetBreakPointObjects(code_offset_); |
| 384 } | 444 } |
| 385 | 445 |
| 386 void DebugFeatureTracker::Track(DebugFeatureTracker::Feature feature) { | 446 void DebugFeatureTracker::Track(DebugFeatureTracker::Feature feature) { |
| (...skipping 769 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1156 public: | 1216 public: |
| 1157 explicit RedirectActiveFunctions(SharedFunctionInfo* shared) | 1217 explicit RedirectActiveFunctions(SharedFunctionInfo* shared) |
| 1158 : shared_(shared) { | 1218 : shared_(shared) { |
| 1159 DCHECK(shared->HasDebugCode()); | 1219 DCHECK(shared->HasDebugCode()); |
| 1160 } | 1220 } |
| 1161 | 1221 |
| 1162 void VisitThread(Isolate* isolate, ThreadLocalTop* top) { | 1222 void VisitThread(Isolate* isolate, ThreadLocalTop* top) { |
| 1163 for (JavaScriptFrameIterator it(isolate, top); !it.done(); it.Advance()) { | 1223 for (JavaScriptFrameIterator it(isolate, top); !it.done(); it.Advance()) { |
| 1164 JavaScriptFrame* frame = it.frame(); | 1224 JavaScriptFrame* frame = it.frame(); |
| 1165 JSFunction* function = frame->function(); | 1225 JSFunction* function = frame->function(); |
| 1226 if (frame->is_interpreted()) { |
| 1227 // TODO(yangguo): replace dispatch table for activated frames. |
| 1228 continue; |
| 1229 } |
| 1166 if (frame->is_optimized()) continue; | 1230 if (frame->is_optimized()) continue; |
| 1167 if (!function->Inlines(shared_)) continue; | 1231 if (!function->Inlines(shared_)) continue; |
| 1168 | 1232 |
| 1169 Code* frame_code = frame->LookupCode(); | 1233 Code* frame_code = frame->LookupCode(); |
| 1170 DCHECK(frame_code->kind() == Code::FUNCTION); | 1234 DCHECK(frame_code->kind() == Code::FUNCTION); |
| 1171 if (frame_code->has_debug_break_slots()) continue; | 1235 if (frame_code->has_debug_break_slots()) continue; |
| 1172 | 1236 |
| 1173 Code* new_code = function->shared()->code(); | 1237 Code* new_code = function->shared()->code(); |
| 1174 Address old_pc = frame->pc(); | 1238 Address old_pc = frame->pc(); |
| 1175 Address new_pc = ComputeNewPcForRedirect(new_code, frame_code, old_pc); | 1239 Address new_pc = ComputeNewPcForRedirect(new_code, frame_code, old_pc); |
| (...skipping 1329 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2505 } | 2569 } |
| 2506 | 2570 |
| 2507 | 2571 |
| 2508 void LockingCommandMessageQueue::Clear() { | 2572 void LockingCommandMessageQueue::Clear() { |
| 2509 base::LockGuard<base::Mutex> lock_guard(&mutex_); | 2573 base::LockGuard<base::Mutex> lock_guard(&mutex_); |
| 2510 queue_.Clear(); | 2574 queue_.Clear(); |
| 2511 } | 2575 } |
| 2512 | 2576 |
| 2513 } // namespace internal | 2577 } // namespace internal |
| 2514 } // namespace v8 | 2578 } // namespace v8 |
| OLD | NEW |