aboutsummaryrefslogblamecommitdiff
path: root/src/plugins/Output/oss/outputoss.cpp
blob: 49a4c579352505e7aea622dcdb1eae1060d72ca9 (plain) (tree)




















                                                                             
                       







                           

                      


                           



                   

                    
































                                                          
                                  
                        
                                    




                                   






                        
                            







                                      





                                                                         
 




                                                                                





                       

                          


                       

                          





                                                                  

                                                                         
                                                                                  


                                                                   
 


                           
 
                                         
 




                     
                        
                            
     
                            
      
                  
 


                        
 
         
 











                                                                                              


                  






                       

                          





                                                             


                                                                      




                                                    

                                          











                                                              

                          





                           
                            








                                                                    


                                                                     
     
      



                       


                                                                                  




                       
               







                                                
               








                                                
               








                                                     
 


                       
                     
                       

                     
 
























                                

                          


                       

                          











                                      

                                                                 










                     
                          
 
               















                                      


                        
 
                                    
 
                          
 


                                                          
 
                              
 

                                                          
 



                              
 





                                   
 

                                      
 




                                 
 





















                                                                               
 
                            
 




                                          
 


                  
 
                          




                     
               










                                       


                

















                                                              

 

                                       
            


                
                            


                                                       
 












                                                           






                                         
 
/***************************************************************************
 *   Copyright (C) 2007 by Uriy Zhuravlev stalkerg@gmail.com               *
 *                                                                         *
 *   Copyright (c) 2000-2001 Brad Hughes bhughes@trolltech.com             *
 *                                                                         *
 *   This program is free software; you can redistribute it and/or modify  *
 *   it under the terms of the GNU General Public License as published by  *
 *   the Free Software Foundation; either version 2 of the License, or     *
 *   (at your option) any later version.                                   *
 *                                                                         *
 *   This program is distributed in the hope that it will be useful,       *
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
 *   GNU General Public License for more details.                          *
 *                                                                         *
 *   You should have received a copy of the GNU General Public License     *
 *   along with this program; if not, write to the                         *
 *   Free Software Foundation, Inc.,                                       *
 *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
 ***************************************************************************/

#include <QApplication>
extern "C"
{
#ifdef HAVE_SYS_SOUNDCARD_H
#include <sys/soundcard.h>
#else
#include <soundcard.h>
#endif
}

#include "outputoss.h"
#include <qmmp/constants.h>
#include <qmmp/buffer.h>
#include <qmmp/visual.h>

#include <stdio.h>
#include <string.h>
#include <QtGlobal>
#include <QSettings>
#include <QDir>

#include <iostream>

//extern Q_EXPORT QApplication* qApp;


void OutputOSS::stop()
{
    m_userStop = TRUE;
}

void OutputOSS::status()
{
    long ct = (m_totalWritten - latency()) / m_bps;

    if (ct < 0)
        ct = 0;

    if (ct > m_currentSeconds)
    {
        m_currentSeconds = ct;
        dispatch(m_currentSeconds, m_totalWritten, m_rate,
                 m_frequency, m_precision, m_channels);
    }
}

long OutputOSS::written()
{
    return m_totalWritten;
}

void OutputOSS::seek(long pos)
{
    /*recycler()->mutex()->lock();
    recycler()->clear();
    recycler()->mutex()->unlock();*/

    m_totalWritten = (pos * m_bps);
    m_currentSeconds = -1;
}


#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/time.h>

#if defined(__FreeBSD__)
#  include <sys/soundcard.h>
#elif defined(__linux__)
#  include <linux/soundcard.h>
#elif defined(__bsdi__)
#  include <sys/soundcard.h>
#endif


OutputOSS::OutputOSS(QObject * parent)
        : Output(parent), m_inited(FALSE), m_pause(FALSE), m_play(FALSE),
        m_userStop(FALSE),
        m_totalWritten(0), m_currentSeconds(-1),
        m_bps(1), m_frequency(-1), m_channels(-1), m_precision(-1),
        do_select(TRUE),
        m_audio_fd(-1), m_mixer_fd(-1)
{
    QSettings settings(QDir::homePath()+"/.qmmp/qmmprc", QSettings::IniFormat);
    m_master = true;
    m_audio_device = settings.value("OSS/device","/dev/dsp").toString();
    m_mixer_device = settings.value("OSS/mixer_device","/dev/mixer").toString();
    openMixer();
}

OutputOSS::~OutputOSS()
{
    if (m_audio_fd > 0)
    {
        close(m_audio_fd);
        m_audio_fd = -1;
    }
    if (m_mixer_fd > 0)
    {
        close(m_mixer_fd);
        m_mixer_fd = -1;
    }
}

