Home | History | Annotate | Download | only in parameter
      1 /*
      2  * Copyright (c) 2011-2015, Intel Corporation
      3  * All rights reserved.
      4  *
      5  * Redistribution and use in source and binary forms, with or without modification,
      6  * are permitted provided that the following conditions are met:
      7  *
      8  * 1. Redistributions of source code must retain the above copyright notice, this
      9  * list of conditions and the following disclaimer.
     10  *
     11  * 2. Redistributions in binary form must reproduce the above copyright notice,
     12  * this list of conditions and the following disclaimer in the documentation and/or
     13  * other materials provided with the distribution.
     14  *
     15  * 3. Neither the name of the copyright holder nor the names of its contributors
     16  * may be used to endorse or promote products derived from this software without
     17  * specific prior written permission.
     18  *
     19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
     20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
     21  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
     22  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
     23  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
     24  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
     25  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
     26  * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
     28  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29  */
     30 #include <cassert>
     31 #include "ConfigurableDomains.h"
     32 #include "ConfigurableDomain.h"
     33 #include "ConfigurableElement.h"
     34 
     35 #define base CElement
     36 
     37 using std::string;
     38 
     39 string CConfigurableDomains::getKind() const
     40 {
     41     return "ConfigurableDomains";
     42 }
     43 
     44 bool CConfigurableDomains::childrenAreDynamic() const
     45 {
     46     return true;
     47 }
     48 
     49 // Ensure validity on whole domains from main blackboard
     50 void CConfigurableDomains::validate(const CParameterBlackboard *pMainBlackboard)
     51 {
     52     // Delegate to domains
     53     size_t uiNbConfigurableDomains = getNbChildren();
     54 
     55     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
     56 
     57         CConfigurableDomain *pChildConfigurableDomain =
     58             static_cast<CConfigurableDomain *>(getChild(child));
     59 
     60         pChildConfigurableDomain->validate(pMainBlackboard);
     61     }
     62 }
     63 
     64 // Configuration application if required
     65 void CConfigurableDomains::apply(CParameterBlackboard *pParameterBlackboard, CSyncerSet &syncerSet,
     66                                  bool bForce, core::Results &infos) const
     67 {
     68     /// Delegate to domains
     69 
     70     // Start with domains that can be synchronized all at once (with passed syncer set)
     71     size_t uiNbConfigurableDomains = getNbChildren();
     72 
     73     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
     74 
     75         const CConfigurableDomain *pChildConfigurableDomain =
     76             static_cast<const CConfigurableDomain *>(getChild(child));
     77 
     78         std::string info;
     79         // Apply and collect syncers when relevant
     80         pChildConfigurableDomain->apply(pParameterBlackboard, &syncerSet, bForce, info);
     81 
     82         if (!info.empty()) {
     83             infos.push_back(info);
     84         }
     85     }
     86     // Synchronize those collected syncers
     87     syncerSet.sync(*pParameterBlackboard, false, nullptr);
     88 
     89     // Then deal with domains that need to synchronize along apply
     90     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
     91 
     92         const CConfigurableDomain *pChildConfigurableDomain =
     93             static_cast<const CConfigurableDomain *>(getChild(child));
     94 
     95         std::string info;
     96         // Apply and synchronize when relevant
     97         pChildConfigurableDomain->apply(pParameterBlackboard, nullptr, bForce, info);
     98         if (!info.empty()) {
     99             infos.push_back(info);
    100         }
    101     }
    102 }
    103 
    104 // From IXmlSource
    105 void CConfigurableDomains::toXml(CXmlElement &xmlElement,
    106                                  CXmlSerializingContext &serializingContext) const
    107 {
    108     // Set attribute
    109     xmlElement.setAttribute("SystemClassName", getName());
    110 
    111     base::childrenToXml(xmlElement, serializingContext);
    112 }
    113 
    114 // Configuration/Domains handling
    115 /// Domains
    116 bool CConfigurableDomains::createDomain(const string &strName, string &strError)
    117 {
    118     // Already exists?
    119     if (findChild(strName)) {
    120 
    121         strError = "Already existing configurable domain";
    122 
    123         return false;
    124     }
    125 
    126     // Creation/Hierarchy
    127     addChild(new CConfigurableDomain(strName));
    128 
    129     return true;
    130 }
    131 
    132 bool CConfigurableDomains::addDomain(CConfigurableDomain &domain, bool bOverwrite, string &strError)
    133 {
    134     string strErrorDrop;
    135 
    136     string strDomainName(domain.getName());
    137     CConfigurableDomain *pExistingDomain = findConfigurableDomain(strDomainName, strErrorDrop);
    138 
    139     if (pExistingDomain) {
    140         if (!bOverwrite) {
    141             strError = "Can't add domain \"" + strDomainName +
    142                        "\" because it already exists and overwrite was not requested.";
    143             return false;
    144         }
    145 
    146         deleteDomain(*pExistingDomain);
    147     }
    148 
    149     addChild(&domain);
    150 
    151     return true;
    152 }
    153 
    154 void CConfigurableDomains::deleteDomain(CConfigurableDomain &configurableDomain)
    155 {
    156     removeChild(&configurableDomain);
    157 
    158     delete &configurableDomain;
    159 }
    160 
    161 bool CConfigurableDomains::deleteDomain(const string &strName, string &strError)
    162 {
    163     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strName, strError);
    164 
    165     if (pConfigurableDomain) {
    166         deleteDomain(*pConfigurableDomain);
    167         return true;
    168     }
    169 
    170     return false;
    171 }
    172 
    173 void CConfigurableDomains::deleteAllDomains()
    174 {
    175     // remove Children
    176     clean();
    177 }
    178 
    179 bool CConfigurableDomains::renameDomain(const string &strName, const string &strNewName,
    180                                         string &strError)
    181 {
    182     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strName, strError);
    183 
    184     if (!pConfigurableDomain) {
    185 
    186         return false;
    187     }
    188 
    189     // Rename
    190     return pConfigurableDomain->rename(strNewName, strError);
    191 }
    192 
    193 bool CConfigurableDomains::setSequenceAwareness(const string &strDomain, bool bSequenceAware,
    194                                                 string &strError)
    195 {
    196     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    197 
    198     if (!pConfigurableDomain) {
    199 
    200         return false;
    201     }
    202 
    203     pConfigurableDomain->setSequenceAwareness(bSequenceAware);
    204 
    205     return true;
    206 }
    207 
    208 bool CConfigurableDomains::getSequenceAwareness(const string &strDomain, bool &bSequenceAware,
    209                                                 string &strError) const
    210 {
    211     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    212 
    213     if (!pConfigurableDomain) {
    214 
    215         return false;
    216     }
    217 
    218     bSequenceAware = pConfigurableDomain->getSequenceAwareness();
    219 
    220     return true;
    221 }
    222 
    223 /// Configurations
    224 bool CConfigurableDomains::listConfigurations(const string &strDomain, string &strResult) const
    225 {
    226     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strResult);
    227 
    228     if (!pConfigurableDomain) {
    229 
    230         return false;
    231     }
    232     // delegate
    233     pConfigurableDomain->listChildren(strResult);
    234 
    235     return true;
    236 }
    237 
    238 bool CConfigurableDomains::createConfiguration(const string &strDomain,
    239                                                const string &strConfiguration,
    240                                                const CParameterBlackboard *pMainBlackboard,
    241                                                string &strError)
    242 {
    243     // Find domain
    244     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    245 
    246     if (!pConfigurableDomain) {
    247 
    248         return false;
    249     }
    250     // Delegate
    251     return pConfigurableDomain->createConfiguration(strConfiguration, pMainBlackboard, strError);
    252 }
    253 
    254 bool CConfigurableDomains::deleteConfiguration(const string &strDomain,
    255                                                const string &strConfiguration, string &strError)
    256 {
    257     // Find domain
    258     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    259 
    260     if (!pConfigurableDomain) {
    261 
    262         return false;
    263     }
    264     // Delegate
    265     return pConfigurableDomain->deleteConfiguration(strConfiguration, strError);
    266 }
    267 
    268 bool CConfigurableDomains::renameConfiguration(const string &strDomain,
    269                                                const string &strConfigurationName,
    270                                                const string &strNewConfigurationName,
    271                                                string &strError)
    272 {
    273     // Find domain
    274     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    275 
    276     if (!pConfigurableDomain) {
    277 
    278         return false;
    279     }
    280     // Delegate
    281     return pConfigurableDomain->renameConfiguration(strConfigurationName, strNewConfigurationName,
    282                                                     strError);
    283 }
    284 
    285 bool CConfigurableDomains::listDomainElements(const string &strDomain, string &strResult) const
    286 {
    287     // Find domain
    288     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strResult);
    289 
    290     if (!pConfigurableDomain) {
    291 
    292         return false;
    293     }
    294     // Delegate
    295     pConfigurableDomain->listAssociatedToElements(strResult);
    296 
    297     return true;
    298 }
    299 
    300 bool CConfigurableDomains::split(const string &domainName, CConfigurableElement *element,
    301                                  core::Results &infos)
    302 {
    303     // Find domain
    304     std::string error;
    305     CConfigurableDomain *domain = findConfigurableDomain(domainName, error);
    306 
    307     if (domain == nullptr) {
    308 
    309         infos.push_back(error);
    310         return false;
    311     }
    312     // Delegate
    313     domain->split(element, infos);
    314 
    315     return true;
    316 }
    317 
    318 void CConfigurableDomains::listAssociatedElements(string &strResult) const
    319 {
    320     std::set<const CConfigurableElement *> configurableElementSet;
    321 
    322     // Get all owned configurable elements
    323     gatherAllOwnedConfigurableElements(configurableElementSet);
    324 
    325     // Fill result
    326     std::set<const CConfigurableElement *>::const_iterator it;
    327 
    328     for (it = configurableElementSet.begin(); it != configurableElementSet.end(); ++it) {
    329 
    330         const CConfigurableElement *pConfigurableElement = *it;
    331 
    332         string strAssociatedDomainList;
    333 
    334         pConfigurableElement->listAssociatedDomains(strAssociatedDomainList, false);
    335 
    336         strResult += pConfigurableElement->getPath() + " [" + strAssociatedDomainList + "]\n";
    337     }
    338 }
    339 
    340 void CConfigurableDomains::listConflictingElements(string &strResult) const
    341 {
    342     std::set<const CConfigurableElement *> configurableElementSet;
    343 
    344     // Get all owned configurable elements
    345     gatherAllOwnedConfigurableElements(configurableElementSet);
    346 
    347     // Fill result
    348     std::set<const CConfigurableElement *>::const_iterator it;
    349 
    350     for (it = configurableElementSet.begin(); it != configurableElementSet.end(); ++it) {
    351 
    352         const CConfigurableElement *pConfigurableElement = *it;
    353 
    354         if (pConfigurableElement->getBelongingDomainCount() > 1) {
    355 
    356             string strBelongingDomainList;
    357 
    358             pConfigurableElement->listBelongingDomains(strBelongingDomainList, false);
    359 
    360             strResult += pConfigurableElement->getPath() + " contained in multiple domains: " +
    361                          strBelongingDomainList + "\n";
    362         }
    363     }
    364 }
    365 
    366 void CConfigurableDomains::listDomains(string &strResult) const
    367 {
    368     // List domains
    369     size_t uiNbConfigurableDomains = getNbChildren();
    370 
    371     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
    372 
    373         const CConfigurableDomain *pChildConfigurableDomain =
    374             static_cast<const CConfigurableDomain *>(getChild(child));
    375 
    376         // Name
    377         strResult += pChildConfigurableDomain->getName();
    378 
    379         // Sequence awareness
    380         if (pChildConfigurableDomain->getSequenceAwareness()) {
    381 
    382             strResult += " [sequence aware]";
    383         }
    384         strResult += "\n";
    385     }
    386 }
    387 
    388 // Gather configurable elements owned by any domain
    389 void CConfigurableDomains::gatherAllOwnedConfigurableElements(
    390     std::set<const CConfigurableElement *> &configurableElementSet) const
    391 {
    392     // Delegate to domains
    393     size_t uiNbConfigurableDomains = getNbChildren();
    394 
    395     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
    396 
    397         const CConfigurableDomain *pChildConfigurableDomain =
    398             static_cast<const CConfigurableDomain *>(getChild(child));
    399 
    400         pChildConfigurableDomain->gatherConfigurableElements(configurableElementSet);
    401     }
    402 }
    403 
    404 // Config restore
    405 bool CConfigurableDomains::restoreConfiguration(const string &domainName,
    406                                                 const string &configurationName,
    407                                                 CParameterBlackboard *mainBlackboard, bool autoSync,
    408                                                 core::Results &errors) const
    409 {
    410     string error;
    411     // Find domain
    412     const CConfigurableDomain *domain = findConfigurableDomain(domainName, error);
    413 
    414     if (domain == nullptr) {
    415 
    416         errors.push_back(error);
    417         return false;
    418     }
    419     // Delegate
    420     return domain->restoreConfiguration(configurationName, mainBlackboard, autoSync, errors);
    421 }
    422 
    423 // Config save
    424 bool CConfigurableDomains::saveConfiguration(const string &strDomain,
    425                                              const string &strConfiguration,
    426                                              const CParameterBlackboard *pMainBlackboard,
    427                                              string &strError)
    428 {
    429     // Find domain
    430     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    431 
    432     if (!pConfigurableDomain) {
    433 
    434         return false;
    435     }
    436     // Delegate
    437     return pConfigurableDomain->saveConfiguration(strConfiguration, pMainBlackboard, strError);
    438 }
    439 
    440 bool CConfigurableDomains::setElementSequence(const string &strDomain,
    441                                               const string &strConfiguration,
    442                                               const std::vector<string> &astrNewElementSequence,
    443                                               string &strError)
    444 {
    445     // Find domain
    446     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    447 
    448     if (!pConfigurableDomain) {
    449 
    450         return false;
    451     }
    452 
    453     // Delegate to domain
    454     return pConfigurableDomain->setElementSequence(strConfiguration, astrNewElementSequence,
    455                                                    strError);
    456 }
    457 
    458 bool CConfigurableDomains::getElementSequence(const string &strDomain,
    459                                               const string &strConfiguration,
    460                                               string &strResult) const
    461 {
    462     // Find domain
    463     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strResult);
    464 
    465     if (!pConfigurableDomain) {
    466 
    467         return false;
    468     }
    469     // Delegate to domain
    470     return pConfigurableDomain->getElementSequence(strConfiguration, strResult);
    471 }
    472 
    473 bool CConfigurableDomains::setApplicationRule(
    474     const string &strDomain, const string &strConfiguration, const string &strApplicationRule,
    475     const CSelectionCriteriaDefinition *pSelectionCriteriaDefinition, string &strError)
    476 {
    477     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    478 
    479     if (!pConfigurableDomain) {
    480 
    481         return false;
    482     }
    483 
    484     // Delegate to domain
    485     return pConfigurableDomain->setApplicationRule(strConfiguration, strApplicationRule,
    486                                                    pSelectionCriteriaDefinition, strError);
    487 }
    488 
    489 bool CConfigurableDomains::clearApplicationRule(const string &strDomain,
    490                                                 const string &strConfiguration, string &strError)
    491 {
    492     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    493 
    494     if (!pConfigurableDomain) {
    495 
    496         return false;
    497     }
    498 
    499     // Delegate to domain
    500     return pConfigurableDomain->clearApplicationRule(strConfiguration, strError);
    501 }
    502 
    503 bool CConfigurableDomains::getApplicationRule(const string &strDomain,
    504                                               const string &strConfiguration,
    505                                               string &strResult) const
    506 {
    507     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strResult);
    508 
    509     if (!pConfigurableDomain) {
    510 
    511         return false;
    512     }
    513 
    514     // Delegate to domain
    515     return pConfigurableDomain->getApplicationRule(strConfiguration, strResult);
    516 }
    517 
    518 // Last applied configurations
    519 void CConfigurableDomains::listLastAppliedConfigurations(string &strResult) const
    520 {
    521     // Browse domains
    522     size_t uiNbConfigurableDomains = getNbChildren();
    523 
    524     for (size_t child = 0; child < uiNbConfigurableDomains; child++) {
    525 
    526         const CConfigurableDomain *pChildConfigurableDomain =
    527             static_cast<const CConfigurableDomain *>(getChild(child));
    528 
    529         strResult += pChildConfigurableDomain->getName() + ": " +
    530                      pChildConfigurableDomain->getLastAppliedConfigurationName() + " [" +
    531                      pChildConfigurableDomain->getPendingConfigurationName() + "]\n";
    532     }
    533 }
    534 
    535 // Configurable element - domain association
    536 bool CConfigurableDomains::addConfigurableElementToDomain(
    537     const string &domainName, CConfigurableElement *element,
    538     const CParameterBlackboard *mainBlackboard, core::Results &infos)
    539 {
    540     // Find domain
    541     std::string error;
    542     CConfigurableDomain *domain = findConfigurableDomain(domainName, error);
    543 
    544     if (domain == nullptr) {
    545 
    546         infos.push_back(error);
    547         return false;
    548     }
    549     // Delegate
    550     return domain->addConfigurableElement(element, mainBlackboard, infos);
    551 }
    552 
    553 bool CConfigurableDomains::removeConfigurableElementFromDomain(
    554     const string &strDomain, CConfigurableElement *pConfigurableElement, string &strError)
    555 {
    556     // Find domain
    557     CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    558 
    559     if (!pConfigurableDomain) {
    560 
    561         return false;
    562     }
    563     // Delegate
    564     return pConfigurableDomain->removeConfigurableElement(pConfigurableElement, strError);
    565 }
    566 
    567 CParameterBlackboard *CConfigurableDomains::findConfigurationBlackboard(
    568     const string &strDomain, const string &strConfiguration,
    569     const CConfigurableElement *pConfigurableElement, size_t &baseOffset, bool &bIsLastApplied,
    570     string &strError) const
    571 {
    572     // Find domain
    573     const CConfigurableDomain *pConfigurableDomain = findConfigurableDomain(strDomain, strError);
    574 
    575     if (!pConfigurableDomain) {
    576 
    577         return nullptr;
    578     }
    579 
    580     // Check that element belongs to the domain
    581     if (!pConfigurableElement->belongsTo(pConfigurableDomain)) {
    582 
    583         strError = "Element \"" + pConfigurableElement->getPath() +
    584                    "\" does not belong to domain \"" + strDomain + "\"";
    585 
    586         return nullptr;
    587     }
    588 
    589     // Find Configuration Blackboard and Base Offset
    590     return pConfigurableDomain->findConfigurationBlackboard(strConfiguration, pConfigurableElement,
    591                                                             baseOffset, bIsLastApplied, strError);
    592 }
    593 
    594 // Domain retrieval
    595 CConfigurableDomain *CConfigurableDomains::findConfigurableDomain(const string &strDomain,
    596                                                                   string &strError)
    597 {
    598     // Call the const equivalent
    599     return const_cast<CConfigurableDomain *>(
    600         static_cast<const CConfigurableDomains *>(this)->findConfigurableDomain(strDomain,
    601                                                                                 strError));
    602 }
    603 
    604 const CConfigurableDomain *CConfigurableDomains::findConfigurableDomain(const string &strDomain,
    605                                                                         string &strError) const
    606 {
    607     // Find domain
    608     const CConfigurableDomain *pConfigurableDomain =
    609         static_cast<const CConfigurableDomain *>(findChild(strDomain));
    610 
    611     if (!pConfigurableDomain) {
    612 
    613         strError = "Configurable domain " + strDomain + " not found";
    614 
    615         return nullptr;
    616     }
    617 
    618     return pConfigurableDomain;
    619 }
    620