Changeset 209101 in webkit
- Timestamp:
- Nov 29, 2016, 4:06:50 PM (8 years ago)
- Location:
- trunk/Source/JavaScriptCore
- Files:
-
- 5 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/Source/JavaScriptCore/ChangeLog
r209088 r209101 1 2016-11-29 Mark Lam <mark.lam@apple.com> 2 3 Fix exception scope verification failures in runtime/RegExp* files. 4 https://bugs.webkit.org/show_bug.cgi?id=165054 5 6 Reviewed by Saam Barati. 7 8 Also replaced returning JSValue() with returning { }. 9 10 * runtime/RegExpConstructor.cpp: 11 (JSC::toFlags): 12 (JSC::regExpCreate): 13 (JSC::constructRegExp): 14 * runtime/RegExpObject.cpp: 15 (JSC::RegExpObject::defineOwnProperty): 16 (JSC::collectMatches): 17 (JSC::RegExpObject::matchGlobal): 18 * runtime/RegExpObjectInlines.h: 19 (JSC::getRegExpObjectLastIndexAsUnsigned): 20 (JSC::RegExpObject::execInline): 21 (JSC::RegExpObject::matchInline): 22 * runtime/RegExpPrototype.cpp: 23 (JSC::regExpProtoFuncCompile): 24 (JSC::flagsString): 25 (JSC::regExpProtoFuncToString): 26 (JSC::regExpProtoFuncSplitFast): 27 1 28 2016-11-29 Andy Estes <aestes@apple.com> 2 29 -
trunk/Source/JavaScriptCore/runtime/RegExpConstructor.cpp
r206386 r209101 215 215 if (flags.isUndefined()) 216 216 return NoFlags; 217 JSString* flagsString = flags.toString (exec);218 ASSERT( scope.exception() ||flagsString);219 if ( !flagsString) {217 JSString* flagsString = flags.toStringOrNull(exec); 218 ASSERT(!!scope.exception() == !flagsString); 219 if (UNLIKELY(!flagsString)) 220 220 return InvalidFlags; 221 }222 221 223 222 RegExpFlags result = regExpFlags(flagsString->value(exec)); … … 237 236 238 237 RegExpFlags flags = toFlags(exec, flagsArg); 239 if (flags == InvalidFlags) 238 ASSERT(!!scope.exception() == (flags == InvalidFlags)); 239 if (UNLIKELY(flags == InvalidFlags)) 240 240 return nullptr; 241 241 … … 258 258 bool isPatternRegExp = patternArg.inherits(RegExpObject::info()); 259 259 bool constructAsRegexp = isRegExp(vm, exec, patternArg); 260 RETURN_IF_EXCEPTION(scope, nullptr); 260 261 261 262 if (newTarget.isUndefined() && constructAsRegexp && flagsArg.isUndefined()) { … … 275 276 if (!flagsArg.isUndefined()) { 276 277 RegExpFlags flags = toFlags(exec, flagsArg); 278 ASSERT(!!scope.exception() == (flags == InvalidFlags)); 277 279 if (flags == InvalidFlags) 278 280 return nullptr; … … 280 282 } 281 283 282 return RegExpObject::create( exec->vm(), structure, regExp);284 return RegExpObject::create(vm, structure, regExp); 283 285 } 284 286 285 287 if (constructAsRegexp) { 286 288 JSValue pattern = patternArg.get(exec, vm.propertyNames->source); 287 if (flagsArg.isUndefined()) 289 RETURN_IF_EXCEPTION(scope, nullptr); 290 if (flagsArg.isUndefined()) { 288 291 flagsArg = patternArg.get(exec, vm.propertyNames->flags); 292 RETURN_IF_EXCEPTION(scope, nullptr); 293 } 289 294 patternArg = pattern; 290 295 } 291 296 297 scope.release(); 292 298 return regExpCreate(exec, globalObject, newTarget, patternArg, flagsArg); 293 299 } -
trunk/Source/JavaScriptCore/runtime/RegExpObject.cpp
r207859 r209101 120 120 return true; 121 121 } 122 if (descriptor.value()) 122 if (descriptor.value()) { 123 123 regExp->setLastIndex(exec, descriptor.value(), false); 124 RETURN_IF_EXCEPTION(scope, false); 125 } 124 126 if (descriptor.writablePresent() && !descriptor.writable()) 125 127 regExp->m_lastIndexIsWritable = false; … … 127 129 } 128 130 131 scope.release(); 129 132 return Base::defineOwnProperty(object, exec, propertyName, descriptor, shouldThrow); 130 133 } … … 180 183 181 184 JSArray* array = constructEmptyArray(exec, nullptr); 182 RETURN_IF_EXCEPTION(scope, JSValue()); 183 185 RETURN_IF_EXCEPTION(scope, { }); 186 187 bool hasException = false; 184 188 auto iterate = [&] () { 185 189 size_t end = result.end; 186 190 size_t length = end - result.start; 187 191 array->push(exec, JSRopeString::createSubstringOfResolved(vm, string, result.start, length)); 192 if (UNLIKELY(scope.exception())) { 193 hasException = true; 194 return; 195 } 188 196 if (!length) 189 197 end = fixEnd(end); … … 217 225 // OK, we have a sensible number of matches. Now we can create them for reals. 218 226 result = savedResult; 219 do 227 do { 220 228 iterate(); 221 while (result); 229 ASSERT(!!scope.exception() == hasException); 230 if (UNLIKELY(hasException)) 231 return { }; 232 } while (result); 222 233 223 234 return array; … … 239 250 240 251 setLastIndex(exec, 0); 241 RETURN_IF_EXCEPTION(scope, JSValue());252 RETURN_IF_EXCEPTION(scope, { }); 242 253 243 254 String s = string->value(exec); … … 246 257 if (regExp->unicode()) { 247 258 unsigned stringLength = s.length(); 259 scope.release(); 248 260 return collectMatches( 249 261 vm, exec, string, s, regExpConstructor, regExp, … … 252 264 }); 253 265 } 254 266 267 scope.release(); 255 268 return collectMatches( 256 269 vm, exec, string, s, regExpConstructor, regExp, -
trunk/Source/JavaScriptCore/runtime/RegExpObjectInlines.h
r208698 r209101 37 37 ExecState* exec, RegExpObject* regExpObject, const String& input) 38 38 { 39 VM& vm = exec->vm(); 40 auto scope = DECLARE_THROW_SCOPE(vm); 39 41 JSValue jsLastIndex = regExpObject->getLastIndex(); 40 42 unsigned lastIndex; … … 42 44 lastIndex = jsLastIndex.asUInt32(); 43 45 if (lastIndex > input.length()) { 46 scope.release(); 44 47 regExpObject->setLastIndex(exec, 0); 45 48 return UINT_MAX; … … 47 50 } else { 48 51 double doubleLastIndex = jsLastIndex.toInteger(exec); 52 RETURN_IF_EXCEPTION(scope, UINT_MAX); 49 53 if (doubleLastIndex < 0 || doubleLastIndex > input.length()) { 54 scope.release(); 50 55 regExpObject->setLastIndex(exec, 0); 51 56 return UINT_MAX; … … 64 69 RegExpConstructor* regExpConstructor = globalObject->regExpConstructor(); 65 70 String input = string->value(exec); 66 RETURN_IF_EXCEPTION(scope, JSValue());71 RETURN_IF_EXCEPTION(scope, { }); 67 72 68 73 bool globalOrSticky = regExp->globalOrSticky(); … … 71 76 if (globalOrSticky) { 72 77 lastIndex = getRegExpObjectLastIndexAsUnsigned(exec, this, input); 78 ASSERT(!scope.exception() || lastIndex == UINT_MAX); 73 79 if (lastIndex == UINT_MAX) 74 80 return jsNull(); … … 80 86 createRegExpMatchesArray(vm, globalObject, string, input, regExp, lastIndex, result); 81 87 if (!array) { 88 scope.release(); 82 89 if (globalOrSticky) 83 90 setLastIndex(exec, 0); … … 87 94 if (globalOrSticky) 88 95 setLastIndex(exec, result.end); 96 RETURN_IF_EXCEPTION(scope, { }); 89 97 regExpConstructor->recordMatch(vm, regExp, string, result); 90 98 return array; … … 101 109 RegExpConstructor* regExpConstructor = globalObject->regExpConstructor(); 102 110 String input = string->value(exec); 103 RETURN_IF_EXCEPTION(scope, MatchResult());111 RETURN_IF_EXCEPTION(scope, { }); 104 112 105 113 if (!regExp->global() && !regExp->sticky()) … … 107 115 108 116 unsigned lastIndex = getRegExpObjectLastIndexAsUnsigned(exec, this, input); 117 ASSERT(!scope.exception() || (lastIndex == UINT_MAX)); 109 118 if (lastIndex == UINT_MAX) 110 119 return MatchResult::failed(); 111 120 112 121 MatchResult result = regExpConstructor->performMatch(vm, regExp, string, input, lastIndex); 122 scope.release(); 113 123 setLastIndex(exec, result.end); 114 124 return result; -
trunk/Source/JavaScriptCore/runtime/RegExpPrototype.cpp
r208698 r209101 184 184 185 185 asRegExpObject(thisValue)->setRegExp(vm, regExp); 186 scope.release(); 186 187 asRegExpObject(thisValue)->setLastIndex(exec, 0); 187 188 return JSValue::encode(thisValue); … … 198 199 auto scope = DECLARE_THROW_SCOPE(vm); 199 200 200 JSValue globalValue = regexp->get(exec, exec->propertyNames().global);201 JSValue globalValue = regexp->get(exec, vm.propertyNames->global); 201 202 RETURN_IF_EXCEPTION(scope, string); 202 JSValue ignoreCaseValue = regexp->get(exec, exec->propertyNames().ignoreCase);203 JSValue ignoreCaseValue = regexp->get(exec, vm.propertyNames->ignoreCase); 203 204 RETURN_IF_EXCEPTION(scope, string); 204 JSValue multilineValue = regexp->get(exec, exec->propertyNames().multiline);205 JSValue multilineValue = regexp->get(exec, vm.propertyNames->multiline); 205 206 RETURN_IF_EXCEPTION(scope, string); 206 JSValue unicodeValue = regexp->get(exec, exec->propertyNames().unicode);207 JSValue unicodeValue = regexp->get(exec, vm.propertyNames->unicode); 207 208 RETURN_IF_EXCEPTION(scope, string); 208 JSValue stickyValue = regexp->get(exec, exec->propertyNames().sticky);209 JSValue stickyValue = regexp->get(exec, vm.propertyNames->sticky); 209 210 RETURN_IF_EXCEPTION(scope, string); 210 211 … … 237 238 238 239 StringRecursionChecker checker(exec, thisObject); 240 ASSERT(!scope.exception() || checker.earlyReturnValue()); 239 241 if (JSValue earlyReturnValue = checker.earlyReturnValue()) 240 242 return JSValue::encode(earlyReturnValue); … … 622 624 // c. Perform ! CreateDataProperty(A, "0", S). 623 625 // d. Return A. 624 if (!regexp->match(vm, input, 0)) 626 if (!regexp->match(vm, input, 0)) { 625 627 result->putDirectIndex(exec, 0, inputString); 628 RETURN_IF_EXCEPTION(scope, encodedJSValue()); 629 } 626 630 return JSValue::encode(result); 627 631 } … … 644 648 [&] (bool isDefined, unsigned start, unsigned length) -> SplitControl { 645 649 result->putDirectIndex(exec, resultLength++, isDefined ? JSRopeString::createSubstringOfResolved(vm, inputString, start, length) : jsUndefined()); 650 RETURN_IF_EXCEPTION(scope, AbortSplit); 646 651 if (resultLength >= limit) 647 652 return AbortSplit; 648 653 return ContinueSplit; 649 654 }); 650 655 RETURN_IF_EXCEPTION(scope, encodedJSValue()); 656 651 657 if (resultLength >= limit) 652 658 return JSValue::encode(result); … … 654 660 // 20. Let T be a String value equal to the substring of S consisting of the elements at indices p (inclusive) through size (exclusive). 655 661 // 21. Perform ! CreateDataProperty(A, ! ToString(lengthA), T). 662 scope.release(); 656 663 result->putDirectIndex(exec, resultLength, JSRopeString::createSubstringOfResolved(vm, inputString, position, inputSize - position)); 657 664 … … 680 687 if (resultLength + dryRunCount >= MAX_STORAGE_VECTOR_LENGTH) { 681 688 throwOutOfMemoryError(exec, scope); 682 return JSValue::encode(jsUndefined());689 return encodedJSValue(); 683 690 } 684 691 … … 694 701 [&] (bool isDefined, unsigned start, unsigned length) -> SplitControl { 695 702 result->putDirectIndex(exec, resultLength++, isDefined ? JSRopeString::createSubstringOfResolved(vm, inputString, start, length) : jsUndefined()); 703 RETURN_IF_EXCEPTION(scope, AbortSplit); 696 704 if (resultLength >= limit) 697 705 return AbortSplit; 698 706 return ContinueSplit; 699 707 }); 700 708 RETURN_IF_EXCEPTION(scope, encodedJSValue()); 709 701 710 if (resultLength >= limit) 702 711 return JSValue::encode(result); … … 704 713 // 20. Let T be a String value equal to the substring of S consisting of the elements at indices p (inclusive) through size (exclusive). 705 714 // 21. Perform ! CreateDataProperty(A, ! ToString(lengthA), T). 715 scope.release(); 706 716 result->putDirectIndex(exec, resultLength, JSRopeString::createSubstringOfResolved(vm, inputString, position, inputSize - position)); 707 717 // 22. Return A.
Note:
See TracChangeset
for help on using the changeset viewer.