Now that we know how to implement a
      simple client, we need to
      implement a server to test it out.
      We have to provide implementations for both the Stock
      and Stock_Factory interfaces,
      and then create an executable that incorporates those implementations.
    
Stock interfaceTo keep things simple, let's implement a Stock
      object with a fixed price.  The constructor will receive all the
      parameters:
    
class Quoter_Stock_i : public POA_Quoter::Stock {
public:
  Quoter_Stock_i (const char *symbol,
                  const char *full_name,
                  CORBA::Double price);
private:
  std::string symbol_;
  std::string full_name_;
  CORBA::Double price_;
};
    In a server, CORBA objects and functions are implemented and represented by programming language data and functions. These programming entities that implement and represent CORBA objects are called servants. Object Adapters link the world of CORBA objects to the world of programming language servants. They provide services for creation of CORBA objects and their object references and for dispatching requests to the appropriate servants.
Notice the name of the base class. TAO implements the CORBA 2.2 specification, which includes the Portable Object Adapter (hence the POA prefix). This new Object Adapter fixes many problems with the previous versions of the CORBA specification, where the so-called Basic Object Adapter was used. Unfortunately, the specification was ambiguous and lead to incompatible (yet compliant) implementations. Code based on the POA, and conforming to the CORBA 2.2 specification, is almost completely portable, the only incompatibilities arising from the names of the generated header files and and other minor things. Those problems can be easily wrapped in helper classes, and the file names can be controlled through IDL compiler options in most cases.
A server application may contain multiple POA instances, but all server applications have at least one POA called the RootPOA.
We have to implement the operations and attributes:
class Quoter_Stock_i : public POA_Quoter::Stock {
public:
  // some details omitted
  char *symbol ();
  char *full_name ();
  CORBA::Double price ();
};
// In the .cpp file:
char *
Quoter_Stock_i::symbol ()
{
  return CORBA::string_dup (this->symbol_.c_str ());
}
    The other attributes and methods are similar, so we don't reproduce them here.
It is important to copy the strings before returning them,
      because the ORB will use CORBA::string_free to release
      them. The rationale is that over the network, the string must be
      copied anyway, hence, the client must be responsible for releasing
      the received string.  When both client and servers are in the same
      address space the ORB can optimize the path and invoke the server
      operation directly, without marshaling or demarshaling.  If the
      client is going to work with both local and remote servers, it
      should always expect to own the string.  In consequence, the server
      implementation must always allocate a copy and return the copy,
      because the server-side must also work identically for local and
      remote clients.  The memory management rules in CORBA are a bit
      subtle, but there are some simple rules to follow:
      
Typing all this code seems tedious. Can't the IDL compiler help with this? After all, it seems as if the method declarations are completely specified! The answer is yes, TAO's IDL compiler can generate empty implementations that you can modify. Simply use the -GI option:
$ $ACE_ROOT/TAO/TAO_IDL/tao_idl -GI Quoter.idlThe empty implementations are generated in the
QuoterI.h and QuoterI.cpp files.
      Be advised that the -GI option overwrites these files
      every time, so it is better to copy your implementation to
      another file.
    
    Our first implementation of the factory will serve only two stocks, "RHAT" and "MSFT":
class Quoter_Stock_Factory_i : public POA_Quoter::Stock_Factory
{
public:
  Quoter_Stock_Factory ();
  Quoter::Stock_ptr get_stock (const char *symbol);
private:
  Quoter_Stock_i rhat_;
  Quoter_Stock_i msft_;
};
    The implementation of the get_stock() method is
      simple, just compare the symbol name and return the appropriate
      object reference:
    
