mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-26 15:23:48 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			788 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			788 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * PortMixer
 | |
|  * Common Windows routines
 | |
|  *
 | |
|  * Copyright (c) 2002, 2006
 | |
|  *
 | |
|  * Written by Dominic Mazzoni and Augustus Saunders
 | |
|  *        and Leland Lucius
 | |
|  *
 | |
|  * PortMixer is intended to work side-by-side with PortAudio,
 | |
|  * the Portable Real-Time Audio Library by Ross Bencina and
 | |
|  * Phil Burk.
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining
 | |
|  * a copy of this software and associated documentation files
 | |
|  * (the "Software"), to deal in the Software without restriction,
 | |
|  * including without limitation the rights to use, copy, modify, merge,
 | |
|  * publish, distribute, sublicense, and/or sell copies of the Software,
 | |
|  * and to permit persons to whom the Software is furnished to do so,
 | |
|  * subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be
 | |
|  * included in all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * Any person wishing to distribute modifications to the Software is
 | |
|  * requested to send the modifications to the original developer so that
 | |
|  * they can be incorporated into the canonical version.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 | |
|  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | |
|  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 | |
|  * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
 | |
|  * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
 | |
|  * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 | |
|  * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #include <windows.h>
 | |
| #include <tchar.h>
 | |
| 
 | |
| #include "portaudio.h"
 | |
| 
 | |
| #include "portmixer.h"
 | |
| #include "px_mixer.h"
 | |
| #include "px_win_common.h"
 | |
| 
 | |
| static UINT get_ctrls(HMIXEROBJ mixer, DWORD lineID, PxCtrl **pctrls);
 | |
| static DWORD find_ctrl(HMIXEROBJ mixer, DWORD lineID, DWORD ctrlID);
 | |
| 
 | |
| #if defined(_DEBUG)
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| static void dprintf(const char *format, ...)
 | |
