Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/usr.bin/audiocfg Update respond to isaki-audio2 branch.



details:   https://anonhg.NetBSD.org/src/rev/11b59eb38fc4
branches:  trunk
changeset: 998938:11b59eb38fc4
user:      isaki <isaki%NetBSD.org@localhost>
date:      Wed May 08 14:36:12 2019 +0000

description:
Update respond to isaki-audio2 branch.
- Extend list command to display supported hardware formats.
- Add set command to set hardware format.
- Use correct /dev/audioctl instead of /dev/audio.

diffstat:

 usr.bin/audiocfg/audiocfg.1 |   29 ++++++-
 usr.bin/audiocfg/audiodev.c |  178 ++++++++++++++++++++++++++++++-------------
 usr.bin/audiocfg/audiodev.h |   18 +++-
 usr.bin/audiocfg/main.c     |  144 ++++++++++++++++++++++++++++++++++-
 4 files changed, 304 insertions(+), 65 deletions(-)

diffs (truncated from 591 to 300 lines):

diff -r 430cef4e86ca -r 11b59eb38fc4 usr.bin/audiocfg/audiocfg.1
--- a/usr.bin/audiocfg/audiocfg.1       Wed May 08 14:25:38 2019 +0000
+++ b/usr.bin/audiocfg/audiocfg.1       Wed May 08 14:36:12 2019 +0000
@@ -1,4 +1,4 @@
-.\"    $NetBSD: audiocfg.1,v 1.3 2010/09/01 09:17:31 wiz Exp $
+.\"    $NetBSD: audiocfg.1,v 1.4 2019/05/08 14:36:12 isaki Exp $
 .\"
 .\" Copyright (c) 2010 The NetBSD Foundation, Inc.
 .\" All rights reserved.
@@ -40,6 +40,14 @@
 .Cm default
 .Ar index
 .Nm
+.Cm set
+.Ar index
+.Ar direction
+.Ar encoding
+.Ar precision
+.Ar channels
+.Ar sample_rate
+.Nm
 .Cm test
 .Ar index
 .Sh DESCRIPTION
@@ -51,6 +59,25 @@
 .Dq Cm list ,
 it lists the available audio devices and shows the currently selected
 default audio device.
+.Pp
+Run with
+.Dq Cm set ,
+if sets the hardware format.
+The
+.Ar direction
+is represented by either of 'p'(playback) or 'r'(record), or both of them,
+indicates direction you want to set to.
+The remaining parameters
+.Ar encoding ,
+.Ar precision ,
+.Ar channels
+and
+.Ar sample_rate
+indicate the hardware format you want to set to.
+These parameters must be selected from the candidates displayed by
+.Nm
+.Cm list .
+.Pp
 Called with
 .Dq Cm test ,
 it plays a tone of 2 seconds for each channel of the device with the index
