/* * Copyright (c) 2001, 2003, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.sun.corba.se.spi.oa ; import org.omg.CORBA.Policy ; import org.omg.PortableInterceptor.ObjectReferenceTemplate ; import org.omg.PortableInterceptor.ObjectReferenceFactory ; import com.sun.corba.se.spi.orb.ORB ; import com.sun.corba.se.spi.oa.OADestroyed ; import com.sun.corba.se.spi.ior.IORTemplate ; // REVISIT: What should the order be? enter/push...pop/exit? /** ObjectAdapter represents the abstract model of an object * adapter that was introduced by ORT. This means that all * object adapters must: * * Other requirements: * *

* Object adapters must also provide mechanisms for: *

* Such mechanisms are all object adapter specific, and so we do not attempt to * create general APIs for these functions here. The object adapter itself * must provide these APIs directly to the user, and they do not affect the rest of the * ORB. This interface basically makes it possible to plug any object adapter into the * ORB and have the OA work propertly with portable interceptors, and also have requests * dispatched properly to the object adapter. *

* The basic function of an ObjectAdapter is to map object IDs to servants and to support * the dispatch operation of the subcontract, which dispatches requests to servants. * This is the purpose of the getInvocationServant method. In addition, ObjectAdapters must be * able to change state gracefully in the presence of executing methods. This * requires the use of the enter/exit methods. Finally, ObjectAdapters often * require access to information about requests. This is accomodated through the * OAInvocationInfo class and the thread local stack maintained by push/pop/peekInvocationInfo * on the ORB. *

* To be useful, this dispatch cycle must be extremely efficient. There are several * scenarios that matter: *

    *
  1. A remote invocation, where the dispatch is handled in the server subcontract.
  2. *
  3. A local invocation, where the dispatch is handled in the client subcontract.
  4. *
  5. A cached local invocation, where the servant is cached when the IOR is established * for the client subcontract, and the dispatch is handled in the client subcontract * to the cached subcontract.
  6. *
*

* Each of these 3 cases is handled a bit differently. On each request, assume as known * ObjectId and ObjectAdapterId, which can be obtained from the object key. * The ObjectAdaptorFactory is available in the subcontract registry, where it is * registered under the subcontract ID. The Subcontract ID is also available in the * object key. *

    *
  1. The remote pattern: *
      *
    1. oa = oaf.find( oaid )
    2. *
    3. oa.enter()
    4. *
    5. info = oa.makeInvocationInfo( oid )
    6. *
    7. info.setOperation( operation )
    8. *
    9. push info
    10. *
    11. oa.getInvocationServant( info )
    12. *
    13. sreq.setExecuteReturnServantInResponseConstructor( true )
    14. *
    15. dispatch to servant
    16. *
    17. oa.returnServant()
    18. *
    19. oa.exit()
    20. *
    21. pop info
    22. *
    *
  2. * REVISIT: Is this the required order for exit/pop? Cna they be nested instead? * Note that getInvocationServant and returnServant may throw exceptions. In such cases, * returnServant, exit, and pop must be called in the correct order. *
  3. The local pattern: *
      *
    1. oa = oaf.find( oaid )
    2. *
    3. oa.enter()
    4. *
    5. info = oa.makeInvocationInfo( oid )
    6. *
    7. info.setOperation( operation )
    8. *
    9. push info
    10. *
    11. oa.getInvocationServant( info )
    12. *
    13. dispatch to servant
    14. *
    15. oa.returnServant()
    16. *
    17. oa.exit()
    18. *
    19. pop info
    20. *
        * * This is the same as the remote case, except that setExecuteReturnServantInResponseConstructor * is not needed (or possible, since there is no server request). *
      1. The fast local pattern: When delegate is constructed, * first extract ObjectKey from IOR in delegate, * then get ObjectId, ObjectAdapterId, and ObjectAdapterFactory (oaf). Then: *
          *
        1. oa = oaf.find( oaid )
        2. *
        3. info = oa.makeInvocationInfo( oid ) (note: no operation!)
        4. *
        5. push info (needed for the correct functioning of getInvocationServant)
        6. *
        7. oa.getInvocationServant( info )
        8. *
        9. pop info *
        * The info instance (which includes the Servant) is cached in the client subcontract. *

        Then, on each invocation:

        *
          *
        1. newinfo = copy of info (clone)
        2. *
        3. info.setOperation( operation )
        4. *
        5. push newinfo
        6. *
        7. oa.enter()
        8. *
        9. dispatch to servant
        10. *
        11. oa.returnServant()
        12. // XXX This is probably wrong: remove it. *
        13. oa.exit()
        14. *
        15. pop info
        16. *
        *
      2. *
      * XXX fast local should not call returnServant: what is correct here? */ public interface ObjectAdapter { //////////////////////////////////////////////////////////////////////////// // Basic methods for supporting interceptors //////////////////////////////////////////////////////////////////////////// /** Returns the ORB associated with this adapter. */ ORB getORB() ; Policy getEffectivePolicy( int type ) ; /** Returns the IOR template of this adapter. The profiles * in this template may be updated only during the AdapterCreated call. * After that call completes, the IOR template must be made immutable. * Note that the server ID, ORB ID, and adapter name are all available * from the IOR template. */ IORTemplate getIORTemplate() ; //////////////////////////////////////////////////////////////////////////// // Methods needed to support ORT. //////////////////////////////////////////////////////////////////////////// /** Return the ID of the AdapterManager for this object adapter. */ int getManagerId() ; /** Return the current state of this object adapter (see * org.omg.PortableInterceptors for states. */ short getState() ; ObjectReferenceTemplate getAdapterTemplate() ; ObjectReferenceFactory getCurrentFactory() ; /** Change the current factory. This may only be called during the * AdapterCreated call. */ void setCurrentFactory( ObjectReferenceFactory factory ) ; //////////////////////////////////////////////////////////////////////////// // Methods required for dispatching to servants //////////////////////////////////////////////////////////////////////////// /** Get the servant corresponding to the given objectId, if this is supported. * This method is only used for models where the servant is an ObjectImpl, * which allows the servant to be used directly as the stub. This allows an object * reference to be replaced by its servant when it is unmarshalled locally. * Such objects are not ORB mediated. */ org.omg.CORBA.Object getLocalServant( byte[] objectId ) ; /** Get the servant for the request given by the parameters. * info must contain a valid objectId in this call. * The servant is set in the InvocationInfo argument that is passed into * this call. * @param info is the InvocationInfo object for the object reference * @exception ForwardException (a runtime exception) is thrown if the request * is to be handled by a different object reference. */ void getInvocationServant( OAInvocationInfo info ) ; /** enter must be called before each request is invoked on a servant. * @exception OADestroyed is thrown when an OA has been destroyed, which * requires a retry in the case where an AdapterActivator is present. */ void enter( ) throws OADestroyed ; /** exit must be called after each request has been completed. If enter * is called, there must always be a corresponding exit. */ void exit( ) ; /** Must be called every time getInvocationServant is called after * the request has completed. */ public void returnServant() ; /** Create an instance of InvocationInfo that is appropriate for this * Object adapter. */ OAInvocationInfo makeInvocationInfo( byte[] objectId ) ; /** Return the most derived interface for the given servant and objectId. */ String[] getInterfaces( Object servant, byte[] objectId ) ; }