Serializer.cpp revision f4ad6e5637b6deccdac4b60615383f290b3806cf
1f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie/*
2f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * Copyright (C) 2015 The Android Open Source Project
3f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie *
4f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * Licensed under the Apache License, Version 2.0 (the "License");
5f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * you may not use this file except in compliance with the License.
6f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * You may obtain a copy of the License at
7f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie *
8f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie *      http://www.apache.org/licenses/LICENSE-2.0
9f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie *
10f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * Unless required by applicable law or agreed to in writing, software
11f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * distributed under the License is distributed on an "AS IS" BASIS,
12f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * See the License for the specific language governing permissions and
14f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie * limitations under the License.
15f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie */
16f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
17f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#define LOG_TAG "APM::Serializer"
18f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie//#define LOG_NDEBUG 0
19f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
20f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include "Serializer.h"
21f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <convert/convert.h>
22f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include "TypeConverter.h"
23f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <libxml/parser.h>
24f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <libxml/xinclude.h>
25f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <string>
26f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <sstream>
27f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie#include <istream>
28f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
29f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieusing std::string;
30f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
31f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffienamespace android {
32f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
33f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestring getXmlAttribute(const xmlNode *cur, const char *attribute)
34f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
35f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    xmlChar *xmlValue = xmlGetProp(cur, (const xmlChar*)attribute);
36f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (xmlValue == NULL) {
37f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return "";
38f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
39f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string value((const char*)xmlValue);
40f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    xmlFree(xmlValue);
41f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return value;
42f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
43f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
44f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieusing utilities::convertTo;
45f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
46f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const PolicySerializer::rootName = "audioPolicyConfiguration";
47f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const PolicySerializer::versionAttribute = "version";
48f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst uint32_t PolicySerializer::gMajor = 1;
49f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst uint32_t PolicySerializer::gMinor = 0;
50f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
51f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffietemplate <class Trait>
52f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatic status_t deserializeCollection(_xmlDoc *doc, const _xmlNode *cur,
53f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                      typename Trait::Collection &collection,
54f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                      typename Trait::PtrSerializingCtx serializingContext)
55f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
56f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    const xmlNode *root = cur->xmlChildrenNode;
57f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    while (root != NULL) {
58f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (xmlStrcmp(root->name, (const xmlChar *)Trait::collectionTag) &&
59f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                xmlStrcmp(root->name, (const xmlChar *)Trait::tag)) {
60f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            root = root->next;
61f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            continue;
62f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
63f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (!xmlStrcmp(root->name, (const xmlChar *)Trait::collectionTag)) {
64f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            root = root->xmlChildrenNode;
65f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
66f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        while (root != NULL) {
67f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (!xmlStrcmp(root->name, (const xmlChar *)Trait::tag)) {
68f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                typename Trait::PtrElement element;
69f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                status_t status = Trait::deserialize(doc, root, element, serializingContext);
70f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                if (status != NO_ERROR) {
71f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    return status;
72f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                }
73f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                if (collection.add(element) < 0) {
74f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    ALOGE("%s: could not add element to %s collection", __FUNCTION__,
75f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                          Trait::collectionTag);
76f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                }
77f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
78f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            root = root->next;
79f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
80f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return NO_ERROR;
81f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
82f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: No %s collection found", __FUNCTION__, Trait::collectionTag);
83f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return BAD_VALUE;
84f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
85f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
86f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const AudioGainTraits::tag = "gain";
87f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const AudioGainTraits::collectionTag = "gains";
88f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
89f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::mode[] = "mode";
90f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::channelMask[] = "channel_mask";
91f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::minValueMB[] = "minValueMB";
92f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::maxValueMB[] = "maxValueMB";
93f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::defaultValueMB[] = "defaultValueMB";
94f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::stepValueMB[] = "stepValueMB";
95f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::minRampMs[] = "minRampMs";
96f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioGainTraits::Attributes::maxRampMs[] = "maxRampMs";
97f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
98f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t AudioGainTraits::deserialize(_xmlDoc */*doc*/, const _xmlNode *root, PtrElement &gain,
99f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                      PtrSerializingCtx /*serializingContext*/)
100f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
101f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    static uint32_t index = 0;
102f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    gain = new Element(index++, true);
103f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
104f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string mode = getXmlAttribute(root, Attributes::mode);
105f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!mode.empty()) {
106f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setMode(GainModeConverter::maskFromString(mode));
107f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
108f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
109f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string channelsLiteral = getXmlAttribute(root, Attributes::channelMask);
110f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!channelsLiteral.empty()) {
111f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setChannelMask(channelMaskFromString(channelsLiteral));
112f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
113f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
114f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string minValueMBLiteral = getXmlAttribute(root, Attributes::minValueMB);
115f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t minValueMB;
116f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!minValueMBLiteral.empty() && convertTo(minValueMBLiteral, minValueMB)) {
117f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setMinValueInMb(minValueMB);
118f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
119f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
120f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string maxValueMBLiteral = getXmlAttribute(root, Attributes::maxValueMB);
121f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t maxValueMB;
122f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!maxValueMBLiteral.empty() && convertTo(maxValueMBLiteral, maxValueMB)) {
123f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setMaxValueInMb(maxValueMB);
124f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
125f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
126f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string defaultValueMBLiteral = getXmlAttribute(root, Attributes::defaultValueMB);
127f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t defaultValueMB;
128f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!defaultValueMBLiteral.empty() && convertTo(defaultValueMBLiteral, defaultValueMB)) {
129f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setDefaultValueInMb(defaultValueMB);
130f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
131f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
132f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string stepValueMBLiteral = getXmlAttribute(root, Attributes::stepValueMB);
133f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t stepValueMB;
134f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!stepValueMBLiteral.empty() && convertTo(stepValueMBLiteral, stepValueMB)) {
135f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setStepValueInMb(stepValueMB);
136f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
137f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
138f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string minRampMsLiteral = getXmlAttribute(root, Attributes::minRampMs);
139f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t minRampMs;
140f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!minRampMsLiteral.empty() && convertTo(minRampMsLiteral, minRampMs)) {
141f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setMinRampInMs(minRampMs);
142f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
143f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
144f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string maxRampMsLiteral = getXmlAttribute(root, Attributes::maxRampMs);
145f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t maxRampMs;
146f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!maxRampMsLiteral.empty() && convertTo(maxRampMsLiteral, maxRampMs)) {
147f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        gain->setMaxRampInMs(maxRampMs);
148f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
149f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: adding new gain mode %08x channel mask %08x min mB %d max mB %d", __FUNCTION__,
150f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie          gain->getMode(), gain->getChannelMask(), gain->getMinValueInMb(),
151f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie          gain->getMaxValueInMb());
152f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
153f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (gain->getMode() == 0) {
154f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
155f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
156f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
157f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
158f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
159f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const AudioProfileTraits::collectionTag = "profiles";
160f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const AudioProfileTraits::tag = "profile";
161f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
162f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioProfileTraits::Attributes::name[] = "name";
163f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioProfileTraits::Attributes::samplingRates[] = "samplingRates";
164f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioProfileTraits::Attributes::format[] = "format";
165f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char AudioProfileTraits::Attributes::channelMasks[] = "channelMasks";
166f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
167f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t AudioProfileTraits::deserialize(_xmlDoc */*doc*/, const _xmlNode *root, PtrElement &profile,
168f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                         PtrSerializingCtx /*serializingContext*/)
169f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
170f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string samplingRates = getXmlAttribute(root, Attributes::samplingRates);
171f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string format = getXmlAttribute(root, Attributes::format);
172f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string channels = getXmlAttribute(root, Attributes::channelMasks);
173f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
174f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    profile = new Element(formatFromString(format), channelMasksFromString(channels, ","),
175f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                          samplingRatesFromString(samplingRates, ","));
176f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
177f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    profile->setDynamicFormat(profile->getFormat() == gDynamicFormat);
178f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    profile->setDynamicChannels(profile->getChannels().isEmpty());
179f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    profile->setDynamicRate(profile->getSampleRates().isEmpty());
180f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
181f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
182f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
183f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
184f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
185f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const MixPortTraits::collectionTag = "mixPorts";
186f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const MixPortTraits::tag = "mixPort";
187f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
188f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char MixPortTraits::Attributes::name[] = "name";
189f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char MixPortTraits::Attributes::role[] = "role";
190f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char MixPortTraits::Attributes::flags[] = "flags";
191f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
192f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t MixPortTraits::deserialize(_xmlDoc *doc, const _xmlNode *child, PtrElement &mixPort,
193f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                    PtrSerializingCtx /*serializingContext*/)
194f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
195f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string name = getXmlAttribute(child, Attributes::name);
196f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (name.empty()) {
197f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::name);
198f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
199f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
200f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::name, name.c_str());
201f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string role = getXmlAttribute(child, Attributes::role);
202f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (role.empty()) {
203f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::role);
204f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
205f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
206f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: Role=%s", __FUNCTION__, role.c_str());
207f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    audio_port_role_t portRole = role == "source" ? AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK;
208f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
209f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    mixPort = new Element(String8(name.c_str()), portRole);
210f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
211f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    AudioProfileTraits::Collection profiles;
212f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<AudioProfileTraits>(doc, child, profiles, NULL);
213f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (profiles.isEmpty()) {
214f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        sp <AudioProfile> dynamicProfile = new AudioProfile(gDynamicFormat,
215f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                                            ChannelsVector(), SampleRateVector());
216f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicFormat(true);
217f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicChannels(true);
218f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicRate(true);
219f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        profiles.add(dynamicProfile);
220f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
221f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    mixPort->setAudioProfiles(profiles);
222f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
223f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string flags = getXmlAttribute(child, Attributes::flags);
224f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!flags.empty()) {
225f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        // Source role
226f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (portRole == AUDIO_PORT_ROLE_SOURCE) {
227f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            mixPort->setFlags(OutputFlagConverter::maskFromString(flags));
228f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        } else {
229f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            // Sink role
230f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            mixPort->setFlags(InputFlagConverter::maskFromString(flags));
231f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
232f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
233f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Deserialize children
234f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    AudioGainTraits::Collection gains;
235f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<AudioGainTraits>(doc, child, gains, NULL);
236f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    mixPort->setGains(gains);
237f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
238f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
239f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
240f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
241f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const DevicePortTraits::tag = "devicePort";
242f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const DevicePortTraits::collectionTag = "devicePorts";
243f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
244f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char DevicePortTraits::Attributes::tagName[] = "tagName";
245f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char DevicePortTraits::Attributes::type[] = "type";
246f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char DevicePortTraits::Attributes::role[] = "role";
247f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char DevicePortTraits::Attributes::address[] = "address";
248f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char DevicePortTraits::Attributes::roleSource[] = "source";
249f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
250f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t DevicePortTraits::deserialize(_xmlDoc *doc, const _xmlNode *root, PtrElement &deviceDesc,
251f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                       PtrSerializingCtx /*serializingContext*/)
252f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
253f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string name = getXmlAttribute(root, Attributes::tagName);
254f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (name.empty()) {
255f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::tagName);
256f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
257f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
258f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::tagName, name.c_str());
259f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string typeName = getXmlAttribute(root, Attributes::type);
260f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (typeName.empty()) {
261f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: no type for %s", __FUNCTION__, name.c_str());
262f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
263f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
264f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::type, typeName.c_str());
265f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string role = getXmlAttribute(root, Attributes::role);
266f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (role.empty()) {
267f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::role);
268f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
269f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
270f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::role, role.c_str());
271f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    audio_port_role_t portRole = (role == Attributes::roleSource) ?
272f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK;
273f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
274f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    audio_devices_t type = AUDIO_DEVICE_NONE;
275f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!DeviceConverter::fromString(typeName, type) ||
276f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            (!audio_is_input_device(type) && portRole == AUDIO_PORT_ROLE_SOURCE) ||
277f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            (!audio_is_output_devices(type) && portRole == AUDIO_PORT_ROLE_SINK)) {
278f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGW("%s: bad type %08x", __FUNCTION__, type);
279f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
280f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
281f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deviceDesc = new Element(type, String8(name.c_str()));
282f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
283f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string address = getXmlAttribute(root, Attributes::address);
284f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!address.empty()) {
285f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGV("%s: address=%s for %s", __FUNCTION__, address.c_str(), name.c_str());
286f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        deviceDesc->mAddress = String8(address.c_str());
287f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
288f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
289f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    AudioProfileTraits::Collection profiles;
290f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<AudioProfileTraits>(doc, root, profiles, NULL);
291f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (profiles.isEmpty()) {
292f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        sp <AudioProfile> dynamicProfile = new AudioProfile(gDynamicFormat,
293f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                                            ChannelsVector(), SampleRateVector());
294f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicFormat(true);
295f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicChannels(true);
296f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        dynamicProfile->setDynamicRate(true);
297f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        profiles.add(dynamicProfile);
298f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
299f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deviceDesc->setAudioProfiles(profiles);
300f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
301f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Deserialize AudioGain children
302f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<AudioGainTraits>(doc, root, deviceDesc->mGains, NULL);
303f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: adding device tag %s type %08x address %s", __FUNCTION__,
304f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie          deviceDesc->getName().string(), type, deviceDesc->mAddress.string());
305f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
306f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
307f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
308f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const RouteTraits::tag = "route";
309f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const RouteTraits::collectionTag = "routes";
310f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
311f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char RouteTraits::Attributes::type[] = "type";
312f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char RouteTraits::Attributes::typeMix[] = "mix";
313f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char RouteTraits::Attributes::sink[] = "sink";
314f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char RouteTraits::Attributes::sources[] = "sources";
315f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
316f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
317f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t RouteTraits::deserialize(_xmlDoc */*doc*/, const _xmlNode *root, PtrElement &element,
318f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                  PtrSerializingCtx ctx)
319f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
320f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string type = getXmlAttribute(root, Attributes::type);
321f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (type.empty()) {
322f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::type);
323f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
324f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
325f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    audio_route_type_t routeType = (type == Attributes::typeMix) ?
326f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                AUDIO_ROUTE_MIX : AUDIO_ROUTE_MUX;
327f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
328f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::type, type.c_str());
329f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element = new Element(routeType);
330f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
331f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string sinkAttr = getXmlAttribute(root, Attributes::sink);
332f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (sinkAttr.empty()) {
333f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::sink);
334f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
335f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
336f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Convert Sink name to port pointer
337f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    sp<AudioPort> sink = ctx->findPortByTagName(String8(sinkAttr.c_str()));
338f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (sink == NULL) {
339f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: no sink found with name=%s", __FUNCTION__, sinkAttr.c_str());
340f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
341f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
342f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element->setSink(sink);
343f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
344f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string sourcesAttr = getXmlAttribute(root, Attributes::sources);
345f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (sourcesAttr.empty()) {
346f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::sources);
347f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
348f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
349f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Tokenize and Convert Sources name to port pointer
350f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    AudioPortVector sources;
351f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    char *sourcesLiteral = strndup(sourcesAttr.c_str(), strlen(sourcesAttr.c_str()));
352f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    char *devTag = strtok(sourcesLiteral, ",");
353f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    while (devTag != NULL) {
354f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (strlen(devTag) != 0) {
355f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            sp<AudioPort> source = ctx->findPortByTagName(String8(devTag));
356f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (source == NULL) {
357f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                ALOGE("%s: no source found with name=%s", __FUNCTION__, devTag);
358f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                return BAD_VALUE;
359f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
360f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            sources.add(source);
361f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
362f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        devTag = strtok(NULL, ",");
363f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
364f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    free(sourcesLiteral);
365f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
366f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    sink->addRoute(element);
367f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    for (size_t i = 0; i < sources.size(); i++) {
368f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        sp<AudioPort> source = sources.itemAt(i);
369f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        source->addRoute(element);
370f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
371f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element->setSources(sources);
372f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
373f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
374f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
375f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const ModuleTraits::childAttachedDevicesTag = "attachedDevices";
376f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const ModuleTraits::childAttachedDeviceTag = "item";
377f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const ModuleTraits::childDefaultOutputDeviceTag = "defaultOutputDevice";
378f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
379f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const ModuleTraits::tag = "module";
380f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const ModuleTraits::collectionTag = "modules";
381f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
382f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char ModuleTraits::Attributes::name[] = "name";
383f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char ModuleTraits::Attributes::version[] = "halVersion";
384f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
385f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t ModuleTraits::deserialize(xmlDocPtr doc, const xmlNode *root, PtrElement &module,
386f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                   PtrSerializingCtx ctx)
387f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
388f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string name = getXmlAttribute(root, Attributes::name);
389f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (name.empty()) {
390f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::name);
391f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
392f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
393f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    uint32_t version = AUDIO_DEVICE_API_VERSION_MIN;
394f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string versionLiteral = getXmlAttribute(root, Attributes::version);
395f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!versionLiteral.empty()) {
396f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        uint32_t major, minor;
397f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        sscanf(versionLiteral.c_str(), "%u.%u", &major, &minor);
398f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        version = HARDWARE_DEVICE_API_VERSION(major, minor);
399f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGV("%s: mHalVersion = %04x major %u minor %u",  __FUNCTION__,
400f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie              version, major, minor);
401f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
402f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
403f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: %s %s=%s", __FUNCTION__, tag, Attributes::name, name.c_str());
404f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
405f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    module = new Element(name.c_str(), version);
406f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
407f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Deserialize childrens: Audio Mix Port, Audio Device Ports (Source/Sink), Audio Routes
408f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    MixPortTraits::Collection mixPorts;
409f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<MixPortTraits>(doc, root, mixPorts, NULL);
410f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    module->setProfiles(mixPorts);
411f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
412f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    DevicePortTraits::Collection devicePorts;
413f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<DevicePortTraits>(doc, root, devicePorts, NULL);
414f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    module->setDeclaredDevices(devicePorts);
415f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
416f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    RouteTraits::Collection routes;
417f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<RouteTraits>(doc, root, routes, module.get());
418f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    module->setRoutes(routes);
419f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
420f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    const xmlNode *children = root->xmlChildrenNode;
421f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    while (children != NULL) {
422f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (!xmlStrcmp(children->name, (const xmlChar *)childAttachedDevicesTag)) {
423f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            ALOGV("%s: %s %s found", __FUNCTION__, tag, childAttachedDevicesTag);
424f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            const xmlNode *child = children->xmlChildrenNode;
425f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            while (child != NULL) {
426f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                if (!xmlStrcmp(child->name, (const xmlChar *)childAttachedDeviceTag)) {
427f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    xmlChar *attachedDevice = xmlNodeListGetString(doc, child->xmlChildrenNode, 1);
428f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    if (attachedDevice != NULL) {
429f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                        ALOGV("%s: %s %s=%s", __FUNCTION__, tag, childAttachedDeviceTag,
430f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                              (const char*)attachedDevice);
431f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                        sp<DeviceDescriptor> device =
432f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                module->getDeclaredDevices().getDeviceFromTagName(String8((const char*)attachedDevice));
433f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                        ctx->addAvailableDevice(device);
434f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                        xmlFree(attachedDevice);
435f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    }
436f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                }
437f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                child = child->next;
438f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
439f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
440f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (!xmlStrcmp(children->name, (const xmlChar *)childDefaultOutputDeviceTag)) {
441f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            xmlChar *defaultOutputDevice = xmlNodeListGetString(doc, children->xmlChildrenNode, 1);;
442f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (defaultOutputDevice != NULL) {
443f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                ALOGV("%s: %s %s=%s", __FUNCTION__, tag, childDefaultOutputDeviceTag,
444f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                      (const char*)defaultOutputDevice);
445f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                sp<DeviceDescriptor> device =
446f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                        module->getDeclaredDevices().getDeviceFromTagName(String8((const char*)defaultOutputDevice));
447f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                if (device != 0 && ctx->getDefaultOutputDevice() == 0) {
448f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    ctx->setDefaultOutputDevice(device);
449f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    ALOGV("%s: default is %08x", __FUNCTION__, ctx->getDefaultOutputDevice()->type());
450f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                }
451f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                xmlFree(defaultOutputDevice);
452f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
453f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
454f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        children = children->next;
455f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
456f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
457f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
458f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
459f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const GlobalConfigTraits::tag = "globalConfiguration";
460f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
461f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char GlobalConfigTraits::Attributes::speakerDrcEnabled[] = "speaker_drc_enabled";
462f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
463f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
464f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t GlobalConfigTraits::deserialize(const xmlNode *cur, AudioPolicyConfig &config)
465f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
466f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    const xmlNode *root = cur->xmlChildrenNode;
467f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    while (root != NULL) {
468f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (!xmlStrcmp(root->name, (const xmlChar *)tag)) {
469f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            string speakerDrcEnabled =
470f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    getXmlAttribute(root, Attributes::speakerDrcEnabled);
471f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            bool isSpeakerDrcEnabled;
472f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (!speakerDrcEnabled.empty() &&
473f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                    convertTo<string, bool>(speakerDrcEnabled, isSpeakerDrcEnabled)) {
474f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                config.setSpeakerDrcEnabled(isSpeakerDrcEnabled);
475f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
476f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            return NO_ERROR;
477f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
478f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        root = root->next;
479f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
480f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
481f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
482f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
483f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
484f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const VolumeTraits::tag = "volume";
485f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const VolumeTraits::collectionTag = "volumes";
486f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char *const VolumeTraits::volumePointTag = "point";
487f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
488f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char VolumeTraits::Attributes::stream[] = "stream";
489f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffieconst char VolumeTraits::Attributes::deviceCategory[] = "deviceCategory";
490f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
491f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
492f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t VolumeTraits::deserialize(_xmlDoc *doc, const _xmlNode *root, PtrElement &element,
493f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                                   PtrSerializingCtx /*serializingContext*/)
494f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
495f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string streamTypeLiteral = getXmlAttribute(root, Attributes::stream);
496f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (streamTypeLiteral.empty()) {
497f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::stream);
498f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
499f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
500f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    audio_stream_type_t streamType;
501f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!StreamTypeConverter::fromString(streamTypeLiteral, streamType)) {
502f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: Invalid %s", __FUNCTION__, Attributes::stream);
503f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
504f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
505f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string deviceCategoryLiteral = getXmlAttribute(root, Attributes::deviceCategory);
506f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (deviceCategoryLiteral.empty()) {
507f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s found", __FUNCTION__, Attributes::deviceCategory);
508f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
509f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
510f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    device_category deviceCategory;
511f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (!DeviceCategoryConverter::fromString(deviceCategoryLiteral, deviceCategory)) {
512f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: Invalid %s=%s", __FUNCTION__, Attributes::deviceCategory,
513f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie              deviceCategoryLiteral.c_str());
514f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
515f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
516f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    CurvePoints points;
517f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    const xmlNode *child = root->xmlChildrenNode;
518f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    while (child != NULL) {
519f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        if (!xmlStrcmp(child->name, (const xmlChar *)volumePointTag)) {
520f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            xmlChar *pointDefinition = xmlNodeListGetString(doc, child->xmlChildrenNode, 1);;
521f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (pointDefinition == NULL) {
522f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                return BAD_VALUE;
523f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
524f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            ALOGV("%s: %s=%s", __FUNCTION__, tag, (const char*)pointDefinition);
525f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            Vector<int32_t> point;
526f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            collectionFromString<DefaultTraits<int32_t> >((const char*)pointDefinition, point, ",");
527f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            if (point.size() != 2) {
528f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                ALOGE("%s: Invalid %s: %s", __FUNCTION__, volumePointTag,
529f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                      (const char*)pointDefinition);
530f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie                return BAD_VALUE;
531f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            }
532f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            points.add(std::make_pair(point[0], point[1]));
533f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie            xmlFree(pointDefinition);
534f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        }
535f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        child = child->next;
536f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
537f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element = new Element();
538f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element->setDeviceCategory(deviceCategory);
539f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element->setStreamType(streamType);
540f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    element->setCurvePoints(points);
541f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return NO_ERROR;
542f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
543f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
544f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois GaffiePolicySerializer::PolicySerializer() : mRootElementName(rootName)
545f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
546f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    std::ostringstream oss;
547f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    oss << gMajor << "." << gMinor;
548f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    mVersion = oss.str();
549f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ALOGV("%s: Version=%s Root=%s", __FUNCTION__, mVersion.c_str(), mRootElementName.c_str());
550f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
551f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
552f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffiestatus_t PolicySerializer::deserialize(const char *configFile, AudioPolicyConfig &config)
553f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie{
554f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    xmlDocPtr doc;
555f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    doc = xmlParseFile(configFile);
556f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (doc == NULL) {
557f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: Could not parse %s document.", __FUNCTION__, configFile);
558f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
559f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
560f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    xmlNodePtr cur = xmlDocGetRootElement(doc);
561f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (cur == NULL) {
562f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: Could not parse %s document: empty.", __FUNCTION__, configFile);
563f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        xmlFreeDoc(doc);
564f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
565f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
566f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (xmlXIncludeProcess(doc) < 0) {
567f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie         ALOGE("%s: libxml failed to resolve XIncludes on %s document.", __FUNCTION__, configFile);
568f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
569f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
570f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (xmlStrcmp(cur->name, (const xmlChar *) mRootElementName.c_str()))  {
571f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No %s root element found in xml data %s.", __FUNCTION__, mRootElementName.c_str(),
572f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie              (const char *)cur->name);
573f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        xmlFreeDoc(doc);
574f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
575f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
576f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
577f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    string version = getXmlAttribute(cur, versionAttribute);
578f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (version.empty()) {
579f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: No version found in root node %s", __FUNCTION__, mRootElementName.c_str());
580f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
581f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
582f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    if (version != mVersion) {
583f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        ALOGE("%s: Version does not match; expect %s got %s", __FUNCTION__, mVersion.c_str(),
584f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie              version.c_str());
585f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie        return BAD_VALUE;
586f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    }
587f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Lets deserialize children
588f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Modules
589f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    ModuleTraits::Collection modules;
590f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<ModuleTraits>(doc, cur, modules, &config);
591f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    config.setHwModules(modules);
592f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
593f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // deserialize volume section
594f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    VolumeTraits::Collection volumes;
595f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    deserializeCollection<VolumeTraits>(doc, cur, volumes, NULL);
596f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
597f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    // Global Configuration
598f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    GlobalConfigTraits::deserialize(cur, config);
599f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
600f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    xmlFreeDoc(doc);
601f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie    return android::OK;
602f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}
603f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie
604f4ad6e5637b6deccdac4b60615383f290b3806cfFrançois Gaffie}; // namespace android
605