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 "ConfigurableDomain.h" 31 #include "DomainConfiguration.h" 32 #include "ConfigurableElement.h" 33 #include "ConfigurationAccessContext.h" 34 #include "XmlDomainSerializingContext.h" 35 #include "XmlDomainImportContext.h" 36 #include "XmlDomainExportContext.h" 37 #include "Utility.h" 38 #include "AlwaysAssert.hpp" 39 #include <cassert> 40 41 #define base CElement 42 43 using std::string; 44 45 CConfigurableDomain::CConfigurableDomain(const string &strName) : base(strName) 46 { 47 } 48 49 CConfigurableDomain::~CConfigurableDomain() 50 { 51 // Remove all configurable elements 52 ConfigurableElementListIterator it; 53 54 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 55 56 CConfigurableElement *pConfigurableElement = *it; 57 58 // Remove from configurable element 59 pConfigurableElement->removeAttachedConfigurableDomain(this); 60 } 61 62 // Remove all associated syncer sets 63 ConfigurableElementToSyncerSetMapIterator mapIt; 64 65 for (mapIt = _configurableElementToSyncerSetMap.begin(); 66 mapIt != _configurableElementToSyncerSetMap.end(); ++mapIt) { 67 68 delete mapIt->second; 69 } 70 } 71 72 string CConfigurableDomain::getKind() const 73 { 74 return "ConfigurableDomain"; 75 } 76 77 bool CConfigurableDomain::childrenAreDynamic() const 78 { 79 return true; 80 } 81 82 // Content dumping 83 string CConfigurableDomain::logValue(utility::ErrorContext & /*ctx*/) const 84 { 85 return string("{") + 86 87 "Sequence aware: " + (_bSequenceAware ? "yes" : "no") + 88 89 ", Last applied configuration: " + 90 (_pLastAppliedConfiguration ? _pLastAppliedConfiguration->getName() : "<none>") + 91 92 "}"; 93 } 94 95 // Sequence awareness 96 void CConfigurableDomain::setSequenceAwareness(bool bSequenceAware) 97 { 98 if (_bSequenceAware != bSequenceAware) { 99 100 _bSequenceAware = bSequenceAware; 101 } 102 } 103 104 bool CConfigurableDomain::getSequenceAwareness() const 105 { 106 return _bSequenceAware; 107 } 108 109 // From IXmlSource 110 void CConfigurableDomain::toXml(CXmlElement &xmlElement, 111 CXmlSerializingContext &serializingContext) const 112 { 113 base::toXml(xmlElement, serializingContext); 114 115 // Sequence awareness 116 xmlElement.setAttribute("SequenceAware", _bSequenceAware); 117 } 118 119 void CConfigurableDomain::childrenToXml(CXmlElement &xmlElement, 120 CXmlSerializingContext &serializingContext) const 121 { 122 // Configurations 123 composeDomainConfigurations(xmlElement, serializingContext); 124 125 // Configurable Elements 126 composeConfigurableElements(xmlElement); 127 128 // Settings 129 composeSettings(xmlElement, static_cast<CXmlDomainExportContext &>(serializingContext)); 130 } 131 132 // XML composing 133 void CConfigurableDomain::composeDomainConfigurations( 134 CXmlElement &xmlElement, CXmlSerializingContext &serializingContext) const 135 { 136 // Create Configurations element 137 CXmlElement xmlConfigurationsElement; 138 139 xmlElement.createChild(xmlConfigurationsElement, "Configurations"); 140 141 // Delegate to base 142 base::childrenToXml(xmlConfigurationsElement, serializingContext); 143 } 144 145 void CConfigurableDomain::composeConfigurableElements(CXmlElement &xmlElement) const 146 { 147 // Create ConfigurableElements element 148 CXmlElement xmlConfigurableElementsElement; 149 150 xmlElement.createChild(xmlConfigurableElementsElement, "ConfigurableElements"); 151 152 // Serialize out all configurable elements settings 153 ConfigurableElementListIterator it; 154 155 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 156 157 const CConfigurableElement *pConfigurableElement = *it; 158 159 // Create corresponding XML child element 160 CXmlElement xmlChildConfigurableElement; 161 162 xmlConfigurableElementsElement.createChild(xmlChildConfigurableElement, 163 "ConfigurableElement"); 164 165 // Set Path attribute 166 xmlChildConfigurableElement.setAttribute("Path", pConfigurableElement->getPath()); 167 } 168 } 169 170 void CConfigurableDomain::composeSettings(CXmlElement &xmlElement, 171 CXmlDomainExportContext &context) const 172 { 173 if (!context.withSettings()) { 174 175 return; 176 } 177 178 // Create Settings element 179 CXmlElement xmlSettingsElement; 180 181 xmlElement.createChild(xmlSettingsElement, "Settings"); 182 183 // Serialize out all configurations settings 184 size_t uiNbConfigurations = getNbChildren(); 185 size_t uiChildConfiguration; 186 187 for (uiChildConfiguration = 0; uiChildConfiguration < uiNbConfigurations; 188 uiChildConfiguration++) { 189 190 const CDomainConfiguration *pDomainConfiguration = 191 static_cast<const CDomainConfiguration *>(getChild(uiChildConfiguration)); 192 193 // Create child xml element for that configuration 194 CXmlElement xmlConfigurationSettingsElement; 195 196 xmlSettingsElement.createChild(xmlConfigurationSettingsElement, 197 pDomainConfiguration->getXmlElementName()); 198 199 // Set its name attribute 200 xmlConfigurationSettingsElement.setNameAttribute(pDomainConfiguration->getName()); 201 202 // Serialize out configuration settings 203 pDomainConfiguration->composeSettings(xmlConfigurationSettingsElement, context); 204 } 205 } 206 207 // From IXmlSink 208 bool CConfigurableDomain::fromXml(const CXmlElement &xmlElement, 209 CXmlSerializingContext &serializingContext) 210 { 211 // Context 212 CXmlDomainImportContext &xmlDomainImportContext = 213 static_cast<CXmlDomainImportContext &>(serializingContext); 214 215 // Sequence awareness (optional) 216 xmlElement.getAttribute("SequenceAware", _bSequenceAware); 217 218 std::string name; 219 xmlElement.getAttribute("Name", name); 220 setName(name); 221 222 // Local parsing. Do not dig 223 if (!parseDomainConfigurations(xmlElement, xmlDomainImportContext) || 224 !parseConfigurableElements(xmlElement, xmlDomainImportContext) || 225 !parseSettings(xmlElement, xmlDomainImportContext)) { 226 227 return false; 228 } 229 230 // All provided configurations are parsed 231 // Attempt validation on areas of non provided configurations for all configurable elements if 232 // required 233 if (xmlDomainImportContext.autoValidationRequired()) { 234 235 autoValidateAll(); 236 } 237 238 return true; 239 } 240 241 // XML parsing 242 bool CConfigurableDomain::parseDomainConfigurations(const CXmlElement &xmlElement, 243 CXmlDomainImportContext &serializingContext) 244 { 245 // We're supposedly clean 246 assert(_configurableElementList.empty()); 247 248 // Get Configurations element 249 CXmlElement xmlConfigurationsElement; 250 251 xmlElement.getChildElement("Configurations", xmlConfigurationsElement); 252 253 // Parse it and create domain configuration objects 254 return base::fromXml(xmlConfigurationsElement, serializingContext); 255 } 256 257 // Parse configurable elements 258 bool CConfigurableDomain::parseConfigurableElements(const CXmlElement &xmlElement, 259 CXmlDomainImportContext &serializingContext) 260 { 261 CSystemClass &systemClass = serializingContext.getSystemClass(); 262 263 // Get ConfigurableElements element 264 CXmlElement xmlConfigurableElementsElement; 265 xmlElement.getChildElement("ConfigurableElements", xmlConfigurableElementsElement); 266 267 // Parse it and associate found configurable elements to it 268 CXmlElement::CChildIterator it(xmlConfigurableElementsElement); 269 270 CXmlElement xmlConfigurableElementElement; 271 272 while (it.next(xmlConfigurableElementElement)) { 273 274 // Locate configurable element 275 string strConfigurableElementPath; 276 xmlConfigurableElementElement.getAttribute("Path", strConfigurableElementPath); 277 278 CPathNavigator pathNavigator(strConfigurableElementPath); 279 string strError; 280 281 // Is there an element and does it match system class name? 282 if (!pathNavigator.navigateThrough(systemClass.getName(), strError)) { 283 284 serializingContext.setError( 285 "Could not find configurable element of path " + strConfigurableElementPath + 286 " from ConfigurableDomain description " + getName() + " (" + strError + ")"); 287 288 return false; 289 } 290 // Browse system class for configurable element 291 CConfigurableElement *pConfigurableElement = 292 static_cast<CConfigurableElement *>(systemClass.findDescendant(pathNavigator)); 293 294 if (!pConfigurableElement) { 295 296 serializingContext.setError("Could not find configurable element of path " + 297 strConfigurableElementPath + 298 " from ConfigurableDomain description " + getName()); 299 300 return false; 301 } 302 // Add found element to domain 303 core::Results infos; 304 if (!addConfigurableElement(pConfigurableElement, nullptr, infos)) { 305 306 strError = utility::asString(infos); 307 serializingContext.setError(strError); 308 309 return false; 310 } 311 } 312 313 return true; 314 } 315 316 // Parse settings 317 bool CConfigurableDomain::parseSettings(const CXmlElement &xmlElement, 318 CXmlDomainImportContext &serializingContext) 319 { 320 // Check we actually need to parse configuration settings 321 if (!serializingContext.withSettings()) { 322 323 // No parsing required 324 return true; 325 } 326 327 // Get Settings element 328 CXmlElement xmlSettingsElement; 329 if (!xmlElement.getChildElement("Settings", xmlSettingsElement)) { 330 331 // No settings, bail out successfully 332 return true; 333 } 334 335 // Parse configuration settings 336 CXmlElement::CChildIterator it(xmlSettingsElement); 337 338 CXmlElement xmlConfigurationSettingsElement; 339 340 while (it.next(xmlConfigurationSettingsElement)) { 341 // Get domain configuration 342 CDomainConfiguration *pDomainConfiguration = static_cast<CDomainConfiguration *>( 343 findChild(xmlConfigurationSettingsElement.getNameAttribute())); 344 345 if (!pDomainConfiguration) { 346 347 serializingContext.setError("Could not find domain configuration referred to by" 348 " configurable domain \"" + 349 getName() + "\"."); 350 351 return false; 352 } 353 // Have domain configuration parse settings for all configurable elements 354 if (!pDomainConfiguration->parseSettings(xmlConfigurationSettingsElement, 355 serializingContext)) { 356 357 return false; 358 } 359 } 360 361 return true; 362 } 363 // Configurable elements association 364 bool CConfigurableDomain::addConfigurableElement(CConfigurableElement *pConfigurableElement, 365 const CParameterBlackboard *pMainBlackboard, 366 core::Results &infos) 367 { 368 // Already associated? 369 if (containsConfigurableElement(pConfigurableElement)) { 370 371 infos.push_back("Configurable element " + pConfigurableElement->getPath() + 372 " already associated to configuration domain " + getName()); 373 374 return false; 375 } 376 377 // Already owned? 378 if (pConfigurableElement->belongsTo(this)) { 379 380 infos.push_back("Configurable element " + pConfigurableElement->getPath() + 381 " already owned by configuration domain " + getName()); 382 383 return false; 384 } 385 386 // Do add 387 doAddConfigurableElement(pConfigurableElement, infos, pMainBlackboard); 388 389 return true; 390 } 391 392 bool CConfigurableDomain::removeConfigurableElement(CConfigurableElement *pConfigurableElement, 393 string &strError) 394 { 395 // Not associated? 396 if (!containsConfigurableElement(pConfigurableElement)) { 397 398 strError = "Configurable element " + pConfigurableElement->getPath() + 399 " not associated to configuration domain " + getName(); 400 401 return false; 402 } 403 404 // Do remove 405 doRemoveConfigurableElement(pConfigurableElement, true); 406 407 return true; 408 } 409 410 /** 411 * Blackboard Configuration and Base Offset retrieval. 412 * 413 * This method fetches the Blackboard associated to the ConfigurableElement 414 * given in parameter, for a specific Configuration. The ConfigurableElement 415 * must belong to the Domain. If a Blackboard is found, the base offset of 416 * the ConfigurableElement is returned as well. This base offset corresponds to 417 * the offset of the ancestor of the ConfigurableElement associated to the Configuration. 418 * 419 * @param[in] strConfiguration Name of the Configuration. 420 * @param[in] pCandidateDescendantConfigurableElement Pointer to a CConfigurableElement that 421 * belongs to the Domain. 422 * @param[out] baseOffset The base offset of the CConfigurableElement. 423 * @param[out] bIsLastApplied Boolean indicating that the Configuration is 424 * the last one applied of the Domain. 425 * @param[out] strError Error message 426 * 427 * return Pointer to the Blackboard of the Configuration. 428 */ 429 CParameterBlackboard *CConfigurableDomain::findConfigurationBlackboard( 430 const string &strConfiguration, 431 const CConfigurableElement *pCandidateDescendantConfigurableElement, size_t &baseOffset, 432 bool &bIsLastApplied, string &strError) const 433 { 434 // Find Configuration 435 const CDomainConfiguration *pDomainConfiguration = 436 static_cast<const CDomainConfiguration *>(findChild(strConfiguration)); 437 438 if (!pDomainConfiguration) { 439 440 strError = "Domain configuration " + strConfiguration + " not found"; 441 442 return nullptr; 443 } 444 445 // Parse all configurable elements 446 ConfigurableElementListIterator it; 447 448 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 449 450 const CConfigurableElement *pAssociatedConfigurableElement = *it; 451 452 // Check if the the associated element is the configurable element or one of its ancestors 453 if ((pCandidateDescendantConfigurableElement == pAssociatedConfigurableElement) || 454 (pCandidateDescendantConfigurableElement->isDescendantOf( 455 pAssociatedConfigurableElement))) { 456 457 baseOffset = pAssociatedConfigurableElement->getOffset(); 458 bIsLastApplied = (pDomainConfiguration == _pLastAppliedConfiguration); 459 460 return pDomainConfiguration->getBlackboard(pAssociatedConfigurableElement); 461 } 462 } 463 464 strError = "Element not associated to the Domain"; 465 466 return nullptr; 467 } 468 469 // Domain splitting 470 bool CConfigurableDomain::split(CConfigurableElement *pConfigurableElement, core::Results &infos) 471 { 472 // Not associated? 473 if (!containsConfigurableElement(pConfigurableElement)) { 474 475 std::string strError = "Configurable element " + pConfigurableElement->getPath() + 476 " not associated to configuration domain " + getName(); 477 infos.push_back(strError); 478 479 return false; 480 } 481 482 // Create sub domain areas for all configurable element's children 483 size_t uiNbConfigurableElementChildren = pConfigurableElement->getNbChildren(); 484 485 if (!uiNbConfigurableElementChildren) { 486 487 std::string strError = "Configurable element " + pConfigurableElement->getPath() + 488 " has no children to split configurable domain to"; 489 infos.push_back(strError); 490 491 return false; 492 } 493 494 for (size_t uiChild = 0; uiChild < uiNbConfigurableElementChildren; uiChild++) { 495 496 CConfigurableElement *pChildConfigurableElement = 497 static_cast<CConfigurableElement *>(pConfigurableElement->getChild(uiChild)); 498 499 doAddConfigurableElement(pChildConfigurableElement, infos); 500 } 501 502 // Delegate to configurations 503 size_t uiNbConfigurations = getNbChildren(); 504 505 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 506 507 CDomainConfiguration *pDomainConfiguration = 508 static_cast<CDomainConfiguration *>(getChild(uiChild)); 509 510 pDomainConfiguration->split(pConfigurableElement); 511 } 512 513 // Remove given configurable element from this domain 514 // Note: we shouldn't need to recompute the sync set in that case, as the splitted element 515 // should include the syncers of its children elements 516 doRemoveConfigurableElement(pConfigurableElement, false); 517 518 return true; 519 } 520 521 // Check if there is a pending configuration for this domain: i.e. an applicable configuration 522 // different from the last applied configuration 523 const CDomainConfiguration *CConfigurableDomain::getPendingConfiguration() const 524 { 525 const CDomainConfiguration *pApplicableDomainConfiguration = 526 findApplicableDomainConfiguration(); 527 528 if (pApplicableDomainConfiguration) { 529 530 // Check not the last one before applying 531 if (!_pLastAppliedConfiguration || 532 (_pLastAppliedConfiguration != pApplicableDomainConfiguration)) { 533 534 return pApplicableDomainConfiguration; 535 } 536 } 537 538 return nullptr; 539 } 540 541 // Configuration application if required 542 void CConfigurableDomain::apply(CParameterBlackboard *pParameterBlackboard, CSyncerSet *pSyncerSet, 543 bool bForce, std::string &strInfo) const 544 { 545 // Apply configuration only if the blackboard will 546 // be synchronized either now or by syncerSet. 547 if (!pSyncerSet ^ _bSequenceAware) { 548 // The configuration can not be syncronised 549 return; 550 } 551 552 if (bForce) { 553 // Force a configuration restore by forgetting about last applied configuration 554 _pLastAppliedConfiguration = nullptr; 555 } 556 const CDomainConfiguration *pApplicableDomainConfiguration = 557 findApplicableDomainConfiguration(); 558 559 if (pApplicableDomainConfiguration) { 560 561 // Check not the last one before applying 562 if (!_pLastAppliedConfiguration || 563 _pLastAppliedConfiguration != pApplicableDomainConfiguration) { 564 565 strInfo = "Applying configuration '" + pApplicableDomainConfiguration->getName() + 566 "' from domain '" + getName() + "'"; 567 568 // Check if we need to synchronize during restore 569 bool bSync = !pSyncerSet && _bSequenceAware; 570 571 // Do the restore 572 pApplicableDomainConfiguration->restore(pParameterBlackboard, bSync, nullptr); 573 574 // Record last applied configuration 575 _pLastAppliedConfiguration = pApplicableDomainConfiguration; 576 577 // Check we need to provide syncer set to caller 578 if (pSyncerSet && !_bSequenceAware) { 579 580 // Since we applied changes, add our own sync set to the given one 581 *pSyncerSet += _syncerSet; 582 } 583 } 584 } 585 } 586 587 // Return applicable configuration validity for given configurable element 588 bool CConfigurableDomain::isApplicableConfigurationValid( 589 const CConfigurableElement *pConfigurableElement) const 590 { 591 const CDomainConfiguration *pApplicableDomainConfiguration = 592 findApplicableDomainConfiguration(); 593 594 return pApplicableDomainConfiguration && 595 pApplicableDomainConfiguration->isValid(pConfigurableElement); 596 } 597 598 // In case configurable element was removed 599 void CConfigurableDomain::computeSyncSet() 600 { 601 // Clean sync set first 602 _syncerSet.clear(); 603 604 // Add syncer sets for all associated configurable elements 605 ConfigurableElementToSyncerSetMapIterator mapIt; 606 607 for (mapIt = _configurableElementToSyncerSetMap.begin(); 608 mapIt != _configurableElementToSyncerSetMap.end(); ++mapIt) { 609 610 const CSyncerSet *pSyncerSet = mapIt->second; 611 612 _syncerSet += *pSyncerSet; 613 } 614 } 615 616 // Configuration Management 617 bool CConfigurableDomain::createConfiguration(const string &strName, 618 const CParameterBlackboard *pMainBlackboard, 619 string &strError) 620 { 621 // Already exists? 622 if (findChild(strName)) { 623 624 strError = "Already existing configuration"; 625 626 return false; 627 } 628 629 // Creation 630 auto pDomainConfiguration = new CDomainConfiguration(strName); 631 632 // Configurable elements association 633 ConfigurableElementListIterator it; 634 635 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 636 637 const CConfigurableElement *pConfigurableElement = *it; 638 ; 639 640 // Retrieve associated syncer set 641 CSyncerSet *pSyncerSet = getSyncerSet(pConfigurableElement); 642 643 // Associate to configuration 644 pDomainConfiguration->addConfigurableElement(pConfigurableElement, pSyncerSet); 645 } 646 647 // Hierarchy 648 addChild(pDomainConfiguration); 649 650 // Ensure validity of fresh new domain configuration 651 // Attempt auto validation, so that the user gets his/her own settings by defaults 652 if (!autoValidateConfiguration(pDomainConfiguration)) { 653 654 // No valid configuration found to copy in from, validate againt main blackboard (will 655 // concerned remaining invalid parts) 656 pDomainConfiguration->validate(pMainBlackboard); 657 } 658 659 return true; 660 } 661 662 bool CConfigurableDomain::deleteConfiguration(const string &strName, string &strError) 663 { 664 CDomainConfiguration *pDomainConfiguration = findConfiguration(strName, strError); 665 666 if (!pDomainConfiguration) { 667 668 return false; 669 } 670 671 // Was the last applied? 672 if (pDomainConfiguration == _pLastAppliedConfiguration) { 673 674 // Forget about it 675 _pLastAppliedConfiguration = nullptr; 676 } 677 678 // Hierarchy 679 removeChild(pDomainConfiguration); 680 681 // Destroy 682 delete pDomainConfiguration; 683 684 return true; 685 } 686 687 void CConfigurableDomain::listAssociatedToElements(string &strResult) const 688 { 689 ConfigurableElementListIterator it; 690 691 // Browse all configurable elements 692 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 693 694 const CConfigurableElement *pConfigurableElement = *it; 695 696 strResult += pConfigurableElement->getPath() + "\n"; 697 } 698 } 699 700 bool CConfigurableDomain::renameConfiguration(const string &strName, const string &strNewName, 701 string &strError) 702 { 703 CDomainConfiguration *pDomainConfiguration = findConfiguration(strName, strError); 704 705 if (!pDomainConfiguration) { 706 707 return false; 708 } 709 710 // Rename 711 return pDomainConfiguration->rename(strNewName, strError); 712 } 713 714 bool CConfigurableDomain::restoreConfiguration(const string &configurationName, 715 CParameterBlackboard *mainBlackboard, bool autoSync, 716 core::Results &errors) const 717 { 718 string error; 719 720 const CDomainConfiguration *configuration = findConfiguration(configurationName, error); 721 722 if (configuration == nullptr) { 723 724 errors.push_back(error); 725 return false; 726 } 727 728 // Delegate 729 bool bSuccess = configuration->restore(mainBlackboard, autoSync && _bSequenceAware, &errors); 730 731 // Record last applied configuration 732 _pLastAppliedConfiguration = configuration; 733 734 // Synchronize 735 if (autoSync && !_bSequenceAware) { 736 737 bSuccess &= _syncerSet.sync(*mainBlackboard, false, &errors); 738 } 739 return bSuccess; 740 } 741 742 bool CConfigurableDomain::saveConfiguration(const string &strName, 743 const CParameterBlackboard *pMainBlackboard, 744 string &strError) 745 { 746 // Find Domain configuration 747 CDomainConfiguration *pDomainConfiguration = findConfiguration(strName, strError); 748 749 if (!pDomainConfiguration) { 750 751 return false; 752 } 753 754 // Delegate 755 pDomainConfiguration->save(pMainBlackboard); 756 757 return true; 758 } 759 760 bool CConfigurableDomain::setElementSequence(const string &strConfiguration, 761 const std::vector<string> &astrNewElementSequence, 762 string &strError) 763 { 764 // Find Domain configuration 765 CDomainConfiguration *pDomainConfiguration = findConfiguration(strConfiguration, strError); 766 767 if (!pDomainConfiguration) { 768 769 return false; 770 } 771 772 // Delegate to configuration 773 return pDomainConfiguration->setElementSequence(astrNewElementSequence, strError); 774 } 775 776 bool CConfigurableDomain::getElementSequence(const string &strConfiguration, 777 string &strResult) const 778 { 779 // Find Domain configuration 780 const CDomainConfiguration *pDomainConfiguration = 781 findConfiguration(strConfiguration, strResult); 782 783 if (!pDomainConfiguration) { 784 785 return false; 786 } 787 788 // Delegate to configuration 789 pDomainConfiguration->getElementSequence(strResult); 790 791 return true; 792 } 793 794 bool CConfigurableDomain::setApplicationRule( 795 const string &strConfiguration, const string &strApplicationRule, 796 const CSelectionCriteriaDefinition *pSelectionCriteriaDefinition, string &strError) 797 { 798 // Find Domain configuration 799 CDomainConfiguration *pDomainConfiguration = findConfiguration(strConfiguration, strError); 800 801 if (!pDomainConfiguration) { 802 803 return false; 804 } 805 806 // Delegate to configuration 807 return pDomainConfiguration->setApplicationRule(strApplicationRule, 808 pSelectionCriteriaDefinition, strError); 809 } 810 811 bool CConfigurableDomain::clearApplicationRule(const string &strConfiguration, string &strError) 812 { 813 // Find Domain configuration 814 CDomainConfiguration *pDomainConfiguration = findConfiguration(strConfiguration, strError); 815 816 if (!pDomainConfiguration) { 817 818 return false; 819 } 820 821 // Delegate to configuration 822 pDomainConfiguration->clearApplicationRule(); 823 824 return true; 825 } 826 827 bool CConfigurableDomain::getApplicationRule(const string &strConfiguration, 828 string &strResult) const 829 { 830 // Find Domain configuration 831 const CDomainConfiguration *pDomainConfiguration = 832 findConfiguration(strConfiguration, strResult); 833 834 if (!pDomainConfiguration) { 835 836 return false; 837 } 838 839 // Delegate to configuration 840 strResult = pDomainConfiguration->getApplicationRule(); 841 842 return true; 843 } 844 845 // Last applied configuration 846 string CConfigurableDomain::getLastAppliedConfigurationName() const 847 { 848 if (_pLastAppliedConfiguration) { 849 850 return _pLastAppliedConfiguration->getName(); 851 } 852 return "<none>"; 853 } 854 855 // Pending configuration 856 string CConfigurableDomain::getPendingConfigurationName() const 857 { 858 const CDomainConfiguration *pApplicableDomainConfiguration = 859 findApplicableDomainConfiguration(); 860 861 if (!pApplicableDomainConfiguration) { 862 863 // No configuration is pending 864 return "<none>"; 865 } 866 867 // Check it will be applied 868 if (pApplicableDomainConfiguration != _pLastAppliedConfiguration) { 869 870 // Found config will get applied 871 return pApplicableDomainConfiguration->getName(); 872 } else { 873 874 // Same configuration as current 875 return ""; 876 } 877 } 878 879 // Ensure validity on whole domain from main blackboard 880 void CConfigurableDomain::validate(const CParameterBlackboard *pMainBlackboard) 881 { 882 883 // Propagate 884 size_t uiNbConfigurations = getNbChildren(); 885 886 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 887 888 CDomainConfiguration *pDomainConfiguration = 889 static_cast<CDomainConfiguration *>(getChild(uiChild)); 890 891 pDomainConfiguration->validate(pMainBlackboard); 892 } 893 } 894 895 // Ensure validity on areas related to configurable element 896 void CConfigurableDomain::validateAreas(const CConfigurableElement *pConfigurableElement, 897 const CParameterBlackboard *pMainBlackboard) 898 { 899 // Propagate 900 size_t uiNbConfigurations = getNbChildren(); 901 902 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 903 904 CDomainConfiguration *pDomainConfiguration = 905 static_cast<CDomainConfiguration *>(getChild(uiChild)); 906 907 pDomainConfiguration->validate(pConfigurableElement, pMainBlackboard); 908 } 909 } 910 911 // Attempt validation for all configurable element's areas, relying on already existing valid 912 // configuration inside domain 913 void CConfigurableDomain::autoValidateAll() 914 { 915 // Validate 916 ConfigurableElementListIterator it; 917 918 // Browse all configurable elements for configuration validation 919 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 920 921 const CConfigurableElement *pConfigurableElement = *it; 922 923 // Auto validate element 924 autoValidateAreas(pConfigurableElement); 925 } 926 } 927 928 // Attempt validation for configurable element's areas, relying on already existing valid 929 // configuration inside domain 930 void CConfigurableDomain::autoValidateAreas(const CConfigurableElement *pConfigurableElement) 931 { 932 // Find first valid configuration for given configurable element 933 const CDomainConfiguration *pValidDomainConfiguration = 934 findValidDomainConfiguration(pConfigurableElement); 935 936 // No valid configuration found, give up 937 if (!pValidDomainConfiguration) { 938 939 return; 940 } 941 942 // Validate all other configurations against found one, if any 943 size_t uiNbConfigurations = getNbChildren(); 944 945 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 946 947 CDomainConfiguration *pDomainConfiguration = 948 static_cast<CDomainConfiguration *>(getChild(uiChild)); 949 950 if (pDomainConfiguration != pValidDomainConfiguration && 951 !pDomainConfiguration->isValid(pConfigurableElement)) { 952 // Validate 953 pDomainConfiguration->validateAgainst(pValidDomainConfiguration, pConfigurableElement); 954 } 955 } 956 } 957 958 // Attempt configuration validation for all configurable elements' areas, relying on already 959 // existing valid configuration inside domain 960 bool CConfigurableDomain::autoValidateConfiguration(CDomainConfiguration *pDomainConfiguration) 961 { 962 // Find another configuration than this one, that ought to be valid! 963 size_t uiNbConfigurations = getNbChildren(); 964 965 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 966 967 const CDomainConfiguration *pPotententialValidDomainConfiguration = 968 static_cast<const CDomainConfiguration *>(getChild(uiChild)); 969 970 if (pPotententialValidDomainConfiguration != pDomainConfiguration) { 971 972 // Validate against it 973 pDomainConfiguration->validateAgainst(pPotententialValidDomainConfiguration); 974 975 return true; 976 } 977 } 978 return false; 979 } 980 981 // Search for a valid configuration for given configurable element 982 const CDomainConfiguration *CConfigurableDomain::findValidDomainConfiguration( 983 const CConfigurableElement *pConfigurableElement) const 984 { 985 size_t uiNbConfigurations = getNbChildren(); 986 987 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 988 989 const CDomainConfiguration *pDomainConfiguration = 990 static_cast<const CDomainConfiguration *>(getChild(uiChild)); 991 992 if (pDomainConfiguration->isValid(pConfigurableElement)) { 993 994 return pDomainConfiguration; 995 } 996 } 997 return nullptr; 998 } 999 1000 // Search for an applicable configuration 1001 const CDomainConfiguration *CConfigurableDomain::findApplicableDomainConfiguration() const 1002 { 1003 size_t uiNbConfigurations = getNbChildren(); 1004 1005 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 1006 1007 const CDomainConfiguration *pDomainConfiguration = 1008 static_cast<const CDomainConfiguration *>(getChild(uiChild)); 1009 1010 if (pDomainConfiguration->isApplicable()) { 1011 1012 return pDomainConfiguration; 1013 } 1014 } 1015 return nullptr; 1016 } 1017 1018 // Gather set of configurable elements 1019 void CConfigurableDomain::gatherConfigurableElements( 1020 std::set<const CConfigurableElement *> &configurableElementSet) const 1021 { 1022 // Insert all configurable elements 1023 configurableElementSet.insert(_configurableElementList.begin(), _configurableElementList.end()); 1024 } 1025 1026 // Check configurable element already attached 1027 bool CConfigurableDomain::containsConfigurableElement( 1028 const CConfigurableElement *pConfigurableCandidateElement) const 1029 { 1030 ConfigurableElementListIterator it; 1031 1032 // Browse all configurable elements for comparison 1033 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 1034 1035 if (pConfigurableCandidateElement == *it) { 1036 1037 return true; 1038 } 1039 } 1040 return false; 1041 } 1042 1043 // Merge any descended configurable element to this one with this one 1044 void CConfigurableDomain::mergeAlreadyAssociatedDescendantConfigurableElements( 1045 CConfigurableElement *newElement, core::Results &infos) 1046 { 1047 std::list<CConfigurableElement *> mergedConfigurableElementList; 1048 1049 ConfigurableElementListIterator it; 1050 1051 // Browse all configurable elements (new one not yet in the list!) 1052 for (it = _configurableElementList.begin(); it != _configurableElementList.end(); ++it) { 1053 1054 CConfigurableElement *pConfigurablePotentialDescendantElement = *it; 1055 1056 if (pConfigurablePotentialDescendantElement->isDescendantOf(newElement)) { 1057 1058 infos.push_back("In domain '" + getName() + 1059 "', merging descendant configurable element's configurations '" + 1060 pConfigurablePotentialDescendantElement->getName() + 1061 "' into its ascendant '" + newElement->getName() + "' ones"); 1062 1063 // Merge configuration data 1064 mergeConfigurations(newElement, pConfigurablePotentialDescendantElement); 1065 1066 // Keep track for removal 1067 mergedConfigurableElementList.push_back(pConfigurablePotentialDescendantElement); 1068 } 1069 } 1070 1071 // Remove all merged elements (new one not yet in the list!) 1072 for (it = mergedConfigurableElementList.begin(); it != mergedConfigurableElementList.end(); 1073 ++it) { 1074 1075 CConfigurableElement *pMergedConfigurableElement = *it; 1076 1077 // Remove merged from configurable element from internal tracking list 1078 // Note: we shouldn't need to recompute the sync set in that case, as the merged to element 1079 // should include the syncers of merged from elements 1080 doRemoveConfigurableElement(pMergedConfigurableElement, false); 1081 } 1082 } 1083 1084 void CConfigurableDomain::mergeConfigurations(CConfigurableElement *pToConfigurableElement, 1085 CConfigurableElement *pFromConfigurableElement) 1086 { 1087 // Propagate to domain configurations 1088 size_t uiNbConfigurations = getNbChildren(); 1089 1090 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 1091 1092 CDomainConfiguration *pDomainConfiguration = 1093 static_cast<CDomainConfiguration *>(getChild(uiChild)); 1094 1095 // Do the merge. 1096 pDomainConfiguration->merge(pToConfigurableElement, pFromConfigurableElement); 1097 } 1098 } 1099 1100 // Configurable elements association 1101 void CConfigurableDomain::doAddConfigurableElement(CConfigurableElement *pConfigurableElement, 1102 core::Results &infos, 1103 const CParameterBlackboard *pMainBlackboard) 1104 { 1105 // Inform configurable element 1106 pConfigurableElement->addAttachedConfigurableDomain(this); 1107 1108 // Create associated syncer set 1109 auto pSyncerSet = new CSyncerSet; 1110 1111 // Add to sync set the configurable element one 1112 pConfigurableElement->fillSyncerSet(*pSyncerSet); 1113 1114 // Store it 1115 _configurableElementToSyncerSetMap[pConfigurableElement] = pSyncerSet; 1116 1117 // Add it to global one 1118 _syncerSet += *pSyncerSet; 1119 1120 // Inform configurations 1121 size_t uiNbConfigurations = getNbChildren(); 1122 1123 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 1124 1125 CDomainConfiguration *pDomainConfiguration = 1126 static_cast<CDomainConfiguration *>(getChild(uiChild)); 1127 1128 pDomainConfiguration->addConfigurableElement(pConfigurableElement, pSyncerSet); 1129 } 1130 1131 // Ensure area validity for that configurable element (if main blackboard provided) 1132 if (pMainBlackboard) { 1133 1134 infos.push_back("Validating domain '" + getName() + 1135 "' against main blackboard for configurable element '" + 1136 pConfigurableElement->getPath() + "'"); 1137 // Need to validate against main blackboard 1138 validateAreas(pConfigurableElement, pMainBlackboard); 1139 } 1140 1141 // Already associated descendend configurable elements need a merge of their configuration data 1142 mergeAlreadyAssociatedDescendantConfigurableElements(pConfigurableElement, infos); 1143 1144 // Add to list 1145 _configurableElementList.push_back(pConfigurableElement); 1146 } 1147 1148 void CConfigurableDomain::doRemoveConfigurableElement(CConfigurableElement *pConfigurableElement, 1149 bool bRecomputeSyncSet) 1150 { 1151 // Remove from list 1152 _configurableElementList.remove(pConfigurableElement); 1153 1154 // Remove associated syncer set 1155 CSyncerSet *pSyncerSet = getSyncerSet(pConfigurableElement); 1156 1157 _configurableElementToSyncerSetMap.erase(pConfigurableElement); 1158 1159 delete pSyncerSet; 1160 1161 // Inform configurable element 1162 pConfigurableElement->removeAttachedConfigurableDomain(this); 1163 1164 // Inform configurations 1165 size_t uiNbConfigurations = getNbChildren(); 1166 1167 for (size_t uiChild = 0; uiChild < uiNbConfigurations; uiChild++) { 1168 1169 CDomainConfiguration *pDomainConfiguration = 1170 static_cast<CDomainConfiguration *>(getChild(uiChild)); 1171 1172 pDomainConfiguration->removeConfigurableElement(pConfigurableElement); 1173 } 1174 // Recompute our sync set if needed 1175 if (bRecomputeSyncSet) { 1176 1177 computeSyncSet(); 1178 } 1179 } 1180 1181 // Syncer set retrieval from configurable element 1182 CSyncerSet *CConfigurableDomain::getSyncerSet( 1183 const CConfigurableElement *pConfigurableElement) const 1184 { 1185 auto mapIt = _configurableElementToSyncerSetMap.find(pConfigurableElement); 1186 1187 ALWAYS_ASSERT(mapIt != _configurableElementToSyncerSetMap.end(), 1188 "Could not find syncer set for " << getName() << " configurable domain"); 1189 1190 return mapIt->second; 1191 } 1192 1193 // Configuration retrieval 1194 CDomainConfiguration *CConfigurableDomain::findConfiguration(const string &strConfiguration, 1195 string &strError) 1196 { 1197 CDomainConfiguration *pDomainConfiguration = 1198 static_cast<CDomainConfiguration *>(findChild(strConfiguration)); 1199 1200 if (!pDomainConfiguration) { 1201 1202 strError = "Domain configuration " + strConfiguration + " not found"; 1203 1204 return nullptr; 1205 } 1206 return pDomainConfiguration; 1207 } 1208 1209 const CDomainConfiguration *CConfigurableDomain::findConfiguration(const string &strConfiguration, 1210 string &strError) const 1211 { 1212 const CDomainConfiguration *pDomainConfiguration = 1213 static_cast<const CDomainConfiguration *>(findChild(strConfiguration)); 1214 1215 if (!pDomainConfiguration) { 1216 1217 strError = "Domain configuration " + strConfiguration + " not found"; 1218 1219 return nullptr; 1220 } 1221 return pDomainConfiguration; 1222 } 1223