wave.c example

This is a sample application that plays back audio data.

/*
 * $QNXLicenseC:
 * Copyright 2007, QNX Software Systems. All Rights Reserved.
 * 
 * You must obtain a written license from and pay applicable license fees to QNX 
 * Software Systems before you may reproduce, modify or distribute this software, 
 * or any work that includes all or part of this software.   Free development 
 * licenses are available for evaluation and non-commercial purposes.  For more 
 * information visit http://licensing.qnx.com or email licensing@qnx.com.
 *  
 * This file may contain contributions from others.  Please review this entire 
 * file for other proprietary rights or license notices, as well as the QNX 
 * Development Suite License Guide at http://licensing.qnx.com/license-guide/ 
 * for other information.
 * $
 */





#include <errno.h>
#include <fcntl.h>
#include <gulliver.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/ioctl.h>
#include <sys/select.h>
#include <sys/stat.h>
#include <sys/termio.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/slogcodes.h>
#include <ctype.h>
#include <limits.h>

#include <sys/asoundlib.h>


const char *kRiffId = "RIFF";
const char *kWaveId = "WAVE";

typedef struct
{
    char    tag[4];
    long    length;
}
RiffTag;

typedef struct
{
    char    Riff[4];
    long    Size;
    char    Wave[4];
}
RiffHdr;


typedef struct
{
    short   FormatTag;
    short   Channels;
    long    SamplesPerSec;
    long    AvgBytesPerSec;
    short   BlockAlign;
    short   BitsPerSample;
}
WaveHdr;


int
err (char *msg)
{
    perror (msg);
    return -1;
}


int
FindTag (FILE * fp, const char *tag)
{
    int     retVal;
    RiffTag tagBfr = { "", 0 };

    retVal = 0;

    // Keep reading until we find the tag or hit the EOF.
    while (fread ((unsigned char *) &tagBfr, sizeof (tagBfr), 1, fp))
    {

        // If this is our tag, set the length and break.    
        if (strncmp (tag, tagBfr.tag, sizeof tagBfr.tag) == 0)
        {
            retVal = ENDIAN_LE32 (tagBfr.length);
            break;
        }

        // Skip ahead the specified number of bytes in the stream           
        fseek (fp, tagBfr.length, SEEK_CUR);
    }

    // Return the result of our operation
    return (retVal);
}


int
CheckHdr (FILE * fp)
{
    RiffHdr riffHdr = { "", 0 };

    // Read the header and, if successful, play the file
    // file or WAVE file.
    if (fread ((unsigned char *) &riffHdr, sizeof (RiffHdr), 1, fp) == 0)
        return 0;

    if (strncmp (riffHdr.Riff, kRiffId, strlen (kRiffId)) ||
        strncmp (riffHdr.Wave, kWaveId, strlen (kWaveId)))
        return -1;

    return 0;
}


int
dev_raw (int fd)
{
    struct termios termios_p;

    if (tcgetattr (fd, &termios_p))
        return (-1);

    termios_p.c_cc[VMIN] = 1;
    termios_p.c_cc[VTIME] = 0;
    termios_p.c_lflag &= ~(ECHO | ICANON | ISIG | ECHOE | ECHOK | ECHONL);
    termios_p.c_oflag &= ~(OPOST);
    return (tcsetattr (fd, TCSANOW, &termios_p));
}

int
dev_unraw (int fd)
{
    struct termios termios_p;

    if (tcgetattr (fd, &termios_p))
        return (-1);

    termios_p.c_lflag |= (ECHO | ICANON | ISIG | ECHOE | ECHOK | ECHONL);
    termios_p.c_oflag |= (OPOST);
    return (tcsetattr (fd, TCSAFLUSH, &termios_p));
}


//*****************************************************************************
/* *INDENT-OFF* */
#ifdef __USAGE
%C[Options] *

