X-Git-Url: https://git.saurik.com/apple/javascriptcore.git/blobdiff_plain/b5422865f473faf3977f31b96a635c4c8c4ede09..9dae56ea45a0f5f8136a5c93d6f3a7f99399ca73:/API/JSObjectRef.cpp diff --git a/API/JSObjectRef.cpp b/API/JSObjectRef.cpp index c6cda25..e81e512 100644 --- a/API/JSObjectRef.cpp +++ b/API/JSObjectRef.cpp @@ -1,6 +1,6 @@ -// -*- mode: c++; c-basic-offset: 4 -*- /* - * Copyright (C) 2006, 2007 Apple Inc. All rights reserved. + * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved. + * Copyright (C) 2008 Kelvin W Sherlock (ksherlock@gmail.com) * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions @@ -27,108 +27,202 @@ #include "config.h" #include "JSObjectRef.h" -#include #include "APICast.h" -#include "JSValueRef.h" +#include "DateConstructor.h" +#include "ErrorConstructor.h" +#include "FunctionConstructor.h" +#include "Identifier.h" +#include "InitializeThreading.h" +#include "JSArray.h" #include "JSCallbackConstructor.h" #include "JSCallbackFunction.h" #include "JSCallbackObject.h" #include "JSClassRef.h" +#include "JSFunction.h" #include "JSGlobalObject.h" - +#include "JSObject.h" +#include "JSRetainPtr.h" +#include "JSString.h" +#include "JSValueRef.h" +#include "ObjectPrototype.h" #include "PropertyNameArray.h" -#include "function.h" -#include "function_object.h" -#include "identifier.h" -#include "internal.h" -#include "object.h" -#include "object_object.h" +#include "RegExpConstructor.h" +#include -using namespace KJS; +using namespace JSC; JSClassRef JSClassCreate(const JSClassDefinition* definition) { - JSLock lock; - JSClassRef jsClass = (definition->attributes & kJSClassAttributeNoAutomaticPrototype) + initializeThreading(); + RefPtr jsClass = (definition->attributes & kJSClassAttributeNoAutomaticPrototype) ? OpaqueJSClass::createNoAutomaticPrototype(definition) : OpaqueJSClass::create(definition); - return JSClassRetain(jsClass); + return jsClass.release().releaseRef(); } JSClassRef JSClassRetain(JSClassRef jsClass) { - JSLock lock; jsClass->ref(); return jsClass; } void JSClassRelease(JSClassRef jsClass) { - JSLock lock; jsClass->deref(); } JSObjectRef JSObjectMake(JSContextRef ctx, JSClassRef jsClass, void* data) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); if (!jsClass) - return toRef(new JSObject(exec->lexicalGlobalObject()->objectPrototype())); // slightly more efficient + return toRef(new (exec) JSObject(exec->lexicalGlobalObject()->emptyObjectStructure())); // slightly more efficient - JSValue* jsPrototype = jsClass->prototype(ctx); - if (!jsPrototype) - jsPrototype = exec->lexicalGlobalObject()->objectPrototype(); + JSCallbackObject* object = new (exec) JSCallbackObject(exec, exec->lexicalGlobalObject()->callbackObjectStructure(), jsClass, data); + if (JSObject* prototype = jsClass->prototype(exec)) + object->setPrototype(prototype); - return toRef(new JSCallbackObject(exec, jsClass, jsPrototype, data)); + return toRef(object); } JSObjectRef JSObjectMakeFunctionWithCallback(JSContextRef ctx, JSStringRef name, JSObjectCallAsFunctionCallback callAsFunction) { - JSLock lock; ExecState* exec = toJS(ctx); - Identifier nameID = name ? Identifier(toJS(name)) : Identifier("anonymous"); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + Identifier nameID = name ? name->identifier(&exec->globalData()) : Identifier(exec, "anonymous"); - return toRef(new JSCallbackFunction(exec, callAsFunction, nameID)); + return toRef(new (exec) JSCallbackFunction(exec, callAsFunction, nameID)); } JSObjectRef JSObjectMakeConstructor(JSContextRef ctx, JSClassRef jsClass, JSObjectCallAsConstructorCallback callAsConstructor) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + JSValuePtr jsPrototype = jsClass + ? jsClass->prototype(exec) + : exec->lexicalGlobalObject()->objectPrototype(); - JSValue* jsPrototype = jsClass - ? jsClass->prototype(ctx) - : exec->dynamicGlobalObject()->objectPrototype(); - - JSCallbackConstructor* constructor = new JSCallbackConstructor(exec, jsClass, callAsConstructor); + JSCallbackConstructor* constructor = new (exec) JSCallbackConstructor(exec->lexicalGlobalObject()->callbackConstructorStructure(), jsClass, callAsConstructor); constructor->putDirect(exec->propertyNames().prototype, jsPrototype, DontEnum | DontDelete | ReadOnly); return toRef(constructor); } JSObjectRef JSObjectMakeFunction(JSContextRef ctx, JSStringRef name, unsigned parameterCount, const JSStringRef parameterNames[], JSStringRef body, JSStringRef sourceURL, int startingLineNumber, JSValueRef* exception) { - JSLock lock; - ExecState* exec = toJS(ctx); - UString::Rep* bodyRep = toJS(body); - UString::Rep* sourceURLRep = sourceURL ? toJS(sourceURL) : &UString::Rep::null; - - Identifier nameID = name ? Identifier(toJS(name)) : Identifier("anonymous"); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + Identifier nameID = name ? name->identifier(&exec->globalData()) : Identifier(exec, "anonymous"); - List args; + ArgList args; for (unsigned i = 0; i < parameterCount; i++) - args.append(jsString(UString(toJS(parameterNames[i])))); - args.append(jsString(UString(bodyRep))); + args.append(jsString(exec, parameterNames[i]->ustring())); + args.append(jsString(exec, body->ustring())); + + JSObject* result = constructFunction(exec, args, nameID, sourceURL->ustring(), startingLineNumber); + if (exec->hadException()) { + if (exception) + *exception = toRef(exec->exception()); + exec->clearException(); + result = 0; + } + return toRef(result); +} + +JSObjectRef JSObjectMakeArray(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) +{ + ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + JSObject* result; + if (argumentCount) { + ArgList argList; + for (size_t i = 0; i < argumentCount; ++i) + argList.append(toJS(arguments[i])); + + result = constructArray(exec, argList); + } else + result = constructEmptyArray(exec); - JSObject* result = exec->dynamicGlobalObject()->functionConstructor()->construct(exec, args, nameID, UString(sourceURLRep), startingLineNumber); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); exec->clearException(); result = 0; } + + return toRef(result); +} + +JSObjectRef JSObjectMakeDate(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) +{ + ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + ArgList argList; + for (size_t i = 0; i < argumentCount; ++i) + argList.append(toJS(arguments[i])); + + JSObject* result = constructDate(exec, argList); + if (exec->hadException()) { + if (exception) + *exception = toRef(exec->exception()); + exec->clearException(); + result = 0; + } + + return toRef(result); +} + +JSObjectRef JSObjectMakeError(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) +{ + ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + ArgList argList; + for (size_t i = 0; i < argumentCount; ++i) + argList.append(toJS(arguments[i])); + + JSObject* result = constructError(exec, argList); + if (exec->hadException()) { + if (exception) + *exception = toRef(exec->exception()); + exec->clearException(); + result = 0; + } + + return toRef(result); +} + +JSObjectRef JSObjectMakeRegExp(JSContextRef ctx, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) +{ + ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + ArgList argList; + for (size_t i = 0; i < argumentCount; ++i) + argList.append(toJS(arguments[i])); + + JSObject* result = constructRegExp(exec, argList); + if (exec->hadException()) { + if (exception) + *exception = toRef(exec->exception()); + exec->clearException(); + result = 0; + } + return toRef(result); } @@ -141,29 +235,31 @@ JSValueRef JSObjectGetPrototype(JSContextRef, JSObjectRef object) void JSObjectSetPrototype(JSContextRef, JSObjectRef object, JSValueRef value) { JSObject* jsObject = toJS(object); - JSValue* jsValue = toJS(value); + JSValuePtr jsValue = toJS(value); - jsObject->setPrototype(jsValue); + jsObject->setPrototype(jsValue.isObject() ? jsValue : jsNull()); } bool JSObjectHasProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - UString::Rep* nameRep = toJS(propertyName); - return jsObject->hasProperty(exec, Identifier(nameRep)); + return jsObject->hasProperty(exec, propertyName->identifier(&exec->globalData())); } JSValueRef JSObjectGetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - UString::Rep* nameRep = toJS(propertyName); - JSValue* jsValue = jsObject->get(exec, Identifier(nameRep)); + JSValuePtr jsValue = jsObject->get(exec, propertyName->identifier(&exec->globalData())); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -174,13 +270,21 @@ JSValueRef JSObjectGetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef void JSObjectSetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef value, JSPropertyAttributes attributes, JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - UString::Rep* nameRep = toJS(propertyName); - JSValue* jsValue = toJS(value); - - jsObject->put(exec, Identifier(nameRep), jsValue, attributes); + Identifier name(propertyName->identifier(&exec->globalData())); + JSValuePtr jsValue = toJS(value); + + if (attributes && !jsObject->hasProperty(exec, name)) + jsObject->putWithAttributes(exec, name, jsValue, attributes); + else { + PutPropertySlot slot; + jsObject->put(exec, name, jsValue, slot); + } + if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -190,11 +294,13 @@ void JSObjectSetProperty(JSContextRef ctx, JSObjectRef object, JSStringRef prope JSValueRef JSObjectGetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - JSValue* jsValue = jsObject->get(exec, propertyIndex); + JSValuePtr jsValue = jsObject->get(exec, propertyIndex); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -206,10 +312,12 @@ JSValueRef JSObjectGetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsi void JSObjectSetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned propertyIndex, JSValueRef value, JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - JSValue* jsValue = toJS(value); + JSValuePtr jsValue = toJS(value); jsObject->put(exec, propertyIndex, jsValue); if (exec->hadException()) { @@ -221,12 +329,13 @@ void JSObjectSetPropertyAtIndex(JSContextRef ctx, JSObjectRef object, unsigned p bool JSObjectDeleteProperty(JSContextRef ctx, JSObjectRef object, JSStringRef propertyName, JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - UString::Rep* nameRep = toJS(propertyName); - bool result = jsObject->deleteProperty(exec, Identifier(nameRep)); + bool result = jsObject->deleteProperty(exec, propertyName->identifier(&exec->globalData())); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -264,25 +373,32 @@ bool JSObjectSetPrivate(JSObjectRef object, void* data) bool JSObjectIsFunction(JSContextRef, JSObjectRef object) { - JSObject* jsObject = toJS(object); - return jsObject->implementsCall(); + CallData callData; + return toJS(object)->getCallData(callData) != CallTypeNone; } JSValueRef JSObjectCallAsFunction(JSContextRef ctx, JSObjectRef object, JSObjectRef thisObject, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); JSObject* jsThisObject = toJS(thisObject); if (!jsThisObject) - jsThisObject = exec->dynamicGlobalObject(); - - List argList; + jsThisObject = exec->globalThisValue(); + + ArgList argList; for (size_t i = 0; i < argumentCount; i++) argList.append(toJS(arguments[i])); - JSValueRef result = toRef(jsObject->call(exec, jsThisObject, argList)); // returns NULL if object->implementsCall() is false + CallData callData; + CallType callType = jsObject->getCallData(callData); + if (callType == CallTypeNone) + return 0; + + JSValueRef result = toRef(call(exec, jsObject, callType, callData, jsThisObject, argList)); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -295,20 +411,27 @@ JSValueRef JSObjectCallAsFunction(JSContextRef ctx, JSObjectRef object, JSObject bool JSObjectIsConstructor(JSContextRef, JSObjectRef object) { JSObject* jsObject = toJS(object); - return jsObject->implementsConstruct(); + ConstructData constructData; + return jsObject->getConstructData(constructData) != ConstructTypeNone; } JSObjectRef JSObjectCallAsConstructor(JSContextRef ctx, JSObjectRef object, size_t argumentCount, const JSValueRef arguments[], JSValueRef* exception) { - JSLock lock; ExecState* exec = toJS(ctx); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + JSObject* jsObject = toJS(object); - - List argList; + + ConstructData constructData; + ConstructType constructType = jsObject->getConstructData(constructData); + if (constructType == ConstructTypeNone) + return 0; + + ArgList argList; for (size_t i = 0; i < argumentCount; i++) argList.append(toJS(arguments[i])); - - JSObjectRef result = toRef(jsObject->construct(exec, argList)); // returns NULL if object->implementsCall() is false + JSObjectRef result = toRef(construct(exec, jsObject, constructType, constructData, argList)); if (exec->hadException()) { if (exception) *exception = toRef(exec->exception()); @@ -318,40 +441,51 @@ JSObjectRef JSObjectCallAsConstructor(JSContextRef ctx, JSObjectRef object, size return result; } -struct OpaqueJSPropertyNameArray -{ - OpaqueJSPropertyNameArray() : refCount(0) +struct OpaqueJSPropertyNameArray { + OpaqueJSPropertyNameArray(JSGlobalData* globalData) + : refCount(0) + , globalData(globalData) { } unsigned refCount; - PropertyNameArray array; + JSGlobalData* globalData; + Vector > array; }; JSPropertyNameArrayRef JSObjectCopyPropertyNames(JSContextRef ctx, JSObjectRef object) { - JSLock lock; JSObject* jsObject = toJS(object); ExecState* exec = toJS(ctx); - - JSPropertyNameArrayRef propertyNames = new OpaqueJSPropertyNameArray(); - jsObject->getPropertyNames(exec, propertyNames->array); + exec->globalData().heap.registerThread(); + JSLock lock(exec); + + JSGlobalData* globalData = &exec->globalData(); + + JSPropertyNameArrayRef propertyNames = new OpaqueJSPropertyNameArray(globalData); + PropertyNameArray array(globalData); + jsObject->getPropertyNames(exec, array); + + size_t size = array.size(); + propertyNames->array.reserveInitialCapacity(size); + for (size_t i = 0; i < size; ++i) + propertyNames->array.append(JSRetainPtr(Adopt, OpaqueJSString::create(array[i].ustring()).releaseRef())); return JSPropertyNameArrayRetain(propertyNames); } JSPropertyNameArrayRef JSPropertyNameArrayRetain(JSPropertyNameArrayRef array) { - JSLock lock; ++array->refCount; return array; } void JSPropertyNameArrayRelease(JSPropertyNameArrayRef array) { - JSLock lock; - if (--array->refCount == 0) + if (--array->refCount == 0) { + JSLock lock(array->globalData->isSharedInstance); delete array; + } } size_t JSPropertyNameArrayGetCount(JSPropertyNameArrayRef array) @@ -361,14 +495,15 @@ size_t JSPropertyNameArrayGetCount(JSPropertyNameArrayRef array) JSStringRef JSPropertyNameArrayGetNameAtIndex(JSPropertyNameArrayRef array, size_t index) { - return toRef(array->array[static_cast(index)].ustring().rep()); + return array->array[static_cast(index)].get(); } void JSPropertyNameAccumulatorAddName(JSPropertyNameAccumulatorRef array, JSStringRef propertyName) { - JSLock lock; PropertyNameArray* propertyNames = toJS(array); - UString::Rep* rep = toJS(propertyName); - - propertyNames->add(Identifier(rep)); + + propertyNames->globalData()->heap.registerThread(); + JSLock lock(propertyNames->globalData()->isSharedInstance); + + propertyNames->add(propertyName->identifier(propertyNames->globalData())); }