/*      $NetBSD: audiotest.c,v 1.33 2025/06/27 19:52:03 andvar Exp $    */

/*
* Copyright (C) 2019 Tetsuya Isaki. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
*    notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
*    notice, this list of conditions and the following disclaimer in the
*    documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/

#include <sys/cdefs.h>
__RCSID("$NetBSD: audiotest.c,v 1.33 2025/06/27 19:52:03 andvar Exp $");

#include <errno.h>
#include <fcntl.h>
#define __STDC_FORMAT_MACROS    /* for PRIx64 */
#include <inttypes.h>
#include <pthread.h>
#include <stdarg.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <util.h>
#include <sys/audioio.h>
#include <sys/event.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
#include <sys/poll.h>
#include <sys/sysctl.h>
#include <sys/time.h>
#include <sys/wait.h>
#if !defined(NO_RUMP)
#include <rump/rump.h>
#include <rump/rump_syscalls.h>
#endif

/* this internal driver option is not exported to userland */
#define AUDIO_SUPPORT_LINEAR24

#if !defined(AUDIO_ENCODING_SLINEAR_NE)
#if BYTE_ORDER == LITTLE_ENDIAN
#define AUDIO_ENCODING_SLINEAR_NE AUDIO_ENCODING_SLINEAR_LE
#define AUDIO_ENCODING_ULINEAR_NE AUDIO_ENCODING_ULINEAR_LE
#define AUDIO_ENCODING_SLINEAR_OE AUDIO_ENCODING_SLINEAR_BE
#define AUDIO_ENCODING_ULINEAR_OE AUDIO_ENCODING_ULINEAR_BE
#else
#define AUDIO_ENCODING_SLINEAR_NE AUDIO_ENCODING_SLINEAR_BE
#define AUDIO_ENCODING_ULINEAR_NE AUDIO_ENCODING_ULINEAR_BE
#define AUDIO_ENCODING_SLINEAR_OE AUDIO_ENCODING_SLINEAR_LE
#define AUDIO_ENCODING_ULINEAR_OE AUDIO_ENCODING_ULINEAR_LE
#endif
#endif

struct testentry {
       const char *name;
       void (*func)(void);
};

void usage(void) __dead;
void xp_err(int, int, const char *, ...) __printflike(3, 4) __dead;
void xp_errx(int, int, const char *, ...) __printflike(3, 4) __dead;
bool match(const char *, const char *);
void xxx_close_wait(void);
int mixer_get_outputs_master(int);
void do_test(int);
int rump_or_open(const char *, int);
int rump_or_write(int, const void *, size_t);
int rump_or_read(int, void *, size_t);
int rump_or_ioctl(int, u_long, void *);
int rump_or_close(int);
int rump_or_fcntl(int, int, ...);
int rump_or_poll(struct pollfd *, nfds_t, int);
int rump_or_kqueue(void);
int rump_or_kevent(int, const struct kevent *, size_t,
       struct kevent *, size_t, const struct timespec *);
int hw_canplay(void);
int hw_canrec(void);
int hw_bidir(void);
int hw_fulldup(void);
void init(int);
void *consumer_thread(void *);
void cleanup_audiofd(void);
void TEST(const char *, ...) __printflike(1, 2);
bool xp_fail(int, const char *, ...) __printflike(2, 3);
void xp_skip(int, const char *, ...) __printflike(2, 3);
bool xp_eq(int, int, int, const char *);
bool xp_eq_str(int, const char *, const char *, const char *);
bool xp_ne(int, int, int, const char *);
bool xp_if(int, bool, const char *);
bool xp_sys_eq(int, int, int, const char *);
bool xp_sys_ok(int, int, const char *);
bool xp_sys_ng(int, int, int, const char *);
bool xp_sys_ptr(int, int, void *, const char *);
int debug_open(int, const char *, int);
int debug_write(int, int, const void *, size_t);
int debug_read(int, int, void *, size_t);
int debug_ioctl(int, int, u_long, const char *, void *, const char *, ...)
       __printflike(6, 7);
int debug_fcntl(int, int, int, const char *, ...) __printflike(4, 5);
int debug_close(int, int);
void *debug_mmap(int, void *, size_t, int, int, int, off_t);
int debug_munmap(int, void *, int);
const char *event_tostr(int);
int debug_poll(int, struct pollfd *, int, int);
int debug_kqueue(int);
int debug_kevent_set(int, int, const struct kevent *, size_t);
int debug_kevent_poll(int, int, struct kevent *, size_t,
       const struct timespec *);
void debug_kev(int, const char *, const struct kevent *);
uid_t debug_getuid(int);
int debug_seteuid(int, uid_t);
int debug_sysctlbyname(int, const char *, void *, size_t *, const void *,
       size_t);

int openable_mode(void);
int mode2aumode(int);
int mode2play(int);
int mode2rec(int);
void reset_after_mmap(void);

/* from audio.c */
static const char *encoding_names[] __unused = {
       "none",
       AudioEmulaw,
       AudioEalaw,
       "pcm16",
       "pcm8",
       AudioEadpcm,
       AudioEslinear_le,
       AudioEslinear_be,
       AudioEulinear_le,
       AudioEulinear_be,
       AudioEslinear,
       AudioEulinear,
       AudioEmpeg_l1_stream,
       AudioEmpeg_l1_packets,
       AudioEmpeg_l1_system,
       AudioEmpeg_l2_stream,
       AudioEmpeg_l2_packets,
       AudioEmpeg_l2_system,
       AudioEac3,
};

int debug;
int props;
int hwfull;
int netbsd;
bool opt_atf;
char testname[64];
int testcount;
int failcount;
int skipcount;
int unit;
bool use_rump;
bool use_pad;
bool exact_match;
int padfd;
int maxfd;
pthread_t th;
char devicename[16];    /* "audioN" */
char devaudio[16];      /* "/dev/audioN" */
char devsound[16];      /* "/dev/soundN" */
char devaudioctl[16];   /* "/dev/audioctlN" */
char devmixer[16];      /* "/dev/mixerN" */
extern struct testentry testtable[];

void
usage(void)
{
       fprintf(stderr, "usage:\t%s [<options>] [<testname>...]\n",
           getprogname());
       fprintf(stderr, "\t-A        : make output suitable for ATF\n");
       fprintf(stderr, "\t-a        : Test all\n");
       fprintf(stderr, "\t-d        : Increase debug level\n");
       fprintf(stderr, "\t-e        : Use exact match for testnames "
           "(default is forward match)\n");
       fprintf(stderr, "\t-l        : List all tests\n");
       fprintf(stderr, "\t-p        : Open pad\n");
#if !defined(NO_RUMP)
       fprintf(stderr, "\t-R        : Use rump (implies -p)\n");
#endif
       fprintf(stderr, "\t-u <unit> : Use audio<unit> (default:0)\n");
       exit(1);
}

/* Customized err(3) */
void
xp_err(int code, int line, const char *fmt, ...)
{
       va_list ap;
       int backup_errno;

       backup_errno = errno;
       printf("%s %d: ", (opt_atf ? "Line" : " ERROR:"), line);
       va_start(ap, fmt);
       vprintf(fmt, ap);
       va_end(ap);
       printf(": %s\n", strerror(backup_errno));

       exit(code);
}

/* Customized errx(3) */
void
xp_errx(int code, int line, const char *fmt, ...)
{
       va_list ap;

       printf("%s %d: ", (opt_atf ? "Line" : " ERROR:"), line);
       va_start(ap, fmt);
       vprintf(fmt, ap);
       va_end(ap);
       printf("\n");

       exit(code);
}

int
main(int argc, char *argv[])
{
       int i;
       int j;
       int c;
       enum {
               CMD_TEST,
               CMD_ALL,
               CMD_LIST,
       } cmd;
       bool found;

       props = -1;
       hwfull = 0;
       unit = 0;
       cmd = CMD_TEST;
       use_pad = false;
       padfd = -1;
       exact_match = false;

       while ((c = getopt(argc, argv, "AadelpRu:")) != -1) {
               switch (c) {
               case 'A':
                       opt_atf = true;
                       break;
               case 'a':
                       cmd = CMD_ALL;
                       break;
               case 'd':
                       debug++;
                       break;
               case 'e':
                       exact_match = true;
                       break;
               case 'l':
                       cmd = CMD_LIST;
                       break;
               case 'p':
                       use_pad = true;
                       break;
               case 'R':
#if !defined(NO_RUMP)
                       use_rump = true;
                       use_pad = true;
#else
                       usage();
#endif
                       break;
               case 'u':
                       unit = atoi(optarg);
                       break;
               default:
                       usage();
               }
       }
       argc -= optind;
       argv += optind;

       if (cmd == CMD_LIST) {
               /* List all */
               for (i = 0; testtable[i].name != NULL; i++)
                       printf("%s\n", testtable[i].name);
               return 0;
       }

       init(unit);

       if (cmd == CMD_ALL) {
               /* Test all */
               if (argc > 0)
                       usage();
               for (i = 0; testtable[i].name != NULL; i++)
                       do_test(i);
       } else {
               /* Test only matched */
               if (argc == 0)
                       usage();

               found = false;
               for (j = 0; j < argc; j++) {
                       for (i = 0; testtable[i].name != NULL; i++) {
                               if (match(argv[j], testtable[i].name)) {
                                       do_test(i);
                                       found = true;
                               }
                       }
               }
               if (!found) {
                       printf("test not found\n");
                       exit(1);
               }
       }

       if (opt_atf == false) {
               printf("Result: %d tests, %d success",
                   testcount,
                   testcount - failcount - skipcount);
               if (failcount > 0)
                       printf(", %d failed", failcount);
               if (skipcount > 0)
                       printf(", %d skipped", skipcount);
               printf("\n");
       }

       if (skipcount > 0)
               return 2;
       if (failcount > 0)
               return 1;

       return 0;
}

bool
match(const char *arg, const char *name)
{
       if (exact_match) {
               /* Exact match */
               if (strcmp(arg, name) == 0)
                       return true;
       } else {
               /* Forward match */
               if (strncmp(arg, name, strlen(arg)) == 0)
                       return true;
       }
       return false;
}

/*
* XXX
* Some hardware drivers (e.g. hdafg(4)) require a little "rest" between
* close(2) and re-open(2).
* audio(4) uses hw_if->close() to tell the hardware to close.  However,
* there is no agreement to wait for completion between MI and MD layer.
* audio(4) immediately shifts the "closed" state, and that is, the next
* open() will be acceptable immediately in audio layer.  But the real
* hardware may not have been closed actually at that point.
* It's troublesome issue but should be fixed...
*
* However, the most frequently used pad(4) (for ATF tests) doesn't have
* such problem, so avoids it to reduce time.
*/
void
xxx_close_wait(void)
{

       if (!use_pad)
               usleep(500 * 1000);
}

void
do_test(int testnumber)
{
       /* Sentinel */
       strlcpy(testname, "<NoName>", sizeof(testname));
       /* Do test */
       testtable[testnumber].func();

       cleanup_audiofd();
       xxx_close_wait();
}

/*
* system call wrappers for rump.
*/

/* open(2) or rump_sys_open(3) */
int
rump_or_open(const char *filename, int flag)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_open(filename, flag);
       else
#endif
               r = open(filename, flag);

       if (r > maxfd)
               maxfd = r;
       return r;
}

/* write(2) or rump_sys_write(3) */
int
rump_or_write(int fd, const void *buf, size_t len)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_write(fd, buf, len);
       else
#endif
               r = write(fd, buf, len);
       return r;
}

/* read(2) or rump_sys_read(3) */
int
rump_or_read(int fd, void *buf, size_t len)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_read(fd, buf, len);
       else
#endif
               r = read(fd, buf, len);
       return r;
}

/* ioctl(2) or rump_sys_ioctl(3) */
int
rump_or_ioctl(int fd, u_long cmd, void *arg)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_ioctl(fd, cmd, arg);
       else
#endif
               r = ioctl(fd, cmd, arg);
       return r;
}

/* close(2) or rump_sys_close(3) */
int
rump_or_close(int fd)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_close(fd);
       else
#endif
               r = close(fd);

       /* maxfd-1 may not valid fd but no matter */
       if (fd == maxfd)
               maxfd--;
       return r;
}

/* fcntl(2) or rump_sys_fcntl(3) */
/* XXX Supported only with no arguments for now */
int
rump_or_fcntl(int fd, int cmd, ...)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_fcntl(fd, cmd);
       else
#endif
               r = fcntl(fd, cmd);
       return r;
}

/* poll(2) or rump_sys_poll(3) */
int
rump_or_poll(struct pollfd *fds, nfds_t nfds, int timeout)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_poll(fds, nfds, timeout);
       else
#endif
               r = poll(fds, nfds, timeout);
       return r;
}

/* kqueue(2) or rump_sys_kqueue(3) */
int
rump_or_kqueue(void)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_kqueue();
       else
#endif
               r = kqueue();
       return r;
}

/* kevent(2) or rump_sys_kevent(3) */
int
rump_or_kevent(int kq, const struct kevent *chlist, size_t nch,
       struct kevent *evlist, size_t nev,
       const struct timespec *timeout)
{
       int r;

#if !defined(NO_RUMP)
       if (use_rump)
               r = rump_sys_kevent(kq, chlist, nch, evlist, nev, timeout);
       else
#endif
               r = kevent(kq, chlist, nch, evlist, nev, timeout);
       return r;
}

int
hw_canplay(void)
{
       return (props & AUDIO_PROP_PLAYBACK) ? 1 : 0;
}

int
hw_canrec(void)
{
       return (props & AUDIO_PROP_CAPTURE) ? 1 : 0;
}

int
hw_bidir(void)
{
       return hw_canplay() & hw_canrec();
}

int
hw_fulldup(void)
{
       return (props & AUDIO_PROP_FULLDUPLEX) ? 1 : 0;
}

#define DPRINTF(fmt...) do {    \
       if (debug)              \
               printf(fmt);    \
} while (0)

#define DPRINTFF(line, fmt...) do {             \
       if (debug) {                            \
               printf("  > %d: ", line);       \
               DPRINTF(fmt);                   \
               fflush(stdout);                 \
       }                                       \
} while (0)

#define DRESULT(r) do {                         \
       int backup_errno = errno;               \
       if (r == -1) {                          \
               DPRINTF(" = %d, err#%d %s\n",   \
                   r, backup_errno,            \
                   strerror(backup_errno));    \
       } else {                                \
               DPRINTF(" = %d\n", r);          \
       }                                       \
       errno = backup_errno;                   \
       return r;                               \
} while (0)

/* pointer variants for mmap */
#define DRESULT_PTR(r) do {                     \
       int backup_errno = errno;               \
       if (r == (void *)-1) {                  \
               DPRINTF(" = -1, err#%d %s\n",   \
                   backup_errno,               \
                   strerror(backup_errno));    \
       } else {                                \
               DPRINTF(" = %p\n", r);          \
       }                                       \
       errno = backup_errno;                   \
       return r;                               \
} while (0)


/*
* requnit <  0: Use auto by pad (not implemented).
* requnit >= 0: Use audio<requnit>.
*/
void
init(int requnit)
{
       struct audio_device devinfo;
       size_t len;
       int rel;
       int fd;
       int r;

       /* XXX */
       atexit(cleanup_audiofd);

       if (requnit < 0) {
               xp_errx(1, __LINE__, "requnit < 0 not implemented.");
       } else {
               unit = requnit;
       }

       /* Set device name */
       snprintf(devicename, sizeof(devicename), "audio%d", unit);
       snprintf(devaudio, sizeof(devaudio), "/dev/audio%d", unit);
       snprintf(devsound, sizeof(devsound), "/dev/sound%d", unit);
       snprintf(devaudioctl, sizeof(devaudioctl), "/dev/audioctl%d", unit);
       snprintf(devmixer, sizeof(devmixer), "/dev/mixer%d", unit);

       /*
        * version
        * audio2 is merged in 8.99.39.
        */
       len = sizeof(rel);
       r = sysctlbyname("kern.osrevision", &rel, &len, NULL, 0);
       if (r == -1)
               xp_err(1, __LINE__, "sysctl kern.osrevision");
       netbsd = rel / 100000000;
       if (rel >=  899003900)
               netbsd = 9;

#if !defined(NO_RUMP)
       if (use_rump) {
               DPRINTF("  use rump\n");
               rump_init();
       }
#endif

       /*
        * Open pad device before all accesses (including /dev/audioctl).
        */
       if (use_pad) {
               padfd = rump_or_open("/dev/pad0", O_RDONLY);
               if (padfd == -1)
                       xp_err(1, __LINE__, "rump_or_open");

               /* Create consumer thread */
               pthread_create(&th, NULL, consumer_thread, NULL);
               /* Set this thread's name */
               pthread_setname_np(pthread_self(), "main", NULL);
       }

       /*
        * Get device properties, etc.
        */
       fd = rump_or_open(devaudioctl, O_RDONLY);
       if (fd == -1)
               xp_err(1, __LINE__, "open %s", devaudioctl);
       r = rump_or_ioctl(fd, AUDIO_GETPROPS, &props);
       if (r == -1)
               xp_err(1, __LINE__, "AUDIO_GETPROPS");
       r = rump_or_ioctl(fd, AUDIO_GETDEV, &devinfo);
       if (r == -1)
               xp_err(1, __LINE__, "AUDIO_GETDEV");
       rump_or_close(fd);

       if (debug) {
               printf("  device = %s, %s, %s\n",
                   devinfo.name, devinfo.version, devinfo.config);
               printf("  hw props =");
               if (hw_canplay())
                       printf(" playback");
               if (hw_canrec())
                       printf(" capture");
               if (hw_fulldup())
                       printf(" fullduplex");
               printf("\n");
       }

}

/* Consumer thread used by pad */
void *
consumer_thread(void *arg)
{
       char buf[1024];
       int r;

       pthread_setname_np(pthread_self(), "consumer", NULL);
       pthread_detach(pthread_self());

       /* throw away data anyway */
       for (;;) {
               r = read(padfd, buf, sizeof(buf));
               if (r < 1)
                       break;
       }

       pthread_exit(NULL);
}

/*
* XXX
* Closing pad descriptor before audio descriptor causes panic (PR kern/54427).
* To avoid this, close non-pad descriptor first using atexit(3) for now.
* This is just a workaround and this function should be removed.
*/
void cleanup_audiofd()
{
       int fd;

       for (fd = 3; fd <= maxfd; fd++) {
               if (fd != padfd)
                       close(fd);
       }
       maxfd = 3;
}

/*
* Support functions
*/

/* Set testname */
void
TEST(const char *name, ...)
{
       va_list ap;

       va_start(ap, name);
       vsnprintf(testname, sizeof(testname), name, ap);
       va_end(ap);
       if (opt_atf == false) {
               printf("%s\n", testname);
               fflush(stdout);
       }
}

/*
* XP_FAIL() should be called when this test fails.
* If caller already count up testcount, call xp_fail() instead.
*/
#define XP_FAIL(fmt...) do {    \
       testcount++;    \
       xp_fail(__LINE__, fmt); \
} while (0)
bool xp_fail(int line, const char *fmt, ...)
{
       va_list ap;

       printf("%s %d: ", (opt_atf ? "Line" : " FAIL:"), line);
       va_start(ap, fmt);
       vprintf(fmt, ap);
       va_end(ap);
       printf("\n");
       fflush(stdout);
       failcount++;

       return false;
}

/*
* XP_SKIP() should be called when you want to skip this test.
* If caller already count up testcount, call xp_skip() instead.
*/
#define XP_SKIP(fmt...) do { \
       testcount++;    \
       xp_skip(__LINE__, fmt); \
} while (0)
void xp_skip(int line, const char *fmt, ...)
{
       va_list ap;

       printf("%s %d: ", (opt_atf ? "Line" : " SKIP:"), line);
       va_start(ap, fmt);
       vprintf(fmt, ap);
       va_end(ap);
       printf("\n");
       fflush(stdout);
       skipcount++;
}

#define XP_EQ(exp, act) xp_eq(__LINE__, exp, act, #act)
bool xp_eq(int line, int exp, int act, const char *varname)
{
       bool r = true;

       testcount++;
       if (exp != act) {
               r = xp_fail(line, "%s expects %d but %d", varname, exp, act);
       }
       return r;
}
#define XP_EQ_STR(exp, act) xp_eq_str(__LINE__, exp, act, #act)
bool xp_eq_str(int line, const char *exp, const char *act, const char *varname)
{
       bool r = true;

       testcount++;
       if (strcmp(exp, act) != 0) {
               r = xp_fail(line, "%s expects \"%s\" but \"%s\"",
                   varname, exp, act);
       }
       return r;
}

#define XP_NE(exp, act) xp_ne(__LINE__, exp, act, #act)
bool xp_ne(int line, int exp, int act, const char *varname)
{
       bool r = true;

       testcount++;
       if (exp == act) {
               r = xp_fail(line, "%s expects != %d but %d", varname, exp, act);
       }
       return r;
}

/* This expects that result is expressed in expr. */
/* GCC extension */
#define XP_IF(expr) xp_if(__LINE__, (expr), #expr)
bool xp_if(int line, bool expr, const char *exprname)
{
       bool r = true;
       testcount++;
       if (!expr) {
               r = xp_fail(__LINE__, "(%s) is expected but not met", exprname);
       }
       return r;
}

/* This expects that the system call returns 'exp'. */
#define XP_SYS_EQ(exp, act)     xp_sys_eq(__LINE__, exp, act, #act)
bool xp_sys_eq(int line, int exp, int act, const char *varname)
{
       bool r = true;

       testcount++;
       if (act == -1) {
               r = xp_fail(line, "%s expects %d but -1,err#%d(%s)",
                   varname, exp, errno, strerror(errno));
       } else {
               r = xp_eq(line, exp, act, varname);
       }
       return r;
}

/*
* This expects that system call succeeds.
* This is useful when you expect the system call succeeds but don't know
* the expected return value, such as open(2).
*/
#define XP_SYS_OK(act)  xp_sys_ok(__LINE__, act, #act)
bool xp_sys_ok(int line, int act, const char *varname)
{
       bool r = true;

       testcount++;
       if (act == -1) {
               r = xp_fail(line, "%s expects success but -1,err#%d(%s)",
                   varname, errno, strerror(errno));
       }
       return r;
}

/* This expects that the system call fails with 'experrno'. */
#define XP_SYS_NG(experrno, act) xp_sys_ng(__LINE__, experrno, act, #act)
bool xp_sys_ng(int line, int experrno, int act, const char *varname)
{
       bool r = true;

       testcount++;
       if (act != -1) {
               r = xp_fail(line, "%s expects -1,err#%d but %d",
                   varname, experrno, act);
       } else if (experrno != errno) {
               char acterrbuf[100];
               int acterrno = errno;
               strlcpy(acterrbuf, strerror(acterrno), sizeof(acterrbuf));
               r = xp_fail(line, "%s expects -1,err#%d(%s) but -1,err#%d(%s)",
                   varname, experrno, strerror(experrno),
                   acterrno, acterrbuf);
       }
       return r;
}

/*
* When exp == 0, this expects that the system call succeeds with returned
* pointer is not -1.
* When exp != 0, this expects that the system call fails with returned
* pointer is -1 and its errno is exp.
* It's only for mmap().
*/
#define XP_SYS_PTR(exp, act) xp_sys_ptr(__LINE__, exp, act, #act)
bool xp_sys_ptr(int line, int exp, void *act, const char *varname)
{
       char errbuf[256];
       int actual_errno;
       bool r = true;

       testcount++;
       if (exp == 0) {
               /* expects to succeed */
               if (act == (void *)-1) {
                       r = xp_fail(line,
                           "%s expects success but -1,err#%d(%s)",
                           varname, errno, strerror(errno));
               }
       } else {
               /* expects to fail */
               if (act != (void *)-1) {
                       r = xp_fail(line,
                           "%s expects -1,err#%d(%s) but success",
                           varname, exp, strerror(exp));
               } else if (exp != errno) {
                       actual_errno = errno;
                       strerror_r(actual_errno, errbuf, sizeof(errbuf));
                       r = xp_fail(line,
                           "%s expects -1,err#%d(%s) but -1,err#%d(%s)",
                           varname, exp, strerror(exp), actual_errno, errbuf);
               }
       }
       return r;
}


/*
* REQUIRED_* return immediately if condition does not meet.
*/
#define REQUIRED_EQ(e, a) do { if (!XP_EQ(e, a)) return; } while (0)
#define REQUIRED_NE(e, a) do { if (!XP_NE(e, a)) return; } while (0)
#define REQUIRED_IF(expr) do { if (!XP_IF(expr)) return; } while (0)
#define REQUIRED_SYS_EQ(e, a) do { if (!XP_SYS_EQ(e, a)) return; } while (0)
#define REQUIRED_SYS_OK(a)    do { if (!XP_SYS_OK(a))    return; } while (0)


static const char *openmode_str[] = {
       "O_RDONLY",
       "O_WRONLY",
       "O_RDWR",
};


/*
* All system calls in following tests should be called with these macros.
*/

#define OPEN(name, mode)        \
       debug_open(__LINE__, name, mode)
int debug_open(int line, const char *name, int mode)
{
       char modestr[32];
       int n;

       if ((mode & 3) != 3) {
               n = snprintf(modestr, sizeof(modestr), "%s",
                   openmode_str[mode & 3]);
       } else {
               n = snprintf(modestr, sizeof(modestr), "%d", mode & 3);
       }
       if ((mode & O_NONBLOCK))
               n += snprintf(modestr + n, sizeof(modestr) - n, "|O_NONBLOCK");

       DPRINTFF(line, "open(\"%s\", %s)", name, modestr);
       int r = rump_or_open(name, mode);
       DRESULT(r);
}

#define WRITE(fd, addr, len)    \
       debug_write(__LINE__, fd, addr, len)
int debug_write(int line, int fd, const void *addr, size_t len)
{
       DPRINTFF(line, "write(%d, %p, %zd)", fd, addr, len);
       int r = rump_or_write(fd, addr, len);
       DRESULT(r);
}

#define READ(fd, addr, len)     \
       debug_read(__LINE__, fd, addr, len)
int debug_read(int line, int fd, void *addr, size_t len)
{
       DPRINTFF(line, "read(%d, %p, %zd)", fd, addr, len);
       int r = rump_or_read(fd, addr, len);
       DRESULT(r);
}

