Patch by Lars Knoll, comment out ICU dependency on Qt platform (unused code).
[WebKit-https.git] / JavaScriptCore / bindings / c / c_utility.cpp
1 /*
2  * Copyright (C) 2004, 2006 Apple Computer, Inc.  All rights reserved.
3  * Copyright (C) 2006 Alexey Proskuryakov (ap@nypop.com)
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 "c_utility.h"
29
30 #include "NP_jsobject.h"
31 #include "c_instance.h" 
32 #include "npruntime_impl.h"
33 #include "npruntime_priv.h"
34 #include "runtime_object.h"
35 #include "runtime_root.h"
36 #include "Platform.h"
37 #if USE(ICU_UNICODE)
38 #include <unicode/ucnv.h>
39 #endif
40
41 namespace KJS { namespace Bindings {
42
43 // Requires free() of returned UTF16Chars.
44 void convertNPStringToUTF16(const NPString *string, NPUTF16 **UTF16Chars, unsigned int *UTF16Length)
45 {
46     convertUTF8ToUTF16(string->UTF8Characters, string->UTF8Length, UTF16Chars, UTF16Length);
47 }
48
49 // Requires free() of returned UTF16Chars.
50 void convertUTF8ToUTF16(const NPUTF8 *UTF8Chars, int UTF8Length, NPUTF16 **UTF16Chars, unsigned int *UTF16Length)
51 {
52 #if USE(ICU_UNICODE)
53     assert(UTF8Chars || UTF8Length == 0);
54     assert(UTF16Chars);
55     
56     if (UTF8Length == -1)
57         UTF8Length = static_cast<int>(strlen(UTF8Chars));
58         
59     // UTF16Length maximum length is the length of the UTF8 string, plus one to include terminator
60     // Without the plus one, it will convert ok, but a warning is generated from the converter as
61     // there is not enough room for a terminating character.
62     *UTF16Length = UTF8Length + 1; 
63         
64     *UTF16Chars = 0;
65     UErrorCode status = U_ZERO_ERROR;
66     UConverter* conv = ucnv_open("utf8", &status);
67     if (U_SUCCESS(status)) { 
68         *UTF16Chars = (NPUTF16 *)malloc(sizeof(NPUTF16) * (*UTF16Length));
69         ucnv_setToUCallBack(conv, UCNV_TO_U_CALLBACK_STOP, 0, 0, 0, &status);
70         *UTF16Length = ucnv_toUChars(conv, (::UChar*)*UTF16Chars, *UTF16Length, UTF8Chars, UTF8Length, &status); 
71         ucnv_close(conv);
72     } 
73     
74     // Check to see if the conversion was successful
75     // Some plugins return invalid UTF-8 in NPVariantType_String, see <http://bugzilla.opendarwin.org/show_bug.cgi?id=5163>
76     // There is no "bad data" for latin1. It is unlikely that the plugin was really sending text in this encoding,
77     // but it should have used UTF-8, and now we are simply avoiding a crash.
78     if (!U_SUCCESS(status)) {
79         *UTF16Length = UTF8Length;
80         
81         if (!*UTF16Chars)   // If the memory wasn't allocated, allocate it.
82             *UTF16Chars = (NPUTF16 *)malloc(sizeof(NPUTF16) * (*UTF16Length));
83  
84         for (unsigned i = 0; i < *UTF16Length; i++)
85             (*UTF16Chars)[i] = UTF8Chars[i] & 0xFF;
86     }
87 #else
88     assert(!"Implement me!");    
89 #endif
90 }
91
92 // Variant value must be released with NPReleaseVariantValue()
93 void coerceValueToNPVariantStringType(ExecState *exec, JSValue *value, NPVariant *result)
94 {
95     UString ustring = value->toString(exec);
96     CString cstring = ustring.UTF8String();
97     NPString string = { (const NPUTF8 *)cstring.c_str(), static_cast<uint32_t>(cstring.size()) };
98     NPN_InitializeVariantWithStringCopy(result, &string);
99 }
100
101 // Variant value must be released with NPReleaseVariantValue()
102 void convertValueToNPVariant(ExecState *exec, JSValue *value, NPVariant *result)
103 {
104     JSType type = value->type();
105     
106     if (type == StringType) {
107         UString ustring = value->toString(exec);
108         CString cstring = ustring.UTF8String();
109         NPString string = { (const NPUTF8 *)cstring.c_str(), static_cast<uint32_t>(cstring.size()) };
110         NPN_InitializeVariantWithStringCopy(result, &string);
111     } else if (type == NumberType) {
112         DOUBLE_TO_NPVARIANT(value->toNumber(exec), *result);
113     } else if (type == BooleanType) {
114         BOOLEAN_TO_NPVARIANT(value->toBoolean(exec), *result);
115     } else if (type == UnspecifiedType) {
116         VOID_TO_NPVARIANT(*result);
117     } else if (type == NullType) {
118         NULL_TO_NPVARIANT(*result);
119     } else if (type == ObjectType) {
120         JSObject *objectImp = static_cast<JSObject*>(value);
121         if (objectImp->classInfo() == &RuntimeObjectImp::info) {
122             RuntimeObjectImp* imp = static_cast<RuntimeObjectImp *>(value);
123             CInstance* instance = static_cast<CInstance*>(imp->getInternalInstance());
124             NPObject* obj = instance->getObject();
125             _NPN_RetainObject(obj);
126             OBJECT_TO_NPVARIANT(obj, *result);
127         } else {
128             Interpreter *originInterpreter = exec->dynamicInterpreter();
129             const Bindings::RootObject *originExecutionContext = rootForInterpreter(originInterpreter);
130
131             Interpreter *interpreter = 0;
132             if (originInterpreter->isGlobalObject(value)) {
133                 interpreter = originInterpreter->interpreterForGlobalObject(value);
134             }
135
136             if (!interpreter)
137                 interpreter = originInterpreter;
138                 
139             const Bindings::RootObject *executionContext = rootForInterpreter(interpreter);
140             if (!executionContext) {
141                 Bindings::RootObject *newExecutionContext = new Bindings::RootObject(0);
142                 newExecutionContext->setInterpreter(interpreter);
143                 executionContext = newExecutionContext;
144             }
145     
146             NPObject* obj = (NPObject *)exec->dynamicInterpreter()->createLanguageInstanceForValue(exec, Instance::CLanguage, value->toObject(exec), originExecutionContext, executionContext);
147             OBJECT_TO_NPVARIANT(obj, *result);
148         }
149     }
150     else
151         VOID_TO_NPVARIANT(*result);
152 }
153
154 JSValue *convertNPVariantToValue(ExecState*, const NPVariant* variant)
155 {
156     NPVariantType type = variant->type;
157
158     if (type == NPVariantType_Bool)
159         return jsBoolean(NPVARIANT_TO_BOOLEAN(*variant));
160     if (type == NPVariantType_Null)
161         return jsNull();
162     if (type == NPVariantType_Void)
163         return jsUndefined();
164     if (type == NPVariantType_Int32)
165         return jsNumber(NPVARIANT_TO_INT32(*variant));
166     if (type == NPVariantType_Double)
167         return jsNumber(NPVARIANT_TO_DOUBLE(*variant));
168     if (type == NPVariantType_String) {
169         NPUTF16 *stringValue;
170         unsigned int UTF16Length;
171         convertNPStringToUTF16(&variant->value.stringValue, &stringValue, &UTF16Length); // requires free() of returned memory
172         UString resultString((const UChar *)stringValue,UTF16Length);
173         free(stringValue);
174         return jsString(resultString);
175     }
176     if (type == NPVariantType_Object) {
177         NPObject *obj = variant->value.objectValue;
178         
179         if (obj->_class == NPScriptObjectClass)
180             // Get JSObject from NP_JavaScriptObject.
181             return ((JavaScriptObject *)obj)->imp;
182
183         // Wrap NPObject in a CInstance.
184         return Instance::createRuntimeObject(Instance::CLanguage, obj);
185     }
186     
187     return jsUndefined();
188 }
189
190 Identifier identifierFromNPIdentifier(const NPUTF8* name)
191 {
192     NPUTF16 *methodName;
193     unsigned UTF16Length;
194     convertUTF8ToUTF16(name, -1, &methodName, &UTF16Length); // requires free() of returned memory.
195     Identifier identifier((const KJS::UChar*)methodName, UTF16Length);
196     free(methodName);
197     return identifier;
198 }
199
200 } }