(file) Return to ProviderManagerService.cpp CVS log (file) (dir) Up to [Pegasus] / pegasus / src / Pegasus / ProviderManager2 / Attic

Diff for /pegasus/src/Pegasus/ProviderManager2/Attic/ProviderManagerService.cpp between version 1.6 and 1.15

version 1.6, 2003/08/21 19:36:25 version 1.15, 2003/10/22 14:26:12
Line 1 
Line 1 
 //%/////////////////////////////////////////////////////////////////////////////  //%2003////////////////////////////////////////////////////////////////////////
 // //
 // Copyright (c) 2000 - 2003 BMC Software, Hewlett-Packard Company, IBM,  // Copyright (c) 2000, 2001, 2002  BMC Software, Hewlett-Packard Development
 // The Open Group, Tivoli Systems  // Company, L. P., IBM Corp., The Open Group, Tivoli Systems.
   // Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L. P.;
   // IBM Corp.; EMC Corporation, The Open Group.
 // //
 // Permission is hereby granted, free of charge, to any person obtaining a copy // Permission is hereby granted, free of charge, to any person obtaining a copy
 // of this software and associated documentation files (the "Software"), to // of this software and associated documentation files (the "Software"), to
Line 44 
Line 46 
  
 PEGASUS_NAMESPACE_BEGIN PEGASUS_NAMESPACE_BEGIN
  
 // ATTN: this section is a temporary solution to populate the list of enabled  
 // provider managers for a given distribution. it includes another temporary  
 // solution for converting a generic file name into a file name useable by  
 // each platform.  
   
 // BEGIN TEMP SECTION // BEGIN TEMP SECTION
 String _resolveFileName(const String & fileName)  class ProviderManagerContainer
   {
   public:
       ProviderManagerContainer(const String & physicalName, const String & logicalName, const String & interfaceName)
 { {
     String temp;  
   
     #if defined(PEGASUS_OS_TYPE_WINDOWS)     #if defined(PEGASUS_OS_TYPE_WINDOWS)
     temp = fileName + String(".dll");          _physicalName = physicalName + String(".dll");
     #elif defined(PEGASUS_OS_HPUX) && defined(PEGASUS_PLATFORM_HPUX_PARISC_ACC)     #elif defined(PEGASUS_OS_HPUX) && defined(PEGASUS_PLATFORM_HPUX_PARISC_ACC)
     temp = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));          _physicalName = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));
     temp.append(String("/lib") + fileName + String(".sl"));          _physicalName.append(String("/lib") + physical + String(".sl"));
     #elif defined(PEGASUS_OS_HPUX) && !defined(PEGASUS_PLATFORM_HPUX_PARISC_ACC)     #elif defined(PEGASUS_OS_HPUX) && !defined(PEGASUS_PLATFORM_HPUX_PARISC_ACC)
     temp = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));          _physicalName = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));
     temp.append(String("/lib") + fileName + String(".so"));          _physicalName.append(String("/lib") + physical + String(".so"));
     #elif defined(PEGASUS_OS_OS400)     #elif defined(PEGASUS_OS_OS400)
     temp = fileName;          _physicalName = physicalName;
     #else     #else
     temp = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));          _physicalName = ConfigManager::getHomedPath(ConfigManager::getInstance()->getCurrentValue("providerDir"));
     temp.append(String("/lib") + fileName + String(".so"));          _physicalName.append(String("/lib") + physical + String(".so"));
     #endif     #endif
  
     return(temp);          _logicalName = logicalName;
 }  
  
 Array<Pair<String, String> > _initializeFileNames(void)          _interfaceName = interfaceName;
 {  
     Array<Pair<String, String> > temp;  
  
     #if defined(ENABLE_DEFAULT_PROVIDER_MANAGER)          _module = ProviderManagerModule(_physicalName);
     temp.append(Pair<String, String>(_resolveFileName("DefaultProviderManager"), String("DEFAULT")));  
     #endif  
  
     #if defined(ENABLE_CMPI_PROVIDER_MANAGER)          _module.load();
     temp.append(Pair<String, String>(_resolveFileName("CMPIProviderManager"), String("CMPI")));  
     #endif  
  
     return(temp);          _manager = _module.getProviderManager(_logicalName);
 } }
  
 static const Array<Pair<String,String> > _fileNames = _initializeFileNames();      ~ProviderManagerContainer(void)
 // END TEMP SECTION      {
           _module.unload();
       }
  
 inline Boolean _isSupportedRequestType(const Message * message)      ProviderManager & getProviderManager(void)
 { {
     Boolean rc = false;          return(*_manager);
       }
  
     if(message == 0)      String & getPhysicalName(void)
     {     {
         return(rc);          return(_physicalName);
     }     }
  
     /*      String & getLogicalName(void)
     // ATTN : need to determine valid request message types      {
     // before enabling.          return(_logicalName);
       }
  
     switch(message->getType())      String & getInterfaceName(void)
     {     {
     case CIM_GET_INSTANCE_REQUEST_MESSAGE:          return(_interfaceName);
     case CIM_ENUMERATE_INSTANCES_REQUEST_MESSAGE:      }
     case CIM_ENUMERATE_INSTANCE_NAMES_REQUEST_MESSAGE:  
     case CIM_CREATE_INSTANCE_REQUEST_MESSAGE:  
     case CIM_MODIFY_INSTANCE_REQUEST_MESSAGE:  
     case CIM_DELETE_INSTANCE_REQUEST_MESSAGE:  
     case CIM_EXEC_QUERY_REQUEST_MESSAGE:  
     case CIM_ASSOCIATORS_REQUEST_MESSAGE:  
     case CIM_ASSOCIATOR_NAMES_REQUEST_MESSAGE:  
     case CIM_REFERENCES_REQUEST_MESSAGE:  
     case CIM_REFERENCE_NAMES_REQUEST_MESSAGE:  
     case CIM_GET_PROPERTY_REQUEST_MESSAGE:  
     case CIM_SET_PROPERTY_REQUEST_MESSAGE:  
     case CIM_INVOKE_METHOD_REQUEST_MESSAGE:  
     case CIM_CREATE_SUBSCRIPTION_REQUEST_MESSAGE:  
     case CIM_MODIFY_SUBSCRIPTION_REQUEST_MESSAGE:  
     case CIM_DELETE_SUBSCRIPTION_REQUEST_MESSAGE:  
     case CIM_ENABLE_INDICATIONS_REQUEST_MESSAGE:  
     case CIM_DISABLE_INDICATIONS_REQUEST_MESSAGE:  
     case CIM_DISABLE_MODULE_REQUEST_MESSAGE:  
     case CIM_ENABLE_MODULE_REQUEST_MESSAGE:  
     case CIM_STOP_ALL_PROVIDERS_REQUEST_MESSAGE:  
     case CIM_CONSUME_INDICATION_REQUEST_MESSAGE:  
         rc = true;  
  
         break;  private:
     default:      String _physicalName;
         rc = false;      String _logicalName;
       String _interfaceName;
  
         break;      ProviderManagerModule _module;
     }      ProviderManager * _manager;
     */  
   };
   
   static Array<ProviderManagerContainer> _providerManagers;
   // END TEMP SECTION
  
     rc = true;  inline Boolean _isSupportedRequestType(const Message * message)
   {
       // ATTN: needs implementation
  
     return(rc);      // for now, assume all requests are valid
   
       return(true);
 } }
  
 inline Boolean _isSupportedResponseType(const Message * message) inline Boolean _isSupportedResponseType(const Message * message)
 { {
     Boolean rc = false;      // ATTN: needs implementation
   
       // for now, assume all responses are invalid
  
     return(rc);      return(false);
 } }
  
 ProviderManagerService::ProviderManagerService(void) ProviderManagerService::ProviderManagerService(void)