/*
* addrstr is the comment for debug message.
*   int onoff = 0;
*   ioctl(fd, SWITCH, onoff); -> IOCTL(fd, SWITCH, onoff, "off");
*/
#define IOCTL(fd, name, addr, addrfmt...)       \
       debug_ioctl(__LINE__, fd, name, #name, addr, addrfmt)
int debug_ioctl(int line, int fd, u_long name, const char *namestr,
       void *addr, const char *addrfmt, ...)
{
       char addrbuf[100];
       va_list ap;

       va_start(ap, addrfmt);
       vsnprintf(addrbuf, sizeof(addrbuf), addrfmt, ap);
       va_end(ap);
       DPRINTFF(line, "ioctl(%d, %s, %s)", fd, namestr, addrbuf);
       int r = rump_or_ioctl(fd, name, addr);
       DRESULT(r);
}

#define FCNTL(fd, name...)      \
       debug_fcntl(__LINE__, fd, name, #name)
int debug_fcntl(int line, int fd, int name, const char *namestr, ...)
{
       int r;

       switch (name) {
        case F_GETFL:  /* no arguments */
               DPRINTFF(line, "fcntl(%d, %s)", fd, namestr);
               r = rump_or_fcntl(fd, name);
               break;
        default:
               __unreachable();
       }
       DRESULT(r);
       return r;
}

#define CLOSE(fd)       \
       debug_close(__LINE__, fd)
int debug_close(int line, int fd)
{
       DPRINTFF(line, "close(%d)", fd);
       int r = rump_or_close(fd);
       DRESULT(r);
}

#define MMAP(ptr, len, prot, flags, fd, offset) \
       debug_mmap(__LINE__, ptr, len, prot, flags, fd, offset)
void *debug_mmap(int line, void *ptr, size_t len, int prot, int flags, int fd,
       off_t offset)
{
       char protbuf[256];
       char flagbuf[256];
       int n;

#define ADDFLAG(buf, var, name) do {                            \
       if (((var) & (name)))                                   \
               n = strlcat(buf, "|" #name, sizeof(buf));       \
       (var) &= ~(name);                                       \
} while (0)

       n = 0;
       protbuf[n] = '\0';
       if (prot == 0) {
               strlcpy(protbuf, "|PROT_NONE", sizeof(protbuf));
       } else {
               ADDFLAG(protbuf, prot, PROT_EXEC);
               ADDFLAG(protbuf, prot, PROT_WRITE);
               ADDFLAG(protbuf, prot, PROT_READ);
               if (prot != 0) {
                       snprintf(protbuf + n, sizeof(protbuf) - n,
                           "|prot=0x%x", prot);
               }
       }

       n = 0;
       flagbuf[n] = '\0';
       if (flags == 0) {
               strlcpy(flagbuf, "|MAP_FILE", sizeof(flagbuf));
       } else {
               ADDFLAG(flagbuf, flags, MAP_SHARED);
               ADDFLAG(flagbuf, flags, MAP_PRIVATE);
               ADDFLAG(flagbuf, flags, MAP_FIXED);
               ADDFLAG(flagbuf, flags, MAP_INHERIT);
               ADDFLAG(flagbuf, flags, MAP_HASSEMAPHORE);
               ADDFLAG(flagbuf, flags, MAP_TRYFIXED);
               ADDFLAG(flagbuf, flags, MAP_WIRED);
               ADDFLAG(flagbuf, flags, MAP_ANON);
               if (flags != 0) {
                       n += snprintf(flagbuf + n, sizeof(flagbuf) - n,
                           "|flag=0x%x", flags);
               }
       }

       DPRINTFF(line, "mmap(%p, %zd, %s, %s, %d, %jd)",
           ptr, len, protbuf + 1, flagbuf + 1, fd, offset);
       void *r = mmap(ptr, len, prot, flags, fd, offset);
       DRESULT_PTR(r);
}

#define MUNMAP(ptr, len)        \
       debug_munmap(__LINE__, ptr, len)
int debug_munmap(int line, void *ptr, int len)
{
#if !defined(NO_RUMP)
       if (use_rump)
               xp_errx(1, __LINE__, "rump doesn't support munmap");
#endif
       DPRINTFF(line, "munmap(%p, %d)", ptr, len);
       int r = munmap(ptr, len);
       DRESULT(r);
}

const char *
event_tostr(int events)
{
       static char buf[64];

       snprintb(buf, sizeof(buf),
           "\177\020" \
           "b\10WRBAND\0" \
           "b\7RDBAND\0" "b\6RDNORM\0" "b\5NVAL\0" "b\4HUP\0" \
           "b\3ERR\0" "b\2OUT\0" "b\1PRI\0" "b\0IN\0",
           events);
       return buf;
}

#define POLL(pfd, nfd, timeout) \
       debug_poll(__LINE__, pfd, nfd, timeout)
int debug_poll(int line, struct pollfd *pfd, int nfd, int timeout)
{
       char buf[256];
       int n = 0;
       buf[n] = '\0';
       for (int i = 0; i < nfd; i++) {
               n += snprintf(buf + n, sizeof(buf) - n, "{fd=%d,events=%s}",
                   pfd[i].fd, event_tostr(pfd[i].events));
       }
       DPRINTFF(line, "poll(%s, %d, %d)", buf, nfd, timeout);
       int r = rump_or_poll(pfd, nfd, timeout);
       DRESULT(r);
}

#define KQUEUE()        \
       debug_kqueue(__LINE__)
int debug_kqueue(int line)
{
       DPRINTFF(line, "kqueue()");
       int r = rump_or_kqueue();
       DRESULT(r);
}

#define KEVENT_SET(kq, kev, nev)        \
       debug_kevent_set(__LINE__, kq, kev, nev)
int debug_kevent_set(int line, int kq, const struct kevent *kev, size_t nev)
{
       DPRINTFF(line, "kevent_set(%d, %p, %zd)", kq, kev, nev);
       int r = rump_or_kevent(kq, kev, nev, NULL, 0, NULL);
       DRESULT(r);
}

#define KEVENT_POLL(kq, kev, nev, ts) \
       debug_kevent_poll(__LINE__, kq, kev, nev, ts)
int debug_kevent_poll(int line, int kq, struct kevent *kev, size_t nev,
       const struct timespec *ts)
{
       char tsbuf[32];

       if (ts == NULL) {
               snprintf(tsbuf, sizeof(tsbuf), "NULL");
       } else if (ts->tv_sec == 0 && ts->tv_nsec == 0) {
               snprintf(tsbuf, sizeof(tsbuf), "0.0");
       } else {
               snprintf(tsbuf, sizeof(tsbuf), "%d.%09ld",
                       (int)ts->tv_sec, ts->tv_nsec);
       }
       DPRINTFF(line, "kevent_poll(%d, %p, %zd, %s)", kq, kev, nev, tsbuf);
       int r = rump_or_kevent(kq, NULL, 0, kev, nev, ts);
       DRESULT(r);
}

#define DEBUG_KEV(name, kev)    \
       debug_kev(__LINE__, name, kev)
void debug_kev(int line, const char *name, const struct kevent *kev)
{
       char flagbuf[256];
       const char *filterbuf;
       uint32_t v;
       int n;

       n = 0;
       flagbuf[n] = '\0';
       if (kev->flags == 0) {
               strcpy(flagbuf, "|0?");
       } else {
               v = kev->flags;
               ADDFLAG(flagbuf, v, EV_ADD);
               if (v != 0)
                       snprintf(flagbuf + n, sizeof(flagbuf)-n, "|0x%x", v);
       }

       switch (kev->filter) {
        case EVFILT_READ:      filterbuf = "EVFILT_READ";      break;
        case EVFILT_WRITE:     filterbuf = "EVFILT_WRITE";     break;
        default:               filterbuf = "EVFILT_?";         break;
       }

       DPRINTFF(line,
           "%s={id:%d,%s,%s,fflags:0x%x,data:0x%" PRIx64 ",udata:0x%x}\n",
           name,
           (int)kev->ident,
           flagbuf + 1,
           filterbuf,
           kev->fflags,
           kev->data,
           (int)(intptr_t)kev->udata);
}

/* XXX rump? */
#define GETUID()        \
       debug_getuid(__LINE__)
uid_t debug_getuid(int line)
{
       DPRINTFF(line, "getuid");
       uid_t r = getuid();
       /* getuid() never fails */
       DPRINTF(" = %u\n", r);
       return r;
}

/* XXX rump? */
#define SETEUID(id)     \
       debug_seteuid(__LINE__, id)
int debug_seteuid(int line, uid_t id)
{
       DPRINTFF(line, "seteuid(%d)", (int)id);
       int r = seteuid(id);
       DRESULT(r);
}

#define SYSCTLBYNAME(name, oldp, oldlenp, newp, newlen) \
       debug_sysctlbyname(__LINE__, name, oldp, oldlenp, newp, newlen)
int debug_sysctlbyname(int line, const char *name, void *oldp, size_t *oldlenp,
       const void *newp, size_t newlen)
{
       DPRINTFF(line, "sysctlbyname(\"%s\")", name);
       int r = sysctlbyname(name, oldp, oldlenp, newp, newlen);
       DRESULT(r);
}


/* Return openable mode on this hardware property */
int
openable_mode(void)
{
       if (hw_bidir())
               return O_RDWR;
       if (hw_canplay())
               return O_WRONLY;
       else
               return O_RDONLY;
}

int mode2aumode_full[] = {
                                       AUMODE_RECORD,  /* O_RDONLY */
       AUMODE_PLAY | AUMODE_PLAY_ALL,                  /* O_WRONLY */
       AUMODE_PLAY | AUMODE_PLAY_ALL | AUMODE_RECORD,  /* O_RDWR   */
};

/* Convert openmode(O_*) to AUMODE_*, with hardware property */
int
mode2aumode(int mode)
{
       int aumode;

       aumode = mode2aumode_full[mode];
       if (hw_canplay() == 0)
               aumode &= ~(AUMODE_PLAY | AUMODE_PLAY_ALL);
       if (hw_canrec() == 0)
               aumode &= ~AUMODE_RECORD;

       if (netbsd >= 9) {
               /* half-duplex treats O_RDWR as O_WRONLY */
               if (mode == O_RDWR && hw_bidir() && hw_fulldup() == 0)
                       aumode &= ~AUMODE_RECORD;
       }

       return aumode;
}

/* Is this mode + hardware playable? */
int
mode2play(int mode)
{
       int aumode;

       aumode = mode2aumode(mode);
       return ((aumode & AUMODE_PLAY)) ? 1 : 0;
}

/* Is this mode + hardware recordable? */
int
mode2rec(int mode)
{
       int aumode;

       aumode = mode2aumode(mode);
       return ((aumode & AUMODE_RECORD)) ? 1 : 0;
}

/*
* On NetBSD7, open() after-closing-mmap fails due to a bug.
* It happens once every two times like flip-flop, so the workaround is
* to open it again.
*/
void
reset_after_mmap(void)
{
       int fd;

       if (netbsd < 8) {
               fd = OPEN(devaudio, O_WRONLY);
               if (fd != -1)
                       CLOSE(fd);
       }
}

/*
* Lookup "outputs.master" and return its mixer device index.
* It may not be strict but I'm not sure.
*/
int
mixer_get_outputs_master(int mixerfd)
{
       const char * const typename[] = { "CLASS", "ENUM", "SET", "VALUE" };
       mixer_devinfo_t di;
       int class_outputs;
       int i;
       int r;

       class_outputs = -1;
       for (i = 0; ; i++) {
               memset(&di, 0, sizeof(di));
               di.index = i;
               r = IOCTL(mixerfd, AUDIO_MIXER_DEVINFO, &di, "index=%d", i);
               if (r < 0)
                       break;
               DPRINTF("  > type=%s(%d) mixer_class=%d name=%s\n",
                   (0 <= di.type && di.type <= 3) ? typename[di.type] : "",
                   di.type, di.mixer_class, di.label.name);
               if (di.type == AUDIO_MIXER_CLASS &&
                   strcmp(di.label.name, "outputs") == 0) {
                       class_outputs = di.mixer_class;
                       DPRINTF("  > class_output=%d\n", class_outputs);
                       continue;
               }
               if (di.type == AUDIO_MIXER_VALUE &&
                   di.mixer_class == class_outputs &&
                   strcmp(di.label.name, "master") == 0) {
                       return i;
               }
       }
       /* Not found */
       return -1;
}

/*
* Tests
*/

void test_open_mode(int);
void test_open(const char *, int);
void test_open_simul(int, int);
void try_open_multiuser(bool);
void test_open_multiuser(bool);
void test_rdwr_fallback(int, bool, bool);
void test_rdwr_two(int, int);
void test_mmap_mode(int, int);
void test_mmap_len(size_t, off_t, int);
void test_poll_mode(int, int, int);
void test_poll_in_open(const char *);
void test_kqueue_mode(int, int, int);
volatile int sigio_caught;
void signal_FIOASYNC(int);
void test_AUDIO_SETFD_xxONLY(int);
void test_AUDIO_SETINFO_mode(int, int, int, int);
void test_AUDIO_SETINFO_params_set(int, int, int);
void test_AUDIO_SETINFO_pause(int, int, int);
int getenc_make_table(int, int[][5]);
void xp_getenc(int[][5], int, int, int, struct audio_prinfo *);
void getenc_check_encodings(int, int[][5]);
void test_AUDIO_ERROR(int);
void test_AUDIO_GETIOFFS_one(int);
void test_AUDIO_GETOOFFS_one(int);
void test_AUDIO_GETOOFFS_wrap(int);
void test_AUDIO_GETOOFFS_flush(int);
void test_AUDIO_GETOOFFS_set(int);
void test_audioctl_open_1(int, int);
void test_audioctl_open_2(int, int);
void try_audioctl_open_multiuser(const char *, const char *);
void test_audioctl_open_multiuser(bool, const char *, const char *);
void test_audioctl_rw(int);

#define DEF(name) \
       void test__ ## name (void); \
       void test__ ## name (void)

/*
* Whether it can be open()ed with specified mode.
*/
void
test_open_mode(int mode)
{
       int fd;
       int r;

       TEST("open_mode_%s", openmode_str[mode] + 2);

       fd = OPEN(devaudio, mode);
       if (mode2aumode(mode) != 0) {
               XP_SYS_OK(fd);
       } else {
               XP_SYS_NG(ENXIO, fd);
       }

       if (fd >= 0) {
               r = CLOSE(fd);
               XP_SYS_EQ(0, r);
       }
}
DEF(open_mode_RDONLY)   { test_open_mode(O_RDONLY); }
DEF(open_mode_WRONLY)   { test_open_mode(O_WRONLY); }
DEF(open_mode_RDWR)     { test_open_mode(O_RDWR);   }

/*
* Check the initial parameters and stickiness.
* /dev/audio
*      The initial parameters are always the same whenever you open.
* /dev/sound and /dev/audioctl
*      The initial parameters are inherited from the last /dev/sound or
*      /dev/audio.
*/
void
test_open(const char *devname, int mode)
{
       struct audio_info ai;
       struct audio_info ai0;
       char devfile[16];
       int fd;
       int r;
       int can_play;
       int can_rec;
       int exp_mode;
       int exp_encoding;
       int exp_precision;
       int exp_channels;
       int exp_sample_rate;
       int exp_pause;
       int exp_popen;
       int exp_ropen;

       TEST("open_%s_%s", devname, openmode_str[mode] + 2);

       snprintf(devfile, sizeof(devfile), "/dev/%s%d", devname, unit);
       can_play = mode2play(mode);
       can_rec  = mode2rec(mode);
       if (strcmp(devname, "audioctl") != 0) {
               if (can_play + can_rec == 0) {
                       /* Check whether it cannot be opened */
                       fd = OPEN(devaudio, mode);
                       XP_SYS_NG(ENXIO, fd);
                       return;
               }
       }

       /* /dev/audio is always initialized */
       if (strcmp(devname, "audio") == 0) {
               exp_encoding = AUDIO_ENCODING_ULAW;
               exp_precision = 8;
               exp_channels = 1;
               exp_sample_rate = 8000;
               exp_pause = 0;
       } else {
               exp_encoding = AUDIO_ENCODING_SLINEAR_LE;
               exp_precision = 16;
               exp_channels = 2;
               exp_sample_rate = 11025;
               exp_pause = 1;
       }

       /* /dev/audioctl is always "not opened" */
       if (strcmp(devname, "audioctl") == 0) {
               exp_mode = 0;
               exp_popen = 0;
               exp_ropen = 0;
       } else {
               exp_mode = mode2aumode(mode);
               exp_popen = can_play;
               exp_ropen = can_rec;
       }


       /*
        * At first, initialize the sticky parameters both of play and rec.
        * This uses /dev/audio to verify /dev/audio.  It's not good way but
        * I don't have better one...
        */
       fd = OPEN(devaudio, openable_mode());
       REQUIRED_SYS_OK(fd);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /*
        * Open target device and check the initial parameters
        * At this moment, all devices are initialized by default.
        */
       fd = OPEN(devfile, mode);
       REQUIRED_SYS_OK(fd);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);

       XP_NE(0, ai.blocksize);
               /* hiwat/lowat */
       XP_EQ(exp_mode, ai.mode);
       /* ai.play */
       XP_EQ(8000, ai.play.sample_rate);
       XP_EQ(1, ai.play.channels);
       XP_EQ(8, ai.play.precision);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.play.encoding);
               /* gain */
               /* port */
       XP_EQ(0, ai.play.seek);
               /* avail_ports */
       XP_NE(0, ai.play.buffer_size);
       XP_EQ(0, ai.play.samples);
       XP_EQ(0, ai.play.eof);
       XP_EQ(0, ai.play.pause);
       XP_EQ(0, ai.play.error);
       XP_EQ(0, ai.play.waiting);
               /* balance */
       XP_EQ(exp_popen, ai.play.open);
       XP_EQ(0, ai.play.active);
       /* ai.record */
       XP_EQ(8000, ai.record.sample_rate);
       XP_EQ(1, ai.record.channels);
       XP_EQ(8, ai.record.precision);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.record.encoding);
               /* gain */
               /* port */
       XP_EQ(0, ai.record.seek);
               /* avail_ports */
       XP_NE(0, ai.record.buffer_size);
       XP_EQ(0, ai.record.samples);
       XP_EQ(0, ai.record.eof);
       XP_EQ(0, ai.record.pause);
       XP_EQ(0, ai.record.error);
       XP_EQ(0, ai.record.waiting);
               /* balance */
       XP_EQ(exp_ropen, ai.record.open);
       if (netbsd < 9 && strcmp(devname, "sound") == 0) {
               /*
                * On NetBSD7/8, it doesn't seem to start recording on open
                * for /dev/sound.  It should be a bug.
                */
               XP_EQ(0, ai.record.active);
       } else {
               XP_EQ(exp_ropen, ai.record.active);
       }
       /* Save it */
       ai0 = ai;

       /*
        * Change much as possible
        */
       AUDIO_INITINFO(&ai);
       ai.mode = ai0.mode ^ AUMODE_PLAY_ALL;
       ai.play.sample_rate = 11025;
       ai.play.channels = 2;
       ai.play.precision = 16;
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.play.pause = 1;
       ai.record.sample_rate = 11025;
       ai.record.channels = 2;
       ai.record.precision = 16;
       ai.record.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.record.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "ai");
       REQUIRED_SYS_EQ(0, r);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /*
        * Open the same target device again and check
        */
       fd = OPEN(devfile, mode);
       REQUIRED_SYS_OK(fd);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);

       XP_NE(0, ai.blocksize);
               /* hiwat/lowat */
       if (netbsd < 8) {
               /*
                * On NetBSD7, the behavior when changing ai.mode on
                * /dev/audioctl can not be explained yet but I won't
                * verify it more over.
                */
       } else {
               /* On NetBSD9, changing mode never affects other fds */
               XP_EQ(exp_mode, ai.mode);
       }
       /* ai.play */
       XP_EQ(exp_sample_rate, ai.play.sample_rate);
       XP_EQ(exp_channels, ai.play.channels);
       XP_EQ(exp_precision, ai.play.precision);
       XP_EQ(exp_encoding, ai.play.encoding);
               /* gain */
               /* port */
       XP_EQ(0, ai.play.seek);
               /* avail_ports */
       XP_NE(0, ai.play.buffer_size);
       XP_EQ(0, ai.play.samples);
       XP_EQ(0, ai.play.eof);
       XP_EQ(exp_pause, ai.play.pause);
       XP_EQ(0, ai.play.error);
       XP_EQ(0, ai.play.waiting);
               /* balance */
       XP_EQ(exp_popen, ai.play.open);
       XP_EQ(0, ai.play.active);
       /* ai.record */
       XP_EQ(exp_sample_rate, ai.record.sample_rate);
       XP_EQ(exp_channels, ai.record.channels);
       XP_EQ(exp_precision, ai.record.precision);
       XP_EQ(exp_encoding, ai.record.encoding);
               /* gain */
               /* port */
       XP_EQ(0, ai.record.seek);
               /* avail_ports */
       XP_NE(0, ai.record.buffer_size);
       XP_EQ(0, ai.record.samples);
       XP_EQ(0, ai.record.eof);
       XP_EQ(exp_pause, ai.record.pause);
       XP_EQ(0, ai.record.error);
       XP_EQ(0, ai.record.waiting);
               /* balance */
       XP_EQ(exp_ropen, ai.record.open);
       if (netbsd < 9 && strcmp(devname, "sound") == 0) {
               /*
                * On NetBSD7/8, it doesn't seem to start recording on open
                * for /dev/sound.  It should be a bug.
                */
               XP_EQ(0, ai.record.active);
       } else {
               XP_EQ(exp_ropen, ai.record.active);
       }

       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);
}
DEF(open_audio_RDONLY)          { test_open("audio", O_RDONLY); }
DEF(open_audio_WRONLY)          { test_open("audio", O_WRONLY); }
DEF(open_audio_RDWR)            { test_open("audio", O_RDWR);   }
DEF(open_sound_RDONLY)          { test_open("sound", O_RDONLY); }
DEF(open_sound_WRONLY)          { test_open("sound", O_WRONLY); }
DEF(open_sound_RDWR)            { test_open("sound", O_RDWR);   }
DEF(open_audioctl_RDONLY)       { test_open("audioctl", O_RDONLY); }
DEF(open_audioctl_WRONLY)       { test_open("audioctl", O_WRONLY); }
DEF(open_audioctl_RDWR)         { test_open("audioctl", O_RDWR);   }

/*
* Open (1) /dev/sound -> (2) /dev/audio -> (3) /dev/sound,
* Both of /dev/audio and /dev/sound share the sticky parameters,
* /dev/sound inherits and use it but /dev/audio initialize and use it.
* So 2nd audio descriptor affects 3rd sound descriptor.
*/
DEF(open_sound_sticky)
{
       struct audio_info ai;
       int fd;
       int r;
       int openmode;

       TEST("open_sound_sticky");

       openmode = openable_mode();

       /* First, open /dev/sound and change encoding as a delegate */
       fd = OPEN(devsound, openmode);
       REQUIRED_SYS_OK(fd);
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.record.encoding = AUDIO_ENCODING_SLINEAR_LE;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /* Next, open /dev/audio.  It makes the encoding mulaw */
       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /* And then, open /dev/sound again */
       fd = OPEN(devsound, openmode);
       REQUIRED_SYS_OK(fd);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.play.encoding);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.record.encoding);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);
}

/*
* /dev/audioctl has stickiness like /dev/sound.
*/
DEF(open_audioctl_sticky)
{
       struct audio_info ai;
       int fd;
       int r;
       int openmode;

       TEST("open_audioctl_sticky");

       openmode = openable_mode();

       /* First, open /dev/audio and change encoding */
       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.play.precision = 16;
       ai.record.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.record.precision = 16;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "SLINEAR_LE");
       REQUIRED_SYS_EQ(0, r);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /* Next, open /dev/audioctl.  It should be affected */
       fd = OPEN(devaudioctl, openmode);
       REQUIRED_SYS_OK(fd);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(AUDIO_ENCODING_SLINEAR_LE, ai.play.encoding);
       XP_EQ(16, ai.play.precision);
       XP_EQ(AUDIO_ENCODING_SLINEAR_LE, ai.record.encoding);
       XP_EQ(16, ai.record.precision);

       /* Then, change /dev/audioctl */
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_ULAW;
       ai.play.precision = 8;
       ai.record.encoding = AUDIO_ENCODING_ULAW;
       ai.record.precision = 8;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "ULAW");
       REQUIRED_SYS_EQ(0, r);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       /* Finally, open /dev/sound.  It also should be affected  */
       fd = OPEN(devsound, openmode);
       REQUIRED_SYS_OK(fd);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.play.encoding);
       XP_EQ(8, ai.play.precision);
       XP_EQ(AUDIO_ENCODING_ULAW, ai.record.encoding);
       XP_EQ(8, ai.record.precision);
       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);
}

