| 1 | /* $Id: mixer.cpp,v 1.26 2002-07-12 08:12:29 sandervl Exp $ */ | 
|---|
| 2 |  | 
|---|
| 3 | /* | 
|---|
| 4 | * Mixer functions | 
|---|
| 5 | * | 
|---|
| 6 | * Copyright 2002 Sander van Leeuwen (sandervl@xs4all.nl) | 
|---|
| 7 | * | 
|---|
| 8 | * TODO: Mixer notification | 
|---|
| 9 | * | 
|---|
| 10 | * NOTE: Not really flexible (capabilities, > 1 audio card) | 
|---|
| 11 | * | 
|---|
| 12 | * Some portions borrowed from Wine (X11): (dll\winmm\mmsystem.c) | 
|---|
| 13 | * (mixerGetLineControlsW/mixerGetControlDetailsW) | 
|---|
| 14 | * Copyright 1993 Martin Ayotte | 
|---|
| 15 | *                Eric POUECH | 
|---|
| 16 | * | 
|---|
| 17 | * Project Odin Software License can be found in LICENSE.TXT | 
|---|
| 18 | * | 
|---|
| 19 | */ | 
|---|
| 20 |  | 
|---|
| 21 | /**************************************************************************** | 
|---|
| 22 | * Includes                                                                 * | 
|---|
| 23 | ****************************************************************************/ | 
|---|
| 24 |  | 
|---|
| 25 | #define NONAMELESSUNION | 
|---|
| 26 | #define NONAMELESSSTRUCT | 
|---|
| 27 |  | 
|---|
| 28 | #include <os2win.h> | 
|---|
| 29 | #include <string.h> | 
|---|
| 30 | #include <dbglog.h> | 
|---|
| 31 | #include <mmsystem.h> | 
|---|
| 32 | #include <winnls.h> | 
|---|
| 33 |  | 
|---|
| 34 | #include "waveoutdart.h" | 
|---|
| 35 | #include "winmm.h" | 
|---|
| 36 | #include "initwinmm.h" | 
|---|
| 37 | #include "mixer.h" | 
|---|
| 38 | #include "mixeros2.h" | 
|---|
| 39 |  | 
|---|
| 40 | #define DBG_LOCALLOG    DBG_mixer | 
|---|
| 41 | #include "dbglocal.h" | 
|---|
| 42 |  | 
|---|
| 43 | static MIXERCONTROLA *mixerAddControl(DWORD dwControl, MIXLINE *pSrcLine); | 
|---|
| 44 | static MIXLINE       *mixerAddSource(MIXLINE *pDestLine, DWORD dwSource); | 
|---|
| 45 | static MIXLINE       *mixerAddDestination(DWORD dwDest); | 
|---|
| 46 |  | 
|---|
| 47 | //array of destination mixer lines | 
|---|
| 48 | static MIXLINE       mixerDest[MAX_MIXER_DESTINATIONS] = {0}; | 
|---|
| 49 | //array of source mixer lines | 
|---|
| 50 | static MIXLINE       mixerSource[MAX_MIXER_SOURCES]    = {0}; | 
|---|
| 51 | //array of all mixer lines | 
|---|
| 52 | static MIXLINE      *pmixerLines[MAX_MIXER_LINES]      = {NULL}; | 
|---|
| 53 | //array of all mixer controls | 
|---|
| 54 | static MIXCONTROL    mixerControls[MAX_MIXER_CONTROLS] = {0}; | 
|---|
| 55 |  | 
|---|
| 56 | static int           nrDestinations    = 0; | 
|---|
| 57 | static int           nrSources         = 0; | 
|---|
| 58 | static int           nrControls        = 0; | 
|---|
| 59 | static int           nrLines           = 0; | 
|---|
| 60 |  | 
|---|
| 61 | /******************************************************************************/ | 
|---|
| 62 | /******************************************************************************/ | 
|---|
| 63 | MMRESULT WINAPI mixerGetControlDetailsA(HMIXEROBJ hmxobj, LPMIXERCONTROLDETAILS lpmcd, DWORD fdwDetails) | 
|---|
| 64 | { | 
|---|
| 65 | DWORD       lineID, controlType; | 
|---|
| 66 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 67 |  | 
|---|
| 68 | if((fdwDetails & 0xF0000000)== MIXER_OBJECTF_HMIXER) { | 
|---|
| 69 | if(!pMixInfo) { | 
|---|
| 70 | return MMSYSERR_INVALHANDLE; | 
|---|
| 71 | } | 
|---|
| 72 | } | 
|---|
| 73 | else | 
|---|
| 74 | if((fdwDetails & 0xF0000000) == MIXER_OBJECTF_MIXER) { | 
|---|
| 75 | if(!HIWORD(hmxobj) && hmxobj > 0) { | 
|---|
| 76 | return MMSYSERR_NODRIVER; | 
|---|
| 77 | } | 
|---|
| 78 | } | 
|---|
| 79 |  | 
|---|
| 80 | if(lpmcd == NULL || lpmcd->cbStruct != sizeof(MIXERCONTROLDETAILS)) { | 
|---|
| 81 | dprintf(("ERROR: invalid pointer or structure size %d (%d)", (lpmcd) ? lpmcd->cbStruct : 0, sizeof(MIXERCONTROLDETAILS))); | 
|---|
| 82 | return MMSYSERR_INVALPARAM; | 
|---|
| 83 | } | 
|---|
| 84 | switch (fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK) { | 
|---|
| 85 | case MIXER_GETCONTROLDETAILSF_VALUE: | 
|---|
| 86 | dprintf(("MIXER_GETCONTROLDETAILSF_VALUE %d (internal id %d)", lpmcd->dwControlID, mixerControls[lpmcd->dwControlID].id)); | 
|---|
| 87 | if(lpmcd->dwControlID >= nrControls) { | 
|---|
| 88 | dprintf(("invalid control %d", lpmcd->dwControlID)); | 
|---|
| 89 | return MIXERR_INVALCONTROL; | 
|---|
| 90 | } | 
|---|
| 91 | if(lpmcd->cbDetails < sizeof(MIXERCONTROLDETAILS_UNSIGNED)) { | 
|---|
| 92 | dprintf(("not enough room in buffer (%d)", lpmcd->cbDetails)); | 
|---|
| 93 | return MMSYSERR_INVALPARAM; | 
|---|
| 94 | } | 
|---|
| 95 | switch(mixerControls[lpmcd->dwControlID].ctrl.dwControlType) { | 
|---|
| 96 | case MIXERCONTROL_CONTROLTYPE_MUX: | 
|---|
| 97 | { | 
|---|
| 98 | MIXERCONTROLDETAILS_BOOLEAN *pDetails = (MIXERCONTROLDETAILS_BOOLEAN *)lpmcd->paDetails; | 
|---|
| 99 | DWORD dwRecSrc; | 
|---|
| 100 |  | 
|---|
| 101 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUX")); | 
|---|
| 102 | if(lpmcd->cChannels != 1) {//only accepts 1 in win2k | 
|---|
| 103 | dprintf(("invalid number of channels %d", lpmcd->cChannels)); | 
|---|
| 104 | return MMSYSERR_INVALPARAM; | 
|---|
| 105 | } | 
|---|
| 106 | //this is also checked in win2k | 
|---|
| 107 | if(lpmcd->u.cMultipleItems != mixerControls[lpmcd->dwControlID].ctrl.cMultipleItems) { | 
|---|
| 108 | dprintf(("invalid cMultipleItems %d", lpmcd->u.cMultipleItems)); | 
|---|
| 109 | return MMSYSERR_INVALPARAM; | 
|---|
| 110 | } | 
|---|
| 111 |  | 
|---|
| 112 | if(mixerControls[lpmcd->dwControlID].id != MIX_CTRL_MUX_IN_W_SRC) { | 
|---|
| 113 | dprintf(("oh, oh. not wavein mux")); | 
|---|
| 114 | DebugInt3(); | 
|---|
| 115 | return MMSYSERR_INVALPARAM; | 
|---|
| 116 | } | 
|---|
| 117 |  | 
|---|
| 118 | if(OSLibMixGetRecSource(&dwRecSrc) == FALSE) { | 
|---|
| 119 | dprintf(("OSLibMixGetRecSource failed!!")); | 
|---|
| 120 | return MIXERR_INVALCONTROL; | 
|---|
| 121 | } | 
|---|
| 122 | //clear the array | 
|---|
| 123 | memset(pDetails, 0, sizeof(MIXERCONTROLDETAILS_BOOLEAN)*lpmcd->u.cMultipleItems); | 
|---|
| 124 | //mark recording source | 
|---|
| 125 | for(int i=0;i<nrDestinations;i++) { | 
|---|
| 126 | if(mixerDest[i].id == MIXER_DEST_WAVEIN) { | 
|---|
| 127 | for(int j=0;j<mixerDest[i].cConnections;j++) { | 
|---|
| 128 | dprintf(("wavein source %s %d (id %d)", pmixerLines[mixerDest[i].Connections[j]]->line.szName, pmixerLines[mixerDest[i].Connections[j]]->id, mixerDest[i].Connections[j])); | 
|---|
| 129 | if(pmixerLines[mixerDest[i].Connections[j]]->id == dwRecSrc) { | 
|---|
| 130 | pDetails[j].fValue = 1; | 
|---|
| 131 | return MMSYSERR_NOERROR; | 
|---|
| 132 | } | 
|---|
| 133 | } | 
|---|
| 134 | } | 
|---|
| 135 | } | 
|---|
| 136 | dprintf(("recording source %d not found!!", dwRecSrc)); | 
|---|
| 137 | break; | 
|---|
| 138 | } | 
|---|
| 139 |  | 
|---|
| 140 | case MIXERCONTROL_CONTROLTYPE_VOLUME: //unsigned | 
|---|
| 141 | { | 
|---|
| 142 | MIXERCONTROLDETAILS_UNSIGNED *pDetails = (MIXERCONTROLDETAILS_UNSIGNED *)lpmcd->paDetails; | 
|---|
| 143 | DWORD dwVolLeft, dwVolRight; | 
|---|
| 144 |  | 
|---|
| 145 | dprintf(("MIXERCONTROL_CONTROLTYPE_VOLUME")); | 
|---|
| 146 | if(OSLibMixGetVolume(mixerControls[lpmcd->dwControlID].id, &dwVolLeft, &dwVolRight) == FALSE) { | 
|---|
| 147 | dprintf(("OSLibMixGetVolume failed!!")); | 
|---|
| 148 | return MIXERR_INVALCONTROL; | 
|---|
| 149 | } | 
|---|
| 150 | pDetails->dwValue = dwVolLeft; | 
|---|
| 151 | dprintf(("%s Left volume %d", mixerControls[lpmcd->dwControlID].ctrl.szName, dwVolLeft)); | 
|---|
| 152 | if(lpmcd->cChannels == 2) { | 
|---|
| 153 | dprintf(("%s Right volume %d", mixerControls[lpmcd->dwControlID].ctrl.szName, dwVolRight)); | 
|---|
| 154 | pDetails += 1; | 
|---|
| 155 | pDetails->dwValue = dwVolRight; | 
|---|
| 156 | } | 
|---|
| 157 | //todo > 2 channels | 
|---|
| 158 | return MMSYSERR_NOERROR; | 
|---|
| 159 | } | 
|---|
| 160 |  | 
|---|
| 161 | case MIXERCONTROL_CONTROLTYPE_MUTE: //assuming boolean | 
|---|
| 162 | { | 
|---|
| 163 | MIXERCONTROLDETAILS_BOOLEAN *pDetails = (MIXERCONTROLDETAILS_BOOLEAN *)lpmcd->paDetails; | 
|---|
| 164 | BOOL  fMute; | 
|---|
| 165 |  | 
|---|
| 166 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUTE")); | 
|---|
| 167 | if(OSLibMixGetMute(mixerControls[lpmcd->dwControlID].id, &fMute) == FALSE) { | 
|---|
| 168 | dprintf(("OSLibMixGetVolume failed!!")); | 
|---|
| 169 | return MIXERR_INVALCONTROL; | 
|---|
| 170 | } | 
|---|
| 171 | pDetails->fValue = fMute; | 
|---|
| 172 | dprintf(("%s Left mute %d", mixerControls[lpmcd->dwControlID].ctrl.szName, fMute)); | 
|---|
| 173 | if(lpmcd->cChannels == 2) { | 
|---|
| 174 | dprintf(("%s Right mute %d", mixerControls[lpmcd->dwControlID].ctrl.szName, fMute)); | 
|---|
| 175 | pDetails += 1; | 
|---|
| 176 | pDetails->fValue = fMute; | 
|---|
| 177 | } | 
|---|
| 178 | //todo > 2 channels (usually only 1 channel is requested though) | 
|---|
| 179 | return MMSYSERR_NOERROR; | 
|---|
| 180 | } | 
|---|
| 181 |  | 
|---|
| 182 | case MIXERCONTROL_CONTROLTYPE_FADER: | 
|---|
| 183 | { | 
|---|
| 184 | MIXERCONTROLDETAILS_UNSIGNED *pDetails = (MIXERCONTROLDETAILS_UNSIGNED *)lpmcd->paDetails; | 
|---|
| 185 | DWORD dwLevelL; | 
|---|
| 186 |  | 
|---|
| 187 | dprintf(("MIXERCONTROL_CONTROLTYPE_FADER")); | 
|---|
| 188 |  | 
|---|
| 189 | if(OSLibMixGetVolume(mixerControls[lpmcd->dwControlID].id, &dwLevelL, NULL) == FALSE) { | 
|---|
| 190 | dprintf(("OSLibMixGetVolume failed!!")); | 
|---|
| 191 | return MIXERR_INVALCONTROL; | 
|---|
| 192 | } | 
|---|
| 193 | pDetails->dwValue = dwLevelL; | 
|---|
| 194 | #ifdef DEBUG | 
|---|
| 195 | if(mixerControls[lpmcd->dwControlID].id == MIX_CTRL_OUT_L_3DCENTER) { | 
|---|
| 196 | dprintf(("3D center %d", dwLevelL)); | 
|---|
| 197 | } | 
|---|
| 198 | else dprintf(("3D depth %d", dwLevelL)); | 
|---|
| 199 | #endif | 
|---|
| 200 | if(lpmcd->cChannels == 2) { | 
|---|
| 201 | pDetails += 1; | 
|---|
| 202 | pDetails->dwValue = dwLevelL; | 
|---|
| 203 | } | 
|---|
| 204 | //todo > 2 channels (??) | 
|---|
| 205 | return MMSYSERR_NOERROR; | 
|---|
| 206 | } | 
|---|
| 207 |  | 
|---|
| 208 | case MIXERCONTROL_CONTROLTYPE_BASS: | 
|---|
| 209 | case MIXERCONTROL_CONTROLTYPE_TREBLE: | 
|---|
| 210 | { | 
|---|
| 211 | MIXERCONTROLDETAILS_UNSIGNED *pDetails = (MIXERCONTROLDETAILS_UNSIGNED *)lpmcd->paDetails; | 
|---|
| 212 | DWORD dwLevelL; | 
|---|
| 213 |  | 
|---|
| 214 | #ifdef DEBUG | 
|---|
| 215 | switch(fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK) { | 
|---|
| 216 | case MIXERCONTROL_CONTROLTYPE_TREBLE: | 
|---|
| 217 | dprintf(("MIXERCONTROL_CONTROLTYPE_TREBLE")); | 
|---|
| 218 | break; | 
|---|
| 219 | case MIXERCONTROL_CONTROLTYPE_BASS: | 
|---|
| 220 | dprintf(("MIXERCONTROL_CONTROLTYPE_BASS")); | 
|---|
| 221 | break; | 
|---|
| 222 | } | 
|---|
| 223 | #endif | 
|---|
| 224 |  | 
|---|
| 225 | if(OSLibMixGetVolume(mixerControls[lpmcd->dwControlID].id, &dwLevelL, NULL) == FALSE) { | 
|---|
| 226 | dprintf(("OSLibMixGetVolume failed!!")); | 
|---|
| 227 | return MIXERR_INVALCONTROL; | 
|---|
| 228 | } | 
|---|
| 229 | pDetails->dwValue = dwLevelL; | 
|---|
| 230 | dprintf(("%s %d", mixerControls[lpmcd->dwControlID].ctrl.szName, dwLevelL)); | 
|---|
| 231 | if(lpmcd->cChannels == 2) { | 
|---|
| 232 | pDetails += 1; | 
|---|
| 233 | pDetails->dwValue = dwLevelL; | 
|---|
| 234 | } | 
|---|
| 235 | return MMSYSERR_NOERROR; | 
|---|
| 236 | } | 
|---|
| 237 |  | 
|---|
| 238 | #ifdef DEBUG | 
|---|
| 239 | case MIXERCONTROL_CONTROLTYPE_MIXER: | 
|---|
| 240 | dprintf(("MIXERCONTROL_CONTROLTYPE_MIXER")); | 
|---|
| 241 | break; | 
|---|
| 242 | case MIXERCONTROL_CONTROLTYPE_CUSTOM: | 
|---|
| 243 | dprintf(("MIXERCONTROL_CONTROLTYPE_CUSTOM")); | 
|---|
| 244 | break; | 
|---|
| 245 | case MIXERCONTROL_CONTROLTYPE_BOOLEANMETER: | 
|---|
| 246 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEANMETER")); | 
|---|
| 247 | break; | 
|---|
| 248 | case MIXERCONTROL_CONTROLTYPE_SIGNEDMETER: | 
|---|
| 249 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNEDMETER")); | 
|---|
| 250 | break; | 
|---|
| 251 | case MIXERCONTROL_CONTROLTYPE_PEAKMETER: | 
|---|
| 252 | dprintf(("MIXERCONTROL_CONTROLTYPE_PEAKMETER")); | 
|---|
| 253 | break; | 
|---|
| 254 | case MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER: | 
|---|
| 255 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER")); | 
|---|
| 256 | break; | 
|---|
| 257 | case MIXERCONTROL_CONTROLTYPE_BOOLEAN: | 
|---|
| 258 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEAN")); | 
|---|
| 259 | break; | 
|---|
| 260 | case MIXERCONTROL_CONTROLTYPE_ONOFF: | 
|---|
| 261 | dprintf(("MIXERCONTROL_CONTROLTYPE_ONOFF")); | 
|---|
| 262 | break; | 
|---|
| 263 | case MIXERCONTROL_CONTROLTYPE_MONO: | 
|---|
| 264 | dprintf(("MIXERCONTROL_CONTROLTYPE_MONO")); | 
|---|
| 265 | break; | 
|---|
| 266 | case MIXERCONTROL_CONTROLTYPE_LOUDNESS: | 
|---|
| 267 | dprintf(("MIXERCONTROL_CONTROLTYPE_LOUDNESS")); | 
|---|
| 268 | break; | 
|---|
| 269 | case MIXERCONTROL_CONTROLTYPE_STEREOENH: | 
|---|
| 270 | dprintf(("MIXERCONTROL_CONTROLTYPE_STEREOENH")); | 
|---|
| 271 | break; | 
|---|
| 272 | case MIXERCONTROL_CONTROLTYPE_BUTTON: | 
|---|
| 273 | dprintf(("MIXERCONTROL_CONTROLTYPE_BUTTON")); | 
|---|
| 274 | break; | 
|---|
| 275 | case MIXERCONTROL_CONTROLTYPE_DECIBELS: | 
|---|
| 276 | dprintf(("MIXERCONTROL_CONTROLTYPE_DECIBELS")); | 
|---|
| 277 | break; | 
|---|
| 278 | case MIXERCONTROL_CONTROLTYPE_SIGNED: | 
|---|
| 279 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNED")); | 
|---|
| 280 | break; | 
|---|
| 281 | case MIXERCONTROL_CONTROLTYPE_UNSIGNED: | 
|---|
| 282 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNED")); | 
|---|
| 283 | break; | 
|---|
| 284 | case MIXERCONTROL_CONTROLTYPE_PERCENT: | 
|---|
| 285 | dprintf(("MIXERCONTROL_CONTROLTYPE_PERCENT")); | 
|---|
| 286 | break; | 
|---|
| 287 | case MIXERCONTROL_CONTROLTYPE_SLIDER: | 
|---|
| 288 | dprintf(("MIXERCONTROL_CONTROLTYPE_SLIDER")); | 
|---|
| 289 | break; | 
|---|
| 290 | case MIXERCONTROL_CONTROLTYPE_PAN: | 
|---|
| 291 | dprintf(("MIXERCONTROL_CONTROLTYPE_PAN")); | 
|---|
| 292 | break; | 
|---|
| 293 | case MIXERCONTROL_CONTROLTYPE_QSOUNDPAN: | 
|---|
| 294 | dprintf(("MIXERCONTROL_CONTROLTYPE_QSOUNDPAN")); | 
|---|
| 295 | break; | 
|---|
| 296 | case MIXERCONTROL_CONTROLTYPE_EQUALIZER: | 
|---|
| 297 | dprintf(("MIXERCONTROL_CONTROLTYPE_EQUALIZER")); | 
|---|
| 298 | break; | 
|---|
| 299 | case MIXERCONTROL_CONTROLTYPE_SINGLESELECT: | 
|---|
| 300 | dprintf(("MIXERCONTROL_CONTROLTYPE_SINGLESELECT")); | 
|---|
| 301 | break; | 
|---|
| 302 | case MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT: | 
|---|
| 303 | dprintf(("MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT")); | 
|---|
| 304 | break; | 
|---|
| 305 | case MIXERCONTROL_CONTROLTYPE_MICROTIME: | 
|---|
| 306 | dprintf(("MIXERCONTROL_CONTROLTYPE_MICROTIME")); | 
|---|
| 307 | break; | 
|---|
| 308 | case MIXERCONTROL_CONTROLTYPE_MILLITIME: | 
|---|
| 309 | dprintf(("MIXERCONTROL_CONTROLTYPE_MILLITIME")); | 
|---|
| 310 | break; | 
|---|
| 311 | #endif | 
|---|
| 312 | default: | 
|---|
| 313 | DebugInt3(); | 
|---|
| 314 | return MIXERR_INVALCONTROL; | 
|---|
| 315 | } | 
|---|
| 316 | return MIXERR_INVALCONTROL; | 
|---|
| 317 |  | 
|---|
| 318 | case MIXER_GETCONTROLDETAILSF_LISTTEXT: | 
|---|
| 319 | dprintf(("MIXER_GETCONTROLDETAILSF_LISTTEXT %d", lpmcd->dwControlID)); | 
|---|
| 320 | if(lpmcd->dwControlID >= nrControls) { | 
|---|
| 321 | dprintf(("invalid control %d", lpmcd->dwControlID)); | 
|---|
| 322 | return MIXERR_INVALCONTROL; | 
|---|
| 323 | } | 
|---|
| 324 | switch(mixerControls[lpmcd->dwControlID].ctrl.dwControlType) { | 
|---|
| 325 | case MIXERCONTROL_CONTROLTYPE_MUX: | 
|---|
| 326 | { | 
|---|
| 327 | MIXERCONTROLDETAILS_LISTTEXTA *pDetails = (MIXERCONTROLDETAILS_LISTTEXTA *)lpmcd->paDetails; | 
|---|
| 328 | if(lpmcd->cChannels != 1) {//only accepts 1 in win2k | 
|---|
| 329 | dprintf(("invalid number of channels %d", lpmcd->cChannels)); | 
|---|
| 330 | return MMSYSERR_INVALPARAM; | 
|---|
| 331 | } | 
|---|
| 332 | //this is also checked in win2k | 
|---|
| 333 | if(lpmcd->u.cMultipleItems != mixerControls[lpmcd->dwControlID].ctrl.cMultipleItems) { | 
|---|
| 334 | dprintf(("invalid cMultipleItems %d", lpmcd->u.cMultipleItems)); | 
|---|
| 335 | return MMSYSERR_INVALPARAM; | 
|---|
| 336 | } | 
|---|
| 337 | //fails otherwise in win2k | 
|---|
| 338 | if(lpmcd->cbDetails != sizeof(MIXERCONTROLDETAILS_LISTTEXTA)) { | 
|---|
| 339 | dprintf(("invalid buffer size %d; should be %d", lpmcd->cbDetails, sizeof(MIXERCONTROLDETAILS_LISTTEXTA))); | 
|---|
| 340 | return MMSYSERR_INVALPARAM; | 
|---|
| 341 | } | 
|---|
| 342 | if(mixerControls[lpmcd->dwControlID].id != MIX_CTRL_MUX_IN_W_SRC) { | 
|---|
| 343 | dprintf(("oh, oh. not wavein mux")); | 
|---|
| 344 | DebugInt3(); | 
|---|
| 345 | return MMSYSERR_INVALPARAM; | 
|---|
| 346 | } | 
|---|
| 347 | //clear the array | 
|---|
| 348 | memset(pDetails, 0, sizeof(MIXERCONTROLDETAILS_LISTTEXTA)*lpmcd->u.cMultipleItems); | 
|---|
| 349 | //set mux source details | 
|---|
| 350 | for(int i=0;i<nrDestinations;i++) { | 
|---|
| 351 | if(mixerDest[i].id == MIXER_DEST_WAVEIN) { | 
|---|
| 352 | for(int j=0;j<mixerDest[i].cConnections;j++) { | 
|---|
| 353 | dprintf(("wavein source %s %d (%s)", pmixerLines[mixerDest[i].Connections[j]]->line.szName, pmixerLines[mixerDest[i].Connections[j]]->id, pmixerLines[mixerDest[i].Connections[j]]->line.szName)); | 
|---|
| 354 | pDetails->dwParam1 = pmixerLines[mixerDest[i].Connections[j]]->line.dwLineID; | 
|---|
| 355 | pDetails->dwParam2 = 0; | 
|---|
| 356 | strncpy(pDetails->szName, pmixerLines[mixerDest[i].Connections[j]]->line.szName, sizeof(pDetails->szName)); | 
|---|
| 357 | pDetails++; | 
|---|
| 358 | } | 
|---|
| 359 | return MMSYSERR_NOERROR; | 
|---|
| 360 | } | 
|---|
| 361 | } | 
|---|
| 362 | DebugInt3(); | 
|---|
| 363 | return MMSYSERR_INVALPARAM; | 
|---|
| 364 | } | 
|---|
| 365 | case MIXERCONTROL_CONTROLTYPE_EQUALIZER: | 
|---|
| 366 | case MIXERCONTROL_CONTROLTYPE_MIXER: | 
|---|
| 367 | case MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT: | 
|---|
| 368 | case MIXERCONTROL_CONTROLTYPE_SINGLESELECT: | 
|---|
| 369 | dprintf(("Only implemented for mux controls")); | 
|---|
| 370 | return MIXERR_INVALCONTROL; | 
|---|
| 371 |  | 
|---|
| 372 | default: | 
|---|
| 373 | dprintf(("Only implemented for multiple item controls")); | 
|---|
| 374 | return MIXERR_INVALCONTROL; | 
|---|
| 375 | } | 
|---|
| 376 |  | 
|---|
| 377 | default: | 
|---|
| 378 | dprintf(("Unknown flag (%08lx)\n", fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK)); | 
|---|
| 379 | break; | 
|---|
| 380 | } | 
|---|
| 381 | return MMSYSERR_NOERROR; | 
|---|
| 382 | } | 
|---|
| 383 | /******************************************************************************/ | 
|---|
| 384 | /******************************************************************************/ | 
|---|
| 385 | MMRESULT WINAPI mixerGetControlDetailsW(HMIXEROBJ hmxobj, LPMIXERCONTROLDETAILS lpmcd, DWORD fdwDetails) | 
|---|
| 386 | { | 
|---|
| 387 | DWORD ret = MMSYSERR_NOTENABLED; | 
|---|
| 388 |  | 
|---|
| 389 | if (lpmcd == NULL || lpmcd->cbStruct != sizeof(*lpmcd)) | 
|---|
| 390 | return MMSYSERR_INVALPARAM; | 
|---|
| 391 |  | 
|---|
| 392 | switch (fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK) { | 
|---|
| 393 | case MIXER_GETCONTROLDETAILSF_VALUE: | 
|---|
| 394 | /* can safely use W structure as it is, no string inside */ | 
|---|
| 395 | ret = mixerGetControlDetailsA(hmxobj, lpmcd, fdwDetails); | 
|---|
| 396 | break; | 
|---|
| 397 |  | 
|---|
| 398 | case MIXER_GETCONTROLDETAILSF_LISTTEXT: | 
|---|
| 399 | { | 
|---|
| 400 | MIXERCONTROLDETAILS_LISTTEXTW *pDetailsW = (MIXERCONTROLDETAILS_LISTTEXTW *)lpmcd->paDetails; | 
|---|
| 401 | MIXERCONTROLDETAILS_LISTTEXTA *pDetailsA; | 
|---|
| 402 | int size = max(1, lpmcd->cChannels) * sizeof(MIXERCONTROLDETAILS_LISTTEXTA); | 
|---|
| 403 | int i; | 
|---|
| 404 |  | 
|---|
| 405 | if (lpmcd->u.cMultipleItems != 0) { | 
|---|
| 406 | size *= lpmcd->u.cMultipleItems; | 
|---|
| 407 | } | 
|---|
| 408 | pDetailsA = (MIXERCONTROLDETAILS_LISTTEXTA *)HeapAlloc(GetProcessHeap(), 0, size); | 
|---|
| 409 | lpmcd->paDetails = pDetailsA; | 
|---|
| 410 | lpmcd->cbDetails = sizeof(MIXERCONTROLDETAILS_LISTTEXTA); | 
|---|
| 411 | /* set up lpmcd->paDetails */ | 
|---|
| 412 | ret = mixerGetControlDetailsA(hmxobj, lpmcd, fdwDetails); | 
|---|
| 413 | /* copy from lpmcd->paDetails back to paDetailsW; */ | 
|---|
| 414 | if(ret == MMSYSERR_NOERROR) { | 
|---|
| 415 | for(i=0;i<lpmcd->u.cMultipleItems*lpmcd->cChannels;i++) { | 
|---|
| 416 | pDetailsW->dwParam1 = pDetailsA->dwParam1; | 
|---|
| 417 | pDetailsW->dwParam2 = pDetailsA->dwParam2; | 
|---|
| 418 | MultiByteToWideChar( CP_ACP, 0, pDetailsA->szName, -1, | 
|---|
| 419 | pDetailsW->szName, | 
|---|
| 420 | sizeof(pDetailsW->szName)/sizeof(WCHAR) ); | 
|---|
| 421 | pDetailsA++; | 
|---|
| 422 | pDetailsW++; | 
|---|
| 423 | } | 
|---|
| 424 | pDetailsA -= lpmcd->u.cMultipleItems*lpmcd->cChannels; | 
|---|
| 425 | pDetailsW -= lpmcd->u.cMultipleItems*lpmcd->cChannels; | 
|---|
| 426 | } | 
|---|
| 427 | HeapFree(GetProcessHeap(), 0, pDetailsA); | 
|---|
| 428 | lpmcd->paDetails = pDetailsW; | 
|---|
| 429 | lpmcd->cbDetails = sizeof(MIXERCONTROLDETAILS_LISTTEXTW); | 
|---|
| 430 | break; | 
|---|
| 431 | } | 
|---|
| 432 | default: | 
|---|
| 433 | dprintf(("Unsupported fdwDetails=0x%08lx\n", fdwDetails)); | 
|---|
| 434 | break; | 
|---|
| 435 | } | 
|---|
| 436 |  | 
|---|
| 437 | return ret; | 
|---|
| 438 | } | 
|---|
| 439 | /******************************************************************************/ | 
|---|
| 440 | /******************************************************************************/ | 
|---|
| 441 | MMRESULT WINAPI mixerSetControlDetails(HMIXEROBJ hmxobj, LPMIXERCONTROLDETAILS lpmcd, DWORD fdwDetails) | 
|---|
| 442 | { | 
|---|
| 443 | DWORD       ret = MMSYSERR_NOTSUPPORTED; | 
|---|
| 444 | DWORD       lineID, controlType; | 
|---|
| 445 | int         val; | 
|---|
| 446 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 447 |  | 
|---|
| 448 | if((fdwDetails & 0xF0000000) == MIXER_OBJECTF_HMIXER) { | 
|---|
| 449 | if(!pMixInfo) { | 
|---|
| 450 | return MMSYSERR_INVALHANDLE; | 
|---|
| 451 | } | 
|---|
| 452 | } | 
|---|
| 453 | else | 
|---|
| 454 | if((fdwDetails & 0xF0000000) == MIXER_OBJECTF_MIXER) { | 
|---|
| 455 | if(!HIWORD(hmxobj) && hmxobj > 0) { | 
|---|
| 456 | return MMSYSERR_NODRIVER; | 
|---|
| 457 | } | 
|---|
| 458 | } | 
|---|
| 459 | if(lpmcd == NULL || lpmcd->cbStruct != sizeof(MIXERCONTROLDETAILS)) { | 
|---|
| 460 | dprintf(("ERROR: invalid pointer or structure size %x %d", lpmcd, lpmcd->cbStruct)); | 
|---|
| 461 | return MMSYSERR_INVALPARAM; | 
|---|
| 462 | } | 
|---|
| 463 |  | 
|---|
| 464 | switch (fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK) { | 
|---|
| 465 | case MIXER_GETCONTROLDETAILSF_VALUE: | 
|---|
| 466 | dprintf(("MIXER_GETCONTROLDETAILSF_VALUE %d (internal id %d)", lpmcd->dwControlID, mixerControls[lpmcd->dwControlID].id)); | 
|---|
| 467 | if(lpmcd->dwControlID >= nrControls) { | 
|---|
| 468 | dprintf(("invalid control %d", lpmcd->dwControlID)); | 
|---|
| 469 | return MIXERR_INVALCONTROL; | 
|---|
| 470 | } | 
|---|
| 471 | if(lpmcd->cbDetails < sizeof(MIXERCONTROLDETAILS_UNSIGNED)) { | 
|---|
| 472 | dprintf(("not enough room in buffer (%d)", lpmcd->cbDetails)); | 
|---|
| 473 | return MMSYSERR_INVALPARAM; | 
|---|
| 474 | } | 
|---|
| 475 | switch(mixerControls[lpmcd->dwControlID].ctrl.dwControlType) { | 
|---|
| 476 | case MIXERCONTROL_CONTROLTYPE_MUX: | 
|---|
| 477 | { | 
|---|
| 478 | MIXERCONTROLDETAILS_BOOLEAN *pDetails = (MIXERCONTROLDETAILS_BOOLEAN *)lpmcd->paDetails; | 
|---|
| 479 | DWORD dwRecSrc; | 
|---|
| 480 |  | 
|---|
| 481 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUX")); | 
|---|
| 482 | if(lpmcd->cChannels != 1) {//only accepts 1 in win2k | 
|---|
| 483 | dprintf(("invalid number of channels %d", lpmcd->cChannels)); | 
|---|
| 484 | return MMSYSERR_INVALPARAM; | 
|---|
| 485 | } | 
|---|
| 486 | //this is also checked in win2k | 
|---|
| 487 | if(lpmcd->u.cMultipleItems != mixerControls[lpmcd->dwControlID].ctrl.cMultipleItems) { | 
|---|
| 488 | dprintf(("invalid cMultipleItems %d", lpmcd->u.cMultipleItems)); | 
|---|
| 489 | return MMSYSERR_INVALPARAM; | 
|---|
| 490 | } | 
|---|
| 491 |  | 
|---|
| 492 | if(mixerControls[lpmcd->dwControlID].id != MIX_CTRL_MUX_IN_W_SRC) { | 
|---|
| 493 | dprintf(("oh, oh. not wavein mux")); | 
|---|
| 494 | DebugInt3(); | 
|---|
| 495 | return MMSYSERR_INVALPARAM; | 
|---|
| 496 | } | 
|---|
| 497 | //select recording source | 
|---|
| 498 | for(int i=0;i<nrDestinations;i++) { | 
|---|
| 499 | if(mixerDest[i].id == MIXER_DEST_WAVEIN) | 
|---|
| 500 | { | 
|---|
| 501 | for(int j=0;j<mixerDest[i].cConnections;j++) { | 
|---|
| 502 | dprintf(("wavein source %s %d (id %d)", pmixerLines[mixerDest[i].Connections[j]]->line.szName, pmixerLines[mixerDest[i].Connections[j]]->id, mixerDest[i].Connections[j])); | 
|---|
| 503 | if(pDetails[j].fValue) | 
|---|
| 504 | { | 
|---|
| 505 | if(OSLibMixSetRecSource(pmixerLines[mixerDest[i].Connections[j]]->id) == FALSE) { | 
|---|
| 506 | dprintf(("OSLibMixGetRecSource failed!!")); | 
|---|
| 507 | return MIXERR_INVALCONTROL; | 
|---|
| 508 | } | 
|---|
| 509 | goto success; | 
|---|
| 510 | } | 
|---|
| 511 | } | 
|---|
| 512 | } | 
|---|
| 513 | } | 
|---|
| 514 | dprintf(("recording source %d not found!!", dwRecSrc)); | 
|---|
| 515 | break; | 
|---|
| 516 | } | 
|---|
| 517 |  | 
|---|
| 518 | case MIXERCONTROL_CONTROLTYPE_VOLUME: //unsigned | 
|---|
| 519 | { | 
|---|
| 520 | MIXERCONTROLDETAILS_UNSIGNED *pDetails = (MIXERCONTROLDETAILS_UNSIGNED *)lpmcd->paDetails; | 
|---|
| 521 | DWORD dwVolumeL, dwVolumeR; | 
|---|
| 522 |  | 
|---|
| 523 | dprintf(("MIXERCONTROL_CONTROLTYPE_VOLUME")); | 
|---|
| 524 | dwVolumeL = pDetails->dwValue; | 
|---|
| 525 |  | 
|---|
| 526 | dprintf(("%s Left volume %d", mixerControls[lpmcd->dwControlID].ctrl.szName, pDetails->dwValue)); | 
|---|
| 527 | if(lpmcd->cChannels == 2) { | 
|---|
| 528 | pDetails += 1; | 
|---|
| 529 | dprintf(("%s Right volume %d", mixerControls[lpmcd->dwControlID].ctrl.szName, pDetails->dwValue)); | 
|---|
| 530 | dwVolumeR = pDetails->dwValue; | 
|---|
| 531 | } | 
|---|
| 532 | else dwVolumeR = dwVolumeL; | 
|---|
| 533 | //todo > 2 channels | 
|---|
| 534 |  | 
|---|
| 535 | if(OSLibMixSetVolume(mixerControls[lpmcd->dwControlID].id, dwVolumeL, dwVolumeR) == FALSE) { | 
|---|
| 536 | return MMSYSERR_INVALPARAM; | 
|---|
| 537 | } | 
|---|
| 538 | mixerControls[lpmcd->dwControlID].val[0].dwValue = dwVolumeL; | 
|---|
| 539 | mixerControls[lpmcd->dwControlID].val[1].dwValue = dwVolumeR; | 
|---|
| 540 | goto success; | 
|---|
| 541 | } | 
|---|
| 542 |  | 
|---|
| 543 | case MIXERCONTROL_CONTROLTYPE_MUTE: //assuming boolean | 
|---|
| 544 | { | 
|---|
| 545 | MIXERCONTROLDETAILS_BOOLEAN *pDetails = (MIXERCONTROLDETAILS_BOOLEAN *)lpmcd->paDetails; | 
|---|
| 546 | BOOL fMute; | 
|---|
| 547 |  | 
|---|
| 548 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUTE")); | 
|---|
| 549 | dprintf(("%s Left mute %d", mixerControls[lpmcd->dwControlID].ctrl.szName, pDetails->fValue)); | 
|---|
| 550 | fMute = pDetails->fValue; | 
|---|
| 551 | if(lpmcd->cChannels == 2) { | 
|---|
| 552 | pDetails += 1; | 
|---|
| 553 | dprintf(("%s Right mute %d", mixerControls[lpmcd->dwControlID].ctrl.szName, pDetails->fValue)); | 
|---|
| 554 | } | 
|---|
| 555 | //todo > 2 channels (usually only 1 channel is requested though) | 
|---|
| 556 |  | 
|---|
| 557 | if(OSLibMixSetMute(mixerControls[lpmcd->dwControlID].id, fMute) == FALSE) { | 
|---|
| 558 | return MMSYSERR_INVALPARAM; | 
|---|
| 559 | } | 
|---|
| 560 | mixerControls[lpmcd->dwControlID].val[0].dwValue = fMute; | 
|---|
| 561 | goto success; | 
|---|
| 562 | } | 
|---|
| 563 |  | 
|---|
| 564 | case MIXERCONTROL_CONTROLTYPE_BASS: | 
|---|
| 565 | case MIXERCONTROL_CONTROLTYPE_TREBLE: | 
|---|
| 566 | case MIXERCONTROL_CONTROLTYPE_FADER: | 
|---|
| 567 | { | 
|---|
| 568 | MIXERCONTROLDETAILS_UNSIGNED *pDetails = (MIXERCONTROLDETAILS_UNSIGNED *)lpmcd->paDetails; | 
|---|
| 569 | DWORD dwLevel; | 
|---|
| 570 |  | 
|---|
| 571 | #ifdef DEBUG | 
|---|
| 572 | switch(fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK) { | 
|---|
| 573 | case MIXERCONTROL_CONTROLTYPE_TREBLE: | 
|---|
| 574 | dprintf(("MIXERCONTROL_CONTROLTYPE_TREBLE")); | 
|---|
| 575 | break; | 
|---|
| 576 | case MIXERCONTROL_CONTROLTYPE_BASS: | 
|---|
| 577 | dprintf(("MIXERCONTROL_CONTROLTYPE_BASS")); | 
|---|
| 578 | break; | 
|---|
| 579 | case MIXERCONTROL_CONTROLTYPE_FADER: | 
|---|
| 580 | dprintf(("MIXERCONTROL_CONTROLTYPE_FADER")); | 
|---|
| 581 | break; | 
|---|
| 582 | } | 
|---|
| 583 | #endif | 
|---|
| 584 | dprintf(("%s %d", mixerControls[lpmcd->dwControlID].ctrl.szName, pDetails->dwValue)); | 
|---|
| 585 | dwLevel = pDetails->dwValue; | 
|---|
| 586 | if(OSLibMixSetVolume(mixerControls[lpmcd->dwControlID].id, dwLevel, dwLevel) == FALSE) { | 
|---|
| 587 | return MMSYSERR_INVALPARAM; | 
|---|
| 588 | } | 
|---|
| 589 | mixerControls[lpmcd->dwControlID].val[0].dwValue = dwLevel; | 
|---|
| 590 | goto success; | 
|---|
| 591 | } | 
|---|
| 592 |  | 
|---|
| 593 | #ifdef DEBUG | 
|---|
| 594 | case MIXERCONTROL_CONTROLTYPE_MIXER: | 
|---|
| 595 | dprintf(("MIXERCONTROL_CONTROLTYPE_MIXER")); | 
|---|
| 596 | break; | 
|---|
| 597 | case MIXERCONTROL_CONTROLTYPE_CUSTOM: | 
|---|
| 598 | dprintf(("MIXERCONTROL_CONTROLTYPE_CUSTOM")); | 
|---|
| 599 | break; | 
|---|
| 600 | case MIXERCONTROL_CONTROLTYPE_BOOLEANMETER: | 
|---|
| 601 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEANMETER")); | 
|---|
| 602 | break; | 
|---|
| 603 | case MIXERCONTROL_CONTROLTYPE_SIGNEDMETER: | 
|---|
| 604 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNEDMETER")); | 
|---|
| 605 | break; | 
|---|
| 606 | case MIXERCONTROL_CONTROLTYPE_PEAKMETER: | 
|---|
| 607 | dprintf(("MIXERCONTROL_CONTROLTYPE_PEAKMETER")); | 
|---|
| 608 | break; | 
|---|
| 609 | case MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER: | 
|---|
| 610 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER")); | 
|---|
| 611 | break; | 
|---|
| 612 | case MIXERCONTROL_CONTROLTYPE_BOOLEAN: | 
|---|
| 613 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEAN")); | 
|---|
| 614 | break; | 
|---|
| 615 | case MIXERCONTROL_CONTROLTYPE_ONOFF: | 
|---|
| 616 | dprintf(("MIXERCONTROL_CONTROLTYPE_ONOFF")); | 
|---|
| 617 | break; | 
|---|
| 618 | case MIXERCONTROL_CONTROLTYPE_MONO: | 
|---|
| 619 | dprintf(("MIXERCONTROL_CONTROLTYPE_MONO")); | 
|---|
| 620 | break; | 
|---|
| 621 | case MIXERCONTROL_CONTROLTYPE_LOUDNESS: | 
|---|
| 622 | dprintf(("MIXERCONTROL_CONTROLTYPE_LOUDNESS")); | 
|---|
| 623 | break; | 
|---|
| 624 | case MIXERCONTROL_CONTROLTYPE_STEREOENH: | 
|---|
| 625 | dprintf(("MIXERCONTROL_CONTROLTYPE_STEREOENH")); | 
|---|
| 626 | break; | 
|---|
| 627 | case MIXERCONTROL_CONTROLTYPE_BUTTON: | 
|---|
| 628 | dprintf(("MIXERCONTROL_CONTROLTYPE_BUTTON")); | 
|---|
| 629 | break; | 
|---|
| 630 | case MIXERCONTROL_CONTROLTYPE_DECIBELS: | 
|---|
| 631 | dprintf(("MIXERCONTROL_CONTROLTYPE_DECIBELS")); | 
|---|
| 632 | break; | 
|---|
| 633 | case MIXERCONTROL_CONTROLTYPE_SIGNED: | 
|---|
| 634 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNED")); | 
|---|
| 635 | break; | 
|---|
| 636 | case MIXERCONTROL_CONTROLTYPE_UNSIGNED: | 
|---|
| 637 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNED")); | 
|---|
| 638 | break; | 
|---|
| 639 | case MIXERCONTROL_CONTROLTYPE_PERCENT: | 
|---|
| 640 | dprintf(("MIXERCONTROL_CONTROLTYPE_PERCENT")); | 
|---|
| 641 | break; | 
|---|
| 642 | case MIXERCONTROL_CONTROLTYPE_SLIDER: | 
|---|
| 643 | dprintf(("MIXERCONTROL_CONTROLTYPE_SLIDER")); | 
|---|
| 644 | break; | 
|---|
| 645 | case MIXERCONTROL_CONTROLTYPE_PAN: | 
|---|
| 646 | dprintf(("MIXERCONTROL_CONTROLTYPE_PAN")); | 
|---|
| 647 | break; | 
|---|
| 648 | case MIXERCONTROL_CONTROLTYPE_QSOUNDPAN: | 
|---|
| 649 | dprintf(("MIXERCONTROL_CONTROLTYPE_QSOUNDPAN")); | 
|---|
| 650 | break; | 
|---|
| 651 | case MIXERCONTROL_CONTROLTYPE_EQUALIZER: | 
|---|
| 652 | dprintf(("MIXERCONTROL_CONTROLTYPE_EQUALIZER")); | 
|---|
| 653 | break; | 
|---|
| 654 | case MIXERCONTROL_CONTROLTYPE_SINGLESELECT: | 
|---|
| 655 | dprintf(("MIXERCONTROL_CONTROLTYPE_SINGLESELECT")); | 
|---|
| 656 | break; | 
|---|
| 657 | case MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT: | 
|---|
| 658 | dprintf(("MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT")); | 
|---|
| 659 | break; | 
|---|
| 660 | case MIXERCONTROL_CONTROLTYPE_MICROTIME: | 
|---|
| 661 | dprintf(("MIXERCONTROL_CONTROLTYPE_MICROTIME")); | 
|---|
| 662 | break; | 
|---|
| 663 | case MIXERCONTROL_CONTROLTYPE_MILLITIME: | 
|---|
| 664 | dprintf(("MIXERCONTROL_CONTROLTYPE_MILLITIME")); | 
|---|
| 665 | break; | 
|---|
| 666 | #endif | 
|---|
| 667 | default: | 
|---|
| 668 | DebugInt3(); | 
|---|
| 669 | return MIXERR_INVALCONTROL; | 
|---|
| 670 | } | 
|---|
| 671 | return MIXERR_INVALCONTROL; | 
|---|
| 672 | case MIXER_GETCONTROLDETAILSF_LISTTEXT: | 
|---|
| 673 | dprintf(("MIXER_GETCONTROLDETAILSF_LISTTEXT")); | 
|---|
| 674 | break; | 
|---|
| 675 | default: | 
|---|
| 676 | dprintf(("Unknown flag (%08lx)\n", fdwDetails & MIXER_GETCONTROLDETAILSF_QUERYMASK)); | 
|---|
| 677 | break; | 
|---|
| 678 | } | 
|---|
| 679 | return MMSYSERR_NOTSUPPORTED; | 
|---|
| 680 |  | 
|---|
| 681 | success: | 
|---|
| 682 | if(pMixInfo->dwFlags & CALLBACK_WINDOW && pMixInfo->dwCallback) { | 
|---|
| 683 | dprintf(("Notify window %x of control change", pMixInfo->dwCallback)); | 
|---|
| 684 | PostMessageA((HWND)pMixInfo->dwCallback, MM_MIXM_CONTROL_CHANGE, (WPARAM)hmxobj, (LPARAM)lpmcd->dwControlID); | 
|---|
| 685 | } | 
|---|
| 686 | return MMSYSERR_NOERROR; | 
|---|
| 687 | } | 
|---|
| 688 | /******************************************************************************/ | 
|---|
| 689 | /******************************************************************************/ | 
|---|
| 690 | MMRESULT WINAPI mixerGetLineControlsA(HMIXEROBJ hmxobj, LPMIXERLINECONTROLSA lpMlc, DWORD fdwControls) | 
|---|
| 691 | { | 
|---|
| 692 | DWORD dwRet = MMSYSERR_NOERROR; | 
|---|
| 693 | DWORD lineID, controlType; | 
|---|
| 694 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 695 |  | 
|---|
| 696 | if((fdwControls & 0xF0000000) == MIXER_OBJECTF_HMIXER) { | 
|---|
| 697 | if(!pMixInfo) { | 
|---|
| 698 | return MMSYSERR_INVALHANDLE; | 
|---|
| 699 | } | 
|---|
| 700 | } | 
|---|
| 701 | else | 
|---|
| 702 | if((fdwControls & 0xF0000000) == MIXER_OBJECTF_MIXER) { | 
|---|
| 703 | if(!HIWORD(hmxobj) && hmxobj > 0) { | 
|---|
| 704 | return MMSYSERR_NODRIVER; | 
|---|
| 705 | } | 
|---|
| 706 | } | 
|---|
| 707 |  | 
|---|
| 708 | if (lpMlc == NULL) return MMSYSERR_INVALPARAM; | 
|---|
| 709 |  | 
|---|
| 710 | if (lpMlc->cbStruct < sizeof(*lpMlc) || lpMlc->cbmxctrl < sizeof(MIXERCONTROLA)) | 
|---|
| 711 | return MMSYSERR_INVALPARAM; | 
|---|
| 712 |  | 
|---|
| 713 | switch(fdwControls & MIXER_GETLINECONTROLSF_QUERYMASK) | 
|---|
| 714 | { | 
|---|
| 715 | case MIXER_GETLINECONTROLSF_ALL: | 
|---|
| 716 | { | 
|---|
| 717 | MIXERLINEA    *pLine; | 
|---|
| 718 | MIXERCONTROLA *pCtrl; | 
|---|
| 719 |  | 
|---|
| 720 | dprintf(("MIXER_GETLINECONTROLSF_ALL for line %d", lpMlc->dwLineID)); | 
|---|
| 721 | if(lpMlc->dwLineID >= nrLines) { | 
|---|
| 722 | dprintf(("ERROR: Invalid line %d", lpMlc->dwLineID)); | 
|---|
| 723 | return MIXERR_INVALLINE; | 
|---|
| 724 | } | 
|---|
| 725 | pLine = &pmixerLines[lpMlc->dwLineID]->line; | 
|---|
| 726 | if(lpMlc->cControls != pLine->cControls) { | 
|---|
| 727 | dprintf(("ERROR: invalid nr of controls %d or structure size %d (%d) (ptr %x)", lpMlc->cControls, lpMlc->cbmxctrl, pLine->cControls*sizeof(MIXERCONTROLA), lpMlc->pamxctrl)); | 
|---|
| 728 | return MMSYSERR_INVALPARAM; | 
|---|
| 729 | } | 
|---|
| 730 | for(int i=0;i<lpMlc->cControls;i++) { | 
|---|
| 731 | pCtrl = lpMlc->pamxctrl+i; | 
|---|
| 732 | memcpy(pCtrl, &mixerControls[pmixerLines[lpMlc->dwLineID]->Controls[i]].ctrl, sizeof(MIXERCONTROLA)); | 
|---|
| 733 | dprintf(("found control %s (%s) control id %d", pCtrl->szName, pCtrl->szShortName, pCtrl->dwControlID)); | 
|---|
| 734 | } | 
|---|
| 735 | break; | 
|---|
| 736 | } | 
|---|
| 737 |  | 
|---|
| 738 | case MIXER_GETLINECONTROLSF_ONEBYID: | 
|---|
| 739 | { | 
|---|
| 740 | dprintf(("MIXER_GETLINECONTROLSF_ONEBYID %x", lpMlc->u.dwControlID)); | 
|---|
| 741 | if(lpMlc->cControls != 1 || lpMlc->cbmxctrl != sizeof(MIXERCONTROLA)) { | 
|---|
| 742 | dprintf(("invalid parameters")); | 
|---|
| 743 | return MMSYSERR_INVALPARAM; | 
|---|
| 744 | } | 
|---|
| 745 | if(lpMlc->u.dwControlID >= nrControls) { | 
|---|
| 746 | dprintf(("invalid control")); | 
|---|
| 747 | return MIXERR_INVALCONTROL; | 
|---|
| 748 | } | 
|---|
| 749 | //find line associated with this control | 
|---|
| 750 | lpMlc->dwLineID = -1; | 
|---|
| 751 | for(int i=0;i<nrLines;i++) { | 
|---|
| 752 | for(int j=0;j<pmixerLines[i]->cControls;j++) { | 
|---|
| 753 | if(pmixerLines[i]->Controls[j] == lpMlc->u.dwControlID) { | 
|---|
| 754 | lpMlc->dwLineID = i; | 
|---|
| 755 | break; | 
|---|
| 756 | } | 
|---|
| 757 | else | 
|---|
| 758 | if(pmixerLines[i]->Controls[j] == -1) { | 
|---|
| 759 | break; | 
|---|
| 760 | } | 
|---|
| 761 | } | 
|---|
| 762 | if(lpMlc->dwLineID != -1) { | 
|---|
| 763 | break; | 
|---|
| 764 | } | 
|---|
| 765 | } | 
|---|
| 766 | if(i == MAX_MIXER_LINES) { | 
|---|
| 767 | dprintf(("Associated line for control %d NOT FOUND", lpMlc->u.dwControlID)); | 
|---|
| 768 | return MIXERR_INVALCONTROL; | 
|---|
| 769 | } | 
|---|
| 770 | memcpy(lpMlc->pamxctrl, &mixerControls[lpMlc->u.dwControlID].ctrl, sizeof(MIXERCONTROLA)); | 
|---|
| 771 |  | 
|---|
| 772 | dprintf(("found control %s (%s) associated line %d", lpMlc->pamxctrl->szName, lpMlc->pamxctrl->szShortName, lpMlc->dwLineID)); | 
|---|
| 773 | break; | 
|---|
| 774 | } | 
|---|
| 775 |  | 
|---|
| 776 | case MIXER_GETLINECONTROLSF_ONEBYTYPE: | 
|---|
| 777 | { | 
|---|
| 778 | dprintf(("MIXER_GETLINECONTROLSF_ONEBYTYPE %x %d", lpMlc->u.dwControlType, lpMlc->dwLineID)); | 
|---|
| 779 | if(lpMlc->dwLineID >= nrLines) { | 
|---|
| 780 | dprintf(("ERROR: Invalid line %d", lpMlc->dwLineID)); | 
|---|
| 781 | return MIXERR_INVALLINE; | 
|---|
| 782 | } | 
|---|
| 783 | #ifdef DEBUG | 
|---|
| 784 | switch (lpMlc->u.dwControlType) { | 
|---|
| 785 | case MIXERCONTROL_CONTROLTYPE_CUSTOM: | 
|---|
| 786 | dprintf(("MIXERCONTROL_CONTROLTYPE_CUSTOM")); | 
|---|
| 787 | break; | 
|---|
| 788 | case MIXERCONTROL_CONTROLTYPE_BOOLEANMETER: | 
|---|
| 789 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEANMETER")); | 
|---|
| 790 | break; | 
|---|
| 791 | case MIXERCONTROL_CONTROLTYPE_SIGNEDMETER: | 
|---|
| 792 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNEDMETER")); | 
|---|
| 793 | break; | 
|---|
| 794 | case MIXERCONTROL_CONTROLTYPE_PEAKMETER: | 
|---|
| 795 | dprintf(("MIXERCONTROL_CONTROLTYPE_PEAKMETER")); | 
|---|
| 796 | break; | 
|---|
| 797 | case MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER: | 
|---|
| 798 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNEDMETER")); | 
|---|
| 799 | break; | 
|---|
| 800 | case MIXERCONTROL_CONTROLTYPE_BOOLEAN: | 
|---|
| 801 | dprintf(("MIXERCONTROL_CONTROLTYPE_BOOLEAN")); | 
|---|
| 802 | break; | 
|---|
| 803 | case MIXERCONTROL_CONTROLTYPE_ONOFF: | 
|---|
| 804 | dprintf(("MIXERCONTROL_CONTROLTYPE_ONOFF")); | 
|---|
| 805 | break; | 
|---|
| 806 | case MIXERCONTROL_CONTROLTYPE_MUTE: | 
|---|
| 807 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUTE")); | 
|---|
| 808 | break; | 
|---|
| 809 | case MIXERCONTROL_CONTROLTYPE_MONO: | 
|---|
| 810 | dprintf(("MIXERCONTROL_CONTROLTYPE_MONO")); | 
|---|
| 811 | break; | 
|---|
| 812 | case MIXERCONTROL_CONTROLTYPE_LOUDNESS: | 
|---|
| 813 | dprintf(("MIXERCONTROL_CONTROLTYPE_LOUDNESS")); | 
|---|
| 814 | break; | 
|---|
| 815 | case MIXERCONTROL_CONTROLTYPE_STEREOENH: | 
|---|
| 816 | dprintf(("MIXERCONTROL_CONTROLTYPE_STEREOENH")); | 
|---|
| 817 | break; | 
|---|
| 818 | case MIXERCONTROL_CONTROLTYPE_BUTTON: | 
|---|
| 819 | dprintf(("MIXERCONTROL_CONTROLTYPE_BUTTON")); | 
|---|
| 820 | break; | 
|---|
| 821 | case MIXERCONTROL_CONTROLTYPE_DECIBELS: | 
|---|
| 822 | dprintf(("MIXERCONTROL_CONTROLTYPE_DECIBELS")); | 
|---|
| 823 | break; | 
|---|
| 824 | case MIXERCONTROL_CONTROLTYPE_SIGNED: | 
|---|
| 825 | dprintf(("MIXERCONTROL_CONTROLTYPE_SIGNED")); | 
|---|
| 826 | break; | 
|---|
| 827 | case MIXERCONTROL_CONTROLTYPE_UNSIGNED: | 
|---|
| 828 | dprintf(("MIXERCONTROL_CONTROLTYPE_UNSIGNED")); | 
|---|
| 829 | break; | 
|---|
| 830 | case MIXERCONTROL_CONTROLTYPE_PERCENT: | 
|---|
| 831 | dprintf(("MIXERCONTROL_CONTROLTYPE_PERCENT")); | 
|---|
| 832 | break; | 
|---|
| 833 | case MIXERCONTROL_CONTROLTYPE_SLIDER: | 
|---|
| 834 | dprintf(("MIXERCONTROL_CONTROLTYPE_SLIDER")); | 
|---|
| 835 | break; | 
|---|
| 836 | case MIXERCONTROL_CONTROLTYPE_PAN: | 
|---|
| 837 | dprintf(("MIXERCONTROL_CONTROLTYPE_PAN")); | 
|---|
| 838 | break; | 
|---|
| 839 | case MIXERCONTROL_CONTROLTYPE_QSOUNDPAN: | 
|---|
| 840 | dprintf(("MIXERCONTROL_CONTROLTYPE_QSOUNDPAN")); | 
|---|
| 841 | break; | 
|---|
| 842 | case MIXERCONTROL_CONTROLTYPE_FADER: | 
|---|
| 843 | dprintf(("MIXERCONTROL_CONTROLTYPE_FADER")); | 
|---|
| 844 | break; | 
|---|
| 845 | case MIXERCONTROL_CONTROLTYPE_VOLUME: | 
|---|
| 846 | dprintf(("MIXERCONTROL_CONTROLTYPE_VOLUME")); | 
|---|
| 847 | break; | 
|---|
| 848 | case MIXERCONTROL_CONTROLTYPE_BASS: | 
|---|
| 849 | dprintf(("MIXERCONTROL_CONTROLTYPE_BASS")); | 
|---|
| 850 | break; | 
|---|
| 851 | case MIXERCONTROL_CONTROLTYPE_TREBLE: | 
|---|
| 852 | dprintf(("MIXERCONTROL_CONTROLTYPE_TREBLE")); | 
|---|
| 853 | break; | 
|---|
| 854 | case MIXERCONTROL_CONTROLTYPE_EQUALIZER: | 
|---|
| 855 | dprintf(("MIXERCONTROL_CONTROLTYPE_EQUALIZER")); | 
|---|
| 856 | break; | 
|---|
| 857 | case MIXERCONTROL_CONTROLTYPE_SINGLESELECT: | 
|---|
| 858 | dprintf(("MIXERCONTROL_CONTROLTYPE_SINGLESELECT")); | 
|---|
| 859 | break; | 
|---|
| 860 | case MIXERCONTROL_CONTROLTYPE_MUX: | 
|---|
| 861 | dprintf(("MIXERCONTROL_CONTROLTYPE_MUX")); | 
|---|
| 862 | break; | 
|---|
| 863 | case MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT: | 
|---|
| 864 | dprintf(("MIXERCONTROL_CONTROLTYPE_MULTIPLESELECT")); | 
|---|
| 865 | break; | 
|---|
| 866 | case MIXERCONTROL_CONTROLTYPE_MIXER: | 
|---|
| 867 | dprintf(("MIXERCONTROL_CONTROLTYPE_MIXER")); | 
|---|
| 868 | break; | 
|---|
| 869 | case MIXERCONTROL_CONTROLTYPE_MICROTIME: | 
|---|
| 870 | dprintf(("MIXERCONTROL_CONTROLTYPE_MICROTIME")); | 
|---|
| 871 | break; | 
|---|
| 872 | case MIXERCONTROL_CONTROLTYPE_MILLITIME: | 
|---|
| 873 | dprintf(("MIXERCONTROL_CONTROLTYPE_MILLITIME")); | 
|---|
| 874 | break; | 
|---|
| 875 | default: | 
|---|
| 876 | return MMSYSERR_INVALPARAM; | 
|---|
| 877 | } | 
|---|
| 878 | #endif | 
|---|
| 879 | int idx; | 
|---|
| 880 |  | 
|---|
| 881 | for(int i=0;i<pmixerLines[lpMlc->dwLineID]->cControls;i++) { | 
|---|
| 882 | idx = pmixerLines[lpMlc->dwLineID]->Controls[i]; | 
|---|
| 883 | if(mixerControls[idx].ctrl.dwControlType == lpMlc->u.dwControlType) { | 
|---|
| 884 | memcpy(lpMlc->pamxctrl, &mixerControls[idx].ctrl, sizeof(MIXERCONTROLA)); | 
|---|
| 885 | dprintf(("found control %s (%s) control id %d", lpMlc->pamxctrl->szName, lpMlc->pamxctrl->szShortName, lpMlc->pamxctrl->dwControlID)); | 
|---|
| 886 | return MMSYSERR_NOERROR; | 
|---|
| 887 | } | 
|---|
| 888 |  | 
|---|
| 889 | } | 
|---|
| 890 | return MIXERR_INVALLINE;    //returned by win2k when type not found | 
|---|
| 891 | } | 
|---|
| 892 | default: | 
|---|
| 893 | dprintf(("Unknown flag %08lx\n", fdwControls & MIXER_GETLINECONTROLSF_QUERYMASK)); | 
|---|
| 894 | dwRet = MMSYSERR_INVALPARAM; | 
|---|
| 895 | break; | 
|---|
| 896 | } | 
|---|
| 897 |  | 
|---|
| 898 | return dwRet; | 
|---|
| 899 | } | 
|---|
| 900 | /******************************************************************************/ | 
|---|
| 901 | /******************************************************************************/ | 
|---|
| 902 | MMRESULT WINAPI mixerGetLineControlsW(HMIXEROBJ hmxobj, LPMIXERLINECONTROLSW lpmlcW, DWORD fdwControls) | 
|---|
| 903 | { | 
|---|
| 904 | MIXERLINECONTROLSA  mlcA; | 
|---|
| 905 | DWORD               ret; | 
|---|
| 906 | int                 i; | 
|---|
| 907 |  | 
|---|
| 908 | if (lpmlcW == NULL || lpmlcW->cbStruct != sizeof(*lpmlcW) || lpmlcW->cbmxctrl != sizeof(MIXERCONTROLW)) | 
|---|
| 909 | return MMSYSERR_INVALPARAM; | 
|---|
| 910 |  | 
|---|
| 911 | mlcA.cbStruct = sizeof(mlcA); | 
|---|
| 912 | mlcA.dwLineID = lpmlcW->dwLineID; | 
|---|
| 913 | mlcA.u.dwControlID = lpmlcW->u.dwControlID; | 
|---|
| 914 | mlcA.u.dwControlType = lpmlcW->u.dwControlType; | 
|---|
| 915 | mlcA.cControls = lpmlcW->cControls; | 
|---|
| 916 | mlcA.cbmxctrl = sizeof(MIXERCONTROLA); | 
|---|
| 917 | mlcA.pamxctrl = (MIXERCONTROLA *)HeapAlloc(GetProcessHeap(), 0, mlcA.cControls * mlcA.cbmxctrl); | 
|---|
| 918 |  | 
|---|
| 919 | ret = mixerGetLineControlsA(hmxobj, &mlcA, fdwControls); | 
|---|
| 920 |  | 
|---|
| 921 | if (ret == MMSYSERR_NOERROR) { | 
|---|
| 922 | lpmlcW->dwLineID = mlcA.dwLineID; | 
|---|
| 923 | lpmlcW->u.dwControlID = mlcA.u.dwControlID; | 
|---|
| 924 | lpmlcW->u.dwControlType = mlcA.u.dwControlType; | 
|---|
| 925 | lpmlcW->cControls = mlcA.cControls; | 
|---|
| 926 |  | 
|---|
| 927 | for (i = 0; i < mlcA.cControls; i++) { | 
|---|
| 928 | lpmlcW->pamxctrl[i].cbStruct = sizeof(MIXERCONTROLW); | 
|---|
| 929 | lpmlcW->pamxctrl[i].dwControlID = mlcA.pamxctrl[i].dwControlID; | 
|---|
| 930 | lpmlcW->pamxctrl[i].dwControlType = mlcA.pamxctrl[i].dwControlType; | 
|---|
| 931 | lpmlcW->pamxctrl[i].fdwControl = mlcA.pamxctrl[i].fdwControl; | 
|---|
| 932 | lpmlcW->pamxctrl[i].cMultipleItems = mlcA.pamxctrl[i].cMultipleItems; | 
|---|
| 933 | MultiByteToWideChar( CP_ACP, 0, mlcA.pamxctrl[i].szShortName, -1, | 
|---|
| 934 | lpmlcW->pamxctrl[i].szShortName, | 
|---|
| 935 | sizeof(lpmlcW->pamxctrl[i].szShortName)/sizeof(WCHAR) ); | 
|---|
| 936 | MultiByteToWideChar( CP_ACP, 0, mlcA.pamxctrl[i].szName, -1, | 
|---|
| 937 | lpmlcW->pamxctrl[i].szName, | 
|---|
| 938 | sizeof(lpmlcW->pamxctrl[i].szName)/sizeof(WCHAR) ); | 
|---|
| 939 | /* sizeof(lpmlcW->pamxctrl[i].Bounds) == | 
|---|
| 940 | * sizeof(mlcA.pamxctrl[i].Bounds) */ | 
|---|
| 941 | memcpy(&lpmlcW->pamxctrl[i].Bounds, &mlcA.pamxctrl[i].Bounds, | 
|---|
| 942 | sizeof(mlcA.pamxctrl[i].Bounds)); | 
|---|
| 943 | /* sizeof(lpmlcW->pamxctrl[i].Metrics) == | 
|---|
| 944 | * sizeof(mlcA.pamxctrl[i].Metrics) */ | 
|---|
| 945 | memcpy(&lpmlcW->pamxctrl[i].Metrics, &mlcA.pamxctrl[i].Metrics, | 
|---|
| 946 | sizeof(mlcA.pamxctrl[i].Metrics)); | 
|---|
| 947 | } | 
|---|
| 948 | } | 
|---|
| 949 |  | 
|---|
| 950 | HeapFree(GetProcessHeap(), 0, mlcA.pamxctrl); | 
|---|
| 951 | return ret; | 
|---|
| 952 | } | 
|---|
| 953 | /******************************************************************************/ | 
|---|
| 954 | /******************************************************************************/ | 
|---|
| 955 | MMRESULT WINAPI mixerGetLineInfoA(HMIXEROBJ hmxobj, LPMIXERLINEA lpMl, DWORD fdwInfo) | 
|---|
| 956 | { | 
|---|
| 957 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 958 |  | 
|---|
| 959 | if((fdwInfo & 0xF0000000) == MIXER_OBJECTF_HMIXER) { | 
|---|
| 960 | if(!pMixInfo) { | 
|---|
| 961 | return MMSYSERR_INVALHANDLE; | 
|---|
| 962 | } | 
|---|
| 963 | } | 
|---|
| 964 | else | 
|---|
| 965 | if((fdwInfo & 0xF0000000) == MIXER_OBJECTF_MIXER) { | 
|---|
| 966 | if(!HIWORD(hmxobj) && hmxobj > 0) { | 
|---|
| 967 | return MMSYSERR_NODRIVER; | 
|---|
| 968 | } | 
|---|
| 969 | } | 
|---|
| 970 |  | 
|---|
| 971 | if (lpMl == NULL || lpMl->cbStruct != sizeof(*lpMl)) { | 
|---|
| 972 | dprintf(("ERROR: mixerGetLineInfoA: invalid paramter!!")); | 
|---|
| 973 | return MMSYSERR_INVALPARAM; | 
|---|
| 974 | } | 
|---|
| 975 |  | 
|---|
| 976 | switch (fdwInfo & MIXER_GETLINEINFOF_QUERYMASK) | 
|---|
| 977 | { | 
|---|
| 978 | case MIXER_GETLINEINFOF_DESTINATION: | 
|---|
| 979 | dprintf(("MIXER_GETLINEINFOF_DESTINATION %d", lpMl->dwDestination)); | 
|---|
| 980 | if(lpMl->dwDestination >= nrDestinations) { | 
|---|
| 981 | dprintf(("ERROR: Invalid destination %d", lpMl->dwDestination)); | 
|---|
| 982 | return MMSYSERR_INVALPARAM; | 
|---|
| 983 | } | 
|---|
| 984 | memcpy(lpMl, &mixerDest[lpMl->dwDestination].line, sizeof(MIXERLINEA)); | 
|---|
| 985 | dprintf(("found line %s (%s) id %d connections %d controls %d", lpMl->szName, lpMl->szShortName, lpMl->dwLineID, lpMl->cConnections, lpMl->cControls)); | 
|---|
| 986 | break; | 
|---|
| 987 |  | 
|---|
| 988 | case MIXER_GETLINEINFOF_LINEID: | 
|---|
| 989 | dprintf(("MIXER_GETLINEINFOF_LINEID %d", lpMl->dwLineID)); | 
|---|
| 990 | if(lpMl->dwLineID >= nrLines) { | 
|---|
| 991 | dprintf(("ERROR: Invalid line %d", lpMl->dwLineID)); | 
|---|
| 992 | return MIXERR_INVALLINE; | 
|---|
| 993 | } | 
|---|
| 994 | memcpy(lpMl, &pmixerLines[lpMl->dwLineID]->line, sizeof(MIXERLINEA)); | 
|---|
| 995 | dprintf(("found line %s (%s) connections %d controls %d", lpMl->szName, lpMl->szShortName, lpMl->cConnections, lpMl->cControls)); | 
|---|
| 996 | break; | 
|---|
| 997 |  | 
|---|
| 998 | case MIXER_GETLINEINFOF_SOURCE: | 
|---|
| 999 | dprintf(("MIXER_GETLINEINFOF_SOURCE %d %d", lpMl->dwDestination, lpMl->dwSource)); | 
|---|
| 1000 | if(lpMl->dwDestination >= nrDestinations) { | 
|---|
| 1001 | dprintf(("ERROR: Invalid destination %d", lpMl->dwDestination)); | 
|---|
| 1002 | return MIXERR_INVALLINE; | 
|---|
| 1003 | } | 
|---|
| 1004 | if(lpMl->dwSource >= MIXER_SRC_MAX) { | 
|---|
| 1005 | dprintf(("ERROR: Invalid source %d", lpMl->dwSource)); | 
|---|
| 1006 | return MIXERR_INVALLINE; | 
|---|
| 1007 | } | 
|---|
| 1008 | if(mixerDest[lpMl->dwDestination].Connections[lpMl->dwSource] == -1) { | 
|---|
| 1009 | dprintf(("ERROR: Invalid destination/source combo (%d,%d)", lpMl->dwDestination, lpMl->dwSource)); | 
|---|
| 1010 | return MIXERR_INVALLINE; | 
|---|
| 1011 | } | 
|---|
| 1012 | memcpy(lpMl, &pmixerLines[mixerDest[lpMl->dwDestination].Connections[lpMl->dwSource]]->line, sizeof(MIXERLINEA)); | 
|---|
| 1013 | dprintf(("found line %s (%s) id %d connections %d controls %d", lpMl->szName, lpMl->szShortName, lpMl->dwLineID, lpMl->cConnections, lpMl->cControls)); | 
|---|
| 1014 | break; | 
|---|
| 1015 |  | 
|---|
| 1016 | case MIXER_GETLINEINFOF_COMPONENTTYPE: | 
|---|
| 1017 | { | 
|---|
| 1018 | dprintf(("MIXER_GETLINEINFOF_COMPONENTTYPE")); | 
|---|
| 1019 | #ifdef DEBUG | 
|---|
| 1020 | switch (lpMl->dwComponentType) | 
|---|
| 1021 | { | 
|---|
| 1022 | case MIXERLINE_COMPONENTTYPE_DST_UNDEFINED: | 
|---|
| 1023 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_UNDEFINED")); | 
|---|
| 1024 | break; | 
|---|
| 1025 | case MIXERLINE_COMPONENTTYPE_DST_DIGITAL: | 
|---|
| 1026 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_DIGITAL")); | 
|---|
| 1027 | break; | 
|---|
| 1028 | case MIXERLINE_COMPONENTTYPE_DST_LINE: | 
|---|
| 1029 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_LINE")); | 
|---|
| 1030 | break; | 
|---|
| 1031 | case MIXERLINE_COMPONENTTYPE_DST_MONITOR: | 
|---|
| 1032 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_MONITOR")); | 
|---|
| 1033 | break; | 
|---|
| 1034 | case MIXERLINE_COMPONENTTYPE_DST_SPEAKERS: | 
|---|
| 1035 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_SPEAKERS")); | 
|---|
| 1036 | break; | 
|---|
| 1037 | case MIXERLINE_COMPONENTTYPE_DST_HEADPHONES: | 
|---|
| 1038 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_HEADPHONES")); | 
|---|
| 1039 | break; | 
|---|
| 1040 | case MIXERLINE_COMPONENTTYPE_DST_TELEPHONE: | 
|---|
| 1041 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_TELEPHONE")); | 
|---|
| 1042 | break; | 
|---|
| 1043 | case MIXERLINE_COMPONENTTYPE_DST_WAVEIN: | 
|---|
| 1044 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_WAVEIN")); | 
|---|
| 1045 | break; | 
|---|
| 1046 | case MIXERLINE_COMPONENTTYPE_DST_VOICEIN: | 
|---|
| 1047 | dprintf(("MIXERLINE_COMPONENTTYPE_DST_VOICEIN")); | 
|---|
| 1048 | break; | 
|---|
| 1049 | case MIXERLINE_COMPONENTTYPE_SRC_UNDEFINED: | 
|---|
| 1050 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_UNDEFINED")); | 
|---|
| 1051 | break; | 
|---|
| 1052 | case MIXERLINE_COMPONENTTYPE_SRC_DIGITAL: | 
|---|
| 1053 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_DIGITAL")); | 
|---|
| 1054 | break; | 
|---|
| 1055 | case MIXERLINE_COMPONENTTYPE_SRC_LINE: | 
|---|
| 1056 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_LINE")); | 
|---|
| 1057 | break; | 
|---|
| 1058 | case MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE: | 
|---|
| 1059 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE")); | 
|---|
| 1060 | break; | 
|---|
| 1061 | case MIXERLINE_COMPONENTTYPE_SRC_SYNTHESIZER: | 
|---|
| 1062 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_SYNTHESIZER")); | 
|---|
| 1063 | break; | 
|---|
| 1064 | case MIXERLINE_COMPONENTTYPE_SRC_COMPACTDISC: | 
|---|
| 1065 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_COMPACTDISC")); | 
|---|
| 1066 | break; | 
|---|
| 1067 | case MIXERLINE_COMPONENTTYPE_SRC_TELEPHONE: | 
|---|
| 1068 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_TELEPHONE")); | 
|---|
| 1069 | break; | 
|---|
| 1070 | case MIXERLINE_COMPONENTTYPE_SRC_PCSPEAKER: | 
|---|
| 1071 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_PCSPEAKER")); | 
|---|
| 1072 | break; | 
|---|
| 1073 | case MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT: | 
|---|
| 1074 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT")); | 
|---|
| 1075 | break; | 
|---|
| 1076 | case MIXERLINE_COMPONENTTYPE_SRC_AUXILIARY: | 
|---|
| 1077 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_AUXILIARY")); | 
|---|
| 1078 | break; | 
|---|
| 1079 | case MIXERLINE_COMPONENTTYPE_SRC_ANALOG: | 
|---|
| 1080 | dprintf(("MIXERLINE_COMPONENTTYPE_SRC_ANALOG")); | 
|---|
| 1081 | break; | 
|---|
| 1082 | default: | 
|---|
| 1083 | dprintf(("Unhandled component type (%08lx)\n", lpMl->dwComponentType)); | 
|---|
| 1084 | return MMSYSERR_INVALPARAM; | 
|---|
| 1085 | } | 
|---|
| 1086 | #endif | 
|---|
| 1087 | //search all lines (src & dest) for one with specified type | 
|---|
| 1088 | for(int i=0;i<nrLines;i++) { | 
|---|
| 1089 | if(pmixerLines[i]->line.dwComponentType == lpMl->dwComponentType) { | 
|---|
| 1090 | break; | 
|---|
| 1091 | } | 
|---|
| 1092 | } | 
|---|
| 1093 | if(i == nrLines) { | 
|---|
| 1094 | dprintf(("Component type %d not found!!", lpMl->dwComponentType)); | 
|---|
| 1095 | return MIXERR_INVALLINE; | 
|---|
| 1096 | } | 
|---|
| 1097 | memcpy(lpMl, &pmixerLines[i]->line, sizeof(MIXERLINEA)); | 
|---|
| 1098 | dprintf(("found line %s (%s) id %d connections %d controls %d", lpMl->szName, lpMl->szShortName, lpMl->dwLineID, lpMl->cConnections, lpMl->cControls)); | 
|---|
| 1099 | break; | 
|---|
| 1100 | } | 
|---|
| 1101 | case MIXER_GETLINEINFOF_TARGETTYPE: | 
|---|
| 1102 | { | 
|---|
| 1103 | dprintf(("MIXER_GETLINEINFOF_TARGETTYPE %x", lpMl->Target.dwType)); | 
|---|
| 1104 | #ifdef DEBUG | 
|---|
| 1105 | switch(lpMl->Target.dwType) { | 
|---|
| 1106 | case MIXERLINE_TARGETTYPE_UNDEFINED: | 
|---|
| 1107 | dprintf(("MIXERLINE_TARGETTYPE_UNDEFINED")); | 
|---|
| 1108 | break; | 
|---|
| 1109 | case MIXERLINE_TARGETTYPE_WAVEOUT: | 
|---|
| 1110 | dprintf(("MIXERLINE_TARGETTYPE_WAVEOUT")); | 
|---|
| 1111 | break; | 
|---|
| 1112 | case MIXERLINE_TARGETTYPE_WAVEIN: | 
|---|
| 1113 | dprintf(("MIXERLINE_TARGETTYPE_WAVEIN")); | 
|---|
| 1114 | break; | 
|---|
| 1115 | case MIXERLINE_TARGETTYPE_MIDIOUT: | 
|---|
| 1116 | dprintf(("MIXERLINE_TARGETTYPE_MIDIOUT")); | 
|---|
| 1117 | break; | 
|---|
| 1118 | case MIXERLINE_TARGETTYPE_MIDIIN: | 
|---|
| 1119 | dprintf(("MIXERLINE_TARGETTYPE_MIDIIN")); | 
|---|
| 1120 | break; | 
|---|
| 1121 | case MIXERLINE_TARGETTYPE_AUX: | 
|---|
| 1122 | dprintf(("MIXERLINE_TARGETTYPE_AUX")); | 
|---|
| 1123 | break; | 
|---|
| 1124 | default: | 
|---|
| 1125 | dprintf(("Unhandled target type (%08lx)\n", lpMl->Target.dwType)); | 
|---|
| 1126 | return MMSYSERR_INVALPARAM; | 
|---|
| 1127 | } | 
|---|
| 1128 | #endif | 
|---|
| 1129 | //search all lines (src & dest) for one with specified type | 
|---|
| 1130 | //TODO: Should we compare mid, pid & pddname too? (must be initialized according to MSDN) | 
|---|
| 1131 | //      (see below; unicode version doesn't copy pddname; change if these checks are required) | 
|---|
| 1132 | for(int i=0;i<nrLines;i++) { | 
|---|
| 1133 | if(pmixerLines[i]->line.Target.dwType == lpMl->Target.dwType) { | 
|---|
| 1134 | break; | 
|---|
| 1135 | } | 
|---|
| 1136 | } | 
|---|
| 1137 | if(i == nrLines) { | 
|---|
| 1138 | dprintf(("Component type %d not found!!", lpMl->Target.dwType)); | 
|---|
| 1139 | return MIXERR_INVALLINE; | 
|---|
| 1140 | } | 
|---|
| 1141 | memcpy(lpMl, &pmixerLines[i]->line, sizeof(MIXERLINEA)); | 
|---|
| 1142 | dprintf(("found line %s (%s) id %d connections %d controls %d", lpMl->szName, lpMl->szShortName, lpMl->dwLineID, lpMl->cConnections, lpMl->cControls)); | 
|---|
| 1143 | break; | 
|---|
| 1144 | } | 
|---|
| 1145 | default: | 
|---|
| 1146 | dprintf(("Unknown flag (%08lx)\n", fdwInfo & MIXER_GETLINEINFOF_QUERYMASK)); | 
|---|
| 1147 | return MMSYSERR_INVALPARAM; | 
|---|
| 1148 | break; | 
|---|
| 1149 | } | 
|---|
| 1150 | return MMSYSERR_NOERROR; | 
|---|
| 1151 | } | 
|---|
| 1152 | /******************************************************************************/ | 
|---|
| 1153 | /******************************************************************************/ | 
|---|
| 1154 | MMRESULT WINAPI mixerGetLineInfoW(HMIXEROBJ hmxobj, LPMIXERLINEW pmxl, DWORD fdwInfo) | 
|---|
| 1155 | { | 
|---|
| 1156 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 1157 | MIXERLINEA     line; | 
|---|
| 1158 | MMRESULT       result; | 
|---|
| 1159 |  | 
|---|
| 1160 | line.cbStruct              = sizeof(MIXERLINEA); | 
|---|
| 1161 | line.cChannels             = pmxl->cChannels; | 
|---|
| 1162 | line.cConnections          = pmxl->cConnections; | 
|---|
| 1163 | line.cControls             = pmxl->cControls; | 
|---|
| 1164 | line.dwComponentType       = pmxl->dwComponentType; | 
|---|
| 1165 | line.dwDestination         = pmxl->dwDestination; | 
|---|
| 1166 | line.dwLineID              = pmxl->dwLineID; | 
|---|
| 1167 | line.dwSource              = pmxl->dwSource; | 
|---|
| 1168 | line.dwUser                = pmxl->dwUser; | 
|---|
| 1169 | line.fdwLine               = pmxl->fdwLine; | 
|---|
| 1170 | line.szName[0]             = 0; | 
|---|
| 1171 | line.szShortName[0]        = 0; | 
|---|
| 1172 | line.Target.dwDeviceID     = pmxl->Target.dwDeviceID; | 
|---|
| 1173 | line.Target.dwType         = pmxl->Target.dwType; | 
|---|
| 1174 | line.Target.vDriverVersion = pmxl->Target.vDriverVersion; | 
|---|
| 1175 | line.Target.wMid           = pmxl->Target.wMid; | 
|---|
| 1176 | line.Target.wPid           = pmxl->Target.wPid; | 
|---|
| 1177 | //TODO: need to copy this?? (MIXER_GETLINEINFOF_TARGETTYPE) | 
|---|
| 1178 | line.Target.szPname[0]     = 0; | 
|---|
| 1179 |  | 
|---|
| 1180 | result = mixerGetLineInfoA(hmxobj, &line, fdwInfo); | 
|---|
| 1181 | if(result != MMSYSERR_NOERROR) { | 
|---|
| 1182 | return result; | 
|---|
| 1183 | } | 
|---|
| 1184 | pmxl->cbStruct              = sizeof(MIXERLINEA); | 
|---|
| 1185 | pmxl->cChannels             = line.cChannels; | 
|---|
| 1186 | pmxl->cConnections          = line.cConnections; | 
|---|
| 1187 | pmxl->cControls             = line.cControls; | 
|---|
| 1188 | pmxl->dwComponentType       = line.dwComponentType; | 
|---|
| 1189 | pmxl->dwDestination         = line.dwDestination; | 
|---|
| 1190 | pmxl->dwLineID              = line.dwLineID; | 
|---|
| 1191 | pmxl->dwSource              = line.dwSource; | 
|---|
| 1192 | pmxl->dwUser                = line.dwUser; | 
|---|
| 1193 | pmxl->fdwLine               = line.fdwLine; | 
|---|
| 1194 | MultiByteToWideChar(CP_ACP, 0, line.szName, -1, pmxl->szName, sizeof(line.szName)); | 
|---|
| 1195 | MultiByteToWideChar(CP_ACP, 0, line.szShortName, -1, pmxl->szShortName, sizeof(line.szShortName)); | 
|---|
| 1196 | pmxl->Target.dwDeviceID     = line.Target.dwDeviceID; | 
|---|
| 1197 | pmxl->Target.dwType         = line.Target.dwType; | 
|---|
| 1198 | pmxl->Target.vDriverVersion = line.Target.vDriverVersion; | 
|---|
| 1199 | pmxl->Target.wMid           = line.Target.wMid; | 
|---|
| 1200 | pmxl->Target.wPid           = line.Target.wPid; | 
|---|
| 1201 | pmxl->Target.szPname[0]     = 0; | 
|---|
| 1202 | MultiByteToWideChar(CP_ACP, 0, line.Target.szPname, -1, pmxl->Target.szPname, sizeof(line.Target.szPname)); | 
|---|
| 1203 | return MMSYSERR_NOERROR; | 
|---|
| 1204 | } | 
|---|
| 1205 | /******************************************************************************/ | 
|---|
| 1206 | /******************************************************************************/ | 
|---|
| 1207 | MMRESULT WINAPI mixerMessage(HMIXER hmx, UINT uMsg, DWORD dwParam1, DWORD dwParam2) | 
|---|
| 1208 | { | 
|---|
| 1209 | dprintf(("WINMM:mixerMessage - NOT IMPLEMENTED" )); | 
|---|
| 1210 | return MMSYSERR_NOTSUPPORTED; | 
|---|
| 1211 | } | 
|---|
| 1212 | /******************************************************************************/ | 
|---|
| 1213 | /******************************************************************************/ | 
|---|
| 1214 | UINT WINAPI mixerGetNumDevs() | 
|---|
| 1215 | { | 
|---|
| 1216 | if(fMMPMAvailable == FALSE || DartWaveOut::getNumDevices() == 0) { | 
|---|
| 1217 | return 0; | 
|---|
| 1218 | } | 
|---|
| 1219 | return 1; | 
|---|
| 1220 | } | 
|---|
| 1221 | /******************************************************************************/ | 
|---|
| 1222 | /******************************************************************************/ | 
|---|
| 1223 | MMRESULT WINAPI mixerGetDevCapsA(UINT uMxId, LPMIXERCAPSA pmxcaps, UINT cbmxcaps) | 
|---|
| 1224 | { | 
|---|
| 1225 | if(fMMPMAvailable == FALSE || DartWaveOut::getNumDevices() == 0) { | 
|---|
| 1226 | return MMSYSERR_BADDEVICEID; | 
|---|
| 1227 | } | 
|---|
| 1228 |  | 
|---|
| 1229 | if(uMxId > 0) { | 
|---|
| 1230 | return MMSYSERR_BADDEVICEID; | 
|---|
| 1231 | } | 
|---|
| 1232 |  | 
|---|
| 1233 | //According to MSDN, nothing is copied when cbmxcaps is zero | 
|---|
| 1234 | if(cbmxcaps >= sizeof(MIXERCAPSA)) { | 
|---|
| 1235 | pmxcaps->wMid = WINMM_MIXER_CAPS_WMID;                  /* manufacturer ID */ | 
|---|
| 1236 | pmxcaps->wPid = WINMM_MIXER_CAPS_WPID;                  /* product ID */ | 
|---|
| 1237 | pmxcaps->vDriverVersion = WINMM_MIXER_CAPS_VERSION;     /* version of the driver */ | 
|---|
| 1238 | strncpy( pmxcaps->szPname, WINMM_MIXERSTRING_A, sizeof(pmxcaps->szPname)); /* product name */ | 
|---|
| 1239 |  | 
|---|
| 1240 | pmxcaps->fdwSupport    = 0;    //no mixer flags exist | 
|---|
| 1241 | pmxcaps->cDestinations = nrDestinations; | 
|---|
| 1242 | dprintf(("mixerGetDevCapsA: cDestinations %d", pmxcaps->cDestinations)); | 
|---|
| 1243 | } | 
|---|
| 1244 |  | 
|---|
| 1245 | return MMSYSERR_NOERROR; | 
|---|
| 1246 | } | 
|---|
| 1247 | /******************************************************************************/ | 
|---|
| 1248 | /******************************************************************************/ | 
|---|
| 1249 | MMRESULT WINAPI mixerGetDevCapsW(UINT uMxId, LPMIXERCAPSW pmxcaps, UINT cbmxcaps) | 
|---|
| 1250 | { | 
|---|
| 1251 | if(fMMPMAvailable == FALSE || DartWaveOut::getNumDevices() == 0) { | 
|---|
| 1252 | return MMSYSERR_BADDEVICEID; | 
|---|
| 1253 | } | 
|---|
| 1254 |  | 
|---|
| 1255 | if(uMxId > 0) { | 
|---|
| 1256 | return MMSYSERR_BADDEVICEID; | 
|---|
| 1257 | } | 
|---|
| 1258 |  | 
|---|
| 1259 | //According to MSDN, nothing is copied when cbmxcaps is zero | 
|---|
| 1260 | if(cbmxcaps >= sizeof(MIXERCAPSW)) { | 
|---|
| 1261 | pmxcaps->wMid = WINMM_MIXER_CAPS_WMID;                  /* manufacturer ID */ | 
|---|
| 1262 | pmxcaps->wPid = WINMM_MIXER_CAPS_WPID;                  /* product ID */ | 
|---|
| 1263 | pmxcaps->vDriverVersion = WINMM_MIXER_CAPS_VERSION;     /* version of the driver */ | 
|---|
| 1264 | lstrcpyW( pmxcaps->szPname, WINMM_MIXERSTRING_W );      /* product name */ | 
|---|
| 1265 |  | 
|---|
| 1266 | pmxcaps->fdwSupport    = 0;    //no mixer flags exist | 
|---|
| 1267 | pmxcaps->cDestinations = nrDestinations; | 
|---|
| 1268 | dprintf(("mixerGetDevCapsW: cDestinations %d", pmxcaps->cDestinations)); | 
|---|
| 1269 | } | 
|---|
| 1270 |  | 
|---|
| 1271 | return MMSYSERR_NOERROR; | 
|---|
| 1272 | } | 
|---|
| 1273 | /******************************************************************************/ | 
|---|
| 1274 | /******************************************************************************/ | 
|---|
| 1275 | MMRESULT WINAPI mixerGetID(HMIXEROBJ hmxobj, UINT * puMxId, DWORD fdwId) | 
|---|
| 1276 | { | 
|---|
| 1277 | switch(fdwId) { | 
|---|
| 1278 | case MIXER_OBJECTF_MIXER: | 
|---|
| 1279 | { | 
|---|
| 1280 | DEVICE_STRUCT *pMixInfo = (DEVICE_STRUCT *)hmxobj; | 
|---|
| 1281 | if(!pMixInfo) { | 
|---|
| 1282 | if(puMxId) { | 
|---|
| 1283 | *puMxId = -1; | 
|---|
| 1284 | } | 
|---|
| 1285 | return MMSYSERR_INVALHANDLE; | 
|---|
| 1286 | } | 
|---|
| 1287 | if(puMxId) { | 
|---|
| 1288 | *puMxId = pMixInfo->uDeviceID; | 
|---|
| 1289 | } | 
|---|
| 1290 | break; | 
|---|
| 1291 | } | 
|---|
| 1292 |  | 
|---|
| 1293 | case MIXER_OBJECTF_HMIXER: | 
|---|
| 1294 | case MIXER_OBJECTF_WAVEOUT: | 
|---|
| 1295 | case MIXER_OBJECTF_HWAVEOUT: | 
|---|
| 1296 | case MIXER_OBJECTF_WAVEIN: | 
|---|
| 1297 | case MIXER_OBJECTF_HWAVEIN: | 
|---|
| 1298 | case MIXER_OBJECTF_MIDIOUT: | 
|---|
| 1299 | case MIXER_OBJECTF_HMIDIOUT: | 
|---|
| 1300 | case MIXER_OBJECTF_MIDIIN: | 
|---|
| 1301 | case MIXER_OBJECTF_HMIDIIN: | 
|---|
| 1302 | case MIXER_OBJECTF_AUX: | 
|---|
| 1303 | //TODO: assume default mixer | 
|---|
| 1304 | if(puMxId) { | 
|---|
| 1305 | *puMxId = 0; | 
|---|
| 1306 | } | 
|---|
| 1307 | break; | 
|---|
| 1308 | default: | 
|---|
| 1309 | if(puMxId) { | 
|---|
| 1310 | *puMxId = -1; | 
|---|
| 1311 | } | 
|---|
| 1312 | return MMSYSERR_INVALPARAM; | 
|---|
| 1313 | } | 
|---|
| 1314 | return MMSYSERR_NOERROR; | 
|---|
| 1315 | } | 
|---|
| 1316 | /******************************************************************************/ | 
|---|
| 1317 | /******************************************************************************/ | 
|---|
| 1318 | MMRESULT WINAPI mixerOpen(LPHMIXER phmx, UINT uMxId, DWORD dwCallback, DWORD dwInstance, | 
|---|
| 1319 | DWORD fdwOpen) | 
|---|
| 1320 | { | 
|---|
| 1321 | DEVICE_STRUCT *pMixInfo; | 
|---|
| 1322 |  | 
|---|
| 1323 | if(DartWaveOut::getNumDevices() == 0) { | 
|---|
| 1324 | if(phmx) *phmx = 0; | 
|---|
| 1325 | return MMSYSERR_NODRIVER; | 
|---|
| 1326 | } | 
|---|
| 1327 | pMixInfo = (DEVICE_STRUCT *)malloc(sizeof(DEVICE_STRUCT)); | 
|---|
| 1328 | if(pMixInfo == NULL) { | 
|---|
| 1329 | return MMSYSERR_NODRIVER; | 
|---|
| 1330 | } | 
|---|
| 1331 | pMixInfo->dwCallback = dwCallback; | 
|---|
| 1332 | pMixInfo->dwDriverInstance = dwInstance; | 
|---|
| 1333 | pMixInfo->dwFlags    = fdwOpen; | 
|---|
| 1334 | pMixInfo->uDeviceID  = uMxId; | 
|---|
| 1335 | pMixInfo->type       = WINMM_MIXER; | 
|---|
| 1336 | if(phmx) | 
|---|
| 1337 | *phmx = (HMIXER)pMixInfo; | 
|---|
| 1338 |  | 
|---|
| 1339 | return MMSYSERR_NOERROR; | 
|---|
| 1340 | } | 
|---|
| 1341 | /******************************************************************************/ | 
|---|
| 1342 | /******************************************************************************/ | 
|---|
| 1343 | MMRESULT WINAPI mixerClose(HMIXER hmx) | 
|---|
| 1344 | { | 
|---|
| 1345 | if(hmx) { | 
|---|
| 1346 | free((void *)hmx); | 
|---|
| 1347 | } | 
|---|
| 1348 | return MMSYSERR_NOERROR; | 
|---|
| 1349 | } | 
|---|
| 1350 | /******************************************************************************/ | 
|---|
| 1351 | /******************************************************************************/ | 
|---|
| 1352 | BOOL mixerInit() | 
|---|
| 1353 | { | 
|---|
| 1354 | MIXLINE *pDestLine; | 
|---|
| 1355 | MIXERCONTROLA *pWaveInMux; | 
|---|
| 1356 |  | 
|---|
| 1357 | if(!fMMPMAvailable) return TRUE; | 
|---|
| 1358 |  | 
|---|
| 1359 | memset(mixerDest, 0, sizeof(mixerDest)); | 
|---|
| 1360 | for(int i=0;i<MIXER_DEST_MAX;i++) { | 
|---|
| 1361 | memset(mixerDest[i].Connections, -1, sizeof(mixerDest[i].Connections)); | 
|---|
| 1362 | memset(mixerDest[i].Controls, -1, sizeof(mixerDest[i].Controls)); | 
|---|
| 1363 | } | 
|---|
| 1364 | memset(mixerSource, 0, sizeof(mixerSource)); | 
|---|
| 1365 | for(i=0;i<MIXER_SRC_MAX;i++) { | 
|---|
| 1366 | memset(mixerSource[i].Connections, -1, sizeof(mixerSource[i].Connections)); | 
|---|
| 1367 | memset(mixerSource[i].Controls, -1, sizeof(mixerSource[i].Controls)); | 
|---|
| 1368 | mixerSource[i].id = -1; | 
|---|
| 1369 | } | 
|---|
| 1370 | if(OSLibMixerOpen() == FALSE) { | 
|---|
| 1371 | //Line out destination | 
|---|
| 1372 | pDestLine = mixerAddDestination(MIXER_DEST_LINEOUT); | 
|---|
| 1373 | mixerAddControl(MIX_CTRL_VOL_OUT_LINE, pDestLine); | 
|---|
| 1374 | mixerAddControl(MIX_CTRL_MUTE_OUT_LINE, pDestLine); | 
|---|
| 1375 |  | 
|---|
| 1376 | //WaveIn destination | 
|---|
| 1377 | pDestLine  = mixerAddDestination(MIXER_DEST_WAVEIN); | 
|---|
| 1378 | pWaveInMux = mixerAddControl(MIX_CTRL_MUX_IN_W_SRC, pDestLine); | 
|---|
| 1379 | mixerAddControl(MIX_CTRL_VOL_IN_W_MIC, pDestLine); | 
|---|
| 1380 | mixerAddControl(MIX_CTRL_VOL_IN_W_LINE, pDestLine); | 
|---|
| 1381 | pWaveInMux->cMultipleItems = pDestLine->line.cConnections; | 
|---|
| 1382 | return TRUE; | 
|---|
| 1383 | } | 
|---|
| 1384 |  | 
|---|
| 1385 | //Line out destination | 
|---|
| 1386 | pDestLine = mixerAddDestination(MIXER_DEST_LINEOUT); | 
|---|
| 1387 |  | 
|---|
| 1388 | mixerAddControl(MIX_CTRL_VOL_OUT_LINE, pDestLine); | 
|---|
| 1389 | mixerAddControl(MIX_CTRL_MUTE_OUT_LINE, pDestLine); | 
|---|
| 1390 | if(OSLibMixIsControlPresent(MIX_CTRL_OUT_L_3DCENTER)) { | 
|---|
| 1391 | mixerAddControl(MIX_CTRL_OUT_L_3DCENTER, pDestLine); | 
|---|
| 1392 | } | 
|---|
| 1393 | if(OSLibMixIsControlPresent(MIX_CTRL_OUT_L_3DDEPTH)) { | 
|---|
| 1394 | mixerAddControl(MIX_CTRL_OUT_L_3DDEPTH, pDestLine); | 
|---|
| 1395 | } | 
|---|
| 1396 | if(OSLibMixIsControlPresent(MIX_CTRL_OUT_L_TREBLE)) { | 
|---|
| 1397 | mixerAddControl(MIX_CTRL_OUT_L_TREBLE, pDestLine); | 
|---|
| 1398 | } | 
|---|
| 1399 | if(OSLibMixIsControlPresent(MIX_CTRL_OUT_L_BASS)) { | 
|---|
| 1400 | mixerAddControl(MIX_CTRL_OUT_L_BASS, pDestLine); | 
|---|
| 1401 | } | 
|---|
| 1402 |  | 
|---|
| 1403 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_PCM)) { | 
|---|
| 1404 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_PCM); | 
|---|
| 1405 | mixerAddControl(MIX_CTRL_VOL_IN_L_PCM, pLine); | 
|---|
| 1406 | mixerAddControl(MIX_CTRL_MUTE_IN_L_PCM, pLine); | 
|---|
| 1407 | } | 
|---|
| 1408 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_WAVETABLE)) { | 
|---|
| 1409 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_WAVETABLE); | 
|---|
| 1410 | mixerAddControl(MIX_CTRL_VOL_IN_L_WAVETABLE, pLine); | 
|---|
| 1411 | mixerAddControl(MIX_CTRL_MUTE_IN_L_WAVETABLE, pLine); | 
|---|
| 1412 | } | 
|---|
| 1413 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_MIDI)) { | 
|---|
| 1414 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_MIDI); | 
|---|
| 1415 | mixerAddControl(MIX_CTRL_VOL_IN_L_MIDI, pLine); | 
|---|
| 1416 | mixerAddControl(MIX_CTRL_MUTE_IN_L_MIDI, pLine); | 
|---|
| 1417 | } | 
|---|
| 1418 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_MIC)) { | 
|---|
| 1419 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_MIC); | 
|---|
| 1420 | mixerAddControl(MIX_CTRL_VOL_IN_L_MIC, pLine); | 
|---|
| 1421 | mixerAddControl(MIX_CTRL_MUTE_IN_L_MIC, pLine); | 
|---|
| 1422 | if(OSLibMixIsControlPresent(MIX_CTRL_BOOST_IN_L_MIC)) { | 
|---|
| 1423 | mixerAddControl(MIX_CTRL_BOOST_IN_L_MIC, pLine); | 
|---|
| 1424 | } | 
|---|
| 1425 | } | 
|---|
| 1426 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_LINE)) { | 
|---|
| 1427 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_LINE); | 
|---|
| 1428 | mixerAddControl(MIX_CTRL_VOL_IN_L_LINE, pLine); | 
|---|
| 1429 | mixerAddControl(MIX_CTRL_MUTE_IN_L_LINE, pLine); | 
|---|
| 1430 | } | 
|---|
| 1431 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_CD)) { | 
|---|
| 1432 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_CD); | 
|---|
| 1433 | mixerAddControl(MIX_CTRL_VOL_IN_L_CD, pLine); | 
|---|
| 1434 | mixerAddControl(MIX_CTRL_MUTE_IN_L_CD, pLine); | 
|---|
| 1435 | } | 
|---|
| 1436 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_AUX)) { | 
|---|
| 1437 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_AUX); | 
|---|
| 1438 | mixerAddControl(MIX_CTRL_VOL_IN_L_AUX, pLine); | 
|---|
| 1439 | mixerAddControl(MIX_CTRL_MUTE_IN_L_AUX, pLine); | 
|---|
| 1440 | } | 
|---|
| 1441 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_SPDIF)) { | 
|---|
| 1442 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_SPDIF); | 
|---|
| 1443 | mixerAddControl(MIX_CTRL_VOL_IN_L_SPDIF, pLine); | 
|---|
| 1444 | mixerAddControl(MIX_CTRL_MUTE_IN_L_SPDIF, pLine); | 
|---|
| 1445 | } | 
|---|
| 1446 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_MONO)) { | 
|---|
| 1447 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_MONOIN); | 
|---|
| 1448 | mixerAddControl(MIX_CTRL_VOL_IN_L_MONO, pLine); | 
|---|
| 1449 | mixerAddControl(MIX_CTRL_MUTE_IN_L_MONO, pLine); | 
|---|
| 1450 | } | 
|---|
| 1451 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_PHONE)) { | 
|---|
| 1452 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_PHONE); | 
|---|
| 1453 | mixerAddControl(MIX_CTRL_VOL_IN_L_PHONE, pLine); | 
|---|
| 1454 | mixerAddControl(MIX_CTRL_MUTE_IN_L_PHONE, pLine); | 
|---|
| 1455 | } | 
|---|
| 1456 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_IN_L_VIDEO)) { | 
|---|
| 1457 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_L_VIDEO); | 
|---|
| 1458 | mixerAddControl(MIX_CTRL_VOL_IN_L_VIDEO, pLine); | 
|---|
| 1459 | mixerAddControl(MIX_CTRL_MUTE_IN_L_VIDEO, pLine); | 
|---|
| 1460 | } | 
|---|
| 1461 |  | 
|---|
| 1462 | //Wave In Destination | 
|---|
| 1463 | if(OSLibMixIsControlPresent(MIX_CTRL_MUX_IN_W_SRC)) { | 
|---|
| 1464 | pDestLine = mixerAddDestination(MIXER_DEST_WAVEIN); | 
|---|
| 1465 | pWaveInMux = mixerAddControl(MIX_CTRL_MUX_IN_W_SRC, pDestLine); | 
|---|
| 1466 |  | 
|---|
| 1467 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_MIC)) { | 
|---|
| 1468 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_MIC); | 
|---|
| 1469 | mixerAddControl(MIX_CTRL_VOL_IN_W_MIC, pLine); | 
|---|
| 1470 | } | 
|---|
| 1471 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_LINE)) { | 
|---|
| 1472 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_LINE); | 
|---|
| 1473 | mixerAddControl(MIX_CTRL_VOL_IN_W_LINE, pLine); | 
|---|
| 1474 | } | 
|---|
| 1475 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_CD)) { | 
|---|
| 1476 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_CD); | 
|---|
| 1477 | mixerAddControl(MIX_CTRL_VOL_IN_W_CD, pLine); | 
|---|
| 1478 | } | 
|---|
| 1479 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_AUX)) { | 
|---|
| 1480 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_AUX); | 
|---|
| 1481 | mixerAddControl(MIX_CTRL_VOL_IN_W_AUX, pLine); | 
|---|
| 1482 | } | 
|---|
| 1483 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_PCM)) { | 
|---|
| 1484 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_PCM); | 
|---|
| 1485 | mixerAddControl(MIX_CTRL_VOL_IN_W_PCM, pLine); | 
|---|
| 1486 | } | 
|---|
| 1487 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_WAVETABLE)) { | 
|---|
| 1488 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_WAVETABLE); | 
|---|
| 1489 | mixerAddControl(MIX_CTRL_VOL_IN_W_WAVETABLE, pLine); | 
|---|
| 1490 | } | 
|---|
| 1491 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_MIDI)) { | 
|---|
| 1492 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_MIDI); | 
|---|
| 1493 | mixerAddControl(MIX_CTRL_VOL_IN_W_MIDI, pLine); | 
|---|
| 1494 | } | 
|---|
| 1495 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_MONOIN)) { | 
|---|
| 1496 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_MONOIN); | 
|---|
| 1497 | mixerAddControl(MIX_CTRL_VOL_IN_W_MONO, pLine); | 
|---|
| 1498 | } | 
|---|
| 1499 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_PHONE)) { | 
|---|
| 1500 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_PHONE); | 
|---|
| 1501 | mixerAddControl(MIX_CTRL_VOL_IN_W_PHONE, pLine); | 
|---|
| 1502 | } | 
|---|
| 1503 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_SPDIF)) { | 
|---|
| 1504 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_SPDIF); | 
|---|
| 1505 | mixerAddControl(MIX_CTRL_VOL_IN_W_SPDIF, pLine); | 
|---|
| 1506 | } | 
|---|
| 1507 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_VIDEO)) { | 
|---|
| 1508 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_VIDEO); | 
|---|
| 1509 | mixerAddControl(MIX_CTRL_VOL_IN_W_VIDEO, pLine); | 
|---|
| 1510 | } | 
|---|
| 1511 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_STEREOMIX)) { | 
|---|
| 1512 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_STEREOMIX); | 
|---|
| 1513 | mixerAddControl(MIX_CTRL_VOL_IN_W_STEREOMIX, pLine); | 
|---|
| 1514 | } | 
|---|
| 1515 | if(OSLibMixIsRecSourcePresent(MIXER_SRC_IN_W_MONOMIX)) { | 
|---|
| 1516 | MIXLINE *pLine = mixerAddSource(pDestLine, MIXER_SRC_IN_W_MONOMIX); | 
|---|
| 1517 | mixerAddControl(MIX_CTRL_VOL_IN_W_MONOMIX, pLine); | 
|---|
| 1518 | } | 
|---|
| 1519 | pWaveInMux->cMultipleItems    = pDestLine->line.cConnections; | 
|---|
| 1520 | pWaveInMux->Bounds.s.lMinimum = 0; | 
|---|
| 1521 | pWaveInMux->Bounds.s.lMaximum = pDestLine->line.cConnections; | 
|---|
| 1522 | pWaveInMux->Metrics.cSteps    = pDestLine->line.cConnections; | 
|---|
| 1523 | } | 
|---|
| 1524 |  | 
|---|
| 1525 | //SPDIF destination | 
|---|
| 1526 | if(OSLibMixIsControlPresent(MIX_CTRL_VOL_OUT_SPDIF)) { | 
|---|
| 1527 | pDestLine = mixerAddDestination(MIXER_DEST_SPDIFOUT); | 
|---|
| 1528 | mixerAddControl(MIX_CTRL_VOL_OUT_SPDIF, pDestLine); | 
|---|
| 1529 | mixerAddControl(MIX_CTRL_MUTE_OUT_SPDIF, pDestLine); | 
|---|
| 1530 | } | 
|---|
| 1531 |  | 
|---|
| 1532 | return TRUE; | 
|---|
| 1533 | } | 
|---|
| 1534 | /******************************************************************************/ | 
|---|
| 1535 | /******************************************************************************/ | 
|---|
| 1536 | void mixerExit() | 
|---|
| 1537 | { | 
|---|
| 1538 | if(fMMPMAvailable == FALSE) return; | 
|---|
| 1539 |  | 
|---|
| 1540 | OSLibMixerClose(); | 
|---|
| 1541 | } | 
|---|
| 1542 | /******************************************************************************/ | 
|---|
| 1543 | /******************************************************************************/ | 
|---|
| 1544 | static MIXLINE *mixerAddSource(MIXLINE *pDestLine, DWORD dwSource) | 
|---|
| 1545 | { | 
|---|
| 1546 | MIXLINE    *pSource = &mixerSource[nrSources]; | 
|---|
| 1547 | MIXERLINEA *pline   = &pSource->line; | 
|---|
| 1548 |  | 
|---|
| 1549 | if(nrSources >= MAX_MIXER_SOURCES) { | 
|---|
| 1550 | dprintf(("ERROR: mixerAddSource: out of room!!!")); | 
|---|
| 1551 | DebugInt3(); | 
|---|
| 1552 | return NULL; | 
|---|
| 1553 | } | 
|---|
| 1554 |  | 
|---|
| 1555 | memset(pline, 0, sizeof(MIXERLINEA)); | 
|---|
| 1556 |  | 
|---|
| 1557 | pline->cbStruct              = sizeof(MIXERLINEA); | 
|---|
| 1558 | pline->cConnections          = 0; | 
|---|
| 1559 | pline->cControls             = 0; | 
|---|
| 1560 | pline->dwDestination         = 0; | 
|---|
| 1561 | pline->fdwLine               = MIXERLINE_LINEF_SOURCE; | 
|---|
| 1562 | pline->Target.dwDeviceID     = 0; | 
|---|
| 1563 | pline->Target.wMid           = WINMM_MIXER_CAPS_WMID; | 
|---|
| 1564 | pline->Target.vDriverVersion = WINMM_MIXER_CAPS_VERSION; | 
|---|
| 1565 | pline->dwSource              = nrSources; | 
|---|
| 1566 | OSLibMixGetLineCaps(dwSource, &pline->cChannels); | 
|---|
| 1567 | nrSources++; | 
|---|
| 1568 |  | 
|---|
| 1569 | switch(dwSource) { | 
|---|
| 1570 | case MIXER_SRC_IN_L_MONOIN: | 
|---|
| 1571 | case MIXER_SRC_IN_W_MONOIN: | 
|---|
| 1572 | case MIXER_SRC_IN_L_PHONE: | 
|---|
| 1573 | case MIXER_SRC_IN_W_PHONE: | 
|---|
| 1574 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_TELEPHONE; | 
|---|
| 1575 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1576 | break; | 
|---|
| 1577 |  | 
|---|
| 1578 | case MIXER_SRC_IN_L_MIC: | 
|---|
| 1579 | case MIXER_SRC_IN_W_MIC: | 
|---|
| 1580 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE; | 
|---|
| 1581 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1582 | break; | 
|---|
| 1583 |  | 
|---|
| 1584 | case MIXER_SRC_IN_L_LINE: | 
|---|
| 1585 | case MIXER_SRC_IN_W_LINE: | 
|---|
| 1586 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_LINE; | 
|---|
| 1587 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1588 | break; | 
|---|
| 1589 |  | 
|---|
| 1590 | case MIXER_SRC_IN_L_CD: | 
|---|
| 1591 | case MIXER_SRC_IN_W_CD: | 
|---|
| 1592 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_COMPACTDISC; | 
|---|
| 1593 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1594 | break; | 
|---|
| 1595 |  | 
|---|
| 1596 | case MIXER_SRC_IN_L_SPDIF: | 
|---|
| 1597 | case MIXER_SRC_IN_W_SPDIF: | 
|---|
| 1598 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_DIGITAL; | 
|---|
| 1599 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1600 | break; | 
|---|
| 1601 |  | 
|---|
| 1602 | case MIXER_SRC_IN_L_VIDEO: | 
|---|
| 1603 | case MIXER_SRC_IN_W_VIDEO: | 
|---|
| 1604 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_UNDEFINED; | 
|---|
| 1605 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1606 | break; | 
|---|
| 1607 |  | 
|---|
| 1608 | case MIXER_SRC_IN_L_AUX: | 
|---|
| 1609 | case MIXER_SRC_IN_W_AUX: | 
|---|
| 1610 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_AUXILIARY; | 
|---|
| 1611 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_AUX; | 
|---|
| 1612 | break; | 
|---|
| 1613 |  | 
|---|
| 1614 | case MIXER_SRC_IN_L_PCM: | 
|---|
| 1615 | case MIXER_SRC_IN_W_PCM: | 
|---|
| 1616 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_WAVEOUT; | 
|---|
| 1617 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_WAVEOUT; | 
|---|
| 1618 | break; | 
|---|
| 1619 |  | 
|---|
| 1620 | case MIXER_SRC_IN_L_WAVETABLE: | 
|---|
| 1621 | case MIXER_SRC_IN_W_WAVETABLE: | 
|---|
| 1622 | case MIXER_SRC_IN_L_MIDI: | 
|---|
| 1623 | case MIXER_SRC_IN_W_MIDI: | 
|---|
| 1624 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_SRC_SYNTHESIZER; | 
|---|
| 1625 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_MIDIOUT; | 
|---|
| 1626 | break; | 
|---|
| 1627 |  | 
|---|
| 1628 | default: | 
|---|
| 1629 | DebugInt3(); | 
|---|
| 1630 | return NULL; | 
|---|
| 1631 | } | 
|---|
| 1632 | strncpy(pline->szShortName, szSourceName[dwSource][0], sizeof(pline->szShortName)); | 
|---|
| 1633 | strncpy(pline->szName,      szSourceName[dwSource][1], sizeof(pline->szName)); | 
|---|
| 1634 | strncpy(pline->Target.szPname, WINMM_MIXERSTRING_A, sizeof(pline->Target.szPname)); | 
|---|
| 1635 |  | 
|---|
| 1636 | //add to array of mixer line pointers | 
|---|
| 1637 | pline->dwLineID      = nrLines; | 
|---|
| 1638 | pmixerLines[nrLines] = pSource; | 
|---|
| 1639 | nrLines++; | 
|---|
| 1640 |  | 
|---|
| 1641 | //store line id in source to line mapping array | 
|---|
| 1642 | pSource->id = dwSource; | 
|---|
| 1643 |  | 
|---|
| 1644 | //increase nr of inputs at destination line | 
|---|
| 1645 | pDestLine->line.cConnections++; | 
|---|
| 1646 | pDestLine->Connections[pDestLine->cConnections] = pline->dwLineID; | 
|---|
| 1647 | pDestLine->cConnections++; | 
|---|
| 1648 |  | 
|---|
| 1649 | dprintf(("Adding Source %s (%s) with destination %s (%d -> %d)", pline->szName, pline->szShortName, pDestLine->line.szName, dwSource, pline->dwLineID)); | 
|---|
| 1650 |  | 
|---|
| 1651 | return pSource; | 
|---|
| 1652 | } | 
|---|
| 1653 | /******************************************************************************/ | 
|---|
| 1654 | /******************************************************************************/ | 
|---|
| 1655 | static MIXLINE *mixerAddDestination(DWORD dwDest) | 
|---|
| 1656 | { | 
|---|
| 1657 | MIXLINE    *pDest  = &mixerDest[nrDestinations]; | 
|---|
| 1658 | MIXERLINEA *pline  = &pDest->line; | 
|---|
| 1659 |  | 
|---|
| 1660 | if(nrDestinations >= MAX_MIXER_DESTINATIONS) { | 
|---|
| 1661 | dprintf(("ERROR: mixerAddDestination: out of room!!!")); | 
|---|
| 1662 | DebugInt3(); | 
|---|
| 1663 | return NULL; | 
|---|
| 1664 | } | 
|---|
| 1665 | memset(pline, 0, sizeof(MIXERLINEA)); | 
|---|
| 1666 | pline->cbStruct        = sizeof(MIXERLINEA); | 
|---|
| 1667 | pline->cConnections    = 0; | 
|---|
| 1668 | pline->cControls       = 0; | 
|---|
| 1669 | pline->dwSource        = (DWORD)-1; | 
|---|
| 1670 | pline->dwDestination   = nrDestinations; | 
|---|
| 1671 |  | 
|---|
| 1672 | switch(dwDest) { | 
|---|
| 1673 | case MIXER_DEST_LINEOUT: | 
|---|
| 1674 | pline->cChannels       = 2; | 
|---|
| 1675 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_DST_SPEAKERS; | 
|---|
| 1676 | pline->fdwLine         = MIXERLINE_LINEF_ACTIVE; | 
|---|
| 1677 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1678 | break; | 
|---|
| 1679 |  | 
|---|
| 1680 | case MIXER_DEST_SPDIFOUT: | 
|---|
| 1681 | pline->cChannels       = 2; | 
|---|
| 1682 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_DST_DIGITAL; | 
|---|
| 1683 | pline->fdwLine         = MIXERLINE_LINEF_ACTIVE; | 
|---|
| 1684 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_UNDEFINED; | 
|---|
| 1685 | break; | 
|---|
| 1686 |  | 
|---|
| 1687 | case MIXER_DEST_WAVEIN: | 
|---|
| 1688 | pline->cChannels       = 2; | 
|---|
| 1689 | pline->dwComponentType = MIXERLINE_COMPONENTTYPE_DST_WAVEIN; | 
|---|
| 1690 | pline->fdwLine         = MIXERLINE_LINEF_ACTIVE; | 
|---|
| 1691 | pline->Target.dwType   = MIXERLINE_TARGETTYPE_WAVEIN; | 
|---|
| 1692 | break; | 
|---|
| 1693 |  | 
|---|
| 1694 | default: | 
|---|
| 1695 | DebugInt3(); | 
|---|
| 1696 | return NULL; | 
|---|
| 1697 | } | 
|---|
| 1698 | strncpy(pline->szShortName, szDestName[dwDest][0], sizeof(pline->szShortName)); | 
|---|
| 1699 | strncpy(pline->szName,      szDestName[dwDest][1], sizeof(pline->szName)); | 
|---|
| 1700 | strncpy(pline->Target.szPname, WINMM_MIXERSTRING_A, sizeof(pline->Target.szPname)); | 
|---|
| 1701 | pline->Target.dwDeviceID     = 0; | 
|---|
| 1702 | pline->Target.wMid           = WINMM_MIXER_CAPS_WMID; | 
|---|
| 1703 | pline->Target.wPid           = WINMM_MIXER_CAPS_WPID; | 
|---|
| 1704 | pline->Target.vDriverVersion = WINMM_MIXER_CAPS_VERSION; | 
|---|
| 1705 |  | 
|---|
| 1706 | dprintf(("Adding destination %s (%s) connections %d controls %d", pline->szName, pline->szShortName, pline->cConnections, pline->cControls)); | 
|---|
| 1707 | nrDestinations++; | 
|---|
| 1708 |  | 
|---|
| 1709 | pline->dwLineID      = nrLines; | 
|---|
| 1710 | pmixerLines[nrLines] = pDest; | 
|---|
| 1711 | nrLines++; | 
|---|
| 1712 |  | 
|---|
| 1713 | //save internal id | 
|---|
| 1714 | pDest->id = dwDest; | 
|---|
| 1715 | return pDest; | 
|---|
| 1716 | } | 
|---|
| 1717 | /******************************************************************************/ | 
|---|
| 1718 | /******************************************************************************/ | 
|---|
| 1719 | static MIXERCONTROLA * mixerAddControl(DWORD dwControl, MIXLINE *pSrcLine) | 
|---|
| 1720 | { | 
|---|
| 1721 | MIXCONTROL    *pmixctrl = &mixerControls[nrControls]; | 
|---|
| 1722 | MIXERCONTROLA *pctrl    = &pmixctrl->ctrl; | 
|---|
| 1723 |  | 
|---|
| 1724 | if(nrControls >= MAX_MIXER_CONTROLS) { | 
|---|
| 1725 | dprintf(("ERROR: mixerAddControl: out of room!!!")); | 
|---|
| 1726 | DebugInt3(); | 
|---|
| 1727 | return NULL; | 
|---|
| 1728 | } | 
|---|
| 1729 |  | 
|---|
| 1730 | memset(pctrl, 0, sizeof(MIXERCONTROLA)); | 
|---|
| 1731 | pctrl->cbStruct = sizeof(MIXERCONTROLA); | 
|---|
| 1732 | pctrl->cMultipleItems   = 0; | 
|---|
| 1733 | pctrl->dwControlID      = nrControls; | 
|---|
| 1734 | pctrl->fdwControl       = 0; | 
|---|
| 1735 | switch(dwControl) { | 
|---|
| 1736 | case MIX_CTRL_MUX_IN_W_SRC: | 
|---|
| 1737 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_MULTIPLE | MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1738 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_MUX; | 
|---|
| 1739 | pctrl->Bounds.s.lMinimum = 0; | 
|---|
| 1740 | pctrl->Bounds.s.lMaximum = 1; | 
|---|
| 1741 | pctrl->Metrics.cSteps    = 1; | 
|---|
| 1742 | pctrl->cMultipleItems    = 0;   //to be filled in later (see caller) | 
|---|
| 1743 | break; | 
|---|
| 1744 |  | 
|---|
| 1745 | case MIX_CTRL_VOL_OUT_LINE: | 
|---|
| 1746 | case MIX_CTRL_VOL_IN_L_MONO: | 
|---|
| 1747 | case MIX_CTRL_VOL_IN_L_PHONE: | 
|---|
| 1748 | case MIX_CTRL_VOL_IN_L_MIC: | 
|---|
| 1749 | case MIX_CTRL_VOL_IN_L_LINE: | 
|---|
| 1750 | case MIX_CTRL_VOL_IN_L_CD: | 
|---|
| 1751 | case MIX_CTRL_VOL_IN_L_SPDIF: | 
|---|
| 1752 | case MIX_CTRL_VOL_IN_L_VIDEO: | 
|---|
| 1753 | case MIX_CTRL_VOL_IN_L_AUX: | 
|---|
| 1754 | case MIX_CTRL_VOL_IN_L_PCM: | 
|---|
| 1755 | case MIX_CTRL_VOL_IN_L_WAVETABLE: | 
|---|
| 1756 | case MIX_CTRL_VOL_IN_L_MIDI: | 
|---|
| 1757 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_VOLUME; | 
|---|
| 1758 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1759 | break; | 
|---|
| 1760 |  | 
|---|
| 1761 | case MIX_CTRL_VOL_IN_W_MONO: | 
|---|
| 1762 | case MIX_CTRL_VOL_IN_W_PHONE: | 
|---|
| 1763 | case MIX_CTRL_VOL_IN_W_MIC: | 
|---|
| 1764 | case MIX_CTRL_VOL_IN_W_LINE: | 
|---|
| 1765 | case MIX_CTRL_VOL_IN_W_CD: | 
|---|
| 1766 | case MIX_CTRL_VOL_IN_W_SPDIF: | 
|---|
| 1767 | case MIX_CTRL_VOL_IN_W_VIDEO: | 
|---|
| 1768 | case MIX_CTRL_VOL_IN_W_AUX: | 
|---|
| 1769 | case MIX_CTRL_VOL_IN_W_PCM: | 
|---|
| 1770 | case MIX_CTRL_VOL_IN_W_WAVETABLE: | 
|---|
| 1771 | case MIX_CTRL_VOL_IN_W_MIDI: | 
|---|
| 1772 | case MIX_CTRL_VOL_IN_W_STEREOMIX: | 
|---|
| 1773 | case MIX_CTRL_VOL_IN_W_MONOMIX: | 
|---|
| 1774 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_VOLUME; | 
|---|
| 1775 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1776 | break; | 
|---|
| 1777 |  | 
|---|
| 1778 | case MIX_CTRL_MUTE_OUT_LINE: | 
|---|
| 1779 | case MIX_CTRL_MUTE_IN_L_MONO: | 
|---|
| 1780 | case MIX_CTRL_MUTE_IN_L_PHONE: | 
|---|
| 1781 | case MIX_CTRL_MUTE_IN_L_MIC: | 
|---|
| 1782 | case MIX_CTRL_MUTE_IN_L_LINE: | 
|---|
| 1783 | case MIX_CTRL_MUTE_IN_L_CD: | 
|---|
| 1784 | case MIX_CTRL_MUTE_IN_L_SPDIF: | 
|---|
| 1785 | case MIX_CTRL_MUTE_IN_L_VIDEO: | 
|---|
| 1786 | case MIX_CTRL_MUTE_IN_L_AUX: | 
|---|
| 1787 | case MIX_CTRL_MUTE_IN_L_PCM: | 
|---|
| 1788 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1789 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_MUTE; | 
|---|
| 1790 | pctrl->Bounds.s.lMinimum = 0; | 
|---|
| 1791 | pctrl->Bounds.s.lMaximum = 1; | 
|---|
| 1792 | pctrl->Metrics.cSteps    = 0; | 
|---|
| 1793 | break; | 
|---|
| 1794 |  | 
|---|
| 1795 | case MIX_CTRL_VOL_OUT_SPDIF: | 
|---|
| 1796 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_VOLUME; | 
|---|
| 1797 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1798 | break; | 
|---|
| 1799 |  | 
|---|
| 1800 | case MIX_CTRL_MUTE_OUT_SPDIF: | 
|---|
| 1801 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1802 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_MUTE; | 
|---|
| 1803 | pctrl->Bounds.s.lMinimum = 0; | 
|---|
| 1804 | pctrl->Bounds.s.lMaximum = 1; | 
|---|
| 1805 | pctrl->Metrics.cSteps    = 0; | 
|---|
| 1806 | break; | 
|---|
| 1807 |  | 
|---|
| 1808 | case MIX_CTRL_BOOST_IN_L_MIC: | 
|---|
| 1809 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1810 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_ONOFF; | 
|---|
| 1811 | pctrl->Bounds.s.lMinimum = 0; | 
|---|
| 1812 | pctrl->Bounds.s.lMaximum = 1; | 
|---|
| 1813 | pctrl->Metrics.cSteps    = 0; | 
|---|
| 1814 | break; | 
|---|
| 1815 |  | 
|---|
| 1816 | case MIX_CTRL_OUT_L_3DDEPTH: | 
|---|
| 1817 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1818 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_FADER; | 
|---|
| 1819 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1820 | break; | 
|---|
| 1821 |  | 
|---|
| 1822 | case MIX_CTRL_OUT_L_3DCENTER: | 
|---|
| 1823 | pctrl->fdwControl       = MIXERCONTROL_CONTROLF_UNIFORM; | 
|---|
| 1824 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_FADER; | 
|---|
| 1825 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1826 | break; | 
|---|
| 1827 |  | 
|---|
| 1828 | case MIX_CTRL_OUT_L_TREBLE: | 
|---|
| 1829 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_TREBLE; | 
|---|
| 1830 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1831 | break; | 
|---|
| 1832 |  | 
|---|
| 1833 | case MIX_CTRL_OUT_L_BASS: | 
|---|
| 1834 | pctrl->dwControlType    = MIXERCONTROL_CONTROLTYPE_BASS; | 
|---|
| 1835 | OSLibMixGetCtrlCaps(dwControl, &pctrl->Bounds.s.lMinimum, &pctrl->Bounds.s.lMaximum, &pctrl->Metrics.cSteps); | 
|---|
| 1836 | break; | 
|---|
| 1837 |  | 
|---|
| 1838 | default: | 
|---|
| 1839 | DebugInt3(); | 
|---|
| 1840 | return FALSE; | 
|---|
| 1841 | } | 
|---|
| 1842 |  | 
|---|
| 1843 | //add control to list of controls associated with source line | 
|---|
| 1844 | pSrcLine->Controls[pSrcLine->cControls] = pctrl->dwControlID; | 
|---|
| 1845 | pSrcLine->line.cControls++; | 
|---|
| 1846 | pSrcLine->cControls++; | 
|---|
| 1847 |  | 
|---|
| 1848 | strncpy(pctrl->szShortName, szCtrlName[dwControl][0], sizeof(pctrl->szShortName)); | 
|---|
| 1849 | strncpy(pctrl->szName,      szCtrlName[dwControl][1], sizeof(pctrl->szName)); | 
|---|
| 1850 |  | 
|---|
| 1851 | //save internal id | 
|---|
| 1852 | pmixctrl->id = dwControl; | 
|---|
| 1853 | nrControls++; | 
|---|
| 1854 | dprintf(("Adding control %s (%s)", pctrl->szName, pctrl->szShortName)); | 
|---|
| 1855 | return pctrl; | 
|---|
| 1856 | } | 
|---|
| 1857 | /******************************************************************************/ | 
|---|
| 1858 | /******************************************************************************/ | 
|---|
| 1859 |  | 
|---|