Options:
    -a[card#:]<dev#>   the card & device number to play out on
    -f<frag_size>      requested fragment size
    -v                 verbose
    -c<args>[,args ..] voice matrix configuration
    -n<num_frags>      requested number of fragments

Args:
    1=<hw_channel_bitmask> hardware channel bitmask for application voice 1
    2=<hw_channel_bitmask> hardware channel bitmask for application voice 2
    3=<hw_channel_bitmask> hardware channel bitmask for application voice 3
    4=<hw_channel_bitmask> hardware channel bitmask for application voice 4
#endif
/* *INDENT-ON* */
//*****************************************************************************


int
main (int argc, char **argv)
{
    int     card = -1;
    int     dev = 0;
    snd_pcm_t *pcm_handle;
    FILE   *file1;
    WaveHdr wavHdr1;
    int     mSamples;
    int     mSampleRate;
    int     mSampleChannels;
    int     mSampleBits;
    char   *mSampleBfr1;
    int     fragsize = -1;
    int     verbose = 0;

    int     rtn;
    snd_pcm_channel_info_t pi;
    snd_mixer_t *mixer_handle;
    snd_mixer_group_t group;
    snd_pcm_channel_params_t pp;
    snd_pcm_channel_setup_t setup;
    int     bsize, n, N = 0, c;
    fd_set  rfds, wfds;
    uint32_t voice_mask[] = { 0, 0, 0, 0 };
    snd_pcm_voice_conversion_t voice_conversion;
    int     voice_override = 0;
    int     num_frags = -1;
    char   *sub_opts, *value;
    char   *dev_opts[] = {
#define CHN1 0
        "1",
#define CHN2 1
        "2",
#define CHN3 2
        "3",
#define CHN4 3
        "4",
        NULL
    };
    char    name[_POSIX_PATH_MAX] = { 0 };

    while ((c = getopt (argc, argv, "a:f:vc:n:")) != EOF)
    {
        switch (c)
        {
        case 'a':
            if (strchr (optarg, ':'))
            {
                card = atoi (optarg);
                dev = atoi (strchr (optarg, ':') + 1);
            }
            else if (isalpha (optarg[0]))
                strcpy (name, optarg);
            else
                dev = atoi (optarg);
            if (name[0] != '\0')
                printf ("Using device /dev/snd/%s\n", name);
            else
                printf ("Using card %d device %d \n", card, dev);
            break;
        case 'f':
            fragsize = atoi (optarg);
            break;
        case 'v':
            verbose = 1;
            break;
        case 'c':
            sub_opts = strdup (optarg);
            while (*sub_opts != '\0')
            {
                switch (getsubopt (&sub_opts, dev_opts, &value))
                {
                case CHN1:
                    voice_mask[0] = strtoul (value, NULL, 0);
                    break;
                case CHN2:
                    voice_mask[1] = strtoul (value, NULL, 0);
                    break;
                case CHN3:
                    voice_mask[2] = strtoul (value, NULL, 0);
                    break;
                case CHN4:
                    voice_mask[3] = strtoul (value, NULL, 0);
                    break;
                default:
                    break;
                }
            }
            voice_override = 1;
            break;
        case 'n':
            num_frags = atoi (optarg) - 1;
            break;
        default:
            return 1;
        }
    }

    setvbuf (stdin, NULL, _IONBF, 0);

    if (name[0] != '\0')
    {
        snd_pcm_info_t info;

        if ((rtn = snd_pcm_open_name (&pcm_handle, name, SND_PCM_OPEN_PLAYBACK)) < 0)
        {
            return err ("open_name");
        }
        rtn = snd_pcm_info (pcm_handle, &info);
        card = info.card;
    }
    else
    {
        if (card == -1)
        {
            if ((rtn =
                    snd_pcm_open_preferred (&pcm_handle, &card, &dev, SND_PCM_OPEN_PLAYBACK)) < 0)
                return err ("device open");
        }
        else
        {
            if ((rtn = snd_pcm_open (&pcm_handle, card, dev, SND_PCM_OPEN_PLAYBACK)) < 0)
                return err ("device open");
        }
    }

    if (argc < 2)
        return err ("no file specified");

    if ((file1 = fopen (argv[optind], "r")) == 0)
        return err ("file open #1");

    if (CheckHdr (file1) == -1)
        return err ("CheckHdr #1");

    mSamples = FindTag (file1, "fmt ");
    fread (&wavHdr1, sizeof (wavHdr1), 1, file1);
    fseek (file1, (mSamples - sizeof (WaveHdr)), SEEK_CUR);

    mSampleRate = ENDIAN_LE32 (wavHdr1.SamplesPerSec);
    mSampleChannels = ENDIAN_LE16 (wavHdr1.Channels);
    mSampleBits = ENDIAN_LE16 (wavHdr1.BitsPerSample);

    printf ("SampleRate = %d, Channels = %d, SampleBits = %d\n", mSampleRate, mSampleChannels,
        mSampleBits);

    /* disabling mmap is not actually required in this example but it is included to 
     * demonstrate how it is used when it is required.
     */
    if ((rtn = snd_pcm_plugin_set_disable (pcm_handle, PLUGIN_DISABLE_MMAP)) < 0)
    {
        fprintf (stderr, "snd_pcm_plugin_set_disable failed: %s\n", snd_strerror (rtn));
        return -1;
    }

    memset (&pi, 0, sizeof (pi));
    pi.channel = SND_PCM_CHANNEL_PLAYBACK;
    if ((rtn = snd_pcm_plugin_info (pcm_handle, &pi)) < 0)
    {
        fprintf (stderr, "snd_pcm_plugin_info failed: %s\n", snd_strerror (rtn));
        return -1;
    }

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

    pp.mode = SND_PCM_MODE_BLOCK;
    pp.channel = SND_PCM_CHANNEL_PLAYBACK;
    pp.start_mode = SND_PCM_START_FULL;
    pp.stop_mode = SND_PCM_STOP_STOP;

    pp.buf.block.frag_size = pi.max_fragment_size;
    if (fragsize != -1)
    {
        pp.buf.block.frag_size = fragsize;
    }
    pp.buf.block.frags_max = num_frags;
    pp.buf.block.frags_min = 1;

    pp.format.interleave = 1;
    pp.format.rate = mSampleRate;
    pp.format.voices = mSampleChannels;

    if (ENDIAN_LE16 (wavHdr1.FormatTag) == 6)
        pp.format.format = SND_PCM_SFMT_A_LAW;
    else if (ENDIAN_LE16 (wavHdr1.FormatTag) == 7)
        pp.format.format = SND_PCM_SFMT_MU_LAW;
    else if (mSampleBits == 8)
        pp.format.format = SND_PCM_SFMT_U8;
    else if (mSampleBits == 24)
        pp.format.format = SND_PCM_SFMT_S24;
    else
        pp.format.format = SND_PCM_SFMT_S16_LE;


    strcpy (pp.sw_mixer_subchn_name, "Wave playback channel");
    if ((rtn = snd_pcm_plugin_params (pcm_handle, &pp)) < 0)
    {
        fprintf (stderr, "snd_pcm_plugin_params failed: %s\n", snd_strerror (rtn));
        return -1;
    }

    if ((rtn = snd_pcm_plugin_prepare (pcm_handle, SND_PCM_CHANNEL_PLAYBACK)) < 0)
        fprintf (stderr, "snd_pcm_plugin_prepare failed: %s\n", snd_strerror (rtn));

    if (voice_override)
    {
        snd_pcm_plugin_get_voice_conversion (pcm_handle, SND_PCM_CHANNEL_PLAYBACK,
            &voice_conversion);
        voice_conversion.matrix[0] = voice_mask[0];
        voice_conversion.matrix[1] = voice_mask[1];
        voice_conversion.matrix[2] = voice_mask[2];
        voice_conversion.matrix[3] = voice_mask[3];
        snd_pcm_plugin_set_voice_conversion (pcm_handle, SND_PCM_CHANNEL_PLAYBACK,
            &voice_conversion);
    }

    memset (&setup, 0, sizeof (setup));
    memset (&group, 0, sizeof (group));
    setup.channel = SND_PCM_CHANNEL_PLAYBACK;
    setup.mixer_gid = &group.gid;
    if ((rtn = snd_pcm_plugin_setup (pcm_handle, &setup)) < 0)
    {
        fprintf (stderr, "snd_pcm_plugin_setup failed: %s\n", snd_strerror (rtn));
        return -1;
    }
    printf ("Format %s \n", snd_pcm_get_format_name (setup.format.format));
    printf ("Frag Size %d \n", setup.buf.block.frag_size);
    printf ("Total Frags %d \n", setup.buf.block.frags);
    printf ("Rate %d \n", setup.format.rate);
    printf ("Voices %d \n", setup.format.voices);
    bsize = setup.buf.block.frag_size;

    if (group.gid.name[0] == 0)
    {
        printf ("Mixer Pcm Group [%s] Not Set \n", group.gid.name);
        exit (-1);
    }
    printf ("Mixer Pcm Group [%s]\n", group.gid.name);
    if ((rtn = snd_mixer_open (&mixer_handle, card, setup.mixer_device)) < 0)
    {
        fprintf (stderr, "snd_mixer_open failed: %s\n", snd_strerror (rtn));
        return -1;
    }

    mSamples = FindTag (file1, "data");

    mSampleBfr1 = malloc (bsize);
    FD_ZERO (&rfds);
    FD_ZERO (&wfds);
    n = 1;
    while (N < mSamples && n > 0)
    {
        if (tcgetpgrp (0) == getpid ())
            FD_SET (STDIN_FILENO, &rfds);
        FD_SET (snd_mixer_file_descriptor (mixer_handle), &rfds);
        FD_SET (snd_pcm_file_descriptor (pcm_handle, SND_PCM_CHANNEL_PLAYBACK), &wfds);

        rtn = max (snd_mixer_file_descriptor (mixer_handle),
            snd_pcm_file_descriptor (pcm_handle, SND_PCM_CHANNEL_PLAYBACK));

        if (select (rtn + 1, &rfds, &wfds, NULL, NULL) == -1)
            return err ("select");


        if (FD_ISSET (STDIN_FILENO, &rfds))
        {
            if ((rtn = snd_mixer_group_read (mixer_handle, &group)) < 0)
                fprintf (stderr, "snd_mixer_group_read failed: %s\n", snd_strerror (rtn));

            dev_raw (fileno (stdin));
            c = getc (stdin);
            dev_unraw (fileno (stdin));

            if (c != EOF)
            {
                switch (c)
                {
                case 'q':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
                        group.volume.names.front_left += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
                        group.volume.names.rear_left += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
                        group.volume.names.woofer += 10;

                    break;
                case 'a':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
                        group.volume.names.front_left -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
                        group.volume.names.rear_left -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
                        group.volume.names.woofer -= 10;
                    break;
                case 'w':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
                        group.volume.names.front_left += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
                        group.volume.names.rear_left += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_CENTER)
                        group.volume.names.front_center += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_RIGHT)
                        group.volume.names.front_right += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_RIGHT)
                        group.volume.names.rear_right += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
                        group.volume.names.woofer += 10;
                    break;
                case 's':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
                        group.volume.names.front_left -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
                        group.volume.names.rear_left -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_CENTER)
                        group.volume.names.front_center -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_RIGHT)
                        group.volume.names.front_right -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_RIGHT)
                        group.volume.names.rear_right -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
                        group.volume.names.woofer -= 10;
                    break;
                case 'e':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_RIGHT)
                        group.volume.names.front_right += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_RIGHT)
                        group.volume.names.rear_right += 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_CENTER)
                        group.volume.names.front_center += 10;
                    break;
                case 'd':
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_RIGHT)
                        group.volume.names.front_right -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_REAR_RIGHT)
                        group.volume.names.rear_right -= 10;
                    if (group.channels & SND_MIXER_CHN_MASK_FRONT_CENTER)
                        group.volume.names.front_center -= 10;
                    break;
                }
                if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
                {
                    if (group.volume.names.front_left > group.max)
                        group.volume.names.front_left = group.max;
                    if (group.volume.names.front_left < group.min)
                        group.volume.names.front_left = group.min;
                }
                if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
                {
                    if (group.volume.names.rear_left > group.max)
                        group.volume.names.rear_left = group.max;
                    if (group.volume.names.rear_left < group.min)
                        group.volume.names.rear_left = group.min;
                }
                if (group.channels & SND_MIXER_CHN_MASK_FRONT_CENTER)
                {
                    if (group.volume.names.front_center > group.max)
                        group.volume.names.front_center = group.max;
                    if (group.volume.names.front_center < group.min)
                        group.volume.names.front_center = group.min;
                }
                if (group.channels & SND_MIXER_CHN_MASK_FRONT_RIGHT)
                {
                    if (group.volume.names.front_right > group.max)
                        group.volume.names.front_right = group.max;
                    if (group.volume.names.front_right < group.min)
                        group.volume.names.front_right = group.min;
                }
                if (group.channels & SND_MIXER_CHN_MASK_REAR_RIGHT)
                {
                    if (group.volume.names.rear_right > group.max)
                        group.volume.names.rear_right = group.max;
                    if (group.volume.names.rear_right < group.min)
                        group.volume.names.rear_right = group.min;
                }
                if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
                {
                    if (group.volume.names.woofer > group.max)
                        group.volume.names.woofer = group.max;
                    if (group.volume.names.woofer < group.min)
                        group.volume.names.woofer = group.min;
                }
                if ((rtn = snd_mixer_group_write (mixer_handle, &group)) < 0)
                    fprintf (stderr, "snd_mixer_group_write failed: %s\n", snd_strerror (rtn));
            }
            else
                exit (0);

            if (group.channels & SND_MIXER_CHN_MASK_FRONT_LEFT)
            {
                printf ("Volume Now at %d:%d \n",
                    100 * (group.volume.names.front_left - group.min) / (group.max - group.min),
                    100 * (group.volume.names.front_right - group.min) / (group.max - group.min));
            }
            else if (group.channels & SND_MIXER_CHN_MASK_REAR_LEFT)
            {
                printf ("Volume Now at %d:%d \n",
                    100 * (group.volume.names.rear_left - group.min) / (group.max - group.min),
                    100 * (group.volume.names.rear_right - group.min) / (group.max - group.min));
            }
            else if (group.channels & SND_MIXER_CHN_MASK_WOOFER)
            {
                printf ("Volume Now at %d:%d \n",
                    100 * (group.volume.names.woofer - group.min) / (group.max - group.min),
                    100 * (group.volume.names.front_center - group.min) / (group.max - group.min));
            }
            else
            {
                printf ("Volume Now at %d:%d \n",
                    100 * (group.volume.names.front_left - group.min) / (group.max - group.min),
                    100 * (group.volume.names.front_right - group.min) / (group.max - group.min));
            }

        }

        if (FD_ISSET (snd_mixer_file_descriptor (mixer_handle), &rfds))
        {
            snd_mixer_callbacks_t callbacks = { 0, 0, 0, 0 };

            snd_mixer_read (mixer_handle, &callbacks);
        }

        if (FD_ISSET (snd_pcm_file_descriptor (pcm_handle, SND_PCM_CHANNEL_PLAYBACK), &wfds))
        {
            snd_pcm_channel_status_t status;
            int     written = 0;

            if ((n = fread (mSampleBfr1, 1, min (mSamples - N, bsize), file1)) <= 0)
                continue;
            written = snd_pcm_plugin_write (pcm_handle, mSampleBfr1, n);
            if (verbose)
                printf ("bytes written = %d \n", written);
            if (written < n)
            {
                memset (&status, 0, sizeof (status));
                status.channel = SND_PCM_CHANNEL_PLAYBACK;
                if (snd_pcm_plugin_status (pcm_handle, &status) < 0)
                {
                    fprintf (stderr, "underrun: playback channel status error\n");
                    exit (1);
                }

                if (status.status == SND_PCM_STATUS_READY ||
                    status.status == SND_PCM_STATUS_UNDERRUN)
                {
                    if (snd_pcm_plugin_prepare (pcm_handle, SND_PCM_CHANNEL_PLAYBACK) < 0)
                    {
                        fprintf (stderr, "underrun: playback channel prepare error\n");
                        exit (1);
                    }
                }
                if (written < 0)
                    written = 0;
                written += snd_pcm_plugin_write (pcm_handle, mSampleBfr1 + written, n - written);
            }
            N += written;
        }
    }

    n = snd_pcm_plugin_flush (pcm_handle, SND_PCM_CHANNEL_PLAYBACK);

    rtn = snd_mixer_close (mixer_handle);
    rtn = snd_pcm_close (pcm_handle);
    fclose (file1);
    return (0);
}