diff -r 430cef4e86ca -r 11b59eb38fc4 usr.bin/audiocfg/audiodev.c
--- a/usr.bin/audiocfg/audiodev.c       Wed May 08 14:25:38 2019 +0000
+++ b/usr.bin/audiocfg/audiodev.c       Wed May 08 14:36:12 2019 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: audiodev.c,v 1.6 2016/03/05 22:10:39 mrg Exp $ */
+/* $NetBSD: audiodev.c,v 1.7 2019/05/08 14:36:12 isaki Exp $ */
 
 /*
  * Copyright (c) 2010 Jared D. McNeill <jmcneill%invisible.ca@localhost>
@@ -31,6 +31,7 @@
 #include <sys/stat.h>
 #include <sys/drvctlio.h>
 
+#include <errno.h>
 #include <fcntl.h>
 #include <paths.h>
 #include <stdio.h>
@@ -47,51 +48,23 @@
 
 #define AUDIODEV_SAMPLE_RATE   44100
 
-static unsigned int
-audiodev_probe_pchans(struct audiodev *adev)
-{
-       audio_info_t info;
-       unsigned int nchans = 0, n;
-       int error;
-
-       AUDIO_INITINFO(&info);
-       info.play.sample_rate = AUDIODEV_SAMPLE_RATE;
-       info.play.precision = 16;
-       info.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
-       info.play.channels = 1;
-       info.mode = AUMODE_PLAY;
-       error = ioctl(adev->fd, AUDIO_SETINFO, &info);
-       if (error == -1)
-               return 0;
-       nchans = 1;
-
-       for (n = 2; n <= 16; n += 2) {
-               info.play.channels = n;
-               error = ioctl(adev->fd, AUDIO_SETINFO, &info);
-               if (error == -1)
-                       break;
-               nchans = info.play.channels;
-       }
-
-       return nchans;
-}
-
 static int
 audiodev_getinfo(struct audiodev *adev)
 {
        struct stat st;
+       struct audiofmt *f;
+       audio_format_query_t query;
+       int i;
 
-       if (stat(adev->path, &st) == -1)
+       if (stat(adev->ctlpath, &st) == -1)
                return -1;
        adev->dev = st.st_rdev;
 
-       if (stat(_PATH_AUDIO, &st) != -1 && st.st_rdev == adev->dev)
+       if (stat(_PATH_AUDIOCTL, &st) != -1 && st.st_rdev == adev->dev)
                adev->defaultdev = true;
 
-       adev->fd = open(adev->path, O_RDWR);
+       adev->fd = open(adev->ctlpath, O_RDONLY);
        if (adev->fd == -1) {
-               adev->fd = open(adev->path, O_WRONLY);
-               if (adev->fd == -1)
                        return -1;
        }
        if (ioctl(adev->fd, AUDIO_GETDEV, &adev->audio_device) == -1) {
@@ -99,7 +72,29 @@
                return -1;
        }
 
-       adev->pchan = audiodev_probe_pchans(adev);
+       for (i = 0; ;i++) {
+               memset(&query, 0, sizeof(query));
+               query.index = i;
+               if (ioctl(adev->fd, AUDIO_QUERYFORMAT, &query) == -1) {
+                       if (errno == ENODEV) {
+                               /* QUERYFORMAT not supported. */
+                               break;
+                       }
+                       if (errno == EINVAL)
+                               break;
+                       close(adev->fd);
+                       return -1;
+               }
+
+               f = calloc(1, sizeof(*f));
+               f->fmt = query.fmt;
+               TAILQ_INSERT_TAIL(&adev->formats, f, next);
+       }
+
+       if (ioctl(adev->fd, AUDIO_GETFORMAT, &adev->info) == -1) {
+               close(adev->fd);
+               return -1;
+       }
 
        return 0;
 }
@@ -115,8 +110,10 @@
 
        strlcpy(adev->pxname, pdev, sizeof(adev->pxname));
        strlcpy(adev->xname, dev, sizeof(adev->xname));
-       snprintf(adev->path, sizeof(adev->path) - 1, "/dev/%s", dev);
+       snprintf(adev->path, sizeof(adev->path), "/dev/%s", dev);
+       snprintf(adev->ctlpath, sizeof(adev->ctlpath), "/dev/audioctl%d", unit);
        adev->unit = unit;
+       TAILQ_INIT(&adev->formats);
 
        if (audiodev_getinfo(adev) == -1) {
                free(adev);
@@ -124,8 +121,16 @@
        }
 
 #ifdef DEBUG
-       printf("[%c] %s: %s\n", adev->defaultdev ? '*' : ' ',
-           adev->path, adev->audio_device.name);
+       printf("DEBUG: [%c] %s(%s): %s\n", adev->defaultdev ? '*' : ' ',
+           adev->path, adev->ctlpath, adev->audio_device.name);
+       struct audiofmt *f;
+       TAILQ_FOREACH(f, &adev->formats, next) {
+               printf("DEBUG: enc%d, %d/%d, %dch\n",
+                   f->fmt.encoding,
+                   f->fmt.validbits,
+                   f->fmt.precision,
+                   f->fmt.channels);
+       }
 #endif
 
        TAILQ_INSERT_TAIL(&audiodevlist, adev, next);
