audSolaris.cpp
上传用户:zhongxx05
上传日期:2007-06-06
资源大小:33641k
文件大小:17k
源码类别:

Symbian

开发平台:

C/C++

  1. /* ***** BEGIN LICENSE BLOCK ***** 
  2.  * Version: RCSL 1.0/RPSL 1.0 
  3.  *  
  4.  * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
  5.  *      
  6.  * The contents of this file, and the files included with this file, are 
  7.  * subject to the current version of the RealNetworks Public Source License 
  8.  * Version 1.0 (the "RPSL") available at 
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed 
  10.  * the file under the RealNetworks Community Source License Version 1.0 
  11.  * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
  12.  * in which case the RCSL will apply. You may also obtain the license terms 
  13.  * directly from RealNetworks.  You may not use this file except in 
  14.  * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
  15.  * applicable to this file, the RCSL.  Please see the applicable RPSL or 
  16.  * RCSL for the rights, obligations and limitations governing use of the 
  17.  * contents of the file.  
  18.  *  
  19.  * This file is part of the Helix DNA Technology. RealNetworks is the 
  20.  * developer of the Original Code and owns the copyrights in the portions 
  21.  * it created. 
  22.  *  
  23.  * This file, and the files included with this file, is distributed and made 
  24.  * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  25.  * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  26.  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
  27.  * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  28.  * 
  29.  * Technology Compatibility Kit Test Suite(s) Location: 
  30.  *    http://www.helixcommunity.org/content/tck 
  31.  * 
  32.  * Contributor(s): 
  33.  *  
  34.  * ***** END LICENSE BLOCK ***** */ 
  35. #include "audSolaris.h"
  36. #include "ihxpckts.h"
  37. #include "hxstrutl.h"
  38. #include <unistd.h>
  39. #include <fcntl.h>
  40. #include <stdlib.h>
  41. #include <errno.h>
  42. #include <sys/ioctl.h>
  43. #include <sys/stropts.h> //for I_FLUSH
  44. #include <stdio.h>
  45. //------------------------------------------
  46. // Ctors and Dtors.
  47. //------------------------------------------
  48. CAudioOutSolaris::CAudioOutSolaris() :
  49.     CAudioOutUNIX(),
  50.     m_nDevID(NO_FILE_DESCRIPTOR),
  51.     m_nMixerID(NO_FILE_DESCRIPTOR),
  52.     m_nBlockDivisions(2)
  53. {
  54. };
  55. CAudioOutSolaris::~CAudioOutSolaris()
  56. {
  57.     //The mixer may be used without the audio device. Make sure
  58.     //it gets closed here.
  59.     _CloseMixer();
  60. };
  61. //-------------------------------------------------------
  62. // These Device Specific methods must be implemented 
  63. // by the platform specific sub-classes.
  64. //-------------------------------------------------------
  65. INT16 CAudioOutSolaris::_Imp_GetAudioFd(void)
  66. {
  67.     return m_nDevID;
  68. }
  69. //Devic specific method to set the audio device characteristics. Sample rate,
  70. //bits-per-sample, etc.
  71. //Method *must* set member vars. m_unSampleRate and m_unNumChannels.
  72. HX_RESULT CAudioOutSolaris::_SetDeviceConfig( const HXAudioFormat* pFormat )
  73. {
  74.     HX_RESULT  retCode = RA_AOE_NOERR;    
  75.     audio_info stAudioinfo;
  76.         
  77.     if( m_nDevID < 0 )
  78.     {
  79.         retCode = RA_AOE_DEVNOTOPEN;
  80.     }
  81.     else
  82.     {
  83.         AUDIO_INITINFO( &stAudioinfo );
  84.         stAudioinfo.play.sample_rate      = pFormat->ulSamplesPerSec;
  85.         stAudioinfo.record.sample_rate    = pFormat->ulSamplesPerSec;
  86.         stAudioinfo.play.channels         = pFormat->uChannels;
  87.         stAudioinfo.record.channels       = pFormat->uChannels;
  88.         stAudioinfo.play.precision        = pFormat->uBitsPerSample;
  89.         stAudioinfo.record.precision      = pFormat->uBitsPerSample;
  90.         stAudioinfo.play.encoding         = AUDIO_ENCODING_LINEAR;
  91.         stAudioinfo.record.encoding       = AUDIO_ENCODING_LINEAR;
  92. //            stAudioinfo.play.buffer_size      = m_ulBytesPerGran;
  93.             //Now set the new config.
  94.         if( ::ioctl(m_nDevID, AUDIO_SETINFO, &stAudioinfo) < 0 ) 
  95.         {
  96. #ifdef _DEBUG                
  97.             fprintf( stderr, "Error setting audio confign");
  98. #endif                
  99.             retCode = RA_AOE_GENERAL;
  100.         }
  101.         else
  102.         {
  103.             //Set member vars from CAudioOutUNIX.
  104.             m_wBlockSize         = m_ulBytesPerGran;
  105.             m_unNumChannels      = pFormat->uChannels;
  106.             m_uSampFrameSize     = pFormat->uBitsPerSample / 8;
  107.             m_unSampleRate       = pFormat->ulSamplesPerSec;
  108.             //XXXgfw Is there a size? Doesn't look like it.
  109.             //XXXgfw Lets just pick one that looks good.
  110.             m_ulDeviceBufferSize = 1<<16;
  111.         }
  112.             
  113.         AUDIO_INITINFO(&stAudioinfo);
  114.         if( ::ioctl(m_nDevID, AUDIO_GETINFO, &stAudioinfo) < 0 ) 
  115.         {
  116. #ifdef _DEBUG            
  117.             fprintf( stderr, "CAudioOutSolaris: Can't get audio configuration.n");
  118. #endif            
  119.             retCode = RA_AOE_GENERAL;
  120.         }
  121.         else
  122.         {
  123.             HX_ASSERT( stAudioinfo.play.sample_rate == m_unSampleRate ); 
  124.             HX_ASSERT( stAudioinfo.play.channels    == m_unNumChannels );
  125.             HX_ASSERT( stAudioinfo.play.precision   == m_uSampFrameSize*8 );
  126.             HX_ASSERT( stAudioinfo.play.encoding    == AUDIO_ENCODING_LINEAR );
  127. //                HX_ASSERT( stAudioinfo.play.buffer_size == m_ulBytesPerGran );
  128.         }
  129.     }
  130.     //XXXgfw 
  131. //      fprintf( stderr, "channels: %d   FrameSize: %d   SampleRate: %d   BuffSize: %d BlockSize: %dn",
  132. //               m_unNumChannels, m_uSampFrameSize, m_unSampleRate, m_ulDeviceBufferSize, m_wBlockSize);
  133.     m_wLastError = retCode;
  134.     return m_wLastError;
  135. }
  136. ULONG32 CAudioOutSolaris::_WriteWithEOF(UCHAR* buffer, ULONG32 ulBuffLength )
  137. {
  138.     HX_ASSERT( ulBuffLength != 0 );
  139.     
  140.     ULONG32 lCount = ::write( m_nDevID, buffer, ulBuffLength);
  141.     if( lCount > 0 )
  142.     {
  143.         //We wrote at least one byte. Write an EOF marker.
  144.         char  szEOF[1]; /* Flawfinder: ignore */
  145.         if( ::write( m_nDevID, szEOF, 0) < 0 )
  146.         {
  147.             //Some sort of error. This is going to throw off our
  148.             //ability to report the number of bytes played. We could
  149.             //keep track of how many times this happens and then just
  150.             //force a restart of the playback or something.
  151.             HX_ASSERT( "_WriteWithEOF: Failed to write EOF" == NULL );
  152.         }
  153.     }
  154.     else
  155.     {
  156.         HX_ASSERT( "_WriteWithEOF: Can't write to device." == NULL );
  157.     }
  158.     return lCount;
  159. }
  160. //Device specific method to write bytes out to the audiodevice and return a 
  161. //count of bytes written. 
  162. HX_RESULT CAudioOutSolaris::_WriteBytes( UCHAR* buffer, ULONG32 ulBuffLength, LONG32& lCount )
  163. {
  164.     HX_ASSERT( ulBuffLength==m_wBlockSize);
  165.     
  166.     HX_RESULT retCode = RA_AOE_NOERR;
  167.     if( m_nDevID < 0 )
  168.     {
  169.         retCode = RA_AOE_DEVNOTOPEN;
  170.     }
  171.     else if( ulBuffLength > 0 ) //Ignore all zero length writes. They screw up eof count.
  172.     {
  173.         //Try to increase resolution by splitting the write. ulBuffLength is 
  174.         //always even.
  175.         HX_ASSERT( (m_wBlockSize%m_nBlockDivisions) == 0 );
  176.         
  177.         int i=0;
  178.         int nChunkSize = m_wBlockSize/m_nBlockDivisions;
  179.         
  180.         for( i=0 ; i< m_nBlockDivisions ; i++ )
  181.         {
  182.             lCount = _WriteWithEOF( buffer+i*nChunkSize, nChunkSize );
  183.             if( lCount < 0 )
  184.                 break;
  185.         }
  186.         
  187.         if( lCount < 0 )
  188.         {
  189.             //Error occurred.
  190.             if( errno == EAGAIN )
  191.                 retCode = RA_AOE_NOERR;
  192.             if( errno == EINTR )
  193.                 retCode = RA_AOE_DEVBUSY;
  194.         }
  195.         else 
  196.         {
  197.             lCount = m_wBlockSize/m_nBlockDivisions*i;
  198.         }
  199.     }
  200.     
  201.     
  202.     return retCode;
  203. }
  204. //Device specific methods to open/close the mixer and audio devices.
  205. HX_RESULT CAudioOutSolaris::_OpenAudio()
  206. {
  207.     HX_RESULT retCode = RA_AOE_NOERR;
  208.     //Check the environmental variable to let user overide default device.
  209.     char *pszOverrideName = getenv( "AUDIO" ); /* Flawfinder: ignore */
  210.     char szDevName[MAX_DEV_NAME+1]; /* Flawfinder: ignore */
  211.     
  212.     // Use defaults if no environment variable is set.
  213.     if ( pszOverrideName && strlen(pszOverrideName)>0 )
  214.     {
  215.         SafeStrCpy( szDevName, pszOverrideName, MAX_DEV_NAME );
  216.     }
  217.     else
  218.     {
  219.         pszOverrideName = getenv( "AUDIODEV" ); /* Flawfinder: ignore */
  220.         if ( pszOverrideName && strlen(pszOverrideName)>0 )
  221.         {
  222.             SafeStrCpy( szDevName, pszOverrideName, MAX_DEV_NAME );
  223.         }
  224.         else
  225.         {
  226.             SafeStrCpy( szDevName, "/dev/audio", MAX_DEV_NAME);
  227.         }
  228.     }
  229.     szDevName[MAX_DEV_NAME]='';
  230.     
  231.     // Open the audio device if it isn't already open
  232.     if ( m_nDevID < 0 )
  233.     {
  234.         //We open in non block mode first, just to test wether or not
  235.         //someone has the audio device. After that we must reopen it
  236.         //so that our writes always block.
  237.         m_nDevID = ::open( szDevName, O_WRONLY | O_EXCL | O_NONBLOCK );
  238.         if( m_nDevID >= 0 )
  239.         {
  240.             //XXXgfw Yeah, I know, why don't I just use ioctl(....,FIONBIO, ...);
  241.             //or fcntl? because it doesn't work!
  242.             close(m_nDevID);
  243.             m_nDevID = ::open( szDevName, O_WRONLY | O_EXCL);
  244.         }
  245.         
  246.     }
  247.     
  248.     if ( m_nDevID < 0 )
  249.     {
  250.         //Error opening device.
  251.         retCode = RA_AOE_BADOPEN;
  252.     }
  253.     m_wLastError = retCode;
  254.     return m_wLastError;
  255. }
  256. HX_RESULT CAudioOutSolaris::_CloseAudio()
  257. {
  258.     HX_RESULT retCode = RA_AOE_NOERR; 
  259.     
  260.     if( m_nDevID >= 0 )
  261.     {
  262.         ::close( m_nDevID );
  263.         m_nDevID = NO_FILE_DESCRIPTOR;
  264.     }
  265.     else
  266.     {
  267.         retCode = RA_AOE_DEVNOTOPEN;
  268.     }
  269.     m_wLastError = retCode;
  270.     return m_wLastError;
  271. }
  272. HX_RESULT CAudioOutSolaris::_OpenMixer()
  273. {
  274.     HX_RESULT retCode = RA_AOE_NOERR;
  275.     if(!m_bMixerPresent)
  276.     {
  277.         //Let user override default device with environ variable.
  278.         char *pszOverrideName = getenv( "MIXER" ); /* Flawfinder: ignore */
  279.         char szDevCtlName[MAX_DEV_NAME]; /* Flawfinder: ignore */
  280.         
  281.         if (pszOverrideName && strlen(pszOverrideName)>0 )
  282.         {
  283.             SafeStrCpy( szDevCtlName , pszOverrideName, MAX_DEV_NAME );
  284.         }
  285.         else
  286.         {
  287.             SafeStrCpy( szDevCtlName, "/dev/audioctl", MAX_DEV_NAME); // default for volume
  288.         }
  289.         
  290.         m_nMixerID = ::open( szDevCtlName, O_RDWR );
  291.         
  292.         if (m_nMixerID > 0)
  293.         {
  294.             m_bMixerPresent = 1;
  295.             _Imp_GetVolume();
  296.         }
  297.         else
  298.         {
  299.             m_nMixerID = NO_FILE_DESCRIPTOR;
  300.             m_bMixerPresent = 0;
  301.         }
  302.     }
  303.     m_wLastError = retCode;
  304.     return m_wLastError;
  305. }
  306. HX_RESULT CAudioOutSolaris::_CloseMixer()
  307. {
  308.     HX_RESULT retCode = RA_AOE_NOERR;
  309.     if( m_nMixerID >= 0 )
  310.     {
  311.         ::close( m_nMixerID );
  312.         m_nMixerID = NO_FILE_DESCRIPTOR;
  313.     }
  314.     
  315.     m_wLastError = retCode;
  316.     return m_wLastError;
  317. }
  318. //Device specific method to reset device and return it to a state that it 
  319. //can accept new sample rates, num channels, etc.
  320. HX_RESULT CAudioOutSolaris::_Reset()
  321. {
  322.     HX_RESULT retCode = RA_AOE_NOERR;
  323.     
  324.     if ( m_nDevID < 0 )
  325.     {
  326.         retCode = RA_AOE_DEVNOTOPEN;
  327.     }
  328.     else if( ::ioctl(m_nDevID, I_FLUSH, FLUSHW) == -1 )
  329.     {
  330.         retCode = RA_AOE_GENERAL;
  331.     }
  332.     //Now, m_ulTotalWritten is going to go to Zero in CAudioOutUNIX.
  333.     //So, we must reset the EOF marker.
  334.     audio_info stAudioInfo;
  335.     AUDIO_INITINFO( &stAudioInfo );
  336.     
  337.     //Fill in the structure with current settings.
  338.     if (::ioctl(m_nDevID, AUDIO_GETINFO, &stAudioInfo) < 0) 
  339.     {
  340. #ifdef _DEBUG        
  341. fprintf( stderr, "CAudioOutSolaris: Just trying to reset EOF in _Reset.n");
  342. #endif        
  343.     }
  344.     //Just change the gain setting. Must map incoming [0,100]-->[0,255]
  345.     stAudioInfo.play.eof     = 0;
  346.     stAudioInfo.play.samples = 0;
  347.     //Now store the new settings.
  348.     if( ::ioctl(m_nMixerID, AUDIO_SETINFO, &stAudioInfo)<0 ) 
  349.     {
  350. #ifdef _DEBUG        
  351. fprintf( stderr, "CAudioOutSolaris: just trying to reset EOFn");
  352. #endif        
  353.     }
  354.     m_wLastError = retCode;
  355.     return m_wLastError;
  356. }
  357. //Device specific method to get/set the devices current volume.
  358. UINT16 CAudioOutSolaris::_GetVolume() const
  359. {
  360.     UINT16     unVolume = 0;
  361.     audio_info stAudioInfo;
  362.     
  363.     if (::ioctl( m_nMixerID, AUDIO_GETINFO, &stAudioInfo) < 0)
  364.     {
  365.         //What to do here? Should we close the mixer and pretend
  366.         //we don't have one anymore?
  367.         unVolume = m_uCurVolume;
  368.     }
  369.     else
  370.     {
  371.         //Map device specific volume levels to [0,100]. 
  372.         unVolume = (UINT16) (((float)(stAudioInfo.play.gain-AUDIO_MIN_GAIN))*100.0 / (float)(AUDIO_MAX_GAIN-AUDIO_MIN_GAIN) + 0.5);
  373.         
  374.     }
  375.     return unVolume;
  376. }
  377. HX_RESULT CAudioOutSolaris::_SetVolume(UINT16 unVolume)
  378. {
  379.     HX_RESULT  retCode = RA_AOE_NOERR;
  380.     audio_info stAudioInfo;
  381.     //Fill in the structure with current settings.
  382.     if (::ioctl(m_nMixerID, AUDIO_GETINFO, &stAudioInfo) < 0) 
  383.     {
  384. #ifdef _DEBUG        
  385. fprintf( stderr, "CAudioOutSolaris: Can't get the audio configurationn");
  386. #endif        
  387. retCode = RA_AOE_NOTSUPPORTED;
  388.     }
  389.     //Just change the gain setting. Must map incoming [0,100]-->[0,255]
  390.     stAudioInfo.play.gain = (UINT16)((float)unVolume/100.0 *(float)(AUDIO_MAX_GAIN-AUDIO_MIN_GAIN) + AUDIO_MIN_GAIN + 0.5);
  391.     //Now store the new settings.
  392.     if( ::ioctl(m_nMixerID, AUDIO_SETINFO, &stAudioInfo)<0 ) 
  393.     {
  394. #ifdef _DEBUG        
  395. fprintf( stderr, "CAudioOutSolaris: Can't set the audio configuration.n");
  396. #endif        
  397.         retCode = RA_AOE_NOTSUPPORTED;
  398.     }
  399.     
  400.     m_wLastError = retCode;
  401.     return m_wLastError;
  402. }
  403. //Device specific method to drain a device. This should play the remaining
  404. //bytes in the devices buffer and then return.
  405. HX_RESULT CAudioOutSolaris::_Drain()
  406. {
  407.     HX_RESULT retCode = RA_AOE_NOERR;
  408.     
  409.     if ( m_nDevID < 0 )
  410.     {
  411.         retCode = RA_AOE_DEVNOTOPEN;
  412.     }
  413.     //XXXgfw We don't use _Drain anywhere right now. The problem
  414.     //XXXgfw is that drain will block the proc until the buffers
  415.     //XXXgfw are all drained. We don't want that. Just let it
  416.     //XXXgfw play.
  417. //      else if( ::ioctl (m_nDevID, AUDIO_DRAIN, 0) == -1 )
  418. //      {
  419. //          retCode = RA_AOE_GENERAL;
  420. //      }
  421.     m_wLastError = retCode;
  422.     return m_wLastError;
  423. }
  424. UINT64 CAudioOutSolaris::_GetBytesActualyPlayed(void) const
  425. {
  426.     UINT64    ulTheAnswer = 0;
  427.     audio_info stAudioInfo;
  428.     //Fill in the structure with current settings.
  429.     AUDIO_INITINFO( &stAudioInfo );
  430.     if (::ioctl(m_nDevID, AUDIO_GETINFO, &stAudioInfo) < 0) 
  431.     {
  432. #ifdef _DEBUG        
  433. fprintf( stderr, "CAudioOutSolaris::_GetBytesActualyPlayed: Failure.n");
  434. #endif
  435.         HX_ASSERT( "_GetBytesActualyPlayed: Error reading audio device"==NULL );
  436.         ulTheAnswer = 0;
  437.     }
  438.     if( stAudioInfo.play.error )
  439.     {
  440.         //Underflow has occured.
  441.         HX_ASSERT( "_GetBytesActualyPlayed: underflow in audio device"==NULL);
  442.     }
  443.     
  444.     //We *always* right full blocks to the device. Well, in the
  445.     //case of solaris, we don't use the rewind mechinism of CAudioOutUNIX 
  446.     //so this is true.
  447.     ulTheAnswer = stAudioInfo.play.eof*m_wBlockSize/m_nBlockDivisions;
  448.     //XXXgfw It would be so cool of we could count on the samples field.
  449. //    ulTheAnswer = (ULONG32)stAudioInfo.play.samples*m_unNumChannels*m_uSampFrameSize;
  450. //        fprintf( stderr, "m_ulTotalWritten: %lu  Bytesplayed:%lu   eof: %d  samples: %d n",
  451. //                 m_ulTotalWritten, ulTheAnswer, stAudioInfo.play.eof, stAudioInfo.play.samples );
  452.     
  453.     return  ulTheAnswer; 
  454. }
  455. HX_RESULT CAudioOutSolaris::_GetRoomOnDevice(ULONG32& ulBytes) const 
  456. {
  457.     HX_RESULT      retCode = RA_AOE_NOERR;
  458.     
  459.     ulBytes = m_ulDeviceBufferSize-(m_ulTotalWritten-_GetBytesActualyPlayed());
  460.     m_wLastError = retCode;
  461.     return m_wLastError;
  462. }
  463. HX_RESULT CAudioOutSolaris::_CheckFormat( const HXAudioFormat* pFormat )
  464. {
  465.     m_wLastError = RA_AOE_NOERR;
  466.     return m_wLastError;
  467. }
  468. HX_RESULT CAudioOutSolaris::_CheckSampleRate( ULONG32 ulSampleRate ) 
  469. {
  470.     m_wLastError = RA_AOE_NOERR;
  471.     //XXXgfw For now, just assume that it is supported.
  472.     
  473.     return m_wLastError;
  474. }
  475. HX_RESULT CAudioOutSolaris::_Pause() 
  476. {
  477.     audio_info_t stAudioInfo;
  478.     m_wLastError = HXR_OK;
  479.     AUDIO_INITINFO(&stAudioInfo);
  480.     stAudioInfo.play.pause = 1;
  481.     if( ::ioctl(m_nDevID, AUDIO_SETINFO, &stAudioInfo) < 0 )
  482.     {
  483.         m_wLastError = RA_AOE_NOTSUPPORTED;
  484.     }
  485.     return m_wLastError;
  486. }
  487. HX_RESULT CAudioOutSolaris::_Resume()
  488. {
  489.     audio_info_t stAudioInfo;
  490.     m_wLastError = HXR_OK;
  491.     AUDIO_INITINFO(&stAudioInfo);
  492.     stAudioInfo.play.pause = 0;
  493.     if( ::ioctl(m_nDevID, AUDIO_SETINFO, &stAudioInfo) < 0 ) 
  494.     {
  495.         m_wLastError = RA_AOE_NOTSUPPORTED;
  496.     }
  497.     return m_wLastError;
  498. }
  499. BOOL CAudioOutSolaris::_IsSelectable() const
  500. {
  501.     return FALSE;
  502. }
  503. BOOL CAudioOutSolaris::_HardwarePauseSupported() const
  504. {
  505.     return TRUE;
  506. }
  507.