blob: 3d13acd4d2befdf244d5d9728733b98615037212 [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{
Paul McLean0f1753e2014-12-02 12:36:45 -070035 ALOGV("proxy_prepare(c:%d, d:%d)", profile->card, profile->device);
Paul McLeanc88e6ae2014-07-16 09:48:34 -070036
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
Paul McLean0f1753e2014-12-02 12:36:45 -070073 if (profile->card < 0 || profile->device < 0) {
74 return -EINVAL;
75 }
76
Paul McLeanc88e6ae2014-07-16 09:48:34 -070077 proxy->pcm = pcm_open(profile->card, profile->device, profile->direction, &proxy->alsa_config);
78 if (proxy->pcm == NULL) {
79 return -ENOMEM;
80 }
81
82 if (!pcm_is_ready(proxy->pcm)) {
Paul McLean0f1753e2014-12-02 12:36:45 -070083 ALOGE(" proxy_open() pcm_open() failed: %s", pcm_get_error(proxy->pcm));
84#if defined(LOG_PCM_PARAMS)
Paul McLeanc88e6ae2014-07-16 09:48:34 -070085 log_pcm_config(&proxy->alsa_config, "config");
86#endif
87 pcm_close(proxy->pcm);
88 proxy->pcm = NULL;
89 return -ENOMEM;
90 }
91
92 return 0;
93}
94
95void proxy_close(alsa_device_proxy * proxy)
96{
Paul McLean2c6196f2014-08-20 16:50:25 -070097 ALOGV("proxy_close() [pcm:%p]", proxy->pcm);
98
99 if (proxy->pcm != NULL) {
100 pcm_close(proxy->pcm);
101 proxy->pcm = NULL;
102 }
Paul McLeanc88e6ae2014-07-16 09:48:34 -0700103}
104
105/*
106 * Sample Rate
107 */
108unsigned proxy_get_sample_rate(const alsa_device_proxy * proxy)
109{
110 return proxy->alsa_config.rate;
111}
112
113/*
114 * Format
115 */
116enum pcm_format proxy_get_format(const alsa_device_proxy * proxy)
117{
118 return proxy->alsa_config.format;
119}
120
121/*
122 * Channel Count
123 */
124unsigned proxy_get_channel_count(const alsa_device_proxy * proxy)
125{
126 return proxy->alsa_config.channels;
127}
128
129/*
130 * Other
131 */
132unsigned int proxy_get_period_size(const alsa_device_proxy * proxy)
133{
134 return proxy->alsa_config.period_size;
135}
136
137unsigned int proxy_get_period_count(const alsa_device_proxy * proxy)
138{
139 return proxy->alsa_config.period_count;
140}
141
142unsigned proxy_get_latency(const alsa_device_proxy * proxy)
143{
144 return (proxy_get_period_size(proxy) * proxy_get_period_count(proxy) * 1000)
145 / proxy_get_sample_rate(proxy);
146}
147
148/*
149 * I/O
150 */
151int proxy_write(const alsa_device_proxy * proxy, const void *data, unsigned int count)
152{
153 return pcm_write(proxy->pcm, data, count);
154}
155
156int proxy_read(const alsa_device_proxy * proxy, void *data, unsigned int count)
157{
158 return pcm_read(proxy->pcm, data, count);
159}