1*ca400598Smlelstv /* $NetBSD: main.c,v 1.17 2023/04/01 12:41:02 mlelstv Exp $ */
2172ca3ccSmrg
3172ca3ccSmrg /*
4172ca3ccSmrg * Copyright (c) 2010 Jared D. McNeill <jmcneill@invisible.ca>
5172ca3ccSmrg * All rights reserved.
6172ca3ccSmrg *
7172ca3ccSmrg * Redistribution and use in source and binary forms, with or without
8172ca3ccSmrg * modification, are permitted provided that the following conditions
9172ca3ccSmrg * are met:
10172ca3ccSmrg * 1. Redistributions of source code must retain the above copyright
11172ca3ccSmrg * notice, this list of conditions and the following disclaimer.
12172ca3ccSmrg * 2. Redistributions in binary form must reproduce the above copyright
13172ca3ccSmrg * notice, this list of conditions and the following disclaimer in the
14172ca3ccSmrg * documentation and/or other materials provided with the distribution.
15172ca3ccSmrg *
16172ca3ccSmrg * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
17172ca3ccSmrg * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
18172ca3ccSmrg * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19172ca3ccSmrg * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
20172ca3ccSmrg * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21172ca3ccSmrg * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22172ca3ccSmrg * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23172ca3ccSmrg * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24172ca3ccSmrg * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25172ca3ccSmrg * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26172ca3ccSmrg * POSSIBILITY OF SUCH DAMAGE.
27172ca3ccSmrg */
28172ca3ccSmrg
29d148107cSjmcneill #include <assert.h>
3016e14080Sisaki #include <err.h>
31172ca3ccSmrg #include <errno.h>
32172ca3ccSmrg #include <fcntl.h>
33172ca3ccSmrg #include <limits.h>
34172ca3ccSmrg #include <stdio.h>
35172ca3ccSmrg #include <stdlib.h>
36172ca3ccSmrg #include <unistd.h>
37172ca3ccSmrg
38172ca3ccSmrg #include "audiodev.h"
39172ca3ccSmrg #include "drvctl.h"
40172ca3ccSmrg
412c7fa373Sjoerg __dead static void
usage(void)4216e14080Sisaki usage(void)
43172ca3ccSmrg {
4416e14080Sisaki const char *p = getprogname();
4516e14080Sisaki
460b88ff99Sisaki fprintf(stderr, "usage: %s list [<index>]\n", p);
477a280120Swiz fprintf(stderr, " %s default <index>\n", p);
480b88ff99Sisaki fprintf(stderr, " %s set <index> [p|r] <enc> <prec> <ch> <freq>\n",
490b88ff99Sisaki p);
507a280120Swiz fprintf(stderr, " %s test <index>\n", p);
51172ca3ccSmrg exit(EXIT_FAILURE);
52172ca3ccSmrg }
53172ca3ccSmrg
540b88ff99Sisaki const char *encoding_names[] = {
550b88ff99Sisaki "none",
560b88ff99Sisaki AudioEmulaw,
570b88ff99Sisaki AudioEalaw,
580b88ff99Sisaki "pcm16",
590b88ff99Sisaki "pcm8",
600b88ff99Sisaki AudioEadpcm,
610b88ff99Sisaki AudioEslinear_le,
620b88ff99Sisaki AudioEslinear_be,
630b88ff99Sisaki AudioEulinear_le,
640b88ff99Sisaki AudioEulinear_be,
650b88ff99Sisaki AudioEslinear,
660b88ff99Sisaki AudioEulinear,
670b88ff99Sisaki AudioEmpeg_l1_stream,
680b88ff99Sisaki AudioEmpeg_l1_packets,
690b88ff99Sisaki AudioEmpeg_l1_system,
700b88ff99Sisaki AudioEmpeg_l2_stream,
710b88ff99Sisaki AudioEmpeg_l2_packets,
720b88ff99Sisaki AudioEmpeg_l2_system,
730b88ff99Sisaki AudioEac3,
740b88ff99Sisaki };
750b88ff99Sisaki u_int encoding_max = __arraycount(encoding_names);
760b88ff99Sisaki
77d148107cSjmcneill static void
print_audiodev(struct audiodev * adev,int i)783dbad286Sisaki print_audiodev(struct audiodev *adev, int i)
79d148107cSjmcneill {
800b88ff99Sisaki struct audiofmt *f;
810b88ff99Sisaki int j;
820b88ff99Sisaki
83d148107cSjmcneill assert(adev != NULL);
84d148107cSjmcneill
853dbad286Sisaki printf("%u: [%c] %s @ %s: ",
863dbad286Sisaki i, adev->defaultdev ? '*' : ' ',
87d148107cSjmcneill adev->xname, adev->pxname);
88d148107cSjmcneill printf("%s", adev->audio_device.name);
89d148107cSjmcneill if (strlen(adev->audio_device.version) > 0)
90d148107cSjmcneill printf(" %s", adev->audio_device.version);
910b88ff99Sisaki printf("\n");
920b88ff99Sisaki printf(" playback: ");
93923f5af2Sisaki if ((adev->hwinfo.mode & AUMODE_PLAY)) {
94*ca400598Smlelstv printf("%u, %uch, %uHz\n",
95*ca400598Smlelstv adev->hwinfo.play.precision,
96923f5af2Sisaki adev->hwinfo.play.channels,
97923f5af2Sisaki adev->hwinfo.play.sample_rate);
98923f5af2Sisaki } else {
990b88ff99Sisaki printf("unavailable\n");
100923f5af2Sisaki }
1010b88ff99Sisaki printf(" record: ");
102923f5af2Sisaki if ((adev->hwinfo.mode & AUMODE_RECORD)) {
103*ca400598Smlelstv printf("%u, %uch, %uHz\n",
104*ca400598Smlelstv adev->hwinfo.record.precision,
105923f5af2Sisaki adev->hwinfo.record.channels,
106923f5af2Sisaki adev->hwinfo.record.sample_rate);
107923f5af2Sisaki } else {
1080b88ff99Sisaki printf("unavailable\n");
109923f5af2Sisaki }
1100b88ff99Sisaki
1110b88ff99Sisaki TAILQ_FOREACH(f, &adev->formats, next) {
1120b88ff99Sisaki printf(" ");
1130b88ff99Sisaki if (f->fmt.priority < 0)
1140b88ff99Sisaki printf("( ) ");
1150b88ff99Sisaki else if ((f->fmt.mode & (AUMODE_PLAY | AUMODE_RECORD))
1160b88ff99Sisaki == (AUMODE_PLAY | AUMODE_RECORD))
1170b88ff99Sisaki printf("(PR) ");
1180b88ff99Sisaki else if ((f->fmt.mode & AUMODE_PLAY))
1190b88ff99Sisaki printf("(P-) ");
1200b88ff99Sisaki else if ((f->fmt.mode & AUMODE_RECORD))
1210b88ff99Sisaki printf("(-R) ");
1220b88ff99Sisaki
1230b88ff99Sisaki if (f->fmt.encoding < encoding_max)
1240b88ff99Sisaki printf("%s", encoding_names[f->fmt.encoding]);
1250b88ff99Sisaki else
1260b88ff99Sisaki printf("unknown_encoding_%d", f->fmt.encoding);
1270b88ff99Sisaki printf(" %d/%d, %dch, ",
1280b88ff99Sisaki f->fmt.validbits,
1290b88ff99Sisaki f->fmt.precision,
1300b88ff99Sisaki f->fmt.channels);
1310b88ff99Sisaki if (f->fmt.frequency_type == 0) {
1320b88ff99Sisaki printf("%d-%dHz",
1330b88ff99Sisaki f->fmt.frequency[0],
1340b88ff99Sisaki f->fmt.frequency[1]);
1350b88ff99Sisaki } else {
1360b88ff99Sisaki for (j = 0; j < (int)f->fmt.frequency_type; j++) {
1370b88ff99Sisaki printf("%s%d",
1380b88ff99Sisaki (j == 0) ? "{ " : ", ",
1390b88ff99Sisaki f->fmt.frequency[j]);
1400b88ff99Sisaki }
1410b88ff99Sisaki printf(" }");
1420b88ff99Sisaki }
1430b88ff99Sisaki printf("\n");
1440b88ff99Sisaki }
145d148107cSjmcneill }
146d148107cSjmcneill
147172ca3ccSmrg int
main(int argc,char * argv[])148172ca3ccSmrg main(int argc, char *argv[])
149172ca3ccSmrg {
150172ca3ccSmrg struct audiodev *adev;
151172ca3ccSmrg unsigned int n, i;
1520b88ff99Sisaki unsigned int j;
1530b88ff99Sisaki const char *enc;
1540b88ff99Sisaki unsigned int prec;
1550b88ff99Sisaki unsigned int ch;
1560b88ff99Sisaki unsigned int freq;
1570b88ff99Sisaki int mode;
158172ca3ccSmrg
159172ca3ccSmrg if (audiodev_refresh() == -1)
160172ca3ccSmrg return EXIT_FAILURE;
161172ca3ccSmrg
162e405ac8dSjmcneill if (argc < 2)
16316e14080Sisaki usage();
164e405ac8dSjmcneill /* NOTREACHED */
165e405ac8dSjmcneill
1660b88ff99Sisaki if (strcmp(argv[1], "list") == 0 && argc == 2) {
1673dbad286Sisaki n = audiodev_count();
1683dbad286Sisaki for (i = 0; i < n; i++)
1693dbad286Sisaki print_audiodev(audiodev_get(i), i);
1700b88ff99Sisaki } else if (strcmp(argv[1], "list") == 0 && argc == 3) {
1713dbad286Sisaki if (*argv[2] < '0' || *argv[2] > '9')
1723dbad286Sisaki usage();
1733dbad286Sisaki /* NOTREACHED */
1743dbad286Sisaki errno = 0;
1753dbad286Sisaki i = strtoul(argv[2], NULL, 10);
1763dbad286Sisaki if (errno)
1773dbad286Sisaki usage();
1783dbad286Sisaki /* NOTREACHED */
1793dbad286Sisaki adev = audiodev_get(i);
1803dbad286Sisaki if (adev == NULL) {
1813dbad286Sisaki errx(EXIT_FAILURE, "no such device");
1823dbad286Sisaki }
1833dbad286Sisaki print_audiodev(adev, i);
184e405ac8dSjmcneill } else if (strcmp(argv[1], "default") == 0 && argc == 3) {
1853dbad286Sisaki if (*argv[2] < '0' || *argv[2] > '9')
1863dbad286Sisaki usage();
1873dbad286Sisaki /* NOTREACHED */
1883dbad286Sisaki errno = 0;
1893dbad286Sisaki i = strtoul(argv[2], NULL, 10);
1903dbad286Sisaki if (errno)
1913dbad286Sisaki usage();
1923dbad286Sisaki /* NOTREACHED */
1933dbad286Sisaki adev = audiodev_get(i);
1943dbad286Sisaki if (adev == NULL) {
1953dbad286Sisaki errx(EXIT_FAILURE, "no such device");
1963dbad286Sisaki }
197172ca3ccSmrg printf("setting default audio device to %s\n", adev->xname);
198172ca3ccSmrg if (audiodev_set_default(adev) == -1) {
19916e14080Sisaki errx(EXIT_FAILURE, "couldn't set default device");
200172ca3ccSmrg }
2010b88ff99Sisaki } else if (strcmp(argv[1], "set") == 0 && argc == 8) {
2020b88ff99Sisaki /* XXX bad commandline... */
2030b88ff99Sisaki /* audiocfg set <index> [p|r] <enc> <prec> <ch> <freq> */
2043dbad286Sisaki if (*argv[2] < '0' || *argv[2] > '9')
2053dbad286Sisaki usage();
2063dbad286Sisaki /* NOTREACHED */
2073dbad286Sisaki errno = 0;
2083dbad286Sisaki i = strtoul(argv[2], NULL, 10);
2093dbad286Sisaki if (errno)
2103dbad286Sisaki usage();
2113dbad286Sisaki /* NOTREACHED */
2123dbad286Sisaki adev = audiodev_get(i);
2133dbad286Sisaki if (adev == NULL) {
2143dbad286Sisaki errx(EXIT_FAILURE, "no such device");
2153dbad286Sisaki }
2163dbad286Sisaki
2170b88ff99Sisaki mode = 0;
2180b88ff99Sisaki for (j = 0; j < strlen(argv[3]); j++) {
2190b88ff99Sisaki if (argv[3][j] == 'p')
2200b88ff99Sisaki mode |= AUMODE_PLAY;
2210b88ff99Sisaki else if (argv[3][j] == 'r')
2220b88ff99Sisaki mode |= AUMODE_RECORD;
2230b88ff99Sisaki else
22416e14080Sisaki usage();
2250b88ff99Sisaki }
2262116fa4fSisaki if (mode == 0)
22716e14080Sisaki usage();
2282116fa4fSisaki /* NOTREACHED */
2290b88ff99Sisaki enc = argv[4];
2300b88ff99Sisaki prec = strtoul(argv[5], NULL, 10);
2310b88ff99Sisaki if (errno)
23216e14080Sisaki usage();
2330b88ff99Sisaki errno = 0;
2340b88ff99Sisaki ch = strtoul(argv[6], NULL, 10);
2350b88ff99Sisaki if (errno)
23616e14080Sisaki usage();
2370b88ff99Sisaki /* NOTREACHED */
2380b88ff99Sisaki errno = 0;
2390b88ff99Sisaki freq = strtoul(argv[7], NULL, 10);
2400b88ff99Sisaki if (errno)
24116e14080Sisaki usage();
2420b88ff99Sisaki /* NOTREACHED */
2430b88ff99Sisaki
2440b88ff99Sisaki if (audiodev_set_param(adev, mode, enc, prec, ch, freq) == -1) {
24516e14080Sisaki errx(EXIT_FAILURE, "couldn't set parameter");
2460b88ff99Sisaki }
247e405ac8dSjmcneill } else if (strcmp(argv[1], "test") == 0 && argc == 3) {
2483dbad286Sisaki if (*argv[2] < '0' || *argv[2] > '9')
2493dbad286Sisaki usage();
2503dbad286Sisaki /* NOTREACHED */
2513dbad286Sisaki errno = 0;
2523dbad286Sisaki i = strtoul(argv[2], NULL, 10);
2533dbad286Sisaki if (errno)
2543dbad286Sisaki usage();
2553dbad286Sisaki /* NOTREACHED */
2563dbad286Sisaki adev = audiodev_get(i);
2573dbad286Sisaki if (adev == NULL) {
2583dbad286Sisaki errx(EXIT_FAILURE, "no such device");
2593dbad286Sisaki }
2603dbad286Sisaki print_audiodev(adev, i);
261ab49aa4fSisaki if (audiodev_test(adev) == -1)
262d148107cSjmcneill return EXIT_FAILURE;
263e405ac8dSjmcneill } else
26416e14080Sisaki usage();
265e405ac8dSjmcneill /* NOTREACHED */
266172ca3ccSmrg
267172ca3ccSmrg return EXIT_SUCCESS;
268172ca3ccSmrg }
269