Line 158 
Line 142 
 ProviderManagerService::ProviderManagerService(ProviderRegistrationManager * providerRegistrationManager) ProviderManagerService::ProviderManagerService(ProviderRegistrationManager * providerRegistrationManager)
     : MessageQueueService(PEGASUS_QUEUENAME_PROVIDERMANAGER_CPP)     : MessageQueueService(PEGASUS_QUEUENAME_PROVIDERMANAGER_CPP)
 { {
     for(Uint32 i = 0, n = _fileNames.size(); i < n; i++)      SetProviderRegistrationManager(providerRegistrationManager);
     {  
         String message;  
   
         message = "ProviderManagerService::ProviderManagerService() loading " +  
             _fileNames[i].first + "(" + _fileNames[i].second + ")";  
   
         PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4, message);  
   
         try  
         {  
             ProviderManagerModule module(_fileNames[i].first);  
   
             if(module.load() == false)  
             {  
                 throw 0;    // ATTN: inefficient  
             }  
  
             ProviderManager * manager = module.getProviderManager(_fileNames[i].second);      // ATTN: this section is a temporary solution to populate the list of enabled
       // provider managers for a given distribution. it includes another temporary
             if(manager == 0)      // solution for converting a generic file name into a file name useable by
             {      // each platform.
                 throw 0;    // ATTN: inefficient  
             }  
  
             // ATTN: only set the hacked/cached provider registration manager pointer after the      // BEGIN TEMP SECTION
             // DEFAULT provider manager is loaded.      //#if defined(PEGASUS_OS_OS400)
             if(String::equalNoCase(_fileNames[i].second, "DEFAULT"))      //_providerManagers.append(ProviderManagerContainer("QSYS/??????????", "INTERNAL", "INTERNAL"));
             {      //#else
                 manager->setProviderRegistrationManager(providerRegistrationManager);      //_providerManager.append(ProviderManagerContainer("InternalProviderManager", "DEFAULT", "INTERNAL"));
             }      //#endif
  
             _providerManagers.append(Pair<ProviderManager *, ProviderManagerModule>(manager, module));      #if defined(ENABLE_DEFAULT_PROVIDER_MANAGER)
         }      #if defined(PEGASUS_OS_OS400)
         catch(...)      _providerManagers.append(ProviderManagerContainer("QSYS/QYCMDFTPVM", "DEFAULT", "C++Default"));
         {      #else
             message = "ProviderManagerService::ProviderManagerService() exception loading " +      _providerManagers.append(ProviderManagerContainer("DefaultProviderManager", "DEFAULT", "C++Default"));
                 _fileNames[i].first + "(" + _fileNames[i].second + ")";      #endif
       #endif
  
             PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4, message);      #if defined(ENABLE_CMPI_PROVIDER_MANAGER)
         }      #if defined(PEGASUS_OS_OS400)
     }      _providerManagers.append(ProviderManagerContainer("QSYS/QYCMCMPIPM", "CMPI", "CMPI"));
       #else
       _providerManagers.append(ProviderManagerContainer("CMPIProviderManager", "CMPI", "CMPI"));
       #endif
       #endif
       // END TEMP SECTION
 } }
  
 ProviderManagerService::~ProviderManagerService(void) ProviderManagerService::~ProviderManagerService(void)
