blob: 4e3158c5ce0e2a6d060cbf3b496873f28202e261 [file] [log] [blame]
Chris Morgancde7f902004-01-20 02:07:35 +00001/*
2 * Audio management UI code
3 *
4 * Copyright 2004 Chris Morgan
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
Jonathan Ernst360a3f92006-05-18 14:49:52 +020018 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
Chris Morgancde7f902004-01-20 02:07:35 +000019 *
20 */
21
Mike McCormack55303932006-01-19 11:55:01 +010022#define WIN32_LEAN_AND_MEAN
Mike McCormackb555e912005-12-08 11:57:44 +010023#define NONAMELESSSTRUCT
24#define NONAMELESSUNION
25
Chris Morgancde7f902004-01-20 02:07:35 +000026#include "config.h"
27#include "wine/port.h"
28
29#include <assert.h>
Chris Morgancde7f902004-01-20 02:07:35 +000030#include <stdlib.h>
31#include <stdio.h>
32#include <string.h>
33
Andrew Eikumdfe73cd2011-08-31 15:16:16 -050034#define COBJMACROS
Mike McCormack55303932006-01-19 11:55:01 +010035#include <windows.h>
Chris Morgancde7f902004-01-20 02:07:35 +000036#include <wine/debug.h>
37#include <shellapi.h>
38#include <objbase.h>
39#include <shlguid.h>
40#include <shlwapi.h>
41#include <shlobj.h>
Robert Reif88812842005-06-28 19:12:52 +000042#include <mmsystem.h>
Robert Reif1758ba62005-12-07 12:50:52 +010043#include <mmreg.h>
Robert Reif1758ba62005-12-07 12:50:52 +010044#include <mmddk.h>
Chris Morgancde7f902004-01-20 02:07:35 +000045
Andrew Eikumdfe73cd2011-08-31 15:16:16 -050046#include "ole2.h"
47#include "initguid.h"
48#include "devpkey.h"
49#include "mmdeviceapi.h"
50#include "audioclient.h"
51#include "audiopolicy.h"
52
Chris Morgancde7f902004-01-20 02:07:35 +000053#include "winecfg.h"
54#include "resource.h"
55
56WINE_DEFAULT_DEBUG_CHANNEL(winecfg);
57
Andrew Eikum58f2a3c2011-10-05 13:40:02 -050058struct DeviceInfo {
59 WCHAR *id;
60 PROPVARIANT name;
61};
62
63static WCHAR g_drv_keyW[256] = {'S','o','f','t','w','a','r','e','\\',
64 'W','i','n','e','\\','D','r','i','v','e','r','s','\\',0};
65
66static const WCHAR reg_out_nameW[] = {'D','e','f','a','u','l','t','O','u','t','p','u','t',0};
67static const WCHAR reg_in_nameW[] = {'D','e','f','a','u','l','t','I','n','p','u','t',0};
68static const WCHAR reg_vout_nameW[] = {'D','e','f','a','u','l','t','V','o','i','c','e','O','u','t','p','u','t',0};
69static const WCHAR reg_vin_nameW[] = {'D','e','f','a','u','l','t','V','o','i','c','e','I','n','p','u','t',0};
70
71static UINT num_render_devs, num_capture_devs;
72static struct DeviceInfo *render_devs, *capture_devs;
73
74static BOOL load_device(IMMDevice *dev, struct DeviceInfo *info)
Frank Richter3713f492006-08-19 05:19:57 +020075{
Andrew Eikum58f2a3c2011-10-05 13:40:02 -050076 IPropertyStore *ps;
77 HRESULT hr;
78
79 hr = IMMDevice_GetId(dev, &info->id);
80 if(FAILED(hr)){
81 info->id = NULL;
82 return FALSE;
83 }
84
85 hr = IMMDevice_OpenPropertyStore(dev, STGM_READ, &ps);
86 if(FAILED(hr)){
87 CoTaskMemFree(info->id);
88 info->id = NULL;
89 return FALSE;
90 }
91
92 PropVariantInit(&info->name);
93
94 hr = IPropertyStore_GetValue(ps,
95 (PROPERTYKEY*)&DEVPKEY_Device_FriendlyName, &info->name);
96 IPropertyStore_Release(ps);
97 if(FAILED(hr)){
98 CoTaskMemFree(info->id);
99 info->id = NULL;
100 return FALSE;
101 }
102
103 return TRUE;
104}
105
106static BOOL load_devices(IMMDeviceEnumerator *devenum, EDataFlow dataflow,
107 UINT *ndevs, struct DeviceInfo **out)
108{
109 IMMDeviceCollection *coll;
110 UINT i;
111 HRESULT hr;
112
113 hr = IMMDeviceEnumerator_EnumAudioEndpoints(devenum, dataflow,
114 DEVICE_STATE_ACTIVE, &coll);
115 if(FAILED(hr))
116 return FALSE;
117
118 hr = IMMDeviceCollection_GetCount(coll, ndevs);
119 if(FAILED(hr)){
120 IMMDeviceCollection_Release(coll);
121 return FALSE;
122 }
123
124 if(*ndevs > 0){
125 *out = HeapAlloc(GetProcessHeap(), 0,
126 sizeof(struct DeviceInfo) * (*ndevs));
127 if(!*out){
128 IMMDeviceCollection_Release(coll);
129 return FALSE;
130 }
131
132 for(i = 0; i < *ndevs; ++i){
133 IMMDevice *dev;
134
135 hr = IMMDeviceCollection_Item(coll, i, &dev);
136 if(FAILED(hr)){
137 (*out)[i].id = NULL;
138 continue;
139 }
140
141 load_device(dev, &(*out)[i]);
142
143 IMMDevice_Release(dev);
144 }
145 }else
146 *out = NULL;
147
148 IMMDeviceCollection_Release(coll);
149
150 return TRUE;
151}
152
153static BOOL get_driver_name(IMMDeviceEnumerator *devenum, PROPVARIANT *pv)
154{
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500155 IMMDevice *device;
156 IPropertyStore *ps;
157 HRESULT hr;
Raphael Junqueira8d9d1fb2005-07-11 10:24:28 +0000158
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500159 static const WCHAR wine_info_deviceW[] = {'W','i','n','e',' ',
160 'i','n','f','o',' ','d','e','v','i','c','e',0};
Robert Reifa76023f2006-04-05 20:53:02 -0400161
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500162 hr = IMMDeviceEnumerator_GetDevice(devenum, wine_info_deviceW, &device);
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500163 if(FAILED(hr))
164 return FALSE;
165
166 hr = IMMDevice_OpenPropertyStore(device, STGM_READ, &ps);
167 if(FAILED(hr)){
168 IMMDevice_Release(device);
169 return FALSE;
170 }
171
172 hr = IPropertyStore_GetValue(ps,
173 (const PROPERTYKEY *)&DEVPKEY_Device_Driver, pv);
174 IPropertyStore_Release(ps);
175 IMMDevice_Release(device);
176 if(FAILED(hr))
177 return FALSE;
178
179 return TRUE;
180}
Robert Reifa76023f2006-04-05 20:53:02 -0400181
Mike McCormackae511352005-06-02 15:11:32 +0000182static void initAudioDlg (HWND hDlg)
Chris Morgancde7f902004-01-20 02:07:35 +0000183{
Andrew Eikum58f2a3c2011-10-05 13:40:02 -0500184 WCHAR display_str[256], format_str[256], sysdefault_str[256], disabled_str[64];
185 IMMDeviceEnumerator *devenum;
186 BOOL have_driver = FALSE;
187 HRESULT hr;
Chris Morgancde7f902004-01-20 02:07:35 +0000188
Crestez Leonard3e55df32005-01-14 19:48:41 +0000189 WINE_TRACE("\n");
190
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100191 LoadStringW(GetModuleHandleW(NULL), IDS_AUDIO_DRIVER,
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500192 format_str, sizeof(format_str) / sizeof(*format_str));
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100193 LoadStringW(GetModuleHandleW(NULL), IDS_AUDIO_DRIVER_NONE,
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500194 disabled_str, sizeof(disabled_str) / sizeof(*disabled_str));
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100195 LoadStringW(GetModuleHandleW(NULL), IDS_AUDIO_SYSDEFAULT,
Andrew Eikum58f2a3c2011-10-05 13:40:02 -0500196 sysdefault_str, sizeof(sysdefault_str) / sizeof(*sysdefault_str));
Maarten Lankhorst58f67032009-08-05 15:16:55 +0200197
Andrew Eikum58f2a3c2011-10-05 13:40:02 -0500198 hr = CoCreateInstance(&CLSID_MMDeviceEnumerator, NULL,
199 CLSCTX_INPROC_SERVER, &IID_IMMDeviceEnumerator, (void**)&devenum);
200 if(SUCCEEDED(hr)){
201 PROPVARIANT pv;
202
203 load_devices(devenum, eRender, &num_render_devs, &render_devs);
204 load_devices(devenum, eCapture, &num_capture_devs, &capture_devs);
205
206 PropVariantInit(&pv);
207 if(get_driver_name(devenum, &pv) && pv.u.pwszVal[0] != '\0'){
208 have_driver = TRUE;
209 wnsprintfW(display_str, sizeof(display_str) / sizeof(*display_str),
210 format_str, pv.u.pwszVal);
211 lstrcatW(g_drv_keyW, pv.u.pwszVal);
212 }
213 PropVariantClear(&pv);
214
215 IMMDeviceEnumerator_Release(devenum);
216 }
217
218 SendDlgItemMessageW(hDlg, IDC_AUDIOOUT_DEVICE, CB_ADDSTRING,
219 0, (LPARAM)sysdefault_str);
220 SendDlgItemMessageW(hDlg, IDC_AUDIOOUT_DEVICE, CB_SETCURSEL, 0, 0);
221 SendDlgItemMessageW(hDlg, IDC_VOICEOUT_DEVICE, CB_ADDSTRING,
222 0, (LPARAM)sysdefault_str);
223 SendDlgItemMessageW(hDlg, IDC_VOICEOUT_DEVICE, CB_SETCURSEL, 0, 0);
224
225 SendDlgItemMessageW(hDlg, IDC_AUDIOIN_DEVICE, CB_ADDSTRING,
226 0, (LPARAM)sysdefault_str);
227 SendDlgItemMessageW(hDlg, IDC_AUDIOIN_DEVICE, CB_SETCURSEL, 0, 0);
228 SendDlgItemMessageW(hDlg, IDC_VOICEIN_DEVICE, CB_ADDSTRING,
229 0, (LPARAM)sysdefault_str);
230 SendDlgItemMessageW(hDlg, IDC_VOICEIN_DEVICE, CB_SETCURSEL, 0, 0);
231
232 if(have_driver){
233 WCHAR *reg_out_dev, *reg_vout_dev, *reg_in_dev, *reg_vin_dev;
234 UINT i;
235
236 reg_out_dev = get_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, reg_out_nameW, NULL);
237 reg_vout_dev = get_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, reg_vout_nameW, NULL);
238 reg_in_dev = get_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, reg_in_nameW, NULL);
239 reg_vin_dev = get_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, reg_vin_nameW, NULL);
240
241 for(i = 0; i < num_render_devs; ++i){
242 if(!render_devs[i].id)
243 continue;
244
245 SendDlgItemMessageW(hDlg, IDC_AUDIOOUT_DEVICE, CB_ADDSTRING,
246 0, (LPARAM)render_devs[i].name.u.pwszVal);
247 SendDlgItemMessageW(hDlg, IDC_AUDIOOUT_DEVICE, CB_SETITEMDATA,
248 i + 1, (LPARAM)&render_devs[i]);
249 if(reg_out_dev && !lstrcmpW(render_devs[i].id, reg_out_dev))
250 SendDlgItemMessageW(hDlg, IDC_AUDIOOUT_DEVICE, CB_SETCURSEL, i + 1, 0);
251
252 SendDlgItemMessageW(hDlg, IDC_VOICEOUT_DEVICE, CB_ADDSTRING,
253 0, (LPARAM)render_devs[i].name.u.pwszVal);
254 SendDlgItemMessageW(hDlg, IDC_VOICEOUT_DEVICE, CB_SETITEMDATA,
255 i + 1, (LPARAM)&render_devs[i]);
256 if(reg_vout_dev && !lstrcmpW(render_devs[i].id, reg_vout_dev))
257 SendDlgItemMessageW(hDlg, IDC_VOICEOUT_DEVICE, CB_SETCURSEL, i + 1, 0);
258 }
259
260 for(i = 0; i < num_capture_devs; ++i){
261 if(!capture_devs[i].id)
262 continue;
263
264 SendDlgItemMessageW(hDlg, IDC_AUDIOIN_DEVICE, CB_ADDSTRING,
265 0, (LPARAM)capture_devs[i].name.u.pwszVal);
266 SendDlgItemMessageW(hDlg, IDC_AUDIOIN_DEVICE, CB_SETITEMDATA,
267 i + 1, (LPARAM)&capture_devs[i]);
268 if(reg_in_dev && !lstrcmpW(capture_devs[i].id, reg_in_dev))
269 SendDlgItemMessageW(hDlg, IDC_AUDIOIN_DEVICE, CB_SETCURSEL, i + 1, 0);
270
271 SendDlgItemMessageW(hDlg, IDC_VOICEIN_DEVICE, CB_ADDSTRING,
272 0, (LPARAM)capture_devs[i].name.u.pwszVal);
273 SendDlgItemMessageW(hDlg, IDC_VOICEIN_DEVICE, CB_SETITEMDATA,
274 i + 1, (LPARAM)&capture_devs[i]);
275 if(reg_vin_dev && !lstrcmpW(capture_devs[i].id, reg_vin_dev))
276 SendDlgItemMessageW(hDlg, IDC_VOICEIN_DEVICE, CB_SETCURSEL, i + 1, 0);
277 }
278
279 HeapFree(GetProcessHeap(), 0, reg_out_dev);
280 HeapFree(GetProcessHeap(), 0, reg_vout_dev);
281 HeapFree(GetProcessHeap(), 0, reg_in_dev);
282 HeapFree(GetProcessHeap(), 0, reg_vin_dev);
283 }else
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500284 wnsprintfW(display_str, sizeof(display_str) / sizeof(*display_str),
285 format_str, disabled_str);
Maarten Lankhorst58f67032009-08-05 15:16:55 +0200286
Andrew Eikum4b8a2962011-09-27 08:49:34 -0500287 SetDlgItemTextW(hDlg, IDC_AUDIO_DRIVER, display_str);
Chris Morgancde7f902004-01-20 02:07:35 +0000288}
289
Andrew Eikum58f2a3c2011-10-05 13:40:02 -0500290static void set_reg_device(HWND hDlg, int dlgitem, const WCHAR *key_name)
291{
292 UINT idx;
293 struct DeviceInfo *info;
294
295 idx = SendDlgItemMessageW(hDlg, dlgitem, CB_GETCURSEL, 0, 0);
296
297 info = (struct DeviceInfo *)SendDlgItemMessageW(hDlg, dlgitem,
298 CB_GETITEMDATA, idx, 0);
299
300 if(!info || info == (void*)CB_ERR)
301 set_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, key_name, NULL);
302 else
303 set_reg_keyW(HKEY_CURRENT_USER, g_drv_keyW, key_name, info->id);
304}
305
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500306static void test_sound(void)
307{
Andrew Eikumb6169512011-11-01 10:18:13 -0500308 if(!PlaySoundW(MAKEINTRESOURCEW(IDW_TESTSOUND), NULL, SND_RESOURCE | SND_ASYNC)){
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500309 WCHAR error_str[256], title_str[256];
310
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100311 LoadStringW(GetModuleHandleW(NULL), IDS_AUDIO_TEST_FAILED,
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500312 error_str, sizeof(error_str) / sizeof(*error_str));
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100313 LoadStringW(GetModuleHandleW(NULL), IDS_AUDIO_TEST_FAILED_TITLE,
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500314 title_str, sizeof(title_str) / sizeof(*title_str));
315
316 MessageBoxW(NULL, error_str, title_str, MB_OK | MB_ICONERROR);
317 }
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500318}
319
Chris Morgancde7f902004-01-20 02:07:35 +0000320INT_PTR CALLBACK
321AudioDlgProc (HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
322{
323 switch (uMsg) {
324 case WM_COMMAND:
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500325 switch (LOWORD(wParam)) {
Steven Edwards8c62cbf2007-04-03 16:09:02 -0400326 case IDC_AUDIO_TEST:
Andrew Eikumb07a82c2011-10-05 13:39:46 -0500327 test_sound();
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500328 break;
Andrew Eikum58f2a3c2011-10-05 13:40:02 -0500329 case IDC_AUDIOOUT_DEVICE:
330 if(HIWORD(wParam) == CBN_SELCHANGE){
331 set_reg_device(hDlg, IDC_AUDIOOUT_DEVICE, reg_out_nameW);
332 SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
333 }
334 break;
335 case IDC_VOICEOUT_DEVICE:
336 if(HIWORD(wParam) == CBN_SELCHANGE){
337 set_reg_device(hDlg, IDC_VOICEOUT_DEVICE, reg_vout_nameW);
338 SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
339 }
340 break;
341 case IDC_AUDIOIN_DEVICE:
342 if(HIWORD(wParam) == CBN_SELCHANGE){
343 set_reg_device(hDlg, IDC_AUDIOIN_DEVICE, reg_in_nameW);
344 SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
345 }
346 break;
347 case IDC_VOICEIN_DEVICE:
348 if(HIWORD(wParam) == CBN_SELCHANGE){
349 set_reg_device(hDlg, IDC_VOICEIN_DEVICE, reg_vin_nameW);
350 SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
351 }
352 break;
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500353 }
Marcus Meissner90c87982011-10-16 13:04:35 +0200354 break;
355
Mike Hearn0af614e2004-09-28 03:55:16 +0000356 case WM_SHOWWINDOW:
357 set_window_title(hDlg);
358 break;
Robert Reiff0c55e72005-12-17 12:30:06 +0100359
Chris Morgancde7f902004-01-20 02:07:35 +0000360 case WM_NOTIFY:
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500361 switch(((LPNMHDR)lParam)->code) {
362 case PSN_KILLACTIVE:
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100363 SetWindowLongPtrW(hDlg, DWLP_MSGRESULT, FALSE);
Robert Reiff0c55e72005-12-17 12:30:06 +0100364 break;
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500365 case PSN_APPLY:
366 apply();
Alexandre Julliardedf44bf2012-01-20 12:42:33 +0100367 SetWindowLongPtrW(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
Andrew Eikumdfe73cd2011-08-31 15:16:16 -0500368 break;
369 case PSN_SETACTIVE:
370 break;
371 }
372 break;
373 case WM_INITDIALOG:
374 initAudioDlg(hDlg);
375 break;
Chris Morgancde7f902004-01-20 02:07:35 +0000376 }
377
378 return FALSE;
379}