]> git.saurik.com Git - apple/javascriptcore.git/blame - API/JSObjectRef.cpp
JavaScriptCore-584.tar.gz
[apple/javascriptcore.git] / API / JSObjectRef.cpp
CommitLineData
b37bf2e1 1/*
9dae56ea
A
2 * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
3 * Copyright (C) 2008 Kelvin W Sherlock (ksherlock@gmail.com)
b37bf2e1
A
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
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.
13 *
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.
25 */
26
27#include "config.h"
28#include "JSObjectRef.h"
29
b37bf2e1 30#include "APICast.h"
f9bf01c6 31#include "CodeBlock.h"
9dae56ea
A
32#include "DateConstructor.h"
33#include "ErrorConstructor.h"
34#include "FunctionConstructor.h"
35#include "Identifier.h"
36#include "InitializeThreading.h"
37#include "JSArray.h"
b37bf2e1
A
38#include "JSCallbackConstructor.h"
39#include "JSCallbackFunction.h"
40#include "JSCallbackObject.h"
41#include "JSClassRef.h"
9dae56ea 42#include "JSFunction.h"
b37bf2e1 43#include "JSGlobalObject.h"
9dae56ea
A
44#include "JSObject.h"
45#include "JSRetainPtr.h"
46#include "JSString.h"
47#include "JSValueRef.h"
48#include "ObjectPrototype.h"
b37bf2e1 49#include "PropertyNameArray.h"
9dae56ea
A
50#include "RegExpConstructor.h"
51#include <wtf/Platform.h>
b37bf2e1 52
9dae56ea 53using namespace JSC;
b37bf2e1
A
54
55JSClassRef JSClassCreate(const JSClassDefinition* definition)
56{
9dae56ea
A
57 initializeThreading();
58 RefPtr<OpaqueJSClass> jsClass = (definition->attributes & kJSClassAttributeNoAutomaticPrototype)
b37bf2e1
A
59 ? OpaqueJSClass::createNoAutomaticPrototype(definition)
60 : OpaqueJSClass::create(definition);
61
9dae56ea 62 return jsClass.release().releaseRef();
b37bf2e1
A
63}
64
65JSClassRef JSClassRetain(JSClassRef jsClass)
66{
b37bf2e1
A
67 jsClass->ref();
68 return jsClass;
69}
70
71void JSClassRelease(JSClassRef jsClass)
72{
b37bf2e1
A
73 jsClass->deref();
74}
75
76JSObjectRef JSObjectMake(JSContextRef ctx, JSClassRef jsClass, void* data)
77{
b37bf2e1 78 ExecState* exec = toJS(ctx);
f9bf01c6 79 APIEntryShim entryShim(exec);
b37bf2e1
A
80
81 if (!jsClass)
9dae56ea 82 return toRef(new (exec) JSObject(exec->lexicalGlobalObject()->emptyObjectStructure())); // slightly more efficient
b37bf2e1 83
9dae56ea
A
84 JSCallbackObject<JSObject>* object = new (exec) JSCallbackObject<JSObject>(exec, exec->lexicalGlobalObject()->callbackObjectStructure(), jsClass, data);
85 if (JSObject* prototype = jsClass->prototype(exec))
86 object->setPrototype(prototype);
b37bf2e1 87
9dae56ea 88 return toRef(object);
b37bf2e1
A
89}
90
91JSObjectRef JSObjectMakeFunctionWithCallback(JSContextRef ctx, JSStringRef name, JSObjectCallAsFunctionCallback callAsFunction)
92{
b37bf2e1 93 ExecState* exec = toJS(ctx);
f9bf01c6 94 APIEntryShim entryShim(exec);
9dae56ea
A
95
96 Identifier nameID = name ? name->identifier(&exec->globalData()) : Identifier(exec, "anonymous");
b37bf2e1 97
9dae56ea 98 return toRef(new (exec) JSCallbackFunction(exec, callAsFunction, nameID));
b37bf2e1
A
99}
100
101JSObjectRef JSObjectMakeConstructor(JSContextRef ctx, JSClassRef jsClass, JSObjectCallAsConstructorCallback callAsConstructor)
102{
b37bf2e1 103 ExecState* exec = toJS(ctx);
f9bf01c6 104 APIEntryShim entryShim(exec);
9dae56ea 105
ba379fdc
A
106 JSValue jsPrototype = jsClass ? jsClass->prototype(exec) : 0;
107 if (!jsPrototype)
108 jsPrototype = exec->lexicalGlobalObject()->objectPrototype();
109
9dae56ea 110 JSCallbackConstructor* constructor = new (exec) JSCallbackConstructor(exec->lexicalGlobalObject()->callbackConstructorStructure(), jsClass, callAsConstructor);
b37bf2e1
A
111 constructor->putDirect(exec->propertyNames().prototype, jsPrototype, DontEnum | DontDelete | ReadOnly);
112 return toRef(constructor);
113}
114
115JSObjectRef JSObjectMakeFunction(JSContextRef ctx, JSStringRef name, unsigned parameterCount, const JSStringRef parameterNames[], JSStringRef body, JSStringRef sourceURL, int startingLineNumber, JSValueRef* exception)
116{
b37bf2e1 117 ExecState* exec = toJS(ctx);
f9bf01c6 118 APIEntryShim entryShim(exec);
9dae56ea
A
119
120 Identifier nameID = name ? name->identifier(&exec->globalData()) : Identifier(exec, "anonymous");
b37bf2e1 121
ba379fdc 122 MarkedArgumentBuffer args;
b37bf2e1 123 for (unsigned i = 0; i < parameterCount; i++)
9dae56ea
A
124 args.append(jsString(exec, parameterNames[i]->ustring()));
125 args.append(jsString(exec, body->ustring()));
126
127 JSObject* result = constructFunction(exec, args, nameID, sourceURL->ustring(), startingLineNumber);
128 if (exec->hadException()) {
129 if (exception)
ba379fdc 130 *exception = toRef(exec, exec->exception());
9dae56ea
A
131 exec->clearException();
132 result = 0;
133 }
134 return toRef(result);
135}
136
137JSObjectRef JSObjectMakeArray(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
138{
139 ExecState* exec = toJS(ctx);
f9bf01c6 140 APIEntryShim entryShim(exec);
9dae56ea
A
141
142 JSObject* result;
143 if (argumentCount) {
ba379fdc 144 MarkedArgumentBuffer argList;
9dae56ea 145 for (size_t i = 0; i < argumentCount; ++i)
ba379fdc 146 argList.append(toJS(exec, arguments[i]));
9dae56ea
A
147
148 result = constructArray(exec, argList);
149 } else
150 result = constructEmptyArray(exec);
b37bf2e1 151
b37bf2e1
A
152 if (exec->hadException()) {
153 if (exception)
ba379fdc 154 *exception = toRef(exec, exec->exception());
b37bf2e1
A
155 exec->clearException();
156 result = 0;
157 }
9dae56ea
A
158
159 return toRef(result);
160}
161
162JSObjectRef JSObjectMakeDate(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
163{
164 ExecState* exec = toJS(ctx);
f9bf01c6 165 APIEntryShim entryShim(exec);
9dae56ea 166
ba379fdc 167 MarkedArgumentBuffer argList;
9dae56ea 168 for (size_t i = 0; i < argumentCount; ++i)
ba379fdc 169 argList.append(toJS(exec, arguments[i]));
9dae56ea
A
170
171 JSObject* result = constructDate(exec, argList);
172 if (exec->hadException()) {
173 if (exception)
ba379fdc 174 *exception = toRef(exec, exec->exception());
9dae56ea
A
175 exec->clearException();
176 result = 0;
177 }
178
179 return toRef(result);
180}
181
182JSObjectRef JSObjectMakeError(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
183{
184 ExecState* exec = toJS(ctx);
f9bf01c6 185 APIEntryShim entryShim(exec);
9dae56ea 186
ba379fdc 187 MarkedArgumentBuffer argList;
9dae56ea 188 for (size_t i = 0; i < argumentCount; ++i)
ba379fdc 189 argList.append(toJS(exec, arguments[i]));
9dae56ea
A
190
191 JSObject* result = constructError(exec, argList);
192 if (exec->hadException()) {
193 if (exception)
ba379fdc 194 *exception = toRef(exec, exec->exception());
9dae56ea
A
195 exec->clearException();
196 result = 0;
197 }
198
199 return toRef(result);
200}
201
202JSObjectRef JSObjectMakeRegExp(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
203{
204 ExecState* exec = toJS(ctx);
f9bf01c6 205 APIEntryShim entryShim(exec);
9dae56ea 206
ba379fdc 207 MarkedArgumentBuffer argList;
9dae56ea 208 for (size_t i = 0; i < argumentCount; ++i)
ba379fdc 209 argList.append(toJS(exec, arguments[i]));
9dae56ea
A
210
211 JSObject* result = constructRegExp(exec, argList);
212 if (exec->hadException()) {
213 if (exception)
ba379fdc 214 *exception = toRef(exec, exec->exception());
9dae56ea
A
215 exec->clearException();
216 result = 0;
217 }
218
b37bf2e1
A
219 return toRef(result);
220}
221
ba379fdc 222JSValueRef JSObjectGetPrototype(JSContextRef ctx, JSObjectRef object)
b37bf2e1 223{
ba379fdc 224 ExecState* exec = toJS(ctx);
f9bf01c6 225 APIEntryShim entryShim(exec);
ba379fdc 226
b37bf2e1 227 JSObject* jsObject = toJS(object);
ba379fdc 228 return toRef(exec, jsObject->prototype());
b37bf2e1
A
229}
230
ba379fdc 231void JSObjectSetPrototype(JSContextRef ctx, JSObjectRef object, JSValueRef value)
b37bf2e1 232{
ba379fdc 233 ExecState* exec = toJS(ctx);
f9bf01c6 234 APIEntryShim entryShim(exec);
ba379fdc 235
b37bf2e1 236 JSObject* jsObject = toJS(object);
ba379fdc 237 JSValue jsValue = toJS(exec, value);
b37bf2e1 238
9dae56ea 239 jsObject->setPrototype(jsValue.isObject() ? jsValue : jsNull());
b37bf2e1
A
240}
241
242bool JSObjectHasProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName)
243{
b37bf2e1 244 ExecState* exec = toJS(ctx);
f9bf01c6 245 APIEntryShim entryShim(exec);
9dae56ea 246
b37bf2e1 247 JSObject* jsObject = toJS(object);
b37bf2e1 248
9dae56ea 249 return jsObject->hasProperty(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
250}
251
252JSValueRef JSObjectGetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception)
253{
b37bf2e1 254 ExecState* exec = toJS(ctx);
f9bf01c6 255 APIEntryShim entryShim(exec);
9dae56ea 256
b37bf2e1 257 JSObject* jsObject = toJS(object);
b37bf2e1 258
ba379fdc 259 JSValue jsValue = jsObject->get(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
260 if (exec->hadException()) {
261 if (exception)
ba379fdc 262 *exception = toRef(exec, exec->exception());
b37bf2e1
A
263 exec->clearException();
264 }
ba379fdc 265 return toRef(exec, jsValue);
b37bf2e1
A
266}
267
268void JSObjectSetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef value, JSPropertyAttributes attributes, JSValueRef* exception)
269{
b37bf2e1 270 ExecState* exec = toJS(ctx);
f9bf01c6 271 APIEntryShim entryShim(exec);
9dae56ea 272
b37bf2e1 273 JSObject* jsObject = toJS(object);
9dae56ea 274 Identifier name(propertyName->identifier(&exec->globalData()));
ba379fdc 275 JSValue jsValue = toJS(exec, value);
9dae56ea
A
276
277 if (attributes && !jsObject->hasProperty(exec, name))
278 jsObject->putWithAttributes(exec, name, jsValue, attributes);
279 else {
280 PutPropertySlot slot;
281 jsObject->put(exec, name, jsValue, slot);
282 }
283
b37bf2e1
A
284 if (exec->hadException()) {
285 if (exception)
ba379fdc 286 *exception = toRef(exec, exec->exception());
b37bf2e1
A
287 exec->clearException();
288 }
289}
290
291JSValueRef JSObjectGetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef* exception)
292{
b37bf2e1 293 ExecState* exec = toJS(ctx);
f9bf01c6 294 APIEntryShim entryShim(exec);
9dae56ea 295
b37bf2e1
A
296 JSObject* jsObject = toJS(object);
297
ba379fdc 298 JSValue jsValue = jsObject->get(exec, propertyIndex);
b37bf2e1
A
299 if (exec->hadException()) {
300 if (exception)
ba379fdc 301 *exception = toRef(exec, exec->exception());
b37bf2e1
A
302 exec->clearException();
303 }
ba379fdc 304 return toRef(exec, jsValue);
b37bf2e1
A
305}
306
307
308void JSObjectSetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef value, JSValueRef* exception)
309{
b37bf2e1 310 ExecState* exec = toJS(ctx);
f9bf01c6 311 APIEntryShim entryShim(exec);
9dae56ea 312
b37bf2e1 313 JSObject* jsObject = toJS(object);
ba379fdc 314 JSValue jsValue = toJS(exec, value);
b37bf2e1
A
315
316 jsObject->put(exec, propertyIndex, jsValue);
317 if (exec->hadException()) {
318 if (exception)
ba379fdc 319 *exception = toRef(exec, exec->exception());
b37bf2e1
A
320 exec->clearException();
321 }
322}
323
324bool JSObjectDeleteProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception)
325{
b37bf2e1 326 ExecState* exec = toJS(ctx);
f9bf01c6 327 APIEntryShim entryShim(exec);
9dae56ea 328
b37bf2e1 329 JSObject* jsObject = toJS(object);
b37bf2e1 330
9dae56ea 331 bool result = jsObject->deleteProperty(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
332 if (exec->hadException()) {
333 if (exception)
ba379fdc 334 *exception = toRef(exec, exec->exception());
b37bf2e1
A
335 exec->clearException();
336 }
337 return result;
338}
339
340void* JSObjectGetPrivate(JSObjectRef object)
341{
342 JSObject* jsObject = toJS(object);
343
344 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::info))
345 return static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->getPrivate();
346 else if (jsObject->inherits(&JSCallbackObject<JSObject>::info))
347 return static_cast<JSCallbackObject<JSObject>*>(jsObject)->getPrivate();
348
349 return 0;
350}
351
352bool JSObjectSetPrivate(JSObjectRef object, void* data)
353{
354 JSObject* jsObject = toJS(object);
355
356 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::info)) {
357 static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->setPrivate(data);
358 return true;
359 } else if (jsObject->inherits(&JSCallbackObject<JSObject>::info)) {
360 static_cast<JSCallbackObject<JSObject>*>(jsObject)->setPrivate(data);
361 return true;
362 }
363
364 return false;
365}
366
367bool JSObjectIsFunction(JSContextRef, JSObjectRef object)
368{
9dae56ea
A
369 CallData callData;
370 return toJS(object)->getCallData(callData) != CallTypeNone;
b37bf2e1
A
371}
372
373JSValueRef JSObjectCallAsFunction(JSContextRef ctx, JSObjectRef object, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
374{
b37bf2e1 375 ExecState* exec = toJS(ctx);
f9bf01c6 376 APIEntryShim entryShim(exec);
9dae56ea 377
b37bf2e1
A
378 JSObject* jsObject = toJS(object);
379 JSObject* jsThisObject = toJS(thisObject);
380
381 if (!jsThisObject)
9dae56ea
A
382 jsThisObject = exec->globalThisValue();
383
ba379fdc 384 MarkedArgumentBuffer argList;
b37bf2e1 385 for (size_t i = 0; i < argumentCount; i++)
ba379fdc 386 argList.append(toJS(exec, arguments[i]));
b37bf2e1 387
9dae56ea
A
388 CallData callData;
389 CallType callType = jsObject->getCallData(callData);
390 if (callType == CallTypeNone)
391 return 0;
392
ba379fdc 393 JSValueRef result = toRef(exec, call(exec, jsObject, callType, callData, jsThisObject, argList));
b37bf2e1
A
394 if (exec->hadException()) {
395 if (exception)
ba379fdc 396 *exception = toRef(exec, exec->exception());
b37bf2e1
A
397 exec->clearException();
398 result = 0;
399 }
400 return result;
401}
402
403bool JSObjectIsConstructor(JSContextRef, JSObjectRef object)
404{
405 JSObject* jsObject = toJS(object);
9dae56ea
A
406 ConstructData constructData;
407 return jsObject->getConstructData(constructData) != ConstructTypeNone;
b37bf2e1
A
408}
409
410JSObjectRef JSObjectCallAsConstructor(JSContextRef ctx, JSObjectRef object, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
411{
b37bf2e1 412 ExecState* exec = toJS(ctx);
f9bf01c6 413 APIEntryShim entryShim(exec);
9dae56ea 414
b37bf2e1 415 JSObject* jsObject = toJS(object);
9dae56ea
A
416
417 ConstructData constructData;
418 ConstructType constructType = jsObject->getConstructData(constructData);
419 if (constructType == ConstructTypeNone)
420 return 0;
421
ba379fdc 422 MarkedArgumentBuffer argList;
b37bf2e1 423 for (size_t i = 0; i < argumentCount; i++)
ba379fdc 424 argList.append(toJS(exec, arguments[i]));
9dae56ea 425 JSObjectRef result = toRef(construct(exec, jsObject, constructType, constructData, argList));
b37bf2e1
A
426 if (exec->hadException()) {
427 if (exception)
ba379fdc 428 *exception = toRef(exec, exec->exception());
b37bf2e1
A
429 exec->clearException();
430 result = 0;
431 }
432 return result;
433}
434
f9bf01c6 435struct OpaqueJSPropertyNameArray : FastAllocBase {
9dae56ea
A
436 OpaqueJSPropertyNameArray(JSGlobalData* globalData)
437 : refCount(0)
438 , globalData(globalData)
b37bf2e1
A
439 {
440 }
441
442 unsigned refCount;
9dae56ea
A
443 JSGlobalData* globalData;
444 Vector<JSRetainPtr<JSStringRef> > array;
b37bf2e1
A
445};
446
447JSPropertyNameArrayRef JSObjectCopyPropertyNames(JSContextRef ctx, JSObjectRef object)
448{
b37bf2e1
A
449 JSObject* jsObject = toJS(object);
450 ExecState* exec = toJS(ctx);
f9bf01c6 451 APIEntryShim entryShim(exec);
9dae56ea
A
452
453 JSGlobalData* globalData = &exec->globalData();
454
455 JSPropertyNameArrayRef propertyNames = new OpaqueJSPropertyNameArray(globalData);
456 PropertyNameArray array(globalData);
457 jsObject->getPropertyNames(exec, array);
458
459 size_t size = array.size();
460 propertyNames->array.reserveInitialCapacity(size);
461 for (size_t i = 0; i < size; ++i)
462 propertyNames->array.append(JSRetainPtr<JSStringRef>(Adopt, OpaqueJSString::create(array[i].ustring()).releaseRef()));
b37bf2e1
A
463
464 return JSPropertyNameArrayRetain(propertyNames);
465}
466
467JSPropertyNameArrayRef JSPropertyNameArrayRetain(JSPropertyNameArrayRef array)
468{
b37bf2e1
A
469 ++array->refCount;
470 return array;
471}
472
473void JSPropertyNameArrayRelease(JSPropertyNameArrayRef array)
474{
9dae56ea 475 if (--array->refCount == 0) {
f9bf01c6 476 APIEntryShim entryShim(array->globalData, false);
b37bf2e1 477 delete array;
9dae56ea 478 }
b37bf2e1
A
479}
480
481size_t JSPropertyNameArrayGetCount(JSPropertyNameArrayRef array)
482{
483 return array->array.size();
484}
485
486JSStringRef JSPropertyNameArrayGetNameAtIndex(JSPropertyNameArrayRef array, size_t index)
487{
9dae56ea 488 return array->array[static_cast<unsigned>(index)].get();
b37bf2e1
A
489}
490
491void JSPropertyNameAccumulatorAddName(JSPropertyNameAccumulatorRef array, JSStringRef propertyName)
492{
b37bf2e1 493 PropertyNameArray* propertyNames = toJS(array);
f9bf01c6 494 APIEntryShim entryShim(propertyNames->globalData());
9dae56ea 495 propertyNames->add(propertyName->identifier(propertyNames->globalData()));
b37bf2e1 496}