/* * Copyright (c) 2011-2015, Intel Corporation * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, * are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation and/or * other materials provided with the distribution. * * 3. Neither the name of the copyright holder nor the names of its contributors * may be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "ConfigurableElement.h" #include "MappingData.h" #include "SyncerSet.h" #include "ConfigurableDomain.h" #include "ConfigurationAccessContext.h" #include "ConfigurableElementAggregator.h" #include "AreaConfiguration.h" #include "Iterator.hpp" #include "Utility.h" #include "XmlParameterSerializingContext.h" #include <assert.h> #define base CElement CConfigurableElement::CConfigurableElement(const std::string &strName) : base(strName) { } bool CConfigurableElement::fromXml(const CXmlElement &xmlElement, CXmlSerializingContext &serializingContext) { auto &context = static_cast<CXmlParameterSerializingContext &>(serializingContext); auto &accessContext = context.getAccessContext(); if (accessContext.serializeSettings()) { // As serialization and deserialisation are handled through the *same* function // the (de)serialize object can not be const in `serializeXmlSettings` signature. // As a result a const_cast is unavoidable :(. // Fixme: split serializeXmlSettings in two functions (in and out) to avoid the `const_cast` return serializeXmlSettings(const_cast<CXmlElement &>(xmlElement), static_cast<CConfigurationAccessContext &>(accessContext)); } return structureFromXml(xmlElement, serializingContext); } void CConfigurableElement::toXml(CXmlElement &xmlElement, CXmlSerializingContext &serializingContext) const { auto &context = static_cast<CXmlParameterSerializingContext &>(serializingContext); auto &accessContext = context.getAccessContext(); if (accessContext.serializeSettings()) { serializeXmlSettings(xmlElement, static_cast<CConfigurationAccessContext &>(accessContext)); } else { structureToXml(xmlElement, serializingContext); } } // XML configuration settings parsing bool CConfigurableElement::serializeXmlSettings( CXmlElement &xmlConfigurationSettingsElementContent, CConfigurationAccessContext &configurationAccessContext) const { size_t uiNbChildren = getNbChildren(); if (!configurationAccessContext.serializeOut()) { // Just do basic checks and propagate to children CXmlElement::CChildIterator it(xmlConfigurationSettingsElementContent); CXmlElement xmlChildConfigurableElementSettingsElement; // Propagate to children for (size_t index = 0; index < uiNbChildren; index++) { // Get child const CConfigurableElement *pChildConfigurableElement = static_cast<const CConfigurableElement *>(getChild(index)); if (!it.next(xmlChildConfigurableElementSettingsElement)) { // Structure error configurationAccessContext.setError( "Configuration settings parsing: missing child node " + pChildConfigurableElement->getXmlElementName() + " (name:" + pChildConfigurableElement->getName() + ") in " + getName()); return false; } // Check element type matches in type if (xmlChildConfigurableElementSettingsElement.getType() != pChildConfigurableElement->getXmlElementName()) { // "Component" tag has been renamed to "ParameterBlock", but retro-compatibility // shall be ensured. // // So checking if this case occurs, i.e. element name is "ParameterBlock" // but xml setting name is "Component". bool compatibilityCase = (pChildConfigurableElement->getXmlElementName() == "ParameterBlock") && (xmlChildConfigurableElementSettingsElement.getType() == "Component"); // Error if the compatibility case does not occur. if (!compatibilityCase) { // Type error configurationAccessContext.setError( "Configuration settings parsing: Settings " "for configurable element " + pChildConfigurableElement->getQualifiedPath() + " does not match expected type: " + xmlChildConfigurableElementSettingsElement.getType() + " instead of " + pChildConfigurableElement->getKind()); return false; } } // Check element type matches in name if (xmlChildConfigurableElementSettingsElement.getNameAttribute() != pChildConfigurableElement->getName()) { // Name error configurationAccessContext.setError( "Configuration settings parsing: Under configurable element " + getQualifiedPath() + ", expected element name " + pChildConfigurableElement->getName() + " but found " + xmlChildConfigurableElementSettingsElement.getNameAttribute() + " instead"); return false; } // Parse child configurable element's settings if (!pChildConfigurableElement->serializeXmlSettings( xmlChildConfigurableElementSettingsElement, configurationAccessContext)) { return false; } } // There should remain no configurable element to parse if (it.next(xmlChildConfigurableElementSettingsElement)) { // Structure error configurationAccessContext.setError( "Configuration settings parsing: Unexpected xml element node " + xmlChildConfigurableElementSettingsElement.getType() + " in " + getQualifiedPath()); return false; } } else { // Handle element name attribute xmlConfigurationSettingsElementContent.setNameAttribute(getName()); // Propagate to children for (size_t index = 0; index < uiNbChildren; index++) { const CConfigurableElement *pChildConfigurableElement = static_cast<const CConfigurableElement *>(getChild(index)); // Create corresponding child element CXmlElement xmlChildConfigurableElementSettingsElement; xmlConfigurationSettingsElementContent.createChild( xmlChildConfigurableElementSettingsElement, pChildConfigurableElement->getXmlElementName()); // Propagate pChildConfigurableElement->serializeXmlSettings( xmlChildConfigurableElementSettingsElement, configurationAccessContext); } } // Done return true; } // AreaConfiguration creation CAreaConfiguration *CConfigurableElement::createAreaConfiguration( const CSyncerSet *pSyncerSet) const { return new CAreaConfiguration(this, pSyncerSet); } // Parameter access bool CConfigurableElement::accessValue(CPathNavigator &pathNavigator, std::string &strValue, bool bSet, CParameterAccessContext ¶meterAccessContext) const { std::string *pStrChildName = pathNavigator.next(); if (!pStrChildName) { parameterAccessContext.setError((bSet ? "Can't set " : "Can't get ") + pathNavigator.getCurrentPath() + " because it is not a parameter"); return false; } const CConfigurableElement *pChild = static_cast<const CConfigurableElement *>(findChild(*pStrChildName)); if (!pChild) { parameterAccessContext.setError("Path not found: " + pathNavigator.getCurrentPath()); return false; } return pChild->accessValue(pathNavigator, strValue, bSet, parameterAccessContext); } // Whole element access void CConfigurableElement::getSettingsAsBytes(std::vector<uint8_t> &bytes, CParameterAccessContext ¶meterAccessContext) const { bytes.resize(getFootPrint()); parameterAccessContext.getParameterBlackboard()->readBytes( bytes, getOffset() - parameterAccessContext.getBaseOffset()); } bool CConfigurableElement::setSettingsAsBytes(const std::vector<uint8_t> &bytes, CParameterAccessContext ¶meterAccessContext) const { CParameterBlackboard *pParameterBlackboard = parameterAccessContext.getParameterBlackboard(); // Size size_t size = getFootPrint(); // Check sizes match if (size != bytes.size()) { parameterAccessContext.setError(std::string("Wrong size: Expected: ") + std::to_string(size) + " Provided: " + std::to_string(bytes.size())); return false; } // Write bytes pParameterBlackboard->writeBytes(bytes, getOffset() - parameterAccessContext.getBaseOffset()); if (not parameterAccessContext.getAutoSync()) { // Auto sync is not activated, sync will be defered until an explicit request return true; } CSyncerSet syncerSet; fillSyncerSet(syncerSet); core::Results res; if (not syncerSet.sync(*parameterAccessContext.getParameterBlackboard(), false, &res)) { parameterAccessContext.setError(utility::asString(res)); return false; } return true; } std::list<const CConfigurableElement *> CConfigurableElement::getConfigurableElementContext() const { std::list<const CConfigurableElement *> configurableElementPath; const CElement *element = this; while (element != nullptr and isOfConfigurableElementType(element)) { auto configurableElement = static_cast<const CConfigurableElement *>(element); configurableElementPath.push_back(configurableElement); element = element->getParent(); } return configurableElementPath; } // Used for simulation and virtual subsystems void CConfigurableElement::setDefaultValues(CParameterAccessContext ¶meterAccessContext) const { // Propagate to children size_t uiNbChildren = getNbChildren(); for (size_t index = 0; index < uiNbChildren; index++) { const CConfigurableElement *pConfigurableElement = static_cast<const CConfigurableElement *>(getChild(index)); pConfigurableElement->setDefaultValues(parameterAccessContext); } } // Element properties void CConfigurableElement::showProperties(std::string &strResult) const { base::showProperties(strResult); strResult += "Total size: " + getFootprintAsString() + "\n"; } std::string CConfigurableElement::logValue(utility::ErrorContext &context) const { return logValue(static_cast<CParameterAccessContext &>(context)); } std::string CConfigurableElement::logValue(CParameterAccessContext & /*ctx*/) const { // By default, an element does not have a value. Only leaf elements have // one. This method could be pure virtual but then, several derived classes // would need to implement it in order to simply return an empty string. return ""; } // Offset void CConfigurableElement::setOffset(size_t offset) { // Assign offset locally _offset = offset; // Propagate to children size_t uiNbChildren = getNbChildren(); for (size_t index = 0; index < uiNbChildren; index++) { CConfigurableElement *pConfigurableElement = static_cast<CConfigurableElement *>(getChild(index)); pConfigurableElement->setOffset(offset); offset += pConfigurableElement->getFootPrint(); } } size_t CConfigurableElement::getOffset() const { return _offset; } // Memory size_t CConfigurableElement::getFootPrint() const { size_t uiSize = 0; size_t uiNbChildren = getNbChildren(); for (size_t index = 0; index < uiNbChildren; index++) { const CConfigurableElement *pConfigurableElement = static_cast<const CConfigurableElement *>(getChild(index)); uiSize += pConfigurableElement->getFootPrint(); } return uiSize; } // Browse parent path to find syncer ISyncer *CConfigurableElement::getSyncer() const { // Check parent const CElement *pParent = getParent(); if (isOfConfigurableElementType(pParent)) { return static_cast<const CConfigurableElement *>(pParent)->getSyncer(); } return nullptr; } // Syncer set (me, ascendant or descendant ones) void CConfigurableElement::fillSyncerSet(CSyncerSet &syncerSet) const { // Try me or ascendants ISyncer *pMineOrAscendantSyncer = getSyncer(); if (pMineOrAscendantSyncer) { // Provide found syncer object syncerSet += pMineOrAscendantSyncer; // Done return; } // Fetch descendant ones fillSyncerSetFromDescendant(syncerSet); } // Syncer set (descendant) void CConfigurableElement::fillSyncerSetFromDescendant(CSyncerSet &syncerSet) const { // Dig size_t uiNbChildren = getNbChildren(); for (size_t index = 0; index < uiNbChildren; index++) { const CConfigurableElement *pConfigurableElement = static_cast<const CConfigurableElement *>(getChild(index)); pConfigurableElement->fillSyncerSetFromDescendant(syncerSet); } } // Configurable domain association void CConfigurableElement::addAttachedConfigurableDomain( const CConfigurableDomain *pConfigurableDomain) { _configurableDomainList.push_back(pConfigurableDomain); } void CConfigurableElement::removeAttachedConfigurableDomain( const CConfigurableDomain *pConfigurableDomain) { _configurableDomainList.remove(pConfigurableDomain); } // Belonging domain bool CConfigurableElement::belongsTo(const CConfigurableDomain *pConfigurableDomain) const { if (containsConfigurableDomain(pConfigurableDomain)) { return true; } return belongsToDomainAscending(pConfigurableDomain); } // Belonging domains void CConfigurableElement::getBelongingDomains( std::list<const CConfigurableDomain *> &configurableDomainList) const { configurableDomainList.insert(configurableDomainList.end(), _configurableDomainList.begin(), _configurableDomainList.end()); // Check parent const CElement *pParent = getParent(); if (isOfConfigurableElementType(pParent)) { static_cast<const CConfigurableElement *>(pParent)->getBelongingDomains( configurableDomainList); } } void CConfigurableElement::listBelongingDomains(std::string &strResult, bool bVertical) const { // Get belonging domain list std::list<const CConfigurableDomain *> configurableDomainList; getBelongingDomains(configurableDomainList); // Fill list listDomains(configurableDomainList, strResult, bVertical); } // Elements with no domains void CConfigurableElement::listRogueElements(std::string &strResult) const { // Get rogue element aggregate list (no associated domain) std::list<const CConfigurableElement *> rogueElementList; CConfigurableElementAggregator configurableElementAggregator( rogueElementList, &CConfigurableElement::hasNoDomainAssociated); configurableElementAggregator.aggegate(this); // Build list as std::string std::list<const CConfigurableElement *>::const_iterator it; for (it = rogueElementList.begin(); it != rogueElementList.end(); ++it) { const CConfigurableElement *pConfigurableElement = *it; strResult += pConfigurableElement->getPath() + "\n"; } } bool CConfigurableElement::isRogue() const { // Check not belonging to any domin from current level and towards ascendents if (getBelongingDomainCount() != 0) { return false; } // Get a list of elements (current + descendants) with no domains associated std::list<const CConfigurableElement *> rogueElementList; CConfigurableElementAggregator agregator(rogueElementList, &CConfigurableElement::hasNoDomainAssociated); agregator.aggegate(this); // Check only one element found which ought to be current one return (rogueElementList.size() == 1) && (rogueElementList.front() == this); } // Footprint as string std::string CConfigurableElement::getFootprintAsString() const { // Get size as string return std::to_string(getFootPrint()) + " byte(s)"; } // Matching check for no domain association bool CConfigurableElement::hasNoDomainAssociated() const { return _configurableDomainList.empty(); } // Matching check for no valid associated domains bool CConfigurableElement::hasNoValidDomainAssociated() const { if (_configurableDomainList.empty()) { // No domains associated return true; } ConfigurableDomainListConstIterator it; // Browse all configurable domains for validity checking for (it = _configurableDomainList.begin(); it != _configurableDomainList.end(); ++it) { const CConfigurableDomain *pConfigurableDomain = *it; if (pConfigurableDomain->isApplicableConfigurationValid(this)) { return false; } } return true; } // Owning domains void CConfigurableElement::listAssociatedDomains(std::string &strResult, bool bVertical) const { // Fill list listDomains(_configurableDomainList, strResult, bVertical); } size_t CConfigurableElement::getBelongingDomainCount() const { // Get belonging domain list std::list<const CConfigurableDomain *> configurableDomainList; getBelongingDomains(configurableDomainList); return configurableDomainList.size(); } void CConfigurableElement::listDomains( const std::list<const CConfigurableDomain *> &configurableDomainList, std::string &strResult, bool bVertical) const { // Fill list ConfigurableDomainListConstIterator it; bool bFirst = true; // Browse all configurable domains for comparison for (it = configurableDomainList.begin(); it != configurableDomainList.end(); ++it) { const CConfigurableDomain *pConfigurableDomain = *it; if (!bVertical && !bFirst) { strResult += ", "; } strResult += pConfigurableDomain->getName(); if (bVertical) { strResult += "\n"; } else { bFirst = false; } } } bool CConfigurableElement::containsConfigurableDomain( const CConfigurableDomain *pConfigurableDomain) const { ConfigurableDomainListConstIterator it; // Browse all configurable domains for comparison for (it = _configurableDomainList.begin(); it != _configurableDomainList.end(); ++it) { if (pConfigurableDomain == *it) { return true; } } return false; } // Belonging domain ascending search bool CConfigurableElement::belongsToDomainAscending( const CConfigurableDomain *pConfigurableDomain) const { // Check parent const CElement *pParent = getParent(); if (isOfConfigurableElementType(pParent)) { return static_cast<const CConfigurableElement *>(pParent)->belongsTo(pConfigurableDomain); } return false; } // Belonging subsystem const CSubsystem *CConfigurableElement::getBelongingSubsystem() const { const CElement *pParent = getParent(); // Stop at system class if (!pParent->getParent()) { return nullptr; } return static_cast<const CConfigurableElement *>(pParent)->getBelongingSubsystem(); } // Check element is a parameter bool CConfigurableElement::isParameter() const { return false; } // Check parent is still of current type (by structure knowledge) bool CConfigurableElement::isOfConfigurableElementType(const CElement *pParent) const { assert(pParent); // Up to system class return !!pParent->getParent(); }