main.c revision 1.8.2.3 1 1.8.2.3 martin /* $NetBSD: main.c,v 1.8.2.3 2019/09/29 07:34:09 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.8.2.3 martin #include <err.h>
31 1.1 mrg #include <errno.h>
32 1.1 mrg #include <fcntl.h>
33 1.1 mrg #include <limits.h>
34 1.1 mrg #include <stdio.h>
35 1.1 mrg #include <stdlib.h>
36 1.1 mrg #include <unistd.h>
37 1.1 mrg
38 1.1 mrg #include "audiodev.h"
39 1.1 mrg #include "drvctl.h"
40 1.1 mrg
41 1.7 joerg __dead static void
42 1.8.2.3 martin usage(void)
43 1.1 mrg {
44 1.8.2.3 martin const char *p = getprogname();
45 1.8.2.3 martin
46 1.8 isaki fprintf(stderr, "usage: %s list [<index>]\n", p);
47 1.3 wiz fprintf(stderr, " %s default <index>\n", p);
48 1.8 isaki fprintf(stderr, " %s set <index> [p|r] <enc> <prec> <ch> <freq>\n",
49 1.8 isaki p);
50 1.3 wiz fprintf(stderr, " %s test <index>\n", p);
51 1.1 mrg exit(EXIT_FAILURE);
52 1.1 mrg }
53 1.1 mrg
54 1.8 isaki const char *encoding_names[] = {
55 1.8 isaki "none",
56 1.8 isaki AudioEmulaw,
57 1.8 isaki AudioEalaw,
58 1.8 isaki "pcm16",
59 1.8 isaki "pcm8",
60 1.8 isaki AudioEadpcm,
61 1.8 isaki AudioEslinear_le,
62 1.8 isaki AudioEslinear_be,
63 1.8 isaki AudioEulinear_le,
64 1.8 isaki AudioEulinear_be,
65 1.8 isaki AudioEslinear,
66 1.8 isaki AudioEulinear,
67 1.8 isaki AudioEmpeg_l1_stream,
68 1.8 isaki AudioEmpeg_l1_packets,
69 1.8 isaki AudioEmpeg_l1_system,
70 1.8 isaki AudioEmpeg_l2_stream,
71 1.8 isaki AudioEmpeg_l2_packets,
72 1.8 isaki AudioEmpeg_l2_system,
73 1.8 isaki AudioEac3,
74 1.8 isaki };
75 1.8 isaki u_int encoding_max = __arraycount(encoding_names);
76 1.8 isaki
77 1.6 jmcneill static void
78 1.6 jmcneill print_audiodev(struct audiodev *adev, int i)
79 1.6 jmcneill {
80 1.8 isaki struct audiofmt *f;
81 1.8 isaki int j;
82 1.8 isaki
83 1.6 jmcneill assert(adev != NULL);
84 1.6 jmcneill
85 1.6 jmcneill printf("%u: [%c] %s @ %s: ",
86 1.6 jmcneill i, adev->defaultdev ? '*' : ' ',
87 1.6 jmcneill adev->xname, adev->pxname);
88 1.6 jmcneill printf("%s", adev->audio_device.name);
89 1.6 jmcneill if (strlen(adev->audio_device.version) > 0)
90 1.6 jmcneill printf(" %s", adev->audio_device.version);
91 1.8 isaki printf("\n");
92 1.8 isaki printf(" playback: ");
93 1.8.2.2 martin if ((adev->hwinfo.mode & AUMODE_PLAY)) {
94 1.8 isaki printf("%uch, %uHz\n",
95 1.8.2.2 martin adev->hwinfo.play.channels,
96 1.8.2.2 martin adev->hwinfo.play.sample_rate);
97 1.8.2.2 martin } else {
98 1.8 isaki printf("unavailable\n");
99 1.8.2.2 martin }
100 1.8 isaki printf(" record: ");
101 1.8.2.2 martin if ((adev->hwinfo.mode & AUMODE_RECORD)) {
102 1.8 isaki printf("%uch, %uHz\n",
103 1.8.2.2 martin adev->hwinfo.record.channels,
104 1.8.2.2 martin adev->hwinfo.record.sample_rate);
105 1.8.2.2 martin } else {
106 1.8 isaki printf("unavailable\n");
107 1.8.2.2 martin }
108 1.8 isaki
109 1.8 isaki TAILQ_FOREACH(f, &adev->formats, next) {
110 1.8 isaki printf(" ");
111 1.8 isaki if (f->fmt.priority < 0)
112 1.8 isaki printf("( ) ");
113 1.8 isaki else if ((f->fmt.mode & (AUMODE_PLAY | AUMODE_RECORD))
114 1.8 isaki == (AUMODE_PLAY | AUMODE_RECORD))
115 1.8 isaki printf("(PR) ");
116 1.8 isaki else if ((f->fmt.mode & AUMODE_PLAY))
117 1.8 isaki printf("(P-) ");
118 1.8 isaki else if ((f->fmt.mode & AUMODE_RECORD))
119 1.8 isaki printf("(-R) ");
120 1.8 isaki
121 1.8 isaki if (f->fmt.encoding < encoding_max)
122 1.8 isaki printf("%s", encoding_names[f->fmt.encoding]);
123 1.8 isaki else
124 1.8 isaki printf("unknown_encoding_%d", f->fmt.encoding);
125 1.8 isaki printf(" %d/%d, %dch, ",
126 1.8 isaki f->fmt.validbits,
127 1.8 isaki f->fmt.precision,
128 1.8 isaki f->fmt.channels);
129 1.8 isaki if (f->fmt.frequency_type == 0) {
130 1.8 isaki printf("%d-%dHz",
131 1.8 isaki f->fmt.frequency[0],
132 1.8 isaki f->fmt.frequency[1]);
133 1.8 isaki } else {
134 1.8 isaki for (j = 0; j < (int)f->fmt.frequency_type; j++) {
135 1.8 isaki printf("%s%d",
136 1.8 isaki (j == 0) ? "{ " : ", ",
137 1.8 isaki f->fmt.frequency[j]);
138 1.8 isaki }
139 1.8 isaki printf(" }");
140 1.8 isaki }
141 1.8 isaki printf("\n");
142 1.8 isaki }
143 1.6 jmcneill }
144 1.6 jmcneill
145 1.1 mrg int
146 1.1 mrg main(int argc, char *argv[])
147 1.1 mrg {
148 1.1 mrg struct audiodev *adev;
149 1.1 mrg unsigned int n, i;
150 1.8 isaki unsigned int j;
151 1.8 isaki const char *enc;
152 1.8 isaki unsigned int prec;
153 1.8 isaki unsigned int ch;
154 1.8 isaki unsigned int freq;
155 1.8 isaki int mode;
156 1.1 mrg
157 1.1 mrg if (audiodev_refresh() == -1)
158 1.1 mrg return EXIT_FAILURE;
159 1.1 mrg
160 1.2 jmcneill if (argc < 2)
161 1.8.2.3 martin usage();
162 1.2 jmcneill /* NOTREACHED */
163 1.2 jmcneill
164 1.8 isaki if (strcmp(argv[1], "list") == 0 && argc == 2) {
165 1.1 mrg n = audiodev_count();
166 1.6 jmcneill for (i = 0; i < n; i++)
167 1.6 jmcneill print_audiodev(audiodev_get(i), i);
168 1.8 isaki } else if (strcmp(argv[1], "list") == 0 && argc == 3) {
169 1.8 isaki errno = 0;
170 1.8 isaki i = strtoul(argv[2], NULL, 10);
171 1.8 isaki if (errno)
172 1.8.2.3 martin usage();
173 1.8 isaki /* NOTREACHED */
174 1.8.2.1 martin adev = audiodev_get(i);
175 1.8.2.1 martin if (adev == NULL) {
176 1.8.2.3 martin errx(EXIT_FAILURE, "no such device");
177 1.8.2.1 martin }
178 1.8.2.1 martin print_audiodev(adev, i);
179 1.2 jmcneill } else if (strcmp(argv[1], "default") == 0 && argc == 3) {
180 1.2 jmcneill if (*argv[2] < '0' || *argv[2] > '9')
181 1.8.2.3 martin usage();
182 1.1 mrg /* NOTREACHED */
183 1.1 mrg errno = 0;
184 1.2 jmcneill i = strtoul(argv[2], NULL, 10);
185 1.1 mrg if (errno)
186 1.8.2.3 martin usage();
187 1.1 mrg /* NOTREACHED */
188 1.1 mrg adev = audiodev_get(i);
189 1.1 mrg if (adev == NULL) {
190 1.8.2.3 martin errx(EXIT_FAILURE, "no such device");
191 1.1 mrg }
192 1.1 mrg printf("setting default audio device to %s\n", adev->xname);
193 1.1 mrg if (audiodev_set_default(adev) == -1) {
194 1.8.2.3 martin errx(EXIT_FAILURE, "couldn't set default device");
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.2.3 martin usage();
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.2.3 martin usage();
206 1.8 isaki /* NOTREACHED */
207 1.8 isaki adev = audiodev_get(i);
208 1.8 isaki if (adev == NULL) {
209 1.8.2.3 martin errx(EXIT_FAILURE, "no such device");
210 1.8 isaki }
211 1.8 isaki
212 1.8 isaki mode = 0;
213 1.8 isaki for (j = 0; j < strlen(argv[3]); j++) {
214 1.8 isaki if (argv[3][j] == 'p')
215 1.8 isaki mode |= AUMODE_PLAY;
216 1.8 isaki else if (argv[3][j] == 'r')
217 1.8 isaki mode |= AUMODE_RECORD;
218 1.8 isaki else
219 1.8.2.3 martin usage();
220 1.8 isaki }
221 1.8 isaki enc = argv[4];
222 1.8 isaki prec = strtoul(argv[5], NULL, 10);
223 1.8 isaki if (errno)
224 1.8.2.3 martin usage();
225 1.8 isaki errno = 0;
226 1.8 isaki ch = strtoul(argv[6], NULL, 10);
227 1.8 isaki if (errno)
228 1.8.2.3 martin usage();
229 1.8 isaki /* NOTREACHED */
230 1.8 isaki errno = 0;
231 1.8 isaki freq = strtoul(argv[7], NULL, 10);
232 1.8 isaki if (errno)
233 1.8.2.3 martin usage();
234 1.8 isaki /* NOTREACHED */
235 1.8 isaki
236 1.8 isaki if (audiodev_set_param(adev, mode, enc, prec, ch, freq) == -1) {
237 1.8.2.3 martin errx(EXIT_FAILURE, "couldn't set parameter");
238 1.8 isaki }
239 1.2 jmcneill } else if (strcmp(argv[1], "test") == 0 && argc == 3) {
240 1.2 jmcneill if (*argv[2] < '0' || *argv[2] > '9')
241 1.8.2.3 martin usage();
242 1.2 jmcneill /* NOTREACHED */
243 1.2 jmcneill errno = 0;
244 1.2 jmcneill i = strtoul(argv[2], NULL, 10);
245 1.2 jmcneill if (errno)
246 1.8.2.3 martin usage();
247 1.2 jmcneill /* NOTREACHED */
248 1.2 jmcneill adev = audiodev_get(i);
249 1.2 jmcneill if (adev == NULL) {
250 1.8.2.3 martin errx(EXIT_FAILURE, "no such device");
251 1.2 jmcneill }
252 1.6 jmcneill print_audiodev(adev, i);
253 1.8.2.2 martin if (audiodev_test(adev) == -1)
254 1.8.2.2 martin return EXIT_FAILURE;
255 1.2 jmcneill } else
256 1.8.2.3 martin usage();
257 1.1 mrg /* NOTREACHED */
258 1.1 mrg
259 1.1 mrg return EXIT_SUCCESS;
260 1.1 mrg }
261