1
0
mirror of https://github.com/cookiengineer/audacity synced 2025-06-30 15:18:42 +02:00
audacity/lib-src/portmixer/src/px_win_endpoint.c
2013-10-31 22:28:21 +00:00

531 lines
13 KiB
C

/*
* PortMixer
* Common Windows routines for Vista/7
*
* Copyright (c) 2002, 2009
*
* Written by Dominic Mazzoni and Augustus Saunders
* and Leland Lucius
*
* PortMixer is intended to work side-by-side with PortAudio,
* the Portable Real-Time Audio Library by Ross Bencina and
* Phil Burk.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files
* (the "Software"), to deal in the Software without restriction,
* including without limitation the rights to use, copy, modify, merge,
* publish, distribute, sublicense, and/or sell copies of the Software,
* and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* Any person wishing to distribute modifications to the Software is
* requested to send the modifications to the original developer so that
* they can be incorporated into the canonical version.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
* ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
*/
#define CINTERFACE 1
#define COBJMACROS 1
#include <windows.h>
#include <tchar.h>
#include <strsafe.h>
#include <mmdeviceapi.h>
#include <initguid.h>
#include <endpointvolume.h>
#include "portaudio.h"
#include "portmixer.h"
#include "px_mixer.h"
#include "px_win_endpoint.h"
#if defined(_DEBUG)
#include <stdio.h>
#include <stdlib.h>
static void dprintf(const char *format, ...)
{
char buf[4096];
va_list args;
int cnt;
va_start(args, format);
cnt = _vsnprintf(buf, sizeof(buf) - 1, format, args);
va_end(args);
if (cnt > 0) {
buf[cnt] = '\0';
OutputDebugString(buf);
}
}
#endif
DEFINE_GUID(CLSID_MMDeviceEnumerator, 0xBCDE0395, 0xE52F, 0x467C,
0x8E, 0x3D, 0xC4, 0x57, 0x92, 0x91, 0x69, 0x2E);
DEFINE_GUID(IID_IMMDeviceEnumerator, 0xA95664D2, 0x9614, 0x4F35,
0xA7, 0x46, 0xDE, 0x8D, 0xB6, 0x36, 0x17, 0xE6);
DEFINE_GUID(IID_IAudioEndpointVolume, 0x5CDF2C82, 0x841E, 0x4546,
0x97, 0x22, 0x0C, 0xF7, 0x40, 0x78, 0x22, 0x9A);
#define DRV_RESERVED 0x0800
#define DRV_QUERYFUNCTIONINSTANCEID (DRV_RESERVED + 17)
#define DRV_QUERYFUNCTIONINSTANCEIDSIZE (DRV_RESERVED + 18)
int open_ep_mixers(px_mixer *Px, UINT deviceIn, UINT deviceOut)
{
PxEPInfo *info;
IMMDeviceEnumerator *denum = NULL;
IMMDevice *device = NULL;
HRESULT hr;
MMRESULT res;
LPWSTR idStr;
size_t idLen;
if (!initialize(Px)) {
goto fail;
}
info = (PxEPInfo *) Px->info;
info->inputEP = NULL;
info->outputEP = NULL;
// Create an audio endpoint device enumerator.
hr = CoCreateInstance(&CLSID_MMDeviceEnumerator,
NULL,
CLSCTX_ALL,
&IID_IMMDeviceEnumerator,
&denum);
if (FAILED(hr)) {
goto fail;
}
if (deviceIn == WAVE_MAPPER) {
hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(denum,
eCapture,
eMultimedia,
&device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->inputEP);
IUnknown_Release(device);
}
if (FAILED(hr)) {
goto fail;
}
}
else {
res = waveInMessage((HWAVEIN)IntToPtr(deviceIn),
DRV_QUERYFUNCTIONINSTANCEIDSIZE,
(DWORD_PTR)&idLen,
(DWORD_PTR)NULL);
if (res != MMSYSERR_NOERROR) {
goto fail;
}
idStr = (WCHAR *) CoTaskMemAlloc(idLen + sizeof(WCHAR));
if (idStr == NULL) {
goto fail;
}
res = waveInMessage((HWAVEIN)IntToPtr(deviceIn),
DRV_QUERYFUNCTIONINSTANCEID,
(DWORD_PTR)idStr,
(DWORD_PTR)idLen);
if (res != MMSYSERR_NOERROR) {
CoTaskMemFree(idStr);
goto fail;
}
hr = IMMDeviceEnumerator_GetDevice(denum, idStr, &device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->inputEP);
IUnknown_Release(device);
}
CoTaskMemFree(idStr);
if (FAILED(hr)) {
goto fail;
}
}
if (deviceOut == WAVE_MAPPER) {
hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(denum,
eRender,
eMultimedia,
&device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->outputEP);
IUnknown_Release(device);
}
if (FAILED(hr)) {
goto fail;
}
}
else {
res = waveOutMessage((HWAVEOUT)IntToPtr(deviceOut),
DRV_QUERYFUNCTIONINSTANCEIDSIZE,
(DWORD_PTR)&idLen,
(DWORD_PTR)NULL);
if (res != MMSYSERR_NOERROR) {
goto fail;
}
idStr = (WCHAR *) CoTaskMemAlloc(idLen + sizeof(WCHAR));
if (idStr == NULL) {
goto fail;
}
res = waveOutMessage((HWAVEOUT)IntToPtr(deviceOut),
DRV_QUERYFUNCTIONINSTANCEID,
(DWORD_PTR)idStr,
(DWORD_PTR)idLen);
if (res != MMSYSERR_NOERROR) {
CoTaskMemFree(idStr);
goto fail;
}
hr = IMMDeviceEnumerator_GetDevice(denum, idStr, &device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->outputEP);
IUnknown_Release(device);
}
CoTaskMemFree(idStr);
if (FAILED(hr)) {
goto fail;
}
}
if (denum) {
IUnknown_Release(denum);
}
return TRUE;
fail:
if (denum) {
IUnknown_Release(denum);
}
return cleanup(Px);
}
int open_ep_mixers_byid(px_mixer *Px, wchar_t *deviceIn, wchar_t *deviceOut)
{
PxEPInfo *info;
IMMDeviceEnumerator *denum = NULL;
IMMDevice *device = NULL;
HRESULT hr;
if (!initialize(Px)) {
goto fail;
}
info = (PxEPInfo *) Px->info;
info->inputEP = NULL;
info->outputEP = NULL;
// Create an audio endpoint device enumerator.
hr = CoCreateInstance(&CLSID_MMDeviceEnumerator,
NULL,
CLSCTX_ALL,
&IID_IMMDeviceEnumerator,
&denum);
if (FAILED(hr)) {
goto fail;
}
if (deviceIn) {
hr = IMMDeviceEnumerator_GetDevice(denum, deviceIn, &device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->inputEP);
IUnknown_Release(device);
}
if (FAILED(hr)) {
goto fail;
}
}
if (deviceOut) {
hr = IMMDeviceEnumerator_GetDevice(denum, deviceOut, &device);
if (SUCCEEDED(hr)) {
hr = IMMDevice_Activate(device,
&IID_IAudioEndpointVolume,
CLSCTX_ALL,
NULL,
&info->outputEP);
IUnknown_Release(device);
}
if (FAILED(hr)) {
goto fail;
}
}
if (denum) {
IUnknown_Release(denum);
}
return TRUE;
fail:
if (denum) {
IUnknown_Release(denum);
}
return cleanup(Px);
}
static int initialize(px_mixer *Px)
{
Px->info = calloc(1, sizeof(PxEPInfo));
if (Px->info == NULL) {
return FALSE;
}
Px->CloseMixer = close_mixer;
// Px->GetNumMixers = get_num_mixers;
// Px->GetMixerName = get_mixer_name;
Px->GetMasterVolume = get_master_volume;
Px->SetMasterVolume = set_master_volume;
Px->SupportsPCMOutputVolume = supports_pcm_output_volume;
Px->GetPCMOutputVolume = get_pcm_output_volume;
Px->SetPCMOutputVolume = set_pcm_output_volume;
Px->GetNumOutputVolumes = get_num_output_volumes;
Px->GetOutputVolumeName = get_output_volume_name;
Px->GetOutputVolume = get_output_volume;
Px->SetOutputVolume = set_output_volume;
Px->GetNumInputSources = get_num_input_sources;
Px->GetInputSourceName = get_input_source_name;
Px->GetCurrentInputSource = get_current_input_source;
Px->SetCurrentInputSource = set_current_input_source;
Px->GetInputVolume = get_input_volume;
Px->SetInputVolume = set_input_volume;
// Px->SupportsOutputBalance = supports_output_balance;
// Px->GetOutputBalance = get_output_balance;
// Px->SetOutputBalance = set_output_balance;
// Px->SupportsPlaythrough = supports_play_through;
// Px->GetPlaythrough = get_play_through;
// Px->SetPlaythrough = set_play_through;
return TRUE;
}
static int cleanup(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
if (info) {
if (info->outputEP) {
IUnknown_Release(info->outputEP);
}
if (info->inputEP) {
IUnknown_Release(info->inputEP);
}
free(info);
Px->info = NULL;
}
return FALSE;
}
static void close_mixer(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
cleanup(Px);
}
static int get_num_mixers(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return 0;
}
static const char *get_mixer_name(px_mixer *Px, int i)
{
return NULL;
}
static PxVolume get_master_volume(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
float volume = 0.0;
if (info->outputEP) {
IAudioEndpointVolume_GetMasterVolumeLevelScalar(info->outputEP,
&volume);
}
return volume;
}
static void set_master_volume(px_mixer *Px, PxVolume volume)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
if (info->outputEP) {
IAudioEndpointVolume_SetMasterVolumeLevelScalar(info->outputEP,
volume,
NULL);
}
return;
}
/*
|| Main output volume
*/
static int supports_pcm_output_volume(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return TRUE;
}
static PxVolume get_pcm_output_volume(px_mixer *Px)
{
return get_master_volume(Px);
}
static void set_pcm_output_volume(px_mixer *Px, PxVolume volume)
{
set_master_volume(Px, volume);
return;
}
/*
|| Output info
*/
static int get_num_output_volumes(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return (info->outputEP ? 1 : 0);
}
static const char *get_output_volume_name(px_mixer *Px, int i)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return (info->outputEP ? "PCM" : NULL);
}
/*
|| Output volume
*/
static PxVolume get_output_volume(px_mixer *Px, int i)
{
return get_master_volume(Px);
}
static void set_output_volume(px_mixer *Px, int i, PxVolume volume)
{
set_master_volume(Px, volume);
return;
}
/*
|| Input source
*/
static int get_num_input_sources(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return (info->inputEP ? 1 : 0);
}
static const char *get_input_source_name(px_mixer *Px, int i)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
return (info->inputEP ? "Master" : NULL);
}
static int get_current_input_source(px_mixer *Px)
{
return 0;
}
static void set_current_input_source(px_mixer *Px, int i)
{
return;
}
/*
|| Input volume
*/
static PxVolume get_input_volume(px_mixer *Px)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
float volume = 0.0;
if (info->inputEP) {
IAudioEndpointVolume_GetMasterVolumeLevelScalar(info->inputEP,
&volume);
}
return volume;
}
static void set_input_volume(px_mixer *Px, PxVolume volume)
{
PxEPInfo *info = (PxEPInfo *)Px->info;
if (info->inputEP) {
IAudioEndpointVolume_SetMasterVolumeLevelScalar(info->inputEP,
volume,
NULL);
}
return;
}