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/arguments.h" | 7 #include "src/arguments.h" |
8 #include "src/jsregexp-inl.h" | 8 #include "src/jsregexp-inl.h" |
9 #include "src/jsregexp.h" | 9 #include "src/jsregexp.h" |
10 #include "src/runtime/runtime-utils.h" | 10 #include "src/runtime/runtime-utils.h" |
(...skipping 741 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
752 if (result->HasFastObjectElements()) { | 752 if (result->HasFastObjectElements()) { |
753 RegExpResultsCache::Enter(isolate, subject, pattern, elements, | 753 RegExpResultsCache::Enter(isolate, subject, pattern, elements, |
754 RegExpResultsCache::STRING_SPLIT_SUBSTRINGS); | 754 RegExpResultsCache::STRING_SPLIT_SUBSTRINGS); |
755 } | 755 } |
756 } | 756 } |
757 | 757 |
758 return *result; | 758 return *result; |
759 } | 759 } |
760 | 760 |
761 | 761 |
762 RUNTIME_FUNCTION(Runtime_RegExpCompile) { | |
763 HandleScope scope(isolate); | |
764 DCHECK(args.length() == 3); | |
765 CONVERT_ARG_HANDLE_CHECKED(JSRegExp, re, 0); | |
766 CONVERT_ARG_HANDLE_CHECKED(String, pattern, 1); | |
767 CONVERT_ARG_HANDLE_CHECKED(String, flags, 2); | |
768 Handle<Object> result; | |
769 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, result, | |
770 RegExpImpl::Compile(re, pattern, flags)); | |
771 return *result; | |
772 } | |
773 | |
774 | |
775 RUNTIME_FUNCTION(Runtime_RegExpExecRT) { | 762 RUNTIME_FUNCTION(Runtime_RegExpExecRT) { |
776 HandleScope scope(isolate); | 763 HandleScope scope(isolate); |
777 DCHECK(args.length() == 4); | 764 DCHECK(args.length() == 4); |
778 CONVERT_ARG_HANDLE_CHECKED(JSRegExp, regexp, 0); | 765 CONVERT_ARG_HANDLE_CHECKED(JSRegExp, regexp, 0); |
779 CONVERT_ARG_HANDLE_CHECKED(String, subject, 1); | 766 CONVERT_ARG_HANDLE_CHECKED(String, subject, 1); |
780 CONVERT_INT32_ARG_CHECKED(index, 2); | 767 CONVERT_INT32_ARG_CHECKED(index, 2); |
781 CONVERT_ARG_HANDLE_CHECKED(JSArray, last_match_info, 3); | 768 CONVERT_ARG_HANDLE_CHECKED(JSArray, last_match_info, 3); |
782 // Due to the way the JS calls are constructed this must be less than the | 769 // Due to the way the JS calls are constructed this must be less than the |
783 // length of a string, i.e. it is always a Smi. We check anyway for security. | 770 // length of a string, i.e. it is always a Smi. We check anyway for security. |
784 RUNTIME_ASSERT(index >= 0); | 771 RUNTIME_ASSERT(index >= 0); |
(...skipping 21 matching lines...) Expand all Loading... |
806 Handle<JSArray> array = Handle<JSArray>::cast(object); | 793 Handle<JSArray> array = Handle<JSArray>::cast(object); |
807 array->set_elements(*elements); | 794 array->set_elements(*elements); |
808 array->set_length(Smi::FromInt(size)); | 795 array->set_length(Smi::FromInt(size)); |
809 // Write in-object properties after the length of the array. | 796 // Write in-object properties after the length of the array. |
810 array->InObjectPropertyAtPut(JSRegExpResult::kIndexIndex, *index); | 797 array->InObjectPropertyAtPut(JSRegExpResult::kIndexIndex, *index); |
811 array->InObjectPropertyAtPut(JSRegExpResult::kInputIndex, *input); | 798 array->InObjectPropertyAtPut(JSRegExpResult::kInputIndex, *input); |
812 return *array; | 799 return *array; |
813 } | 800 } |
814 | 801 |
815 | 802 |
816 RUNTIME_FUNCTION(Runtime_RegExpInitializeObject) { | 803 static JSRegExp::Flags RegExpFlagsFromString(Handle<String> flags, |
| 804 bool* success) { |
| 805 uint32_t value = JSRegExp::NONE; |
| 806 int length = flags->length(); |
| 807 // A longer flags string cannot be valid. |
| 808 if (length > 4) return JSRegExp::Flags(0); |
| 809 for (int i = 0; i < length; i++) { |
| 810 uint32_t flag = JSRegExp::NONE; |
| 811 switch (flags->Get(i)) { |
| 812 case 'g': |
| 813 flag = JSRegExp::GLOBAL; |
| 814 break; |
| 815 case 'i': |
| 816 flag = JSRegExp::IGNORE_CASE; |
| 817 break; |
| 818 case 'm': |
| 819 flag = JSRegExp::MULTILINE; |
| 820 break; |
| 821 case 'y': |
| 822 if (!FLAG_harmony_regexps) return JSRegExp::Flags(0); |
| 823 flag = JSRegExp::STICKY; |
| 824 break; |
| 825 default: |
| 826 return JSRegExp::Flags(0); |
| 827 } |
| 828 // Duplicate flag. |
| 829 if (value & flag) return JSRegExp::Flags(0); |
| 830 value |= flag; |
| 831 } |
| 832 *success = true; |
| 833 return JSRegExp::Flags(value); |
| 834 } |
| 835 |
| 836 |
| 837 RUNTIME_FUNCTION(Runtime_RegExpInitializeAndCompile) { |
817 HandleScope scope(isolate); | 838 HandleScope scope(isolate); |
818 DCHECK(args.length() == 6); | 839 DCHECK(args.length() == 3); |
819 CONVERT_ARG_HANDLE_CHECKED(JSRegExp, regexp, 0); | 840 CONVERT_ARG_HANDLE_CHECKED(JSRegExp, regexp, 0); |
820 CONVERT_ARG_HANDLE_CHECKED(String, source, 1); | 841 CONVERT_ARG_HANDLE_CHECKED(String, source, 1); |
| 842 CONVERT_ARG_HANDLE_CHECKED(String, flags_string, 2); |
| 843 Factory* factory = isolate->factory(); |
821 // If source is the empty string we set it to "(?:)" instead as | 844 // If source is the empty string we set it to "(?:)" instead as |
822 // suggested by ECMA-262, 5th, section 15.10.4.1. | 845 // suggested by ECMA-262, 5th, section 15.10.4.1. |
823 if (source->length() == 0) source = isolate->factory()->query_colon_string(); | 846 if (source->length() == 0) source = factory->query_colon_string(); |
824 | 847 |
825 CONVERT_ARG_HANDLE_CHECKED(Object, global, 2); | 848 bool success = false; |
826 if (!global->IsTrue()) global = isolate->factory()->false_value(); | 849 JSRegExp::Flags flags = RegExpFlagsFromString(flags_string, &success); |
| 850 if (!success) { |
| 851 Handle<FixedArray> element = factory->NewFixedArray(1); |
| 852 element->set(0, *flags_string); |
| 853 Handle<JSArray> args = factory->NewJSArrayWithElements(element); |
| 854 THROW_NEW_ERROR_RETURN_FAILURE( |
| 855 isolate, NewSyntaxError("invalid_regexp_flags", args)); |
| 856 } |
827 | 857 |
828 CONVERT_ARG_HANDLE_CHECKED(Object, ignoreCase, 3); | 858 Handle<Object> global = factory->ToBoolean(flags.is_global()); |
829 if (!ignoreCase->IsTrue()) ignoreCase = isolate->factory()->false_value(); | 859 Handle<Object> ignore_case = factory->ToBoolean(flags.is_ignore_case()); |
830 | 860 Handle<Object> multiline = factory->ToBoolean(flags.is_multiline()); |
831 CONVERT_ARG_HANDLE_CHECKED(Object, multiline, 4); | 861 Handle<Object> sticky = factory->ToBoolean(flags.is_sticky()); |
832 if (!multiline->IsTrue()) multiline = isolate->factory()->false_value(); | |
833 | |
834 CONVERT_ARG_HANDLE_CHECKED(Object, sticky, 5); | |
835 if (!sticky->IsTrue()) sticky = isolate->factory()->false_value(); | |
836 | 862 |
837 Map* map = regexp->map(); | 863 Map* map = regexp->map(); |
838 Object* constructor = map->constructor(); | 864 Object* constructor = map->constructor(); |
839 if (!FLAG_harmony_regexps && constructor->IsJSFunction() && | 865 if (!FLAG_harmony_regexps && constructor->IsJSFunction() && |
840 JSFunction::cast(constructor)->initial_map() == map) { | 866 JSFunction::cast(constructor)->initial_map() == map) { |
841 // If we still have the original map, set in-object properties directly. | 867 // If we still have the original map, set in-object properties directly. |
842 regexp->InObjectPropertyAtPut(JSRegExp::kSourceFieldIndex, *source); | 868 regexp->InObjectPropertyAtPut(JSRegExp::kSourceFieldIndex, *source); |
843 // Both true and false are immovable immortal objects so no need for write | 869 // Both true and false are immovable immortal objects so no need for write |
844 // barrier. | 870 // barrier. |
845 regexp->InObjectPropertyAtPut(JSRegExp::kGlobalFieldIndex, *global, | 871 regexp->InObjectPropertyAtPut(JSRegExp::kGlobalFieldIndex, *global, |
846 SKIP_WRITE_BARRIER); | 872 SKIP_WRITE_BARRIER); |
847 regexp->InObjectPropertyAtPut(JSRegExp::kIgnoreCaseFieldIndex, *ignoreCase, | 873 regexp->InObjectPropertyAtPut(JSRegExp::kIgnoreCaseFieldIndex, *ignore_case, |
848 SKIP_WRITE_BARRIER); | 874 SKIP_WRITE_BARRIER); |
849 regexp->InObjectPropertyAtPut(JSRegExp::kMultilineFieldIndex, *multiline, | 875 regexp->InObjectPropertyAtPut(JSRegExp::kMultilineFieldIndex, *multiline, |
850 SKIP_WRITE_BARRIER); | 876 SKIP_WRITE_BARRIER); |
851 regexp->InObjectPropertyAtPut(JSRegExp::kLastIndexFieldIndex, | 877 regexp->InObjectPropertyAtPut(JSRegExp::kLastIndexFieldIndex, |
852 Smi::FromInt(0), SKIP_WRITE_BARRIER); | 878 Smi::FromInt(0), SKIP_WRITE_BARRIER); |
853 return *regexp; | 879 } else { |
| 880 // Map has changed, so use generic, but slower, method. We also end here if |
| 881 // the --harmony-regexp flag is set, because the initial map does not have |
| 882 // space for the 'sticky' flag, since it is from the snapshot, but must work |
| 883 // both with and without --harmony-regexp. When sticky comes out from under |
| 884 // the flag, we will be able to use the fast initial map. |
| 885 PropertyAttributes final = |
| 886 static_cast<PropertyAttributes>(READ_ONLY | DONT_ENUM | DONT_DELETE); |
| 887 PropertyAttributes writable = |
| 888 static_cast<PropertyAttributes>(DONT_ENUM | DONT_DELETE); |
| 889 Handle<Object> zero(Smi::FromInt(0), isolate); |
| 890 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->source_string(), |
| 891 source, final).Check(); |
| 892 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->global_string(), |
| 893 global, final).Check(); |
| 894 JSObject::SetOwnPropertyIgnoreAttributes( |
| 895 regexp, factory->ignore_case_string(), ignore_case, final).Check(); |
| 896 JSObject::SetOwnPropertyIgnoreAttributes( |
| 897 regexp, factory->multiline_string(), multiline, final).Check(); |
| 898 if (FLAG_harmony_regexps) { |
| 899 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->sticky_string(), |
| 900 sticky, final).Check(); |
| 901 } |
| 902 JSObject::SetOwnPropertyIgnoreAttributes( |
| 903 regexp, factory->last_index_string(), zero, writable).Check(); |
854 } | 904 } |
855 | 905 |
856 // Map has changed, so use generic, but slower, method. We also end here if | 906 Handle<Object> result; |
857 // the --harmony-regexp flag is set, because the initial map does not have | 907 ASSIGN_RETURN_FAILURE_ON_EXCEPTION( |
858 // space for the 'sticky' flag, since it is from the snapshot, but must work | 908 isolate, result, RegExpImpl::Compile(regexp, source, flags)); |
859 // both with and without --harmony-regexp. When sticky comes out from under | 909 return *result; |
860 // the flag, we will be able to use the fast initial map. | |
861 PropertyAttributes final = | |
862 static_cast<PropertyAttributes>(READ_ONLY | DONT_ENUM | DONT_DELETE); | |
863 PropertyAttributes writable = | |
864 static_cast<PropertyAttributes>(DONT_ENUM | DONT_DELETE); | |
865 Handle<Object> zero(Smi::FromInt(0), isolate); | |
866 Factory* factory = isolate->factory(); | |
867 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->source_string(), | |
868 source, final).Check(); | |
869 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->global_string(), | |
870 global, final).Check(); | |
871 JSObject::SetOwnPropertyIgnoreAttributes( | |
872 regexp, factory->ignore_case_string(), ignoreCase, final).Check(); | |
873 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->multiline_string(), | |
874 multiline, final).Check(); | |
875 if (FLAG_harmony_regexps) { | |
876 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->sticky_string(), | |
877 sticky, final).Check(); | |
878 } | |
879 JSObject::SetOwnPropertyIgnoreAttributes(regexp, factory->last_index_string(), | |
880 zero, writable).Check(); | |
881 return *regexp; | |
882 } | 910 } |
883 | 911 |
884 | 912 |
885 RUNTIME_FUNCTION(Runtime_MaterializeRegExpLiteral) { | 913 RUNTIME_FUNCTION(Runtime_MaterializeRegExpLiteral) { |
886 HandleScope scope(isolate); | 914 HandleScope scope(isolate); |
887 DCHECK(args.length() == 4); | 915 DCHECK(args.length() == 4); |
888 CONVERT_ARG_HANDLE_CHECKED(FixedArray, literals, 0); | 916 CONVERT_ARG_HANDLE_CHECKED(FixedArray, literals, 0); |
889 CONVERT_SMI_ARG_CHECKED(index, 1); | 917 CONVERT_SMI_ARG_CHECKED(index, 1); |
890 CONVERT_ARG_HANDLE_CHECKED(String, pattern, 2); | 918 CONVERT_ARG_HANDLE_CHECKED(String, pattern, 2); |
891 CONVERT_ARG_HANDLE_CHECKED(String, flags, 3); | 919 CONVERT_ARG_HANDLE_CHECKED(String, flags, 3); |
(...skipping 196 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1088 | 1116 |
1089 | 1117 |
1090 RUNTIME_FUNCTION(RuntimeReference_IsRegExp) { | 1118 RUNTIME_FUNCTION(RuntimeReference_IsRegExp) { |
1091 SealHandleScope shs(isolate); | 1119 SealHandleScope shs(isolate); |
1092 DCHECK(args.length() == 1); | 1120 DCHECK(args.length() == 1); |
1093 CONVERT_ARG_CHECKED(Object, obj, 0); | 1121 CONVERT_ARG_CHECKED(Object, obj, 0); |
1094 return isolate->heap()->ToBoolean(obj->IsJSRegExp()); | 1122 return isolate->heap()->ToBoolean(obj->IsJSRegExp()); |
1095 } | 1123 } |
1096 } | 1124 } |
1097 } // namespace v8::internal | 1125 } // namespace v8::internal |
OLD | NEW |