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::ConfigParsingUtils"
18 //#define LOG_NDEBUG 0
19
20 #include "ConfigParsingUtils.h"
21 #include "AudioGain.h"
22 #include "IOProfile.h"
23 #include <system/audio.h>
24 #include <media/AudioParameter.h>
25 #include <media/TypeConverter.h>
26 #include <utils/Log.h>
27 #include <cutils/misc.h>
28
29 namespace android {
30
31 // --- audio_policy.conf file parsing
32
33 //static
loadAudioPortGain(cnode * root,AudioPort & audioPort,int index)34 void ConfigParsingUtils::loadAudioPortGain(cnode *root, AudioPort &audioPort, int index)
35 {
36 cnode *node = root->first_child;
37
38 sp<AudioGain> gain = new AudioGain(index, audioPort.useInputChannelMask());
39
40 while (node) {
41 if (strcmp(node->name, GAIN_MODE) == 0) {
42 gain->setMode(GainModeConverter::maskFromString(node->value));
43 } else if (strcmp(node->name, GAIN_CHANNELS) == 0) {
44 audio_channel_mask_t mask;
45 if (audioPort.useInputChannelMask()) {
46 if (InputChannelConverter::fromString(node->value, mask)) {
47 gain->setChannelMask(mask);
48 }
49 } else {
50 if (OutputChannelConverter::fromString(node->value, mask)) {
51 gain->setChannelMask(mask);
52 }
53 }
54 } else if (strcmp(node->name, GAIN_MIN_VALUE) == 0) {
55 gain->setMinValueInMb(atoi(node->value));
56 } else if (strcmp(node->name, GAIN_MAX_VALUE) == 0) {
57 gain->setMaxValueInMb(atoi(node->value));
58 } else if (strcmp(node->name, GAIN_DEFAULT_VALUE) == 0) {
59 gain->setDefaultValueInMb(atoi(node->value));
60 } else if (strcmp(node->name, GAIN_STEP_VALUE) == 0) {
61 gain->setStepValueInMb(atoi(node->value));
62 } else if (strcmp(node->name, GAIN_MIN_RAMP_MS) == 0) {
63 gain->setMinRampInMs(atoi(node->value));
64 } else if (strcmp(node->name, GAIN_MAX_RAMP_MS) == 0) {
65 gain->setMaxRampInMs(atoi(node->value));
66 }
67 node = node->next;
68 }
69
70 ALOGV("loadGain() adding new gain mode %08x channel mask %08x min mB %d max mB %d",
71 gain->getMode(), gain->getChannelMask(), gain->getMinValueInMb(),
72 gain->getMaxValueInMb());
73
74 if (gain->getMode() == 0) {
75 return;
76 }
77 audioPort.mGains.add(gain);
78 }
79
loadAudioPortGains(cnode * root,AudioPort & audioPort)80 void ConfigParsingUtils::loadAudioPortGains(cnode *root, AudioPort &audioPort)
81 {
82 cnode *node = root->first_child;
83 int index = 0;
84 while (node) {
85 ALOGV("loadGains() loading gain %s", node->name);
86 loadAudioPortGain(node, audioPort, index++);
87 node = node->next;
88 }
89 }
90
91 //static
loadDeviceDescriptorGains(cnode * root,sp<DeviceDescriptor> & deviceDesc)92 void ConfigParsingUtils::loadDeviceDescriptorGains(cnode *root, sp<DeviceDescriptor> &deviceDesc)
93 {
94 loadAudioPortGains(root, *deviceDesc);
95 if (deviceDesc->mGains.size() > 0) {
96 deviceDesc->mGains[0]->getDefaultConfig(&deviceDesc->mGain);
97 }
98 }
99
100 //static
loadHwModuleDevice(cnode * root,DeviceVector & devices)101 status_t ConfigParsingUtils::loadHwModuleDevice(cnode *root, DeviceVector &devices)
102 {
103 cnode *node = root->first_child;
104
105 audio_devices_t type = AUDIO_DEVICE_NONE;
106 while (node) {
107 if (strcmp(node->name, APM_DEVICE_TYPE) == 0) {
108 deviceFromString(node->value, type);
109 break;
110 }
111 node = node->next;
112 }
113 if (type == AUDIO_DEVICE_NONE ||
114 (!audio_is_input_device(type) && !audio_is_output_device(type))) {
115 ALOGW("loadDevice() bad type %08x", type);
116 return BAD_VALUE;
117 }
118 sp<DeviceDescriptor> deviceDesc = new DeviceDescriptor(type, String8(root->name));
119
120 node = root->first_child;
121 while (node) {
122 if (strcmp(node->name, APM_DEVICE_ADDRESS) == 0) {
123 deviceDesc->mAddress = String8((char *)node->value);
124 } else if (strcmp(node->name, CHANNELS_TAG) == 0) {
125 if (audio_is_input_device(type)) {
126 deviceDesc->addAudioProfile(
127 new AudioProfile(gDynamicFormat,
128 inputChannelMasksFromString(node->value),
129 SampleRateVector()));
130 } else {
131 deviceDesc->addAudioProfile(
132 new AudioProfile(gDynamicFormat,
133 outputChannelMasksFromString(node->value),
134 SampleRateVector()));
135 }
136 } else if (strcmp(node->name, GAINS_TAG) == 0) {
137 loadDeviceDescriptorGains(node, deviceDesc);
138 }
139 node = node->next;
140 }
141
142 ALOGV("loadDevice() adding device tag (literal type) %s type %08x address %s",
143 deviceDesc->getTagName().string(), type, deviceDesc->mAddress.string());
144
145 devices.add(deviceDesc);
146 return NO_ERROR;
147 }
148
149 //static
loadHwModuleProfile(cnode * root,sp<HwModule> & module,audio_port_role_t role)150 status_t ConfigParsingUtils::loadHwModuleProfile(cnode *root, sp<HwModule> &module,
151 audio_port_role_t role)
152 {
153 cnode *node = root->first_child;
154
155 sp<IOProfile> profile = new IOProfile(String8(root->name), role);
156
157 AudioProfileVector audioProfiles;
158 SampleRateVector sampleRates;
159 ChannelsVector channels;
160 FormatVector formats;
161
162 while (node) {
163 if (strcmp(node->name, FORMATS_TAG) == 0 &&
164 strcmp(node->value, DYNAMIC_VALUE_TAG) != 0) {
165 formats = formatsFromString(node->value);
166 } else if (strcmp(node->name, SAMPLING_RATES_TAG) == 0 &&
167 strcmp(node->value, DYNAMIC_VALUE_TAG) != 0) {
168 collectionFromString<SampleRateTraits>(node->value, sampleRates);
169 } else if (strcmp(node->name, CHANNELS_TAG) == 0 &&
170 strcmp(node->value, DYNAMIC_VALUE_TAG) != 0) {
171 if (role == AUDIO_PORT_ROLE_SINK) {
172 channels = inputChannelMasksFromString(node->value);
173 } else {
174 channels = outputChannelMasksFromString(node->value);
175 }
176 } else if (strcmp(node->name, DEVICES_TAG) == 0) {
177 DeviceVector devices;
178 loadDevicesFromTag(node->value, devices, module->getDeclaredDevices());
179 profile->setSupportedDevices(devices);
180 } else if (strcmp(node->name, FLAGS_TAG) == 0) {
181 if (role == AUDIO_PORT_ROLE_SINK) {
182 profile->setFlags(InputFlagConverter::maskFromString(node->value));
183 } else {
184 profile->setFlags(OutputFlagConverter::maskFromString(node->value));
185 }
186 } else if (strcmp(node->name, GAINS_TAG) == 0) {
187 loadAudioPortGains(node, *profile);
188 }
189 node = node->next;
190 }
191 if (formats.isEmpty()) {
192 sp<AudioProfile> profileToAdd = new AudioProfile(gDynamicFormat, channels, sampleRates);
193 profileToAdd->setDynamicFormat(true);
194 profileToAdd->setDynamicChannels(channels.isEmpty());
195 profileToAdd->setDynamicRate(sampleRates.isEmpty());
196 audioProfiles.add(profileToAdd);
197 } else {
198 for (size_t i = 0; i < formats.size(); i++) {
199 // For compatibility reason, for each format, creates a profile with the same
200 // collection of rate and channels.
201 sp<AudioProfile> profileToAdd = new AudioProfile(formats[i], channels, sampleRates);
202 profileToAdd->setDynamicFormat(formats[i] == gDynamicFormat);
203 profileToAdd->setDynamicChannels(channels.isEmpty());
204 profileToAdd->setDynamicRate(sampleRates.isEmpty());
205 audioProfiles.add(profileToAdd);
206 }
207 }
208 profile->setAudioProfiles(audioProfiles);
209 ALOGW_IF(!profile->hasSupportedDevices(), "load%s() invalid supported devices",
210 role == AUDIO_PORT_ROLE_SINK ? "Input" : "Output");
211 if (profile->hasSupportedDevices()) {
212 ALOGV("load%s() adding Supported Devices %04x, mFlags %04x",
213 role == AUDIO_PORT_ROLE_SINK ? "Input" : "Output",
214 profile->getSupportedDevicesType(), profile->getFlags());
215 return module->addProfile(profile);
216 }
217 return BAD_VALUE;
218 }
219
220 //static
loadHwModule(cnode * root,sp<HwModule> & module,AudioPolicyConfig & config)221 status_t ConfigParsingUtils::loadHwModule(cnode *root, sp<HwModule> &module,
222 AudioPolicyConfig &config)
223 {
224 status_t status = NAME_NOT_FOUND;
225 cnode *node = config_find(root, DEVICES_TAG);
226 if (node != NULL) {
227 node = node->first_child;
228 DeviceVector devices;
229 while (node) {
230 ALOGV("loadHwModule() loading device %s", node->name);
231 status_t tmpStatus = loadHwModuleDevice(node, devices);
232 if (status == NAME_NOT_FOUND || status == NO_ERROR) {
233 status = tmpStatus;
234 }
235 node = node->next;
236 }
237 module->setDeclaredDevices(devices);
238 }
239 node = config_find(root, OUTPUTS_TAG);
240 if (node != NULL) {
241 node = node->first_child;
242 while (node) {
243 ALOGV("loadHwModule() loading output %s", node->name);
244 status_t tmpStatus = loadHwModuleProfile(node, module, AUDIO_PORT_ROLE_SOURCE);
245 if (status == NAME_NOT_FOUND || status == NO_ERROR) {
246 status = tmpStatus;
247 }
248 node = node->next;
249 }
250 }
251 node = config_find(root, INPUTS_TAG);
252 if (node != NULL) {
253 node = node->first_child;
254 while (node) {
255 ALOGV("loadHwModule() loading input %s", node->name);
256 status_t tmpStatus = loadHwModuleProfile(node, module, AUDIO_PORT_ROLE_SINK);
257 if (status == NAME_NOT_FOUND || status == NO_ERROR) {
258 status = tmpStatus;
259 }
260 node = node->next;
261 }
262 }
263 loadModuleGlobalConfig(root, module, config);
264 return status;
265 }
266
267 //static
loadHwModules(cnode * root,HwModuleCollection & hwModules,AudioPolicyConfig & config)268 void ConfigParsingUtils::loadHwModules(cnode *root, HwModuleCollection &hwModules,
269 AudioPolicyConfig &config)
270 {
271 cnode *node = config_find(root, AUDIO_HW_MODULE_TAG);
272 if (node == NULL) {
273 return;
274 }
275
276 node = node->first_child;
277 while (node) {
278 ALOGV("loadHwModules() loading module %s", node->name);
279 sp<HwModule> module = new HwModule(node->name);
280 if (loadHwModule(node, module, config) == NO_ERROR) {
281 hwModules.add(module);
282 }
283 node = node->next;
284 }
285 }
286
287 //static
loadDevicesFromTag(const char * tag,DeviceVector & devices,const DeviceVector & declaredDevices)288 void ConfigParsingUtils::loadDevicesFromTag(const char *tag, DeviceVector &devices,
289 const DeviceVector &declaredDevices)
290 {
291 char *tagLiteral = strndup(tag, strlen(tag));
292 char *devTag = strtok(tagLiteral, AudioParameter::valueListSeparator);
293 while (devTag != NULL) {
294 if (strlen(devTag) != 0) {
295 audio_devices_t type;
296 if (deviceFromString(devTag, type)) {
297 uint32_t inBit = type & AUDIO_DEVICE_BIT_IN;
298 type &= ~AUDIO_DEVICE_BIT_IN;
299 while (type) {
300 audio_devices_t singleType =
301 inBit | (1 << (31 - __builtin_clz(type)));
302 type &= ~singleType;
303 sp<DeviceDescriptor> dev = new DeviceDescriptor(singleType);
304 devices.add(dev);
305 }
306 } else {
307 sp<DeviceDescriptor> deviceDesc =
308 declaredDevices.getDeviceFromTagName(String8(devTag));
309 if (deviceDesc != 0) {
310 devices.add(deviceDesc);
311 }
312 }
313 }
314 devTag = strtok(NULL, AudioParameter::valueListSeparator);
315 }
316 free(tagLiteral);
317 }
318
319 //static
loadModuleGlobalConfig(cnode * root,const sp<HwModule> & module,AudioPolicyConfig & config)320 void ConfigParsingUtils::loadModuleGlobalConfig(cnode *root, const sp<HwModule> &module,
321 AudioPolicyConfig &config)
322 {
323 cnode *node = config_find(root, GLOBAL_CONFIG_TAG);
324
325 if (node == NULL) {
326 return;
327 }
328 DeviceVector declaredDevices;
329 if (module != NULL) {
330 declaredDevices = module->getDeclaredDevices();
331 }
332
333 node = node->first_child;
334 while (node) {
335 if (strcmp(ATTACHED_OUTPUT_DEVICES_TAG, node->name) == 0) {
336 DeviceVector availableOutputDevices;
337 loadDevicesFromTag(node->value, availableOutputDevices, declaredDevices);
338 ALOGV("loadGlobalConfig() Attached Output Devices %08x",
339 availableOutputDevices.types());
340 config.addAvailableOutputDevices(availableOutputDevices);
341 } else if (strcmp(DEFAULT_OUTPUT_DEVICE_TAG, node->name) == 0) {
342 audio_devices_t device = AUDIO_DEVICE_NONE;
343 deviceFromString(node->value, device);
344 if (device != AUDIO_DEVICE_NONE) {
345 sp<DeviceDescriptor> defaultOutputDevice = new DeviceDescriptor(device);
346 config.setDefaultOutputDevice(defaultOutputDevice);
347 ALOGV("loadGlobalConfig() mDefaultOutputDevice %08x", defaultOutputDevice->type());
348 } else {
349 ALOGW("loadGlobalConfig() default device not specified");
350 }
351 } else if (strcmp(ATTACHED_INPUT_DEVICES_TAG, node->name) == 0) {
352 DeviceVector availableInputDevices;
353 loadDevicesFromTag(node->value, availableInputDevices, declaredDevices);
354 ALOGV("loadGlobalConfig() Available InputDevices %08x", availableInputDevices.types());
355 config.addAvailableInputDevices(availableInputDevices);
356 } else if (strcmp(AUDIO_HAL_VERSION_TAG, node->name) == 0) {
357 uint32_t major, minor;
358 sscanf((char *)node->value, "%u.%u", &major, &minor);
359 module->setHalVersion(major, minor);
360 ALOGV("loadGlobalConfig() mHalVersion = major %u minor %u", major, minor);
361 }
362 node = node->next;
363 }
364 }
365
366 //static
loadGlobalConfig(cnode * root,AudioPolicyConfig & config,const sp<HwModule> & primaryModule)367 void ConfigParsingUtils::loadGlobalConfig(cnode *root, AudioPolicyConfig &config,
368 const sp<HwModule>& primaryModule)
369 {
370 cnode *node = config_find(root, GLOBAL_CONFIG_TAG);
371
372 if (node == NULL) {
373 return;
374 }
375 node = node->first_child;
376 while (node) {
377 if (strcmp(SPEAKER_DRC_ENABLED_TAG, node->name) == 0) {
378 bool speakerDrcEnabled;
379 if (utilities::convertTo<std::string, bool>(node->value, speakerDrcEnabled)) {
380 ALOGV("loadGlobalConfig() mSpeakerDrcEnabled = %d", speakerDrcEnabled);
381 config.setSpeakerDrcEnabled(speakerDrcEnabled);
382 }
383 }
384 node = node->next;
385 }
386 loadModuleGlobalConfig(root, primaryModule, config);
387 }
388
389 //static
loadConfig(const char * path,AudioPolicyConfig & config)390 status_t ConfigParsingUtils::loadConfig(const char *path, AudioPolicyConfig &config)
391 {
392 cnode *root;
393 char *data;
394
395 data = (char *)load_file(path, NULL);
396 if (data == NULL) {
397 return -ENODEV;
398 }
399 root = config_node("", "");
400 config_load(root, data);
401
402 HwModuleCollection hwModules;
403 loadHwModules(root, hwModules, config);
404
405 // legacy audio_policy.conf files have one global_configuration section, attached to primary.
406 loadGlobalConfig(root, config, hwModules.getModuleFromName(AUDIO_HARDWARE_MODULE_ID_PRIMARY));
407
408 config.setHwModules(hwModules);
409
410 config_free(root);
411 free(root);
412 free(data);
413
414 ALOGI("loadAudioPolicyConfig() loaded %s\n", path);
415
416 return NO_ERROR;
417 }
418
419 } // namespace android
420