1
/* $Id: sound_legacy.c 3553 2011-05-05 06:14:19Z nanang $ */
3
* Copyright (C) 2008-2011 Teluu Inc. (http://www.teluu.com)
5
* This program is free software; you can redistribute it and/or modify
6
* it under the terms of the GNU General Public License as published by
7
* the Free Software Foundation; either version 2 of the License, or
8
* (at your option) any later version.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21
* This is implementation of legacy sound device API, for applications
22
* that still use the old/deprecated sound device API. This implementation
23
* uses the new Audio Device API.
25
* Please see http://trac.pjsip.org/repos/wiki/Audio_Dev_API for more
29
#include <pjmedia/sound.h>
30
#include <pjmedia-audiodev/errno.h>
31
#include <pj/assert.h>
33
#if PJMEDIA_HAS_LEGACY_SOUND_API
35
static struct legacy_subsys
37
pjmedia_snd_dev_info info[4];
38
unsigned info_counter;
39
unsigned user_rec_latency;
40
unsigned user_play_latency;
43
struct pjmedia_snd_stream
46
pjmedia_aud_stream *aud_strm;
47
pjmedia_snd_rec_cb user_rec_cb;
48
pjmedia_snd_play_cb user_play_cb;
52
PJ_DEF(pj_status_t) pjmedia_snd_init(pj_pool_factory *factory)
54
return pjmedia_aud_subsys_init(factory);
57
PJ_DEF(pj_status_t) pjmedia_snd_deinit(void)
59
return pjmedia_aud_subsys_shutdown();
62
PJ_DEF(int) pjmedia_snd_get_dev_count(void)
64
return pjmedia_aud_dev_count();
67
PJ_DEF(const pjmedia_snd_dev_info*) pjmedia_snd_get_dev_info(unsigned index)
69
pjmedia_snd_dev_info *oi = &g_sys.info[g_sys.info_counter];
70
pjmedia_aud_dev_info di;
72
g_sys.info_counter = (g_sys.info_counter+1) % PJ_ARRAY_SIZE(g_sys.info);
74
if (pjmedia_aud_dev_get_info(index, &di) != PJ_SUCCESS)
77
pj_bzero(oi, sizeof(*oi));
78
pj_ansi_strncpy(oi->name, di.name, sizeof(oi->name));
79
oi->name[sizeof(oi->name)-1] = '\0';
80
oi->input_count = di.input_count;
81
oi->output_count = di.output_count;
82
oi->default_samples_per_sec = di.default_samples_per_sec;
88
static pj_status_t snd_play_cb(void *user_data,
91
pjmedia_snd_stream *strm = (pjmedia_snd_stream*)user_data;
93
frame->type = PJMEDIA_FRAME_TYPE_AUDIO;
94
return strm->user_play_cb(strm->user_user_data,
95
frame->timestamp.u32.lo,
100
static pj_status_t snd_rec_cb(void *user_data,
101
pjmedia_frame *frame)
103
pjmedia_snd_stream *strm = (pjmedia_snd_stream*)user_data;
104
return strm->user_rec_cb(strm->user_user_data,
105
frame->timestamp.u32.lo,
110
static pj_status_t open_stream( pjmedia_dir dir,
114
unsigned channel_count,
115
unsigned samples_per_frame,
116
unsigned bits_per_sample,
117
pjmedia_snd_rec_cb rec_cb,
118
pjmedia_snd_play_cb play_cb,
120
pjmedia_snd_stream **p_snd_strm)
123
pjmedia_snd_stream *snd_strm;
124
pjmedia_aud_param param;
127
/* Initialize parameters */
128
if (dir & PJMEDIA_DIR_CAPTURE) {
129
status = pjmedia_aud_dev_default_param(rec_id, ¶m);
131
status = pjmedia_aud_dev_default_param(play_id, ¶m);
133
if (status != PJ_SUCCESS)
137
param.rec_id = rec_id;
138
param.play_id = play_id;
139
param.clock_rate = clock_rate;
140
param.channel_count = channel_count;
141
param.samples_per_frame = samples_per_frame;
142
param.bits_per_sample = bits_per_sample;
144
/* Latencies setting */
145
if ((dir & PJMEDIA_DIR_CAPTURE) && g_sys.user_rec_latency) {
146
param.flags |= PJMEDIA_AUD_DEV_CAP_INPUT_LATENCY;
147
param.input_latency_ms = g_sys.user_rec_latency;
149
if ((dir & PJMEDIA_DIR_PLAYBACK) && g_sys.user_play_latency) {
150
param.flags |= PJMEDIA_AUD_DEV_CAP_OUTPUT_LATENCY;
151
param.output_latency_ms = g_sys.user_play_latency;
154
/* Create sound wrapper */
155
pool = pj_pool_create(pjmedia_aud_subsys_get_pool_factory(),
156
"legacy-snd", 512, 512, NULL);
157
snd_strm = PJ_POOL_ZALLOC_T(pool, pjmedia_snd_stream);
158
snd_strm->pool = pool;
159
snd_strm->user_rec_cb = rec_cb;
160
snd_strm->user_play_cb = play_cb;
161
snd_strm->user_user_data = user_data;
163
/* Create the stream */
164
status = pjmedia_aud_stream_create(¶m, &snd_rec_cb,
165
&snd_play_cb, snd_strm,
166
&snd_strm->aud_strm);
167
if (status != PJ_SUCCESS) {
168
pj_pool_release(pool);
172
*p_snd_strm = snd_strm;
176
PJ_DEF(pj_status_t) pjmedia_snd_open_rec( int index,
178
unsigned channel_count,
179
unsigned samples_per_frame,
180
unsigned bits_per_sample,
181
pjmedia_snd_rec_cb rec_cb,
183
pjmedia_snd_stream **p_snd_strm)
185
return open_stream(PJMEDIA_DIR_CAPTURE, index, PJMEDIA_AUD_INVALID_DEV,
186
clock_rate, channel_count, samples_per_frame,
187
bits_per_sample, rec_cb, NULL,
188
user_data, p_snd_strm);
191
PJ_DEF(pj_status_t) pjmedia_snd_open_player( int index,
193
unsigned channel_count,
194
unsigned samples_per_frame,
195
unsigned bits_per_sample,
196
pjmedia_snd_play_cb play_cb,
198
pjmedia_snd_stream **p_snd_strm )
200
return open_stream(PJMEDIA_DIR_PLAYBACK, PJMEDIA_AUD_INVALID_DEV, index,
201
clock_rate, channel_count, samples_per_frame,
202
bits_per_sample, NULL, play_cb,
203
user_data, p_snd_strm);
206
PJ_DEF(pj_status_t) pjmedia_snd_open( int rec_id,
209
unsigned channel_count,
210
unsigned samples_per_frame,
211
unsigned bits_per_sample,
212
pjmedia_snd_rec_cb rec_cb,
213
pjmedia_snd_play_cb play_cb,
215
pjmedia_snd_stream **p_snd_strm)
217
return open_stream(PJMEDIA_DIR_CAPTURE_PLAYBACK, rec_id, play_id,
218
clock_rate, channel_count, samples_per_frame,
219
bits_per_sample, rec_cb, play_cb,
220
user_data, p_snd_strm);
223
PJ_DEF(pj_status_t) pjmedia_snd_stream_start(pjmedia_snd_stream *stream)
225
return pjmedia_aud_stream_start(stream->aud_strm);
228
PJ_DEF(pj_status_t) pjmedia_snd_stream_stop(pjmedia_snd_stream *stream)
230
return pjmedia_aud_stream_stop(stream->aud_strm);
233
PJ_DEF(pj_status_t) pjmedia_snd_stream_get_info(pjmedia_snd_stream *strm,
234
pjmedia_snd_stream_info *pi)
236
pjmedia_aud_param param;
239
status = pjmedia_aud_stream_get_param(strm->aud_strm, ¶m);
240
if (status != PJ_SUCCESS)
243
pj_bzero(pi, sizeof(*pi));
245
pi->play_id = param.play_id;
246
pi->rec_id = param.rec_id;
247
pi->clock_rate = param.clock_rate;
248
pi->channel_count = param.channel_count;
249
pi->samples_per_frame = param.samples_per_frame;
250
pi->bits_per_sample = param.bits_per_sample;
252
if (param.flags & PJMEDIA_AUD_DEV_CAP_INPUT_LATENCY) {
253
pi->rec_latency = param.input_latency_ms;
255
if (param.flags & PJMEDIA_AUD_DEV_CAP_OUTPUT_LATENCY) {
256
pi->play_latency = param.output_latency_ms;
263
PJ_DEF(pj_status_t) pjmedia_snd_stream_close(pjmedia_snd_stream *stream)
267
status = pjmedia_aud_stream_destroy(stream->aud_strm);
268
if (status != PJ_SUCCESS)
271
pj_pool_release(stream->pool);
275
PJ_DEF(pj_status_t) pjmedia_snd_set_latency(unsigned input_latency,
276
unsigned output_latency)
278
g_sys.user_rec_latency = input_latency;
279
g_sys.user_play_latency = output_latency;
283
#endif /* PJMEDIA_HAS_LEGACY_SOUND_API */