* Recoded converters (in a cleaner way) git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@5980 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
		
			
				
	
	
		
			340 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			340 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // --------------------------------------------------------------------------
 | |
| // Name: sndoss.cpp
 | |
| // Purpose:
 | |
| // Date: 08/11/1999
 | |
| // Author: Guilhem Lavaux <lavaux@easynet.fr> (C) 1999, 2000
 | |
| // CVSID: $Id$
 | |
| // --------------------------------------------------------------------------
 | |
| #ifdef __GNUG__
 | |
| #pragma implementation "sndoss.cpp"
 | |
| #endif
 | |
| 
 | |
| #include <sys/soundcard.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <wx/defs.h>
 | |
| #include <wx/string.h>
 | |
| #include "sndbase.h"
 | |
| #include "sndoss.h"
 | |
| #include "sndpcm.h"
 | |
| #ifdef __WXGTK__
 | |
| #include <gdk/gdk.h>
 | |
| #endif
 | |
| 
 | |
| wxSoundStreamOSS::wxSoundStreamOSS(const wxString& dev_name)
 | |
| {
 | |
|   wxSoundFormatPcm pcm_default;
 | |
| 
 | |
|   m_fd = open(dev_name.mb_str(), O_RDWR);
 | |
| 
 | |
|   if (m_fd == -1) {
 | |
|     m_snderror = wxSOUND_INVDEV;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   m_devname = dev_name;
 | |
|  
 | |
|   wxSoundStreamOSS::SetSoundFormat(pcm_default);
 | |
| 
 | |
|   ioctl(m_fd, SNDCTL_DSP_GETBLKSIZE, &m_bufsize);
 | |
| 
 | |
|   m_snderror = wxSOUND_NOERR;
 | |
| 
 | |
|   close(m_fd);
 | |
| 
 | |
|   m_oss_stop = TRUE;
 | |
|   m_q_filled = TRUE;
 | |
| }
 | |
| 
 | |
| wxSoundStreamOSS::~wxSoundStreamOSS()
 | |
| {
 | |
|   if (m_fd > 0)
 | |
|     close(m_fd);
 | |
| }
 | |
| 
 | |
| wxUint32 wxSoundStreamOSS::GetBestSize() const
 | |
| {
 | |
|   return m_bufsize;
 | |
| }
 | |
| 
 | |
| wxSoundStream& wxSoundStreamOSS::Read(void *buffer, wxUint32 len)
 | |
| {
 | |
|   int ret;
 | |
| 
 | |
|   m_lastcount = (wxUint32)ret = read(m_fd, buffer, len);
 | |
|   m_q_filled  = TRUE;
 | |
| 
 | |
|   if (ret < 0)
 | |
|     m_snderror = wxSOUND_IOERR;
 | |
|   else
 | |
|     m_snderror = wxSOUND_NOERR;
 | |
| 
 | |
|   return *this;
 | |
| }
 | |
| 
 | |
| wxSoundStream& wxSoundStreamOSS::Write(const void *buffer, wxUint32 len)
 | |
| {
 | |
|   int ret;
 | |
| 
 | |
|   m_lastcount = (wxUint32)ret = write(m_fd, buffer, len);
 | |
|   m_q_filled  = TRUE;
 | |
| 
 | |
|   if (ret < 0)
 | |
|     m_snderror = wxSOUND_IOERR;
 | |
|   else
 | |
|     m_snderror = wxSOUND_NOERR;
 | |
| 
 | |
|   return *this;
 | |
| }
 | |
| 
 | |
| bool wxSoundStreamOSS::SetSoundFormat(const wxSoundFormatBase& format)
 | |
| {
 | |
|   int tmp;
 | |
|   wxSoundFormatPcm *pcm_format;
 | |
| 
 | |
|   if (format.GetType() != wxSOUND_PCM) {
 | |
|     m_snderror = wxSOUND_INVFRMT;
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (m_fd == -1) {
 | |
|     m_snderror = wxSOUND_INVDEV;
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (m_sndformat)
 | |
|     delete m_sndformat;
 | |
| 
 | |
|   m_sndformat = format.Clone();
 | |
|   if (!m_sndformat) {
 | |
|     m_snderror = wxSOUND_MEMERR;
 | |
|     return FALSE;
 | |
|   }
 | |
|   pcm_format = (wxSoundFormatPcm *)m_sndformat;
 | |
| 
 | |
|   // Set the sample rate field.
 | |
|   tmp = pcm_format->GetSampleRate();
 | |
|   ioctl(m_fd, SNDCTL_DSP_SPEED, &tmp);
 | |
| 
 | |
|   pcm_format->SetSampleRate(tmp);
 | |
| 
 | |
|   // Detect the best format
 | |
|   DetectBest(pcm_format);
 | |
|   SetupFormat(pcm_format);
 | |
| 
 | |
|   tmp = pcm_format->GetChannels();
 | |
|   ioctl(m_fd, SNDCTL_DSP_CHANNELS, &tmp);
 | |
|   pcm_format->SetChannels(tmp);
 | |
| 
 | |
|   m_snderror = wxSOUND_NOERR;
 | |
|   if (*pcm_format != format) {
 | |
|     m_snderror = wxSOUND_NOEXACT;
 | |
|     return FALSE;
 | |
|   }
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool wxSoundStreamOSS::SetupFormat(wxSoundFormatPcm *pcm_format)
 | |
| {
 | |
|   int tmp;
 | |
| 
 | |
|   switch(pcm_format->GetBPS()) {
 | |
|   case 8:
 | |
|     if (pcm_format->Signed())
 | |
|       tmp = AFMT_S8;
 | |
|     else
 | |
|       tmp = AFMT_U8;
 | |
|     break;
 | |
|   case 16:
 | |
|     switch (pcm_format->GetOrder()) {
 | |
|     case wxBIG_ENDIAN:
 | |
|       if (pcm_format->Signed())
 | |
|         tmp = AFMT_S16_BE;
 | |
|       else
 | |
|         tmp = AFMT_U16_BE;
 | |
|       break;
 | |
|     case wxLITTLE_ENDIAN:
 | |
|       if (pcm_format->Signed())
 | |
|         tmp = AFMT_S16_LE;
 | |
|       else
 | |
|         tmp = AFMT_U16_LE;
 | |
|       break;
 | |
|     }
 | |
|     break;
 | |
|   }
 | |
| 
 | |
|   ioctl(m_fd, SNDCTL_DSP_SETFMT, &tmp);
 | |
| 
 | |
|   // Demangling.
 | |
|   switch (tmp) {
 | |
|   case AFMT_U8:
 | |
|     pcm_format->SetBPS(8);
 | |
|     pcm_format->Signed(FALSE);
 | |
|     break;
 | |
|   case AFMT_S8:
 | |
|     pcm_format->SetBPS(8);
 | |
|     pcm_format->Signed(TRUE);
 | |
|     break;
 | |
|   case AFMT_U16_LE:
 | |
|     pcm_format->SetBPS(16);
 | |
|     pcm_format->Signed(FALSE);
 | |
|     pcm_format->SetOrder(wxLITTLE_ENDIAN);
 | |
|     break;
 | |
|   case AFMT_U16_BE:
 | |
|     pcm_format->SetBPS(16);
 | |
|     pcm_format->Signed(FALSE);
 | |
|     pcm_format->SetOrder(wxBIG_ENDIAN);
 | |
|     break;
 | |
|   case AFMT_S16_LE:
 | |
|     pcm_format->SetBPS(16);
 | |
|     pcm_format->Signed(TRUE);
 | |
|     pcm_format->SetOrder(wxLITTLE_ENDIAN);
 | |
|     break;
 | |
|   case AFMT_S16_BE:
 | |
|     pcm_format->SetBPS(16);
 | |
|     pcm_format->Signed(TRUE);
 | |
|     pcm_format->SetOrder(wxBIG_ENDIAN);
 | |
|     break;
 | |
|   }
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| #ifdef __WXGTK__
 | |
| static void _wxSound_OSS_CBack(gpointer data, int source,
 | |
|                                GdkInputCondition condition)
 | |
| {
 | |
|   wxSoundStreamOSS *oss = (wxSoundStreamOSS *)data;
 | |
| 
 | |
|   switch (condition) {
 | |
|   case GDK_INPUT_READ:
 | |
|     oss->WakeUpEvt(wxSOUND_INPUT);
 | |
|     break;
 | |
|   case GDK_INPUT_WRITE:
 | |
|     oss->WakeUpEvt(wxSOUND_OUTPUT);
 | |
|     break;
 | |
|   default:
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void wxSoundStreamOSS::WakeUpEvt(int evt)
 | |
| {
 | |
|   m_q_filled = FALSE;
 | |
|   OnSoundEvent(evt);
 | |
| }
 | |
| 
 | |
| bool wxSoundStreamOSS::StartProduction(int evt)
 | |
| {
 | |
|   wxSoundFormatBase *old_frmt;
 | |
| 
 | |
|   if (!m_oss_stop)
 | |
|     StopProduction();
 | |
| 
 | |
|   old_frmt = m_sndformat->Clone();
 | |
|   if (!old_frmt) {
 | |
|     m_snderror = wxSOUND_MEMERR;
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (evt == wxSOUND_OUTPUT)
 | |
|     m_fd = open(m_devname.mb_str(), O_WRONLY);
 | |
|   else if (evt == wxSOUND_INPUT)
 | |
|     m_fd = open(m_devname.mb_str(), O_RDONLY);
 | |
| 
 | |
|   if (m_fd == -1) {
 | |
|     m_snderror = wxSOUND_INVDEV;
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   SetSoundFormat(*old_frmt);
 | |
|   delete old_frmt;
 | |
| 
 | |
| #ifdef __WXGTK__
 | |
|   int trig;
 | |
| 
 | |
|   if (evt == wxSOUND_OUTPUT) {
 | |
|     m_tag = gdk_input_add(m_fd, GDK_INPUT_WRITE, _wxSound_OSS_CBack, (gpointer)this);
 | |
|     trig = PCM_ENABLE_OUTPUT;
 | |
|   } else {
 | |
|     m_tag = gdk_input_add(m_fd, GDK_INPUT_READ, _wxSound_OSS_CBack, (gpointer)this);
 | |
|     trig = PCM_ENABLE_INPUT;
 | |
|   }
 | |
| #else
 | |
|   while (!m_oss_stop)
 | |
|     OnSoundEvent(evt);
 | |
| #endif
 | |
| 
 | |
|   ioctl(m_fd, SNDCTL_DSP_SETTRIGGER, &trig);
 | |
| 
 | |
|   m_oss_stop = FALSE;
 | |
|   m_q_filled = FALSE;
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool wxSoundStreamOSS::StopProduction()
 | |
| {
 | |
|   if (m_oss_stop)
 | |
|     return FALSE;
 | |
| 
 | |
| #ifdef __WXGTK__
 | |
|   gdk_input_remove(m_tag);
 | |
| #endif
 | |
| 
 | |
|   close(m_fd);
 | |
|   m_oss_stop = TRUE;
 | |
|   m_q_filled = TRUE;
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| bool wxSoundStreamOSS::QueueFilled() const
 | |
| {
 | |
|   return m_q_filled;
 | |
| }
 | |
| 
 | |
| //
 | |
| // Detect the closest format (The best).
 | |
| //
 | |
| void wxSoundStreamOSS::DetectBest(wxSoundFormatPcm *pcm)
 | |
| {
 | |
| #define MASK_16BITS (AFMT_S16_LE | AFMT_S16_BE | AFMT_U16_LE | AFMT_U16_BE)
 | |
| 
 | |
|   int fmt_mask;
 | |
|   wxSoundFormatPcm best_pcm;
 | |
| 
 | |
|   // We change neither the number of channels nor the sample rate
 | |
| 
 | |
|   best_pcm.SetSampleRate(pcm->GetSampleRate());
 | |
|   best_pcm.SetChannels(pcm->GetChannels());
 | |
| 
 | |
|   // Get the supported format by the sound card
 | |
|   ioctl(m_fd, SNDCTL_DSP_GETFMTS, &fmt_mask);
 | |
| 
 | |
|   // It supports 16 bits
 | |
|   if (pcm->GetBPS() == 16 && ((fmt_mask & MASK_16BITS) != 0))
 | |
|     best_pcm.SetBPS(16);
 | |
| 
 | |
|   // It supports big endianness
 | |
|   if (pcm->GetOrder() == wxBIG_ENDIAN && ((fmt_mask & (AFMT_S16_BE | AFMT_U16_BE)) != 0))
 | |
|     best_pcm.SetOrder(wxBIG_ENDIAN);
 | |
| 
 | |
|   // It supports little endianness
 | |
|   if (pcm->GetOrder() == wxLITTLE_ENDIAN && ((fmt_mask & (AFMT_S16_LE | AFMT_U16_LE)) != 0))
 | |
|     best_pcm.SetOrder(wxLITTLE_ENDIAN);
 | |
| 
 | |
|   // It supports signed samples
 | |
|   if (pcm->Signed() && ((fmt_mask & (AFMT_S16_LE | AFMT_S16_BE | AFMT_S8)) != 0))
 | |
|     best_pcm.Signed(TRUE);
 | |
| 
 | |
|   // It supports unsigned samples
 | |
|   if (!pcm->Signed() && ((fmt_mask & (AFMT_U16_LE | AFMT_U16_BE | AFMT_U8)) != 0))
 | |
|     best_pcm.Signed(FALSE);
 | |
| 
 | |
|   // Finally recopy the new format
 | |
|   *pcm = best_pcm;
 | |
| }
 |