blob: f9b01686894245730bcc26b107906b561172facf [file] [log] [blame]
/*
* Copyright (C) 2014-2018 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#pragma once
#include "ObjectPropertyConditionSet.h"
#include "PropertyOffset.h"
#include "StructureSet.h"
namespace JSC {
class CallLinkStatus;
class PutByIdVariant {
WTF_MAKE_FAST_ALLOCATED;
public:
enum Kind {
NotSet,
Replace,
Transition,
Setter
};
PutByIdVariant()
: m_kind(NotSet)
, m_offset(invalidOffset)
, m_newStructure(nullptr)
{
}
PutByIdVariant(const PutByIdVariant&);
PutByIdVariant& operator=(const PutByIdVariant&);
static PutByIdVariant replace(const StructureSet&, PropertyOffset);
static PutByIdVariant transition(
const StructureSet& oldStructure, Structure* newStructure,
const ObjectPropertyConditionSet&, PropertyOffset);
static PutByIdVariant setter(
const StructureSet&, PropertyOffset, const ObjectPropertyConditionSet&,
std::unique_ptr<CallLinkStatus>);
Kind kind() const { return m_kind; }
bool isSet() const { return kind() != NotSet; }
bool operator!() const { return !isSet(); }
const StructureSet& structure() const
{
ASSERT(kind() == Replace || kind() == Setter);
return m_oldStructure;
}
const StructureSet& oldStructure() const
{
ASSERT(kind() == Transition || kind() == Replace || kind() == Setter);
return m_oldStructure;
}
const StructureSet& structureSet() const
{
return oldStructure();
}
StructureSet& oldStructure()
{
ASSERT(kind() == Transition || kind() == Replace || kind() == Setter);
return m_oldStructure;
}
StructureSet& structureSet()
{
return oldStructure();
}
Structure* oldStructureForTransition() const;
Structure* newStructure() const
{
ASSERT(kind() == Transition);
return m_newStructure;
}
void fixTransitionToReplaceIfNecessary();
bool writesStructures() const;
bool reallocatesStorage() const;
bool makesCalls() const;
const ObjectPropertyConditionSet& conditionSet() const { return m_conditionSet; }
// We don't support intrinsics for Setters (it would be sweet if we did) but we need this for templated helpers.
Intrinsic intrinsic() const { return NoIntrinsic; }
// This is needed for templated helpers.
bool isPropertyUnset() const { return false; }
PropertyOffset offset() const
{
ASSERT(isSet());
return m_offset;
}
CallLinkStatus* callLinkStatus() const
{
ASSERT(kind() == Setter);
return m_callLinkStatus.get();
}
bool attemptToMerge(const PutByIdVariant& other);
void markIfCheap(SlotVisitor&);
bool finalize(VM&);
void dump(PrintStream&) const;
void dumpInContext(PrintStream&, DumpContext*) const;
private:
bool attemptToMergeTransitionWithReplace(const PutByIdVariant& replace);
Kind m_kind;
PropertyOffset m_offset;
StructureSet m_oldStructure;
Structure* m_newStructure { nullptr };
ObjectPropertyConditionSet m_conditionSet;
std::unique_ptr<CallLinkStatus> m_callLinkStatus;
};
} // namespace JSC