void OutputOSS::configure(long freq, int chan, int prec, int rate)
{
    // we need to configure
    if (freq != m_frequency || chan != m_channels || prec != m_precision)
    {
        qDebug("OutputOSS: frequency=%d, channels=%d, bits=%d", freq, chan, prec);
        // we have already configured, but are changing settings...
        // reset the device
        resetDSP();

        m_frequency = freq;
        m_channels = chan;
        m_precision = prec;

        m_bps = freq * chan * (prec / 8);

        int p;
        switch (prec)
        {
        default:
        case 16:
#if defined(AFMT_S16_NE)
            p = AFMT_S16_NE;
#else
            p = AFMT_S16_LE;
#endif
            break;

        case 8:
            p = AFMT_S8;
            break;

        }

        if (ioctl(m_audio_fd, SNDCTL_DSP_SETFMT, &p) == -1)
            qWarning("OutputOSS: can't set audio format");
        /*if(ioctl(m_audio_fd, SNDCTL_DSP_SAMPLESIZE, &prec) == -1)
            qDebug("OutputOSS: can't set audio format");*/
        /*int stereo = (chan > 1) ? 1 : 0;
         ioctl(m_audio_fd, SNDCTL_DSP_STEREO, &stereo);*/
        if (ioctl(m_audio_fd, SNDCTL_DSP_SPEED, &m_channels) == -1)
            qWarning("OutputOSS: can't set number of channels");
        if (chan != m_channels)
            qWarning("OutputOSS: can't set number of channels, using %d instead", m_channels);
        if (ioctl(m_audio_fd, SNDCTL_DSP_SPEED, &freq) == -1)
            qWarning("OutputOSS: can't set audio format");
    }

    m_rate = rate;
}


void OutputOSS::reset()
{
    if (m_audio_fd > 0)
    {
        close(m_audio_fd);
        m_audio_fd = -1;
    }

    m_audio_fd = open(m_audio_device.toAscii(), O_WRONLY, 0);

    if (m_audio_fd < 0)
    {
        error(QString("OSSOutput: failed to open output device '%1'").
              arg(m_audio_device));
        return;
    }

    int flags;
    if ((flags = fcntl(m_audio_fd, F_GETFL, 0)) > 0)
    {
        flags &= O_NDELAY;
        fcntl(m_audio_fd, F_SETFL, flags);
    }

    fd_set afd;
    FD_ZERO(&afd);
    FD_SET(m_audio_fd, &afd);
    struct timeval tv;
    tv.tv_sec = 0l;
    tv.tv_usec = 50000l;
    do_select = (select(m_audio_fd + 1, 0, &afd, 0, &tv) > 0);

    if (m_audio_fd > 0)
    {
        close(m_mixer_fd);
        m_mixer_fd = -1;
    }
    openMixer();
}

void OutputOSS::openMixer()
{
#if SOUND_VERSION < 0x040000
    if (m_mixer_fd != -1)
        return;

    if ((m_mixer_fd = open(m_mixer_device.toAscii(), O_RDWR)) == -1)
    {
        return;
    }
    if (m_audio_fd < 0)
    {
        error(QString("OSSOutput: failed to open mixer device '%1'").
              arg(m_mixer_device));
        return;
    }
#endif
}

void OutputOSS::pause()
{
    m_pause = !m_pause;
    OutputState::Type state = m_pause ? OutputState::Paused: OutputState::Playing;
    dispatch(state);
}

void OutputOSS::post()
{
    if (m_audio_fd < 1)
        return;

    int unused;
    ioctl(m_audio_fd, SNDCTL_DSP_POST, &unused);
}

void OutputOSS::sync()
{
    if (m_audio_fd < 1)
        return;

    int unused;
    ioctl(m_audio_fd, SNDCTL_DSP_SYNC, &unused);
}


void OutputOSS::resetDSP()
{
    if (m_audio_fd < 1)
        return;

    int unused;
    ioctl(m_audio_fd, SNDCTL_DSP_RESET, &unused);
}


bool OutputOSS::initialize()
{
    m_inited = m_pause = m_play = m_userStop = FALSE;


    reset();
    if (m_audio_fd < 0)
        return FALSE;
    if (m_mixer_fd < 0)
        return FALSE;


    m_currentSeconds = -1;
    m_totalWritten = 0;
    stat = OutputState::Stopped;

    m_inited = TRUE;
    return TRUE;
}

void OutputOSS::uninitialize()
{
    if (!m_inited)
        return;
    m_inited = FALSE;
    m_pause = FALSE;
    m_play = FALSE;
    m_userStop  = FALSE;
    m_totalWritten = 0;
    m_currentSeconds = -1;
    m_bps = -1;
    m_frequency = -1;
    m_channels = -1;
    m_precision = -1;
    resetDSP();
    if (m_audio_fd > 0)
    {
        close(m_audio_fd);
        m_audio_fd = -1;
    }
    if (m_audio_fd > 0)
    {
        close(m_mixer_fd);
        m_mixer_fd = -1;
    }

    qDebug("OutputOSS: uninitialize");
    dispatch(OutputState::Stopped);
}

