Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Disregard lastIndex for non-global non-sticky regexps #627

Conversation

claudepache
Copy link
Contributor

With these changes , non-global non-sticky regexps with nonwritable
lastIndex property should just work:

  • In RegExpBuiltinExec, remove any access to the lastIndex property for
    non-global, non-sticky regexp.
  • In RegExp.prototype.@@search, avoid to to set the lastIndex property
    when not necessary.

With these changes , non-global non-sticky regexps with nonwritable
lastIndex property should just work:
* In RegExpBuiltinExec, remove any access to the lastIndex property for
non-global, non-sticky regexp.
* In RegExp.prototype.@@search, avoid to to set the lastIndex property
when not necessary.
@claudepache
Copy link
Contributor Author

The intention of this PR is to fix #625. I have attempted to find and remove all plausible write accesses to the lastIndex property on non-global, non-sticky regexps.

@claudepache
Copy link
Contributor Author

I have attempted to find and remove all plausible write accesses to the lastIndex property on non-global, non-sticky regexps.

There remains the unconditional initialisation of rx.lastIndex that is part of the RegExpInitialize abstract operation (step 12). However, I don’t think it is worth to "fix" it; for I think it could go wrong here only when applying the deprecated RegExp#compile().

@littledan
Copy link
Member

littledan commented Jun 30, 2016

Do you think this will be web-compatible? Does this correspond to Firefox or Safari semantics?

@claudepache
Copy link
Contributor Author

claudepache commented Jun 30, 2016

Do you think this will be web-compatible? Does this correspond to Firefox or Safari semantics?

Good question. First, note that setting a value to the lastIndex property of a non-global, non-sticky regexp is already useless, as it is not used when performing a match (see Step 8 of RegExpBuiltinExec). The only significant difference is that that property is no longer updated to 0 after a failed match. Test cases:

(function() {
    var rx = /a/
    rx.lastIndex = 3
    rx.exec('b')
    return rx.lastIndex
})()
  • 0 in Firefox, Chrome, Edge (in accordance to current and ES5 spec);
  • 3 in Safari (in accordance to the proposed changes).
(function() {
    var rx = /a/
    rx.lastIndex = 3
    'b'.match(rx)
    return rx.lastIndex
})()
  • 0 in Chrome, Edge, Firefox Nightly (in accordance to current and ES5 spec);
  • 3 in Safari and stable Firefox (in accordance to the proposed changes).

@claudepache
Copy link
Contributor Author

Also, just in case it was not clear (note the /g flag):

'b'.match(Object.freeze(/a/g))

Throws in current Firefox, Safari, Chrome, as well as any version of spec that supports Object.freeze.

@littledan
Copy link
Member

Would we need all of the changes in this patch to get the intended effect, or just the changes in lines 28700-28701?

@claudepache
Copy link
Contributor Author

Would we need all of the changes in this patch to get the intended effect, or just the changes in lines 28700-28701?

  • The changes in lines 28690/28694 just avoids a useless read access to rx.lastIndex.
  • The changes of RegExp.prototype.@@search in lines 28940ff are needed in order to support String#search on frozen non-global regexps. (More precisely, it supports the most likely case, where lastIndex is blocked to 0: if we want to support the other cases, it becomes slightly more complicated.)

@bterlson
Copy link
Member

bterlson commented Jul 1, 2016

This change seems good to me - no need to do puts that are useless. However, still needs-consensus. Any volunteers to lead the discussion at TC39?

@bterlson bterlson added the needs consensus This needs committee consensus before it can be eligible to be merged. label Jul 1, 2016
@leobalter
Copy link
Member

Count me in. I don't have anything complex to bring so I'll have time to do this.

@littledan
Copy link
Member

I'd be happier with this if we made this a more minimal change, avoiding the useless yet observable changes to reading lastIndex. It sounds like the changes to search are needed, however. I will unfortunately not have data beyond the bug report to contribute to the meeting.

@claudepache
Copy link
Contributor Author

claudepache commented Jul 12, 2016

I'd be happier with this if we made this a more minimal change, avoiding the useless yet observable changes to reading lastIndex. It sounds like the changes to search are needed, however. I will unfortunately not have data beyond the bug report to contribute to the meeting.

About the useless read of lastIndex. This is part of the RegExpBuiltinExec abstract operation, which is applied only to objects that have a [[RegExpMatcher]] internal slots; and those objects always have an own nonconfigurable data property named lastIndex (see RegExpAlloc). So that any useless reading of that property on such object is truly unobservable.

