| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 | /*=============================================================================  Library: CTK  Copyright (c) 2010 German Cancer Research Center,    Division of Medical and Biological Informatics  Licensed under the Apache License, Version 2.0 (the "License");  you may not use this file except in compliance with the License.  You may obtain a copy of the License at    http://www.apache.org/licenses/LICENSE-2.0  Unless required by applicable law or agreed to in writing, software  distributed under the License is distributed on an "AS IS" BASIS,  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the License for the specific language governing permissions and  limitations under the License.=============================================================================*/#include "ctkServiceReferencePrivate.h"#include <QObject>#include <QMutexLocker>#include "ctkPluginConstants.h"#include "ctkServiceFactory.h"#include "ctkServiceException.h"#include "ctkPluginPrivate_p.h"#include "ctkServiceRegistrationPrivate.h"#include "ctkPluginFrameworkContext_p.h"  ctkServiceReferencePrivate::ctkServiceReferencePrivate(ctkServiceRegistrationPrivate* reg)    : registration(reg)  {  }  QObject* ctkServiceReferencePrivate::getService(ctkPlugin* plugin)  {    QObject* s = 0;    {      QMutexLocker lock(®istration->propsLock);      if (registration->available)      {        int count = registration->dependents.value(plugin);        if (count == 0)        {          QStringList classes =              registration->properties.value(ctkPluginConstants::OBJECTCLASS).toStringList();          registration->dependents[plugin] = 1;          if (ctkServiceFactory* serviceFactory = qobject_cast<ctkServiceFactory*>(registration->getService()))          {            try            {              s = serviceFactory->getService(plugin, registration->q_func());            }            catch (const std::exception& pe)            {              ctkServiceException se("ctkServiceFactory throw an exception",                                  ctkServiceException::FACTORY_EXCEPTION, pe);              plugin->d_func()->fwCtx->listeners.frameworkError                (registration->plugin->q_func(), se);              return 0;            }            if (s == 0) {              ctkServiceException se("ctkServiceFactory produced null",                                  ctkServiceException::FACTORY_ERROR);              plugin->d_func()->fwCtx->listeners.frameworkError                (registration->plugin->q_func(), se);              return 0;            }            for (QStringListIterator i(classes); i.hasNext(); )            {              QString cls = i.next();              if (!registration->plugin->fwCtx->services.checkServiceClass(s, cls))              {                ctkServiceException se(QString("ctkServiceFactory produced an object ") +                                     "that did not implement: " + cls,                                     ctkServiceException::FACTORY_ERROR);                plugin->d_func()->fwCtx->listeners.frameworkError                  (registration->plugin->q_func(), se);                return 0;              }            }            registration->serviceInstances.insert(plugin, s);          }          else          {            s = registration->getService();          }        }        else        {          registration->dependents.insert(plugin, count + 1);          if (qobject_cast<ctkServiceFactory*>(registration->getService()))          {            s = registration->serviceInstances.value(plugin);          }          else          {            s = registration->getService();          }        }      }    }    return s;  }  /**     * Unget the service object.     *     * @param bundle Bundle who wants remove service.     * @param checkRefCounter If true decrement refence counter and remove service     *                        if we reach zero. If false remove service without     *                        checking refence counter.     * @return True if service was remove or false if only refence counter was     *         decremented.     */  bool ctkServiceReferencePrivate::ungetService(ctkPlugin* plugin, bool checkRefCounter)  {    QMutexLocker lock(®istration->propsLock);    bool hadReferences = false;    if (registration->reference != 0)    {      bool removeService = false;      int count= registration->dependents.value(plugin);      if (count > 0)      {        hadReferences = true;      }      if(checkRefCounter)      {        if (count > 1)        {            registration->dependents[plugin] = count - 1;        }        else if(count == 1)        {          removeService = true;        }      }      else      {        removeService = true;      }      if (removeService)      {        QObject* sfi = registration->serviceInstances[plugin];        registration->serviceInstances.remove(plugin);        if (sfi != 0)        {          try          {            qobject_cast<ctkServiceFactory*>(registration->getService())->ungetService(plugin,                registration->q_func(), sfi);          }          catch (const std::exception& e)          {            plugin->d_func()->fwCtx->listeners.frameworkError(registration->plugin->q_func(), e);          }        }        registration->dependents.remove(plugin);      }    }    return hadReferences;  }
 |