Changeset 252160 in webkit


Ignore:
Timestamp:
Nov 6, 2019, 4:29:19 PM (5 years ago)
Author:
mark.lam@apple.com
Message:

JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut() should fire its watchpoint as the last step.
https://bugs.webkit.org/show_bug.cgi?id=203867
<rdar://problem/56813514>

Reviewed by Saam Barati.

JSTests:

  • stress/racy-slow-put-cloned-arguments-when-having-a-bad-time.js: Added.

Source/JavaScriptCore:

JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut() should make all
the array structures SlowPut before firing the watchpoint. Otherwise, the
concurrent JIT may think it's grabbing the slow put version of the structure, but
is actually grabbing the non-SlowPut version because it happened to beat the
mutator in a race to read the structure before the mutator makes it SlowPut.

Also removed some assertions in DFGSpeculativeJIT.cpp that are vulnerable to races
between when the mutator makes all array structures SlowPut and when it fires the
HavingABadTime watchpoint. The FTL equivalent did not have these assertions.

  • dfg/DFGSpeculativeJIT.cpp:

(JSC::DFG::SpeculativeJIT::compileCreateRest):
(JSC::DFG::SpeculativeJIT::compileNewArray):
(JSC::DFG::SpeculativeJIT::compileNewArrayWithSpread):

  • runtime/JSGlobalObject.cpp:

(JSC::JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut):

Location:
trunk
Files:
1 added
4 edited

Legend:

Unmodified
Added
Removed
  • trunk/JSTests/ChangeLog

    r252158 r252160  
     12019-11-06  Mark Lam  <mark.lam@apple.com>
     2
     3        JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut() should fire its watchpoint as the last step.
     4        https://bugs.webkit.org/show_bug.cgi?id=203867
     5        <rdar://problem/56813514>
     6
     7        Reviewed by Saam Barati.
     8
     9        * stress/racy-slow-put-cloned-arguments-when-having-a-bad-time.js: Added.
     10
    1112019-11-06  Commit Queue  <commit-queue@webkit.org>
    212
  • trunk/Source/JavaScriptCore/ChangeLog

    r252158 r252160  
     12019-11-06  Mark Lam  <mark.lam@apple.com>
     2
     3        JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut() should fire its watchpoint as the last step.
     4        https://bugs.webkit.org/show_bug.cgi?id=203867
     5        <rdar://problem/56813514>
     6
     7        Reviewed by Saam Barati.
     8
     9        JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut() should make all
     10        the array structures SlowPut before firing the watchpoint.  Otherwise, the
     11        concurrent JIT may think it's grabbing the slow put version of the structure, but
     12        is actually grabbing the non-SlowPut version because it happened to beat the
     13        mutator in a race to read the structure before the mutator makes it SlowPut.
     14
     15        Also removed some assertions in DFGSpeculativeJIT.cpp that are vulnerable to races
     16        between when the mutator makes all array structures SlowPut and when it fires the
     17        HavingABadTime watchpoint.  The FTL equivalent did not have these assertions.
     18
     19        * dfg/DFGSpeculativeJIT.cpp:
     20        (JSC::DFG::SpeculativeJIT::compileCreateRest):
     21        (JSC::DFG::SpeculativeJIT::compileNewArray):
     22        (JSC::DFG::SpeculativeJIT::compileNewArrayWithSpread):
     23        * runtime/JSGlobalObject.cpp:
     24        (JSC::JSGlobalObject::fireWatchpointAndMakeAllArrayStructuresSlowPut):
     25
    1262019-11-06  Commit Queue  <commit-queue@webkit.org>
    227
  • trunk/Source/JavaScriptCore/dfg/DFGSpeculativeJIT.cpp

    r252021 r252160  
    78167816        // arguments to have arrayLength exceed MIN_ARRAY_STORAGE_CONSTRUCTION_LENGTH.
    78177817        bool shouldAllowForArrayStorageStructureForLargeArrays = false;
    7818         ASSERT(m_jit.graph().globalObjectFor(node->origin.semantic)->restParameterStructure()->indexingMode() == ArrayWithContiguous || m_jit.graph().globalObjectFor(node->origin.semantic)->isHavingABadTime());
    78197818        compileAllocateNewArrayWithSize(m_jit.graph().globalObjectFor(node->origin.semantic), arrayResultGPR, arrayLengthGPR, ArrayWithContiguous, shouldAllowForArrayStorageStructureForLargeArrays);
    78207819
     
    79807979    RegisteredStructure structure = m_jit.graph().registerStructure(globalObject->arrayStructureForIndexingTypeDuringAllocation(node->indexingType()));
    79817980    if (!globalObject->isHavingABadTime() && !hasAnyArrayStorage(node->indexingType())) {
    7982         DFG_ASSERT(m_jit.graph(), node, structure->indexingType() == node->indexingType(), structure->indexingType(), node->indexingType());
    79837981        ASSERT(
    79847982            hasUndecided(structure->indexingType())
     
    81748172            // non-ArrayStorage shaped array.
    81758173            bool shouldAllowForArrayStorageStructureForLargeArrays = false;
    8176             ASSERT(m_jit.graph().globalObjectFor(node->origin.semantic)->restParameterStructure()->indexingType() == ArrayWithContiguous || m_jit.graph().globalObjectFor(node->origin.semantic)->isHavingABadTime());
    81778174            compileAllocateNewArrayWithSize(m_jit.graph().globalObjectFor(node->origin.semantic), resultGPR, lengthGPR, ArrayWithContiguous, shouldAllowForArrayStorageStructureForLargeArrays);
    81788175        }
  • trunk/Source/JavaScriptCore/runtime/JSGlobalObject.cpp

    r252032 r252160  
    15661566        return;
    15671567
    1568     // Make sure that all allocations or indexed storage transitions that are inlining
    1569     // the assumption that it's safe to transition to a non-SlowPut array storage don't
    1570     // do so anymore.
    1571     m_havingABadTimeWatchpoint->fireAll(vm, "Having a bad time");
    1572     ASSERT(isHavingABadTime()); // The watchpoint is what tells us that we're having a bad time.
    1573    
    15741568    // Make sure that all JSArray allocations that load the appropriate structure from
    15751569    // this object now load a structure that uses SlowPut.
     
    15851579    slowPutStructure = ClonedArguments::createSlowPutStructure(vm, this, m_objectPrototype.get());
    15861580    m_clonedArgumentsStructure.set(vm, this, slowPutStructure);
     1581
     1582    // Make sure that all allocations or indexed storage transitions that are inlining
     1583    // the assumption that it's safe to transition to a non-SlowPut array storage don't
     1584    // do so anymore.
     1585    // Note: we are deliberately firing the watchpoint here at the end only after
     1586    // making all the array structures SlowPut. This ensures that the concurrent
     1587    // JIT threads will always get the SlowPut versions of the structures if
     1588    // isHavingABadTime() returns true. The concurrent JIT relies on this.
     1589    m_havingABadTimeWatchpoint->fireAll(vm, "Having a bad time");
     1590    ASSERT(isHavingABadTime()); // The watchpoint is what tells us that we're having a bad time.
    15871591};
    15881592
Note: See TracChangeset for help on using the changeset viewer.