blob: 61d1c8e8d516d01bc0f711e971bd82e09a6c325e [file] [log] [blame]
jat@google.com134be542009-08-03 15:30:11 +00001/*
2 * Copyright 2008 Google Inc.
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
5 * use this file except in compliance with the License. You may obtain a copy of
6 * the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13 * License for the specific language governing permissions and limitations under
14 * the License.
15 */
16
17#include "JavaObject.h"
18#include "FFSessionHandler.h"
19#include "SessionData.h"
20#include "ServerMethods.h"
21#include "Debug.h"
22#include "XpcomDebug.h"
23#include "HostChannel.h"
24#include "InvokeMessage.h"
25#include "ReturnMessage.h"
26#include "scoped_ptr/scoped_ptr.h"
27
28static JSClass JavaObjectClass = {
29 "GWTJavaObject", /* class name */
30 JSCLASS_HAS_PRIVATE | JSCLASS_HAS_RESERVED_SLOTS(1) | JSCLASS_NEW_ENUMERATE, /* flags */
31
32 JS_PropertyStub, /* add property */
33 JS_PropertyStub, /* delete property */
34 JavaObject::getProperty, /* get property */
35 JavaObject::setProperty, /* set property */
36
37 reinterpret_cast<JSEnumerateOp>(JavaObject::enumerate), /* enumerate */
38 JS_ResolveStub, /* resolve */
39 JS_ConvertStub, // JavaObject::convert, /* convert */
40 JavaObject::finalize, /* finalize */ //TODO
41
42 NULL, /* object hooks */
43 NULL, /* check access */
conroy@google.com36cf7482011-01-12 19:43:57 +000044#if GECKO_VERSION < 2000
jat@google.com134be542009-08-03 15:30:11 +000045 JavaObject::call, /* call */ //TODO
conroy@google.com36cf7482011-01-12 19:43:57 +000046#else
47 JavaObject::call20, /* call: compatability wrapper for 2.0+ */
48#endif //GECKO_VERSION
jat@google.com134be542009-08-03 15:30:11 +000049 NULL, /* construct */
50 NULL, /* object serialization */
51 NULL, /* has instance */
52 NULL, /* mark */
53 NULL /* reserve slots */
54};
55
56int JavaObject::getObjectId(JSContext* ctx, JSObject* obj) {
57 jsval val;
58 JSClass* jsClass = JS_GET_CLASS(ctx, obj);
jat@google.com134be542009-08-03 15:30:11 +000059 if (jsClass != &JavaObjectClass) {
60 Debug::log(Debug::Error)
61 << "JavaObject::getObjectId called on non-JavaObject: " << jsClass->name
62 << Debug::flush;
63 return -1;
64 }
65 if (JSCLASS_RESERVED_SLOTS(jsClass) < 1) {
66 Debug::log(Debug::Error)
67 << "JavaObject::getObjectId -- " << static_cast<void*>(obj)
68 << " has only " << (JSCLASS_RESERVED_SLOTS(jsClass))
69 << " reserved slots, no objectId present" << Debug::flush;
70 return -1;
71 }
jat@google.com134be542009-08-03 15:30:11 +000072 if (!JS_GetReservedSlot(ctx, obj, 0, &val)) {
73 Debug::log(Debug::Error) << "Error getting reserved slot" << Debug::flush;
74 return -1;
75 }
76 // TODO: assert JSVAL_IS_INT(val)
77 return JSVAL_TO_INT(val);
78}
79
80SessionData* JavaObject::getSessionData(JSContext* ctx, JSObject* obj) {
81 void* data = JS_GetInstancePrivate(ctx, obj, &JavaObjectClass, NULL);
82 return static_cast<SessionData*>(data);
83}
84
85
86bool JavaObject::isJavaObject(JSContext* ctx, JSObject* obj) {
87 return JS_GET_CLASS(ctx, obj) == &JavaObjectClass;
88}
89
90JSObject* JavaObject::construct(JSContext* ctx, SessionData* data, int objectRef) {
91 // TODO: prototype? parent?
92 Debug::log(Debug::Spam) << "JavaObject::construct objectId=" << objectRef << Debug::flush;
93 JSObject* obj = JS_NewObject(ctx, &JavaObjectClass, NULL, NULL);
94 Debug::log(Debug::Spam) << " obj=" << obj << Debug::flush;
95 if (!obj) {
96 return NULL;
97 }
98 // set the session data
99 if (!JS_SetPrivate(ctx, obj, data)) {
100 Debug::log(Debug::Error) << "Could not set private data" << Debug::flush;
101 return NULL;
102 }
103 // set the objectId
104 if (!JS_SetReservedSlot(ctx, obj, 0, INT_TO_JSVAL(objectRef))) {
105 Debug::log(Debug::Error) << "Could not set reserved slot" << Debug::flush;
106 return NULL;
107 }
108 // define toString (TODO: some way to avoid doing this each time)
conroy@google.com36cf7482011-01-12 19:43:57 +0000109#if GECKO_VERSION < 2000
jat@google.com134be542009-08-03 15:30:11 +0000110 if (!JS_DefineFunction(ctx, obj, "toString", JavaObject::toString, 0, 0)) {
111 Debug::log(Debug::Error) << "Could not define toString method on object"
112 << Debug::flush;
113 }
conroy@google.com36cf7482011-01-12 19:43:57 +0000114#else
115 if (!JS_DefineFunction(ctx, obj, "toString", JavaObject::toString20, 0, 0)) {
116 Debug::log(Debug::Error) << "Could not define toString method on object"
117 << Debug::flush;
118 }
119#endif //GECKO_VERSION
jat@google.com134be542009-08-03 15:30:11 +0000120 return obj;
121}
122
conroy@google.com36cf7482011-01-12 19:43:57 +0000123JSBool JavaObject::getProperty(JSContext* ctx, JSObject* obj, jsid id,
jat@google.com134be542009-08-03 15:30:11 +0000124 jsval* rval) {
125 Debug::log(Debug::Spam) << "JavaObject::getProperty obj=" << obj << Debug::flush;
126 SessionData* data = JavaObject::getSessionData(ctx, obj);
127 if (!data) {
128 // TODO: replace the frame with an error page instead?
129 *rval = JSVAL_VOID;
130 return JS_TRUE;
131 }
132 int objectRef = JavaObject::getObjectId(ctx, obj);
conroy@google.com36cf7482011-01-12 19:43:57 +0000133 if (JSID_IS_STRING(id)) {
134 JSString* str = JSID_TO_STRING(id);
135 if ((JS_GetStringEncodingLength(ctx, str) == 8) && !strncmp("toString",
136 JS_EncodeString(ctx, str), 8)) {
jat@google.com134be542009-08-03 15:30:11 +0000137 *rval = data->getToStringTearOff();
138 return JS_TRUE;
139 }
conroy@google.com36cf7482011-01-12 19:43:57 +0000140 if ((JS_GetStringEncodingLength(ctx, str) == 2) && !strncmp("id",
141 JS_EncodeString(ctx, str), 2)) {
jat@google.com134be542009-08-03 15:30:11 +0000142 *rval = INT_TO_JSVAL(objectRef);
143 return JS_TRUE;
144 }
conroy@google.com36cf7482011-01-12 19:43:57 +0000145 if ((JS_GetStringEncodingLength(ctx, str) == 16) && !strncmp("__noSuchMethod__",
146 JS_EncodeString(ctx, str), 16)) {
jat@google.com5e86cbd2009-08-22 23:59:24 +0000147 // Avoid error spew if we are disconnected
148 *rval = JSVAL_VOID;
149 return JS_TRUE;
150 }
jat@google.com134be542009-08-03 15:30:11 +0000151 Debug::log(Debug::Error) << "Getting unexpected string property "
152 << dumpJsVal(ctx, id) << Debug::flush;
153 // TODO: throw a better exception here
154 return JS_FALSE;
155 }
conroy@google.com36cf7482011-01-12 19:43:57 +0000156 if (!JSID_IS_INT(id)) {
jat@google.com134be542009-08-03 15:30:11 +0000157 Debug::log(Debug::Error) << "Getting non-int/non-string property "
158 << dumpJsVal(ctx, id) << Debug::flush;
159 // TODO: throw a better exception here
160 return JS_FALSE;
161 }
conroy@google.com36cf7482011-01-12 19:43:57 +0000162 int dispId = JSID_TO_INT(id);
jat@google.com134be542009-08-03 15:30:11 +0000163
164 HostChannel* channel = data->getHostChannel();
165 SessionHandler* handler = data->getSessionHandler();
166
167 Value value = ServerMethods::getProperty(*channel, handler, objectRef, dispId);
168 data->makeJsvalFromValue(*rval, ctx, value);
169 return JS_TRUE;
170}
171
conroy@google.com36cf7482011-01-12 19:43:57 +0000172JSBool JavaObject::setProperty(JSContext* ctx, JSObject* obj, jsid id,
jat@google.com134be542009-08-03 15:30:11 +0000173 jsval* vp) {
174 Debug::log(Debug::Spam) << "JavaObject::setProperty obj=" << obj << Debug::flush;
conroy@google.com36cf7482011-01-12 19:43:57 +0000175 if (!JSID_IS_INT(id)) {
jat@google.com134be542009-08-03 15:30:11 +0000176 Debug::log(Debug::Error) << " Error: setting string property id" << Debug::flush;
177 // TODO: throw a better exception here
178 return JS_FALSE;
179 }
180
181 SessionData* data = JavaObject::getSessionData(ctx, obj);
182 if (!data) {
183 return JS_TRUE;
184 }
185
186 int objectRef = JavaObject::getObjectId(ctx, obj);
conroy@google.com36cf7482011-01-12 19:43:57 +0000187 int dispId = JSID_TO_INT(id);
jat@google.com134be542009-08-03 15:30:11 +0000188
189 Value value;
190 data->makeValueFromJsval(value, ctx, *vp);
191
192 HostChannel* channel = data->getHostChannel();
193 SessionHandler* handler = data->getSessionHandler();
194
195 if (!ServerMethods::setProperty(*channel, handler, objectRef, dispId, value)) {
196 // TODO: throw a better exception here
197 return JS_FALSE;
198 }
199 return JS_TRUE;
200}
201
202// TODO: can this be removed now?
203JSBool JavaObject::convert(JSContext* ctx, JSObject* obj, JSType type, jsval* vp) {
204 Debug::log(Debug::Spam) << "JavaObject::convert obj=" << obj
205 << " type=" << type << Debug::flush;
206 switch (type) {
207 case JSTYPE_STRING:
208 return toString(ctx, obj, 0, NULL, vp);
209 case JSTYPE_VOID:
210 *vp = JSVAL_VOID;
211 return JS_TRUE;
212 case JSTYPE_NULL:
213 *vp = JSVAL_NULL;
214 return JS_TRUE;
215 case JSTYPE_OBJECT:
216 *vp = OBJECT_TO_JSVAL(obj);
217 return JS_TRUE;
218 default:
219 break;
220 }
221 return JS_FALSE;
222}
223
224/**
225 * List of property names we want to fake on wrapped Java objects.
226 */
227static const char* propertyNames[] = {
228 "toString",
229 "id",
230};
231#define NUM_PROPERTY_NAMES (sizeof(propertyNames) / sizeof(propertyNames[0]))
232
233JSBool JavaObject::enumerate(JSContext* ctx, JSObject* obj, JSIterateOp op,
234 jsval* statep, jsid* idp) {
235 int objectId = JavaObject::getObjectId(ctx, obj);
236 switch (op) {
237 case JSENUMERATE_INIT:
238 Debug::log(Debug::Spam) << "JavaObject::enumerate(oid=" << objectId
239 << ", INIT)" << Debug::flush;
240 *statep = JSVAL_ZERO;
241 if (idp) {
conroy@google.com36cf7482011-01-12 19:43:57 +0000242 *idp = INT_TO_JSID(NUM_PROPERTY_NAMES);
jat@google.com134be542009-08-03 15:30:11 +0000243 }
244 break;
245 case JSENUMERATE_NEXT:
246 {
247 int idNum = JSVAL_TO_INT(*statep);
248 Debug::log(Debug::Spam) << "JavaObject::enumerate(oid=" << objectId
249 << ", NEXT " << idNum << ")" << Debug::flush;
250 *statep = INT_TO_JSVAL(idNum + 1);
251 if (idNum >= NUM_PROPERTY_NAMES) {
252 *statep = JSVAL_NULL;
conroy@google.com36cf7482011-01-12 19:43:57 +0000253#if GECKO_VERSION < 2000
254 //TODO(jat): do we need to do this?
jat@google.com134be542009-08-03 15:30:11 +0000255 *idp = JSVAL_NULL;
conroy@google.com36cf7482011-01-12 19:43:57 +0000256#endif //GECKO_VERSION
jat@google.com134be542009-08-03 15:30:11 +0000257 } else {
258 const char* propName = propertyNames[idNum];
259 JSString* str = JS_NewStringCopyZ(ctx, propName);
260 return JS_ValueToId(ctx, STRING_TO_JSVAL(str), idp);
261 }
262 break;
263 }
264 case JSENUMERATE_DESTROY:
265 Debug::log(Debug::Spam) << "JavaObject::enumerate(oid=" << objectId
266 << ", DESTROY)" << Debug::flush;
267 *statep = JSVAL_NULL;
268 break;
269 default:
270 Debug::log(Debug::Error) << "Unknown Enumerate op " <<
271 static_cast<int>(op) << Debug::flush;
272 return JS_FALSE;
273 }
274 return JS_TRUE;
275}
276
277void JavaObject::finalize(JSContext* ctx, JSObject* obj) {
scottb@google.com12c8f152009-12-01 00:12:57 +0000278 Debug::log(Debug::Spam) << "JavaObject::finalize obj=" << obj
jat@google.com134be542009-08-03 15:30:11 +0000279 << " objId=" << JavaObject::getObjectId(ctx, obj) << Debug::flush;
280 SessionData* data = JavaObject::getSessionData(ctx, obj);
281 if (data) {
282 int objectId = JavaObject::getObjectId(ctx, obj);
283 data->freeJavaObject(objectId);
284 JS_SetPrivate(ctx, obj, NULL);
285 }
286}
287
288JSBool JavaObject::toString(JSContext* ctx, JSObject* obj, uintN argc,
289 jsval* argv, jsval* rval) {
290 SessionData* data = JavaObject::getSessionData(ctx, obj);
291 if (!data) {
292 *rval = JSVAL_VOID;
293 return JS_TRUE;
294 }
295 int oid = getObjectId(ctx, obj);
296 Debug::log(Debug::Spam) << "JavaObject::toString(id=" << oid << ")"
297 << Debug::flush;
298 Value javaThis;
299 javaThis.setJavaObject(oid);
300 // we ignore any supplied parameters
301 return invokeJava(ctx, data, javaThis, InvokeMessage::TOSTRING_DISP_ID, 0,
302 NULL, rval);
303}
304
305/**
306 * Called when the JavaObject is invoked as a function.
307 * We ignore the JSObject* argument, which is the 'this' context, which is
308 * usually the window object. The JavaObject instance is in argv[-2].
309 *
310 * Returns a JS array, with the first element being a boolean indicating that
311 * an exception occured, and the second element is either the return value or
312 * the exception which was thrown. In this case, we always return false and
313 * raise the exception ourselves.
314 */
315JSBool JavaObject::call(JSContext* ctx, JSObject*, uintN argc, jsval* argv,
316 jsval* rval) {
317 // Get the JavaObject called as a function
318 JSObject* obj = JSVAL_TO_OBJECT(argv[-2]);
319 if (argc < 2 || !JSVAL_IS_INT(argv[0]) || !JSVAL_IS_OBJECT(argv[1])) {
320 Debug::log(Debug::Error) << "JavaObject::call incorrect arguments" << Debug::flush;
321 return JS_FALSE;
322 }
323 int dispId = JSVAL_TO_INT(argv[0]);
324 if (Debug::level(Debug::Spam)) {
325 Debug::DebugStream& dbg = Debug::log(Debug::Spam) << "JavaObject::call oid="
326 << JavaObject::getObjectId(ctx, obj) << ",dispId=" << dispId << " (";
327 for (unsigned i = 2; i < argc; ++i) {
328 if (i > 2) {
329 dbg << ", ";
330 }
331 dbg << dumpJsVal(ctx, argv[i]);
332 }
333 dbg << ")" << Debug::flush;
334 }
335
336 SessionData* data = JavaObject::getSessionData(ctx, obj);
337 if (!data) {
338 *rval = JSVAL_VOID;
339 return JS_TRUE;
340 }
341 Debug::log(Debug::Spam) << "Data = " << data << Debug::flush;
342
343 Value javaThis;
344 if (!JSVAL_IS_NULL(argv[1])) {
345 JSObject* thisObj = JSVAL_TO_OBJECT(argv[1]);
346 if (isJavaObject(ctx, thisObj)) {
347 javaThis.setJavaObject(getObjectId(ctx, thisObj));
348 } else {
349 data->makeValueFromJsval(javaThis, ctx, argv[1]);
350 }
351 } else {
352 int oid = getObjectId(ctx, obj);
353 javaThis.setJavaObject(oid);
354 }
355 return invokeJava(ctx, data, javaThis, dispId, argc - 2, &argv[2], rval);
356}
357
conroy@google.com36cf7482011-01-12 19:43:57 +0000358#if GECKO_VERSION >= 2000
359/**
360 * Compatability wrapper for Gecko 2.0+
361 */
362JSBool JavaObject::toString20(JSContext* ctx, uintN argc, jsval* vp) {
363 jsval rval = JSVAL_VOID;
364 JSBool success;
365 success = toString(ctx, JS_THIS_OBJECT(ctx, vp), argc, JS_ARGV(ctx, vp),
366 &rval);
367 JS_SET_RVAL(cx, vp, rval);
368 return success;
369}
370
371/**
372 * Compatability wrapper method for Gecko 2.0+
373 */
374JSBool JavaObject::call20(JSContext* ctx, uintN argc, jsval* vp) {
375 jsval rval = JSVAL_VOID;
376 JSBool success;
377 success = call(ctx, NULL, argc, JS_ARGV(ctx, vp), &rval);
378 JS_SET_RVAL(ctx, vp, rval);
379 return success;
380}
381#endif //GECKO_VERSION
382
jat@google.com134be542009-08-03 15:30:11 +0000383/**
384 * Calls a method on a Java object and returns a two-element JS array, with
385 * the first element being a boolean flag indicating an exception was thrown,
386 * and the second element is the actual return value or exception.
387 */
388JSBool JavaObject::invokeJava(JSContext* ctx, SessionData* data,
389 const Value& javaThis, int dispId, int numArgs, const jsval* jsargs,
390 jsval* rval) {
391 HostChannel* channel = data->getHostChannel();
392 SessionHandler* handler = data->getSessionHandler();
393 scoped_array<Value> args(new Value[numArgs]);
394 for (int i = 0; i < numArgs; ++i) {
395 data->makeValueFromJsval(args[i], ctx, jsargs[i]);
396 }
scottb@google.comb0dbdff2009-11-23 21:18:40 +0000397
398 bool isException = false;
399 Value returnValue;
jat@google.com134be542009-08-03 15:30:11 +0000400 if (!InvokeMessage::send(*channel, javaThis, dispId, numArgs, args.get())) {
jat@google.com5e86cbd2009-08-22 23:59:24 +0000401 Debug::log(Debug::Debugging) << "JavaObject::call failed to send invoke message" << Debug::flush;
scottb@google.comb0dbdff2009-11-23 21:18:40 +0000402 } else {
403 Debug::log(Debug::Spam) << " return from invoke" << Debug::flush;
404 scoped_ptr<ReturnMessage> retMsg(channel->reactToMessagesWhileWaitingForReturn(handler));
405 if (!retMsg.get()) {
406 Debug::log(Debug::Debugging) << "JavaObject::call failed to get return value" << Debug::flush;
407 } else {
408 isException = retMsg->isException();
409 returnValue = retMsg->getReturnValue();
410 }
jat@google.com134be542009-08-03 15:30:11 +0000411 }
jat@google.com134be542009-08-03 15:30:11 +0000412 // Since we can set exceptions normally, we always return false to the
413 // wrapper function and set the exception ourselves if one occurs.
414 // TODO: cleanup exception case
415 jsval retvalArray[] = {JSVAL_FALSE, JSVAL_VOID};
416 JSObject* retval = JS_NewArrayObject(ctx, 2, retvalArray);
417 *rval = OBJECT_TO_JSVAL(retval);
418 jsval retJsVal;
419 Debug::log(Debug::Spam) << " result is " << returnValue << Debug::flush;
420 data->makeJsvalFromValue(retJsVal, ctx, returnValue);
scottb@google.comb0dbdff2009-11-23 21:18:40 +0000421 if (isException) {
jat@google.com134be542009-08-03 15:30:11 +0000422 JS_SetPendingException(ctx, retJsVal);
423 return false;
424 }
425 if (!JS_SetElement(ctx, retval, 1, &retJsVal)) {
426 Debug::log(Debug::Error) << "Error setting return value element in array"
427 << Debug::flush;
428 return false;
429 }
430 return true;
431}