| // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| #include "platform/assert.h" |
| #include "vm/bootstrap_natives.h" |
| #include "vm/exceptions.h" |
| #include "vm/native_entry.h" |
| #include "vm/object.h" |
| #include "vm/regexp_assembler_bytecode.h" |
| #include "vm/regexp_parser.h" |
| #include "vm/thread.h" |
| |
| #if !defined(DART_PRECOMPILED_RUNTIME) |
| #include "vm/regexp_assembler_ir.h" |
| #endif // !defined(DART_PRECOMPILED_RUNTIME) |
| |
| namespace dart { |
| |
| DEFINE_NATIVE_ENTRY(RegExp_factory, 0, 6) { |
| ASSERT( |
| TypeArguments::CheckedHandle(zone, arguments->NativeArgAt(0)).IsNull()); |
| GET_NON_NULL_NATIVE_ARGUMENT(String, pattern, arguments->NativeArgAt(1)); |
| GET_NON_NULL_NATIVE_ARGUMENT(Instance, handle_multi_line, |
| arguments->NativeArgAt(2)); |
| GET_NON_NULL_NATIVE_ARGUMENT(Instance, handle_case_sensitive, |
| arguments->NativeArgAt(3)); |
| GET_NON_NULL_NATIVE_ARGUMENT(Instance, handle_unicode, |
| arguments->NativeArgAt(4)); |
| GET_NON_NULL_NATIVE_ARGUMENT(Instance, handle_dot_all, |
| arguments->NativeArgAt(5)); |
| bool ignore_case = handle_case_sensitive.ptr() != Bool::True().ptr(); |
| bool multi_line = handle_multi_line.ptr() == Bool::True().ptr(); |
| bool unicode = handle_unicode.ptr() == Bool::True().ptr(); |
| bool dot_all = handle_dot_all.ptr() == Bool::True().ptr(); |
| |
| RegExpFlags flags; |
| |
| if (ignore_case) flags.SetIgnoreCase(); |
| if (multi_line) flags.SetMultiLine(); |
| if (unicode) flags.SetUnicode(); |
| if (dot_all) flags.SetDotAll(); |
| |
| // Parse the pattern once in order to throw any format exceptions within |
| // the factory constructor. It is parsed again upon compilation. |
| RegExpCompileData compileData; |
| // Throws an exception on parsing failure. |
| RegExpParser::ParseRegExp(pattern, flags, &compileData); |
| |
| // Create a RegExp object containing only the initial parameters. |
| return RegExpEngine::CreateRegExp(thread, pattern, flags); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getPattern, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| return regexp.pattern(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getIsMultiLine, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| return Bool::Get(regexp.flags().IsMultiLine()).ptr(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getIsUnicode, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| return Bool::Get(regexp.flags().IsUnicode()).ptr(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getIsDotAll, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| return Bool::Get(regexp.flags().IsDotAll()).ptr(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getIsCaseSensitive, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| return Bool::Get(!regexp.flags().IgnoreCase()).ptr(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getGroupCount, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| if (regexp.is_initialized()) { |
| return Smi::New(regexp.num_bracket_expressions()); |
| } |
| const String& pattern = String::Handle(regexp.pattern()); |
| const String& errmsg = |
| String::Handle(String::New("Regular expression is not initialized yet.")); |
| const String& message = String::Handle(String::Concat(errmsg, pattern)); |
| const Array& args = Array::Handle(Array::New(1)); |
| args.SetAt(0, message); |
| Exceptions::ThrowByType(Exceptions::kFormat, args); |
| return Object::null(); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_getGroupNameMap, 0, 1) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| if (regexp.is_initialized()) { |
| return regexp.capture_name_map(); |
| } |
| const String& pattern = String::Handle(regexp.pattern()); |
| const String& errmsg = String::Handle( |
| String::New("Regular expression is not initialized yet. ")); |
| const String& message = String::Handle(String::Concat(errmsg, pattern)); |
| const Array& args = Array::Handle(Array::New(1)); |
| args.SetAt(0, message); |
| Exceptions::ThrowByType(Exceptions::kFormat, args); |
| return Object::null(); |
| } |
| |
| static ObjectPtr ExecuteMatch(Zone* zone, |
| NativeArguments* arguments, |
| bool sticky) { |
| const RegExp& regexp = RegExp::CheckedHandle(zone, arguments->NativeArgAt(0)); |
| ASSERT(!regexp.IsNull()); |
| GET_NON_NULL_NATIVE_ARGUMENT(String, subject, arguments->NativeArgAt(1)); |
| GET_NON_NULL_NATIVE_ARGUMENT(Smi, start_index, arguments->NativeArgAt(2)); |
| |
| #if !defined(DART_PRECOMPILED_RUNTIME) |
| if (!FLAG_interpret_irregexp) { |
| return IRRegExpMacroAssembler::Execute(regexp, subject, start_index, |
| /*sticky=*/sticky, zone); |
| } |
| #endif |
| return BytecodeRegExpMacroAssembler::Interpret(regexp, subject, start_index, |
| /*sticky=*/sticky, zone); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_ExecuteMatch, 0, 3) { |
| // This function is intrinsified. See Intrinsifier::RegExp_ExecuteMatch. |
| return ExecuteMatch(zone, arguments, /*sticky=*/false); |
| } |
| |
| DEFINE_NATIVE_ENTRY(RegExp_ExecuteMatchSticky, 0, 3) { |
| // This function is intrinsified. See Intrinsifier::RegExp_ExecuteMatchSticky. |
| return ExecuteMatch(zone, arguments, /*sticky=*/true); |
| } |
| |
| } // namespace dart |