1 /* 2 * Copyright (c) 1999, 2012, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. Oracle designates this 8 * particular file as subject to the "Classpath" exception as provided 9 * by Oracle in the LICENSE file that accompanied this code. 10 * 11 * This code is distributed in the hope that it will be useful, but WITHOUT 12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 14 * version 2 for more details (a copy is included in the LICENSE file that 15 * accompanied this code). 16 * 17 * You should have received a copy of the GNU General Public License version 18 * 2 along with this work; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 20 * 21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 22 * or visit www.oracle.com if you need additional information or have any 23 * questions. 24 */ 25 /* 26 * Licensed Materials - Property of IBM 27 * RMI-IIOP v1.0 28 * Copyright IBM Corp. 1998 1999 All Rights Reserved 29 * 30 */ 31 32 package com.sun.corba.se.impl.javax.rmi.CORBA; // Util (sed marker, don't remove!) 33 34 import java.rmi.RemoteException; 35 import java.rmi.UnexpectedException; 36 import java.rmi.MarshalException; 37 38 import java.rmi.server.RMIClassLoader; 39 40 import java.util.Hashtable; 41 import java.util.Enumeration; 42 import java.util.Properties; 43 import java.util.Map; 44 import java.util.WeakHashMap; 45 46 import java.io.Serializable; 47 import java.io.NotSerializableException; 48 49 import java.lang.reflect.Constructor; 50 51 import javax.rmi.CORBA.ValueHandler; 52 import javax.rmi.CORBA.Tie; 53 54 import java.security.AccessController; 55 import java.security.PrivilegedAction; 56 57 import java.rmi.MarshalException; 58 import java.rmi.NoSuchObjectException; 59 import java.rmi.AccessException; 60 import java.rmi.Remote; 61 import java.rmi.ServerError; 62 import java.rmi.ServerException; 63 import java.rmi.ServerRuntimeException; 64 65 import javax.transaction.TransactionRequiredException; 66 import javax.transaction.TransactionRolledbackException; 67 import javax.transaction.InvalidTransactionException; 68 69 import org.omg.CORBA.SystemException; 70 import org.omg.CORBA.Any; 71 import org.omg.CORBA.TypeCode; 72 import org.omg.CORBA.COMM_FAILURE; 73 import org.omg.CORBA.BAD_PARAM; 74 import org.omg.CORBA.INV_OBJREF; 75 import org.omg.CORBA.NO_PERMISSION; 76 import org.omg.CORBA.MARSHAL; 77 import org.omg.CORBA.OBJECT_NOT_EXIST; 78 import org.omg.CORBA.TRANSACTION_REQUIRED; 79 import org.omg.CORBA.TRANSACTION_ROLLEDBACK; 80 import org.omg.CORBA.INVALID_TRANSACTION; 81 import org.omg.CORBA.BAD_OPERATION; 82 import org.omg.CORBA.ACTIVITY_REQUIRED; 83 import org.omg.CORBA.ACTIVITY_COMPLETED; 84 import org.omg.CORBA.INVALID_ACTIVITY; 85 import org.omg.CORBA.CompletionStatus; 86 import org.omg.CORBA.TCKind; 87 import org.omg.CORBA.portable.UnknownException; 88 import org.omg.CORBA.portable.InputStream; 89 import org.omg.CORBA.portable.OutputStream; 90 91 // This class must be able to function with non-Sun ORBs. 92 // This means that any of the following com.sun.corba classes 93 // must only occur in contexts that also handle the non-Sun case. 94 95 import com.sun.corba.se.pept.transport.ContactInfoList ; 96 import com.sun.corba.se.spi.orb.ORB; 97 import com.sun.corba.se.spi.orb.ORBVersionFactory; 98 import com.sun.corba.se.spi.protocol.CorbaClientDelegate; 99 import com.sun.corba.se.spi.transport.CorbaContactInfoList ; 100 import com.sun.corba.se.spi.protocol.LocalClientRequestDispatcher ; 101 import com.sun.corba.se.spi.copyobject.ReflectiveCopyException ; 102 import com.sun.corba.se.spi.copyobject.CopierManager ; 103 import com.sun.corba.se.spi.copyobject.ObjectCopierFactory ; 104 import com.sun.corba.se.spi.copyobject.ObjectCopier ; 105 import com.sun.corba.se.impl.io.ValueHandlerImpl; 106 import com.sun.corba.se.impl.orbutil.ORBConstants; 107 import com.sun.corba.se.impl.orbutil.ORBUtility; 108 import com.sun.corba.se.impl.logging.OMGSystemException; 109 import com.sun.corba.se.impl.util.Utility; 110 import com.sun.corba.se.impl.util.IdentityHashtable; 111 import com.sun.corba.se.impl.util.JDKBridge; 112 import com.sun.corba.se.impl.orbutil.ORBClassLoader; 113 import com.sun.corba.se.impl.logging.UtilSystemException; 114 import com.sun.corba.se.spi.logging.CORBALogDomains; 115 116 /** 117 * Provides utility methods that can be used by stubs and ties to 118 * perform common operations. 119 */ 120 public class Util implements javax.rmi.CORBA.UtilDelegate 121 { 122 // Runs as long as there are exportedServants 123 private static KeepAlive keepAlive = null; 124 125 // Maps targets to ties. 126 private static IdentityHashtable exportedServants = new IdentityHashtable(); 127 128 private static ValueHandlerImpl valueHandlerSingleton = new ValueHandlerImpl(); 129 130 private UtilSystemException utilWrapper = UtilSystemException.get( 131 CORBALogDomains.RPC_ENCODING); 132 133 private static Util instance = null; 134 135 public Util() { 136 setInstance(this); 137 } 138 139 private static void setInstance( Util util ) { 140 assert instance == null : "Instance already defined"; 141 instance = util; 142 } 143 144 public static Util getInstance() { 145 return instance; 146 } 147 148 public static boolean isInstanceDefined() { 149 return instance != null; 150 } 151 152 // Used by TOAFactory.shutdown to unexport all targets for this 153 // particular ORB. This happens during ORB shutdown. 154 public void unregisterTargetsForORB(org.omg.CORBA.ORB orb) 155 { 156 for (Enumeration e = exportedServants.keys(); e.hasMoreElements(); ) 157 { 158 java.lang.Object key = e.nextElement(); 159 Remote target = (Remote)(key instanceof Tie ? ((Tie)key).getTarget() : key); 160 161 // Bug 4476347: BAD_OPERATION is thrown if the ties delegate isn't set. 162 // We can ignore this because it means the tie is not connected to an ORB. 163 try { 164 if (orb == getTie(target).orb()) { 165 try { 166 unexportObject(target); 167 } catch( java.rmi.NoSuchObjectException ex ) { 168 // We neglect this exception if at all if it is 169 // raised. It is not harmful. 170 } 171 } 172 } catch (BAD_OPERATION bad) { 173 /* Ignore */ 174 } 175 } 176 } 177 178 /** 179 * Maps a SystemException to a RemoteException. 180 * @param ex the SystemException to map. 181 * @return the mapped exception. 182 */ 183 public RemoteException mapSystemException(SystemException ex) 184 { 185 if (ex instanceof UnknownException) { 186 Throwable orig = ((UnknownException)ex).originalEx; 187 if (orig instanceof Error) { 188 return new ServerError("Error occurred in server thread",(Error)orig); 189 } else if (orig instanceof RemoteException) { 190 return new ServerException("RemoteException occurred in server thread", 191 (Exception)orig); 192 } else if (orig instanceof RuntimeException) { 193 throw (RuntimeException) orig; 194 } 195 } 196 197 // Build the message string... 198 String name = ex.getClass().getName(); 199 String corbaName = name.substring(name.lastIndexOf('.')+1); 200 String status; 201 switch (ex.completed.value()) { 202 case CompletionStatus._COMPLETED_YES: 203 status = "Yes"; 204 break; 205 case CompletionStatus._COMPLETED_NO: 206 status = "No"; 207 break; 208 case CompletionStatus._COMPLETED_MAYBE: 209 default: 210 status = "Maybe"; 211 break; 212 } 213 214 String message = "CORBA " + corbaName + " " + ex.minor + " " + status; 215 216 // Now map to the correct RemoteException type... 217 if (ex instanceof COMM_FAILURE) { 218 return new MarshalException(message, ex); 219 } else if (ex instanceof INV_OBJREF) { 220 RemoteException newEx = new NoSuchObjectException(message); 221 newEx.detail = ex; 222 return newEx; 223 } else if (ex instanceof NO_PERMISSION) { 224 return new AccessException(message, ex); 225 } else if (ex instanceof MARSHAL) { 226 return new MarshalException(message, ex); 227 } else if (ex instanceof OBJECT_NOT_EXIST) { 228 RemoteException newEx = new NoSuchObjectException(message); 229 newEx.detail = ex; 230 return newEx; 231 } else if (ex instanceof TRANSACTION_REQUIRED) { 232 RemoteException newEx = new TransactionRequiredException(message); 233 newEx.detail = ex; 234 return newEx; 235 } else if (ex instanceof TRANSACTION_ROLLEDBACK) { 236 RemoteException newEx = new TransactionRolledbackException(message); 237 newEx.detail = ex; 238 return newEx; 239 } else if (ex instanceof INVALID_TRANSACTION) { 240 RemoteException newEx = new InvalidTransactionException(message); 241 newEx.detail = ex; 242 return newEx; 243 } else if (ex instanceof BAD_PARAM) { 244 Exception inner = ex; 245 246 // Pre-Merlin Sun ORBs used the incorrect minor code for 247 // this case. See Java to IDL ptc-00-01-08 1.4.8. 248 if (ex.minor == ORBConstants.LEGACY_SUN_NOT_SERIALIZABLE || 249 ex.minor == OMGSystemException.NOT_SERIALIZABLE) { 250 251 if (ex.getMessage() != null) 252 inner = new NotSerializableException(ex.getMessage()); 253 else 254 inner = new NotSerializableException(); 255 256 inner.initCause( ex ) ; 257 } 258 259 return new MarshalException(message,inner); 260 } else if (ex instanceof ACTIVITY_REQUIRED) { 261 try { 262 Class cl = ORBClassLoader.loadClass( 263 "javax.activity.ActivityRequiredException"); 264 Class[] params = new Class[2]; 265 params[0] = java.lang.String.class; 266 params[1] = java.lang.Throwable.class; 267 Constructor cr = cl.getConstructor(params); 268 Object[] args = new Object[2]; 269 args[0] = message; 270 args[1] = ex; 271 return (RemoteException) cr.newInstance(args); 272 } catch (Throwable e) { 273 utilWrapper.classNotFound( 274 e, "javax.activity.ActivityRequiredException"); 275 } 276 } else if (ex instanceof ACTIVITY_COMPLETED) { 277 try { 278 Class cl = ORBClassLoader.loadClass( 279 "javax.activity.ActivityCompletedException"); 280 Class[] params = new Class[2]; 281 params[0] = java.lang.String.class; 282 params[1] = java.lang.Throwable.class; 283 Constructor cr = cl.getConstructor(params); 284 Object[] args = new Object[2]; 285 args[0] = message; 286 args[1] = ex; 287 return (RemoteException) cr.newInstance(args); 288 } catch (Throwable e) { 289 utilWrapper.classNotFound( 290 e, "javax.activity.ActivityCompletedException"); 291 } 292 } else if (ex instanceof INVALID_ACTIVITY) { 293 try { 294 Class cl = ORBClassLoader.loadClass( 295 "javax.activity.InvalidActivityException"); 296 Class[] params = new Class[2]; 297 params[0] = java.lang.String.class; 298 params[1] = java.lang.Throwable.class; 299 Constructor cr = cl.getConstructor(params); 300 Object[] args = new Object[2]; 301 args[0] = message; 302 args[1] = ex; 303 return (RemoteException) cr.newInstance(args); 304 } catch (Throwable e) { 305 utilWrapper.classNotFound( 306 e, "javax.activity.InvalidActivityException"); 307 } 308 } 309 310 // Just map to a generic RemoteException... 311 return new RemoteException(message, ex); 312 } 313 314 /** 315 * Writes any java.lang.Object as a CORBA any. 316 * @param out the stream in which to write the any. 317 * @param obj the object to write as an any. 318 */ 319 public void writeAny( org.omg.CORBA.portable.OutputStream out, 320 java.lang.Object obj) 321 { 322 org.omg.CORBA.ORB orb = out.orb(); 323 324 // Create Any 325 Any any = orb.create_any(); 326 327 // Make sure we have a connected object... 328 java.lang.Object newObj = Utility.autoConnect(obj,orb,false); 329 330 if (newObj instanceof org.omg.CORBA.Object) { 331 any.insert_Object((org.omg.CORBA.Object)newObj); 332 } else { 333 if (newObj == null) { 334 // Handle the null case, including backwards 335 // compatibility issues 336 any.insert_Value(null, createTypeCodeForNull(orb)); 337 } else { 338 if (newObj instanceof Serializable) { 339 // If they're our Any and ORB implementations, 340 // we may want to do type code related versioning. 341 TypeCode tc = createTypeCode((Serializable)newObj, any, orb); 342 if (tc == null) 343 any.insert_Value((Serializable)newObj); 344 else 345 any.insert_Value((Serializable)newObj, tc); 346 } else if (newObj instanceof Remote) { 347 ORBUtility.throwNotSerializableForCorba(newObj.getClass().getName()); 348 } else { 349 ORBUtility.throwNotSerializableForCorba(newObj.getClass().getName()); 350 } 351 } 352 } 353 354 out.write_any(any); 355 } 356 357 /** 358 * When using our own ORB and Any implementations, we need to get 359 * the ORB version and create the type code appropriately. This is 360 * to overcome a bug in which the JDK 1.3.x ORBs used a tk_char 361 * rather than a tk_wchar to describe a Java char field. 362 * 363 * This only works in RMI-IIOP with Util.writeAny since we actually 364 * know what ORB and stream we're writing with when we insert 365 * the value. 366 * 367 * Returns null if it wasn't possible to create the TypeCode (means 368 * it wasn't our ORB or Any implementation). 369 * 370 * This does not handle null objs. 371 */ 372 private TypeCode createTypeCode(Serializable obj, 373 org.omg.CORBA.Any any, 374 org.omg.CORBA.ORB orb) { 375 376 if (any instanceof com.sun.corba.se.impl.corba.AnyImpl && 377 orb instanceof ORB) { 378 379 com.sun.corba.se.impl.corba.AnyImpl anyImpl 380 = (com.sun.corba.se.impl.corba.AnyImpl)any; 381 382 ORB ourORB = (ORB)orb; 383 384 return anyImpl.createTypeCodeForClass(obj.getClass(), ourORB); 385 386 } else 387 return null; 388 } 389 390 391 /** 392 * This is used to create the TypeCode for a null reference. 393 * It also handles backwards compatibility with JDK 1.3.x. 394 * 395 * This method will not return null. 396 */ 397 private TypeCode createTypeCodeForNull(org.omg.CORBA.ORB orb) 398 { 399 if (orb instanceof ORB) { 400 401 ORB ourORB = (ORB)orb; 402 403 // Preserve backwards compatibility with Kestrel and Ladybird 404 // by not fully implementing interop issue resolution 3857, 405 // and returning a null TypeCode with a tk_value TCKind. 406 // If we're not talking to Kestrel or Ladybird, fall through 407 // to the abstract interface case (also used for foreign ORBs). 408 if (!ORBVersionFactory.getFOREIGN().equals(ourORB.getORBVersion()) && 409 ORBVersionFactory.getNEWER().compareTo(ourORB.getORBVersion()) > 0) { 410 411 return orb.get_primitive_tc(TCKind.tk_value); 412 } 413 } 414 415 // Use tk_abstract_interface as detailed in the resolution 416 417 // REVISIT: Define this in IDL and get the ID in generated code 418 String abstractBaseID = "IDL:omg.org/CORBA/AbstractBase:1.0"; 419 420 return orb.create_abstract_interface_tc(abstractBaseID, ""); 421 } 422 423 /** 424 * Reads a java.lang.Object as a CORBA any. 425 * @param in the stream from which to read the any. 426 * @return the object read from the stream. 427 */ 428 public Object readAny(InputStream in) 429 { 430 Any any = in.read_any(); 431 if ( any.type().kind().value() == TCKind._tk_objref ) 432 return any.extract_Object (); 433 else 434 return any.extract_Value(); 435 } 436 437 /** 438 * Writes a java.lang.Object as a CORBA Object. If <code>obj</code> is 439 * an exported RMI-IIOP server object, the tie is found 440 * and wired to <code>obj</code>, then written to <code>out.write_Object(org.omg.CORBA.Object)</code>. 441 * If <code>obj</code> is a CORBA Object, it is written to 442 * <code>out.write_Object(org.omg.CORBA.Object)</code>. 443 * @param out the stream in which to write the object. 444 * @param obj the object to write. 445 */ 446 public void writeRemoteObject(OutputStream out, java.lang.Object obj) 447 { 448 // Make sure we have a connected object, then 449 // write it out... 450 451 Object newObj = Utility.autoConnect(obj,out.orb(),false); 452 out.write_Object((org.omg.CORBA.Object)newObj); 453 } 454 455 /** 456 * Writes a java.lang.Object as either a value or a CORBA Object. 457 * If <code>obj</code> is a value object or a stub object, it is written to 458 * <code>out.write_abstract_interface(java.lang.Object)</code>. If <code>obj</code> is an exported 459 * RMI-IIOP server object, the tie is found and wired to <code>obj</code>, 460 * then written to <code>out.write_abstract_interface(java.lang.Object)</code>. 461 * @param out the stream in which to write the object. 462 * @param obj the object to write. 463 */ 464 public void writeAbstractObject( OutputStream out, java.lang.Object obj ) 465 { 466 // Make sure we have a connected object, then 467 // write it out... 468 469 Object newObj = Utility.autoConnect(obj,out.orb(),false); 470 ((org.omg.CORBA_2_3.portable.OutputStream)out).write_abstract_interface(newObj); 471 } 472 473 /** 474 * Registers a target for a tie. Adds the tie to an internal table and calls 475 * {@link Tie#setTarget} on the tie object. 476 * @param tie the tie to register. 477 * @param target the target for the tie. 478 */ 479 public void registerTarget(javax.rmi.CORBA.Tie tie, java.rmi.Remote target) 480 { 481 synchronized (exportedServants) { 482 // Do we already have this target registered? 483 if (lookupTie(target) == null) { 484 // No, so register it and set the target... 485 exportedServants.put(target,tie); 486 tie.setTarget(target); 487 488 // Do we need to instantiate our keep-alive thread? 489 if (keepAlive == null) { 490 // Yes. Instantiate our keep-alive thread and start 491 // it up... 492 keepAlive = (KeepAlive)AccessController.doPrivileged(new PrivilegedAction() { 493 public java.lang.Object run() { 494 return new KeepAlive(); 495 } 496 }); 497 keepAlive.start(); 498 } 499 } 500 } 501 } 502 503 /** 504 * Removes the associated tie from an internal table and calls {@link Tie#deactivate} 505 * to deactivate the object. 506 * @param target the object to unexport. 507 */ 508 public void unexportObject(java.rmi.Remote target) 509 throws java.rmi.NoSuchObjectException 510 { 511 synchronized (exportedServants) { 512 Tie cachedTie = lookupTie(target); 513 if (cachedTie != null) { 514 exportedServants.remove(target); 515 Utility.purgeStubForTie(cachedTie); 516 Utility.purgeTieAndServant(cachedTie); 517 try { 518 cleanUpTie(cachedTie); 519 } catch (BAD_OPERATION e) { 520 // ignore 521 } catch (org.omg.CORBA.OBJ_ADAPTER e) { 522 // This can happen when the target was never associated with a POA. 523 // We can safely ignore this case. 524 } 525 526 // Is it time to shut down our keep alive thread? 527 if (exportedServants.isEmpty()) { 528 keepAlive.quit(); 529 keepAlive = null; 530 } 531 } else { 532 throw new java.rmi.NoSuchObjectException("Tie not found" ); 533 } 534 } 535 } 536 537 protected void cleanUpTie(Tie cachedTie) 538 throws java.rmi.NoSuchObjectException 539 { 540 cachedTie.setTarget(null); 541 cachedTie.deactivate(); 542 } 543 544 /** 545 * Returns the tie (if any) for a given target object. 546 * @return the tie or null if no tie is registered for the given target. 547 */ 548 public Tie getTie (Remote target) 549 { 550 synchronized (exportedServants) { 551 return lookupTie(target); 552 } 553 } 554 555 /** 556 * An unsynchronized version of getTie() for internal use. 557 */ 558 private static Tie lookupTie (Remote target) 559 { 560 Tie result = (Tie)exportedServants.get(target); 561 if (result == null && target instanceof Tie) { 562 if (exportedServants.contains(target)) { 563 result = (Tie)target; 564 } 565 } 566 return result; 567 } 568 569 /** 570 * Returns a singleton instance of a class that implements the 571 * {@link ValueHandler} interface. 572 * @return a class which implements the ValueHandler interface. 573 */ 574 public ValueHandler createValueHandler() 575 { 576 return valueHandlerSingleton; 577 } 578 579 /** 580 * Returns the codebase, if any, for the given class. 581 * @param clz the class to get a codebase for. 582 * @return a space-separated list of URLs, or null. 583 */ 584 public String getCodebase(java.lang.Class clz) { 585 return RMIClassLoader.getClassAnnotation(clz); 586 } 587 588 /** 589 * Returns a class instance for the specified class. 590 * @param className the name of the class. 591 * @param remoteCodebase a space-separated list of URLs at which 592 * the class might be found. May be null. 593 * @param loadingContext a class whose ClassLoader may be used to 594 * load the class if all other methods fail. 595 * @return the <code>Class</code> object representing the loaded class. 596 * @exception ClassNotFoundException if class cannot be loaded. 597 */ 598 public Class loadClass( String className, String remoteCodebase, 599 ClassLoader loader) throws ClassNotFoundException 600 { 601 return JDKBridge.loadClass(className,remoteCodebase,loader); 602 } 603 604 /** 605 * The <tt>isLocal</tt> method has the same semantics as the 606 * ObjectImpl._is_local method, except that it can throw a RemoteException. 607 * (no it doesn't but the spec says it should.) 608 * 609 * The <tt>_is_local()</tt> method is provided so that stubs may determine 610 * if a particular object is implemented by a local servant and hence local 611 * invocation APIs may be used. 612 * 613 * @param stub the stub to test. 614 * 615 * @return The <tt>_is_local()</tt> method returns true if 616 * the servant incarnating the object is located in the same process as 617 * the stub and they both share the same ORB instance. The <tt>_is_local()</tt> 618 * method returns false otherwise. The default behavior of <tt>_is_local()</tt> is 619 * to return false. 620 * 621 * @throws RemoteException The Java to IDL specification does to 622 * specify the conditions that cause a RemoteException to be thrown. 623 */ 624 public boolean isLocal(javax.rmi.CORBA.Stub stub) throws RemoteException 625 { 626 boolean result = false ; 627 628 try { 629 org.omg.CORBA.portable.Delegate delegate = stub._get_delegate() ; 630 if (delegate instanceof CorbaClientDelegate) { 631 // For the Sun ORB 632 CorbaClientDelegate cdel = (CorbaClientDelegate)delegate ; 633 ContactInfoList cil = cdel.getContactInfoList() ; 634 if (cil instanceof CorbaContactInfoList) { 635 CorbaContactInfoList ccil = (CorbaContactInfoList)cil ; 636 LocalClientRequestDispatcher lcs = ccil.getLocalClientRequestDispatcher() ; 637 result = lcs.useLocalInvocation( null ) ; 638 } 639 } else { 640 // For a non-Sun ORB 641 result = delegate.is_local( stub ) ; 642 } 643 } catch (SystemException e) { 644 throw javax.rmi.CORBA.Util.mapSystemException(e); 645 } 646 647 return result ; 648 } 649 650 /** 651 * Wraps an exception thrown by an implementation 652 * method. It returns the corresponding client-side exception. 653 * @param orig the exception to wrap. 654 * @return the wrapped exception. 655 */ 656 public RemoteException wrapException(Throwable orig) 657 { 658 if (orig instanceof SystemException) { 659 return mapSystemException((SystemException)orig); 660 } 661 662 if (orig instanceof Error) { 663 return new ServerError("Error occurred in server thread",(Error)orig); 664 } else if (orig instanceof RemoteException) { 665 return new ServerException("RemoteException occurred in server thread", 666 (Exception)orig); 667 } else if (orig instanceof RuntimeException) { 668 throw (RuntimeException) orig; 669 } 670 671 if (orig instanceof Exception) 672 return new UnexpectedException( orig.toString(), (Exception)orig ); 673 else 674 return new UnexpectedException( orig.toString()); 675 } 676 677 /** 678 * Copies or connects an array of objects. Used by local stubs 679 * to copy any number of actual parameters, preserving sharing 680 * across parameters as necessary to support RMI semantics. 681 * @param obj the objects to copy or connect. 682 * @param orb the ORB. 683 * @return the copied or connected objects. 684 * @exception RemoteException if any object could not be copied or connected. 685 */ 686 public Object[] copyObjects (Object[] obj, org.omg.CORBA.ORB orb) 687 throws RemoteException 688 { 689 if (obj == null) 690 // Bug fix for 5018613: JCK test expects copyObjects to throw 691 // NPE when obj==null. This is actually not in the spec, since 692 // obj is not really an RMI-IDL data type, but we follow our 693 // test here, and force this error to be thrown. 694 throw new NullPointerException() ; 695 696 Class compType = obj.getClass().getComponentType() ; 697 if (Remote.class.isAssignableFrom( compType ) && !compType.isInterface()) { 698 // obj is an array of remote impl types. This 699 // causes problems with stream copier, so we copy 700 // it over to an array of Remotes instead. 701 Remote[] result = new Remote[obj.length] ; 702 System.arraycopy( (Object)obj, 0, (Object)result, 0, obj.length ) ; 703 return (Object[])copyObject( result, orb ) ; 704 } else 705 return (Object[])copyObject( obj, orb ) ; 706 } 707 708 /** 709 * Copies or connects an object. Used by local stubs to copy 710 * an actual parameter, result object, or exception. 711 * @param obj the object to copy. 712 * @param orb the ORB. 713 * @return the copy or connected object. 714 * @exception RemoteException if the object could not be copied or connected. 715 */ 716 public Object copyObject (Object obj, org.omg.CORBA.ORB orb) 717 throws RemoteException 718 { 719 if (orb instanceof ORB) { 720 ORB lorb = (ORB)orb ; 721 722 try { 723 try { 724 // This gets the copier for the current invocation, which was 725 // previously set by preinvoke. 726 return lorb.peekInvocationInfo().getCopierFactory().make().copy( obj ) ; 727 } catch (java.util.EmptyStackException exc) { 728 // copyObject was invoked outside of an invocation, probably by 729 // a test. Get the default copier from the ORB. 730 // XXX should we just make the default copier available directly 731 // and avoid constructing one on each call? 732 CopierManager cm = lorb.getCopierManager() ; 733 ObjectCopier copier = cm.getDefaultObjectCopierFactory().make() ; 734 return copier.copy( obj ) ; 735 } 736 } catch (ReflectiveCopyException exc) { 737 RemoteException rexc = new RemoteException() ; 738 rexc.initCause( exc ) ; 739 throw rexc ; 740 } 741 } else { 742 org.omg.CORBA_2_3.portable.OutputStream out = 743 (org.omg.CORBA_2_3.portable.OutputStream)orb.create_output_stream(); 744 out.write_value((Serializable)obj); 745 org.omg.CORBA_2_3.portable.InputStream in = 746 (org.omg.CORBA_2_3.portable.InputStream)out.create_input_stream(); 747 return in.read_value(); 748 } 749 } 750 } 751 752 class KeepAlive extends Thread 753 { 754 boolean quit = false; 755 756 public KeepAlive () 757 { 758 setDaemon(false); 759 } 760 761 public synchronized void run () 762 { 763 while (!quit) { 764 try { 765 wait(); 766 } catch (InterruptedException e) {} 767 } 768 } 769 770 public synchronized void quit () 771 { 772 quit = true; 773 notifyAll(); 774 } 775 }