OLD | NEW |
1 // Copyright 2015 the V8 project authors. All rights reserved. | 1 // Copyright 2015 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/interpreter/bytecodes.h" | 5 #include "src/interpreter/bytecodes.h" |
6 | 6 |
7 #include "src/interpreter/bytecode-array-builder.h" | 7 #include "src/frames.h" |
8 | 8 |
9 namespace v8 { | 9 namespace v8 { |
10 namespace internal { | 10 namespace internal { |
11 namespace interpreter { | 11 namespace interpreter { |
12 | 12 |
13 // Maximum number of operands a bytecode may have. | 13 // Maximum number of operands a bytecode may have. |
14 static const int kMaxOperands = 3; | 14 static const int kMaxOperands = 3; |
15 | 15 |
16 // kBytecodeTable relies on kNone being the same as zero to detect length. | 16 // kBytecodeTable relies on kNone being the same as zero to detect length. |
17 STATIC_ASSERT(static_cast<int>(OperandType::kNone) == 0); | 17 STATIC_ASSERT(static_cast<int>(OperandType::kNone) == 0); |
(...skipping 78 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
96 | 96 |
97 // static | 97 // static |
98 int Bytecodes::MaximumNumberOfOperands() { return kMaxOperands; } | 98 int Bytecodes::MaximumNumberOfOperands() { return kMaxOperands; } |
99 | 99 |
100 | 100 |
101 // static | 101 // static |
102 int Bytecodes::MaximumSize() { return 1 + kMaxOperands; } | 102 int Bytecodes::MaximumSize() { return 1 + kMaxOperands; } |
103 | 103 |
104 | 104 |
105 // static | 105 // static |
106 std::ostream& Bytecodes::Decode(std::ostream& os, | 106 std::ostream& Bytecodes::Decode(std::ostream& os, const uint8_t* bytecode_start, |
107 const uint8_t* bytecode_start) { | 107 int parameter_count) { |
108 Vector<char> buf = Vector<char>::New(50); | 108 Vector<char> buf = Vector<char>::New(50); |
109 | 109 |
110 Bytecode bytecode = Bytecodes::FromByte(bytecode_start[0]); | 110 Bytecode bytecode = Bytecodes::FromByte(bytecode_start[0]); |
111 int bytecode_size = Bytecodes::Size(bytecode); | 111 int bytecode_size = Bytecodes::Size(bytecode); |
112 | 112 |
113 for (int i = 0; i < bytecode_size; i++) { | 113 for (int i = 0; i < bytecode_size; i++) { |
114 SNPrintF(buf, "%02x ", bytecode_start[i]); | 114 SNPrintF(buf, "%02x ", bytecode_start[i]); |
115 os << buf.start(); | 115 os << buf.start(); |
116 } | 116 } |
117 for (int i = bytecode_size; i < Bytecodes::MaximumSize(); i++) { | 117 for (int i = bytecode_size; i < Bytecodes::MaximumSize(); i++) { |
118 os << " "; | 118 os << " "; |
119 } | 119 } |
120 | 120 |
121 os << bytecode << " "; | 121 os << bytecode << " "; |
122 | 122 |
123 const uint8_t* operands_start = bytecode_start + 1; | 123 const uint8_t* operands_start = bytecode_start + 1; |
124 int operands_size = bytecode_size - 1; | 124 int operands_size = bytecode_size - 1; |
125 for (int i = 0; i < operands_size; i++) { | 125 for (int i = 0; i < operands_size; i++) { |
126 OperandType op_type = GetOperandType(bytecode, i); | 126 OperandType op_type = GetOperandType(bytecode, i); |
127 uint8_t operand = operands_start[i]; | 127 uint8_t operand = operands_start[i]; |
128 switch (op_type) { | 128 switch (op_type) { |
129 case interpreter::OperandType::kIdx: | 129 case interpreter::OperandType::kIdx: |
130 os << "[" << static_cast<unsigned int>(operand) << "]"; | 130 os << "[" << static_cast<unsigned int>(operand) << "]"; |
131 break; | 131 break; |
132 case interpreter::OperandType::kImm8: | 132 case interpreter::OperandType::kImm8: |
133 os << "#" << static_cast<int>(operand); | 133 os << "#" << static_cast<int>(operand); |
134 break; | 134 break; |
135 case interpreter::OperandType::kReg: | 135 case interpreter::OperandType::kReg: { |
136 os << "r" << Register::FromOperand(operand).index(); | 136 Register reg = Register::FromOperand(operand); |
| 137 if (reg.is_parameter()) { |
| 138 int parameter_index = reg.ToParameterIndex(parameter_count); |
| 139 if (parameter_index == 0) { |
| 140 os << "<this>"; |
| 141 } else { |
| 142 os << "a" << parameter_index - 1; |
| 143 } |
| 144 } else { |
| 145 os << "r" << reg.index(); |
| 146 } |
137 break; | 147 break; |
| 148 } |
138 case interpreter::OperandType::kNone: | 149 case interpreter::OperandType::kNone: |
139 UNREACHABLE(); | 150 UNREACHABLE(); |
140 break; | 151 break; |
141 } | 152 } |
142 if (i != operands_size - 1) { | 153 if (i != operands_size - 1) { |
143 os << ", "; | 154 os << ", "; |
144 } | 155 } |
145 } | 156 } |
146 return os; | 157 return os; |
147 } | 158 } |
148 | 159 |
149 | 160 |
150 std::ostream& operator<<(std::ostream& os, const Bytecode& bytecode) { | 161 std::ostream& operator<<(std::ostream& os, const Bytecode& bytecode) { |
151 return os << Bytecodes::ToString(bytecode); | 162 return os << Bytecodes::ToString(bytecode); |
152 } | 163 } |
153 | 164 |
154 | 165 |
155 std::ostream& operator<<(std::ostream& os, const OperandType& operand_type) { | 166 std::ostream& operator<<(std::ostream& os, const OperandType& operand_type) { |
156 return os << Bytecodes::OperandTypeToString(operand_type); | 167 return os << Bytecodes::OperandTypeToString(operand_type); |
157 } | 168 } |
158 | 169 |
| 170 |
| 171 static const int kLastParamRegisterIndex = |
| 172 -InterpreterFrameConstants::kLastParamFromRegisterPointer / kPointerSize; |
| 173 |
| 174 |
| 175 // Registers occupy range 0-127 in 8-bit value leaving 128 unused values. |
| 176 // Parameter indices are biased with the negative value kLastParamRegisterIndex |
| 177 // for ease of access in the interpreter. |
| 178 static const int kMaxParameterIndex = 128 + kLastParamRegisterIndex; |
| 179 |
| 180 |
| 181 Register Register::FromParameterIndex(int index, int parameter_count) { |
| 182 DCHECK_GE(index, 0); |
| 183 DCHECK_LT(index, parameter_count); |
| 184 DCHECK_LE(parameter_count, kMaxParameterIndex + 1); |
| 185 int register_index = kLastParamRegisterIndex - parameter_count + index + 1; |
| 186 DCHECK_LT(register_index, 0); |
| 187 DCHECK_GE(register_index, Register::kMinRegisterIndex); |
| 188 return Register(register_index); |
| 189 } |
| 190 |
| 191 |
| 192 int Register::ToParameterIndex(int parameter_count) const { |
| 193 DCHECK(is_parameter()); |
| 194 return index() - kLastParamRegisterIndex + parameter_count - 1; |
| 195 } |
| 196 |
| 197 |
| 198 int Register::MaxParameterIndex() { return kMaxParameterIndex; } |
| 199 |
| 200 |
| 201 uint8_t Register::ToOperand() const { return static_cast<uint8_t>(-index_); } |
| 202 |
| 203 |
| 204 Register Register::FromOperand(uint8_t operand) { |
| 205 return Register(-static_cast<int8_t>(operand)); |
| 206 } |
| 207 |
159 } // namespace interpreter | 208 } // namespace interpreter |
160 } // namespace internal | 209 } // namespace internal |
161 } // namespace v8 | 210 } // namespace v8 |
OLD | NEW |