1 // --------------------------------------------------------------------------
5 // Author: Guilhem Lavaux <lavaux@easynet.fr> (C) 1999, 2000
7 // --------------------------------------------------------------------------
9 #pragma implementation "sndwin.cpp"
12 #include "wx/wxprec.h"
19 #include "wx/module.h"
20 #include "wx/msw/private.h"
21 #include "wx/string.h"
24 // -------------------------------------------------------------------------
26 // -------------------------------------------------------------------------
28 #include "wx/mmedia/sndbase.h"
29 #include "wx/mmedia/sndwin.h"
30 #include "wx/mmedia/sndpcm.h"
32 // -------------------------------------------------------------------------
34 // -------------------------------------------------------------------------
39 // -------------------------------------------------------------------------
40 // External definitions, forward, ...
41 // -------------------------------------------------------------------------
43 typedef struct _wxSoundInternal wxSoundInternal
;
44 typedef struct _wxSoundInfoHeader wxSoundInfoHeader
;
46 extern const wxChar
*wxCanvasClassName
;
48 wxList
*wxSoundHandleList
= NULL
;
50 static inline wxSoundStreamWin
*wxFindSoundFromHandle(WXHWND hWnd
)
52 wxNode
*node
= wxSoundHandleList
->Find((long)hWnd
);
55 return (wxSoundStreamWin
*)node
->Data();
58 struct _wxSoundInternal
{
62 bool m_output_enabled
, m_input_enabled
;
65 struct _wxSoundInfoHeader
{
66 HGLOBAL m_h_header
, m_h_data
;
70 bool m_playing
, m_recording
;
71 wxUint32 m_position
, m_size
;
73 wxSoundStreamWin
*m_driver
;
76 #define WXSOUND_MAX_QUEUE 10
78 wxSoundStreamWin::wxSoundStreamWin()
82 m_production_started
= FALSE
;
83 m_internal
= new wxSoundInternal
;
85 m_snderror
= wxSOUND_MEMERROR
;
89 m_snderror
= wxSOUND_NOERROR
;
95 m_internal
->m_input_enabled
= FALSE
;
96 m_internal
->m_output_enabled
= FALSE
;
98 m_waiting_for
= FALSE
;
100 if (!OpenDevice(wxSOUND_OUTPUT
))
106 wxSoundStreamWin::~wxSoundStreamWin()
109 if (m_production_started
)
117 // -----------------------------------------------------------------------
118 // _wxSoundHandlerWndProc: Window callback to handle buffer completion
119 // -----------------------------------------------------------------------
120 LRESULT APIENTRY _EXPORT
_wxSoundHandlerWndProc(HWND hWnd
, UINT message
,
121 WPARAM wParam
, LPARAM lParam
)
123 wxSoundStreamWin
*sndwin
;
125 sndwin
= wxFindSoundFromHandle((WXHWND
)hWnd
);
131 sndwin
->NotifyDoneBuffer(wParam
, wxSOUND_OUTPUT
);
134 sndwin
->NotifyDoneBuffer(wParam
, wxSOUND_INPUT
);
142 // -----------------------------------------------------------------------
143 // CreateSndWindow() creates an hidden window which will receive the sound
145 // -----------------------------------------------------------------------
147 void wxSoundStreamWin::CreateSndWindow()
149 FARPROC proc
= MakeProcInstance((FARPROC
)_wxSoundHandlerWndProc
,
153 m_internal
->m_sndWin
= ::CreateWindow(wxCanvasClassName
, NULL
, 0,
154 0, 0, 0, 0, NULL
, (HMENU
) NULL
,
155 wxGetInstance(), NULL
);
157 error
= GetLastError();
159 ::SetWindowLong(m_internal
->m_sndWin
, GWL_WNDPROC
, (LONG
)proc
);
161 // Add this window to the sound handle list so we'll be able to redecode
162 // the "magic" number.
163 wxSoundHandleList
->Append((long)m_internal
->m_sndWin
, (wxObject
*)this);
166 // -----------------------------------------------------------------------
167 // DestroySndWindow() destroys the hidden window
168 // -----------------------------------------------------------------------
170 void wxSoundStreamWin::DestroySndWindow()
172 if (m_internal
->m_sndWin
) {
173 ::DestroyWindow(m_internal
->m_sndWin
);
174 wxSoundHandleList
->DeleteObject((wxObject
*)this);
178 // -------------------------------------------------------------------------
179 // OpenDevice(int mode) initializes the windows driver for a "mode"
180 // operation. mode is a bit mask: if the bit "wxSOUND_OUTPUT" is set,
181 // the driver is opened for output operation, and if the bit "wxSOUND_INPUT"
182 // is set, then the driver is opened for input operation. The two modes
184 // The initialization parameters (sample rate, ...) are taken from the
185 // m_sndformat object.
186 // At the end, OpenDevice() calls AllocHeaders() to initialize the Sound IO
188 // -------------------------------------------------------------------------
189 bool wxSoundStreamWin::OpenDevice(int mode
)
191 wxSoundFormatPcm
*pcm
;
192 WAVEFORMATEX wformat
;
195 m_snderror
= wxSOUND_INVFRMT
;
199 pcm
= (wxSoundFormatPcm
*)m_sndformat
;
201 wformat
.wFormatTag
= WAVE_FORMAT_PCM
;
202 wformat
.nChannels
= pcm
->GetChannels();
203 wformat
.nBlockAlign
= wformat
.nChannels
* pcm
->GetBPS() / 8;
204 wformat
.nSamplesPerSec
= pcm
->GetSampleRate();
205 wformat
.nAvgBytesPerSec
= wformat
.nSamplesPerSec
* wformat
.nBlockAlign
;
206 wformat
.wBitsPerSample
= pcm
->GetBPS();
209 // -----------------------------------
210 // Open the driver for Output operation
211 // -----------------------------------
212 if (mode
& wxSOUND_OUTPUT
) {
215 result
= waveOutOpen(&m_internal
->m_devout
,
216 WAVE_MAPPER
, &wformat
,
217 (DWORD
)m_internal
->m_sndWin
, 0,
220 if (result
!= MMSYSERR_NOERROR
) {
221 m_snderror
= wxSOUND_INVDEV
;
225 m_output_frag_out
= WXSOUND_MAX_QUEUE
-1;
226 m_current_frag_out
= 0;
228 m_internal
->m_output_enabled
= TRUE
;
230 // -----------------------------------
231 // Open the driver for Input operation
232 // -----------------------------------
233 if (mode
& wxSOUND_INPUT
) {
236 result
= waveInOpen(&m_internal
->m_devin
,
237 WAVE_MAPPER
, &wformat
,
238 (DWORD
)m_internal
->m_sndWin
, 0,
241 if (result
!= MMSYSERR_NOERROR
) {
242 m_snderror
= wxSOUND_INVDEV
;
246 m_current_frag_in
= WXSOUND_MAX_QUEUE
-1;
249 m_internal
->m_input_enabled
= TRUE
;
252 if (mode
& wxSOUND_OUTPUT
) {
253 if (!AllocHeaders(wxSOUND_OUTPUT
)) {
258 if (mode
& wxSOUND_INPUT
) {
259 if (!AllocHeaders(wxSOUND_INPUT
)) {
268 // -------------------------------------------------------------------------
269 // CloseDevice() closes the driver handles and frees memory allocated for
271 // -------------------------------------------------------------------------
272 void wxSoundStreamWin::CloseDevice()
274 if (m_internal
->m_output_enabled
) {
275 FreeHeaders(wxSOUND_OUTPUT
);
276 m_internal
->m_output_enabled
= FALSE
;
277 waveOutClose(m_internal
->m_devout
);
280 if (m_internal
->m_input_enabled
) {
281 FreeHeaders(wxSOUND_INPUT
);
282 m_internal
->m_input_enabled
= FALSE
;
283 waveInClose(m_internal
->m_devin
);
287 // -------------------------------------------------------------------------
288 // AllocHeader(int mode)
290 // mode has the same mean as in OpenDevice() except that here the two flags
291 // must be exclusive.
292 // AllocHeader() initializes an element of an operation (this can be input
293 // or output). It means it allocates the sound header's memory block
294 // and "prepares" it (It is needed by Windows). At the same time, it sets
295 // private datas so we can the header's owner (See callback).
297 // It returns the new allocated block or NULL.
298 // -------------------------------------------------------------------------
299 wxSoundInfoHeader
*wxSoundStreamWin::AllocHeader(int mode
)
301 wxSoundInfoHeader
*info
;
304 // Some memory allocation
305 info
= new wxSoundInfoHeader
;
306 info
->m_h_data
= GlobalAlloc(GMEM_MOVEABLE
| GMEM_SHARE
, GetBestSize());
307 info
->m_h_header
= GlobalAlloc(GMEM_MOVEABLE
| GMEM_SHARE
, sizeof(WAVEHDR
));
308 if (!info
->m_h_data
|| !info
->m_h_header
) {
310 m_snderror
= wxSOUND_MEMERR
;
314 // Get the two pointers from the system
315 info
->m_data
= (char *)GlobalLock(info
->m_h_data
);
316 info
->m_header
= (WAVEHDR
*)GlobalLock(info
->m_h_header
);
317 // Set the header's mode
319 // Set the parent of the header
320 info
->m_driver
= this;
324 header
= info
->m_header
;
325 // Initialize Windows variables
326 header
->lpData
= info
->m_data
;
327 header
->dwBufferLength
= GetBestSize();
328 header
->dwUser
= (DWORD
)info
;
329 header
->dwFlags
= WHDR_DONE
;
331 // "Prepare" the header
332 if (mode
== wxSOUND_INPUT
) {
335 result
= waveInPrepareHeader(m_internal
->m_devin
, header
,
338 if (result
!= MMSYSERR_NOERROR
) {
339 // If something goes wrong, free everything.
340 GlobalUnlock(info
->m_data
);
341 GlobalUnlock(info
->m_header
);
342 GlobalFree(info
->m_h_data
);
343 GlobalFree(info
->m_h_header
);
346 m_snderror
= wxSOUND_IOERROR
;
349 } else if (mode
== wxSOUND_OUTPUT
) {
352 result
= waveOutPrepareHeader(m_internal
->m_devout
, header
,
355 if (result
!= MMSYSERR_NOERROR
) {
356 // If something goes wrong, free everything.
357 GlobalUnlock(info
->m_data
);
358 GlobalUnlock(info
->m_header
);
359 GlobalFree(info
->m_h_data
);
360 GlobalFree(info
->m_h_header
);
363 m_snderror
= wxSOUND_IOERROR
;
370 // -------------------------------------------------------------------------
371 // AllocHeaders(int mode)
373 // "mode" has the same mean as for OpenDevice() except that the two flags must
375 // AllocHeaders() allocates WXSOUND_MAX_QUEUE (= 128) blocks for an operation
376 // queue. It uses AllocHeader() for each element.
378 // Once it has allocated all blocks, it returns TRUE and if an error occured
380 // -------------------------------------------------------------------------
381 bool wxSoundStreamWin::AllocHeaders(int mode
)
384 wxSoundInfoHeader
**headers
;
386 if (mode
== wxSOUND_OUTPUT
)
387 headers
= m_headers_play
= new wxSoundInfoHeader
*[WXSOUND_MAX_QUEUE
];
389 headers
= m_headers_rec
= new wxSoundInfoHeader
*[WXSOUND_MAX_QUEUE
];
391 memset(headers
, 0, WXSOUND_MAX_QUEUE
*sizeof(wxSoundInfoHeader
*));
393 for (i
=0;i
<WXSOUND_MAX_QUEUE
;i
++) {
394 headers
[i
] = AllocHeader(mode
);
403 // -------------------------------------------------------------------------
404 // FreeHeader(int mode)
406 // "mode" has the same mean as for OpenDevice() except that the two flags must
408 // FreeHeader() frees a memory block and "unprepares" it.
409 // -------------------------------------------------------------------------
410 void wxSoundStreamWin::FreeHeader(wxSoundInfoHeader
*header
, int mode
)
412 if (mode
== wxSOUND_OUTPUT
)
413 waveOutUnprepareHeader(m_internal
->m_devout
, header
->m_header
, sizeof(WAVEHDR
));
415 waveInUnprepareHeader(m_internal
->m_devin
, header
->m_header
, sizeof(WAVEHDR
));
417 GlobalUnlock(header
->m_data
);
418 GlobalUnlock(header
->m_header
);
419 GlobalFree(header
->m_h_header
);
420 GlobalFree(header
->m_h_data
);
424 // -------------------------------------------------------------------------
425 // FreeHeaders(int mode)
427 // "mode" has the same mean as for OpenDevice() except that the two flags must
429 // FreeHeaders() frees all an operation queue once it has checked that
430 // all buffers have been terminated.
431 // -------------------------------------------------------------------------
432 void wxSoundStreamWin::FreeHeaders(int mode
)
435 wxSoundInfoHeader
***headers
;
437 if (mode
== wxSOUND_OUTPUT
)
438 headers
= &m_headers_play
;
440 headers
= &m_headers_rec
;
442 for (i
=0;i
<WXSOUND_MAX_QUEUE
;i
++) {
444 // We wait for the end of the buffer
445 WaitFor((*headers
)[i
]);
446 // Then, we free the header
447 FreeHeader((*headers
)[i
], mode
);
454 // -------------------------------------------------------------------------
455 // WaitFor(wxSoundInfoHeader *info)
457 // "info" is one element of an IO queue
458 // WaitFor() checks whether the specified block has been terminated.
459 // If it hasn't been terminated, it waits for its termination.
461 // NB: if it's a partially filled buffer it adds it to the Windows queue
462 // -------------------------------------------------------------------------
463 void wxSoundStreamWin::WaitFor(wxSoundInfoHeader
*info
)
465 // If the buffer is finished, we return immediately
466 if (!info
->m_playing
) {
468 // We begun filling it: we must send it to the Windows queue
469 if (info
->m_position
!= 0) {
470 memset(info
->m_data
+ info
->m_position
, 0, info
->m_size
);
479 m_waiting_for
= TRUE
;
480 // Else, we wait for its termination
481 while (info
->m_playing
|| info
->m_recording
)
483 m_waiting_for
= FALSE
;
486 // -------------------------------------------------------------------------
487 // AddToQueue(wxSoundInfoHeader *info)
489 // For "info", see WaitFor()
490 // AddToQueue() sends the IO queue element to the Windows queue.
492 // Warning: in the current implementation, it partially assume we send the
493 // element in the right order. This is true in that implementation but if
494 // you use it elsewhere, be careful: it may shuffle all your sound datas.
495 // -------------------------------------------------------------------------
496 bool wxSoundStreamWin::AddToQueue(wxSoundInfoHeader
*info
)
500 if (info
->m_mode
== wxSOUND_INPUT
) {
501 // Increment the input fragment pointer
502 result
= waveInAddBuffer(m_internal
->m_devin
,
503 info
->m_header
, sizeof(WAVEHDR
));
504 if (result
== MMSYSERR_NOERROR
)
505 info
->m_recording
= TRUE
;
508 } else if (info
->m_mode
== wxSOUND_OUTPUT
) {
509 result
= waveOutWrite(m_internal
->m_devout
,
510 info
->m_header
, sizeof(WAVEHDR
));
511 if (result
== MMSYSERR_NOERROR
)
512 info
->m_playing
= TRUE
;
519 // -------------------------------------------------------------------------
520 // ClearHeader(wxSoundInfoHeader *info)
522 // ClearHeader() reinitializes the parameters of "info" to their default
524 // -------------------------------------------------------------------------
525 void wxSoundStreamWin::ClearHeader(wxSoundInfoHeader
*info
)
527 info
->m_playing
= FALSE
;
528 info
->m_recording
= FALSE
;
529 info
->m_position
= 0;
530 info
->m_size
= GetBestSize();
533 // -------------------------------------------------------------------------
534 // wxSoundInfoHeader *NextFragmentOutput()
536 // NextFragmentOutput() looks for a free output block. It will always
537 // return you a non-NULL pointer but it may waits for an empty buffer a long
539 // -------------------------------------------------------------------------
540 wxSoundInfoHeader
*wxSoundStreamWin::NextFragmentOutput()
542 if (m_headers_play
[m_current_frag_out
]->m_playing
) {
543 m_current_frag_out
= (m_current_frag_out
+ 1) % WXSOUND_MAX_QUEUE
;
545 if (m_headers_play
[m_current_frag_out
]->m_playing
)
546 WaitFor(m_headers_play
[m_current_frag_out
]);
548 if (m_current_frag_out
== m_output_frag_out
)
549 m_queue_filled
= TRUE
;
550 return m_headers_play
[m_current_frag_out
];
553 // -------------------------------------------------------------------------
554 // The behaviour of Write is documented in the global documentation.
555 // -------------------------------------------------------------------------
556 wxSoundStream
& wxSoundStreamWin::Write(const void *buffer
, wxUint32 len
)
559 if (!m_internal
->m_output_enabled
) {
560 m_snderror
= wxSOUND_NOTSTARTED
;
566 wxSoundInfoHeader
*header
;
569 // Get a new output fragment
570 header
= NextFragmentOutput();
572 to_copy
= (len
> header
->m_size
) ? header
->m_size
: len
;
573 memcpy(header
->m_data
+ header
->m_position
, buffer
, to_copy
);
575 header
->m_position
+= to_copy
;
576 header
->m_size
-= to_copy
;
577 buffer
= (((const char *)buffer
) + to_copy
);
579 m_lastcount
+= to_copy
;
581 // If the fragment is full, we send it to the Windows queue.
582 if (header
->m_size
== 0)
583 if (!AddToQueue(header
)) {
584 m_snderror
= wxSOUND_IOERROR
;
591 // -------------------------------------------------------------------------
592 // NextFragmentInput is not functional.
593 // -------------------------------------------------------------------------
594 wxSoundInfoHeader
*wxSoundStreamWin::NextFragmentInput()
596 wxSoundInfoHeader
*header
;
598 // Queue pointer: reader
599 m_current_frag_in
= (m_current_frag_in
+ 1) % WXSOUND_MAX_QUEUE
;
601 header
= m_headers_rec
[m_current_frag_in
];
602 // If the current buffer is in recording mode, we must wait for its
604 if (header
->m_recording
)
607 // We reached the writer position: the queue is full.
608 if (m_current_frag_in
== m_input_frag_in
)
609 m_queue_filled
= TRUE
;
614 // -------------------------------------------------------------------------
615 // The behaviour of Read is documented in the global documentation.
616 // -------------------------------------------------------------------------
617 wxSoundStream
& wxSoundStreamWin::Read(void *buffer
, wxUint32 len
)
619 wxSoundInfoHeader
*header
;
623 if (!m_internal
->m_input_enabled
)
627 header
= NextFragmentInput();
629 to_copy
= (len
> header
->m_size
) ? header
->m_size
: len
;
630 memcpy(buffer
, header
->m_data
+ header
->m_position
, to_copy
);
632 header
->m_position
+= to_copy
;
633 header
->m_size
-= to_copy
;
634 buffer
= (((char *)buffer
) + to_copy
);
636 m_lastcount
+= to_copy
;
638 if (header
->m_size
== 0) {
640 if (!AddToQueue(header
)) {
641 m_snderror
= wxSOUND_IOERROR
;
649 // -------------------------------------------------------------------------
650 // NotifyDoneBuffer(wxUint32 dev_handle)
652 // NotifyDoneBuffer() is called by wxSoundHandlerProc each time a sound
653 // fragment finished. It reinitializes the parameters of the fragment and
654 // sends an event to the clients.
655 // -------------------------------------------------------------------------
656 void wxSoundStreamWin::NotifyDoneBuffer(wxUint32 dev_handle
, int flag
)
658 wxSoundInfoHeader
*info
;
660 if (flag
== wxSOUND_OUTPUT
) {
661 if (!m_internal
->m_output_enabled
)
664 // Queue pointer: reader
665 m_output_frag_out
= (m_output_frag_out
+ 1) % WXSOUND_MAX_QUEUE
;
666 info
= m_headers_play
[m_output_frag_out
];
667 // Clear header to tell the system the buffer is free now
669 m_queue_filled
= FALSE
;
671 // Try to requeue a new buffer.
672 OnSoundEvent(wxSOUND_OUTPUT
);
674 if (!m_internal
->m_input_enabled
)
677 // Recording completed
678 m_headers_rec
[m_input_frag_in
]->m_recording
= FALSE
;
679 // Queue pointer: writer
680 m_input_frag_in
= (m_input_frag_in
+ 1) % WXSOUND_MAX_QUEUE
;
682 OnSoundEvent(wxSOUND_INPUT
);
683 m_queue_filled
= FALSE
;
687 // -------------------------------------------------------------------------
689 // -------------------------------------------------------------------------
690 bool wxSoundStreamWin::SetSoundFormat(wxSoundFormatBase
& base
)
692 // TODO: detect best format
693 return wxSoundStream::SetSoundFormat(base
);
696 // -------------------------------------------------------------------------
698 // -------------------------------------------------------------------------
699 bool wxSoundStreamWin::StartProduction(int evt
)
704 if ((m_internal
->m_output_enabled
&& (evt
& wxSOUND_OUTPUT
)) ||
705 (m_internal
->m_input_enabled
&& (evt
& wxSOUND_INPUT
)))
708 if (!OpenDevice(evt
))
711 m_production_started
= TRUE
;
712 m_queue_filled
= FALSE
;
713 // Send a dummy event to start.
714 if (evt
& wxSOUND_OUTPUT
)
715 OnSoundEvent(wxSOUND_OUTPUT
);
717 if (evt
& wxSOUND_INPUT
) {
719 for (i
=0;i
<WXSOUND_MAX_QUEUE
;i
++)
720 AddToQueue(m_headers_rec
[i
]);
722 waveInStart(m_internal
->m_devin
);
728 // -------------------------------------------------------------------------
730 // ------------------------------------------------------------------------
731 bool wxSoundStreamWin::StopProduction()
733 if (!m_production_started
) {
734 m_snderror
= wxSOUND_NOTSTARTED
;
738 m_snderror
= wxSOUND_NOERROR
;
739 m_production_started
= FALSE
;
744 // -------------------------------------------------------------------------
746 // -------------------------------------------------------------------------
747 bool wxSoundStreamWin::QueueFilled() const
749 return (!m_production_started
|| m_queue_filled
);
753 // --------------------------------------------------------------------------
755 // --------------------------------------------------------------------------
757 class WXDLLEXPORT wxSoundWinModule
: public wxModule
{
758 DECLARE_DYNAMIC_CLASS(wxSoundWinModule
)
764 IMPLEMENT_DYNAMIC_CLASS(wxSoundWinModule
, wxModule
)
766 bool wxSoundWinModule::OnInit() {
767 wxSoundHandleList
= new wxList(wxKEY_INTEGER
);
771 void wxSoundWinModule::OnExit() {
772 delete wxSoundHandleList
;