/* * * * INTEL CONFIDENTIAL * Copyright © 2011 Intel * Corporation All Rights Reserved. * * The source code contained or described herein and all documents related to * the source code ("Material") are owned by Intel Corporation or its suppliers * or licensors. Title to the Material remains with Intel Corporation or its * suppliers and licensors. The Material contains trade secrets and proprietary * and confidential information of Intel or its suppliers and licensors. The * Material is protected by worldwide copyright and trade secret laws and * treaty provisions. No part of the Material may be used, copied, reproduced, * modified, published, uploaded, posted, transmitted, distributed, or * disclosed in any way without Intel’s prior express written permission. * * No license under any patent, copyright, trade secret or other intellectual * property right is granted to or conferred upon you by disclosure or delivery * of the Materials, either expressly, by implication, inducement, estoppel or * otherwise. Any license under such intellectual property rights must be * express and approved by Intel in writing. * * AUTHOR: Patrick Benavoli (patrickx.benavoli@intel.com) * CREATED: 2011-06-01 * UPDATED: 2011-07-27 * * * */ #include "ArrayParameter.h" #include // for istringstream #include "Tokenizer.h" #include "ParameterType.h" #include "ParameterAccessContext.h" #include "ConfigurationAccessContext.h" #include "ParameterBlackboard.h" #define base CParameter CArrayParameter::CArrayParameter(const string& strName, const CTypeElement* pTypeElement) : base(strName, pTypeElement) { } uint32_t CArrayParameter::getFootPrint() const { return getSize() * getArrayLength(); } // Array length uint32_t CArrayParameter::getArrayLength() const { return getTypeElement()->getArrayLength(); } // Element properties void CArrayParameter::showProperties(string& strResult) const { base::showProperties(strResult); // Array length strResult += "Array length: "; strResult += toString(getArrayLength()); strResult += "\n"; } // XML configuration settings parsing bool CArrayParameter::serializeXmlSettings(CXmlElement& xmlConfigurationSettingsElementContent, CConfigurationAccessContext& configurationAccessContext) const { // Check for value space handleValueSpaceAttribute(xmlConfigurationSettingsElementContent, configurationAccessContext); // Handle access if (!configurationAccessContext.serializeOut()) { // Actually set values to blackboard if (!setValues(0, configurationAccessContext.getBaseOffset(), xmlConfigurationSettingsElementContent.getTextContent(), configurationAccessContext)) { return false; } } else { // Get string value string strValue; // Whole array requested getValues(configurationAccessContext.getBaseOffset(), strValue, configurationAccessContext); // Populate value into xml text node xmlConfigurationSettingsElementContent.setTextContent(strValue); } // Done return true; } // User set/get bool CArrayParameter::setValue(CPathNavigator& pathNavigator, const string& strValue, CParameterAccessContext& parameterContext) const { uint32_t uiStartIndex; if (!getIndex(pathNavigator, uiStartIndex, parameterContext)) { return false; } if (uiStartIndex == (uint32_t)-1) { // No index provided, start with 0 uiStartIndex = 0; } // Actually set values if (!setValues(uiStartIndex, 0, strValue, parameterContext)) { return false; } // Synchronize if (parameterContext.getAutoSync() && !sync(parameterContext)) { // Append parameter path to error parameterContext.appendToError(" " + getPath()); return false; } return true; } bool CArrayParameter::getValue(CPathNavigator& pathNavigator, string& strValue, CParameterAccessContext& parameterContext) const { uint32_t uiIndex; if (!getIndex(pathNavigator, uiIndex, parameterContext)) { return false; } if (uiIndex == (uint32_t)-1) { // Whole array requested getValues(0, strValue, parameterContext); } else { // Scalar requested doGetValue(strValue, getOffset() + uiIndex * getSize(), parameterContext); } return true; } void CArrayParameter::logValue(string& strValue, CErrorContext& errorContext) const { // Parameter context CParameterAccessContext& parameterContext = static_cast(errorContext); // Dump values getValues(0, strValue, parameterContext); } // Used for simulation only void CArrayParameter::setDefaultValues(CParameterAccessContext& parameterAccessContext) const { // Get default value from type uint32_t uiDefaultValue = static_cast(getTypeElement())->getDefaultValue(); // Write blackboard CParameterBlackboard* pBlackboard = parameterAccessContext.getParameterBlackboard(); // Process uint32_t uiValueIndex; uint32_t uiSize = getSize(); uint32_t uiOffset = getOffset(); bool bSubsystemIsBigEndian = parameterAccessContext.isBigEndianSubsystem(); uint32_t uiArrayLength = getArrayLength(); for (uiValueIndex = 0; uiValueIndex < uiArrayLength; uiValueIndex++) { // Beware this code works on little endian architectures only! pBlackboard->write(&uiDefaultValue, uiSize, uiOffset, bSubsystemIsBigEndian); uiOffset += uiSize; } } // Index from path bool CArrayParameter::getIndex(CPathNavigator& pathNavigator, uint32_t& uiIndex, CParameterAccessContext& parameterContext) const { uiIndex = (uint32_t)-1; string* pStrChildName = pathNavigator.next(); if (pStrChildName) { // Check index is numeric istringstream iss(*pStrChildName); iss >> uiIndex; if (!iss) { parameterContext.setError("Expected numerical expression as last item in " + pathNavigator.getCurrentPath()); return false; } if (uiIndex >= getArrayLength()) { ostringstream oss; oss << "Provided index out of range (max is " << getArrayLength() - 1 << ")"; parameterContext.setError(oss.str()); return false; } // Check no other item provided in path pStrChildName = pathNavigator.next(); if (pStrChildName) { // Should be leaf element parameterContext.setError("Path not found: " + pathNavigator.getCurrentPath()); return false; } } return true; } // Common set value processing bool CArrayParameter::setValues(uint32_t uiStartIndex, uint32_t uiBaseOffset, const string& strValue, CParameterAccessContext& parameterContext) const { // Deal with value(s) Tokenizer tok(strValue, DEFAULT_DELIMITER + ","); vector astrValues = tok.split(); uint32_t uiNbValues = astrValues.size(); // Check number of provided values if (uiNbValues + uiStartIndex > getArrayLength()) { // Out of bounds parameterContext.setError("Too many values provided"); return false; } // Process uint32_t uiValueIndex; uint32_t uiSize = getSize(); uint32_t uiOffset = getOffset() + uiStartIndex * uiSize - uiBaseOffset; for (uiValueIndex = 0; uiValueIndex < uiNbValues; uiValueIndex++) { if (!doSetValue(astrValues[uiValueIndex], uiOffset, parameterContext)) { // Append parameter path to error parameterContext.appendToError(" " + getPath() + "/" + toString(uiValueIndex + uiStartIndex)); return false; } uiOffset += uiSize; } return true; } // Common get value processing void CArrayParameter::getValues(uint32_t uiBaseOffset, string& strValues, CParameterAccessContext& parameterContext) const { uint32_t uiValueIndex; uint32_t uiSize = getSize(); uint32_t uiOffset = getOffset() - uiBaseOffset; uint32_t uiArrayLength = getArrayLength(); bool bFirst = true; for (uiValueIndex = 0; uiValueIndex < uiArrayLength; uiValueIndex++) { string strReadValue; doGetValue(strReadValue, uiOffset, parameterContext); if (!bFirst) { strValues += " "; } else { bFirst = false; } strValues += strReadValue; uiOffset += uiSize; } }