]> git.saurik.com Git - apple/javascriptcore.git/blame - API/JSObjectRef.cpp
JavaScriptCore-903.5.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"
4e4e5a6f 29#include "JSObjectRefPrivate.h"
b37bf2e1 30
b37bf2e1 31#include "APICast.h"
f9bf01c6 32#include "CodeBlock.h"
9dae56ea
A
33#include "DateConstructor.h"
34#include "ErrorConstructor.h"
35#include "FunctionConstructor.h"
36#include "Identifier.h"
37#include "InitializeThreading.h"
38#include "JSArray.h"
b37bf2e1
A
39#include "JSCallbackConstructor.h"
40#include "JSCallbackFunction.h"
41#include "JSCallbackObject.h"
42#include "JSClassRef.h"
9dae56ea 43#include "JSFunction.h"
b37bf2e1 44#include "JSGlobalObject.h"
9dae56ea
A
45#include "JSObject.h"
46#include "JSRetainPtr.h"
47#include "JSString.h"
48#include "JSValueRef.h"
49#include "ObjectPrototype.h"
b37bf2e1 50#include "PropertyNameArray.h"
9dae56ea 51#include "RegExpConstructor.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
14957cd0 62 return jsClass.release().leakRef();
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)
14957cd0 82 return toRef(constructEmptyObject(exec));
b37bf2e1 83
14957cd0 84 JSCallbackObject<JSObjectWithGlobalObject>* object = new (exec) JSCallbackObject<JSObjectWithGlobalObject>(exec, exec->lexicalGlobalObject(), exec->lexicalGlobalObject()->callbackObjectStructure(), jsClass, data);
9dae56ea 85 if (JSObject* prototype = jsClass->prototype(exec))
14957cd0 86 object->setPrototype(exec->globalData(), 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
14957cd0 98 return toRef(new (exec) JSCallbackFunction(exec, exec->lexicalGlobalObject(), 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
14957cd0
A
110 JSCallbackConstructor* constructor = new (exec) JSCallbackConstructor(exec->lexicalGlobalObject(), exec->lexicalGlobalObject()->callbackConstructorStructure(), jsClass, callAsConstructor);
111 constructor->putDirect(exec->globalData(), exec->propertyNames().prototype, jsPrototype, DontEnum | DontDelete | ReadOnly);
b37bf2e1
A
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
14957cd0 127 JSObject* result = constructFunction(exec, exec->lexicalGlobalObject(), args, nameID, sourceURL->ustring(), startingLineNumber);
9dae56ea
A
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 170
14957cd0 171 JSObject* result = constructDate(exec, exec->lexicalGlobalObject(), argList);
9dae56ea
A
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
14957cd0
A
187 JSValue message = argumentCount ? toJS(exec, arguments[0]) : jsUndefined();
188 Structure* errorStructure = exec->lexicalGlobalObject()->errorStructure();
189 JSObject* result = ErrorInstance::create(exec, errorStructure, message);
9dae56ea 190
9dae56ea
A
191 if (exec->hadException()) {
192 if (exception)
ba379fdc 193 *exception = toRef(exec, exec->exception());
9dae56ea
A
194 exec->clearException();
195 result = 0;
196 }
197
198 return toRef(result);
199}
200
201JSObjectRef JSObjectMakeRegExp(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
202{
203 ExecState* exec = toJS(ctx);
f9bf01c6 204 APIEntryShim entryShim(exec);
9dae56ea 205
ba379fdc 206 MarkedArgumentBuffer argList;
9dae56ea 207 for (size_t i = 0; i < argumentCount; ++i)
ba379fdc 208 argList.append(toJS(exec, arguments[i]));
9dae56ea 209
14957cd0 210 JSObject* result = constructRegExp(exec, exec->lexicalGlobalObject(), argList);
9dae56ea
A
211 if (exec->hadException()) {
212 if (exception)
ba379fdc 213 *exception = toRef(exec, exec->exception());
9dae56ea
A
214 exec->clearException();
215 result = 0;
216 }
217
b37bf2e1
A
218 return toRef(result);
219}
220
ba379fdc 221JSValueRef JSObjectGetPrototype(JSContextRef ctx, JSObjectRef object)
b37bf2e1 222{
ba379fdc 223 ExecState* exec = toJS(ctx);
f9bf01c6 224 APIEntryShim entryShim(exec);
ba379fdc 225
b37bf2e1 226 JSObject* jsObject = toJS(object);
ba379fdc 227 return toRef(exec, jsObject->prototype());
b37bf2e1
A
228}
229
ba379fdc 230void JSObjectSetPrototype(JSContextRef ctx, JSObjectRef object, JSValueRef value)
b37bf2e1 231{
ba379fdc 232 ExecState* exec = toJS(ctx);
f9bf01c6 233 APIEntryShim entryShim(exec);
ba379fdc 234
b37bf2e1 235 JSObject* jsObject = toJS(object);
ba379fdc 236 JSValue jsValue = toJS(exec, value);
b37bf2e1 237
14957cd0 238 jsObject->setPrototypeWithCycleCheck(exec->globalData(), jsValue.isObject() ? jsValue : jsNull());
b37bf2e1
A
239}
240
241bool JSObjectHasProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName)
242{
b37bf2e1 243 ExecState* exec = toJS(ctx);
f9bf01c6 244 APIEntryShim entryShim(exec);
9dae56ea 245
b37bf2e1 246 JSObject* jsObject = toJS(object);
b37bf2e1 247
9dae56ea 248 return jsObject->hasProperty(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
249}
250
251JSValueRef JSObjectGetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception)
252{
b37bf2e1 253 ExecState* exec = toJS(ctx);
f9bf01c6 254 APIEntryShim entryShim(exec);
9dae56ea 255
b37bf2e1 256 JSObject* jsObject = toJS(object);
b37bf2e1 257
ba379fdc 258 JSValue jsValue = jsObject->get(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
259 if (exec->hadException()) {
260 if (exception)
ba379fdc 261 *exception = toRef(exec, exec->exception());
b37bf2e1
A
262 exec->clearException();
263 }
ba379fdc 264 return toRef(exec, jsValue);
b37bf2e1
A
265}
266
267void JSObjectSetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef value, JSPropertyAttributes attributes, JSValueRef* exception)
268{
b37bf2e1 269 ExecState* exec = toJS(ctx);
f9bf01c6 270 APIEntryShim entryShim(exec);
9dae56ea 271
b37bf2e1 272 JSObject* jsObject = toJS(object);
9dae56ea 273 Identifier name(propertyName->identifier(&exec->globalData()));
ba379fdc 274 JSValue jsValue = toJS(exec, value);
9dae56ea
A
275
276 if (attributes && !jsObject->hasProperty(exec, name))
277 jsObject->putWithAttributes(exec, name, jsValue, attributes);
278 else {
279 PutPropertySlot slot;
280 jsObject->put(exec, name, jsValue, slot);
281 }
282
b37bf2e1
A
283 if (exec->hadException()) {
284 if (exception)
ba379fdc 285 *exception = toRef(exec, exec->exception());
b37bf2e1
A
286 exec->clearException();
287 }
288}
289
290JSValueRef JSObjectGetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef* exception)
291{
b37bf2e1 292 ExecState* exec = toJS(ctx);
f9bf01c6 293 APIEntryShim entryShim(exec);
9dae56ea 294
b37bf2e1
A
295 JSObject* jsObject = toJS(object);
296
ba379fdc 297 JSValue jsValue = jsObject->get(exec, propertyIndex);
b37bf2e1
A
298 if (exec->hadException()) {
299 if (exception)
ba379fdc 300 *exception = toRef(exec, exec->exception());
b37bf2e1
A
301 exec->clearException();
302 }
ba379fdc 303 return toRef(exec, jsValue);
b37bf2e1
A
304}
305
306
307void JSObjectSetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef value, JSValueRef* exception)
308{
b37bf2e1 309 ExecState* exec = toJS(ctx);
f9bf01c6 310 APIEntryShim entryShim(exec);
9dae56ea 311
b37bf2e1 312 JSObject* jsObject = toJS(object);
ba379fdc 313 JSValue jsValue = toJS(exec, value);
b37bf2e1
A
314
315 jsObject->put(exec, propertyIndex, jsValue);
316 if (exec->hadException()) {
317 if (exception)
ba379fdc 318 *exception = toRef(exec, exec->exception());
b37bf2e1
A
319 exec->clearException();
320 }
321}
322
323bool JSObjectDeleteProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception)
324{
b37bf2e1 325 ExecState* exec = toJS(ctx);
f9bf01c6 326 APIEntryShim entryShim(exec);
9dae56ea 327
b37bf2e1 328 JSObject* jsObject = toJS(object);
b37bf2e1 329
9dae56ea 330 bool result = jsObject->deleteProperty(exec, propertyName->identifier(&exec->globalData()));
b37bf2e1
A
331 if (exec->hadException()) {
332 if (exception)
ba379fdc 333 *exception = toRef(exec, exec->exception());
b37bf2e1
A
334 exec->clearException();
335 }
336 return result;
337}
338
339void* JSObjectGetPrivate(JSObjectRef object)
340{
341 JSObject* jsObject = toJS(object);
342
14957cd0 343 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::s_info))
b37bf2e1 344 return static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->getPrivate();
14957cd0
A
345 if (jsObject->inherits(&JSCallbackObject<JSObjectWithGlobalObject>::s_info))
346 return static_cast<JSCallbackObject<JSObjectWithGlobalObject>*>(jsObject)->getPrivate();
b37bf2e1
A
347
348 return 0;
349}
350
351bool JSObjectSetPrivate(JSObjectRef object, void* data)
352{
353 JSObject* jsObject = toJS(object);
354
14957cd0 355 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::s_info)) {
b37bf2e1
A
356 static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->setPrivate(data);
357 return true;
14957cd0
A
358 }
359 if (jsObject->inherits(&JSCallbackObject<JSObjectWithGlobalObject>::s_info)) {
360 static_cast<JSCallbackObject<JSObjectWithGlobalObject>*>(jsObject)->setPrivate(data);
b37bf2e1
A
361 return true;
362 }
363
364 return false;
365}
366
4e4e5a6f
A
367JSValueRef JSObjectGetPrivateProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName)
368{
369 ExecState* exec = toJS(ctx);
370 APIEntryShim entryShim(exec);
371 JSObject* jsObject = toJS(object);
372 JSValue result;
373 Identifier name(propertyName->identifier(&exec->globalData()));
14957cd0 374 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::s_info))
4e4e5a6f 375 result = static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->getPrivateProperty(name);
14957cd0
A
376 else if (jsObject->inherits(&JSCallbackObject<JSObjectWithGlobalObject>::s_info))
377 result = static_cast<JSCallbackObject<JSObjectWithGlobalObject>*>(jsObject)->getPrivateProperty(name);
4e4e5a6f
A
378 return toRef(exec, result);
379}
380
381bool JSObjectSetPrivateProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef value)
382{
383 ExecState* exec = toJS(ctx);
384 APIEntryShim entryShim(exec);
385 JSObject* jsObject = toJS(object);
14957cd0 386 JSValue jsValue = value ? toJS(exec, value) : JSValue();
4e4e5a6f 387 Identifier name(propertyName->identifier(&exec->globalData()));
14957cd0
A
388 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::s_info)) {
389 static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->setPrivateProperty(exec->globalData(), name, jsValue);
4e4e5a6f
A
390 return true;
391 }
14957cd0
A
392 if (jsObject->inherits(&JSCallbackObject<JSObjectWithGlobalObject>::s_info)) {
393 static_cast<JSCallbackObject<JSObjectWithGlobalObject>*>(jsObject)->setPrivateProperty(exec->globalData(), name, jsValue);
4e4e5a6f
A
394 return true;
395 }
396 return false;
397}
398
399bool JSObjectDeletePrivateProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName)
400{
401 ExecState* exec = toJS(ctx);
402 APIEntryShim entryShim(exec);
403 JSObject* jsObject = toJS(object);
404 Identifier name(propertyName->identifier(&exec->globalData()));
14957cd0 405 if (jsObject->inherits(&JSCallbackObject<JSGlobalObject>::s_info)) {
4e4e5a6f
A
406 static_cast<JSCallbackObject<JSGlobalObject>*>(jsObject)->deletePrivateProperty(name);
407 return true;
408 }
14957cd0
A
409 if (jsObject->inherits(&JSCallbackObject<JSObjectWithGlobalObject>::s_info)) {
410 static_cast<JSCallbackObject<JSObjectWithGlobalObject>*>(jsObject)->deletePrivateProperty(name);
4e4e5a6f
A
411 return true;
412 }
413 return false;
414}
415
b37bf2e1
A
416bool JSObjectIsFunction(JSContextRef, JSObjectRef object)
417{
9dae56ea
A
418 CallData callData;
419 return toJS(object)->getCallData(callData) != CallTypeNone;
b37bf2e1
A
420}
421
422JSValueRef JSObjectCallAsFunction(JSContextRef ctx, JSObjectRef object, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
423{
b37bf2e1 424 ExecState* exec = toJS(ctx);
f9bf01c6 425 APIEntryShim entryShim(exec);
9dae56ea 426
b37bf2e1
A
427 JSObject* jsObject = toJS(object);
428 JSObject* jsThisObject = toJS(thisObject);
429
430 if (!jsThisObject)
9dae56ea
A
431 jsThisObject = exec->globalThisValue();
432
ba379fdc 433 MarkedArgumentBuffer argList;
b37bf2e1 434 for (size_t i = 0; i < argumentCount; i++)
ba379fdc 435 argList.append(toJS(exec, arguments[i]));
b37bf2e1 436
9dae56ea
A
437 CallData callData;
438 CallType callType = jsObject->getCallData(callData);
439 if (callType == CallTypeNone)
440 return 0;
441
ba379fdc 442 JSValueRef result = toRef(exec, call(exec, jsObject, callType, callData, jsThisObject, argList));
b37bf2e1
A
443 if (exec->hadException()) {
444 if (exception)
ba379fdc 445 *exception = toRef(exec, exec->exception());
b37bf2e1
A
446 exec->clearException();
447 result = 0;
448 }
449 return result;
450}
451
452bool JSObjectIsConstructor(JSContextRef, JSObjectRef object)
453{
454 JSObject* jsObject = toJS(object);
9dae56ea
A
455 ConstructData constructData;
456 return jsObject->getConstructData(constructData) != ConstructTypeNone;
b37bf2e1
A
457}
458
459JSObjectRef JSObjectCallAsConstructor(JSContextRef ctx, JSObjectRef object, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception)
460{
b37bf2e1 461 ExecState* exec = toJS(ctx);
f9bf01c6 462 APIEntryShim entryShim(exec);
9dae56ea 463
b37bf2e1 464 JSObject* jsObject = toJS(object);
9dae56ea
A
465
466 ConstructData constructData;
467 ConstructType constructType = jsObject->getConstructData(constructData);
468 if (constructType == ConstructTypeNone)
469 return 0;
470
ba379fdc 471 MarkedArgumentBuffer argList;
b37bf2e1 472 for (size_t i = 0; i < argumentCount; i++)
ba379fdc 473 argList.append(toJS(exec, arguments[i]));
9dae56ea 474 JSObjectRef result = toRef(construct(exec, jsObject, constructType, constructData, argList));
b37bf2e1
A
475 if (exec->hadException()) {
476 if (exception)
ba379fdc 477 *exception = toRef(exec, exec->exception());
b37bf2e1
A
478 exec->clearException();
479 result = 0;
480 }
481 return result;
482}
483
14957cd0
A
484struct OpaqueJSPropertyNameArray {
485 WTF_MAKE_FAST_ALLOCATED;
486public:
9dae56ea
A
487 OpaqueJSPropertyNameArray(JSGlobalData* globalData)
488 : refCount(0)
489 , globalData(globalData)
b37bf2e1
A
490 {
491 }
492
493 unsigned refCount;
9dae56ea
A
494 JSGlobalData* globalData;
495 Vector<JSRetainPtr<JSStringRef> > array;
b37bf2e1
A
496};
497
498JSPropertyNameArrayRef JSObjectCopyPropertyNames(JSContextRef ctx, JSObjectRef object)
499{
b37bf2e1
A
500 JSObject* jsObject = toJS(object);
501 ExecState* exec = toJS(ctx);
f9bf01c6 502 APIEntryShim entryShim(exec);
9dae56ea
A
503
504 JSGlobalData* globalData = &exec->globalData();
505
506 JSPropertyNameArrayRef propertyNames = new OpaqueJSPropertyNameArray(globalData);
507 PropertyNameArray array(globalData);
508 jsObject->getPropertyNames(exec, array);
509
510 size_t size = array.size();
511 propertyNames->array.reserveInitialCapacity(size);
512 for (size_t i = 0; i < size; ++i)
14957cd0 513 propertyNames->array.append(JSRetainPtr<JSStringRef>(Adopt, OpaqueJSString::create(array[i].ustring()).leakRef()));
b37bf2e1
A
514
515 return JSPropertyNameArrayRetain(propertyNames);
516}
517
518JSPropertyNameArrayRef JSPropertyNameArrayRetain(JSPropertyNameArrayRef array)
519{
b37bf2e1
A
520 ++array->refCount;
521 return array;
522}
523
524void JSPropertyNameArrayRelease(JSPropertyNameArrayRef array)
525{
9dae56ea 526 if (--array->refCount == 0) {
f9bf01c6 527 APIEntryShim entryShim(array->globalData, false);
b37bf2e1 528 delete array;
9dae56ea 529 }
b37bf2e1
A
530}
531
532size_t JSPropertyNameArrayGetCount(JSPropertyNameArrayRef array)
533{
534 return array->array.size();
535}
536
537JSStringRef JSPropertyNameArrayGetNameAtIndex(JSPropertyNameArrayRef array, size_t index)
538{
9dae56ea 539 return array->array[static_cast<unsigned>(index)].get();
b37bf2e1
A
540}
541
542void JSPropertyNameAccumulatorAddName(JSPropertyNameAccumulatorRef array, JSStringRef propertyName)
543{
b37bf2e1 544 PropertyNameArray* propertyNames = toJS(array);
f9bf01c6 545 APIEntryShim entryShim(propertyNames->globalData());
9dae56ea 546 propertyNames->add(propertyName->identifier(propertyNames->globalData()));
b37bf2e1 547}