| {
 | |
|    char buf[4096];
 | |
|    va_list args;
 | |
|    int cnt;
 | |
| 
 | |
|    va_start(args, format);
 | |
|    cnt = _vsnprintf(buf, sizeof(buf) - 1, format, args);
 | |
|    va_end(args);
 | |
| 
 | |
|    if (cnt > 0) {
 | |
|       buf[cnt] = '\0';
 | |
|       OutputDebugString(buf);
 | |
|    }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static BOOL is_vista_or_later() 
 | |
| {
 | |
|    OSVERSIONINFOEX osvi;
 | |
|    DWORDLONG dwlConditionMask = 0;
 | |
|    int op=VER_GREATER_EQUAL;
 | |
| 
 | |
|    // Initialize the OSVERSIONINFOEX structure.
 | |
| 
 | |
|    ZeroMemory(&osvi, sizeof(OSVERSIONINFOEX));
 | |
|    osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
 | |
|    osvi.dwMajorVersion = 6;
 | |
|    osvi.dwMinorVersion = 0;
 | |
|    osvi.wServicePackMajor = 0;
 | |
|    osvi.wServicePackMinor = 0;
 | |
| 
 | |
|    // Initialize the condition mask.
 | |
| 
 | |
|    VER_SET_CONDITION( dwlConditionMask, VER_MAJORVERSION, op );
 | |
|    VER_SET_CONDITION( dwlConditionMask, VER_MINORVERSION, op );
 | |
| 
 | |
|    // Perform the test.
 | |
| 
 | |
|    return VerifyVersionInfo(
 | |
|       &osvi, 
 | |
|       VER_MAJORVERSION | VER_MINORVERSION,
 | |
|       dwlConditionMask);
 | |
| }
 | |
| 
 | |
| int open_mixers(px_mixer *Px, UINT deviceIn, UINT deviceOut)
 | |
| {
 | |
|    PxInfo*info;
 | |
|    MMRESULT res;
 | |
|   
 | |
|    if (is_vista_or_later()) {
 | |
|       return open_ep_mixers(Px, deviceIn, deviceOut);
 | |
|    }
 | |
| 
 | |
|    res = mixerGetID((HMIXEROBJ) (deviceIn == WAVE_MAPPER ? 0 : deviceIn),
 | |
|                     &deviceIn,
 | |
|                     MIXER_OBJECTF_WAVEIN);
 | |
|    if (res != MMSYSERR_NOERROR) {
 | |
|       return FALSE;
 | |
|    }
 | |
| 
 | |
|    res = mixerGetID((HMIXEROBJ) (deviceOut == WAVE_MAPPER ? 0 : deviceOut),
 | |
|                     &deviceOut,
 | |
|                     MIXER_OBJECTF_WAVEOUT);
 | |
|    if (res != MMSYSERR_NOERROR) {
 | |
|       return FALSE;
 | |
|    }
 | |
| 
 | |
|    if (!initialize(Px)) {
 | |
|       return FALSE;
 | |
|    }
 | |
| 
 | |
|    info = (PxInfo *) Px->info;
 | |
|    info->hInputMixer = NULL;
 | |
|    info->hOutputMixer = NULL;
 | |
|    info->numInputs = 0;
 | |
|    info->muxID = 0;
 | |
|    info->speakerID = 0;
 | |
|    info->waveID = 0;
 | |
| 
 | |
|    if (deviceIn != UINT_MAX) {
 | |
|       res = mixerOpen((LPHMIXER) &info->hInputMixer,
 | |
|                       deviceIn,
 | |
|                       0,
 | |
|                       0,
 | |
|                       MIXER_OBJECTF_MIXER);
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          return cleanup(Px);
 | |
|       }
 | |
| 
 | |
|       info->muxID = find_ctrl(info->hInputMixer,
 | |
|                               MIXERLINE_COMPONENTTYPE_DST_WAVEIN,
 | |
|                               MIXERCONTROL_CONTROLTYPE_MUX);
 | |
|       if (info->muxID == -1) {
 | |
|          info->muxID = find_ctrl(info->hInputMixer,
 | |
|                                  MIXERLINE_COMPONENTTYPE_DST_WAVEIN,
 | |
|                                  MIXERCONTROL_CONTROLTYPE_MIXER);
 | |
|       }
 | |
| 
 | |
|       info->numInputs = get_ctrls(info->hInputMixer,
 | |
|                                   MIXERLINE_COMPONENTTYPE_DST_WAVEIN,
 | |
|                                   &info->src);
 | |
| 
 | |
|       if (info->numInputs == 0) {
 | |
|          return cleanup(Px);
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    if (deviceOut != UINT_MAX) {
 | |
|       res = mixerOpen((LPHMIXER) &info->hOutputMixer,
 | |
|                       deviceOut,
 | |
|                       0,
 | |
|                       0,
 | |
|                       MIXER_OBJECTF_MIXER);
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          return cleanup(Px);
 | |
|       }
 | |
| 
 | |
|       info->speakerID = find_ctrl(info->hOutputMixer,
 | |
|                                   MIXERLINE_COMPONENTTYPE_DST_SPEAKERS,
 | |
|                                   MIXERCONTROL_CONTROLTYPE_VOLUME);
 | |
| 
 | |
|       info->waveID = find_ctrl(info->hOutputMixer,
 | |
|                                MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT,
 | |
|                                MIXERCONTROL_CONTROLTYPE_VOLUME);
 | |
| 
 | |
|       info->numOutputs = get_ctrls(info->hOutputMixer,
 | |
|                                    MIXERLINE_COMPONENTTYPE_DST_SPEAKERS,
 | |
|                                    &info->dst);
 | |
| 
 | |
|       if (info->numOutputs == 0) {
 | |
|          return cleanup(Px);
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    return TRUE;
 | |
| }
 | |
| 
 | |
| static UINT get_ctrls(HMIXEROBJ mixer, DWORD lineID, PxCtrl **pctrls)
 | |
| {
 | |
|    MMRESULT res;
 | |
|    MIXERLINE line;
 | |
|    MIXERLINECONTROLS controls;
 | |
|    MIXERCONTROL control;
 | |
|    PxCtrl *ctrls = NULL;
 | |
|    UINT num;
 | |
|    UINT s;
 | |
| 
 | |
|    do {
 | |
|       line.cbStruct = sizeof(MIXERLINE);
 | |
|       line.dwComponentType = lineID;
 | |
| 
 | |
|       res = mixerGetLineInfo(mixer,
 | |
|                              &line,
 | |
|                              MIXER_GETLINEINFOF_COMPONENTTYPE);
 | |
| 
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       num = (UINT) line.cConnections;
 | |
| 
 | |
|       ctrls = calloc(num, sizeof(PxInfo));
 | |
|       if (ctrls == NULL) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       for (s = 0; s < num; s++)
 | |
|       {
 | |
|          line.dwSource      = s;
 | |
| 
 | |
|          res = mixerGetLineInfo(mixer,
 | |
|                                 &line,
 | |
|                                 MIXER_GETLINEINFOF_SOURCE);
 | |
| 
 | |
|          if (res != MMSYSERR_NOERROR) {
 | |
|             break;
 | |
|          }
 | |
| 
 | |
|          ctrls[s].lineID = line.dwLineID;
 | |
|          ctrls[s].name = strdup(line.szName);
 | |
|          if (ctrls[s].name == NULL) {
 | |
|             break;
 | |
|          }
 | |
| 
 | |
|          controls.cbStruct = sizeof(MIXERLINECONTROLS);
 | |
|          controls.dwLineID = line.dwLineID;
 | |
|          controls.dwControlType = MIXERCONTROL_CONTROLTYPE_VOLUME;
 | |
|          controls.cbmxctrl = sizeof(MIXERCONTROL);
 | |
|          controls.pamxctrl = &control;
 | |
| 
 | |
|          control.cbStruct = sizeof(MIXERCONTROL);
 | |
| 
 | |
|          res = mixerGetLineControls(mixer,
 | |
|                                     &controls,
 | |
|                                     MIXER_GETLINECONTROLSF_ONEBYTYPE);
 | |
| 
 | |
|          if (res == MMSYSERR_NOERROR)
 | |
|             ctrls[s].controlID = control.dwControlID;
 | |
|          else
 | |
|             ctrls[s].controlID = 0;
 | |
|       }
 | |
| 
 | |
|       if (s != num) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       *pctrls = ctrls;
 | |
| 
 | |
|       return num;
 | |
| 
 | |
|    } while (FALSE);
 | |
| 
 | |
|    if (ctrls) {
 | |
|       for (s = 0; s < num; s++) {
 | |
|          if (ctrls[s].name) {
 | |
|             free(ctrls[s].name);
 | |
|          }
 | |
|       }
 | |
| 
 | |
|       free(ctrls);
 | |
|    }
 | |
| 
 | |
|    return 0;
 | |
| }
 | |
| 
 | |
| static DWORD find_ctrl(HMIXEROBJ mixer, DWORD lineID, DWORD ctrlID)
 | |
| {
 | |
|    MMRESULT res;
 | |
|    MIXERLINE line;
 | |
|    MIXERLINECONTROLS controls;
 | |
|    MIXERCONTROL control;
 | |
| 
 | |
|    line.cbStruct = sizeof(MIXERLINE);
 | |
|    line.dwComponentType = lineID;
 | |
| 
 | |
|    res = mixerGetLineInfo(mixer,
 | |
|                           &line,
 | |
|                           MIXER_GETLINEINFOF_COMPONENTTYPE);
 | |
| 
 | |
|    if (res != MMSYSERR_NOERROR) {
 | |
|       return -1;
 | |
|    }
 | |
| 
 | |
|    controls.cbStruct = sizeof(MIXERLINECONTROLS);
 | |
|    controls.dwLineID = line.dwLineID;
 | |
|    controls.dwControlType = ctrlID;
 | |
|    controls.cbmxctrl = sizeof(MIXERCONTROL);
 | |
|    controls.pamxctrl = &control;
 | |
| 
 | |
|    control.cbStruct = sizeof(MIXERCONTROL);
 | |
| 
 | |
|    res = mixerGetLineControls(mixer,
 | |
|                               &controls,
 | |
|                               MIXER_GETLINECONTROLSF_ONEBYTYPE);
 | |
| 
 | |
|    if (res != MMSYSERR_NOERROR) {
 | |
|       return -1;
 | |
|    }
 | |
| 
 | |
|    return control.dwControlID;
 | |
| }
 | |
| 
 | |
| static int initialize(px_mixer *Px)
 | |
| {
 | |
|    Px->info = calloc(1, sizeof(PxInfo));
 | |
|    if (Px->info == NULL) {
 | |
|       return FALSE;
 | |
|    }
 | |
| 
 | |
|    Px->CloseMixer = close_mixer;
 | |
|    Px->GetNumMixers = get_num_mixers;
 | |
|    Px->GetMixerName = get_mixer_name;
 | |
|    Px->GetMasterVolume = get_master_volume;
 | |
|    Px->SetMasterVolume = set_master_volume;
 | |
|    Px->SupportsPCMOutputVolume = supports_pcm_output_volume;
 | |
|    Px->GetPCMOutputVolume = get_pcm_output_volume;
 | |
|    Px->SetPCMOutputVolume = set_pcm_output_volume;
 | |
|    Px->GetNumOutputVolumes = get_num_output_volumes;
 | |
|    Px->GetOutputVolumeName = get_output_volume_name;
 | |
|    Px->GetOutputVolume = get_output_volume;
 | |
|    Px->SetOutputVolume = set_output_volume;
 | |
|    Px->GetNumInputSources = get_num_input_sources;
 | |
|    Px->GetInputSourceName = get_input_source_name;
 | |
|    Px->GetCurrentInputSource = get_current_input_source;
 | |
|    Px->SetCurrentInputSource = set_current_input_source;
 | |
|    Px->GetInputVolume = get_input_volume;
 | |
|    Px->SetInputVolume = set_input_volume;
 | |
|    
 | |
| // Px->SupportsOutputBalance = supports_output_balance;
 | |
| // Px->GetOutputBalance = get_output_balance;
 | |
| // Px->SetOutputBalance = set_output_balance;
 | |
|    
 | |
| // Px->SupportsPlaythrough = supports_play_through;
 | |
| // Px->GetPlaythrough = get_play_through;
 | |
| // Px->SetPlaythrough = set_play_through;
 | |
|    
 | |
|    return TRUE;
 | |
| }
 | |
| 
 | |
| static int cleanup(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    int i;
 | |
| 
 | |
|    if (info) {
 | |
|       if (info->hInputMixer) {
 | |
|          mixerClose((HMIXER) info->hInputMixer);
 | |
|       }
 | |
|    
 | |
|       if (info->hOutputMixer) {
 | |
|          mixerClose((HMIXER) info->hOutputMixer);
 | |
|       }
 | |
| 
 | |
|       if (info->inName) {
 | |
|          free(info->inName);
 | |
|       }
 | |
| 
 | |
|       if (info->outName) {
 | |
|          free(info->outName);
 | |
|       }
 | |
| 
 | |
|       if (info->src) {
 | |
|          for (i = 0; i < info->numInputs; i++) {
 | |
|             if (info->src[i].name) {
 | |
|                free(info->src[i].name);
 | |
|             }
 | |
|          }
 | |
| 
 | |
|          free(info->src);
 | |
|       }
 | |
| 
 | |
|       if (info->dst) {
 | |
|          for (i = 0; i < info->numOutputs; i++) {
 | |
|             if (info->dst[i].name) {
 | |
|                free(info->dst[i].name);
 | |
|             }
 | |
|          }
 | |
| 
 | |
|          free(info->dst);
 | |
|       }
 | |
| 
 | |
|       free(info);
 | |
|       Px->info = NULL;
 | |
|    }
 | |
| 
 | |
|    return FALSE;
 | |
| }
 | |
| 
 | |
| static void close_mixer(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    cleanup(Px);
 | |
| }
 | |
| 
 | |
| static int get_num_mixers(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    return (info->hInputMixer != NULL) + (info->hOutputMixer != NULL);
 | |
| }
 | |
| 
 | |
| static const char *get_mixer_name(px_mixer *Px, int i)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    MIXERCAPS caps;
 | |
|    MMRESULT res;
 | |
| 
 | |
|    if (i > 1) {
 | |
|       return NULL;
 | |
|    }
 | |
| 
 | |
|    if (i == 0) {
 | |
|       if (!info->inName) {
 | |
|          res = mixerGetDevCaps((UINT) info->hInputMixer, &caps, sizeof(caps));
 | |
|          if (res == MMSYSERR_NOERROR) {
 | |
|             info->inName = strdup(caps.szPname);
 | |
|          }
 | |
|       }
 | |
|       return info->inName;
 | |
|    }
 | |
| 
 | |
|    if (!info->outName) {
 | |
|       res = mixerGetDevCaps((UINT) info->hOutputMixer, &caps, sizeof(caps));
 | |
|       if (res == MMSYSERR_NOERROR) {
 | |
|          info->outName = strdup(caps.szPname);
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    return info->outName;
 | |
| }
 | |
| 
 | |
| static PxVolume VolumeFunction(HMIXEROBJ hMixer, DWORD controlID, PxVolume volume)
 | |
| {
 | |
|    MIXERCONTROLDETAILS details;
 | |
|    MMRESULT result;
 | |
|    MIXERCONTROLDETAILS_UNSIGNED value;
 | |
| 
 | |
|    if (hMixer == NULL) {
 | |
|       return -1.0;
 | |
|    }
 | |
|    
 | |
|    memset(&value, 0, sizeof(MIXERCONTROLDETAILS_UNSIGNED));
 | |
|    
 | |
|    details.cbStruct = sizeof(MIXERCONTROLDETAILS);
 | |
|    details.dwControlID = controlID;
 | |
|    details.cChannels = 1; /* all channels */
 | |
|    details.cMultipleItems = 0;
 | |
|    details.cbDetails = sizeof(MIXERCONTROLDETAILS_UNSIGNED);
 | |
|    details.paDetails = &value;
 | |
| 
 | |
|    if (volume < 0.0) {
 | |
|       result = mixerGetControlDetails(hMixer,
 | |
|                                       &details,
 | |
|                                       MIXER_GETCONTROLDETAILSF_VALUE);
 | |
|       if (result != MMSYSERR_NOERROR)
 | |
|          return -1.0;
 | |
|       
 | |
|       return (PxVolume)(value.dwValue / 65535.0);
 | |
|    }
 | |
|    
 | |
|    value.dwValue = (unsigned short)(volume * 65535.0);
 | |
|    result = mixerSetControlDetails(hMixer,
 | |
|                                    &details,
 | |
|                                    MIXER_GETCONTROLDETAILSF_VALUE);
 | |
|    
 | |
|    if (result != MMSYSERR_NOERROR)
 | |
|       return -1.0;
 | |
|    
 | |
|    return 0.0;
 | |
| }
 | |
| 
 | |
| static PxVolume get_master_volume(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    
 | |
|    return VolumeFunction(info->hOutputMixer, info->speakerID, -1.0);
 | |
| }
 | |
| 
 | |
| static void set_master_volume(px_mixer *Px, PxVolume volume)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    VolumeFunction(info->hOutputMixer, info->speakerID, volume);
 | |
|    
 | |
|    return;
 | |
| }
 | |
| 
 | |
| /*
 | |
| || Main output volume
 | |
| */
 | |
| 
 | |
| static int supports_pcm_output_volume(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    
 | |
|    return (info->waveID != -1);
 | |
| }
 | |
| 
 | |
| static PxVolume get_pcm_output_volume(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    
 | |
|    return VolumeFunction(info->hOutputMixer, info->waveID, -1.0);
 | |
| }
 | |
| 
 | |
| static void set_pcm_output_volume(px_mixer *Px, PxVolume volume)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    VolumeFunction(info->hOutputMixer, info->waveID, volume);
 | |
| 
 | |
|    return;
 | |
| }
 | |
| 
 | |
| /*
 | |
| || Output info
 | |
| */
 | |
| 
 | |
| static int get_num_output_volumes(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    return info->numOutputs;
 | |
| }
 | |
| 
 | |
| static const char *get_output_volume_name(px_mixer *Px, int i)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    if (i < info->numOutputs) {
 | |
|       return info->dst[i].name;
 | |
|    }
 | |
| 
 | |
|    return NULL;
 | |
| }
 | |
| 
 | |
| /*
 | |
| || Output volume
 | |
| */
 | |
| 
 | |
| static PxVolume get_output_volume(px_mixer *Px, int i)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    if (i < info->numOutputs) {
 | |
|       return VolumeFunction(info->hOutputMixer,
 | |
|                             info->src[i].controlID,
 | |
|                             -1.0);
 | |
|    }
 | |
| 
 | |
|    return -1.0;
 | |
| }
 | |
