// 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 "src/runtime/runtime-utils.h" #include #include "src/arguments.h" #include "src/frames-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/isolate-inl.h" #include "src/ostreams.h" namespace v8 { namespace internal { RUNTIME_FUNCTION(Runtime_InterpreterNewClosure) { HandleScope scope(isolate); DCHECK_EQ(2, args.length()); CONVERT_ARG_HANDLE_CHECKED(SharedFunctionInfo, shared, 0); CONVERT_SMI_ARG_CHECKED(pretenured_flag, 1); Handle context(isolate->context(), isolate); return *isolate->factory()->NewFunctionFromSharedFunctionInfo( shared, context, static_cast(pretenured_flag)); } 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 PrintRegisters(std::ostream& os, bool is_input, interpreter::BytecodeArrayIterator& bytecode_iterator, Handle accumulator) { static const char kAccumulator[] = "accumulator"; static const int kRegFieldWidth = static_cast(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 << " ]" << std::endl; } // Print the registers. JavaScriptFrameIterator frame_iterator( bytecode_iterator.bytecode_array()->GetIsolate()); InterpretedFrame* frame = reinterpret_cast(frame_iterator.frame()); 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); 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(kRegFieldWidth) << interpreter::Register(reg_index).ToString( bytecode_iterator.bytecode_array()->parameter_count()) << kArrowDirection; reg_object->ShortPrint(os); os << " ]" << std::endl; } } } if (FLAG_log_colour) { os << kNormalColourCode; } } } // namespace RUNTIME_FUNCTION(Runtime_InterpreterTraceBytecodeEntry) { SealHandleScope shs(isolate); DCHECK_EQ(3, args.length()); CONVERT_ARG_HANDLE_CHECKED(BytecodeArray, bytecode_array, 0); CONVERT_SMI_ARG_CHECKED(bytecode_offset, 1); CONVERT_ARG_HANDLE_CHECKED(Object, accumulator, 2); OFStream os(stdout); int offset = bytecode_offset - BytecodeArray::kHeaderSize + kHeapObjectTag; interpreter::BytecodeArrayIterator bytecode_iterator(bytecode_array); AdvanceToOffsetForTracing(bytecode_iterator, offset); if (offset == bytecode_iterator.current_offset()) { // Print bytecode. const uint8_t* base_address = bytecode_array->GetFirstBytecodeAddress(); const uint8_t* bytecode_address = base_address + offset; os << " -> " << static_cast(bytecode_address) << " @ " << std::setw(4) << offset << " : "; interpreter::BytecodeDecoder::Decode(os, bytecode_address, bytecode_array->parameter_count()); os << std::endl; // Print all input registers and accumulator. PrintRegisters(os, true, bytecode_iterator, accumulator); os << std::flush; } return isolate->heap()->undefined_value(); } RUNTIME_FUNCTION(Runtime_InterpreterTraceBytecodeExit) { SealHandleScope shs(isolate); DCHECK_EQ(3, args.length()); CONVERT_ARG_HANDLE_CHECKED(BytecodeArray, bytecode_array, 0); CONVERT_SMI_ARG_CHECKED(bytecode_offset, 1); CONVERT_ARG_HANDLE_CHECKED(Object, accumulator, 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()) { OFStream os(stdout); // Print all output registers and accumulator. PrintRegisters(os, false, bytecode_iterator, accumulator); os << std::flush; } return isolate->heap()->undefined_value(); } RUNTIME_FUNCTION(Runtime_InterpreterClearPendingMessage) { SealHandleScope shs(isolate); DCHECK_EQ(0, args.length()); Object* message = isolate->thread_local_top()->pending_message_obj_; isolate->clear_pending_message(); return message; } RUNTIME_FUNCTION(Runtime_InterpreterSetPendingMessage) { SealHandleScope shs(isolate); DCHECK_EQ(1, args.length()); CONVERT_ARG_HANDLE_CHECKED(Object, message, 0); isolate->thread_local_top()->pending_message_obj_ = *message; return isolate->heap()->undefined_value(); } RUNTIME_FUNCTION(Runtime_InterpreterAdvanceBytecodeOffset) { SealHandleScope shs(isolate); DCHECK_EQ(2, args.length()); CONVERT_ARG_HANDLE_CHECKED(BytecodeArray, bytecode_array, 0); CONVERT_SMI_ARG_CHECKED(bytecode_offset, 1); interpreter::BytecodeArrayIterator it(bytecode_array); int offset = bytecode_offset - BytecodeArray::kHeaderSize + kHeapObjectTag; while (it.current_offset() < offset) it.Advance(); DCHECK_EQ(offset, it.current_offset()); it.Advance(); // Advance by one bytecode. offset = it.current_offset() + BytecodeArray::kHeaderSize - kHeapObjectTag; return Smi::FromInt(offset); } } // namespace internal } // namespace v8