⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 audlinux_oss.cpp

📁 著名的 helix realplayer 基于手机 symbian 系统的 播放器全套源代码
💻 CPP
📖 第 1 页 / 共 2 页
字号:
    }
    
    if ( m_nDevID < 0 )
    {
#ifdef _DEBUG        
        fprintf( stderr, "Failed to open audio(%s)!!!!!!! Code is: %d  errno: %d\n",
                 szDevName, m_nDevID, errno );
#endif        
        
        //Error opening device.
        retCode = RA_AOE_BADOPEN;
    }
    
    m_wLastError = retCode;
    return m_wLastError;
}

HX_RESULT CAudioOutLinux::_CloseAudio()
{
    HX_RESULT retCode = RA_AOE_NOERR;
    
    if( m_nDevID >= 0 )
    {
        ::close( m_nDevID );
        m_nDevID = NO_FILE_DESCRIPTOR;
    }
    else
    {
        retCode = RA_AOE_DEVNOTOPEN;
    }
    m_wLastError = retCode;
    return m_wLastError;
}

HX_RESULT CAudioOutLinux::_OpenMixer()
{
    HX_RESULT retCode = RA_AOE_NOERR;

    if(!m_bMixerPresent)
    {
        //Let user override default device with environ variable.
        char *pszOverrideName = getenv( "MIXER" ); /* Flawfinder: ignore */
        char szDevCtlName[MAX_DEV_NAME]; /* Flawfinder: ignore */
        
        if (pszOverrideName && strlen(pszOverrideName)>0 )
	{
            SafeStrCpy( szDevCtlName , pszOverrideName, MAX_DEV_NAME );
	}
        else
	{
            SafeStrCpy( szDevCtlName , "/dev/mixer", MAX_DEV_NAME );	// default for volume
	}
        
        m_nMixerID = ::open( szDevCtlName, O_RDWR );
        
        if (m_nMixerID > 0)
        {
            m_bMixerPresent = 1;
            _Imp_GetVolume();
        }
        else
        {
            m_nMixerID = NO_FILE_DESCRIPTOR;
            m_bMixerPresent = 0;
        }
    }

    m_wLastError = retCode;
    return m_wLastError;
}

HX_RESULT CAudioOutLinux::_CloseMixer()
{
    HX_RESULT retCode = RA_AOE_NOERR;

    if( m_nMixerID >= 0 )
    {
        ::close( m_nMixerID );
        m_nMixerID = NO_FILE_DESCRIPTOR;
    }
    
    m_wLastError = retCode;
    return m_wLastError;
}

//Device specific method to reset device and return it to a state that it 
//can accept new sample rates, num channels, etc.
HX_RESULT CAudioOutLinux::_Reset()
{
    HX_RESULT retCode = RA_AOE_NOERR;

    m_ulPausePosition = 0;
    
    if ( m_nDevID < 0 )
    {
        retCode = RA_AOE_DEVNOTOPEN;
    }
    else if( ::ioctl (m_nDevID, SOUND_PCM_RESET, 0) == -1 )
    {
        retCode = RA_AOE_GENERAL;
    }

    m_wLastError = retCode;
    return m_wLastError;
}

//Device specific method to get/set the devices current volume.
UINT16 CAudioOutLinux::_GetVolume() const
{
    int nVolume      = 0;
    int nRetVolume   = 0;
    int nLeftVolume  = 0;
    int nRightVolume = 0;
    
    if (::ioctl( m_nMixerID, MIXER_READ(HX_VOLUME), &nVolume) < 0)
    {
        nRetVolume = 0;
    }
    nLeftVolume  = (nVolume & 0x000000ff); 
    nRightVolume = (nVolume & 0x0000ff00) >> 8;

    //Which one to use? Average them?
    nRetVolume = nLeftVolume ;

    return nRetVolume; 
}

