1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
6 * Copyright 1998 Patrik Stridvall
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
25 #include "wine/port.h"
40 #include "wine/debug.h"
41 #include "wine/unicode.h"
43 WINE_DEFAULT_DEBUG_CHANNEL(msacm
);
45 /***********************************************************************
46 * acmDriverAddA (MSACM32.@)
48 MMRESULT WINAPI
acmDriverAddA(PHACMDRIVERID phadid
, HINSTANCE hinstModule
,
49 LPARAM lParam
, DWORD dwPriority
, DWORD fdwAdd
)
52 WCHAR
* driverW
= NULL
;
53 LPARAM lParamW
= lParam
;
55 TRACE("(%p, %p, %08lx, %08x, %08x)\n",
56 phadid
, hinstModule
, lParam
, dwPriority
, fdwAdd
);
59 WARN("invalid parameter\n");
60 return MMSYSERR_INVALPARAM
;
63 /* Check if any unknown flags */
65 ~(ACM_DRIVERADDF_FUNCTION
|ACM_DRIVERADDF_NOTIFYHWND
|
66 ACM_DRIVERADDF_GLOBAL
)) {
67 WARN("invalid flag\n");
68 return MMSYSERR_INVALFLAG
;
71 /* Check if any incompatible flags */
72 if ((fdwAdd
& ACM_DRIVERADDF_FUNCTION
) &&
73 (fdwAdd
& ACM_DRIVERADDF_NOTIFYHWND
)) {
74 WARN("invalid flag\n");
75 return MMSYSERR_INVALFLAG
;
78 /* A->W translation of name */
79 if ((fdwAdd
& ACM_DRIVERADDF_TYPEMASK
) == ACM_DRIVERADDF_NAME
) {
82 if (lParam
== 0) return MMSYSERR_INVALPARAM
;
83 len
= MultiByteToWideChar(CP_ACP
, 0, (LPSTR
)lParam
, -1, NULL
, 0);
84 driverW
= HeapAlloc(MSACM_hHeap
, 0, len
* sizeof(WCHAR
));
85 if (!driverW
) return MMSYSERR_NOMEM
;
86 MultiByteToWideChar(CP_ACP
, 0, (LPSTR
)lParam
, -1, driverW
, len
);
87 lParamW
= (LPARAM
)driverW
;
90 resultW
= acmDriverAddW(phadid
, hinstModule
, lParamW
, dwPriority
, fdwAdd
);
91 HeapFree(MSACM_hHeap
, 0, driverW
);
95 /***********************************************************************
96 * acmDriverAddW (MSACM32.@)
99 MMRESULT WINAPI
acmDriverAddW(PHACMDRIVERID phadid
, HINSTANCE hinstModule
,
100 LPARAM lParam
, DWORD dwPriority
, DWORD fdwAdd
)
102 PWINE_ACMLOCALDRIVER pLocalDrv
= NULL
;
104 TRACE("(%p, %p, %08lx, %08x, %08x)\n",
105 phadid
, hinstModule
, lParam
, dwPriority
, fdwAdd
);
108 WARN("invalid parameter\n");
109 return MMSYSERR_INVALPARAM
;
112 /* Check if any unknown flags */
114 ~(ACM_DRIVERADDF_FUNCTION
|ACM_DRIVERADDF_NOTIFYHWND
|
115 ACM_DRIVERADDF_GLOBAL
)) {
116 WARN("invalid flag\n");
117 return MMSYSERR_INVALFLAG
;
120 /* Check if any incompatible flags */
121 if ((fdwAdd
& ACM_DRIVERADDF_FUNCTION
) &&
122 (fdwAdd
& ACM_DRIVERADDF_NOTIFYHWND
)) {
123 WARN("invalid flag\n");
124 return MMSYSERR_INVALFLAG
;
127 switch (fdwAdd
& ACM_DRIVERADDF_TYPEMASK
) {
128 case ACM_DRIVERADDF_NAME
:
131 lParam name of value in HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Drivers32
132 dwPriority (unused, set to 0)
134 *phadid
= (HACMDRIVERID
) MSACM_RegisterDriverFromRegistry((LPCWSTR
)lParam
);
136 ERR("Unable to register driver via ACM_DRIVERADDF_NAME\n");
137 return MMSYSERR_INVALPARAM
;
140 case ACM_DRIVERADDF_FUNCTION
:
142 hInstModule Handle of module which contains driver entry proc
143 lParam Driver function address
144 dwPriority (unused, set to 0)
146 fdwAdd
&= ~ACM_DRIVERADDF_TYPEMASK
;
147 /* FIXME: fdwAdd ignored */
148 /* Application-supplied acmDriverProc's are placed at the top of the priority unless
149 fdwAdd indicates ACM_DRIVERADDF_GLOBAL
151 pLocalDrv
= MSACM_RegisterLocalDriver(hinstModule
, (DRIVERPROC
)lParam
);
152 *phadid
= pLocalDrv
? (HACMDRIVERID
) MSACM_RegisterDriver(NULL
, NULL
, pLocalDrv
) : NULL
;
154 ERR("Unable to register driver via ACM_DRIVERADDF_FUNCTION\n");
155 return MMSYSERR_INVALPARAM
;
158 case ACM_DRIVERADDF_NOTIFYHWND
:
161 lParam Handle of notification window
162 dwPriority Window message to send for notification broadcasts
164 *phadid
= (HACMDRIVERID
) MSACM_RegisterNotificationWindow((HWND
)lParam
, dwPriority
);
166 ERR("Unable to register driver via ACM_DRIVERADDF_NOTIFYHWND\n");
167 return MMSYSERR_INVALPARAM
;
171 ERR("invalid flag value 0x%08x for fdwAdd\n", fdwAdd
& ACM_DRIVERADDF_TYPEMASK
);
172 return MMSYSERR_INVALFLAG
;
175 MSACM_BroadcastNotification();
176 return MMSYSERR_NOERROR
;
179 /***********************************************************************
180 * acmDriverClose (MSACM32.@)
182 MMRESULT WINAPI
acmDriverClose(HACMDRIVER had
, DWORD fdwClose
)
185 PWINE_ACMDRIVERID padid
;
186 PWINE_ACMDRIVER
* tpad
;
188 TRACE("(%p, %08x)\n", had
, fdwClose
);
191 WARN("invalid flag\n");
192 return MMSYSERR_INVALFLAG
;
195 pad
= MSACM_GetDriver(had
);
197 WARN("invalid handle\n");
198 return MMSYSERR_INVALHANDLE
;
201 padid
= pad
->obj
.pACMDriverID
;
203 /* remove driver from list */
204 for (tpad
= &(padid
->pACMDriverList
); *tpad
; tpad
= &((*tpad
)->pNextACMDriver
)) {
206 *tpad
= (*tpad
)->pNextACMDriver
;
211 /* close driver if it has been opened */
212 if (pad
->hDrvr
&& !pad
->pLocalDrvrInst
)
213 CloseDriver(pad
->hDrvr
, 0, 0);
214 else if (pad
->pLocalDrvrInst
)
215 MSACM_CloseLocalDriver(pad
->pLocalDrvrInst
);
218 HeapFree(MSACM_hHeap
, 0, pad
);
220 return MMSYSERR_NOERROR
;
223 /***********************************************************************
224 * acmDriverDetailsA (MSACM32.@)
226 MMRESULT WINAPI
acmDriverDetailsA(HACMDRIVERID hadid
, PACMDRIVERDETAILSA padd
, DWORD fdwDetails
)
229 ACMDRIVERDETAILSW addw
;
231 TRACE("(%p, %p, %08x)\n", hadid
, padd
, fdwDetails
);
234 WARN("invalid parameter\n");
235 return MMSYSERR_INVALPARAM
;
238 if (padd
->cbStruct
< 4) {
239 WARN("invalid parameter\n");
240 return MMSYSERR_INVALPARAM
;
243 addw
.cbStruct
= sizeof(addw
);
244 mmr
= acmDriverDetailsW(hadid
, &addw
, fdwDetails
);
246 ACMDRIVERDETAILSA padda
;
248 padda
.fccType
= addw
.fccType
;
249 padda
.fccComp
= addw
.fccComp
;
250 padda
.wMid
= addw
.wMid
;
251 padda
.wPid
= addw
.wPid
;
252 padda
.vdwACM
= addw
.vdwACM
;
253 padda
.vdwDriver
= addw
.vdwDriver
;
254 padda
.fdwSupport
= addw
.fdwSupport
;
255 padda
.cFormatTags
= addw
.cFormatTags
;
256 padda
.cFilterTags
= addw
.cFilterTags
;
257 padda
.hicon
= addw
.hicon
;
258 WideCharToMultiByte( CP_ACP
, 0, addw
.szShortName
, -1, padda
.szShortName
,
259 sizeof(padda
.szShortName
), NULL
, NULL
);
260 WideCharToMultiByte( CP_ACP
, 0, addw
.szLongName
, -1, padda
.szLongName
,
261 sizeof(padda
.szLongName
), NULL
, NULL
);
262 WideCharToMultiByte( CP_ACP
, 0, addw
.szCopyright
, -1, padda
.szCopyright
,
263 sizeof(padda
.szCopyright
), NULL
, NULL
);
264 WideCharToMultiByte( CP_ACP
, 0, addw
.szLicensing
, -1, padda
.szLicensing
,
265 sizeof(padda
.szLicensing
), NULL
, NULL
);
266 WideCharToMultiByte( CP_ACP
, 0, addw
.szFeatures
, -1, padda
.szFeatures
,
267 sizeof(padda
.szFeatures
), NULL
, NULL
);
268 padda
.cbStruct
= min(padd
->cbStruct
, sizeof(*padd
));
269 memcpy(padd
, &padda
, padda
.cbStruct
);
274 /***********************************************************************
275 * acmDriverDetailsW (MSACM32.@)
277 MMRESULT WINAPI
acmDriverDetailsW(HACMDRIVERID hadid
, PACMDRIVERDETAILSW padd
, DWORD fdwDetails
)
282 TRACE("(%p, %p, %08x)\n", hadid
, padd
, fdwDetails
);
285 WARN("invalid parameter\n");
286 return MMSYSERR_INVALPARAM
;
289 if (padd
->cbStruct
< 4) {
290 WARN("invalid parameter\n");
291 return MMSYSERR_INVALPARAM
;
295 WARN("invalid flag\n");
296 return MMSYSERR_INVALFLAG
;
299 mmr
= acmDriverOpen(&acmDrvr
, hadid
, 0);
300 if (mmr
== MMSYSERR_NOERROR
) {
301 ACMDRIVERDETAILSW paddw
;
302 paddw
.cbStruct
= sizeof(paddw
);
303 mmr
= MSACM_Message(acmDrvr
, ACMDM_DRIVER_DETAILS
, (LPARAM
)&paddw
, 0);
305 acmDriverClose(acmDrvr
, 0);
306 paddw
.cbStruct
= min(padd
->cbStruct
, sizeof(*padd
));
307 memcpy(padd
, &paddw
, paddw
.cbStruct
);
309 else if (mmr
== MMSYSERR_NODRIVER
)
310 return MMSYSERR_NOTSUPPORTED
;
315 /***********************************************************************
316 * acmDriverEnum (MSACM32.@)
318 MMRESULT WINAPI
acmDriverEnum(ACMDRIVERENUMCB fnCallback
, DWORD_PTR dwInstance
,
321 PWINE_ACMDRIVERID padid
;
324 TRACE("(%p, %08lx, %08x)\n", fnCallback
, dwInstance
, fdwEnum
);
327 WARN("invalid parameter\n");
328 return MMSYSERR_INVALPARAM
;
331 if (fdwEnum
& ~(ACM_DRIVERENUMF_NOLOCAL
|ACM_DRIVERENUMF_DISABLED
)) {
332 WARN("invalid flag\n");
333 return MMSYSERR_INVALFLAG
;
336 for (padid
= MSACM_pFirstACMDriverID
; padid
; padid
= padid
->pNextACMDriverID
) {
337 fdwSupport
= padid
->fdwSupport
;
339 if (padid
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_DISABLED
) {
340 if (fdwEnum
& ACM_DRIVERENUMF_DISABLED
)
341 fdwSupport
|= ACMDRIVERDETAILS_SUPPORTF_DISABLED
;
345 if (!(*fnCallback
)((HACMDRIVERID
)padid
, dwInstance
, fdwSupport
))
349 return MMSYSERR_NOERROR
;
352 /***********************************************************************
353 * acmDriverID (MSACM32.@)
355 MMRESULT WINAPI
acmDriverID(HACMOBJ hao
, PHACMDRIVERID phadid
, DWORD fdwDriverID
)
359 TRACE("(%p, %p, %08x)\n", hao
, phadid
, fdwDriverID
);
362 WARN("invalid flag\n");
363 return MMSYSERR_INVALFLAG
;
366 pao
= MSACM_GetObj(hao
, WINE_ACMOBJ_DONTCARE
);
368 WARN("invalid handle\n");
369 return MMSYSERR_INVALHANDLE
;
373 WARN("invalid parameter\n");
374 return MMSYSERR_INVALPARAM
;
377 *phadid
= (HACMDRIVERID
) pao
->pACMDriverID
;
379 return MMSYSERR_NOERROR
;
382 /***********************************************************************
383 * acmDriverMessage (MSACM32.@)
385 * Note: MSDN documentation (July 2001) is incomplete. This function
386 * accepts sending messages to an HACMDRIVERID in addition to the
387 * documented HACMDRIVER. In fact, for DRV_QUERYCONFIGURE and DRV_CONFIGURE,
388 * this might actually be the required mode of operation.
390 * Note: For DRV_CONFIGURE, msacm supplies its own DRVCONFIGINFO structure
391 * when the application fails to supply one. Some native drivers depend on
392 * this and refuse to display unless a valid DRVCONFIGINFO structure is
393 * built and supplied.
395 LRESULT WINAPI
acmDriverMessage(HACMDRIVER had
, UINT uMsg
, LPARAM lParam1
, LPARAM lParam2
)
397 TRACE("(%p, %04x, %08lx, %08lx\n", had
, uMsg
, lParam1
, lParam2
);
399 if ((uMsg
>= ACMDM_USER
&& uMsg
< ACMDM_RESERVED_LOW
) ||
400 uMsg
== ACMDM_DRIVER_ABOUT
||
401 uMsg
== DRV_QUERYCONFIGURE
||
402 uMsg
== DRV_CONFIGURE
)
404 PWINE_ACMDRIVERID padid
;
406 LPDRVCONFIGINFO pConfigInfo
= NULL
;
407 LPWSTR section_name
= NULL
;
408 LPWSTR alias_name
= NULL
;
410 /* Check whether handle is an HACMDRIVERID */
411 padid
= MSACM_GetDriverID((HACMDRIVERID
)had
);
413 /* If the message is DRV_CONFIGURE, and the application provides no
414 DRVCONFIGINFO structure, msacm must supply its own.
416 if (uMsg
== DRV_CONFIGURE
&& lParam2
== 0) {
419 /* Get the alias from the HACMDRIVERID */
421 pAlias
= padid
->pszDriverAlias
;
422 if (pAlias
== NULL
) {
423 WARN("DRV_CONFIGURE: no alias for this driver, cannot self-supply alias\n");
426 FIXME("DRV_CONFIGURE: reverse lookup HACMDRIVER -> HACMDRIVERID not implemented\n");
430 if (pAlias
!= NULL
) {
431 /* DRVCONFIGINFO is only 12 bytes long, but native msacm
432 * reports a 16-byte structure to codecs, so allocate 16 bytes,
433 * just to be on the safe side.
435 const unsigned int iStructSize
= 16;
436 pConfigInfo
= HeapAlloc(MSACM_hHeap
, 0, iStructSize
);
438 ERR("OOM while supplying DRVCONFIGINFO for DRV_CONFIGURE, using NULL\n");
440 static const WCHAR drivers32
[] = {'D','r','i','v','e','r','s','3','2','\0'};
442 pConfigInfo
->dwDCISize
= iStructSize
;
444 section_name
= HeapAlloc(MSACM_hHeap
, 0, (strlenW(drivers32
) + 1) * sizeof(WCHAR
));
445 if (section_name
) strcpyW(section_name
, drivers32
);
446 pConfigInfo
->lpszDCISectionName
= section_name
;
447 alias_name
= HeapAlloc(MSACM_hHeap
, 0, (strlenW(pAlias
) + 1) * sizeof(WCHAR
));
448 if (alias_name
) strcpyW(alias_name
, pAlias
);
449 pConfigInfo
->lpszDCIAliasName
= alias_name
;
451 if (pConfigInfo
->lpszDCISectionName
== NULL
|| pConfigInfo
->lpszDCIAliasName
== NULL
) {
452 HeapFree(MSACM_hHeap
, 0, alias_name
);
453 HeapFree(MSACM_hHeap
, 0, section_name
);
454 HeapFree(MSACM_hHeap
, 0, pConfigInfo
);
456 ERR("OOM while supplying DRVCONFIGINFO for DRV_CONFIGURE, using NULL\n");
461 lParam2
= (LPARAM
)pConfigInfo
;
465 /* Handle is really an HACMDRIVERID, must have an open session to get an HACMDRIVER */
466 if (padid
->pACMDriverList
!= NULL
) {
467 lResult
= MSACM_Message((HACMDRIVER
)padid
->pACMDriverList
, uMsg
, lParam1
, lParam2
);
469 MMRESULT mmr
= acmDriverOpen(&had
, (HACMDRIVERID
)padid
, 0);
470 if (mmr
!= MMSYSERR_NOERROR
) {
471 lResult
= MMSYSERR_INVALPARAM
;
473 lResult
= acmDriverMessage(had
, uMsg
, lParam1
, lParam2
);
474 acmDriverClose(had
, 0);
478 lResult
= MSACM_Message(had
, uMsg
, lParam1
, lParam2
);
481 HeapFree(MSACM_hHeap
, 0, alias_name
);
482 HeapFree(MSACM_hHeap
, 0, section_name
);
483 HeapFree(MSACM_hHeap
, 0, pConfigInfo
);
487 WARN("invalid parameter\n");
488 return MMSYSERR_INVALPARAM
;
491 /***********************************************************************
492 * acmDriverOpen (MSACM32.@)
494 MMRESULT WINAPI
acmDriverOpen(PHACMDRIVER phad
, HACMDRIVERID hadid
, DWORD fdwOpen
)
496 PWINE_ACMDRIVERID padid
;
497 PWINE_ACMDRIVER pad
= NULL
;
500 TRACE("(%p, %p, %08u)\n", phad
, hadid
, fdwOpen
);
503 WARN("invalid parameter\n");
504 return MMSYSERR_INVALPARAM
;
508 WARN("invalid flag\n");
509 return MMSYSERR_INVALFLAG
;
512 padid
= MSACM_GetDriverID(hadid
);
514 WARN("invalid handle\n");
515 return MMSYSERR_INVALHANDLE
;
518 pad
= HeapAlloc(MSACM_hHeap
, 0, sizeof(WINE_ACMDRIVER
));
521 return MMSYSERR_NOMEM
;
524 pad
->obj
.dwType
= WINE_ACMOBJ_DRIVER
;
525 pad
->obj
.pACMDriverID
= padid
;
527 pad
->pLocalDrvrInst
= NULL
;
529 if (padid
->pLocalDriver
== NULL
)
531 ACMDRVOPENDESCW adod
;
535 /* this is not an externally added driver... need to actually load it */
536 if (!padid
->pszDriverAlias
)
538 ret
= MMSYSERR_ERROR
;
542 adod
.cbStruct
= sizeof(adod
);
543 adod
.fccType
= ACMDRIVERDETAILS_FCCTYPE_AUDIOCODEC
;
544 adod
.fccComp
= ACMDRIVERDETAILS_FCCCOMP_UNDEFINED
;
545 adod
.dwVersion
= acmGetVersion();
546 adod
.dwFlags
= fdwOpen
;
548 len
= strlen("Drivers32") + 1;
549 section_name
= HeapAlloc(MSACM_hHeap
, 0, len
* sizeof(WCHAR
));
550 MultiByteToWideChar(CP_ACP
, 0, "Drivers32", -1, section_name
, len
);
551 adod
.pszSectionName
= section_name
;
552 adod
.pszAliasName
= padid
->pszDriverAlias
;
555 pad
->hDrvr
= OpenDriver(padid
->pszDriverAlias
, NULL
, (DWORD_PTR
)&adod
);
557 HeapFree(MSACM_hHeap
, 0, section_name
);
561 if (ret
== MMSYSERR_NOERROR
)
562 ret
= MMSYSERR_NODRIVER
;
568 ACMDRVOPENDESCW adod
;
572 adod
.cbStruct
= sizeof(adod
);
573 adod
.fccType
= ACMDRIVERDETAILS_FCCTYPE_AUDIOCODEC
;
574 adod
.fccComp
= ACMDRIVERDETAILS_FCCCOMP_UNDEFINED
;
575 adod
.dwVersion
= acmGetVersion();
576 adod
.dwFlags
= fdwOpen
;
578 adod
.pszSectionName
= NULL
;
579 adod
.pszAliasName
= NULL
;
582 pad
->pLocalDrvrInst
= MSACM_OpenLocalDriver(padid
->pLocalDriver
, (DWORD_PTR
)&adod
);
583 if (!pad
->pLocalDrvrInst
)
586 if (ret
== MMSYSERR_NOERROR
)
587 ret
= MMSYSERR_NODRIVER
;
592 /* insert new pad at beg of list */
593 pad
->pNextACMDriver
= padid
->pACMDriverList
;
594 padid
->pACMDriverList
= pad
;
596 /* FIXME: Create a WINE_ACMDRIVER32 */
597 *phad
= (HACMDRIVER
)pad
;
598 TRACE("%s => %p\n", debugstr_w(padid
->pszDriverAlias
), pad
);
600 return MMSYSERR_NOERROR
;
602 WARN("failed: ret = %08x\n", ret
);
603 if (pad
&& !pad
->hDrvr
)
604 HeapFree(MSACM_hHeap
, 0, pad
);
608 /***********************************************************************
609 * acmDriverPriority (MSACM32.@)
611 MMRESULT WINAPI
acmDriverPriority(HACMDRIVERID hadid
, DWORD dwPriority
, DWORD fdwPriority
)
614 TRACE("(%p, %08x, %08x)\n", hadid
, dwPriority
, fdwPriority
);
616 /* Check for unknown flags */
618 ~(ACM_DRIVERPRIORITYF_ENABLE
|ACM_DRIVERPRIORITYF_DISABLE
|
619 ACM_DRIVERPRIORITYF_BEGIN
|ACM_DRIVERPRIORITYF_END
)) {
620 WARN("invalid flag\n");
621 return MMSYSERR_INVALFLAG
;
624 /* Check for incompatible flags */
625 if ((fdwPriority
& ACM_DRIVERPRIORITYF_ENABLE
) &&
626 (fdwPriority
& ACM_DRIVERPRIORITYF_DISABLE
)) {
627 WARN("invalid flag\n");
628 return MMSYSERR_INVALFLAG
;
631 /* Check for incompatible flags */
632 if ((fdwPriority
& ACM_DRIVERPRIORITYF_BEGIN
) &&
633 (fdwPriority
& ACM_DRIVERPRIORITYF_END
)) {
634 WARN("invalid flag\n");
635 return MMSYSERR_INVALFLAG
;
638 /* According to MSDN, ACM_DRIVERPRIORITYF_BEGIN and ACM_DRIVERPRIORITYF_END
639 may only appear by themselves, and in addition, hadid and dwPriority must
641 if ((fdwPriority
& ACM_DRIVERPRIORITYF_BEGIN
) ||
642 (fdwPriority
& ACM_DRIVERPRIORITYF_END
)) {
643 if (fdwPriority
& ~(ACM_DRIVERPRIORITYF_BEGIN
|ACM_DRIVERPRIORITYF_END
)) {
644 WARN("ACM_DRIVERPRIORITYF_[BEGIN|END] cannot be used with any other flags\n");
645 return MMSYSERR_INVALPARAM
;
648 WARN("priority invalid with ACM_DRIVERPRIORITYF_[BEGIN|END]\n");
649 return MMSYSERR_INVALPARAM
;
652 WARN("non-null hadid invalid with ACM_DRIVERPRIORITYF_[BEGIN|END]\n");
653 return MMSYSERR_INVALPARAM
;
655 /* FIXME: MSDN wording suggests that deferred notification should be
656 implemented as a system-wide lock held by a calling task, and that
657 re-enabling notifications should broadcast them across all processes.
658 This implementation uses a simple DWORD counter. One consequence of the
659 current implementation is that applications will never see
660 MMSYSERR_ALLOCATED as a return error.
662 if (fdwPriority
& ACM_DRIVERPRIORITYF_BEGIN
) {
663 MSACM_DisableNotifications();
664 } else if (fdwPriority
& ACM_DRIVERPRIORITYF_END
) {
665 MSACM_EnableNotifications();
667 return MMSYSERR_NOERROR
;
669 PWINE_ACMDRIVERID padid
;
670 PWINE_ACMNOTIFYWND panwnd
;
671 BOOL bPerformBroadcast
= FALSE
;
673 /* Fetch driver ID */
674 padid
= MSACM_GetDriverID(hadid
);
675 panwnd
= MSACM_GetNotifyWnd(hadid
);
676 if (!padid
&& !panwnd
) {
677 WARN("invalid handle\n");
678 return MMSYSERR_INVALHANDLE
;
682 /* Check whether driver ID is appropriate for requested op */
684 if (padid
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_LOCAL
) {
685 return MMSYSERR_NOTSUPPORTED
;
687 if (dwPriority
!= 1 && dwPriority
!= (DWORD
)-1) {
688 FIXME("unexpected priority %d, using sign only\n", dwPriority
);
689 if ((signed)dwPriority
< 0) dwPriority
= (DWORD
)-1;
690 if (dwPriority
> 0) dwPriority
= 1;
693 if (dwPriority
== 1 && (padid
->pPrevACMDriverID
== NULL
||
694 (padid
->pPrevACMDriverID
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_LOCAL
))) {
695 /* do nothing - driver is first of list, or first after last
697 } else if (dwPriority
== (DWORD
)-1 && padid
->pNextACMDriverID
== NULL
) {
698 /* do nothing - driver is last of list */
700 MSACM_RePositionDriver(padid
, dwPriority
);
701 bPerformBroadcast
= TRUE
;
705 /* Check whether driver ID should be enabled or disabled */
706 if (fdwPriority
& ACM_DRIVERPRIORITYF_DISABLE
) {
707 if (!(padid
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_DISABLED
)) {
708 padid
->fdwSupport
|= ACMDRIVERDETAILS_SUPPORTF_DISABLED
;
709 bPerformBroadcast
= TRUE
;
711 } else if (fdwPriority
& ACM_DRIVERPRIORITYF_ENABLE
) {
712 if (padid
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_DISABLED
) {
713 padid
->fdwSupport
&= ~ACMDRIVERDETAILS_SUPPORTF_DISABLED
;
714 bPerformBroadcast
= TRUE
;
721 return MMSYSERR_NOTSUPPORTED
;
724 /* Check whether notify window should be enabled or disabled */
725 if (fdwPriority
& ACM_DRIVERPRIORITYF_DISABLE
) {
726 if (!(panwnd
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_DISABLED
)) {
727 panwnd
->fdwSupport
|= ACMDRIVERDETAILS_SUPPORTF_DISABLED
;
728 bPerformBroadcast
= TRUE
;
730 } else if (fdwPriority
& ACM_DRIVERPRIORITYF_ENABLE
) {
731 if (panwnd
->fdwSupport
& ACMDRIVERDETAILS_SUPPORTF_DISABLED
) {
732 panwnd
->fdwSupport
&= ~ACMDRIVERDETAILS_SUPPORTF_DISABLED
;
733 bPerformBroadcast
= TRUE
;
738 /* Perform broadcast of changes */
739 if (bPerformBroadcast
) {
740 MSACM_WriteCurrentPriorities();
741 MSACM_BroadcastNotification();
743 return MMSYSERR_NOERROR
;
747 /***********************************************************************
748 * acmDriverRemove (MSACM32.@)
750 MMRESULT WINAPI
acmDriverRemove(HACMDRIVERID hadid
, DWORD fdwRemove
)
752 PWINE_ACMDRIVERID padid
;
753 PWINE_ACMNOTIFYWND panwnd
;
755 TRACE("(%p, %08x)\n", hadid
, fdwRemove
);
757 padid
= MSACM_GetDriverID(hadid
);
758 panwnd
= MSACM_GetNotifyWnd(hadid
);
759 if (!padid
&& !panwnd
) {
760 WARN("invalid handle\n");
761 return MMSYSERR_INVALHANDLE
;
765 WARN("invalid flag\n");
766 return MMSYSERR_INVALFLAG
;
769 if (padid
) MSACM_UnregisterDriver(padid
);
770 if (panwnd
) MSACM_UnRegisterNotificationWindow(panwnd
);
771 MSACM_BroadcastNotification();
773 return MMSYSERR_NOERROR
;