@@ -207,13 +212,13 @@
        char audioctlpath[PATH_MAX+1];
        char mixerpath[PATH_MAX+1];
 
-       snprintf(audiopath, sizeof(audiopath) - 1,
+       snprintf(audiopath, sizeof(audiopath),
            _PATH_AUDIO "%u", adev->unit);
-       snprintf(soundpath, sizeof(soundpath) - 1,
+       snprintf(soundpath, sizeof(soundpath),
            _PATH_SOUND "%u", adev->unit);
-       snprintf(audioctlpath, sizeof(audioctlpath) - 1,
+       snprintf(audioctlpath, sizeof(audioctlpath),
            _PATH_AUDIOCTL "%u", adev->unit);
-       snprintf(mixerpath, sizeof(mixerpath) - 1,
+       snprintf(mixerpath, sizeof(mixerpath),
            _PATH_MIXER "%u", adev->unit);
 
        unlink(_PATH_AUDIO);
@@ -242,33 +247,92 @@
 }
 
 int
+audiodev_set_param(struct audiodev *adev, int mode,
+       const char *encname, unsigned int prec, unsigned int ch, unsigned int freq)
+{
+       struct audio_info ai;
+       int setmode;
+       u_int enc;
+
+       setmode = 0;
+       ai = adev->info;
+
+       for (enc = 0; enc < encoding_max; enc++) {
+               if (strcmp(encname, encoding_names[enc]) == 0)
+                       break;
+       }
+       if (enc >= encoding_max) {
+               fprintf(stderr, "unknown encoding name: %s\n", encname);
+               errno = EINVAL;
+               return -1;
+       }
+
+       if ((ai.mode & mode & AUMODE_PLAY)) {
+               setmode |= AUMODE_PLAY;
+               ai.play.encoding = enc;
+               ai.play.precision = prec;
+               ai.play.channels = ch;
+               ai.play.sample_rate = freq;
+       }
+       if ((ai.mode & mode & AUMODE_RECORD)) {
+               setmode |= AUMODE_RECORD;
+               ai.record.encoding = enc;
+               ai.record.precision = prec;
+               ai.record.channels = ch;
+               ai.record.sample_rate = freq;
+       }
+
+       if (setmode == 0) {
+               errno = EINVAL;
+               return -1;
+       }
+
+       ai.mode = setmode;
+       printf("setting %s to %s:%u, %uch, %uHz\n",
+           adev->xname, encname, prec, ch, freq);
+       if (ioctl(adev->fd, AUDIO_SETFORMAT, &ai) == -1) {
+               perror("ioctl AUDIO_SETFORMAT");
+               return -1;
+       }
+       return 0;
+}
+
+int
 audiodev_test(struct audiodev *adev, unsigned int chanmask)
 {
        audio_info_t info;
        int16_t *buf;
        size_t buflen;
        off_t off;
-       int rv = 0;
+       int fd;
+       int rv = -1;
+
+       fd = open(adev->path, O_WRONLY);
+       if (fd == -1) {
+               perror("open");
+               return -1;
+       }
 
        AUDIO_INITINFO(&info);
        info.play.sample_rate = AUDIODEV_SAMPLE_RATE;
-       info.play.channels = adev->pchan;
+       info.play.channels = adev->info.play.channels;
        info.play.precision = 16;
        info.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
        info.mode = AUMODE_PLAY;
-       if (ioctl(adev->fd, AUDIO_SETINFO, &info) == -1) {
+       if (ioctl(fd, AUDIO_SETINFO, &info) == -1) {
                perror("ioctl AUDIO_SETINFO");
-               return -1;
+               goto abort;
        }
-       if (ioctl(adev->fd, AUDIO_GETINFO, &info) == -1) {
+       if (ioctl(fd, AUDIO_GETINFO, &info) == -1) {
                perror("ioctl AUDIO_GETINFO");
-               return -1;
+               goto abort;
        }
 
        dtmf_new(&buf, &buflen, info.play.sample_rate, 2,
-           adev->pchan, chanmask, 350.0, 440.0);
-       if (buf == NULL)
-               return -1;



Home | Main Index | Thread Index | Old Index