1*b9df5ad1SAndroid Build Coastguard Worker /*
2*b9df5ad1SAndroid Build Coastguard Worker * Copyright (C) 2014 The Android Open Source Project
3*b9df5ad1SAndroid Build Coastguard Worker *
4*b9df5ad1SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*b9df5ad1SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*b9df5ad1SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*b9df5ad1SAndroid Build Coastguard Worker *
8*b9df5ad1SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*b9df5ad1SAndroid Build Coastguard Worker *
10*b9df5ad1SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*b9df5ad1SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*b9df5ad1SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*b9df5ad1SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*b9df5ad1SAndroid Build Coastguard Worker * limitations under the License.
15*b9df5ad1SAndroid Build Coastguard Worker */
16*b9df5ad1SAndroid Build Coastguard Worker
17*b9df5ad1SAndroid Build Coastguard Worker #define LOG_TAG "alsa_device_profile"
18*b9df5ad1SAndroid Build Coastguard Worker /*#define LOG_NDEBUG 0*/
19*b9df5ad1SAndroid Build Coastguard Worker /*#define LOG_PCM_PARAMS 0*/
20*b9df5ad1SAndroid Build Coastguard Worker
21*b9df5ad1SAndroid Build Coastguard Worker #include <errno.h>
22*b9df5ad1SAndroid Build Coastguard Worker #include <inttypes.h>
23*b9df5ad1SAndroid Build Coastguard Worker #include <stdint.h>
24*b9df5ad1SAndroid Build Coastguard Worker #include <stdlib.h>
25*b9df5ad1SAndroid Build Coastguard Worker #include <cutils/properties.h>
26*b9df5ad1SAndroid Build Coastguard Worker
27*b9df5ad1SAndroid Build Coastguard Worker #include <log/log.h>
28*b9df5ad1SAndroid Build Coastguard Worker
29*b9df5ad1SAndroid Build Coastguard Worker #include "include/alsa_device_profile.h"
30*b9df5ad1SAndroid Build Coastguard Worker #include "include/alsa_format.h"
31*b9df5ad1SAndroid Build Coastguard Worker #include "include/alsa_logging.h"
32*b9df5ad1SAndroid Build Coastguard Worker
33*b9df5ad1SAndroid Build Coastguard Worker #define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
34*b9df5ad1SAndroid Build Coastguard Worker
35*b9df5ad1SAndroid Build Coastguard Worker #define PERIOD_DURATION_US (5 * 1000)
36*b9df5ad1SAndroid Build Coastguard Worker
37*b9df5ad1SAndroid Build Coastguard Worker #define DEFAULT_PERIOD_SIZE 1024
38*b9df5ad1SAndroid Build Coastguard Worker
39*b9df5ad1SAndroid Build Coastguard Worker static const char * const format_string_map[] = {
40*b9df5ad1SAndroid Build Coastguard Worker "AUDIO_FORMAT_PCM_16_BIT", /* "PCM_FORMAT_S16_LE", */
41*b9df5ad1SAndroid Build Coastguard Worker "AUDIO_FORMAT_PCM_32_BIT", /* "PCM_FORMAT_S32_LE", */
42*b9df5ad1SAndroid Build Coastguard Worker "AUDIO_FORMAT_PCM_8_BIT", /* "PCM_FORMAT_S8", */
43*b9df5ad1SAndroid Build Coastguard Worker "AUDIO_FORMAT_PCM_8_24_BIT", /* "PCM_FORMAT_S24_LE", */
44*b9df5ad1SAndroid Build Coastguard Worker "AUDIO_FORMAT_PCM_24_BIT_PACKED"/* "PCM_FORMAT_S24_3LE" */
45*b9df5ad1SAndroid Build Coastguard Worker };
46*b9df5ad1SAndroid Build Coastguard Worker
47*b9df5ad1SAndroid Build Coastguard Worker extern int8_t const pcm_format_value_map[50];
48*b9df5ad1SAndroid Build Coastguard Worker
49*b9df5ad1SAndroid Build Coastguard Worker /* Sort these in terms of preference (best first).
50*b9df5ad1SAndroid Build Coastguard Worker 192 kHz is not first because it requires significant resources for possibly worse
51*b9df5ad1SAndroid Build Coastguard Worker quality and driver instability (depends on device).
52*b9df5ad1SAndroid Build Coastguard Worker The order here determines the default sample rate for the device.
53*b9df5ad1SAndroid Build Coastguard Worker AudioPolicyManager may not respect this ordering when picking sample rates.
54*b9df5ad1SAndroid Build Coastguard Worker Update MAX_PROFILE_SAMPLE_RATES after changing the array size.
55*b9df5ad1SAndroid Build Coastguard Worker
56*b9df5ad1SAndroid Build Coastguard Worker TODO: remove 32000, 22050, 12000, 11025? Each sample rate check
57*b9df5ad1SAndroid Build Coastguard Worker requires opening the device which may cause pops. */
58*b9df5ad1SAndroid Build Coastguard Worker static const unsigned std_sample_rates[] =
59*b9df5ad1SAndroid Build Coastguard Worker {96000, 88200, 192000, 176400, 48000, 44100, 32000, 24000, 22050, 16000, 12000, 11025, 8000};
60*b9df5ad1SAndroid Build Coastguard Worker
profile_reset(alsa_device_profile * profile)61*b9df5ad1SAndroid Build Coastguard Worker static void profile_reset(alsa_device_profile* profile)
62*b9df5ad1SAndroid Build Coastguard Worker {
63*b9df5ad1SAndroid Build Coastguard Worker profile->card = profile->device = -1;
64*b9df5ad1SAndroid Build Coastguard Worker profile->extra_latency_ms = 0;
65*b9df5ad1SAndroid Build Coastguard Worker
66*b9df5ad1SAndroid Build Coastguard Worker /* terminate the attribute arrays with invalid values */
67*b9df5ad1SAndroid Build Coastguard Worker profile->formats[0] = PCM_FORMAT_INVALID;
68*b9df5ad1SAndroid Build Coastguard Worker profile->sample_rates[0] = 0;
69*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[0] = 0;
70*b9df5ad1SAndroid Build Coastguard Worker
71*b9df5ad1SAndroid Build Coastguard Worker profile->min_period_size = profile->max_period_size = 0;
72*b9df5ad1SAndroid Build Coastguard Worker profile->min_channel_count = profile->max_channel_count = DEFAULT_CHANNEL_COUNT;
73*b9df5ad1SAndroid Build Coastguard Worker
74*b9df5ad1SAndroid Build Coastguard Worker profile->is_valid = false;
75*b9df5ad1SAndroid Build Coastguard Worker }
76*b9df5ad1SAndroid Build Coastguard Worker
profile_init(alsa_device_profile * profile,int direction)77*b9df5ad1SAndroid Build Coastguard Worker void profile_init(alsa_device_profile* profile, int direction)
78*b9df5ad1SAndroid Build Coastguard Worker {
79*b9df5ad1SAndroid Build Coastguard Worker profile->direction = direction;
80*b9df5ad1SAndroid Build Coastguard Worker profile_reset(profile);
81*b9df5ad1SAndroid Build Coastguard Worker }
82*b9df5ad1SAndroid Build Coastguard Worker
profile_is_initialized(const alsa_device_profile * profile)83*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_initialized(const alsa_device_profile* profile)
84*b9df5ad1SAndroid Build Coastguard Worker {
85*b9df5ad1SAndroid Build Coastguard Worker return profile->card >= 0 && profile->device >= 0;
86*b9df5ad1SAndroid Build Coastguard Worker }
87*b9df5ad1SAndroid Build Coastguard Worker
profile_is_valid(const alsa_device_profile * profile)88*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_valid(const alsa_device_profile* profile) {
89*b9df5ad1SAndroid Build Coastguard Worker return profile->is_valid;
90*b9df5ad1SAndroid Build Coastguard Worker }
91*b9df5ad1SAndroid Build Coastguard Worker
profile_is_cached_for(const alsa_device_profile * profile,int card,int device)92*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_cached_for(const alsa_device_profile* profile, int card, int device) {
93*b9df5ad1SAndroid Build Coastguard Worker return card == profile->card && device == profile->device;
94*b9df5ad1SAndroid Build Coastguard Worker }
95*b9df5ad1SAndroid Build Coastguard Worker
profile_decache(alsa_device_profile * profile)96*b9df5ad1SAndroid Build Coastguard Worker void profile_decache(alsa_device_profile* profile) {
97*b9df5ad1SAndroid Build Coastguard Worker profile_reset(profile);
98*b9df5ad1SAndroid Build Coastguard Worker }
99*b9df5ad1SAndroid Build Coastguard Worker
100*b9df5ad1SAndroid Build Coastguard Worker /*
101*b9df5ad1SAndroid Build Coastguard Worker * Returns the supplied value rounded up to the next even multiple of 16
102*b9df5ad1SAndroid Build Coastguard Worker */
round_to_16_mult(unsigned int size)103*b9df5ad1SAndroid Build Coastguard Worker static unsigned int round_to_16_mult(unsigned int size)
104*b9df5ad1SAndroid Build Coastguard Worker {
105*b9df5ad1SAndroid Build Coastguard Worker return (size + 15) & ~15; /* 0xFFFFFFF0; */
106*b9df5ad1SAndroid Build Coastguard Worker }
107*b9df5ad1SAndroid Build Coastguard Worker
108*b9df5ad1SAndroid Build Coastguard Worker /*
109*b9df5ad1SAndroid Build Coastguard Worker * Returns the system defined minimum period size based on the supplied sample rate.
110*b9df5ad1SAndroid Build Coastguard Worker */
profile_calc_min_period_size(const alsa_device_profile * profile,unsigned sample_rate)111*b9df5ad1SAndroid Build Coastguard Worker unsigned profile_calc_min_period_size(const alsa_device_profile* profile, unsigned sample_rate)
112*b9df5ad1SAndroid Build Coastguard Worker {
113*b9df5ad1SAndroid Build Coastguard Worker ALOGV("profile_calc_min_period_size(%p, rate:%d)", profile, sample_rate);
114*b9df5ad1SAndroid Build Coastguard Worker if (profile == NULL) {
115*b9df5ad1SAndroid Build Coastguard Worker return DEFAULT_PERIOD_SIZE;
116*b9df5ad1SAndroid Build Coastguard Worker } else {
117*b9df5ad1SAndroid Build Coastguard Worker unsigned period_us = property_get_int32("ro.audio.usb.period_us", PERIOD_DURATION_US);
118*b9df5ad1SAndroid Build Coastguard Worker unsigned num_sample_frames = ((uint64_t)sample_rate * period_us) / 1000000;
119*b9df5ad1SAndroid Build Coastguard Worker
120*b9df5ad1SAndroid Build Coastguard Worker if (num_sample_frames < profile->min_period_size) {
121*b9df5ad1SAndroid Build Coastguard Worker num_sample_frames = profile->min_period_size;
122*b9df5ad1SAndroid Build Coastguard Worker }
123*b9df5ad1SAndroid Build Coastguard Worker return round_to_16_mult(num_sample_frames);
124*b9df5ad1SAndroid Build Coastguard Worker }
125*b9df5ad1SAndroid Build Coastguard Worker }
126*b9df5ad1SAndroid Build Coastguard Worker
profile_get_period_size(const alsa_device_profile * profile,unsigned sample_rate)127*b9df5ad1SAndroid Build Coastguard Worker unsigned int profile_get_period_size(const alsa_device_profile* profile, unsigned sample_rate)
128*b9df5ad1SAndroid Build Coastguard Worker {
129*b9df5ad1SAndroid Build Coastguard Worker unsigned int period_size = profile_calc_min_period_size(profile, sample_rate);
130*b9df5ad1SAndroid Build Coastguard Worker ALOGV("profile_get_period_size(rate:%d) = %d", sample_rate, period_size);
131*b9df5ad1SAndroid Build Coastguard Worker return period_size;
132*b9df5ad1SAndroid Build Coastguard Worker }
133*b9df5ad1SAndroid Build Coastguard Worker
134*b9df5ad1SAndroid Build Coastguard Worker /*
135*b9df5ad1SAndroid Build Coastguard Worker * Sample Rate
136*b9df5ad1SAndroid Build Coastguard Worker */
profile_get_default_sample_rate(const alsa_device_profile * profile)137*b9df5ad1SAndroid Build Coastguard Worker unsigned profile_get_default_sample_rate(const alsa_device_profile* profile)
138*b9df5ad1SAndroid Build Coastguard Worker {
139*b9df5ad1SAndroid Build Coastguard Worker /*
140*b9df5ad1SAndroid Build Coastguard Worker * This is probably a poor algorithm. The default sample rate should be the highest (within
141*b9df5ad1SAndroid Build Coastguard Worker * limits) rate that is available for both input and output. HOWEVER, the profile has only
142*b9df5ad1SAndroid Build Coastguard Worker * one or the other, so that will need to be done at a higher level, like in the HAL.
143*b9df5ad1SAndroid Build Coastguard Worker */
144*b9df5ad1SAndroid Build Coastguard Worker /*
145*b9df5ad1SAndroid Build Coastguard Worker * TODO this won't be right in general. we should store a preferred rate as we are scanning.
146*b9df5ad1SAndroid Build Coastguard Worker * But right now it will return the highest rate, which may be correct.
147*b9df5ad1SAndroid Build Coastguard Worker */
148*b9df5ad1SAndroid Build Coastguard Worker return profile_is_valid(profile) ? profile->sample_rates[0] : DEFAULT_SAMPLE_RATE;
149*b9df5ad1SAndroid Build Coastguard Worker }
150*b9df5ad1SAndroid Build Coastguard Worker
profile_get_highest_sample_rate(const alsa_device_profile * profile)151*b9df5ad1SAndroid Build Coastguard Worker unsigned profile_get_highest_sample_rate(const alsa_device_profile* profile) {
152*b9df5ad1SAndroid Build Coastguard Worker /* The hightest sample rate is always stored in the first element of sample_rates.
153*b9df5ad1SAndroid Build Coastguard Worker * Note that profile_reset() initiaizes the first element of samples_rates to 0
154*b9df5ad1SAndroid Build Coastguard Worker * Which is what we want to return if the profile had not been read anyway.
155*b9df5ad1SAndroid Build Coastguard Worker */
156*b9df5ad1SAndroid Build Coastguard Worker return profile->sample_rates[0];
157*b9df5ad1SAndroid Build Coastguard Worker }
158*b9df5ad1SAndroid Build Coastguard Worker
profile_is_sample_rate_valid(const alsa_device_profile * profile,unsigned rate)159*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_sample_rate_valid(const alsa_device_profile* profile, unsigned rate)
160*b9df5ad1SAndroid Build Coastguard Worker {
161*b9df5ad1SAndroid Build Coastguard Worker if (profile_is_valid(profile)) {
162*b9df5ad1SAndroid Build Coastguard Worker size_t index;
163*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; profile->sample_rates[index] != 0; index++) {
164*b9df5ad1SAndroid Build Coastguard Worker if (profile->sample_rates[index] == rate) {
165*b9df5ad1SAndroid Build Coastguard Worker return true;
166*b9df5ad1SAndroid Build Coastguard Worker }
167*b9df5ad1SAndroid Build Coastguard Worker }
168*b9df5ad1SAndroid Build Coastguard Worker
169*b9df5ad1SAndroid Build Coastguard Worker return false;
170*b9df5ad1SAndroid Build Coastguard Worker } else {
171*b9df5ad1SAndroid Build Coastguard Worker ALOGW("**** PROFILE NOT VALID!");
172*b9df5ad1SAndroid Build Coastguard Worker return rate == DEFAULT_SAMPLE_RATE;
173*b9df5ad1SAndroid Build Coastguard Worker }
174*b9df5ad1SAndroid Build Coastguard Worker }
175*b9df5ad1SAndroid Build Coastguard Worker
176*b9df5ad1SAndroid Build Coastguard Worker /*
177*b9df5ad1SAndroid Build Coastguard Worker * Format
178*b9df5ad1SAndroid Build Coastguard Worker */
profile_get_default_format(const alsa_device_profile * profile)179*b9df5ad1SAndroid Build Coastguard Worker enum pcm_format profile_get_default_format(const alsa_device_profile* profile)
180*b9df5ad1SAndroid Build Coastguard Worker {
181*b9df5ad1SAndroid Build Coastguard Worker /*
182*b9df5ad1SAndroid Build Coastguard Worker * TODO this won't be right in general. we should store a preferred format as we are scanning.
183*b9df5ad1SAndroid Build Coastguard Worker */
184*b9df5ad1SAndroid Build Coastguard Worker return profile_is_valid(profile) ? profile->formats[0] : DEFAULT_SAMPLE_FORMAT;
185*b9df5ad1SAndroid Build Coastguard Worker }
186*b9df5ad1SAndroid Build Coastguard Worker
profile_is_format_valid(const alsa_device_profile * profile,enum pcm_format fmt)187*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_format_valid(const alsa_device_profile* profile, enum pcm_format fmt) {
188*b9df5ad1SAndroid Build Coastguard Worker if (profile_is_valid(profile)) {
189*b9df5ad1SAndroid Build Coastguard Worker size_t index;
190*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; profile->formats[index] != PCM_FORMAT_INVALID; index++) {
191*b9df5ad1SAndroid Build Coastguard Worker if (profile->formats[index] == fmt) {
192*b9df5ad1SAndroid Build Coastguard Worker return true;
193*b9df5ad1SAndroid Build Coastguard Worker }
194*b9df5ad1SAndroid Build Coastguard Worker }
195*b9df5ad1SAndroid Build Coastguard Worker
196*b9df5ad1SAndroid Build Coastguard Worker return false;
197*b9df5ad1SAndroid Build Coastguard Worker } else {
198*b9df5ad1SAndroid Build Coastguard Worker return fmt == DEFAULT_SAMPLE_FORMAT;
199*b9df5ad1SAndroid Build Coastguard Worker }
200*b9df5ad1SAndroid Build Coastguard Worker }
201*b9df5ad1SAndroid Build Coastguard Worker
202*b9df5ad1SAndroid Build Coastguard Worker /*
203*b9df5ad1SAndroid Build Coastguard Worker * Channels
204*b9df5ad1SAndroid Build Coastguard Worker */
profile_get_default_channel_count(const alsa_device_profile * profile)205*b9df5ad1SAndroid Build Coastguard Worker unsigned profile_get_default_channel_count(const alsa_device_profile* profile)
206*b9df5ad1SAndroid Build Coastguard Worker {
207*b9df5ad1SAndroid Build Coastguard Worker return profile_is_valid(profile) ? profile->channel_counts[0] : DEFAULT_CHANNEL_COUNT;
208*b9df5ad1SAndroid Build Coastguard Worker }
209*b9df5ad1SAndroid Build Coastguard Worker
profile_get_closest_channel_count(const alsa_device_profile * profile,unsigned count)210*b9df5ad1SAndroid Build Coastguard Worker unsigned profile_get_closest_channel_count(const alsa_device_profile* profile, unsigned count)
211*b9df5ad1SAndroid Build Coastguard Worker {
212*b9df5ad1SAndroid Build Coastguard Worker if (profile_is_valid(profile)) {
213*b9df5ad1SAndroid Build Coastguard Worker if (count < profile->min_channel_count) {
214*b9df5ad1SAndroid Build Coastguard Worker return profile->min_channel_count;
215*b9df5ad1SAndroid Build Coastguard Worker } else if (count > profile->max_channel_count) {
216*b9df5ad1SAndroid Build Coastguard Worker return profile->max_channel_count;
217*b9df5ad1SAndroid Build Coastguard Worker } else {
218*b9df5ad1SAndroid Build Coastguard Worker return count;
219*b9df5ad1SAndroid Build Coastguard Worker }
220*b9df5ad1SAndroid Build Coastguard Worker } else {
221*b9df5ad1SAndroid Build Coastguard Worker return 0;
222*b9df5ad1SAndroid Build Coastguard Worker }
223*b9df5ad1SAndroid Build Coastguard Worker }
224*b9df5ad1SAndroid Build Coastguard Worker
profile_is_channel_count_valid(const alsa_device_profile * profile,unsigned count)225*b9df5ad1SAndroid Build Coastguard Worker bool profile_is_channel_count_valid(const alsa_device_profile* profile, unsigned count)
226*b9df5ad1SAndroid Build Coastguard Worker {
227*b9df5ad1SAndroid Build Coastguard Worker if (profile_is_initialized(profile)) {
228*b9df5ad1SAndroid Build Coastguard Worker return count >= profile->min_channel_count && count <= profile->max_channel_count;
229*b9df5ad1SAndroid Build Coastguard Worker } else {
230*b9df5ad1SAndroid Build Coastguard Worker return count == DEFAULT_CHANNEL_COUNT;
231*b9df5ad1SAndroid Build Coastguard Worker }
232*b9df5ad1SAndroid Build Coastguard Worker }
233*b9df5ad1SAndroid Build Coastguard Worker
profile_test_sample_rate(const alsa_device_profile * profile,unsigned rate)234*b9df5ad1SAndroid Build Coastguard Worker static bool profile_test_sample_rate(const alsa_device_profile* profile, unsigned rate)
235*b9df5ad1SAndroid Build Coastguard Worker {
236*b9df5ad1SAndroid Build Coastguard Worker struct pcm_config config = profile->default_config;
237*b9df5ad1SAndroid Build Coastguard Worker config.rate = rate;
238*b9df5ad1SAndroid Build Coastguard Worker // This method tests whether a sample rate is supported by the USB device
239*b9df5ad1SAndroid Build Coastguard Worker // by attempting to open it.
240*b9df5ad1SAndroid Build Coastguard Worker //
241*b9df5ad1SAndroid Build Coastguard Worker // The profile default_config currently contains the minimum channel count.
242*b9df5ad1SAndroid Build Coastguard Worker // As some usb devices cannot sustain the sample rate across all its supported
243*b9df5ad1SAndroid Build Coastguard Worker // channel counts, we try the largest usable channel count. This is
244*b9df5ad1SAndroid Build Coastguard Worker // bounded by FCC_LIMIT.
245*b9df5ad1SAndroid Build Coastguard Worker //
246*b9df5ad1SAndroid Build Coastguard Worker // If config.channels > FCC_LIMIT then we still test it for sample rate compatibility.
247*b9df5ad1SAndroid Build Coastguard Worker // It is possible that the USB device does not support less than a certain number
248*b9df5ad1SAndroid Build Coastguard Worker // of channels, and that minimum number is > FCC_LIMIT. Then the default_config
249*b9df5ad1SAndroid Build Coastguard Worker // channels will be > FCC_LIMIT (and we still proceed with the test).
250*b9df5ad1SAndroid Build Coastguard Worker //
251*b9df5ad1SAndroid Build Coastguard Worker // For example, the FocusRite Scarlett 18i20 supports between 16 to 20 playback
252*b9df5ad1SAndroid Build Coastguard Worker // channels and between 14 to 18 capture channels.
253*b9df5ad1SAndroid Build Coastguard Worker // If FCC_LIMIT is 8, we still need to use and test 16 output channels for playback
254*b9df5ad1SAndroid Build Coastguard Worker // and 14 input channels for capture, as that will be the ALSA opening configuration.
255*b9df5ad1SAndroid Build Coastguard Worker // The Android USB audio HAL layer will automatically zero pad to accommodate the
256*b9df5ad1SAndroid Build Coastguard Worker // 16 playback or 14 capture channel configuration from the (up to FCC_LIMIT)
257*b9df5ad1SAndroid Build Coastguard Worker // channels delivered by AudioFlinger.
258*b9df5ad1SAndroid Build Coastguard Worker if (config.channels < FCC_LIMIT) {
259*b9df5ad1SAndroid Build Coastguard Worker config.channels = profile->max_channel_count;
260*b9df5ad1SAndroid Build Coastguard Worker if (config.channels > FCC_LIMIT) config.channels = FCC_LIMIT;
261*b9df5ad1SAndroid Build Coastguard Worker }
262*b9df5ad1SAndroid Build Coastguard Worker bool works = false; /* let's be pessimistic */
263*b9df5ad1SAndroid Build Coastguard Worker struct pcm * pcm = pcm_open(profile->card, profile->device,
264*b9df5ad1SAndroid Build Coastguard Worker profile->direction, &config);
265*b9df5ad1SAndroid Build Coastguard Worker
266*b9df5ad1SAndroid Build Coastguard Worker if (pcm != NULL) {
267*b9df5ad1SAndroid Build Coastguard Worker works = pcm_is_ready(pcm);
268*b9df5ad1SAndroid Build Coastguard Worker pcm_close(pcm);
269*b9df5ad1SAndroid Build Coastguard Worker }
270*b9df5ad1SAndroid Build Coastguard Worker
271*b9df5ad1SAndroid Build Coastguard Worker return works;
272*b9df5ad1SAndroid Build Coastguard Worker }
273*b9df5ad1SAndroid Build Coastguard Worker
profile_enum_sample_rates(alsa_device_profile * profile,unsigned min,unsigned max)274*b9df5ad1SAndroid Build Coastguard Worker static unsigned profile_enum_sample_rates(alsa_device_profile* profile, unsigned min, unsigned max)
275*b9df5ad1SAndroid Build Coastguard Worker {
276*b9df5ad1SAndroid Build Coastguard Worker unsigned num_entries = 0;
277*b9df5ad1SAndroid Build Coastguard Worker unsigned index;
278*b9df5ad1SAndroid Build Coastguard Worker
279*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; index < ARRAY_SIZE(std_sample_rates) &&
280*b9df5ad1SAndroid Build Coastguard Worker num_entries < ARRAY_SIZE(profile->sample_rates) - 1;
281*b9df5ad1SAndroid Build Coastguard Worker index++) {
282*b9df5ad1SAndroid Build Coastguard Worker if (std_sample_rates[index] >= min && std_sample_rates[index] <= max
283*b9df5ad1SAndroid Build Coastguard Worker && profile_test_sample_rate(profile, std_sample_rates[index])) {
284*b9df5ad1SAndroid Build Coastguard Worker profile->sample_rates[num_entries++] = std_sample_rates[index];
285*b9df5ad1SAndroid Build Coastguard Worker }
286*b9df5ad1SAndroid Build Coastguard Worker }
287*b9df5ad1SAndroid Build Coastguard Worker profile->sample_rates[num_entries] = 0; /* terminate */
288*b9df5ad1SAndroid Build Coastguard Worker return num_entries; /* return # of supported rates */
289*b9df5ad1SAndroid Build Coastguard Worker }
290*b9df5ad1SAndroid Build Coastguard Worker
profile_enum_sample_formats(alsa_device_profile * profile,const struct pcm_mask * mask)291*b9df5ad1SAndroid Build Coastguard Worker static unsigned profile_enum_sample_formats(alsa_device_profile* profile,
292*b9df5ad1SAndroid Build Coastguard Worker const struct pcm_mask * mask)
293*b9df5ad1SAndroid Build Coastguard Worker {
294*b9df5ad1SAndroid Build Coastguard Worker const int num_slots = ARRAY_SIZE(mask->bits);
295*b9df5ad1SAndroid Build Coastguard Worker const int bits_per_slot = sizeof(mask->bits[0]) * 8;
296*b9df5ad1SAndroid Build Coastguard Worker
297*b9df5ad1SAndroid Build Coastguard Worker const int table_size = ARRAY_SIZE(pcm_format_value_map);
298*b9df5ad1SAndroid Build Coastguard Worker
299*b9df5ad1SAndroid Build Coastguard Worker int slot_index, bit_index, table_index;
300*b9df5ad1SAndroid Build Coastguard Worker table_index = 0;
301*b9df5ad1SAndroid Build Coastguard Worker int num_written = 0;
302*b9df5ad1SAndroid Build Coastguard Worker for (slot_index = 0; slot_index < num_slots && table_index < table_size;
303*b9df5ad1SAndroid Build Coastguard Worker slot_index++) {
304*b9df5ad1SAndroid Build Coastguard Worker unsigned bit_mask = 1;
305*b9df5ad1SAndroid Build Coastguard Worker for (bit_index = 0;
306*b9df5ad1SAndroid Build Coastguard Worker bit_index < bits_per_slot && table_index < table_size;
307*b9df5ad1SAndroid Build Coastguard Worker bit_index++) {
308*b9df5ad1SAndroid Build Coastguard Worker if ((mask->bits[slot_index] & bit_mask) != 0) {
309*b9df5ad1SAndroid Build Coastguard Worker enum pcm_format format = pcm_format_value_map[table_index];
310*b9df5ad1SAndroid Build Coastguard Worker /* Never return invalid (unrecognized) or 8-bit */
311*b9df5ad1SAndroid Build Coastguard Worker if (format != PCM_FORMAT_INVALID && format != PCM_FORMAT_S8) {
312*b9df5ad1SAndroid Build Coastguard Worker profile->formats[num_written++] = format;
313*b9df5ad1SAndroid Build Coastguard Worker if (num_written == ARRAY_SIZE(profile->formats) - 1) {
314*b9df5ad1SAndroid Build Coastguard Worker /* leave at least one PCM_FORMAT_INVALID at the end */
315*b9df5ad1SAndroid Build Coastguard Worker goto end;
316*b9df5ad1SAndroid Build Coastguard Worker }
317*b9df5ad1SAndroid Build Coastguard Worker }
318*b9df5ad1SAndroid Build Coastguard Worker }
319*b9df5ad1SAndroid Build Coastguard Worker bit_mask <<= 1;
320*b9df5ad1SAndroid Build Coastguard Worker table_index++;
321*b9df5ad1SAndroid Build Coastguard Worker }
322*b9df5ad1SAndroid Build Coastguard Worker }
323*b9df5ad1SAndroid Build Coastguard Worker end:
324*b9df5ad1SAndroid Build Coastguard Worker profile->formats[num_written] = PCM_FORMAT_INVALID;
325*b9df5ad1SAndroid Build Coastguard Worker return num_written;
326*b9df5ad1SAndroid Build Coastguard Worker }
327*b9df5ad1SAndroid Build Coastguard Worker
profile_enum_channel_counts(alsa_device_profile * profile,unsigned min,unsigned max)328*b9df5ad1SAndroid Build Coastguard Worker static unsigned profile_enum_channel_counts(alsa_device_profile* profile, unsigned min,
329*b9df5ad1SAndroid Build Coastguard Worker unsigned max)
330*b9df5ad1SAndroid Build Coastguard Worker {
331*b9df5ad1SAndroid Build Coastguard Worker /* modify alsa_device_profile.h if you change the std_channel_counts[] array. */
332*b9df5ad1SAndroid Build Coastguard Worker // The order of this array controls the order for channel mask generation.
333*b9df5ad1SAndroid Build Coastguard Worker // In general, this is just counting from max to min not skipping anything,
334*b9df5ad1SAndroid Build Coastguard Worker // but need not be that way.
335*b9df5ad1SAndroid Build Coastguard Worker static const unsigned std_channel_counts[FCC_24] = {
336*b9df5ad1SAndroid Build Coastguard Worker 24, 23, 22, 21, 20, 19, 18, 17,
337*b9df5ad1SAndroid Build Coastguard Worker 16, 15, 14, 13, 12, 11, 10, 9,
338*b9df5ad1SAndroid Build Coastguard Worker 8, 7, 6, 5, 4, 3, 2, 1
339*b9df5ad1SAndroid Build Coastguard Worker };
340*b9df5ad1SAndroid Build Coastguard Worker
341*b9df5ad1SAndroid Build Coastguard Worker unsigned num_counts = 0;
342*b9df5ad1SAndroid Build Coastguard Worker unsigned index;
343*b9df5ad1SAndroid Build Coastguard Worker int max_allowed_index = -1; // index of maximum allowed channel count reported by device.
344*b9df5ad1SAndroid Build Coastguard Worker /* TODO write a profile_test_channel_count() */
345*b9df5ad1SAndroid Build Coastguard Worker /* Ensure there is at least one invalid channel count to terminate the channel counts array */
346*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; index < ARRAY_SIZE(std_channel_counts) &&
347*b9df5ad1SAndroid Build Coastguard Worker num_counts < ARRAY_SIZE(profile->channel_counts) - 1;
348*b9df5ad1SAndroid Build Coastguard Worker index++) {
349*b9df5ad1SAndroid Build Coastguard Worker const unsigned test_count = std_channel_counts[index];
350*b9df5ad1SAndroid Build Coastguard Worker /* TODO Do we want a channel counts test? */
351*b9df5ad1SAndroid Build Coastguard Worker if (test_count <= FCC_LIMIT) {
352*b9df5ad1SAndroid Build Coastguard Worker if (test_count >= min && test_count <= max /* &&
353*b9df5ad1SAndroid Build Coastguard Worker profile_test_channel_count(profile, channel_counts[index])*/) {
354*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[num_counts++] = test_count;
355*b9df5ad1SAndroid Build Coastguard Worker }
356*b9df5ad1SAndroid Build Coastguard Worker if (max_allowed_index < 0 ||
357*b9df5ad1SAndroid Build Coastguard Worker std_channel_counts[max_allowed_index] < test_count) {
358*b9df5ad1SAndroid Build Coastguard Worker max_allowed_index = index;
359*b9df5ad1SAndroid Build Coastguard Worker }
360*b9df5ad1SAndroid Build Coastguard Worker }
361*b9df5ad1SAndroid Build Coastguard Worker }
362*b9df5ad1SAndroid Build Coastguard Worker // if we have no match with the standard counts, we use the largest (preferred) std count.
363*b9df5ad1SAndroid Build Coastguard Worker // Note: the usb hal will adjust channel data properly to fit.
364*b9df5ad1SAndroid Build Coastguard Worker if (num_counts == 0 && max_allowed_index >= 0) {
365*b9df5ad1SAndroid Build Coastguard Worker ALOGW("usb device does not match std channel counts, setting to %d",
366*b9df5ad1SAndroid Build Coastguard Worker std_channel_counts[max_allowed_index]);
367*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[num_counts++] = std_channel_counts[max_allowed_index];
368*b9df5ad1SAndroid Build Coastguard Worker }
369*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[num_counts] = 0;
370*b9df5ad1SAndroid Build Coastguard Worker return num_counts; /* return # of supported counts */
371*b9df5ad1SAndroid Build Coastguard Worker }
372*b9df5ad1SAndroid Build Coastguard Worker
373*b9df5ad1SAndroid Build Coastguard Worker /*
374*b9df5ad1SAndroid Build Coastguard Worker * Reads and decodes configuration info from the specified ALSA card/device.
375*b9df5ad1SAndroid Build Coastguard Worker */
read_alsa_device_config(alsa_device_profile * profile,struct pcm_config * config)376*b9df5ad1SAndroid Build Coastguard Worker static int read_alsa_device_config(alsa_device_profile * profile, struct pcm_config * config)
377*b9df5ad1SAndroid Build Coastguard Worker {
378*b9df5ad1SAndroid Build Coastguard Worker ALOGV("usb:audio_hw - read_alsa_device_config(c:%d d:%d t:0x%X)",
379*b9df5ad1SAndroid Build Coastguard Worker profile->card, profile->device, profile->direction);
380*b9df5ad1SAndroid Build Coastguard Worker
381*b9df5ad1SAndroid Build Coastguard Worker if (profile->card < 0 || profile->device < 0) {
382*b9df5ad1SAndroid Build Coastguard Worker return -EINVAL;
383*b9df5ad1SAndroid Build Coastguard Worker }
384*b9df5ad1SAndroid Build Coastguard Worker
385*b9df5ad1SAndroid Build Coastguard Worker struct pcm_params * alsa_hw_params =
386*b9df5ad1SAndroid Build Coastguard Worker pcm_params_get(profile->card, profile->device, profile->direction);
387*b9df5ad1SAndroid Build Coastguard Worker if (alsa_hw_params == NULL) {
388*b9df5ad1SAndroid Build Coastguard Worker return -EINVAL;
389*b9df5ad1SAndroid Build Coastguard Worker }
390*b9df5ad1SAndroid Build Coastguard Worker
391*b9df5ad1SAndroid Build Coastguard Worker profile->min_period_size = pcm_params_get_min(alsa_hw_params, PCM_PARAM_PERIOD_SIZE);
392*b9df5ad1SAndroid Build Coastguard Worker profile->max_period_size = pcm_params_get_max(alsa_hw_params, PCM_PARAM_PERIOD_SIZE);
393*b9df5ad1SAndroid Build Coastguard Worker
394*b9df5ad1SAndroid Build Coastguard Worker profile->min_channel_count = pcm_params_get_min(alsa_hw_params, PCM_PARAM_CHANNELS);
395*b9df5ad1SAndroid Build Coastguard Worker profile->max_channel_count = pcm_params_get_max(alsa_hw_params, PCM_PARAM_CHANNELS);
396*b9df5ad1SAndroid Build Coastguard Worker
397*b9df5ad1SAndroid Build Coastguard Worker int ret = 0;
398*b9df5ad1SAndroid Build Coastguard Worker
399*b9df5ad1SAndroid Build Coastguard Worker /*
400*b9df5ad1SAndroid Build Coastguard Worker * This Logging will be useful when testing new USB devices.
401*b9df5ad1SAndroid Build Coastguard Worker */
402*b9df5ad1SAndroid Build Coastguard Worker #ifdef LOG_PCM_PARAMS
403*b9df5ad1SAndroid Build Coastguard Worker log_pcm_params(alsa_hw_params);
404*b9df5ad1SAndroid Build Coastguard Worker #endif
405*b9df5ad1SAndroid Build Coastguard Worker
406*b9df5ad1SAndroid Build Coastguard Worker config->channels = pcm_params_get_min(alsa_hw_params, PCM_PARAM_CHANNELS);
407*b9df5ad1SAndroid Build Coastguard Worker // For output devices, let's make sure we choose at least stereo
408*b9df5ad1SAndroid Build Coastguard Worker // (assuming the device supports it).
409*b9df5ad1SAndroid Build Coastguard Worker if (profile->direction == PCM_OUT &&
410*b9df5ad1SAndroid Build Coastguard Worker config->channels < 2 && pcm_params_get_max(alsa_hw_params, PCM_PARAM_CHANNELS) >= 2) {
411*b9df5ad1SAndroid Build Coastguard Worker config->channels = 2;
412*b9df5ad1SAndroid Build Coastguard Worker }
413*b9df5ad1SAndroid Build Coastguard Worker config->rate = pcm_params_get_min(alsa_hw_params, PCM_PARAM_RATE);
414*b9df5ad1SAndroid Build Coastguard Worker // Prefer 48K or 44.1K
415*b9df5ad1SAndroid Build Coastguard Worker if (config->rate < 48000 &&
416*b9df5ad1SAndroid Build Coastguard Worker pcm_params_get_max(alsa_hw_params, PCM_PARAM_RATE) >= 48000) {
417*b9df5ad1SAndroid Build Coastguard Worker config->rate = 48000;
418*b9df5ad1SAndroid Build Coastguard Worker } else if (config->rate < 44100 &&
419*b9df5ad1SAndroid Build Coastguard Worker pcm_params_get_max(alsa_hw_params, PCM_PARAM_RATE) >= 44100) {
420*b9df5ad1SAndroid Build Coastguard Worker config->rate = 44100;
421*b9df5ad1SAndroid Build Coastguard Worker }
422*b9df5ad1SAndroid Build Coastguard Worker config->period_size = profile_calc_min_period_size(profile, config->rate);
423*b9df5ad1SAndroid Build Coastguard Worker config->period_count = pcm_params_get_min(alsa_hw_params, PCM_PARAM_PERIODS);
424*b9df5ad1SAndroid Build Coastguard Worker config->format = get_pcm_format_for_mask(pcm_params_get_mask(alsa_hw_params, PCM_PARAM_FORMAT));
425*b9df5ad1SAndroid Build Coastguard Worker #ifdef LOG_PCM_PARAMS
426*b9df5ad1SAndroid Build Coastguard Worker log_pcm_config(config, "read_alsa_device_config");
427*b9df5ad1SAndroid Build Coastguard Worker #endif
428*b9df5ad1SAndroid Build Coastguard Worker if (config->format == PCM_FORMAT_INVALID) {
429*b9df5ad1SAndroid Build Coastguard Worker ret = -EINVAL;
430*b9df5ad1SAndroid Build Coastguard Worker }
431*b9df5ad1SAndroid Build Coastguard Worker
432*b9df5ad1SAndroid Build Coastguard Worker pcm_params_free(alsa_hw_params);
433*b9df5ad1SAndroid Build Coastguard Worker
434*b9df5ad1SAndroid Build Coastguard Worker return ret;
435*b9df5ad1SAndroid Build Coastguard Worker }
436*b9df5ad1SAndroid Build Coastguard Worker
profile_fill_builtin_device_info(alsa_device_profile * profile,struct pcm_config * config,unsigned buffer_frame_count)437*b9df5ad1SAndroid Build Coastguard Worker bool profile_fill_builtin_device_info(alsa_device_profile* profile, struct pcm_config* config,
438*b9df5ad1SAndroid Build Coastguard Worker unsigned buffer_frame_count) {
439*b9df5ad1SAndroid Build Coastguard Worker if (!profile_is_initialized(profile)) {
440*b9df5ad1SAndroid Build Coastguard Worker return false;
441*b9df5ad1SAndroid Build Coastguard Worker }
442*b9df5ad1SAndroid Build Coastguard Worker profile->extra_latency_ms = property_get_int32(
443*b9df5ad1SAndroid Build Coastguard Worker "ro.hardware.audio.tinyalsa.host_latency_ms", 0);
444*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.channels = config->channels;
445*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.rate = config->rate;
446*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.format = config->format;
447*b9df5ad1SAndroid Build Coastguard Worker int period_count = property_get_int32(
448*b9df5ad1SAndroid Build Coastguard Worker "ro.hardware.audio.tinyalsa.period_count", DEFAULT_PERIOD_COUNT);
449*b9df5ad1SAndroid Build Coastguard Worker if (period_count <= 0) period_count = DEFAULT_PERIOD_COUNT;
450*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.period_count = period_count;
451*b9df5ad1SAndroid Build Coastguard Worker int period_size_multiplier = property_get_int32(
452*b9df5ad1SAndroid Build Coastguard Worker "ro.hardware.audio.tinyalsa.period_size_multiplier", 1);
453*b9df5ad1SAndroid Build Coastguard Worker if (period_size_multiplier <= 0) period_size_multiplier = 1;
454*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.period_size =
455*b9df5ad1SAndroid Build Coastguard Worker period_size_multiplier * buffer_frame_count / period_count;
456*b9df5ad1SAndroid Build Coastguard Worker profile->min_period_size = profile->max_period_size = profile->default_config.period_size;
457*b9df5ad1SAndroid Build Coastguard Worker profile->formats[0] = config->format;
458*b9df5ad1SAndroid Build Coastguard Worker profile->formats[1] = PCM_FORMAT_INVALID;
459*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[0] = config->channels;
460*b9df5ad1SAndroid Build Coastguard Worker profile->channel_counts[1] = 0;
461*b9df5ad1SAndroid Build Coastguard Worker profile->min_channel_count = profile->max_channel_count = config->channels;
462*b9df5ad1SAndroid Build Coastguard Worker profile->sample_rates[0] = config->rate;
463*b9df5ad1SAndroid Build Coastguard Worker profile->sample_rates[1] = 0;
464*b9df5ad1SAndroid Build Coastguard Worker profile->is_valid = true;
465*b9df5ad1SAndroid Build Coastguard Worker return true;
466*b9df5ad1SAndroid Build Coastguard Worker }
467*b9df5ad1SAndroid Build Coastguard Worker
profile_read_device_info(alsa_device_profile * profile)468*b9df5ad1SAndroid Build Coastguard Worker bool profile_read_device_info(alsa_device_profile* profile)
469*b9df5ad1SAndroid Build Coastguard Worker {
470*b9df5ad1SAndroid Build Coastguard Worker if (!profile_is_initialized(profile)) {
471*b9df5ad1SAndroid Build Coastguard Worker return false;
472*b9df5ad1SAndroid Build Coastguard Worker }
473*b9df5ad1SAndroid Build Coastguard Worker
474*b9df5ad1SAndroid Build Coastguard Worker /* let's get some defaults */
475*b9df5ad1SAndroid Build Coastguard Worker read_alsa_device_config(profile, &profile->default_config);
476*b9df5ad1SAndroid Build Coastguard Worker ALOGV("default_config chans:%d rate:%d format:%d count:%d size:%d",
477*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.channels, profile->default_config.rate,
478*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.format, profile->default_config.period_count,
479*b9df5ad1SAndroid Build Coastguard Worker profile->default_config.period_size);
480*b9df5ad1SAndroid Build Coastguard Worker
481*b9df5ad1SAndroid Build Coastguard Worker struct pcm_params * alsa_hw_params = pcm_params_get(profile->card,
482*b9df5ad1SAndroid Build Coastguard Worker profile->device,
483*b9df5ad1SAndroid Build Coastguard Worker profile->direction);
484*b9df5ad1SAndroid Build Coastguard Worker if (alsa_hw_params == NULL) {
485*b9df5ad1SAndroid Build Coastguard Worker return false;
486*b9df5ad1SAndroid Build Coastguard Worker }
487*b9df5ad1SAndroid Build Coastguard Worker
488*b9df5ad1SAndroid Build Coastguard Worker /* Formats */
489*b9df5ad1SAndroid Build Coastguard Worker const struct pcm_mask * format_mask = pcm_params_get_mask(alsa_hw_params, PCM_PARAM_FORMAT);
490*b9df5ad1SAndroid Build Coastguard Worker profile_enum_sample_formats(profile, format_mask);
491*b9df5ad1SAndroid Build Coastguard Worker
492*b9df5ad1SAndroid Build Coastguard Worker /* Channels */
493*b9df5ad1SAndroid Build Coastguard Worker profile_enum_channel_counts(
494*b9df5ad1SAndroid Build Coastguard Worker profile, pcm_params_get_min(alsa_hw_params, PCM_PARAM_CHANNELS),
495*b9df5ad1SAndroid Build Coastguard Worker pcm_params_get_max(alsa_hw_params, PCM_PARAM_CHANNELS));
496*b9df5ad1SAndroid Build Coastguard Worker
497*b9df5ad1SAndroid Build Coastguard Worker /* Sample Rates */
498*b9df5ad1SAndroid Build Coastguard Worker profile_enum_sample_rates(
499*b9df5ad1SAndroid Build Coastguard Worker profile, pcm_params_get_min(alsa_hw_params, PCM_PARAM_RATE),
500*b9df5ad1SAndroid Build Coastguard Worker pcm_params_get_max(alsa_hw_params, PCM_PARAM_RATE));
501*b9df5ad1SAndroid Build Coastguard Worker
502*b9df5ad1SAndroid Build Coastguard Worker profile->is_valid = true;
503*b9df5ad1SAndroid Build Coastguard Worker
504*b9df5ad1SAndroid Build Coastguard Worker pcm_params_free(alsa_hw_params);
505*b9df5ad1SAndroid Build Coastguard Worker return true;
506*b9df5ad1SAndroid Build Coastguard Worker }
507*b9df5ad1SAndroid Build Coastguard Worker
profile_get_sample_rate_strs(const alsa_device_profile * profile)508*b9df5ad1SAndroid Build Coastguard Worker char * profile_get_sample_rate_strs(const alsa_device_profile* profile)
509*b9df5ad1SAndroid Build Coastguard Worker {
510*b9df5ad1SAndroid Build Coastguard Worker /* if we assume that rate strings are about 5 characters (48000 is 5), plus ~1 for a
511*b9df5ad1SAndroid Build Coastguard Worker * delimiter "|" this buffer has room for about 22 rate strings which seems like
512*b9df5ad1SAndroid Build Coastguard Worker * way too much, but it's a stack variable so only temporary.
513*b9df5ad1SAndroid Build Coastguard Worker */
514*b9df5ad1SAndroid Build Coastguard Worker char buffer[128];
515*b9df5ad1SAndroid Build Coastguard Worker buffer[0] = '\0';
516*b9df5ad1SAndroid Build Coastguard Worker size_t buffSize = ARRAY_SIZE(buffer);
517*b9df5ad1SAndroid Build Coastguard Worker size_t curStrLen = 0;
518*b9df5ad1SAndroid Build Coastguard Worker
519*b9df5ad1SAndroid Build Coastguard Worker char numBuffer[32];
520*b9df5ad1SAndroid Build Coastguard Worker
521*b9df5ad1SAndroid Build Coastguard Worker size_t numEntries = 0;
522*b9df5ad1SAndroid Build Coastguard Worker size_t index;
523*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; profile->sample_rates[index] != 0; index++) {
524*b9df5ad1SAndroid Build Coastguard Worker snprintf(numBuffer, sizeof(numBuffer), "%u", profile->sample_rates[index]);
525*b9df5ad1SAndroid Build Coastguard Worker // account for both the null, and potentially the bar.
526*b9df5ad1SAndroid Build Coastguard Worker if (buffSize - curStrLen < strlen(numBuffer) + (numEntries != 0 ? 2 : 1)) {
527*b9df5ad1SAndroid Build Coastguard Worker /* we don't have room for another, so bail at this point rather than
528*b9df5ad1SAndroid Build Coastguard Worker * return a malformed rate string
529*b9df5ad1SAndroid Build Coastguard Worker */
530*b9df5ad1SAndroid Build Coastguard Worker break;
531*b9df5ad1SAndroid Build Coastguard Worker }
532*b9df5ad1SAndroid Build Coastguard Worker if (numEntries++ != 0) {
533*b9df5ad1SAndroid Build Coastguard Worker strlcat(buffer, "|", buffSize);
534*b9df5ad1SAndroid Build Coastguard Worker }
535*b9df5ad1SAndroid Build Coastguard Worker curStrLen = strlcat(buffer, numBuffer, buffSize);
536*b9df5ad1SAndroid Build Coastguard Worker }
537*b9df5ad1SAndroid Build Coastguard Worker
538*b9df5ad1SAndroid Build Coastguard Worker return strdup(buffer);
539*b9df5ad1SAndroid Build Coastguard Worker }
540*b9df5ad1SAndroid Build Coastguard Worker
profile_get_format_strs(const alsa_device_profile * profile)541*b9df5ad1SAndroid Build Coastguard Worker char * profile_get_format_strs(const alsa_device_profile* profile)
542*b9df5ad1SAndroid Build Coastguard Worker {
543*b9df5ad1SAndroid Build Coastguard Worker /* if we assume that format strings are about 24 characters (AUDIO_FORMAT_PCM_16_BIT is 23),
544*b9df5ad1SAndroid Build Coastguard Worker * plus ~1 for a delimiter "|" this buffer has room for about 10 format strings which seems
545*b9df5ad1SAndroid Build Coastguard Worker * like way too much, but it's a stack variable so only temporary.
546*b9df5ad1SAndroid Build Coastguard Worker */
547*b9df5ad1SAndroid Build Coastguard Worker char buffer[256];
548*b9df5ad1SAndroid Build Coastguard Worker buffer[0] = '\0';
549*b9df5ad1SAndroid Build Coastguard Worker size_t buffSize = ARRAY_SIZE(buffer);
550*b9df5ad1SAndroid Build Coastguard Worker size_t curStrLen = 0;
551*b9df5ad1SAndroid Build Coastguard Worker
552*b9df5ad1SAndroid Build Coastguard Worker size_t numEntries = 0;
553*b9df5ad1SAndroid Build Coastguard Worker size_t index = 0;
554*b9df5ad1SAndroid Build Coastguard Worker for (index = 0; profile->formats[index] != PCM_FORMAT_INVALID; index++) {
555*b9df5ad1SAndroid Build Coastguard Worker // account for both the null, and potentially the bar.
556*b9df5ad1SAndroid Build Coastguard Worker if (buffSize - curStrLen < strlen(format_string_map[profile->formats[index]])
557*b9df5ad1SAndroid Build Coastguard Worker + (numEntries != 0 ? 2 : 1)) {
558*b9df5ad1SAndroid Build Coastguard Worker /* we don't have room for another, so bail at this point rather than
559*b9df5ad1SAndroid Build Coastguard Worker * return a malformed rate string
560*b9df5ad1SAndroid Build Coastguard Worker */
561*b9df5ad1SAndroid Build Coastguard Worker break;
562*b9df5ad1SAndroid Build Coastguard Worker }
563*b9df5ad1SAndroid Build Coastguard Worker if (numEntries++ != 0) {
564*b9df5ad1SAndroid Build Coastguard Worker strlcat(buffer, "|", buffSize);
565*b9df5ad1SAndroid Build Coastguard Worker }
566*b9df5ad1SAndroid Build Coastguard Worker curStrLen = strlcat(buffer, format_string_map[profile->formats[index]], buffSize);
567*b9df5ad1SAndroid Build Coastguard Worker }
568*b9df5ad1SAndroid Build Coastguard Worker
569*b9df5ad1SAndroid Build Coastguard Worker return strdup(buffer);
570*b9df5ad1SAndroid Build Coastguard Worker }
571*b9df5ad1SAndroid Build Coastguard Worker
profile_get_channel_count_strs(const alsa_device_profile * profile)572*b9df5ad1SAndroid Build Coastguard Worker char * profile_get_channel_count_strs(const alsa_device_profile* profile)
573*b9df5ad1SAndroid Build Coastguard Worker {
574*b9df5ad1SAndroid Build Coastguard Worker // we use only the canonical even number channel position masks.
575*b9df5ad1SAndroid Build Coastguard Worker static const char * const out_chans_strs[] = {
576*b9df5ad1SAndroid Build Coastguard Worker [0] = "AUDIO_CHANNEL_NONE", /* will never be taken as this is a terminator */
577*b9df5ad1SAndroid Build Coastguard Worker [1] = "AUDIO_CHANNEL_OUT_MONO",
578*b9df5ad1SAndroid Build Coastguard Worker [2] = "AUDIO_CHANNEL_OUT_STEREO",
579*b9df5ad1SAndroid Build Coastguard Worker [4] = "AUDIO_CHANNEL_OUT_QUAD",
580*b9df5ad1SAndroid Build Coastguard Worker [6] = "AUDIO_CHANNEL_OUT_5POINT1",
581*b9df5ad1SAndroid Build Coastguard Worker [FCC_8] = "AUDIO_CHANNEL_OUT_7POINT1",
582*b9df5ad1SAndroid Build Coastguard Worker [FCC_12] = "AUDIO_CHANNEL_OUT_7POINT1POINT4",
583*b9df5ad1SAndroid Build Coastguard Worker [FCC_24] = "AUDIO_CHANNEL_OUT_22POINT2",
584*b9df5ad1SAndroid Build Coastguard Worker };
585*b9df5ad1SAndroid Build Coastguard Worker
586*b9df5ad1SAndroid Build Coastguard Worker static const char * const in_chans_strs[] = {
587*b9df5ad1SAndroid Build Coastguard Worker [0] = "AUDIO_CHANNEL_NONE", /* will never be taken as this is a terminator */
588*b9df5ad1SAndroid Build Coastguard Worker [1] = "AUDIO_CHANNEL_IN_MONO",
589*b9df5ad1SAndroid Build Coastguard Worker [2] = "AUDIO_CHANNEL_IN_STEREO",
590*b9df5ad1SAndroid Build Coastguard Worker /* channel counts greater than this not considered */
591*b9df5ad1SAndroid Build Coastguard Worker };
592*b9df5ad1SAndroid Build Coastguard Worker
593*b9df5ad1SAndroid Build Coastguard Worker static const char * const index_chans_strs[] = {
594*b9df5ad1SAndroid Build Coastguard Worker [0] = "AUDIO_CHANNEL_NONE", /* will never be taken as this is a terminator */
595*b9df5ad1SAndroid Build Coastguard Worker
596*b9df5ad1SAndroid Build Coastguard Worker [1] = "AUDIO_CHANNEL_INDEX_MASK_1",
597*b9df5ad1SAndroid Build Coastguard Worker [2] = "AUDIO_CHANNEL_INDEX_MASK_2",
598*b9df5ad1SAndroid Build Coastguard Worker [3] = "AUDIO_CHANNEL_INDEX_MASK_3",
599*b9df5ad1SAndroid Build Coastguard Worker [4] = "AUDIO_CHANNEL_INDEX_MASK_4",
600*b9df5ad1SAndroid Build Coastguard Worker [5] = "AUDIO_CHANNEL_INDEX_MASK_5",
601*b9df5ad1SAndroid Build Coastguard Worker [6] = "AUDIO_CHANNEL_INDEX_MASK_6",
602*b9df5ad1SAndroid Build Coastguard Worker [7] = "AUDIO_CHANNEL_INDEX_MASK_7",
603*b9df5ad1SAndroid Build Coastguard Worker [8] = "AUDIO_CHANNEL_INDEX_MASK_8",
604*b9df5ad1SAndroid Build Coastguard Worker
605*b9df5ad1SAndroid Build Coastguard Worker [9] = "AUDIO_CHANNEL_INDEX_MASK_9",
606*b9df5ad1SAndroid Build Coastguard Worker [10] = "AUDIO_CHANNEL_INDEX_MASK_10",
607*b9df5ad1SAndroid Build Coastguard Worker [11] = "AUDIO_CHANNEL_INDEX_MASK_11",
608*b9df5ad1SAndroid Build Coastguard Worker [12] = "AUDIO_CHANNEL_INDEX_MASK_12",
609*b9df5ad1SAndroid Build Coastguard Worker [13] = "AUDIO_CHANNEL_INDEX_MASK_13",
610*b9df5ad1SAndroid Build Coastguard Worker [14] = "AUDIO_CHANNEL_INDEX_MASK_14",
611*b9df5ad1SAndroid Build Coastguard Worker [15] = "AUDIO_CHANNEL_INDEX_MASK_15",
612*b9df5ad1SAndroid Build Coastguard Worker [16] = "AUDIO_CHANNEL_INDEX_MASK_16",
613*b9df5ad1SAndroid Build Coastguard Worker
614*b9df5ad1SAndroid Build Coastguard Worker [17] = "AUDIO_CHANNEL_INDEX_MASK_17",
615*b9df5ad1SAndroid Build Coastguard Worker [18] = "AUDIO_CHANNEL_INDEX_MASK_18",
616*b9df5ad1SAndroid Build Coastguard Worker [19] = "AUDIO_CHANNEL_INDEX_MASK_19",
617*b9df5ad1SAndroid Build Coastguard Worker [20] = "AUDIO_CHANNEL_INDEX_MASK_20",
618*b9df5ad1SAndroid Build Coastguard Worker [21] = "AUDIO_CHANNEL_INDEX_MASK_21",
619*b9df5ad1SAndroid Build Coastguard Worker [22] = "AUDIO_CHANNEL_INDEX_MASK_22",
620*b9df5ad1SAndroid Build Coastguard Worker [23] = "AUDIO_CHANNEL_INDEX_MASK_23",
621*b9df5ad1SAndroid Build Coastguard Worker [24] = "AUDIO_CHANNEL_INDEX_MASK_24",
622*b9df5ad1SAndroid Build Coastguard Worker };
623*b9df5ad1SAndroid Build Coastguard Worker
624*b9df5ad1SAndroid Build Coastguard Worker const bool isOutProfile = profile->direction == PCM_OUT;
625*b9df5ad1SAndroid Build Coastguard Worker
626*b9df5ad1SAndroid Build Coastguard Worker const char * const * const chans_strs = isOutProfile ? out_chans_strs : in_chans_strs;
627*b9df5ad1SAndroid Build Coastguard Worker size_t chans_strs_size =
628*b9df5ad1SAndroid Build Coastguard Worker isOutProfile ? ARRAY_SIZE(out_chans_strs) : ARRAY_SIZE(in_chans_strs);
629*b9df5ad1SAndroid Build Coastguard Worker if (chans_strs_size > FCC_LIMIT + 1) chans_strs_size = FCC_LIMIT + 1; // starts with 0.
630*b9df5ad1SAndroid Build Coastguard Worker
631*b9df5ad1SAndroid Build Coastguard Worker /*
632*b9df5ad1SAndroid Build Coastguard Worker * MAX_CHANNEL_NAME_LEN: The longest channel name so far is "AUDIO_CHANNEL_OUT_7POINT1POINT4"
633*b9df5ad1SAndroid Build Coastguard Worker * at 31 chars, add 1 for the "|" delimiter, so we allocate 48 chars to be safe.
634*b9df5ad1SAndroid Build Coastguard Worker *
635*b9df5ad1SAndroid Build Coastguard Worker * We allocate room for channel index and channel position strings (2x).
636*b9df5ad1SAndroid Build Coastguard Worker */
637*b9df5ad1SAndroid Build Coastguard Worker const size_t MAX_CHANNEL_NAME_LEN = 48;
638*b9df5ad1SAndroid Build Coastguard Worker char buffer[MAX_CHANNEL_NAME_LEN * (FCC_LIMIT * 2) + 1];
639*b9df5ad1SAndroid Build Coastguard Worker buffer[0] = '\0';
640*b9df5ad1SAndroid Build Coastguard Worker size_t buffSize = ARRAY_SIZE(buffer);
641*b9df5ad1SAndroid Build Coastguard Worker size_t curStrLen = 0;
642*b9df5ad1SAndroid Build Coastguard Worker
643*b9df5ad1SAndroid Build Coastguard Worker /* We currently support MONO and STEREO, and always report STEREO but some (many)
644*b9df5ad1SAndroid Build Coastguard Worker * USB Audio Devices may only announce support for MONO (a headset mic for example), or
645*b9df5ad1SAndroid Build Coastguard Worker * The total number of output channels. SO, if the device itself doesn't explicitly
646*b9df5ad1SAndroid Build Coastguard Worker * support STEREO, append to the channel config strings we are generating.
647*b9df5ad1SAndroid Build Coastguard Worker *
648*b9df5ad1SAndroid Build Coastguard Worker * The MONO and STEREO positional channel masks are provided for legacy compatibility.
649*b9df5ad1SAndroid Build Coastguard Worker * For multichannel (n > 2) we only expose channel index masks.
650*b9df5ad1SAndroid Build Coastguard Worker */
651*b9df5ad1SAndroid Build Coastguard Worker // Always support stereo
652*b9df5ad1SAndroid Build Coastguard Worker curStrLen = strlcat(buffer, chans_strs[2], buffSize);
653*b9df5ad1SAndroid Build Coastguard Worker
654*b9df5ad1SAndroid Build Coastguard Worker size_t index;
655*b9df5ad1SAndroid Build Coastguard Worker unsigned channel_count;
656*b9df5ad1SAndroid Build Coastguard Worker for (index = 0;
657*b9df5ad1SAndroid Build Coastguard Worker (channel_count = profile->channel_counts[index]) != 0;
658*b9df5ad1SAndroid Build Coastguard Worker index++) {
659*b9df5ad1SAndroid Build Coastguard Worker
660*b9df5ad1SAndroid Build Coastguard Worker if (channel_count > FCC_LIMIT) continue;
661*b9df5ad1SAndroid Build Coastguard Worker
662*b9df5ad1SAndroid Build Coastguard Worker /* we only show positional information for mono (stereo handled already) */
663*b9df5ad1SAndroid Build Coastguard Worker if (channel_count < chans_strs_size
664*b9df5ad1SAndroid Build Coastguard Worker && chans_strs[channel_count] != NULL
665*b9df5ad1SAndroid Build Coastguard Worker && channel_count < 2 /* positional only for fewer than 2 channels */) {
666*b9df5ad1SAndroid Build Coastguard Worker // account for the '|' and the '\0'
667*b9df5ad1SAndroid Build Coastguard Worker if (buffSize - curStrLen < strlen(chans_strs[channel_count]) + 2) {
668*b9df5ad1SAndroid Build Coastguard Worker /* we don't have room for another, so bail at this point rather than
669*b9df5ad1SAndroid Build Coastguard Worker * return a malformed rate string
670*b9df5ad1SAndroid Build Coastguard Worker */
671*b9df5ad1SAndroid Build Coastguard Worker break;
672*b9df5ad1SAndroid Build Coastguard Worker }
673*b9df5ad1SAndroid Build Coastguard Worker
674*b9df5ad1SAndroid Build Coastguard Worker if (curStrLen != 0) strlcat(buffer, "|", buffSize);
675*b9df5ad1SAndroid Build Coastguard Worker curStrLen = strlcat(buffer, chans_strs[channel_count], buffSize);
676*b9df5ad1SAndroid Build Coastguard Worker }
677*b9df5ad1SAndroid Build Coastguard Worker
678*b9df5ad1SAndroid Build Coastguard Worker // handle channel index masks for both input and output
679*b9df5ad1SAndroid Build Coastguard Worker // +2 to account for the '|' and the '\0'
680*b9df5ad1SAndroid Build Coastguard Worker if (buffSize - curStrLen < strlen(index_chans_strs[channel_count]) + 2) {
681*b9df5ad1SAndroid Build Coastguard Worker /* we don't have room for another, so bail at this point rather than
682*b9df5ad1SAndroid Build Coastguard Worker * return a malformed rate string
683*b9df5ad1SAndroid Build Coastguard Worker */
684*b9df5ad1SAndroid Build Coastguard Worker break;
685*b9df5ad1SAndroid Build Coastguard Worker }
686*b9df5ad1SAndroid Build Coastguard Worker
687*b9df5ad1SAndroid Build Coastguard Worker if (curStrLen != 0) strlcat(buffer, "|", buffSize);
688*b9df5ad1SAndroid Build Coastguard Worker curStrLen = strlcat(buffer, index_chans_strs[channel_count], buffSize);
689*b9df5ad1SAndroid Build Coastguard Worker }
690*b9df5ad1SAndroid Build Coastguard Worker
691*b9df5ad1SAndroid Build Coastguard Worker return strdup(buffer);
692*b9df5ad1SAndroid Build Coastguard Worker }
693*b9df5ad1SAndroid Build Coastguard Worker
profile_dump(const alsa_device_profile * profile,int fd)694*b9df5ad1SAndroid Build Coastguard Worker void profile_dump(const alsa_device_profile* profile, int fd)
695*b9df5ad1SAndroid Build Coastguard Worker {
696*b9df5ad1SAndroid Build Coastguard Worker if (profile == NULL) {
697*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " %s\n", "No USB Profile");
698*b9df5ad1SAndroid Build Coastguard Worker return; /* bail early */
699*b9df5ad1SAndroid Build Coastguard Worker }
700*b9df5ad1SAndroid Build Coastguard Worker
701*b9df5ad1SAndroid Build Coastguard Worker if (!profile->is_valid) {
702*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " Profile is INVALID");
703*b9df5ad1SAndroid Build Coastguard Worker }
704*b9df5ad1SAndroid Build Coastguard Worker
705*b9df5ad1SAndroid Build Coastguard Worker /* card/device/direction */
706*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " card:%d, device:%d - %s\n",
707*b9df5ad1SAndroid Build Coastguard Worker profile->card, profile->device, profile->direction == PCM_OUT ? "OUT" : "IN");
708*b9df5ad1SAndroid Build Coastguard Worker
709*b9df5ad1SAndroid Build Coastguard Worker /* formats */
710*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " Formats: ");
711*b9df5ad1SAndroid Build Coastguard Worker for (int fmtIndex = 0;
712*b9df5ad1SAndroid Build Coastguard Worker fmtIndex < MAX_PROFILE_FORMATS && profile->formats[fmtIndex] != PCM_FORMAT_INVALID;
713*b9df5ad1SAndroid Build Coastguard Worker fmtIndex++) {
714*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "%d ", profile->formats[fmtIndex]);
715*b9df5ad1SAndroid Build Coastguard Worker }
716*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "\n");
717*b9df5ad1SAndroid Build Coastguard Worker
718*b9df5ad1SAndroid Build Coastguard Worker /* sample rates */
719*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " Rates: ");
720*b9df5ad1SAndroid Build Coastguard Worker for (int rateIndex = 0;
721*b9df5ad1SAndroid Build Coastguard Worker rateIndex < MAX_PROFILE_SAMPLE_RATES && profile->sample_rates[rateIndex] != 0;
722*b9df5ad1SAndroid Build Coastguard Worker rateIndex++) {
723*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "%u ", profile->sample_rates[rateIndex]);
724*b9df5ad1SAndroid Build Coastguard Worker }
725*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "\n");
726*b9df5ad1SAndroid Build Coastguard Worker
727*b9df5ad1SAndroid Build Coastguard Worker // channel counts
728*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " Channel Counts: ");
729*b9df5ad1SAndroid Build Coastguard Worker for (int cntIndex = 0;
730*b9df5ad1SAndroid Build Coastguard Worker cntIndex < MAX_PROFILE_CHANNEL_COUNTS && profile->channel_counts[cntIndex] != 0;
731*b9df5ad1SAndroid Build Coastguard Worker cntIndex++) {
732*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "%u ", profile->channel_counts[cntIndex]);
733*b9df5ad1SAndroid Build Coastguard Worker }
734*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, "\n");
735*b9df5ad1SAndroid Build Coastguard Worker
736*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " min/max period size [%u : %u]\n",
737*b9df5ad1SAndroid Build Coastguard Worker profile->min_period_size,profile-> max_period_size);
738*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " min/max channel count [%u : %u]\n",
739*b9df5ad1SAndroid Build Coastguard Worker profile->min_channel_count, profile->max_channel_count);
740*b9df5ad1SAndroid Build Coastguard Worker
741*b9df5ad1SAndroid Build Coastguard Worker // struct pcm_config default_config;
742*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " Default Config:\n");
743*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " channels: %d\n", profile->default_config.channels);
744*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " rate: %d\n", profile->default_config.rate);
745*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " period_size: %d\n", profile->default_config.period_size);
746*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " period_count: %d\n", profile->default_config.period_count);
747*b9df5ad1SAndroid Build Coastguard Worker dprintf(fd, " format: %d\n", profile->default_config.format);
748*b9df5ad1SAndroid Build Coastguard Worker }
749