HX_RESULT CAudioOutLinux::_SetVolume(UINT16 unVolume)
{
    HX_RESULT retCode = RA_AOE_NOERR;
    
    int nNewVolume=0;

    //Set both left and right volumes.
    nNewVolume = (unVolume & 0xff) | ((unVolume &0xff) << 8);
    
    if (::ioctl( m_nMixerID, MIXER_WRITE(HX_VOLUME), &nNewVolume) < 0)
    {
        retCode = RA_AOE_NOTSUPPORTED;
    }

    m_wLastError = retCode;
    return m_wLastError;
}

//Device specific method to drain a device. This should play the remaining
//bytes in the devices buffer and then return.
HX_RESULT CAudioOutLinux::_Drain()
{
    HX_RESULT retCode = RA_AOE_NOERR;
    
    if ( m_nDevID < 0 )
    {
        retCode = RA_AOE_DEVNOTOPEN;
    }
    else if( ::ioctl (m_nDevID, SNDCTL_DSP_SYNC, 0) == -1 )
    {
        retCode = RA_AOE_GENERAL;
    }

    m_wLastError = retCode;
    return m_wLastError;
}



UINT64 CAudioOutLinux::_GetBytesActualyPlayed(void) const
{
    /* Get current playback position in device DMA. */
    int     bytes2 = 0;
    UINT64  ulTheAnswer = 0;

    //What versions of the linux kernel do we want to support?
    if( !m_bGetODelayFailed )
    {
        if( m_ulTotalWritten > 0 )
        {
            HX_ASSERT( m_unSampleRate!=0 && m_uSampFrameSize!=0 );
            ULONG32 ulTick = GetTickCount();

            //We need to update the timestamps every so often.
            //This make sure that if the XServer was blocked, and
            //we ran dry, that we re-sync up.
            if( (ulTick-m_ulLastTimeStamp)>200 )
            {
                ((CAudioOutLinux*)this)->_SyncUpTimeStamps();
                ulTick = GetTickCount();
            }
            
            ulTheAnswer = (UINT64)(m_ulLastBytesPlayed+
            ((float)(ulTick-m_ulLastTimeStamp)*
             (float)m_unNumChannels/1000.0*
             m_unSampleRate*m_uSampFrameSize) +0.5 );
        }
    }
    else
    {
        //We will assume that the error is because of an incomplete 
        //implementation of the oss compatible driver. So, just 
        //fake it with time stamps.
        if( m_ulTotalWritten > 0 )
        {
            ulTheAnswer = (UINT64)((float)(GetTickCount()-m_ulTickCount)*(float)m_unNumChannels/1000.0*m_unSampleRate*m_uSampFrameSize);
            ulTheAnswer += m_ulPausePosition;
        }
    }
    return  ulTheAnswer;
}


//this must return the number of bytes that can be written without blocking.
//Don't use SNDCTL_DSP_GETODELAY here as it can't compute that amount
//correctly.
HX_RESULT CAudioOutLinux::_GetRoomOnDevice(ULONG32& ulBytes) const
{
    HX_RESULT      retCode = RA_AOE_NOERR;
    audio_buf_info stBuffInfo;
    int            theErr=0;

    //What versions of the linux kernel do we want to support?
    if( m_bGetOSpaceFailed )
    {
        theErr = -1;
    }
    else
    {
        theErr = ::ioctl(m_nDevID, SNDCTL_DSP_GETOSPACE, &stBuffInfo);
    }

    if ( theErr != -1 )
    {
        ulBytes = stBuffInfo.bytes;
    }
    else
    {
        //So we just try it once.
        m_bGetOSpaceFailed = TRUE;
        ulBytes = m_ulDeviceBufferSize-(m_ulTotalWritten-_GetBytesActualyPlayed() );
    }

    m_wLastError = retCode;
    return m_wLastError;
}

