blob: ea47bdc550a04396499569c339a77a328cab08eb [file] [log] [blame]
Benny Prijono204ce5e2006-03-20 16:59:23 +00001/* $Id$ */
2/*
3 * Copyright (C) 2003-2006 Benny Prijono <benny@prijono.org>
4 *
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.
9 *
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.
14 *
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
18 */
19
20static const char *desc =
21 " sndinfo.c \n"
22 " \n"
23 " PURPOSE: \n"
24 " Print sound device info and test open device. \n"
25 " \n"
26 " USAGE: \n"
Benny Prijonoa8df6542006-03-23 14:20:42 +000027 " sndinfo [id rec/play/both clockrate nchan bits] \n"
Benny Prijono204ce5e2006-03-20 16:59:23 +000028 " \n"
29 " DESCRIPTION: \n"
30 " When invoked without any arguments, it displays information about all \n"
31 " sound devices in the system. \n"
32 " \n"
33 " When invoked with arguments, the program tests if device can be opened \n"
34 " with the specified arguments. All these arguments must be specified: \n"
Benny Prijonoa8df6542006-03-23 14:20:42 +000035 " - id The device ID (-1 for the first capable device) \n"
36 " - rec/play/both Specify which streams to open. \n"
37 " - clockrate Specify clock rate (e.g. 8000, 11025, etc.) \n"
38 " - nchan Number of channels (1=mono, 2=stereo). \n"
39 " - bits Number of bits per sample (normally 16). \n";
Benny Prijono204ce5e2006-03-20 16:59:23 +000040
41#include <pjmedia.h>
42#include <pjlib.h>
43
44#include <stdlib.h> /* atoi() */
45#include <stdio.h>
46
47
48static void enum_devices(void)
49{
50 int i, count;
51
52 count = pjmedia_snd_get_dev_count();
53 if (count == 0) {
54 puts("No devices found");
55 return;
56 }
57
58 for (i=0; i<count; ++i) {
59 const pjmedia_snd_dev_info *info;
60
61 info = pjmedia_snd_get_dev_info(i);
62 pj_assert(info != NULL);
63
64 printf( "Device #%02d: \n"
65 " Name : %s\n"
66 " # of input channels : %d\n"
67 " # of output channels: %d\n"
68 " Default clock rate : %d Hz\n\n",
69 i, info->name, info->input_count, info->output_count,
70 info->default_samples_per_sec);
71 }
Benny Prijonoa8df6542006-03-23 14:20:42 +000072 puts("");
73 puts("Run with -h to get more options");
Benny Prijono204ce5e2006-03-20 16:59:23 +000074}
75
Benny Prijonoa8df6542006-03-23 14:20:42 +000076static int play_counter;
77static int rec_counter;
Benny Prijonoe452dda2006-04-03 09:43:36 +000078static int min_delay = 0xFFFF, max_delay;
79static char play_delays[1000];
Benny Prijono204ce5e2006-03-20 16:59:23 +000080
81static pj_status_t play_cb(void *user_data, pj_uint32_t timestamp,
82 void *output, unsigned size)
83{
Benny Prijonoc708f932006-04-03 11:42:56 +000084 static pj_timestamp last_cb;
Benny Prijonoe452dda2006-04-03 09:43:36 +000085
Benny Prijonoa8df6542006-03-23 14:20:42 +000086 ++play_counter;
Benny Prijonoe452dda2006-04-03 09:43:36 +000087
Benny Prijonoc708f932006-04-03 11:42:56 +000088 if (last_cb.u64 == 0) {
89 pj_get_timestamp(&last_cb);
Benny Prijonoe452dda2006-04-03 09:43:36 +000090 } else {
Benny Prijonoc708f932006-04-03 11:42:56 +000091 pj_timestamp now;
Benny Prijonoe452dda2006-04-03 09:43:36 +000092 int delay;
93
Benny Prijonoc708f932006-04-03 11:42:56 +000094 pj_get_timestamp(&now);
95
96 delay = pj_elapsed_msec(&last_cb, &now);
Benny Prijonoe452dda2006-04-03 09:43:36 +000097 if (delay < min_delay)
98 min_delay = delay;
99 if (delay > max_delay)
100 max_delay = delay;
101
Benny Prijonoc708f932006-04-03 11:42:56 +0000102 last_cb = now;
Benny Prijonoe452dda2006-04-03 09:43:36 +0000103
104 play_delays[play_counter-1] = delay;
105 }
106
Benny Prijono204ce5e2006-03-20 16:59:23 +0000107 return PJ_SUCCESS;
108}
109
110static pj_status_t rec_cb(void *user_data, pj_uint32_t timestamp,
111 const void *input, unsigned size)
112{
Benny Prijonoa8df6542006-03-23 14:20:42 +0000113 ++rec_counter;
Benny Prijono204ce5e2006-03-20 16:59:23 +0000114 return PJ_SUCCESS;
115}
116
Benny Prijonoa8df6542006-03-23 14:20:42 +0000117static void app_perror(const char *title, pj_status_t status)
Benny Prijono204ce5e2006-03-20 16:59:23 +0000118{
Benny Prijonoa8df6542006-03-23 14:20:42 +0000119 char errmsg[PJ_ERR_MSG_SIZE];
120
121 pj_strerror(status, errmsg, sizeof(errmsg));
122 printf( "%s: %s (err=%d)\n",
123 title, errmsg, status);
124}
Benny Prijonoe452dda2006-04-03 09:43:36 +0000125
Benny Prijonoa8df6542006-03-23 14:20:42 +0000126static int open_device(int dev_id, pjmedia_dir dir,
127 int clock_rate, int nchannel, int bits)
128{
129 pj_status_t status = PJ_SUCCESS;
Benny Prijono204ce5e2006-03-20 16:59:23 +0000130 unsigned nsamples;
131 pjmedia_snd_stream *strm;
Benny Prijonoa8df6542006-03-23 14:20:42 +0000132 const char *dirtype;
Benny Prijonoe452dda2006-04-03 09:43:36 +0000133 unsigned i;
Benny Prijono204ce5e2006-03-20 16:59:23 +0000134
Benny Prijonoa8df6542006-03-23 14:20:42 +0000135 switch (dir) {
136 case PJMEDIA_DIR_CAPTURE:
137 dirtype = "capture"; break;
138 case PJMEDIA_DIR_PLAYBACK:
139 dirtype = "playback"; break;
140 case PJMEDIA_DIR_CAPTURE_PLAYBACK:
141 dirtype = "capture/playback"; break;
142 default:
143 return 1;
144 }
145
Benny Prijono204ce5e2006-03-20 16:59:23 +0000146 nsamples = clock_rate * 20 / 1000;
147
Benny Prijonoa8df6542006-03-23 14:20:42 +0000148 printf( "Opening device %d for %s: clockrate=%d, nchannel=%d, "
149 "bits=%d, nsamples=%d..\n",
150 dev_id, dirtype, clock_rate, nchannel, bits, nsamples);
Benny Prijono204ce5e2006-03-20 16:59:23 +0000151
Benny Prijonoa8df6542006-03-23 14:20:42 +0000152 if (dir == PJMEDIA_DIR_CAPTURE) {
153 status = pjmedia_snd_open_rec( dev_id, clock_rate, nchannel,
154 nsamples, bits, &rec_cb, NULL,
155 &strm);
156 } else if (dir == PJMEDIA_DIR_PLAYBACK) {
Benny Prijono204ce5e2006-03-20 16:59:23 +0000157 status = pjmedia_snd_open_player( dev_id, clock_rate, nchannel,
158 nsamples, bits, &play_cb, NULL,
159 &strm);
Benny Prijonoa8df6542006-03-23 14:20:42 +0000160 } else {
161 status = pjmedia_snd_open( dev_id, dev_id, clock_rate, nchannel,
162 nsamples, bits, &rec_cb, &play_cb, NULL,
163 &strm);
Benny Prijono204ce5e2006-03-20 16:59:23 +0000164 }
165
166 if (status != PJ_SUCCESS) {
Benny Prijonoa8df6542006-03-23 14:20:42 +0000167 app_perror("Unable to open device for capture", status);
168 return 1;
Benny Prijono204ce5e2006-03-20 16:59:23 +0000169 }
170
Benny Prijonoa8df6542006-03-23 14:20:42 +0000171 status = pjmedia_snd_stream_start(strm);
172 if (status != PJ_SUCCESS) {
173 app_perror("Unable to start capture stream", status);
174 return 1;
175 }
176
177 /* Let playback/capture runs for a while */
178 pj_thread_sleep(1000);
179
Benny Prijono204ce5e2006-03-20 16:59:23 +0000180 pjmedia_snd_stream_close(strm);
Benny Prijonoa8df6542006-03-23 14:20:42 +0000181
182 if ((dir & PJMEDIA_DIR_CAPTURE) && rec_counter==0) {
183 printf("Error: capture stream was not running\n");
184 return 1;
185 }
186
187 if ((dir & PJMEDIA_DIR_PLAYBACK) && play_counter==0) {
188 printf("Error: playback stream was not running\n");
189 return 1;
190 }
191
192 puts("Success.");
Benny Prijonoe452dda2006-04-03 09:43:36 +0000193
194 printf("Delay: ");
195 for (i=0; i<play_counter; ++i)
196 printf("%d ", play_delays[i]);
197
198 puts("");
199 if (dir & PJMEDIA_DIR_PLAYBACK) {
200 printf("Callback interval: min interval=%d ms, max interval=%d ms\n",
201 min_delay, max_delay);
202 }
203
204
Benny Prijono204ce5e2006-03-20 16:59:23 +0000205 return 0;
206}
207
208
209int main(int argc, char *argv[])
210{
211 pj_caching_pool cp;
212 pjmedia_endpt *med_endpt;
213 pj_status_t status;
214
215 /* Init pjlib */
216 status = pj_init();
217 PJ_ASSERT_RETURN(status==PJ_SUCCESS, 1);
218
219 /* Must create a pool factory before we can allocate any memory. */
220 pj_caching_pool_init(&cp, &pj_pool_factory_default_policy, 0);
221
222 /*
223 * Initialize media endpoint.
224 * This will implicitly initialize PJMEDIA too.
225 */
Benny Prijono275fd682006-03-22 11:59:11 +0000226 status = pjmedia_endpt_create(&cp.factory, NULL, 1, &med_endpt);
Benny Prijono204ce5e2006-03-20 16:59:23 +0000227 PJ_ASSERT_RETURN(status == PJ_SUCCESS, 1);
228
229
230 if (argc == 1) {
231 enum_devices();
232 return 0;
233 } else if (argc == 6) {
234
235 int dev_id;
Benny Prijonoa8df6542006-03-23 14:20:42 +0000236 pjmedia_dir dir;
Benny Prijono204ce5e2006-03-20 16:59:23 +0000237 int clock_rate;
238 int nchannel;
239 int bits;
240
241 dev_id = atoi(argv[1]);
Benny Prijonoa8df6542006-03-23 14:20:42 +0000242
243 if (strcmp(argv[2], "rec")==0)
244 dir = PJMEDIA_DIR_CAPTURE;
245 else if (strcmp(argv[2], "play")==0)
246 dir = PJMEDIA_DIR_PLAYBACK;
247 else if (strcmp(argv[2], "both")==0)
248 dir = PJMEDIA_DIR_CAPTURE_PLAYBACK;
249
Benny Prijono204ce5e2006-03-20 16:59:23 +0000250 clock_rate = atoi(argv[3]);
251 nchannel = atoi(argv[4]);
252 bits = atoi(argv[5]);
253
Benny Prijonoa8df6542006-03-23 14:20:42 +0000254 return open_device(dev_id, dir, clock_rate, nchannel, bits);
Benny Prijono204ce5e2006-03-20 16:59:23 +0000255
256 } else {
257 puts("Error: invalid arguments");
258 puts(desc);
259 return 1;
260 }
261
262 return 0;
263}
264
265