/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "APM::Serializer" //#define LOG_NDEBUG 0 #include #include #include #include #include #include #include #include #include #include #include #include "IOProfile.h" #include "Serializer.h" #include "TypeConverter.h" namespace android { namespace { using utilities::convertTo; static inline bool maybeVendorExtension(const std::string& s) { // Only checks whether the string starts with the "vendor prefix". static const std::string vendorPrefix = "VX_"; return s.size() > vendorPrefix.size() && s.substr(0, vendorPrefix.size()) == vendorPrefix; } template struct AndroidCollectionTraits { typedef sp Element; typedef C Collection; typedef void* PtrSerializingCtx; static status_t addElementToCollection(const Element &element, Collection *collection) { return collection->add(element) >= 0 ? NO_ERROR : BAD_VALUE; } }; template struct StdCollectionTraits { typedef C Collection; typedef typename C::value_type Element; typedef void* PtrSerializingCtx; static status_t addElementToCollection(const Element &element, Collection *collection) { auto pair = collection->insert(element); return pair.second ? NO_ERROR : BAD_VALUE; } }; struct AudioGainTraits : public AndroidCollectionTraits { static constexpr const char *tag = "gain"; static constexpr const char *collectionTag = "gains"; struct Attributes { /** gain modes supported, e.g. AUDIO_GAIN_MODE_CHANNELS. */ static constexpr const char *mode = "mode"; /** controlled channels, needed if mode AUDIO_GAIN_MODE_CHANNELS. */ static constexpr const char *channelMask = "channel_mask"; static constexpr const char *minValueMB = "minValueMB"; /**< min value in millibel. */ static constexpr const char *maxValueMB = "maxValueMB"; /**< max value in millibel. */ /** default value in millibel. */ static constexpr const char *defaultValueMB = "defaultValueMB"; static constexpr const char *stepValueMB = "stepValueMB"; /**< step value in millibel. */ /** needed if mode AUDIO_GAIN_MODE_RAMP. */ static constexpr const char *minRampMs = "minRampMs"; /** needed if mode AUDIO_GAIN_MODE_RAMP. */ static constexpr const char *maxRampMs = "maxRampMs"; /** needed to allow use setPortGain instead of setStreamVolume. */ static constexpr const char *useForVolume = "useForVolume"; }; // No children }; // A profile section contains a name, one audio format and the list of supported sampling rates // and channel masks for this format struct AudioProfileTraits : public AndroidCollectionTraits { static constexpr const char *tag = "profile"; static constexpr const char *collectionTag = "profiles"; struct Attributes { static constexpr const char *samplingRates = "samplingRates"; static constexpr const char *format = "format"; static constexpr const char *channelMasks = "channelMasks"; }; }; struct MixPortTraits : public AndroidCollectionTraits { static constexpr const char *tag = "mixPort"; static constexpr const char *collectionTag = "mixPorts"; struct Attributes { static constexpr const char *name = "name"; static constexpr const char *role = "role"; static constexpr const char *roleSource = "source"; /**< . */ static constexpr const char *flags = "flags"; static constexpr const char *maxOpenCount = "maxOpenCount"; static constexpr const char *maxActiveCount = "maxActiveCount"; static constexpr const char *recommendedMuteDurationMs = "recommendedMuteDurationMs"; }; // Children: GainTraits }; struct DevicePortTraits : public AndroidCollectionTraits { static constexpr const char *tag = "devicePort"; static constexpr const char *collectionTag = "devicePorts"; struct Attributes { /** : any string without space. */ static constexpr const char *tagName = "tagName"; static constexpr const char *type = "type"; /**< . */ static constexpr const char *role = "role"; /**< . */ static constexpr const char *roleSource = "source"; /**< . */ /** optional: device address, char string less than 64. */ static constexpr const char *address = "address"; /** optional: the list of encoded audio formats that are known to be supported. */ static constexpr const char *encodedFormats = "encodedFormats"; }; // Children: GainTraits (optional) }; struct RouteTraits : public AndroidCollectionTraits { static constexpr const char *tag = "route"; static constexpr const char *collectionTag = "routes"; struct Attributes { static constexpr const char *type = "type"; /**< : mix or mux. */ static constexpr const char *typeMix = "mix"; /**< type attribute mix value. */ static constexpr const char *sink = "sink"; /**< . */ /** sources: all source that can be involved in this route. */ static constexpr const char *sources = "sources"; }; typedef HwModule *PtrSerializingCtx; }; struct ModuleTraits : public AndroidCollectionTraits { static constexpr const char *tag = "module"; static constexpr const char *collectionTag = "modules"; static constexpr const char *childAttachedDevicesTag = "attachedDevices"; static constexpr const char *childAttachedDeviceTag = "item"; static constexpr const char *childDefaultOutputDeviceTag = "defaultOutputDevice"; struct Attributes { static constexpr const char *name = "name"; static constexpr const char *version = "halVersion"; }; typedef AudioPolicyConfig *PtrSerializingCtx; // Children: mixPortTraits, devicePortTraits, and routeTraits // Need to call deserialize on each child }; struct GlobalConfigTraits { typedef std::monostate Element; static constexpr const char *tag = "globalConfiguration"; struct Attributes { static constexpr const char *callScreenModeSupported= "call_screen_mode_supported"; static constexpr const char *engineLibrarySuffix = "engine_library"; }; typedef AudioPolicyConfig *PtrSerializingCtx; }; struct SurroundSoundTraits { typedef std::monostate Element; static constexpr const char *tag = "surroundSound"; typedef AudioPolicyConfig *PtrSerializingCtx; // Children: SurroundSoundFormatTraits }; struct SurroundSoundFormatTraits : public StdCollectionTraits { static constexpr const char *tag = "format"; static constexpr const char *collectionTag = "formats"; struct Attributes { static constexpr const char *name = "name"; static constexpr const char *subformats = "subformats"; }; }; class PolicySerializer { public: status_t deserialize(const char *configFile, AudioPolicyConfig *config, bool ignoreVendorExtensions = false); template status_t deserializeCollection(const xmlNode *cur, typename Trait::Collection *collection, typename Trait::PtrSerializingCtx serializingContext); template std::variant deserialize(const xmlNode *cur, typename Trait::PtrSerializingCtx serializingContext); private: static constexpr const char *rootName = "audioPolicyConfiguration"; static constexpr const char *versionAttribute = "version"; typedef AudioPolicyConfig Element; bool mIgnoreVendorExtensions = false; std::string mChannelMasksSeparator = ","; std::string mSamplingRatesSeparator = ","; std::string mFlagsSeparator = "|"; // Children: ModulesTraits, VolumeTraits, SurroundSoundTraits (optional) }; // Deleter using free() for use with std::unique_ptr<>. See also UniqueCPtr<> below. struct FreeDelete { // NOTE: Deleting a const object is valid but free() takes a non-const pointer. void operator()(const void* ptr) const { free(const_cast(ptr)); } }; // Alias for std::unique_ptr<> that uses the C function free() to delete objects. template using UniqueCPtr = std::unique_ptr; template constexpr void (*xmlDeleter)(T* t); template <> constexpr auto xmlDeleter = xmlFreeDoc; template <> constexpr auto xmlDeleter = [](xmlChar *s) { xmlFree(s); }; /** @return a unique_ptr with the correct deleter for the libxml2 object. */ template constexpr auto make_xmlUnique(T *t) { // Wrap deleter in lambda to enable empty base optimization auto deleter = [](T *t) { xmlDeleter(t); }; return std::unique_ptr{t, deleter}; } std::string getXmlAttribute(const xmlNode *cur, const char *attribute) { auto xmlValue = make_xmlUnique(xmlGetProp(cur, reinterpret_cast(attribute))); if (xmlValue == nullptr) { return ""; } std::string value(reinterpret_cast(xmlValue.get())); return value; } template const xmlNode* getReference(const xmlNode *cur, const std::string &refName) { for (; cur != NULL; cur = cur->next) { if (!xmlStrcmp(cur->name, reinterpret_cast(Trait::collectionTag))) { for (const xmlNode *child = cur->children; child != NULL; child = child->next) { if ((!xmlStrcmp(child->name, reinterpret_cast(Trait::referenceTag)))) { std::string name = getXmlAttribute(child, Trait::Attributes::referenceName); if (refName == name) { return child; } } } } } return NULL; } template status_t PolicySerializer::deserializeCollection(const xmlNode *cur, typename Trait::Collection *collection, typename Trait::PtrSerializingCtx serializingContext) { for (cur = cur->xmlChildrenNode; cur != NULL; cur = cur->next) { const xmlNode *child = NULL; if (!xmlStrcmp(cur->name, reinterpret_cast(Trait::collectionTag))) { child = cur->xmlChildrenNode; } else if (!xmlStrcmp(cur->name, reinterpret_cast(Trait::tag))) { child = cur; } for (; child != NULL; child = child->next) { if (!xmlStrcmp(child->name, reinterpret_cast(Trait::tag))) { auto maybeElement = deserialize(child, serializingContext); if (maybeElement.index() == 1) { status_t status = Trait::addElementToCollection( std::get<1>(maybeElement), collection); if (status != NO_ERROR) { ALOGE("%s: could not add element to %s collection", __func__, Trait::collectionTag); return status; } } else if (mIgnoreVendorExtensions && std::get(maybeElement) == NO_INIT) { // Skip a vendor extension element. } else { return BAD_VALUE; } } } if (!xmlStrcmp(cur->name, reinterpret_cast(Trait::tag))) { return NO_ERROR; } } return NO_ERROR; } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, AudioGainTraits::PtrSerializingCtx /*serializingContext*/) { using Attributes = AudioGainTraits::Attributes; static uint32_t index = 0; AudioGainTraits::Element gain = new AudioGain(index++, true); std::string mode = getXmlAttribute(cur, Attributes::mode); if (!mode.empty()) { gain->setMode(GainModeConverter::maskFromString(mode, " ")); } std::string channelsLiteral = getXmlAttribute(cur, Attributes::channelMask); if (!channelsLiteral.empty()) { gain->setChannelMask(channelMaskFromString(channelsLiteral)); } std::string minValueMBLiteral = getXmlAttribute(cur, Attributes::minValueMB); int32_t minValueMB; if (!minValueMBLiteral.empty() && convertTo(minValueMBLiteral, minValueMB)) { gain->setMinValueInMb(minValueMB); } std::string maxValueMBLiteral = getXmlAttribute(cur, Attributes::maxValueMB); int32_t maxValueMB; if (!maxValueMBLiteral.empty() && convertTo(maxValueMBLiteral, maxValueMB)) { gain->setMaxValueInMb(maxValueMB); } std::string defaultValueMBLiteral = getXmlAttribute(cur, Attributes::defaultValueMB); int32_t defaultValueMB; if (!defaultValueMBLiteral.empty() && convertTo(defaultValueMBLiteral, defaultValueMB)) { gain->setDefaultValueInMb(defaultValueMB); } std::string stepValueMBLiteral = getXmlAttribute(cur, Attributes::stepValueMB); uint32_t stepValueMB; if (!stepValueMBLiteral.empty() && convertTo(stepValueMBLiteral, stepValueMB)) { gain->setStepValueInMb(stepValueMB); } std::string minRampMsLiteral = getXmlAttribute(cur, Attributes::minRampMs); uint32_t minRampMs; if (!minRampMsLiteral.empty() && convertTo(minRampMsLiteral, minRampMs)) { gain->setMinRampInMs(minRampMs); } std::string maxRampMsLiteral = getXmlAttribute(cur, Attributes::maxRampMs); uint32_t maxRampMs; if (!maxRampMsLiteral.empty() && convertTo(maxRampMsLiteral, maxRampMs)) { gain->setMaxRampInMs(maxRampMs); } std::string useForVolumeLiteral = getXmlAttribute(cur, Attributes::useForVolume); bool useForVolume = false; if (!useForVolumeLiteral.empty() && convertTo(useForVolumeLiteral, useForVolume)) { gain->setUseForVolume(useForVolume); } ALOGV("%s: adding new gain mode %08x channel mask %08x min mB %d max mB %d UseForVolume: %d", __func__, gain->getMode(), gain->getChannelMask(), gain->getMinValueInMb(), gain->getMaxValueInMb(), useForVolume); if (gain->getMode() != 0) { return gain; } else { return BAD_VALUE; } } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, AudioProfileTraits::PtrSerializingCtx /*serializingContext*/) { using Attributes = AudioProfileTraits::Attributes; std::string samplingRates = getXmlAttribute(cur, Attributes::samplingRates); std::string format = getXmlAttribute(cur, Attributes::format); std::string channels = getXmlAttribute(cur, Attributes::channelMasks); if (mIgnoreVendorExtensions && maybeVendorExtension(format)) { ALOGI("%s: vendor extension format \"%s\" skipped", __func__, format.c_str()); return NO_INIT; } AudioProfileTraits::Element profile = new AudioProfile(formatFromString(format, gDynamicFormat), channelMasksFromString(channels, mChannelMasksSeparator.c_str()), samplingRatesFromString(samplingRates, mSamplingRatesSeparator.c_str())); profile->setDynamicFormat(profile->getFormat() == gDynamicFormat); profile->setDynamicChannels(profile->getChannels().empty()); profile->setDynamicRate(profile->getSampleRates().empty()); return profile; } template<> std::variant PolicySerializer::deserialize( const xmlNode *child, MixPortTraits::PtrSerializingCtx /*serializingContext*/) { using Attributes = MixPortTraits::Attributes; std::string name = getXmlAttribute(child, Attributes::name); if (name.empty()) { ALOGE("%s: No %s found", __func__, Attributes::name); return BAD_VALUE; } ALOGV("%s: %s %s=%s", __func__, MixPortTraits::tag, Attributes::name, name.c_str()); std::string role = getXmlAttribute(child, Attributes::role); if (role.empty()) { ALOGE("%s: No %s found", __func__, Attributes::role); return BAD_VALUE; } ALOGV("%s: Role=%s", __func__, role.c_str()); audio_port_role_t portRole = (role == Attributes::roleSource) ? AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK; MixPortTraits::Element mixPort = new IOProfile(name, portRole); AudioProfileTraits::Collection profiles; status_t status = deserializeCollection(child, &profiles, NULL); if (status != NO_ERROR) { return status; } if (profiles.empty()) { profiles.add(AudioProfile::createFullDynamic(gDynamicFormat)); } // The audio profiles are in order of listed in audio policy configuration file. // Sort audio profiles accroding to the format. sortAudioProfiles(profiles); mixPort->setAudioProfiles(profiles); std::string flags = getXmlAttribute(child, Attributes::flags); if (!flags.empty()) { // Source role if (portRole == AUDIO_PORT_ROLE_SOURCE) { //TODO: b/193496180 use spatializer flag at audio HAL when available until then, // use DEEP_BUFFER+FAST flag combo to indicate the spatializer output profile uint32_t intFlags = OutputFlagConverter::maskFromString(flags, mFlagsSeparator.c_str()); if (intFlags == (AUDIO_OUTPUT_FLAG_FAST | AUDIO_OUTPUT_FLAG_DEEP_BUFFER)) { intFlags = AUDIO_OUTPUT_FLAG_SPATIALIZER; } mixPort->setFlags(intFlags); } else { // Sink role mixPort->setFlags(InputFlagConverter::maskFromString(flags, mFlagsSeparator.c_str())); } } std::string maxOpenCount = getXmlAttribute(child, Attributes::maxOpenCount); if (!maxOpenCount.empty()) { convertTo(maxOpenCount, mixPort->maxOpenCount); } std::string maxActiveCount = getXmlAttribute(child, Attributes::maxActiveCount); if (!maxActiveCount.empty()) { convertTo(maxActiveCount, mixPort->maxActiveCount); } std::string recommendedmuteDurationMsLiteral = getXmlAttribute(child, Attributes::recommendedMuteDurationMs); if (!recommendedmuteDurationMsLiteral.empty()) { convertTo(recommendedmuteDurationMsLiteral, mixPort->recommendedMuteDurationMs); } // Deserialize children AudioGainTraits::Collection gains; status = deserializeCollection(child, &gains, NULL); if (status != NO_ERROR) { return status; } mixPort->setGains(gains); return mixPort; } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, DevicePortTraits::PtrSerializingCtx /*serializingContext*/) { using Attributes = DevicePortTraits::Attributes; auto& tag = DevicePortTraits::tag; std::string name = getXmlAttribute(cur, Attributes::tagName); if (name.empty()) { ALOGE("%s: No %s found", __func__, Attributes::tagName); return BAD_VALUE; } ALOGV("%s: %s %s=%s", __func__, tag, Attributes::tagName, name.c_str()); std::string typeName = getXmlAttribute(cur, Attributes::type); if (typeName.empty()) { ALOGE("%s: no type for %s", __func__, name.c_str()); return BAD_VALUE; } ALOGV("%s: %s %s=%s", __func__, tag, Attributes::type, typeName.c_str()); std::string role = getXmlAttribute(cur, Attributes::role); if (role.empty()) { ALOGE("%s: No %s found", __func__, Attributes::role); return BAD_VALUE; } ALOGV("%s: %s %s=%s", __func__, tag, Attributes::role, role.c_str()); audio_port_role_t portRole = (role == Attributes::roleSource) ? AUDIO_PORT_ROLE_SOURCE : AUDIO_PORT_ROLE_SINK; if (mIgnoreVendorExtensions && maybeVendorExtension(typeName)) { ALOGI("%s: vendor extension device type \"%s\" skipped", __func__, typeName.c_str()); return NO_INIT; } audio_devices_t type = AUDIO_DEVICE_NONE; if (!DeviceConverter::fromString(typeName, type) || (!audio_is_input_device(type) && portRole == AUDIO_PORT_ROLE_SOURCE) || (!audio_is_output_devices(type) && portRole == AUDIO_PORT_ROLE_SINK)) { ALOGW("%s: bad type %08x", __func__, type); return BAD_VALUE; } std::string encodedFormatsLiteral = getXmlAttribute(cur, Attributes::encodedFormats); ALOGV("%s: %s %s=%s", __func__, tag, Attributes::encodedFormats, encodedFormatsLiteral.c_str()); FormatVector encodedFormats; if (!encodedFormatsLiteral.empty()) { encodedFormats = formatsFromString(encodedFormatsLiteral, " "); } std::string address = getXmlAttribute(cur, Attributes::address); DevicePortTraits::Element deviceDesc = new DeviceDescriptor(type, name, address, encodedFormats); AudioProfileTraits::Collection profiles; status_t status = deserializeCollection(cur, &profiles, NULL); if (status != NO_ERROR) { return status; } if (profiles.empty()) { profiles.add(AudioProfile::createFullDynamic(gDynamicFormat)); } // The audio profiles are in order of listed in audio policy configuration file. // Sort audio profiles accroding to the format. sortAudioProfiles(profiles); deviceDesc->setAudioProfiles(profiles); // Deserialize AudioGain children status = deserializeCollection(cur, &deviceDesc->mGains, NULL); if (status != NO_ERROR) { return status; } ALOGV("%s: adding device tag %s type %08x address %s", __func__, deviceDesc->getName().c_str(), type, deviceDesc->address().c_str()); return deviceDesc; } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, RouteTraits::PtrSerializingCtx ctx) { using Attributes = RouteTraits::Attributes; std::string type = getXmlAttribute(cur, Attributes::type); if (type.empty()) { ALOGE("%s: No %s found", __func__, Attributes::type); return BAD_VALUE; } audio_route_type_t routeType = (type == Attributes::typeMix) ? AUDIO_ROUTE_MIX : AUDIO_ROUTE_MUX; ALOGV("%s: %s %s=%s", __func__, RouteTraits::tag, Attributes::type, type.c_str()); RouteTraits::Element route = new AudioRoute(routeType); std::string sinkAttr = getXmlAttribute(cur, Attributes::sink); if (sinkAttr.empty()) { ALOGE("%s: No %s found", __func__, Attributes::sink); return BAD_VALUE; } // Convert Sink name to port pointer sp sink = ctx->findPortByTagName(sinkAttr); if (sink == NULL) { if (!mIgnoreVendorExtensions) { ALOGE("%s: no sink found with name \"%s\"", __func__, sinkAttr.c_str()); return BAD_VALUE; } else { ALOGW("%s: skipping route to sink \"%s\" as it likely has vendor extension type", __func__, sinkAttr.c_str()); return NO_INIT; } } route->setSink(sink); std::string sourcesAttr = getXmlAttribute(cur, Attributes::sources); if (sourcesAttr.empty()) { ALOGE("%s: No %s found", __func__, Attributes::sources); return BAD_VALUE; } // Tokenize and Convert Sources name to port pointer PolicyAudioPortVector sources; UniqueCPtr sourcesLiteral{strndup( sourcesAttr.c_str(), strlen(sourcesAttr.c_str()))}; char *devTag = strtok(sourcesLiteral.get(), ","); while (devTag != NULL) { if (strlen(devTag) != 0) { sp source = ctx->findPortByTagName(devTag); if (source == NULL) { if (!mIgnoreVendorExtensions) { ALOGE("%s: no source found with name \"%s\"", __func__, devTag); return BAD_VALUE; } else { ALOGW("%s: skipping route source \"%s\" as it likely has vendor extension type", __func__, devTag); } } else { sources.add(source); } } devTag = strtok(NULL, ","); } sink->addRoute(route); for (size_t i = 0; i < sources.size(); i++) { sp source = sources.itemAt(i); source->addRoute(route); } route->setSources(sources); return route; } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, ModuleTraits::PtrSerializingCtx ctx) { using Attributes = ModuleTraits::Attributes; auto& tag = ModuleTraits::tag; auto& childAttachedDevicesTag = ModuleTraits::childAttachedDevicesTag; auto& childAttachedDeviceTag = ModuleTraits::childAttachedDeviceTag; auto& childDefaultOutputDeviceTag = ModuleTraits::childDefaultOutputDeviceTag; std::string name = getXmlAttribute(cur, Attributes::name); if (name.empty()) { ALOGE("%s: No %s found", __func__, Attributes::name); return BAD_VALUE; } uint32_t versionMajor = 0, versionMinor = 0; std::string versionLiteral = getXmlAttribute(cur, Attributes::version); if (!versionLiteral.empty()) { sscanf(versionLiteral.c_str(), "%u.%u", &versionMajor, &versionMinor); ALOGV("%s: mHalVersion = major %u minor %u", __func__, versionMajor, versionMajor); } ALOGV("%s: %s %s=%s", __func__, ModuleTraits::tag, Attributes::name, name.c_str()); ModuleTraits::Element module = new HwModule(name.c_str(), versionMajor, versionMinor); // Deserialize children: Audio Mix Port, Audio Device Ports (Source/Sink), Audio Routes MixPortTraits::Collection mixPorts; status_t status = deserializeCollection(cur, &mixPorts, NULL); if (status != NO_ERROR) { return status; } module->setProfiles(mixPorts); DevicePortTraits::Collection devicePorts; status = deserializeCollection(cur, &devicePorts, NULL); if (status != NO_ERROR) { return status; } module->setDeclaredDevices(devicePorts); RouteTraits::Collection routes; status = deserializeCollection(cur, &routes, module.get()); if (status != NO_ERROR) { return status; } module->setRoutes(routes); for (const xmlNode *children = cur->xmlChildrenNode; children != NULL; children = children->next) { if (!xmlStrcmp(children->name, reinterpret_cast(childAttachedDevicesTag))) { ALOGV("%s: %s %s found", __func__, tag, childAttachedDevicesTag); for (const xmlNode *child = children->xmlChildrenNode; child != NULL; child = child->next) { if (!xmlStrcmp(child->name, reinterpret_cast(childAttachedDeviceTag))) { auto attachedDevice = make_xmlUnique(xmlNodeListGetString( child->doc, child->xmlChildrenNode, 1)); if (attachedDevice != nullptr) { ALOGV("%s: %s %s=%s", __func__, tag, childAttachedDeviceTag, reinterpret_cast(attachedDevice.get())); sp device = module->getDeclaredDevices(). getDeviceFromTagName(std::string(reinterpret_cast( attachedDevice.get()))); if (device == NULL) { if (mIgnoreVendorExtensions) { ALOGW("%s: skipped attached device \"%s\" because it likely uses a " "vendor extension type", __func__, reinterpret_cast(attachedDevice.get())); } else { ALOGE("%s: got null device in %s, \"%s\"", __func__, child->name, reinterpret_cast(attachedDevice.get())); } continue; } ctx->addDevice(device); } } } } if (!xmlStrcmp(children->name, reinterpret_cast(childDefaultOutputDeviceTag))) { auto defaultOutputDevice = make_xmlUnique(xmlNodeListGetString( children->doc, children->xmlChildrenNode, 1)); if (defaultOutputDevice != nullptr) { ALOGV("%s: %s %s=%s", __func__, tag, childDefaultOutputDeviceTag, reinterpret_cast(defaultOutputDevice.get())); sp device = module->getDeclaredDevices().getDeviceFromTagName( std::string(reinterpret_cast(defaultOutputDevice.get()))); if (device != 0 && ctx->getDefaultOutputDevice() == 0) { ctx->setDefaultOutputDevice(device); ALOGV("%s: default is %08x", __func__, ctx->getDefaultOutputDevice()->type()); } } } } return module; } template<> std::variant PolicySerializer::deserialize( const xmlNode *root, GlobalConfigTraits::PtrSerializingCtx config) { using Attributes = GlobalConfigTraits::Attributes; for (const xmlNode *cur = root->xmlChildrenNode; cur != NULL; cur = cur->next) { if (!xmlStrcmp(cur->name, reinterpret_cast(GlobalConfigTraits::tag))) { bool value; std::string attr = getXmlAttribute(cur, Attributes::callScreenModeSupported); if (!attr.empty() && convertTo(attr, value)) { config->setCallScreenModeSupported(value); } std::string engineLibrarySuffix = getXmlAttribute(cur, Attributes::engineLibrarySuffix); if (!engineLibrarySuffix.empty()) { config->setEngineLibraryNameSuffix(engineLibrarySuffix); } return NO_ERROR; } } return NO_ERROR; } template<> std::variant PolicySerializer::deserialize( const xmlNode *root, SurroundSoundTraits::PtrSerializingCtx config) { config->setDefaultSurroundFormats(); for (const xmlNode *cur = root->xmlChildrenNode; cur != NULL; cur = cur->next) { if (!xmlStrcmp(cur->name, reinterpret_cast(SurroundSoundTraits::tag))) { AudioPolicyConfig::SurroundFormats formats; status_t status = deserializeCollection( cur, &formats, nullptr); if (status == NO_ERROR) { config->setSurroundFormats(formats); } return NO_ERROR; } } return NO_ERROR; } template<> std::variant PolicySerializer::deserialize( const xmlNode *cur, SurroundSoundFormatTraits::PtrSerializingCtx /*serializingContext*/) { using Attributes = SurroundSoundFormatTraits::Attributes; std::string formatLiteral = getXmlAttribute(cur, Attributes::name); if (formatLiteral.empty()) { ALOGE("%s: No %s found for a surround format", __func__, Attributes::name); return BAD_VALUE; } if (mIgnoreVendorExtensions && maybeVendorExtension(formatLiteral)) { ALOGI("%s: vendor extension format \"%s\" skipped", __func__, formatLiteral.c_str()); return NO_INIT; } audio_format_t format = formatFromString(formatLiteral); if (format == AUDIO_FORMAT_DEFAULT) { ALOGE("%s: Unrecognized format %s", __func__, formatLiteral.c_str()); return BAD_VALUE; } SurroundSoundFormatTraits::Element pair = std::make_pair( format, SurroundSoundFormatTraits::Collection::mapped_type{}); std::string subformatsLiteral = getXmlAttribute(cur, Attributes::subformats); if (subformatsLiteral.empty()) return pair; FormatVector subformats = formatsFromString(subformatsLiteral, " "); for (const auto& subformat : subformats) { auto result = pair.second.insert(subformat); if (!result.second) { ALOGE("%s: could not add subformat %x to collection", __func__, subformat); return BAD_VALUE; } } return pair; } status_t PolicySerializer::deserialize(const char *configFile, AudioPolicyConfig *config, bool ignoreVendorExtensions) { mIgnoreVendorExtensions = ignoreVendorExtensions; auto doc = make_xmlUnique(xmlParseFile(configFile)); if (doc == nullptr) { ALOGE("%s: Could not parse %s document.", __func__, configFile); return BAD_VALUE; } xmlNodePtr root = xmlDocGetRootElement(doc.get()); if (root == NULL) { ALOGE("%s: Could not parse %s document: empty.", __func__, configFile); return BAD_VALUE; } if (xmlXIncludeProcess(doc.get()) < 0) { ALOGE("%s: libxml failed to resolve XIncludes on %s document.", __func__, configFile); } if (xmlStrcmp(root->name, reinterpret_cast(rootName))) { ALOGE("%s: No %s root element found in xml data %s.", __func__, rootName, reinterpret_cast(root->name)); return BAD_VALUE; } std::string version = getXmlAttribute(root, versionAttribute); if (version.empty()) { ALOGE("%s: No version found in root node %s", __func__, rootName); return BAD_VALUE; } if (version == "7.0" || version == "7.1") { mChannelMasksSeparator = mSamplingRatesSeparator = mFlagsSeparator = " "; } else if (version != "1.0") { ALOGE("%s: Version does not match; expected \"1.0\", \"7.0\", or \"7.1\" got \"%s\"", __func__, version.c_str()); return BAD_VALUE; } // Let's deserialize children // Modules ModuleTraits::Collection modules; status_t status = deserializeCollection(root, &modules, config); if (status != NO_ERROR) { return status; } config->setHwModules(modules); // Global Configuration deserialize(root, config); // Surround configuration deserialize(root, config); return android::OK; } } // namespace status_t deserializeAudioPolicyFile(const char *fileName, AudioPolicyConfig *config) { PolicySerializer serializer; status_t status = serializer.deserialize(fileName, config); return status; } status_t deserializeAudioPolicyFileForVts(const char *fileName, AudioPolicyConfig *config) { PolicySerializer serializer; status_t status = serializer.deserialize(fileName, config, true /*ignoreVendorExtensions*/); return status; } } // namespace android