/*
* Open two descriptors simultaneously.
*/
void
test_open_simul(int mode0, int mode1)
{
       struct audio_info ai;
       int fd0, fd1;
       int i;
       int r;
       int actmode;
#define AUMODE_BOTH (AUMODE_PLAY | AUMODE_RECORD)
       struct {
               int mode0;
               int mode1;
       } expfulltable[] = {
               /* expected fd0         expected fd1 (-errno expects error) */
               { AUMODE_RECORD,        AUMODE_RECORD },        // REC, REC
               { AUMODE_RECORD,        AUMODE_PLAY },          // REC, PLAY
               { AUMODE_RECORD,        AUMODE_BOTH },          // REC, BOTH
               { AUMODE_PLAY,          AUMODE_RECORD },        // PLAY, REC
               { AUMODE_PLAY,          AUMODE_PLAY },          // PLAY, PLAY
               { AUMODE_PLAY,          AUMODE_BOTH },          // PLAY, BOTH
               { AUMODE_BOTH,          AUMODE_RECORD },        // BOTH, REC
               { AUMODE_BOTH,          AUMODE_PLAY },          // BOTH, PLAY
               { AUMODE_BOTH,          AUMODE_BOTH },          // BOTH, BOTH
       },
       exphalftable[] = {
               /* expected fd0         expected fd1 (-errno expects error) */
               { AUMODE_RECORD,        AUMODE_RECORD },        // REC, REC
               { AUMODE_RECORD,        -ENODEV },              // REC, PLAY
               { AUMODE_RECORD,        -ENODEV },              // REC, BOTH
               { AUMODE_PLAY,          -ENODEV },              // PLAY, REC
               { AUMODE_PLAY,          AUMODE_PLAY },          // PLAY, PLAY
               { AUMODE_PLAY,          AUMODE_PLAY },          // PLAY, BOTH
               { AUMODE_PLAY,          -ENODEV },              // BOTH, REC
               { AUMODE_PLAY,          AUMODE_PLAY },          // BOTH, PLAY
               { AUMODE_PLAY,          AUMODE_PLAY },          // BOTH, BOTH
       }, *exptable;

       /* The expected values are different in half-duplex or full-duplex */
       if (hw_fulldup()) {
               exptable = expfulltable;
       } else {
               exptable = exphalftable;
       }

       TEST("open_simul_%s_%s",
           openmode_str[mode0] + 2,
           openmode_str[mode1] + 2);

       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }

       if (mode2aumode(mode0) == 0 || mode2aumode(mode1) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       i = mode0 * 3 + mode1;

       /* Open first one */
       fd0 = OPEN(devaudio, mode0);
       REQUIRED_SYS_OK(fd0);
       r = IOCTL(fd0, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       actmode = ai.mode & AUMODE_BOTH;
       XP_EQ(exptable[i].mode0, actmode);

       /* Open second one */
       fd1 = OPEN(devaudio, mode1);
       if (exptable[i].mode1 >= 0) {
               /* Case to expect to be able to open */
               REQUIRED_SYS_OK(fd1);
               r = IOCTL(fd1, AUDIO_GETBUFINFO, &ai, "");
               XP_SYS_EQ(0, r);
               if (r == 0) {
                       actmode = ai.mode & AUMODE_BOTH;
                       XP_EQ(exptable[i].mode1, actmode);
               }
       } else {
               /* Case to expect not to be able to open */
               XP_SYS_NG(ENODEV, fd1);
               if (fd1 == -1) {
                       XP_EQ(-exptable[i].mode1, errno);
               } else {
                       r = IOCTL(fd1, AUDIO_GETBUFINFO, &ai, "");
                       XP_SYS_EQ(0, r);
                       if (r == 0) {
                               actmode = ai.mode & AUMODE_BOTH;
                               XP_FAIL("expects error but %d", actmode);
                       }
               }
       }

       if (fd1 >= 0) {
               r = CLOSE(fd1);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);
}
DEF(open_simul_RDONLY_RDONLY)   { test_open_simul(O_RDONLY, O_RDONLY);  }
DEF(open_simul_RDONLY_WRONLY)   { test_open_simul(O_RDONLY, O_WRONLY);  }
DEF(open_simul_RDONLY_RDWR)     { test_open_simul(O_RDONLY, O_RDWR);    }
DEF(open_simul_WRONLY_RDONLY)   { test_open_simul(O_WRONLY, O_RDONLY);  }
DEF(open_simul_WRONLY_WRONLY)   { test_open_simul(O_WRONLY, O_WRONLY);  }
DEF(open_simul_WRONLY_RDWR)     { test_open_simul(O_WRONLY, O_RDWR);    }
DEF(open_simul_RDWR_RDONLY)     { test_open_simul(O_RDWR, O_RDONLY);    }
DEF(open_simul_RDWR_WRONLY)     { test_open_simul(O_RDWR, O_WRONLY);    }
DEF(open_simul_RDWR_RDWR)       { test_open_simul(O_RDWR, O_RDWR);      }

/*
* /dev/audio can be opened by other user who opens /dev/audio.
*/
void
try_open_multiuser(bool multiuser)
{
       int fd0;
       int fd1;
       int r;
       uid_t ouid;

       /*
        * Test1: Open as root first and then unprivileged user.
        */

       /* At first, open as root */
       fd0 = OPEN(devaudio, openable_mode());
       REQUIRED_SYS_OK(fd0);

       ouid = GETUID();
       r = SETEUID(1);
       REQUIRED_SYS_EQ(0, r);

       /* Then, open as unprivileged user */
       fd1 = OPEN(devaudio, openable_mode());
       if (multiuser) {
               /* If multiuser, another user also can open */
               XP_SYS_OK(fd1);
       } else {
               /* If not multiuser, another user cannot open */
               XP_SYS_NG(EPERM, fd1);
       }
       if (fd1 != -1) {
               r = CLOSE(fd1);
               XP_SYS_EQ(0, r);
       }

       r = SETEUID(ouid);
       REQUIRED_SYS_EQ(0, r);

       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);

       /*
        * Test2: Open as unprivileged user first and then root.
        */

       /* At first, open as unprivileged user */
       ouid = GETUID();
       r = SETEUID(1);
       REQUIRED_SYS_EQ(0, r);

       fd0 = OPEN(devaudio, openable_mode());
       REQUIRED_SYS_OK(fd0);

       /* Then open as root */
       r = SETEUID(ouid);
       REQUIRED_SYS_EQ(0, r);

       /* root always can open */
       fd1 = OPEN(devaudio, openable_mode());
       XP_SYS_OK(fd1);
       if (fd1 != -1) {
               r = CLOSE(fd1);
               XP_SYS_EQ(0, r);
       }

       /* Close first one as unprivileged user */
       r = SETEUID(1);
       REQUIRED_SYS_EQ(0, r);
       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);
       r = SETEUID(ouid);
       REQUIRED_SYS_EQ(0, r);
}
/*
* This is a wrapper for open_multiuser.
* XXX XP_* macros are not compatible with on-error-goto, we need try-catch...
*/
void
test_open_multiuser(bool multiuser)
{
       char mibname[32];
       bool oldval;
       size_t oldlen;
       int r;

       TEST("open_multiuser_%d", multiuser);
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (netbsd < 9) {
               /* NetBSD8 has no way (difficult) to determine device name */
               XP_SKIP("NetBSD8 cannot determine device name");
               return;
       }
       if (geteuid() != 0) {
               XP_SKIP("Must be run as a privileged user");
               return;
       }

       /* Get current multiuser mode (and save it) */
       snprintf(mibname, sizeof(mibname), "hw.%s.multiuser", devicename);
       oldlen = sizeof(oldval);
       r = SYSCTLBYNAME(mibname, &oldval, &oldlen, NULL, 0);
       REQUIRED_SYS_EQ(0, r);
       DPRINTF("  > multiuser=%d\n", oldval);

       /* Change if necessary */
       if (oldval != multiuser) {
               r = SYSCTLBYNAME(mibname, NULL, NULL, &multiuser,
                   sizeof(multiuser));
               REQUIRED_SYS_EQ(0, r);
               DPRINTF("  > new multiuser=%d\n", multiuser);
       }

       /* Do test */
       try_open_multiuser(multiuser);

       /* Restore multiuser mode */
       if (oldval != multiuser) {
               DPRINTF("  > restore multiuser to %d\n", oldval);
               r = SYSCTLBYNAME(mibname, NULL, NULL, &oldval, sizeof(oldval));
               REQUIRED_SYS_EQ(0, r);
       }
}
DEF(open_multiuser_0)   { test_open_multiuser(false); }
DEF(open_multiuser_1)   { test_open_multiuser(true); }

/*
* Normal playback (with PLAY_ALL).
* It does not verify real playback data.
*/
DEF(write_PLAY_ALL)
{
       char buf[8000];
       int fd;
       int r;

       TEST("write_PLAY_ALL");

       fd = OPEN(devaudio, O_WRONLY);
       if (hw_canplay()) {
               REQUIRED_SYS_OK(fd);
       } else {
               XP_SYS_NG(ENXIO, fd);
               return;
       }

       /* mulaw 1sec silence */
       memset(buf, 0xff, sizeof(buf));
       r = WRITE(fd, buf, sizeof(buf));
       XP_SYS_EQ(sizeof(buf), r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Normal playback (without PLAY_ALL).
* It does not verify real playback data.
*/
DEF(write_PLAY)
{
       struct audio_info ai;
       char *wav;
       int wavsize;
       int totalsize;
       int fd;
       int r;

       TEST("write_PLAY");

       fd = OPEN(devaudio, O_WRONLY);
       if (hw_canplay()) {
               REQUIRED_SYS_OK(fd);
       } else {
               XP_SYS_NG(ENXIO, fd);
               return;
       }

       /* Drop PLAY_ALL */
       AUDIO_INITINFO(&ai);
       ai.mode = AUMODE_PLAY;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "mode");
       REQUIRED_SYS_EQ(0, r);

       /* Check mode and get blocksize */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(AUMODE_PLAY, ai.mode);

       wavsize = ai.blocksize;
       wav = (char *)malloc(wavsize);
       REQUIRED_IF(wav != NULL);
       memset(wav, 0xff, wavsize);

       /* Write blocks until 1sec */
       for (totalsize = 0; totalsize < 8000; ) {
               r = WRITE(fd, wav, wavsize);
               XP_SYS_EQ(wavsize, r);
               if (r == -1)
                       break;  /* XXX */
               totalsize += r;
       }

       /* XXX What should I test it? */
       /* Check ai.play.error */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(0, ai.play.error);

       /* Playback data is no longer necessary */
       r = IOCTL(fd, AUDIO_FLUSH, NULL, "");
       REQUIRED_SYS_EQ(0, r);

       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);

       free(wav);
}

