xref: /aosp_15_r20/external/parameter-framework/upstream/parameter/Subsystem.cpp (revision c33452fb792a5495ec310a9626f2638b053af5dd)
1*c33452fbSAndroid Build Coastguard Worker /*
2*c33452fbSAndroid Build Coastguard Worker  * Copyright (c) 2011-2015, Intel Corporation
3*c33452fbSAndroid Build Coastguard Worker  * All rights reserved.
4*c33452fbSAndroid Build Coastguard Worker  *
5*c33452fbSAndroid Build Coastguard Worker  * Redistribution and use in source and binary forms, with or without modification,
6*c33452fbSAndroid Build Coastguard Worker  * are permitted provided that the following conditions are met:
7*c33452fbSAndroid Build Coastguard Worker  *
8*c33452fbSAndroid Build Coastguard Worker  * 1. Redistributions of source code must retain the above copyright notice, this
9*c33452fbSAndroid Build Coastguard Worker  * list of conditions and the following disclaimer.
10*c33452fbSAndroid Build Coastguard Worker  *
11*c33452fbSAndroid Build Coastguard Worker  * 2. Redistributions in binary form must reproduce the above copyright notice,
12*c33452fbSAndroid Build Coastguard Worker  * this list of conditions and the following disclaimer in the documentation and/or
13*c33452fbSAndroid Build Coastguard Worker  * other materials provided with the distribution.
14*c33452fbSAndroid Build Coastguard Worker  *
15*c33452fbSAndroid Build Coastguard Worker  * 3. Neither the name of the copyright holder nor the names of its contributors
16*c33452fbSAndroid Build Coastguard Worker  * may be used to endorse or promote products derived from this software without
17*c33452fbSAndroid Build Coastguard Worker  * specific prior written permission.
18*c33452fbSAndroid Build Coastguard Worker  *
19*c33452fbSAndroid Build Coastguard Worker  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
20*c33452fbSAndroid Build Coastguard Worker  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21*c33452fbSAndroid Build Coastguard Worker  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22*c33452fbSAndroid Build Coastguard Worker  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
23*c33452fbSAndroid Build Coastguard Worker  * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24*c33452fbSAndroid Build Coastguard Worker  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25*c33452fbSAndroid Build Coastguard Worker  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
26*c33452fbSAndroid Build Coastguard Worker  * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27*c33452fbSAndroid Build Coastguard Worker  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
28*c33452fbSAndroid Build Coastguard Worker  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29*c33452fbSAndroid Build Coastguard Worker  */
30*c33452fbSAndroid Build Coastguard Worker #include "Subsystem.h"
31*c33452fbSAndroid Build Coastguard Worker #include "ComponentLibrary.h"
32*c33452fbSAndroid Build Coastguard Worker #include "InstanceDefinition.h"
33*c33452fbSAndroid Build Coastguard Worker #include "XmlParameterSerializingContext.h"
34*c33452fbSAndroid Build Coastguard Worker #include "ParameterAccessContext.h"
35*c33452fbSAndroid Build Coastguard Worker #include "ConfigurationAccessContext.h"
36*c33452fbSAndroid Build Coastguard Worker #include "SubsystemObjectCreator.h"
37*c33452fbSAndroid Build Coastguard Worker #include "MappingData.h"
38*c33452fbSAndroid Build Coastguard Worker #include <assert.h>
39*c33452fbSAndroid Build Coastguard Worker #include <sstream>
40*c33452fbSAndroid Build Coastguard Worker 
41*c33452fbSAndroid Build Coastguard Worker #define base CConfigurableElement
42*c33452fbSAndroid Build Coastguard Worker 
43*c33452fbSAndroid Build Coastguard Worker using std::string;
44*c33452fbSAndroid Build Coastguard Worker using std::list;
45*c33452fbSAndroid Build Coastguard Worker 
CSubsystem(const string & strName,core::log::Logger & logger)46*c33452fbSAndroid Build Coastguard Worker CSubsystem::CSubsystem(const string &strName, core::log::Logger &logger)
47*c33452fbSAndroid Build Coastguard Worker     : base(strName), _pComponentLibrary(new CComponentLibrary),
48*c33452fbSAndroid Build Coastguard Worker       _pInstanceDefinition(new CInstanceDefinition), _logger(logger)
49*c33452fbSAndroid Build Coastguard Worker {
50*c33452fbSAndroid Build Coastguard Worker     // Note: A subsystem contains instance components
51*c33452fbSAndroid Build Coastguard Worker     // InstanceDefintion and ComponentLibrary objects are then not chosen to be children
52*c33452fbSAndroid Build Coastguard Worker     // They'll be delt with locally
53*c33452fbSAndroid Build Coastguard Worker }
54*c33452fbSAndroid Build Coastguard Worker 
~CSubsystem()55*c33452fbSAndroid Build Coastguard Worker CSubsystem::~CSubsystem()
56*c33452fbSAndroid Build Coastguard Worker {
57*c33452fbSAndroid Build Coastguard Worker     // FIXME use unique_ptr, would make this method empty
58*c33452fbSAndroid Build Coastguard Worker 
59*c33452fbSAndroid Build Coastguard Worker     for (auto *subsystemObject : _subsystemObjectList) {
60*c33452fbSAndroid Build Coastguard Worker 
61*c33452fbSAndroid Build Coastguard Worker         delete subsystemObject;
62*c33452fbSAndroid Build Coastguard Worker     }
63*c33452fbSAndroid Build Coastguard Worker 
64*c33452fbSAndroid Build Coastguard Worker     // Remove susbsystem creators
65*c33452fbSAndroid Build Coastguard Worker     for (auto *subsystemObjectCreator : _subsystemObjectCreatorArray) {
66*c33452fbSAndroid Build Coastguard Worker 
67*c33452fbSAndroid Build Coastguard Worker         delete subsystemObjectCreator;
68*c33452fbSAndroid Build Coastguard Worker     }
69*c33452fbSAndroid Build Coastguard Worker 
70*c33452fbSAndroid Build Coastguard Worker     // Order matters!
71*c33452fbSAndroid Build Coastguard Worker     delete _pInstanceDefinition;
72*c33452fbSAndroid Build Coastguard Worker     delete _pComponentLibrary;
73*c33452fbSAndroid Build Coastguard Worker 
74*c33452fbSAndroid Build Coastguard Worker     delete _pMappingData;
75*c33452fbSAndroid Build Coastguard Worker }
76*c33452fbSAndroid Build Coastguard Worker 
getKind() const77*c33452fbSAndroid Build Coastguard Worker string CSubsystem::getKind() const
78*c33452fbSAndroid Build Coastguard Worker {
79*c33452fbSAndroid Build Coastguard Worker     return "Subsystem";
80*c33452fbSAndroid Build Coastguard Worker }
81*c33452fbSAndroid Build Coastguard Worker 
82*c33452fbSAndroid Build Coastguard Worker // Susbsystem sanity
isAlive() const83*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::isAlive() const
84*c33452fbSAndroid Build Coastguard Worker {
85*c33452fbSAndroid Build Coastguard Worker     return true;
86*c33452fbSAndroid Build Coastguard Worker }
87*c33452fbSAndroid Build Coastguard Worker 
88*c33452fbSAndroid Build Coastguard Worker // Resynchronization after subsystem restart needed
needResync(bool)89*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::needResync(bool /*bClear*/)
90*c33452fbSAndroid Build Coastguard Worker {
91*c33452fbSAndroid Build Coastguard Worker     return false;
92*c33452fbSAndroid Build Coastguard Worker }
93*c33452fbSAndroid Build Coastguard Worker 
structureFromXml(const CXmlElement & xmlElement,CXmlSerializingContext & serializingContext)94*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::structureFromXml(const CXmlElement &xmlElement,
95*c33452fbSAndroid Build Coastguard Worker                                   CXmlSerializingContext &serializingContext)
96*c33452fbSAndroid Build Coastguard Worker {
97*c33452fbSAndroid Build Coastguard Worker     // Subsystem class does not rely on generic fromXml algorithm of Element class.
98*c33452fbSAndroid Build Coastguard Worker     // So, setting here the description if found as XML attribute.
99*c33452fbSAndroid Build Coastguard Worker     string description;
100*c33452fbSAndroid Build Coastguard Worker     xmlElement.getAttribute(gDescriptionPropertyName, description);
101*c33452fbSAndroid Build Coastguard Worker     setDescription(description);
102*c33452fbSAndroid Build Coastguard Worker 
103*c33452fbSAndroid Build Coastguard Worker     // Context
104*c33452fbSAndroid Build Coastguard Worker     CXmlParameterSerializingContext &parameterBuildContext =
105*c33452fbSAndroid Build Coastguard Worker         static_cast<CXmlParameterSerializingContext &>(serializingContext);
106*c33452fbSAndroid Build Coastguard Worker 
107*c33452fbSAndroid Build Coastguard Worker     // Install temporary component library for further component creation
108*c33452fbSAndroid Build Coastguard Worker     parameterBuildContext.setComponentLibrary(_pComponentLibrary);
109*c33452fbSAndroid Build Coastguard Worker 
110*c33452fbSAndroid Build Coastguard Worker     CXmlElement childElement;
111*c33452fbSAndroid Build Coastguard Worker 
112*c33452fbSAndroid Build Coastguard Worker     // Manage mapping attribute
113*c33452fbSAndroid Build Coastguard Worker     string rawMapping;
114*c33452fbSAndroid Build Coastguard Worker     xmlElement.getAttribute("Mapping", rawMapping);
115*c33452fbSAndroid Build Coastguard Worker     if (!rawMapping.empty()) {
116*c33452fbSAndroid Build Coastguard Worker 
117*c33452fbSAndroid Build Coastguard Worker         std::string error;
118*c33452fbSAndroid Build Coastguard Worker         _pMappingData = new CMappingData;
119*c33452fbSAndroid Build Coastguard Worker         if (!_pMappingData->init(rawMapping, error)) {
120*c33452fbSAndroid Build Coastguard Worker 
121*c33452fbSAndroid Build Coastguard Worker             serializingContext.setError("Invalid Mapping data from XML element '" +
122*c33452fbSAndroid Build Coastguard Worker                                         xmlElement.getPath() + "': " + error);
123*c33452fbSAndroid Build Coastguard Worker             return false;
124*c33452fbSAndroid Build Coastguard Worker         }
125*c33452fbSAndroid Build Coastguard Worker     }
126*c33452fbSAndroid Build Coastguard Worker 
127*c33452fbSAndroid Build Coastguard Worker     // XML populate ComponentLibrary
128*c33452fbSAndroid Build Coastguard Worker     xmlElement.getChildElement("ComponentLibrary", childElement);
129*c33452fbSAndroid Build Coastguard Worker 
130*c33452fbSAndroid Build Coastguard Worker     if (!_pComponentLibrary->fromXml(childElement, serializingContext)) {
131*c33452fbSAndroid Build Coastguard Worker 
132*c33452fbSAndroid Build Coastguard Worker         return false;
133*c33452fbSAndroid Build Coastguard Worker     }
134*c33452fbSAndroid Build Coastguard Worker 
135*c33452fbSAndroid Build Coastguard Worker     // XML populate InstanceDefintion
136*c33452fbSAndroid Build Coastguard Worker     xmlElement.getChildElement("InstanceDefintion", childElement);
137*c33452fbSAndroid Build Coastguard Worker     if (!_pInstanceDefinition->fromXml(childElement, serializingContext)) {
138*c33452fbSAndroid Build Coastguard Worker 
139*c33452fbSAndroid Build Coastguard Worker         return false;
140*c33452fbSAndroid Build Coastguard Worker     }
141*c33452fbSAndroid Build Coastguard Worker 
142*c33452fbSAndroid Build Coastguard Worker     // Create components
143*c33452fbSAndroid Build Coastguard Worker     _pInstanceDefinition->createInstances(this);
144*c33452fbSAndroid Build Coastguard Worker 
145*c33452fbSAndroid Build Coastguard Worker     // Execute mapping to create subsystem mapping entities
146*c33452fbSAndroid Build Coastguard Worker     string strError;
147*c33452fbSAndroid Build Coastguard Worker     if (!mapSubsystemElements(strError)) {
148*c33452fbSAndroid Build Coastguard Worker 
149*c33452fbSAndroid Build Coastguard Worker         serializingContext.setError(strError);
150*c33452fbSAndroid Build Coastguard Worker 
151*c33452fbSAndroid Build Coastguard Worker         return false;
152*c33452fbSAndroid Build Coastguard Worker     }
153*c33452fbSAndroid Build Coastguard Worker 
154*c33452fbSAndroid Build Coastguard Worker     return true;
155*c33452fbSAndroid Build Coastguard Worker }
156*c33452fbSAndroid Build Coastguard Worker 
mapSubsystemElements(string & strError)157*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::mapSubsystemElements(string &strError)
158*c33452fbSAndroid Build Coastguard Worker {
159*c33452fbSAndroid Build Coastguard Worker     // Default mapping context
160*c33452fbSAndroid Build Coastguard Worker     CMappingContext context(_contextMappingKeyArray.size());
161*c33452fbSAndroid Build Coastguard Worker     // Add Subsystem-level mapping data, which will be propagated to all children
162*c33452fbSAndroid Build Coastguard Worker     handleMappingContext(this, context, strError);
163*c33452fbSAndroid Build Coastguard Worker 
164*c33452fbSAndroid Build Coastguard Worker     _contextStack.push(context);
165*c33452fbSAndroid Build Coastguard Worker 
166*c33452fbSAndroid Build Coastguard Worker     // Map all instantiated subelements in subsystem
167*c33452fbSAndroid Build Coastguard Worker     size_t nbChildren = getNbChildren();
168*c33452fbSAndroid Build Coastguard Worker 
169*c33452fbSAndroid Build Coastguard Worker     for (size_t child = 0; child < nbChildren; child++) {
170*c33452fbSAndroid Build Coastguard Worker 
171*c33452fbSAndroid Build Coastguard Worker         CInstanceConfigurableElement *pInstanceConfigurableChildElement =
172*c33452fbSAndroid Build Coastguard Worker             static_cast<CInstanceConfigurableElement *>(getChild(child));
173*c33452fbSAndroid Build Coastguard Worker 
174*c33452fbSAndroid Build Coastguard Worker         if (!pInstanceConfigurableChildElement->map(*this, strError)) {
175*c33452fbSAndroid Build Coastguard Worker 
176*c33452fbSAndroid Build Coastguard Worker             return false;
177*c33452fbSAndroid Build Coastguard Worker         }
178*c33452fbSAndroid Build Coastguard Worker     }
179*c33452fbSAndroid Build Coastguard Worker     return true;
180*c33452fbSAndroid Build Coastguard Worker }
181*c33452fbSAndroid Build Coastguard Worker 
182*c33452fbSAndroid Build Coastguard Worker // Formats the mapping of the ConfigurableElements
formatMappingDataList(const list<const CConfigurableElement * > & configurableElementPath) const183*c33452fbSAndroid Build Coastguard Worker string CSubsystem::formatMappingDataList(
184*c33452fbSAndroid Build Coastguard Worker     const list<const CConfigurableElement *> &configurableElementPath) const
185*c33452fbSAndroid Build Coastguard Worker {
186*c33452fbSAndroid Build Coastguard Worker     // The list is parsed in reverse order because it has been filled from the leaf to the trunk
187*c33452fbSAndroid Build Coastguard Worker     // of the tree. When formatting the mapping, we want to start from the subsystem level
188*c33452fbSAndroid Build Coastguard Worker     std::list<string> mappings;
189*c33452fbSAndroid Build Coastguard Worker     list<const CConfigurableElement *>::const_reverse_iterator it;
190*c33452fbSAndroid Build Coastguard Worker     for (it = configurableElementPath.rbegin(); it != configurableElementPath.rend(); ++it) {
191*c33452fbSAndroid Build Coastguard Worker 
192*c33452fbSAndroid Build Coastguard Worker         auto maybeMapping = (*it)->getFormattedMapping();
193*c33452fbSAndroid Build Coastguard Worker         if (not maybeMapping.empty()) {
194*c33452fbSAndroid Build Coastguard Worker             mappings.push_back(maybeMapping);
195*c33452fbSAndroid Build Coastguard Worker         }
196*c33452fbSAndroid Build Coastguard Worker     }
197*c33452fbSAndroid Build Coastguard Worker 
198*c33452fbSAndroid Build Coastguard Worker     return utility::asString(mappings, ", ");
199*c33452fbSAndroid Build Coastguard Worker }
200*c33452fbSAndroid Build Coastguard Worker 
201*c33452fbSAndroid Build Coastguard Worker // Find the CSubystemObject containing a specific CInstanceConfigurableElement
findSubsystemObjectFromConfigurableElement(const CInstanceConfigurableElement * pInstanceConfigurableElement) const202*c33452fbSAndroid Build Coastguard Worker const CSubsystemObject *CSubsystem::findSubsystemObjectFromConfigurableElement(
203*c33452fbSAndroid Build Coastguard Worker     const CInstanceConfigurableElement *pInstanceConfigurableElement) const
204*c33452fbSAndroid Build Coastguard Worker {
205*c33452fbSAndroid Build Coastguard Worker 
206*c33452fbSAndroid Build Coastguard Worker     list<CSubsystemObject *>::const_iterator it;
207*c33452fbSAndroid Build Coastguard Worker     for (it = _subsystemObjectList.begin(); it != _subsystemObjectList.end(); ++it) {
208*c33452fbSAndroid Build Coastguard Worker 
209*c33452fbSAndroid Build Coastguard Worker         // Check if one of the SubsystemObjects is associated with a ConfigurableElement
210*c33452fbSAndroid Build Coastguard Worker         // corresponding to the expected one
211*c33452fbSAndroid Build Coastguard Worker         const CSubsystemObject *pSubsystemObject = *it;
212*c33452fbSAndroid Build Coastguard Worker 
213*c33452fbSAndroid Build Coastguard Worker         if (pSubsystemObject->getConfigurableElement() == pInstanceConfigurableElement) {
214*c33452fbSAndroid Build Coastguard Worker 
215*c33452fbSAndroid Build Coastguard Worker             return pSubsystemObject;
216*c33452fbSAndroid Build Coastguard Worker         }
217*c33452fbSAndroid Build Coastguard Worker     }
218*c33452fbSAndroid Build Coastguard Worker 
219*c33452fbSAndroid Build Coastguard Worker     return nullptr;
220*c33452fbSAndroid Build Coastguard Worker }
221*c33452fbSAndroid Build Coastguard Worker 
findSubsystemLevelMappingKeyValue(const CInstanceConfigurableElement * pInstanceConfigurableElement,string & strMappingKey,string & strMappingValue) const222*c33452fbSAndroid Build Coastguard Worker void CSubsystem::findSubsystemLevelMappingKeyValue(
223*c33452fbSAndroid Build Coastguard Worker     const CInstanceConfigurableElement *pInstanceConfigurableElement, string &strMappingKey,
224*c33452fbSAndroid Build Coastguard Worker     string &strMappingValue) const
225*c33452fbSAndroid Build Coastguard Worker {
226*c33452fbSAndroid Build Coastguard Worker     // Find creator to get key name
227*c33452fbSAndroid Build Coastguard Worker     std::vector<CSubsystemObjectCreator *>::const_iterator it;
228*c33452fbSAndroid Build Coastguard Worker     for (it = _subsystemObjectCreatorArray.begin(); it != _subsystemObjectCreatorArray.end();
229*c33452fbSAndroid Build Coastguard Worker          ++it) {
230*c33452fbSAndroid Build Coastguard Worker 
231*c33452fbSAndroid Build Coastguard Worker         const CSubsystemObjectCreator *pSubsystemObjectCreator = *it;
232*c33452fbSAndroid Build Coastguard Worker 
233*c33452fbSAndroid Build Coastguard Worker         strMappingKey = pSubsystemObjectCreator->getMappingKey();
234*c33452fbSAndroid Build Coastguard Worker 
235*c33452fbSAndroid Build Coastguard Worker         // Check if the ObjectCreator MappingKey corresponds to the element mapping data
236*c33452fbSAndroid Build Coastguard Worker         const string *pStrValue;
237*c33452fbSAndroid Build Coastguard Worker         if (pInstanceConfigurableElement->getMappingData(strMappingKey, pStrValue)) {
238*c33452fbSAndroid Build Coastguard Worker 
239*c33452fbSAndroid Build Coastguard Worker             strMappingValue = *pStrValue;
240*c33452fbSAndroid Build Coastguard Worker             return;
241*c33452fbSAndroid Build Coastguard Worker         }
242*c33452fbSAndroid Build Coastguard Worker     }
243*c33452fbSAndroid Build Coastguard Worker     assert(0);
244*c33452fbSAndroid Build Coastguard Worker }
245*c33452fbSAndroid Build Coastguard Worker 
246*c33452fbSAndroid Build Coastguard Worker // Formats the mapping data as a comma separated list of key value pairs
getFormattedSubsystemMappingData(const CInstanceConfigurableElement * pInstanceConfigurableElement) const247*c33452fbSAndroid Build Coastguard Worker string CSubsystem::getFormattedSubsystemMappingData(
248*c33452fbSAndroid Build Coastguard Worker     const CInstanceConfigurableElement *pInstanceConfigurableElement) const
249*c33452fbSAndroid Build Coastguard Worker {
250*c33452fbSAndroid Build Coastguard Worker     // Find the SubsystemObject related to pInstanceConfigurableElement
251*c33452fbSAndroid Build Coastguard Worker     const CSubsystemObject *pSubsystemObject =
252*c33452fbSAndroid Build Coastguard Worker         findSubsystemObjectFromConfigurableElement(pInstanceConfigurableElement);
253*c33452fbSAndroid Build Coastguard Worker 
254*c33452fbSAndroid Build Coastguard Worker     // Exit if node does not correspond to a SubsystemObject
255*c33452fbSAndroid Build Coastguard Worker     if (pSubsystemObject == nullptr) {
256*c33452fbSAndroid Build Coastguard Worker 
257*c33452fbSAndroid Build Coastguard Worker         return "";
258*c33452fbSAndroid Build Coastguard Worker     }
259*c33452fbSAndroid Build Coastguard Worker 
260*c33452fbSAndroid Build Coastguard Worker     // Find SubsystemCreator mapping key
261*c33452fbSAndroid Build Coastguard Worker     string strMappingKey;
262*c33452fbSAndroid Build Coastguard Worker     string strMappingValue; // mapping value where amends are not replaced by their value
263*c33452fbSAndroid Build Coastguard Worker     findSubsystemLevelMappingKeyValue(pInstanceConfigurableElement, strMappingKey, strMappingValue);
264*c33452fbSAndroid Build Coastguard Worker 
265*c33452fbSAndroid Build Coastguard Worker     // Find SubSystemObject mapping value (with amends replaced by their value)
266*c33452fbSAndroid Build Coastguard Worker     return strMappingKey + ":" + pSubsystemObject->getFormattedMappingValue();
267*c33452fbSAndroid Build Coastguard Worker }
268*c33452fbSAndroid Build Coastguard Worker 
getMapping(list<const CConfigurableElement * > & configurableElementPath) const269*c33452fbSAndroid Build Coastguard Worker string CSubsystem::getMapping(list<const CConfigurableElement *> &configurableElementPath) const
270*c33452fbSAndroid Build Coastguard Worker {
271*c33452fbSAndroid Build Coastguard Worker     if (configurableElementPath.empty()) {
272*c33452fbSAndroid Build Coastguard Worker 
273*c33452fbSAndroid Build Coastguard Worker         return "";
274*c33452fbSAndroid Build Coastguard Worker     }
275*c33452fbSAndroid Build Coastguard Worker 
276*c33452fbSAndroid Build Coastguard Worker     // Get the first element, which is the element containing the amended mapping
277*c33452fbSAndroid Build Coastguard Worker     const CInstanceConfigurableElement *pInstanceConfigurableElement =
278*c33452fbSAndroid Build Coastguard Worker         static_cast<const CInstanceConfigurableElement *>(configurableElementPath.front());
279*c33452fbSAndroid Build Coastguard Worker 
280*c33452fbSAndroid Build Coastguard Worker     // Format context mapping data
281*c33452fbSAndroid Build Coastguard Worker     string strValue = formatMappingDataList(configurableElementPath);
282*c33452fbSAndroid Build Coastguard Worker 
283*c33452fbSAndroid Build Coastguard Worker     // Print the mapping of the first node, which corresponds to a SubsystemObject
284*c33452fbSAndroid Build Coastguard Worker     auto subsystemObjectAmendedMapping =
285*c33452fbSAndroid Build Coastguard Worker         getFormattedSubsystemMappingData(pInstanceConfigurableElement);
286*c33452fbSAndroid Build Coastguard Worker     if (not subsystemObjectAmendedMapping.empty()) {
287*c33452fbSAndroid Build Coastguard Worker         strValue += ", " + subsystemObjectAmendedMapping;
288*c33452fbSAndroid Build Coastguard Worker     }
289*c33452fbSAndroid Build Coastguard Worker 
290*c33452fbSAndroid Build Coastguard Worker     return strValue;
291*c33452fbSAndroid Build Coastguard Worker }
292*c33452fbSAndroid Build Coastguard Worker 
293*c33452fbSAndroid Build Coastguard Worker // Used for simulation and virtual subsystems
setDefaultValues(CParameterAccessContext & parameterAccessContext) const294*c33452fbSAndroid Build Coastguard Worker void CSubsystem::setDefaultValues(CParameterAccessContext &parameterAccessContext) const
295*c33452fbSAndroid Build Coastguard Worker {
296*c33452fbSAndroid Build Coastguard Worker     base::setDefaultValues(parameterAccessContext);
297*c33452fbSAndroid Build Coastguard Worker }
298*c33452fbSAndroid Build Coastguard Worker 
299*c33452fbSAndroid Build Coastguard Worker // Belonging subsystem
getBelongingSubsystem() const300*c33452fbSAndroid Build Coastguard Worker const CSubsystem *CSubsystem::getBelongingSubsystem() const
301*c33452fbSAndroid Build Coastguard Worker {
302*c33452fbSAndroid Build Coastguard Worker     return this;
303*c33452fbSAndroid Build Coastguard Worker }
304*c33452fbSAndroid Build Coastguard Worker 
305*c33452fbSAndroid Build Coastguard Worker // Subsystem context mapping keys publication
addContextMappingKey(const string & strMappingKey)306*c33452fbSAndroid Build Coastguard Worker void CSubsystem::addContextMappingKey(const string &strMappingKey)
307*c33452fbSAndroid Build Coastguard Worker {
308*c33452fbSAndroid Build Coastguard Worker     _contextMappingKeyArray.push_back(strMappingKey);
309*c33452fbSAndroid Build Coastguard Worker }
310*c33452fbSAndroid Build Coastguard Worker 
311*c33452fbSAndroid Build Coastguard Worker // Subsystem object creator publication (strong reference)
addSubsystemObjectFactory(CSubsystemObjectCreator * pSubsystemObjectCreator)312*c33452fbSAndroid Build Coastguard Worker void CSubsystem::addSubsystemObjectFactory(CSubsystemObjectCreator *pSubsystemObjectCreator)
313*c33452fbSAndroid Build Coastguard Worker {
314*c33452fbSAndroid Build Coastguard Worker     _subsystemObjectCreatorArray.push_back(pSubsystemObjectCreator);
315*c33452fbSAndroid Build Coastguard Worker }
316*c33452fbSAndroid Build Coastguard Worker 
317*c33452fbSAndroid Build Coastguard Worker // Generic error handling from derived subsystem classes
getMappingError(const string & strKey,const string & strMessage,const CConfigurableElement * pConfigurableElement) const318*c33452fbSAndroid Build Coastguard Worker string CSubsystem::getMappingError(const string &strKey, const string &strMessage,
319*c33452fbSAndroid Build Coastguard Worker                                    const CConfigurableElement *pConfigurableElement) const
320*c33452fbSAndroid Build Coastguard Worker {
321*c33452fbSAndroid Build Coastguard Worker     return getName() + " " + getKind() + " " + "mapping:\n" + strKey + " " + "error: \"" +
322*c33452fbSAndroid Build Coastguard Worker            strMessage + "\" " + "for element " + pConfigurableElement->getPath();
323*c33452fbSAndroid Build Coastguard Worker }
324*c33452fbSAndroid Build Coastguard Worker 
getMappingData(const std::string & strKey,const std::string * & pStrValue) const325*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::getMappingData(const std::string &strKey, const std::string *&pStrValue) const
326*c33452fbSAndroid Build Coastguard Worker {
327*c33452fbSAndroid Build Coastguard Worker     if (_pMappingData) {
328*c33452fbSAndroid Build Coastguard Worker 
329*c33452fbSAndroid Build Coastguard Worker         return _pMappingData->getValue(strKey, pStrValue);
330*c33452fbSAndroid Build Coastguard Worker     }
331*c33452fbSAndroid Build Coastguard Worker     return false;
332*c33452fbSAndroid Build Coastguard Worker }
333*c33452fbSAndroid Build Coastguard Worker 
334*c33452fbSAndroid Build Coastguard Worker // Returns the formatted mapping
getFormattedMapping() const335*c33452fbSAndroid Build Coastguard Worker std::string CSubsystem::getFormattedMapping() const
336*c33452fbSAndroid Build Coastguard Worker {
337*c33452fbSAndroid Build Coastguard Worker     if (!_pMappingData) {
338*c33452fbSAndroid Build Coastguard Worker         return "";
339*c33452fbSAndroid Build Coastguard Worker     }
340*c33452fbSAndroid Build Coastguard Worker     return _pMappingData->asString();
341*c33452fbSAndroid Build Coastguard Worker }
342*c33452fbSAndroid Build Coastguard Worker 
343*c33452fbSAndroid Build Coastguard Worker // Mapping generic context handling
handleMappingContext(const CConfigurableElement * pConfigurableElement,CMappingContext & context,string & strError) const344*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::handleMappingContext(const CConfigurableElement *pConfigurableElement,
345*c33452fbSAndroid Build Coastguard Worker                                       CMappingContext &context, string &strError) const
346*c33452fbSAndroid Build Coastguard Worker {
347*c33452fbSAndroid Build Coastguard Worker     // Feed context with found mapping data
348*c33452fbSAndroid Build Coastguard Worker     for (size_t item = 0; item < _contextMappingKeyArray.size(); item++) {
349*c33452fbSAndroid Build Coastguard Worker 
350*c33452fbSAndroid Build Coastguard Worker         const string &strKey = _contextMappingKeyArray[item];
351*c33452fbSAndroid Build Coastguard Worker         const string *pStrValue;
352*c33452fbSAndroid Build Coastguard Worker 
353*c33452fbSAndroid Build Coastguard Worker         if (pConfigurableElement->getMappingData(strKey, pStrValue)) {
354*c33452fbSAndroid Build Coastguard Worker             // Assign item to context
355*c33452fbSAndroid Build Coastguard Worker             if (!context.setItem(item, &strKey, pStrValue)) {
356*c33452fbSAndroid Build Coastguard Worker 
357*c33452fbSAndroid Build Coastguard Worker                 strError = getMappingError(strKey, "Already set", pConfigurableElement);
358*c33452fbSAndroid Build Coastguard Worker 
359*c33452fbSAndroid Build Coastguard Worker                 return false;
360*c33452fbSAndroid Build Coastguard Worker             }
361*c33452fbSAndroid Build Coastguard Worker         }
362*c33452fbSAndroid Build Coastguard Worker     }
363*c33452fbSAndroid Build Coastguard Worker     return true;
364*c33452fbSAndroid Build Coastguard Worker }
365*c33452fbSAndroid Build Coastguard Worker 
366*c33452fbSAndroid Build Coastguard Worker // Subsystem object creation handling
handleSubsystemObjectCreation(CInstanceConfigurableElement * pInstanceConfigurableElement,CMappingContext & context,bool & bHasCreatedSubsystemObject,string & strError)367*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::handleSubsystemObjectCreation(
368*c33452fbSAndroid Build Coastguard Worker     CInstanceConfigurableElement *pInstanceConfigurableElement, CMappingContext &context,
369*c33452fbSAndroid Build Coastguard Worker     bool &bHasCreatedSubsystemObject, string &strError)
370*c33452fbSAndroid Build Coastguard Worker {
371*c33452fbSAndroid Build Coastguard Worker     bHasCreatedSubsystemObject = false;
372*c33452fbSAndroid Build Coastguard Worker 
373*c33452fbSAndroid Build Coastguard Worker     for (const auto *pSubsystemObjectCreator : _subsystemObjectCreatorArray) {
374*c33452fbSAndroid Build Coastguard Worker 
375*c33452fbSAndroid Build Coastguard Worker         // Mapping key
376*c33452fbSAndroid Build Coastguard Worker         string strKey = pSubsystemObjectCreator->getMappingKey();
377*c33452fbSAndroid Build Coastguard Worker         // Object id
378*c33452fbSAndroid Build Coastguard Worker         const string *pStrValue;
379*c33452fbSAndroid Build Coastguard Worker 
380*c33452fbSAndroid Build Coastguard Worker         if (pInstanceConfigurableElement->getMappingData(strKey, pStrValue)) {
381*c33452fbSAndroid Build Coastguard Worker 
382*c33452fbSAndroid Build Coastguard Worker             // First check context consistency
383*c33452fbSAndroid Build Coastguard Worker             // (required ancestors must have been set prior to object creation)
384*c33452fbSAndroid Build Coastguard Worker             uint32_t uiAncestorMask = pSubsystemObjectCreator->getAncestorMask();
385*c33452fbSAndroid Build Coastguard Worker 
386*c33452fbSAndroid Build Coastguard Worker             for (size_t uiAncestorKey = 0; uiAncestorKey < _contextMappingKeyArray.size();
387*c33452fbSAndroid Build Coastguard Worker                  uiAncestorKey++) {
388*c33452fbSAndroid Build Coastguard Worker 
389*c33452fbSAndroid Build Coastguard Worker                 if (!((1 << uiAncestorKey) & uiAncestorMask)) {
390*c33452fbSAndroid Build Coastguard Worker                     // Ancestor not required
391*c33452fbSAndroid Build Coastguard Worker                     continue;
392*c33452fbSAndroid Build Coastguard Worker                 }
393*c33452fbSAndroid Build Coastguard Worker                 // Check ancestor was provided
394*c33452fbSAndroid Build Coastguard Worker                 if (!context.iSet(uiAncestorKey)) {
395*c33452fbSAndroid Build Coastguard Worker 
396*c33452fbSAndroid Build Coastguard Worker                     strError =
397*c33452fbSAndroid Build Coastguard Worker                         getMappingError(strKey, _contextMappingKeyArray[uiAncestorKey] + " not set",
398*c33452fbSAndroid Build Coastguard Worker                                         pInstanceConfigurableElement);
399*c33452fbSAndroid Build Coastguard Worker 
400*c33452fbSAndroid Build Coastguard Worker                     return false;
401*c33452fbSAndroid Build Coastguard Worker                 }
402*c33452fbSAndroid Build Coastguard Worker             }
403*c33452fbSAndroid Build Coastguard Worker 
404*c33452fbSAndroid Build Coastguard Worker             // Then check configurable element size is correct
405*c33452fbSAndroid Build Coastguard Worker             if (pInstanceConfigurableElement->getFootPrint() >
406*c33452fbSAndroid Build Coastguard Worker                 pSubsystemObjectCreator->getMaxConfigurableElementSize()) {
407*c33452fbSAndroid Build Coastguard Worker 
408*c33452fbSAndroid Build Coastguard Worker                 string strSizeError =
409*c33452fbSAndroid Build Coastguard Worker                     "Size should not exceed " +
410*c33452fbSAndroid Build Coastguard Worker                     std::to_string(pSubsystemObjectCreator->getMaxConfigurableElementSize());
411*c33452fbSAndroid Build Coastguard Worker 
412*c33452fbSAndroid Build Coastguard Worker                 strError = getMappingError(strKey, strSizeError, pInstanceConfigurableElement);
413*c33452fbSAndroid Build Coastguard Worker 
414*c33452fbSAndroid Build Coastguard Worker                 return false;
415*c33452fbSAndroid Build Coastguard Worker             }
416*c33452fbSAndroid Build Coastguard Worker 
417*c33452fbSAndroid Build Coastguard Worker             // Do create object and keep its track
418*c33452fbSAndroid Build Coastguard Worker             _subsystemObjectList.push_back(pSubsystemObjectCreator->objectCreate(
419*c33452fbSAndroid Build Coastguard Worker                 *pStrValue, pInstanceConfigurableElement, context, _logger));
420*c33452fbSAndroid Build Coastguard Worker 
421*c33452fbSAndroid Build Coastguard Worker             // Indicate subsytem creation to caller
422*c33452fbSAndroid Build Coastguard Worker             bHasCreatedSubsystemObject = true;
423*c33452fbSAndroid Build Coastguard Worker 
424*c33452fbSAndroid Build Coastguard Worker             // The subsystem Object has been instantiated, no need to continue looking for an
425*c33452fbSAndroid Build Coastguard Worker             // instantiation mapping
426*c33452fbSAndroid Build Coastguard Worker             break;
427*c33452fbSAndroid Build Coastguard Worker         }
428*c33452fbSAndroid Build Coastguard Worker     }
429*c33452fbSAndroid Build Coastguard Worker 
430*c33452fbSAndroid Build Coastguard Worker     return true;
431*c33452fbSAndroid Build Coastguard Worker }
432*c33452fbSAndroid Build Coastguard Worker 
433*c33452fbSAndroid Build Coastguard Worker // From IMapper
434*c33452fbSAndroid Build Coastguard Worker // Handle a configurable element mapping
mapBegin(CInstanceConfigurableElement * pInstanceConfigurableElement,bool & bKeepDiving,string & strError)435*c33452fbSAndroid Build Coastguard Worker bool CSubsystem::mapBegin(CInstanceConfigurableElement *pInstanceConfigurableElement,
436*c33452fbSAndroid Build Coastguard Worker                           bool &bKeepDiving, string &strError)
437*c33452fbSAndroid Build Coastguard Worker {
438*c33452fbSAndroid Build Coastguard Worker     // Get current context
439*c33452fbSAndroid Build Coastguard Worker     CMappingContext context = _contextStack.top();
440*c33452fbSAndroid Build Coastguard Worker 
441*c33452fbSAndroid Build Coastguard Worker     // Add mapping in context
442*c33452fbSAndroid Build Coastguard Worker     if (!handleMappingContext(pInstanceConfigurableElement, context, strError)) {
443*c33452fbSAndroid Build Coastguard Worker 
444*c33452fbSAndroid Build Coastguard Worker         return false;
445*c33452fbSAndroid Build Coastguard Worker     }
446*c33452fbSAndroid Build Coastguard Worker 
447*c33452fbSAndroid Build Coastguard Worker     // Push context
448*c33452fbSAndroid Build Coastguard Worker     _contextStack.push(context);
449*c33452fbSAndroid Build Coastguard Worker 
450*c33452fbSAndroid Build Coastguard Worker     // Assume diving by default
451*c33452fbSAndroid Build Coastguard Worker     bKeepDiving = true;
452*c33452fbSAndroid Build Coastguard Worker 
453*c33452fbSAndroid Build Coastguard Worker     // Deal with ambiguous usage of parameter blocks
454*c33452fbSAndroid Build Coastguard Worker     bool bShouldCreateSubsystemObject = true;
455*c33452fbSAndroid Build Coastguard Worker 
456*c33452fbSAndroid Build Coastguard Worker     switch (pInstanceConfigurableElement->getType()) {
457*c33452fbSAndroid Build Coastguard Worker 
458*c33452fbSAndroid Build Coastguard Worker     case CInstanceConfigurableElement::EComponent:
459*c33452fbSAndroid Build Coastguard Worker     case CInstanceConfigurableElement::EParameterBlock:
460*c33452fbSAndroid Build Coastguard Worker         // Subsystem object creation is optional in parameter blocks
461*c33452fbSAndroid Build Coastguard Worker         bShouldCreateSubsystemObject = false;
462*c33452fbSAndroid Build Coastguard Worker     // No break
463*c33452fbSAndroid Build Coastguard Worker     case CInstanceConfigurableElement::EBitParameterBlock:
464*c33452fbSAndroid Build Coastguard Worker     case CInstanceConfigurableElement::EParameter:
465*c33452fbSAndroid Build Coastguard Worker     case CInstanceConfigurableElement::EStringParameter:
466*c33452fbSAndroid Build Coastguard Worker 
467*c33452fbSAndroid Build Coastguard Worker         bool bHasCreatedSubsystemObject;
468*c33452fbSAndroid Build Coastguard Worker 
469*c33452fbSAndroid Build Coastguard Worker         if (!handleSubsystemObjectCreation(pInstanceConfigurableElement, context,
470*c33452fbSAndroid Build Coastguard Worker                                            bHasCreatedSubsystemObject, strError)) {
471*c33452fbSAndroid Build Coastguard Worker 
472*c33452fbSAndroid Build Coastguard Worker             return false;
473*c33452fbSAndroid Build Coastguard Worker         }
474*c33452fbSAndroid Build Coastguard Worker         // Check for creation error
475*c33452fbSAndroid Build Coastguard Worker         if (bShouldCreateSubsystemObject && !bHasCreatedSubsystemObject) {
476*c33452fbSAndroid Build Coastguard Worker 
477*c33452fbSAndroid Build Coastguard Worker             strError = getMappingError("Not found", "Subsystem object mapping key is missing",
478*c33452fbSAndroid Build Coastguard Worker                                        pInstanceConfigurableElement);
479*c33452fbSAndroid Build Coastguard Worker             return false;
480*c33452fbSAndroid Build Coastguard Worker         }
481*c33452fbSAndroid Build Coastguard Worker         // Not created and no error, keep diving
482*c33452fbSAndroid Build Coastguard Worker         bKeepDiving = !bHasCreatedSubsystemObject;
483*c33452fbSAndroid Build Coastguard Worker 
484*c33452fbSAndroid Build Coastguard Worker         return true;
485*c33452fbSAndroid Build Coastguard Worker 
486*c33452fbSAndroid Build Coastguard Worker     default:
487*c33452fbSAndroid Build Coastguard Worker         assert(0);
488*c33452fbSAndroid Build Coastguard Worker         return false;
489*c33452fbSAndroid Build Coastguard Worker     }
490*c33452fbSAndroid Build Coastguard Worker }
491*c33452fbSAndroid Build Coastguard Worker 
mapEnd()492*c33452fbSAndroid Build Coastguard Worker void CSubsystem::mapEnd()
493*c33452fbSAndroid Build Coastguard Worker {
494*c33452fbSAndroid Build Coastguard Worker     // Unstack context
495*c33452fbSAndroid Build Coastguard Worker     _contextStack.pop();
496*c33452fbSAndroid Build Coastguard Worker }
497