1 /* 2 * Copyright (C) 2004, 2008, 2009 Apple Inc. All rights reserved. 3 * Copyright (C) 2008 Collabora Ltd. 4 * Copyright (C) 2011 Peter Varga (pvarga (at) webkit.org), University of Szeged 5 * Copyright (C) 2013 Google Inc. All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY 17 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 19 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR 20 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 21 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 22 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 23 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY 24 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 26 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 27 */ 28 29 #include "config.h" 30 #include "bindings/v8/ScriptRegexp.h" 31 32 #include "bindings/v8/V8Binding.h" 33 #include "bindings/v8/V8PerIsolateData.h" 34 #include "bindings/v8/V8RecursionScope.h" 35 36 namespace WebCore { 37 38 ScriptRegexp::ScriptRegexp(const String& pattern, TextCaseSensitivity caseSensitivity, MultilineMode multilineMode) 39 { 40 v8::Isolate* isolate = v8::Isolate::GetCurrent(); 41 v8::HandleScope handleScope(isolate); 42 v8::Local<v8::Context> context = V8PerIsolateData::from(isolate)->ensureRegexContext(); 43 v8::Context::Scope scope(context); 44 45 unsigned flags = v8::RegExp::kNone; 46 if (caseSensitivity == TextCaseInsensitive) 47 flags |= v8::RegExp::kIgnoreCase; 48 if (multilineMode == MultilineEnabled) 49 flags |= v8::RegExp::kMultiline; 50 51 v8::TryCatch tryCatch; 52 v8::Local<v8::RegExp> regex = v8::RegExp::New(v8String(context->GetIsolate(), pattern), static_cast<v8::RegExp::Flags>(flags)); 53 54 // If the regex failed to compile we'll get an empty handle. 55 if (!regex.IsEmpty()) 56 m_regex.set(isolate, regex); 57 } 58 59 int ScriptRegexp::match(const String& string, int startFrom, int* matchLength) const 60 { 61 if (matchLength) 62 *matchLength = 0; 63 64 if (m_regex.isEmpty() || string.isNull()) 65 return -1; 66 67 // v8 strings are limited to int. 68 if (string.length() > INT_MAX) 69 return -1; 70 71 v8::Isolate* isolate = v8::Isolate::GetCurrent(); 72 v8::HandleScope handleScope(isolate); 73 v8::Local<v8::Context> context = V8PerIsolateData::current()->ensureRegexContext(); 74 v8::Context::Scope scope(context); 75 v8::TryCatch tryCatch; 76 77 V8RecursionScope::MicrotaskSuppression microtaskScope; 78 79 v8::Local<v8::RegExp> regex = m_regex.newLocal(isolate); 80 v8::Local<v8::Function> exec = regex->Get(v8AtomicString(isolate, "exec")).As<v8::Function>(); 81 82 v8::Handle<v8::Value> argv[] = { v8String(context->GetIsolate(), string.substring(startFrom)) }; 83 v8::Local<v8::Value> returnValue = exec->Call(regex, 1, argv); 84 85 // RegExp#exec returns null if there's no match, otherwise it returns an 86 // Array of strings with the first being the whole match string and others 87 // being subgroups. The Array also has some random properties tacked on like 88 // "index" which is the offset of the match. 89 // 90 // https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/RegExp/exec 91 92 if (!returnValue->IsArray()) 93 return -1; 94 95 v8::Local<v8::Array> result = returnValue.As<v8::Array>(); 96 int matchOffset = result->Get(v8AtomicString(isolate, "index"))->ToInt32()->Value(); 97 98 if (matchLength) { 99 v8::Local<v8::String> match = result->Get(0).As<v8::String>(); 100 *matchLength = match->Length(); 101 } 102 103 return matchOffset + startFrom; 104 } 105 106 } // namespace WebCore 107