source: trunk/src/msacm32/format.c

Last change on this file was 6712, checked in by sandervl, 24 years ago

restored old version

File size: 23.7 KB
RevLine 
[5434]1/* -*- tab-width: 8; c-basic-offset: 4 -*- */
[6712]2
[5434]3/*
4 * MSACM32 library
5 *
6 * Copyright 1998 Patrik Stridvall
[6712]7 * 2000 Eric Pouech
[5434]8 */
9
10#include <string.h>
11#include "winbase.h"
12#include "winnls.h"
13#include "winerror.h"
14#include "windef.h"
15#include "wingdi.h"
16#include "winuser.h"
17#include "wine/unicode.h"
18#include "debugtools.h"
19#include "mmsystem.h"
20#include "msacm.h"
21#include "msacmdrv.h"
22#include "wineacm.h"
23
24DEFAULT_DEBUG_CHANNEL(msacm);
25
[6712]26static PACMFORMATCHOOSEA afc;
[5434]27
28struct MSACM_FillFormatData {
[6712]29 HWND hWnd;
30#define WINE_ACMFF_TAG 0
31#define WINE_ACMFF_FORMAT 1
32#define WINE_ACMFF_WFX 2
33 int mode;
34 char szFormatTag[ACMFORMATTAGDETAILS_FORMATTAG_CHARS];
35 PACMFORMATCHOOSEA afc;
36 DWORD ret;
[5434]37};
38
[6712]39static BOOL CALLBACK MSACM_FillFormatTagsCB(HACMDRIVERID hadid,
40 PACMFORMATTAGDETAILSA paftd,
41 DWORD dwInstance, DWORD fdwSupport)
[5434]42{
[6712]43 struct MSACM_FillFormatData* affd = (struct MSACM_FillFormatData*)dwInstance;
[5434]44
45 switch (affd->mode) {
46 case WINE_ACMFF_TAG:
[6712]47 if (SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
48 CB_FINDSTRINGEXACT,
49 (WPARAM)-1, (LPARAM)paftd->szFormatTag) == CB_ERR)
50 SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
51 CB_ADDSTRING, 0, (DWORD)paftd->szFormatTag);
52 break;
[5434]53 case WINE_ACMFF_FORMAT:
[6712]54 if (strcmp(affd->szFormatTag, paftd->szFormatTag) == 0) {
55 HACMDRIVER had;
[5434]56
[6712]57 if (acmDriverOpen(&had, hadid, 0) == MMSYSERR_NOERROR) {
58 ACMFORMATDETAILSA afd;
59 int i, idx;
60 MMRESULT mmr;
61 char buffer[64];
[5434]62
[6712]63 afd.cbStruct = sizeof(afd);
64 afd.dwFormatTag = paftd->dwFormatTag;
65 afd.pwfx = HeapAlloc(GetProcessHeap(), 0, paftd->cbFormatSize);
66 afd.pwfx->wFormatTag = paftd->dwFormatTag;
67 afd.pwfx->cbSize = paftd->cbFormatSize;
68 afd.cbwfx = paftd->cbFormatSize;
[5434]69
[6712]70 for (i = 0; i < paftd->cStandardFormats; i++) {
71 afd.dwFormatIndex = i;
72 mmr = acmFormatDetailsA(had, &afd, ACM_FORMATDETAILSF_INDEX);
73 if (mmr == MMSYSERR_NOERROR) {
74 strcpy(buffer, afd.szFormat);
75 for (idx = strlen(buffer);
76 idx < ACMFORMATTAGDETAILS_FORMATTAG_CHARS; idx++)
77 buffer[idx] = ' ';
78 wsprintfA(buffer + ACMFORMATTAGDETAILS_FORMATTAG_CHARS,
79 "%d Ko/s",
80 (afd.pwfx->nAvgBytesPerSec + 512) / 1024);
81 SendDlgItemMessageA(affd->hWnd,
82 IDD_ACMFORMATCHOOSE_CMB_FORMAT,
83 CB_ADDSTRING, 0, (DWORD)buffer);
84 }
85 }
86 acmDriverClose(had, 0);
87 SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT,
88 CB_SETCURSEL, 0, 0);
89 }
90 }
91 break;
[5434]92 case WINE_ACMFF_WFX:
[6712]93 if (strcmp(affd->szFormatTag, paftd->szFormatTag) == 0) {
94 HACMDRIVER had;
[5434]95
[6712]96 if (acmDriverOpen(&had, hadid, 0) == MMSYSERR_NOERROR) {
97 ACMFORMATDETAILSA afd;
[5434]98
[6712]99 afd.cbStruct = sizeof(afd);
100 afd.dwFormatTag = paftd->dwFormatTag;
101 afd.pwfx = affd->afc->pwfx;
102 afd.cbwfx = affd->afc->cbwfx;
[5434]103
[6712]104 afd.dwFormatIndex = SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT, CB_GETCURSEL, 0, 0);;
105 affd->ret = acmFormatDetailsA(had, &afd, ACM_FORMATDETAILSF_INDEX);
106 acmDriverClose(had, 0);
107 return TRUE;
108 }
109 }
110 break;
[5434]111 default:
[6712]112 FIXME("Unknown mode (%d)\n", affd->mode);
113 break;
[5434]114 }
115 return TRUE;
116}
117
118static BOOL MSACM_FillFormatTags(HWND hWnd)
119{
[6712]120 ACMFORMATTAGDETAILSA aftd;
121 struct MSACM_FillFormatData affd;
[5434]122
123 memset(&aftd, 0, sizeof(aftd));
124 aftd.cbStruct = sizeof(aftd);
125
126 affd.hWnd = hWnd;
127 affd.mode = WINE_ACMFF_TAG;
128
129 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
130 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG, CB_SETCURSEL, 0, 0);
131 return TRUE;
132}
133
134static BOOL MSACM_FillFormat(HWND hWnd)
135{
[6712]136 ACMFORMATTAGDETAILSA aftd;
137 struct MSACM_FillFormatData affd;
[5434]138
139 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT, CB_RESETCONTENT, 0, 0);
140
141 memset(&aftd, 0, sizeof(aftd));
142 aftd.cbStruct = sizeof(aftd);
143
144 affd.hWnd = hWnd;
145 affd.mode = WINE_ACMFF_FORMAT;
[6712]146 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
147 CB_GETLBTEXT,
148 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
149 CB_GETCURSEL, 0, 0),
150 (DWORD)affd.szFormatTag);
151
[5434]152 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
153 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT, CB_SETCURSEL, 0, 0);
154 return TRUE;
155}
156
157static MMRESULT MSACM_GetWFX(HWND hWnd, PACMFORMATCHOOSEA afc)
158{
[6712]159 ACMFORMATTAGDETAILSA aftd;
160 struct MSACM_FillFormatData affd;
[5434]161
162 memset(&aftd, 0, sizeof(aftd));
163 aftd.cbStruct = sizeof(aftd);
164
165 affd.hWnd = hWnd;
166 affd.mode = WINE_ACMFF_WFX;
167 affd.afc = afc;
168 affd.ret = MMSYSERR_NOERROR;
169
170 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
171 return affd.ret;
172}
173
[6712]174static BOOL WINAPI FormatChooseDlgProc(HWND hWnd, UINT msg,
175 WPARAM wParam, LPARAM lParam)
[5434]176{
[6712]177
[5434]178 TRACE("hwnd=%i msg=%i 0x%08x 0x%08lx\n", hWnd, msg, wParam, lParam );
[6712]179
[5434]180 switch (msg) {
181 case WM_INITDIALOG:
[6712]182 afc = (PACMFORMATCHOOSEA)lParam;
183 MSACM_FillFormatTags(hWnd);
184 MSACM_FillFormat(hWnd);
185 if ((afc->fdwStyle & ~(ACMFORMATCHOOSE_STYLEF_CONTEXTHELP|
186 ACMFORMATCHOOSE_STYLEF_SHOWHELP)) != 0)
187 FIXME("Unsupported style %08lx\n", ((PACMFORMATCHOOSEA)lParam)->fdwStyle);
188 if (!(afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_SHOWHELP))
189 ShowWindow(GetDlgItem(hWnd, IDD_ACMFORMATCHOOSE_BTN_HELP), SW_HIDE);
190 return TRUE;
191
[5434]192 case WM_COMMAND:
[6712]193 switch (LOWORD(wParam)) {
194 case IDOK:
195 EndDialog(hWnd, MSACM_GetWFX(hWnd, afc));
196 return TRUE;
197 case IDCANCEL:
198 EndDialog(hWnd, ACMERR_CANCELED);
199 return TRUE;
200 case IDD_ACMFORMATCHOOSE_CMB_FORMATTAG:
201 switch (HIWORD(wParam)) {
202 case CBN_SELCHANGE:
203 MSACM_FillFormat(hWnd);
204 break;
205 default:
206 TRACE("Dropped dlgNotif (fmtTag): 0x%08x 0x%08lx\n",
207 HIWORD(wParam), lParam);
208 break;
209 }
210 break;
211 case IDD_ACMFORMATCHOOSE_BTN_HELP:
212 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_SHOWHELP)
213 SendMessageA(afc->hwndOwner,
214 RegisterWindowMessageA(ACMHELPMSGSTRINGA), 0L, 0L);
215 break;
216
217 default:
218 TRACE("Dropped dlgCmd: ctl=%d ntf=0x%04x 0x%08lx\n",
219 LOWORD(wParam), HIWORD(wParam), lParam);
220 break;
221 }
222 break;
[5434]223 case WM_CONTEXTMENU:
[6712]224 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_CONTEXTHELP)
225 SendMessageA(afc->hwndOwner,
226 RegisterWindowMessageA(ACMHELPMSGCONTEXTMENUA),
227 wParam, lParam);
228 break;
[5434]229#if defined(WM_CONTEXTHELP)
230 case WM_CONTEXTHELP:
[6712]231 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_CONTEXTHELP)
232 SendMessageA(afc->hwndOwner,
233 RegisterWindowMessageA(ACMHELPMSGCONTEXTHELPA),
234 wParam, lParam);
235 break;
236#endif
[5434]237 default:
[6712]238 TRACE("Dropped dlgMsg: hwnd=%i msg=%i 0x%08x 0x%08lx\n",
239 hWnd, msg, wParam, lParam );
240 break;
[5434]241 }
242 return FALSE;
243}
244
245/***********************************************************************
246 * acmFormatChooseA (MSACM32.23)
247 */
248MMRESULT WINAPI acmFormatChooseA(PACMFORMATCHOOSEA pafmtc)
249{
250 return DialogBoxParamA(MSACM_hInstance32, MAKEINTRESOURCEA(DLG_ACMFORMATCHOOSE_ID),
[6712]251 pafmtc->hwndOwner, FormatChooseDlgProc, (INT)pafmtc);
[5434]252}
253
254/***********************************************************************
255 * acmFormatChooseW (MSACM32.24)
256 */
257MMRESULT WINAPI acmFormatChooseW(PACMFORMATCHOOSEW pafmtc)
258{
259 FIXME("(%p): stub\n", pafmtc);
260 SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
261 return MMSYSERR_ERROR;
262}
263
264/***********************************************************************
265 * acmFormatDetailsA (MSACM32.25)
266 */
[6712]267MMRESULT WINAPI acmFormatDetailsA(HACMDRIVER had, PACMFORMATDETAILSA pafd,
268 DWORD fdwDetails)
[5434]269{
[6712]270 ACMFORMATDETAILSW afdw;
271 MMRESULT mmr;
[5434]272
273 memset(&afdw, 0, sizeof(afdw));
274 afdw.cbStruct = sizeof(afdw);
275 afdw.dwFormatIndex = pafd->dwFormatIndex;
[6712]276 afdw.dwFormatTag = pafd->dwFormatTag;
277 afdw.pwfx = pafd->pwfx;
278 afdw.cbwfx = pafd->cbwfx;
[5434]279
280 mmr = acmFormatDetailsW(had, &afdw, fdwDetails);
281 if (mmr == MMSYSERR_NOERROR) {
[6712]282 pafd->dwFormatTag = afdw.dwFormatTag;
283 pafd->fdwSupport = afdw.fdwSupport;
[5434]284 WideCharToMultiByte( CP_ACP, 0, afdw.szFormat, -1,
285 pafd->szFormat, sizeof(pafd->szFormat), NULL, NULL );
286 }
287 return mmr;
288}
289
290/***********************************************************************
291 * acmFormatDetailsW (MSACM32.26)
292 */
[6712]293MMRESULT WINAPI acmFormatDetailsW(HACMDRIVER had, PACMFORMATDETAILSW pafd,
294 DWORD fdwDetails)
[5434]295{
[6712]296 MMRESULT mmr;
297 static WCHAR fmt1[] = {'%','d',' ','H','z',0};
298 static WCHAR fmt2[] = {';',' ','%','d',' ','b','i','t','s',0};
299 ACMFORMATTAGDETAILSA aftd;
[5434]300
301 TRACE("(0x%08x, %p, %ld)\n", had, pafd, fdwDetails);
302
303 memset(&aftd, 0, sizeof(aftd));
304 aftd.cbStruct = sizeof(aftd);
305
306 if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM;
[6712]307
[5434]308 switch (fdwDetails) {
309 case ACM_FORMATDETAILSF_FORMAT:
[6712]310 if (pafd->dwFormatTag != pafd->pwfx->wFormatTag) {
311 mmr = MMSYSERR_INVALPARAM;
312 break;
313 }
314 if (had == (HACMDRIVER)NULL) {
315 PWINE_ACMDRIVERID padid;
[5434]316
[6712]317 mmr = ACMERR_NOTPOSSIBLE;
318 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
319 /* should check for codec only */
320 if (padid->bEnabled &&
321 acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == 0) {
322 mmr = MSACM_Message(had, ACMDM_FORMAT_DETAILS,
323 (LPARAM)pafd, (LPARAM)fdwDetails);
324 acmDriverClose(had, 0);
325 if (mmr == MMSYSERR_NOERROR) break;
326 }
327 }
328 } else {
329 mmr = MSACM_Message(had, ACMDM_FORMAT_DETAILS,
330 (LPARAM)pafd, (LPARAM)fdwDetails);
331 }
332 break;
[5434]333 case ACM_FORMATDETAILSF_INDEX:
[6712]334 /* should check pafd->dwFormatIndex < aftd->cStandardFormats */
335 mmr = MSACM_Message(had, ACMDM_FORMAT_DETAILS,
336 (LPARAM)pafd, (LPARAM)fdwDetails);
337 break;
[5434]338 default:
[6712]339 WARN("Unknown fdwDetails %08lx\n", fdwDetails);
340 mmr = MMSYSERR_INVALFLAG;
341 break;
[5434]342 }
343
344 if (mmr == MMSYSERR_NOERROR && pafd->szFormat[0] == (WCHAR)0) {
[6712]345 wsprintfW(pafd->szFormat, fmt1, pafd->pwfx->nSamplesPerSec);
346 if (pafd->pwfx->wBitsPerSample) {
347 wsprintfW(pafd->szFormat + lstrlenW(pafd->szFormat), fmt2,
348 pafd->pwfx->wBitsPerSample);
349 }
[5434]350 MultiByteToWideChar( CP_ACP, 0, (pafd->pwfx->nChannels == 1) ? "; Mono" : "; Stereo", -1,
351 pafd->szFormat + strlenW(pafd->szFormat),
352 sizeof(pafd->szFormat)/sizeof(WCHAR) - strlenW(pafd->szFormat) );
353 }
354
355 TRACE("=> %d\n", mmr);
356 return mmr;
357}
358
359struct MSACM_FormatEnumWtoA_Instance {
[6712]360 PACMFORMATDETAILSA pafda;
361 DWORD dwInstance;
362 ACMFORMATENUMCBA fnCallback;
[5434]363};
364
365static BOOL CALLBACK MSACM_FormatEnumCallbackWtoA(HACMDRIVERID hadid,
[6712]366 PACMFORMATDETAILSW pafdw,
367 DWORD dwInstance,
368 DWORD fdwSupport)
[5434]369{
370 struct MSACM_FormatEnumWtoA_Instance* pafei;
371
372 pafei = (struct MSACM_FormatEnumWtoA_Instance*)dwInstance;
373
[6712]374 pafei->pafda->dwFormatIndex = pafdw->dwFormatIndex;
375 pafei->pafda->dwFormatTag = pafdw->dwFormatTag;
376 pafei->pafda->fdwSupport = pafdw->fdwSupport;
[5434]377 WideCharToMultiByte( CP_ACP, 0, pafdw->szFormat, -1,
378 pafei->pafda->szFormat, sizeof(pafei->pafda->szFormat), NULL, NULL );
379
[6712]380 return (pafei->fnCallback)(hadid, pafei->pafda,
381 pafei->dwInstance, fdwSupport);
[5434]382}
383
384/***********************************************************************
385 * acmFormatEnumA (MSACM32.27)
386 */
387MMRESULT WINAPI acmFormatEnumA(HACMDRIVER had, PACMFORMATDETAILSA pafda,
[6712]388 ACMFORMATENUMCBA fnCallback, DWORD dwInstance,
389 DWORD fdwEnum)
[5434]390{
[6712]391 ACMFORMATDETAILSW afdw;
[5434]392 struct MSACM_FormatEnumWtoA_Instance afei;
393
394 memset(&afdw, 0, sizeof(afdw));
395 afdw.cbStruct = sizeof(afdw);
396 afdw.dwFormatIndex = pafda->dwFormatIndex;
397 afdw.dwFormatTag = pafda->dwFormatTag;
398 afdw.pwfx = pafda->pwfx;
399 afdw.cbwfx = pafda->cbwfx;
400
401 afei.pafda = pafda;
402 afei.dwInstance = dwInstance;
403 afei.fnCallback = fnCallback;
404
[6712]405 return acmFormatEnumW(had, &afdw, MSACM_FormatEnumCallbackWtoA,
406 (DWORD)&afei, fdwEnum);
[5434]407}
408
409/***********************************************************************
410 * acmFormatEnumW (MSACM32.28)
411 */
[6712]412static BOOL MSACM_FormatEnumHelper(PWINE_ACMDRIVERID padid, HACMDRIVER had,
413 PACMFORMATDETAILSW pafd, PWAVEFORMATEX pwfxRef,
414 ACMFORMATENUMCBW fnCallback, DWORD dwInstance,
415 DWORD fdwEnum)
[5434]416{
[6712]417 ACMDRIVERDETAILSW add;
418 ACMFORMATTAGDETAILSW aftd;
419 int i, j;
[5434]420
421 add.cbStruct = sizeof(add);
[6712]422
[5434]423 if (acmDriverDetailsW((HACMDRIVERID)padid, &add, 0) != MMSYSERR_NOERROR) return FALSE;
424
425 for (i = 0; i < add.cFormatTags; i++) {
[6712]426 memset(&aftd, 0, sizeof(aftd));
427 aftd.cbStruct = sizeof(aftd);
428 aftd.dwFormatTagIndex = i;
429 if (acmFormatTagDetailsW(had, &aftd, ACM_FORMATTAGDETAILSF_INDEX) != MMSYSERR_NOERROR)
430 continue;
431
432 if ((fdwEnum & ACM_FORMATENUMF_WFORMATTAG) && aftd.dwFormatTag != pwfxRef->wFormatTag)
433 continue;
434
435 for (j = 0; j < aftd.cStandardFormats; j++) {
436 pafd->dwFormatIndex = j;
437 pafd->dwFormatTag = aftd.dwFormatTag;
438 if (acmFormatDetailsW(had, pafd, ACM_FORMATDETAILSF_INDEX) != MMSYSERR_NOERROR)
439 continue;
440
441 if ((fdwEnum & ACM_FORMATENUMF_NCHANNELS) &&
442 pafd->pwfx->nChannels != pwfxRef->nChannels)
443 continue;
444 if ((fdwEnum & ACM_FORMATENUMF_NSAMPLESPERSEC) &&
445 pafd->pwfx->nSamplesPerSec != pwfxRef->nSamplesPerSec)
446 continue;
447 if ((fdwEnum & ACM_FORMATENUMF_WBITSPERSAMPLE) &&
448 pafd->pwfx->wBitsPerSample != pwfxRef->wBitsPerSample)
449 continue;
450 if ((fdwEnum & ACM_FORMATENUMF_HARDWARE) &&
451 !(pafd->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_HARDWARE))
452 continue;
453
454 /* more checks to be done on fdwEnum */
[5434]455
[6712]456 if (!(fnCallback)((HACMDRIVERID)padid, pafd, dwInstance, add.fdwSupport))
457 return FALSE;
458 }
459 /* the "formats" used by the filters are also reported */
[5434]460 }
461 return TRUE;
462}
463
464/**********************************************************************/
465
466MMRESULT WINAPI acmFormatEnumW(HACMDRIVER had, PACMFORMATDETAILSW pafd,
[6712]467 ACMFORMATENUMCBW fnCallback, DWORD dwInstance,
468 DWORD fdwEnum)
[5434]469{
[6712]470 PWINE_ACMDRIVERID padid;
471 WAVEFORMATEX wfxRef;
472 BOOL ret;
[5434]473
474 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
[6712]475 had, pafd, fnCallback, dwInstance, fdwEnum);
[5434]476
477 if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM;
478
479 if (fdwEnum & (ACM_FORMATENUMF_WFORMATTAG|ACM_FORMATENUMF_NCHANNELS|
[6712]480 ACM_FORMATENUMF_NSAMPLESPERSEC|ACM_FORMATENUMF_WBITSPERSAMPLE|
481 ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST))
[5434]482 wfxRef = *pafd->pwfx;
483
[6712]484 if ((fdwEnum & ACM_FORMATENUMF_HARDWARE) &&
485 !(fdwEnum & (ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT)))
486 return MMSYSERR_INVALPARAM;
[5434]487
488 if ((fdwEnum & ACM_FORMATENUMF_WFORMATTAG) &&
[6712]489 (pafd->dwFormatTag != pafd->pwfx->wFormatTag))
490 return MMSYSERR_INVALPARAM;
[5434]491
492 if (fdwEnum & (ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST|
[6712]493 ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT))
494 FIXME("Unsupported fdwEnum values %08lx\n", fdwEnum);
[5434]495
496 if (had) {
[6712]497 HACMDRIVERID hadid;
[5434]498
[6712]499 if (acmDriverID(had, &hadid, 0) != MMSYSERR_NOERROR)
500 return MMSYSERR_INVALHANDLE;
501 MSACM_FormatEnumHelper(MSACM_GetDriverID(hadid), had, pafd, &wfxRef,
502 fnCallback, dwInstance, fdwEnum);
503 return MMSYSERR_NOERROR;
[5434]504 }
505 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
[6712]506 /* should check for codec only */
507 if (!padid->bEnabled || acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != MMSYSERR_NOERROR)
508 continue;
509 ret = MSACM_FormatEnumHelper(padid, had, pafd, &wfxRef,
510 fnCallback, dwInstance, fdwEnum);
511 acmDriverClose(had, 0);
512 if (!ret) break;
[5434]513 }
514 return MMSYSERR_NOERROR;
515}
516
517/***********************************************************************
518 * acmFormatSuggest (MSACM32.29)
519 */
[6712]520MMRESULT WINAPI acmFormatSuggest(HACMDRIVER had, PWAVEFORMATEX pwfxSrc,
521 PWAVEFORMATEX pwfxDst, DWORD cbwfxDst, DWORD fdwSuggest)
[5434]522{
[6712]523 ACMDRVFORMATSUGGEST adfg;
524 MMRESULT mmr;
[5434]525
[6712]526 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
527 had, pwfxSrc, pwfxDst, cbwfxDst, fdwSuggest);
[5434]528
529 if (fdwSuggest & ~(ACM_FORMATSUGGESTF_NCHANNELS|ACM_FORMATSUGGESTF_NSAMPLESPERSEC|
[6712]530 ACM_FORMATSUGGESTF_WBITSPERSAMPLE|ACM_FORMATSUGGESTF_WFORMATTAG))
531 return MMSYSERR_INVALFLAG;
[5434]532
533 adfg.cbStruct = sizeof(adfg);
534 adfg.fdwSuggest = fdwSuggest;
535 adfg.pwfxSrc = pwfxSrc;
536 adfg.cbwfxSrc = (pwfxSrc->wFormatTag == WAVE_FORMAT_PCM) ?
[6712]537 sizeof(WAVEFORMATEX) : pwfxSrc->cbSize;
[5434]538 adfg.pwfxDst = pwfxDst;
539 adfg.cbwfxDst = cbwfxDst;
540
541 if (had == (HACMDRIVER)NULL) {
[6712]542 PWINE_ACMDRIVERID padid;
543
544 /* MS doc says: ACM finds the best suggestion.
545 * Well, first found will be the "best"
546 */
547 mmr = ACMERR_NOTPOSSIBLE;
548 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
549 /* should check for codec only */
550 if (!padid->bEnabled ||
551 acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != MMSYSERR_NOERROR)
552 continue;
553
554 if (MSACM_Message(had, ACMDM_FORMAT_SUGGEST,
555 (LPARAM)&adfg, 0L) == MMSYSERR_NOERROR) {
556 mmr = MMSYSERR_NOERROR;
557 break;
558 }
559 acmDriverClose(had, 0);
560 }
[5434]561 } else {
[6712]562 mmr = MSACM_Message(had, ACMDM_FORMAT_SUGGEST, (LPARAM)&adfg, 0L);
[5434]563 }
564 return mmr;
565}
566
567/***********************************************************************
568 * acmFormatTagDetailsA (MSACM32.30)
569 */
[6712]570MMRESULT WINAPI acmFormatTagDetailsA(HACMDRIVER had, PACMFORMATTAGDETAILSA paftda,
571 DWORD fdwDetails)
[5434]572{
[6712]573 ACMFORMATTAGDETAILSW aftdw;
574 MMRESULT mmr;
[5434]575
576 memset(&aftdw, 0, sizeof(aftdw));
577 aftdw.cbStruct = sizeof(aftdw);
578 aftdw.dwFormatTagIndex = paftda->dwFormatTagIndex;
579 aftdw.dwFormatTag = paftda->dwFormatTag;
580
581 mmr = acmFormatTagDetailsW(had, &aftdw, fdwDetails);
582 if (mmr == MMSYSERR_NOERROR) {
[6712]583 paftda->dwFormatTag = aftdw.dwFormatTag;
584 paftda->dwFormatTagIndex = aftdw.dwFormatTagIndex;
585 paftda->cbFormatSize = aftdw.cbFormatSize;
586 paftda->fdwSupport = aftdw.fdwSupport;
587 paftda->cStandardFormats = aftdw.cStandardFormats;
[5434]588 WideCharToMultiByte( CP_ACP, 0, aftdw.szFormatTag, -1, paftda->szFormatTag,
589 sizeof(paftda->szFormatTag), NULL, NULL );
590 }
591 return mmr;
592}
593
594/***********************************************************************
595 * acmFormatTagDetailsW (MSACM32.31)
596 */
[6712]597MMRESULT WINAPI acmFormatTagDetailsW(HACMDRIVER had, PACMFORMATTAGDETAILSW paftd,
598 DWORD fdwDetails)
[5434]599{
[6712]600 PWINE_ACMDRIVERID padid;
601 MMRESULT mmr;
[5434]602
603 TRACE("(0x%08x, %p, %ld)\n", had, paftd, fdwDetails);
604
605 if (fdwDetails & ~(ACM_FORMATTAGDETAILSF_FORMATTAG|ACM_FORMATTAGDETAILSF_INDEX|
[6712]606 ACM_FORMATTAGDETAILSF_LARGESTSIZE))
607 return MMSYSERR_INVALFLAG;
[5434]608
609 switch (fdwDetails) {
610 case ACM_FORMATTAGDETAILSF_FORMATTAG:
[6712]611 if (had == (HACMDRIVER)NULL) {
612 mmr = ACMERR_NOTPOSSIBLE;
613 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
614 /* should check for codec only */
615 if (padid->bEnabled && acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == 0) {
616 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
617 (LPARAM)paftd, (LPARAM)fdwDetails);
618 acmDriverClose(had, 0);
619 if (mmr == MMSYSERR_NOERROR) break;
620 }
621 }
622 } else {
623 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
624 (LPARAM)paftd, (LPARAM)fdwDetails);
625 }
626 break;
[5434]627
628 case ACM_FORMATTAGDETAILSF_INDEX:
[6712]629 /* FIXME should check paftd->dwFormatTagIndex < add.cFormatTags */
630 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
631 (LPARAM)paftd, (LPARAM)fdwDetails);
632 break;
[5434]633
634 case ACM_FORMATTAGDETAILSF_LARGESTSIZE:
[6712]635 if (had == (HACMDRIVER)NULL) {
636 ACMFORMATTAGDETAILSW tmp;
637 DWORD ft = paftd->dwFormatTag;
[5434]638
[6712]639 mmr = ACMERR_NOTPOSSIBLE;
640 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
641 /* should check for codec only */
642 if (padid->bEnabled &&
643 acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == 0) {
[5434]644
[6712]645 memset(&tmp, 0, sizeof(tmp));
646 tmp.cbStruct = sizeof(tmp);
647 tmp.dwFormatTag = ft;
[5434]648
[6712]649 if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
650 (LPARAM)&tmp,
651 (LPARAM)fdwDetails) == MMSYSERR_NOERROR) {
652 if (mmr == ACMERR_NOTPOSSIBLE ||
653 paftd->cbFormatSize < tmp.cbFormatSize) {
654 *paftd = tmp;
655 mmr = MMSYSERR_NOERROR;
656 }
657 }
658 acmDriverClose(had, 0);
659 }
660 }
661 } else {
662 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
663 (LPARAM)paftd, (LPARAM)fdwDetails);
664 }
665 break;
[5434]666
667 default:
[6712]668 WARN("Unsupported fdwDetails=%08lx\n", fdwDetails);
669 mmr = MMSYSERR_ERROR;
[5434]670 }
671
[6712]672 if (mmr == MMSYSERR_NOERROR &&
673 paftd->dwFormatTag == WAVE_FORMAT_PCM && paftd->szFormatTag[0] == 0)
[5434]674 MultiByteToWideChar( CP_ACP, 0, "PCM", -1, paftd->szFormatTag,
675 sizeof(paftd->szFormatTag)/sizeof(WCHAR) );
676
677 return mmr;
678}
679
680struct MSACM_FormatTagEnumWtoA_Instance {
[6712]681 PACMFORMATTAGDETAILSA paftda;
682 DWORD dwInstance;
683 ACMFORMATTAGENUMCBA fnCallback;
[5434]684};
685
686static BOOL CALLBACK MSACM_FormatTagEnumCallbackWtoA(HACMDRIVERID hadid,
[6712]687 PACMFORMATTAGDETAILSW paftdw,
688 DWORD dwInstance,
689 DWORD fdwSupport)
[5434]690{
691 struct MSACM_FormatTagEnumWtoA_Instance* paftei;
692
693 paftei = (struct MSACM_FormatTagEnumWtoA_Instance*)dwInstance;
694
[6712]695 paftei->paftda->dwFormatTagIndex = paftdw->dwFormatTagIndex;
696 paftei->paftda->dwFormatTag = paftdw->dwFormatTag;
697 paftei->paftda->cbFormatSize = paftdw->cbFormatSize;
698 paftei->paftda->fdwSupport = paftdw->fdwSupport;
699 paftei->paftda->cStandardFormats = paftdw->cStandardFormats;
[5434]700 WideCharToMultiByte( CP_ACP, 0, paftdw->szFormatTag, -1, paftei->paftda->szFormatTag,
701 sizeof(paftei->paftda->szFormatTag), NULL, NULL );
702
[6712]703 return (paftei->fnCallback)(hadid, paftei->paftda,
704 paftei->dwInstance, fdwSupport);
[5434]705}
706
707/***********************************************************************
708 * acmFormatTagEnumA (MSACM32.32)
709 */
710MMRESULT WINAPI acmFormatTagEnumA(HACMDRIVER had, PACMFORMATTAGDETAILSA paftda,
[6712]711 ACMFORMATTAGENUMCBA fnCallback, DWORD dwInstance,
712 DWORD fdwEnum)
[5434]713{
[6712]714 ACMFORMATTAGDETAILSW aftdw;
[5434]715 struct MSACM_FormatTagEnumWtoA_Instance aftei;
716
717 memset(&aftdw, 0, sizeof(aftdw));
718 aftdw.cbStruct = sizeof(aftdw);
719 aftdw.dwFormatTagIndex = paftda->dwFormatTagIndex;
720 aftdw.dwFormatTag = paftda->dwFormatTag;
721
722 aftei.paftda = paftda;
723 aftei.dwInstance = dwInstance;
724 aftei.fnCallback = fnCallback;
725
[6712]726 return acmFormatTagEnumW(had, &aftdw, MSACM_FormatTagEnumCallbackWtoA,
727 (DWORD)&aftei, fdwEnum);
[5434]728}
729
730/***********************************************************************
731 * acmFormatTagEnumW (MSACM32.33)
732 */
733MMRESULT WINAPI acmFormatTagEnumW(HACMDRIVER had, PACMFORMATTAGDETAILSW paftd,
[6712]734 ACMFORMATTAGENUMCBW fnCallback, DWORD dwInstance,
735 DWORD fdwEnum)
[5434]736{
[6712]737 PWINE_ACMDRIVERID padid;
738 ACMDRIVERDETAILSW add;
739 int i;
740 BOOL bPcmDone = FALSE;
[5434]741
742 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
[6712]743 had, paftd, fnCallback, dwInstance, fdwEnum);
[5434]744
745 if (paftd->cbStruct < sizeof(*paftd)) return MMSYSERR_INVALPARAM;
746
747 if (had) FIXME("had != NULL, not supported\n");
[6712]748
[5434]749 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
[6712]750 /* should check for codec only */
751 if (padid->bEnabled && acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == MMSYSERR_NOERROR) {
752 add.cbStruct = sizeof(add);
[5434]753
[6712]754 if (acmDriverDetailsW((HACMDRIVERID)padid, &add, 0) == MMSYSERR_NOERROR) {
755 for (i = 0; i < add.cFormatTags; i++) {
756 paftd->dwFormatTagIndex = i;
757 if (acmFormatTagDetailsW(had, paftd, ACM_FORMATTAGDETAILSF_INDEX) == MMSYSERR_NOERROR) {
758 if (paftd->dwFormatTag == WAVE_FORMAT_PCM) {
759 /* FIXME (EPP): I'm not sure this is the correct
760 * algorithm (should make more sense to apply the same
761 * for all already loaded formats, but this will do
762 * for now
763 */
764 if (bPcmDone) continue;
765 bPcmDone = TRUE;
766 }
767 if (!(fnCallback)((HACMDRIVERID)padid, paftd, dwInstance,
768 add.fdwSupport)) {
769 padid = NULL;
770 break;
771 }
772 }
773 }
774 }
775 }
776 acmDriverClose(had, 0);
[5434]777 }
778 return MMSYSERR_NOERROR;
779}
Note: See TracBrowser for help on using the repository browser.