Quoter::Stock_ptr
Quoter_Stock_Factory_i::get_stock (const char *symbol)
{
  if (strcmp (symbol, "RHAT") == 0) {
    return this->rhat_._this();
  } else if (strcmp (symbol, "MSFT") == 0) {
    return this->msft_._this ();
  }
  throw Quoter::Invalid_Stock_Symbol ();
}
    So what is that _this() method?  In the POA mapping
      the client-side stubs and server-side skeletons are not related
      through inheritance.  You must either explicitly activate the
      servant (your implementation object) or use _this() to
      activate it with its default POA.  _this() creates and
      registers a CORBA object under the RootPOA, and returns the created
      object reference for the new object. We will discuss more about
      explicit and implicit activation later, but at this point it is
      important to remove any thoughts about converting pointers to
      servants to object references or vice-versa, it just does not
      work that way.
    
Now that we have all the object implementations in place, we must create the server executable. We start with the ORB initialization:
int main (int argc, char* argv[])
{
  try {
    // First initialize the ORB, that will remove some arguments...
    CORBA::ORB_var orb =
      CORBA::ORB_init (argc, argv,
                       "" /* the ORB name, it can be anything! */);
    On startup, the ORB starts the POA in the {holding state}, where all requests received are not processed until the POA is activated. Meanwhile the requests are stored to some implementation limit. TAO sets this limit to 0, because queueing is a severe source of overhead and unpredictability in real-time systems.
What does this means for us? Well, we have to activate the POA. The process is a bit tedious. First we gain access to the RootPOA:
    CORBA::Object_var poa_object =
      orb->resolve_initial_references ("RootPOA");
    PortableServer::POA_var poa =
      PortableServer::POA::_narrow (poa_object.in ());
    resolve_initial_references() is used to bootstrap
      all kinds of services, like the Naming Service and the Trading
      Service, but it is also used to gain access to other ORB
      interfaces, such as the RootPOA, the Current objects, and the
      Policy Managers.
    
Now that we have gained access to the Root POA, we must obtain its POA manager. The POA managers provide interfaces to activate and deactivate one or more POAs:
    PortableServer::POAManager_var poa_manager =
      poa->the_POAManager ();
    and now we activate the POA:
    poa_manager->activate ();
    The shutdown process is similar to the client side, but now we must also remember to destroy the POA. Putting all the code above together, we get:
int main (int argc, char* argv[])
{
  try {
    // First initialize the ORB, that will remove some arguments...
    CORBA::ORB_var orb =
      CORBA::ORB_init (argc, argv,
                       "" /* the ORB name, it can be anything! */);
    CORBA::Object_var poa_object =
      orb->resolve_initial_references ("RootPOA");
    PortableServer::POA_var poa =
      PortableServer::POA::_narrow (poa_object.in ());
    PortableServer::POAManager_var poa_manager =
      poa->the_POAManager ();
    poa_manager->activate ();
    // The application code goes here!
    // Destroy the POA, waiting until the destruction terminates
    poa->destroy (1, 1);
    orb->destroy ();
  }
  catch (const CORBA::Exception &ex) {
    std::cerr << "CORBA exception raised!" << std::endl;
  }
  return 0;
}
    Now we create an instance of our stock factory implementation
      and activate it using _this()
    
    Quoter_Stock_Factory_i stock_factory_i;
    Quoter::Stock_Factory_var stock_factory =
      stock_factory_i._this ();
    Next we convert the object reference into an IOR string, so it can be used by the client:
    CORBA::String_var ior = orb->object_to_string (stock_factory.in ());
    std::cerr << ior.in () << std::endl;
    There is only one final detail. We must run the ORB event loop to start processing requests from the client:
    orb->run ();
    There are many details that we have left out from this server, such as how to terminate the event loop, how to perform servant memory management, orderly deactivation of servants, and the fact is that it is not very flexible, but we have covered a number of other things, and more importantly we can test the client now!
Flesh out the implementation. You don't have to do it from scratch, as we provide you with the following files: Stock_i.h, Stock_i.cpp, Stock_Factory_i.h Stock_Factory_i.cpp, Quoter.idl and the always useful MPC file
Compare your solution with server.cpp.
To test this application we need a client. We just run both of them:
$ server > ior_file $ client file://ior_file MSFT RHATAlso test invalid symbols!