2015-08-13 02:26:07 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 Andrew Kelley
|
|
|
|
*
|
|
|
|
* This file is part of libsoundio, which is MIT licensed.
|
|
|
|
* See http://opensource.org/licenses/MIT
|
|
|
|
*/
|
|
|
|
|
2015-08-08 22:22:50 +00:00
|
|
|
#include "wasapi.hpp"
|
|
|
|
#include "soundio.hpp"
|
|
|
|
|
2015-08-10 22:22:08 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#define INITGUID
|
|
|
|
#define CINTERFACE
|
|
|
|
#define COBJMACROS
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
2015-08-11 06:18:08 +00:00
|
|
|
#include <windows.h>
|
2015-08-10 22:22:08 +00:00
|
|
|
#include <mmdeviceapi.h>
|
2015-08-11 06:18:08 +00:00
|
|
|
#include <functiondiscoverykeys_devpkey.h>
|
2015-08-12 18:01:06 +00:00
|
|
|
#include <mmreg.h>
|
2015-08-12 22:32:44 +00:00
|
|
|
#include <audioclient.h>
|
2015-08-11 06:18:08 +00:00
|
|
|
|
2015-08-12 18:01:06 +00:00
|
|
|
// Attempting to use the Windows-supplied versions of these constants resulted
|
|
|
|
// in `undefined reference` linker errors.
|
|
|
|
const static GUID SOUNDIO_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT = {
|
|
|
|
0x00000003,0x0000,0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}};
|
|
|
|
|
|
|
|
const static GUID SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM = {
|
|
|
|
0x00000001,0x0000,0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}};
|
2015-08-11 06:18:08 +00:00
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
// Adding more common sample rates helps the heuristics; feel free to do that.
|
|
|
|
static int test_sample_rates[] = {
|
|
|
|
8000,
|
|
|
|
11025,
|
|
|
|
16000,
|
|
|
|
22050,
|
|
|
|
32000,
|
|
|
|
37800,
|
|
|
|
44056,
|
|
|
|
44100,
|
|
|
|
47250,
|
|
|
|
48000,
|
|
|
|
50000,
|
|
|
|
50400,
|
|
|
|
88200,
|
|
|
|
96000,
|
|
|
|
176400,
|
|
|
|
192000,
|
|
|
|
352800,
|
|
|
|
2822400,
|
|
|
|
5644800,
|
|
|
|
};
|
|
|
|
|
|
|
|
// If you modify this list, also modify `to_wave_format` appropriately.
|
|
|
|
static SoundIoFormat test_formats[] = {
|
|
|
|
SoundIoFormatU8,
|
|
|
|
SoundIoFormatS16LE,
|
|
|
|
SoundIoFormatS24LE,
|
|
|
|
SoundIoFormatS32LE,
|
|
|
|
SoundIoFormatFloat32LE,
|
|
|
|
SoundIoFormatFloat64LE,
|
|
|
|
};
|
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
// converts a windows wide string to a UTF-8 encoded char *
|
|
|
|
// Possible errors:
|
|
|
|
// * SoundIoErrorNoMem
|
|
|
|
// * SoundIoErrorEncodingString
|
|
|
|
static int from_lpwstr(LPWSTR lpwstr, char **out_str, int *out_str_len) {
|
|
|
|
DWORD flags = 0;
|
|
|
|
int buf_size = WideCharToMultiByte(CP_UTF8, flags, lpwstr, -1, nullptr, 0, nullptr, nullptr);
|
|
|
|
|
|
|
|
if (buf_size == 0)
|
|
|
|
return SoundIoErrorEncodingString;
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
char *buf = allocate<char>(buf_size);
|
2015-08-11 06:18:08 +00:00
|
|
|
if (!buf)
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
|
|
|
|
if (WideCharToMultiByte(CP_UTF8, flags, lpwstr, -1, buf, buf_size, nullptr, nullptr) != buf_size) {
|
|
|
|
free(buf);
|
|
|
|
return SoundIoErrorEncodingString;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_str = buf;
|
2015-08-13 02:26:07 +00:00
|
|
|
*out_str_len = buf_size - 1;
|
2015-08-11 06:18:08 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-08-13 00:50:36 +00:00
|
|
|
static void from_wave_format_layout(WAVEFORMATEXTENSIBLE *wave_format, SoundIoChannelLayout *layout) {
|
|
|
|
assert(wave_format->Format.wFormatTag == WAVE_FORMAT_EXTENSIBLE);
|
|
|
|
layout->channel_count = 0;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_FRONT_LEFT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdFrontLeft;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_FRONT_RIGHT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdFrontRight;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_FRONT_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdFrontCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_LOW_FREQUENCY)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdLfe;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_BACK_LEFT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdBackLeft;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_BACK_RIGHT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdBackRight;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_FRONT_LEFT_OF_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdFrontLeftCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_FRONT_RIGHT_OF_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdFrontRightCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_BACK_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdBackCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_SIDE_LEFT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdSideLeft;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_SIDE_RIGHT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdSideRight;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_FRONT_LEFT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopFrontLeft;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_FRONT_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopFrontCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_FRONT_RIGHT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopFrontRight;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_BACK_LEFT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopBackLeft;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_BACK_CENTER)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopBackCenter;
|
|
|
|
if (wave_format->dwChannelMask & SPEAKER_TOP_BACK_RIGHT)
|
|
|
|
layout->channels[layout->channel_count++] = SoundIoChannelIdTopBackRight;
|
|
|
|
|
|
|
|
soundio_channel_layout_detect_builtin(layout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static SoundIoFormat from_wave_format_format(WAVEFORMATEXTENSIBLE *wave_format) {
|
2015-08-13 00:11:03 +00:00
|
|
|
assert(wave_format->Format.wFormatTag == WAVE_FORMAT_EXTENSIBLE);
|
|
|
|
bool is_pcm = IsEqualGUID(wave_format->SubFormat, SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM);
|
|
|
|
bool is_float = IsEqualGUID(wave_format->SubFormat, SOUNDIO_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT);
|
|
|
|
|
|
|
|
if (wave_format->Samples.wValidBitsPerSample == wave_format->Format.wBitsPerSample) {
|
|
|
|
if (wave_format->Format.wBitsPerSample == 8) {
|
|
|
|
if (is_pcm)
|
|
|
|
return SoundIoFormatU8;
|
|
|
|
} else if (wave_format->Format.wBitsPerSample == 16) {
|
|
|
|
if (is_pcm)
|
|
|
|
return SoundIoFormatS16LE;
|
|
|
|
} else if (wave_format->Format.wBitsPerSample == 32) {
|
|
|
|
if (is_pcm)
|
|
|
|
return SoundIoFormatS32LE;
|
|
|
|
else if (is_float)
|
|
|
|
return SoundIoFormatFloat32LE;
|
|
|
|
} else if (wave_format->Format.wBitsPerSample == 64) {
|
|
|
|
if (is_float)
|
|
|
|
return SoundIoFormatFloat64LE;
|
|
|
|
}
|
|
|
|
} else if (wave_format->Format.wBitsPerSample == 32 &&
|
|
|
|
wave_format->Samples.wValidBitsPerSample == 24)
|
|
|
|
{
|
|
|
|
return SoundIoFormatS24LE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SoundIoFormatInvalid;
|
|
|
|
}
|
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
// only needs to support the formats in test_formats
|
|
|
|
static void to_wave_format(SoundIoFormat format, WAVEFORMATEXTENSIBLE *wave_format) {
|
|
|
|
switch (format) {
|
|
|
|
case SoundIoFormatU8:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM;
|
|
|
|
wave_format->Format.wBitsPerSample = 8;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 8;
|
|
|
|
break;
|
|
|
|
case SoundIoFormatS16LE:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM;
|
|
|
|
wave_format->Format.wBitsPerSample = 16;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 16;
|
|
|
|
break;
|
|
|
|
case SoundIoFormatS24LE:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM;
|
|
|
|
wave_format->Format.wBitsPerSample = 32;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 24;
|
|
|
|
break;
|
|
|
|
case SoundIoFormatS32LE:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_PCM;
|
|
|
|
wave_format->Format.wBitsPerSample = 32;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 32;
|
|
|
|
break;
|
|
|
|
case SoundIoFormatFloat32LE:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT;
|
|
|
|
wave_format->Format.wBitsPerSample = 32;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 32;
|
|
|
|
break;
|
|
|
|
case SoundIoFormatFloat64LE:
|
|
|
|
wave_format->SubFormat = SOUNDIO_KSDATAFORMAT_SUBTYPE_IEEE_FLOAT;
|
|
|
|
wave_format->Format.wBitsPerSample = 64;
|
|
|
|
wave_format->Samples.wValidBitsPerSample = 64;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
soundio_panic("to_wave_format: unsupported format");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
static SoundIoDeviceAim data_flow_to_aim(EDataFlow data_flow) {
|
|
|
|
return (data_flow == eRender) ? SoundIoDeviceAimOutput : SoundIoDeviceAimInput;
|
|
|
|
}
|
2015-08-10 22:22:08 +00:00
|
|
|
|
2015-08-12 22:32:44 +00:00
|
|
|
|
|
|
|
static double from_reference_time(REFERENCE_TIME rt) {
|
|
|
|
return ((double)rt) / 10000000.0;
|
|
|
|
}
|
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
static void destruct_device(SoundIoDevicePrivate *dev) {
|
|
|
|
SoundIoDeviceWasapi *dw = &dev->backend_data.wasapi;
|
|
|
|
if (dw->audio_client)
|
|
|
|
IUnknown_Release(dw->audio_client);
|
|
|
|
dw->sample_rates.deinit();
|
|
|
|
}
|
|
|
|
|
2015-08-10 22:22:08 +00:00
|
|
|
struct RefreshDevices {
|
|
|
|
IMMDeviceCollection *collection;
|
2015-08-11 06:18:08 +00:00
|
|
|
IMMDevice *mm_device;
|
2015-08-12 18:01:06 +00:00
|
|
|
IMMDevice *default_render_device;
|
|
|
|
IMMDevice *default_capture_device;
|
2015-08-11 06:18:08 +00:00
|
|
|
IMMEndpoint *endpoint;
|
|
|
|
IPropertyStore *prop_store;
|
|
|
|
LPWSTR lpwstr;
|
|
|
|
PROPVARIANT prop_variant_value;
|
2015-08-13 00:11:03 +00:00
|
|
|
WAVEFORMATEXTENSIBLE *wave_format;
|
2015-08-11 06:18:08 +00:00
|
|
|
bool prop_variant_value_inited;
|
|
|
|
SoundIoDevicesInfo *devices_info;
|
2015-08-13 02:26:07 +00:00
|
|
|
SoundIoDevice *device_shared;
|
|
|
|
SoundIoDevice *device_raw;
|
2015-08-12 18:01:06 +00:00
|
|
|
char *default_render_id;
|
|
|
|
int default_render_id_len;
|
|
|
|
char *default_capture_id;
|
|
|
|
int default_capture_id_len;
|
2015-08-10 22:22:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void deinit_refresh_devices(RefreshDevices *rd) {
|
2015-08-11 06:18:08 +00:00
|
|
|
soundio_destroy_devices_info(rd->devices_info);
|
2015-08-13 02:26:07 +00:00
|
|
|
soundio_device_unref(rd->device_shared);
|
|
|
|
soundio_device_unref(rd->device_raw);
|
2015-08-11 06:18:08 +00:00
|
|
|
if (rd->mm_device)
|
|
|
|
IMMDevice_Release(rd->mm_device);
|
2015-08-12 18:01:06 +00:00
|
|
|
if (rd->default_render_device)
|
|
|
|
IMMDevice_Release(rd->default_render_device);
|
|
|
|
if (rd->default_capture_device)
|
|
|
|
IMMDevice_Release(rd->default_capture_device);
|
2015-08-10 22:22:08 +00:00
|
|
|
if (rd->collection)
|
|
|
|
IMMDeviceCollection_Release(rd->collection);
|
2015-08-11 06:18:08 +00:00
|
|
|
if (rd->lpwstr)
|
|
|
|
CoTaskMemFree(rd->lpwstr);
|
|
|
|
if (rd->endpoint)
|
|
|
|
IMMEndpoint_Release(rd->endpoint);
|
|
|
|
if (rd->prop_store)
|
|
|
|
IPropertyStore_Release(rd->prop_store);
|
|
|
|
if (rd->prop_variant_value_inited)
|
|
|
|
PropVariantClear(&rd->prop_variant_value);
|
2015-08-13 00:11:03 +00:00
|
|
|
if (rd->wave_format)
|
|
|
|
CoTaskMemFree(rd->wave_format);
|
2015-08-10 22:22:08 +00:00
|
|
|
}
|
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
static int detect_valid_formats(RefreshDevices *rd, WAVEFORMATEXTENSIBLE *wave_format,
|
|
|
|
SoundIoDevicePrivate *dev, AUDCLNT_SHAREMODE share_mode)
|
|
|
|
{
|
|
|
|
SoundIoDevice *device = &dev->pub;
|
2015-08-12 22:32:44 +00:00
|
|
|
SoundIoDeviceWasapi *dw = &dev->backend_data.wasapi;
|
2015-08-13 07:11:54 +00:00
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
device->format_count = 0;
|
|
|
|
device->formats = allocate<SoundIoFormat>(array_length(test_formats));
|
|
|
|
if (!device->formats)
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
|
|
|
|
WAVEFORMATEX *closest_match = nullptr;
|
|
|
|
WAVEFORMATEXTENSIBLE orig_wave_format = *wave_format;
|
|
|
|
|
|
|
|
for (int i = 0; i < array_length(test_formats); i += 1) {
|
|
|
|
SoundIoFormat test_format = test_formats[i];
|
|
|
|
to_wave_format(test_format, wave_format);
|
|
|
|
|
|
|
|
HRESULT hr = IAudioClient_IsFormatSupported(dw->audio_client, share_mode,
|
|
|
|
(WAVEFORMATEX*)wave_format, &closest_match);
|
|
|
|
if (closest_match) {
|
|
|
|
CoTaskMemFree(closest_match);
|
|
|
|
closest_match = nullptr;
|
|
|
|
}
|
|
|
|
if (hr == S_OK) {
|
|
|
|
device->formats[device->format_count++] = test_format;
|
|
|
|
} else if (hr == AUDCLNT_E_UNSUPPORTED_FORMAT || hr == S_FALSE || hr == E_INVALIDARG) {
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
*wave_format = orig_wave_format;
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*wave_format = orig_wave_format;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int add_sample_rate(SoundIoList<SoundIoSampleRateRange> *sample_rates, int *current_min, int the_max) {
|
|
|
|
int err;
|
|
|
|
if ((err = sample_rates->add_one()))
|
|
|
|
return err;
|
|
|
|
|
|
|
|
SoundIoSampleRateRange *last_range = &sample_rates->last();
|
|
|
|
last_range->min = *current_min;
|
|
|
|
last_range->max = the_max;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int do_sample_rate_test(SoundIoDevicePrivate *dev, WAVEFORMATEXTENSIBLE *wave_format,
|
|
|
|
int test_sample_rate, AUDCLNT_SHAREMODE share_mode, int *current_min, int *last_success_rate)
|
|
|
|
{
|
|
|
|
SoundIoDeviceWasapi *dw = &dev->backend_data.wasapi;
|
|
|
|
WAVEFORMATEX *closest_match = nullptr;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
wave_format->Format.nSamplesPerSec = test_sample_rate;
|
|
|
|
HRESULT hr = IAudioClient_IsFormatSupported(dw->audio_client, share_mode,
|
|
|
|
(WAVEFORMATEX*)wave_format, &closest_match);
|
|
|
|
if (closest_match) {
|
|
|
|
CoTaskMemFree(closest_match);
|
|
|
|
closest_match = nullptr;
|
|
|
|
}
|
|
|
|
if (hr == S_OK) {
|
|
|
|
if (*current_min == -1) {
|
|
|
|
*current_min = test_sample_rate;
|
|
|
|
}
|
|
|
|
*last_success_rate = test_sample_rate;
|
|
|
|
} else if (hr == AUDCLNT_E_UNSUPPORTED_FORMAT || hr == S_FALSE || hr == E_INVALIDARG) {
|
|
|
|
if (*current_min != -1) {
|
|
|
|
if ((err = add_sample_rate(&dw->sample_rates, current_min, *last_success_rate)))
|
|
|
|
return err;
|
|
|
|
*current_min = -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2015-08-12 22:32:44 +00:00
|
|
|
}
|
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
static int detect_valid_sample_rates(RefreshDevices *rd, WAVEFORMATEXTENSIBLE *wave_format,
|
|
|
|
SoundIoDevicePrivate *dev, AUDCLNT_SHAREMODE share_mode)
|
|
|
|
{
|
|
|
|
SoundIoDeviceWasapi *dw = &dev->backend_data.wasapi;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
DWORD orig_sample_rate = wave_format->Format.nSamplesPerSec;
|
|
|
|
|
|
|
|
assert(dw->sample_rates.length == 0);
|
|
|
|
|
|
|
|
int current_min = -1;
|
|
|
|
int last_success_rate = -1;
|
|
|
|
for (int i = 0; i < array_length(test_sample_rates); i += 1) {
|
|
|
|
for (int offset = -1; offset <= 1; offset += 1) {
|
|
|
|
int test_sample_rate = test_sample_rates[i] + offset;
|
|
|
|
if ((err = do_sample_rate_test(dev, wave_format, test_sample_rate, share_mode,
|
|
|
|
¤t_min, &last_success_rate)))
|
|
|
|
{
|
|
|
|
wave_format->Format.nSamplesPerSec = orig_sample_rate;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (current_min != -1) {
|
|
|
|
if ((err = add_sample_rate(&dw->sample_rates, ¤t_min, last_success_rate))) {
|
|
|
|
wave_format->Format.nSamplesPerSec = orig_sample_rate;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SoundIoDevice *device = &dev->pub;
|
|
|
|
|
|
|
|
device->sample_rate_count = dw->sample_rates.length;
|
|
|
|
device->sample_rates = dw->sample_rates.items;
|
|
|
|
|
|
|
|
wave_format->Format.nSamplesPerSec = orig_sample_rate;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-10 22:22:08 +00:00
|
|
|
static int refresh_devices(SoundIoPrivate *si) {
|
2015-08-11 06:18:08 +00:00
|
|
|
SoundIo *soundio = &si->pub;
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
RefreshDevices rd = {0};
|
2015-08-11 06:18:08 +00:00
|
|
|
int err;
|
2015-08-10 22:22:08 +00:00
|
|
|
HRESULT hr;
|
2015-08-12 18:01:06 +00:00
|
|
|
|
|
|
|
if (FAILED(hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(siw->device_enumerator, eRender,
|
|
|
|
eMultimedia, &rd.default_render_device)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if (rd.lpwstr)
|
|
|
|
CoTaskMemFree(rd.lpwstr);
|
|
|
|
if (FAILED(hr = IMMDevice_GetId(rd.default_render_device, &rd.lpwstr))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if ((err = from_lpwstr(rd.lpwstr, &rd.default_render_id, &rd.default_render_id_len))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (FAILED(hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(siw->device_enumerator, eCapture,
|
|
|
|
eMultimedia, &rd.default_capture_device)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if (rd.lpwstr)
|
|
|
|
CoTaskMemFree(rd.lpwstr);
|
|
|
|
if (FAILED(hr = IMMDevice_GetId(rd.default_capture_device, &rd.lpwstr))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if ((err = from_lpwstr(rd.lpwstr, &rd.default_capture_id, &rd.default_capture_id_len))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-10 22:22:08 +00:00
|
|
|
if (FAILED(hr = IMMDeviceEnumerator_EnumAudioEndpoints(siw->device_enumerator,
|
|
|
|
eAll, DEVICE_STATE_ACTIVE, &rd.collection)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
UINT unsigned_count;
|
|
|
|
if (FAILED(hr = IMMDeviceCollection_GetCount(rd.collection, &unsigned_count))) {
|
2015-08-10 22:22:08 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
if (unsigned_count > (UINT)INT_MAX) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorIncompatibleDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
int device_count = unsigned_count;
|
|
|
|
|
|
|
|
if (!(rd.devices_info = allocate<SoundIoDevicesInfo>(1))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-08-13 00:50:36 +00:00
|
|
|
rd.devices_info->default_input_index = -1;
|
|
|
|
rd.devices_info->default_output_index = -1;
|
2015-08-11 06:18:08 +00:00
|
|
|
|
|
|
|
for (int device_i = 0; device_i < device_count; device_i += 1) {
|
|
|
|
if (rd.mm_device)
|
|
|
|
IMMDevice_Release(rd.mm_device);
|
|
|
|
if (FAILED(hr = IMMDeviceCollection_Item(rd.collection, device_i, &rd.mm_device))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if (rd.lpwstr)
|
|
|
|
CoTaskMemFree(rd.lpwstr);
|
|
|
|
if (FAILED(hr = IMMDevice_GetId(rd.mm_device, &rd.lpwstr))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-12 22:32:44 +00:00
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
SoundIoDevicePrivate *dev_shared = allocate<SoundIoDevicePrivate>(1);
|
|
|
|
if (!dev_shared) {
|
2015-08-11 06:18:08 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-08-13 02:26:07 +00:00
|
|
|
SoundIoDeviceWasapi *dev_w_shared = &dev_shared->backend_data.wasapi;
|
|
|
|
dev_shared->destruct = destruct_device;
|
|
|
|
assert(!rd.device_shared);
|
|
|
|
rd.device_shared = &dev_shared->pub;
|
|
|
|
rd.device_shared->ref_count = 1;
|
|
|
|
rd.device_shared->soundio = soundio;
|
|
|
|
rd.device_shared->is_raw = false;
|
|
|
|
|
|
|
|
SoundIoDevicePrivate *dev_raw = allocate<SoundIoDevicePrivate>(1);
|
|
|
|
if (!dev_raw) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
SoundIoDeviceWasapi *dev_w_raw = &dev_raw->backend_data.wasapi;
|
|
|
|
dev_raw->destruct = destruct_device;
|
|
|
|
assert(!rd.device_raw);
|
|
|
|
rd.device_raw = &dev_raw->pub;
|
|
|
|
rd.device_raw->ref_count = 1;
|
|
|
|
rd.device_raw->soundio = soundio;
|
|
|
|
rd.device_raw->is_raw = true;
|
2015-08-11 06:18:08 +00:00
|
|
|
|
|
|
|
int device_id_len;
|
2015-08-13 02:26:07 +00:00
|
|
|
if ((err = from_lpwstr(rd.lpwstr, &rd.device_shared->id, &device_id_len))) {
|
2015-08-11 06:18:08 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_raw->id = soundio_str_dupe(rd.device_shared->id, device_id_len);
|
|
|
|
if (!rd.device_raw->id) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-08-12 22:32:44 +00:00
|
|
|
if (FAILED(hr = IMMDevice_Activate(rd.mm_device, IID_IAudioClient,
|
2015-08-13 02:26:07 +00:00
|
|
|
CLSCTX_ALL, nullptr, (void**)&dev_w_shared->audio_client)))
|
2015-08-12 22:32:44 +00:00
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
if (FAILED(hr = IAudioClient_AddRef(dev_w_shared->audio_client))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
dev_w_raw->audio_client = dev_w_shared->audio_client;
|
|
|
|
|
2015-08-12 22:32:44 +00:00
|
|
|
REFERENCE_TIME default_device_period;
|
|
|
|
REFERENCE_TIME min_device_period;
|
2015-08-13 02:26:07 +00:00
|
|
|
if (FAILED(hr = IAudioClient_GetDevicePeriod(dev_w_shared->audio_client,
|
2015-08-12 22:32:44 +00:00
|
|
|
&default_device_period, &min_device_period)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared->period_duration_current = from_reference_time(default_device_period);
|
|
|
|
rd.device_shared->period_duration_min = from_reference_time(min_device_period);
|
2015-08-12 22:32:44 +00:00
|
|
|
|
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
if (rd.endpoint)
|
|
|
|
IMMEndpoint_Release(rd.endpoint);
|
|
|
|
if (FAILED(hr = IMMDevice_QueryInterface(rd.mm_device, IID_IMMEndpoint, (void**)&rd.endpoint))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
EDataFlow data_flow;
|
|
|
|
if (FAILED(hr = IMMEndpoint_GetDataFlow(rd.endpoint, &data_flow))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared->aim = data_flow_to_aim(data_flow);
|
|
|
|
rd.device_raw->aim = rd.device_shared->aim;
|
2015-08-11 06:18:08 +00:00
|
|
|
|
|
|
|
if (rd.prop_store)
|
|
|
|
IPropertyStore_Release(rd.prop_store);
|
|
|
|
if (FAILED(hr = IMMDevice_OpenPropertyStore(rd.mm_device, STGM_READ, &rd.prop_store))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rd.prop_variant_value_inited)
|
|
|
|
PropVariantClear(&rd.prop_variant_value);
|
|
|
|
PropVariantInit(&rd.prop_variant_value);
|
|
|
|
rd.prop_variant_value_inited = true;
|
|
|
|
if (FAILED(hr = IPropertyStore_GetValue(rd.prop_store,
|
|
|
|
PKEY_Device_FriendlyName, &rd.prop_variant_value)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if (!rd.prop_variant_value.pwszVal) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
int device_name_len;
|
2015-08-13 02:26:07 +00:00
|
|
|
if ((err = from_lpwstr(rd.prop_variant_value.pwszVal, &rd.device_shared->name, &device_name_len))) {
|
2015-08-11 06:18:08 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_raw->name = soundio_str_dupe(rd.device_shared->name, device_name_len);
|
|
|
|
if (!rd.device_raw->name) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
// Get the format that WASAPI opens the device with for shared streams.
|
|
|
|
// This is guaranteed to work, so we use this to modulate the sample
|
|
|
|
// rate while holding the format constant and vice versa.
|
|
|
|
if (rd.prop_variant_value_inited)
|
|
|
|
PropVariantClear(&rd.prop_variant_value);
|
|
|
|
PropVariantInit(&rd.prop_variant_value);
|
|
|
|
rd.prop_variant_value_inited = true;
|
|
|
|
if (FAILED(hr = IPropertyStore_GetValue(rd.prop_store, PKEY_AudioEngine_DeviceFormat,
|
|
|
|
&rd.prop_variant_value)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
WAVEFORMATEXTENSIBLE *valid_wave_format = (WAVEFORMATEXTENSIBLE *)rd.prop_variant_value.blob.pBlobData;
|
|
|
|
if (valid_wave_format->Format.wFormatTag != WAVE_FORMAT_EXTENSIBLE) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
|
|
|
if ((err = detect_valid_sample_rates(&rd, valid_wave_format, dev_raw,
|
|
|
|
AUDCLNT_SHAREMODE_EXCLUSIVE)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if ((err = detect_valid_formats(&rd, valid_wave_format, dev_raw,
|
|
|
|
AUDCLNT_SHAREMODE_EXCLUSIVE)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2015-08-13 00:11:03 +00:00
|
|
|
if (rd.wave_format)
|
|
|
|
CoTaskMemFree(rd.wave_format);
|
2015-08-13 02:26:07 +00:00
|
|
|
if (FAILED(hr = IAudioClient_GetMixFormat(dev_w_shared->audio_client, (WAVEFORMATEX**)&rd.wave_format))) {
|
2015-08-12 18:01:06 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
2015-08-13 00:11:03 +00:00
|
|
|
if (rd.wave_format->Format.wFormatTag != WAVE_FORMAT_EXTENSIBLE) {
|
2015-08-12 18:01:06 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorOpeningDevice;
|
|
|
|
}
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared->sample_rate_current = rd.wave_format->Format.nSamplesPerSec;
|
2015-08-13 07:11:54 +00:00
|
|
|
rd.device_shared->current_format = from_wave_format_format(rd.wave_format);
|
|
|
|
|
|
|
|
|
2015-08-13 01:57:34 +00:00
|
|
|
// WASAPI performs resampling in shared mode, so any value is valid.
|
|
|
|
// Let's pick some reasonable min and max values.
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared->sample_rate_count = 1;
|
|
|
|
rd.device_shared->sample_rates = &dev_shared->prealloc_sample_rate_range;
|
2015-08-13 07:11:54 +00:00
|
|
|
rd.device_shared->sample_rates[0].min = min(SOUNDIO_MIN_SAMPLE_RATE,
|
|
|
|
rd.device_shared->sample_rate_current);
|
|
|
|
rd.device_shared->sample_rates[0].max = max(SOUNDIO_MAX_SAMPLE_RATE,
|
|
|
|
rd.device_shared->sample_rate_current);
|
2015-08-12 18:01:06 +00:00
|
|
|
|
2015-08-13 07:11:54 +00:00
|
|
|
if ((err = detect_valid_formats(&rd, rd.wave_format, dev_shared,
|
|
|
|
AUDCLNT_SHAREMODE_SHARED)))
|
|
|
|
{
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
2015-08-13 00:11:03 +00:00
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
from_wave_format_layout(rd.wave_format, &rd.device_shared->current_layout);
|
|
|
|
rd.device_shared->layout_count = 1;
|
|
|
|
rd.device_shared->layouts = allocate<SoundIoChannelLayout>(1);
|
2015-08-13 00:50:36 +00:00
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
if (!rd.device_shared->layouts) {
|
2015-08-13 00:50:36 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared->layouts[0] = rd.device_shared->current_layout;
|
2015-08-13 00:11:03 +00:00
|
|
|
|
|
|
|
|
2015-08-12 18:01:06 +00:00
|
|
|
|
2015-08-11 06:18:08 +00:00
|
|
|
SoundIoList<SoundIoDevice *> *device_list;
|
2015-08-13 02:26:07 +00:00
|
|
|
if (rd.device_shared->aim == SoundIoDeviceAimOutput) {
|
2015-08-11 06:18:08 +00:00
|
|
|
device_list = &rd.devices_info->output_devices;
|
2015-08-13 02:26:07 +00:00
|
|
|
if (soundio_streql(rd.device_shared->id, device_id_len,
|
|
|
|
rd.default_render_id, rd.default_render_id_len))
|
|
|
|
{
|
2015-08-12 18:01:06 +00:00
|
|
|
rd.devices_info->default_output_index = device_list->length;
|
2015-08-13 02:26:07 +00:00
|
|
|
}
|
2015-08-11 06:18:08 +00:00
|
|
|
} else {
|
2015-08-13 02:26:07 +00:00
|
|
|
assert(rd.device_shared->aim == SoundIoDeviceAimInput);
|
2015-08-11 06:18:08 +00:00
|
|
|
device_list = &rd.devices_info->input_devices;
|
2015-08-13 02:26:07 +00:00
|
|
|
if (soundio_streql(rd.device_shared->id, device_id_len,
|
|
|
|
rd.default_capture_id, rd.default_capture_id_len))
|
|
|
|
{
|
2015-08-12 18:01:06 +00:00
|
|
|
rd.devices_info->default_input_index = device_list->length;
|
2015-08-13 02:26:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((err = device_list->append(rd.device_shared))) {
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
2015-08-11 06:18:08 +00:00
|
|
|
}
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_shared = nullptr;
|
2015-08-11 06:18:08 +00:00
|
|
|
|
2015-08-13 02:26:07 +00:00
|
|
|
if ((err = device_list->append(rd.device_raw))) {
|
2015-08-11 06:18:08 +00:00
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
return err;
|
|
|
|
}
|
2015-08-13 02:26:07 +00:00
|
|
|
rd.device_raw = nullptr;
|
2015-08-11 06:18:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
soundio_os_mutex_lock(siw->mutex);
|
|
|
|
soundio_destroy_devices_info(siw->ready_devices_info);
|
|
|
|
siw->ready_devices_info = rd.devices_info;
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
soundio_os_mutex_unlock(siw->mutex);
|
|
|
|
|
|
|
|
rd.devices_info = nullptr;
|
|
|
|
deinit_refresh_devices(&rd);
|
|
|
|
|
2015-08-10 22:22:08 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void shutdown_backend(SoundIoPrivate *si, int err) {
|
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
soundio_os_mutex_lock(siw->mutex);
|
|
|
|
siw->shutdown_err = err;
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
soundio_os_mutex_unlock(siw->mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void device_thread_run(void *arg) {
|
|
|
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
HRESULT hr = CoCreateInstance(CLSID_MMDeviceEnumerator, nullptr,
|
|
|
|
CLSCTX_ALL, IID_IMMDeviceEnumerator, (void**)&siw->device_enumerator);
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
shutdown_backend(si, SoundIoErrorSystemResources);
|
|
|
|
if (!siw->have_devices_flag.exchange(true)) {
|
|
|
|
soundio_os_cond_signal(siw->cond, nullptr);
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
if (!siw->abort_flag.test_and_set())
|
|
|
|
break;
|
|
|
|
if (siw->device_scan_queued.exchange(false)) {
|
|
|
|
err = refresh_devices(si);
|
|
|
|
if (err)
|
|
|
|
shutdown_backend(si, err);
|
|
|
|
if (!siw->have_devices_flag.exchange(true)) {
|
2015-08-12 18:01:06 +00:00
|
|
|
// TODO separate cond for signaling devices like coreaudio
|
2015-08-10 22:22:08 +00:00
|
|
|
soundio_os_cond_signal(siw->cond, nullptr);
|
|
|
|
soundio->on_events_signal(soundio);
|
|
|
|
}
|
|
|
|
if (err)
|
|
|
|
break;
|
|
|
|
soundio_os_cond_signal(siw->cond, nullptr);
|
|
|
|
}
|
|
|
|
soundio_os_cond_wait(siw->cond, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
IMMDeviceEnumerator_Release(siw->device_enumerator);
|
|
|
|
siw->device_enumerator = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void block_until_have_devices(SoundIoWasapi *siw) {
|
|
|
|
if (siw->have_devices_flag.load())
|
|
|
|
return;
|
|
|
|
while (!siw->have_devices_flag.load())
|
|
|
|
soundio_os_cond_wait(siw->cond, nullptr);
|
|
|
|
}
|
|
|
|
|
2015-08-08 22:22:50 +00:00
|
|
|
static void flush_events_wasapi(struct SoundIoPrivate *si) {
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
block_until_have_devices(siw);
|
|
|
|
|
|
|
|
bool change = false;
|
|
|
|
bool cb_shutdown = false;
|
|
|
|
SoundIoDevicesInfo *old_devices_info = nullptr;
|
|
|
|
|
|
|
|
soundio_os_mutex_lock(siw->mutex);
|
|
|
|
|
|
|
|
if (siw->shutdown_err && !siw->emitted_shutdown_cb) {
|
|
|
|
siw->emitted_shutdown_cb = true;
|
|
|
|
cb_shutdown = true;
|
|
|
|
} else if (siw->ready_devices_info) {
|
|
|
|
old_devices_info = si->safe_devices_info;
|
|
|
|
si->safe_devices_info = siw->ready_devices_info;
|
|
|
|
siw->ready_devices_info = nullptr;
|
|
|
|
change = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
soundio_os_mutex_unlock(siw->mutex);
|
|
|
|
|
|
|
|
if (cb_shutdown)
|
|
|
|
soundio->on_backend_disconnect(soundio, siw->shutdown_err);
|
|
|
|
else if (change)
|
|
|
|
soundio->on_devices_change(soundio);
|
|
|
|
|
|
|
|
soundio_destroy_devices_info(old_devices_info);
|
2015-08-08 22:22:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void wait_events_wasapi(struct SoundIoPrivate *si) {
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
flush_events_wasapi(si);
|
|
|
|
soundio_os_cond_wait(siw->cond, nullptr);
|
2015-08-08 22:22:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void wakeup_wasapi(struct SoundIoPrivate *si) {
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
soundio_os_cond_signal(siw->cond, nullptr);
|
2015-08-08 22:22:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void outstream_destroy_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_open_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_pause_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os, bool pause) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_start_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_begin_write_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os,
|
|
|
|
SoundIoChannelArea **out_areas, int *frame_count)
|
|
|
|
{
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_end_write_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_clear_buffer_wasapi(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void instream_destroy_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int instream_open_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int instream_pause_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is, bool pause) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int instream_start_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int instream_begin_read_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is,
|
|
|
|
SoundIoChannelArea **out_areas, int *frame_count)
|
|
|
|
{
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int instream_end_read_wasapi(struct SoundIoPrivate *si, struct SoundIoInStreamPrivate *is) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void destroy_wasapi(struct SoundIoPrivate *si) {
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
|
|
|
|
if (siw->thread) {
|
|
|
|
siw->abort_flag.clear();
|
|
|
|
soundio_os_cond_signal(siw->cond, nullptr);
|
|
|
|
soundio_os_thread_destroy(siw->thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (siw->cond)
|
|
|
|
soundio_os_cond_destroy(siw->cond);
|
|
|
|
|
|
|
|
if (siw->mutex)
|
|
|
|
soundio_os_mutex_destroy(siw->mutex);
|
|
|
|
|
|
|
|
soundio_destroy_devices_info(siw->ready_devices_info);
|
2015-08-08 22:22:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int soundio_wasapi_init(SoundIoPrivate *si) {
|
2015-08-10 22:22:08 +00:00
|
|
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
siw->have_devices_flag.store(false);
|
|
|
|
siw->device_scan_queued.store(true);
|
|
|
|
siw->abort_flag.test_and_set();
|
|
|
|
|
|
|
|
siw->mutex = soundio_os_mutex_create();
|
|
|
|
if (!siw->mutex) {
|
|
|
|
destroy_wasapi(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
siw->cond = soundio_os_cond_create();
|
|
|
|
if (!siw->cond) {
|
|
|
|
destroy_wasapi(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((err = soundio_os_thread_create(device_thread_run, si, false, &siw->thread))) {
|
|
|
|
destroy_wasapi(si);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2015-08-08 22:22:50 +00:00
|
|
|
si->destroy = destroy_wasapi;
|
|
|
|
si->flush_events = flush_events_wasapi;
|
|
|
|
si->wait_events = wait_events_wasapi;
|
|
|
|
si->wakeup = wakeup_wasapi;
|
|
|
|
|
|
|
|
si->outstream_open = outstream_open_wasapi;
|
|
|
|
si->outstream_destroy = outstream_destroy_wasapi;
|
|
|
|
si->outstream_start = outstream_start_wasapi;
|
|
|
|
si->outstream_begin_write = outstream_begin_write_wasapi;
|
|
|
|
si->outstream_end_write = outstream_end_write_wasapi;
|
|
|
|
si->outstream_clear_buffer = outstream_clear_buffer_wasapi;
|
|
|
|
si->outstream_pause = outstream_pause_wasapi;
|
|
|
|
|
|
|
|
si->instream_open = instream_open_wasapi;
|
|
|
|
si->instream_destroy = instream_destroy_wasapi;
|
|
|
|
si->instream_start = instream_start_wasapi;
|
|
|
|
si->instream_begin_read = instream_begin_read_wasapi;
|
|
|
|
si->instream_end_read = instream_end_read_wasapi;
|
|
|
|
si->instream_pause = instream_pause_wasapi;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|