blob: 269e911d7a7f7b39e81cf93c8d2def7fbc9acc3d [file] [log] [blame]
// -*- mode: c++; c-basic-offset: 4 -*-
/*
* Copyright (C) 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
*/
#ifndef KJS_PROPERTY_MAP_H_
#define KJS_PROPERTY_MAP_H_
#include "identifier.h"
#include "protect.h"
#include <wtf/OwnArrayPtr.h>
namespace KJS {
class JSObject;
class JSValue;
class PropertyNameArray;
struct PropertyMapEntry;
struct PropertyMapHashTable;
class SavedProperty : Noncopyable {
public:
// Since we use this in arrays, we allocate it uninitialized
// and then explicitly initialize. This means we can allocate
// the array without initializing every saved property in the
// array twice. To accomplish this, the class uses data members
// with types that don't have constructors.
SavedProperty();
void init(UString::Rep* name, JSValue*, unsigned attributes);
~SavedProperty();
UString::Rep* name() const;
JSValue* value() const;
unsigned attributes() const;
private:
UString::Rep* m_name;
JSValue* m_value;
unsigned m_attributes;
};
struct SavedProperties {
SavedProperties();
~SavedProperties();
unsigned count;
OwnArrayPtr<SavedProperty> properties;
};
class PropertyMap : Noncopyable {
public:
PropertyMap();
~PropertyMap();
void clear();
void put(const Identifier&, JSValue*, unsigned attributes, bool checkReadOnly = false);
void remove(const Identifier&);
JSValue* get(const Identifier&) const;
JSValue* get(const Identifier&, unsigned& attributes) const;
JSValue** getLocation(const Identifier& name);
void mark() const;
void getEnumerablePropertyNames(PropertyNameArray&) const;
void save(SavedProperties&) const;
void restore(const SavedProperties&);
bool hasGetterSetterProperties() const { return m_getterSetterFlag; }
void setHasGetterSetterProperties(bool f) { m_getterSetterFlag = f; }
bool containsGettersOrSetters() const;
private:
typedef PropertyMapEntry Entry;
typedef PropertyMapHashTable Table;
static bool keysMatch(const UString::Rep*, const UString::Rep*);
void expand();
void rehash();
void rehash(unsigned newTableSize);
void createTable();
void insert(const Entry&);
void checkConsistency();
UString::Rep* m_singleEntryKey;
union {
JSValue* singleEntryValue;
Table* table;
} m_u;
short m_singleEntryAttributes;
bool m_getterSetterFlag : 1;
bool m_usingTable : 1;
};
inline PropertyMap::PropertyMap()
: m_singleEntryKey(0)
, m_getterSetterFlag(false)
, m_usingTable(false)
{
}
inline SavedProperty::SavedProperty()
#ifndef NDEBUG
: m_name(0)
, m_value(0)
, m_attributes(0)
#endif
{
}
inline void SavedProperty::init(UString::Rep* name, JSValue* value, unsigned attributes)
{
ASSERT(name);
ASSERT(value);
ASSERT(!m_name);
ASSERT(!m_value);
ASSERT(!m_attributes);
m_name = name;
m_value = value;
m_attributes = attributes;
name->ref();
gcProtect(value);
}
inline SavedProperty::~SavedProperty()
{
ASSERT(m_name);
ASSERT(m_value);
m_name->deref();
gcUnprotect(m_value);
}
inline UString::Rep* SavedProperty::name() const
{
ASSERT(m_name);
ASSERT(m_value);
return m_name;
}
inline JSValue* SavedProperty::value() const
{
ASSERT(m_name);
ASSERT(m_value);
return m_value;
}
inline unsigned SavedProperty::attributes() const
{
ASSERT(m_name);
ASSERT(m_value);
return m_attributes;
}
} // namespace
#endif // _KJS_PROPERTY_MAP_H_