/*
* Copyright (C) 2011 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. AND ITS CONTRIBUTORS ``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 ITS 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.
*/
#ifndef WriteBarrier_h
#define WriteBarrier_h
#include "JSValue.h"
namespace JSC {
class JSCell;
class JSGlobalData;
inline void writeBarrier(JSGlobalData&, const JSCell*, JSValue)
{
}
inline void writeBarrier(JSGlobalData&, const JSCell*, JSCell*)
{
}
typedef enum { } Unknown;
typedef JSValue* HandleSlot;
template <typename T> struct JSValueChecker {
static const bool IsJSValue = false;
};
template <> struct JSValueChecker<JSValue> {
static const bool IsJSValue = true;
};
// We have a separate base class with no constructors for use in Unions.
template <typename T> class WriteBarrierBase {
public:
COMPILE_ASSERT(!JSValueChecker<T>::IsJSValue, WriteBarrier_JSValue_is_invalid__use_unknown);
void set(JSGlobalData& globalData, const JSCell* owner, T* value)
{
this->m_cell = reinterpret_cast<JSCell*>(value);
writeBarrier(globalData, owner, this->m_cell);
#if ENABLE(JSC_ZOMBIES)
ASSERT(!isZombie(owner));
ASSERT(!isZombie(m_cell));
#endif
}
T* get() const
{
return reinterpret_cast<T*>(m_cell);
}
T* operator*() const
{
ASSERT(m_cell);
#if ENABLE(JSC_ZOMBIES)
ASSERT(!isZombie(m_cell));
#endif
return static_cast<T*>(m_cell);
}
T* operator->() const
{
ASSERT(m_cell);
return static_cast<T*>(m_cell);
}
void clear() { m_cell = 0; }
JSCell** slot() { return &m_cell; }
typedef T* (WriteBarrierBase::*UnspecifiedBoolType);
operator UnspecifiedBoolType*() const { return m_cell ? reinterpret_cast<UnspecifiedBoolType*>(1) : 0; }
bool operator!() const { return !m_cell; }
void setWithoutWriteBarrier(T* value)
{
this->m_cell = reinterpret_cast<JSCell*>(value);
#if ENABLE(JSC_ZOMBIES)
ASSERT(!m_cell || !isZombie(m_cell));
#endif
}
private:
JSCell* m_cell;
};
template <> class WriteBarrierBase<Unknown> {
public:
void set(JSGlobalData& globalData, const JSCell* owner, JSValue value)
{
#if ENABLE(JSC_ZOMBIES)
ASSERT(!isZombie(owner));
ASSERT(!value.isZombie());
#endif
m_value = JSValue::encode(value);
writeBarrier(globalData, owner, value);
}
void setWithoutWriteBarrier(JSValue value)
{
#if ENABLE(JSC_ZOMBIES)
ASSERT(!value.isZombie());
#endif
m_value = JSValue::encode(value);
}
JSValue get() const
{
return JSValue::decode(m_value);
}
void clear() { m_value = JSValue::encode(JSValue()); }
void setUndefined() { m_value = JSValue::encode(jsUndefined()); }
bool isNumber() const { return get().isNumber(); }
bool isObject() const { return get().isObject(); }
bool isNull() const { return get().isNull(); }
bool isGetterSetter() const { return get().isGetterSetter(); }
JSValue* slot()
{
union {
EncodedJSValue* v;
JSValue* slot;
} u;
u.v = &m_value;
return u.slot;
}
typedef JSValue (WriteBarrierBase::*UnspecifiedBoolType);
operator UnspecifiedBoolType*() const { return get() ? reinterpret_cast<UnspecifiedBoolType*>(1) : 0; }
bool operator!() const { return !get(); }
private:
EncodedJSValue m_value;
};
template <typename T> class WriteBarrier : public WriteBarrierBase<T> {
public:
WriteBarrier()
{
this->setWithoutWriteBarrier(0);
}
WriteBarrier(JSGlobalData& globalData, const JSCell* owner, T* value)
{
this->set(globalData, owner, value);
}
};
template <> class WriteBarrier<Unknown> : public WriteBarrierBase<Unknown> {
public:
WriteBarrier()
{
this->setWithoutWriteBarrier(JSValue());
}
WriteBarrier(JSGlobalData& globalData, const JSCell* owner, JSValue value)
{
this->set(globalData, owner, value);
}
};
template <typename U, typename V> inline bool operator==(const WriteBarrierBase<U>& lhs, const WriteBarrierBase<V>& rhs)
{
return lhs.get() == rhs.get();
}
} // namespace JSC
#endif // WriteBarrier_h