jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 1 | /* |
| 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 | |
| 28 | static 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 44 | #if GECKO_VERSION < 2000 |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 45 | JavaObject::call, /* call */ //TODO |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 46 | #else |
| 47 | JavaObject::call20, /* call: compatability wrapper for 2.0+ */ |
| 48 | #endif //GECKO_VERSION |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 49 | NULL, /* construct */ |
| 50 | NULL, /* object serialization */ |
| 51 | NULL, /* has instance */ |
| 52 | NULL, /* mark */ |
| 53 | NULL /* reserve slots */ |
| 54 | }; |
| 55 | |
| 56 | int JavaObject::getObjectId(JSContext* ctx, JSObject* obj) { |
| 57 | jsval val; |
| 58 | JSClass* jsClass = JS_GET_CLASS(ctx, obj); |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 59 | 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.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 72 | 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 | |
| 80 | SessionData* JavaObject::getSessionData(JSContext* ctx, JSObject* obj) { |
| 81 | void* data = JS_GetInstancePrivate(ctx, obj, &JavaObjectClass, NULL); |
| 82 | return static_cast<SessionData*>(data); |
| 83 | } |
| 84 | |
| 85 | |
| 86 | bool JavaObject::isJavaObject(JSContext* ctx, JSObject* obj) { |
| 87 | return JS_GET_CLASS(ctx, obj) == &JavaObjectClass; |
| 88 | } |
| 89 | |
| 90 | JSObject* 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 109 | #if GECKO_VERSION < 2000 |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 110 | 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 114 | #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.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 120 | return obj; |
| 121 | } |
| 122 | |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 123 | JSBool JavaObject::getProperty(JSContext* ctx, JSObject* obj, jsid id, |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 124 | 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 133 | 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.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 137 | *rval = data->getToStringTearOff(); |
| 138 | return JS_TRUE; |
| 139 | } |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 140 | if ((JS_GetStringEncodingLength(ctx, str) == 2) && !strncmp("id", |
| 141 | JS_EncodeString(ctx, str), 2)) { |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 142 | *rval = INT_TO_JSVAL(objectRef); |
| 143 | return JS_TRUE; |
| 144 | } |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 145 | if ((JS_GetStringEncodingLength(ctx, str) == 16) && !strncmp("__noSuchMethod__", |
| 146 | JS_EncodeString(ctx, str), 16)) { |
jat@google.com | 5e86cbd | 2009-08-22 23:59:24 +0000 | [diff] [blame] | 147 | // Avoid error spew if we are disconnected |
| 148 | *rval = JSVAL_VOID; |
| 149 | return JS_TRUE; |
| 150 | } |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 151 | 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 156 | if (!JSID_IS_INT(id)) { |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 157 | 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 162 | int dispId = JSID_TO_INT(id); |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 163 | |
| 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 172 | JSBool JavaObject::setProperty(JSContext* ctx, JSObject* obj, jsid id, |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 173 | jsval* vp) { |
| 174 | Debug::log(Debug::Spam) << "JavaObject::setProperty obj=" << obj << Debug::flush; |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 175 | if (!JSID_IS_INT(id)) { |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 176 | 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 187 | int dispId = JSID_TO_INT(id); |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 188 | |
| 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? |
| 203 | JSBool 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 | */ |
| 227 | static const char* propertyNames[] = { |
| 228 | "toString", |
| 229 | "id", |
| 230 | }; |
| 231 | #define NUM_PROPERTY_NAMES (sizeof(propertyNames) / sizeof(propertyNames[0])) |
| 232 | |
| 233 | JSBool 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 242 | *idp = INT_TO_JSID(NUM_PROPERTY_NAMES); |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 243 | } |
| 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 253 | #if GECKO_VERSION < 2000 |
| 254 | //TODO(jat): do we need to do this? |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 255 | *idp = JSVAL_NULL; |
conroy@google.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 256 | #endif //GECKO_VERSION |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 257 | } 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 | |
| 277 | void JavaObject::finalize(JSContext* ctx, JSObject* obj) { |
scottb@google.com | 12c8f15 | 2009-12-01 00:12:57 +0000 | [diff] [blame] | 278 | Debug::log(Debug::Spam) << "JavaObject::finalize obj=" << obj |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 279 | << " 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 | |
| 288 | JSBool 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 | */ |
| 315 | JSBool 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.com | 36cf748 | 2011-01-12 19:43:57 +0000 | [diff] [blame] | 358 | #if GECKO_VERSION >= 2000 |
| 359 | /** |
| 360 | * Compatability wrapper for Gecko 2.0+ |
| 361 | */ |
| 362 | JSBool 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 | */ |
| 374 | JSBool 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.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 383 | /** |
| 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 | */ |
| 388 | JSBool 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.com | b0dbdff | 2009-11-23 21:18:40 +0000 | [diff] [blame] | 397 | |
| 398 | bool isException = false; |
| 399 | Value returnValue; |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 400 | if (!InvokeMessage::send(*channel, javaThis, dispId, numArgs, args.get())) { |
jat@google.com | 5e86cbd | 2009-08-22 23:59:24 +0000 | [diff] [blame] | 401 | Debug::log(Debug::Debugging) << "JavaObject::call failed to send invoke message" << Debug::flush; |
scottb@google.com | b0dbdff | 2009-11-23 21:18:40 +0000 | [diff] [blame] | 402 | } 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.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 411 | } |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 412 | // 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.com | b0dbdff | 2009-11-23 21:18:40 +0000 | [diff] [blame] | 421 | if (isException) { |
jat@google.com | 134be54 | 2009-08-03 15:30:11 +0000 | [diff] [blame] | 422 | 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 | } |