OLD | NEW |
1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 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/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #include "src/interpreter/bytecode-array-builder.h" | 7 #include "src/interpreter/bytecode-array-builder.h" |
| 8 #include "src/interpreter/bytecode-array-iterator.h" |
8 #include "test/unittests/test-utils.h" | 9 #include "test/unittests/test-utils.h" |
9 | 10 |
10 namespace v8 { | 11 namespace v8 { |
11 namespace internal { | 12 namespace internal { |
12 namespace interpreter { | 13 namespace interpreter { |
13 | 14 |
14 class BytecodeArrayBuilderTest : public TestWithIsolateAndZone { | 15 class BytecodeArrayBuilderTest : public TestWithIsolateAndZone { |
15 public: | 16 public: |
16 BytecodeArrayBuilderTest() {} | 17 BytecodeArrayBuilderTest() {} |
17 ~BytecodeArrayBuilderTest() override {} | 18 ~BytecodeArrayBuilderTest() override {} |
(...skipping 23 matching lines...) Expand all Loading... |
41 | 42 |
42 // Emit load / store property operations. | 43 // Emit load / store property operations. |
43 builder.LoadNamedProperty(reg, 0, LanguageMode::SLOPPY) | 44 builder.LoadNamedProperty(reg, 0, LanguageMode::SLOPPY) |
44 .LoadKeyedProperty(reg, 0, LanguageMode::SLOPPY) | 45 .LoadKeyedProperty(reg, 0, LanguageMode::SLOPPY) |
45 .StoreNamedProperty(reg, reg, 0, LanguageMode::SLOPPY) | 46 .StoreNamedProperty(reg, reg, 0, LanguageMode::SLOPPY) |
46 .StoreKeyedProperty(reg, reg, 0, LanguageMode::SLOPPY); | 47 .StoreKeyedProperty(reg, reg, 0, LanguageMode::SLOPPY); |
47 | 48 |
48 // Call operations. | 49 // Call operations. |
49 builder.Call(reg, reg, 0); | 50 builder.Call(reg, reg, 0); |
50 | 51 |
51 // Emit binary operators invocations. | 52 // Emit binary operator invocations. |
52 builder.BinaryOperation(Token::Value::ADD, reg) | 53 builder.BinaryOperation(Token::Value::ADD, reg) |
53 .BinaryOperation(Token::Value::SUB, reg) | 54 .BinaryOperation(Token::Value::SUB, reg) |
54 .BinaryOperation(Token::Value::MUL, reg) | 55 .BinaryOperation(Token::Value::MUL, reg) |
55 .BinaryOperation(Token::Value::DIV, reg) | 56 .BinaryOperation(Token::Value::DIV, reg) |
56 .BinaryOperation(Token::Value::MOD, reg); | 57 .BinaryOperation(Token::Value::MOD, reg); |
57 | 58 |
| 59 // Emit test operator invocations. |
| 60 builder.CompareOperation(Token::Value::EQ, reg, LanguageMode::SLOPPY) |
| 61 .CompareOperation(Token::Value::NE, reg, LanguageMode::SLOPPY) |
| 62 .CompareOperation(Token::Value::EQ_STRICT, reg, LanguageMode::SLOPPY) |
| 63 .CompareOperation(Token::Value::NE_STRICT, reg, LanguageMode::SLOPPY) |
| 64 .CompareOperation(Token::Value::LT, reg, LanguageMode::SLOPPY) |
| 65 .CompareOperation(Token::Value::GT, reg, LanguageMode::SLOPPY) |
| 66 .CompareOperation(Token::Value::LTE, reg, LanguageMode::SLOPPY) |
| 67 .CompareOperation(Token::Value::GTE, reg, LanguageMode::SLOPPY) |
| 68 .CompareOperation(Token::Value::INSTANCEOF, reg, LanguageMode::SLOPPY) |
| 69 .CompareOperation(Token::Value::IN, reg, LanguageMode::SLOPPY); |
| 70 |
| 71 // Emit cast operator invocations. |
| 72 builder.LoadNull().CastAccumulatorToBoolean(); |
| 73 |
58 // Emit control flow. Return must be the last instruction. | 74 // Emit control flow. Return must be the last instruction. |
| 75 BytecodeLabel start; |
| 76 builder.Bind(&start); |
| 77 // Short jumps with Imm8 operands |
| 78 builder.Jump(&start).JumpIfTrue(&start).JumpIfFalse(&start); |
| 79 // Insert dummy ops to force longer jumps |
| 80 for (int i = 0; i < 128; i++) { |
| 81 builder.LoadTrue(); |
| 82 } |
| 83 // Longer jumps requiring Constant operand |
| 84 builder.Jump(&start).JumpIfTrue(&start).JumpIfFalse(&start); |
59 builder.Return(); | 85 builder.Return(); |
60 | 86 |
61 // Generate BytecodeArray. | 87 // Generate BytecodeArray. |
62 Handle<BytecodeArray> the_array = builder.ToBytecodeArray(); | 88 Handle<BytecodeArray> the_array = builder.ToBytecodeArray(); |
63 CHECK_EQ(the_array->frame_size(), builder.locals_count() * kPointerSize); | 89 CHECK_EQ(the_array->frame_size(), builder.locals_count() * kPointerSize); |
64 | 90 |
65 // Build scorecard of bytecodes encountered in the BytecodeArray. | 91 // Build scorecard of bytecodes encountered in the BytecodeArray. |
66 std::vector<int> scorecard(Bytecodes::ToByte(Bytecode::kLast) + 1); | 92 std::vector<int> scorecard(Bytecodes::ToByte(Bytecode::kLast) + 1); |
67 Bytecode final_bytecode = Bytecode::kLdaZero; | 93 Bytecode final_bytecode = Bytecode::kLdaZero; |
68 for (int i = 0; i < the_array->length(); i++) { | 94 for (int i = 0; i < the_array->length(); i++) { |
(...skipping 104 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
173 .LoadLiteral(large_smi) | 199 .LoadLiteral(large_smi) |
174 .LoadLiteral(heap_num_1) | 200 .LoadLiteral(heap_num_1) |
175 .LoadLiteral(heap_num_1) | 201 .LoadLiteral(heap_num_1) |
176 .LoadLiteral(heap_num_2_copy); | 202 .LoadLiteral(heap_num_2_copy); |
177 | 203 |
178 Handle<BytecodeArray> array = builder.ToBytecodeArray(); | 204 Handle<BytecodeArray> array = builder.ToBytecodeArray(); |
179 // Should only have one entry for each identical constant. | 205 // Should only have one entry for each identical constant. |
180 CHECK_EQ(array->constant_pool()->length(), 3); | 206 CHECK_EQ(array->constant_pool()->length(), 3); |
181 } | 207 } |
182 | 208 |
| 209 |
| 210 TEST_F(BytecodeArrayBuilderTest, ForwardJumps) { |
| 211 static const int kFarJumpDistance = 256; |
| 212 |
| 213 BytecodeArrayBuilder builder(isolate(), zone()); |
| 214 builder.set_parameter_count(0); |
| 215 builder.set_locals_count(0); |
| 216 |
| 217 BytecodeLabel far0, far1, far2; |
| 218 BytecodeLabel near0, near1, near2; |
| 219 |
| 220 builder.Jump(&near0) |
| 221 .JumpIfTrue(&near1) |
| 222 .JumpIfFalse(&near2) |
| 223 .Bind(&near0) |
| 224 .Bind(&near1) |
| 225 .Bind(&near2) |
| 226 .Jump(&far0) |
| 227 .JumpIfTrue(&far1) |
| 228 .JumpIfFalse(&far2); |
| 229 for (int i = 0; i < kFarJumpDistance - 6; i++) { |
| 230 builder.LoadUndefined(); |
| 231 } |
| 232 builder.Bind(&far0).Bind(&far1).Bind(&far2); |
| 233 builder.Return(); |
| 234 |
| 235 Handle<BytecodeArray> array = builder.ToBytecodeArray(); |
| 236 DCHECK_EQ(array->length(), 12 + kFarJumpDistance - 6 + 1); |
| 237 |
| 238 BytecodeArrayIterator iterator(array); |
| 239 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 240 CHECK_EQ(iterator.GetSmi8Operand(0), 6); |
| 241 iterator.Advance(); |
| 242 |
| 243 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfTrue); |
| 244 CHECK_EQ(iterator.GetSmi8Operand(0), 4); |
| 245 iterator.Advance(); |
| 246 |
| 247 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfFalse); |
| 248 CHECK_EQ(iterator.GetSmi8Operand(0), 2); |
| 249 iterator.Advance(); |
| 250 |
| 251 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpConstant); |
| 252 CHECK_EQ(*iterator.GetConstantForIndexOperand(0), |
| 253 Smi::FromInt(kFarJumpDistance)); |
| 254 CHECK_EQ( |
| 255 array->get(iterator.current_offset() + |
| 256 Smi::cast(*iterator.GetConstantForIndexOperand(0))->value()), |
| 257 Bytecodes::ToByte(Bytecode::kReturn)); |
| 258 iterator.Advance(); |
| 259 |
| 260 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfTrueConstant); |
| 261 CHECK_EQ(*iterator.GetConstantForIndexOperand(0), |
| 262 Smi::FromInt(kFarJumpDistance - 2)); |
| 263 CHECK_EQ( |
| 264 array->get(iterator.current_offset() + |
| 265 Smi::cast(*iterator.GetConstantForIndexOperand(0))->value()), |
| 266 Bytecodes::ToByte(Bytecode::kReturn)); |
| 267 iterator.Advance(); |
| 268 |
| 269 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfFalseConstant); |
| 270 CHECK_EQ(*iterator.GetConstantForIndexOperand(0), |
| 271 Smi::FromInt(kFarJumpDistance - 4)); |
| 272 CHECK_EQ( |
| 273 array->get(iterator.current_offset() + |
| 274 Smi::cast(*iterator.GetConstantForIndexOperand(0))->value()), |
| 275 Bytecodes::ToByte(Bytecode::kReturn)); |
| 276 iterator.Advance(); |
| 277 } |
| 278 |
| 279 |
| 280 TEST_F(BytecodeArrayBuilderTest, BackwardJumps) { |
| 281 BytecodeArrayBuilder builder(isolate(), zone()); |
| 282 builder.set_parameter_count(0); |
| 283 builder.set_locals_count(0); |
| 284 |
| 285 BytecodeLabel label0, label1, label2; |
| 286 builder.Bind(&label0) |
| 287 .Jump(&label0) |
| 288 .Bind(&label1) |
| 289 .JumpIfTrue(&label1) |
| 290 .Bind(&label2) |
| 291 .JumpIfFalse(&label2); |
| 292 for (int i = 0; i < 64; i++) { |
| 293 builder.Jump(&label2); |
| 294 } |
| 295 builder.JumpIfFalse(&label2); |
| 296 builder.JumpIfTrue(&label1); |
| 297 builder.Jump(&label0); |
| 298 builder.Return(); |
| 299 |
| 300 Handle<BytecodeArray> array = builder.ToBytecodeArray(); |
| 301 BytecodeArrayIterator iterator(array); |
| 302 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 303 CHECK_EQ(iterator.GetSmi8Operand(0), 0); |
| 304 iterator.Advance(); |
| 305 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfTrue); |
| 306 CHECK_EQ(iterator.GetSmi8Operand(0), 0); |
| 307 iterator.Advance(); |
| 308 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfFalse); |
| 309 CHECK_EQ(iterator.GetSmi8Operand(0), 0); |
| 310 iterator.Advance(); |
| 311 for (int i = 0; i < 64; i++) { |
| 312 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 313 CHECK_EQ(iterator.GetSmi8Operand(0), -i * 2 - 2); |
| 314 iterator.Advance(); |
| 315 } |
| 316 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfFalseConstant); |
| 317 CHECK_EQ(Smi::cast(*iterator.GetConstantForIndexOperand(0))->value(), -130); |
| 318 iterator.Advance(); |
| 319 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpIfTrueConstant); |
| 320 CHECK_EQ(Smi::cast(*iterator.GetConstantForIndexOperand(0))->value(), -134); |
| 321 iterator.Advance(); |
| 322 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJumpConstant); |
| 323 CHECK_EQ(Smi::cast(*iterator.GetConstantForIndexOperand(0))->value(), -138); |
| 324 iterator.Advance(); |
| 325 CHECK_EQ(iterator.current_bytecode(), Bytecode::kReturn); |
| 326 iterator.Advance(); |
| 327 CHECK(iterator.done()); |
| 328 } |
| 329 |
| 330 |
| 331 TEST_F(BytecodeArrayBuilderTest, LabelReuse) { |
| 332 BytecodeArrayBuilder builder(isolate(), zone()); |
| 333 builder.set_parameter_count(0); |
| 334 builder.set_locals_count(0); |
| 335 |
| 336 // Labels can only have 1 forward reference, but |
| 337 // can be referred to mulitple times once bound. |
| 338 BytecodeLabel label; |
| 339 |
| 340 builder.Jump(&label).Bind(&label).Jump(&label).Jump(&label).Return(); |
| 341 |
| 342 Handle<BytecodeArray> array = builder.ToBytecodeArray(); |
| 343 BytecodeArrayIterator iterator(array); |
| 344 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 345 CHECK_EQ(iterator.GetSmi8Operand(0), 2); |
| 346 iterator.Advance(); |
| 347 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 348 CHECK_EQ(iterator.GetSmi8Operand(0), 0); |
| 349 iterator.Advance(); |
| 350 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 351 CHECK_EQ(iterator.GetSmi8Operand(0), -2); |
| 352 iterator.Advance(); |
| 353 CHECK_EQ(iterator.current_bytecode(), Bytecode::kReturn); |
| 354 iterator.Advance(); |
| 355 CHECK(iterator.done()); |
| 356 } |
| 357 |
| 358 |
| 359 TEST_F(BytecodeArrayBuilderTest, LabelAddressReuse) { |
| 360 static const int kRepeats = 3; |
| 361 |
| 362 BytecodeArrayBuilder builder(isolate(), zone()); |
| 363 builder.set_parameter_count(0); |
| 364 builder.set_locals_count(0); |
| 365 |
| 366 for (int i = 0; i < kRepeats; i++) { |
| 367 BytecodeLabel label; |
| 368 builder.Jump(&label).Bind(&label).Jump(&label).Jump(&label); |
| 369 } |
| 370 |
| 371 builder.Return(); |
| 372 |
| 373 Handle<BytecodeArray> array = builder.ToBytecodeArray(); |
| 374 BytecodeArrayIterator iterator(array); |
| 375 for (int i = 0; i < kRepeats; i++) { |
| 376 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 377 CHECK_EQ(iterator.GetSmi8Operand(0), 2); |
| 378 iterator.Advance(); |
| 379 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 380 CHECK_EQ(iterator.GetSmi8Operand(0), 0); |
| 381 iterator.Advance(); |
| 382 CHECK_EQ(iterator.current_bytecode(), Bytecode::kJump); |
| 383 CHECK_EQ(iterator.GetSmi8Operand(0), -2); |
| 384 iterator.Advance(); |
| 385 } |
| 386 CHECK_EQ(iterator.current_bytecode(), Bytecode::kReturn); |
| 387 iterator.Advance(); |
| 388 CHECK(iterator.done()); |
| 389 } |
| 390 |
| 391 |
183 } // namespace interpreter | 392 } // namespace interpreter |
184 } // namespace internal | 393 } // namespace internal |
185 } // namespace v8 | 394 } // namespace v8 |
OLD | NEW |