HX_RESULT CAudioOutLinux::_CheckFormat( const HXAudioFormat* pFormat )
{
    int   nBitsPerSample = pFormat->uBitsPerSample;
    int   ulTmp          = pFormat->ulSamplesPerSec;    
    int   nNumChannels   = pFormat->uChannels;
    float fTmp           = 0.0;

    HX_RESULT retCode = RA_AOE_NOERR;
    
    //Is the device already open?
    if( m_nDevID > 0 || RA_AOE_NOERR != _OpenAudio() )
    {
        retCode = RA_AOE_DEVBUSY;
        return retCode;
    }

    //See if the sample rate is supported.
    if (ioctl(m_nDevID, SOUND_PCM_WRITE_RATE, &ulTmp) == -1)
    {
        //Not quite the real error, but it is what we need to return.
        retCode = RA_AOE_DEVBUSY;
        goto donechecking;
    }

    if (ulTmp == 0)
    {
	/* 
	 * Some drivers actually set the sample rate on the device, but
	 * return 0 for the sample rate. On these platforms we just ignore
	 * the return value and assume the sample rate is set to what was
	 * requested.
	 */

	ulTmp = pFormat->ulSamplesPerSec;
    }

    //The ESS 1688 Sound card (not the ESS Solo-1) will return sample
    //rates that are close but not quite the ones we asked for (the
    //ESS Solo-1 doesn't play mono). I have see freqs like 44194
    //instead of 44100 (.2%) and 7984 instead of 8000 (.6%).
    //So, if we are close enough just say it is OK. 
    //How about 1%?
    fTmp = (float)ulTmp/(float)pFormat->ulSamplesPerSec;
    if( fabs(1.0-fTmp) > .01 )
    {
        //It is NOT supported
        retCode = RA_AOE_BADFORMAT;
        goto donechecking;
    }

    //Check num channels.
    if (ioctl(m_nDevID, SOUND_PCM_WRITE_CHANNELS, &nNumChannels) == -1)
    {
        retCode = RA_AOE_DEVBUSY;
        goto donechecking;
    }
    else if ( nNumChannels != pFormat->uChannels )
    {
        retCode = RA_AOE_BADFORMAT;
        goto donechecking;
    }

    //Check the frame size.
    if (ioctl(m_nDevID, SNDCTL_DSP_SETFMT, &nBitsPerSample) == -1)
    {
       retCode = RA_AOE_DEVBUSY;
       goto donechecking;
    }
    else if ( nBitsPerSample != pFormat->uBitsPerSample )
    {
        retCode = RA_AOE_BADFORMAT;
        goto donechecking;
    }

    //Close the audio device.
  donechecking:
    _CloseAudio();
    m_wLastError = retCode;
    return retCode;
}


HX_RESULT CAudioOutLinux::_CheckSampleRate( ULONG32 ulSampleRate )
{
    ULONG32 ulTmp = ulSampleRate;

    m_wLastError = RA_AOE_NOERR;
    
    //Is the device already open?
    if( m_nDevID > 0 || RA_AOE_NOERR != _OpenAudio() )
    {
        m_wLastError = RA_AOE_DEVBUSY;
    }
    else
    {   
        //See if the sample rate is supported.
        if (ioctl(m_nDevID, SOUND_PCM_WRITE_RATE, &ulTmp) == -1)
        {
            //Not quite the real error, but it is what we need to return.
            m_wLastError = RA_AOE_DEVBUSY;
        }
        else if( ulSampleRate != ulTmp )
        {
            //It is NOT supported
            m_wLastError = RA_AOE_BADFORMAT;
        }
        
        _CloseAudio();
    }
    
    return m_wLastError;
}


HX_RESULT CAudioOutLinux::_Pause() 
{
    m_wLastError = HXR_OK;
    m_ulPausePosition = m_ulTotalWritten;
    m_ulTickCount = 0;
    m_ulLastTimeStamp   = 0;
    return m_wLastError;
}

HX_RESULT CAudioOutLinux::_Resume()
{
    m_wLastError = HXR_OK;

    if( m_ulTotalWritten > 0 )
    {
        m_ulTickCount = GetTickCount();
        m_ulLastTimeStamp = m_ulTickCount;
    }
    return m_wLastError;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -