MMRESULT WINAPI acmFormatEnumW(HACMDRIVER had, PACMFORMATDETAILSW pafd, ACMFORMATENUMCBW fnCallback, DWORD dwInstance, DWORD fdwEnum) { PWINE_ACMDRIVERID padid; WAVEFORMATEX wfxRef; BOOL ret; TRACE("(%p, %p, %p, %d, %d)\n", had, pafd, fnCallback, dwInstance, fdwEnum); if (!pafd) return MMSYSERR_INVALPARAM; if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM; if (fdwEnum & (ACM_FORMATENUMF_WFORMATTAG|ACM_FORMATENUMF_NCHANNELS| ACM_FORMATENUMF_NSAMPLESPERSEC|ACM_FORMATENUMF_WBITSPERSAMPLE| ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST)) wfxRef = *pafd->pwfx; if ((fdwEnum & ACM_FORMATENUMF_HARDWARE) && !(fdwEnum & (ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT))) return MMSYSERR_INVALPARAM; if ((fdwEnum & ACM_FORMATENUMF_WFORMATTAG) && (pafd->dwFormatTag != pafd->pwfx->wFormatTag)) return MMSYSERR_INVALPARAM; if (fdwEnum & (ACM_FORMATENUMF_CONVERT|ACM_FORMATENUMF_SUGGEST| ACM_FORMATENUMF_INPUT|ACM_FORMATENUMF_OUTPUT)) FIXME("Unsupported fdwEnum values %08x\n", fdwEnum); if (had) { HACMDRIVERID hadid; if (acmDriverID((HACMOBJ)had, &hadid, 0) != MMSYSERR_NOERROR) return MMSYSERR_INVALHANDLE; MSACM_FormatEnumHelper(MSACM_GetDriverID(hadid), had, pafd, &wfxRef, fnCallback, dwInstance, fdwEnum); return MMSYSERR_NOERROR; } for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) { /* should check for codec only */ if ((padid->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_DISABLED) || acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != MMSYSERR_NOERROR) continue; ret = MSACM_FormatEnumHelper(padid, had, pafd, &wfxRef, fnCallback, dwInstance, fdwEnum); acmDriverClose(had, 0); if (!ret) break; } return MMSYSERR_NOERROR; }
/*********************************************************************** * acmFilterEnumW (MSACM32.@) */ MMRESULT WINAPI acmFilterEnumW(HACMDRIVER had, PACMFILTERDETAILSW pafd, ACMFILTERENUMCBW fnCallback, DWORD_PTR dwInstance, DWORD fdwEnum) { PWINE_ACMDRIVERID padid; BOOL ret; TRACE("(%p, %p, %p, %ld, %d)\n", had, pafd, fnCallback, dwInstance, fdwEnum); if (pafd->cbStruct < sizeof(*pafd)) return MMSYSERR_INVALPARAM; if (fdwEnum & ~(ACM_FILTERENUMF_DWFILTERTAG)) FIXME("Unsupported fdwEnum values\n"); if (had) { HACMDRIVERID hadid; if (acmDriverID((HACMOBJ)had, &hadid, 0) != MMSYSERR_NOERROR) return MMSYSERR_INVALHANDLE; MSACM_FilterEnumHelper(MSACM_GetDriverID(hadid), had, pafd, fnCallback, dwInstance, fdwEnum); return MMSYSERR_NOERROR; } for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) { /* should check for codec only */ if ((padid->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_DISABLED) || acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != MMSYSERR_NOERROR) continue; ret = MSACM_FilterEnumHelper(padid, had, pafd, fnCallback, dwInstance, fdwEnum); acmDriverClose(had, 0); if (!ret) break; } return MMSYSERR_NOERROR; }
/*********************************************************************** * acmFormatTagEnumW (MSACM32.@) */ MMRESULT WINAPI acmFormatTagEnumW(HACMDRIVER had, PACMFORMATTAGDETAILSW paftd, ACMFORMATTAGENUMCBW fnCallback, DWORD dwInstance, DWORD fdwEnum) { PWINE_ACMDRIVERID padid; unsigned int i; BOOL bPcmDone = FALSE; TRACE("(%p, %p, %p, %d, %d)\n", had, paftd, fnCallback, dwInstance, fdwEnum); if (!paftd) return MMSYSERR_INVALPARAM; if (paftd->cbStruct < sizeof(*paftd)) return MMSYSERR_INVALPARAM; if (fdwEnum != 0) return MMSYSERR_INVALFLAG; /* (WS) MSDN info page says that if had != 0, then we should find * the specific driver to get its tags from. Therefore I'm removing * the FIXME call and adding a search block below. It also seems * that the lack of this functionality was the responsible for * codecs to be multiply and incorrectly listed. */ /* if (had) FIXME("had != NULL, not supported\n"); */ if (had) { if (acmDriverID((HACMOBJ)had, (HACMDRIVERID *)&padid, 0) != MMSYSERR_NOERROR) return MMSYSERR_INVALHANDLE; for (i = 0; i < padid->cFormatTags; i++) { paftd->dwFormatTagIndex = i; if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS, (LPARAM)paftd, ACM_FORMATTAGDETAILSF_INDEX) == MMSYSERR_NOERROR) { if (paftd->dwFormatTag == WAVE_FORMAT_PCM) { if (paftd->szFormatTag[0] == 0) MultiByteToWideChar( CP_ACP, 0, "PCM", -1, paftd->szFormatTag, sizeof(paftd->szFormatTag)/sizeof(WCHAR) ); /* (WS) I'm preserving this PCM hack since it seems to be * correct. Please notice this block was borrowed from * below. */ if (bPcmDone) continue; bPcmDone = TRUE; } if (!(fnCallback)((HACMDRIVERID)padid, paftd, dwInstance, padid->fdwSupport)) return MMSYSERR_NOERROR; } } } /* if had==0 then search for the first suitable driver */ else { for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) { /* should check for codec only */ if (!(padid->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_DISABLED) && acmDriverOpen(&had, (HACMDRIVERID)padid, 0) == MMSYSERR_NOERROR) { for (i = 0; i < padid->cFormatTags; i++) { paftd->dwFormatTagIndex = i; if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS, (LPARAM)paftd, ACM_FORMATTAGDETAILSF_INDEX) == MMSYSERR_NOERROR) { if (paftd->dwFormatTag == WAVE_FORMAT_PCM) { if (paftd->szFormatTag[0] == 0) MultiByteToWideChar( CP_ACP, 0, "PCM", -1, paftd->szFormatTag, sizeof(paftd->szFormatTag)/sizeof(WCHAR) ); /* FIXME (EPP): I'm not sure this is the correct * algorithm (should make more sense to apply the same * for all already loaded formats, but this will do * for now */ if (bPcmDone) continue; bPcmDone = TRUE; } if (!(fnCallback)((HACMDRIVERID)padid, paftd, dwInstance, padid->fdwSupport)) { acmDriverClose(had, 0); return MMSYSERR_NOERROR; } } } } acmDriverClose(had, 0); } } return MMSYSERR_NOERROR; }
static BOOL CALLBACK DriverEnumProc(HACMDRIVERID hadid, DWORD_PTR dwInstance, DWORD fdwSupport) { MMRESULT rc; ACMDRIVERDETAILSA dd; HACMDRIVER had; DWORD dwDriverPriority; DWORD dwDriverSupport; if (winetest_interactive) { trace("id: %p\n", hadid); trace(" Supports:\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_ASYNC) trace(" async conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_CODEC) trace(" different format conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_CONVERTER) trace(" same format conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_FILTER) trace(" filtering\n"); } /* try an invalid pointer */ rc = acmDriverDetailsA(hadid, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try an invalid structure size */ ZeroMemory(&dd, sizeof(dd)); rc = acmDriverDetailsA(hadid, &dd, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* MSDN says this should fail but it doesn't in practice */ dd.cbStruct = 4; rc = acmDriverDetailsA(hadid, &dd, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NOTSUPPORTED, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try an invalid handle */ dd.cbStruct = sizeof(dd); rc = acmDriverDetailsA((HACMDRIVERID)1, &dd, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try an invalid handle and pointer */ rc = acmDriverDetailsA((HACMDRIVERID)1, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try invalid details */ rc = acmDriverDetailsA(hadid, &dd, -1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverDetailsA(hadid, &dd, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NOTSUPPORTED, "acmDriverDetailsA(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* cbStruct should contain size of returned data (at most sizeof(dd)) TODO: should it be *exactly* sizeof(dd), as tested here? */ if (rc == MMSYSERR_NOERROR) { static const struct { const char *shortname; WORD mid; WORD pid; WORD pid_alt; } *iter, expected_ids[] = { { "Microsoft IMA ADPCM", MM_MICROSOFT, MM_MSFT_ACM_IMAADPCM }, { "MS-ADPCM", MM_MICROSOFT, MM_MSFT_ACM_MSADPCM }, { "Microsoft CCITT G.711", MM_MICROSOFT, MM_MSFT_ACM_G711}, { "MPEG Layer-3 Codec", MM_FRAUNHOFER_IIS, MM_FHGIIS_MPEGLAYER3_DECODE, MM_FHGIIS_MPEGLAYER3_PROFESSIONAL }, { "MS-PCM", MM_MICROSOFT, MM_MSFT_ACM_PCM }, { 0 } }; ok(dd.cbStruct == sizeof(dd), "acmDriverDetailsA(): cbStruct = %08x\n", dd.cbStruct); for (iter = expected_ids; iter->shortname; ++iter) { if (!strcmp(iter->shortname, dd.szShortName)) { /* try alternative product id on mismatch */ if (iter->pid_alt && iter->pid != dd.wPid) ok(iter->mid == dd.wMid && iter->pid_alt == dd.wPid, "Got wrong manufacturer (0x%x vs 0x%x) or product (0x%x vs 0x%x)\n", dd.wMid, iter->mid, dd.wPid, iter->pid_alt); else ok(iter->mid == dd.wMid && iter->pid == dd.wPid, "Got wrong manufacturer (0x%x vs 0x%x) or product (0x%x vs 0x%x)\n", dd.wMid, iter->mid, dd.wPid, iter->pid); } } } if (rc == MMSYSERR_NOERROR && winetest_interactive) { trace(" Short name: %s\n", dd.szShortName); trace(" Long name: %s\n", dd.szLongName); trace(" Copyright: %s\n", dd.szCopyright); trace(" Licensing: %s\n", dd.szLicensing); trace(" Features: %s\n", dd.szFeatures); trace(" Supports %u formats\n", dd.cFormatTags); trace(" Supports %u filter formats\n", dd.cFilterTags); trace(" Mid: 0x%x\n", dd.wMid); trace(" Pid: 0x%x\n", dd.wPid); } /* try bad pointer */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_PRIORITY, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_PRIORITY, &dwDriverPriority); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_PRIORITY, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_PRIORITY, &dwDriverSupport); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try bad pointer */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_SUPPORT, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_SUPPORT, &dwDriverSupport); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_SUPPORT, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_SUPPORT, &dwDriverSupport); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try invalid pointer */ rc = acmDriverOpen(0, hadid, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try invalid handle */ rc = acmDriverOpen(&had, (HACMDRIVERID)1, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try invalid open */ rc = acmDriverOpen(&had, hadid, -1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverOpen(&had, hadid, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NODRIVER, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); if (rc == MMSYSERR_NOERROR) { DWORD dwSize; HACMDRIVERID hid; /* try bad pointer */ rc = acmDriverID((HACMOBJ)had, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmDriverID((HACMOBJ)1, &hid, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad handle and pointer */ rc = acmDriverID((HACMOBJ)1, 0, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad flag */ rc = acmDriverID((HACMOBJ)had, &hid, 1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverID((HACMOBJ)had, &hid, 0); ok(rc == MMSYSERR_NOERROR, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); ok(hid == hadid, "acmDriverID() returned ID %p doesn't equal %p\n", hid, hadid); /* try bad pointer */ rc = acmMetrics((HACMOBJ)had, ACM_METRIC_MAX_SIZE_FORMAT, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_MAX_SIZE_FORMAT, &dwSize); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_MAX_SIZE_FORMAT, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)had, ACM_METRIC_MAX_SIZE_FORMAT, &dwSize); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); if (rc == MMSYSERR_NOERROR) { ACMFORMATDETAILSA fd; WAVEFORMATEX * pwfx; ACMFORMATTAGDETAILSA aftd; /* try bad pointer */ rc = acmFormatEnumA(had, 0, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad structure size */ ZeroMemory(&fd, sizeof(fd)); rc = acmFormatEnumA(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); fd.cbStruct = sizeof(fd) - 1; rc = acmFormatEnumA(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); if (dwSize < sizeof(WAVEFORMATEX)) dwSize = sizeof(WAVEFORMATEX); pwfx = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, dwSize); pwfx->cbSize = LOWORD(dwSize) - sizeof(WAVEFORMATEX); pwfx->wFormatTag = WAVE_FORMAT_UNKNOWN; fd.cbStruct = sizeof(fd); fd.pwfx = pwfx; fd.cbwfx = dwSize; fd.dwFormatTag = WAVE_FORMAT_UNKNOWN; /* try valid parameters */ rc = acmFormatEnumA(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_NOERROR, "acmFormatEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try bad pointer */ rc = acmFormatTagEnumA(had, 0, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad structure size */ ZeroMemory(&aftd, sizeof(aftd)); rc = acmFormatTagEnumA(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); aftd.cbStruct = sizeof(aftd) - 1; rc = acmFormatTagEnumA(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); aftd.cbStruct = sizeof(aftd); aftd.dwFormatTag = WAVE_FORMAT_UNKNOWN; /* try bad flag */ rc = acmFormatTagEnumA(had, &aftd, FormatTagEnumProc, 0, 1); ok(rc == MMSYSERR_INVALFLAG, "acmFormatTagEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmFormatTagEnumA(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_NOERROR, "acmFormatTagEnumA(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); HeapFree(GetProcessHeap(), 0, pwfx); /* try invalid handle */ rc = acmDriverClose((HACMDRIVER)1, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try invalid flag */ rc = acmDriverClose(had, 1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverClose(had, 0); ok(rc == MMSYSERR_NOERROR, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try closing again */ rc = acmDriverClose(had, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); } } return TRUE; }
static BOOL CALLBACK DriverEnumProc(HACMDRIVERID hadid, DWORD_PTR dwInstance, DWORD fdwSupport) { MMRESULT rc; ACMDRIVERDETAILS dd; HACMDRIVER had; DWORD dwDriverPriority; DWORD dwDriverSupport; if (winetest_interactive) { trace("id: %p\n", hadid); trace(" Supports:\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_ASYNC) trace(" async conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_CODEC) trace(" different format conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_CONVERTER) trace(" same format conversions\n"); if (fdwSupport & ACMDRIVERDETAILS_SUPPORTF_FILTER) trace(" filtering\n"); } /* try an invalid pointer */ rc = acmDriverDetails(hadid, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try an invalid structure size */ ZeroMemory(&dd, sizeof(dd)); rc = acmDriverDetails(hadid, &dd, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* MSDN says this should fail but it doesn't in practice */ dd.cbStruct = 4; rc = acmDriverDetails(hadid, &dd, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NOTSUPPORTED, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try an invalid handle */ dd.cbStruct = sizeof(dd); rc = acmDriverDetails((HACMDRIVERID)1, &dd, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try an invalid handle and pointer */ rc = acmDriverDetails((HACMDRIVERID)1, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try invalid details */ rc = acmDriverDetails(hadid, &dd, -1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverDetails(hadid, &dd, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NOTSUPPORTED, "acmDriverDetails(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* cbStruct should contain size of returned data (at most sizeof(dd)) TODO: should it be *exactly* sizeof(dd), as tested here? */ if (rc == MMSYSERR_NOERROR) { ok(dd.cbStruct == sizeof(dd), "acmDriverDetails(): cbStruct = %08x\n", dd.cbStruct); } if (rc == MMSYSERR_NOERROR && winetest_interactive) { trace(" Short name: %s\n", dd.szShortName); trace(" Long name: %s\n", dd.szLongName); trace(" Copyright: %s\n", dd.szCopyright); trace(" Licensing: %s\n", dd.szLicensing); trace(" Features: %s\n", dd.szFeatures); trace(" Supports %u formats\n", dd.cFormatTags); trace(" Supports %u filter formats\n", dd.cFilterTags); } /* try bad pointer */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_PRIORITY, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_PRIORITY, &dwDriverPriority); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_PRIORITY, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_PRIORITY, &dwDriverSupport); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try bad pointer */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_SUPPORT, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_SUPPORT, &dwDriverSupport); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_DRIVER_SUPPORT, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)hadid, ACM_METRIC_DRIVER_SUPPORT, &dwDriverSupport); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try invalid pointer */ rc = acmDriverOpen(0, hadid, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try invalid handle */ rc = acmDriverOpen(&had, (HACMDRIVERID)1, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try invalid open */ rc = acmDriverOpen(&had, hadid, -1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverOpen(&had, hadid, 0); ok(rc == MMSYSERR_NOERROR || rc == MMSYSERR_NODRIVER, "acmDriverOpen(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); if (rc == MMSYSERR_NOERROR) { DWORD dwSize; HACMDRIVERID hid; /* try bad pointer */ rc = acmDriverID((HACMOBJ)had, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmDriverID((HACMOBJ)1, &hid, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad handle and pointer */ rc = acmDriverID((HACMOBJ)1, 0, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad flag */ rc = acmDriverID((HACMOBJ)had, &hid, 1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverID((HACMOBJ)had, &hid, 0); ok(rc == MMSYSERR_NOERROR, "acmDriverID(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); ok(hid == hadid, "acmDriverID() returned ID %p doesn't equal %p\n", hid, hadid); /* try bad pointer */ rc = acmMetrics((HACMOBJ)had, ACM_METRIC_MAX_SIZE_FORMAT, 0); ok(rc == MMSYSERR_INVALPARAM, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_MAX_SIZE_FORMAT, &dwSize); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try bad pointer and handle */ rc = acmMetrics((HACMOBJ)1, ACM_METRIC_MAX_SIZE_FORMAT, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try valid parameters */ rc = acmMetrics((HACMOBJ)had, ACM_METRIC_MAX_SIZE_FORMAT, &dwSize); ok(rc == MMSYSERR_NOERROR, "acmMetrics(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); if (rc == MMSYSERR_NOERROR) { ACMFORMATDETAILS fd; WAVEFORMATEX * pwfx; ACMFORMATTAGDETAILS aftd; /* try bad pointer */ rc = acmFormatEnum(had, 0, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad structure size */ ZeroMemory(&fd, sizeof(fd)); rc = acmFormatEnum(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); fd.cbStruct = sizeof(fd) - 1; rc = acmFormatEnum(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); if (dwSize < sizeof(WAVEFORMATEX)) dwSize = sizeof(WAVEFORMATEX); pwfx = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, dwSize); pwfx->cbSize = LOWORD(dwSize) - sizeof(WAVEFORMATEX); pwfx->wFormatTag = WAVE_FORMAT_UNKNOWN; fd.cbStruct = sizeof(fd); fd.pwfx = pwfx; fd.cbwfx = dwSize; fd.dwFormatTag = WAVE_FORMAT_UNKNOWN; /* try valid parameters */ rc = acmFormatEnum(had, &fd, FormatEnumProc, 0, 0); ok(rc == MMSYSERR_NOERROR, "acmFormatEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try bad pointer */ rc = acmFormatTagEnum(had, 0, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); /* try bad structure size */ ZeroMemory(&aftd, sizeof(aftd)); rc = acmFormatTagEnum(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); aftd.cbStruct = sizeof(aftd) - 1; rc = acmFormatTagEnum(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_INVALPARAM, "acmFormatTagEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALPARAM); aftd.cbStruct = sizeof(aftd); aftd.dwFormatTag = WAVE_FORMAT_UNKNOWN; /* try bad flag */ rc = acmFormatTagEnum(had, &aftd, FormatTagEnumProc, 0, 1); ok(rc == MMSYSERR_INVALFLAG, "acmFormatTagEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmFormatTagEnum(had, &aftd, FormatTagEnumProc, 0, 0); ok(rc == MMSYSERR_NOERROR, "acmFormatTagEnum(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); HeapFree(GetProcessHeap(), 0, pwfx); /* try invalid handle */ rc = acmDriverClose((HACMDRIVER)1, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); /* try invalid flag */ rc = acmDriverClose(had, 1); ok(rc == MMSYSERR_INVALFLAG, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALFLAG); /* try valid parameters */ rc = acmDriverClose(had, 0); ok(rc == MMSYSERR_NOERROR, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_NOERROR); /* try closing again */ rc = acmDriverClose(had, 0); ok(rc == MMSYSERR_INVALHANDLE, "acmDriverClose(): rc = %08x, should be %08x\n", rc, MMSYSERR_INVALHANDLE); } } return TRUE; }