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/runtime/runtime-utils.h" | 5 #include "src/runtime/runtime-utils.h" |
6 | 6 |
7 #include "src/arguments.h" | 7 #include "src/arguments.h" |
8 #include "src/conversions-inl.h" | 8 #include "src/conversions-inl.h" |
9 #include "src/isolate-inl.h" | 9 #include "src/isolate-inl.h" |
10 #include "src/messages.h" | 10 #include "src/messages.h" |
(...skipping 1482 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1493 DCHECK(args.length() == 3); | 1493 DCHECK(args.length() == 3); |
1494 | 1494 |
1495 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, recv, 0); | 1495 CONVERT_ARG_HANDLE_CHECKED(JSReceiver, recv, 0); |
1496 CONVERT_ARG_HANDLE_CHECKED(String, string, 1); | 1496 CONVERT_ARG_HANDLE_CHECKED(String, string, 1); |
1497 Handle<Object> replace_obj = args.at<Object>(2); | 1497 Handle<Object> replace_obj = args.at<Object>(2); |
1498 | 1498 |
1499 Factory* factory = isolate->factory(); | 1499 Factory* factory = isolate->factory(); |
1500 | 1500 |
1501 string = String::Flatten(string); | 1501 string = String::Flatten(string); |
1502 | 1502 |
1503 // Fast-path for unmodified JSRegExps. | |
1504 if (RegExpUtils::IsUnmodifiedRegExp(isolate, recv)) { | |
1505 RETURN_RESULT_OR_FAILURE( | |
1506 isolate, RegExpReplace(isolate, Handle<JSRegExp>::cast(recv), string, | |
1507 replace_obj)); | |
1508 } | |
1509 | |
1510 const int length = string->length(); | 1503 const int length = string->length(); |
1511 const bool functional_replace = replace_obj->IsCallable(); | 1504 const bool functional_replace = replace_obj->IsCallable(); |
1512 | 1505 |
1513 Handle<String> replace; | 1506 Handle<String> replace; |
1514 if (!functional_replace) { | 1507 if (!functional_replace) { |
1515 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, replace, | 1508 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, replace, |
1516 Object::ToString(isolate, replace_obj)); | 1509 Object::ToString(isolate, replace_obj)); |
1517 } | 1510 } |
1518 | 1511 |
1519 Handle<Object> global_obj; | 1512 Handle<Object> global_obj; |
1520 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( | 1513 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( |
1521 isolate, global_obj, | 1514 isolate, global_obj, |
1522 JSReceiver::GetProperty(recv, factory->global_string())); | 1515 JSReceiver::GetProperty(recv, factory->global_string())); |
1523 const bool global = global_obj->BooleanValue(); | 1516 const bool global = global_obj->BooleanValue(); |
1524 | 1517 |
1525 bool unicode = false; | 1518 bool unicode = false; |
1526 if (global) { | 1519 if (global) { |
1527 Handle<Object> unicode_obj; | 1520 Handle<Object> unicode_obj; |
1528 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( | 1521 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( |
1529 isolate, unicode_obj, | 1522 isolate, unicode_obj, |
1530 JSReceiver::GetProperty(recv, factory->unicode_string())); | 1523 JSReceiver::GetProperty(recv, factory->unicode_string())); |
1531 unicode = unicode_obj->BooleanValue(); | 1524 unicode = unicode_obj->BooleanValue(); |
1532 | 1525 |
1533 RETURN_FAILURE_ON_EXCEPTION(isolate, | 1526 RETURN_FAILURE_ON_EXCEPTION(isolate, |
1534 RegExpUtils::SetLastIndex(isolate, recv, 0)); | 1527 RegExpUtils::SetLastIndex(isolate, recv, 0)); |
1535 } | 1528 } |
1536 | 1529 |
| 1530 // TODO(jgruber): Here and at all other fast path checks, rely on map checks |
| 1531 // instead. |
| 1532 // TODO(jgruber): We could speed up the fast path by checking flags |
| 1533 // afterwards, but that would violate the spec (which states that exec is |
| 1534 // accessed after global and unicode). |
| 1535 // TODO(adamk): this fast path is wrong as we doesn't ensure that 'exec' |
| 1536 // is actually a data property on RegExp.prototype. |
| 1537 Handle<Object> exec = factory->undefined_value(); |
| 1538 if (recv->IsJSRegExp()) { |
| 1539 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( |
| 1540 isolate, exec, JSObject::GetProperty( |
| 1541 recv, factory->NewStringFromAsciiChecked("exec"))); |
| 1542 if (RegExpUtils::IsBuiltinExec(exec)) { |
| 1543 RETURN_RESULT_OR_FAILURE( |
| 1544 isolate, RegExpReplace(isolate, Handle<JSRegExp>::cast(recv), string, |
| 1545 replace_obj)); |
| 1546 } |
| 1547 } |
| 1548 |
1537 Zone zone(isolate->allocator(), ZONE_NAME); | 1549 Zone zone(isolate->allocator(), ZONE_NAME); |
1538 ZoneVector<Handle<Object>> results(&zone); | 1550 ZoneVector<Handle<Object>> results(&zone); |
1539 | 1551 |
1540 while (true) { | 1552 while (true) { |
1541 Handle<Object> result; | 1553 Handle<Object> result; |
1542 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( | 1554 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( |
1543 isolate, result, RegExpUtils::RegExpExec(isolate, recv, string, | 1555 isolate, result, RegExpUtils::RegExpExec(isolate, recv, string, exec)); |
1544 factory->undefined_value())); | 1556 |
| 1557 // Ensure exec will be read again on the next loop through. |
| 1558 exec = factory->undefined_value(); |
1545 | 1559 |
1546 if (result->IsNull(isolate)) break; | 1560 if (result->IsNull(isolate)) break; |
1547 | 1561 |
1548 results.push_back(result); | 1562 results.push_back(result); |
1549 if (!global) break; | 1563 if (!global) break; |
1550 | 1564 |
1551 Handle<Object> match_obj; | 1565 Handle<Object> match_obj; |
1552 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, match_obj, | 1566 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, match_obj, |
1553 Object::GetElement(isolate, result, 0)); | 1567 Object::GetElement(isolate, result, 0)); |
1554 | 1568 |
(...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1667 | 1681 |
1668 RUNTIME_FUNCTION(Runtime_IsRegExp) { | 1682 RUNTIME_FUNCTION(Runtime_IsRegExp) { |
1669 SealHandleScope shs(isolate); | 1683 SealHandleScope shs(isolate); |
1670 DCHECK(args.length() == 1); | 1684 DCHECK(args.length() == 1); |
1671 CONVERT_ARG_CHECKED(Object, obj, 0); | 1685 CONVERT_ARG_CHECKED(Object, obj, 0); |
1672 return isolate->heap()->ToBoolean(obj->IsJSRegExp()); | 1686 return isolate->heap()->ToBoolean(obj->IsJSRegExp()); |
1673 } | 1687 } |
1674 | 1688 |
1675 } // namespace internal | 1689 } // namespace internal |
1676 } // namespace v8 | 1690 } // namespace v8 |
OLD | NEW |