2 * Copyright (C) 2006, 2008 Apple Inc. All rights reserved.
3 * Copyright (C) 2007 Eric Seidel <eric@webkit.org>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
15 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
17 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
18 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
19 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
20 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
21 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
22 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 #include "ExceptionHelpers.h"
31 #include "JSCallbackFunction.h"
32 #include "JSClassRef.h"
33 #include "JSFunction.h"
34 #include "JSGlobalObject.h"
36 #include "JSObjectRef.h"
38 #include "JSStringRef.h"
39 #include "OpaqueJSString.h"
40 #include "PropertyNameArray.h"
41 #include <wtf/Vector.h>
46 inline JSCallbackObject
<Base
>* JSCallbackObject
<Base
>::asCallbackObject(JSValue value
)
48 ASSERT(asObject(value
)->inherits(&s_info
));
49 return static_cast<JSCallbackObject
*>(asObject(value
));
53 JSCallbackObject
<Base
>::JSCallbackObject(ExecState
* exec
, JSGlobalObject
* globalObject
, Structure
* structure
, JSClassRef jsClass
, void* data
)
54 : Base(globalObject
, structure
)
55 , m_callbackObjectData(adoptPtr(new JSCallbackObjectData(data
, jsClass
)))
57 ASSERT(Base::inherits(&s_info
));
61 // Global object constructor.
62 // FIXME: Move this into a separate JSGlobalCallbackObject class derived from this one.
64 JSCallbackObject
<Base
>::JSCallbackObject(JSGlobalData
& globalData
, JSClassRef jsClass
, Structure
* structure
)
65 : Base(globalData
, structure
)
66 , m_callbackObjectData(adoptPtr(new JSCallbackObjectData(0, jsClass
)))
68 ASSERT(Base::inherits(&s_info
));
69 ASSERT(Base::isGlobalObject());
70 init(static_cast<JSGlobalObject
*>(this)->globalExec());
74 void JSCallbackObject
<Base
>::init(ExecState
* exec
)
78 Vector
<JSObjectInitializeCallback
, 16> initRoutines
;
79 JSClassRef jsClass
= classRef();
81 if (JSObjectInitializeCallback initialize
= jsClass
->initialize
)
82 initRoutines
.append(initialize
);
83 } while ((jsClass
= jsClass
->parentClass
));
85 // initialize from base to derived
86 for (int i
= static_cast<int>(initRoutines
.size()) - 1; i
>= 0; i
--) {
87 APICallbackShim
callbackShim(exec
);
88 JSObjectInitializeCallback initialize
= initRoutines
[i
];
89 initialize(toRef(exec
), toRef(this));
92 bool needsFinalizer
= false;
93 for (JSClassRef jsClassPtr
= classRef(); jsClassPtr
&& !needsFinalizer
; jsClassPtr
= jsClassPtr
->parentClass
)
94 needsFinalizer
= jsClassPtr
->finalize
;
96 HandleSlot slot
= exec
->globalData().allocateGlobalHandle();
97 HandleHeap::heapFor(slot
)->makeWeak(slot
, m_callbackObjectData
.get(), classRef());
98 HandleHeap::heapFor(slot
)->writeBarrier(slot
, this);
103 template <class Base
>
104 UString JSCallbackObject
<Base
>::className() const
106 UString thisClassName
= classRef()->className();
107 if (!thisClassName
.isEmpty())
108 return thisClassName
;
110 return Base::className();
113 template <class Base
>
114 bool JSCallbackObject
<Base
>::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
116 JSContextRef ctx
= toRef(exec
);
117 JSObjectRef thisRef
= toRef(this);
118 RefPtr
<OpaqueJSString
> propertyNameRef
;
120 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
121 // optional optimization to bypass getProperty in cases when we only need to know if the property exists
122 if (JSObjectHasPropertyCallback hasProperty
= jsClass
->hasProperty
) {
123 if (!propertyNameRef
)
124 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
125 APICallbackShim
callbackShim(exec
);
126 if (hasProperty(ctx
, thisRef
, propertyNameRef
.get())) {
127 slot
.setCustom(this, callbackGetter
);
130 } else if (JSObjectGetPropertyCallback getProperty
= jsClass
->getProperty
) {
131 if (!propertyNameRef
)
132 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
133 JSValueRef exception
= 0;
136 APICallbackShim
callbackShim(exec
);
137 value
= getProperty(ctx
, thisRef
, propertyNameRef
.get(), &exception
);
140 throwError(exec
, toJS(exec
, exception
));
141 slot
.setValue(jsUndefined());
145 slot
.setValue(toJS(exec
, value
));
150 if (OpaqueJSClassStaticValuesTable
* staticValues
= jsClass
->staticValues(exec
)) {
151 if (staticValues
->contains(propertyName
.impl())) {
152 JSValue value
= getStaticValue(exec
, propertyName
);
154 slot
.setValue(value
);
160 if (OpaqueJSClassStaticFunctionsTable
* staticFunctions
= jsClass
->staticFunctions(exec
)) {
161 if (staticFunctions
->contains(propertyName
.impl())) {
162 slot
.setCustom(this, staticFunctionGetter
);
168 return Base::getOwnPropertySlot(exec
, propertyName
, slot
);
171 template <class Base
>
172 bool JSCallbackObject
<Base
>::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
175 if (getOwnPropertySlot(exec
, propertyName
, slot
)) {
176 // Ideally we should return an access descriptor, but returning a value descriptor is better than nothing.
177 JSValue value
= slot
.getValue(exec
, propertyName
);
178 if (!exec
->hadException())
179 descriptor
.setValue(value
);
180 // We don't know whether the property is configurable, but assume it is.
181 descriptor
.setConfigurable(true);
182 // We don't know whether the property is enumerable (we could call getOwnPropertyNames() to find out), but assume it isn't.
183 descriptor
.setEnumerable(false);
187 return Base::getOwnPropertyDescriptor(exec
, propertyName
, descriptor
);
190 template <class Base
>
191 void JSCallbackObject
<Base
>::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue value
, PutPropertySlot
& slot
)
193 JSContextRef ctx
= toRef(exec
);
194 JSObjectRef thisRef
= toRef(this);
195 RefPtr
<OpaqueJSString
> propertyNameRef
;
196 JSValueRef valueRef
= toRef(exec
, value
);
198 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
199 if (JSObjectSetPropertyCallback setProperty
= jsClass
->setProperty
) {
200 if (!propertyNameRef
)
201 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
202 JSValueRef exception
= 0;
205 APICallbackShim
callbackShim(exec
);
206 result
= setProperty(ctx
, thisRef
, propertyNameRef
.get(), valueRef
, &exception
);
209 throwError(exec
, toJS(exec
, exception
));
210 if (result
|| exception
)
214 if (OpaqueJSClassStaticValuesTable
* staticValues
= jsClass
->staticValues(exec
)) {
215 if (StaticValueEntry
* entry
= staticValues
->get(propertyName
.impl())) {
216 if (entry
->attributes
& kJSPropertyAttributeReadOnly
)
218 if (JSObjectSetPropertyCallback setProperty
= entry
->setProperty
) {
219 if (!propertyNameRef
)
220 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
221 JSValueRef exception
= 0;
224 APICallbackShim
callbackShim(exec
);
225 result
= setProperty(ctx
, thisRef
, propertyNameRef
.get(), valueRef
, &exception
);
228 throwError(exec
, toJS(exec
, exception
));
229 if (result
|| exception
)
235 if (OpaqueJSClassStaticFunctionsTable
* staticFunctions
= jsClass
->staticFunctions(exec
)) {
236 if (StaticFunctionEntry
* entry
= staticFunctions
->get(propertyName
.impl())) {
237 if (entry
->attributes
& kJSPropertyAttributeReadOnly
)
239 JSCallbackObject
<Base
>::putDirect(exec
->globalData(), propertyName
, value
); // put as override property
245 return Base::put(exec
, propertyName
, value
, slot
);
248 template <class Base
>
249 bool JSCallbackObject
<Base
>::deleteProperty(ExecState
* exec
, const Identifier
& propertyName
)
251 JSContextRef ctx
= toRef(exec
);
252 JSObjectRef thisRef
= toRef(this);
253 RefPtr
<OpaqueJSString
> propertyNameRef
;
255 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
256 if (JSObjectDeletePropertyCallback deleteProperty
= jsClass
->deleteProperty
) {
257 if (!propertyNameRef
)
258 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
259 JSValueRef exception
= 0;
262 APICallbackShim
callbackShim(exec
);
263 result
= deleteProperty(ctx
, thisRef
, propertyNameRef
.get(), &exception
);
266 throwError(exec
, toJS(exec
, exception
));
267 if (result
|| exception
)
271 if (OpaqueJSClassStaticValuesTable
* staticValues
= jsClass
->staticValues(exec
)) {
272 if (StaticValueEntry
* entry
= staticValues
->get(propertyName
.impl())) {
273 if (entry
->attributes
& kJSPropertyAttributeDontDelete
)
279 if (OpaqueJSClassStaticFunctionsTable
* staticFunctions
= jsClass
->staticFunctions(exec
)) {
280 if (StaticFunctionEntry
* entry
= staticFunctions
->get(propertyName
.impl())) {
281 if (entry
->attributes
& kJSPropertyAttributeDontDelete
)
288 return Base::deleteProperty(exec
, propertyName
);
291 template <class Base
>
292 bool JSCallbackObject
<Base
>::deleteProperty(ExecState
* exec
, unsigned propertyName
)
294 return deleteProperty(exec
, Identifier::from(exec
, propertyName
));
297 template <class Base
>
298 ConstructType JSCallbackObject
<Base
>::getConstructData(ConstructData
& constructData
)
300 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
301 if (jsClass
->callAsConstructor
) {
302 constructData
.native
.function
= construct
;
303 return ConstructTypeHost
;
306 return ConstructTypeNone
;
309 template <class Base
>
310 EncodedJSValue JSCallbackObject
<Base
>::construct(ExecState
* exec
)
312 JSObject
* constructor
= exec
->callee();
313 JSContextRef execRef
= toRef(exec
);
314 JSObjectRef constructorRef
= toRef(constructor
);
316 for (JSClassRef jsClass
= static_cast<JSCallbackObject
<Base
>*>(constructor
)->classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
317 if (JSObjectCallAsConstructorCallback callAsConstructor
= jsClass
->callAsConstructor
) {
318 int argumentCount
= static_cast<int>(exec
->argumentCount());
319 Vector
<JSValueRef
, 16> arguments(argumentCount
);
320 for (int i
= 0; i
< argumentCount
; i
++)
321 arguments
[i
] = toRef(exec
, exec
->argument(i
));
322 JSValueRef exception
= 0;
325 APICallbackShim
callbackShim(exec
);
326 result
= toJS(callAsConstructor(execRef
, constructorRef
, argumentCount
, arguments
.data(), &exception
));
329 throwError(exec
, toJS(exec
, exception
));
330 return JSValue::encode(result
);
334 ASSERT_NOT_REACHED(); // getConstructData should prevent us from reaching here
335 return JSValue::encode(JSValue());
338 template <class Base
>
339 bool JSCallbackObject
<Base
>::hasInstance(ExecState
* exec
, JSValue value
, JSValue
)
341 JSContextRef execRef
= toRef(exec
);
342 JSObjectRef thisRef
= toRef(this);
344 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
345 if (JSObjectHasInstanceCallback hasInstance
= jsClass
->hasInstance
) {
346 JSValueRef valueRef
= toRef(exec
, value
);
347 JSValueRef exception
= 0;
350 APICallbackShim
callbackShim(exec
);
351 result
= hasInstance(execRef
, thisRef
, valueRef
, &exception
);
354 throwError(exec
, toJS(exec
, exception
));
361 template <class Base
>
362 CallType JSCallbackObject
<Base
>::getCallData(CallData
& callData
)
364 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
365 if (jsClass
->callAsFunction
) {
366 callData
.native
.function
= call
;
373 template <class Base
>
374 EncodedJSValue JSCallbackObject
<Base
>::call(ExecState
* exec
)
376 JSContextRef execRef
= toRef(exec
);
377 JSObjectRef functionRef
= toRef(exec
->callee());
378 JSObjectRef thisObjRef
= toRef(exec
->hostThisValue().toThisObject(exec
));
380 for (JSClassRef jsClass
= static_cast<JSCallbackObject
<Base
>*>(toJS(functionRef
))->classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
381 if (JSObjectCallAsFunctionCallback callAsFunction
= jsClass
->callAsFunction
) {
382 int argumentCount
= static_cast<int>(exec
->argumentCount());
383 Vector
<JSValueRef
, 16> arguments(argumentCount
);
384 for (int i
= 0; i
< argumentCount
; i
++)
385 arguments
[i
] = toRef(exec
, exec
->argument(i
));
386 JSValueRef exception
= 0;
389 APICallbackShim
callbackShim(exec
);
390 result
= toJS(exec
, callAsFunction(execRef
, functionRef
, thisObjRef
, argumentCount
, arguments
.data(), &exception
));
393 throwError(exec
, toJS(exec
, exception
));
394 return JSValue::encode(result
);
398 ASSERT_NOT_REACHED(); // getCallData should prevent us from reaching here
399 return JSValue::encode(JSValue());
402 template <class Base
>
403 void JSCallbackObject
<Base
>::getOwnPropertyNames(ExecState
* exec
, PropertyNameArray
& propertyNames
, EnumerationMode mode
)
405 JSContextRef execRef
= toRef(exec
);
406 JSObjectRef thisRef
= toRef(this);
408 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
409 if (JSObjectGetPropertyNamesCallback getPropertyNames
= jsClass
->getPropertyNames
) {
410 APICallbackShim
callbackShim(exec
);
411 getPropertyNames(execRef
, thisRef
, toRef(&propertyNames
));
414 if (OpaqueJSClassStaticValuesTable
* staticValues
= jsClass
->staticValues(exec
)) {
415 typedef OpaqueJSClassStaticValuesTable::const_iterator iterator
;
416 iterator end
= staticValues
->end();
417 for (iterator it
= staticValues
->begin(); it
!= end
; ++it
) {
418 StringImpl
* name
= it
->first
.get();
419 StaticValueEntry
* entry
= it
->second
;
420 if (entry
->getProperty
&& (!(entry
->attributes
& kJSPropertyAttributeDontEnum
) || (mode
== IncludeDontEnumProperties
)))
421 propertyNames
.add(Identifier(exec
, name
));
425 if (OpaqueJSClassStaticFunctionsTable
* staticFunctions
= jsClass
->staticFunctions(exec
)) {
426 typedef OpaqueJSClassStaticFunctionsTable::const_iterator iterator
;
427 iterator end
= staticFunctions
->end();
428 for (iterator it
= staticFunctions
->begin(); it
!= end
; ++it
) {
429 StringImpl
* name
= it
->first
.get();
430 StaticFunctionEntry
* entry
= it
->second
;
431 if (!(entry
->attributes
& kJSPropertyAttributeDontEnum
) || (mode
== IncludeDontEnumProperties
))
432 propertyNames
.add(Identifier(exec
, name
));
437 Base::getOwnPropertyNames(exec
, propertyNames
, mode
);
440 template <class Base
>
441 double JSCallbackObject
<Base
>::toNumber(ExecState
* exec
) const
443 // We need this check to guard against the case where this object is rhs of
444 // a binary expression where lhs threw an exception in its conversion to
446 if (exec
->hadException())
448 JSContextRef ctx
= toRef(exec
);
449 JSObjectRef thisRef
= toRef(this);
451 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
)
452 if (JSObjectConvertToTypeCallback convertToType
= jsClass
->convertToType
) {
453 JSValueRef exception
= 0;
456 APICallbackShim
callbackShim(exec
);
457 value
= convertToType(ctx
, thisRef
, kJSTypeNumber
, &exception
);
460 throwError(exec
, toJS(exec
, exception
));
466 return toJS(exec
, value
).getNumber(dValue
) ? dValue
: NaN
;
469 return Base::toNumber(exec
);
472 template <class Base
>
473 UString JSCallbackObject
<Base
>::toString(ExecState
* exec
) const
475 JSContextRef ctx
= toRef(exec
);
476 JSObjectRef thisRef
= toRef(this);
478 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
)
479 if (JSObjectConvertToTypeCallback convertToType
= jsClass
->convertToType
) {
480 JSValueRef exception
= 0;
483 APICallbackShim
callbackShim(exec
);
484 value
= convertToType(ctx
, thisRef
, kJSTypeString
, &exception
);
487 throwError(exec
, toJS(exec
, exception
));
491 return toJS(exec
, value
).getString(exec
);
494 return Base::toString(exec
);
497 template <class Base
>
498 void JSCallbackObject
<Base
>::setPrivate(void* data
)
500 m_callbackObjectData
->privateData
= data
;
503 template <class Base
>
504 void* JSCallbackObject
<Base
>::getPrivate()
506 return m_callbackObjectData
->privateData
;
509 template <class Base
>
510 bool JSCallbackObject
<Base
>::inherits(JSClassRef c
) const
512 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
)
519 template <class Base
>
520 JSValue JSCallbackObject
<Base
>::getStaticValue(ExecState
* exec
, const Identifier
& propertyName
)
522 JSObjectRef thisRef
= toRef(this);
523 RefPtr
<OpaqueJSString
> propertyNameRef
;
525 for (JSClassRef jsClass
= classRef(); jsClass
; jsClass
= jsClass
->parentClass
)
526 if (OpaqueJSClassStaticValuesTable
* staticValues
= jsClass
->staticValues(exec
))
527 if (StaticValueEntry
* entry
= staticValues
->get(propertyName
.impl()))
528 if (JSObjectGetPropertyCallback getProperty
= entry
->getProperty
) {
529 if (!propertyNameRef
)
530 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
531 JSValueRef exception
= 0;
534 APICallbackShim
callbackShim(exec
);
535 value
= getProperty(toRef(exec
), thisRef
, propertyNameRef
.get(), &exception
);
538 throwError(exec
, toJS(exec
, exception
));
539 return jsUndefined();
542 return toJS(exec
, value
);
548 template <class Base
>
549 JSValue JSCallbackObject
<Base
>::staticFunctionGetter(ExecState
* exec
, JSValue slotBase
, const Identifier
& propertyName
)
551 JSCallbackObject
* thisObj
= asCallbackObject(slotBase
);
553 // Check for cached or override property.
554 PropertySlot
slot2(thisObj
);
555 if (thisObj
->Base::getOwnPropertySlot(exec
, propertyName
, slot2
))
556 return slot2
.getValue(exec
, propertyName
);
558 for (JSClassRef jsClass
= thisObj
->classRef(); jsClass
; jsClass
= jsClass
->parentClass
) {
559 if (OpaqueJSClassStaticFunctionsTable
* staticFunctions
= jsClass
->staticFunctions(exec
)) {
560 if (StaticFunctionEntry
* entry
= staticFunctions
->get(propertyName
.impl())) {
561 if (JSObjectCallAsFunctionCallback callAsFunction
= entry
->callAsFunction
) {
563 JSObject
* o
= new (exec
) JSCallbackFunction(exec
, asGlobalObject(thisObj
->getAnonymousValue(0)), callAsFunction
, propertyName
);
564 thisObj
->putDirect(exec
->globalData(), propertyName
, o
, entry
->attributes
);
571 return throwError(exec
, createReferenceError(exec
, "Static function property defined with NULL callAsFunction callback."));
574 template <class Base
>
575 JSValue JSCallbackObject
<Base
>::callbackGetter(ExecState
* exec
, JSValue slotBase
, const Identifier
& propertyName
)
577 JSCallbackObject
* thisObj
= asCallbackObject(slotBase
);
579 JSObjectRef thisRef
= toRef(thisObj
);
580 RefPtr
<OpaqueJSString
> propertyNameRef
;
582 for (JSClassRef jsClass
= thisObj
->classRef(); jsClass
; jsClass
= jsClass
->parentClass
)
583 if (JSObjectGetPropertyCallback getProperty
= jsClass
->getProperty
) {
584 if (!propertyNameRef
)
585 propertyNameRef
= OpaqueJSString::create(propertyName
.ustring());
586 JSValueRef exception
= 0;
589 APICallbackShim
callbackShim(exec
);
590 value
= getProperty(toRef(exec
), thisRef
, propertyNameRef
.get(), &exception
);
593 throwError(exec
, toJS(exec
, exception
));
594 return jsUndefined();
597 return toJS(exec
, value
);
600 return throwError(exec
, createReferenceError(exec
, "hasProperty callback returned true for a property that doesn't exist."));