| // Copyright 2015 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <iomanip> |
| |
| #include "src/execution/arguments-inl.h" |
| #include "src/execution/frames-inl.h" |
| #include "src/execution/isolate-inl.h" |
| #include "src/interpreter/bytecode-array-iterator.h" |
| #include "src/interpreter/bytecode-decoder.h" |
| #include "src/interpreter/bytecode-flags.h" |
| #include "src/interpreter/bytecode-register.h" |
| #include "src/interpreter/bytecodes.h" |
| #include "src/interpreter/interpreter.h" |
| #include "src/logging/counters.h" |
| #include "src/runtime/runtime-utils.h" |
| #include "src/snapshot/snapshot.h" |
| #include "src/utils/ostreams.h" |
| |
| namespace v8 { |
| namespace internal { |
| |
| #ifdef V8_TRACE_UNOPTIMIZED |
| |
| namespace { |
| |
| void AdvanceToOffsetForTracing( |
| interpreter::BytecodeArrayIterator& bytecode_iterator, int offset) { |
| while (bytecode_iterator.current_offset() + |
| bytecode_iterator.current_bytecode_size() <= |
| offset) { |
| bytecode_iterator.Advance(); |
| } |
| DCHECK(bytecode_iterator.current_offset() == offset || |
| ((bytecode_iterator.current_offset() + 1) == offset && |
| bytecode_iterator.current_operand_scale() > |
| interpreter::OperandScale::kSingle)); |
| } |
| |
| void PrintRegisterRange(UnoptimizedFrame* frame, std::ostream& os, |
| interpreter::BytecodeArrayIterator& bytecode_iterator, |
| const int& reg_field_width, const char* arrow_direction, |
| interpreter::Register first_reg, int range) { |
| for (int reg_index = first_reg.index(); reg_index < first_reg.index() + range; |
| reg_index++) { |
| Object reg_object = frame->ReadInterpreterRegister(reg_index); |
| os << " [ " << std::setw(reg_field_width) |
| << interpreter::Register(reg_index).ToString() << arrow_direction; |
| reg_object.ShortPrint(os); |
| os << " ]" << std::endl; |
| } |
| } |
| |
| void PrintRegisters(UnoptimizedFrame* frame, std::ostream& os, bool is_input, |
| interpreter::BytecodeArrayIterator& bytecode_iterator, |
| Handle<Object> accumulator) { |
| static const char kAccumulator[] = "accumulator"; |
| static const int kRegFieldWidth = static_cast<int>(sizeof(kAccumulator) - 1); |
| static const char* kInputColourCode = "\033[0;36m"; |
| static const char* kOutputColourCode = "\033[0;35m"; |
| static const char* kNormalColourCode = "\033[0;m"; |
| const char* kArrowDirection = is_input ? " -> " : " <- "; |
| if (FLAG_log_colour) { |
| os << (is_input ? kInputColourCode : kOutputColourCode); |
| } |
| |
| interpreter::Bytecode bytecode = bytecode_iterator.current_bytecode(); |
| |
| // Print accumulator. |
| if ((is_input && interpreter::Bytecodes::ReadsAccumulator(bytecode)) || |
| (!is_input && interpreter::Bytecodes::WritesAccumulator(bytecode))) { |
| os << " [ " << kAccumulator << kArrowDirection; |
| accumulator->ShortPrint(os); |
| os << " ]" << std::endl; |
| } |
| |
| // Print the registers. |
| int operand_count = interpreter::Bytecodes::NumberOfOperands(bytecode); |
| for (int operand_index = 0; operand_index < operand_count; operand_index++) { |
| interpreter::OperandType operand_type = |
| interpreter::Bytecodes::GetOperandType(bytecode, operand_index); |
| bool should_print = |
| is_input |
| ? interpreter::Bytecodes::IsRegisterInputOperandType(operand_type) |
| : interpreter::Bytecodes::IsRegisterOutputOperandType(operand_type); |
| if (should_print) { |
| interpreter::Register first_reg = |
| bytecode_iterator.GetRegisterOperand(operand_index); |
| int range = bytecode_iterator.GetRegisterOperandRange(operand_index); |
| PrintRegisterRange(frame, os, bytecode_iterator, kRegFieldWidth, |
| kArrowDirection, first_reg, range); |
| } |
| } |
| if (!is_input && interpreter::Bytecodes::IsShortStar(bytecode)) { |
| PrintRegisterRange(frame, os, bytecode_iterator, kRegFieldWidth, |
| kArrowDirection, |
| interpreter::Register::FromShortStar(bytecode), 1); |
| } |
| if (FLAG_log_colour) { |
| os << kNormalColourCode; |
| } |
| } |
| |
| } // namespace |
| |
| RUNTIME_FUNCTION(Runtime_TraceUnoptimizedBytecodeEntry) { |
| if (!FLAG_trace_ignition && !FLAG_trace_baseline_exec) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| JavaScriptFrameIterator frame_iterator(isolate); |
| UnoptimizedFrame* frame = |
| reinterpret_cast<UnoptimizedFrame*>(frame_iterator.frame()); |
| |
| if (frame->is_interpreted() && !FLAG_trace_ignition) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| if (frame->is_baseline() && !FLAG_trace_baseline_exec) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| SealHandleScope shs(isolate); |
| DCHECK_EQ(3, args.length()); |
| Handle<BytecodeArray> bytecode_array = args.at<BytecodeArray>(0); |
| int bytecode_offset = args.smi_value_at(1); |
| Handle<Object> accumulator = args.at(2); |
| |
| int offset = bytecode_offset - BytecodeArray::kHeaderSize + kHeapObjectTag; |
| interpreter::BytecodeArrayIterator bytecode_iterator(bytecode_array); |
| AdvanceToOffsetForTracing(bytecode_iterator, offset); |
| if (offset == bytecode_iterator.current_offset()) { |
| StdoutStream os; |
| |
| // Print bytecode. |
| const uint8_t* base_address = reinterpret_cast<const uint8_t*>( |
| bytecode_array->GetFirstBytecodeAddress()); |
| const uint8_t* bytecode_address = base_address + offset; |
| |
| if (frame->is_baseline()) { |
| os << "B-> "; |
| } else { |
| os << " -> "; |
| } |
| os << static_cast<const void*>(bytecode_address) << " @ " << std::setw(4) |
| << offset << " : "; |
| interpreter::BytecodeDecoder::Decode(os, bytecode_address); |
| os << std::endl; |
| // Print all input registers and accumulator. |
| PrintRegisters(frame, os, true, bytecode_iterator, accumulator); |
| |
| os << std::flush; |
| } |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| RUNTIME_FUNCTION(Runtime_TraceUnoptimizedBytecodeExit) { |
| if (!FLAG_trace_ignition && !FLAG_trace_baseline_exec) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| JavaScriptFrameIterator frame_iterator(isolate); |
| UnoptimizedFrame* frame = |
| reinterpret_cast<UnoptimizedFrame*>(frame_iterator.frame()); |
| |
| if (frame->is_interpreted() && !FLAG_trace_ignition) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| if (frame->is_baseline() && !FLAG_trace_baseline_exec) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| SealHandleScope shs(isolate); |
| DCHECK_EQ(3, args.length()); |
| Handle<BytecodeArray> bytecode_array = args.at<BytecodeArray>(0); |
| int bytecode_offset = args.smi_value_at(1); |
| Handle<Object> accumulator = args.at(2); |
| |
| int offset = bytecode_offset - BytecodeArray::kHeaderSize + kHeapObjectTag; |
| interpreter::BytecodeArrayIterator bytecode_iterator(bytecode_array); |
| AdvanceToOffsetForTracing(bytecode_iterator, offset); |
| // The offset comparison here ensures registers only printed when the |
| // (potentially) widened bytecode has completed. The iterator reports |
| // the offset as the offset of the prefix bytecode. |
| if (bytecode_iterator.current_operand_scale() == |
| interpreter::OperandScale::kSingle || |
| offset > bytecode_iterator.current_offset()) { |
| StdoutStream os; |
| |
| // Print all output registers and accumulator. |
| PrintRegisters(frame, os, false, bytecode_iterator, accumulator); |
| os << std::flush; |
| } |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| #endif |
| |
| #ifdef V8_TRACE_FEEDBACK_UPDATES |
| |
| RUNTIME_FUNCTION(Runtime_TraceUpdateFeedback) { |
| if (!FLAG_trace_feedback_updates) { |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| SealHandleScope shs(isolate); |
| DCHECK_EQ(3, args.length()); |
| Handle<JSFunction> function = args.at<JSFunction>(0); |
| int slot = args.smi_value_at(1); |
| auto reason = String::cast(args[2]); |
| |
| int slot_count = function->feedback_vector().metadata().slot_count(); |
| |
| StdoutStream os; |
| os << "[Feedback slot " << slot << "/" << slot_count << " in "; |
| function->shared().ShortPrint(os); |
| os << " updated to "; |
| function->feedback_vector().FeedbackSlotPrint(os, FeedbackSlot(slot)); |
| os << " - "; |
| |
| StringCharacterStream stream(reason); |
| while (stream.HasMore()) { |
| uint16_t character = stream.GetNext(); |
| PrintF("%c", character); |
| } |
| |
| os << "]" << std::endl; |
| |
| return ReadOnlyRoots(isolate).undefined_value(); |
| } |
| |
| #endif |
| |
| } // namespace internal |
| } // namespace v8 |