Changeset 172741 in webkit


Ignore:
Timestamp:
Aug 18, 2014, 7:48:00 PM (11 years ago)
Author:
fpizlo@apple.com
Message:

REGRESSION(r172401): for-in optimization no longer works at all
https://bugs.webkit.org/show_bug.cgi?id=136056

Reviewed by Mark Hahnenberg.

This is a partial roll-out of r172401. It turns out that the fix wasn't actually fixing a
real bug (since it's fine to use op_get_direct_pname on the wrong base because it has a
structure check) and it was actually breaking the entire for-in optimization (since there is
no way that we can statically prove that the base matches, because the base we see is a
newly created temporary, and anyway doing it right would be really hard in our bytecode
because it's 3AC form).

But, I added a new test for the problem, and kept the original test. Both the old test and
the new test prove that r172401 wasn't fixing what it thought it was fixing. To the extent
that it resolved crashes it was because it just disabled the for-in optimization entirely.

  • bytecompiler/BytecodeGenerator.cpp:

(JSC::BytecodeGenerator::emitGetByVal):
(JSC::BytecodeGenerator::pushIndexedForInScope):
(JSC::BytecodeGenerator::pushStructureForInScope):

  • bytecompiler/BytecodeGenerator.h:

(JSC::ForInContext::ForInContext):
(JSC::StructureForInContext::StructureForInContext):
(JSC::IndexedForInContext::IndexedForInContext):
(JSC::ForInContext::base): Deleted.

  • bytecompiler/NodesCodegen.cpp:

(JSC::ForInNode::emitMultiLoopBytecode):

  • tests/stress/for-in-base-reassigned.js: Added.
  • tests/stress/for-in-base-reassigned-later.js: Added.
  • tests/stress/for-in-base-reassigned-later-and-change-structure.js: Added.
Location:
trunk/Source/JavaScriptCore
Files:
3 added
4 edited

Legend:

Unmodified
Added
Removed
  • trunk/Source/JavaScriptCore/ChangeLog

    r172739 r172741  
     12014-08-18  Filip Pizlo  <fpizlo@apple.com>
     2
     3        REGRESSION(r172401): for-in optimization no longer works at all
     4        https://bugs.webkit.org/show_bug.cgi?id=136056
     5
     6        Reviewed by Mark Hahnenberg.
     7       
     8        This is a partial roll-out of r172401. It turns out that the fix wasn't actually fixing a
     9        real bug (since it's fine to use op_get_direct_pname on the wrong base because it has a
     10        structure check) and it was actually breaking the entire for-in optimization (since there is
     11        no way that we can statically prove that the base matches, because the base we see is a
     12        newly created temporary, and anyway doing it right would be really hard in our bytecode
     13        because it's 3AC form).
     14       
     15        But, I added a new test for the problem, and kept the original test. Both the old test and
     16        the new test prove that r172401 wasn't fixing what it thought it was fixing. To the extent
     17        that it resolved crashes it was because it just disabled the for-in optimization entirely.
     18
     19        * bytecompiler/BytecodeGenerator.cpp:
     20        (JSC::BytecodeGenerator::emitGetByVal):
     21        (JSC::BytecodeGenerator::pushIndexedForInScope):
     22        (JSC::BytecodeGenerator::pushStructureForInScope):
     23        * bytecompiler/BytecodeGenerator.h:
     24        (JSC::ForInContext::ForInContext):
     25        (JSC::StructureForInContext::StructureForInContext):
     26        (JSC::IndexedForInContext::IndexedForInContext):
     27        (JSC::ForInContext::base): Deleted.
     28        * bytecompiler/NodesCodegen.cpp:
     29        (JSC::ForInNode::emitMultiLoopBytecode):
     30        * tests/stress/for-in-base-reassigned.js: Added.
     31        * tests/stress/for-in-base-reassigned-later.js: Added.
     32        * tests/stress/for-in-base-reassigned-later-and-change-structure.js: Added.
     33
    1342014-08-18  Mark Lam  <mark.lam@apple.com>
    235
  • trunk/Source/JavaScriptCore/bytecompiler/BytecodeGenerator.cpp

    r172614 r172741  
    14231423    for (size_t i = m_forInContextStack.size(); i > 0; i--) {
    14241424        ForInContext* context = m_forInContextStack[i - 1].get();
    1425         if (context->base() != base)
    1426             continue;
    1427 
    14281425        if (context->local() != property)
    14291426            continue;
     
    25872584}
    25882585
    2589 void BytecodeGenerator::pushIndexedForInScope(RegisterID* baseRegister, RegisterID* localRegister, RegisterID* indexRegister)
     2586void BytecodeGenerator::pushIndexedForInScope(RegisterID* localRegister, RegisterID* indexRegister)
    25902587{
    25912588    if (!localRegister)
    25922589        return;
    2593     m_forInContextStack.append(std::make_unique<IndexedForInContext>(baseRegister, localRegister, indexRegister));
     2590    m_forInContextStack.append(std::make_unique<IndexedForInContext>(localRegister, indexRegister));
    25942591}
    25952592
     
    26012598}
    26022599
    2603 void BytecodeGenerator::pushStructureForInScope(RegisterID* baseRegister, RegisterID* localRegister, RegisterID* indexRegister, RegisterID* propertyRegister, RegisterID* enumeratorRegister)
     2600void BytecodeGenerator::pushStructureForInScope(RegisterID* localRegister, RegisterID* indexRegister, RegisterID* propertyRegister, RegisterID* enumeratorRegister)
    26042601{
    26052602    if (!localRegister)
    26062603        return;
    2607     m_forInContextStack.append(std::make_unique<StructureForInContext>(baseRegister, localRegister, indexRegister, propertyRegister, enumeratorRegister));
     2604    m_forInContextStack.append(std::make_unique<StructureForInContext>(localRegister, indexRegister, propertyRegister, enumeratorRegister));
    26082605}
    26092606
  • trunk/Source/JavaScriptCore/bytecompiler/BytecodeGenerator.h

    r172614 r172741  
    100100    class ForInContext {
    101101    public:
    102         ForInContext(RegisterID* baseRegister, RegisterID* localRegister)
    103             : m_baseRegister(baseRegister)
    104             , m_localRegister(localRegister)
     102        ForInContext(RegisterID* localRegister)
     103            : m_localRegister(localRegister)
    105104            , m_isValid(true)
    106105        {
     
    120119        virtual ForInContextType type() const = 0;
    121120
    122         RegisterID* base() const { return m_baseRegister.get(); }
    123121        RegisterID* local() const { return m_localRegister.get(); }
    124122
    125123    private:
    126         RefPtr<RegisterID> m_baseRegister;
    127124        RefPtr<RegisterID> m_localRegister;
    128125        bool m_isValid;
     
    131128    class StructureForInContext : public ForInContext {
    132129    public:
    133         StructureForInContext(RegisterID* baseRegister, RegisterID* localRegister, RegisterID* indexRegister, RegisterID* propertyRegister, RegisterID* enumeratorRegister)
    134             : ForInContext(baseRegister, localRegister)
     130        StructureForInContext(RegisterID* localRegister, RegisterID* indexRegister, RegisterID* propertyRegister, RegisterID* enumeratorRegister)
     131            : ForInContext(localRegister)
    135132            , m_indexRegister(indexRegister)
    136133            , m_propertyRegister(propertyRegister)
     
    156153    class IndexedForInContext : public ForInContext {
    157154    public:
    158         IndexedForInContext(RegisterID* baseRegister, RegisterID* localRegister, RegisterID* indexRegister)
    159             : ForInContext(baseRegister, localRegister)
     155        IndexedForInContext(RegisterID* localRegister, RegisterID* indexRegister)
     156            : ForInContext(localRegister)
    160157            , m_indexRegister(indexRegister)
    161158        {
     
    528525        void popFinallyContext();
    529526
    530         void pushIndexedForInScope(RegisterID* base, RegisterID* local, RegisterID* index);
     527        void pushIndexedForInScope(RegisterID* local, RegisterID* index);
    531528        void popIndexedForInScope(RegisterID* local);
    532         void pushStructureForInScope(RegisterID* base, RegisterID* local, RegisterID* index, RegisterID* property, RegisterID* enumerator);
     529        void pushStructureForInScope(RegisterID* local, RegisterID* index, RegisterID* property, RegisterID* enumerator);
    533530        void popStructureForInScope(RegisterID* local);
    534531        void invalidateForInContextForLocal(RegisterID* local);
  • trunk/Source/JavaScriptCore/bytecompiler/NodesCodegen.cpp

    r172717 r172741  
    20712071        this->emitLoopHeader(generator, propertyName.get());
    20722072
    2073         generator.pushIndexedForInScope(base.get(), local.get(), i.get());
     2073        generator.pushIndexedForInScope(local.get(), i.get());
    20742074        generator.emitNode(dst, m_statement);
    20752075        generator.popIndexedForInScope(local.get());
     
    21052105        this->emitLoopHeader(generator, propertyName.get());
    21062106
    2107         generator.pushStructureForInScope(base.get(), local.get(), i.get(), propertyName.get(), structureEnumerator.get());
     2107        generator.pushStructureForInScope(local.get(), i.get(), propertyName.get(), structureEnumerator.get());
    21082108        generator.emitNode(dst, m_statement);
    21092109        generator.popStructureForInScope(local.get());
Note: See TracChangeset for help on using the changeset viewer.