2014-03-09 23:13:36 +00:00
|
|
|
/*
|
|
|
|
* This file is part of mpv.
|
|
|
|
*
|
|
|
|
* Original author: Jonathan Yong <10walls@gmail.com>
|
|
|
|
*
|
|
|
|
* mpv is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* mpv is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with mpv. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef MP_AO_WASAPI_H_
|
|
|
|
#define MP_AO_WASAPI_H_
|
|
|
|
|
2016-01-05 03:47:38 +00:00
|
|
|
#include <stdlib.h>
|
2015-03-31 08:56:17 +00:00
|
|
|
#include <stdbool.h>
|
2016-01-05 03:47:38 +00:00
|
|
|
#include <windows.h>
|
|
|
|
#include <mmdeviceapi.h>
|
2014-03-09 23:13:36 +00:00
|
|
|
#include <audioclient.h>
|
2014-03-11 06:46:22 +00:00
|
|
|
#include <audiopolicy.h>
|
2016-01-05 03:47:38 +00:00
|
|
|
#include <endpointvolume.h>
|
2014-03-09 23:13:36 +00:00
|
|
|
|
2016-01-05 03:47:38 +00:00
|
|
|
#include "common/msg.h"
|
2014-08-29 10:09:04 +00:00
|
|
|
#include "osdep/atomics.h"
|
2016-01-05 03:47:38 +00:00
|
|
|
#include "internal.h"
|
|
|
|
#include "ao.h"
|
2014-05-20 23:04:47 +00:00
|
|
|
|
2014-11-17 11:37:51 +00:00
|
|
|
typedef struct change_notify {
|
2015-12-21 00:43:23 +00:00
|
|
|
IMMNotificationClient client; // this must be first in the structure!
|
2015-12-29 09:13:17 +00:00
|
|
|
IMMDeviceEnumerator *pEnumerator; // object where client is registered
|
2015-12-21 00:43:23 +00:00
|
|
|
LPWSTR monitored; // Monitored device
|
2015-03-31 08:56:17 +00:00
|
|
|
bool is_hotplug;
|
2014-11-13 09:09:47 +00:00
|
|
|
struct ao *ao;
|
2014-11-17 11:37:51 +00:00
|
|
|
} change_notify;
|
|
|
|
|
2015-03-31 08:56:17 +00:00
|
|
|
HRESULT wasapi_change_init(struct ao* ao, bool is_hotplug);
|
2014-11-13 09:09:47 +00:00
|
|
|
void wasapi_change_uninit(struct ao* ao);
|
|
|
|
|
|
|
|
#define EXIT_ON_ERROR(hres) \
|
|
|
|
do { if (FAILED(hres)) { goto exit_label; } } while(0)
|
|
|
|
#define SAFE_RELEASE(unk, release) \
|
|
|
|
do { if ((unk) != NULL) { release; (unk) = NULL; } } while(0)
|
2014-11-17 11:37:51 +00:00
|
|
|
|
2015-03-29 00:12:48 +00:00
|
|
|
enum wasapi_thread_state {
|
|
|
|
WASAPI_THREAD_FEED = 0,
|
|
|
|
WASAPI_THREAD_RESUME,
|
|
|
|
WASAPI_THREAD_RESET,
|
|
|
|
WASAPI_THREAD_SHUTDOWN
|
|
|
|
};
|
|
|
|
|
2014-03-09 23:13:36 +00:00
|
|
|
typedef struct wasapi_state {
|
|
|
|
struct mp_log *log;
|
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// Thread handles
|
|
|
|
HRESULT init_ret; // status of init phase
|
|
|
|
HANDLE hInitDone; // set when init is complete in audio thread
|
|
|
|
HANDLE hAudioThread; // the audio thread itself
|
|
|
|
HANDLE hWake; // thread wakeup event
|
|
|
|
atomic_int thread_state; // enum wasapi_thread_state (what to do on wakeup)
|
2014-03-09 23:13:36 +00:00
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// for setting the audio thread priority
|
|
|
|
HANDLE hTask;
|
|
|
|
|
2015-12-26 20:57:16 +00:00
|
|
|
// ID of the device to use
|
|
|
|
LPWSTR deviceID;
|
2015-12-21 02:19:12 +00:00
|
|
|
// WASAPI object handles owned and used by audio thread
|
2014-03-09 23:13:36 +00:00
|
|
|
IMMDevice *pDevice;
|
|
|
|
IAudioClient *pAudioClient;
|
|
|
|
IAudioRenderClient *pRenderClient;
|
2014-11-13 09:09:47 +00:00
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// WASAPI internal clock information, for estimating delay
|
|
|
|
IAudioClock *pAudioClock;
|
|
|
|
atomic_ullong sample_count; // samples per channel written by GetBuffer
|
|
|
|
UINT64 clock_frequency; // scale for position returned by GetPosition
|
|
|
|
LARGE_INTEGER qpc_frequency; // frequency of Windows' high resolution timer
|
|
|
|
|
|
|
|
// WASAPI control (handles owned by audio thread but used by main thread)
|
|
|
|
IAudioSessionControl *pSessionControl; // setting the stream title
|
|
|
|
IAudioEndpointVolume *pEndpointVolume; // exclusive mode volume/mute
|
|
|
|
ISimpleAudioVolume *pAudioVolume; // shared mode volume/mute
|
|
|
|
DWORD vol_hw_support; // is hardware volume supported for exclusive-mode?
|
|
|
|
|
|
|
|
// Streams used to marshal the proxy objects. The thread owning the actual
|
|
|
|
// objects needs to marshal proxy objects into these streams, and the thread
|
|
|
|
// that wants the proxies unmarshals them from here.
|
2014-03-11 06:46:22 +00:00
|
|
|
IStream *sSessionControl;
|
2015-12-21 02:19:12 +00:00
|
|
|
IStream *sEndpointVolume;
|
|
|
|
IStream *sAudioVolume;
|
2014-03-11 03:47:33 +00:00
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// WASAPI proxy handles, for Single-Threaded Apartment communication. One is
|
|
|
|
// needed for each audio thread object that's accessed from the main thread.
|
|
|
|
IAudioSessionControl *pSessionControlProxy;
|
|
|
|
IAudioEndpointVolume *pEndpointVolumeProxy;
|
|
|
|
ISimpleAudioVolume *pAudioVolumeProxy;
|
2014-03-09 23:13:36 +00:00
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// ao options
|
2014-03-09 23:13:36 +00:00
|
|
|
int opt_exclusive;
|
|
|
|
int opt_list;
|
|
|
|
char *opt_device;
|
|
|
|
|
2015-12-21 02:19:12 +00:00
|
|
|
// format info
|
2015-03-27 23:48:39 +00:00
|
|
|
WAVEFORMATEXTENSIBLE format;
|
2015-12-21 02:19:12 +00:00
|
|
|
AUDCLNT_SHAREMODE share_mode; // AUDCLNT_SHAREMODE_EXCLUSIVE / SHARED
|
|
|
|
UINT32 bufferFrameCount; // number of frames in buffer
|
2015-03-27 23:48:39 +00:00
|
|
|
|
2014-11-13 09:09:47 +00:00
|
|
|
change_notify change;
|
2014-03-09 23:13:36 +00:00
|
|
|
} wasapi_state;
|
|
|
|
|
2016-01-05 03:47:38 +00:00
|
|
|
char *mp_GUID_to_str_buf(char *buf, size_t buf_size, const GUID *guid);
|
|
|
|
char *mp_PKEY_to_str_buf(char *buf, size_t buf_size, const PROPERTYKEY *pkey);
|
|
|
|
char *mp_HRESULT_to_str_buf(char *buf, size_t buf_size, HRESULT hr);
|
|
|
|
#define mp_GUID_to_str(guid) mp_GUID_to_str_buf((char[40]){0}, 40, (guid))
|
|
|
|
#define mp_PKEY_to_str(pkey) mp_PKEY_to_str_buf((char[42]){0}, 42, (pkey))
|
|
|
|
#define mp_HRESULT_to_str(hr) mp_HRESULT_to_str_buf((char[60]){0}, 60, (hr))
|
|
|
|
#define mp_LastError_to_str() mp_HRESULT_to_str(HRESULT_FROM_WIN32(GetLastError()))
|
|
|
|
|
|
|
|
void wasapi_list_devs(struct ao *ao, struct ao_device_list *list);
|
|
|
|
LPWSTR find_deviceID(struct ao *ao);
|
|
|
|
|
|
|
|
void wasapi_dispatch(struct ao *ao);
|
|
|
|
HRESULT wasapi_thread_init(struct ao *ao);
|
|
|
|
void wasapi_thread_uninit(struct ao *ao);
|
|
|
|
|
|
|
|
void wasapi_receive_proxies(wasapi_state *state);
|
|
|
|
void wasapi_release_proxies(wasapi_state *state);
|
|
|
|
|
2014-03-11 03:47:33 +00:00
|
|
|
#endif
|