IOProfile.cpp revision f129b03fa583d4cc26fd9c9171b8fb3b0ed8d4f4
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::IOProfile"
18//#define LOG_NDEBUG 0
19
20#include "IOProfile.h"
21#include "HwModule.h"
22#include "AudioGain.h"
23
24namespace android {
25
26IOProfile::IOProfile(const String8& name, audio_port_role_t role)
27    : AudioPort(name, AUDIO_PORT_TYPE_MIX, role)
28{
29}
30
31IOProfile::~IOProfile()
32{
33}
34
35// checks if the IO profile is compatible with specified parameters.
36// Sampling rate, format and channel mask must be specified in order to
37// get a valid a match
38bool IOProfile::isCompatibleProfile(audio_devices_t device,
39                                    String8 address,
40                                    uint32_t samplingRate,
41                                    uint32_t *updatedSamplingRate,
42                                    audio_format_t format,
43                                    audio_format_t *updatedFormat,
44                                    audio_channel_mask_t channelMask,
45                                    audio_channel_mask_t *updatedChannelMask,
46                                    uint32_t flags) const
47{
48    const bool isPlaybackThread = mType == AUDIO_PORT_TYPE_MIX && mRole == AUDIO_PORT_ROLE_SOURCE;
49    const bool isRecordThread = mType == AUDIO_PORT_TYPE_MIX && mRole == AUDIO_PORT_ROLE_SINK;
50    ALOG_ASSERT(isPlaybackThread != isRecordThread);
51
52
53    if (device != AUDIO_DEVICE_NONE) {
54        // just check types if multiple devices are selected
55        if (popcount(device & ~AUDIO_DEVICE_BIT_IN) > 1) {
56            if ((mSupportedDevices.types() & device) != device) {
57                return false;
58            }
59        } else if (mSupportedDevices.getDevice(device, address) == 0) {
60            return false;
61        }
62    }
63
64    if (samplingRate == 0) {
65         return false;
66    }
67    uint32_t myUpdatedSamplingRate = samplingRate;
68    if (isPlaybackThread && checkExactSamplingRate(samplingRate) != NO_ERROR) {
69         return false;
70    }
71    if (isRecordThread && checkCompatibleSamplingRate(samplingRate, &myUpdatedSamplingRate) !=
72            NO_ERROR) {
73         return false;
74    }
75
76    if (!audio_is_valid_format(format)) {
77        return false;
78    }
79    if (isPlaybackThread && checkExactFormat(format) != NO_ERROR) {
80        return false;
81    }
82    audio_format_t myUpdatedFormat = format;
83    if (isRecordThread && checkCompatibleFormat(format, &myUpdatedFormat) != NO_ERROR) {
84        return false;
85    }
86
87    if (isPlaybackThread && (!audio_is_output_channel(channelMask) ||
88            checkExactChannelMask(channelMask) != NO_ERROR)) {
89        return false;
90    }
91    audio_channel_mask_t myUpdatedChannelMask = channelMask;
92    if (isRecordThread && (!audio_is_input_channel(channelMask) ||
93            checkCompatibleChannelMask(channelMask, &myUpdatedChannelMask) != NO_ERROR)) {
94        return false;
95    }
96
97    if (isPlaybackThread && (mFlags & flags) != flags) {
98        return false;
99    }
100    // The only input flag that is allowed to be different is the fast flag.
101    // An existing fast stream is compatible with a normal track request.
102    // An existing normal stream is compatible with a fast track request,
103    // but the fast request will be denied by AudioFlinger and converted to normal track.
104    if (isRecordThread && ((mFlags ^ flags) &
105            ~AUDIO_INPUT_FLAG_FAST)) {
106        return false;
107    }
108
109    if (updatedSamplingRate != NULL) {
110        *updatedSamplingRate = myUpdatedSamplingRate;
111    }
112    if (updatedFormat != NULL) {
113        *updatedFormat = myUpdatedFormat;
114    }
115    if (updatedChannelMask != NULL) {
116        *updatedChannelMask = myUpdatedChannelMask;
117    }
118    return true;
119}
120
121void IOProfile::dump(int fd)
122{
123    const size_t SIZE = 256;
124    char buffer[SIZE];
125    String8 result;
126
127    AudioPort::dump(fd, 4);
128
129    snprintf(buffer, SIZE, "    - flags: 0x%04x\n", mFlags);
130    result.append(buffer);
131    snprintf(buffer, SIZE, "    - devices:\n");
132    result.append(buffer);
133    write(fd, result.string(), result.size());
134    for (size_t i = 0; i < mSupportedDevices.size(); i++) {
135        mSupportedDevices[i]->dump(fd, 6, i);
136    }
137}
138
139void IOProfile::log()
140{
141    const size_t SIZE = 256;
142    char buffer[SIZE];
143    String8 result;
144
145    ALOGV("    - sampling rates: ");
146    for (size_t i = 0; i < mSamplingRates.size(); i++) {
147        ALOGV("  %d", mSamplingRates[i]);
148    }
149
150    ALOGV("    - channel masks: ");
151    for (size_t i = 0; i < mChannelMasks.size(); i++) {
152        ALOGV("  0x%04x", mChannelMasks[i]);
153    }
154
155    ALOGV("    - formats: ");
156    for (size_t i = 0; i < mFormats.size(); i++) {
157        ALOGV("  0x%08x", mFormats[i]);
158    }
159
160    ALOGV("    - devices: 0x%04x\n", mSupportedDevices.types());
161    ALOGV("    - flags: 0x%04x\n", mFlags);
162}
163
164}; // namespace android
165