+/////////////////////////////////////////////////////////////////////////////
+// Name: sound.cpp
+// Purpose: wxSound
+// Author: Marcel Rasche, Vaclav Slavik
+// Modified by:
+// Created: 25/10/98
+// RCS-ID: $Id$
+// Copyright: (c) Julian Smart, Vaclav Slavik
+// Licence: wxWindows licence
+/////////////////////////////////////////////////////////////////////////////
+
+#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
+#pragma implementation "sound.h"
+#pragma implementation "soundbase.h"
+#endif
+
+// for compilers that support precompilation, includes "wx.h".
+#include "wx/wxprec.h"
+
+#include "wx/setup.h"
+
+#if defined(__BORLANDC__)
+#pragma hdrstop
+#endif
+
+#if wxUSE_WAVE
+
+#include <stdio.h>
+#include <unistd.h>
+#include <fcntl.h>
+#include <sys/ioctl.h>
+
+#if HAVE_SYS_SOUNDCARD_H
+#include <sys/soundcard.h>
+#endif
+
+#ifndef WX_PRECOMP
+ #include "wx/event.h"
+ #include "wx/intl.h"
+ #include "wx/log.h"
+#endif
+
+#include "wx/thread.h"
+#include "wx/file.h"
+#include "wx/module.h"
+#include "wx/sound.h"
+#include "wx/dynlib.h"
+
+// ----------------------------------------------------------------------------
+// wxSoundBackendNull, used in absence of audio API or card
+// ----------------------------------------------------------------------------
+
+class wxSoundBackendNull : public wxSoundBackend
+{
+public:
+ wxString GetName() const { return _("No sound"); }
+ int GetPriority() const { return 0; }
+ bool IsAvailable() const { return true; }
+ bool HasNativeAsyncPlayback() const { return true; }
+ bool Play(wxSoundData *WXUNUSED(data), unsigned WXUNUSED(flags))
+ { return true; }
+};
+
+
+// ----------------------------------------------------------------------------
+// wxSoundBackendOSS, for Linux
+// ----------------------------------------------------------------------------
+
+#ifdef HAVE_SYS_SOUNDCARD_H
+
+#ifndef AUDIODEV
+#define AUDIODEV "/dev/dsp" // Default path for audio device
+#endif
+
+class wxSoundBackendOSS : public wxSoundBackend
+{
+public:
+ wxString GetName() const { return _T("Open Sound System"); }
+ int GetPriority() const { return 10; }
+ bool IsAvailable() const;
+ bool HasNativeAsyncPlayback() const { return false; }
+ bool Play(wxSoundData *data, unsigned flags);
+
+private:
+ int OpenDSP(const wxSoundData *data);
+ bool InitDSP(int dev, int iDataBits, int iChannel,
+ unsigned long ulSamplingRate);
+
+ int m_DSPblkSize; // Size of the DSP buffer
+};
+
+bool wxSoundBackendOSS::IsAvailable() const
+{
+ int fd;
+ fd = open(AUDIODEV, O_WRONLY | O_NONBLOCK);
+ if (fd < 0)
+ return false;
+ close(fd);
+ return true;
+}
+
+bool wxSoundBackendOSS::Play(wxSoundData *data, unsigned flags)
+{
+ int dev = OpenDSP(data);
+
+ if (dev < 0)
+ return false;
+
+ ioctl(dev, SNDCTL_DSP_SYNC, 0);
+
+ do
+ {
+ bool play = true;
+ int i;
+ unsigned l = 0;
+ size_t datasize = data->m_dataBytes;
+
+ do
+ {
+ i= (int)((l + m_DSPblkSize) < datasize ?
+ m_DSPblkSize : (datasize - l));
+ if (write(dev, &data->m_data[l], i) != i)
+ {
+ play = false;
+ }
+ l += i;
+ } while (play && l < datasize);
+ } while (flags & wxSOUND_LOOP);
+
+ close(dev);
+ return true;
+}
+
+int wxSoundBackendOSS::OpenDSP(const wxSoundData *data)
+{
+ int dev = -1;
+
+ if ((dev = open(AUDIODEV, O_WRONLY, 0)) <0)
+ return -1;
+
+ if (!InitDSP(dev,
+ (int)data->m_bitsPerSample,
+ data->m_channels == 1 ? 0 : 1,
+ data->m_samplingRate))
+ {
+ close(dev);
+ return -1;
+ }
+
+ return dev;
+}
+
+bool wxSoundBackendOSS::InitDSP(int dev, int iDataBits, int iChannel,
+ unsigned long ulSamplingRate)
+{
+ if (ioctl(dev, SNDCTL_DSP_GETBLKSIZE, &m_DSPblkSize) < 0)
+ return false;
+ if (m_DSPblkSize < 4096 || m_DSPblkSize > 65536)
+ return false;
+ if (ioctl(dev, SNDCTL_DSP_SAMPLESIZE, &iDataBits) < 0)
+ return false;
+ if (ioctl(dev, SNDCTL_DSP_STEREO, &iChannel) < 0)
+ return false;
+ if (ioctl(dev, SNDCTL_DSP_SPEED, &ulSamplingRate) < 0)
+ return false;
+ return true;
+}
+
+#endif // HAVE_SYS_SOUNDCARD_H
+
+
+// ----------------------------------------------------------------------------
+// wxSoundData
+// ----------------------------------------------------------------------------
+
+void wxSoundData::IncRef()
+{
+ m_refCnt++;
+}
+
+void wxSoundData::DecRef()
+{
+ if (--m_refCnt == 0)
+ delete this;
+}
+
+wxSoundData::~wxSoundData()
+{
+ delete[] m_dataWithHeader;
+}
+
+
+// ----------------------------------------------------------------------------
+// wxSoundAsyncPlaybackThread
+// ----------------------------------------------------------------------------
+
+#if wxUSE_THREADS
+
+// mutex for all wxSound's synchronization
+static wxMutex gs_soundMutex;
+
+// this class manages asynchronous playback of audio if the backend doesn't
+// support it natively (e.g. OSS backend)
+class wxSoundAsyncPlaybackThread : public wxThread
+{
+public:
+ wxSoundAsyncPlaybackThread(wxSoundBackend *backend,
+ wxSoundData *data, unsigned flags)
+ : wxThread(), m_backend(backend), m_data(data), m_flags(flags) {}
+ virtual ExitCode Entry()
+ {
+ m_backend->Play(m_data, m_flags & ~wxSOUND_ASYNC);
+ wxMutexLocker locker(gs_soundMutex);
+ m_data->DecRef();
+ wxLogTrace(_T("sound"), _T("terminated async playback thread"));
+ return 0;
+ }
+
+protected:
+ wxSoundBackend *m_backend;
+ wxSoundData *m_data;
+ unsigned m_flags;
+};
+
+#endif // wxUSE_THREADS
+
+// ----------------------------------------------------------------------------
+// wxSound
+// ----------------------------------------------------------------------------
+
+wxSoundBackend *wxSound::ms_backend = NULL;
+
+// FIXME - temporary, until we have plugins architecture
+#if wxUSE_LIBSDL
+ #if wxUSE_PLUGINS
+ wxDynamicLibrary *wxSound::ms_backendSDL = NULL;
+ #else
+ extern "C" wxSoundBackend *wxCreateSoundBackendSDL();
+ #endif
+#endif
+
+wxSound::wxSound() : m_data(NULL)
+{
+}
+
+wxSound::wxSound(const wxString& sFileName, bool isResource) : m_data(NULL)
+{
+ Create(sFileName, isResource);
+}
+
+wxSound::wxSound(int size, const wxByte* data) : m_data(NULL)
+{
+ Create(size, data);
+}
+
+wxSound::~wxSound()
+{
+ Free();
+}
+
+bool wxSound::Create(const wxString& fileName, bool isResource)
+{
+ wxASSERT_MSG( !isResource,
+ _T("Loading sound from resources is only supported on Windows") );
+
+ Free();
+
+ wxFile fileWave;
+ if (!fileWave.Open(fileName, wxFile::read))
+ {
+ return false;
+ }
+
+ size_t len = fileWave.Length();
+ wxUint8 *data = new wxUint8[len];
+ if (fileWave.Read(data, len) != len)
+ {
+ wxLogError(_("Couldn't load sound data from '%s'."), fileName.c_str());
+ return false;
+ }
+
+ if (!LoadWAV(data, len, false))
+ {
+ wxLogError(_("Sound file '%s' is in unsupported format."),
+ fileName.c_str());
+ return false;
+ }
+
+ return true;
+}
+
+bool wxSound::Create(int size, const wxByte* data)
+{
+ wxASSERT( data != NULL );
+
+ Free();
+ if (!LoadWAV(data, size, true))
+ {
+ wxLogError(_("Sound data are in unsupported format."));
+ return false;
+ }
+ return true;
+}
+
+/*static*/ void wxSound::EnsureBackend()
+{
+ if (!ms_backend)
+ {
+ // FIXME -- make this fully dynamic when plugins architecture is in
+ // place
+#ifdef HAVE_SYS_SOUNDCARD_H
+ ms_backend = new wxSoundBackendOSS();
+ if (!ms_backend->IsAvailable())
+ {
+ wxDELETE(ms_backend);
+ }
+#endif
+
+#if wxUSE_LIBSDL
+ if (!ms_backend)
+ {
+#if !wxUSE_PLUGINS
+ ms_backend = wxCreateSoundBackendSDL();
+#else
+ wxString dllname;
+ dllname.Printf(_T("%s/%s"),
+ wxDynamicLibrary::GetPluginsDirectory().c_str(),
+ wxDynamicLibrary::CanonicalizePluginName(
+ _T("sound_sdl"), wxDL_PLUGIN_BASE).c_str());
+ wxLogTrace(_T("sound"),
+ _T("trying to load SDL plugin from '%s'..."),
+ dllname.c_str());
+ wxLogNull null;
+ ms_backendSDL = new wxDynamicLibrary(dllname, wxDL_NOW);
+ if (!ms_backendSDL->IsLoaded())
+ {
+ wxDELETE(ms_backendSDL);
+ }
+ else
+ {
+ typedef wxSoundBackend *(*wxCreateSoundBackend_t)();
+ wxDYNLIB_FUNCTION(wxCreateSoundBackend_t,
+ wxCreateSoundBackendSDL, *ms_backendSDL);
+ if (pfnwxCreateSoundBackendSDL)
+ {
+ ms_backend = (*pfnwxCreateSoundBackendSDL)();
+ }
+ }
+#endif
+ if (ms_backend && !ms_backend->IsAvailable())
+ {
+ wxDELETE(ms_backend);
+ }
+ }
+#endif
+
+ if (!ms_backend)
+ ms_backend = new wxSoundBackendNull();
+
+ wxLogTrace(_T("sound"),
+ _T("using backend '%s'"), ms_backend->GetName().c_str());
+ }
+}
+
+/*static*/ void wxSound::UnloadBackend()
+{
+ if (ms_backend)
+ {
+ wxLogTrace(_T("sound"), _T("unloading backend"));
+ delete ms_backend;
+ ms_backend = NULL;
+#if wxUSE_LIBSDL && wxUSE_PLUGINS
+ delete ms_backendSDL;
+#endif
+ }
+}
+
+bool wxSound::DoPlay(unsigned flags)
+{
+ wxCHECK_MSG( IsOk(), false, _T("Attempt to play invalid wave data") );
+
+ EnsureBackend();
+
+ if ((flags & wxSOUND_ASYNC) && !ms_backend->HasNativeAsyncPlayback())
+ {
+#if wxUSE_THREADS
+ wxMutexLocker locker(gs_soundMutex);
+ m_data->IncRef();
+ wxThread *th = new wxSoundAsyncPlaybackThread(ms_backend, m_data, flags);
+ th->Create();
+ th->Run();
+ wxLogTrace(_T("sound"), _T("launched async playback thread"));
+#else
+ wxLogError(_("Unable to play sound asynchronously."));
+ return false;
+#endif
+ }
+ else
+ {
+ ms_backend->Play(m_data, flags);
+ }
+ return true;
+}
+
+void wxSound::Free()
+{
+#if wxUSE_THREADS
+ wxMutexLocker locker(gs_soundMutex);
+#endif
+ if (m_data)
+ m_data->DecRef();
+}
+
+typedef struct
+{
+ wxUint32 uiSize;
+ wxUint16 uiFormatTag;
+ wxUint16 uiChannels;
+ wxUint32 ulSamplesPerSec;
+ wxUint32 ulAvgBytesPerSec;
+ wxUint16 uiBlockAlign;
+ wxUint16 uiBitsPerSample;
+} WAVEFORMAT;
+
+#define MONO 1 // and stereo is 2 by wav format
+#define WAVE_FORMAT_PCM 1
+#define WAVE_INDEX 8
+#define FMT_INDEX 12
+
+bool wxSound::LoadWAV(const wxUint8 *data, size_t length, bool copyData)
+{
+ WAVEFORMAT waveformat;
+ wxUint32 ul;
+
+ if (length < 32 + sizeof(WAVEFORMAT))
+ return false;
+
+ memcpy(&waveformat, &data[FMT_INDEX + 4], sizeof(WAVEFORMAT));
+ waveformat.uiSize = wxUINT32_SWAP_ON_BE(waveformat.uiSize);
+ waveformat.uiFormatTag = wxUINT16_SWAP_ON_BE(waveformat.uiFormatTag);
+ waveformat.uiChannels = wxUINT16_SWAP_ON_BE(waveformat.uiChannels);
+ waveformat.ulSamplesPerSec = wxUINT32_SWAP_ON_BE(waveformat.ulSamplesPerSec);
+ waveformat.ulAvgBytesPerSec = wxUINT32_SWAP_ON_BE(waveformat.ulAvgBytesPerSec);
+ waveformat.uiBlockAlign = wxUINT16_SWAP_ON_BE(waveformat.uiBlockAlign);
+ waveformat.uiBitsPerSample = wxUINT16_SWAP_ON_BE(waveformat.uiBitsPerSample);
+
+ if (memcmp(data, "RIFF", 4) != 0)
+ return false;
+ if (memcmp(&data[WAVE_INDEX], "WAVE", 4) != 0)
+ return false;
+ if (memcmp(&data[FMT_INDEX], "fmt ", 4) != 0)
+ return false;
+ if (memcmp(&data[FMT_INDEX + waveformat.uiSize + 8], "data", 4) != 0)
+ return false;
+ memcpy(&ul,&data[FMT_INDEX + waveformat.uiSize + 12], 4);
+ ul = wxUINT32_SWAP_ON_BE(ul);
+
+ //WAS: if (ul + FMT_INDEX + waveformat.uiSize + 16 != length)
+ if (ul + FMT_INDEX + waveformat.uiSize + 16 > length)
+ return false;
+
+ if (waveformat.uiFormatTag != WAVE_FORMAT_PCM)
+ return false;
+
+ if (waveformat.ulSamplesPerSec !=
+ waveformat.ulAvgBytesPerSec / waveformat.uiBlockAlign)
+ return false;
+
+ m_data = new wxSoundData;
+ m_data->m_channels = waveformat.uiChannels;
+ m_data->m_samplingRate = waveformat.ulSamplesPerSec;
+ m_data->m_bitsPerSample = waveformat.uiBitsPerSample;
+ m_data->m_samples = ul / (m_data->m_channels * m_data->m_bitsPerSample / 8);
+ m_data->m_dataBytes = ul;
+
+ if (copyData)
+ {
+ m_data->m_dataWithHeader = new wxUint8[length];
+ memcpy(m_data->m_dataWithHeader, data, length);
+ }
+ else
+ m_data->m_dataWithHeader = (wxUint8*)data;
+
+ m_data->m_data =
+ (&m_data->m_dataWithHeader[FMT_INDEX + waveformat.uiSize + 8]);
+
+ return true;
+}
+
+
+// ----------------------------------------------------------------------------
+// wxSoundCleanupModule
+// ----------------------------------------------------------------------------
+
+class wxSoundCleanupModule: public wxModule
+{
+public:
+ bool OnInit() { return true; }
+ void OnExit() { wxSound::UnloadBackend(); }
+ DECLARE_DYNAMIC_CLASS(wxSoundCleanupModule)
+};
+
+IMPLEMENT_DYNAMIC_CLASS(wxSoundCleanupModule, wxModule)
+
+#endif