/*
* Normal recording.
* It does not verify real recorded data.
*/
DEF(read)
{
       char buf[8000];
       int fd;
       int r;

       TEST("read");

       fd = OPEN(devaudio, O_RDONLY);
       if (hw_canrec()) {
               REQUIRED_SYS_OK(fd);
       } else {
               XP_SYS_NG(ENXIO, fd);
               return;
       }

       /* mulaw 1sec */
       r = READ(fd, buf, sizeof(buf));
       XP_SYS_EQ(sizeof(buf), r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Repeat open-write-close cycle.
*/
DEF(rept_write)
{
       struct timeval start, end, result;
       double res;
       char buf[8000]; /* 1sec in 8bit-mulaw,1ch,8000Hz */
       int fd;
       int r;
       int n;

       TEST("rept_write");

       if (hw_canplay() == 0) {
               XP_SKIP("This test is only for playable device");
               return;
       }

       /* XXX It may timeout on some hardware driver. */
       XP_SKIP("not yet");
       return;

       memset(buf, 0xff, sizeof(buf));
       n = 3;
       gettimeofday(&start, NULL);
       for (int i = 0; i < n; i++) {
               fd = OPEN(devaudio, O_WRONLY);
               REQUIRED_SYS_OK(fd);

               r = WRITE(fd, buf, sizeof(buf));
               XP_SYS_EQ(sizeof(buf), r);

               r = CLOSE(fd);
               XP_SYS_EQ(0, r);
       }
       gettimeofday(&end, NULL);
       timersub(&end, &start, &result);
       res = (double)result.tv_sec + (double)result.tv_usec / 1000000;
       /* Make judgement but not too strict */
       if (res >= n * 1.5) {
               XP_FAIL("expects %d sec but %4.1f sec", n, res);
               return;
       }
}

/*
* Repeat open-read-close cycle.
*/
DEF(rept_read)
{
       struct timeval start, end, result;
       double res;
       char buf[8000]; /* 1sec in 8bit-mulaw,1ch,8000Hz */
       int fd;
       int r;
       int n;

       TEST("rept_read");

       if (hw_canrec() == 0) {
               XP_SKIP("This test is only for recordable device");
               return;
       }

       /* XXX It may timeout on some hardware driver. */
       XP_SKIP("not yet");
       return;

       n = 3;
       gettimeofday(&start, NULL);
       for (int i = 0; i < n; i++) {
               fd = OPEN(devaudio, O_RDONLY);
               REQUIRED_SYS_OK(fd);

               r = READ(fd, buf, sizeof(buf));
               XP_SYS_EQ(sizeof(buf), r);

               r = CLOSE(fd);
               XP_SYS_EQ(0, r);
       }
       gettimeofday(&end, NULL);
       timersub(&end, &start, &result);
       res = (double)result.tv_sec + (double)result.tv_usec / 1000000;
       /* Make judgement but not too strict */
       if (res >= n * 1.5) {
               XP_FAIL("expects %d sec but %4.1f sec", n, res);
               return;
       }
}

/*
* Opening with O_RDWR on half-duplex hardware falls back to O_WRONLY.
* expwrite: expected to be able to play.
* expread : expected to be able to record.
*/
void
test_rdwr_fallback(int openmode, bool expwrite, bool expread)
{
       struct audio_info ai;
       char buf[10];
       int fd;
       int r;

       TEST("rdwr_fallback_%s", openmode_str[openmode] + 2);

       if (hw_bidir() == 0) {
               XP_SKIP("This test is only for bi-directional device");
               return;
       }

       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       ai.record.pause = 1;

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /* Set pause not to play noise */
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause");
       REQUIRED_SYS_EQ(0, r);

       memset(buf, 0xff, sizeof(buf));
       r = WRITE(fd, buf, sizeof(buf));
       if (expwrite) {
               XP_SYS_EQ(sizeof(buf), r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       r = READ(fd, buf, 0);
       if (expread) {
               XP_SYS_EQ(0, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       r = CLOSE(fd);
       REQUIRED_SYS_EQ(0, r);
}
DEF(rdwr_fallback_RDONLY) { test_rdwr_fallback(O_RDONLY, false, true); }
DEF(rdwr_fallback_WRONLY) { test_rdwr_fallback(O_WRONLY, true, false); }
DEF(rdwr_fallback_RDWR) {
       bool expread;
       /*
        * On NetBSD7, O_RDWR on half-duplex is accepted. It's possible to
        * read and write if they don't occur at the same time.
        * On NetBSD9, O_RDWR on half-duplex falls back O_WRONLY.
        */
       if (netbsd < 8) {
               expread = true;
       } else {
               expread = hw_fulldup() ? true : false;
       }
       test_rdwr_fallback(O_RDWR, true, expread);
}

/*
* On full-duplex hardware, the second descriptor's readability/writability
* does not depend on the first descriptor's open mode.
* On half-duplex hardware, it depends on the first descriptor's open mode.
*/
void
test_rdwr_two(int mode0, int mode1)
{
       struct audio_info ai;
       char wbuf[100]; /* 1/80sec in 8bit-mulaw,1ch,8000Hz */
       char rbuf[100]; /* 1/80sec in 8bit-mulaw,1ch,8000Hz */
       bool canopen;
       bool canwrite;
       bool canread;
       int fd0;
       int fd1;
       int r;
       struct {
               bool canopen;
               bool canwrite;
               bool canread;
       } exptable_full[] = {
       /*      open write read    1st, 2nd mode */
               { 1, 0, 1 },    /* REC, REC */
               { 1, 1, 0 },    /* REC, PLAY */
               { 1, 1, 1 },    /* REC, BOTH */
               { 1, 0, 1 },    /* PLAY, REC */
               { 1, 1, 0 },    /* PLAY, PLAY */
               { 1, 1, 1 },    /* PLAY, BOTH */
               { 1, 0, 1 },    /* BOTH, REC */
               { 1, 1, 0 },    /* BOTH, PLAY */
               { 1, 1, 1 },    /* BOTH, BOTH */
       },
       exptable_half[] = {
               { 1, 0, 1 },    /* REC, REC */
               { 0, 0, 0 },    /* REC, PLAY */
               { 0, 0, 0 },    /* REC, BOTH */
               { 0, 0, 0 },    /* PLAY, REC */
               { 1, 1, 0 },    /* PLAY, PLAY */
               { 1, 1, 0 },    /* PLAY, BOTH */
               { 0, 0, 0 },    /* BOTH, REC */
               { 1, 1, 0 },    /* BOTH, PLAY */
               { 0, 0, 0 },    /* BOTH, BOTH */
       }, *exptable;

       TEST("rdwr_two_%s_%s",
           openmode_str[mode0] + 2,
           openmode_str[mode1] + 2);

       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (hw_bidir() == 0) {
               XP_SKIP("This test is only for bi-directional device");
               return;
       }

       exptable = hw_fulldup() ? exptable_full : exptable_half;

       canopen  = exptable[mode0 * 3 + mode1].canopen;
       canwrite = exptable[mode0 * 3 + mode1].canwrite;
       canread  = exptable[mode0 * 3 + mode1].canread;

       if (!canopen) {
               XP_SKIP("This combination is not openable on half-duplex");
               return;
       }

       fd0 = OPEN(devaudio, mode0);
       REQUIRED_SYS_OK(fd0);

       fd1 = OPEN(devaudio, mode1);
       REQUIRED_SYS_OK(fd1);

       /* Silent data to make no sound */
       memset(&wbuf, 0xff, sizeof(wbuf));
       /* Pause to make no sound */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd0, AUDIO_SETINFO, &ai, "pause");
       XP_SYS_EQ(0, r);

       /* write(fd1) */
       r = WRITE(fd1, wbuf, sizeof(wbuf));
       if (canwrite) {
               XP_SYS_EQ(100, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       /* read(fd1) */
       r = READ(fd1, rbuf, sizeof(rbuf));
       if (canread) {
               XP_SYS_EQ(100, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);
       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);
}
DEF(rdwr_two_RDONLY_RDONLY)     { test_rdwr_two(O_RDONLY, O_RDONLY);    }
DEF(rdwr_two_RDONLY_WRONLY)     { test_rdwr_two(O_RDONLY, O_WRONLY);    }
DEF(rdwr_two_RDONLY_RDWR)       { test_rdwr_two(O_RDONLY, O_RDWR);      }
DEF(rdwr_two_WRONLY_RDONLY)     { test_rdwr_two(O_WRONLY, O_RDONLY);    }
DEF(rdwr_two_WRONLY_WRONLY)     { test_rdwr_two(O_WRONLY, O_WRONLY);    }
DEF(rdwr_two_WRONLY_RDWR)       { test_rdwr_two(O_WRONLY, O_RDWR);      }
DEF(rdwr_two_RDWR_RDONLY)       { test_rdwr_two(O_RDWR, O_RDONLY);      }
DEF(rdwr_two_RDWR_WRONLY)       { test_rdwr_two(O_RDWR, O_WRONLY);      }
DEF(rdwr_two_RDWR_RDWR)         { test_rdwr_two(O_RDWR, O_RDWR);        }

/*
* Read and write different descriptors simultaneously.
* Only on full-duplex.
*/
DEF(rdwr_simul)
{
       char wbuf[1000];        /* 1/8sec in mulaw,1ch,8kHz */
       char rbuf[1000];
       int fd0;
       int fd1;
       int r;
       int status;
       pid_t pid;

       TEST("rdwr_simul");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (!hw_fulldup()) {
               XP_SKIP("This test is only for full-duplex device");
               return;
       }

       /* Silence data to make no sound */
       memset(wbuf, 0xff, sizeof(wbuf));

       fd0 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd0);
       fd1 = OPEN(devaudio, O_RDONLY);
       REQUIRED_SYS_OK(fd1);

       fflush(stdout);
       fflush(stderr);
       pid = fork();
       if (pid == -1)
               xp_err(1, __LINE__, "fork");

       if (pid == 0) {
               /* child (read) */
               for (int i = 0; i < 10; i++) {
                       r = READ(fd1, rbuf, sizeof(rbuf));
                       if (r == -1)
                               xp_err(1, __LINE__, "read(i=%d)", i);
               }
               exit(0);
       } else {
               /* parent (write) */
               for (int i = 0; i < 10; i++) {
                       r = WRITE(fd0, wbuf, sizeof(wbuf));
                       if (r == -1)
                               xp_err(1, __LINE__, "write(i=%d)", i);
               }
               waitpid(pid, &status, 0);
       }

       CLOSE(fd0);
       CLOSE(fd1);
       /* If you reach here, consider as success */
       XP_EQ(0, 0);
}

/*
* DRAIN should work even on incomplete data left.
*/
DEF(drain_incomplete)
{
       struct audio_info ai;
       int r;
       int fd;

       TEST("drain_incomplete");

       if (hw_canplay() == 0) {
               XP_SKIP("This test is only for playable device");
               return;
       }

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       AUDIO_INITINFO(&ai);
       /* let precision > 8 */
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
       ai.play.precision = 16;
       ai.mode = AUMODE_PLAY;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       /* Write one byte and then close */
       r = WRITE(fd, &r, 1);
       XP_SYS_EQ(1, r);
       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* DRAIN should work even in pause.
*/
DEF(drain_pause)
{
       struct audio_info ai;
       int r;
       int fd;

       TEST("drain_pause");

       if (hw_canplay() == 0) {
               XP_SKIP("This test is only for playable device");
               return;
       }

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       /* Set pause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);
       /* Write some data and then close */
       r = WRITE(fd, &r, 4);
       XP_SYS_EQ(4, r);
       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* DRAIN does not affect for record-only descriptor.
*/
DEF(drain_onrec)
{
       int fd;
       int r;

       TEST("drain_onrec");

       if (hw_canrec() == 0) {
               XP_SKIP("This test is only for recordable device");
               return;
       }

       fd = OPEN(devaudio, O_RDONLY);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Whether mmap() succeeds with specified parameter.
*/
void
test_mmap_mode(int mode, int prot)
{
       char buf[10];
       struct audio_info ai;
       const char *protstr;
       int expected;
       int fd;
       int r;
       int len;
       void *ptr;

       if (prot == PROT_NONE) {
               protstr = "NONE";
       } else if (prot == PROT_READ) {
               protstr = "READ";
       } else if (prot == PROT_WRITE) {
               protstr = "WRITE";
       } else if (prot == (PROT_READ | PROT_WRITE)) {
               protstr = "READWRITE";
       } else {
               xp_errx(1, __LINE__, "unknown prot %x\n", prot);
       }
       TEST("mmap_%s_%s", openmode_str[mode] + 2, protstr);
       if ((props & AUDIO_PROP_MMAP) == 0) {
               XP_SKIP("This test is only for mmap-able device");
               return;
       }
       if (mode2aumode(mode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }
#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       /*
        * On NetBSD7 and 8, mmap() always succeeds regardless of open mode.
        * On NetBSD9, mmap() succeeds only for writable descriptor.
        */
       expected = mode2play(mode);
       if (netbsd < 9) {
               expected = true;
       }

       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "get");
       REQUIRED_SYS_EQ(0, r);

       len = ai.play.buffer_size;

       /* Make it pause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause");
       REQUIRED_SYS_EQ(0, r);

       ptr = MMAP(NULL, len, prot, MAP_FILE, fd, 0);
       XP_SYS_PTR(expected ? 0 : EACCES, ptr);
       if (expected) {
               /* XXX Doing mmap(2) doesn't inhibit read(2) */
               if (mode2rec(mode)) {
                       r = READ(fd, buf, 0);
                       XP_SYS_EQ(0, r);
               }
               /* Doing mmap(2) inhibits write(2) */
               if (mode2play(mode)) {
                       /* NetBSD9 changes errno */
                       r = WRITE(fd, buf, 0);
                       if (netbsd < 9) {
                               XP_SYS_NG(EINVAL, r);
                       } else {
                               XP_SYS_NG(EPERM, r);
                       }
               }
       }
       if (ptr != MAP_FAILED) {
               r = MUNMAP(ptr, len);
               XP_SYS_EQ(0, r);
       }

       /* Whether the pause is still valid */
       if (mode2play(mode)) {
               r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
               XP_SYS_EQ(0, r);
               XP_EQ(1, ai.play.pause);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}
#define PROT_READWRITE  (PROT_READ | PROT_WRITE)
DEF(mmap_mode_RDONLY_NONE)      { test_mmap_mode(O_RDONLY, PROT_NONE); }
DEF(mmap_mode_RDONLY_READ)      { test_mmap_mode(O_RDONLY, PROT_READ); }
DEF(mmap_mode_RDONLY_WRITE)     { test_mmap_mode(O_RDONLY, PROT_WRITE); }
DEF(mmap_mode_RDONLY_READWRITE) { test_mmap_mode(O_RDONLY, PROT_READWRITE); }
DEF(mmap_mode_WRONLY_NONE)      { test_mmap_mode(O_WRONLY, PROT_NONE); }
DEF(mmap_mode_WRONLY_READ)      { test_mmap_mode(O_WRONLY, PROT_READ); }
DEF(mmap_mode_WRONLY_WRITE)     { test_mmap_mode(O_WRONLY, PROT_WRITE); }
DEF(mmap_mode_WRONLY_READWRITE) { test_mmap_mode(O_WRONLY, PROT_READWRITE); }
DEF(mmap_mode_RDWR_NONE)        { test_mmap_mode(O_RDWR, PROT_NONE); }
DEF(mmap_mode_RDWR_READ)        { test_mmap_mode(O_RDWR, PROT_READ); }
DEF(mmap_mode_RDWR_WRITE)       { test_mmap_mode(O_RDWR, PROT_WRITE); }
DEF(mmap_mode_RDWR_READWRITE)   { test_mmap_mode(O_RDWR, PROT_READWRITE); }

/*
* Check mmap()'s length and offset.
*
* Actual len and offset cannot be determined before open.  So that,
* pass pre-defined constant as argument, and convert it after open.
*/
#define LS      (100)   /* lsize     */
#define LS1     (101)   /* lsize + 1 */
void
test_mmap_len(size_t len, off_t offset, int exp)
{
       struct audio_info ai;
       int fd;
       int r;
       size_t plen;
       void *ptr;
       int bufsize;
       int pagesize;
       int lsize;

       TEST("mmap_len(%zd, %jd, %d)", len, offset, exp);
       if ((props & AUDIO_PROP_MMAP) == 0) {
               XP_SKIP("This test is only for mmap-able device");
               return;
       }
#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       plen = sizeof(pagesize);
       r = SYSCTLBYNAME("hw.pagesize", &pagesize, &plen, NULL, 0);
       REQUIRED_SYS_EQ(0, r);

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(r);

       /* Get buffer_size */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       bufsize = ai.play.buffer_size;

       /*
        * XXX someone refers bufsize and another one does pagesize.
        * I'm not sure.
        */
       lsize = roundup2(bufsize, pagesize);

       /* Here, lsize can be assigned */
       if (len == LS) {
               len = lsize;
       } else if (len == LS1) {
               len = lsize + 1;
       }
       if (offset == LS) {
               offset = lsize;
       } else if (offset == LS1) {
               offset = lsize + 1;
       }

       ptr = MMAP(NULL, len, PROT_WRITE, MAP_FILE, fd, offset);
       if (exp == 0) {
               XP_SYS_PTR(0, ptr);
       } else {
               /* NetBSD8 introduces EOVERFLOW */
               if (netbsd < 8 && exp == EOVERFLOW)
                       exp = EINVAL;
               XP_SYS_PTR(exp, ptr);
       }

       if (ptr != MAP_FAILED) {
               r = MUNMAP(ptr, len);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}
#define f(l, o, e)      test_mmap_len(l, o, e)
DEF(mmap_len_0) { f(0,   0,   EINVAL); }        /* len is 0 */
DEF(mmap_len_1) { f(1,   0,   0); }             /* len is smaller than lsize */
DEF(mmap_len_2) { f(LS,  0,   0); }             /* len is the same as lsize */
DEF(mmap_len_3) { f(LS1, 0,   EOVERFLOW); }     /* len is larger */
DEF(mmap_len_4) { f(0,   -1,  EINVAL); }        /* offset is negative */
DEF(mmap_len_5) { f(0,   LS,  EINVAL); }        /* len is 0 */
DEF(mmap_len_6) { f(0,   LS1, EINVAL); }        /* len is 0 */
DEF(mmap_len_7) { f(1,   LS,  EOVERFLOW); }     /* exceed */
/*
* When you treat the offset as 32bit, offset will be 0 and thus it
* incorrectly succeeds.
*/
DEF(mmap_len_8) { f(LS, 1ULL << 32, EOVERFLOW); }
#undef f

/*
* mmap() the same descriptor twice.
*/
DEF(mmap_twice)
{
       struct audio_info ai;
       int fd;
       int r;
       int len;
       void *ptr1;
       void *ptr2;

       TEST("mmap_twice");
       if ((props & AUDIO_PROP_MMAP) == 0) {
               XP_SKIP("This test is only for mmap-able device");
               return;
       }
#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "get");
       REQUIRED_SYS_EQ(0, r);
       len = ai.play.buffer_size;

       ptr1 = MMAP(NULL, len, PROT_WRITE, MAP_FILE, fd, 0);
       XP_SYS_PTR(0, ptr1);

       /* XXX I'm not sure this sucess is intended.  Anyway I follow it */
       ptr2 = MMAP(NULL, len, PROT_WRITE, MAP_FILE, fd, 0);
       XP_SYS_PTR(0, ptr2);

       if (ptr2 != MAP_FAILED) {
               r = MUNMAP(ptr2, len);
               XP_SYS_EQ(0, r);
       }
       if (ptr1 != MAP_FAILED) {
               r = MUNMAP(ptr1, len);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}

/*
* mmap() different descriptors.
*/
DEF(mmap_multi)
{
       struct audio_info ai;
       int fd0;
       int fd1;
       int r;
       int len;
       void *ptr0;
       void *ptr1;

       TEST("mmap_multi");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if ((props & AUDIO_PROP_MMAP) == 0) {
               XP_SKIP("This test is only for mmap-able device");
               return;
       }
#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       fd0 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd0);

       r = IOCTL(fd0, AUDIO_GETBUFINFO, &ai, "get");
       REQUIRED_SYS_EQ(0, r);
       len = ai.play.buffer_size;

       fd1 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd1);

       ptr0 = MMAP(NULL, len, PROT_WRITE, MAP_FILE, fd0, 0);
       XP_SYS_PTR(0, ptr0);

       ptr1 = MMAP(NULL, len,  PROT_WRITE, MAP_FILE, fd1, 0);
       XP_SYS_PTR(0, ptr1);

       if (ptr0 != MAP_FAILED) {
               r = MUNMAP(ptr1, len);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);

       if (ptr1 != MAP_FAILED) {
               r = MUNMAP(ptr0, len);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}

#define IN      POLLIN
#define OUT     POLLOUT
/*
* Whether poll() succeeds with specified mode.
*/
void
test_poll_mode(int mode, int events, int expected_revents)
{
       struct pollfd pfd;
       const char *events_str;
       int fd;
       int r;
       int expected_r;

       if (events == IN) {
               events_str = "IN";
       } else if (events == OUT) {
               events_str = "OUT";
       } else if (events == (IN | OUT)) {
               events_str = "INOUT";
       } else {
               events_str = "?";
       }
       TEST("poll_mode_%s_%s", openmode_str[mode] + 2, events_str);
       if (mode2aumode(mode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       expected_r = (expected_revents != 0) ? 1 : 0;

       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       /* Wait a bit to be recorded. */
       usleep(100 * 1000);

       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = events;

       r = POLL(&pfd, 1, 100);
       /* It's a bit complicated..  */
       if (r < 0 || r > 1) {
               /*
                * Check these two cases first:
                * - system call fails.
                * - poll() with one nfds returns >1.  It's strange.
                */
               XP_SYS_EQ(expected_r, r);
       } else {
               /*
                * Otherwise, poll() returned 0 or 1.
                */
               DPRINTF("  > pfd.revents=%s\n", event_tostr(pfd.revents));

               /* NetBSD7,8 have several strange behavior.  It must be bug. */

               XP_SYS_EQ(expected_r, r);
               XP_EQ(expected_revents, pfd.revents);
       }
       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(poll_mode_RDONLY_IN)        { test_poll_mode(O_RDONLY, IN,     IN); }
DEF(poll_mode_RDONLY_OUT)       { test_poll_mode(O_RDONLY, OUT,    0); }
DEF(poll_mode_RDONLY_INOUT)     { test_poll_mode(O_RDONLY, IN|OUT, IN); }
DEF(poll_mode_WRONLY_IN)        { test_poll_mode(O_WRONLY, IN,     0); }
DEF(poll_mode_WRONLY_OUT)       { test_poll_mode(O_WRONLY, OUT,    OUT); }
DEF(poll_mode_WRONLY_INOUT)     { test_poll_mode(O_WRONLY, IN|OUT, OUT); }
DEF(poll_mode_RDWR_IN)          {
       /* On half-duplex, O_RDWR is the same as O_WRONLY. */
       if (hw_fulldup()) test_poll_mode(O_RDWR,   IN,     IN);
       else              test_poll_mode(O_RDWR,   IN,     0);
}
DEF(poll_mode_RDWR_OUT)         { test_poll_mode(O_RDWR,   OUT,    OUT); }
DEF(poll_mode_RDWR_INOUT)       {
       /* On half-duplex, O_RDWR is the same as O_WRONLY. */
       if (hw_fulldup()) test_poll_mode(O_RDWR,   IN|OUT, IN|OUT);
       else              test_poll_mode(O_RDWR,   IN|OUT,    OUT);
}

/*
* Poll(OUT) when buffer is empty.
*/
DEF(poll_out_empty)
{
       struct pollfd pfd;
       int fd;
       int r;

       TEST("poll_out_empty");

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = POLLOUT;

       /* Check when empty.  It should succeed even if timeout == 0 */
       r = POLL(&pfd, 1, 0);
       XP_SYS_EQ(1, r);
       XP_EQ(POLLOUT, pfd.revents);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Poll(OUT) when buffer is full.
*/
DEF(poll_out_full)
{
       struct audio_info ai;
       struct pollfd pfd;
       int fd;
       int r;
       char *buf;
       int buflen;

       TEST("poll_out_full");

       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Pause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Get buffer size */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Write until full */
       buflen = ai.play.buffer_size;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       /* Do poll */
       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = POLLOUT;
       r = POLL(&pfd, 1, 0);
       XP_SYS_EQ(0, r);
       XP_EQ(0, pfd.revents);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* Poll(OUT) when buffer is full but hiwat sets lower than full.
*/
DEF(poll_out_hiwat)
{
       struct audio_info ai;
       struct pollfd pfd;
       int fd;
       int r;
       char *buf;
       int buflen;
       int newhiwat;

       TEST("poll_out_hiwat");

       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Get buffer size and hiwat */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       /* Change hiwat some different value */
       newhiwat = ai.lowat;

       /* Set pause and hiwat */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       ai.hiwat = newhiwat;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=1;hiwat");
       XP_SYS_EQ(0, r);

       /* Get the set hiwat again */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Write until full */
       buflen = ai.blocksize * ai.hiwat;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       /* Do poll */
       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = POLLOUT;
       r = POLL(&pfd, 1, 0);
       XP_SYS_EQ(0, r);
       XP_EQ(0, pfd.revents);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* Unpause from buffer full, POLLOUT should raise.
* XXX poll(2) on NetBSD7 is really incomplete and weird.  I don't test it.
*/
DEF(poll_out_unpause)
{
       struct audio_info ai;
       struct pollfd pfd;
       int fd;
       int r;
       char *buf;
       int buflen;
       u_int blocksize;
       int hiwat;
       int lowat;

       TEST("poll_out_unpause");
       if (netbsd < 8) {
               XP_SKIP("NetBSD7's poll() is too incomplete to test.");
               return;
       }

       /* Non-blocking open */
       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Adjust block size and hiwat/lowat to make the test time 1sec */
       blocksize = 1000;       /* 1/8 sec in mulaw,1ch,8000Hz */
       hiwat = 12;             /* 1.5sec */
       lowat = 4;              /* 0.5sec */
       AUDIO_INITINFO(&ai);
       ai.blocksize = blocksize;
       ai.hiwat = hiwat;
       ai.lowat = lowat;
       /* and also set encoding */
       /*
        * XXX NetBSD7 has different results depending on whether the input
        * encoding is emulated (AUDIO_ENCODINGFLAG_EMULATED) or not.  It's
        * not easy to ensure this situation on all hardware environment.
        * On NetBSD9, the result is the same regardless of input encoding.
        */
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "blocksize=%d", blocksize);
       XP_SYS_EQ(0, r);
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       if (ai.blocksize != blocksize) {
               /*
                * NetBSD9 can not change the blocksize.  Then,
                * adjust using hiwat/lowat.
                */
               blocksize = ai.blocksize;
               hiwat = howmany(8000 * 1.5, blocksize);
               lowat = howmany(8000 * 0.5, blocksize);
       }
       /* Anyway, set the parameters */
       AUDIO_INITINFO(&ai);
       ai.blocksize = blocksize;
       ai.hiwat = hiwat;
       ai.lowat = lowat;
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=1");
       XP_SYS_EQ(0, r);

       /* Get the set parameters again */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Write until full */
       buflen = ai.blocksize * ai.hiwat;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       /* At this time, POLLOUT should not be set because buffer is full */
       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = POLLOUT;
       r = POLL(&pfd, 1, 0);
       XP_SYS_EQ(0, r);
       XP_EQ(0, pfd.revents);

       /* Unpause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 0;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=0");
       XP_SYS_EQ(0, r);

       /*
        * When unpause occurs:
        * - NetBSD7 (emul=0) -> the buffer remains.
        * - NetBSD7 (emul=1) -> the buffer is cleared.
        * - NetBSD8          -> the buffer remains.
        * - NetBSD9          -> the buffer remains.
        */

       /* Check poll() up to 2sec */
       pfd.revents = 0;
       r = POLL(&pfd, 1, 2000);
       XP_SYS_EQ(1, r);
       XP_EQ(POLLOUT, pfd.revents);

       /*
        * Since POLLOUT is set, it should be writable.
        * But at this time, no all buffer may be writable.
        */
       r = WRITE(fd, buf, buflen);
       XP_SYS_OK(r);

       /* Flush it because there is no need to play it */
       r = IOCTL(fd, AUDIO_FLUSH, NULL, "");
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* poll(2) must not be affected by playback of other descriptors.
*/
DEF(poll_out_simul)
{
       struct audio_info ai;
       struct pollfd pfd[2];
       int fd[2];
       int r;
       char *buf;
       u_int blocksize;
       int hiwat;
       int lowat;
       int buflen;
       int time;

       TEST("poll_out_simul");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }

       /* Make sure that it's not affected by descriptor order */
       for (int i = 0; i < 2; i++) {
               int a = i;
               int b = 1 - i;

               fd[0] = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
               REQUIRED_SYS_OK(fd[0]);
               fd[1] = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
               REQUIRED_SYS_OK(fd[1]);

               /*
                * Adjust block size and hiwat/lowat.
                * I want to choice suitable blocksize (if possible).
                */
               blocksize = 1000;       /* 1/8 sec in mulaw,1ch,8000Hz */
               hiwat = 12;             /* 1.5sec */
               lowat = 4;              /* 0.5sec */
               AUDIO_INITINFO(&ai);
               ai.blocksize = blocksize;
               ai.hiwat = hiwat;
               ai.lowat = lowat;
               r = IOCTL(fd[0], AUDIO_SETINFO, &ai, "blocksize=1000");
               XP_SYS_EQ(0, r);
               r = IOCTL(fd[0], AUDIO_GETBUFINFO, &ai, "read back blocksize");
               if (ai.blocksize != blocksize) {
                       /*
                        * NetBSD9 can not change the blocksize.  Then,
                        * adjust using hiwat/lowat.
                        */
                       blocksize = ai.blocksize;
                       hiwat = howmany(8000 * 1.5, blocksize);
                       lowat = howmany(8000 * 0.5, blocksize);
               }
               /* Anyway, set the parameters */
               AUDIO_INITINFO(&ai);
               ai.blocksize = blocksize;
               ai.hiwat = hiwat;
               ai.lowat = lowat;
               /* Pause fdA */
               ai.play.pause = 1;
               r = IOCTL(fd[a], AUDIO_SETINFO, &ai, "pause=1");
               XP_SYS_EQ(0, r);
               /* Unpause fdB */
               ai.play.pause = 0;
               r = IOCTL(fd[b], AUDIO_SETINFO, &ai, "pause=0");
               XP_SYS_EQ(0, r);

               /* Get again. XXX two individual ioctls are correct */
               r = IOCTL(fd[0], AUDIO_GETBUFINFO, &ai, "");
               XP_SYS_EQ(0, r);
               DPRINTF("  > blocksize=%d lowat=%d hiwat=%d\n",
                       ai.blocksize, ai.lowat, ai.hiwat);

               /* Enough long time than the playback time */
               time = (ai.hiwat - ai.lowat) * blocksize / 8;  /*[msec]*/
               time *= 2;

               /* Write fdA full */
               buflen = blocksize * ai.lowat;
               buf = (char *)malloc(buflen);
               REQUIRED_IF(buf != NULL);
               memset(buf, 0xff, buflen);
               do {
                       r = WRITE(fd[a], buf, buflen);
               } while (r == buflen);
               if (r == -1) {
                       XP_SYS_NG(EAGAIN, r);
               }

               /* POLLOUT should not be set, because fdA is buffer full */
               memset(pfd, 0, sizeof(pfd));
               pfd[0].fd = fd[a];
               pfd[0].events = POLLOUT;
               r = POLL(pfd, 1, 0);
               XP_SYS_EQ(0, r);
               XP_EQ(0, pfd[0].revents);

               /* Write fdB at least lowat */
               r = WRITE(fd[b], buf, buflen);
               XP_SYS_EQ(buflen, r);
               r = WRITE(fd[b], buf, buflen);
               if (r == -1) {
                       XP_SYS_NG(EAGAIN, r);
               }

               /* Only fdB should become POLLOUT */
               memset(pfd, 0, sizeof(pfd));
               pfd[0].fd = fd[0];
               pfd[0].events = POLLOUT;
               pfd[1].fd = fd[1];
               pfd[1].events = POLLOUT;
               r = POLL(pfd, 2, time);
               XP_SYS_EQ(1, r);
               if (r != -1) {
                       XP_EQ(0, pfd[a].revents);
                       XP_EQ(POLLOUT, pfd[b].revents);
               }

               /* Drop the rest */
               r = IOCTL(fd[0], AUDIO_FLUSH, NULL, "");
               XP_SYS_EQ(0, r);
               r = IOCTL(fd[1], AUDIO_FLUSH, NULL, "");
               XP_SYS_EQ(0, r);

               r = CLOSE(fd[0]);
               XP_SYS_EQ(0, r);
               r = CLOSE(fd[1]);
               XP_SYS_EQ(0, r);
               free(buf);

               xxx_close_wait();
       }
}

/*
* Open with READ mode starts recording immediately.
* Of course, audioctl doesn't start.
*/
void
test_poll_in_open(const char *devname)
{
       struct audio_info ai;
       struct pollfd pfd;
       char buf[4096];
       char devfile[16];
       int fd;
       int r;
       bool is_audioctl;

       TEST("poll_in_open_%s", devname);
       if (hw_canrec() == 0) {
               XP_SKIP("This test is only for recordable device");
               return;
       }

       snprintf(devfile, sizeof(devfile), "/dev/%s%d", devname, unit);
       is_audioctl = (strcmp(devname, "audioctl") == 0);

       fd = OPEN(devfile, O_RDONLY);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       if (is_audioctl) {
               /* opening /dev/audioctl doesn't start recording. */
               XP_EQ(0, ai.record.active);
       } else {
               /* opening /dev/{audio,sound} starts recording. */
               /*
                * On NetBSD7/8, opening /dev/sound doesn't start recording.
                * It must be a bug.
                */
               XP_EQ(1, ai.record.active);
       }

       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd;
       pfd.events = POLLIN;
       r = POLL(&pfd, 1, 1000);
       if (is_audioctl) {
               /*
                * poll-ing /dev/audioctl always fails.
                * XXX Returning error instead of timeout should be better(?).
                */
               REQUIRED_SYS_EQ(0, r);
       } else {
               /*
                * poll-ing /dev/{audio,sound} will succeed when recorded
                * data is arrived.
                */
               /*
                * On NetBSD7/8, opening /dev/sound doesn't start recording.
                * It must be a bug.
                */
               REQUIRED_SYS_EQ(1, r);

               /* In this case, read() should succeed. */
               r = READ(fd, buf, sizeof(buf));
               XP_SYS_OK(r);
               XP_NE(0, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(poll_in_open_audio)         { test_poll_in_open("audio"); }
DEF(poll_in_open_sound)         { test_poll_in_open("sound"); }
DEF(poll_in_open_audioctl)      { test_poll_in_open("audioctl"); }

/*
* poll(2) must not be affected by other recording descriptors even if
* playback descriptor waits with POLLIN (though it's not normal usage).
* In other words, two POLLIN must not interfere.
*/
DEF(poll_in_simul)
{
       struct audio_info ai;
       struct pollfd pfd;
       int fd[2];
       int r;
       char *buf;
       int blocksize;

       TEST("poll_in_simul");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (hw_fulldup() == 0) {
               XP_SKIP("This test is only for full-duplex device");
               return;
       }

       int play = 0;
       int rec = 1;

       fd[play] = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd[play]);
       fd[rec] = OPEN(devaudio, O_RDONLY);
       REQUIRED_SYS_OK(fd[rec]);

       /* Get block size */
       r = IOCTL(fd[rec], AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       blocksize = ai.blocksize;

       buf = (char *)malloc(blocksize);
       REQUIRED_IF(buf != NULL);

       /*
        * At first, make sure the playback one doesn't return POLLIN.
        */
       memset(&pfd, 0, sizeof(pfd));
       pfd.fd = fd[play];
       pfd.events = POLLIN;
       r = POLL(&pfd, 1, 0);
       if (r == 0 && pfd.revents == 0) {
               XP_SYS_EQ(0, r);
               XP_EQ(0, pfd.revents);
       } else {
               XP_FAIL("play fd returns POLLIN");
               goto abort;
       }

       /* Start recording */
       r = READ(fd[rec], buf, blocksize);
       XP_SYS_EQ(blocksize, r);

       /* Poll()ing playback descriptor with POLLIN should not raise */
       r = POLL(&pfd, 1, 1000);
       XP_SYS_EQ(0, r);
       XP_EQ(0, pfd.revents);

       /* Poll()ing recording descriptor with POLLIN should raise */
       pfd.fd = fd[rec];
       r = POLL(&pfd, 1, 0);
       XP_SYS_EQ(1, r);
       XP_EQ(POLLIN, pfd.revents);

abort:
       r = CLOSE(fd[play]);
       XP_SYS_EQ(0, r);
       r = CLOSE(fd[rec]);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* Whether kqueue() succeeds with specified mode.
*/
void
test_kqueue_mode(int openmode, int filt, int expected)
{
       struct kevent kev;
       struct timespec ts;
       int fd;
       int kq;
       int r;

       TEST("kqueue_mode_%s_%s",
           openmode_str[openmode] + 2,
           (filt == EVFILT_READ) ? "READ" : "WRITE");
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       ts.tv_sec = 0;
       ts.tv_nsec = 100 * 1000 * 1000; // 100msec

       kq = KQUEUE();
       XP_SYS_OK(kq);

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /*
        * Check whether the specified filter can be set.
        * Any filters can always be set, even if pointless combination.
        * For example, EVFILT_READ can be set on O_WRONLY descriptor
        * though it will never raise.
        * I will not mention about good or bad of this behavior here.
        */
       EV_SET(&kev, fd, filt, EV_ADD, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       XP_SYS_EQ(0, r);

       if (r == 0) {
               /* If the filter can be set, try kevent(poll) */
               r = KEVENT_POLL(kq, &kev, 1, &ts);
               XP_SYS_EQ(expected, r);

               /* Delete it */
               EV_SET(&kev, fd, filt, EV_DELETE, 0, 0, 0);
               r = KEVENT_SET(kq, &kev, 1);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(kq);
       XP_SYS_EQ(0, r);
}
DEF(kqueue_mode_RDONLY_READ) {
       /* Should raise */
       test_kqueue_mode(O_RDONLY, EVFILT_READ, 1);
}
DEF(kqueue_mode_RDONLY_WRITE) {
       /* Should never raise (NetBSD7 has bugs) */
       int expected = (netbsd < 8) ? 1 : 0;
       test_kqueue_mode(O_RDONLY, EVFILT_WRITE, expected);
}
DEF(kqueue_mode_WRONLY_READ) {
       /* Should never raise */
       test_kqueue_mode(O_WRONLY, EVFILT_READ, 0);
}
DEF(kqueue_mode_WRONLY_WRITE) {
       /* Should raise */
       test_kqueue_mode(O_WRONLY, EVFILT_WRITE, 1);
}
DEF(kqueue_mode_RDWR_READ) {
       /* Should raise on fulldup but not on halfdup, on NetBSD9 */
       int expected = hw_fulldup() ? 1 : 0;
       test_kqueue_mode(O_RDWR, EVFILT_READ, expected);
}
DEF(kqueue_mode_RDWR_WRITE) {
       /* Should raise */
       test_kqueue_mode(O_RDWR, EVFILT_WRITE, 1);
}

/*
* kqueue(2) when buffer is empty.
*/
DEF(kqueue_empty)
{
       struct audio_info ai;
       struct kevent kev;
       struct timespec ts;
       int kq;
       int fd;
       int r;

       TEST("kqueue_empty");

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       kq = KQUEUE();
       XP_SYS_OK(kq);

       EV_SET(&kev, fd, EV_ADD, EVFILT_WRITE, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       XP_SYS_EQ(0, r);

       /* When the buffer is empty, it should succeed even if timeout == 0 */
       memset(&ts, 0, sizeof(ts));
       r = KEVENT_POLL(kq, &kev, 1, &ts);
       XP_SYS_EQ(1, r);
       XP_EQ(fd, kev.ident);
       /*
        * XXX According to kqueue(2) manpage, returned kev.data contains
        * "the amount of space remaining in the write buffer".
        * NetBSD7 returns buffer_size.  Shouldn't it be blocksize * hiwat?
        */
       /* XP_EQ(ai.blocksize * ai.hiwat, kev.data); */
       XP_EQ(ai.play.buffer_size, kev.data);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(kq);
       XP_SYS_EQ(0, r);
}

/*
* kqueue(2) when buffer is full.
*/
DEF(kqueue_full)
{
       struct audio_info ai;
       struct kevent kev;
       struct timespec ts;
       int kq;
       int fd;
       int r;
       char *buf;
       int buflen;

       TEST("kqueue_full");

       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Pause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Get buffer size */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Write until full */
       buflen = ai.play.buffer_size;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       kq = KQUEUE();
       XP_SYS_OK(kq);

       EV_SET(&kev, fd, EV_ADD, EVFILT_WRITE, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       XP_SYS_EQ(0, r);

       /* kevent() should not raise */
       ts.tv_sec = 0;
       ts.tv_nsec = 100L * 1000 * 1000;        /* 100msec */
       r = KEVENT_POLL(kq, &kev, 1, &ts);
       XP_SYS_EQ(0, r);
       if (r > 0) {
               XP_EQ(fd, kev.ident);
               XP_EQ(0, kev.data);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(kq);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* kqueue(2) when buffer is full but hiwat sets lower than full.
*/
DEF(kqueue_hiwat)
{
       struct audio_info ai;
       struct kevent kev;
       struct timespec ts;
       int kq;
       int fd;
       int r;
       char *buf;
       int buflen;
       int newhiwat;

       TEST("kqueue_hiwat");

       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Get buffer size and hiwat */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "hiwat");
       XP_SYS_EQ(0, r);
       /* Change hiwat some different value */
       newhiwat = ai.hiwat - 1;

       /* Set pause and hiwat */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       ai.hiwat = newhiwat;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=1;hiwat");
       XP_SYS_EQ(0, r);

       /* Get the set parameters again */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(1, ai.play.pause);
       XP_EQ(newhiwat, ai.hiwat);

       /* Write until full */
       buflen = ai.blocksize * ai.hiwat;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       kq = KQUEUE();
       XP_SYS_OK(kq);

       EV_SET(&kev, fd, EV_ADD, EVFILT_WRITE, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       XP_SYS_EQ(0, r);

       /* Should not raise because it's not possible to write */
       ts.tv_sec = 0;
       ts.tv_nsec = 100L * 1000 * 1000;        /* 100msec */
       r = KEVENT_POLL(kq, &kev, 1, &ts);
       if (r > 0)
               DEBUG_KEV("kev", &kev);
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(kq);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* Unpause from buffer full, kevent() should raise.
*/
DEF(kqueue_unpause)
{
       struct audio_info ai;
       struct kevent kev;
       struct timespec ts;
       int fd;
       int r;
       int kq;
       char *buf;
       int buflen;
       u_int blocksize;
       int hiwat;
       int lowat;

       TEST("kqueue_unpause");

       /* Non-blocking open */
       fd = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
       REQUIRED_SYS_OK(fd);

       /* Adjust block size and hiwat/lowat to make the test time 1sec */
       blocksize = 1000;       /* 1/8 sec in mulaw,1ch,8000Hz */
       hiwat = 12;             /* 1.5sec */
       lowat = 4;              /* 0.5sec */
       AUDIO_INITINFO(&ai);
       ai.blocksize = blocksize;
       ai.hiwat = hiwat;
       ai.lowat = lowat;
       /* and also set encoding */
       /*
        * XXX NetBSD7 has different results depending on whether the input
        * encoding is emulated (AUDIO_ENCODINGFLAG_EMULATED) or not.  It's
        * not easy to ensure this situation on all hardware environment.
        * On NetBSD9, the result is the same regardless of input encoding.
        */
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "blocksize=%d", blocksize);
       XP_SYS_EQ(0, r);
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       if (ai.blocksize != blocksize) {
               /*
                * NetBSD9 can not change the blocksize.  Then,
                * adjust using hiwat/lowat.
                */
               blocksize = ai.blocksize;
               hiwat = howmany(8000 * 1.5, blocksize);
               lowat = howmany(8000 * 0.5, blocksize);
       }
       /* Anyway, set the parameters */
       AUDIO_INITINFO(&ai);
       ai.blocksize = blocksize;
       ai.hiwat = hiwat;
       ai.lowat = lowat;
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=1");
       XP_SYS_EQ(0, r);

       /* Get the set parameters again */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       DPRINTF("  > blocksize=%d hiwat=%d lowat=%d buffer_size=%d\n",
           ai.blocksize, ai.hiwat, ai.lowat, ai.play.buffer_size);

       /* Write until full */
       buflen = ai.blocksize * ai.hiwat;
       buf = (char *)malloc(buflen);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buflen);
       do {
               r = WRITE(fd, buf, buflen);
       } while (r == buflen);
       if (r == -1) {
               XP_SYS_NG(EAGAIN, r);
       }

       kq = KQUEUE();
       XP_SYS_OK(kq);

       EV_SET(&kev, fd, EV_ADD, EVFILT_WRITE, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       XP_SYS_EQ(0, r);

       /* Unpause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 0;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=0");
       XP_SYS_EQ(0, r);

       /* Check kevent() up to 2sec */
       ts.tv_sec = 2;
       ts.tv_nsec = 0;
       r = KEVENT_POLL(kq, &kev, 1, &ts);
       if (r >= 1)
               DEBUG_KEV("kev", &kev);
       if (netbsd < 8) {
               /*
                * NetBSD7 with EMULATED_FLAG unset has bugs.  Unpausing
                * unintentionally clears buffer (and therefore it becomes
                * writable) but it doesn't raise EVFILT_WRITE.
                */
       } else {
               XP_SYS_EQ(1, r);
       }

       /* Flush it because there is no need to play it */
       r = IOCTL(fd, AUDIO_FLUSH, NULL, "");
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(kq);
       XP_SYS_EQ(0, r);
       free(buf);
}

/*
* kevent(2) must not be affected by other audio descriptors.
*/
DEF(kqueue_simul)
{
       struct audio_info ai;
       struct audio_info ai2;
       struct kevent kev[2];
       struct timespec ts;
       int fd[2];
       int r;
       int kq;
       u_int blocksize;
       int hiwat;
       int lowat;
       char *buf;
       int buflen;

       TEST("kqueue_simul");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }

       memset(&ts, 0, sizeof(ts));

       /* Make sure that it's not affected by descriptor order */
       for (int i = 0; i < 2; i++) {
               int a = i;
               int b = 1 - i;

               fd[0] = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
               REQUIRED_SYS_OK(fd[0]);
               fd[1] = OPEN(devaudio, O_WRONLY | O_NONBLOCK);
               REQUIRED_SYS_OK(fd[1]);

               /*
                * Adjust block size and hiwat/lowat.
                * I want to choice suitable blocksize (if possible).
                */
               blocksize = 1000;       /* 1/8 sec in mulaw,1ch,8000Hz */
               hiwat = 12;             /* 1.5sec */
               lowat = 4;              /* 0.5sec */
               AUDIO_INITINFO(&ai);
               ai.blocksize = blocksize;
               ai.hiwat = hiwat;
               ai.lowat = lowat;
               r = IOCTL(fd[0], AUDIO_SETINFO, &ai, "blocksize=1000");
               XP_SYS_EQ(0, r);
               r = IOCTL(fd[0], AUDIO_GETBUFINFO, &ai, "read back blocksize");
               if (ai.blocksize != blocksize) {
                       /*
                        * NetBSD9 can not change the blocksize.  Then,
                        * adjust using hiwat/lowat.
                        */
                       blocksize = ai.blocksize;
                       hiwat = howmany(8000 * 1.5, blocksize);
                       lowat = howmany(8000 * 0.5, blocksize);
               }
               /* Anyway, set the parameters to both */
               AUDIO_INITINFO(&ai);
               ai.blocksize = blocksize;
               ai.hiwat = hiwat;
               ai.lowat = lowat;
               ai.play.pause = 1;
               r = IOCTL(fd[a], AUDIO_SETINFO, &ai, "pause=1");
               XP_SYS_EQ(0, r);
               r = IOCTL(fd[b], AUDIO_SETINFO, &ai, "pause=1");
               XP_SYS_EQ(0, r);

               /* Write both until full */
               buflen = ai.blocksize * ai.hiwat;
               buf = (char *)malloc(buflen);
               REQUIRED_IF(buf != NULL);
               memset(buf, 0xff, buflen);
               /* Write fdA */
               do {
                       r = WRITE(fd[a], buf, buflen);
               } while (r == buflen);
               if (r == -1) {
                       XP_SYS_NG(EAGAIN, r);
               }
               /* Write fdB */
               do {
                       r = WRITE(fd[b], buf, buflen);
               } while (r == buflen);
               if (r == -1) {
                       XP_SYS_NG(EAGAIN, r);
               }

               /* Get fdB's initial seek for later */
               r = IOCTL(fd[b], AUDIO_GETBUFINFO, &ai2, "");
               XP_SYS_EQ(0, r);

               kq = KQUEUE();
               XP_SYS_OK(kq);

               /* Both aren't raised at this point */
               EV_SET(&kev[0], fd[a], EV_ADD, EVFILT_WRITE, 0, 0, 0);
               EV_SET(&kev[1], fd[b], EV_ADD, EVFILT_WRITE, 0, 0, 0);
               r = KEVENT_SET(kq, kev, 2);
               XP_SYS_EQ(0, r);

               /* Unpause only fdA */
               AUDIO_INITINFO(&ai);
               ai.play.pause = 0;
               r = IOCTL(fd[a], AUDIO_SETINFO, &ai, "pause=0");
               XP_SYS_EQ(0, r);

               /* kevent() up to 2sec */
               ts.tv_sec = 2;
               ts.tv_nsec = 0;
               r = KEVENT_POLL(kq, &kev[0], 1, &ts);
               if (r >= 1)
                       DEBUG_KEV("kev", &kev[0]);
               /* fdA should raise */
               XP_SYS_EQ(1, r);
               XP_EQ(fd[a], kev[0].ident);

               /* Make sure that fdB keeps whole data */
               r = IOCTL(fd[b], AUDIO_GETBUFINFO, &ai, "");
               XP_EQ(ai2.play.seek, ai.play.seek);

               /* Flush it because there is no need to play it */
               r = IOCTL(fd[0], AUDIO_FLUSH, NULL, "");
               XP_SYS_EQ(0, r);
               r = IOCTL(fd[1], AUDIO_FLUSH, NULL, "");
               XP_SYS_EQ(0, r);

               r = CLOSE(fd[0]);
               XP_SYS_EQ(0, r);
               r = CLOSE(fd[1]);
               XP_SYS_EQ(0, r);
               r = CLOSE(kq);
               XP_SYS_EQ(0, r);
               free(buf);

               xxx_close_wait();
       }
}

/* Shared data between threads for ioctl_while_write */
struct ioctl_while_write_data {
       int fd;
       struct timeval start;
       int terminated;
};

/* Test thread for ioctl_while_write */
void *thread_ioctl_while_write(void *);
void *
thread_ioctl_while_write(void *arg)
{
       struct ioctl_while_write_data *data = arg;
       struct timeval now, res;
       struct audio_info ai;
       int r;

       /* If 0.5 seconds have elapsed since writing, assume it's blocked */
       do {
               usleep(100);
               gettimeofday(&now, NULL);
               timersub(&now, &data->start, &res);
       } while (res.tv_usec < 500000);

       /* Then, do ioctl() */
       r = IOCTL(data->fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /* Terminate */
       data->terminated = 1;

       /* Resume write() by unpause */
       AUDIO_INITINFO(&ai);
       if (netbsd < 8) {
               /*
                * XXX NetBSD7 has bugs and it cannot be unpaused.
                * However, it also has another bug and it clears buffer
                * when encoding is changed.  I use it. :-P
                */
               ai.play.encoding = AUDIO_ENCODING_SLINEAR_LE;
       }
       ai.play.pause = 0;
       r = IOCTL(data->fd, AUDIO_SETINFO, &ai, "pause=0");
       XP_SYS_EQ(0, r);

       return NULL;
}

/*
* ioctl(2) can be issued while write(2)-ing.
*/
DEF(ioctl_while_write)
{
       struct audio_info ai;
       struct ioctl_while_write_data data0, *data;
       char buf[8000]; /* 1sec in mulaw,1ch,8000Hz */
       pthread_t tid;
       int r;

       TEST("ioctl_while_write");

       data = &data0;
       memset(data, 0, sizeof(*data));
       memset(buf, 0xff, sizeof(buf));

       data->fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(data->fd);

       /* Pause to block write(2)ing */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(data->fd, AUDIO_SETINFO, &ai, "pause=1");
       XP_SYS_EQ(0, r);

       gettimeofday(&data->start, NULL);

       pthread_create(&tid, NULL, thread_ioctl_while_write, data);

       /* Write until blocking */
       for (;;) {
               r = WRITE(data->fd, buf, sizeof(buf));
               if (data->terminated)
                       break;
               XP_SYS_EQ(sizeof(buf), r);

               /* Update written time */
               gettimeofday(&data->start, NULL);
       }

       pthread_join(tid, NULL);

       /* Flush */
       r = IOCTL(data->fd, AUDIO_FLUSH, NULL, "");
       XP_SYS_EQ(0, r);
       r = CLOSE(data->fd);
       XP_SYS_EQ(0, r);
}

volatile int sigio_caught;
void
signal_FIOASYNC(int signo)
{
       if (signo == SIGIO) {
               sigio_caught = 1;
               DPRINTF("  > %d: pid %d got SIGIO\n", __LINE__, (int)getpid());
       }
}

/*
* FIOASYNC between two descriptors should be split.
*/
DEF(FIOASYNC_reset)
{
       int fd0, fd1;
       int r;
       int val;

       TEST("FIOASYNC_reset");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }

       /* The first one opens */
       fd0 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd0);

       /* The second one opens, enables ASYNC, and closes */
       fd1 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd1);
       val = 1;
       r = IOCTL(fd1, FIOASYNC, &val, "on");
       XP_SYS_EQ(0, r);
       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);

       /* Again, the second one opens and enables ASYNC */
       fd1 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd1);
       val = 1;
       r = IOCTL(fd1, FIOASYNC, &val, "on");
       XP_SYS_EQ(0, r);        /* NetBSD8 fails */
       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);
       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);
}

/*
* Whether SIGIO is emitted on playback.
* XXX I don't understand conditions that NetBSD7 emits signal.
*/
DEF(FIOASYNC_play_signal)
{
       struct audio_info ai;
       int r;
       int fd;
       int val;
       char *data;
       int i;

       TEST("FIOASYNC_play_signal");
       if (hw_canplay() == 0) {
               XP_SKIP("This test is only for playable device");
               return;
       }

       signal(SIGIO, signal_FIOASYNC);
       sigio_caught = 0;

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       REQUIRED_IF(ai.blocksize != 0);
       data = (char *)malloc(ai.blocksize);
       REQUIRED_IF(data != NULL);
       memset(data, 0xff, ai.blocksize);

       val = 1;
       r = IOCTL(fd, FIOASYNC, &val, "on");
       XP_SYS_EQ(0, r);

       r = WRITE(fd, data, ai.blocksize);
       XP_SYS_EQ(ai.blocksize, r);

       /* Waits signal until 1sec */
       for (i = 0; i < 100 && sigio_caught == 0; i++) {
               usleep(10000);
       }
       signal(SIGIO, SIG_IGN);
       XP_EQ(1, sigio_caught);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       free(data);
       signal(SIGIO, SIG_IGN);
       sigio_caught = 0;
}

/*
* Whether SIGIO is emitted on recording.
*/
DEF(FIOASYNC_rec_signal)
{
       char buf[10];
       int r;
       int fd;
       int val;
       int i;

       TEST("FIOASYNC_rec_signal");
       if (hw_canrec() == 0) {
               XP_SKIP("This test is only for recordable device");
               return;
       }

       signal(SIGIO, signal_FIOASYNC);
       sigio_caught = 0;

       fd = OPEN(devaudio, O_RDONLY);
       REQUIRED_SYS_OK(fd);

       val = 1;
       r = IOCTL(fd, FIOASYNC, &val, "on");
       XP_SYS_EQ(0, r);

       r = READ(fd, buf, sizeof(buf));
       XP_SYS_EQ(sizeof(buf), r);

       /* Wait signal until 1sec */
       for (i = 0; i < 100 && sigio_caught == 0; i++) {
               usleep(10000);
       }
       signal(SIGIO, SIG_IGN);
       XP_EQ(1, sigio_caught);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       signal(SIGIO, SIG_IGN);
       sigio_caught = 0;
}

/*
* FIOASYNC doesn't affect other descriptor.
* For simplify, test only for playback...
*/
DEF(FIOASYNC_multi)
{
       struct audio_info ai;
       char *buf;
       char pipebuf[1];
       int r;
       int i;
       int fd1;
       int fd2;
       int pd[2];
       int val;
       pid_t pid;
       int status;

       TEST("FIOASYNC_multi");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (hw_canplay() == 0) {
               XP_SKIP("This test is only for playable device");
               return;
       }

       /* Pipe used between parent and child */
       r = pipe(pd);
       REQUIRED_SYS_EQ(0, r);

       fd1 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd1);
       fd2 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd2);

       /* Pause fd2 */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd2, AUDIO_SETINFO, &ai, "pause");
       REQUIRED_SYS_EQ(0, r);

       /* Fill both */
       r = IOCTL(fd1, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       REQUIRED_IF(ai.blocksize != 0);
       buf = (char *)malloc(ai.blocksize);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, ai.blocksize);
       r = WRITE(fd1, buf, ai.blocksize);
       XP_SYS_EQ(ai.blocksize, r);

       sigio_caught = 0;
       val = 1;

       fflush(stdout);
       fflush(stderr);
       pid = fork();
       if (pid == -1) {
               REQUIRED_SYS_OK(pid);
       }
       if (pid == 0) {
               /* Child */
               close(fd1);

               /* Child enables ASYNC on fd2 */
               signal(SIGIO, signal_FIOASYNC);
               r = IOCTL(fd2, FIOASYNC, &val, "on");
               /* It cannot count errors because here is a child process */
               /* XP_SYS_EQ(0, r); */

               /*
                * Waits signal until 1sec.
                * But fd2 is paused so it should never raise.
                */
               for (i = 0; i < 100 && sigio_caught == 0; i++) {
                       usleep(10000);
               }
               signal(SIGIO, SIG_IGN);
               pipebuf[0] = sigio_caught;
               /* This is not WRITE() macro here */
               write(pd[1], pipebuf, sizeof(pipebuf));

               /* XXX? */
               close(fd2);
               sleep(1);
               exit(0);
       } else {
               /* Parent */
               DPRINTF("  > fork() = %d\n", (int)pid);

               /* Parent enables ASYNC on fd1 */
               signal(SIGIO, signal_FIOASYNC);
               r = IOCTL(fd1, FIOASYNC, &val, "on");
               XP_SYS_EQ(0, r);

               /* Waits signal until 1sec */
               for (i = 0; i < 100 && sigio_caught == 0; i++) {
                       usleep(10000);
               }
               signal(SIGIO, SIG_IGN);
               XP_EQ(1, sigio_caught);

               /* Then read child's result from pipe */
               r = read(pd[0], pipebuf, sizeof(pipebuf));
               if (r != 1) {
                       XP_FAIL("reading from child failed");
               }
               DPRINTF("  > child's sigio_cauht = %d\n", pipebuf[0]);
               XP_EQ(0, pipebuf[0]);

               waitpid(pid, &status, 0);
       }

       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);
       r = CLOSE(fd2);
       XP_SYS_EQ(0, r);

       signal(SIGIO, SIG_IGN);
       sigio_caught = 0;
       free(buf);
}

/*
* Check AUDIO_WSEEK behavior.
*/
DEF(AUDIO_WSEEK)
{
       char buf[4];
       struct audio_info ai;
       int r;
       int fd;
       u_long n;

       TEST("AUDIO_WSEEK");

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       /* Pause to count sample data */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause=1");
       REQUIRED_SYS_EQ(0, r);

       /* On the initial state, it should be 0 bytes */
       n = 0;
       r = IOCTL(fd, AUDIO_WSEEK, &n, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, n);

       /* When writing 4 bytes, it should be 4 bytes */
       memset(buf, 0xff, sizeof(buf));
       r = WRITE(fd, buf, sizeof(buf));
       REQUIRED_EQ(sizeof(buf), r);
       r = IOCTL(fd, AUDIO_WSEEK, &n, "");
       XP_SYS_EQ(0, r);
       if (netbsd < 9) {
               /*
                * On NetBSD7, it will return 0.
                * Perhaps, WSEEK returns the number of pustream bytes but
                * data has already advanced...
                */
               XP_EQ(0, n);
       } else {
               /* Data less than one block remains here */
               XP_EQ(4, n);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check AUDIO_SETFD behavior for O_*ONLY descriptor.
* On NetBSD7, SETFD modify audio layer's state (and MD driver's state)
* regardless of open mode.  GETFD obtains audio layer's duplex.
* On NetBSD9, SETFD is obsoleted.  GETFD obtains hardware's duplex.
*/
void
test_AUDIO_SETFD_xxONLY(int openmode)
{
       struct audio_info ai;
       int r;
       int fd;
       int n;

       TEST("AUDIO_SETFD_%s", openmode_str[openmode] + 2);
       if (openmode == O_RDONLY && hw_canrec() == 0) {
               XP_SKIP("This test is for recordable device");
               return;
       }
       if (openmode == O_WRONLY && hw_canplay() == 0) {
               XP_SKIP("This test is for playable device");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /*
        * Just after open(2),
        * - On NetBSD7, it's always half-duplex.
        * - On NetBSD9, it's the same as hardware one regardless of openmode.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       if (netbsd < 9) {
               XP_EQ(0, n);
       } else {
               XP_EQ(hw_fulldup(), n);
       }

       /*
        * When trying to set to full-duplex,
        * - On NetBSD7, it will succeed if the hardware is full-duplex, or
        *   will fail if the hardware is half-duplex.
        * - On NetBSD9, it will always succeed but will not be modified.
        */
       n = 1;
       r = IOCTL(fd, AUDIO_SETFD, &n, "on");
       if (netbsd < 8) {
               if (hw_fulldup()) {
                       XP_SYS_EQ(0, r);
               } else {
                       XP_SYS_NG(ENOTTY, r);
               }
       } else if (netbsd == 8) {
               XP_FAIL("expected result is unknown");
       } else {
               XP_SYS_EQ(0, r);
       }

       /*
        * When obtain it,
        * - On NetBSD7, it will be 1 if the hardware is full-duplex or
        *   0 if half-duplex.
        * - On NetBSD9, it will never be changed because it's the hardware
        *   property.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       if (netbsd < 8) {
               XP_EQ(hw_fulldup(), n);
       } else if (netbsd == 8) {
               XP_FAIL("expected result is unknown");
       } else {
               XP_EQ(hw_fulldup(), n);
       }

       /* Some track parameters like ai.*.open should not change */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(mode2play(openmode), ai.play.open);
       XP_EQ(mode2rec(openmode), ai.record.open);

       /*
        * When trying to set to half-duplex,
        * - On NetBSD7, it will succeed if the hardware is full-duplex, or
        *   it will succeed with nothing happens.
        * - On NetBSD9, it will always succeed but nothing happens.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_SETFD, &n, "off");
       XP_SYS_EQ(0, r);

       /*
        * When obtain it again,
        * - On NetBSD7, it will be 0 if the hardware is full-duplex, or
        *   still 0 if half-duplex.
        * - On NetBSD9, it should not change.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       if (netbsd < 9) {
               XP_EQ(0, n);
       } else {
               XP_EQ(hw_fulldup(), n);
       }

       /* Some track parameters like ai.*.open should not change */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(mode2play(openmode), ai.play.open);
       XP_EQ(mode2rec(openmode), ai.record.open);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(AUDIO_SETFD_RDONLY) { test_AUDIO_SETFD_xxONLY(O_RDONLY); }
DEF(AUDIO_SETFD_WRONLY) { test_AUDIO_SETFD_xxONLY(O_WRONLY); }

/*
* Check AUDIO_SETFD behavior for O_RDWR descriptor.
*/
DEF(AUDIO_SETFD_RDWR)
{
       struct audio_info ai;
       int r;
       int fd;
       int n;

       TEST("AUDIO_SETFD_RDWR");
       if (!hw_fulldup()) {
               XP_SKIP("This test is only for full-duplex device");
               return;
       }

       fd = OPEN(devaudio, O_RDWR);
       REQUIRED_SYS_OK(fd);

       /*
        * - audio(4) manpage until NetBSD7 said "If a full-duplex capable
        *   audio device is opened for both reading and writing it will
        *   start in half-duplex play mode", but implementation doesn't
        *   seem to follow it.  It returns full-duplex.
        * - On NetBSD9, it should return full-duplex on full-duplex, or
        *   half-duplex on half-duplex.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       XP_EQ(hw_fulldup(), n);

       /*
        * When trying to set to full-duplex,
        * - On NetBSD7, it will succeed with nothing happens if full-duplex,
        *   or will fail if half-duplex.
        * - On NetBSD9, it will always succeed with nothing happens.
        */
       n = 1;
       r = IOCTL(fd, AUDIO_SETFD, &n, "on");
       if (netbsd < 9) {
               if (hw_fulldup()) {
                       XP_SYS_EQ(0, r);
               } else {
                       XP_SYS_NG(ENOTTY, r);
               }
       } else {
               XP_SYS_EQ(0, r);
       }

       /* When obtains it, it returns half/full-duplex as is */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       XP_EQ(hw_fulldup(), n);

       /* Some track parameters like ai.*.open should not change */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(1, ai.play.open);
       XP_EQ(mode2rec(O_RDWR), ai.record.open);

       /*
        * When trying to set to half-duplex,
        * - On NetBSD7, it will succeed if the hardware is full-duplex, or
        *   it will succeed with nothing happens.
        * - On NetBSD9, it will always succeed but nothing happens.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_SETFD, &n, "off");
       if (netbsd < 8) {
               XP_SYS_EQ(0, r);
       } else if (netbsd == 8) {
               XP_FAIL("expected result is unknown");
       } else {
               XP_SYS_EQ(0, r);
       }

       /*
        * When obtain it again,
        * - On NetBSD7, it will be 0 if the hardware is full-duplex, or
        *   still 0 if half-duplex.
        * - On NetBSD9, it should be 1 if the hardware is full-duplex, or
        *   0 if half-duplex.
        */
       n = 0;
       r = IOCTL(fd, AUDIO_GETFD, &n, "");
       XP_SYS_EQ(0, r);
       if (netbsd < 8) {
               XP_EQ(0, n);
       } else if (netbsd == 8) {
               XP_FAIL("expected result is unknown");
       } else {
               XP_EQ(hw_fulldup(), n);
       }

       /* Some track parameters like ai.*.open should not change */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(1, ai.play.open);
       XP_EQ(mode2rec(O_RDWR), ai.record.open);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check AUDIO_GETINFO.eof behavior.
*/
DEF(AUDIO_GETINFO_eof)
{
       struct audio_info ai;
       char buf[4];
       int r;
       int fd, fd1;

       TEST("AUDIO_GETINFO_eof");
       if (hw_canplay() == 0) {
               XP_SKIP("This test is for playable device");
               return;
       }

       fd = OPEN(devaudio, O_RDWR);
       REQUIRED_SYS_OK(fd);

       /* Pause to make no sound */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "pause");
       REQUIRED_SYS_EQ(0, r);

       /* It should be 0 initially */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, ai.play.eof);
       XP_EQ(0, ai.record.eof);

       /* Writing zero bytes should increment it */
       r = WRITE(fd, &r, 0);
       REQUIRED_SYS_OK(r);
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(1, ai.play.eof);
       XP_EQ(0, ai.record.eof);

       /* Writing one ore more bytes should noto increment it */
       memset(buf, 0xff, sizeof(buf));
       r = WRITE(fd, buf, sizeof(buf));
       REQUIRED_SYS_OK(r);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(1, ai.play.eof);
       XP_EQ(0, ai.record.eof);

       /* Writing zero bytes again should increment it */
       r = WRITE(fd, buf, 0);
       REQUIRED_SYS_OK(r);
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(2, ai.play.eof);
       XP_EQ(0, ai.record.eof);

       /* Reading zero bytes should not increment it */
       if (hw_fulldup()) {
               r = READ(fd, buf, 0);
               REQUIRED_SYS_OK(r);
               memset(&ai, 0, sizeof(ai));
               r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
               XP_SYS_EQ(0, r);
               XP_EQ(2, ai.play.eof);
               XP_EQ(0, ai.record.eof);
       }

       /* should not interfere with other descriptor */
       if (netbsd >= 8) {
               fd1 = OPEN(devaudio, O_RDWR);
               REQUIRED_SYS_OK(fd1);
               memset(&ai, 0, sizeof(ai));
               r = IOCTL(fd1, AUDIO_GETBUFINFO, &ai, "");
               XP_SYS_EQ(0, r);
               XP_EQ(0, ai.play.eof);
               XP_EQ(0, ai.record.eof);
               r = CLOSE(fd1);
               XP_SYS_EQ(0, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       xxx_close_wait();

       /* When reopen, it should reset the counter */
       fd = OPEN(devaudio, O_RDWR);
       REQUIRED_SYS_OK(fd);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, ai.play.eof);
       XP_EQ(0, ai.record.eof);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check relationship between openmode and mode set by AUDIO_SETINFO.
*/
void
test_AUDIO_SETINFO_mode(int openmode, int index, int setmode, int expected)
{
       struct audio_info ai;
       char buf[10];
       int inimode;
       int r;
       int fd;
       bool canwrite;
       bool canread;

       /* index was passed only for displaying here */
       TEST("AUDIO_SETINFO_mode_%s_%d", openmode_str[openmode] + 2, index);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       inimode = mode2aumode(openmode);

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /* When just after opening */
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       XP_EQ(inimode, ai.mode);
       XP_EQ(mode2play(openmode), ai.play.open);
       XP_EQ(mode2rec(openmode),  ai.record.open);
       XP_NE(0, ai.play.buffer_size);
       XP_NE(0, ai.record.buffer_size);

       /* Change mode (and pause here) */
       ai.mode = setmode;
       ai.play.pause = 1;
       ai.record.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "mode");
       XP_SYS_EQ(0, r);
       if (r == 0) {
               r = IOCTL(fd, AUDIO_GETINFO, &ai, "");
               XP_SYS_EQ(0, r);
               XP_EQ(expected, ai.mode);

               /* It seems to keep the initial openmode regardless of mode */
               XP_EQ(mode2play(openmode), ai.play.open);
               XP_EQ(mode2rec(openmode), ai.record.open);
               XP_NE(0, ai.play.buffer_size);
               XP_NE(0, ai.record.buffer_size);
       }

       /*
        * On NetBSD7, whether writable depends openmode when open.
        * On NetBSD9, whether writable should depend inimode when open.
        * Modifying after open should not affect this mode.
        */
       if (netbsd < 9) {
               canwrite = (openmode != O_RDONLY);
       } else {
               canwrite = ((inimode & AUMODE_PLAY) != 0);
       }
       r = WRITE(fd, buf, 0);
       if (canwrite) {
               XP_SYS_EQ(0, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       /*
        * On NetBSD7, whether readable depends openmode when open.
        * On NetBSD9, whether readable should depend inimode when open.
        * Modifying after open should not affect this mode.
        */
       if (netbsd < 9) {
               canread = (openmode != O_WRONLY);
       } else {
               canread = ((inimode & AUMODE_RECORD) != 0);
       }
       r = READ(fd, buf, 0);
       if (canread) {
               XP_SYS_EQ(0, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
/*
* XXX hmm... it's too complex
*/
/* shortcut for table form */
#define P       AUMODE_PLAY
#define A       AUMODE_PLAY_ALL
#define R       AUMODE_RECORD
struct setinfo_mode_t {
       int setmode;    /* mode used in SETINFO */
       int expmode7;   /* expected mode on NetBSD7 */
       int expmode9;   /* expected mode on NetBSD9 */
};
/*
* The following tables show this operation on NetBSD7 is almost 'undefined'.
* In contrast, NetBSD9 never changes mode by AUDIO_SETINFO except
* AUMODE_PLAY_ALL.
*
* setmode == 0 and 8 are out of range and invalid input samples.
* But NetBSD7 seems to accept it as is.
*/
struct setinfo_mode_t table_SETINFO_mode_O_RDONLY[] = {
       /* setmode      expmode7        expmode9 */
       {     0,             0,          R    },
       {     P,             P,          R    },
       {   A  ,           A|P,          R    },
       {   A|P,           A|P,          R    },
       { R    ,         R    ,          R    },
       { R|  P,             P,          R    },
       { R|A  ,           A|P,          R    },
       { R|A|P,           A|P,          R    },
       {     8,             8,          R    },
};
struct setinfo_mode_t table_SETINFO_mode_O_WRONLY[] = {
       /* setmode      expmode7        expmode9 */
       {     0,             0,              P },
       {     P,             P,              P },
       {   A  ,           A|P,            A|P },
       {   A|P,           A|P,            A|P },
       { R    ,         R    ,              P },
       { R|  P,             P,              P },
       { R|A  ,           A|P,            A|P },
       { R|A|P,           A|P,            A|P },
       {     8,             8,              P },
};
#define f(openmode, index)      do {                                    \
       struct setinfo_mode_t *table = table_SETINFO_mode_##openmode;   \
       int setmode = table[index].setmode;                             \
       int expected = (netbsd < 9)                                     \
           ? table[index].expmode7                                     \
           : table[index].expmode9;                                    \
       test_AUDIO_SETINFO_mode(openmode, index, setmode, expected);    \
} while (0)
DEF(AUDIO_SETINFO_mode_RDONLY_0) { f(O_RDONLY, 0); }
DEF(AUDIO_SETINFO_mode_RDONLY_1) { f(O_RDONLY, 1); }
DEF(AUDIO_SETINFO_mode_RDONLY_2) { f(O_RDONLY, 2); }
DEF(AUDIO_SETINFO_mode_RDONLY_3) { f(O_RDONLY, 3); }
DEF(AUDIO_SETINFO_mode_RDONLY_4) { f(O_RDONLY, 4); }
DEF(AUDIO_SETINFO_mode_RDONLY_5) { f(O_RDONLY, 5); }
DEF(AUDIO_SETINFO_mode_RDONLY_6) { f(O_RDONLY, 6); }
DEF(AUDIO_SETINFO_mode_RDONLY_7) { f(O_RDONLY, 7); }
DEF(AUDIO_SETINFO_mode_RDONLY_8) { f(O_RDONLY, 8); }
DEF(AUDIO_SETINFO_mode_WRONLY_0) { f(O_WRONLY, 0); }
DEF(AUDIO_SETINFO_mode_WRONLY_1) { f(O_WRONLY, 1); }
DEF(AUDIO_SETINFO_mode_WRONLY_2) { f(O_WRONLY, 2); }
DEF(AUDIO_SETINFO_mode_WRONLY_3) { f(O_WRONLY, 3); }
DEF(AUDIO_SETINFO_mode_WRONLY_4) { f(O_WRONLY, 4); }
DEF(AUDIO_SETINFO_mode_WRONLY_5) { f(O_WRONLY, 5); }
DEF(AUDIO_SETINFO_mode_WRONLY_6) { f(O_WRONLY, 6); }
DEF(AUDIO_SETINFO_mode_WRONLY_7) { f(O_WRONLY, 7); }
DEF(AUDIO_SETINFO_mode_WRONLY_8) { f(O_WRONLY, 8); }
#undef f
/*
* The following tables also show that NetBSD7's behavior is almost
* 'undefined'.
*/
struct setinfo_mode_t table_SETINFO_mode_O_RDWR_full[] = {
       /* setmode      expmode7        expmode9 */
       {     0,            0,          R|  P },
       {     P,            P,          R|  P },
       {   A  ,          A|P,          R|A|P },
       {   A|P,          A|P,          R|A|P },
       { R    ,        R    ,          R|  P },
       { R|  P,        R|  P,          R|  P },
       { R|A  ,        R|A|P,          R|A|P },
       { R|A|P,        R|A|P,          R|A|P },
       {     8,            8,          R|  P },
};
struct setinfo_mode_t table_SETINFO_mode_O_RDWR_half[] = {
       /* setmode      expmode7        expmode9 */
       {     0,            0,              P },
       {     P,            P,              P },
       {   A  ,          A|P,            A|P },
       {   A|P,          A|P,            A|P },
       { R    ,        R    ,              P },
       { R|  P,            P,              P },
       { R|A  ,          A|P,            A|P },
       { R|A|P,          A|P,            A|P },
       {     8,            8,              P },
};
#define f(index)        do {                                            \
       struct setinfo_mode_t *table = (hw_fulldup())                   \
           ? table_SETINFO_mode_O_RDWR_full                            \
           : table_SETINFO_mode_O_RDWR_half;                           \
       int setmode = table[index].setmode;                             \
       int expected = (netbsd < 9)                                     \
           ? table[index].expmode7                                     \
           : table[index].expmode9;                                    \
       test_AUDIO_SETINFO_mode(O_RDWR, index, setmode, expected);      \
} while (0)
DEF(AUDIO_SETINFO_mode_RDWR_0) { f(0); }
DEF(AUDIO_SETINFO_mode_RDWR_1) { f(1); }
DEF(AUDIO_SETINFO_mode_RDWR_2) { f(2); }
DEF(AUDIO_SETINFO_mode_RDWR_3) { f(3); }
DEF(AUDIO_SETINFO_mode_RDWR_4) { f(4); }
DEF(AUDIO_SETINFO_mode_RDWR_5) { f(5); }
DEF(AUDIO_SETINFO_mode_RDWR_6) { f(6); }
DEF(AUDIO_SETINFO_mode_RDWR_7) { f(7); }
DEF(AUDIO_SETINFO_mode_RDWR_8) { f(8); }
#undef f
#undef P
#undef A
#undef R

/*
* Check whether encoding params can be set.
*/
void
test_AUDIO_SETINFO_params_set(int openmode, int aimode, int pause)
{
       struct audio_info ai;
       int r;
       int fd;

       /*
        * aimode is bool value that indicates whether to change ai.mode.
        * pause is bool value that indicates whether to change ai.*.pause.
        */

       TEST("AUDIO_SETINFO_params_%s_%d_%d",
           openmode_str[openmode] + 2, aimode, pause);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       /* On half-duplex, O_RDWR is the same as O_WRONLY, so skip it */
       if (!hw_fulldup() && openmode == O_RDWR) {
               XP_SKIP("This is the same with O_WRONLY on half-duplex");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       AUDIO_INITINFO(&ai);
       /*
        * It takes time and effort to check all parameters independently,
        * so that use sample_rate as a representative.
        */
       ai.play.sample_rate = 11025;
       ai.record.sample_rate = 11025;
       if (aimode)
               ai.mode = mode2aumode(openmode) & ~AUMODE_PLAY_ALL;
       if (pause) {
               ai.play.pause = 1;
               ai.record.pause = 1;
       }

       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       int expmode = (aimode)
           ? (mode2aumode(openmode) & ~AUMODE_PLAY_ALL)
           : mode2aumode(openmode);
       XP_EQ(expmode, ai.mode);
       XP_EQ(11025, ai.play.sample_rate);
       XP_EQ(pause, ai.play.pause);
       XP_EQ(11025, ai.record.sample_rate);
       XP_EQ(pause, ai.record.pause);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
#define f(a,b,c) test_AUDIO_SETINFO_params_set(a, b, c)
DEF(AUDIO_SETINFO_params_set_RDONLY_0)  { f(O_RDONLY, 0, 0); }
DEF(AUDIO_SETINFO_params_set_RDONLY_1)  { f(O_RDONLY, 0, 1); }
/* On RDONLY, ai.mode is not changeable
*  AUDIO_SETINFO_params_set_RDONLY_2)  { f(O_RDONLY, 1, 0); }
*  AUDIO_SETINFO_params_set_RDONLY_3)  { f(O_RDONLY, 1, 1); }
*/
DEF(AUDIO_SETINFO_params_set_WRONLY_0)  { f(O_WRONLY, 0, 0); }
DEF(AUDIO_SETINFO_params_set_WRONLY_1)  { f(O_WRONLY, 0, 1); }
DEF(AUDIO_SETINFO_params_set_WRONLY_2)  { f(O_WRONLY, 1, 0); }
DEF(AUDIO_SETINFO_params_set_WRONLY_3)  { f(O_WRONLY, 1, 1); }
DEF(AUDIO_SETINFO_params_set_RDWR_0)    { f(O_RDWR, 0, 0); }
DEF(AUDIO_SETINFO_params_set_RDWR_1)    { f(O_RDWR, 0, 1); }
DEF(AUDIO_SETINFO_params_set_RDWR_2)    { f(O_RDWR, 1, 0); }
DEF(AUDIO_SETINFO_params_set_RDWR_3)    { f(O_RDWR, 1, 1); }
#undef f

/*
* AUDIO_SETINFO for existing track should not be interfered by other
* descriptor.
* AUDIO_SETINFO for non-existing track affects/is affected sticky parameters
* for backward compatibility.
*/
DEF(AUDIO_SETINFO_params_simul)
{
       struct audio_info ai;
       int fd0;
       int fd1;
       int r;

       TEST("AUDIO_SETINFO_params_simul");
       if (netbsd < 8) {
               XP_SKIP("Multiple open is not supported");
               return;
       }
       if (hw_canplay() == 0) {
               XP_SKIP("This test is for playable device");
               return;
       }

       /* Open the 1st one as playback only */
       fd0 = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd0);

       /* Open the 2nd one as both of playback and recording */
       fd1 = OPEN(devaudio, O_RDWR);
       REQUIRED_SYS_OK(fd1);

       /* Change some parameters of both track on the 2nd one */
       AUDIO_INITINFO(&ai);
       ai.play.sample_rate = 11025;
       ai.record.sample_rate = 11025;
       r = IOCTL(fd1, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /*
        * The 1st one doesn't have recording track so that only recording
        * parameter is affected by sticky parameter.
        */
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd0, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(8000, ai.play.sample_rate);
       XP_EQ(11025, ai.record.sample_rate);

       /* Next, change some parameters of both track on the 1st one */
       AUDIO_INITINFO(&ai);
       ai.play.sample_rate = 16000;
       ai.record.sample_rate = 16000;
       r = IOCTL(fd0, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       /*
        * On full-duplex device, the 2nd one has both track so that
        * both track are not affected by sticky parameter.
        * Otherwise, the 2nd one has only playback track so that
        * playback track is not affected by sticky parameter.
        */
       memset(&ai, 0, sizeof(ai));
       r = IOCTL(fd1, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(11025, ai.play.sample_rate);
       if (hw_fulldup()) {
               XP_EQ(11025, ai.record.sample_rate);
       } else {
               XP_EQ(16000, ai.record.sample_rate);
       }

       r = CLOSE(fd0);
       XP_SYS_EQ(0, r);
       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);
}

/*
* AUDIO_SETINFO(encoding/precision) is tested in AUDIO_GETENC_range below.
*/

/*
* Check whether the number of channels can be set.
*/
DEF(AUDIO_SETINFO_channels)
{
       struct audio_info hwinfo;
       struct audio_info ai;
       int mode;
       int r;
       int fd;
       int i;
       unsigned int ch;
       struct {
               int ch;
               bool expected;
       } table[] = {
               {  0,   false },
               {  1,   true }, /* monaural */
               {  2,   true }, /* stereo */
       };

       TEST("AUDIO_SETINFO_channels");
       if (netbsd < 8) {
               /*
                * On NetBSD7, the result depends the hardware and there is
                * no way to know it.
                */
               XP_SKIP("The test doesn't make sense on NetBSD7");
               return;
       }

       mode = openable_mode();
       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       /*
        * The audio layer always supports monaural and stereo regardless of
        * the hardware capability.
        */
       for (i = 0; i < (int)__arraycount(table); i++) {
               ch = table[i].ch;
               bool expected = table[i].expected;

               AUDIO_INITINFO(&ai);
               if (mode != O_RDONLY)
                       ai.play.channels = ch;
               if (mode != O_WRONLY)
                       ai.record.channels = ch;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=%d", ch);
               if (expected) {
                       /* Expects to succeed */
                       XP_SYS_EQ(0, r);

                       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
                       XP_SYS_EQ(0, r);
                       if (mode != O_RDONLY)
                               XP_EQ(ch, ai.play.channels);
                       if (mode != O_WRONLY)
                               XP_EQ(ch, ai.record.channels);
               } else {
                       /* Expects to fail */
                       XP_SYS_NG(EINVAL, r);
               }
       }

       /*
        * The maximum number of supported channels depends the hardware.
        */
       /* Get the number of channels that the hardware supports */
       r = IOCTL(fd, AUDIO_GETFORMAT, &hwinfo, "");
       REQUIRED_SYS_EQ(0, r);

       if ((hwinfo.mode & AUMODE_PLAY)) {
               DPRINTF("  > hwinfo.play.channels = %d\n",
                   hwinfo.play.channels);
               for (ch = 3; ch <= hwinfo.play.channels; ch++) {
                       AUDIO_INITINFO(&ai);
                       ai.play.channels = ch;
                       r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=%d", ch);
                       XP_SYS_EQ(0, r);

                       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
                       XP_SYS_EQ(0, r);
                       XP_EQ(ch, ai.play.channels);
               }

               AUDIO_INITINFO(&ai);
               ai.play.channels = ch;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=%d", ch);
               XP_SYS_NG(EINVAL, r);
       }
       if ((hwinfo.mode & AUMODE_RECORD)) {
               DPRINTF("  > hwinfo.record.channels = %d\n",
                   hwinfo.record.channels);
               for (ch = 3; ch <= hwinfo.record.channels; ch++) {
                       AUDIO_INITINFO(&ai);
                       ai.record.channels = ch;
                       r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=%d", ch);
                       XP_SYS_EQ(0, r);

                       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
                       XP_SYS_EQ(0, r);
                       XP_EQ(ch, ai.record.channels);
               }

               AUDIO_INITINFO(&ai);
               ai.record.channels = ch;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=%d", ch);
               XP_SYS_NG(EINVAL, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check whether the sample rate can be set.
*/
DEF(AUDIO_SETINFO_sample_rate)
{
       struct audio_info ai;
       int mode;
       int r;
       int fd;
       int i;
       struct {
               int freq;
               bool expected;
       } table[] = {
               {    999,       false },
               {   1000,       true }, /* lower limit */
               {  48000,       true },
               { 192000,       true }, /* upper limit */
               { 192001,       false },
       };

       TEST("AUDIO_SETINFO_sample_rate");
       if (netbsd < 8) {
               /*
                * On NetBSD7, the result depends the hardware and there is
                * no way to know it.
                */
               XP_SKIP("The test doesn't make sense on NetBSD7");
               return;
       }

       mode = openable_mode();
       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       for (i = 0; i < (int)__arraycount(table); i++) {
               int freq = table[i].freq;
               bool expected = table[i].expected;

               AUDIO_INITINFO(&ai);
               if (mode != O_RDONLY)
                       ai.play.sample_rate = freq;
               if (mode != O_WRONLY)
                       ai.record.sample_rate = freq;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "sample_rate=%d", freq);
               if (expected) {
                       /* Expects to succeed */
                       XP_SYS_EQ(0, r);

                       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
                       XP_SYS_EQ(0, r);
                       if (mode != O_RDONLY)
                               XP_EQ(freq, ai.play.sample_rate);
                       if (mode != O_WRONLY)
                               XP_EQ(freq, ai.record.sample_rate);
               } else {
                       /* Expects to fail */
                       XP_SYS_NG(EINVAL, r);
               }
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* SETINFO(sample_rate = 0) should fail correctly.
*/
DEF(AUDIO_SETINFO_sample_rate_0)
{
       struct audio_info ai;
       int mode;
       int r;
       int fd;

       TEST("AUDIO_SETINFO_sample_rate_0");
       if (netbsd < 9) {
               /*
                * On NetBSD7,8 this will block system call and you will not
                * even be able to shutdown...
                */
               XP_SKIP("This will cause an infinite loop in the kernel");
               return;
       }

       mode = openable_mode();
       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       AUDIO_INITINFO(&ai);
       ai.play.sample_rate = 0;
       ai.record.sample_rate = 0;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "sample_rate=0");
       /* Expects to fail */
       XP_SYS_NG(EINVAL, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check whether the pause/unpause works.
*/
void
test_AUDIO_SETINFO_pause(int openmode, int aimode, int param)
{
       struct audio_info ai;
       int r;
       int fd;

       /*
        * aimode is bool value that indicates whether to change ai.mode.
        * param is bool value that indicates whether to change encoding
        * parameters of ai.{play,record}.*.
        */

       TEST("AUDIO_SETINFO_pause_%s_%d_%d",
           openmode_str[openmode] + 2, aimode, param);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       /* On half-duplex, O_RDWR is the same as O_WRONLY, so skip it */
       if (!hw_fulldup() && openmode == O_RDWR) {
               XP_SKIP("This is the same with O_WRONLY on half-duplex");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /* Set pause */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       ai.record.pause = 1;
       if (aimode)
               ai.mode = mode2aumode(openmode) & ~AUMODE_PLAY_ALL;
       if (param) {
               ai.play.sample_rate = 11025;
               ai.record.sample_rate = 11025;
       }

       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       int expmode = (aimode)
           ? (mode2aumode(openmode) & ~AUMODE_PLAY_ALL)
           : mode2aumode(openmode);
       XP_EQ(expmode, ai.mode);
       XP_EQ(1, ai.play.pause);
       XP_EQ(param ? 11025 : 8000, ai.play.sample_rate);
       XP_EQ(1, ai.record.pause);
       XP_EQ(param ? 11025 : 8000, ai.record.sample_rate);

       /* Set unpause (?) */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 0;
       ai.record.pause = 0;
       if (aimode)
               ai.mode = mode2aumode(openmode);
       if (param) {
               ai.play.sample_rate = 16000;
               ai.record.sample_rate = 16000;
       }

       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       XP_SYS_EQ(0, r);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(mode2aumode(openmode), ai.mode);
       XP_EQ(0, ai.play.pause);
       XP_EQ(0, ai.record.pause);
       if (openmode != O_RDONLY)
               XP_EQ(param ? 16000 : 8000, ai.play.sample_rate);
       if (openmode != O_WRONLY)
               XP_EQ(param ? 16000 : 8000, ai.record.sample_rate);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(AUDIO_SETINFO_pause_RDONLY_0) { test_AUDIO_SETINFO_pause(O_RDONLY, 0, 0); }
DEF(AUDIO_SETINFO_pause_RDONLY_1) { test_AUDIO_SETINFO_pause(O_RDONLY, 0, 1); }
/* On RDONLY, ai.mode is not changeable
*  AUDIO_SETINFO_pause_RDONLY_2) { test_AUDIO_SETINFO_pause(O_RDONLY, 1, 0); }
*  AUDIO_SETINFO_pause_RDONLY_3) { test_AUDIO_SETINFO_pause(O_RDONLY, 1, 1); }
*/
DEF(AUDIO_SETINFO_pause_WRONLY_0) { test_AUDIO_SETINFO_pause(O_WRONLY, 0, 0); }
DEF(AUDIO_SETINFO_pause_WRONLY_1) { test_AUDIO_SETINFO_pause(O_WRONLY, 0, 1); }
DEF(AUDIO_SETINFO_pause_WRONLY_2) { test_AUDIO_SETINFO_pause(O_WRONLY, 1, 0); }
DEF(AUDIO_SETINFO_pause_WRONLY_3) { test_AUDIO_SETINFO_pause(O_WRONLY, 1, 1); }
DEF(AUDIO_SETINFO_pause_RDWR_0)   { test_AUDIO_SETINFO_pause(O_RDWR, 0, 0); }
DEF(AUDIO_SETINFO_pause_RDWR_1)   { test_AUDIO_SETINFO_pause(O_RDWR, 0, 1); }
DEF(AUDIO_SETINFO_pause_RDWR_2)   { test_AUDIO_SETINFO_pause(O_RDWR, 1, 0); }
DEF(AUDIO_SETINFO_pause_RDWR_3)   { test_AUDIO_SETINFO_pause(O_RDWR, 1, 1); }

/*
* Check whether gain can be obtained/set.
* And the gain should work with rich mixer.
* PR kern/52781
*/
DEF(AUDIO_SETINFO_gain)
{
       struct audio_info ai;
       mixer_ctrl_t m;
       int index;
       int master;
       int master_backup;
       int gain;
       int fd;
       int mixerfd;
       int r;

       TEST("AUDIO_SETINFO_gain");

       /* Open /dev/mixer */
       mixerfd = OPEN(devmixer, O_RDWR);
       REQUIRED_SYS_OK(mixerfd);
       index = mixer_get_outputs_master(mixerfd);
       if (index == -1) {
               XP_SKIP("Hardware has no outputs.master");
               CLOSE(mixerfd);
               return;
       }

       /*
        * Get current outputs.master.
        * auich(4) requires class type (m.type) and number of channels
        * (un.value.num_channels) in addition to the index (m.dev)...
        * What is the index...?
        */
       memset(&m, 0, sizeof(m));
       m.dev = index;
       m.type = AUDIO_MIXER_VALUE;
       m.un.value.num_channels = 1; /* dummy */
       r = IOCTL(mixerfd, AUDIO_MIXER_READ, &m, "m.dev=%d", m.dev);
       REQUIRED_SYS_EQ(0, r);
       master = m.un.value.level[0];
       DPRINTF("  > outputs.master = %d\n", master);
       master_backup = master;

       /* Open /dev/audio */
       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       /* Check ai.play.gain */
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "");
       XP_SYS_EQ(0, r);
       XP_EQ(master, ai.play.gain);

       /* Change it some different value */
       AUDIO_INITINFO(&ai);
       if (master == 0)
               gain = 255;
       else
               gain = 0;
       ai.play.gain = gain;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "play.gain=%d", ai.play.gain);
       XP_SYS_EQ(0, r);

       /* Check gain has changed */
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "play.gain");
       XP_SYS_EQ(0, r);
       XP_NE(master, ai.play.gain);

       /* Check whether outputs.master work with gain */
       r = IOCTL(mixerfd, AUDIO_MIXER_READ, &m, "");
       XP_SYS_EQ(0, r);
       XP_EQ(ai.play.gain, m.un.value.level[0]);

       /* Restore outputs.master */
       AUDIO_INITINFO(&ai);
       ai.play.gain = master_backup;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "play.gain=%d", ai.play.gain);
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
       r = CLOSE(mixerfd);
       XP_SYS_EQ(0, r);
}

/*
* Look if there are any (non-zero) gain values that can be changed.
* If any gain can be set, it is set to gain[0].
* If another gain can be set, it is set to gain[1], otherwise gain[1] = -1.
* This is for AUDIO_SETINFO_gain_balance.
*/
static void
get_changeable_gain(int fd, int *gain, const char *dir, int offset)
{
       struct audio_info ai;
       int *ai_gain;
       int hi;
       int lo;
       int r;

       /* A hack to handle ai.{play,record}.gain in the same code.. */
       ai_gain = (int *)(((char *)&ai) + offset);

       /* Try to set the maximum gain */
       AUDIO_INITINFO(&ai);
       *ai_gain = AUDIO_MAX_GAIN;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "%s.gain=%d", dir, *ai_gain);
       XP_SYS_EQ(0, r);
       /* Get again.  The value you set is not always used as is. */
       AUDIO_INITINFO(&ai);
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
       XP_SYS_EQ(0, r);
       hi = *ai_gain;

       /* Look for next configurable value. */
       for (lo = hi - 1; lo >= 0; lo--) {
               AUDIO_INITINFO(&ai);
               *ai_gain = lo;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "%s.gain=%d", dir, *ai_gain);
               XP_SYS_EQ(0, r);
               /* Get again */
               r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
               XP_SYS_EQ(0, r);
               if (*ai_gain != hi) {
                       lo = *ai_gain;
                       break;
               }
       }

       /* Now gain is lo(=gain[0]). */

       /*
        * hi  lo
        * --- ---
        *  <0  <0          : not available.
        * >=0  <0          : available but not changeable.
        * >=0 >=0 (hi!=lo) : available and changeable.
        */
       if (hi < 0) {
               gain[0] = -1;
               gain[1] = -1;
               DPRINTF("  > %s.gain cannot be set\n", dir);
       } else if (lo < 0) {
               gain[0] = hi;
               gain[1] = -1;
               DPRINTF("  > %s.gain can only be set %d\n", dir, gain[0]);
       } else {
               gain[0] = lo;
               gain[1] = hi;
               DPRINTF("  > %s.gain can be set %d, %d\n",
                   dir, gain[0], gain[1]);
       }
}

/*
* Look if there are any balance values that can be changed.
* If any balance value can be set, it is set to balance[0].
* If another balance value can be set, it is set to balance[1],
* otherwise balance[1] = -1.
* This is for AUDIO_SETINFO_gain_balance.
*/
static void
get_changeable_balance(int fd, int *balance, const char *dir, int offset)
{
       struct audio_info ai;
       u_char *ai_balance;
       u_char left;
       u_char right;
       int r;

       /* A hack to handle ai.{play,record}.balance in the same code.. */
       ai_balance = ((u_char *)&ai) + offset;

       /* Look for the right side configurable value. */
       AUDIO_INITINFO(&ai);
       *ai_balance = AUDIO_RIGHT_BALANCE;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "%s.balance=%d", dir, *ai_balance);
       XP_SYS_EQ(0, r);
       /* Get again.  The value you set is not always used as is. */
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
       XP_SYS_EQ(0, r);
       right = *ai_balance;

       /* Look for the left side configurable value. */
       AUDIO_INITINFO(&ai);
       *ai_balance = AUDIO_LEFT_BALANCE;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "%s.balance=%d", dir, *ai_balance);
       XP_SYS_EQ(0, r);
       /* Get again */
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
       XP_SYS_EQ(0, r);
       left = *ai_balance;

       /* Now balance is the left(=balance[0]). */

       if (left == right) {
               /* The driver has no balance feature. */
               balance[0] = left;
               balance[1] = -1;
               DPRINTF("  > %s.balance can only be set %d\n",
                   dir, balance[0]);
       } else {
               balance[0] = left;
               balance[1] = right;
               DPRINTF("  > %s.balance can be set %d, %d\n",
                   dir, balance[0], balance[1]);
       }
}

/*
* Check whether gain and balance can be set at the same time.
* PR kern/56308
*/
DEF(AUDIO_SETINFO_gain_balance)
{
       struct audio_info oai;
       struct audio_info ai;
       int i;
       int mode;
       int fd;
       int r;
       int pgain[2];
       int pbalance[2];
       int rgain[2];
       int rbalance[2];
       bool ptest;
       bool rtest;

       TEST("AUDIO_SETINFO_gain_balance");

       mode = openable_mode();
       fd = OPEN(devaudio, mode);
       REQUIRED_SYS_OK(fd);

       /* Backup current gain and balance */
       r = IOCTL(fd, AUDIO_GETINFO, &oai, "&oai");
       XP_SYS_EQ(0, r);

       if (debug) {
               printf("  > old play.gain      = %d\n", oai.play.gain);
               printf("  > old play.balance   = %d\n", oai.play.balance);
               printf("  > old record.gain    = %d\n", oai.record.gain);
               printf("  > old record.balance = %d\n", oai.record.balance);
       }

       for (i = 0; i < 2; i++) {
               pgain[i]    = -1;
               pbalance[i] = -1;
               rgain[i]    = -1;
               rbalance[i] = -1;
       }

       /*
        * First, check each one separately can be changed.
        *
        * The simplest two different gain values are zero and non-zero.
        * But some device drivers seem to process balance differently
        * when the gain is high enough and when the gain is zero or near.
        * So I needed to select two different "non-zero (and high if
        * possible)" gains.
        */
       if (hw_canplay()) {
               get_changeable_gain(fd, pgain, "play",
                   offsetof(struct audio_info, play.gain));
               get_changeable_balance(fd, pbalance, "play",
                   offsetof(struct audio_info, play.balance));
       }
       if (hw_canrec()) {
               get_changeable_gain(fd, rgain, "record",
                   offsetof(struct audio_info, record.gain));
               get_changeable_balance(fd, rbalance, "record",
                   offsetof(struct audio_info, record.balance));
       }

       /*
        * [0] [1]
        * --- ---
        *  -1  *  : not available.
        * >=0  -1 : available but not changeable.
        * >=0 >=0 : available and changeable.  It can be tested.
        */
       ptest = (pgain[0]    >= 0 && pgain[1]    >= 0 &&
                pbalance[0] >= 0 && pbalance[1] >= 0);
       rtest = (rgain[0]    >= 0 && rgain[1]    >= 0 &&
                rbalance[0] >= 0 && rbalance[1] >= 0);

       if (ptest == false && rtest == false) {
               XP_SKIP(
                   "The test requires changeable gain and changeable balance");

               /* Restore as possible */
               AUDIO_INITINFO(&ai);
               ai.play.gain      = oai.play.gain;
               ai.play.balance   = oai.play.balance;
               ai.record.gain    = oai.record.gain;
               ai.record.balance = oai.record.balance;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "restore all");
               XP_SYS_EQ(0, r);

               r = CLOSE(fd);
               XP_SYS_EQ(0, r);
               return;
       }

       /*
        * If both play.gain and play.balance are changeable,
        * it should be able to set both at the same time.
        */
       if (ptest) {
               AUDIO_INITINFO(&ai);
               ai.play.gain    = pgain[1];
               ai.play.balance = pbalance[1];
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "play.gain=%d/balance=%d",
                   ai.play.gain, ai.play.balance);
               XP_SYS_EQ(0, r);

               AUDIO_INITINFO(&ai);
               r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
               XP_SYS_EQ(0, r);

               DPRINTF("  > setting play.gain=%d/balance=%d: "
                   "result gain=%d/balance=%d\n",
                   pgain[1], pbalance[1], ai.play.gain, ai.play.balance);
               XP_EQ(ai.play.gain,    pgain[1]);
               XP_EQ(ai.play.balance, pbalance[1]);
       }
       /*
        * If both record.gain and record.balance are changeable,
        * it should be able to set both at the same time.
        */
       if (rtest) {
               AUDIO_INITINFO(&ai);
               ai.record.gain    = rgain[1];
               ai.record.balance = rbalance[1];
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "record.gain=%d/balance=%d",
                   ai.record.gain, ai.record.balance);
               XP_SYS_EQ(0, r);

               AUDIO_INITINFO(&ai);
               r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
               XP_SYS_EQ(0, r);

               DPRINTF("  > setting record.gain=%d/balance=%d: "
                   "result gain=%d/balance=%d\n",
                   rgain[1], rbalance[1], ai.record.gain, ai.record.balance);
               XP_EQ(ai.record.gain,    rgain[1]);
               XP_EQ(ai.record.balance, rbalance[1]);
       }

       /*
        * Restore all values as possible at the same time.
        * This restore is also a test.
        */
       AUDIO_INITINFO(&ai);
       ai.play.gain      = oai.play.gain;
       ai.play.balance   = oai.play.balance;
       ai.record.gain    = oai.record.gain;
       ai.record.balance = oai.record.balance;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "restore all");
       XP_SYS_EQ(0, r);

       AUDIO_INITINFO(&ai);
       r = IOCTL(fd, AUDIO_GETINFO, &ai, "&ai");
       XP_SYS_EQ(0, r);
       XP_EQ(oai.play.gain,      ai.play.gain);
       XP_EQ(oai.play.balance,   ai.play.balance);
       XP_EQ(oai.record.gain,    ai.record.gain);
       XP_EQ(oai.record.balance, ai.record.balance);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Changing track formats after mmap should fail.
*/
DEF(AUDIO_SETINFO_mmap_enc)
{
       struct audio_info ai;
       void *ptr;
       int fd;
       int r;

       TEST("AUDIO_SETINFO_mmap");

#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       ptr = MMAP(NULL, 1, PROT_WRITE, MAP_FILE, fd, 0);
       XP_SYS_PTR(0, ptr);

       /*
        * SETINFO after mmap should fail.
        * NetBSD9 changes errno.
        */
       AUDIO_INITINFO(&ai);
       ai.play.channels = 2;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "channels=2");
       if (netbsd < 9) {
               XP_SYS_NG(EINVAL, r);
       } else {
               XP_SYS_NG(EIO, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}

/*
* Even after mmap, changing pause should succeed.
*/
DEF(AUDIO_SETINFO_mmap_pause)
{
       struct audio_info ai;
       void *ptr;
       int fd;
       int r;

       TEST("AUDIO_SETINFO_mmap");

#if !defined(NO_RUMP)
       if (use_rump) {
               XP_SKIP("rump doesn't support mmap");
               return;
       }
#endif

       fd = OPEN(devaudio, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       ptr = MMAP(NULL, 1, PROT_WRITE, MAP_FILE, fd, 0);
       XP_SYS_PTR(0, ptr);

       /* SETINFO after mmap should fail */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 1;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "set pause");
       XP_SYS_EQ(0, r);

       AUDIO_INITINFO(&ai);
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "get pause");
       XP_SYS_EQ(0, r);

       XP_EQ(1, ai.play.pause);

       /*
        * Unpause before close.  Unless, subsequent audioplay(1) which use
        * /dev/sound by default will pause...
        */
       AUDIO_INITINFO(&ai);
       ai.play.pause = 0;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "reset pause");
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       reset_after_mmap();
}

#define NENC    (AUDIO_ENCODING_AC3 + 1)
#define NPREC   (5)
/*
* Make table of encoding+precision supported by this device.
* Return last used index .
* This function is called from test_AUDIO_GETENC_*()
*/
int
getenc_make_table(int fd, int expected[][5])
{
       audio_encoding_t ae;
       int idx;
       int p;
       int r;

       /*
        * expected[][] is two dimensional table.
        * encoding \ precision| 4  8  16  24  32
        * --------------------+-----------------
        * AUDIO_ENCODING_NONE |
        * AUDIO_ENCODING_ULAW |
        *  :
        *
        * Each cell has expected behavior.
        *  0: the hardware doesn't support this encoding/precision.
        *  1: the hardware supports this encoding/precision.
        *  2: the hardware doesn't support this encoding/precision but
        *     audio layer will respond as supported for compatibility.
        */
       for (idx = 0; ; idx++) {
               memset(&ae, 0, sizeof(ae));
               ae.index = idx;
               r = IOCTL(fd, AUDIO_GETENC, &ae, "index=%d", idx);
               if (r != 0) {
                       XP_SYS_NG(EINVAL, r);
                       break;
               }

               XP_EQ(idx, ae.index);
               if (0 <= ae.encoding && ae.encoding <= AUDIO_ENCODING_AC3) {
                       XP_EQ_STR(encoding_names[ae.encoding], ae.name);
               } else {
                       XP_FAIL("ae.encoding %d", ae.encoding);
               }

               if (ae.precision != 4 &&
                   ae.precision != 8 &&
                   ae.precision != 16 &&
                   ae.precision != 24 &&
                   ae.precision != 32)
               {
                       XP_FAIL("ae.precision %d", ae.precision);
               }
               /* Other bits should not be set */
               XP_EQ(0, (ae.flags & ~AUDIO_ENCODINGFLAG_EMULATED));

               expected[ae.encoding][ae.precision / 8] = 1;
               DPRINTF("  > encoding=%s precision=%d\n",
                   encoding_names[ae.encoding], ae.precision);
       }

       /*
        * Backward compatibility bandaid.
        *
        * - Some encoding/precision pairs are obviously inconsistent
        *   (e.g., encoding=AUDIO_ENCODING_PCM8, precision=16) but
        *   it's due to historical reasons.
        * - It's incomplete for NetBSD7 and NetBSD8.  I don't really
        *   understand their rule...  This is just memo, not specification.
        */
#define SET(x) do {     \
       if ((x) == 0)   \
               x = 2;  \
} while (0)
#define p4 (0)
#define p8 (1)
#define p16 (2)
#define p24 (3)
#define p32 (4)

       if (expected[AUDIO_ENCODING_SLINEAR][p8]) {
               SET(expected[AUDIO_ENCODING_SLINEAR_LE][p8]);
               SET(expected[AUDIO_ENCODING_SLINEAR_BE][p8]);
       }
       if (expected[AUDIO_ENCODING_ULINEAR][p8]) {
               SET(expected[AUDIO_ENCODING_ULINEAR_LE][p8]);
               SET(expected[AUDIO_ENCODING_ULINEAR_BE][p8]);
               SET(expected[AUDIO_ENCODING_PCM8][p8]);
               SET(expected[AUDIO_ENCODING_PCM16][p8]);
       }
       for (p = p16; p <= p32; p++) {
#if !defined(AUDIO_SUPPORT_LINEAR24)
               if (p == p24)
                       continue;
#endif
               if (expected[AUDIO_ENCODING_SLINEAR_NE][p]) {
                       SET(expected[AUDIO_ENCODING_SLINEAR][p]);
                       SET(expected[AUDIO_ENCODING_PCM16][p]);
               }
               if (expected[AUDIO_ENCODING_ULINEAR_NE][p]) {
                       SET(expected[AUDIO_ENCODING_ULINEAR][p]);
               }
       }

       if (netbsd < 9) {
               if (expected[AUDIO_ENCODING_SLINEAR_LE][p16] ||
                   expected[AUDIO_ENCODING_SLINEAR_BE][p16] ||
                   expected[AUDIO_ENCODING_ULINEAR_LE][p16] ||
                   expected[AUDIO_ENCODING_ULINEAR_BE][p16])
               {
                       SET(expected[AUDIO_ENCODING_PCM8][p8]);
                       SET(expected[AUDIO_ENCODING_PCM16][p8]);
                       SET(expected[AUDIO_ENCODING_SLINEAR_LE][p8]);
                       SET(expected[AUDIO_ENCODING_SLINEAR_BE][p8]);
                       SET(expected[AUDIO_ENCODING_ULINEAR_LE][p8]);
                       SET(expected[AUDIO_ENCODING_ULINEAR_BE][p8]);
                       SET(expected[AUDIO_ENCODING_SLINEAR][p8]);
                       SET(expected[AUDIO_ENCODING_ULINEAR][p8]);
               }
       }

       /* Return last used index */
       return idx;
#undef SET
#undef p4
#undef p8
#undef p16
#undef p24
#undef p32
}

/*
* This function is called from test_AUDIO_GETENC below.
*/
void
xp_getenc(int expected[][5], int enc, int j, int r, struct audio_prinfo *pr)
{
       int prec = (j == 0) ? 4 : j * 8;

       if (expected[enc][j]) {
               /* expect to succeed */
               XP_SYS_EQ(0, r);

               XP_EQ(enc, pr->encoding);
               XP_EQ(prec, pr->precision);
       } else {
               /* expect to fail */
               XP_SYS_NG(EINVAL, r);
       }
}

/*
* This function is called from test_AUDIO_GETENC below.
*/
void
getenc_check_encodings(int openmode, int expected[][5])
{
       struct audio_info ai;
       int fd;
       int i, j;
       int r;

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       for (i = 0; i < NENC; i++) {
               for (j = 0; j < NPREC; j++) {
                       /* precisions are 4 and 8, 16, 24, 32 */
                       int prec = (j == 0) ? 4 : j * 8;

                       /*
                        * AUDIO_GETENC has no way to know range of
                        * supported channels and sample_rate.
                        */
                       AUDIO_INITINFO(&ai);
                       ai.play.encoding = i;
                       ai.play.precision = prec;
                       ai.record.encoding = i;
                       ai.record.precision = prec;

                       r = IOCTL(fd, AUDIO_SETINFO, &ai, "%s:%d",
                           encoding_names[i], prec);
                       if (mode2play(openmode))
                               xp_getenc(expected, i, j, r, &ai.play);
                       if (mode2rec(openmode))
                               xp_getenc(expected, i, j, r, &ai.record);
               }
       }
       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* Check whether encoding+precision obtained by AUDIO_GETENC can be set.
*/
DEF(AUDIO_GETENC_range)
{
       audio_encoding_t ae;
       int fd;
       int r;
       int expected[NENC][NPREC];
       int i, j;

       TEST("AUDIO_GETENC_range");

       fd = OPEN(devaudio, openable_mode());
       REQUIRED_SYS_OK(fd);

       memset(&expected, 0, sizeof(expected));
       i = getenc_make_table(fd, expected);

       /* When error has occurred, the next index should also occur error */
       ae.index = i + 1;
       r = IOCTL(fd, AUDIO_GETENC, &ae, "index=%d", ae.index);
       XP_SYS_NG(EINVAL, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       /* For debug */
       if (debug) {
               for (i = 0; i < NENC; i++) {
                       printf("expected[%2d] %15s", i, encoding_names[i]);
                       for (j = 0; j < NPREC; j++) {
                               printf(" %d", expected[i][j]);
                       }
                       printf("\n");
               }
       }

       /* Whether obtained encodings can be actually set */
       if (hw_fulldup()) {
               /* Test both R/W at once using single descriptor */
               getenc_check_encodings(O_RDWR, expected);
       } else {
               /* Test playback and recording if available */
               if (hw_canplay()) {
                       getenc_check_encodings(O_WRONLY, expected);
               }
               if (hw_canplay() && hw_canrec()) {
                       xxx_close_wait();
               }
               if (hw_canrec()) {
                       getenc_check_encodings(O_RDONLY, expected);
               }
       }
}
#undef NENC
#undef NPREC

/*
* Check AUDIO_GETENC out of range.
*/
DEF(AUDIO_GETENC_error)
{
       audio_encoding_t e;
       int fd;
       int r;

       TEST("AUDIO_GETENC_error");

       fd = OPEN(devaudio, openable_mode());
       REQUIRED_SYS_OK(fd);

       memset(&e, 0, sizeof(e));
       e.index = -1;
       r = IOCTL(fd, AUDIO_GETENC, &e, "index=-1");
       /* NetBSD7 may not fail depending on hardware driver */
       XP_SYS_NG(EINVAL, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* AUDIO_[PR]ERROR should be zero on the initial state even on non-existent
* track.
*/
void
test_AUDIO_ERROR(int openmode)
{
       int fd;
       int r;
       int errors;

       TEST("AUDIO_ERROR_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /* Check PERROR */
       errors = 0xdeadbeef;
       r = IOCTL(fd, AUDIO_PERROR, &errors, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, errors);

       /* Check RERROR */
       errors = 0xdeadbeef;
       r = IOCTL(fd, AUDIO_RERROR, &errors, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, errors);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(AUDIO_ERROR_RDONLY) { test_AUDIO_ERROR(O_RDONLY); }
DEF(AUDIO_ERROR_WRONLY) { test_AUDIO_ERROR(O_WRONLY); }
DEF(AUDIO_ERROR_RDWR)   { test_AUDIO_ERROR(O_RDWR); }

/*
* AUDIO_GETIOFFS at least one block.
*/
void
test_AUDIO_GETIOFFS_one(int openmode)
{
       struct audio_info ai;
       audio_offset_t o;
       int fd;
       int r;
       u_int blocksize;
       u_int blk_ms;

       TEST("AUDIO_GETIOFFS_one_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

#if 0
       /*
        * On NetBSD7/8, native encodings and emulated encodings behave
        * differently.  But it's hard to identify which encoding is native.
        * If you try other encodings, edit these parameters manually.
        */
       AUDIO_INITINFO(&ai);
       ai.record.encoding = AUDIO_ENCODING_SLINEAR_NE;
       ai.record.precision = 16;
       ai.record.channels = 2;
       ai.record.sample_rate = 48000;
       /* ai.blocksize is shared by play and record, so set both the same. */
       *ai.play = *ai.record;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
#endif

       /* Get blocksize to calc blk_ms. */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       blocksize = ai.blocksize;
       if (netbsd < 9) {
               blk_ms = 0;
       } else {
               /* On NetBSD9, blocktime can always be calculated. */
               blk_ms = blocksize * 1000 /
                   (ai.play.precision / 8 * ai.play.channels *
                    ai.play.sample_rate);
       }
       if (blk_ms == 0)
               blk_ms = 50;
       DPRINTF("  > blocksize=%u, estimated blk_ms=%u\n", blocksize, blk_ms);

       /*
        * Even when just opened, recording counters will start.
        * Wait a moment, about one block time.
        */
       usleep(blk_ms * 1000);

       r = IOCTL(fd, AUDIO_GETIOFFS, &o, "");
       XP_SYS_EQ(0, r);
       if (mode2rec(openmode)) {
               /*
                * It's difficult to know exact values.
                * But at least these should not be zero.
                */
               DPRINTF("  > %d: samples=%u deltablks=%u offset=%u\n",
                   __LINE__, o.samples, o.deltablks, o.offset);
               XP_NE(0, o.samples);
               XP_NE(0, o.deltablks);
               XP_NE(0, o.offset);
       } else {
               /* All are zero on playback track. */
               XP_EQ(0, o.samples);
               XP_EQ(0, o.deltablks);
               XP_EQ(0, o.offset);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(AUDIO_GETIOFFS_one_RDONLY) { test_AUDIO_GETIOFFS_one(O_RDONLY); }
DEF(AUDIO_GETIOFFS_one_WRONLY) { test_AUDIO_GETIOFFS_one(O_WRONLY); }
DEF(AUDIO_GETIOFFS_one_RDWR)   { test_AUDIO_GETIOFFS_one(O_RDWR); }

/*
* AUDIO_GETOOFFS for one block.
*/
void
test_AUDIO_GETOOFFS_one(int openmode)
{
       struct audio_info ai;
       audio_offset_t o;
       char *buf;
       int fd;
       int r;
       u_int blocksize;
       u_int initial_offset;
       u_int blk_ms;

       TEST("AUDIO_GETOOFFS_one_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

#if 0
       /*
        * On NetBSD7/8, native encodings and emulated encodings behave
        * differently.  But it's hard to identify which encoding is native.
        * If you try other encodings, edit these parameters manually.
        */
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_NE;
       ai.play.precision = 16;
       ai.play.channels = 2;
       ai.play.sample_rate = 48000;
       /* ai.blocksize is shared by play and record, so set both the same. */
       *ai.record = *ai.play;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "slinear16/2ch/48000");
       REQUIRED_SYS_EQ(0, r);
#endif

       /* Get blocksize to calc blk_ms. */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       blocksize = ai.blocksize;
       if (netbsd < 9) {
               blk_ms = 0;
       } else {
               /* On NetBSD9, blocktime can always be calculated. */
               blk_ms = blocksize * 1000 /
                   (ai.play.precision / 8 * ai.play.channels *
                    ai.play.sample_rate);
       }
       if (blk_ms == 0)
               blk_ms = 50;
       DPRINTF("  > blocksize=%u, estimated blk_ms=%u\n", blocksize, blk_ms);

       buf = (char *)malloc(blocksize);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, blocksize);

       /*
        * On NetBSD7, .offset starts from one block.  What is the block??
        * On NetBSD9, .offset starts from zero.
        */
       if (netbsd < 9) {
               initial_offset = blocksize;
       } else {
               initial_offset = 0;
       }

       /* When just opened, all are zero. */
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, o.samples);
       XP_EQ(0, o.deltablks);
       XP_EQ(initial_offset, o.offset);

       /* Even if wait (at least) one block, these remain unchanged. */
       usleep(blk_ms * 1000);
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, o.samples);
       XP_EQ(0, o.deltablks);
       XP_EQ(initial_offset, o.offset);

       /* Write one block. */
       r = WRITE(fd, buf, blocksize);
       if (mode2play(openmode)) {
               XP_SYS_EQ(blocksize, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }
       r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
       REQUIRED_SYS_EQ(0, r);

       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       if (mode2play(openmode)) {
               /* All advance one block. */
               XP_EQ(blocksize, o.samples);
               XP_EQ(1, o.deltablks);
               XP_EQ(initial_offset + blocksize, o.offset);
       } else {
               /*
                * All are zero on non-play track.
                * On NetBSD7, the rec track has play buffer, too.
                */
               XP_EQ(0, o.samples);
               XP_EQ(0, o.deltablks);
               XP_EQ(initial_offset, o.offset);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       free(buf);
}
DEF(AUDIO_GETOOFFS_one_RDONLY) { test_AUDIO_GETOOFFS_one(O_RDONLY); }
DEF(AUDIO_GETOOFFS_one_WRONLY) { test_AUDIO_GETOOFFS_one(O_WRONLY); }
DEF(AUDIO_GETOOFFS_one_RDWR)   { test_AUDIO_GETOOFFS_one(O_RDWR); }

/*
* AUDIO_GETOOFFS when wrap around buffer.
*/
void
test_AUDIO_GETOOFFS_wrap(int openmode)
{
       struct audio_info ai;
       audio_offset_t o;
       char *buf;
       int fd;
       int r;
       u_int blocksize;
       u_int buffer_size;
       u_int initial_offset;
       u_int nblks;

       TEST("AUDIO_GETOOFFS_wrap_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

#if 1
       /* To save test time, use larger format if possible. */
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_NE;
       ai.play.precision = 16;
       ai.play.channels = 2;
       ai.play.sample_rate = 48000;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "slinear16/2/48000");
       if (r != 0)
#endif
       {
               /*
                * If it cannot be set, use common format instead.
                * May be happened on NetBSD7/8.
                */
               AUDIO_INITINFO(&ai);
               ai.play.encoding = AUDIO_ENCODING_ULAW;
               ai.play.precision = 8;
               ai.play.channels = 1;
               ai.play.sample_rate = 8000;
               r = IOCTL(fd, AUDIO_SETINFO, &ai, "ulaw/1/8000");
       }
       REQUIRED_SYS_EQ(0, r);

       /* Get buffer_size and blocksize. */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       buffer_size = ai.play.buffer_size;
       blocksize = ai.blocksize;
       nblks = buffer_size / blocksize;
       DPRINTF("  > buffer_size=%u blocksize=%u nblks=%u\n",
           buffer_size, blocksize, nblks);

       buf = (char *)malloc(buffer_size);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, buffer_size);

       /*
        * On NetBSD7, .offset starts from one block.  What is the block??
        * On NetBSD9, .offset starts from zero.
        */
       if (netbsd < 9) {
               initial_offset = blocksize;
       } else {
               initial_offset = 0;
       }

       /* Write full buffer. */
       r = WRITE(fd, buf, buffer_size);
       if (mode2play(openmode)) {
               XP_SYS_EQ(buffer_size, r);

               /* Then, wait. */
               r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
               REQUIRED_SYS_EQ(0, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }

       /*
        * .deltablks is number of blocks since last checked.
        * .offset is wrapped around to zero.
        */
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       if (mode2play(openmode)) {
               /*
                * On NetBSD7, samples may be blocksize * nblks or buffer_size
                * depending on native/emulated encoding.
                * On NetBSD9, samples is always equal to buffer_size.
                */
               if (buffer_size != blocksize * nblks &&
                   o.samples == blocksize * nblks) {
                       DPRINTF("  > %d: samples(%u) == blocksize * nblks\n",
                           __LINE__, o.samples);
               } else {
                       XP_EQ(buffer_size, o.samples);
               }
               XP_EQ(nblks, o.deltablks);
               XP_EQ(initial_offset, o.offset);
       } else {
               /*
                * On non-play track, it silently succeeds with zero.
                * But on NetBSD7, RDONLY descriptor also has play buffer.
                */
               XP_EQ(0, o.samples);
               XP_EQ(0, o.deltablks);
               XP_EQ(initial_offset, o.offset);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       free(buf);
}
DEF(AUDIO_GETOOFFS_wrap_RDONLY) { test_AUDIO_GETOOFFS_wrap(O_RDONLY); }
DEF(AUDIO_GETOOFFS_wrap_WRONLY) { test_AUDIO_GETOOFFS_wrap(O_WRONLY); }
DEF(AUDIO_GETOOFFS_wrap_RDWR)   { test_AUDIO_GETOOFFS_wrap(O_RDWR); }

/*
* Check whether AUDIO_FLUSH clears AUDIO_GETOOFFS.
*/
void
test_AUDIO_GETOOFFS_flush(int openmode)
{
       struct audio_info ai;
       audio_offset_t o;
       char *buf;
       int fd;
       int r;
       u_int initial_offset;
       u_int last_offset;

       TEST("AUDIO_GETOOFFS_flush_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

#if 0
       /* On NetBSD7/8, native encoding changes buffer behavior. */
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_NE;
       ai.play.precision = 16;
       ai.play.channels = 2;
       ai.play.sample_rate = 48000;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
#endif

       /* Get blocksize. */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);

       buf = (char *)malloc(ai.blocksize);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, ai.blocksize);

       /*
        * On NetBSD7, .offset starts from one block.  What is the block??
        * On NetBSD9, .offset starts from zero.
        */
       if (netbsd < 9) {
               initial_offset = ai.blocksize;
       } else {
               initial_offset = 0;
       }

       /* Write one block. */
       r = WRITE(fd, buf, ai.blocksize);
       if (mode2play(openmode)) {
               XP_SYS_EQ(ai.blocksize, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }
       r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
       XP_SYS_EQ(0, r);

       /* Obtain once. */
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       if (mode2play(openmode)) {
               XP_EQ(ai.blocksize, o.samples);
               XP_EQ(1, o.deltablks);
               XP_EQ(initial_offset + ai.blocksize, o.offset);
       } else {
               /*
                * On non-play track, it silently succeeds with zero.
                * But on NetBSD7, RDONLY descriptor also has play buffer.
                */
               XP_EQ(0, o.samples);
               XP_EQ(0, o.deltablks);
               XP_EQ(initial_offset, o.offset);
       }

       /* Write one more block to advance .offset. */
       r = WRITE(fd, buf, ai.blocksize);
       if (mode2play(openmode)) {
               XP_SYS_EQ(ai.blocksize, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }
       r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
       XP_SYS_EQ(0, r);

       /* If offset remains unchanged, this is expected offset. */
       last_offset = initial_offset + ai.blocksize * 2;

       /* Then, flush. */
       r = IOCTL(fd, AUDIO_FLUSH, NULL, "");
       REQUIRED_SYS_EQ(0, r);

       /* All should be cleared. */
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, o.samples);
       XP_EQ(0, o.deltablks);
       if (mode2play(openmode)) {
               /*
                * On NetBSD7,
                * offset is cleared if native encodings(?), but remains
                * unchanged if emulated encodings(?).  Looks a bug.
                * On NetBSD9, it should always be cleared.
                */
               if (netbsd < 9 && o.offset == last_offset) {
                       DPRINTF("  > %d: offset(%u) == last_offset\n",
                           __LINE__, o.offset);
               } else {
                       XP_EQ(initial_offset, o.offset);
               }
       } else {
               XP_EQ(initial_offset, o.offset);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       free(buf);
}
DEF(AUDIO_GETOOFFS_flush_RDONLY) { test_AUDIO_GETOOFFS_flush(O_RDONLY); }
DEF(AUDIO_GETOOFFS_flush_WRONLY) { test_AUDIO_GETOOFFS_flush(O_WRONLY); }
DEF(AUDIO_GETOOFFS_flush_RDWR)   { test_AUDIO_GETOOFFS_flush(O_RDWR); }

/*
* Check whether AUDIO_SETINFO(encoding) clears AUDIO_GETOOFFS.
*/
void
test_AUDIO_GETOOFFS_set(int openmode)
{
       struct audio_info ai;
       audio_offset_t o;
       char *buf;
       int fd;
       int r;
       u_int initial_offset;

       TEST("AUDIO_GETOOFFS_set_%s", openmode_str[openmode] + 2);
       if (mode2aumode(openmode) == 0) {
               XP_SKIP("Operation not allowed on this hardware property");
               return;
       }

       fd = OPEN(devaudio, openmode);
       REQUIRED_SYS_OK(fd);

       /* Get blocksize. */
       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       XP_SYS_EQ(0, r);

       buf = (char *)malloc(ai.blocksize);
       REQUIRED_IF(buf != NULL);
       memset(buf, 0xff, ai.blocksize);

       /*
        * On NetBSD7, .offset starts from one block.  What is the block??
        * On NetBSD9, .offset starts from zero.
        */
       if (netbsd < 9) {
               initial_offset = ai.blocksize;
       } else {
               initial_offset = 0;
       }

       /* Write one block. */
       r = WRITE(fd, buf, ai.blocksize);
       if (mode2play(openmode)) {
               XP_SYS_EQ(ai.blocksize, r);
       } else {
               XP_SYS_NG(EBADF, r);
       }
       r = IOCTL(fd, AUDIO_DRAIN, NULL, "");
       XP_SYS_EQ(0, r);

       /*
        * Then, change encoding.
        * If we fail to change it, we cannot continue.  This may happen
        * on NetBSD7/8.
        */
       AUDIO_INITINFO(&ai);
       ai.play.encoding = AUDIO_ENCODING_SLINEAR_NE;
       ai.play.precision = 16;
       ai.play.channels = 2;
       ai.play.sample_rate = 48000;
       r = IOCTL(fd, AUDIO_SETINFO, &ai, "slinear16/2ch/48000");
       REQUIRED_SYS_EQ(0, r);

       r = IOCTL(fd, AUDIO_GETBUFINFO, &ai, "");
       REQUIRED_SYS_EQ(0, r);
       if (netbsd < 9) {
               initial_offset = ai.blocksize;
       } else {
               initial_offset = 0;
       }

       /* Clear counters? */
       r = IOCTL(fd, AUDIO_GETOOFFS, &o, "");
       XP_SYS_EQ(0, r);
       XP_EQ(0, o.samples);
       XP_EQ(0, o.deltablks);
       XP_EQ(initial_offset, o.offset);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       free(buf);
}
DEF(AUDIO_GETOOFFS_set_RDONLY) { test_AUDIO_GETOOFFS_set(O_RDONLY); }
DEF(AUDIO_GETOOFFS_set_WRONLY) { test_AUDIO_GETOOFFS_set(O_WRONLY); }
DEF(AUDIO_GETOOFFS_set_RDWR)   { test_AUDIO_GETOOFFS_set(O_RDWR); }

/*
* /dev/audioctl can always be opened while /dev/audio is open.
*/
void
test_audioctl_open_1(int fmode, int cmode)
{
       int fd;
       int ctl;
       int r;

       TEST("audioctl_open_1_%s_%s",
           openmode_str[fmode] + 2, openmode_str[cmode] + 2);
       if (hw_canplay() == 0 && fmode == O_WRONLY) {
               XP_SKIP("This test is for playable device");
               return;
       }
       if (hw_canrec() == 0 && fmode == O_RDONLY) {
               XP_SKIP("This test is for recordable device");
               return;
       }

       fd = OPEN(devaudio, fmode);
       REQUIRED_SYS_OK(fd);

       ctl = OPEN(devaudioctl, cmode);
       XP_SYS_OK(ctl);

       r = CLOSE(ctl);
       XP_SYS_EQ(0, r);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(audioctl_open_1_RDONLY_RDONLY) { test_audioctl_open_1(O_RDONLY, O_RDONLY); }
DEF(audioctl_open_1_RDONLY_RWONLY) { test_audioctl_open_1(O_RDONLY, O_WRONLY); }
DEF(audioctl_open_1_RDONLY_RDWR)   { test_audioctl_open_1(O_RDONLY, O_RDWR); }
DEF(audioctl_open_1_WRONLY_RDONLY) { test_audioctl_open_1(O_WRONLY, O_RDONLY); }
DEF(audioctl_open_1_WRONLY_RWONLY) { test_audioctl_open_1(O_WRONLY, O_WRONLY); }
DEF(audioctl_open_1_WRONLY_RDWR)   { test_audioctl_open_1(O_WRONLY, O_RDWR); }
DEF(audioctl_open_1_RDWR_RDONLY)   { test_audioctl_open_1(O_RDWR, O_RDONLY); }
DEF(audioctl_open_1_RDWR_RWONLY)   { test_audioctl_open_1(O_RDWR, O_WRONLY); }
DEF(audioctl_open_1_RDWR_RDWR)     { test_audioctl_open_1(O_RDWR, O_RDWR); }

/*
* /dev/audio can always be opened while /dev/audioctl is open.
*/
void
test_audioctl_open_2(int fmode, int cmode)
{
       int fd;
       int ctl;
       int r;

       TEST("audioctl_open_2_%s_%s",
           openmode_str[fmode] + 2, openmode_str[cmode] + 2);
       if (hw_canplay() == 0 && fmode == O_WRONLY) {
               XP_SKIP("This test is for playable device");
               return;
       }
       if (hw_canrec() == 0 && fmode == O_RDONLY) {
               XP_SKIP("This test is for recordable device");
               return;
       }

       ctl = OPEN(devaudioctl, cmode);
       REQUIRED_SYS_OK(ctl);

       fd = OPEN(devaudio, fmode);
       XP_SYS_OK(fd);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);

       r = CLOSE(ctl);
       XP_SYS_EQ(0, r);
}
DEF(audioctl_open_2_RDONLY_RDONLY) { test_audioctl_open_2(O_RDONLY, O_RDONLY); }
DEF(audioctl_open_2_RDONLY_RWONLY) { test_audioctl_open_2(O_RDONLY, O_WRONLY); }
DEF(audioctl_open_2_RDONLY_RDWR)   { test_audioctl_open_2(O_RDONLY, O_RDWR); }
DEF(audioctl_open_2_WRONLY_RDONLY) { test_audioctl_open_2(O_WRONLY, O_RDONLY); }
DEF(audioctl_open_2_WRONLY_RWONLY) { test_audioctl_open_2(O_WRONLY, O_WRONLY); }
DEF(audioctl_open_2_WRONLY_RDWR)   { test_audioctl_open_2(O_WRONLY, O_RDWR); }
DEF(audioctl_open_2_RDWR_RDONLY)   { test_audioctl_open_2(O_RDWR, O_RDONLY); }
DEF(audioctl_open_2_RDWR_RWONLY)   { test_audioctl_open_2(O_RDWR, O_WRONLY); }
DEF(audioctl_open_2_RDWR_RDWR)     { test_audioctl_open_2(O_RDWR, O_RDWR); }

/*
* Open multiple /dev/audioctl.
*/
DEF(audioctl_open_simul)
{
       int ctl0;
       int ctl1;
       int r;

       TEST("audioctl_open_simul");

       ctl0 = OPEN(devaudioctl, O_RDWR);
       REQUIRED_SYS_OK(ctl0);

       ctl1 = OPEN(devaudioctl, O_RDWR);
       XP_SYS_OK(ctl1);

       r = CLOSE(ctl0);
       XP_SYS_EQ(0, r);

       r = CLOSE(ctl1);
       XP_SYS_EQ(0, r);
}

/*
* /dev/audioctl can be opened by other user who opens /dev/audioctl,
* /dev/audioctl can be opened by other user who opens /dev/audio,
* /dev/audio    can be opened by other user who opens /dev/audioctl,
* regardless of multiuser mode.
*/
void
try_audioctl_open_multiuser(const char *dev1, const char *dev2)
{
       int fd1;
       int fd2;
       int r;
       uid_t ouid;

       /*
        * At first, open dev1 as root.
        * And then open dev2 as unprivileged user.
        */

       fd1 = OPEN(dev1, O_RDWR);
       REQUIRED_SYS_OK(fd1);

       ouid = GETUID();
       r = SETEUID(1);
       REQUIRED_SYS_EQ(0, r);

       fd2 = OPEN(dev2, O_RDWR);
       XP_SYS_OK(fd2);

       /* Close */
       r = CLOSE(fd2);
       XP_SYS_EQ(0, r);

       r = SETEUID(ouid);
       REQUIRED_SYS_EQ(0, r);

       r = CLOSE(fd1);
       XP_SYS_EQ(0, r);
}
/*
* This is a wrapper for audioctl_open_multiuser.
* XXX XP_* macros are not compatible with on-error-goto, we need try-catch...
*/
void
test_audioctl_open_multiuser(bool multiuser,
       const char *dev1, const char *dev2)
{
       char mibname[32];
       bool oldval;
       size_t oldlen;
       int r;

       if (netbsd < 8 && multiuser == 1) {
               XP_SKIP("multiuser is not supported");
               return;
       }
       if (netbsd < 9) {
               /* NetBSD8 has no way (difficult) to determine device name */
               XP_SKIP("NetBSD8 cannot determine device name");
               return;
       }
       if (geteuid() != 0) {
               XP_SKIP("This test must be privileged user");
               return;
       }

       /* Get current multiuser mode (and save it) */
       snprintf(mibname, sizeof(mibname), "hw.%s.multiuser", devicename);
       oldlen = sizeof(oldval);
       r = SYSCTLBYNAME(mibname, &oldval, &oldlen, NULL, 0);
       REQUIRED_SYS_EQ(0, r);
       DPRINTF("  > multiuser=%d\n", oldval);

       /* Change if necessary */
       if (oldval != multiuser) {
               r = SYSCTLBYNAME(mibname, NULL, NULL, &multiuser,
                   sizeof(multiuser));
               REQUIRED_SYS_EQ(0, r);
               DPRINTF("  > new multiuser=%d\n", multiuser);
       }

       /* Do test */
       try_audioctl_open_multiuser(dev1, dev2);

       /* Restore multiuser mode */
       if (oldval != multiuser) {
               DPRINTF("  > restore multiuser to %d\n", oldval);
               r = SYSCTLBYNAME(mibname, NULL, NULL, &oldval, sizeof(oldval));
               XP_SYS_EQ(0, r);
       }
}
DEF(audioctl_open_multiuser0_audio1) {
       TEST("audioctl_open_multiuser0_audio1");
       test_audioctl_open_multiuser(false, devaudio, devaudioctl);
}
DEF(audioctl_open_multiuser1_audio1) {
       TEST("audioctl_open_multiuser1_audio1");
       test_audioctl_open_multiuser(true, devaudio, devaudioctl);
}
DEF(audioctl_open_multiuser0_audio2) {
       TEST("audioctl_open_multiuser0_audio2");
       test_audioctl_open_multiuser(false, devaudioctl, devaudio);
}
DEF(audioctl_open_multiuser1_audio2) {
       TEST("audioctl_open_multiuser1_audio2");
       test_audioctl_open_multiuser(true, devaudioctl, devaudio);
}
DEF(audioctl_open_multiuser0_audioctl) {
       TEST("audioctl_open_multiuser0_audioctl");
       test_audioctl_open_multiuser(false, devaudioctl, devaudioctl);
}
DEF(audioctl_open_multiuser1_audioctl) {
       TEST("audioctl_open_multiuser1_audioctl");
       test_audioctl_open_multiuser(true, devaudioctl, devaudioctl);
}

/*
* /dev/audioctl cannot be read/written regardless of its open mode.
*/
void
test_audioctl_rw(int openmode)
{
       char buf[1];
       int fd;
       int r;

       TEST("audioctl_rw_%s", openmode_str[openmode] + 2);

       fd = OPEN(devaudioctl, openmode);
       REQUIRED_SYS_OK(fd);

       if (mode2play(openmode)) {
               r = WRITE(fd, buf, sizeof(buf));
               XP_SYS_NG(ENODEV, r);
       }

       if (mode2rec(openmode)) {
               r = READ(fd, buf, sizeof(buf));
               XP_SYS_NG(ENODEV, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}
DEF(audioctl_rw_RDONLY) { test_audioctl_rw(O_RDONLY); }
DEF(audioctl_rw_WRONLY) { test_audioctl_rw(O_WRONLY); }
DEF(audioctl_rw_RDWR)   { test_audioctl_rw(O_RDWR); }

/*
* poll(2) for /dev/audioctl should never raise.
* I'm not sure about consistency between poll(2) and kqueue(2) but
* anyway I follow it.
* XXX Omit checking each openmode
*/
DEF(audioctl_poll)
{
       struct pollfd pfd;
       int fd;
       int r;

       TEST("audioctl_poll");

       fd = OPEN(devaudioctl, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       pfd.fd = fd;
       pfd.events = POLLOUT;
       r = POLL(&pfd, 1, 100);
       XP_SYS_EQ(0, r);
       XP_EQ(0, pfd.revents);

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}

/*
* kqueue(2) for /dev/audioctl fails.
* I'm not sure about consistency between poll(2) and kqueue(2) but
* anyway I follow it.
* XXX Omit checking each openmode
*/
DEF(audioctl_kqueue)
{
       struct kevent kev;
       int fd;
       int kq;
       int r;

       TEST("audioctl_kqueue");

       fd = OPEN(devaudioctl, O_WRONLY);
       REQUIRED_SYS_OK(fd);

       kq = KQUEUE();
       XP_SYS_OK(kq);

       EV_SET(&kev, fd, EVFILT_WRITE, EV_ADD, 0, 0, 0);
       r = KEVENT_SET(kq, &kev, 1);
       /*
        * NetBSD7 has a bug.  It looks to wanted to treat it as successful
        * but returned 1(== EPERM).
        * On NetBSD9, I decided to return ENODEV.
        */
       if (netbsd < 8) {
               XP_SYS_NG(1/*EPERM*/, r);
       } else {
               XP_SYS_NG(ENODEV, r);
       }

       r = CLOSE(fd);
       XP_SYS_EQ(0, r);
}


/*
* This table is processed by t_audio.awk!
* Keep /^\tENT(testname),/ format in order to add to atf.
*/
#define ENT(x) { #x, test__ ## x }
struct testentry testtable[] = {
       ENT(open_mode_RDONLY),
       ENT(open_mode_WRONLY),
       ENT(open_mode_RDWR),
       ENT(open_audio_RDONLY),
       ENT(open_audio_WRONLY),
       ENT(open_audio_RDWR),
       ENT(open_sound_RDONLY),
       ENT(open_sound_WRONLY),
       ENT(open_sound_RDWR),
       ENT(open_audioctl_RDONLY),
       ENT(open_audioctl_WRONLY),
       ENT(open_audioctl_RDWR),
       ENT(open_sound_sticky),
       ENT(open_audioctl_sticky),
       ENT(open_simul_RDONLY_RDONLY),
       ENT(open_simul_RDONLY_WRONLY),
       ENT(open_simul_RDONLY_RDWR),
       ENT(open_simul_WRONLY_RDONLY),
       ENT(open_simul_WRONLY_WRONLY),
       ENT(open_simul_WRONLY_RDWR),
       ENT(open_simul_RDWR_RDONLY),
       ENT(open_simul_RDWR_WRONLY),
       ENT(open_simul_RDWR_RDWR),
/**/    ENT(open_multiuser_0),          // XXX TODO sysctl
/**/    ENT(open_multiuser_1),          // XXX TODO sysctl
       ENT(write_PLAY_ALL),
       ENT(write_PLAY),
       ENT(read),
       ENT(rept_write),
       ENT(rept_read),
       ENT(rdwr_fallback_RDONLY),
       ENT(rdwr_fallback_WRONLY),
       ENT(rdwr_fallback_RDWR),
       ENT(rdwr_two_RDONLY_RDONLY),
       ENT(rdwr_two_RDONLY_WRONLY),
       ENT(rdwr_two_RDONLY_RDWR),
       ENT(rdwr_two_WRONLY_RDONLY),
       ENT(rdwr_two_WRONLY_WRONLY),
       ENT(rdwr_two_WRONLY_RDWR),
       ENT(rdwr_two_RDWR_RDONLY),
       ENT(rdwr_two_RDWR_WRONLY),
       ENT(rdwr_two_RDWR_RDWR),
       ENT(rdwr_simul),
       ENT(drain_incomplete),
       ENT(drain_pause),
       ENT(drain_onrec),
/**/    ENT(mmap_mode_RDONLY_NONE),     // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDONLY_READ),     // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDONLY_WRITE),    // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDONLY_READWRITE),// XXX rump doesn't support mmap
/**/    ENT(mmap_mode_WRONLY_NONE),     // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_WRONLY_READ),     // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_WRONLY_WRITE),    // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_WRONLY_READWRITE),// XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDWR_NONE),       // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDWR_READ),       // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDWR_WRITE),      // XXX rump doesn't support mmap
/**/    ENT(mmap_mode_RDWR_READWRITE),  // XXX rump doesn't support mmap
/**/    ENT(mmap_len_0),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_1),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_2),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_3),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_4),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_5),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_6),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_7),                // XXX rump doesn't support mmap
/**/    ENT(mmap_len_8),                // XXX rump doesn't support mmap
/**/    ENT(mmap_twice),                // XXX rump doesn't support mmap
/**/    ENT(mmap_multi),                // XXX rump doesn't support mmap
       ENT(poll_mode_RDONLY_IN),
       ENT(poll_mode_RDONLY_OUT),
       ENT(poll_mode_RDONLY_INOUT),
       ENT(poll_mode_WRONLY_IN),
       ENT(poll_mode_WRONLY_OUT),
       ENT(poll_mode_WRONLY_INOUT),
       ENT(poll_mode_RDWR_IN),
       ENT(poll_mode_RDWR_OUT),
       ENT(poll_mode_RDWR_INOUT),
       ENT(poll_out_empty),
       ENT(poll_out_full),
       ENT(poll_out_hiwat),
/**/    ENT(poll_out_unpause),          // XXX does not seem to work on rump
/**/    ENT(poll_out_simul),            // XXX does not seem to work on rump
       ENT(poll_in_open_audio),
       ENT(poll_in_open_sound),
       ENT(poll_in_open_audioctl),
       ENT(poll_in_simul),
       ENT(kqueue_mode_RDONLY_READ),
       ENT(kqueue_mode_RDONLY_WRITE),
       ENT(kqueue_mode_WRONLY_READ),
       ENT(kqueue_mode_WRONLY_WRITE),
       ENT(kqueue_mode_RDWR_READ),
       ENT(kqueue_mode_RDWR_WRITE),
       ENT(kqueue_empty),
       ENT(kqueue_full),
       ENT(kqueue_hiwat),
/**/    ENT(kqueue_unpause),            // XXX does not seem to work on rump
/**/    ENT(kqueue_simul),              // XXX does not seem to work on rump
       ENT(ioctl_while_write),
       ENT(FIOASYNC_reset),
       ENT(FIOASYNC_play_signal),
       ENT(FIOASYNC_rec_signal),
/**/    ENT(FIOASYNC_multi),            // XXX does not seem to work on rump
       ENT(AUDIO_WSEEK),
       ENT(AUDIO_SETFD_RDONLY),
       ENT(AUDIO_SETFD_WRONLY),
       ENT(AUDIO_SETFD_RDWR),
       ENT(AUDIO_GETINFO_eof),
       ENT(AUDIO_SETINFO_mode_RDONLY_0),
       ENT(AUDIO_SETINFO_mode_RDONLY_1),
       ENT(AUDIO_SETINFO_mode_RDONLY_2),
       ENT(AUDIO_SETINFO_mode_RDONLY_3),
       ENT(AUDIO_SETINFO_mode_RDONLY_4),
       ENT(AUDIO_SETINFO_mode_RDONLY_5),
       ENT(AUDIO_SETINFO_mode_RDONLY_6),
       ENT(AUDIO_SETINFO_mode_RDONLY_7),
       ENT(AUDIO_SETINFO_mode_RDONLY_8),
       ENT(AUDIO_SETINFO_mode_WRONLY_0),
       ENT(AUDIO_SETINFO_mode_WRONLY_1),
       ENT(AUDIO_SETINFO_mode_WRONLY_2),
       ENT(AUDIO_SETINFO_mode_WRONLY_3),
       ENT(AUDIO_SETINFO_mode_WRONLY_4),
       ENT(AUDIO_SETINFO_mode_WRONLY_5),
       ENT(AUDIO_SETINFO_mode_WRONLY_6),
       ENT(AUDIO_SETINFO_mode_WRONLY_7),
       ENT(AUDIO_SETINFO_mode_WRONLY_8),
       ENT(AUDIO_SETINFO_mode_RDWR_0),
       ENT(AUDIO_SETINFO_mode_RDWR_1),
       ENT(AUDIO_SETINFO_mode_RDWR_2),
       ENT(AUDIO_SETINFO_mode_RDWR_3),
       ENT(AUDIO_SETINFO_mode_RDWR_4),
       ENT(AUDIO_SETINFO_mode_RDWR_5),
       ENT(AUDIO_SETINFO_mode_RDWR_6),
       ENT(AUDIO_SETINFO_mode_RDWR_7),
       ENT(AUDIO_SETINFO_mode_RDWR_8),
       ENT(AUDIO_SETINFO_params_set_RDONLY_0),
       ENT(AUDIO_SETINFO_params_set_RDONLY_1),
       ENT(AUDIO_SETINFO_params_set_WRONLY_0),
       ENT(AUDIO_SETINFO_params_set_WRONLY_1),
       ENT(AUDIO_SETINFO_params_set_WRONLY_2),
       ENT(AUDIO_SETINFO_params_set_WRONLY_3),
       ENT(AUDIO_SETINFO_params_set_RDWR_0),
       ENT(AUDIO_SETINFO_params_set_RDWR_1),
       ENT(AUDIO_SETINFO_params_set_RDWR_2),
       ENT(AUDIO_SETINFO_params_set_RDWR_3),
       ENT(AUDIO_SETINFO_params_simul),
       ENT(AUDIO_SETINFO_channels),
       ENT(AUDIO_SETINFO_sample_rate),
       ENT(AUDIO_SETINFO_sample_rate_0),
       ENT(AUDIO_SETINFO_pause_RDONLY_0),
       ENT(AUDIO_SETINFO_pause_RDONLY_1),
       ENT(AUDIO_SETINFO_pause_WRONLY_0),
       ENT(AUDIO_SETINFO_pause_WRONLY_1),
       ENT(AUDIO_SETINFO_pause_WRONLY_2),
       ENT(AUDIO_SETINFO_pause_WRONLY_3),
       ENT(AUDIO_SETINFO_pause_RDWR_0),
       ENT(AUDIO_SETINFO_pause_RDWR_1),
       ENT(AUDIO_SETINFO_pause_RDWR_2),
       ENT(AUDIO_SETINFO_pause_RDWR_3),
       ENT(AUDIO_SETINFO_gain),
       ENT(AUDIO_SETINFO_gain_balance),
/**/    ENT(AUDIO_SETINFO_mmap_enc),    // XXX rump doesn't support mmap
/**/    ENT(AUDIO_SETINFO_mmap_pause),  // XXX rump doesn't support mmap
       ENT(AUDIO_GETENC_range),
       ENT(AUDIO_GETENC_error),
       ENT(AUDIO_ERROR_RDONLY),
       ENT(AUDIO_ERROR_WRONLY),
       ENT(AUDIO_ERROR_RDWR),
       ENT(AUDIO_GETIOFFS_one_RDONLY),
       ENT(AUDIO_GETIOFFS_one_WRONLY),
       ENT(AUDIO_GETIOFFS_one_RDWR),
       ENT(AUDIO_GETOOFFS_one_RDONLY),
       ENT(AUDIO_GETOOFFS_one_WRONLY),
       ENT(AUDIO_GETOOFFS_one_RDWR),
       ENT(AUDIO_GETOOFFS_wrap_RDONLY),
       ENT(AUDIO_GETOOFFS_wrap_WRONLY),
       ENT(AUDIO_GETOOFFS_wrap_RDWR),
       ENT(AUDIO_GETOOFFS_flush_RDONLY),
       ENT(AUDIO_GETOOFFS_flush_WRONLY),
       ENT(AUDIO_GETOOFFS_flush_RDWR),
       ENT(AUDIO_GETOOFFS_set_RDONLY),
       ENT(AUDIO_GETOOFFS_set_WRONLY),
       ENT(AUDIO_GETOOFFS_set_RDWR),
       ENT(audioctl_open_1_RDONLY_RDONLY),
       ENT(audioctl_open_1_RDONLY_RWONLY),
       ENT(audioctl_open_1_RDONLY_RDWR),
       ENT(audioctl_open_1_WRONLY_RDONLY),
       ENT(audioctl_open_1_WRONLY_RWONLY),
       ENT(audioctl_open_1_WRONLY_RDWR),
       ENT(audioctl_open_1_RDWR_RDONLY),
       ENT(audioctl_open_1_RDWR_RWONLY),
       ENT(audioctl_open_1_RDWR_RDWR),
       ENT(audioctl_open_2_RDONLY_RDONLY),
       ENT(audioctl_open_2_RDONLY_RWONLY),
       ENT(audioctl_open_2_RDONLY_RDWR),
       ENT(audioctl_open_2_WRONLY_RDONLY),
       ENT(audioctl_open_2_WRONLY_RWONLY),
       ENT(audioctl_open_2_WRONLY_RDWR),
       ENT(audioctl_open_2_RDWR_RDONLY),
       ENT(audioctl_open_2_RDWR_RWONLY),
       ENT(audioctl_open_2_RDWR_RDWR),
       ENT(audioctl_open_simul),
/**/    ENT(audioctl_open_multiuser0_audio1),   // XXX TODO sysctl
/**/    ENT(audioctl_open_multiuser1_audio1),   // XXX TODO sysctl
/**/    ENT(audioctl_open_multiuser0_audio2),   // XXX TODO sysctl
/**/    ENT(audioctl_open_multiuser1_audio2),   // XXX TODO sysctl
/**/    ENT(audioctl_open_multiuser0_audioctl), // XXX TODO sysctl
/**/    ENT(audioctl_open_multiuser1_audioctl), // XXX TODO sysctl
       ENT(audioctl_rw_RDONLY),
       ENT(audioctl_rw_WRONLY),
       ENT(audioctl_rw_RDWR),
       ENT(audioctl_poll),
       ENT(audioctl_kqueue),
       {.name = NULL},
};