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::HwModule"
18 //#define LOG_NDEBUG 0
19 
20 #include "HwModule.h"
21 #include "IOProfile.h"
22 #include <policy.h>
23 #include <system/audio.h>
24 
25 namespace android {
26 
HwModule(const char * name,uint32_t halVersionMajor,uint32_t halVersionMinor)27 HwModule::HwModule(const char *name, uint32_t halVersionMajor, uint32_t halVersionMinor)
28     : mName(String8(name)),
29       mHandle(AUDIO_MODULE_HANDLE_NONE)
30 {
31     setHalVersion(halVersionMajor, halVersionMinor);
32 }
33 
~HwModule()34 HwModule::~HwModule()
35 {
36     for (size_t i = 0; i < mOutputProfiles.size(); i++) {
37         mOutputProfiles[i]->clearSupportedDevices();
38     }
39     for (size_t i = 0; i < mInputProfiles.size(); i++) {
40         mInputProfiles[i]->clearSupportedDevices();
41     }
42 }
43 
addOutputProfile(const std::string & name,const audio_config_t * config,audio_devices_t device,const String8 & address)44 status_t HwModule::addOutputProfile(const std::string& name, const audio_config_t *config,
45                                     audio_devices_t device, const String8& address)
46 {
47     sp<IOProfile> profile = new OutputProfile(name);
48 
49     profile->addAudioProfile(new AudioProfile(config->format, config->channel_mask,
50                                               config->sample_rate));
51 
52     sp<DeviceDescriptor> devDesc = new DeviceDescriptor(device, "" /*tagName*/, address.string());
53     addDynamicDevice(devDesc);
54     // Reciprocally attach the device to the module
55     devDesc->attach(this);
56     profile->addSupportedDevice(devDesc);
57 
58     return addOutputProfile(profile);
59 }
60 
addOutputProfile(const sp<IOProfile> & profile)61 status_t HwModule::addOutputProfile(const sp<IOProfile> &profile)
62 {
63     profile->attach(this);
64     mOutputProfiles.add(profile);
65     mPorts.add(profile);
66     return NO_ERROR;
67 }
68 
addInputProfile(const sp<IOProfile> & profile)69 status_t HwModule::addInputProfile(const sp<IOProfile> &profile)
70 {
71     profile->attach(this);
72     mInputProfiles.add(profile);
73     mPorts.add(profile);
74     return NO_ERROR;
75 }
76 
addProfile(const sp<IOProfile> & profile)77 status_t HwModule::addProfile(const sp<IOProfile> &profile)
78 {
79     switch (profile->getRole()) {
80     case AUDIO_PORT_ROLE_SOURCE:
81         return addOutputProfile(profile);
82     case AUDIO_PORT_ROLE_SINK:
83         return addInputProfile(profile);
84     case AUDIO_PORT_ROLE_NONE:
85         return BAD_VALUE;
86     }
87     return BAD_VALUE;
88 }
89 
setProfiles(const IOProfileCollection & profiles)90 void HwModule::setProfiles(const IOProfileCollection &profiles)
91 {
92     for (size_t i = 0; i < profiles.size(); i++) {
93         addProfile(profiles[i]);
94     }
95 }
96 
removeOutputProfile(const std::string & name)97 status_t HwModule::removeOutputProfile(const std::string& name)
98 {
99     for (size_t i = 0; i < mOutputProfiles.size(); i++) {
100         if (mOutputProfiles[i]->getName() == name) {
101             for (const auto &device : mOutputProfiles[i]->getSupportedDevices()) {
102                 removeDynamicDevice(device);
103             }
104             mOutputProfiles.removeAt(i);
105             break;
106         }
107     }
108 
109     return NO_ERROR;
110 }
111 
addInputProfile(const std::string & name,const audio_config_t * config,audio_devices_t device,const String8 & address)112 status_t HwModule::addInputProfile(const std::string& name, const audio_config_t *config,
113                                    audio_devices_t device, const String8& address)
114 {
115     sp<IOProfile> profile = new InputProfile(name);
116     profile->addAudioProfile(new AudioProfile(config->format, config->channel_mask,
117                                               config->sample_rate));
118 
119     sp<DeviceDescriptor> devDesc = new DeviceDescriptor(device, "" /*tagName*/, address.string());
120     addDynamicDevice(devDesc);
121     // Reciprocally attach the device to the module
122     devDesc->attach(this);
123     profile->addSupportedDevice(devDesc);
124 
125     ALOGV("addInputProfile() name %s rate %d mask 0x%08x",
126           name.c_str(), config->sample_rate, config->channel_mask);
127 
128     return addInputProfile(profile);
129 }
130 
removeInputProfile(const std::string & name)131 status_t HwModule::removeInputProfile(const std::string& name)
132 {
133     for (size_t i = 0; i < mInputProfiles.size(); i++) {
134         if (mInputProfiles[i]->getName() == name) {
135             for (const auto &device : mInputProfiles[i]->getSupportedDevices()) {
136                 removeDynamicDevice(device);
137             }
138             mInputProfiles.removeAt(i);
139             break;
140         }
141     }
142 
143     return NO_ERROR;
144 }
145 
setDeclaredDevices(const DeviceVector & devices)146 void HwModule::setDeclaredDevices(const DeviceVector &devices)
147 {
148     mDeclaredDevices = devices;
149     for (size_t i = 0; i < devices.size(); i++) {
150         mPorts.add(devices[i]);
151     }
152 }
153 
getRouteSinkDevice(const sp<AudioRoute> & route) const154 sp<DeviceDescriptor> HwModule::getRouteSinkDevice(const sp<AudioRoute> &route) const
155 {
156     sp<DeviceDescriptor> sinkDevice = 0;
157     if (route->getSink()->asAudioPort()->getType() == AUDIO_PORT_TYPE_DEVICE) {
158         sinkDevice = mDeclaredDevices.getDeviceFromTagName(route->getSink()->getTagName());
159     }
160     return sinkDevice;
161 }
162 
getRouteSourceDevices(const sp<AudioRoute> & route) const163 DeviceVector HwModule::getRouteSourceDevices(const sp<AudioRoute> &route) const
164 {
165     DeviceVector sourceDevices;
166     for (const auto& source : route->getSources()) {
167         if (source->asAudioPort()->getType() == AUDIO_PORT_TYPE_DEVICE) {
168             sourceDevices.add(mDeclaredDevices.getDeviceFromTagName(source->getTagName()));
169         }
170     }
171     return sourceDevices;
172 }
173 
setRoutes(const AudioRouteVector & routes)174 void HwModule::setRoutes(const AudioRouteVector &routes)
175 {
176     mRoutes = routes;
177     // Now updating the streams (aka IOProfile until now) supported devices
178     refreshSupportedDevices();
179 }
180 
refreshSupportedDevices()181 void HwModule::refreshSupportedDevices()
182 {
183     // Now updating the streams (aka IOProfile until now) supported devices
184     for (const auto& stream : mInputProfiles) {
185         DeviceVector sourceDevices;
186         for (const auto& route : stream->getRoutes()) {
187             sp<PolicyAudioPort> sink = route->getSink();
188             if (sink == 0 || stream != sink) {
189                 ALOGE("%s: Invalid route attached to input stream", __FUNCTION__);
190                 continue;
191             }
192             DeviceVector sourceDevicesForRoute = getRouteSourceDevices(route);
193             if (sourceDevicesForRoute.isEmpty()) {
194                 ALOGE("%s: invalid source devices for %s", __FUNCTION__, stream->getName().c_str());
195                 continue;
196             }
197             sourceDevices.add(sourceDevicesForRoute);
198         }
199         if (sourceDevices.isEmpty()) {
200             ALOGE("%s: invalid source devices for %s", __FUNCTION__, stream->getName().c_str());
201             continue;
202         }
203         stream->setSupportedDevices(sourceDevices);
204     }
205     for (const auto& stream : mOutputProfiles) {
206         DeviceVector sinkDevices;
207         for (const auto& route : stream->getRoutes()) {
208             sp<PolicyAudioPort> source = findByTagName(route->getSources(), stream->getTagName());
209             if (source == 0 || stream != source) {
210                 ALOGE("%s: Invalid route attached to output stream", __FUNCTION__);
211                 continue;
212             }
213             sp<DeviceDescriptor> sinkDevice = getRouteSinkDevice(route);
214             if (sinkDevice == 0) {
215                 ALOGE("%s: invalid sink device for %s", __FUNCTION__, stream->getName().c_str());
216                 continue;
217             }
218             sinkDevices.add(sinkDevice);
219         }
220         stream->setSupportedDevices(sinkDevices);
221     }
222 }
223 
setHandle(audio_module_handle_t handle)224 void HwModule::setHandle(audio_module_handle_t handle) {
225     ALOGW_IF(mHandle != AUDIO_MODULE_HANDLE_NONE,
226             "HwModule handle is changing from %d to %d", mHandle, handle);
227     mHandle = handle;
228 }
229 
supportsPatch(const sp<PolicyAudioPort> & srcPort,const sp<PolicyAudioPort> & dstPort) const230 bool HwModule::supportsPatch(const sp<PolicyAudioPort> &srcPort,
231                              const sp<PolicyAudioPort> &dstPort) const {
232     for (const auto &route : mRoutes) {
233         if (route->supportsPatch(srcPort, dstPort)) {
234             return true;
235         }
236     }
237     return false;
238 }
239 
dump(String8 * dst) const240 void HwModule::dump(String8 *dst) const
241 {
242     dst->appendFormat("  - name: %s\n", getName());
243     dst->appendFormat("  - handle: %d\n", mHandle);
244     dst->appendFormat("  - version: %u.%u\n", getHalVersionMajor(), getHalVersionMinor());
245     if (mOutputProfiles.size()) {
246         dst->append("  - outputs:\n");
247         for (size_t i = 0; i < mOutputProfiles.size(); i++) {
248             dst->appendFormat("    output %zu:\n", i);
249             mOutputProfiles[i]->dump(dst);
250         }
251     }
252     if (mInputProfiles.size()) {
253         dst->append("  - inputs:\n");
254         for (size_t i = 0; i < mInputProfiles.size(); i++) {
255             dst->appendFormat("    input %zu:\n", i);
256             mInputProfiles[i]->dump(dst);
257         }
258     }
259     mDeclaredDevices.dump(dst, String8("Declared"), 2, true);
260     mDynamicDevices.dump(dst, String8("Dynamic"),  2, true);
261     dumpAudioRouteVector(mRoutes, dst, 2);
262 }
263 
getModuleFromName(const char * name) const264 sp <HwModule> HwModuleCollection::getModuleFromName(const char *name) const
265 {
266     for (const auto& module : *this) {
267         if (strcmp(module->getName(), name) == 0) {
268             return module;
269         }
270     }
271     return nullptr;
272 }
273 
getModuleForDeviceType(audio_devices_t type,audio_format_t encodedFormat) const274 sp <HwModule> HwModuleCollection::getModuleForDeviceType(audio_devices_t type,
275                                                          audio_format_t encodedFormat) const
276 {
277     for (const auto& module : *this) {
278         const auto& profiles = audio_is_output_device(type) ?
279                 module->getOutputProfiles() : module->getInputProfiles();
280         for (const auto& profile : profiles) {
281             if (profile->supportsDeviceTypes({type})) {
282                 if (encodedFormat != AUDIO_FORMAT_DEFAULT) {
283                     DeviceVector declaredDevices = module->getDeclaredDevices();
284                     sp <DeviceDescriptor> deviceDesc =
285                             declaredDevices.getDevice(type, String8(), encodedFormat);
286                     if (deviceDesc) {
287                         return module;
288                     }
289                 } else {
290                     return module;
291                 }
292             }
293         }
294     }
295     return nullptr;
296 }
297 
getModuleForDevice(const sp<DeviceDescriptor> & device,audio_format_t encodedFormat) const298 sp<HwModule> HwModuleCollection::getModuleForDevice(const sp<DeviceDescriptor> &device,
299                                                      audio_format_t encodedFormat) const
300 {
301     return getModuleForDeviceType(device->type(), encodedFormat);
302 }
303 
getAvailableDevicesFromModuleName(const char * name,const DeviceVector & availableDevices) const304 DeviceVector HwModuleCollection::getAvailableDevicesFromModuleName(
305         const char *name, const DeviceVector &availableDevices) const
306 {
307     sp<HwModule> module = getModuleFromName(name);
308     if (module == nullptr) {
309         return DeviceVector();
310     }
311     return availableDevices.getDevicesFromHwModule(module->getHandle());
312 }
313 
getDeviceDescriptor(const audio_devices_t deviceType,const char * address,const char * name,const audio_format_t encodedFormat,bool allowToCreate,bool matchAddress) const314 sp<DeviceDescriptor> HwModuleCollection::getDeviceDescriptor(const audio_devices_t deviceType,
315                                                              const char *address,
316                                                              const char *name,
317                                                              const audio_format_t encodedFormat,
318                                                              bool allowToCreate,
319                                                              bool matchAddress) const
320 {
321     String8 devAddress = (address == nullptr || !matchAddress) ? String8("") : String8(address);
322     // handle legacy remote submix case where the address was not always specified
323     if (audio_is_remote_submix_device(deviceType) && (devAddress.length() == 0)) {
324         devAddress = String8("0");
325     }
326 
327     for (const auto& hwModule : *this) {
328         DeviceVector moduleDevices = hwModule->getAllDevices();
329         auto moduleDevice = moduleDevices.getDevice(deviceType, devAddress, encodedFormat);
330         if (moduleDevice) {
331             if (encodedFormat != AUDIO_FORMAT_DEFAULT) {
332                 moduleDevice->setEncodedFormat(encodedFormat);
333             }
334             if (allowToCreate) {
335                 moduleDevice->attach(hwModule);
336                 moduleDevice->setAddress(devAddress.string());
337                 moduleDevice->setName(name);
338             }
339             return moduleDevice;
340         }
341     }
342     if (!allowToCreate) {
343         ALOGV("%s: could not find HW module for device %s %04x address %s", __FUNCTION__,
344               name, deviceType, address);
345         return nullptr;
346     }
347     return createDevice(deviceType, address, name, encodedFormat);
348 }
349 
createDevice(const audio_devices_t type,const char * address,const char * name,const audio_format_t encodedFormat) const350 sp<DeviceDescriptor> HwModuleCollection::createDevice(const audio_devices_t type,
351                                                       const char *address,
352                                                       const char *name,
353                                                       const audio_format_t encodedFormat) const
354 {
355     sp<HwModule> hwModule = getModuleForDeviceType(type, encodedFormat);
356     if (hwModule == 0) {
357         ALOGE("%s: could not find HW module for device %04x address %s", __FUNCTION__, type,
358               address);
359         return nullptr;
360     }
361 
362     sp<DeviceDescriptor> device = new DeviceDescriptor(type, name, address);
363     device->setName(name);
364     device->setEncodedFormat(encodedFormat);
365 
366   // Add the device to the list of dynamic devices
367     hwModule->addDynamicDevice(device);
368     // Reciprocally attach the device to the module
369     device->attach(hwModule);
370     ALOGD("%s: adding dynamic device %s to module %s", __FUNCTION__,
371           device->toString().c_str(), hwModule->getName());
372 
373     const auto &profiles = (audio_is_output_device(type) ? hwModule->getOutputProfiles() :
374                                                              hwModule->getInputProfiles());
375     for (const auto &profile : profiles) {
376         // Add the device as supported to all profile supporting "weakly" or not the device
377         // according to its type
378         if (profile->supportsDevice(device, false /*matchAdress*/)) {
379 
380             // @todo quid of audio profile? import the profile from device of the same type?
381             const auto &isoTypeDeviceForProfile =
382                 profile->getSupportedDevices().getDevice(type, String8(), AUDIO_FORMAT_DEFAULT);
383             device->importAudioPortAndPickAudioProfile(isoTypeDeviceForProfile, true /* force */);
384 
385             ALOGV("%s: adding device %s to profile %s", __FUNCTION__,
386                   device->toString().c_str(), profile->getTagName().c_str());
387             profile->addSupportedDevice(device);
388         }
389     }
390     return device;
391 }
392 
cleanUpForDevice(const sp<DeviceDescriptor> & device)393 void HwModuleCollection::cleanUpForDevice(const sp<DeviceDescriptor> &device)
394 {
395     for (const auto& hwModule : *this) {
396         DeviceVector moduleDevices = hwModule->getAllDevices();
397         if (!moduleDevices.contains(device)) {
398             continue;
399         }
400 
401         // removal of remote submix devices associated with a dynamic policy is
402         // handled by removeOutputProfile() and removeInputProfile()
403         if (audio_is_remote_submix_device(device->type()) && device->address() != "0") {
404             continue;
405         }
406 
407         device->detach();
408         // Only remove from dynamic list, not from declared list!!!
409         if (!hwModule->getDynamicDevices().contains(device)) {
410             return;
411         }
412         hwModule->removeDynamicDevice(device);
413         ALOGV("%s: removed dynamic device %s from module %s", __FUNCTION__,
414               device->toString().c_str(), hwModule->getName());
415 
416         const IOProfileCollection &profiles = audio_is_output_device(device->type()) ?
417                     hwModule->getOutputProfiles() : hwModule->getInputProfiles();
418         for (const auto &profile : profiles) {
419             // For cleanup, strong match is required
420             if (profile->supportsDevice(device, true /*matchAdress*/)) {
421                 ALOGV("%s: removing device %s from profile %s", __FUNCTION__,
422                       device->toString().c_str(), profile->getTagName().c_str());
423                 profile->removeSupportedDevice(device);
424             }
425         }
426     }
427 }
428 
dump(String8 * dst) const429 void HwModuleCollection::dump(String8 *dst) const
430 {
431     dst->append("\nHW Modules dump:\n");
432     for (size_t i = 0; i < size(); i++) {
433         dst->appendFormat("- HW Module %zu:\n", i + 1);
434         itemAt(i)->dump(dst);
435     }
436 }
437 
438 
439 } //namespace android
440