| 
 | |
| static void set_output_volume(px_mixer *Px, int i, PxVolume volume)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    if (i < info->numOutputs) {
 | |
|       VolumeFunction(info->hOutputMixer,
 | |
|                      info->src[i].controlID,
 | |
|                      volume);
 | |
|    }
 | |
| 
 | |
|    return;
 | |
| }
 | |
| 
 | |
| /*
 | |
| || Input source
 | |
| */
 | |
| 
 | |
| static int get_num_input_sources(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    return info->numInputs;
 | |
| }
 | |
| 
 | |
| static const char *get_input_source_name(px_mixer *Px, int i)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
| 
 | |
|    if (i < info->numInputs) {
 | |
|       return info->src[i].name;
 | |
|    }
 | |
| 
 | |
|    return NULL;
 | |
| }
 | |
| 
 | |
| static int get_current_input_source(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    MIXERCONTROLDETAILS details;
 | |
|    MIXERCONTROLDETAILS_LISTTEXT *list;
 | |
|    MIXERCONTROLDETAILS_BOOLEAN *flags;
 | |
|    MMRESULT res;
 | |
|    int i = 0;
 | |
|    int j;
 | |
| 
 | |
|    if (!info->hInputMixer) {
 | |
|       return -1;
 | |
|    }
 | |
| 
 | |
|    do {
 | |
|       list = calloc(info->numInputs, sizeof(MIXERCONTROLDETAILS_LISTTEXT));
 | |
|       flags = calloc(info->numInputs, sizeof(MIXERCONTROLDETAILS_BOOLEAN));
 | |
|       if (list == NULL || flags == NULL) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       details.cbStruct = sizeof(MIXERCONTROLDETAILS);
 | |
|       details.dwControlID = info->muxID;
 | |
|       details.cMultipleItems = info->numInputs;
 | |
|       details.cChannels = 1;   
 | |
|       details.cbDetails = sizeof(MIXERCONTROLDETAILS_LISTTEXT);
 | |
|       details.paDetails = &list[0];
 | |
| 
 | |
|       res = mixerGetControlDetails(info->hInputMixer,
 | |
|                                    &details,
 | |
|                                    MIXER_GETCONTROLDETAILSF_LISTTEXT);
 | |
| 
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       details.cbStruct = sizeof(MIXERCONTROLDETAILS);
 | |
|       details.dwControlID = info->muxID;
 | |
|       details.cMultipleItems = info->numInputs;
 | |
|       details.cChannels = 1;   
 | |
|       details.cbDetails = sizeof(MIXERCONTROLDETAILS_BOOLEAN);
 | |
|       details.paDetails = &flags[0];
 | |
| 
 | |
|       res = mixerGetControlDetails(info->hInputMixer,
 | |
|                                    &details,
 | |
|                                    MIXER_SETCONTROLDETAILSF_VALUE);
 | |
| 
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       for (j = 0; j < info->numInputs; j++) {
 | |
|          if (flags[j].fValue) {
 | |
|             i = j;
 | |
|             break;
 | |
|          }
 | |
|       }
 | |
| 
 | |
|       for (j = 0; j < info->numInputs; j++) {
 | |
|          if (info->src[j].lineID == list[i].dwParam1 ) {
 | |
|             i = j;
 | |
|             break;
 | |
|          }
 | |
|       }
 | |
| 
 | |
|    } while(FALSE);
 | |
| 
 | |
|    if (list) {
 | |
|       free(list);
 | |
|    }
 | |
| 
 | |
|    if (flags) {
 | |
|       free(flags);
 | |
|    }
 | |
| 
 | |
|    return i;
 | |
| }
 | |
