main.c revision 1.8.2.2 1 1.8.2.2 martin /* $NetBSD: main.c,v 1.8.2.2 2019/09/28 07:41:18 martin Exp $ */
2 1.1 mrg
3 1.1 mrg /*
4 1.1 mrg * Copyright (c) 2010 Jared D. McNeill <jmcneill (at) invisible.ca>
5 1.1 mrg * All rights reserved.
6 1.1 mrg *
7 1.1 mrg * Redistribution and use in source and binary forms, with or without
8 1.1 mrg * modification, are permitted provided that the following conditions
9 1.1 mrg * are met:
10 1.1 mrg * 1. Redistributions of source code must retain the above copyright
11 1.1 mrg * notice, this list of conditions and the following disclaimer.
12 1.1 mrg * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 mrg * notice, this list of conditions and the following disclaimer in the
14 1.1 mrg * documentation and/or other materials provided with the distribution.
15 1.1 mrg *
16 1.1 mrg * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
17 1.1 mrg * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
18 1.1 mrg * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 1.1 mrg * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
20 1.1 mrg * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21 1.1 mrg * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22 1.1 mrg * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23 1.1 mrg * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24 1.1 mrg * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25 1.1 mrg * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26 1.1 mrg * POSSIBILITY OF SUCH DAMAGE.
27 1.1 mrg */
28 1.1 mrg
29 1.6 jmcneill #include <assert.h>
30 1.1 mrg #include <errno.h>
31 1.1 mrg #include <fcntl.h>
32 1.1 mrg #include <limits.h>
33 1.1 mrg #include <stdio.h>
34 1.1 mrg #include <stdlib.h>
35 1.1 mrg #include <unistd.h>
36 1.1 mrg
37 1.1 mrg #include "audiodev.h"
38 1.1 mrg #include "drvctl.h"
39 1.1 mrg
40 1.7 joerg __dead static void
41 1.1 mrg usage(const char *p)
42 1.1 mrg {
43 1.8 isaki fprintf(stderr, "usage: %s list [<index>]\n", p);
44 1.3 wiz fprintf(stderr, " %s default <index>\n", p);
45 1.8 isaki fprintf(stderr, " %s set <index> [p|r] <enc> <prec> <ch> <freq>\n",
46 1.8 isaki p);
47 1.3 wiz fprintf(stderr, " %s test <index>\n", p);
48 1.1 mrg exit(EXIT_FAILURE);
49 1.1 mrg }
50 1.1 mrg
51 1.8 isaki const char *encoding_names[] = {
52 1.8 isaki "none",
53 1.8 isaki AudioEmulaw,
54 1.8 isaki AudioEalaw,
55 1.8 isaki "pcm16",
56 1.8 isaki "pcm8",
57 1.8 isaki AudioEadpcm,
58 1.8 isaki AudioEslinear_le,
59 1.8 isaki AudioEslinear_be,
60 1.8 isaki AudioEulinear_le,
61 1.8 isaki AudioEulinear_be,
62 1.8 isaki AudioEslinear,
63 1.8 isaki AudioEulinear,
64 1.8 isaki AudioEmpeg_l1_stream,
65 1.8 isaki AudioEmpeg_l1_packets,
66 1.8 isaki AudioEmpeg_l1_system,
67 1.8 isaki AudioEmpeg_l2_stream,
68 1.8 isaki AudioEmpeg_l2_packets,
69 1.8 isaki AudioEmpeg_l2_system,
70 1.8 isaki AudioEac3,
71 1.8 isaki };
72 1.8 isaki u_int encoding_max = __arraycount(encoding_names);
73 1.8 isaki
74 1.6 jmcneill static void
75 1.6 jmcneill print_audiodev(struct audiodev *adev, int i)
76 1.6 jmcneill {
77 1.8 isaki struct audiofmt *f;
78 1.8 isaki int j;
79 1.8 isaki
80 1.6 jmcneill assert(adev != NULL);
81 1.6 jmcneill
82 1.6 jmcneill printf("%u: [%c] %s @ %s: ",
83 1.6 jmcneill i, adev->defaultdev ? '*' : ' ',
84 1.6 jmcneill adev->xname, adev->pxname);
85 1.6 jmcneill printf("%s", adev->audio_device.name);
86 1.6 jmcneill if (strlen(adev->audio_device.version) > 0)
87 1.6 jmcneill printf(" %s", adev->audio_device.version);
88 1.8 isaki printf("\n");
89 1.8 isaki printf(" playback: ");
90 1.8.2.2 martin if ((adev->hwinfo.mode & AUMODE_PLAY)) {
91 1.8 isaki printf("%uch, %uHz\n",
92 1.8.2.2 martin adev->hwinfo.play.channels,
93 1.8.2.2 martin adev->hwinfo.play.sample_rate);
94 1.8.2.2 martin } else {
95 1.8 isaki printf("unavailable\n");
96 1.8.2.2 martin }
97 1.8 isaki printf(" record: ");
98 1.8.2.2 martin if ((adev->hwinfo.mode & AUMODE_RECORD)) {
99 1.8 isaki printf("%uch, %uHz\n",
100 1.8.2.2 martin adev->hwinfo.record.channels,
101 1.8.2.2 martin adev->hwinfo.record.sample_rate);
102 1.8.2.2 martin } else {
103 1.8 isaki printf("unavailable\n");
104 1.8.2.2 martin }
105 1.8 isaki
106 1.8 isaki TAILQ_FOREACH(f, &adev->formats, next) {
107 1.8 isaki printf(" ");
108 1.8 isaki if (f->fmt.priority < 0)
109 1.8 isaki printf("( ) ");
110 1.8 isaki else if ((f->fmt.mode & (AUMODE_PLAY | AUMODE_RECORD))
111 1.8 isaki == (AUMODE_PLAY | AUMODE_RECORD))
112 1.8 isaki printf("(PR) ");
113 1.8 isaki else if ((f->fmt.mode & AUMODE_PLAY))
114 1.8 isaki printf("(P-) ");
115 1.8 isaki else if ((f->fmt.mode & AUMODE_RECORD))
116 1.8 isaki printf("(-R) ");
117 1.8 isaki
118 1.8 isaki if (f->fmt.encoding < encoding_max)
119 1.8 isaki printf("%s", encoding_names[f->fmt.encoding]);
120 1.8 isaki else
121 1.8 isaki printf("unknown_encoding_%d", f->fmt.encoding);
122 1.8 isaki printf(" %d/%d, %dch, ",
123 1.8 isaki f->fmt.validbits,
124 1.8 isaki f->fmt.precision,
125 1.8 isaki f->fmt.channels);
126 1.8 isaki if (f->fmt.frequency_type == 0) {
127 1.8 isaki printf("%d-%dHz",
128 1.8 isaki f->fmt.frequency[0],
129 1.8 isaki f->fmt.frequency[1]);
130 1.8 isaki } else {
131 1.8 isaki for (j = 0; j < (int)f->fmt.frequency_type; j++) {
132 1.8 isaki printf("%s%d",
133 1.8 isaki (j == 0) ? "{ " : ", ",
134 1.8 isaki f->fmt.frequency[j]);
135 1.8 isaki }
136 1.8 isaki printf(" }");
137 1.8 isaki }
138 1.8 isaki printf("\n");
139 1.8 isaki }
140 1.6 jmcneill }
141 1.6 jmcneill
142 1.1 mrg int
143 1.1 mrg main(int argc, char *argv[])
144 1.1 mrg {
145 1.1 mrg struct audiodev *adev;
146 1.1 mrg unsigned int n, i;
147 1.8 isaki unsigned int j;
148 1.8 isaki const char *enc;
149 1.8 isaki unsigned int prec;
150 1.8 isaki unsigned int ch;
151 1.8 isaki unsigned int freq;
152 1.8 isaki int mode;
153 1.1 mrg
154 1.1 mrg if (audiodev_refresh() == -1)
155 1.1 mrg return EXIT_FAILURE;
156 1.1 mrg
157 1.2 jmcneill if (argc < 2)
158 1.2 jmcneill usage(argv[0]);
159 1.2 jmcneill /* NOTREACHED */
160 1.2 jmcneill
161 1.8 isaki if (strcmp(argv[1], "list") == 0 && argc == 2) {
162 1.1 mrg n = audiodev_count();
163 1.6 jmcneill for (i = 0; i < n; i++)
164 1.6 jmcneill print_audiodev(audiodev_get(i), i);
165 1.8 isaki } else if (strcmp(argv[1], "list") == 0 && argc == 3) {
166 1.8 isaki errno = 0;
167 1.8 isaki i = strtoul(argv[2], NULL, 10);
168 1.8 isaki if (errno)
169 1.8 isaki usage(argv[0]);
170 1.8 isaki /* NOTREACHED */
171 1.8.2.1 martin adev = audiodev_get(i);
172 1.8.2.1 martin if (adev == NULL) {
173 1.8.2.1 martin fprintf(stderr, "no such device\n");
174 1.8.2.1 martin return EXIT_FAILURE;
175 1.8.2.1 martin }
176 1.8.2.1 martin print_audiodev(adev, i);
177 1.2 jmcneill } else if (strcmp(argv[1], "default") == 0 && argc == 3) {
178 1.2 jmcneill if (*argv[2] < '0' || *argv[2] > '9')
179 1.1 mrg usage(argv[0]);
180 1.1 mrg /* NOTREACHED */
181 1.1 mrg errno = 0;
182 1.2 jmcneill i = strtoul(argv[2], NULL, 10);
183 1.1 mrg if (errno)
184 1.1 mrg usage(argv[0]);
185 1.1 mrg /* NOTREACHED */
186 1.1 mrg adev = audiodev_get(i);
187 1.1 mrg if (adev == NULL) {
188 1.1 mrg fprintf(stderr, "no such device\n");
189 1.1 mrg return EXIT_FAILURE;
190 1.1 mrg }
191 1.1 mrg printf("setting default audio device to %s\n", adev->xname);
192 1.1 mrg if (audiodev_set_default(adev) == -1) {
193 1.1 mrg perror("couldn't set default device");
194 1.1 mrg return EXIT_FAILURE;
195 1.1 mrg }
196 1.8 isaki } else if (strcmp(argv[1], "set") == 0 && argc == 8) {
197 1.8 isaki /* XXX bad commandline... */
198 1.8 isaki /* audiocfg set <index> [p|r] <enc> <prec> <ch> <freq> */
199 1.8 isaki if (*argv[2] < '0' || *argv[2] > '9')
200 1.8 isaki usage(argv[0]);
201 1.8 isaki /* NOTREACHED */
202 1.8 isaki errno = 0;
203 1.8 isaki i = strtoul(argv[2], NULL, 10);
204 1.8 isaki if (errno)
205 1.8 isaki usage(argv[0]);
206 1.8 isaki /* NOTREACHED */
207 1.8 isaki adev = audiodev_get(i);
208 1.8 isaki if (adev == NULL) {
209 1.8 isaki fprintf(stderr, "no such device\n");
210 1.8 isaki return EXIT_FAILURE;
211 1.8 isaki }
212 1.8 isaki
213 1.8 isaki mode = 0;
214 1.8 isaki for (j = 0; j < strlen(argv[3]); j++) {
215 1.8 isaki if (argv[3][j] == 'p')
216 1.8 isaki mode |= AUMODE_PLAY;
217 1.8 isaki else if (argv[3][j] == 'r')
218 1.8 isaki mode |= AUMODE_RECORD;
219 1.8 isaki else
220 1.8 isaki usage(argv[0]);
221 1.8 isaki }
222 1.8 isaki enc = argv[4];
223 1.8 isaki prec = strtoul(argv[5], NULL, 10);
224 1.8 isaki if (errno)
225 1.8 isaki usage(argv[0]);
226 1.8 isaki errno = 0;
227 1.8 isaki ch = strtoul(argv[6], NULL, 10);
228 1.8 isaki if (errno)
229 1.8 isaki usage(argv[0]);
230 1.8 isaki /* NOTREACHED */
231 1.8 isaki errno = 0;
232 1.8 isaki freq = strtoul(argv[7], NULL, 10);
233 1.8 isaki if (errno)
234 1.8 isaki usage(argv[0]);
235 1.8 isaki /* NOTREACHED */
236 1.8 isaki
237 1.8 isaki if (audiodev_set_param(adev, mode, enc, prec, ch, freq) == -1) {
238 1.8 isaki perror("couldn't set parameter");
239 1.8 isaki return EXIT_FAILURE;
240 1.8 isaki }
241 1.2 jmcneill } else if (strcmp(argv[1], "test") == 0 && argc == 3) {
242 1.2 jmcneill if (*argv[2] < '0' || *argv[2] > '9')
243 1.2 jmcneill usage(argv[0]);
244 1.2 jmcneill /* NOTREACHED */
245 1.2 jmcneill errno = 0;
246 1.2 jmcneill i = strtoul(argv[2], NULL, 10);
247 1.2 jmcneill if (errno)
248 1.2 jmcneill usage(argv[0]);
249 1.2 jmcneill /* NOTREACHED */
250 1.2 jmcneill adev = audiodev_get(i);
251 1.2 jmcneill if (adev == NULL) {
252 1.2 jmcneill fprintf(stderr, "no such device\n");
253 1.2 jmcneill return EXIT_FAILURE;
254 1.2 jmcneill }
255 1.6 jmcneill print_audiodev(adev, i);
256 1.8.2.2 martin if (audiodev_test(adev) == -1)
257 1.8.2.2 martin return EXIT_FAILURE;
258 1.2 jmcneill } else
259 1.1 mrg usage(argv[0]);
260 1.1 mrg /* NOTREACHED */
261 1.1 mrg
262 1.1 mrg return EXIT_SUCCESS;
263 1.1 mrg }
264