blob: 77e8ca32875ce61c06f0d5a1d4ba7f377ec3a2c8 [file] [log] [blame]
Paul McLeanc88e6ae2014-07-16 09:48:34 -07001/*
2 * Copyright (C) 2014 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 "alsa_device_proxy"
18/*#define LOG_NDEBUG 0*/
19/*#define LOG_PCM_PARAMS 0*/
20
21#include <log/log.h>
22
Elliott Hughes6cdd2b32014-12-29 12:59:11 -080023#include <errno.h>
24
Paul McLeanc88e6ae2014-07-16 09:48:34 -070025#include "alsa_device_proxy.h"
26
Paul McLean271444a2014-12-15 09:44:42 -080027#include "audio_logging.h"
Paul McLeanc88e6ae2014-07-16 09:48:34 -070028
29#define DEFAULT_PERIOD_SIZE 1024
30#define DEFAULT_PERIOD_COUNT 2
31
Paul McLeanc88e6ae2014-07-16 09:48:34 -070032void proxy_prepare(alsa_device_proxy * proxy, alsa_device_profile* profile,
33 struct pcm_config * config)
34{
35 ALOGV("proxy_prepare()");
36
37 proxy->profile = profile;
38
Paul McLeanaf5187e2014-08-12 13:57:10 -070039#ifdef LOG_PCM_PARAMS
Paul McLeanc88e6ae2014-07-16 09:48:34 -070040 log_pcm_config(config, "proxy_setup()");
41#endif
42
43 proxy->alsa_config.format =
44 config->format != PCM_FORMAT_INVALID && profile_is_format_valid(profile, config->format)
45 ? config->format : profile->default_config.format;
46 proxy->alsa_config.rate =
47 config->rate != 0 && profile_is_sample_rate_valid(profile, config->rate)
48 ? config->rate : profile->default_config.rate;
49 proxy->alsa_config.channels =
50 config->channels != 0 && profile_is_channel_count_valid(profile, config->channels)
51 ? config->channels : profile->default_config.channels;
52
53 proxy->alsa_config.period_count = profile->default_config.period_count;
54 proxy->alsa_config.period_size =
55 profile_get_period_size(proxy->profile, proxy->alsa_config.rate);
56
Mike Lockwood7de76b22014-08-14 09:34:54 -070057 // Hack for USB accessory audio.
58 // Here we set the correct value for period_count if tinyalsa fails to get it from the
59 // f_audio_source driver.
60 if (proxy->alsa_config.period_count == 0) {
61 proxy->alsa_config.period_count = 4;
62 }
63
Paul McLeanc88e6ae2014-07-16 09:48:34 -070064 proxy->pcm = NULL;
65}
66
67int proxy_open(alsa_device_proxy * proxy)
68{
69 alsa_device_profile* profile = proxy->profile;
Paul McLeanc88e6ae2014-07-16 09:48:34 -070070 ALOGV("proxy_open(card:%d device:%d %s)", profile->card, profile->device,
71 profile->direction == PCM_OUT ? "PCM_OUT" : "PCM_IN");
72
73 proxy->pcm = pcm_open(profile->card, profile->device, profile->direction, &proxy->alsa_config);
74 if (proxy->pcm == NULL) {
75 return -ENOMEM;
76 }
77
78 if (!pcm_is_ready(proxy->pcm)) {
79 ALOGE("[%s] proxy_open() pcm_open() failed: %s", LOG_TAG, pcm_get_error(proxy->pcm));
80#ifdef LOG_PCM_PARAMS
81 log_pcm_config(&proxy->alsa_config, "config");
82#endif
83 pcm_close(proxy->pcm);
84 proxy->pcm = NULL;
85 return -ENOMEM;
86 }
87
88 return 0;
89}
90
91void proxy_close(alsa_device_proxy * proxy)
92{
Paul McLean2c6196f2014-08-20 16:50:25 -070093 ALOGV("proxy_close() [pcm:%p]", proxy->pcm);
94
95 if (proxy->pcm != NULL) {
96 pcm_close(proxy->pcm);
97 proxy->pcm = NULL;
98 }
Paul McLeanc88e6ae2014-07-16 09:48:34 -070099}
100
101/*
102 * Sample Rate
103 */
104unsigned proxy_get_sample_rate(const alsa_device_proxy * proxy)
105{
106 return proxy->alsa_config.rate;
107}
108
109/*
110 * Format
111 */
112enum pcm_format proxy_get_format(const alsa_device_proxy * proxy)
113{
114 return proxy->alsa_config.format;
115}
116
117/*
118 * Channel Count
119 */
120unsigned proxy_get_channel_count(const alsa_device_proxy * proxy)
121{
122 return proxy->alsa_config.channels;
123}
124
125/*
126 * Other
127 */
128unsigned int proxy_get_period_size(const alsa_device_proxy * proxy)
129{
130 return proxy->alsa_config.period_size;
131}
132
133unsigned int proxy_get_period_count(const alsa_device_proxy * proxy)
134{
135 return proxy->alsa_config.period_count;
136}
137
138unsigned proxy_get_latency(const alsa_device_proxy * proxy)
139{
140 return (proxy_get_period_size(proxy) * proxy_get_period_count(proxy) * 1000)
141 / proxy_get_sample_rate(proxy);
142}
143
144/*
145 * I/O
146 */
147int proxy_write(const alsa_device_proxy * proxy, const void *data, unsigned int count)
148{
149 return pcm_write(proxy->pcm, data, count);
150}
151
152int proxy_read(const alsa_device_proxy * proxy, void *data, unsigned int count)
153{
154 return pcm_read(proxy->pcm, data, count);
155}