Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1305 lines
32 KiB

// File: AudioCpl.cpp
#include "precomp.h"
#include "resource.h"
#include "avdefs.h"
#include "ConfCpl.h"
#include "help_ids.h"
#include <ih323cc.h>
#include <initguid.h>
#include <nacguids.h>
#include <auformats.h>
#include "ConfPolicies.h"
const int SLOW_CPU_MHZ = 110; // pentiums 110mhz - 180mhz are "slow"
const int FAST_CPU_MHZ = 200; // fast machines are 200mhz and faster
const int VERYFAST_CPU_MHZ = 500; // un-normalized, this will include 400 mhz PentIIs
// 486 and slow pentium (<= 100 mhz) settings (VERY SLOW)
const UINT CIF_RATE_VERYSLOW = 3;
const UINT SQCIF_RATE_VERYSLOW = 7;
const UINT QCIF_RATE_VERYSLOW = 7;
// pentium 75mhz settings (SLOW)
const UINT CIF_RATE_SLOW = 7;
const UINT SQCIF_RATE_SLOW = 15;
const UINT QCIF_RATE_SLOW = 15;
// pentium 200mhz settings (FAST)
const UINT CIF_RATE_FAST = 15;
const UINT SQCIF_RATE_FAST = 30;
const UINT QCIF_RATE_FAST = 30;
const UINT CIF_RATE_VERYFAST = 30;
typedef struct _CODECINFO
{
HINSTANCE hLib;
IH323CallControl* lpIH323;
UINT uNumFormats;
PBASIC_AUDCAP_INFO pCodecCapList;
PWORD pOldCodecOrderList;
PWORD pCurCodecOrderList;
LPAPPCAPPIF lpIAppCap;
LPAPPVIDCAPPIF lpIVidAppCap;
LPCAPSIF lpICapsCtl;
} CODECINFO, *PCODECINFO;
typedef struct
{
BOOL fManual;
UINT uBandwidth;
PCODECINFO pCodecInfo;
} ADVCODEC, *PADVCODEC;
//SS: the cpu utilization is bogus so for now hide it
//#define CODEC_LV_NUM_COLUMNS 3
#define CODEC_LV_MAX_COLUMNS 3
#define CODEC_LV_NUM_COLUMNS 2
#define MAGIC_CPU_DO_NOT_EXCEED_PERCENTAGE 50 //Don't use more than this % of the CPU for encoding
//Also in nac\balance.cpp
// given the bandwidth identifier (1-4) and the CPU megahertz, return
// the actual bandwidth amount in bits/sec
int GetBandwidthBits(int id, int megahertz)
{
int nBits=BW_144KBS_BITS;
switch (id)
{
case BW_144KBS:
nBits=BW_144KBS_BITS;
break;
case BW_288KBS:
nBits=BW_288KBS_BITS;
break;
case BW_ISDN:
nBits=BW_ISDN_BITS;
break;
case BW_MOREKBS: // LAN
if (megahertz >= VERYFAST_CPU_MHZ)
{
nBits = BW_FASTLAN_BITS;
}
else
{
nBits = BW_SLOWLAN_BITS;
}
break;
}
return nBits;
}
// begin data types used for ChooseCodecByBw
#define CODEC_DISABLED 99
#define CODEC_UNKNOWN 98
typedef struct _codecprefrow
{
WORD wFormatTag;
WORD wOrder586;
WORD wOrder486;
WORD wMinBW;
} CODECPREFROW;
typedef CODECPREFROW CODECPREFTABLE[];
// FORMAT NAME 586 Order 486 Order Minimum BW
static const CODECPREFTABLE g_CodecPrefTable =
{
WAVE_FORMAT_MSG723, 1, CODEC_DISABLED, BW_144KBS,
WAVE_FORMAT_LH_SB16, 2, 1, BW_288KBS,
WAVE_FORMAT_LH_SB8, 3, 2, BW_144KBS,
WAVE_FORMAT_LH_SB12, 4, 3, BW_144KBS,
WAVE_FORMAT_MULAW, 5, 4, BW_ISDN,
WAVE_FORMAT_ALAW, 6, 5, BW_ISDN,
WAVE_FORMAT_ADPCM, 7, 6, BW_ISDN,
WAVE_FORMAT_LH_CELP, 8, CODEC_DISABLED, BW_144KBS,
WAVE_FORMAT_MSRT24, 9, 7, BW_144KBS
};
// end of stuff used by ChooseCodecByBw
static const int g_ColHdrStrId[CODEC_LV_MAX_COLUMNS] =
{
IDS_CODECNAME,
IDS_MAXBITRATE,
IDS_CPUUTIL,
};
//prototypes
INT_PTR CALLBACK AdvCodecDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
int CALLBACK CodecLVCompareProc (LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort);
BOOL FillCodecListView(HWND hCB, PCODECINFO pCodecInfo);
BOOL ChooseCodecByBw(UINT uBandWidthId, PCODECINFO pCodecInfo);
void SortCodecs(PADVCODEC pAdvCodec, int nSelection);
BOOL SetAppCodecPrefs(PCODECINFO pCodecInfo,UINT uBandwith);
BOOL GetIAppCap(PCODECINFO pCodecInfo);
BOOL GetAppCapFormats(PCODECINFO pCodecInfo);
void FreeAppCapFormats(PCODECINFO pCodecInfo);
void ReleaseIAppCap(PCODECINFO pCodecInfo);
BOOL ApplyUserPrefs(PCODECINFO pCodecInfo, UINT uMaximumBandwidth);
static const DWORD _rgHelpIdsAudio[] = {
IDC_GENERAL_GROUP, IDH_AUDIO_GENERAL,
IDC_FULLDUPLEX, IDH_AUDIO_FULL_DUPLEX,
IDC_AUTOGAIN, IDH_AUDIO_AUTO_GAIN,
IDC_AUTOMIX, IDH_AUDIO_AUTOMIXER,
IDC_DIRECTSOUND, IDH_AUDIO_DIRECTSOUND,
IDC_START_AUDIO_WIZ, IDH_AUDIO_TUNING_WIZARD,
IDC_ADVANCEDCODEC, IDH_AUDIO_ADVANCED_CODEC_SETTINGS,
IDC_MICSENSE_GROUP, IDH_AUDIO_MIC_SENSITIVITY,
IDC_MICSENSE_AUTO, IDH_AUDIO_AUTO_SENSITIVITY,
IDC_MICSENSE_MANUAL, IDH_AUDIO_MANUAL_SENSITIVITY,
IDC_TRK_MIC, IDH_AUDIO_MANUAL_SENSITIVITY,
0, 0 // terminator
};
VOID InitAudioSettings(HWND hDlg, BOOL *pfFullDuplex, BOOL *pfAgc, BOOL *pfAutoMix, BOOL *pfDirectSound)
{
BOOL fFullDuplex = FALSE;
BOOL fAgc = FALSE;
BOOL fDirectSound;
RegEntry re( AUDIO_KEY, HKEY_CURRENT_USER );
UINT uSoundCardCaps = re.GetNumber(REGVAL_SOUNDCARDCAPS,SOUNDCARD_NONE);
ASSERT(ISSOUNDCARDPRESENT(uSoundCardCaps) && (!SysPol::NoAudio()));
if (ISSOUNDCARDFULLDUPLEX(uSoundCardCaps) && ConfPolicies::IsFullDuplexAllowed())
{
::EnableWindow(::GetDlgItem(hDlg, IDC_FULLDUPLEX), TRUE);
// read settings from registry
fFullDuplex = (BOOL)
( re.GetNumber(REGVAL_FULLDUPLEX,0) == FULLDUPLEX_ENABLED );
}
else
{
DisableControl(hDlg, IDC_FULLDUPLEX);
}
if (DOESSOUNDCARDHAVEAGC(uSoundCardCaps))
{
::EnableWindow(::GetDlgItem(hDlg, IDC_AUTOGAIN), TRUE);
// read settings from registry
fAgc = (BOOL)
( re.GetNumber(REGVAL_AUTOGAIN,AUTOGAIN_ENABLED) == AUTOGAIN_ENABLED );
}
else
{
DisableControl(hDlg, IDC_AUTOGAIN);
}
*pfFullDuplex = fFullDuplex;
// for automix and agc, don't try updating the check-mark if
// NULL has been passed in
if (pfAutoMix)
{
*pfAutoMix = (BOOL)(re.GetNumber(REGVAL_AUTOMIX, AUTOMIX_ENABLED) == AUTOMIX_ENABLED);
SendDlgItemMessage(hDlg, IDC_AUTOMIX, BM_SETCHECK, *pfAutoMix, 0L);
}
if (pfAgc)
{
*pfAgc = fAgc;
SendDlgItemMessage ( hDlg, IDC_AUTOGAIN, BM_SETCHECK, fAgc, 0L );
}
if (ISDIRECTSOUNDAVAILABLE(uSoundCardCaps))
{
RegEntry rePol(POLICIES_KEY, HKEY_CURRENT_USER);
fDirectSound = (BOOL)(re.GetNumber(REGVAL_DIRECTSOUND, DSOUND_USER_DISABLED) == DSOUND_USER_ENABLED);
::EnableWindow(::GetDlgItem(hDlg, IDC_DIRECTSOUND),
!rePol.GetNumber(REGVAL_POL_NOCHANGE_DIRECTSOUND, DEFAULT_POL_NOCHANGE_DIRECTSOUND));
}
else
{
fDirectSound = FALSE;
::EnableWindow(::GetDlgItem(hDlg, IDC_DIRECTSOUND), FALSE);
SendDlgItemMessage(hDlg, IDC_DIRECTSOUND, BM_SETCHECK, FALSE, 0L);
}
// don't check the checkbox if the caller didn't pass in a var to be updated
if (pfDirectSound)
{
*pfDirectSound = fDirectSound;
SendDlgItemMessage(hDlg, IDC_DIRECTSOUND, BM_SETCHECK, *pfDirectSound, 0L);
}
// set the check boxes for those that are enabled
SendDlgItemMessage ( hDlg, IDC_FULLDUPLEX,
BM_SETCHECK, *pfFullDuplex, 0L );
}
INT_PTR APIENTRY AudioDlgProc( HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
static PROPSHEETPAGE * ps;
static UINT uOldCodecChoice;
static UINT uNewCodecChoice;
static BOOL fOldFullDuplex;
static BOOL fOldAgc;
static BOOL fOldAutoMic;
static BOOL fOldAutoMix;
static BOOL fOldDirectSound;
static BOOL fOldH323GatewayEnabled;
static UINT uOldMicSense;
static CODECINFO CodecInfo;
static BOOL bAdvDlg;
switch (message) {
case WM_INITDIALOG:
{
// Save the PROPSHEETPAGE information.
ps = (PROPSHEETPAGE *)lParam;
RegEntry re( AUDIO_KEY, HKEY_CURRENT_USER );
InitAudioSettings(hDlg, &fOldFullDuplex, &fOldAgc, &fOldAutoMix, &fOldDirectSound);
//initialize the codecinfo structure, these will be set as and when needed
ZeroMemory(&CodecInfo, sizeof(CodecInfo));
uNewCodecChoice = uOldCodecChoice = re.GetNumber(REGVAL_CODECCHOICE, CODECCHOICE_AUTO);
// Get Audio settings
fOldAutoMic = (BOOL)
( re.GetNumber(REGVAL_MICROPHONE_AUTO,
DEFAULT_MICROPHONE_AUTO) == MICROPHONE_AUTO_YES );
SendDlgItemMessage ( hDlg,
fOldAutoMic ? IDC_MICSENSE_AUTO : IDC_MICSENSE_MANUAL,
BM_SETCHECK, TRUE, 0L );
EnableWindow ( GetDlgItem ( hDlg, IDC_TRK_MIC ),
(BOOL)SendDlgItemMessage ( hDlg, IDC_MICSENSE_MANUAL,
BM_GETCHECK, 0,0));
SendDlgItemMessage (hDlg, IDC_TRK_MIC, TBM_SETRANGE, FALSE,
MAKELONG (MIN_MICROPHONE_SENSITIVITY,
MAX_MICROPHONE_SENSITIVITY ));
uOldMicSense = re.GetNumber ( REGVAL_MICROPHONE_SENSITIVITY,
DEFAULT_MICROPHONE_SENSITIVITY );
SendDlgItemMessage (hDlg, IDC_TRK_MIC, TBM_SETTICFREQ,
( MAX_MICROPHONE_SENSITIVITY - MIN_MICROPHONE_SENSITIVITY )
/ 10, 0 );
SendDlgItemMessage (hDlg, IDC_TRK_MIC, TBM_SETPOS, TRUE,
uOldMicSense );
bAdvDlg = FALSE;
return (TRUE);
}
case WM_NOTIFY:
switch (((NMHDR FAR *) lParam)->code) {
case PSN_APPLY:
{
RegEntry re( AUDIO_KEY, HKEY_CURRENT_USER );
BOOL fFullDuplex = SendDlgItemMessage
(hDlg, IDC_FULLDUPLEX, BM_GETCHECK, 0, 0 ) ?
FULLDUPLEX_ENABLED : FULLDUPLEX_DISABLED;
BOOL fAgc = SendDlgItemMessage
(hDlg, IDC_AUTOGAIN, BM_GETCHECK, 0, 0 ) ?
AUTOGAIN_ENABLED : AUTOGAIN_DISABLED;
BOOL fAutoMix = SendDlgItemMessage
(hDlg, IDC_AUTOMIX, BM_GETCHECK, 0, 0) ?
AUTOMIX_ENABLED : AUTOMIX_DISABLED;
BOOL fDirectSound = SendDlgItemMessage
(hDlg, IDC_DIRECTSOUND, BM_GETCHECK, 0, 0) ?
DSOUND_USER_ENABLED : DSOUND_USER_DISABLED;
if ( fFullDuplex != fOldFullDuplex )
{
re.SetValue ( REGVAL_FULLDUPLEX, fFullDuplex );
g_dwChangedSettings |= CSETTING_L_FULLDUPLEX;
}
if ( fAgc != fOldAgc )
{
re.SetValue ( REGVAL_AUTOGAIN, fAgc );
g_dwChangedSettings |= CSETTING_L_AGC;
}
// use same flag bit as agc as for automix
// since this automix and agc may evenutally
// get combined into 1 ui choice
if (fAutoMix != fOldAutoMix)
{
re.SetValue(REGVAL_AUTOMIX, fAutoMix);
g_dwChangedSettings |= CSETTING_L_AGC;
}
if (fDirectSound != fOldDirectSound)
{
re.SetValue(REGVAL_DIRECTSOUND, fDirectSound);
g_dwChangedSettings |= CSETTING_L_DIRECTSOUND;
}
UINT uBandWidth = re.GetNumber(REGVAL_TYPICALBANDWIDTH,BW_DEFAULT);
// if the advanced dialog has not been accessed,
// then there is nothing to changes as far as the
// codecs go
if (bAdvDlg)
{
if (uNewCodecChoice == CODECCHOICE_AUTO)
{
//apply heuristics and apply the prefs to registry
ChooseCodecByBw(uBandWidth, &CodecInfo);
SetAppCodecPrefs(&CodecInfo, uBandWidth);
}
else
{
for (int i = 0; i < (int)CodecInfo.uNumFormats &&
CodecInfo.pCodecCapList; i++)
{
if (CodecInfo.pCodecCapList[i].wSortIndex !=
CodecInfo.pOldCodecOrderList[i])
{
// oder has changed, save the new order
SetAppCodecPrefs(&CodecInfo, uBandWidth);
break;
}
}
}
}
FreeAppCapFormats(&CodecInfo);
// Handle the Trackbar controls:
BOOL fAutoMic = (BOOL)SendDlgItemMessage ( hDlg, IDC_MICSENSE_AUTO,
BM_GETCHECK, 0, 0 );
if ( fAutoMic != fOldAutoMic ) {
g_dwChangedSettings |= CSETTING_L_AUTOMIC;
re.SetValue ( REGVAL_MICROPHONE_AUTO, fAutoMic ?
MICROPHONE_AUTO_YES : MICROPHONE_AUTO_NO );
}
UINT uMicSense = (UINT)SendDlgItemMessage( hDlg, IDC_TRK_MIC,
TBM_GETPOS, 0, 0 );
if ( uMicSense != uOldMicSense ) {
g_dwChangedSettings |= CSETTING_L_MICSENSITIVITY;
re.SetValue ( REGVAL_MICROPHONE_SENSITIVITY, uMicSense);
}
break;
}
case PSN_RESET:
//reset the codec choice in the registry if it has changed
if (uNewCodecChoice != uOldCodecChoice)
{
RegEntry re( AUDIO_KEY, HKEY_CURRENT_USER );
re.SetValue(REGVAL_CODECCHOICE, uOldCodecChoice);
}
//free the capformats if allocated
FreeAppCapFormats(&CodecInfo);
break;
}
break;
case WM_COMMAND:
switch (GET_WM_COMMAND_ID (wParam, lParam))
{
case IDC_START_AUDIO_WIZ:
{
AUDIOWIZOUTPUT AwOutput;
BOOL fFullDuplex, fAgc, fDirectSound, fCurrentDS;
if (::FIsConferenceActive())
{
ConfMsgBox(hDlg,
(LPCTSTR)IDS_NOAUDIOTUNING);
break;
}
fAgc = SendDlgItemMessage
(hDlg, IDC_AUTOGAIN, BM_GETCHECK, 0, 0 ) ?
AUTOGAIN_ENABLED : AUTOGAIN_DISABLED;
//SS:note the calibrated value can get out of sync
//need a warning
CallAudioCalibWizard(hDlg,RUNDUE_USERINVOKED,WAVE_MAPPER,&AwOutput,(INT)fAgc);
if (AwOutput.uChanged & SOUNDCARD_CHANGED)
g_dwChangedSettings |= CSETTING_L_AUDIODEVICE;
// wizard can change duplex, and agc availability, and DirectSound
// don't pass in AGC and AUTOMIX pointers, because we don't want
// the checkboxes to revet back to what they were initialized at.
InitAudioSettings(hDlg, &fFullDuplex, NULL, NULL, NULL);
break;
}
case IDC_ADVANCEDCODEC:
{
RegEntry re( AUDIO_KEY, HKEY_CURRENT_USER );
//SS ::: get the cap formats,
//this will retrieve the codec caps if not allready rerieved
if (GetAppCapFormats(&CodecInfo))
{
ADVCODEC AdvCodec;
AdvCodec.fManual = (uNewCodecChoice == CODECCHOICE_MANUAL);
AdvCodec.uBandwidth = re.GetNumber(REGVAL_TYPICALBANDWIDTH, BW_DEFAULT);
AdvCodec.pCodecInfo = &CodecInfo;
if ((DialogBoxParam(GetInstanceHandle(),MAKEINTRESOURCE(IDD_ADVANCEDCODEC),hDlg,AdvCodecDlgProc,
(LPARAM)&AdvCodec)) == IDCANCEL)
{
//the dialog box changes it in place so current settings need to be restored into
//update the pcodecinfo sort order from the previous current settings
for (int i = 0; i < (int)CodecInfo.uNumFormats; i++)
(CodecInfo.pCodecCapList[i]).wSortIndex = CodecInfo.pCurCodecOrderList[i];
}
else
{
uNewCodecChoice = (AdvCodec.fManual ?
CODECCHOICE_MANUAL : CODECCHOICE_AUTO);
re.SetValue(REGVAL_CODECCHOICE, uNewCodecChoice);
//else update the current
for (int i = 0; i < (int)CodecInfo.uNumFormats; i++)
CodecInfo.pCurCodecOrderList[i] = (CodecInfo.pCodecCapList[i]).wSortIndex;
bAdvDlg = TRUE;
}
}
break;
}
case IDC_MICSENSE_AUTO:
case IDC_MICSENSE_MANUAL:
EnableWindow ( GetDlgItem ( hDlg, IDC_TRK_MIC ),
(BOOL)SendDlgItemMessage ( hDlg, IDC_MICSENSE_MANUAL,
BM_GETCHECK, 0,0));
break;
default:
break;
}
break;
case WM_CONTEXTMENU:
DoHelpWhatsThis(wParam, _rgHelpIdsAudio);
break;
case WM_HELP:
DoHelp(lParam, _rgHelpIdsAudio);
break;
}
return (FALSE);
}
static const DWORD aAdvCodecHelpIds[] = {
IDC_CODECMANUAL, IDH_AUDIO_MANUAL_CODEC_SETTINGS,
IDC_COMBO_CODEC, IDH_ADVCOMP_CODECS,
IDC_CODECDEFAULT, IDH_ADVCOMP_DEFAULTS,
IDC_CODECLISTLABEL, IDH_ADVCOMP_CODECS,
0, 0 // terminator
};
INT_PTR CALLBACK AdvCodecDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
static PADVCODEC pAdvCodec = NULL;
static nCodecSelection=0;
WORD wCmdId;
switch(uMsg)
{
case WM_INITDIALOG:
//get the list of codecs
pAdvCodec = (PADVCODEC)lParam;
ASSERT(pAdvCodec);
SendDlgItemMessage ( hDlg, IDC_CODECMANUAL,
BM_SETCHECK, pAdvCodec->fManual, 0L );
EnableWindow(GetDlgItem(hDlg, IDC_COMBO_CODEC), pAdvCodec->fManual);
if (!pAdvCodec->fManual)
{
ChooseCodecByBw(pAdvCodec->uBandwidth, pAdvCodec->pCodecInfo);
}
//fill the list box;
FillCodecListView(GetDlgItem(hDlg, IDC_COMBO_CODEC),
pAdvCodec->pCodecInfo);
if (pAdvCodec->fManual)
{
nCodecSelection = (int)SendMessage((HWND)lParam, CB_GETCURSEL, 0,0);
// convert index of combo box choice to index in capability list
nCodecSelection = (int)SendMessage((HWND)lParam, CB_GETITEMDATA, nCodecSelection,0);
}
return(TRUE);
case WM_CONTEXTMENU:
DoHelpWhatsThis(wParam, aAdvCodecHelpIds);
break;
case WM_HELP:
DoHelp(lParam, aAdvCodecHelpIds);
break;
case WM_COMMAND:
wCmdId = GET_WM_COMMAND_ID (wParam, lParam); // LOWORD (uParam)
switch (wCmdId)
{
case IDOK:
if (pAdvCodec->fManual)
{
nCodecSelection = (int)SendDlgItemMessage(hDlg, IDC_COMBO_CODEC, CB_GETCURSEL, 0,0);
nCodecSelection = (int)SendDlgItemMessage(hDlg, IDC_COMBO_CODEC, CB_GETITEMDATA, nCodecSelection,0);
SortCodecs(pAdvCodec, nCodecSelection);
}
EndDialog (hDlg, IDOK);
return(TRUE);
case IDCANCEL:
EndDialog(hDlg, IDCANCEL);
return(TRUE);
case IDC_CODECMANUAL:
pAdvCodec->fManual = (BOOL)SendDlgItemMessage ( hDlg, IDC_CODECMANUAL,
BM_GETCHECK, 0,0);
EnableWindow(GetDlgItem(hDlg, IDC_COMBO_CODEC), pAdvCodec->fManual);
if (pAdvCodec->fManual)
{
break;
}
else
{
ChooseCodecByBw(pAdvCodec->uBandwidth, pAdvCodec->pCodecInfo);
FillCodecListView(GetDlgItem(hDlg, IDC_COMBO_CODEC),
pAdvCodec->pCodecInfo);
}
break;
} // WM_COMMAND
default:
return FALSE;
}
return (FALSE);
}
BOOL FillCodecListView(HWND hCB, PCODECINFO pCodecInfo)
{
PBASIC_AUDCAP_INFO pCurCodecCap;
UINT uIndex, uTotal, uItemIndex;
UINT uSortTop, uSortTopIndex;
SendMessage(hCB, CB_RESETCONTENT,0,0); // erase all items in CB
uTotal = pCodecInfo->uNumFormats;
uSortTop = uTotal-1;
for (uIndex = 0; uIndex < uTotal; uIndex++)
{
pCurCodecCap = &(pCodecInfo->pCodecCapList[uIndex]);
if ((!pCurCodecCap->bRecvEnabled) && (!pCurCodecCap->bSendEnabled))
{
continue;
}
uItemIndex = (UINT)SendMessage(hCB, CB_ADDSTRING, 0, (LPARAM)(pCurCodecCap->szFormat));
SendMessage(hCB, CB_SETITEMDATA, uItemIndex, uIndex);
if (pCurCodecCap->wSortIndex < uSortTop)
{
uSortTop = pCurCodecCap->wSortIndex;
uSortTopIndex = uItemIndex;
}
}
SendMessage(hCB, CB_SETCURSEL, uSortTopIndex, 0);
return TRUE;
}
void SortCodecs(PADVCODEC pAdvCodec, int nSelection)
{
PBASIC_AUDCAP_INFO pDefaultCodec, pSelectedCodec, pCodec;
WORD wSortOrderBest, wSortOrderSelected;
UINT uIndex;
if (!pAdvCodec->fManual)
return;
if ((nSelection < 0) || ((UINT)nSelection > (pAdvCodec->pCodecInfo->uNumFormats)))
{
return;
}
ChooseCodecByBw(pAdvCodec->uBandwidth, pAdvCodec->pCodecInfo);
// this is the codec the user selected
pSelectedCodec = &(pAdvCodec->pCodecInfo->pCodecCapList[nSelection]);
// all codecs that have a sort index less than the selected codec
// get moved down one
for (uIndex = 0; uIndex < pAdvCodec->pCodecInfo->uNumFormats; uIndex++)
{
pCodec = &(pAdvCodec->pCodecInfo->pCodecCapList[uIndex]);
if (pCodec->wSortIndex < pSelectedCodec->wSortIndex)
{
pCodec->wSortIndex = pCodec->wSortIndex + 1;
}
}
pSelectedCodec->wSortIndex = 0;
}
#define SQCIF 0x1
#define QCIF 0x2
#define CIF 0x4
#define UNKNOWN 0x8
#define get_format(s) (s == Small ? SQCIF : (s == Medium ? QCIF: (s == Large ? CIF : UNKNOWN)))
BOOL SetAppCodecPrefs(PCODECINFO pCodecInfo,UINT uBandwidth)
{
BOOL fRet = FALSE;
HRESULT hr;
DWORD dwcFormats,dwcFormatsReturned;
BASIC_VIDCAP_INFO *pvidcaps = NULL;
UINT dwBitsPerSec,x;
int iFamily,format, nNormalizedSpeed;
int nCifIncrease;
DWORD dwSysPolBandwidth;
// frame rates initialized to 486/P60 settings
UINT uRateCIF=CIF_RATE_VERYSLOW, uRateQCIF=QCIF_RATE_VERYSLOW, uRateSQCIF=SQCIF_RATE_VERYSLOW;
if (!pCodecInfo->lpIAppCap)
{
if (!GetIAppCap(pCodecInfo))
goto MyExit;
}
if (FAILED(hr = ((pCodecInfo->lpIAppCap)->ApplyAppFormatPrefs(pCodecInfo->pCodecCapList,
pCodecInfo->uNumFormats))))
goto MyExit;
//Set the BW to something sane
#ifdef _M_IX86
GetNormalizedCPUSpeed (&nNormalizedSpeed,&iFamily);
#else
//BUGBUG, setting things really high, otherwise,
iFamily=6;
nNormalizedSpeed=300;
#endif
dwBitsPerSec = GetBandwidthBits(uBandwidth,nNormalizedSpeed);
dwSysPolBandwidth = SysPol::GetMaximumBandwidth();
// apply bandwidth policy key if the user's setting is set to
// a LAN speed
if ((dwSysPolBandwidth > 0) && (dwBitsPerSec >= BW_SLOWLAN_BITS))
{
dwBitsPerSec = dwSysPolBandwidth;
}
if ((iFamily >= 5) && (nNormalizedSpeed >= SLOW_CPU_MHZ))
{
// normal pentiums (75mhz - 180mhz)
if (nNormalizedSpeed < FAST_CPU_MHZ)
{
uRateCIF = CIF_RATE_SLOW;
uRateQCIF = QCIF_RATE_SLOW;
uRateSQCIF= SQCIF_RATE_SLOW;
}
// pentiums between 200-350 mhz
else if (nNormalizedSpeed < VERYFAST_CPU_MHZ)
{
uRateCIF = CIF_RATE_FAST;
uRateQCIF = QCIF_RATE_FAST;
uRateSQCIF = SQCIF_RATE_FAST;
}
// really fast pentiums (400mhz and greater)
else
{
// it would be better if we could scale between 15 and 30 frames/sec
// depending on the CPU speed. But H.245 doesn't have any values
// between 15 and 30. (See definition of Minimum Picture Interval)
// So for now, 30 frames per sec CIF for all 400mhz and faster machines
uRateCIF = CIF_RATE_VERYFAST;
uRateQCIF = QCIF_RATE_FAST;
uRateSQCIF = SQCIF_RATE_FAST;
}
}
// Get the number of BASIC_VIDCAP_INFO structures available
if (pCodecInfo->lpIVidAppCap->GetNumFormats((UINT*)&dwcFormats) != S_OK)
goto MyExit;
if (dwcFormats > 0)
{
// Allocate some memory to hold the list in
if (!(pvidcaps = (BASIC_VIDCAP_INFO*)LocalAlloc(LPTR,dwcFormats * sizeof (BASIC_VIDCAP_INFO))))
goto MyExit;
// Get the list
if (pCodecInfo->lpIVidAppCap->EnumFormats(pvidcaps, dwcFormats * sizeof (BASIC_VIDCAP_INFO),
(UINT*)&dwcFormatsReturned) != S_OK)
goto MyExit;
//Setup the Bandwitdh, and the frame rate (according to philf's #s)
for (x=0;x<dwcFormatsReturned;x++)
{
// If the codec is "hardware accelerated" (0 for cpu usage), then we assume
// that the maximum bitrate is whatever is specified at install
// time.
pvidcaps[x].uMaxBitrate=dwBitsPerSec;
if (pvidcaps[x].wCPUUtilizationEncode == 0)
{
// hardware acceleration - don't change the frame rate
continue;
}
format=get_format (pvidcaps[x].enumVideoSize);
//Which format
switch (format)
{
case SQCIF:
pvidcaps[x].uFrameRate=uRateSQCIF;
break;
case QCIF:
pvidcaps[x].uFrameRate=uRateQCIF;
break;
case CIF:
pvidcaps[x].uFrameRate=uRateCIF;
break;
default:
WARNING_OUT(("Incorrect Frame size discovered\r\n"));
pvidcaps[x].uFrameRate=uRateCIF;
}
}
}
// Ok, now submit this list
if (pCodecInfo->lpIVidAppCap->ApplyAppFormatPrefs(pvidcaps, dwcFormats) != S_OK)
goto MyExit;
// Free the memory, we're done
LocalFree(pvidcaps);
// Set the bandwidth limit, which rebuilds capability sets
hr = pCodecInfo->lpIH323->SetMaxPPBandwidth(dwBitsPerSec);
fRet = !FAILED(hr);
MyExit:
return (fRet);
}
BOOL GetIAppCap(PCODECINFO pCodecInfo)
{
BOOL fRet=TRUE;
HRESULT hr;
if (pCodecInfo->lpIAppCap && pCodecInfo->lpIVidAppCap && pCodecInfo->lpICapsCtl)
return TRUE;
if(NULL == pCodecInfo->lpIH323)
{
IH323CallControl *pIH323 = NULL;
pCodecInfo->hLib = ::NmLoadLibrary(H323DLL);
if ((pCodecInfo->hLib) == NULL)
{
WARNING_OUT(("LoadLibrary(H323DLL) failed"));
return FALSE;
}
CREATEH323CC pfnCreateH323 = (CREATEH323CC) ::GetProcAddress(pCodecInfo->hLib, SZ_FNCREATEH323CC);
if (pfnCreateH323 == NULL)
{
ERROR_OUT(("GetProcAddress(CreateH323) failed"));
return FALSE;
}
hr = pfnCreateH323(&pIH323, FALSE, 0);
if (FAILED(hr))
{
ERROR_OUT(("CreateH323 failed, hr=0x%lx", hr));
return FALSE;
}
pCodecInfo->lpIH323 = pIH323;
}
else
{
// going to Release() later, so AddRef()
pCodecInfo->lpIH323->AddRef();
}
if(!pCodecInfo->lpIAppCap)
{
hr = pCodecInfo->lpIH323->QueryInterface(IID_IAppAudioCap, (void **)&(pCodecInfo->lpIAppCap));
if (FAILED(hr))
{
fRet=FALSE;
goto MyExit;
}
}
if(!pCodecInfo->lpIVidAppCap)
{
hr = pCodecInfo->lpIH323->QueryInterface(IID_IAppVidCap, (void **)&(pCodecInfo->lpIVidAppCap));
if (FAILED(hr))
{
fRet=FALSE;
goto MyExit;
}
}
if(!pCodecInfo->lpICapsCtl)
{
hr = pCodecInfo->lpIH323->QueryInterface(IID_IDualPubCap, (void **)&(pCodecInfo->lpICapsCtl));
if (FAILED(hr))
{
fRet=FALSE;
goto MyExit;
}
}
MyExit:
if (!fRet)
{
ReleaseIAppCap(pCodecInfo);
}
return (fRet);
}
BOOL GetAppCapFormats(PCODECINFO pCodecInfo)
{
BOOL fRet = FALSE;
UINT uNumFormats = 0;
HRESULT hr;
int i;
if (pCodecInfo->pCodecCapList)
return TRUE;
if (!pCodecInfo->lpIAppCap)
{
if (!GetIAppCap(pCodecInfo))
goto MyExit;
}
if (FAILED(hr = ((pCodecInfo->lpIAppCap)->GetNumFormats(&uNumFormats))))
goto MyExit;
if (!uNumFormats)
goto MyExit;
if (!(pCodecInfo->pCodecCapList = (PBASIC_AUDCAP_INFO)LocalAlloc
(LPTR,uNumFormats * sizeof(BASIC_AUDCAP_INFO))))
goto MyExit;
if (!(pCodecInfo->pOldCodecOrderList = (PWORD)LocalAlloc
(LPTR,uNumFormats * sizeof(WORD))))
goto MyExit;
if (!(pCodecInfo->pCurCodecOrderList = (PWORD)LocalAlloc
(LPTR,uNumFormats * sizeof(WORD))))
goto MyExit;
if (FAILED(hr = ((pCodecInfo->lpIAppCap)->EnumFormats(pCodecInfo->pCodecCapList,
uNumFormats * sizeof(BASIC_AUDCAP_INFO), &(pCodecInfo->uNumFormats)))))
{
//free the memory
LocalFree(pCodecInfo->pCodecCapList);
pCodecInfo->pCodecCapList = NULL;
LocalFree(pCodecInfo->pOldCodecOrderList);
pCodecInfo->pOldCodecOrderList = NULL;
LocalFree(pCodecInfo->pCurCodecOrderList);
pCodecInfo->pCurCodecOrderList = NULL;
pCodecInfo->uNumFormats = 0;
goto MyExit;
}
// initialize the old and the current state
for (i=0; i<(int)pCodecInfo->uNumFormats;i++)
{
pCodecInfo->pCurCodecOrderList[i] = pCodecInfo->pCodecCapList[i].wSortIndex;
pCodecInfo->pOldCodecOrderList[i] = pCodecInfo->pCodecCapList[i].wSortIndex;
//SS:hack if we dont have the average, use the max as average
if (!(pCodecInfo->pCodecCapList[i].uAvgBitrate))
pCodecInfo->pCodecCapList[i].uAvgBitrate = pCodecInfo->pCodecCapList[i].uMaxBitrate;
}
fRet = TRUE;
MyExit:
if (!fRet)
{
FreeAppCapFormats(pCodecInfo);
}
return (fRet);
}
void FreeAppCapFormats(PCODECINFO pCodecInfo)
{
if (pCodecInfo->pCodecCapList)
{
LocalFree(pCodecInfo->pCodecCapList);
pCodecInfo->pCodecCapList = NULL;
pCodecInfo->uNumFormats = 0;
}
if (pCodecInfo->pOldCodecOrderList)
{
LocalFree(pCodecInfo->pOldCodecOrderList);
pCodecInfo->pOldCodecOrderList = NULL;
}
if (pCodecInfo->pCurCodecOrderList)
{
LocalFree(pCodecInfo->pCurCodecOrderList);
pCodecInfo->pCurCodecOrderList = NULL;
}
ReleaseIAppCap(pCodecInfo);
}
void ReleaseIAppCap(PCODECINFO pCodecInfo)
{
if(pCodecInfo->lpIAppCap)
{
pCodecInfo->lpIAppCap->Release();
pCodecInfo->lpIAppCap = NULL;
}
if(pCodecInfo->lpIVidAppCap)
{
pCodecInfo->lpIVidAppCap->Release();
pCodecInfo->lpIVidAppCap = NULL;
}
if(pCodecInfo->lpICapsCtl)
{
pCodecInfo->lpICapsCtl->Release();
pCodecInfo->lpICapsCtl = NULL;
}
if(pCodecInfo->lpIH323)
{
pCodecInfo->lpIH323->Release();
pCodecInfo->lpIH323 = NULL;
}
if (pCodecInfo->hLib)
{
FreeLibrary(pCodecInfo->hLib);
pCodecInfo->hLib = NULL;
}
}
static void IsCodecDisabled(WORD wFormatTag, int cpu_family,
UINT uBandwidthID, BOOL *pbSendEnabled, BOOL *pbRecvEnabled)
{
int index;
int size = sizeof(g_CodecPrefTable)/sizeof(CODECPREFROW);
for (index = 0; index < size; index++)
{
if (g_CodecPrefTable[index].wFormatTag == wFormatTag)
{
// does bandwidth limit the use of this codec ?
if (uBandwidthID < g_CodecPrefTable[index].wMinBW)
{
*pbSendEnabled = FALSE;
*pbRecvEnabled = FALSE;
return;
}
*pbRecvEnabled = TRUE; // all codecs can decode on 486
if ((cpu_family <= 4) &&
(CODEC_DISABLED == g_CodecPrefTable[index].wOrder486))
{
*pbSendEnabled = FALSE;
}
// otherwise, the codec can be used for sending and receiving
else
{
*pbSendEnabled = TRUE;
}
return;
}
}
WARNING_OUT(("Audiocpl.cpp:IsCodecDisabled - Unknown Codec!"));
// it may be unknown, but enable it anyway
*pbSendEnabled = TRUE;
*pbRecvEnabled = TRUE;
return;
}
static int GetCodecPrefOrder(WORD wFormatTag, int cpu_family)
{
int index;
int size = sizeof(g_CodecPrefTable)/sizeof(CODECPREFROW);
for (index = 0; index < size; index++)
{
if (g_CodecPrefTable[index].wFormatTag == wFormatTag)
{
if (cpu_family > 4)
{
return g_CodecPrefTable[index].wOrder586;
}
else
{
return g_CodecPrefTable[index].wOrder486;
}
}
}
WARNING_OUT(("Audiocpl.cpp:GetCodecPrefOrder - Unknown Codec!"));
// this will put the codec at the bottom of the list
return CODEC_UNKNOWN;
}
// called by the sort routine that is within ChooseCodecByBW()
// returns -1 if v1 is more prefered. +1 if v2 is more preferred over
// v1. Returns 0 on tie.
static int codec_compare(const void *v1, const void *v2, int nCpuFamily)
{
PBASIC_AUDCAP_INFO pCap1, pCap2;
pCap1 = (PBASIC_AUDCAP_INFO)v1;
pCap2 = (PBASIC_AUDCAP_INFO)v2;
int pref1, pref2;
// get preference order
// if we can't send with this codec, we compare it as disabled
// so that it appears on the bottom of the Codec Selection UI list
if (pCap1->bSendEnabled == TRUE)
pref1 = GetCodecPrefOrder(pCap1->wFormatTag, nCpuFamily);
else
pref1 = CODEC_DISABLED;
if (pCap2->bSendEnabled == TRUE)
pref2 = GetCodecPrefOrder(pCap2->wFormatTag, nCpuFamily);
else
pref2 = CODEC_DISABLED;
if (pref1 < pref2)
return -1;
if (pref1 > pref2)
return 1;
// pref1==pref2
// special case, G723.1 has two formats. Higher bitrate is prefered
if (pCap1->wFormatTag == WAVE_FORMAT_MSG723)
{
if (pCap1->uMaxBitrate < pCap2->uMaxBitrate)
return 1;
else
return -1;
}
return 0;
}
BOOL ChooseCodecByBw(UINT uBandWidthId, PCODECINFO pCodecInfo)
{
PBASIC_AUDCAP_INFO pCodecCapList; // list of capabilities
#ifdef _M_IX86
int nNormalizedSpeed;
#endif
int nFamily;
UINT uNumFormats, uNumDisabled = 0;
UINT index;
BOOL bSendEnabled, bRecvEnabled, bCompletelySorted;
int *OrderList, ret, temp;
//Fill out the PCodecInfo Structure
if (!GetAppCapFormats(pCodecInfo))
return FALSE;
if (NULL == pCodecInfo->pCodecCapList)
return FALSE;
pCodecCapList = pCodecInfo->pCodecCapList;
uNumFormats = pCodecInfo->uNumFormats;
// figure out what type of CPU we have
#ifdef _M_IX86
GetNormalizedCPUSpeed (&nNormalizedSpeed,&nFamily);
// A Pentium with FP emumalation is assumed to be a 486
if (TRUE == IsFloatingPointEmulated())
{
nFamily = 4;
}
#else
// assume a DEC Alpha is a really fast pentium
nFamily=5;
#endif
// scan once to see which codecs should be disabled
for (index=0; index < uNumFormats; index++)
{
IsCodecDisabled(pCodecCapList[index].wFormatTag,
nFamily,uBandWidthId, &bSendEnabled, &bRecvEnabled);
pCodecCapList[index].bSendEnabled = bSendEnabled;
pCodecCapList[index].bRecvEnabled = bRecvEnabled;
}
// sort the capability list based on the preference table
OrderList = (int *)LocalAlloc(LPTR, sizeof(int)*uNumFormats);
if (OrderList == NULL)
return FALSE;
for (index = 0; index < uNumFormats; index++)
{
OrderList[index] = index;
}
// bubble sort routine
do
{
bCompletelySorted = TRUE;
for (index=0; index < (uNumFormats-1); index++)
{
ret = codec_compare(&pCodecCapList[OrderList[index]],
&pCodecCapList[OrderList[index+1]], nFamily);
if (ret > 0)
{
// swap
temp = OrderList[index];
OrderList[index] = OrderList[index+1];
OrderList[index+1] = temp;
bCompletelySorted = FALSE;
}
}
} while (bCompletelySorted == FALSE);
// set the selection index from the result of the sort
for (index = 0; index < uNumFormats; index++)
{
pCodecCapList[OrderList[index]].wSortIndex = (WORD)index;
}
LocalFree(OrderList);
return TRUE;
}
// this is called by the audio tuning wizard
VOID SaveDefaultCodecSettings(UINT uBandWidth)
{
CODECINFO CodecInfo;
ZeroMemory(&CodecInfo, sizeof(CodecInfo));
RegEntry re(AUDIO_KEY, HKEY_CURRENT_USER);
re.SetValue(REGVAL_CODECCHOICE, CODECCHOICE_AUTO);
ChooseCodecByBw(uBandWidth, &CodecInfo);
//set the ordering
SetAppCodecPrefs(&CodecInfo, uBandWidth);
FreeAppCapFormats(&CodecInfo);
}
// this is called by the general property page
VOID UpdateCodecSettings(UINT uBandWidth)
{
RegEntry re(AUDIO_KEY, HKEY_CURRENT_USER);
if (re.GetNumber(REGVAL_CODECCHOICE, CODECCHOICE_AUTO) ==
CODECCHOICE_AUTO)
{
CODECINFO CodecInfo;
ZeroMemory(&CodecInfo, sizeof(CodecInfo));
ChooseCodecByBw(uBandWidth, &CodecInfo);
//set the ordering
SetAppCodecPrefs(&CodecInfo, uBandWidth);
FreeAppCapFormats(&CodecInfo);
}
}