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_
|
|
|
|
|
|
|
|
#define COBJMACROS 1
|
|
|
|
#define _WIN32_WINNT 0x600
|
|
|
|
|
|
|
|
#include <audioclient.h>
|
2014-03-11 06:46:22 +00:00
|
|
|
#include <audiopolicy.h>
|
2014-03-09 23:13:36 +00:00
|
|
|
#include <mmdeviceapi.h>
|
|
|
|
#include <avrt.h>
|
|
|
|
|
2014-08-29 10:09:04 +00:00
|
|
|
#include "osdep/atomics.h"
|
2014-05-20 23:04:47 +00:00
|
|
|
|
2014-11-17 11:37:51 +00:00
|
|
|
typedef struct change_notify {
|
2014-11-13 09:09:47 +00:00
|
|
|
IMMNotificationClient client; /* this must be first in the structure! */
|
2014-11-17 11:37:51 +00:00
|
|
|
LPWSTR monitored; /* Monitored device */
|
2014-11-13 09:09:47 +00:00
|
|
|
struct ao *ao;
|
2014-11-17 11:37:51 +00:00
|
|
|
} change_notify;
|
|
|
|
|
2014-11-13 09:09:47 +00:00
|
|
|
HRESULT wasapi_change_init(struct ao* ao);
|
|
|
|
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
|
|
|
|
2014-03-09 23:13:36 +00:00
|
|
|
typedef struct wasapi_state {
|
|
|
|
struct mp_log *log;
|
|
|
|
HANDLE threadLoop;
|
|
|
|
|
|
|
|
/* Init phase */
|
2014-11-17 10:48:02 +00:00
|
|
|
HRESULT init_ret;
|
2014-03-09 23:13:36 +00:00
|
|
|
HANDLE init_done;
|
|
|
|
int share_mode;
|
|
|
|
|
|
|
|
HANDLE hUninit;
|
|
|
|
|
|
|
|
/* volume control */
|
|
|
|
DWORD vol_hw_support, status;
|
|
|
|
float audio_volume;
|
2014-03-11 05:50:49 +00:00
|
|
|
float previous_volume;
|
|
|
|
float initial_volume;
|
2014-03-09 23:13:36 +00:00
|
|
|
|
|
|
|
/* Buffers */
|
|
|
|
size_t buffer_block_size; /* Size of each block in bytes */
|
|
|
|
UINT32 bufferFrameCount; /* wasapi buffer block size, number of frames, frame size at format.nBlockAlign */
|
|
|
|
|
|
|
|
/* WASAPI handles, owned by other thread */
|
|
|
|
IMMDevice *pDevice;
|
|
|
|
IAudioClient *pAudioClient;
|
|
|
|
IAudioRenderClient *pRenderClient;
|
2014-03-11 05:50:49 +00:00
|
|
|
ISimpleAudioVolume *pAudioVolume;
|
2014-03-09 23:13:36 +00:00
|
|
|
IAudioEndpointVolume *pEndpointVolume;
|
2014-03-11 06:46:22 +00:00
|
|
|
IAudioSessionControl *pSessionControl;
|
2014-11-13 09:09:47 +00:00
|
|
|
IMMDeviceEnumerator *pEnumerator;
|
|
|
|
|
2014-03-09 23:13:36 +00:00
|
|
|
HANDLE hFeed; /* wasapi event */
|
2014-03-11 03:47:33 +00:00
|
|
|
HANDLE hForceFeed; /* forces writing a buffer (e.g. before audio_resume) */
|
|
|
|
HANDLE hFeedDone; /* set only after a hForceFeed */
|
2014-03-09 23:13:36 +00:00
|
|
|
HANDLE hTask; /* AV thread */
|
|
|
|
DWORD taskIndex; /* AV task ID */
|
|
|
|
WAVEFORMATEXTENSIBLE format;
|
|
|
|
|
2014-03-11 03:47:33 +00:00
|
|
|
/* WASAPI proxy handles, for Single-Threaded Apartment communication.
|
|
|
|
One is needed for each object that's accessed by a different thread. */
|
|
|
|
IAudioClient *pAudioClientProxy;
|
2014-03-11 05:50:49 +00:00
|
|
|
ISimpleAudioVolume *pAudioVolumeProxy;
|
2014-03-11 03:47:33 +00:00
|
|
|
IAudioEndpointVolume *pEndpointVolumeProxy;
|
2014-03-11 06:46:22 +00:00
|
|
|
IAudioSessionControl *pSessionControlProxy;
|
2014-03-11 03:47:33 +00:00
|
|
|
|
|
|
|
/* 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. */
|
|
|
|
IStream *sAudioClient;
|
2014-03-11 05:50:49 +00:00
|
|
|
IStream *sAudioVolume;
|
2014-03-11 03:47:33 +00:00
|
|
|
IStream *sEndpointVolume;
|
2014-03-11 06:46:22 +00:00
|
|
|
IStream *sSessionControl;
|
2014-03-11 03:47:33 +00:00
|
|
|
|
2014-03-09 23:13:36 +00:00
|
|
|
/* WASAPI internal clock information, for estimating delay */
|
|
|
|
IAudioClock *pAudioClock;
|
|
|
|
UINT64 clock_frequency; /* scale for the "samples" returned by the clock */
|
2014-05-20 23:04:47 +00:00
|
|
|
atomic_ullong sample_count; /* the amount of samples per channel written to a GetBuffer buffer */
|
2014-03-09 23:13:36 +00:00
|
|
|
LARGE_INTEGER qpc_frequency; /* frequency of windows' high resolution timer */
|
|
|
|
|
|
|
|
int opt_exclusive;
|
|
|
|
int opt_list;
|
|
|
|
char *opt_device;
|
|
|
|
|
2014-11-28 18:43:48 +00:00
|
|
|
/* Don't use these functions directly in case
|
|
|
|
they are unimplemented for some reason.
|
|
|
|
(XP shouldn't be an issue since it doesn't support wasapi, maybe wine?)
|
|
|
|
Blob is owned by the main thread */
|
2014-03-09 23:13:36 +00:00
|
|
|
struct {
|
|
|
|
HMODULE hAvrt;
|
|
|
|
HANDLE (WINAPI *pAvSetMmThreadCharacteristicsW)(LPCWSTR, LPDWORD);
|
|
|
|
WINBOOL (WINAPI *pAvRevertMmThreadCharacteristics)(HANDLE);
|
|
|
|
} VistaBlob;
|
2014-11-13 09:09:47 +00:00
|
|
|
|
|
|
|
change_notify change;
|
2014-03-09 23:13:36 +00:00
|
|
|
} wasapi_state;
|
|
|
|
|
2014-03-11 03:47:33 +00:00
|
|
|
#endif
|