mirror of
https://github.com/mpv-player/mpv
synced 2025-04-04 23:40:47 +00:00
ao_wasapi: Refactor device listing/loading
Omit "{0.0.0.00000000}." on devices that start with that substring, re-add when searching for devices by ID. Log the device ID of the default device. Log the friendly name of the used device. Consistently refer to endpoints/devices as devices, as this is more consistent with mpv terminology.
This commit is contained in:
parent
d5adaed9d8
commit
efc3668fbe
@ -564,6 +564,77 @@ exit_label:
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char* wstring_to_utf8(wchar_t *wstring) {
|
||||||
|
if (wstring) {
|
||||||
|
int len = WideCharToMultiByte(CP_UTF8, 0, wstring, -1, NULL, 0, NULL, NULL);
|
||||||
|
char *ret = malloc(len);
|
||||||
|
WideCharToMultiByte(CP_UTF8, 0, wstring, -1, ret, len, NULL, NULL);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char* get_device_id(IMMDevice *pDevice) {
|
||||||
|
LPWSTR devid = NULL;
|
||||||
|
char *idstr = NULL;
|
||||||
|
|
||||||
|
HRESULT hr = IMMDevice_GetId(pDevice, &devid);
|
||||||
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
|
idstr = wstring_to_utf8(devid);
|
||||||
|
|
||||||
|
if (strstr(idstr, "{0.0.0.00000000}.")) {
|
||||||
|
char *stripped = strdup(idstr + strlen("{0.0.0.00000000}."));
|
||||||
|
free(idstr);
|
||||||
|
idstr = stripped;
|
||||||
|
}
|
||||||
|
|
||||||
|
exit_label:
|
||||||
|
SAFE_RELEASE(devid, CoTaskMemFree(devid));
|
||||||
|
return idstr;
|
||||||
|
}
|
||||||
|
|
||||||
|
static char* get_device_name(IMMDevice *pDevice) {
|
||||||
|
IPropertyStore *pProps = NULL;
|
||||||
|
char *namestr = NULL;
|
||||||
|
|
||||||
|
HRESULT hr = IMMDevice_OpenPropertyStore(pDevice, STGM_READ, &pProps);
|
||||||
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
|
PROPVARIANT devname;
|
||||||
|
PropVariantInit(&devname);
|
||||||
|
|
||||||
|
hr = IPropertyStore_GetValue(pProps, &PKEY_Device_FriendlyName, &devname);
|
||||||
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
|
namestr = wstring_to_utf8(devname.pwszVal);
|
||||||
|
|
||||||
|
exit_label:
|
||||||
|
PropVariantClear(&devname);
|
||||||
|
SAFE_RELEASE(pProps, IPropertyStore_Release(pProps));
|
||||||
|
return namestr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// frees *idstr
|
||||||
|
static int device_id_match(char *idstr, char *candidate) {
|
||||||
|
if (idstr == NULL || candidate == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
int found = 0;
|
||||||
|
#define FOUND(x) do { found = (x); goto end; } while(0)
|
||||||
|
if (strcmp(idstr, candidate) == 0)
|
||||||
|
FOUND(1);
|
||||||
|
if (strstr(idstr, "{0.0.0.00000000}.")) {
|
||||||
|
char *start = idstr + strlen("{0.0.0.00000000}.");
|
||||||
|
if (strcmp(start, candidate) == 0)
|
||||||
|
FOUND(1);
|
||||||
|
}
|
||||||
|
#undef FOUND
|
||||||
|
end:
|
||||||
|
free(idstr);
|
||||||
|
return found;
|
||||||
|
}
|
||||||
|
|
||||||
static HRESULT enumerate_with_state(char *header, int status, int with_id) {
|
static HRESULT enumerate_with_state(char *header, int status, int with_id) {
|
||||||
HRESULT hr;
|
HRESULT hr;
|
||||||
IMMDeviceEnumerator *pEnumerator = NULL;
|
IMMDeviceEnumerator *pEnumerator = NULL;
|
||||||
@ -591,30 +662,19 @@ static HRESULT enumerate_with_state(char *header, int status, int with_id) {
|
|||||||
hr = IMMDeviceCollection_Item(pDevices, i, &pDevice);
|
hr = IMMDeviceCollection_Item(pDevices, i, &pDevice);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
hr = IMMDevice_GetId(pDevice, &idStr);
|
char *name = get_device_name(pDevice);
|
||||||
EXIT_ON_ERROR(hr)
|
char *id = get_device_id(pDevice);
|
||||||
|
|
||||||
hr = IMMDevice_OpenPropertyStore(pDevice, STGM_READ, &pProps);
|
|
||||||
EXIT_ON_ERROR(hr)
|
|
||||||
|
|
||||||
PROPVARIANT varName;
|
|
||||||
PropVariantInit(&varName);
|
|
||||||
|
|
||||||
hr = IPropertyStore_GetValue(pProps, &PKEY_Device_FriendlyName, &varName);
|
|
||||||
EXIT_ON_ERROR(hr)
|
|
||||||
|
|
||||||
if (with_id) {
|
if (with_id) {
|
||||||
mp_msg(MSGT_AO, MSGL_INFO, "ao-wasapi: Endpoint #%d: \"%S\" (ID \"%S\")\n",
|
mp_msg(MSGT_AO, MSGL_INFO, "ao-wasapi: Device #%d: %s, ID: %s\n",
|
||||||
i, varName.pwszVal, idStr);
|
i, name, id);
|
||||||
} else {
|
} else {
|
||||||
mp_msg(MSGT_AO, MSGL_INFO, "ao-wasapi: Endpoint: \"%S\" (ID \"%S\")\n",
|
mp_msg(MSGT_AO, MSGL_INFO, "ao-wasapi: %s, ID: %s\n",
|
||||||
varName.pwszVal, idStr);
|
name, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
CoTaskMemFree(idStr);
|
free(name);
|
||||||
idStr = NULL;
|
free(id);
|
||||||
PropVariantClear(&varName);
|
|
||||||
SAFE_RELEASE(pProps, IPropertyStore_Release(pProps));
|
|
||||||
SAFE_RELEASE(pDevice, IMMDevice_Release(pDevice));
|
SAFE_RELEASE(pDevice, IMMDevice_Release(pDevice));
|
||||||
}
|
}
|
||||||
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices));
|
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices));
|
||||||
@ -648,8 +708,6 @@ static HRESULT find_and_load_device(wasapi_state *state, int devno, char *devid)
|
|||||||
IMMDeviceCollection *pDevices = NULL;
|
IMMDeviceCollection *pDevices = NULL;
|
||||||
IMMDevice *pTempDevice = NULL;
|
IMMDevice *pTempDevice = NULL;
|
||||||
LPWSTR deviceID = NULL;
|
LPWSTR deviceID = NULL;
|
||||||
LPWSTR tmpID = NULL;
|
|
||||||
char *tmpStr = NULL;
|
|
||||||
|
|
||||||
hr = IMMDeviceEnumerator_EnumAudioEndpoints(state->pEnumerator, eRender,
|
hr = IMMDeviceEnumerator_EnumAudioEndpoints(state->pEnumerator, eRender,
|
||||||
DEVICE_STATE_ACTIVE, &pDevices);
|
DEVICE_STATE_ACTIVE, &pDevices);
|
||||||
@ -662,40 +720,32 @@ static HRESULT find_and_load_device(wasapi_state *state, int devno, char *devid)
|
|||||||
if (devno >= count) {
|
if (devno >= count) {
|
||||||
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: no endpoind #%d!\n", devno);
|
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: no endpoind #%d!\n", devno);
|
||||||
} else {
|
} else {
|
||||||
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: finding endpoint #%d\n", devno);
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: finding device #%d\n", devno);
|
||||||
hr = IMMDeviceCollection_Item(pDevices, devno, &pTempDevice);
|
hr = IMMDeviceCollection_Item(pDevices, devno, &pTempDevice);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
hr = IMMDevice_GetId(pTempDevice, &deviceID);
|
hr = IMMDevice_GetId(pTempDevice, &deviceID);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: found endpoint #%d\n", devno);
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: found device #%d\n", devno);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: finding endpoint \"%s\"\n", devid);
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: finding device %s\n", devid);
|
||||||
|
|
||||||
for (int i = 0; i < count; i++) {
|
for (int i = 0; i < count; i++) {
|
||||||
hr = IMMDeviceCollection_Item(pDevices, i, &pTempDevice);
|
hr = IMMDeviceCollection_Item(pDevices, i, &pTempDevice);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
hr = IMMDevice_GetId(pTempDevice, &tmpID);
|
if (device_id_match(get_device_id(pTempDevice), devid)) {
|
||||||
|
hr = IMMDevice_GetId(pTempDevice, &deviceID);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
int len = WideCharToMultiByte(CP_UTF8, 0, tmpID, -1, NULL, 0, NULL, NULL);
|
|
||||||
tmpStr = malloc(len);
|
|
||||||
WideCharToMultiByte(CP_UTF8, 0, tmpID, -1, tmpStr, len, NULL, NULL);
|
|
||||||
|
|
||||||
if (strcmp(devid, tmpStr) == 0) {
|
|
||||||
deviceID = tmpID;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
SAFE_RELEASE(tmpStr, free(tmpStr));
|
|
||||||
SAFE_RELEASE(tmpID, CoTaskMemFree(tmpID));
|
|
||||||
SAFE_RELEASE(pTempDevice, IMMDevice_Release(pTempDevice));
|
SAFE_RELEASE(pTempDevice, IMMDevice_Release(pTempDevice));
|
||||||
}
|
}
|
||||||
if (deviceID == NULL) {
|
if (deviceID == NULL) {
|
||||||
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: could not find endpoint \"%s\"!\n", devid);
|
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: could not find device %s!\n", devid);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -703,20 +753,18 @@ static HRESULT find_and_load_device(wasapi_state *state, int devno, char *devid)
|
|||||||
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices))
|
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices))
|
||||||
|
|
||||||
if (deviceID == NULL) {
|
if (deviceID == NULL) {
|
||||||
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: no endpoint to load!\n");
|
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: no device to load!\n");
|
||||||
} else {
|
} else {
|
||||||
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: loading endpoint %S\n", deviceID);
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: loading device %S\n", deviceID);
|
||||||
|
|
||||||
hr = IMMDeviceEnumerator_GetDevice(state->pEnumerator, deviceID, &state->pDevice);
|
hr = IMMDeviceEnumerator_GetDevice(state->pEnumerator, deviceID, &state->pDevice);
|
||||||
|
|
||||||
if (FAILED(hr)) {
|
if (FAILED(hr)) {
|
||||||
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: could not load requested endpoint!\n");
|
mp_msg(MSGT_AO, MSGL_ERR, "ao-wasapi: could not load requested device!\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
exit_label:
|
exit_label:
|
||||||
SAFE_RELEASE(tmpStr, free(tmpStr));
|
|
||||||
SAFE_RELEASE(tmpID, CoTaskMemFree(tmpID));
|
|
||||||
SAFE_RELEASE(pTempDevice, IMMDevice_Release(pTempDevice));
|
SAFE_RELEASE(pTempDevice, IMMDevice_Release(pTempDevice));
|
||||||
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices));
|
SAFE_RELEASE(pDevices, IMMDeviceCollection_Release(pDevices));
|
||||||
return hr;
|
return hr;
|
||||||
@ -732,10 +780,13 @@ static int thread_init(struct ao *ao)
|
|||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
if (!state->opt_device) {
|
if (!state->opt_device) {
|
||||||
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: using default endpoint\n");
|
|
||||||
hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(state->pEnumerator,
|
hr = IMMDeviceEnumerator_GetDefaultAudioEndpoint(state->pEnumerator,
|
||||||
eRender, eConsole,
|
eRender, eConsole,
|
||||||
&state->pDevice);
|
&state->pDevice);
|
||||||
|
|
||||||
|
char *id = get_device_id(state->pDevice);
|
||||||
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: default device ID: %s\n", id);
|
||||||
|
free(id);
|
||||||
} else {
|
} else {
|
||||||
int devno = -1;
|
int devno = -1;
|
||||||
char *devid = NULL;
|
char *devid = NULL;
|
||||||
@ -755,6 +806,10 @@ static int thread_init(struct ao *ao)
|
|||||||
}
|
}
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
|
||||||
|
char *name = get_device_name(state->pDevice);
|
||||||
|
mp_msg(MSGT_AO, MSGL_V, "ao-wasapi: device loaded: %s\n", name);
|
||||||
|
free(name);
|
||||||
|
|
||||||
hr = IMMDeviceActivator_Activate(state->pDevice, &IID_IAudioClient,
|
hr = IMMDeviceActivator_Activate(state->pDevice, &IID_IAudioClient,
|
||||||
CLSCTX_ALL, NULL, (void **)&state->pAudioClient);
|
CLSCTX_ALL, NULL, (void **)&state->pAudioClient);
|
||||||
EXIT_ON_ERROR(hr)
|
EXIT_ON_ERROR(hr)
|
||||||
|
Loading…
Reference in New Issue
Block a user