Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2016 the V8 project authors. All rights reserved. | 1 // Copyright 2016 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/builtins/builtins.h" | 5 #include "src/builtins/builtins.h" |
| 6 #include "src/builtins/builtins-utils.h" | 6 #include "src/builtins/builtins-utils.h" |
| 7 | |
| 8 #include "src/code-factory.h" | |
| 7 #include "src/elements.h" | 9 #include "src/elements.h" |
| 8 | 10 |
| 9 namespace v8 { | 11 namespace v8 { |
| 10 namespace internal { | 12 namespace internal { |
| 11 | 13 |
| 12 namespace { | 14 namespace { |
| 13 | 15 |
| 14 inline bool ClampedToInteger(Isolate* isolate, Object* object, int* out) { | 16 inline bool ClampedToInteger(Isolate* isolate, Object* object, int* out) { |
| 15 // This is an extended version of ECMA-262 7.1.11 handling signed values | 17 // This is an extended version of ECMA-262 7.1.11 handling signed values |
| 16 // Try to convert object to a number and clamp values to [kMinInt, kMaxInt] | 18 // Try to convert object to a number and clamp values to [kMinInt, kMaxInt] |
| (...skipping 1247 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1264 assembler->Return(assembler->BooleanConstant(true)); | 1266 assembler->Return(assembler->BooleanConstant(true)); |
| 1265 | 1267 |
| 1266 assembler->Bind(&return_false); | 1268 assembler->Bind(&return_false); |
| 1267 assembler->Return(assembler->BooleanConstant(false)); | 1269 assembler->Return(assembler->BooleanConstant(false)); |
| 1268 | 1270 |
| 1269 assembler->Bind(&call_runtime); | 1271 assembler->Bind(&call_runtime); |
| 1270 assembler->Return( | 1272 assembler->Return( |
| 1271 assembler->CallRuntime(Runtime::kArrayIsArray, context, object)); | 1273 assembler->CallRuntime(Runtime::kArrayIsArray, context, object)); |
| 1272 } | 1274 } |
| 1273 | 1275 |
| 1276 void Builtins::Generate_ArrayIncludes(CodeStubAssembler* assembler) { | |
| 1277 typedef compiler::Node Node; | |
| 1278 typedef CodeStubAssembler::Label Label; | |
| 1279 typedef CodeStubAssembler::Variable Variable; | |
| 1280 | |
| 1281 Node* array = assembler->Parameter(0); | |
| 1282 Node* search_element = assembler->Parameter(1); | |
| 1283 Node* start_from = assembler->Parameter(2); | |
| 1284 Node* context = assembler->Parameter(3 + 2); | |
| 1285 | |
| 1286 Node* int32_zero = assembler->Int32Constant(0); | |
| 1287 Node* int32_one = assembler->Int32Constant(1); | |
| 1288 | |
| 1289 Node* the_hole = assembler->TheHoleConstant(); | |
| 1290 Node* undefined = assembler->UndefinedConstant(); | |
| 1291 Node* heap_number_map = assembler->HeapNumberMapConstant(); | |
| 1292 | |
| 1293 Variable len(assembler, MachineRepresentation::kWord32), | |
|
Benedikt Meurer
2016/07/19 03:53:19
Nit: Can you rename those to len_var, k_var and n_
| |
| 1294 k(assembler, MachineRepresentation::kWord32), | |
| 1295 n(assembler, MachineRepresentation::kWord32); | |
|
Camillo Bruni
2016/07/19 12:08:37
pretty please: use real variable names, k = key, n
| |
| 1296 | |
| 1297 Label init_k(assembler), return_true(assembler), return_false(assembler), | |
| 1298 call_runtime(assembler); | |
| 1299 | |
| 1300 { | |
| 1301 // Take slow path if not a JSArray, if retrieving elements requires | |
| 1302 // traversing prototype, or if access checks are required. | |
| 1303 Label init_len(assembler); | |
| 1304 assembler->BranchIfFastJSArray(array, context, &init_len, &call_runtime); | |
|
Benedikt Meurer
2016/07/19 03:53:19
Nit: Move this before the { } block, and start the
| |
| 1305 | |
| 1306 assembler->Bind(&init_len); | |
| 1307 len.Bind(assembler->SmiToWord( | |
| 1308 assembler->LoadObjectField(array, JSArray::kLengthOffset))); | |
| 1309 | |
| 1310 assembler->GotoUnless(assembler->Word32Equal(len.value(), int32_zero), | |
| 1311 &init_k); | |
| 1312 assembler->Return(assembler->BooleanConstant(false)); | |
| 1313 } | |
| 1314 | |
| 1315 assembler->Bind(&init_k); | |
| 1316 { | |
| 1317 Label done(assembler), init_k_smi(assembler), init_k_heap_num(assembler), | |
| 1318 init_k_zero(assembler), init_k_n(assembler); | |
|
Camillo Bruni
2016/07/19 12:08:37
again, I really like having readable variable name
| |
| 1319 Callable call_to_integer = CodeFactory::ToInteger(assembler->isolate()); | |
| 1320 Node* tagged_n = assembler->CallStub(call_to_integer, context, start_from); | |
| 1321 | |
| 1322 assembler->Branch(assembler->WordIsSmi(tagged_n), &init_k_smi, | |
| 1323 &init_k_heap_num); | |
| 1324 | |
| 1325 assembler->Bind(&init_k_smi); | |
| 1326 { | |
| 1327 n.Bind(assembler->SmiToWord32(tagged_n)); | |
| 1328 assembler->Goto(&init_k_n); | |
| 1329 } | |
| 1330 | |
| 1331 assembler->Bind(&init_k_heap_num); | |
| 1332 { | |
| 1333 Label abort(assembler); | |
| 1334 Node* fp_len = assembler->ChangeInt32ToFloat64(len.value()); | |
| 1335 Node* fp_n = assembler->LoadHeapNumberValue(tagged_n); | |
| 1336 assembler->GotoIf(assembler->Float64GreaterThanOrEqual(fp_n, fp_len), | |
| 1337 &abort); | |
|
Benedikt Meurer
2016/07/19 03:53:19
Instead of the abort label, go to the shared retur
caitp
2016/07/19 19:59:21
The issue with that is that the return_false label
Benedikt Meurer
2016/07/20 03:55:01
That seems suspicious. Maybe a bug in the CodeAsse
danno
2016/07/25 13:53:55
I don't think this is a bug (although I haven't de
| |
| 1338 n.Bind(assembler->TruncateFloat64ToWord32(fp_n)); | |
|
Benedikt Meurer
2016/07/19 03:53:18
The fp_n can be outside the valid word32 range. An
caitp
2016/07/19 19:59:21
`len` is pulled directly from kLengthOffset of JSA
Benedikt Meurer
2016/07/20 03:55:01
It could still be negative, i.e. -2^32. Truncating
caitp
2016/07/20 16:07:55
well, here's the rationale for this I came up with
| |
| 1339 assembler->Goto(&init_k_n); | |
| 1340 | |
| 1341 assembler->Bind(&abort); | |
| 1342 assembler->Return(assembler->BooleanConstant(false)); | |
| 1343 } | |
| 1344 | |
| 1345 assembler->Bind(&init_k_n); | |
| 1346 { | |
| 1347 Label if_positive(assembler), if_negative(assembler), done(assembler); | |
| 1348 assembler->Branch(assembler->Int32LessThan(n.value(), int32_zero), | |
| 1349 &if_negative, &if_positive); | |
| 1350 | |
| 1351 assembler->Bind(&if_positive); | |
| 1352 { | |
| 1353 k.Bind(n.value()); | |
| 1354 assembler->Goto(&done); | |
| 1355 } | |
| 1356 | |
| 1357 assembler->Bind(&if_negative); | |
| 1358 { | |
| 1359 k.Bind(assembler->Int32Add(len.value(), n.value())); | |
| 1360 assembler->Branch(assembler->Int32LessThan(k.value(), int32_zero), | |
| 1361 &init_k_zero, &done); | |
| 1362 } | |
| 1363 | |
| 1364 assembler->Bind(&init_k_zero); | |
| 1365 { | |
| 1366 k.Bind(int32_zero); | |
| 1367 assembler->Goto(&done); | |
| 1368 } | |
| 1369 | |
| 1370 assembler->Bind(&done); | |
| 1371 } | |
| 1372 } | |
| 1373 | |
| 1374 int32_t element_kinds[] = { | |
|
Benedikt Meurer
2016/07/19 03:53:19
Nit: Make this static const. And rename to kElemen
caitp
2016/07/19 19:59:21
Acknowledged. Requires a change to the Switch() pr
Benedikt Meurer
2016/07/20 03:55:01
Acknowledged.
| |
| 1375 FAST_SMI_ELEMENTS, FAST_HOLEY_SMI_ELEMENTS, FAST_ELEMENTS, | |
| 1376 FAST_HOLEY_ELEMENTS, FAST_DOUBLE_ELEMENTS, FAST_HOLEY_DOUBLE_ELEMENTS, | |
| 1377 }; | |
| 1378 | |
| 1379 Label if_smiorobjects(assembler), if_packed_doubles(assembler), | |
| 1380 if_holey_doubles(assembler); | |
| 1381 Label* element_kind_handlers[] = {&if_smiorobjects, &if_smiorobjects, | |
| 1382 &if_smiorobjects, &if_smiorobjects, | |
| 1383 &if_packed_doubles, &if_holey_doubles}; | |
| 1384 | |
| 1385 Node* map = assembler->LoadMap(array); | |
| 1386 Node* bit_field2 = assembler->LoadMapBitField2(map); | |
| 1387 Node* elements_kind = | |
| 1388 assembler->BitFieldDecode<Map::ElementsKindBits>(bit_field2); | |
| 1389 Node* elements = assembler->LoadElements(array); | |
| 1390 assembler->Switch(elements_kind, &return_false, element_kinds, | |
|
Benedikt Meurer
2016/07/19 03:53:19
Nice idea!
| |
| 1391 element_kind_handlers, arraysize(element_kinds)); | |
| 1392 | |
| 1393 assembler->Bind(&if_smiorobjects); | |
| 1394 { | |
| 1395 Variable search_num(assembler, MachineRepresentation::kFloat64); | |
| 1396 Label ident_loop(assembler, &k), heap_num_loop(assembler, &search_num), | |
| 1397 string_loop(assembler, &k), simd_loop(assembler), | |
| 1398 undef_loop(assembler, &k), not_smi(assembler), not_heap_num(assembler); | |
| 1399 | |
| 1400 assembler->GotoUnless(assembler->WordIsSmi(search_element), ¬_smi); | |
| 1401 search_num.Bind(assembler->SmiToFloat64(search_element)); | |
| 1402 assembler->Goto(&heap_num_loop); | |
| 1403 | |
| 1404 assembler->Bind(¬_smi); | |
| 1405 assembler->GotoIf(assembler->WordEqual(search_element, undefined), | |
| 1406 &undef_loop); | |
| 1407 Node* map = assembler->LoadMap(search_element); | |
| 1408 assembler->GotoIf(assembler->WordNotEqual(map, heap_number_map), | |
| 1409 ¬_heap_num); | |
| 1410 search_num.Bind(assembler->LoadHeapNumberValue(search_element)); | |
| 1411 assembler->Goto(&heap_num_loop); | |
| 1412 | |
| 1413 assembler->Bind(¬_heap_num); | |
| 1414 Node* search_type = assembler->LoadMapInstanceType(map); | |
| 1415 assembler->GotoIf( | |
| 1416 assembler->Int32LessThan( | |
| 1417 search_type, assembler->Int32Constant(FIRST_NONSTRING_TYPE)), | |
| 1418 &string_loop); | |
| 1419 assembler->GotoIf( | |
| 1420 assembler->WordEqual(search_type, | |
| 1421 assembler->Int32Constant(SIMD128_VALUE_TYPE)), | |
| 1422 &simd_loop); | |
| 1423 assembler->Goto(&ident_loop); | |
| 1424 | |
| 1425 assembler->Bind(&ident_loop); | |
| 1426 { | |
| 1427 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1428 &return_false); | |
| 1429 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1430 assembler->GotoIf(assembler->WordEqual(element_k, search_element), | |
| 1431 &return_true); | |
| 1432 | |
| 1433 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1434 assembler->Goto(&ident_loop); | |
| 1435 } | |
| 1436 | |
| 1437 assembler->Bind(&undef_loop); | |
| 1438 { | |
| 1439 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1440 &return_false); | |
| 1441 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1442 assembler->GotoIf(assembler->WordEqual(element_k, undefined), | |
| 1443 &return_true); | |
| 1444 assembler->GotoIf(assembler->WordEqual(element_k, the_hole), | |
| 1445 &return_true); | |
| 1446 | |
| 1447 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1448 assembler->Goto(&undef_loop); | |
| 1449 } | |
| 1450 | |
| 1451 assembler->Bind(&heap_num_loop); | |
| 1452 { | |
| 1453 Label nan_loop(assembler, &k), not_nan_loop(assembler, &k); | |
| 1454 assembler->BranchIfFloat64IsNaN(search_num.value(), &nan_loop, | |
| 1455 ¬_nan_loop); | |
| 1456 | |
| 1457 assembler->Bind(¬_nan_loop); | |
| 1458 { | |
| 1459 Label continue_loop(assembler), not_smi(assembler); | |
| 1460 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1461 &return_false); | |
| 1462 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1463 assembler->GotoUnless(assembler->WordIsSmi(element_k), ¬_smi); | |
| 1464 assembler->Branch( | |
| 1465 assembler->Float64Equal(search_num.value(), | |
| 1466 assembler->SmiToFloat64(element_k)), | |
| 1467 &return_true, &continue_loop); | |
| 1468 | |
| 1469 assembler->Bind(¬_smi); | |
| 1470 assembler->GotoIf(assembler->WordNotEqual(assembler->LoadMap(element_k), | |
| 1471 heap_number_map), | |
| 1472 &continue_loop); | |
| 1473 assembler->BranchIfFloat64Equal( | |
| 1474 search_num.value(), assembler->LoadHeapNumberValue(element_k), | |
| 1475 &return_true, &continue_loop); | |
| 1476 | |
| 1477 assembler->Bind(&continue_loop); | |
| 1478 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1479 assembler->Goto(¬_nan_loop); | |
| 1480 } | |
| 1481 | |
| 1482 assembler->Bind(&nan_loop); | |
| 1483 { | |
| 1484 Label continue_loop(assembler); | |
| 1485 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1486 &return_false); | |
| 1487 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1488 assembler->GotoIf(assembler->WordIsSmi(element_k), &continue_loop); | |
| 1489 assembler->GotoIf(assembler->WordNotEqual(assembler->LoadMap(element_k), | |
| 1490 heap_number_map), | |
| 1491 &continue_loop); | |
| 1492 assembler->BranchIfFloat64IsNaN( | |
| 1493 assembler->LoadHeapNumberValue(element_k), &return_true, | |
| 1494 &continue_loop); | |
| 1495 | |
| 1496 assembler->Bind(&continue_loop); | |
| 1497 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1498 assembler->Goto(&nan_loop); | |
| 1499 } | |
| 1500 } | |
| 1501 | |
| 1502 assembler->Bind(&string_loop); | |
| 1503 { | |
| 1504 Label continue_loop(assembler); | |
| 1505 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1506 &return_false); | |
| 1507 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1508 assembler->GotoIf(assembler->WordIsSmi(element_k), &continue_loop); | |
| 1509 assembler->GotoUnless(assembler->Int32LessThan( | |
| 1510 assembler->LoadMapInstanceType(element_k), | |
| 1511 assembler->Int32Constant(FIRST_NONSTRING_TYPE)), | |
| 1512 &continue_loop); | |
| 1513 | |
| 1514 Callable callable = CodeFactory::StringEqual(assembler->isolate()); | |
|
Benedikt Meurer
2016/07/19 03:53:18
Please add a TODO here, that at some point we want
caitp
2016/07/19 19:59:21
Acknowledged.
| |
| 1515 Node* result = | |
| 1516 assembler->CallStub(callable, context, search_element, element_k); | |
| 1517 assembler->Branch( | |
| 1518 assembler->WordEqual(assembler->BooleanConstant(true), result), | |
| 1519 &return_true, &continue_loop); | |
| 1520 | |
| 1521 assembler->Bind(&continue_loop); | |
| 1522 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1523 assembler->Goto(&string_loop); | |
| 1524 } | |
| 1525 | |
| 1526 assembler->Bind(&simd_loop); | |
| 1527 { | |
| 1528 Label continue_loop(assembler, &k), loop_body(assembler, &k); | |
| 1529 Node* map = assembler->LoadMap(search_element); | |
| 1530 | |
| 1531 assembler->Goto(&loop_body); | |
| 1532 assembler->Bind(&loop_body); | |
| 1533 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1534 &return_false); | |
| 1535 | |
| 1536 Node* element_k = assembler->LoadFixedArrayElement(elements, k.value()); | |
| 1537 assembler->GotoIf(assembler->WordIsSmi(element_k), &continue_loop); | |
| 1538 | |
| 1539 Node* map_k = assembler->LoadMap(element_k); | |
| 1540 assembler->BranchIfSimd128Equal(search_element, map, element_k, map_k, | |
| 1541 &return_true, &continue_loop); | |
| 1542 | |
| 1543 assembler->Bind(&continue_loop); | |
| 1544 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1545 assembler->Goto(&loop_body); | |
| 1546 } | |
| 1547 } | |
| 1548 | |
| 1549 assembler->Bind(&if_packed_doubles); | |
| 1550 { | |
| 1551 Label nan_loop(assembler, &k), not_nan_loop(assembler, &k), | |
| 1552 hole_loop(assembler, &k), search_notnan(assembler); | |
| 1553 Variable search_num(assembler, MachineRepresentation::kFloat64); | |
| 1554 | |
| 1555 assembler->GotoUnless(assembler->WordIsSmi(search_element), &search_notnan); | |
| 1556 search_num.Bind(assembler->SmiToFloat64(search_element)); | |
| 1557 assembler->Goto(¬_nan_loop); | |
| 1558 | |
| 1559 assembler->Bind(&search_notnan); | |
| 1560 assembler->GotoIf(assembler->WordNotEqual( | |
| 1561 assembler->LoadMap(search_element), heap_number_map), | |
| 1562 &return_false); | |
| 1563 | |
| 1564 search_num.Bind(assembler->LoadHeapNumberValue(search_element)); | |
| 1565 | |
| 1566 assembler->BranchIfFloat64IsNaN(search_num.value(), &nan_loop, | |
| 1567 ¬_nan_loop); | |
| 1568 | |
| 1569 // Search for HeapNumber | |
| 1570 assembler->Bind(¬_nan_loop); | |
| 1571 { | |
| 1572 Label continue_loop(assembler); | |
| 1573 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1574 &return_false); | |
| 1575 Node* element_k = assembler->LoadFixedDoubleArrayElement( | |
| 1576 elements, k.value(), MachineType::Float64()); | |
| 1577 assembler->BranchIfFloat64Equal(element_k, search_num.value(), | |
| 1578 &return_true, &continue_loop); | |
| 1579 assembler->Bind(&continue_loop); | |
| 1580 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1581 assembler->Goto(¬_nan_loop); | |
| 1582 } | |
| 1583 | |
| 1584 // Search for NaN | |
| 1585 assembler->Bind(&nan_loop); | |
| 1586 { | |
| 1587 Label continue_loop(assembler); | |
| 1588 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1589 &return_false); | |
| 1590 Node* element_k = assembler->LoadFixedDoubleArrayElement( | |
| 1591 elements, k.value(), MachineType::Float64()); | |
| 1592 assembler->BranchIfFloat64IsNaN(element_k, &return_true, &continue_loop); | |
| 1593 assembler->Bind(&continue_loop); | |
| 1594 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1595 assembler->Goto(&nan_loop); | |
| 1596 } | |
| 1597 } | |
| 1598 | |
| 1599 assembler->Bind(&if_holey_doubles); | |
| 1600 { | |
| 1601 Label nan_loop(assembler, &k), not_nan_loop(assembler, &k), | |
| 1602 hole_loop(assembler, &k), search_notnan(assembler); | |
| 1603 Variable search_num(assembler, MachineRepresentation::kFloat64); | |
| 1604 | |
| 1605 assembler->GotoUnless(assembler->WordIsSmi(search_element), &search_notnan); | |
| 1606 search_num.Bind(assembler->SmiToFloat64(search_element)); | |
| 1607 assembler->Goto(¬_nan_loop); | |
| 1608 | |
| 1609 assembler->Bind(&search_notnan); | |
| 1610 assembler->GotoIf(assembler->WordEqual(search_element, undefined), | |
| 1611 &hole_loop); | |
| 1612 assembler->GotoIf(assembler->WordNotEqual( | |
| 1613 assembler->LoadMap(search_element), heap_number_map), | |
| 1614 &return_false); | |
| 1615 | |
| 1616 search_num.Bind(assembler->LoadHeapNumberValue(search_element)); | |
| 1617 | |
| 1618 assembler->BranchIfFloat64IsNaN(search_num.value(), &nan_loop, | |
| 1619 ¬_nan_loop); | |
| 1620 | |
| 1621 // Search for HeapNumber | |
| 1622 assembler->Bind(¬_nan_loop); | |
| 1623 { | |
| 1624 Label continue_loop(assembler); | |
| 1625 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1626 &return_false); | |
| 1627 | |
| 1628 if (kPointerSize == kDoubleSize) { | |
|
Benedikt Meurer
2016/07/19 03:53:19
You don't need this distinction. You can always co
caitp
2016/07/19 19:59:21
Acknowledged. This pattern is used elsewhere thoug
Benedikt Meurer
2016/07/20 03:55:01
Hm, ok, we could also simplify that.
| |
| 1629 Node* element = assembler->LoadFixedDoubleArrayElement( | |
| 1630 elements, k.value(), MachineType::Uint64()); | |
| 1631 Node* the_hole = assembler->Int64Constant(kHoleNanInt64); | |
| 1632 assembler->GotoIf(assembler->Word64Equal(element, the_hole), | |
| 1633 &continue_loop); | |
| 1634 } else { | |
| 1635 Node* element_upper = assembler->LoadFixedDoubleArrayElement( | |
| 1636 elements, k.value(), MachineType::Uint32(), | |
| 1637 kIeeeDoubleExponentWordOffset); | |
| 1638 assembler->GotoIf( | |
| 1639 assembler->Word32Equal(element_upper, | |
| 1640 assembler->Int32Constant(kHoleNanUpper32)), | |
| 1641 &continue_loop); | |
| 1642 } | |
| 1643 | |
| 1644 Node* element_k = assembler->LoadFixedDoubleArrayElement( | |
| 1645 elements, k.value(), MachineType::Float64()); | |
| 1646 assembler->BranchIfFloat64Equal(element_k, search_num.value(), | |
| 1647 &return_true, &continue_loop); | |
| 1648 assembler->Bind(&continue_loop); | |
| 1649 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1650 assembler->Goto(¬_nan_loop); | |
| 1651 } | |
| 1652 | |
| 1653 // Search for NaN | |
| 1654 assembler->Bind(&nan_loop); | |
| 1655 { | |
| 1656 Label continue_loop(assembler); | |
| 1657 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1658 &return_false); | |
| 1659 | |
| 1660 if (kPointerSize == kDoubleSize) { | |
|
Benedikt Meurer
2016/07/19 03:53:19
See above.
| |
| 1661 Node* element = assembler->LoadFixedDoubleArrayElement( | |
| 1662 elements, k.value(), MachineType::Uint64()); | |
| 1663 Node* the_hole = assembler->Int64Constant(kHoleNanInt64); | |
| 1664 assembler->GotoIf(assembler->Word64Equal(element, the_hole), | |
| 1665 &continue_loop); | |
| 1666 } else { | |
| 1667 Node* element_upper = assembler->LoadFixedDoubleArrayElement( | |
| 1668 elements, k.value(), MachineType::Uint32(), | |
| 1669 kIeeeDoubleExponentWordOffset); | |
| 1670 assembler->GotoIf( | |
| 1671 assembler->Word32Equal(element_upper, | |
| 1672 assembler->Int32Constant(kHoleNanUpper32)), | |
| 1673 &continue_loop); | |
| 1674 } | |
| 1675 | |
| 1676 Node* element_k = assembler->LoadFixedDoubleArrayElement( | |
| 1677 elements, k.value(), MachineType::Float64()); | |
| 1678 assembler->BranchIfFloat64IsNaN(element_k, &return_true, &continue_loop); | |
| 1679 assembler->Bind(&continue_loop); | |
| 1680 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1681 assembler->Goto(&nan_loop); | |
| 1682 } | |
| 1683 | |
| 1684 // Search for the Hole | |
| 1685 assembler->Bind(&hole_loop); | |
| 1686 { | |
| 1687 assembler->GotoUnless(assembler->Int32LessThan(k.value(), len.value()), | |
| 1688 &return_false); | |
| 1689 | |
| 1690 if (kPointerSize == kDoubleSize) { | |
| 1691 Node* element = assembler->LoadFixedDoubleArrayElement( | |
| 1692 elements, k.value(), MachineType::Uint64()); | |
| 1693 Node* the_hole = assembler->Int64Constant(kHoleNanInt64); | |
| 1694 assembler->GotoIf(assembler->Word64Equal(element, the_hole), | |
| 1695 &return_true); | |
| 1696 } else { | |
| 1697 Node* element_upper = assembler->LoadFixedDoubleArrayElement( | |
| 1698 elements, k.value(), MachineType::Uint32(), | |
| 1699 kIeeeDoubleExponentWordOffset); | |
| 1700 assembler->GotoIf( | |
| 1701 assembler->Word32Equal(element_upper, | |
| 1702 assembler->Int32Constant(kHoleNanUpper32)), | |
| 1703 &return_true); | |
| 1704 } | |
| 1705 | |
| 1706 k.Bind(assembler->Int32Add(k.value(), int32_one)); | |
| 1707 assembler->Goto(&hole_loop); | |
| 1708 } | |
| 1709 } | |
| 1710 | |
| 1711 assembler->Bind(&return_true); | |
| 1712 assembler->Return(assembler->BooleanConstant(true)); | |
| 1713 | |
| 1714 assembler->Bind(&return_false); | |
| 1715 assembler->Return(assembler->BooleanConstant(false)); | |
| 1716 | |
| 1717 assembler->Bind(&call_runtime); | |
| 1718 assembler->Return(assembler->CallRuntime(Runtime::kArrayIncludes_Slow, | |
| 1719 context, array, search_element, | |
| 1720 start_from)); | |
| 1721 } | |
| 1722 | |
| 1274 } // namespace internal | 1723 } // namespace internal |
| 1275 } // namespace v8 | 1724 } // namespace v8 |
| OLD | NEW |