long OutputOSS::latency()
{
    ulong used = 0;

    if (! m_pause)
    {
        if (ioctl(m_audio_fd, SNDCTL_DSP_GETODELAY, &used) == -1)
            used = 0;
    }

    return used;
}

void OutputOSS::run()
{
    mutex()->lock();

    if (! m_inited)
    {
        mutex()->unlock();

        return;
    }

    m_play = TRUE;

    mutex()->unlock();

    fd_set afd;
    struct timeval tv;
    Buffer *b = 0;
    bool done = FALSE;
    unsigned long n = 0, m = 0, l = 0;

    dispatch(OutputState::Playing);

    FD_ZERO(&afd);

    while (! done)
    {
        mutex()->lock();

        recycler()->mutex()->lock();

        done = m_userStop;

        while (! done && (recycler()->empty() || m_pause))
        {
            post();

            mutex()->unlock();

            recycler()->cond()->wakeOne();
            recycler()->cond()->wait(recycler()->mutex());

            mutex()->lock();
            done = m_userStop;
            status();
        }

        if (! b)
        {
            b = recycler()->next();
            if (b->rate)
                m_rate = b->rate;
        }

        recycler()->cond()->wakeOne();
        recycler()->mutex()->unlock();

        FD_ZERO(&afd);
        FD_SET(m_audio_fd, &afd);
        // nice long poll timeout
        tv.tv_sec = 5l;
        tv.tv_usec = 0l;

        if (b &&
                (! do_select || (select(m_audio_fd + 1, 0, &afd, 0, &tv) > 0 &&
                                 FD_ISSET(m_audio_fd, &afd))))
        {
            l = qMin(int(2048), int(b->nbytes - n));
            if (l > 0)
            {
                mutex()->unlock();
                m = write(m_audio_fd, b->data + n, l);
                mutex()->lock();
                n += m;

                status();
                dispatchVisual(b, m_totalWritten, m_channels, m_precision);
            }
            else
            {
                // force buffer change
                n = b->nbytes;
                m = 0;
            }
        }

        m_totalWritten += m;

        if (n == b->nbytes)
        {
            recycler()->mutex()->lock();
            recycler()->done();
            recycler()->mutex()->unlock();

            b = 0;
            n = 0;
        }

        mutex()->unlock();
    }

    mutex()->lock();

    if (! m_userStop)
        sync();
    resetDSP();

    m_play = FALSE;

    dispatch(OutputState::Stopped);
    mutex()->unlock();
}


void OutputOSS::setVolume(int l, int r)
{
    int v, devs;
    long cmd;

#if SOUND_VERSION < 0x040000
    ioctl(m_mixer_fd, SOUND_MIXER_READ_DEVMASK, &devs);
    if ((devs & SOUND_MASK_PCM) && (m_master == false))
        cmd = SOUND_MIXER_WRITE_PCM;
    else if ((devs & SOUND_MASK_VOLUME) && (m_master == true))
        cmd = SOUND_MIXER_WRITE_VOLUME;
    else
    {
        //close(mifd);
        return;
    }
    v = (r << 8) | l;
    ioctl(m_mixer_fd, cmd, &v);
#else
    cmd = SNDCTL_DSP_SETPLAYVOL;
    v = (r << 8) | l;
    ioctl(m_audio_fd, cmd, &v);
#endif
}

void OutputOSS::volume(int *ll,int *rr)
{
    *ll = 0;
    *rr = 0;
    int  cmd;
    int v, devs;
#if SOUND_VERSION < 0x040000
    ioctl(m_mixer_fd, SOUND_MIXER_READ_DEVMASK, &devs);
    if ((devs & SOUND_MASK_PCM) && (m_master == 0))
        cmd = SOUND_MIXER_READ_PCM;

    else if ((devs & SOUND_MASK_VOLUME) && (m_master == 1))
        cmd = SOUND_MIXER_READ_VOLUME;
    else
        return;

    ioctl(m_mixer_fd, cmd, &v);
    *ll = (v & 0xFF00) >> 8;
    *rr = (v & 0x00FF);

    *ll = (*ll > 100) ? 100 : *ll;
    *rr = (*rr > 100) ? 100 : *rr;
    *ll = (*ll < 0) ? 0 : *ll;
    *rr = (*rr < 0) ? 0 : *rr;
#else
    cmd = SNDCTL_DSP_GETPLAYVOL;
    if (ioctl(m_audio_fd, cmd, &v) == -1)
        v = 0;
    *rr = (v & 0xFF00) >> 8;
    *ll = (v & 0x00FF);
#endif
}