Line 322 
Line 294 
 { {
     PEG_METHOD_ENTER(TRC_PROVIDERMANAGER, "ProviderManagerService::handleCimOperation");     PEG_METHOD_ENTER(TRC_PROVIDERMANAGER, "ProviderManagerService::handleCimOperation");
  
       if(arg == 0)
       {
           // thread started with invalid argument.
           return(PEGASUS_THREAD_RETURN(1));
       }
   
     // get the service from argument     // get the service from argument
     ProviderManagerService * service = reinterpret_cast<ProviderManagerService *>(arg);     ProviderManagerService * service = reinterpret_cast<ProviderManagerService *>(arg);
  
     PEGASUS_ASSERT(service != 0);  
   
     if(service->_incomingQueue.size() == 0)     if(service->_incomingQueue.size() == 0)
     {     {
         PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4,         PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4,
Line 340 
Line 316 
  
     AsyncOpNode * op = service->_incomingQueue.dequeue();     AsyncOpNode * op = service->_incomingQueue.dequeue();
  
     PEGASUS_ASSERT(op != 0 );      if((op == 0) || (op->_request.count() == 0))
   
     if(op->_request.count() == 0)  
     {     {
         MessageQueue * queue = MessageQueue::lookup(op->_source_queue);         MessageQueue * queue = MessageQueue::lookup(op->_source_queue);
  
Line 356 
Line 330 
  
     AsyncRequest * request = static_cast<AsyncRequest *>(op->_request.next(0));     AsyncRequest * request = static_cast<AsyncRequest *>(op->_request.next(0));
  
     PEGASUS_ASSERT(request != 0);      if((request == 0) || (request->getType() != async_messages::ASYNC_LEGACY_OP_START))
   
     if(request->getType() != async_messages::ASYNC_LEGACY_OP_START)  
     {     {
         // reply with NAK         // reply with NAK
  
Line 367 
Line 339 
         return(PEGASUS_THREAD_RETURN(0));         return(PEGASUS_THREAD_RETURN(0));
     }     }
  
       try
       {
     Message * legacy = static_cast<AsyncLegacyOperationStart *>(request)->get_action();     Message * legacy = static_cast<AsyncLegacyOperationStart *>(request)->get_action();
  
     if(_isSupportedRequestType(legacy))     if(_isSupportedRequestType(legacy))
Line 389 
Line 363 
             Thread::clearLanguages();             Thread::clearLanguages();
         }         }
  
         try  
         {  
             service->handleCimRequest(op, legacy);             service->handleCimRequest(op, legacy);
         }         }
       }
         catch(...)         catch(...)
         {         {
             // ATTN: log error             // ATTN: log error
         }         }
     }  
  
     PEG_METHOD_EXIT();     PEG_METHOD_EXIT();
  
     return(PEGASUS_THREAD_RETURN(0));     return(PEGASUS_THREAD_RETURN(0));
 } }
  
 void ProviderManagerService::handleCimRequest(AsyncOpNode * op, const Message * message) throw()  void ProviderManagerService::handleCimRequest(AsyncOpNode * op, const Message * message)
 { {
     PEG_METHOD_ENTER(TRC_PROVIDERMANAGER, "ProviderManagerService::handleCimRequest");     PEG_METHOD_ENTER(TRC_PROVIDERMANAGER, "ProviderManagerService::handleCimRequest");
  
Line 418 
Line 390 
  
     Message * response = 0;     Message * response = 0;
  
       // get namespace and class name from message
       String nameSpace;
       String className;
   
       switch(message->getType())
       {
       case CIM_GET_CLASS_REQUEST_MESSAGE:
           {
               const CIMGetClassRequestMessage * p = dynamic_cast<const CIMGetClassRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_ENUMERATE_CLASSES_REQUEST_MESSAGE:
           {
               const CIMEnumerateClassesRequestMessage * p = dynamic_cast<const CIMEnumerateClassesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_ENUMERATE_CLASS_NAMES_REQUEST_MESSAGE:
           {
               const CIMEnumerateClassNamesRequestMessage * p = dynamic_cast<const CIMEnumerateClassNamesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_CREATE_CLASS_REQUEST_MESSAGE:
           {
               const CIMCreateClassRequestMessage * p = dynamic_cast<const CIMCreateClassRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_MODIFY_CLASS_REQUEST_MESSAGE:
           {
               const CIMModifyClassRequestMessage * p = dynamic_cast<const CIMModifyClassRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_DELETE_CLASS_REQUEST_MESSAGE:
           {
               const CIMDeleteClassRequestMessage * p = dynamic_cast<const CIMDeleteClassRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_GET_INSTANCE_REQUEST_MESSAGE:
           {
               const CIMGetInstanceRequestMessage * p = dynamic_cast<const CIMGetInstanceRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_ENUMERATE_INSTANCES_REQUEST_MESSAGE:
           {
               const CIMEnumerateInstancesRequestMessage * p = dynamic_cast<const CIMEnumerateInstancesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_ENUMERATE_INSTANCE_NAMES_REQUEST_MESSAGE:
           {
               const CIMEnumerateInstanceNamesRequestMessage * p = dynamic_cast<const CIMEnumerateInstanceNamesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_CREATE_INSTANCE_REQUEST_MESSAGE:
           {
               const CIMCreateInstanceRequestMessage * p = dynamic_cast<const CIMCreateInstanceRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_MODIFY_INSTANCE_REQUEST_MESSAGE:
           {
               const CIMModifyInstanceRequestMessage * p = dynamic_cast<const CIMModifyInstanceRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_DELETE_INSTANCE_REQUEST_MESSAGE:
           {
               const CIMDeleteInstanceRequestMessage * p = dynamic_cast<const CIMDeleteInstanceRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_EXEC_QUERY_REQUEST_MESSAGE:
           break;
       case CIM_ASSOCIATORS_REQUEST_MESSAGE:
           {
               const CIMAssociatorsRequestMessage * p = dynamic_cast<const CIMAssociatorsRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_ASSOCIATOR_NAMES_REQUEST_MESSAGE:
           {
               const CIMAssociatorNamesRequestMessage * p = dynamic_cast<const CIMAssociatorNamesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_REFERENCES_REQUEST_MESSAGE:
           {
               const CIMReferencesRequestMessage * p = dynamic_cast<const CIMReferencesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_REFERENCE_NAMES_REQUEST_MESSAGE:
           {
               const CIMReferenceNamesRequestMessage * p = dynamic_cast<const CIMReferenceNamesRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_GET_PROPERTY_REQUEST_MESSAGE:
           {
               const CIMGetPropertyRequestMessage * p = dynamic_cast<const CIMGetPropertyRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_SET_PROPERTY_REQUEST_MESSAGE:
           {
               const CIMSetPropertyRequestMessage * p = dynamic_cast<const CIMSetPropertyRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       case CIM_INVOKE_METHOD_REQUEST_MESSAGE:
           {
               const CIMInvokeMethodRequestMessage * p = dynamic_cast<const CIMInvokeMethodRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               //className = p->className.getString();
           }
   
           break;
       /*
       case CIM_ENABLE_INDICATION_SUBSCRIPTION_REQUEST_MESSAGE:
           {
               const CIMEnableIndicationsSubscriptionRequestMessage * p = dynamic_cast<const CIMEnableIndicationsRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_MODIFY_INDICATION_SUBSCRIPTION_REQUEST_MESSAGE:
           {
               const CIMModifyIndicationsRequestMessage * p = dynamic_cast<const CIMModifyIndicationsRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       case CIM_DISABLE_INDICATION_SUBSCRIPTION_REQUEST_MESSAGE:
           {
               const CIMGetClassRequestMessage * p = dynamic_cast<const CIMGetClassRequestMessage *>(message);
   
               PEGASUS_ASSERT(p != 0);
   
               nameSpace = p->nameSpace.getString();
               className = p->className.getString();
           }
   
           break;
       */
       default:
           break;
       }
   
       ProviderName name(
           CIMObjectPath(String::EMPTY, nameSpace, className).toString(),
           String::EMPTY,
           String::EMPTY,
           String::EMPTY,
           0);
   
     // find provider manager     // find provider manager
     // ATTN: implement efficient lookup      name = ProviderRegistrar().findProvider(name);
     ProviderManager * manager = _providerManagers[0].first;  
  
       // find provider manager for provider interface
       for(Uint32 i = 0, n = _providerManagers.size(); i < n; i++)
       {
           if(String::equalNoCase(name.getInterfaceName(), _providerManagers[i].getInterfaceName()))
           {
     try     try
     {     {
         PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4,         PEG_TRACE_STRING(TRC_PROVIDERMANAGER, Tracer::LEVEL4,
             "ProviderManagerService::handleCimRequest() passing control to provider manager.");             "ProviderManagerService::handleCimRequest() passing control to provider manager.");
  
         // forward request         // forward request
         response = manager->processMessage(request);                  response = _providerManagers[0].getProviderManager().processMessage(request);
     }     }
     catch(...)     catch(...)
     {     {
         // ATTN: create response with error message         // ATTN: create response with error message
     }     }
  
               break;
           }
       }
   
     // preserve message key     // preserve message key
     response->setKey(request->getKey());     response->setKey(request->getKey());
  
Line 458 
Line 701 
 } }
  
 PEGASUS_NAMESPACE_END PEGASUS_NAMESPACE_END
   


Legend:
Removed from v.1.6  
changed lines
  Added in v.1.15

No CVS admin address has been configured
Powered by
ViewCVS 0.9.2