ReactOS 0.4.16-dev-2206-gc56950d
sup.c
Go to the documentation of this file.
1/*
2 * COPYRIGHT: See COPYING in the top level directory
3 * PROJECT: ReactOS Kernel Streaming
4 * FILE: lib/drivers/sound/mmixer/sup.c
5 * PURPOSE: Mixer Support Functions
6 * PROGRAMMER: Johannes Anderwald
7 */
8
9#include "precomp.h"
10
11// #define NDEBUG
12#include <debug.h>
13
14const GUID KSNODETYPE_SUM = {0xDA441A60L, 0xC556, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
15const GUID KSNODETYPE_DAC = {0x507AE360L, 0xC554, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
16const GUID KSNODETYPE_ADC = {0x4D837FE0L, 0xC555, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
17const GUID KSNODETYPE_AGC = {0xE88C9BA0L, 0xC557, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
18const GUID KSNODETYPE_LOUDNESS = {0x41887440L, 0xC558, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
19const GUID KSNODETYPE_MUTE = {0x02B223C0L, 0xC557, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
20const GUID KSNODETYPE_TONE = {0x7607E580L, 0xC557, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
21const GUID KSNODETYPE_VOLUME = {0x3A5ACC00L, 0xC557, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
22const GUID KSNODETYPE_PEAKMETER = {0xa085651e, 0x5f0d, 0x4b36, {0xa8, 0x69, 0xd1, 0x95, 0xd6, 0xab, 0x4b, 0x9e}};
23const GUID KSNODETYPE_MUX = {0x2CEAF780, 0xC556, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
24const GUID KSNODETYPE_STEREO_WIDE = {0xA9E69800L, 0xC558, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
25const GUID KSNODETYPE_CHORUS = {0x20173F20L, 0xC559, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
26const GUID KSNODETYPE_REVERB = {0xEF0328E0L, 0xC558, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
27const GUID KSNODETYPE_SUPERMIX = {0xE573ADC0L, 0xC555, 0x11D0, {0x8A, 0x2B, 0x00, 0xA0, 0xC9, 0x25, 0x5A, 0xC1}};
28
29const GUID KSPROPSETID_Audio = {0x45FFAAA0L, 0x6E1B, 0x11D0, {0xBC, 0xF2, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}};
30const GUID KSPROPSETID_Pin = {0x8C134960L, 0x51AD, 0x11CF, {0x87, 0x8A, 0x94, 0xF8, 0x01, 0xC1, 0x00, 0x00}};
31const GUID KSPROPSETID_General = {0x1464EDA5L, 0x6A8F, 0x11D1, {0x9A, 0xA7, 0x00, 0xA0, 0xC9, 0x22, 0x31, 0x96}};
32const GUID KSPROPSETID_Topology = {0x720D4AC0L, 0x7533, 0x11D0, {0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00}};
33const GUID KSEVENTSETID_AudioControlChange = {0xE85E9698L, 0xFA2F, 0x11D1, {0x95, 0xBD, 0x00, 0xC0, 0x4F, 0xB9, 0x25, 0xD3}};
34
35const GUID KSDATAFORMAT_TYPE_MUSIC = {0xE725D360L, 0x62CC, 0x11CF, {0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00}};
36const GUID KSDATAFORMAT_SUBTYPE_MIDI = {0x1D262760L, 0xE957, 0x11CF, {0xA5, 0xD6, 0x28, 0xDB, 0x04, 0xC1, 0x00, 0x00}};
37const GUID KSDATAFORMAT_SPECIFIER_NONE = {0x0F6417D6L, 0xC318, 0x11D0, {0xA4, 0x3F, 0x00, 0xA0, 0xC9, 0x22, 0x31, 0x96}};
38
42{
45
50
53
54 return MM_STATUS_SUCCESS;
55}
56
59 IN LPMIXER_INFO MixerInfo,
60 IN ULONG NodeID)
61{
62 PLIST_ENTRY Entry, ControlEntry;
63 LPMIXERLINE_EXT MixerLineSrc;
64 LPMIXERCONTROL_EXT MixerControl;
65
66 /* get first entry */
67 Entry = MixerInfo->LineList.Flink;
68
69 while(Entry != &MixerInfo->LineList)
70 {
72
73 ControlEntry = MixerLineSrc->ControlsList.Flink;
74 while(ControlEntry != &MixerLineSrc->ControlsList)
75 {
76 MixerControl = (LPMIXERCONTROL_EXT)CONTAINING_RECORD(ControlEntry, MIXERCONTROL_EXT, Entry);
77 if (MixerControl->NodeID == NodeID)
78 {
79 return MixerLineSrc;
80 }
81 ControlEntry = ControlEntry->Flink;
82 }
83 Entry = Entry->Flink;
84 }
85
86 return NULL;
87}
88
89VOID
91 IN ULONG LogicalPinArrayCount,
92 IN PULONG LogicalPinArray,
93 OUT PULONG PinOffset)
94{
96 ULONG LowestId = 0;
97
98 for(Index = 1; Index < LogicalPinArrayCount; Index++)
99 {
100 if (LogicalPinArray[Index] != MAXULONG)
101 {
102 /* sanity check: logical pin id must be unique */
103 ASSERT(LogicalPinArray[Index] != LogicalPinArray[LowestId]);
104 }
105
106 if (LogicalPinArray[Index] < LogicalPinArray[LowestId])
107 LowestId = Index;
108 }
109
110 /* store result */
111 *PinOffset = LowestId;
112}
113
114VOID
117 IN LPMIXER_INFO MixerInfo)
118{
119 /* UNIMPLEMENTED;
120 * FIXME
121 * free all lines
122 */
123
124 MixerContext->Free((PVOID)MixerInfo);
125}
126
130 IN HANDLE hDevice)
131{
132 LPMIXER_DATA MixerData;
134 PMIXER_LIST MixerList;
135
136 /* get mixer list */
138
139 if (!MixerList->MixerDataCount)
140 return NULL;
141
142 Entry = MixerList->MixerData.Flink;
143
144 while(Entry != &MixerList->MixerData)
145 {
147
148 if (MixerData->hDevice == hDevice)
149 return MixerData;
150
151 /* move to next mixer entry */
152 Entry = Entry->Flink;
153 }
154 return NULL;
155}
156
160 IN ULONG MixerIndex)
161{
162 LPMIXER_INFO MixerInfo;
164 PMIXER_LIST MixerList;
165 ULONG Index = 0;
166
167 /* get mixer list */
169
170 if (!MixerList->MixerListCount)
171 return NULL;
172
173 Entry = MixerList->MixerList.Flink;
174
175 while(Entry != &MixerList->MixerList)
176 {
178
179 if (Index == MixerIndex)
180 return MixerInfo;
181
182 /* move to next mixer entry */
183 Index++;
184 Entry = Entry->Flink;
185 }
186
187 return NULL;
188}
189
192 IN PMIXER_LIST MixerList,
193 IN LPWSTR MixerName,
194 OUT LPMIXER_INFO *OutMixerInfo)
195{
196 LPMIXER_INFO MixerInfo;
198
199 Entry = MixerList->MixerList.Flink;
200 while(Entry != &MixerList->MixerList)
201 {
203
204 DPRINT1("MixerName %S MixerName %S\n", MixerInfo->MixCaps.szPname, MixerName);
205 if (_wcsicmp(MixerInfo->MixCaps.szPname, MixerName) == 0)
206 {
207 *OutMixerInfo = MixerInfo;
208 return MM_STATUS_SUCCESS;
209 }
210 /* move to next mixer entry */
211 Entry = Entry->Flink;
212 }
213
215}
216
219 LPMIXER_INFO MixerInfo,
220 DWORD dwLineID)
221{
223 LPMIXERLINE_EXT MixerLineSrc;
224
225 /* get first entry */
226 Entry = MixerInfo->LineList.Flink;
227
228 while(Entry != &MixerInfo->LineList)
229 {
231 DPRINT("dwLineID %x dwLineID %x MixerLineSrc %p\n", MixerLineSrc->Line.dwLineID, dwLineID, MixerLineSrc);
232 if (MixerLineSrc->Line.dwLineID == dwLineID)
233 return MixerLineSrc;
234
235 Entry = Entry->Flink;
236 }
237
238 return NULL;
239}
240
241LPGUID
243 IN PKSMULTIPLE_ITEM MultipleItem,
244 IN ULONG Index)
245{
247
248 ASSERT(Index < MultipleItem->Count);
249
250 NodeType = (LPGUID)(MultipleItem + 1);
251 return &NodeType[Index];
252}
253
256 LPMIXER_INFO MixerInfo,
257 DWORD dwComponentType)
258{
260 LPMIXERLINE_EXT MixerLineSrc;
261
262 /* get first entry */
263 Entry = MixerInfo->LineList.Flink;
264
265 while(Entry != &MixerInfo->LineList)
266 {
268 if (MixerLineSrc->Line.dwComponentType == dwComponentType)
269 return MixerLineSrc;
270
271 Entry = Entry->Flink;
272 }
273
274 return NULL;
275}
276
279 LPMIXER_INFO MixerInfo,
280 DWORD dwControlID,
281 LPMIXERLINE_EXT *OutMixerLine,
282 LPMIXERCONTROL_EXT *OutMixerControl,
283 PULONG NodeId)
284{
285 PLIST_ENTRY Entry, ControlEntry;
286 LPMIXERLINE_EXT MixerLineSrc;
287 LPMIXERCONTROL_EXT MixerControl;
288
289 /* get first entry */
290 Entry = MixerInfo->LineList.Flink;
291
292 while(Entry != &MixerInfo->LineList)
293 {
295
296 ControlEntry = MixerLineSrc->ControlsList.Flink;
297 while(ControlEntry != &MixerLineSrc->ControlsList)
298 {
299 MixerControl = (LPMIXERCONTROL_EXT)CONTAINING_RECORD(ControlEntry, MIXERCONTROL_EXT, Entry);
300 if (MixerControl->Control.dwControlID == dwControlID)
301 {
302 if (OutMixerLine)
303 *OutMixerLine = MixerLineSrc;
304 if (OutMixerControl)
305 *OutMixerControl = MixerControl;
306 if (NodeId)
307 *NodeId = MixerControl->NodeID;
308 return MM_STATUS_SUCCESS;
309 }
310 ControlEntry = ControlEntry->Flink;
311 }
312 Entry = Entry->Flink;
313 }
314
316}
317
318VOID
321 IN LPMIXER_INFO MixerInfo,
323 IN ULONG Value)
324{
327
328 /* enumerate list and perform notification */
329 Entry = MixerInfo->EventList.Flink;
330 while(Entry != &MixerInfo->EventList)
331 {
332 /* get notification entry offset */
334
335 if (NotificationEntry->MixerEventRoutine)
336 {
337 /* now perform the callback */
338 NotificationEntry->MixerEventRoutine(NotificationEntry->MixerEventContext, (HANDLE)MixerInfo, NotificationType, Value);
339 }
340
341 /* move to next notification entry */
342 Entry = Entry->Flink;
343 }
344}
345
349 IN LPMIXER_INFO MixerInfo,
350 IN LPMIXERCONTROL_EXT MixerControl,
351 IN ULONG dwLineID,
352 IN LPMIXERCONTROLDETAILS MixerControlDetails,
353 IN ULONG bSet)
354{
356 LONG Value;
358
359 if (MixerControlDetails->cbDetails != sizeof(MIXERCONTROLDETAILS_BOOLEAN))
361
362 /* get input */
363 Input = (LPMIXERCONTROLDETAILS_BOOLEAN)MixerControlDetails->paDetails;
364
365 /* FIXME SEH */
366 if (bSet)
367 Value = Input->fValue;
368
369 /* set control details */
370 Status = MMixerSetGetControlDetails(MixerContext, MixerControl->hDevice, MixerControl->NodeID, bSet, KSPROPERTY_AUDIO_MUTE, 0, &Value);
371
373 return Status;
374
375 /* FIXME SEH */
376 if (!bSet)
377 {
378 Input->fValue = Value;
379 return Status;
380 }
381 else
382 {
383 /* notify wdmaud clients MM_MIXM_LINE_CHANGE dwLineID */
385 }
386
387 return Status;
388}
389
393 IN LPMIXER_INFO MixerInfo,
394 IN ULONG NodeId,
395 IN ULONG bSet,
396 IN ULONG Flags,
397 IN LPMIXERCONTROL_EXT MixerControl,
398 IN LPMIXERCONTROLDETAILS MixerControlDetails,
399 IN LPMIXERLINE_EXT MixerLine)
400{
402 PULONG LogicalNodes, ConnectedNodes;
403 ULONG LogicalNodesCount, ConnectedNodesCount, Index, CurLogicalPinOffset, BytesReturned, OldLogicalPinOffset;
404 LPMIXER_DATA MixerData;
407 LPMIXERLINE_EXT SourceLine;
409
410 DPRINT("MixerControlDetails %p\n", MixerControlDetails);
411 DPRINT("bSet %lx\n", bSet);
412 DPRINT("Flags %lx\n", Flags);
413 DPRINT("NodeId %lu\n", MixerControl->NodeID);
414 DPRINT("MixerControlDetails dwControlID %lu\n", MixerControlDetails->dwControlID);
415 DPRINT("MixerControlDetails cChannels %lu\n", MixerControlDetails->cChannels);
416 DPRINT("MixerControlDetails cMultipleItems %lu\n", MixerControlDetails->cMultipleItems);
417 DPRINT("MixerControlDetails cbDetails %lu\n", MixerControlDetails->cbDetails);
418 DPRINT("MixerControlDetails paDetails %p\n", MixerControlDetails->paDetails);
419
420 if (MixerControl->Control.fdwControl & MIXERCONTROL_CONTROLF_UNIFORM)
421 {
422 /* control acts uniform */
423 if (MixerControlDetails->cChannels != 1)
424 {
425 /* expected 1 channel */
426 DPRINT1("Expected 1 channel but got %lu\n", MixerControlDetails->cChannels);
428 }
429 }
430
431 /* check if multiple items match */
432 if (MixerControlDetails->cMultipleItems != MixerControl->Control.cMultipleItems)
433 {
434 DPRINT1("MultipleItems mismatch %lu expected %lu\n", MixerControlDetails->cMultipleItems, MixerControl->Control.cMultipleItems);
436 }
437
438 if (bSet)
439 {
441 {
442 /* tell me when this is hit */
443 ASSERT(FALSE);
444 }
446 {
447 /* sanity check */
448 ASSERT(bSet == TRUE);
449 ASSERT(MixerControlDetails->cbDetails == sizeof(MIXERCONTROLDETAILS_BOOLEAN));
450
451 Values = (LPMIXERCONTROLDETAILS_BOOLEAN)MixerControlDetails->paDetails;
452 CurLogicalPinOffset = MAXULONG;
453 for(Index = 0; Index < MixerControlDetails->cMultipleItems; Index++)
454 {
455 if (Values[Index].fValue)
456 {
457 /* mux can only activate one line at a time */
458 ASSERT(CurLogicalPinOffset == MAXULONG);
459 CurLogicalPinOffset = Index;
460 }
461 }
462
463 /* setup request */
464 Request.NodeId = NodeId;
465 Request.Reserved = 0;
468 Request.Property.Set = KSPROPSETID_Audio;
469
470 /* perform getting source */
471 Status = MixerContext->Control(MixerControl->hDevice, IOCTL_KS_PROPERTY, (PVOID)&Request, sizeof(KSNODEPROPERTY), &OldLogicalPinOffset, sizeof(ULONG), &BytesReturned);
473 {
474 /* failed to get source */
475 return Status;
476 }
477
478 DPRINT("OldLogicalPinOffset %lu CurLogicalPinOffset %lu\n", OldLogicalPinOffset, CurLogicalPinOffset);
479
480 if (OldLogicalPinOffset == CurLogicalPinOffset)
481 {
482 /* cannot be unselected */
484 }
485
486 /* perform setting source */
488 Status = MixerContext->Control(MixerControl->hDevice, IOCTL_KS_PROPERTY, (PVOID)&Request, sizeof(KSNODEPROPERTY), &CurLogicalPinOffset, sizeof(ULONG), &BytesReturned);
490 {
491 /* failed to set source */
492 return Status;
493 }
494
495 /* notify control change */
496 MMixerNotifyControlChange(MixerContext, MixerInfo, MM_MIXM_CONTROL_CHANGE, MixerControl->Control.dwControlID );
497
498 return Status;
499 }
500 }
501 else
502 {
504 {
505 /* setup request */
506 Request.NodeId = NodeId;
507 Request.Reserved = 0;
510 Request.Property.Set = KSPROPSETID_Audio;
511
512 /* perform getting source */
513 Status = MixerContext->Control(MixerControl->hDevice, IOCTL_KS_PROPERTY, (PVOID)&Request, sizeof(KSNODEPROPERTY), &OldLogicalPinOffset, sizeof(ULONG), &BytesReturned);
515 {
516 /* failed to get source */
517 return Status;
518 }
519
520 /* gets the corresponding mixer data */
521 MixerData = MMixerGetMixerDataByDeviceHandle(MixerContext, MixerControl->hDevice);
522
523 /* sanity check */
524 ASSERT(MixerData);
525 ASSERT(MixerData->Topology);
526 ASSERT(MixerData->MixerInfo == MixerInfo);
527
528 /* now allocate logical pin array */
529 Status = MMixerAllocateTopologyNodeArray(MixerContext, MixerData->Topology, &LogicalNodes);
531 {
532 /* no memory */
533 return MM_STATUS_NO_MEMORY;
534 }
535
536 /* get logical pin nodes */
537 MMixerGetConnectedFromLogicalTopologyPins(MixerData->Topology, MixerControl->NodeID, &LogicalNodesCount, LogicalNodes);
538
539 /* sanity check */
540 ASSERT(LogicalNodesCount == MixerControlDetails->cMultipleItems);
541 ASSERT(LogicalNodesCount == MixerControl->Control.Metrics.dwReserved[0]);
542
543 Values = (LPMIXERCONTROLDETAILS_BOOLEAN)MixerControlDetails->paDetails;
544 for(Index = 0; Index < LogicalNodesCount; Index++)
545 {
546 /* getting logical pin offset */
547 MMixerGetLowestLogicalTopologyPinOffsetFromArray(LogicalNodesCount, LogicalNodes, &CurLogicalPinOffset);
548
549 if (CurLogicalPinOffset == OldLogicalPinOffset)
550 {
551 /* mark index as active */
552 Values[Index].fValue = TRUE;
553 }
554 else
555 {
556 /* index not active */
557 Values[Index].fValue = FALSE;
558 }
559
560 /* mark offset as consumed */
561 LogicalNodes[CurLogicalPinOffset] = MAXULONG;
562 }
563
564 /* cleanup */
565 MixerContext->Free(LogicalNodes);
566
567 /* done */
568 return MM_STATUS_SUCCESS;
569 }
571 {
572 /* sanity check */
573 ASSERT(bSet == FALSE);
574
575 /* gets the corresponding mixer data */
576 MixerData = MMixerGetMixerDataByDeviceHandle(MixerContext, MixerControl->hDevice);
577
578 /* sanity check */
579 ASSERT(MixerData);
580 ASSERT(MixerData->Topology);
581 ASSERT(MixerData->MixerInfo == MixerInfo);
582
583 /* now allocate logical pin array */
584 Status = MMixerAllocateTopologyNodeArray(MixerContext, MixerData->Topology, &LogicalNodes);
586 {
587 /* no memory */
588 return MM_STATUS_NO_MEMORY;
589 }
590
591 /* allocate connected node array */
592 Status = MMixerAllocateTopologyNodeArray(MixerContext, MixerData->Topology, &ConnectedNodes);
594 {
595 /* no memory */
596 MixerContext->Free(LogicalNodes);
597 return MM_STATUS_NO_MEMORY;
598 }
599
600 /* get logical pin nodes */
601 MMixerGetConnectedFromLogicalTopologyPins(MixerData->Topology, MixerControl->NodeID, &LogicalNodesCount, LogicalNodes);
602
603 /* get connected nodes */
604 MMixerGetNextNodesFromNodeIndex(MixerContext, MixerData->Topology, MixerControl->NodeID, TRUE, &ConnectedNodesCount, ConnectedNodes);
605
606 /* sanity check */
607 ASSERT(ConnectedNodesCount == LogicalNodesCount);
608 ASSERT(ConnectedNodesCount == MixerControlDetails->cMultipleItems);
609 ASSERT(ConnectedNodesCount == MixerControl->Control.Metrics.dwReserved[0]);
610
611 ListText = (LPMIXERCONTROLDETAILS_LISTTEXTW)MixerControlDetails->paDetails;
612
613 for(Index = 0; Index < ConnectedNodesCount; Index++)
614 {
615 /* getting logical pin offset */
616 MMixerGetLowestLogicalTopologyPinOffsetFromArray(LogicalNodesCount, LogicalNodes, &CurLogicalPinOffset);
617
618 /* get mixer line with that node */
619 SourceLine = MMixerGetMixerLineContainingNodeId(MixerInfo, ConnectedNodes[CurLogicalPinOffset]);
620
621 /* sanity check */
622 ASSERT(SourceLine);
623
624 DPRINT1("PinOffset %lu LogicalPin %lu NodeId %lu LineName %S\n", CurLogicalPinOffset, LogicalNodes[CurLogicalPinOffset], ConnectedNodes[CurLogicalPinOffset], SourceLine->Line.szName);
625
626 /* copy details */
627 ListText[Index].dwParam1 = SourceLine->Line.dwLineID;
628 ListText[Index].dwParam2 = SourceLine->Line.dwComponentType;
629 MixerContext->Copy(ListText[Index].szName, SourceLine->Line.szName, (wcslen(SourceLine->Line.szName) + 1) * sizeof(WCHAR));
630
631 /* mark offset as consumed */
632 LogicalNodes[CurLogicalPinOffset] = MAXULONG;
633 }
634
635 /* cleanup */
636 MixerContext->Free(LogicalNodes);
637 MixerContext->Free(ConnectedNodes);
638
639 /* done */
640 return MM_STATUS_SUCCESS;
641 }
642 }
643
645}
646
650 IN LPMIXER_INFO MixerInfo,
651 IN ULONG NodeId,
652 IN ULONG bSet,
653 LPMIXERCONTROL_EXT MixerControl,
654 IN LPMIXERCONTROLDETAILS MixerControlDetails,
655 LPMIXERLINE_EXT MixerLine)
656{
658 LONG MaxRange, Value;
659 ULONG Channel;
661 LPMIXERVOLUME_DATA VolumeData;
662
663 if (MixerControlDetails->cbDetails != sizeof(MIXERCONTROLDETAILS_UNSIGNED))
665
666 VolumeData = (LPMIXERVOLUME_DATA)MixerControl->ExtraData;
667 if (!VolumeData)
669
670 /* Get input */
671 Input = (LPMIXERCONTROLDETAILS_UNSIGNED)MixerControlDetails->paDetails;
672 if (!Input)
673 return MM_STATUS_UNSUCCESSFUL; /* To prevent dereferencing NULL */
674
675 /* Get maximum available range */
676 MaxRange = VolumeData->SignedMaximum - VolumeData->SignedMinimum;
677
678 /* Loop for each channel */
679 for (Channel = 0; Channel < MixerControlDetails->cChannels; Channel++)
680 {
681 if (bSet)
682 {
683 /* FIXME SEH */
684 /* Convert from logical units to hardware range (DB) */
685 Value = (LONG)((INT64)Input[Channel].dwValue * MaxRange / 0x10000 + VolumeData->SignedMinimum);
686 }
687
688 /* Get/set control details */
690
691 if (!bSet)
692 {
693 /* FIXME SEH */
694 /* Convert from hardware range (DB) to logical units */
695 Input[Channel].dwValue = (ULONG)(((INT64)Value - VolumeData->SignedMinimum + 1) * 0x10000 / MaxRange);
696 }
697 }
698
699 if (bSet)
700 {
701 /* Notify clients of a line change */
703 }
704 return Status;
705}
706
709 IN PMIXER_LIST MixerList,
710 IN ULONG DeviceId)
711{
713 LPMIXER_DATA MixerData;
714
715 Entry = MixerList->MixerData.Flink;
716 while(Entry != &MixerList->MixerData)
717 {
719 if (MixerData->DeviceId == DeviceId)
720 {
721 return MixerData;
722 }
723 Entry = Entry->Flink;
724 }
725 return NULL;
726}
727
730 IN PMIXER_LIST MixerList,
732{
734 LPMIXER_DATA MixerData;
735
736 Entry = MixerList->MixerData.Flink;
737 while(Entry != &MixerList->MixerData)
738 {
740 if (_wcsicmp(&DeviceName[2], &MixerData->DeviceName[2]) == 0)
741 {
742 /* found entry */
743 return MixerData;
744 }
745 Entry = Entry->Flink;
746 }
747 return NULL;
748}
749
753 IN PMIXER_LIST MixerList,
754 IN ULONG DeviceId,
756 IN HANDLE hDevice,
757 IN HANDLE hKey)
758{
759 LPMIXER_DATA MixerData;
760
761 MixerData = (LPMIXER_DATA)MixerContext->Alloc(sizeof(MIXER_DATA));
762 if (!MixerData)
763 return MM_STATUS_NO_MEMORY;
764
765 MixerData->DeviceId = DeviceId;
766 MixerData->DeviceName = DeviceName;
767 MixerData->hDevice = hDevice;
768 MixerData->hDeviceInterfaceKey = hKey;
769 MixerData->Topology = NULL;
770
771 InsertTailList(&MixerList->MixerData, &MixerData->Entry);
772 MixerList->MixerDataCount++;
773 return MM_STATUS_SUCCESS;
774}
775
780 OUT LPWSTR OutDeviceName)
781{
787 ULONG ResultLength, KeyType;
788 HANDLE hMediaKey, hGuidKey;
790
791 /* prepare property */
795
796 /* try get component id */
798
800 {
801 Status = MixerContext->OpenKey(NULL, L"\\Registry\\Machine\\SYSTEM\\CurrentControlSet\\Control\\MediaCategories", KEY_READ, &hMediaKey);
803 {
805 Status = MixerContext->OpenKey(hMediaKey, GuidString.Buffer, KEY_READ, &hGuidKey);
808 {
809 Status = MixerContext->QueryKeyValue(hGuidKey, L"Name", (PVOID*)&DeviceName, &ResultLength, &KeyType);
811 {
812 MixerContext->Copy(OutDeviceName, DeviceName, min(ResultLength, (MAXPNAMELEN-1)*2));
814 }
815
816 MixerContext->CloseKey(hGuidKey);
817 }
818 MixerContext->CloseKey(hMediaKey);
819 }
820 }
821 return Status;
822}
823
828 IN HANDLE hKey)
829{
830 LPWSTR Name;
831 HANDLE hTemp;
833 ULONG Type;
835
836 Status = MixerContext->QueryKeyValue(hKey, L"FriendlyName", (PVOID*)&Name, &Length, &Type);
838 {
839 /* copy device name */
841
842 /* make sure its null terminated */
843 DeviceName[MAXPNAMELEN-1] = L'\0';
844
845 /* free device name */
847
848 /* done */
849 return Status;
850 }
851
852 Status = MixerContext->OpenKey(hKey, L"Device Parameters", KEY_READ, &hTemp);
854 return Status;
855
856 Status = MixerContext->QueryKeyValue(hTemp, L"FriendlyName", (PVOID*)&Name, &Length, &Type);
858 {
859 /* copy device name */
861
862 /* make sure its null terminated */
863 DeviceName[MAXPNAMELEN-1] = L'\0';
864
865 /* free device name */
867 }
868
869 MixerContext->CloseKey(hTemp);
870 return Status;
871}
872
873VOID
875 IN OUT PKSPIN_CONNECT PinConnect,
876 IN ULONG PinId)
877{
878 PinConnect->Interface.Set = KSINTERFACESETID_Standard;
879 PinConnect->Interface.Id = KSINTERFACE_STANDARD_STREAMING;
880 PinConnect->Interface.Flags = 0;
881 PinConnect->Medium.Set = KSMEDIUMSETID_Standard;
882 PinConnect->Medium.Id = KSMEDIUM_TYPE_ANYINSTANCE;
883 PinConnect->Medium.Flags = 0;
884 PinConnect->PinToHandle = NULL;
885 PinConnect->PinId = PinId;
886 PinConnect->Priority.PriorityClass = KSPRIORITY_NORMAL;
887 PinConnect->Priority.PrioritySubClass = 1;
888}
NodeType
Definition: Node.h:6
signed long long INT64
Type
Definition: Type.h:7
ACPI_BUFFER *RetBuffer ACPI_BUFFER *RetBuffer char ACPI_WALK_RESOURCE_CALLBACK void *Context ACPI_BUFFER *RetBuffer UINT16 ACPI_RESOURCE **ResourcePtr ACPI_GENERIC_ADDRESS *Reg UINT32 *ReturnValue UINT8 UINT8 *Slp_TypB ACPI_PHYSICAL_ADDRESS PhysicalAddress64 UINT32 UINT32 *TimeElapsed UINT32 ACPI_STATUS const char UINT32 ACPI_STATUS const char UINT32 const char const char UINT32 ComponentId
Definition: acpixf.h:1281
#define DPRINT1
Definition: precomp.h:8
LPWSTR Name
Definition: desk.c:124
#define NULL
Definition: types.h:112
#define TRUE
Definition: types.h:120
#define FALSE
Definition: types.h:117
_ACRTIMP int __cdecl _wcsicmp(const wchar_t *, const wchar_t *)
Definition: wcs.c:159
_ACRTIMP size_t __cdecl wcslen(const wchar_t *)
Definition: wcs.c:2983
MIXER_CONTEXT MixerContext
Definition: mmixer.c:41
#define L(x)
Definition: resources.c:13
#define NodeType(P)
Definition: nodetype.h:51
#define KSPROPERTY_TYPE_SET
Definition: dmksctrl.h:43
#define KSPROPERTY_TYPE_TOPOLOGY
Definition: dmksctrl.h:53
#define KSPROPERTY_TYPE_GET
Definition: dmksctrl.h:42
#define InsertTailList(ListHead, Entry)
unsigned long DWORD
Definition: ntddk_ex.h:95
FxAutoRegKey hKey
Status
Definition: gdiplustypes.h:25
NTSYSAPI NTSTATUS WINAPI RtlStringFromGUID(REFGUID, PUNICODE_STRING)
#define KSPRIORITY_NORMAL
Definition: ks.h:1386
#define KSMEDIUM_TYPE_ANYINSTANCE
Definition: ks.h:301
#define KSMEDIUMSETID_Standard
Definition: ks.h:306
#define IOCTL_KS_PROPERTY
Definition: ks.h:127
@ KSINTERFACE_STANDARD_STREAMING
Definition: ks.h:283
#define KSINTERFACESETID_Standard
Definition: ks.h:280
@ KSPROPERTY_GENERAL_COMPONENTID
Definition: ks.h:447
@ KSPROPERTY_AUDIO_VOLUMELEVEL
Definition: ksmedia.h:1057
@ KSPROPERTY_AUDIO_MUTE
Definition: ksmedia.h:1066
@ KSPROPERTY_AUDIO_MUX_SOURCE
Definition: ksmedia.h:1065
if(dx< 0)
Definition: linetemp.h:194
MIXER_STATUS
Definition: mmixer.h:4
@ MM_STATUS_NOT_IMPLEMENTED
Definition: mmixer.h:7
@ MM_STATUS_UNSUCCESSFUL
Definition: mmixer.h:11
@ MM_STATUS_INVALID_PARAMETER
Definition: mmixer.h:10
@ MM_STATUS_NO_MEMORY
Definition: mmixer.h:12
@ MM_STATUS_SUCCESS
Definition: mmixer.h:5
#define MIXERCONTROL_CONTROLF_UNIFORM
Definition: mmsystem.h:350
#define MIXER_SETCONTROLDETAILSF_VALUE
Definition: mmsystem.h:415
struct tMIXERCONTROLDETAILS_BOOLEAN * LPMIXERCONTROLDETAILS_BOOLEAN
#define MIXER_GETCONTROLDETAILSF_VALUE
Definition: mmsystem.h:412
#define MIXER_GETCONTROLDETAILSF_QUERYMASK
Definition: mmsystem.h:414
#define MM_MIXM_CONTROL_CHANGE
Definition: mmsystem.h:83
#define MIXER_SETCONTROLDETAILSF_CUSTOM
Definition: mmsystem.h:416
#define MIXER_SETCONTROLDETAILSF_QUERYMASK
Definition: mmsystem.h:417
#define MIXER_GETCONTROLDETAILSF_LISTTEXT
Definition: mmsystem.h:413
#define MM_MIXM_LINE_CHANGE
Definition: mmsystem.h:82
struct tagMIXERCONTROLDETAILS_LISTTEXTW * LPMIXERCONTROLDETAILS_LISTTEXTW
#define MAXPNAMELEN
Definition: mmsystem.h:24
struct tMIXERCONTROLDETAILS_UNSIGNED * LPMIXERCONTROLDETAILS_UNSIGNED
#define ASSERT(a)
Definition: mode.c:44
static PWSTR GuidString
Definition: apphelp.c:93
HMIXER hMixer
Definition: test.c:10
#define min(a, b)
Definition: monoChain.cc:55
int Count
Definition: noreturn.cpp:7
#define KEY_READ
Definition: nt_native.h:1026
NTSYSAPI VOID NTAPI RtlFreeUnicodeString(PUNICODE_STRING UnicodeString)
_In_ ULONG _In_ ULONG _In_ ULONG Length
Definition: ntddpcm.h:102
long LONG
Definition: pedump.c:60
static const WCHAR szName[]
Definition: powrprof.c:45
GUID * LPGUID
Definition: guiddef.h:81
@ Input
Definition: arc.h:93
MIXER_STATUS MMixerSetGetControlDetails(IN PMIXER_CONTEXT MixerContext, IN HANDLE hMixer, IN ULONG NodeId, IN ULONG bSet, IN ULONG PropertyId, IN ULONG Channel, IN PLONG InputValue)
Definition: filter.c:214
struct MIXERLINE_EXT * LPMIXERLINE_EXT
struct EVENT_NOTIFICATION_ENTRY * PEVENT_NOTIFICATION_ENTRY
struct MIXERVOLUME_DATA * LPMIXERVOLUME_DATA
struct MIXER_DATA * LPMIXER_DATA
struct MIXERCONTROL_EXT * LPMIXERCONTROL_EXT
struct MIXER_LIST * PMIXER_LIST
VOID MMixerGetConnectedFromLogicalTopologyPins(IN PTOPOLOGY Topology, IN ULONG NodeIndex, OUT PULONG OutPinCount, OUT PULONG OutPins)
Definition: topology.c:1158
MIXER_STATUS MMixerAllocateTopologyNodeArray(IN PMIXER_CONTEXT MixerContext, IN PTOPOLOGY Topology, OUT PULONG *OutPins)
Definition: topology.c:1045
VOID MMixerGetNextNodesFromNodeIndex(IN PMIXER_CONTEXT MixerContext, IN PTOPOLOGY Topology, IN ULONG NodeIndex, IN ULONG bUpStream, OUT PULONG OutNodesCount, OUT PULONG OutNodes)
Definition: topology.c:965
struct MIXER_INFO * LPMIXER_INFO
const GUID KSNODETYPE_DAC
Definition: sup.c:15
const GUID KSPROPSETID_Audio
Definition: sup.c:29
const GUID KSPROPSETID_Pin
Definition: sup.c:30
const GUID KSNODETYPE_PEAKMETER
Definition: sup.c:22
MIXER_STATUS MMixerSetGetMuteControlDetails(IN PMIXER_CONTEXT MixerContext, IN LPMIXER_INFO MixerInfo, IN LPMIXERCONTROL_EXT MixerControl, IN ULONG dwLineID, IN LPMIXERCONTROLDETAILS MixerControlDetails, IN ULONG bSet)
Definition: sup.c:347
VOID MMixerFreeMixerInfo(IN PMIXER_CONTEXT MixerContext, IN LPMIXER_INFO MixerInfo)
Definition: sup.c:115
LPMIXER_INFO MMixerGetMixerInfoByIndex(IN PMIXER_CONTEXT MixerContext, IN ULONG MixerIndex)
Definition: sup.c:158
const GUID KSDATAFORMAT_TYPE_MUSIC
Definition: sup.c:35
LPMIXER_DATA MMixerGetDataByDeviceName(IN PMIXER_LIST MixerList, IN LPWSTR DeviceName)
Definition: sup.c:729
const GUID KSEVENTSETID_AudioControlChange
Definition: sup.c:33
VOID MMixerInitializePinConnect(IN OUT PKSPIN_CONNECT PinConnect, IN ULONG PinId)
Definition: sup.c:874
const GUID KSNODETYPE_VOLUME
Definition: sup.c:21
MIXER_STATUS MMixerGetDeviceNameWithComponentId(IN PMIXER_CONTEXT MixerContext, IN HANDLE hMixer, OUT LPWSTR OutDeviceName)
Definition: sup.c:777
const GUID KSNODETYPE_CHORUS
Definition: sup.c:25
const GUID KSNODETYPE_AGC
Definition: sup.c:17
const GUID KSNODETYPE_SUPERMIX
Definition: sup.c:27
LPGUID MMixerGetNodeType(IN PKSMULTIPLE_ITEM MultipleItem, IN ULONG Index)
Definition: sup.c:242
MIXER_STATUS MMixerSetGetMuxControlDetails(IN PMIXER_CONTEXT MixerContext, IN LPMIXER_INFO MixerInfo, IN ULONG NodeId, IN ULONG bSet, IN ULONG Flags, IN LPMIXERCONTROL_EXT MixerControl, IN LPMIXERCONTROLDETAILS MixerControlDetails, IN LPMIXERLINE_EXT MixerLine)
Definition: sup.c:391
const GUID KSNODETYPE_SUM
Definition: sup.c:14
MIXER_STATUS MMixerGetMixerByName(IN PMIXER_LIST MixerList, IN LPWSTR MixerName, OUT LPMIXER_INFO *OutMixerInfo)
Definition: sup.c:191
MIXER_STATUS MMixerGetDeviceName(IN PMIXER_CONTEXT MixerContext, OUT LPWSTR DeviceName, IN HANDLE hKey)
Definition: sup.c:825
const GUID KSDATAFORMAT_SPECIFIER_NONE
Definition: sup.c:37
const GUID KSNODETYPE_ADC
Definition: sup.c:16
MIXER_STATUS MMixerGetMixerControlById(LPMIXER_INFO MixerInfo, DWORD dwControlID, LPMIXERLINE_EXT *OutMixerLine, LPMIXERCONTROL_EXT *OutMixerControl, PULONG NodeId)
Definition: sup.c:278
const GUID KSNODETYPE_MUX
Definition: sup.c:23
const GUID KSNODETYPE_MUTE
Definition: sup.c:19
MIXER_STATUS MMixerVerifyContext(IN PMIXER_CONTEXT MixerContext)
Definition: sup.c:40
LPMIXERLINE_EXT MMixerGetSourceMixerLineByComponentType(LPMIXER_INFO MixerInfo, DWORD dwComponentType)
Definition: sup.c:255
VOID MMixerGetLowestLogicalTopologyPinOffsetFromArray(IN ULONG LogicalPinArrayCount, IN PULONG LogicalPinArray, OUT PULONG PinOffset)
Definition: sup.c:90
MIXER_STATUS MMixerSetGetVolumeControlDetails(IN PMIXER_CONTEXT MixerContext, IN LPMIXER_INFO MixerInfo, IN ULONG NodeId, IN ULONG bSet, LPMIXERCONTROL_EXT MixerControl, IN LPMIXERCONTROLDETAILS MixerControlDetails, LPMIXERLINE_EXT MixerLine)
Definition: sup.c:648
MIXER_STATUS MMixerCreateMixerData(IN PMIXER_CONTEXT MixerContext, IN PMIXER_LIST MixerList, IN ULONG DeviceId, IN LPWSTR DeviceName, IN HANDLE hDevice, IN HANDLE hKey)
Definition: sup.c:751
LPMIXERLINE_EXT MMixerGetMixerLineContainingNodeId(IN LPMIXER_INFO MixerInfo, IN ULONG NodeID)
Definition: sup.c:58
const GUID KSNODETYPE_STEREO_WIDE
Definition: sup.c:24
VOID MMixerNotifyControlChange(IN PMIXER_CONTEXT MixerContext, IN LPMIXER_INFO MixerInfo, IN ULONG NotificationType, IN ULONG Value)
Definition: sup.c:319
const GUID KSNODETYPE_REVERB
Definition: sup.c:26
const GUID KSNODETYPE_TONE
Definition: sup.c:20
LPMIXERLINE_EXT MMixerGetSourceMixerLineByLineId(LPMIXER_INFO MixerInfo, DWORD dwLineID)
Definition: sup.c:218
const GUID KSPROPSETID_General
Definition: sup.c:31
LPMIXER_DATA MMixerGetMixerDataByDeviceHandle(IN PMIXER_CONTEXT MixerContext, IN HANDLE hDevice)
Definition: sup.c:128
const GUID KSDATAFORMAT_SUBTYPE_MIDI
Definition: sup.c:36
const GUID KSPROPSETID_Topology
Definition: sup.c:32
LPMIXER_DATA MMixerGetDataByDeviceId(IN PMIXER_LIST MixerList, IN ULONG DeviceId)
Definition: sup.c:708
const GUID KSNODETYPE_LOUDNESS
Definition: sup.c:18
#define DPRINT
Definition: sndvol32.h:73
Definition: precomp.h:165
base of all file and directory entries
Definition: entries.h:83
HANDLE hDevice
Definition: precomp.h:81
MIXERCONTROLW Control
Definition: precomp.h:79
PVOID ExtraData
Definition: precomp.h:82
ULONG NodeID
Definition: precomp.h:80
MIXERLINEW Line
Definition: precomp.h:89
LIST_ENTRY ControlsList
Definition: precomp.h:90
LONG SignedMinimum
Definition: precomp.h:103
LONG SignedMaximum
Definition: precomp.h:104
PMIXER_ALLOC_EVENT_DATA AllocEventData
Definition: mmixer.h:103
PMIXER_OPEN_KEY OpenKey
Definition: mmixer.h:100
PMIXER_DEVICE_CONTROL Control
Definition: mmixer.h:95
PMIXER_CLOSE Close
Definition: mmixer.h:98
ULONG SizeOfStruct
Definition: mmixer.h:91
PMIXER_CLOSEKEY CloseKey
Definition: mmixer.h:102
PMIXER_COPY Copy
Definition: mmixer.h:99
PMIXER_QUERY_KEY_VALUE QueryKeyValue
Definition: mmixer.h:101
PMIXER_FREE_EVENT_DATA FreeEventData
Definition: mmixer.h:104
PVOID MixerContext
Definition: mmixer.h:92
PMIXER_OPEN Open
Definition: mmixer.h:97
PMIXER_FREE Free
Definition: mmixer.h:96
PMIXER_ALLOC Alloc
Definition: mmixer.h:94
LIST_ENTRY Entry
Definition: precomp.h:109
LPMIXER_INFO MixerInfo
Definition: precomp.h:115
HANDLE hDevice
Definition: precomp.h:111
HANDLE hDeviceInterfaceKey
Definition: precomp.h:112
LPWSTR DeviceName
Definition: precomp.h:113
ULONG DeviceId
Definition: precomp.h:110
PTOPOLOGY Topology
Definition: precomp.h:114
LIST_ENTRY LineList
Definition: precomp.h:71
MIXERCAPSW MixCaps
Definition: precomp.h:70
ULONG MixerDataCount
Definition: precomp.h:148
LIST_ENTRY MixerData
Definition: precomp.h:149
ULONG MixerListCount
Definition: precomp.h:145
LIST_ENTRY MixerList
Definition: precomp.h:146
Definition: typedefs.h:120
struct _LIST_ENTRY * Flink
Definition: typedefs.h:121
WCHAR szPname[MAXPNAMELEN]
Definition: mmsystem.h:1226
DWORD dwLineID
Definition: mmsystem.h:1258
WCHAR szName[MIXER_LONG_NAME_CHARS]
Definition: mmsystem.h:1266
DWORD dwComponentType
Definition: mmsystem.h:1261
#define MAXULONG
Definition: typedefs.h:251
uint32_t * PULONG
Definition: typedefs.h:59
#define IN
Definition: typedefs.h:39
#define CONTAINING_RECORD(address, type, field)
Definition: typedefs.h:260
uint32_t ULONG
Definition: typedefs.h:59
#define OUT
Definition: typedefs.h:40
_In_ WDFCOLLECTION _In_ ULONG Index
_Must_inspect_result_ _In_ WDFDEVICE _In_ DEVICE_REGISTRY_PROPERTY _In_ ULONG _Out_ PULONG ResultLength
Definition: wdfdevice.h:3782
_In_ WDF_SPECIAL_FILE_TYPE NotificationType
Definition: wdfdevice.h:1024
_Must_inspect_result_ _In_ PWDFDEVICE_INIT _In_opt_ PCUNICODE_STRING DeviceName
Definition: wdfdevice.h:3281
_In_ WDFREQUEST Request
Definition: wdfdevice.h:547
_Must_inspect_result_ _In_ WDFIOTARGET _In_opt_ WDFREQUEST _In_ ULONG _In_opt_ PWDF_MEMORY_DESCRIPTOR _In_opt_ PWDF_MEMORY_DESCRIPTOR _In_opt_ PWDF_REQUEST_SEND_OPTIONS _Out_opt_ PULONG_PTR BytesReturned
Definition: wdfiotarget.h:1052
_Must_inspect_result_ _In_ WDFKEY _In_ PCUNICODE_STRING _Out_opt_ PUSHORT _Inout_opt_ PUNICODE_STRING Value
Definition: wdfregistry.h:413
_Must_inspect_result_ _In_ ULONG Flags
Definition: wsk.h:170
_In_ ULONG _In_opt_ PVOID _In_ PDRIVER_OBJECT _In_ PDRIVER_NOTIFICATION_CALLBACK_ROUTINE _Inout_opt_ __drv_aliasesMem PVOID _Outptr_result_nullonfailure_ _At_ * NotificationEntry(return==0, __drv_allocatesMem(Mem))) PVOID *NotificationEntry
__wchar_t WCHAR
Definition: xmlstorage.h:180
WCHAR * LPWSTR
Definition: xmlstorage.h:184