summaryrefslogtreecommitdiffstats
path: root/Source/JavaScriptGlue/JavaScriptGlue.cpp
diff options
context:
space:
mode:
authorSteve Block <steveblock@google.com>2011-05-06 11:45:16 +0100
committerSteve Block <steveblock@google.com>2011-05-12 13:44:10 +0100
commitcad810f21b803229eb11403f9209855525a25d57 (patch)
tree29a6fd0279be608e0fe9ffe9841f722f0f4e4269 /Source/JavaScriptGlue/JavaScriptGlue.cpp
parent121b0cf4517156d0ac5111caf9830c51b69bae8f (diff)
downloadexternal_webkit-cad810f21b803229eb11403f9209855525a25d57.zip
external_webkit-cad810f21b803229eb11403f9209855525a25d57.tar.gz
external_webkit-cad810f21b803229eb11403f9209855525a25d57.tar.bz2
Merge WebKit at r75315: Initial merge by git.
Change-Id: I570314b346ce101c935ed22a626b48c2af266b84
Diffstat (limited to 'Source/JavaScriptGlue/JavaScriptGlue.cpp')
-rw-r--r--Source/JavaScriptGlue/JavaScriptGlue.cpp661
1 files changed, 661 insertions, 0 deletions
diff --git a/Source/JavaScriptGlue/JavaScriptGlue.cpp b/Source/JavaScriptGlue/JavaScriptGlue.cpp
new file mode 100644
index 0000000..134d310
--- /dev/null
+++ b/Source/JavaScriptGlue/JavaScriptGlue.cpp
@@ -0,0 +1,661 @@
+/*
+ * Copyright (C) 2005 Apple Computer, Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "JavaScriptGlue.h"
+#include "JSUtils.h"
+#include "JSBase.h"
+#include "JSObject.h"
+#include "JSRun.h"
+#include <JavaScriptCore/Completion.h>
+#include <JavaScriptCore/InitializeThreading.h>
+
+static CFTypeRef sJSCFNullRef = 0;
+
+static void CFJSObjectDispose(void *data);
+static JSObjectRef CFJSObjectCopyProperty(void *data, CFStringRef propertyName);
+static void CFJSObjectSetProperty(void *data, CFStringRef propertyName, JSObjectRef jsValue);
+static CFTypeRef CFJSObjectCopyCFValue(void *data);
+static UInt8 CFJSObjectEqual(void *data1, void *data2);
+static CFArrayRef CFJSObjectCopyPropertyNames(void *data);
+
+void *JSCFRetain(CFAllocatorRef allocator, const void *value);
+void JSCFRelease(CFAllocatorRef allocator, const void *value);
+
+
+void JSSetCFNull(CFTypeRef nullRef)
+{
+ ReleaseCFType(sJSCFNullRef);
+ sJSCFNullRef = RetainCFType(nullRef);
+}
+
+CFTypeRef JSGetCFNull(void)
+{
+ return sJSCFNullRef;
+}
+
+/*
+ JSRetain
+*/
+JSTypeRef JSRetain(JSTypeRef ref)
+{
+ if (ref)
+ {
+ JSBase* ptr = (JSBase*)ref;
+ ptr->Retain();
+ }
+ return ref;
+}
+
+/*
+ JSRelease
+*/
+void JSRelease(JSTypeRef ref)
+{
+ if (ref)
+ {
+ JSBase* ptr = (JSBase*)ref;
+ ptr->Release();
+ }
+}
+
+/*
+ JSCopyDescription
+*/
+CFStringRef JSCopyDescription(JSTypeRef ref)
+{
+ CFStringRef result = 0;
+ if (ref)
+ {
+ JSBase* ptr = (JSBase*)ref;
+ ptr->CopyDescription();
+ }
+ return result;
+}
+
+/*
+ JSEqual
+*/
+UInt8 JSEqual(JSTypeRef ref1, JSTypeRef ref2)
+{
+ UInt8 result = false;
+ if (ref1 && ref2)
+ {
+ JSBase* ptr = (JSBase*)ref1;
+ result = ptr->Equal((JSBase*)ref2);
+ }
+ return result;
+}
+
+
+/*
+ JSGetTypeID
+*/
+JSTypeID JSGetTypeID(JSTypeRef ref)
+{
+ JSTypeID result = kJSInvalidTypeID;
+ if (ref)
+ {
+ JSBase* ptr = (JSBase*)ref;
+ result = ptr->GetTypeID();
+ }
+ return result;
+}
+
+
+/*
+ JSGetRetainCount
+*/
+CFIndex JSGetRetainCount(JSTypeRef ref)
+{
+ CFIndex result = -1;
+ if (ref)
+ {
+ JSBase* ptr = (JSBase*)ref;
+ result = ptr->RetainCount();
+ }
+ return result;
+}
+
+
+
+/*
+ JSObjectCreate
+*/
+JSObjectRef JSObjectCreate(void *data, JSObjectCallBacksPtr callBacks)
+{
+ JSObjectRef result = JSObjectCreateInternal(data, callBacks, 0, kJSUserObjectDataTypeUnknown);
+ return result;
+}
+
+/*
+ JSObjectCreateInternal
+*/
+JSObjectRef JSObjectCreateInternal(void *data, JSObjectCallBacksPtr callBacks, JSObjectMarkProcPtr markProc, int type)
+{
+ JSObjectRef result = 0;
+ JSUserObject* ptr = new JSUserObject(callBacks, markProc, data, type);
+ result = (JSObjectRef)ptr;
+ return result;
+}
+
+/*
+ JSObjectCopyCFValue
+*/
+CFTypeRef JSObjectCopyCFValue(JSObjectRef ref)
+{
+ CFTypeRef result = 0;
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ result = ptr->CopyCFValue();
+ }
+ return result;
+}
+
+/*
+ JSObjectGetData
+*/
+void *JSObjectGetData(JSObjectRef ref)
+{
+ void *result = 0;
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ result = ptr->GetData();
+ }
+ return result;
+}
+
+
+/*
+ JSObjectCopyProperty
+*/
+JSObjectRef JSObjectCopyProperty(JSObjectRef ref, CFStringRef propertyName)
+{
+ JSObjectRef result = 0;
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ result = (JSObjectRef)ptr->CopyProperty(propertyName);
+ }
+ return result;
+}
+
+
+/*
+ JSObjectSetProperty
+*/
+void JSObjectSetProperty(JSObjectRef ref, CFStringRef propertyName, JSObjectRef value)
+{
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ ptr->SetProperty(propertyName, (JSUserObject*)value);
+ }
+}
+
+
+/*
+ JSObjectCallFunction
+*/
+JSObjectRef JSObjectCallFunction(JSObjectRef ref, JSObjectRef thisObj, CFArrayRef args)
+{
+ JSObjectRef result = 0;
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ result = (JSObjectRef)ptr->CallFunction((JSUserObject*)thisObj, args);
+ }
+ return result;
+}
+
+
+/*
+ JSRunCreate
+*/
+JSRunRef JSRunCreate(CFStringRef jsSource, JSFlags inFlags)
+{
+ initializeThreading();
+
+ JSRunRef result = 0;
+ if (jsSource)
+ {
+ JSGlueAPIEntry entry;
+ result = (JSRunRef) new JSRun(jsSource, inFlags);
+ }
+ return result;
+}
+
+/*
+ JSRunCopySource
+*/
+CFStringRef JSRunCopySource(JSRunRef ref)
+{
+ CFStringRef result = 0;
+ JSRun* ptr = (JSRun*)ref;
+ if (ptr)
+ {
+ result = UStringToCFString(ptr->GetSource());
+ }
+ return result;
+}
+
+
+/*
+ JSRunCopyGlobalObject
+*/
+JSObjectRef JSRunCopyGlobalObject(JSRunRef ref)
+{
+ JSObjectRef result = 0;
+ JSRun* ptr = (JSRun*)ref;
+ if (ptr)
+ {
+ JSGlobalObject* globalObject = ptr->GlobalObject();
+ result = (JSObjectRef)KJSValueToJSObject(globalObject, globalObject->globalExec());
+ }
+ return result;
+}
+
+/*
+ JSRunEvaluate
+*/
+JSObjectRef JSRunEvaluate(JSRunRef ref)
+{
+ JSObjectRef result = 0;
+ JSRun* ptr = (JSRun*)ref;
+ if (ptr)
+ {
+ JSGlueAPIEntry entry;
+ Completion completion = ptr->Evaluate();
+ if (completion.isValueCompletion())
+ {
+ result = (JSObjectRef)KJSValueToJSObject(completion.value(), ptr->GlobalObject()->globalExec());
+ }
+
+ if (completion.complType() == Throw)
+ {
+ JSFlags flags = ptr->Flags();
+ if (flags & kJSFlagDebug)
+ {
+ CFTypeRef error = JSObjectCopyCFValue(result);
+ if (error)
+ {
+ CFShow(error);
+ CFRelease(error);
+ }
+ }
+ }
+ }
+ return result;
+}
+
+/*
+ JSRunCheckSyntax
+ Return true if no syntax error
+*/
+bool JSRunCheckSyntax(JSRunRef ref)
+{
+ bool result = false;
+ JSRun* ptr = (JSRun*)ref;
+ if (ptr)
+ {
+ JSGlueAPIEntry entry;
+ result = ptr->CheckSyntax();
+ }
+ return result;
+}
+
+/*
+ JSCollect - trigger garbage collection
+*/
+void JSCollect()
+{
+ initializeThreading();
+
+ JSGlueAPIEntry entry;
+ Heap* heap = getThreadGlobalExecState()->heap();
+ if (!heap->isBusy())
+ heap->collectAllGarbage();
+}
+
+/*
+ JSTypeGetCFArrayCallBacks
+*/
+void JSTypeGetCFArrayCallBacks(CFArrayCallBacks* outCallBacks)
+{
+ if (outCallBacks)
+ {
+ outCallBacks->version = 1;
+ outCallBacks->retain = (CFArrayRetainCallBack)JSCFRetain;
+ outCallBacks->release = (CFArrayReleaseCallBack)JSCFRelease;
+ outCallBacks->copyDescription = (CFArrayCopyDescriptionCallBack)JSCopyDescription;
+ outCallBacks->equal = (CFArrayEqualCallBack)JSEqual;
+ }
+}
+
+
+/*
+ JSCFRetain
+*/
+void *JSCFRetain(CFAllocatorRef allocator, const void *value)
+{
+ JSRetain((JSTypeRef)value);
+ return (void*)value;
+}
+
+/*
+ JSCFRelease
+*/
+void JSCFRelease(CFAllocatorRef allocator, const void *value)
+{
+ JSRelease((JSTypeRef)value);
+}
+
+
+/*
+ JSObjectCreateWithCFType
+*/
+JSObjectRef JSObjectCreateWithCFType(CFTypeRef inRef)
+{
+ JSObjectCallBacks callBacks;
+ JSObjectRef cfJSObject = nil;
+ if (inRef)
+ {
+ callBacks.dispose = CFJSObjectDispose;
+ callBacks.equal = CFJSObjectEqual;
+ callBacks.copyCFValue = CFJSObjectCopyCFValue;
+ callBacks.copyProperty = CFJSObjectCopyProperty;
+ callBacks.setProperty = CFJSObjectSetProperty;
+ callBacks.callFunction = 0;
+ callBacks.copyPropertyNames = CFJSObjectCopyPropertyNames;
+ cfJSObject = JSObjectCreateInternal((void*)CFRetain(inRef), &callBacks, 0, kJSUserObjectDataTypeCFType );
+ }
+ return cfJSObject;
+}
+
+/*
+ CFJSObjectDispose
+*/
+void CFJSObjectDispose(void *data)
+{
+ if (data)
+ {
+ CFRelease((JSTypeRef)data);
+ }
+}
+
+CFArrayRef JSObjectCopyPropertyNames(JSObjectRef ref)
+{
+ CFArrayRef result = 0;
+ JSUserObject* ptr = (JSUserObject*)ref;
+ if (ptr && (ptr->GetTypeID() == kJSObjectTypeID))
+ {
+ result = ptr->CopyPropertyNames();
+ }
+ return result;
+}
+/*
+ CFJSObjectCopyProperty
+*/
+JSObjectRef CFJSObjectCopyProperty(void *data, CFStringRef propertyName)
+{
+ JSObjectRef result = 0;
+ if (data && propertyName)
+ {
+ CFTypeRef cfResult = 0;
+ if (CFGetTypeID(data) == CFDictionaryGetTypeID())
+ {
+ if (CFStringCompare(propertyName, CFSTR("length"), 0) == kCFCompareEqualTo)
+ {
+ int len = CFDictionaryGetCount((CFDictionaryRef)data);
+ cfResult = CFNumberCreate(0, kCFNumberIntType, &len);
+ }
+ else
+ {
+ cfResult = RetainCFType(CFDictionaryGetValue((CFDictionaryRef)data, propertyName));
+ }
+ }
+ else if (CFGetTypeID(data) == CFArrayGetTypeID())
+ {
+ if (CFStringCompare(propertyName, CFSTR("length"), 0) == kCFCompareEqualTo)
+ {
+ int len = CFArrayGetCount((CFArrayRef)data);
+ cfResult = CFNumberCreate(0, kCFNumberIntType, &len);
+ }
+ else
+ {
+ SInt32 index = CFStringGetIntValue(propertyName);
+ CFIndex arrayCount = CFArrayGetCount((CFArrayRef)data);
+ if (index >= 0 && index < arrayCount)
+ {
+ cfResult = RetainCFType(CFArrayGetValueAtIndex((CFArrayRef)data, index));
+ }
+ }
+ }
+ else if (CFGetTypeID(data) == CFStringGetTypeID())
+ {
+ if (CFStringCompare(propertyName, CFSTR("length"), 0) == kCFCompareEqualTo)
+ {
+ int len = CFStringGetLength((CFStringRef)data);
+ cfResult = CFNumberCreate(0, kCFNumberIntType, &len);
+ }
+ }
+ if (cfResult)
+ {
+ result = JSObjectCreateWithCFType(cfResult);
+ CFRelease(cfResult);
+ }
+ }
+ return result;
+}
+
+
+/*
+ CFJSObjectSetProperty
+*/
+void CFJSObjectSetProperty(void *data, CFStringRef propertyName, JSObjectRef jsValue)
+{
+ if (data && propertyName)
+ {
+ CFTypeRef cfValue = JSObjectCopyCFValue(jsValue);
+
+ if (cfValue)
+ {
+ if (CFGetTypeID(data) == CFDictionaryGetTypeID())
+ {
+ CFDictionarySetValue((CFMutableDictionaryRef)data, propertyName, cfValue);
+ }
+ else if (CFGetTypeID(data) == CFArrayGetTypeID())
+ {
+ SInt32 index = CFStringGetIntValue(propertyName);
+ CFIndex arrayCount = CFArrayGetCount((CFArrayRef)data);
+ if (index >= 0)
+ {
+ for (; arrayCount < index; arrayCount++)
+ {
+ CFArrayAppendValue((CFMutableArrayRef)data, GetCFNull());
+ }
+ CFArraySetValueAtIndex((CFMutableArrayRef)data, index, cfValue);
+ }
+ }
+ CFRelease(cfValue);
+ }
+ else
+ {
+ if (CFGetTypeID(data) == CFDictionaryGetTypeID())
+ {
+ CFDictionaryRemoveValue((CFMutableDictionaryRef)data, propertyName);
+ }
+ else if (CFGetTypeID(data) == CFArrayGetTypeID())
+ {
+ SInt32 index = CFStringGetIntValue(propertyName);
+ CFIndex arrayCount = CFArrayGetCount((CFArrayRef)data);
+ if (index >= 0)
+ {
+ for (; arrayCount < index; arrayCount++)
+ {
+ CFArrayAppendValue((CFMutableArrayRef)data, GetCFNull());
+ }
+ CFArraySetValueAtIndex((CFMutableArrayRef)data, index, GetCFNull());
+ }
+ }
+ }
+ }
+}
+
+
+/*
+ CFJSObjectCopyCFValue
+*/
+CFTypeRef CFJSObjectCopyCFValue(void *data)
+{
+ CFTypeRef result = 0;
+ if (data)
+ {
+ result = (CFTypeRef)CFRetain(data);
+ }
+ return result;
+}
+
+/*
+ CFJSObjectCopyCFValue
+*/
+UInt8 CFJSObjectEqual(void *data1, void *data2)
+{
+ UInt8 result = false;
+ if (data1 && data2)
+ {
+ CFEqual((CFTypeRef)data1, (CFTypeRef)data2);
+ }
+ return result;
+}
+
+
+/*
+ CFJSObjectCopyPropertyNames
+*/
+CFArrayRef CFJSObjectCopyPropertyNames(void *data)
+{
+ CFMutableArrayRef result = 0;
+ if (data)
+ {
+ CFTypeID cfType = CFGetTypeID(data);
+ if (cfType == CFDictionaryGetTypeID())
+ {
+ CFIndex count = CFDictionaryGetCount((CFDictionaryRef)data);
+ if (count)
+ {
+ CFTypeRef* keys = (CFTypeRef*)malloc(sizeof(CFTypeRef)*count);
+ if (keys)
+ {
+ int i;
+ CFDictionaryGetKeysAndValues((CFDictionaryRef)data, (const void **)keys, 0);
+ for (i = 0; i < count; i++)
+ {
+ CFStringRef key = (CFStringRef)keys[i];
+ if (CFGetTypeID(key) != CFStringGetTypeID()) continue;
+
+ if (!result) result = CFArrayCreateMutable(0, 0, &kCFTypeArrayCallBacks);
+ if (!result) continue;
+
+ CFArrayAppendValue(result, key);
+ }
+ free(keys);
+ }
+ }
+ }
+ }
+ return result;
+}
+
+
+
+
+CFMutableArrayRef JSCreateCFArrayFromJSArray(CFArrayRef array)
+{
+ CFIndex count = array ? CFArrayGetCount(array) : 0;
+ CFMutableArrayRef cfArray = CFArrayCreateMutable(0, 0, &kCFTypeArrayCallBacks);
+ CFIndex i;
+
+ for (i = 0; cfArray && i < count; i++)
+ {
+ JSObjectRef jsValue = (JSObjectRef)CFArrayGetValueAtIndex(array, i);
+ CFTypeRef cfvalue = JSObjectCopyCFValue(jsValue);
+ if (cfvalue)
+ {
+ CFArrayAppendValue(cfArray, cfvalue);
+ CFRelease(cfvalue);
+ }
+ else
+ {
+ CFArrayAppendValue(cfArray, GetCFNull());
+ }
+ }
+ return cfArray;
+}
+
+CFMutableArrayRef JSCreateJSArrayFromCFArray(CFArrayRef array)
+{
+ initializeThreading();
+
+ CFIndex count = array ? CFArrayGetCount(array) : 0;
+ CFArrayCallBacks arrayCallbacks;
+ CFMutableArrayRef jsArray;
+ CFIndex i;
+
+ JSTypeGetCFArrayCallBacks(&arrayCallbacks);
+ jsArray = CFArrayCreateMutable(0, 0, &arrayCallbacks);
+
+ for (i = 0; array && i < count; i++)
+ {
+ CFTypeRef cfValue = (CFTypeRef)CFArrayGetValueAtIndex(array, i);
+ JSObjectRef jsValue = JSObjectCreateWithCFType(cfValue);
+
+ if (!jsValue) jsValue = JSObjectCreateWithCFType(GetCFNull());
+ if (jsValue)
+ {
+ CFArrayAppendValue(jsArray, jsValue);
+ JSRelease(jsValue);
+ }
+ }
+ return jsArray;
+}
+
+
+void JSLockInterpreter()
+{
+ initializeThreading();
+ JSLock::lock(LockForReal);
+}
+
+
+void JSUnlockInterpreter()
+{
+ JSLock::unlock(LockForReal);
+}