Remove m_subclassData from JSArray, move the attribute to subclass as needed
[WebKit-https.git] / Source / WebCore / bridge / runtime_array.cpp
1 /*
2  * Copyright (C) 2003, 2008 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "runtime_array.h"
28
29 #include <runtime/ArrayPrototype.h>
30 #include <runtime/Error.h>
31 #include <runtime/PropertyNameArray.h>
32 #include "JSDOMBinding.h"
33
34 using namespace WebCore;
35
36 namespace JSC {
37
38 const ClassInfo RuntimeArray::s_info = { "RuntimeArray", &JSArray::s_info, 0, 0, CREATE_METHOD_TABLE(RuntimeArray) };
39
40 RuntimeArray::RuntimeArray(ExecState* exec, Structure* structure)
41     : JSArray(exec->globalData(), structure)
42     , m_array(0)
43 {
44 }
45
46 void RuntimeArray::finishCreation(JSGlobalData& globalData, Bindings::Array* array)
47 {
48     Base::finishCreation(globalData);
49     ASSERT(inherits(&s_info));
50     m_array = array;
51 }
52
53 RuntimeArray::~RuntimeArray()
54 {
55     delete getConcreteArray();
56 }
57
58 void RuntimeArray::destroy(JSCell* cell)
59 {
60     jsCast<RuntimeArray*>(cell)->RuntimeArray::~RuntimeArray();
61 }
62
63 JSValue RuntimeArray::lengthGetter(ExecState*, JSValue slotBase, const Identifier&)
64 {
65     RuntimeArray* thisObj = static_cast<RuntimeArray*>(asObject(slotBase));
66     return jsNumber(thisObj->getLength());
67 }
68
69 JSValue RuntimeArray::indexGetter(ExecState* exec, JSValue slotBase, unsigned index)
70 {
71     RuntimeArray* thisObj = static_cast<RuntimeArray*>(asObject(slotBase));
72     return thisObj->getConcreteArray()->valueAt(exec, index);
73 }
74
75 void RuntimeArray::getOwnPropertyNames(JSObject* object, ExecState* exec, PropertyNameArray& propertyNames, EnumerationMode mode)
76 {
77     RuntimeArray* thisObject = jsCast<RuntimeArray*>(object);
78     unsigned length = thisObject->getLength();
79     for (unsigned i = 0; i < length; ++i)
80         propertyNames.add(Identifier::from(exec, i));
81
82     if (mode == IncludeDontEnumProperties)
83         propertyNames.add(exec->propertyNames().length);
84
85     JSObject::getOwnPropertyNames(thisObject, exec, propertyNames, mode);
86 }
87
88 bool RuntimeArray::getOwnPropertySlot(JSCell* cell, ExecState* exec, const Identifier& propertyName, PropertySlot& slot)
89 {
90     RuntimeArray* thisObject = jsCast<RuntimeArray*>(cell);
91     if (propertyName == exec->propertyNames().length) {
92         slot.setCacheableCustom(thisObject, thisObject->lengthGetter);
93         return true;
94     }
95     
96     bool ok;
97     unsigned index = propertyName.toArrayIndex(ok);
98     if (ok) {
99         if (index < thisObject->getLength()) {
100             slot.setCustomIndex(thisObject, index, thisObject->indexGetter);
101             return true;
102         }
103     }
104     
105     return JSObject::getOwnPropertySlot(thisObject, exec, propertyName, slot);
106 }
107
108 bool RuntimeArray::getOwnPropertyDescriptor(JSObject* object, ExecState* exec, const Identifier& propertyName, PropertyDescriptor& descriptor)
109 {
110     RuntimeArray* thisObject = jsCast<RuntimeArray*>(object);
111     if (propertyName == exec->propertyNames().length) {
112         PropertySlot slot;
113         slot.setCustom(thisObject, lengthGetter);
114         descriptor.setDescriptor(slot.getValue(exec, propertyName), ReadOnly | DontDelete | DontEnum);
115         return true;
116     }
117     
118     bool ok;
119     unsigned index = propertyName.toArrayIndex(ok);
120     if (ok) {
121         if (index < thisObject->getLength()) {
122             PropertySlot slot;
123             slot.setCustomIndex(thisObject, index, indexGetter);
124             descriptor.setDescriptor(slot.getValue(exec, propertyName), DontDelete | DontEnum);
125             return true;
126         }
127     }
128     
129     return JSObject::getOwnPropertyDescriptor(thisObject, exec, propertyName, descriptor);
130 }
131
132 bool RuntimeArray::getOwnPropertySlotByIndex(JSCell* cell, ExecState *exec, unsigned index, PropertySlot& slot)
133 {
134     RuntimeArray* thisObject = jsCast<RuntimeArray*>(cell);
135     if (index < thisObject->getLength()) {
136         slot.setCustomIndex(thisObject, index, thisObject->indexGetter);
137         return true;
138     }
139     
140     return JSObject::getOwnPropertySlotByIndex(thisObject, exec, index, slot);
141 }
142
143 void RuntimeArray::put(JSCell* cell, ExecState* exec, const Identifier& propertyName, JSValue value, PutPropertySlot& slot)
144 {
145     RuntimeArray* thisObject = jsCast<RuntimeArray*>(cell);
146     if (propertyName == exec->propertyNames().length) {
147         throwError(exec, createRangeError(exec, "Range error"));
148         return;
149     }
150     
151     bool ok;
152     unsigned index = propertyName.toArrayIndex(ok);
153     if (ok) {
154         thisObject->getConcreteArray()->setValueAt(exec, index, value);
155         return;
156     }
157     
158     JSObject::put(thisObject, exec, propertyName, value, slot);
159 }
160
161 void RuntimeArray::putByIndex(JSCell* cell, ExecState* exec, unsigned index, JSValue value, bool)
162 {
163     RuntimeArray* thisObject = jsCast<RuntimeArray*>(cell);
164     if (index >= thisObject->getLength()) {
165         throwError(exec, createRangeError(exec, "Range error"));
166         return;
167     }
168     
169     thisObject->getConcreteArray()->setValueAt(exec, index, value);
170 }
171
172 bool RuntimeArray::deleteProperty(JSCell*, ExecState*, const Identifier&)
173 {
174     return false;
175 }
176
177 bool RuntimeArray::deletePropertyByIndex(JSCell*, ExecState*, unsigned)
178 {
179     return false;
180 }
181
182 }