1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
6 * Copyright 1998 Patrik Stridvall
17 #include "wine/unicode.h"
18 #include "debugtools.h"
24 DEFAULT_DEBUG_CHANNEL(msacm);
26 static PACMFORMATCHOOSEA afc;
28 struct MSACM_FillFormatData {
30 #define WINE_ACMFF_TAG 0
31 #define WINE_ACMFF_FORMAT 1
32 #define WINE_ACMFF_WFX 2
34 char szFormatTag[ACMFORMATTAGDETAILS_FORMATTAG_CHARS];
35 PACMFORMATCHOOSEA afc;
39 static BOOL CALLBACK MSACM_FillFormatTagsCB(HACMDRIVERID hadid,
40 PACMFORMATTAGDETAILSA paftd,
41 DWORD dwInstance, DWORD fdwSupport)
43 struct MSACM_FillFormatData* affd = (struct MSACM_FillFormatData*)dwInstance;
47 if (SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
49 (WPARAM)-1, (LPARAM)paftd->szFormatTag) == CB_ERR)
50 SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
51 CB_ADDSTRING, 0, (DWORD)paftd->szFormatTag);
53 case WINE_ACMFF_FORMAT:
54 if (strcmp(affd->szFormatTag, paftd->szFormatTag) == 0) {
57 if (acmDriverOpen(&had, hadid, 0) == MMSYSERR_NOERROR) {
58 ACMFORMATDETAILSA afd;
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;
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++)
78 wsprintfA(buffer + ACMFORMATTAGDETAILS_FORMATTAG_CHARS,
80 (afd.pwfx->nAvgBytesPerSec + 512) / 1024);
81 SendDlgItemMessageA(affd->hWnd,
82 IDD_ACMFORMATCHOOSE_CMB_FORMAT,
83 CB_ADDSTRING, 0, (DWORD)buffer);
86 acmDriverClose(had, 0);
87 SendDlgItemMessageA(affd->hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT,
93 if (strcmp(affd->szFormatTag, paftd->szFormatTag) == 0) {
96 if (acmDriverOpen(&had, hadid, 0) == MMSYSERR_NOERROR) {
97 ACMFORMATDETAILSA afd;
99 afd.cbStruct = sizeof(afd);
100 afd.dwFormatTag = paftd->dwFormatTag;
101 afd.pwfx = affd->afc->pwfx;
102 afd.cbwfx = affd->afc->cbwfx;
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);
112 FIXME("Unknown mode (%d)\n", affd->mode);
118 static BOOL MSACM_FillFormatTags(HWND hWnd)
120 ACMFORMATTAGDETAILSA aftd;
121 struct MSACM_FillFormatData affd;
123 memset(&aftd, 0, sizeof(aftd));
124 aftd.cbStruct = sizeof(aftd);
127 affd.mode = WINE_ACMFF_TAG;
129 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
130 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG, CB_SETCURSEL, 0, 0);
134 static BOOL MSACM_FillFormat(HWND hWnd)
136 ACMFORMATTAGDETAILSA aftd;
137 struct MSACM_FillFormatData affd;
139 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT, CB_RESETCONTENT, 0, 0);
141 memset(&aftd, 0, sizeof(aftd));
142 aftd.cbStruct = sizeof(aftd);
145 affd.mode = WINE_ACMFF_FORMAT;
146 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
148 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMATTAG,
150 (DWORD)affd.szFormatTag);
152 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
153 SendDlgItemMessageA(hWnd, IDD_ACMFORMATCHOOSE_CMB_FORMAT, CB_SETCURSEL, 0, 0);
157 static MMRESULT MSACM_GetWFX(HWND hWnd, PACMFORMATCHOOSEA afc)
159 ACMFORMATTAGDETAILSA aftd;
160 struct MSACM_FillFormatData affd;
162 memset(&aftd, 0, sizeof(aftd));
163 aftd.cbStruct = sizeof(aftd);
166 affd.mode = WINE_ACMFF_WFX;
168 affd.ret = MMSYSERR_NOERROR;
170 acmFormatTagEnumA((HACMDRIVER)0, &aftd, MSACM_FillFormatTagsCB, (DWORD)&affd, 0);
174 static BOOL WINAPI FormatChooseDlgProc(HWND hWnd, UINT msg,
175 WPARAM wParam, LPARAM lParam)
178 TRACE("hwnd=%i msg=%i 0x%08x 0x%08lx\n", hWnd, msg, wParam, lParam );
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);
193 switch (LOWORD(wParam)) {
195 EndDialog(hWnd, MSACM_GetWFX(hWnd, afc));
198 EndDialog(hWnd, ACMERR_CANCELED);
200 case IDD_ACMFORMATCHOOSE_CMB_FORMATTAG:
201 switch (HIWORD(wParam)) {
203 MSACM_FillFormat(hWnd);
206 TRACE("Dropped dlgNotif (fmtTag): 0x%08x 0x%08lx\n",
207 HIWORD(wParam), lParam);
211 case IDD_ACMFORMATCHOOSE_BTN_HELP:
212 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_SHOWHELP)
213 SendMessageA(afc->hwndOwner,
214 RegisterWindowMessageA(ACMHELPMSGSTRINGA), 0L, 0L);
218 TRACE("Dropped dlgCmd: ctl=%d ntf=0x%04x 0x%08lx\n",
219 LOWORD(wParam), HIWORD(wParam), lParam);
224 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_CONTEXTHELP)
225 SendMessageA(afc->hwndOwner,
226 RegisterWindowMessageA(ACMHELPMSGCONTEXTMENUA),
229 #if defined(WM_CONTEXTHELP)
231 if (afc->fdwStyle & ACMFORMATCHOOSE_STYLEF_CONTEXTHELP)
232 SendMessageA(afc->hwndOwner,
233 RegisterWindowMessageA(ACMHELPMSGCONTEXTHELPA),
238 TRACE("Dropped dlgMsg: hwnd=%i msg=%i 0x%08x 0x%08lx\n",
239 hWnd, msg, wParam, lParam );
245 /***********************************************************************
246 * acmFormatChooseA (MSACM32.23)
248 MMRESULT WINAPI acmFormatChooseA(PACMFORMATCHOOSEA pafmtc)
250 return DialogBoxParamA(MSACM_hInstance32, MAKEINTRESOURCEA(DLG_ACMFORMATCHOOSE_ID),
251 pafmtc->hwndOwner, FormatChooseDlgProc, (INT)pafmtc);
254 /***********************************************************************
255 * acmFormatChooseW (MSACM32.24)
257 MMRESULT WINAPI acmFormatChooseW(PACMFORMATCHOOSEW pafmtc)
259 FIXME("(%p): stub\n", pafmtc);
260 SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
261 return MMSYSERR_ERROR;
264 /***********************************************************************
265 * acmFormatDetailsA (MSACM32.25)
267 MMRESULT WINAPI acmFormatDetailsA(HACMDRIVER had, PACMFORMATDETAILSA pafd,
270 ACMFORMATDETAILSW afdw;
273 memset(&afdw, 0, sizeof(afdw));
274 afdw.cbStruct = sizeof(afdw);
275 afdw.dwFormatIndex = pafd->dwFormatIndex;
276 afdw.dwFormatTag = pafd->dwFormatTag;
277 afdw.pwfx = pafd->pwfx;
278 afdw.cbwfx = pafd->cbwfx;
280 mmr = acmFormatDetailsW(had, &afdw, fdwDetails);
281 if (mmr == MMSYSERR_NOERROR) {
282 pafd->dwFormatTag = afdw.dwFormatTag;
283 pafd->fdwSupport = afdw.fdwSupport;
284 WideCharToMultiByte( CP_ACP, 0, afdw.szFormat, -1,
285 pafd->szFormat, sizeof(pafd->szFormat), NULL, NULL );
290 /***********************************************************************
291 * acmFormatDetailsW (MSACM32.26)
293 MMRESULT WINAPI acmFormatDetailsW(HACMDRIVER had, PACMFORMATDETAILSW pafd,
297 static WCHAR fmt1[] = {'%','d',' ','H','z',0};
298 static WCHAR fmt2[] = {';',' ','%','d',' ','b','i','t','s',0};
299 ACMFORMATTAGDETAILSA aftd;
301 TRACE("(0x%08x, %p, %ld)\n", had, pafd, fdwDetails);
303 memset(&aftd, 0, sizeof(aftd));
304 aftd.cbStruct = sizeof(aftd);
306 if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM;
308 switch (fdwDetails) {
309 case ACM_FORMATDETAILSF_FORMAT:
310 if (pafd->dwFormatTag != pafd->pwfx->wFormatTag) {
311 mmr = MMSYSERR_INVALPARAM;
314 if (had == (HACMDRIVER)NULL) {
315 PWINE_ACMDRIVERID padid;
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;
329 mmr = MSACM_Message(had, ACMDM_FORMAT_DETAILS,
330 (LPARAM)pafd, (LPARAM)fdwDetails);
333 case ACM_FORMATDETAILSF_INDEX:
334 /* should check pafd->dwFormatIndex < aftd->cStandardFormats */
335 mmr = MSACM_Message(had, ACMDM_FORMAT_DETAILS,
336 (LPARAM)pafd, (LPARAM)fdwDetails);
339 WARN("Unknown fdwDetails %08lx\n", fdwDetails);
340 mmr = MMSYSERR_INVALFLAG;
344 if (mmr == MMSYSERR_NOERROR && pafd->szFormat[0] == (WCHAR)0) {
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);
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) );
355 TRACE("=> %d\n", mmr);
359 struct MSACM_FormatEnumWtoA_Instance {
360 PACMFORMATDETAILSA pafda;
362 ACMFORMATENUMCBA fnCallback;
365 static BOOL CALLBACK MSACM_FormatEnumCallbackWtoA(HACMDRIVERID hadid,
366 PACMFORMATDETAILSW pafdw,
370 struct MSACM_FormatEnumWtoA_Instance* pafei;
372 pafei = (struct MSACM_FormatEnumWtoA_Instance*)dwInstance;
374 pafei->pafda->dwFormatIndex = pafdw->dwFormatIndex;
375 pafei->pafda->dwFormatTag = pafdw->dwFormatTag;
376 pafei->pafda->fdwSupport = pafdw->fdwSupport;
377 WideCharToMultiByte( CP_ACP, 0, pafdw->szFormat, -1,
378 pafei->pafda->szFormat, sizeof(pafei->pafda->szFormat), NULL, NULL );
380 return (pafei->fnCallback)(hadid, pafei->pafda,
381 pafei->dwInstance, fdwSupport);
384 /***********************************************************************
385 * acmFormatEnumA (MSACM32.27)
387 MMRESULT WINAPI acmFormatEnumA(HACMDRIVER had, PACMFORMATDETAILSA pafda,
388 ACMFORMATENUMCBA fnCallback, DWORD dwInstance,
391 ACMFORMATDETAILSW afdw;
392 struct MSACM_FormatEnumWtoA_Instance afei;
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;
402 afei.dwInstance = dwInstance;
403 afei.fnCallback = fnCallback;
405 return acmFormatEnumW(had, &afdw, MSACM_FormatEnumCallbackWtoA,
406 (DWORD)&afei, fdwEnum);
409 /***********************************************************************
410 * acmFormatEnumW (MSACM32.28)
412 static BOOL MSACM_FormatEnumHelper(PWINE_ACMDRIVERID padid, HACMDRIVER had,
413 PACMFORMATDETAILSW pafd, PWAVEFORMATEX pwfxRef,
414 ACMFORMATENUMCBW fnCallback, DWORD dwInstance,
417 ACMDRIVERDETAILSW add;
418 ACMFORMATTAGDETAILSW aftd;
421 add.cbStruct = sizeof(add);
423 if (acmDriverDetailsW((HACMDRIVERID)padid, &add, 0) != MMSYSERR_NOERROR) return FALSE;
425 for (i = 0; i < add.cFormatTags; i++) {
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)
432 if ((fdwEnum & ACM_FORMATENUMF_WFORMATTAG) && aftd.dwFormatTag != pwfxRef->wFormatTag)
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)
441 if ((fdwEnum & ACM_FORMATENUMF_NCHANNELS) &&
442 pafd->pwfx->nChannels != pwfxRef->nChannels)
444 if ((fdwEnum & ACM_FORMATENUMF_NSAMPLESPERSEC) &&
445 pafd->pwfx->nSamplesPerSec != pwfxRef->nSamplesPerSec)
447 if ((fdwEnum & ACM_FORMATENUMF_WBITSPERSAMPLE) &&
448 pafd->pwfx->wBitsPerSample != pwfxRef->wBitsPerSample)
450 if ((fdwEnum & ACM_FORMATENUMF_HARDWARE) &&
451 !(pafd->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_HARDWARE))
454 /* more checks to be done on fdwEnum */
456 if (!(fnCallback)((HACMDRIVERID)padid, pafd, dwInstance, add.fdwSupport))
459 /* the "formats" used by the filters are also reported */
464 /**********************************************************************/
466 MMRESULT WINAPI acmFormatEnumW(HACMDRIVER had, PACMFORMATDETAILSW pafd,
467 ACMFORMATENUMCBW fnCallback, DWORD dwInstance,
470 PWINE_ACMDRIVERID padid;
474 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
475 had, pafd, fnCallback, dwInstance, fdwEnum);
477 if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM;
479 if (fdwEnum & (ACM_FORMATENUMF_WFORMATTAG|ACM_FORMATENUMF_NCHANNELS|
480 ACM_FORMATENUMF_NSAMPLESPERSEC|ACM_FORMATENUMF_WBITSPERSAMPLE|
481 ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST))
482 wfxRef = *pafd->pwfx;
484 if ((fdwEnum & ACM_FORMATENUMF_HARDWARE) &&
485 !(fdwEnum & (ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT)))
486 return MMSYSERR_INVALPARAM;
488 if ((fdwEnum & ACM_FORMATENUMF_WFORMATTAG) &&
489 (pafd->dwFormatTag != pafd->pwfx->wFormatTag))
490 return MMSYSERR_INVALPARAM;
492 if (fdwEnum & (ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST|
493 ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT))
494 FIXME("Unsupported fdwEnum values %08lx\n", fdwEnum);
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;
505 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
506 /* should check for codec only */
507 if (!padid->bEnabled || acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != MMSYSERR_NOERROR)
509 ret = MSACM_FormatEnumHelper(padid, had, pafd, &wfxRef,
510 fnCallback, dwInstance, fdwEnum);
511 acmDriverClose(had, 0);
514 return MMSYSERR_NOERROR;
517 /***********************************************************************
518 * acmFormatSuggest (MSACM32.29)
520 MMRESULT WINAPI acmFormatSuggest(HACMDRIVER had, PWAVEFORMATEX pwfxSrc,
521 PWAVEFORMATEX pwfxDst, DWORD cbwfxDst, DWORD fdwSuggest)
523 ACMDRVFORMATSUGGEST adfg;
526 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
527 had, pwfxSrc, pwfxDst, cbwfxDst, fdwSuggest);
529 if (fdwSuggest & ~(ACM_FORMATSUGGESTF_NCHANNELS|ACM_FORMATSUGGESTF_NSAMPLESPERSEC|
530 ACM_FORMATSUGGESTF_WBITSPERSAMPLE|ACM_FORMATSUGGESTF_WFORMATTAG))
531 return MMSYSERR_INVALFLAG;
533 adfg.cbStruct = sizeof(adfg);
534 adfg.fdwSuggest = fdwSuggest;
535 adfg.pwfxSrc = pwfxSrc;
536 adfg.cbwfxSrc = (pwfxSrc->wFormatTag == WAVE_FORMAT_PCM) ?
537 sizeof(WAVEFORMATEX) : pwfxSrc->cbSize;
538 adfg.pwfxDst = pwfxDst;
539 adfg.cbwfxDst = cbwfxDst;
541 if (had == (HACMDRIVER)NULL) {
542 PWINE_ACMDRIVERID padid;
544 /* MS doc says: ACM finds the best suggestion.
545 * Well, first found will be the "best"
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)
554 if (MSACM_Message(had, ACMDM_FORMAT_SUGGEST,
555 (LPARAM)&adfg, 0L) == MMSYSERR_NOERROR) {
556 mmr = MMSYSERR_NOERROR;
559 acmDriverClose(had, 0);
562 mmr = MSACM_Message(had, ACMDM_FORMAT_SUGGEST, (LPARAM)&adfg, 0L);
567 /***********************************************************************
568 * acmFormatTagDetailsA (MSACM32.30)
570 MMRESULT WINAPI acmFormatTagDetailsA(HACMDRIVER had, PACMFORMATTAGDETAILSA paftda,
573 ACMFORMATTAGDETAILSW aftdw;
576 memset(&aftdw, 0, sizeof(aftdw));
577 aftdw.cbStruct = sizeof(aftdw);
578 aftdw.dwFormatTagIndex = paftda->dwFormatTagIndex;
579 aftdw.dwFormatTag = paftda->dwFormatTag;
581 mmr = acmFormatTagDetailsW(had, &aftdw, fdwDetails);
582 if (mmr == MMSYSERR_NOERROR) {
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;
588 WideCharToMultiByte( CP_ACP, 0, aftdw.szFormatTag, -1, paftda->szFormatTag,
589 sizeof(paftda->szFormatTag), NULL, NULL );
594 /***********************************************************************
595 * acmFormatTagDetailsW (MSACM32.31)
597 MMRESULT WINAPI acmFormatTagDetailsW(HACMDRIVER had, PACMFORMATTAGDETAILSW paftd,
600 PWINE_ACMDRIVERID padid;
603 TRACE("(0x%08x, %p, %ld)\n", had, paftd, fdwDetails);
605 if (fdwDetails & ~(ACM_FORMATTAGDETAILSF_FORMATTAG|ACM_FORMATTAGDETAILSF_INDEX|
606 ACM_FORMATTAGDETAILSF_LARGESTSIZE))
607 return MMSYSERR_INVALFLAG;
609 switch (fdwDetails) {
610 case ACM_FORMATTAGDETAILSF_FORMATTAG:
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;
623 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
624 (LPARAM)paftd, (LPARAM)fdwDetails);
628 case ACM_FORMATTAGDETAILSF_INDEX:
629 /* FIXME should check paftd->dwFormatTagIndex < add.cFormatTags */
630 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
631 (LPARAM)paftd, (LPARAM)fdwDetails);
634 case ACM_FORMATTAGDETAILSF_LARGESTSIZE:
635 if (had == (HACMDRIVER)NULL) {
636 ACMFORMATTAGDETAILSW tmp;
637 DWORD ft = paftd->dwFormatTag;
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) {
645 memset(&tmp, 0, sizeof(tmp));
646 tmp.cbStruct = sizeof(tmp);
647 tmp.dwFormatTag = ft;
649 if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
651 (LPARAM)fdwDetails) == MMSYSERR_NOERROR) {
652 if (mmr == ACMERR_NOTPOSSIBLE ||
653 paftd->cbFormatSize < tmp.cbFormatSize) {
655 mmr = MMSYSERR_NOERROR;
658 acmDriverClose(had, 0);
662 mmr = MSACM_Message(had, ACMDM_FORMATTAG_DETAILS,
663 (LPARAM)paftd, (LPARAM)fdwDetails);
668 WARN("Unsupported fdwDetails=%08lx\n", fdwDetails);
669 mmr = MMSYSERR_ERROR;
672 if (mmr == MMSYSERR_NOERROR &&
673 paftd->dwFormatTag == WAVE_FORMAT_PCM && paftd->szFormatTag[0] == 0)
674 MultiByteToWideChar( CP_ACP, 0, "PCM", -1, paftd->szFormatTag,
675 sizeof(paftd->szFormatTag)/sizeof(WCHAR) );
680 struct MSACM_FormatTagEnumWtoA_Instance {
681 PACMFORMATTAGDETAILSA paftda;
683 ACMFORMATTAGENUMCBA fnCallback;
686 static BOOL CALLBACK MSACM_FormatTagEnumCallbackWtoA(HACMDRIVERID hadid,
687 PACMFORMATTAGDETAILSW paftdw,
691 struct MSACM_FormatTagEnumWtoA_Instance* paftei;
693 paftei = (struct MSACM_FormatTagEnumWtoA_Instance*)dwInstance;
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;
700 WideCharToMultiByte( CP_ACP, 0, paftdw->szFormatTag, -1, paftei->paftda->szFormatTag,
701 sizeof(paftei->paftda->szFormatTag), NULL, NULL );
703 return (paftei->fnCallback)(hadid, paftei->paftda,
704 paftei->dwInstance, fdwSupport);
707 /***********************************************************************
708 * acmFormatTagEnumA (MSACM32.32)
710 MMRESULT WINAPI acmFormatTagEnumA(HACMDRIVER had, PACMFORMATTAGDETAILSA paftda,
711 ACMFORMATTAGENUMCBA fnCallback, DWORD dwInstance,
714 ACMFORMATTAGDETAILSW aftdw;
715 struct MSACM_FormatTagEnumWtoA_Instance aftei;
717 memset(&aftdw, 0, sizeof(aftdw));
718 aftdw.cbStruct = sizeof(aftdw);
719 aftdw.dwFormatTagIndex = paftda->dwFormatTagIndex;
720 aftdw.dwFormatTag = paftda->dwFormatTag;
722 aftei.paftda = paftda;
723 aftei.dwInstance = dwInstance;
724 aftei.fnCallback = fnCallback;
726 return acmFormatTagEnumW(had, &aftdw, MSACM_FormatTagEnumCallbackWtoA,
727 (DWORD)&aftei, fdwEnum);
730 /***********************************************************************
731 * acmFormatTagEnumW (MSACM32.33)
733 MMRESULT WINAPI acmFormatTagEnumW(HACMDRIVER had, PACMFORMATTAGDETAILSW paftd,
734 ACMFORMATTAGENUMCBW fnCallback, DWORD dwInstance,
737 PWINE_ACMDRIVERID padid;
738 ACMDRIVERDETAILSW add;
740 BOOL bPcmDone = FALSE;
742 TRACE("(0x%08x, %p, %p, %ld, %ld)\n",
743 had, paftd, fnCallback, dwInstance, fdwEnum);
745 if (paftd->cbStruct < sizeof(*paftd)) return MMSYSERR_INVALPARAM;
747 if (had) FIXME("had != NULL, not supported\n");
749 for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
750 /* should check for codec only */
751 if (padid->bEnabled && acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == MMSYSERR_NOERROR) {
752 add.cbStruct = sizeof(add);
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
764 if (bPcmDone) continue;
767 if (!(fnCallback)((HACMDRIVERID)padid, paftd, dwInstance,
776 acmDriverClose(had, 0);
778 return MMSYSERR_NOERROR;