| 
 | |
| static void set_current_input_source(px_mixer *Px, int i)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    MIXERCONTROLDETAILS details;
 | |
|    MIXERCONTROLDETAILS_LISTTEXT *list;
 | |
|    MIXERCONTROLDETAILS_BOOLEAN *flags;
 | |
|    MMRESULT res;
 | |
|    int j;
 | |
| 
 | |
|    if (!info->hInputMixer) {
 | |
|       return;
 | |
|    }
 | |
| 
 | |
|    do {
 | |
|       list = calloc(info->numInputs, sizeof(MIXERCONTROLDETAILS_LISTTEXT));
 | |
|       flags = calloc(info->numInputs, sizeof(MIXERCONTROLDETAILS_BOOLEAN));
 | |
|       if (list == NULL || flags == NULL) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       details.cbStruct = sizeof(MIXERCONTROLDETAILS);
 | |
|       details.dwControlID = info->muxID;
 | |
|       details.cMultipleItems = info->numInputs;
 | |
|       details.cChannels = 1;   
 | |
|       details.cbDetails = sizeof(MIXERCONTROLDETAILS_LISTTEXT);
 | |
|       details.paDetails = &list[0];
 | |
| 
 | |
|       res = mixerGetControlDetails(info->hInputMixer,
 | |
|                                    &details,
 | |
|                                    MIXER_GETCONTROLDETAILSF_LISTTEXT);
 | |
| 
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          break;
 | |
|       }
 | |
