11#include "../openttd.h"
14#include "../core/bitmath_func.hpp"
15#include "../core/math_func.hpp"
19#include <versionhelpers.h>
20#include "../os/windows/win32.h"
23#include "../safeguards.h"
27using HeaderDataPair = std::pair<WAVEHDR, std::unique_ptr<CHAR[]>>;
29static HWAVEOUT _waveout;
30static HeaderDataPair _wave_hdr[2];
33static DWORD _threadId;
36static void PrepareHeader(HeaderDataPair &hdr)
38 hdr.second = std::make_unique<CHAR[]>(_bufsize * 4);
39 hdr.first.dwBufferLength = _bufsize * 4;
40 hdr.first.dwFlags = 0;
41 hdr.first.lpData = hdr.second.get();
42 if (waveOutPrepareHeader(_waveout, &hdr.first,
sizeof(WAVEHDR)) != MMSYSERR_NOERROR)
throw "waveOutPrepareHeader failed";
45static DWORD WINAPI SoundThread(LPVOID)
50 for (
auto &hdr : _wave_hdr) {
51 if ((hdr.first.dwFlags & WHDR_INQUEUE) != 0)
continue;
52 MxMixSamples(hdr.first.lpData, hdr.first.dwBufferLength / 4);
53 if (waveOutWrite(_waveout, &hdr.first,
sizeof(WAVEHDR)) != MMSYSERR_NOERROR) {
54 MessageBox(
nullptr, L
"Sounds are disabled until restart.", L
"waveOutWrite failed", MB_ICONINFORMATION);
58 WaitForSingleObject(_event, INFINITE);
59 }
while (_waveout !=
nullptr);
67 wfex.wFormatTag = WAVE_FORMAT_PCM;
69 wfex.wBitsPerSample = 16;
71 wfex.nBlockAlign = (wfex.nChannels * wfex.wBitsPerSample) / 8;
72 wfex.nAvgBytesPerSec = wfex.nSamplesPerSec * wfex.nBlockAlign;
76 _bufsize = std::min<int>(_bufsize, UINT16_MAX);
79 if (
nullptr == (_event = CreateEvent(
nullptr, FALSE, FALSE,
nullptr)))
throw "Failed to create event";
81 if (waveOutOpen(&_waveout, WAVE_MAPPER, &wfex, (DWORD_PTR)_event, 0, CALLBACK_EVENT) != MMSYSERR_NOERROR)
throw "waveOutOpen failed";
83 MxInitialize(wfex.nSamplesPerSec);
85 for (
auto &hdr : _wave_hdr) PrepareHeader(hdr);
87 if (
nullptr == (_thread = CreateThread(
nullptr, 8192, SoundThread, 0, 0, &_threadId)))
throw "Failed to create thread";
88 }
catch (
const char *error) {
98 HWAVEOUT waveout = _waveout;
102 SignalObjectAndWait(_event, _thread, INFINITE, FALSE);
105 waveOutReset(waveout);
106 for (
auto &hdr : _wave_hdr) {
107 waveOutUnprepareHeader(waveout, &hdr.first,
sizeof(WAVEHDR));
110 waveOutClose(waveout);
112 CloseHandle(_thread);
Factory for the sound driver for Windows.
void Stop() override
Stop this driver.
std::optional< std::string_view > Start(const StringList ¶m) override
Start this driver.
int GetDriverParamInt(const StringList &parm, const char *name, int def)
Get an integer parameter the list of parameters.
std::vector< std::string > StringList
Type for a list of strings.
void SetCurrentThreadName(const char *threadName)
Name the thread this function is called on for the debugger.
Base for Windows sound handling.