// This may look like C, but it's really -*- C++ -*- // $Id$ #include "tao/IIOP_Acceptor.h" #include "tao/IIOP_Profile.h" #include "tao/MProfile.h" #include "tao/ORB_Core.h" #include "tao/Server_Strategy_Factory.h" #include "tao/debug.h" #include "tao/RT_Policy_i.h" #include "ace/Auto_Ptr.h" #if !defined(__ACE_INLINE__) #include "tao/IIOP_Acceptor.i" #endif /* __ACE_INLINE__ */ ACE_RCSID(tao, IIOP_Acceptor, "$Id$") #if defined (ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION) template class ACE_Acceptor; template class ACE_Strategy_Acceptor; template class ACE_Accept_Strategy; template class ACE_Creation_Strategy; template class ACE_Concurrency_Strategy; template class ACE_Scheduling_Strategy; template class TAO_Creation_Strategy; template class TAO_Concurrency_Strategy; template class TAO_Accept_Strategy; #elif defined (ACE_HAS_TEMPLATE_INSTANTIATION_PRAGMA) #pragma instantiate ACE_Acceptor #pragma instantiate ACE_Strategy_Acceptor #pragma instantiate ACE_Accept_Strategy #pragma instantiate ACE_Creation_Strategy #pragma instantiate ACE_Concurrency_Strategy #pragma instantiate ACE_Scheduling_Strategy #pragma instantiate TAO_Creation_Strategy #pragma instantiate TAO_Concurrency_Strategy #pragma instantiate TAO_Accept_Strategy #endif /* ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION */ TAO_IIOP_Acceptor::TAO_IIOP_Acceptor (CORBA::Boolean flag) : TAO_Acceptor (TAO_TAG_IIOP_PROFILE), addrs_ (0), hosts_ (0), endpoint_count_ (0), version_ (TAO_DEF_GIOP_MAJOR, TAO_DEF_GIOP_MINOR), orb_core_ (0), lite_flag_ (flag), base_acceptor_ (), creation_strategy_ (0), concurrency_strategy_ (0), accept_strategy_ (0) { } TAO_IIOP_Acceptor::~TAO_IIOP_Acceptor (void) { // Make sure we are closed before we start destroying the // strategies. this->close (); delete this->creation_strategy_; delete this->concurrency_strategy_; delete this->accept_strategy_; delete [] this->addrs_; for (size_t i = 0; i < this->endpoint_count_; ++i) CORBA::string_free (this->hosts_[i]); delete [] this->hosts_; } // TODO = // 2) For V1.[1,2] there are tagged components int TAO_IIOP_Acceptor::create_mprofile (const TAO_ObjectKey &object_key, TAO_MProfile &mprofile) { // Sanity check. if (this->endpoint_count_ == 0) return -1; // If RT_CORBA is enabled, only one IIOP profile is created per // , and all IIOP endpoints are added into that profile. // If RT_CORBA is not enabled, we create a separate profile for each // acceptor endpoint. #if (TAO_HAS_RT_CORBA == 1) return create_rt_mprofile (object_key, mprofile); #else /* TAO_HAS_RT_CORBA == 1 */ // Adding this->endpoint_count_ to the TAO_MProfile. int count = mprofile.profile_count (); if ((mprofile.size () - count) < this->endpoint_count_ && mprofile.grow (count + this->endpoint_count_) == -1) return -1; // Create a profile for each acceptor endpoint. for (size_t i = 0; i < this->endpoint_count_; ++i) { TAO_IIOP_Profile *pfile = 0; ACE_NEW_RETURN (pfile, TAO_IIOP_Profile (this->hosts_[i], this->addrs_[i].get_port_number (), object_key, this->addrs_[i], this->version_, this->orb_core_), -1); if (mprofile.give_profile (pfile) == -1) { pfile->_decr_refcnt (); pfile = 0; return -1; } if (this->orb_core_->orb_params ()->std_profile_components () == 0) continue; pfile->tagged_components ().set_orb_type (TAO_ORB_TYPE); CONV_FRAME::CodeSetComponentInfo code_set_info; code_set_info.ForCharData.native_code_set = TAO_DEFAULT_CHAR_CODESET_ID; code_set_info.ForWcharData.native_code_set = TAO_DEFAULT_WCHAR_CODESET_ID; pfile->tagged_components ().set_code_sets (code_set_info); } return 0; #endif /* TAO_HAS_RT_CORBA == 1 */ } int TAO_IIOP_Acceptor::create_rt_mprofile (const TAO_ObjectKey &object_key, TAO_MProfile &mprofile) { size_t index = 0; TAO_Profile *pfile = 0; TAO_IIOP_Profile *iiop_profile = 0; // First see if already contains a IIOP profile. for (TAO_PHandle i = 0; i != mprofile.profile_count (); ++i) { pfile = mprofile.get_profile (i); if (pfile->tag () == TAO_TAG_IIOP_PROFILE) { iiop_profile = ACE_dynamic_cast (TAO_IIOP_Profile *, pfile); break; } } // If doesn't contain a IIOP_Profile, we need to create // one. if (iiop_profile == 0) { ACE_NEW_RETURN (iiop_profile, TAO_IIOP_Profile (this->hosts_[0], this->addrs_[0].get_port_number (), object_key, this->addrs_[0], this->version_, this->orb_core_), -1); iiop_profile->endpoint ()->priority (this->priority ()); if (mprofile.give_profile (iiop_profile) == -1) { iiop_profile->_decr_refcnt (); iiop_profile = 0; return -1; } if (this->orb_core_->orb_params ()->std_profile_components () != 0) { iiop_profile->tagged_components ().set_orb_type (TAO_ORB_TYPE); CONV_FRAME::CodeSetComponentInfo code_set_info; code_set_info.ForCharData.native_code_set = TAO_DEFAULT_CHAR_CODESET_ID; code_set_info.ForWcharData.native_code_set = TAO_DEFAULT_WCHAR_CODESET_ID; iiop_profile->tagged_components ().set_code_sets (code_set_info); } index = 1; } // Add any remaining acceptor endpoints to the IIOP_Profile. for (; index < this->endpoint_count_; ++index) { TAO_IIOP_Endpoint *endpoint = 0; ACE_NEW_RETURN (endpoint, TAO_IIOP_Endpoint (this->hosts_[index], this->addrs_[index].get_port_number (), this->addrs_[index]), -1); endpoint->priority (this->priority_); iiop_profile->add_endpoint (endpoint); } return 0; } int TAO_IIOP_Acceptor::is_collocated (const TAO_Endpoint *endpoint) { const TAO_IIOP_Endpoint *endp = ACE_dynamic_cast (const TAO_IIOP_Endpoint *, endpoint); // Make sure the dynamically cast pointer is valid. if (endp == 0) return 0; for (size_t i = 0; i < this->endpoint_count_; ++i) { // compare the port and sin_addr (numeric host address) if (endp->object_addr () == this->addrs_[i]) return 1; // Collocated } return 0; // Not collocated } int TAO_IIOP_Acceptor::close (void) { return this->base_acceptor_.close (); } int TAO_IIOP_Acceptor::open (TAO_ORB_Core *orb_core, int major, int minor, const char *address, const char *options) { this->orb_core_ = orb_core; if (this->init_tcp_properties () != 0) return -1; if (this->hosts_ != 0) { // The hostname cache has already been set! // This is bad mojo, i.e. an internal TAO error. ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) ") ACE_TEXT ("IIOP_Acceptor::open - "), ACE_TEXT ("hostname already set\n\n")), -1); } if (address == 0) return -1; if (major >=0 && minor >= 0) this->version_.set_version (ACE_static_cast (CORBA::Octet, major), ACE_static_cast (CORBA::Octet, minor)); // Parse options if (this->parse_options (options) == -1) return -1; ACE_INET_Addr addr; if (ACE_OS::strchr (address, ':') == address) { // The address is a port number or port name. No hostname was // specified. The hostname for each network interface and the // fully qualified domain name must be obtained. // Check for multiple network interfaces. if (this->probe_interfaces (orb_core) == -1) return -1; // First convert the port into a usable form. if (addr.set (address + sizeof (':')) != 0) return -1; // Now reset the port and set the host. if (addr.set (addr.get_port_number (), ACE_static_cast (ACE_UINT32, INADDR_ANY), 1) != 0) return -1; else return this->open_i (addr); } else if (ACE_OS::strchr (address, ':') == 0) { // The address is a hostname. No port was specified, so assume // port zero (port will be chosen for us). if (addr.set ((unsigned short) 0, address) != 0) return -1; } else if (addr.set (address) != 0) // Host and port were specified. return -1; this->endpoint_count_ = 1; // Only one hostname to store ACE_NEW_RETURN (this->addrs_, ACE_INET_Addr[this->endpoint_count_], -1); ACE_NEW_RETURN (this->hosts_, char *[this->endpoint_count_], -1); if (this->hostname (orb_core, addr, this->hosts_[0]) != 0) return -1; // Copy the addr. The port is (re)set in // TAO_IIOP_Acceptor::open_i(). if (this->addrs_[0].set (addr) != 0) return -1; return this->open_i (addr); } int TAO_IIOP_Acceptor::open_default (TAO_ORB_Core *orb_core, int major, int minor, const char *options) { this->orb_core_ = orb_core; if (this->init_tcp_properties () != 0) return -1; if (this->hosts_ != 0) { // The hostname cache has already been set! // This is bad mojo, i.e. an internal TAO error. ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) ") ACE_TEXT ("IIOP_Acceptor::open_default - "), ACE_TEXT ("hostname already set\n\n")), -1); } if (major >=0 && minor >= 0) this->version_.set_version (ACE_static_cast (CORBA::Octet, major), ACE_static_cast (CORBA::Octet, minor)); // Parse options if (this->parse_options (options) == -1) return -1; // Check for multiple network interfaces. if (this->probe_interfaces (orb_core) == -1) return -1; // Now that each network interface's hostname has been cached, open // an endpoint on each network interface using the INADDR_ANY // address. ACE_INET_Addr addr; if (addr.set (ACE_static_cast(u_short, 0), ACE_static_cast(ACE_UINT32, INADDR_ANY), 1) != 0) return -1; return this->open_i (addr); } int TAO_IIOP_Acceptor::open_i (const ACE_INET_Addr& addr) { ACE_NEW_RETURN (this->creation_strategy_, TAO_IIOP_CREATION_STRATEGY (this->orb_core_, &(this->tcp_properties_), this->lite_flag_), -1); ACE_NEW_RETURN (this->concurrency_strategy_, TAO_IIOP_CONCURRENCY_STRATEGY (this->orb_core_), -1); ACE_NEW_RETURN (this->accept_strategy_, TAO_IIOP_ACCEPT_STRATEGY (this->orb_core_), -1); if (this->base_acceptor_.open (addr, this->orb_core_->reactor (this), this->creation_strategy_, this->accept_strategy_, this->concurrency_strategy_) == -1) { if (TAO_debug_level > 0) ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("\n\nTAO (%P|%t) IIOP_Acceptor::open_i ") ACE_TEXT ("- %p\n\n"), ACE_TEXT ("cannot open acceptor"))); return -1; } ACE_INET_Addr address; // We do this make sure the port number the endpoint is listening on // gets set in the addr. if (this->base_acceptor_.acceptor ().get_local_addr (address) != 0) { // @@ Should this be a catastrophic error??? if (TAO_debug_level > 0) ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("\n\nTAO (%P|%t) IIOP_Acceptor::open_i ") ACE_TEXT ("- %p\n\n"), ACE_TEXT ("cannot get local addr"))); return -1; } // Set the port for each addr. If there is more than one network // interface then the endpoint created on each interface will be on // the same port. This is how a wildcard socket bind() is supposed // to work. u_short port = address.get_port_number (); for (size_t j = 0; j < this->endpoint_count_; ++j) this->addrs_[j].set_port_number (port, 1); if (TAO_debug_level > 5) { for (size_t i = 0; i < this->endpoint_count_; ++i) { ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("\nTAO (%P|%t) IIOP_Acceptor::open_i - ") ACE_TEXT ("listening on: <%s:%u>\n"), this->hosts_[i], this->addrs_[i].get_port_number ())); } } return 0; } int TAO_IIOP_Acceptor::hostname (TAO_ORB_Core *orb_core, ACE_INET_Addr &addr, char *&host) { char tmp_host[MAXHOSTNAMELEN + 1]; if (orb_core->orb_params ()->use_dotted_decimal_addresses () || addr.get_host_name (tmp_host, sizeof (tmp_host)) != 0) { const char *tmp = addr.get_host_addr (); if (tmp == 0) { if (TAO_debug_level > 0) ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("\n\nTAO (%P|%t) ") ACE_TEXT ("IIOP_Acceptor::hostname ") ACE_TEXT ("- %p\n\n"), ACE_TEXT ("cannot determine hostname"))); return -1; } host = CORBA::string_dup (tmp); } else host = CORBA::string_dup (tmp_host); return 0; } int TAO_IIOP_Acceptor::probe_interfaces (TAO_ORB_Core *orb_core) { // Extract the hostname for each network interface, and then cache // it. The hostnames will then be used when creating a // TAO_IIOP_Profile for each endpoint setup on the probed // network interfaces. ACE_INET_Addr *if_addrs = 0; size_t if_cnt = 0; if (ACE::get_ip_interfaces (if_cnt, if_addrs) != 0 && errno != ENOTSUP) { // In the case where errno == ENOTSUP, if_cnt and if_addrs will // not be modified, and will each remain equal to zero. This // causes the default interface to be used. return -1; } if (if_cnt == 0 || if_addrs == 0) { if (TAO_debug_level > 0) { ACE_DEBUG ((LM_WARNING, ACE_TEXT ("TAO (%P|%t) Unable to probe network ") ACE_TEXT ("interfaces. Using default."))); } if_cnt = 1; // Force the network interface count to be one. delete [] if_addrs; ACE_NEW_RETURN (if_addrs, ACE_INET_Addr[if_cnt], -1); } // Scan for the loopback interface since it shouldn't be included in // the list of cached hostnames unless it is the only interface. size_t lo_cnt = 0; // Loopback interface count for (size_t j = 0; j < if_cnt; ++j) if (if_addrs[j].get_ip_address () == INADDR_LOOPBACK) lo_cnt++; // The instantiation for this template is in // tao/IIOP_Connector.cpp. ACE_Auto_Basic_Array_Ptr safe_if_addrs (if_addrs); // If the loopback interface is the only interface then include it // in the list of interfaces to query for a hostname, otherwise // exclude it from the list. if (if_cnt == lo_cnt) this->endpoint_count_ = if_cnt; else this->endpoint_count_ = if_cnt - lo_cnt; ACE_NEW_RETURN (this->addrs_, ACE_INET_Addr[this->endpoint_count_], -1); ACE_NEW_RETURN (this->hosts_, char *[this->endpoint_count_], -1); // The number of hosts/interfaces we want to cache may not be the // same as the number of detected interfaces so keep a separate // count. size_t host_cnt = 0; for (size_t i = 0; i < if_cnt; ++i) { // Ignore any loopback interface if there are other // non-loopback interfaces. if (if_cnt != lo_cnt && if_addrs[i].get_ip_address() == INADDR_LOOPBACK) continue; if (this->hostname (orb_core, if_addrs[i], this->hosts_[host_cnt]) != 0) return -1; // Copy the addr. The port is (re)set in // TAO_IIOP_Acceptor::open_i(). if (this->addrs_[host_cnt].set (if_addrs[i]) != 0) return -1; host_cnt++; } return 0; } CORBA::ULong TAO_IIOP_Acceptor::endpoint_count (void) { return this->endpoint_count_; } int TAO_IIOP_Acceptor::object_key (IOP::TaggedProfile &profile, TAO_ObjectKey &object_key) { // Create the decoding stream from the encapsulation in the buffer, #if (TAO_NO_COPY_OCTET_SEQUENCES == 1) TAO_InputCDR cdr (profile.profile_data.mb ()); #else TAO_InputCDR cdr (ACE_reinterpret_cast(char*,profile.profile_data.get_buffer ()), profile.profile_data.length ()); #endif /* TAO_NO_COPY_OCTET_SEQUENCES == 1 */ CORBA::Octet major, minor; // Read the version. We just read it here. We don't*do any* // processing. if (!(cdr.read_octet (major) && cdr.read_octet (minor))) { if (TAO_debug_level > 0) { ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("TAO (%P|%t) IIOP_Profile::decode - v%d.%d\n"), major, minor)); } return -1; } CORBA::String_var host; CORBA::UShort port = 0; // Get host and port. No processing here too.. if (cdr.read_string (host.out ()) == 0 || cdr.read_ushort (port) == 0) { if (TAO_debug_level > 0) { ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("TAO (%P|%t) TAO_Tagged_Profile::decode - ") ACE_TEXT ("error while decoding host/port"))); } return -1; } // ... and object key. if ((cdr >> object_key) == 0) return -1; // We are NOT bothered about the rest. return 1; } int TAO_IIOP_Acceptor::parse_options (const char *str) { if (str == 0) return 0; // No options to parse. Not a problem. // Use an option format similar to the one used for CGI scripts in // HTTP URLs. // e.g.: option1=foo&option2=bar ACE_CString options (str); size_t len = options.length (); const char option_delimiter = '&'; // Count the number of options. CORBA::ULong option_count = 1; // Number of endpoints in the string (initialized to 1). // Only check for endpoints after the protocol specification and // before the object key. for (size_t i = 0; i < len; ++i) if (options[i] == option_delimiter) option_count++; // The idea behind the following loop is to split the options into // (option, name) pairs. // For example, // `option1=foo&option2=bar' // will be parsed into: // `option1=foo' // `option2=bar' int begin = 0; int end = -1; for (CORBA::ULong j = 0; j < option_count; ++j) { begin += end + 1; if (j < option_count - 1) end = options.find (option_delimiter, begin); else end = len - begin; // Handle last endpoint differently if (end == begin) ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) Zero length IIOP option.\n")), -1); else if (end != ACE_CString::npos) { ACE_CString opt = options.substring (begin, end); int slot = opt.find ("="); if (slot == ACE_static_cast (int, len - 1) || slot == ACE_CString::npos) ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) IIOP option <%s> is ") ACE_TEXT ("missing a value.\n"), opt.c_str ()), -1); ACE_CString name = opt.substring (0, slot); ACE_CString value = opt.substring (slot + 1); if (name.length () == 0) ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) Zero length IIOP ") ACE_TEXT ("option name.\n")), -1); if (name == "priority") { CORBA::Short corba_priority = ACE_static_cast (CORBA::Short, ACE_OS::atoi (value.c_str ())); if (corba_priority >= 0 /* && corba_priority < 32768 */) // priority_ and corba_priority will always be less // than 32768 since CORBA::Short is a signed 16 bit // integer. this->priority_ = corba_priority; else ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) Invalid IIOP endpoint ") ACE_TEXT ("priority: <%s>\n"), value.c_str ()), -1); } else ACE_ERROR_RETURN ((LM_ERROR, ACE_TEXT ("TAO (%P|%t) Invalid IIOP option: <%s>\n"), name.c_str ()), -1); } } return 0; } int TAO_IIOP_Acceptor::init_tcp_properties (void) { #if (TAO_HAS_RT_CORBA == 1) // @@ Currently (in the code below), we obtain protocol properties from // ORB-level ServerProtocol, even though the policy may // have been overridden on POA level. That's because currently all // endpoints (acceptors) are global. Once endpoints become per POA, // the code below will have to be changed to look at the POA-level // ServerProtocol policy first. // @@ Later we may want to factor some of the code below // among different protocols and place it into TAO_Acceptor, for // example. // ServerProtocolProperties policy controls protocols configuration. // Look for protocol properties in the effective ServerProtocolPolicy. TAO_ServerProtocolPolicy *server_protocols = this->orb_core_->server_protocol (); // Automatically release the policy. CORBA::Object_var auto_release = server_protocols; RTCORBA::TCPProtocolProperties_var tcp_properties = RTCORBA::TCPProtocolProperties::_nil (); RTCORBA::ProtocolList & protocols = server_protocols->protocols_rep (); // Find protocol properties for TCP. ACE_DECLARE_NEW_CORBA_ENV; for (CORBA::ULong j = 0; j < protocols.length (); ++j) if (protocols[j].protocol_type == TAO_TAG_IIOP_PROFILE) { tcp_properties = RTCORBA::TCPProtocolProperties::_narrow (protocols[j].transport_protocol_properties.in (), ACE_TRY_ENV); ACE_CHECK_RETURN (-1); break; } if (CORBA::is_nil (tcp_properties.in ())) { // TCP Properties were not specified in the effective policy. // We must use orb defaults. server_protocols = this->orb_core_->default_server_protocol (); // Automatically release the policy. auto_release = server_protocols; // Find protocol properties for IIOP. RTCORBA::ProtocolList & protocols = server_protocols->protocols_rep (); for (CORBA::ULong j = 0; j < protocols.length (); ++j) if (protocols[j].protocol_type == TAO_TAG_IIOP_PROFILE) { tcp_properties = RTCORBA::TCPProtocolProperties::_narrow (protocols[j].transport_protocol_properties.in (), ACE_TRY_ENV); ACE_CHECK_RETURN (-1); break; } // Orb defaults should never be null, since the ORB initializes // them in ORB_init ... } // Extract and locally store properties of interest. this->tcp_properties_.send_buffer_size = tcp_properties->send_buffer_size (); this->tcp_properties_.recv_buffer_size = tcp_properties->recv_buffer_size (); this->tcp_properties_.no_delay = tcp_properties->no_delay (); // @@ NOTE. RTCORBA treats a combination of transport+messaging // as a single protocol. Keep this in mind for when we adopt // RTCORBA approach to protocols configuration for nonRT use. In // particular, what are the semantics of independent variation of // messaging and transport layers, when one transport appears in // combination with several messaging protocols, for example. #else /* TAO_HAS_RT_CORBA == 1 */ this->tcp_properties_.send_buffer_size = this->orb_core_->orb_params ()->sock_sndbuf_size (); this->tcp_properties_.recv_buffer_size = this->orb_core_->orb_params ()->sock_rcvbuf_size (); this->tcp_properties_.no_delay = this->orb_core_->orb_params ()->nodelay (); #endif /* TAO_HAS_RT_CORBA == 1 */ return 0; }