| 
 | |
|       for (j = 0; j < info->numInputs; j++) {
 | |
|          if (list[j].dwParam1 == info->src[i].lineID) {
 | |
|             flags[j].fValue = TRUE;
 | |
|             break;
 | |
|          }
 | |
|       }
 | |
| 
 | |
|       details.cbStruct = sizeof(MIXERCONTROLDETAILS);
 | |
|       details.dwControlID = info->muxID;
 | |
|       details.cMultipleItems = info->numInputs;
 | |
|       details.cChannels = 1;   
 | |
|       details.cbDetails = sizeof(MIXERCONTROLDETAILS_BOOLEAN);
 | |
|       details.paDetails = &flags[0];
 | |
| 
 | |
|       res = mixerSetControlDetails(info->hInputMixer,
 | |
|                                    &details,
 | |
|                                    MIXER_SETCONTROLDETAILSF_VALUE);
 | |
| 
 | |
|       if (res != MMSYSERR_NOERROR) {
 | |
|          break;
 | |
|       }
 | |
|    } while(FALSE);
 | |
| 
 | |
|    if (list) {
 | |
|       free(list);
 | |
|    }
 | |
| 
 | |
|    if (flags) {
 | |
|       free(flags);
 | |
|    }
 | |
| 
 | |
|    return;
 | |
| }
 | |
| 
 | |
| /*
 | |
| || Input volume
 | |
| */
 | |
| 
 | |
| static PxVolume get_input_volume(px_mixer *Px)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    int src;
 | |
| 
 | |
|    if (info->hInputMixer) {
 | |
|       src = get_current_input_source(Px);
 | |
|       if (src < info->numInputs) {
 | |
|          return VolumeFunction(info->hInputMixer,
 | |
|                                info->src[src].controlID,
 | |
|                                -1.0);
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    return -1.0;
 | |
| }
 | |
| 
 | |
| static void set_input_volume(px_mixer *Px, PxVolume volume)
 | |
| {
 | |
|    PxInfo *info = (PxInfo *)Px->info;
 | |
|    int src;
 | |
| 
 | |
|    if (info->hInputMixer) {
 | |
|       src = get_current_input_source(Px);
 | |
|       if (src < info->numInputs) {
 | |
|          VolumeFunction(info->hInputMixer,
 | |
|                         info->src[src].controlID,
 | |
|                         volume);
 | |
|       }
 | |
|    }
 | |
| 
 | |
|    return;
 | |
| }
 |