• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #define LOG_TAG "APM::Serializer"
18 //#define LOG_NDEBUG 0
19 
20 #include <memory>
21 #include <string>
22 #include <utility>
23 
24 #include <hidl/Status.h>
25 #include <libxml/parser.h>
26 #include <libxml/xinclude.h>
27 #include <media/convert.h>
28 #include <utils/Log.h>
29 #include <utils/StrongPointer.h>
30 #include <utils/Errors.h>
31 #include <utils/RefBase.h>
32 #include "Serializer.h"
33 #include "TypeConverter.h"
34 
35 namespace android {
36 
37 namespace {
38 
39 // TODO(mnaganov): Consider finding an alternative for using HIDL code.
40 using hardware::Return;
41 using hardware::Status;
42 using utilities::convertTo;
43 
44 template<typename E, typename C>
45 struct AndroidCollectionTraits {
46     typedef sp<E> Element;
47     typedef C Collection;
48     typedef void* PtrSerializingCtx;
49 
addElementToCollectionandroid::__anon4f25d9440111::AndroidCollectionTraits50     static status_t addElementToCollection(const Element &element, Collection *collection) {
51         return collection->add(element) >= 0 ? NO_ERROR : BAD_VALUE;
52     }
53 };
54 
55 template<typename C>
56 struct StdCollectionTraits {
57     typedef C Collection;
58     typedef typename C::value_type Element;
59     typedef void* PtrSerializingCtx;
60 
addElementToCollectionandroid::__anon4f25d9440111::StdCollectionTraits61     static status_t addElementToCollection(const Element &element, Collection *collection) {
62         auto pair = collection->insert(element);
63         return pair.second ? NO_ERROR : BAD_VALUE;
64     }
65 };
66 
67 struct AudioGainTraits : public AndroidCollectionTraits<AudioGain, AudioGains>
68 {
69     static constexpr const char *tag = "gain";
70     static constexpr const char *collectionTag = "gains";
71 
72     struct Attributes
73     {
74         /** gain modes supported, e.g. AUDIO_GAIN_MODE_CHANNELS. */
75         static constexpr const char *mode = "mode";
76         /** controlled channels, needed if mode AUDIO_GAIN_MODE_CHANNELS. */
77         static constexpr const char *channelMask = "channel_mask";
78         static constexpr const char *minValueMB = "minValueMB"; /**< min value in millibel. */
79         static constexpr const char *maxValueMB = "maxValueMB"; /**< max value in millibel. */
80         /** default value in millibel. */
81         static constexpr const char *defaultValueMB = "defaultValueMB";
82         static constexpr const char *stepValueMB = "stepValueMB"; /**< step value in millibel. */
83         /** needed if mode AUDIO_GAIN_MODE_RAMP. */
84         static constexpr const char *minRampMs = "minRampMs";
85         /** needed if mode AUDIO_GAIN_MODE_RAMP. */
86         static constexpr const char *maxRampMs = "maxRampMs";
87         /** needed to allow use setPortGain instead of setStreamVolume. */
88         static constexpr const char *useForVolume = "useForVolume";
89 
90     };
91 
92     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
93     // No children
94 };
95 
96 // A profile section contains a name,  one audio format and the list of supported sampling rates
97 // and channel masks for this format
98 struct AudioProfileTraits : public AndroidCollectionTraits<AudioProfile, AudioProfileVector>
99 {
100     static constexpr const char *tag = "profile";
101     static constexpr const char *collectionTag = "profiles";
102 
103     struct Attributes
104     {
105         static constexpr const char *samplingRates = "samplingRates";
106         static constexpr const char *format = "format";
107         static constexpr const char *channelMasks = "channelMasks";
108     };
109 
110     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
111 };
112 
113 struct MixPortTraits : public AndroidCollectionTraits<IOProfile, IOProfileCollection>
114 {
115     static constexpr const char *tag = "mixPort";
116     static constexpr const char *collectionTag = "mixPorts";
117 
118     struct Attributes
119     {
120         static constexpr const char *name = "name";
121         static constexpr const char *role = "role";
122         static constexpr const char *roleSource = "source"; /**< <attribute role source value>. */
123         static constexpr const char *flags = "flags";
124         static constexpr const char *maxOpenCount = "maxOpenCount";
125         static constexpr const char *maxActiveCount = "maxActiveCount";
126     };
127 
128     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
129     // Children: GainTraits
130 };
131 
132 struct DevicePortTraits : public AndroidCollectionTraits<DeviceDescriptor, DeviceVector>
133 {
134     static constexpr const char *tag = "devicePort";
135     static constexpr const char *collectionTag = "devicePorts";
136 
137     struct Attributes
138     {
139         /**  <device tag name>: any string without space. */
140         static constexpr const char *tagName = "tagName";
141         static constexpr const char *type = "type"; /**< <device type>. */
142         static constexpr const char *role = "role"; /**< <device role: sink or source>. */
143         static constexpr const char *roleSource = "source"; /**< <attribute role source value>. */
144         /** optional: device address, char string less than 64. */
145         static constexpr const char *address = "address";
146         /** optional: the list of encoded audio formats that are known to be supported. */
147         static constexpr const char *encodedFormats = "encodedFormats";
148     };
149 
150     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
151     // Children: GainTraits (optional)
152 };
153 
154 struct RouteTraits : public AndroidCollectionTraits<AudioRoute, AudioRouteVector>
155 {
156     static constexpr const char *tag = "route";
157     static constexpr const char *collectionTag = "routes";
158 
159     struct Attributes
160     {
161         static constexpr const char *type = "type"; /**< <route type>: mix or mux. */
162         static constexpr const char *typeMix = "mix"; /**< type attribute mix value. */
163         static constexpr const char *sink = "sink"; /**< <sink: involved in this route>. */
164         /** sources: all source that can be involved in this route. */
165         static constexpr const char *sources = "sources";
166     };
167 
168     typedef HwModule *PtrSerializingCtx;
169 
170     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
171 };
172 
173 struct ModuleTraits : public AndroidCollectionTraits<HwModule, HwModuleCollection>
174 {
175     static constexpr const char *tag = "module";
176     static constexpr const char *collectionTag = "modules";
177 
178     static constexpr const char *childAttachedDevicesTag = "attachedDevices";
179     static constexpr const char *childAttachedDeviceTag = "item";
180     static constexpr const char *childDefaultOutputDeviceTag = "defaultOutputDevice";
181 
182     struct Attributes
183     {
184         static constexpr const char *name = "name";
185         static constexpr const char *version = "halVersion";
186     };
187 
188     typedef AudioPolicyConfig *PtrSerializingCtx;
189 
190     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
191     // Children: mixPortTraits, devicePortTraits, and routeTraits
192     // Need to call deserialize on each child
193 };
194 
195 struct GlobalConfigTraits
196 {
197     static constexpr const char *tag = "globalConfiguration";
198 
199     struct Attributes
200     {
201         static constexpr const char *speakerDrcEnabled = "speaker_drc_enabled";
202         static constexpr const char *callScreenModeSupported= "call_screen_mode_supported";
203         static constexpr const char *engineLibrarySuffix = "engine_library";
204     };
205 
206     static status_t deserialize(const xmlNode *root, AudioPolicyConfig *config);
207 };
208 
209 struct SurroundSoundTraits
210 {
211     static constexpr const char *tag = "surroundSound";
212 
213     static status_t deserialize(const xmlNode *root, AudioPolicyConfig *config);
214     // Children: SurroundSoundFormatTraits
215 };
216 
217 struct SurroundSoundFormatTraits : public StdCollectionTraits<AudioPolicyConfig::SurroundFormats>
218 {
219     static constexpr const char *tag = "format";
220     static constexpr const char *collectionTag = "formats";
221 
222     struct Attributes
223     {
224         static constexpr const char *name = "name";
225         static constexpr const char *subformats = "subformats";
226     };
227 
228     static Return<Element> deserialize(const xmlNode *cur, PtrSerializingCtx serializingContext);
229 };
230 
231 class PolicySerializer
232 {
233 public:
PolicySerializer()234     PolicySerializer() : mVersion{std::to_string(gMajor) + "." + std::to_string(gMinor)}
235     {
236         ALOGV("%s: Version=%s Root=%s", __func__, mVersion.c_str(), rootName);
237     }
238     status_t deserialize(const char *configFile, AudioPolicyConfig *config);
239 
240 private:
241     static constexpr const char *rootName = "audioPolicyConfiguration";
242     static constexpr const char *versionAttribute = "version";
243     static constexpr uint32_t gMajor = 1; /**< the major number of the policy xml format version. */
244     static constexpr uint32_t gMinor = 0; /**< the minor number of the policy xml format version. */
245 
246     typedef AudioPolicyConfig Element;
247 
248     const std::string mVersion;
249 
250     // Children: ModulesTraits, VolumeTraits, SurroundSoundTraits (optional)
251 };
252 
253 template <class T>
254 constexpr void (*xmlDeleter)(T* t);
255 template <>
256 constexpr auto xmlDeleter<xmlDoc> = xmlFreeDoc;
257 // http://b/111067277 - Add back constexpr when we switch to C++17.
258 template <>
__anon4f25d9440202(xmlChar *s) 259 auto xmlDeleter<xmlChar> = [](xmlChar *s) { xmlFree(s); };
260 
261 /** @return a unique_ptr with the correct deleter for the libxml2 object. */
262 template <class T>
make_xmlUnique(T * t)263 constexpr auto make_xmlUnique(T *t) {
264     // Wrap deleter in lambda to enable empty base optimization
265     auto deleter = [](T *t) { xmlDeleter<T>(t); };
266     return std::unique_ptr<T, decltype(deleter)>{t, deleter};
267 }
268 
getXmlAttribute(const xmlNode * cur,const char * attribute)269 std::string getXmlAttribute(const xmlNode *cur, const char *attribute)
270 {
271     auto xmlValue = make_xmlUnique(xmlGetProp(cur, reinterpret_cast<const xmlChar*>(attribute)));
272     if (xmlValue == nullptr) {
273         return "";
274     }
275     std::string value(reinterpret_cast<const char*>(xmlValue.get()));
276     return value;
277 }
278 
279 template <class Trait>
getReference(const xmlNode * cur,const std::string & refName)280 const xmlNode* getReference(const xmlNode *cur, const std::string &refName)
281 {
282     for (; cur != NULL; cur = cur->next) {
283         if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(Trait::collectionTag))) {
284             for (const xmlNode *child = cur->children; child != NULL; child = child->next) {
285                 if ((!xmlStrcmp(child->name,
286                                         reinterpret_cast<const xmlChar*>(Trait::referenceTag)))) {
287                     std::string name = getXmlAttribute(child, Trait::Attributes::referenceName);
288                     if (refName == name) {
289                         return child;
290                     }
291                 }
292             }
293         }
294     }
295     return NULL;
296 }
297 
298 template <class Trait>
deserializeCollection(const xmlNode * cur,typename Trait::Collection * collection,typename Trait::PtrSerializingCtx serializingContext)299 status_t deserializeCollection(const xmlNode *cur,
300         typename Trait::Collection *collection,
301         typename Trait::PtrSerializingCtx serializingContext)
302 {
303     for (cur = cur->xmlChildrenNode; cur != NULL; cur = cur->next) {
304         const xmlNode *child = NULL;
305         if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(Trait::collectionTag))) {
306             child = cur->xmlChildrenNode;
307         } else if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(Trait::tag))) {
308             child = cur;
309         }
310         for (; child != NULL; child = child->next) {
311             if (!xmlStrcmp(child->name, reinterpret_cast<const xmlChar*>(Trait::tag))) {
312                 auto element = Trait::deserialize(child, serializingContext);
313                 if (element.isOk()) {
314                     status_t status = Trait::addElementToCollection(element, collection);
315                     if (status != NO_ERROR) {
316                         ALOGE("%s: could not add element to %s collection", __func__,
317                             Trait::collectionTag);
318                         return status;
319                     }
320                 } else {
321                     return BAD_VALUE;
322                 }
323             }
324         }
325         if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(Trait::tag))) {
326             return NO_ERROR;
327         }
328     }
329     return NO_ERROR;
330 }
331 
deserialize(const xmlNode * cur,PtrSerializingCtx)332 Return<AudioGainTraits::Element> AudioGainTraits::deserialize(const xmlNode *cur,
333         PtrSerializingCtx /*serializingContext*/)
334 {
335     static uint32_t index = 0;
336     Element gain = new AudioGain(index++, true);
337 
338     std::string mode = getXmlAttribute(cur, Attributes::mode);
339     if (!mode.empty()) {
340         gain->setMode(GainModeConverter::maskFromString(mode));
341     }
342 
343     std::string channelsLiteral = getXmlAttribute(cur, Attributes::channelMask);
344     if (!channelsLiteral.empty()) {
345         gain->setChannelMask(channelMaskFromString(channelsLiteral));
346     }
347 
348     std::string minValueMBLiteral = getXmlAttribute(cur, Attributes::minValueMB);
349     int32_t minValueMB;
350     if (!minValueMBLiteral.empty() && convertTo(minValueMBLiteral, minValueMB)) {
351         gain->setMinValueInMb(minValueMB);
352     }
353 
354     std::string maxValueMBLiteral = getXmlAttribute(cur, Attributes::maxValueMB);
355     int32_t maxValueMB;
356     if (!maxValueMBLiteral.empty() && convertTo(maxValueMBLiteral, maxValueMB)) {
357         gain->setMaxValueInMb(maxValueMB);
358     }
359 
360     std::string defaultValueMBLiteral = getXmlAttribute(cur, Attributes::defaultValueMB);
361     int32_t defaultValueMB;
362     if (!defaultValueMBLiteral.empty() && convertTo(defaultValueMBLiteral, defaultValueMB)) {
363         gain->setDefaultValueInMb(defaultValueMB);
364     }
365 
366     std::string stepValueMBLiteral = getXmlAttribute(cur, Attributes::stepValueMB);
367     uint32_t stepValueMB;
368     if (!stepValueMBLiteral.empty() && convertTo(stepValueMBLiteral, stepValueMB)) {
369         gain->setStepValueInMb(stepValueMB);
370     }
371 
372     std::string minRampMsLiteral = getXmlAttribute(cur, Attributes::minRampMs);
373     uint32_t minRampMs;
374     if (!minRampMsLiteral.empty() && convertTo(minRampMsLiteral, minRampMs)) {
375         gain->setMinRampInMs(minRampMs);
376     }
377 
378     std::string maxRampMsLiteral = getXmlAttribute(cur, Attributes::maxRampMs);
379     uint32_t maxRampMs;
380     if (!maxRampMsLiteral.empty() && convertTo(maxRampMsLiteral, maxRampMs)) {
381         gain->setMaxRampInMs(maxRampMs);
382     }
383     std::string useForVolumeLiteral = getXmlAttribute(cur, Attributes::useForVolume);
384     bool useForVolume = false;
385     if (!useForVolumeLiteral.empty() && convertTo(useForVolumeLiteral, useForVolume)) {
386         gain->setUseForVolume(useForVolume);
387     }
388     ALOGV("%s: adding new gain mode %08x channel mask %08x min mB %d max mB %d UseForVolume: %d",
389           __func__, gain->getMode(), gain->getChannelMask(), gain->getMinValueInMb(),
390           gain->getMaxValueInMb(), useForVolume);
391 
392     if (gain->getMode() != 0) {
393         return gain;
394     } else {
395         return Status::fromStatusT(BAD_VALUE);
396     }
397 }
398 
deserialize(const xmlNode * cur,PtrSerializingCtx)399 Return<AudioProfileTraits::Element> AudioProfileTraits::deserialize(const xmlNode *cur,
400         PtrSerializingCtx /*serializingContext*/)
401 {
402     std::string samplingRates = getXmlAttribute(cur, Attributes::samplingRates);
403     std::string format = getXmlAttribute(cur, Attributes::format);
404     std::string channels = getXmlAttribute(cur, Attributes::channelMasks);
405 
406     Element profile = new AudioProfile(formatFromString(format, gDynamicFormat),
407             channelMasksFromString(channels, ","),
408             samplingRatesFromString(samplingRates, ","));
409 
410     profile->setDynamicFormat(profile->getFormat() == gDynamicFormat);
411     profile->setDynamicChannels(profile->getChannels().empty());
412     profile->setDynamicRate(profile->getSampleRates().empty());
413 
414     return profile;
415 }
416 
deserialize(const xmlNode * child,PtrSerializingCtx)417 Return<MixPortTraits::Element> MixPortTraits::deserialize(const xmlNode *child,
418         PtrSerializingCtx /*serializingContext*/)
419 {
420     std::string name = getXmlAttribute(child, Attributes::name);
421     if (name.empty()) {
422         ALOGE("%s: No %s found", __func__, Attributes::name);
423         return Status::fromStatusT(BAD_VALUE);
424     }
425     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::name, name.c_str());
426     std::string role = getXmlAttribute(child, Attributes::role);
427     if (role.empty()) {
428         ALOGE("%s: No %s found", __func__, Attributes::role);
429         return Status::fromStatusT(BAD_VALUE);
430     }
431     ALOGV("%s: Role=%s", __func__, role.c_str());
432     audio_port_role_t portRole = (role == Attributes::roleSource) ?
433             AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK;
434 
435     Element mixPort = new IOProfile(name, portRole);
436 
437     AudioProfileTraits::Collection profiles;
438     status_t status = deserializeCollection<AudioProfileTraits>(child, &profiles, NULL);
439     if (status != NO_ERROR) {
440         return Status::fromStatusT(status);
441     }
442     if (profiles.empty()) {
443         profiles.add(AudioProfile::createFullDynamic(gDynamicFormat));
444     }
445     // The audio profiles are in order of listed in audio policy configuration file.
446     // Sort audio profiles accroding to the format.
447     sortAudioProfiles(profiles);
448     mixPort->setAudioProfiles(profiles);
449 
450     std::string flags = getXmlAttribute(child, Attributes::flags);
451     if (!flags.empty()) {
452         // Source role
453         if (portRole == AUDIO_PORT_ROLE_SOURCE) {
454             mixPort->setFlags(OutputFlagConverter::maskFromString(flags));
455         } else {
456             // Sink role
457             mixPort->setFlags(InputFlagConverter::maskFromString(flags));
458         }
459     }
460     std::string maxOpenCount = getXmlAttribute(child, Attributes::maxOpenCount);
461     if (!maxOpenCount.empty()) {
462         convertTo(maxOpenCount, mixPort->maxOpenCount);
463     }
464     std::string maxActiveCount = getXmlAttribute(child, Attributes::maxActiveCount);
465     if (!maxActiveCount.empty()) {
466         convertTo(maxActiveCount, mixPort->maxActiveCount);
467     }
468     // Deserialize children
469     AudioGainTraits::Collection gains;
470     status = deserializeCollection<AudioGainTraits>(child, &gains, NULL);
471     if (status != NO_ERROR) {
472         return Status::fromStatusT(status);
473     }
474     mixPort->setGains(gains);
475 
476     return mixPort;
477 }
478 
deserialize(const xmlNode * cur,PtrSerializingCtx)479 Return<DevicePortTraits::Element> DevicePortTraits::deserialize(const xmlNode *cur,
480         PtrSerializingCtx /*serializingContext*/)
481 {
482     std::string name = getXmlAttribute(cur, Attributes::tagName);
483     if (name.empty()) {
484         ALOGE("%s: No %s found", __func__, Attributes::tagName);
485         return Status::fromStatusT(BAD_VALUE);
486     }
487     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::tagName, name.c_str());
488     std::string typeName = getXmlAttribute(cur, Attributes::type);
489     if (typeName.empty()) {
490         ALOGE("%s: no type for %s", __func__, name.c_str());
491         return Status::fromStatusT(BAD_VALUE);
492     }
493     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::type, typeName.c_str());
494     std::string role = getXmlAttribute(cur, Attributes::role);
495     if (role.empty()) {
496         ALOGE("%s: No %s found", __func__, Attributes::role);
497         return Status::fromStatusT(BAD_VALUE);
498     }
499     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::role, role.c_str());
500     audio_port_role_t portRole = (role == Attributes::roleSource) ?
501                 AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK;
502 
503     audio_devices_t type = AUDIO_DEVICE_NONE;
504     if (!deviceFromString(typeName, type) ||
505             (!audio_is_input_device(type) && portRole == AUDIO_PORT_ROLE_SOURCE) ||
506             (!audio_is_output_devices(type) && portRole == AUDIO_PORT_ROLE_SINK)) {
507         ALOGW("%s: bad type %08x", __func__, type);
508         return Status::fromStatusT(BAD_VALUE);
509     }
510     std::string encodedFormatsLiteral = getXmlAttribute(cur, Attributes::encodedFormats);
511     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::encodedFormats, encodedFormatsLiteral.c_str());
512     FormatVector encodedFormats;
513     if (!encodedFormatsLiteral.empty()) {
514         encodedFormats = formatsFromString(encodedFormatsLiteral, " ");
515     }
516     std::string address = getXmlAttribute(cur, Attributes::address);
517     Element deviceDesc = new DeviceDescriptor(type, name, address, encodedFormats);
518 
519     AudioProfileTraits::Collection profiles;
520     status_t status = deserializeCollection<AudioProfileTraits>(cur, &profiles, NULL);
521     if (status != NO_ERROR) {
522         return Status::fromStatusT(status);
523     }
524     if (profiles.empty()) {
525         profiles.add(AudioProfile::createFullDynamic(gDynamicFormat));
526     }
527     // The audio profiles are in order of listed in audio policy configuration file.
528     // Sort audio profiles accroding to the format.
529     sortAudioProfiles(profiles);
530     deviceDesc->setAudioProfiles(profiles);
531 
532     // Deserialize AudioGain children
533     status = deserializeCollection<AudioGainTraits>(cur, &deviceDesc->mGains, NULL);
534     if (status != NO_ERROR) {
535         return Status::fromStatusT(status);
536     }
537     ALOGV("%s: adding device tag %s type %08x address %s", __func__,
538           deviceDesc->getName().c_str(), type, deviceDesc->address().c_str());
539     return deviceDesc;
540 }
541 
deserialize(const xmlNode * cur,PtrSerializingCtx ctx)542 Return<RouteTraits::Element> RouteTraits::deserialize(const xmlNode *cur, PtrSerializingCtx ctx)
543 {
544     std::string type = getXmlAttribute(cur, Attributes::type);
545     if (type.empty()) {
546         ALOGE("%s: No %s found", __func__, Attributes::type);
547         return Status::fromStatusT(BAD_VALUE);
548     }
549     audio_route_type_t routeType = (type == Attributes::typeMix) ?
550                 AUDIO_ROUTE_MIX : AUDIO_ROUTE_MUX;
551 
552     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::type, type.c_str());
553     Element route = new AudioRoute(routeType);
554 
555     std::string sinkAttr = getXmlAttribute(cur, Attributes::sink);
556     if (sinkAttr.empty()) {
557         ALOGE("%s: No %s found", __func__, Attributes::sink);
558         return Status::fromStatusT(BAD_VALUE);
559     }
560     // Convert Sink name to port pointer
561     sp<PolicyAudioPort> sink = ctx->findPortByTagName(sinkAttr);
562     if (sink == NULL) {
563         ALOGE("%s: no sink found with name=%s", __func__, sinkAttr.c_str());
564         return Status::fromStatusT(BAD_VALUE);
565     }
566     route->setSink(sink);
567 
568     std::string sourcesAttr = getXmlAttribute(cur, Attributes::sources);
569     if (sourcesAttr.empty()) {
570         ALOGE("%s: No %s found", __func__, Attributes::sources);
571         return Status::fromStatusT(BAD_VALUE);
572     }
573     // Tokenize and Convert Sources name to port pointer
574     PolicyAudioPortVector sources;
575     std::unique_ptr<char[]> sourcesLiteral{strndup(
576                 sourcesAttr.c_str(), strlen(sourcesAttr.c_str()))};
577     char *devTag = strtok(sourcesLiteral.get(), ",");
578     while (devTag != NULL) {
579         if (strlen(devTag) != 0) {
580             sp<PolicyAudioPort> source = ctx->findPortByTagName(devTag);
581             if (source == NULL) {
582                 ALOGE("%s: no source found with name=%s", __func__, devTag);
583                 return Status::fromStatusT(BAD_VALUE);
584             }
585             sources.add(source);
586         }
587         devTag = strtok(NULL, ",");
588     }
589 
590     sink->addRoute(route);
591     for (size_t i = 0; i < sources.size(); i++) {
592         sp<PolicyAudioPort> source = sources.itemAt(i);
593         source->addRoute(route);
594     }
595     route->setSources(sources);
596     return route;
597 }
598 
deserialize(const xmlNode * cur,PtrSerializingCtx ctx)599 Return<ModuleTraits::Element> ModuleTraits::deserialize(const xmlNode *cur, PtrSerializingCtx ctx)
600 {
601     std::string name = getXmlAttribute(cur, Attributes::name);
602     if (name.empty()) {
603         ALOGE("%s: No %s found", __func__, Attributes::name);
604         return Status::fromStatusT(BAD_VALUE);
605     }
606     uint32_t versionMajor = 0, versionMinor = 0;
607     std::string versionLiteral = getXmlAttribute(cur, Attributes::version);
608     if (!versionLiteral.empty()) {
609         sscanf(versionLiteral.c_str(), "%u.%u", &versionMajor, &versionMinor);
610         ALOGV("%s: mHalVersion = major %u minor %u",  __func__,
611               versionMajor, versionMajor);
612     }
613 
614     ALOGV("%s: %s %s=%s", __func__, tag, Attributes::name, name.c_str());
615 
616     Element module = new HwModule(name.c_str(), versionMajor, versionMinor);
617 
618     // Deserialize childrens: Audio Mix Port, Audio Device Ports (Source/Sink), Audio Routes
619     MixPortTraits::Collection mixPorts;
620     status_t status = deserializeCollection<MixPortTraits>(cur, &mixPorts, NULL);
621     if (status != NO_ERROR) {
622         return Status::fromStatusT(status);
623     }
624     module->setProfiles(mixPorts);
625 
626     DevicePortTraits::Collection devicePorts;
627     status = deserializeCollection<DevicePortTraits>(cur, &devicePorts, NULL);
628     if (status != NO_ERROR) {
629         return Status::fromStatusT(status);
630     }
631     module->setDeclaredDevices(devicePorts);
632 
633     RouteTraits::Collection routes;
634     status = deserializeCollection<RouteTraits>(cur, &routes, module.get());
635     if (status != NO_ERROR) {
636         return Status::fromStatusT(status);
637     }
638     module->setRoutes(routes);
639 
640     for (const xmlNode *children = cur->xmlChildrenNode; children != NULL;
641          children = children->next) {
642         if (!xmlStrcmp(children->name, reinterpret_cast<const xmlChar*>(childAttachedDevicesTag))) {
643             ALOGV("%s: %s %s found", __func__, tag, childAttachedDevicesTag);
644             for (const xmlNode *child = children->xmlChildrenNode; child != NULL;
645                  child = child->next) {
646                 if (!xmlStrcmp(child->name,
647                                 reinterpret_cast<const xmlChar*>(childAttachedDeviceTag))) {
648                     auto attachedDevice = make_xmlUnique(xmlNodeListGetString(
649                                     child->doc, child->xmlChildrenNode, 1));
650                     if (attachedDevice != nullptr) {
651                         ALOGV("%s: %s %s=%s", __func__, tag, childAttachedDeviceTag,
652                                 reinterpret_cast<const char*>(attachedDevice.get()));
653                         sp<DeviceDescriptor> device = module->getDeclaredDevices().
654                                 getDeviceFromTagName(std::string(reinterpret_cast<const char*>(
655                                                         attachedDevice.get())));
656                         ctx->addDevice(device);
657                     }
658                 }
659             }
660         }
661         if (!xmlStrcmp(children->name,
662                         reinterpret_cast<const xmlChar*>(childDefaultOutputDeviceTag))) {
663             auto defaultOutputDevice = make_xmlUnique(xmlNodeListGetString(
664                             children->doc, children->xmlChildrenNode, 1));
665             if (defaultOutputDevice != nullptr) {
666                 ALOGV("%s: %s %s=%s", __func__, tag, childDefaultOutputDeviceTag,
667                         reinterpret_cast<const char*>(defaultOutputDevice.get()));
668                 sp<DeviceDescriptor> device = module->getDeclaredDevices().getDeviceFromTagName(
669                         std::string(reinterpret_cast<const char*>(defaultOutputDevice.get())));
670                 if (device != 0 && ctx->getDefaultOutputDevice() == 0) {
671                     ctx->setDefaultOutputDevice(device);
672                     ALOGV("%s: default is %08x",
673                             __func__, ctx->getDefaultOutputDevice()->type());
674                 }
675             }
676         }
677     }
678     return module;
679 }
680 
deserialize(const xmlNode * root,AudioPolicyConfig * config)681 status_t GlobalConfigTraits::deserialize(const xmlNode *root, AudioPolicyConfig *config)
682 {
683     for (const xmlNode *cur = root->xmlChildrenNode; cur != NULL; cur = cur->next) {
684         if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(tag))) {
685             bool value;
686             std::string attr = getXmlAttribute(cur, Attributes::speakerDrcEnabled);
687             if (!attr.empty() &&
688                     convertTo<std::string, bool>(attr, value)) {
689                 config->setSpeakerDrcEnabled(value);
690             }
691             attr = getXmlAttribute(cur, Attributes::callScreenModeSupported);
692             if (!attr.empty() &&
693                     convertTo<std::string, bool>(attr, value)) {
694                 config->setCallScreenModeSupported(value);
695             }
696             std::string engineLibrarySuffix = getXmlAttribute(cur, Attributes::engineLibrarySuffix);
697             if (!engineLibrarySuffix.empty()) {
698                 config->setEngineLibraryNameSuffix(engineLibrarySuffix);
699             }
700             return NO_ERROR;
701         }
702     }
703     return NO_ERROR;
704 }
705 
deserialize(const xmlNode * root,AudioPolicyConfig * config)706 status_t SurroundSoundTraits::deserialize(const xmlNode *root, AudioPolicyConfig *config)
707 {
708     config->setDefaultSurroundFormats();
709 
710     for (const xmlNode *cur = root->xmlChildrenNode; cur != NULL; cur = cur->next) {
711         if (!xmlStrcmp(cur->name, reinterpret_cast<const xmlChar*>(tag))) {
712             AudioPolicyConfig::SurroundFormats formats;
713             status_t status = deserializeCollection<SurroundSoundFormatTraits>(
714                     cur, &formats, nullptr);
715             if (status == NO_ERROR) {
716                 config->setSurroundFormats(formats);
717             }
718             return NO_ERROR;
719         }
720     }
721     return NO_ERROR;
722 }
723 
deserialize(const xmlNode * cur,PtrSerializingCtx)724 Return<SurroundSoundFormatTraits::Element> SurroundSoundFormatTraits::deserialize(
725         const xmlNode *cur, PtrSerializingCtx /*serializingContext*/)
726 {
727     std::string formatLiteral = getXmlAttribute(cur, Attributes::name);
728     if (formatLiteral.empty()) {
729         ALOGE("%s: No %s found for a surround format", __func__, Attributes::name);
730         return Status::fromStatusT(BAD_VALUE);
731     }
732     audio_format_t format = formatFromString(formatLiteral);
733     if (format == AUDIO_FORMAT_DEFAULT) {
734         ALOGE("%s: Unrecognized format %s", __func__, formatLiteral.c_str());
735         return Status::fromStatusT(BAD_VALUE);
736     }
737     Element pair = std::make_pair(format, Collection::mapped_type{});
738 
739     std::string subformatsLiteral = getXmlAttribute(cur, Attributes::subformats);
740     if (subformatsLiteral.empty()) return pair;
741     FormatVector subformats = formatsFromString(subformatsLiteral, " ");
742     for (const auto& subformat : subformats) {
743         auto result = pair.second.insert(subformat);
744         if (!result.second) {
745             ALOGE("%s: could not add subformat %x to collection", __func__, subformat);
746             return Status::fromStatusT(BAD_VALUE);
747         }
748     }
749     return pair;
750 }
751 
deserialize(const char * configFile,AudioPolicyConfig * config)752 status_t PolicySerializer::deserialize(const char *configFile, AudioPolicyConfig *config)
753 {
754     auto doc = make_xmlUnique(xmlParseFile(configFile));
755     if (doc == nullptr) {
756         ALOGE("%s: Could not parse %s document.", __func__, configFile);
757         return BAD_VALUE;
758     }
759     xmlNodePtr root = xmlDocGetRootElement(doc.get());
760     if (root == NULL) {
761         ALOGE("%s: Could not parse %s document: empty.", __func__, configFile);
762         return BAD_VALUE;
763     }
764     if (xmlXIncludeProcess(doc.get()) < 0) {
765         ALOGE("%s: libxml failed to resolve XIncludes on %s document.", __func__, configFile);
766     }
767 
768     if (xmlStrcmp(root->name, reinterpret_cast<const xmlChar*>(rootName)))  {
769         ALOGE("%s: No %s root element found in xml data %s.", __func__, rootName,
770                 reinterpret_cast<const char*>(root->name));
771         return BAD_VALUE;
772     }
773 
774     std::string version = getXmlAttribute(root, versionAttribute);
775     if (version.empty()) {
776         ALOGE("%s: No version found in root node %s", __func__, rootName);
777         return BAD_VALUE;
778     }
779     if (version != mVersion) {
780         ALOGE("%s: Version does not match; expect %s got %s", __func__, mVersion.c_str(),
781               version.c_str());
782         return BAD_VALUE;
783     }
784     // Lets deserialize children
785     // Modules
786     ModuleTraits::Collection modules;
787     status_t status = deserializeCollection<ModuleTraits>(root, &modules, config);
788     if (status != NO_ERROR) {
789         return status;
790     }
791     config->setHwModules(modules);
792 
793     // Global Configuration
794     GlobalConfigTraits::deserialize(root, config);
795 
796     // Surround configuration
797     SurroundSoundTraits::deserialize(root, config);
798 
799     return android::OK;
800 }
801 
802 }  // namespace
803 
deserializeAudioPolicyFile(const char * fileName,AudioPolicyConfig * config)804 status_t deserializeAudioPolicyFile(const char *fileName, AudioPolicyConfig *config)
805 {
806     PolicySerializer serializer;
807     return serializer.deserialize(fileName, config);
808 }
809 
810 } // namespace android
811