About the changes in RegExp.prototype.@@search: It might be possible that the change is not needed for webcompat; however, I have tried to be consistent in making frozen regexp to just work under reasonable conditions, rather than to modify the strict minimum number of lines needed for resolving the particular reported issue. Consistency is important in order to avoid bugs.

@arai-a
Copy link
Contributor

arai-a commented Jul 15, 2016

ToLength is observable there.

let r = /a/;
a.lastIndex = { valueOf() { throw "hello" } };
r.test("a"); // this throws "hello", while ToLength operation

@claudepache
Copy link
Contributor Author

ToLength is observable there.

True. This is probably only significant for checking how scrupulously implementations follow the spec. Test case:

function test(rx) {
    rx.lastIndex = { valueOf: function () { throw "ok" } }
    try {
        rx.exec('foo')
    }
    catch (e) {
        if (e === "ok")
            return true
    }
    return false
}

Results:

test(/a/g) test(/a/)
ECMA262, Firefox, Chrome true true
Safari, IE, Edge true false

@leobalter
Copy link
Member

This reached consensus on the July meeting. :shipit:

@littledan
Copy link
Member

littledan commented Jul 28, 2016

Sounds like we need test262 tests before merging this PR. Who wants to write them?

Edit: Oh, I see a bug was filed at tc39/test262#737 .

@leobalter
Copy link
Member

it's good to verify the work on tc39/test262#711 to avoid any collision.

I can help @claudepache to get the tests done.

@bterlson bterlson added the needs test262 tests The proposal should specify how to test an implementation. Ideally via github.com/tc39/test262 label Jul 28, 2016
@tcare
Copy link
Member

tcare commented Aug 4, 2016

No collision that I could see.

kisg pushed a commit to paul99/v8mips that referenced this pull request Sep 14, 2016
This implements tc39/ecma262#627.

BUG=v8:5360

Review-Url: https://codereview.chromium.org/2339443002
Cr-Commit-Position: refs/heads/master@{#39402}
schuay added a commit to schuay/test262 that referenced this pull request Sep 16, 2016
Update tests for the new lastIndex semantics introduced in
tc39/ecma262#627.
schuay added a commit to schuay/test262 that referenced this pull request Sep 16, 2016
Add new tests for the lastIndex semantic change introduced in
tc39/ecma262#627.
schuay added a commit to schuay/test262 that referenced this pull request Sep 16, 2016
Add new tests for the lastIndex semantic change introduced in
tc39/ecma262#627.
schuay added a commit to schuay/test262 that referenced this pull request Sep 19, 2016
Add and update tests for the lastIndex semantic change introduced in
tc39/ecma262#627.
schuay added a commit to schuay/test262 that referenced this pull request Sep 19, 2016
Add and update tests for the lastIndex semantic change introduced in
tc39/ecma262#627.
tcare pushed a commit to tc39/test262 that referenced this pull request Sep 29, 2016
* Tests for new lastIndex semantics

Add and update tests for the lastIndex semantic change introduced in
tc39/ecma262#627.

* Address comments
@bterlson
Copy link
Member

@tcare / @leobalter we merged test262 tests for this right? Shall I take this PR?

@bterlson bterlson removed the needs consensus This needs committee consensus before it can be eligible to be merged. label Sep 29, 2016
@leobalter
Copy link
Member

yes, please

kisg pushed a commit to paul99/v8mips that referenced this pull request Nov 15, 2016
Implements upcoming changes to @@search according to
tc39/ecma262#627.

This also adds SameValue to CodeStubAssembler and extracts a part of
CSA::TruncateTaggedToFloat64.

BUG=v8:5339

Review-Url: https://codereview.chromium.org/2438683005
Cr-Commit-Position: refs/heads/master@{#41000}
@schuay
Copy link
Contributor

schuay commented Dec 21, 2016

@bterlson Any news on merging this?

@bterlson
Copy link
Member

@schuay thanks for the ping, I think it's good to go. I removed the label and then promptly lost it :-P

@bterlson bterlson merged commit 395fede into tc39:master Dec 21, 2016
@bakkot bakkot removed the needs test262 tests The proposal should specify how to test an implementation. Ideally via github.com/tc39/test262 label Feb 8, 2017
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

9 participants