2009-01-14 Jeremy Moskovich <jeremy@chromium.org>
[WebKit-https.git] / WebCore / bridge / c / c_instance.cpp
1 /*
2  * Copyright (C) 2003, 2006 Apple Computer, Inc.  All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27
28 #if ENABLE(NETSCAPE_PLUGIN_API)
29
30 #include "c_instance.h"
31
32 #include "c_class.h"
33 #include "c_runtime.h"
34 #include "c_utility.h"
35 #include "npruntime_impl.h"
36 #include "runtime_root.h"
37 #include <runtime/ArgList.h>
38 #include <runtime/Error.h>
39 #include <interpreter/CallFrame.h>
40 #include <runtime/JSLock.h>
41 #include <runtime/JSNumberCell.h>
42 #include <runtime/PropertyNameArray.h>
43 #include <wtf/Assertions.h>
44 #include <wtf/StdLibExtras.h>
45 #include <wtf/StringExtras.h>
46 #include <wtf/Vector.h>
47
48 namespace JSC {
49 namespace Bindings {
50
51 using JSC::UString;
52
53 JSC::UString& globalExceptionString()
54 {
55     DEFINE_STATIC_LOCAL(JSC::UString, exceptionStr, ());
56     return exceptionStr;
57 }
58
59 void CInstance::setGlobalException(UString exception)
60 {
61     globalExceptionString() = exception;
62 }
63
64 void CInstance::moveGlobalExceptionToExecState(ExecState* exec)
65 {
66     if (globalExceptionString().isNull())
67         return;
68
69     {
70         JSLock lock(false);
71         throwError(exec, GeneralError, globalExceptionString());
72     }
73
74     globalExceptionString() = UString();
75 }
76
77 CInstance::CInstance(NPObject* o, PassRefPtr<RootObject> rootObject)
78     : Instance(rootObject)
79 {
80     _object = _NPN_RetainObject(o);
81     _class = 0;
82 }
83
84 CInstance::~CInstance() 
85 {
86     _NPN_ReleaseObject(_object);
87 }
88
89 Class *CInstance::getClass() const
90 {
91     if (!_class)
92         _class = CClass::classForIsA(_object->_class);
93     return _class;
94 }
95
96 bool CInstance::supportsInvokeDefaultMethod() const
97 {
98     return _object->_class->invokeDefault;
99 }
100
101 JSValuePtr CInstance::invokeMethod(ExecState* exec, const MethodList& methodList, const ArgList& args)
102 {
103     // Overloading methods are not allowed by NPObjects.  Should only be one
104     // name match for a particular method.
105     ASSERT(methodList.size() == 1);
106
107     CMethod* method = static_cast<CMethod*>(methodList[0]);
108
109     NPIdentifier ident = _NPN_GetStringIdentifier(method->name());
110     if (!_object->_class->hasMethod(_object, ident))
111         return jsUndefined();
112
113     unsigned count = args.size();
114     Vector<NPVariant, 8> cArgs(count);
115
116     unsigned i;
117     for (i = 0; i < count; i++)
118         convertValueToNPVariant(exec, args.at(exec, i), &cArgs[i]);
119
120     // Invoke the 'C' method.
121     NPVariant resultVariant;
122     VOID_TO_NPVARIANT(resultVariant);
123
124     {
125         JSLock::DropAllLocks dropAllLocks(false);
126         ASSERT(globalExceptionString().isNull());
127         _object->_class->invoke(_object, ident, cArgs.data(), count, &resultVariant);
128         moveGlobalExceptionToExecState(exec);
129     }
130
131     for (i = 0; i < count; i++)
132         _NPN_ReleaseVariantValue(&cArgs[i]);
133
134     JSValuePtr resultValue = convertNPVariantToValue(exec, &resultVariant, _rootObject.get());
135     _NPN_ReleaseVariantValue(&resultVariant);
136     return resultValue;
137 }
138
139
140 JSValuePtr CInstance::invokeDefaultMethod(ExecState* exec, const ArgList& args)
141 {
142     if (!_object->_class->invokeDefault)
143         return jsUndefined();
144
145     unsigned count = args.size();
146     Vector<NPVariant, 8> cArgs(count);
147
148     unsigned i;
149     for (i = 0; i < count; i++)
150         convertValueToNPVariant(exec, args.at(exec, i), &cArgs[i]);
151
152     // Invoke the 'C' method.
153     NPVariant resultVariant;
154     VOID_TO_NPVARIANT(resultVariant);
155     {
156         JSLock::DropAllLocks dropAllLocks(false);
157         ASSERT(globalExceptionString().isNull());
158         _object->_class->invokeDefault(_object, cArgs.data(), count, &resultVariant);
159         moveGlobalExceptionToExecState(exec);
160     }
161
162     for (i = 0; i < count; i++)
163         _NPN_ReleaseVariantValue(&cArgs[i]);
164
165     JSValuePtr resultValue = convertNPVariantToValue(exec, &resultVariant, _rootObject.get());
166     _NPN_ReleaseVariantValue(&resultVariant);
167     return resultValue;
168 }
169
170 bool CInstance::supportsConstruct() const
171 {
172     return _object->_class->construct;
173 }
174     
175 JSValuePtr CInstance::invokeConstruct(ExecState* exec, const ArgList& args)
176 {
177     if (!_object->_class->construct)
178         return jsUndefined();
179
180     unsigned count = args.size();
181     Vector<NPVariant, 8> cArgs(count);
182
183     unsigned i;
184     for (i = 0; i < count; i++)
185         convertValueToNPVariant(exec, args.at(exec, i), &cArgs[i]);
186
187     // Invoke the 'C' method.
188     NPVariant resultVariant;
189     VOID_TO_NPVARIANT(resultVariant);
190     {
191         JSLock::DropAllLocks dropAllLocks(false);
192         ASSERT(globalExceptionString().isNull());
193         _object->_class->construct(_object, cArgs.data(), count, &resultVariant);
194         moveGlobalExceptionToExecState(exec);
195     }
196
197     for (i = 0; i < count; i++)
198         _NPN_ReleaseVariantValue(&cArgs[i]);
199
200     JSValuePtr resultValue = convertNPVariantToValue(exec, &resultVariant, _rootObject.get());
201     _NPN_ReleaseVariantValue(&resultVariant);
202     return resultValue;
203 }
204
205 JSValuePtr CInstance::defaultValue(ExecState* exec, PreferredPrimitiveType hint) const
206 {
207     if (hint == PreferString)
208         return stringValue(exec);
209     if (hint == PreferNumber)
210         return numberValue(exec);
211     return valueOf(exec);
212 }
213
214 JSValuePtr CInstance::stringValue(ExecState* exec) const
215 {
216     char buf[1024];
217     snprintf(buf, sizeof(buf), "NPObject %p, NPClass %p", _object, _object->_class);
218     return jsString(exec, buf);
219 }
220
221 JSValuePtr CInstance::numberValue(ExecState* exec) const
222 {
223     // FIXME: Implement something sensible.
224     return jsNumber(exec, 0);
225 }
226
227 JSValuePtr CInstance::booleanValue() const
228 {
229     // FIXME: Implement something sensible.
230     return jsBoolean(false);
231 }
232
233 JSValuePtr CInstance::valueOf(ExecState* exec) const 
234 {
235     return stringValue(exec);
236 }
237
238 void CInstance::getPropertyNames(ExecState* exec, PropertyNameArray& nameArray)
239 {
240     if (!NP_CLASS_STRUCT_VERSION_HAS_ENUM(_object->_class) || !_object->_class->enumerate)
241         return;
242
243     uint32_t count;
244     NPIdentifier* identifiers;
245
246     {
247         JSLock::DropAllLocks dropAllLocks(false);
248         ASSERT(globalExceptionString().isNull());
249         bool ok = _object->_class->enumerate(_object, &identifiers, &count);
250         moveGlobalExceptionToExecState(exec);
251         if (!ok)
252             return;
253     }
254
255     for (uint32_t i = 0; i < count; i++) {
256         PrivateIdentifier* identifier = static_cast<PrivateIdentifier*>(identifiers[i]);
257
258         if (identifier->isString)
259             nameArray.add(identifierFromNPIdentifier(identifier->value.string));
260         else
261             nameArray.add(Identifier::from(exec, identifier->value.number));
262     }
263
264     // FIXME: This should really call NPN_MemFree but that's in WebKit
265     free(identifiers);
266 }
267
268 }
269 }
270
271 #endif // ENABLE(NETSCAPE_PLUGIN_API)