|
|
//
// Driver Verifier UI
// Copyright (c) Microsoft Corporation, 1999
//
//
//
// module: GCntPage.cpp
// author: DMihai
// created: 11/1/00
//
// Description:
//
#include "stdafx.h"
#include "verifier.h"
#include "GCntPage.h"
#include "VrfUtil.h"
#include "VGlobal.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__; #endif
//
// Timer ID
//
#define REFRESH_TIMER_ID 0x1234
//
// Help IDs
//
static DWORD MyHelpIds[] = { IDC_GLOBC_LIST, IDH_DV_GlobalCounters, 0, 0 };
/////////////////////////////////////////////////////////////////////////////
// CGlobalCountPage property page
IMPLEMENT_DYNCREATE(CGlobalCountPage, CVerifierPropertyPage)
CGlobalCountPage::CGlobalCountPage() : CVerifierPropertyPage(CGlobalCountPage::IDD) { //{{AFX_DATA_INIT(CGlobalCountPage)
//}}AFX_DATA_INIT
m_nSortColumnIndex = 0; m_bAscendSortName = FALSE; m_bAscendSortValue = FALSE;
m_uTimerHandler = 0;
m_pParentSheet = NULL; }
CGlobalCountPage::~CGlobalCountPage() { }
void CGlobalCountPage::DoDataExchange(CDataExchange* pDX) { static BOOL bShownPoolCoverageWarning = FALSE;
if( ! pDX->m_bSaveAndValidate ) { //
// Query the kernel
//
VrfGetRuntimeVerifierData( &m_RuntimeVerifierData );
if( FALSE == bShownPoolCoverageWarning ) { bShownPoolCoverageWarning = CheckAndShowPoolCoverageWarning(); } }
CVerifierPropertyPage::DoDataExchange(pDX); //{{AFX_DATA_MAP(CGlobalCountPage)
DDX_Control(pDX, IDC_GLOBC_LIST, m_CountersList); DDX_Control(pDX, IDC_GLOBC_NEXT_DESCR_STATIC, m_NextDescription); //}}AFX_DATA_MAP
}
BEGIN_MESSAGE_MAP(CGlobalCountPage, CVerifierPropertyPage) //{{AFX_MSG_MAP(CGlobalCountPage)
ON_WM_TIMER() ON_NOTIFY(LVN_COLUMNCLICK, IDC_GLOBC_LIST, OnColumnclickGlobcList) ON_WM_CONTEXTMENU() ON_MESSAGE( WM_HELP, OnHelp ) //}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::SetupListHeader() { LVCOLUMN lvColumn; CRect rectWnd; CString strCounter, strValue; VERIFY( strCounter.LoadString( IDS_COUNTER ) ); VERIFY( strValue.LoadString( IDS_VALUE ) );
//
// List's regtangle
//
m_CountersList.GetClientRect( &rectWnd ); //
// Column 0 - counter
//
ZeroMemory( &lvColumn, sizeof( lvColumn ) ); lvColumn.mask = LVCF_FMT | LVCF_SUBITEM | LVCF_TEXT | LVCF_WIDTH; lvColumn.fmt = LVCFMT_LEFT; lvColumn.iSubItem = 0; lvColumn.pszText = strCounter.GetBuffer( strCounter.GetLength() + 1 ); lvColumn.cx = (int)( rectWnd.Width() * 0.50 ); VERIFY( m_CountersList.InsertColumn( 0, &lvColumn ) != -1 ); strCounter.ReleaseBuffer();
//
// Column 1
//
lvColumn.iSubItem = 1; lvColumn.pszText = strValue.GetBuffer( strValue.GetLength() + 1 ); lvColumn.cx = (int)( rectWnd.Width() * 0.47 ); VERIFY( m_CountersList.InsertColumn( 1, &lvColumn ) != -1 ); strValue.ReleaseBuffer(); }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::FillTheList() { //
// N.B.
//
// If you change the first parameter (index - stored in the item's data)
// you need to change the switch statement in GetCounterValue as well
//
AddCounterInList( 0, IDS_ALLOCATIONSATTEMPTED_LIST, m_RuntimeVerifierData.AllocationsAttempted ); AddCounterInList( 1, IDS_ALLOCATIONSSUCCEEDED_LIST, m_RuntimeVerifierData.AllocationsSucceeded ); AddCounterInList( 2, IDS_ALLOCATIONSSUCCEEDEDSPECIALPOOL_LIST, m_RuntimeVerifierData.AllocationsSucceededSpecialPool ); AddCounterInList( 3, IDS_ALLOCATIONSWITHNOTAG_LIST, m_RuntimeVerifierData.AllocationsWithNoTag ); AddCounterInList( 4, IDS_UNTRACKEDPOOL_LIST, m_RuntimeVerifierData.UnTrackedPool ); AddCounterInList( 5, IDS_ALLOCATIONSFAILED_LIST, m_RuntimeVerifierData.AllocationsFailed ); AddCounterInList( 6, IDS_ALLOCATIONSFAILEDDELIBERATELY_LIST, m_RuntimeVerifierData.AllocationsFailedDeliberately ); AddCounterInList( 7, IDS_RAISEIRQLS_LIST, m_RuntimeVerifierData.RaiseIrqls ); AddCounterInList( 8, IDS_ACQUIRESPINLOCKS_LIST, m_RuntimeVerifierData.AcquireSpinLocks ); AddCounterInList( 9, IDS_SYNCHRONIZEEXECUTIONS_LIST, m_RuntimeVerifierData.SynchronizeExecutions ); AddCounterInList( 10, IDS_TRIMS_LIST, m_RuntimeVerifierData.Trims ); }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::RefreshTheList() { INT nListItems; INT nCrtListItem; INT_PTR nCrtCounterIndex; SIZE_T sizeValue; nListItems = m_CountersList.GetItemCount();
for( nCrtListItem = 0; nCrtListItem < nListItems; nCrtListItem += 1 ) { nCrtCounterIndex = m_CountersList.GetItemData( nCrtListItem );
sizeValue = GetCounterValue( nCrtCounterIndex );
UpdateCounterValueInList( nCrtListItem, sizeValue ); } }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::SortTheList() { if( 0 != m_nSortColumnIndex ) { //
// Sort by counter value - this is probably not very useful
// but we are providing it to be consistent with all
// the lists being sortable by any column
//
m_CountersList.SortItems( CounterValueCmpFunc, (LPARAM)this ); } else { //
// Sort by driver name
//
m_CountersList.SortItems( CounterNameCmpFunc, (LPARAM)this ); } }
/////////////////////////////////////////////////////////////
SIZE_T CGlobalCountPage::GetCounterValue( INT_PTR nCounterIndex ) { SIZE_T sizeValue;
//
// N.B.
//
// If you change this switch statement you need to change FillTheList as well
//
switch( nCounterIndex ) { case 0: sizeValue = m_RuntimeVerifierData.AllocationsAttempted; break;
case 1: sizeValue = m_RuntimeVerifierData.AllocationsSucceeded; break;
case 2: sizeValue = m_RuntimeVerifierData.AllocationsSucceededSpecialPool; break;
case 3: sizeValue = m_RuntimeVerifierData.AllocationsWithNoTag; break;
case 4: sizeValue = m_RuntimeVerifierData.UnTrackedPool; break;
case 5: sizeValue = m_RuntimeVerifierData.AllocationsFailed; break;
case 6: sizeValue = m_RuntimeVerifierData.AllocationsFailedDeliberately; break;
case 7: sizeValue = m_RuntimeVerifierData.RaiseIrqls; break;
case 8: sizeValue = m_RuntimeVerifierData.AcquireSpinLocks; break;
case 9: sizeValue = m_RuntimeVerifierData.SynchronizeExecutions; break;
case 10: sizeValue = m_RuntimeVerifierData.Trims; break;
default: //
// Oops, how did we get here ?!?
//
ASSERT( FALSE );
sizeValue = 0;
break; }
return sizeValue; }
/////////////////////////////////////////////////////////////
BOOL CGlobalCountPage::GetCounterName( LPARAM lItemData, TCHAR *szCounterName, ULONG uCounterNameBufferLen ) { INT nItemIndex; BOOL bResult; LVFINDINFO FindInfo; LVITEM lvItem;
bResult = FALSE;
ZeroMemory( &FindInfo, sizeof( FindInfo ) ); FindInfo.flags = LVFI_PARAM; FindInfo.lParam = lItemData;
nItemIndex = m_CountersList.FindItem( &FindInfo );
if( nItemIndex < 0 || nItemIndex > 10 ) { ASSERT( FALSE ); } else { //
// Found our item - get the name
//
ZeroMemory( &lvItem, sizeof( lvItem ) );
lvItem.mask = LVIF_TEXT; lvItem.iItem = nItemIndex; lvItem.iSubItem = 0; lvItem.pszText = szCounterName; lvItem.cchTextMax = uCounterNameBufferLen;
bResult = m_CountersList.GetItem( &lvItem ); if( bResult == FALSE ) { //
// Could not get the current item's attributes?!?
//
ASSERT( FALSE ); } }
return bResult; }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::AddCounterInList( INT nItemData, ULONG uIdResourceString, SIZE_T sizeValue ) { INT nActualIndex; LVITEM lvItem; CString strName;
VERIFY( strName.LoadString( uIdResourceString ) );
ZeroMemory( &lvItem, sizeof( lvItem ) );
//
// LVITEM's member pszText is not a const pointer
// so we need to GetBuffer here :-(
//
//
// Sub-item 0 - counter's name
//
lvItem.pszText = strName.GetBuffer( strName.GetLength() + 1 ); if( NULL == lvItem.pszText ) { goto Done; }
lvItem.mask = LVIF_TEXT | LVIF_PARAM; lvItem.lParam = nItemData; lvItem.iItem = m_CountersList.GetItemCount();
nActualIndex = m_CountersList.InsertItem( &lvItem );
if( nActualIndex < 0 ) { //
// Could not add an item in the list - give up
//
goto Done; }
//
// Sub-item 1 - counter's value
//
UpdateCounterValueInList( nActualIndex, sizeValue );
Done: //
// All done
//
NOTHING; }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::RefreshInfo() { if( UpdateData( FALSE ) ) { //
// Refresh the settings bits list
//
RefreshTheList(); } }
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::UpdateCounterValueInList( INT nItemIndex, SIZE_T sizeValue ) { TCHAR szValue[ 32 ]; LVITEM lvItem;
#ifndef _WIN64
//
// 32 bit SIZE_T
//
_sntprintf( szValue, ARRAY_LENGTH( szValue ), _T( "%u" ), sizeValue );
#else
//
// 64 bit SIZE_T
//
_sntprintf( szValue, ARRAY_LENGTH( szValue ), _T( "%I64u" ), sizeValue );
#endif
szValue[ ARRAY_LENGTH( szValue ) - 1 ] = 0;
//
// Update the list item
//
ZeroMemory( &lvItem, sizeof( lvItem ) ); lvItem.mask = LVIF_TEXT; lvItem.iItem = nItemIndex; lvItem.iSubItem = 1; lvItem.pszText = szValue; VERIFY( m_CountersList.SetItem( &lvItem ) ); }
/////////////////////////////////////////////////////////////
//
// Other methods
//
/////////////////////////////////////////////////////////////
#define MIN_MEM_SIZE_TO_DISABLE_WARNING 0x80000000 // 2 Gb
#define MIN_ALLOCATIONS_SIGNIFICANT 100
#define MIN_PERCENTAGE_AVOID_WARNING 95
BOOL CGlobalCountPage::CheckAndShowPoolCoverageWarning() { BOOL bWarningDisplayed; ULONGLONG ullPercentageCoverage; TCHAR *szMessage; CString strMsgFormat; CString strWarnMsg;
bWarningDisplayed = FALSE;
if( m_RuntimeVerifierData.m_bSpecialPool && m_RuntimeVerifierData.AllocationsSucceeded >= MIN_ALLOCATIONS_SIGNIFICANT ) { //
// Special pool verification is enabled &&
// There is a significant number of allocations
//
ASSERT( m_RuntimeVerifierData.AllocationsSucceeded >= m_RuntimeVerifierData.AllocationsSucceededSpecialPool );
//
// The coverage percentage
//
ullPercentageCoverage = ( (ULONGLONG)m_RuntimeVerifierData.AllocationsSucceededSpecialPool * (ULONGLONG) 100 ) / (ULONGLONG)m_RuntimeVerifierData.AllocationsSucceeded;
ASSERT( ullPercentageCoverage <= 100 );
if( ullPercentageCoverage < MIN_PERCENTAGE_AVOID_WARNING ) { //
// Warn the user
//
if( strMsgFormat.LoadString( IDS_COVERAGE_WARNING_FORMAT ) ) { szMessage = strWarnMsg.GetBuffer( strMsgFormat.GetLength() + 32 );
if( szMessage != NULL ) { _stprintf( szMessage, (LPCTSTR)strMsgFormat, ullPercentageCoverage ); strWarnMsg.ReleaseBuffer();
AfxMessageBox( strWarnMsg, MB_OK | MB_ICONINFORMATION ); } } else { ASSERT( FALSE ); }
bWarningDisplayed = TRUE; } }
return bWarningDisplayed; }
/////////////////////////////////////////////////////////////
int CALLBACK CGlobalCountPage::CounterValueCmpFunc( LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort) { SIZE_T size1; SIZE_T size2; int nCmpRez = 0;
CGlobalCountPage *pThis = (CGlobalCountPage *)lParamSort; ASSERT_VALID( pThis );
size1 = pThis->GetCounterValue( (INT) lParam1 ); size2 = pThis->GetCounterValue( (INT) lParam2 );
if( size1 > size2 ) { nCmpRez = 1; } else { if( size1 < size2 ) { nCmpRez = -1; } }
if( FALSE != pThis->m_bAscendSortValue ) { nCmpRez *= -1; }
return nCmpRez; }
/////////////////////////////////////////////////////////////
int CALLBACK CGlobalCountPage::CounterNameCmpFunc( LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort) { int nCmpRez = 0; BOOL bSuccess; TCHAR szCounterName1[ _MAX_PATH ]; TCHAR szCounterName2[ _MAX_PATH ];
CGlobalCountPage *pThis = (CGlobalCountPage *)lParamSort; ASSERT_VALID( pThis );
//
// Get the first counter name
//
bSuccess = pThis->GetCounterName( lParam1, szCounterName1, ARRAY_LENGTH( szCounterName1 ) );
if( FALSE == bSuccess ) { goto Done; }
//
// Get the second counter name
//
bSuccess = pThis->GetCounterName( lParam2, szCounterName2, ARRAY_LENGTH( szCounterName2 ) );
if( FALSE == bSuccess ) { goto Done; }
//
// Compare the names
//
nCmpRez = _tcsicmp( szCounterName1, szCounterName2 ); if( FALSE != pThis->m_bAscendSortName ) { nCmpRez *= -1; }
Done:
return nCmpRez; }
/////////////////////////////////////////////////////////////
// CGlobalCountPage message handlers
BOOL CGlobalCountPage::OnInitDialog() { CPropertyPage::OnInitDialog();
//
// Setup the settings bits list
//
m_CountersList.SetExtendedStyle( LVS_EX_FULLROWSELECT | m_CountersList.GetExtendedStyle() );
m_CountersList.SetBkColor( ::GetSysColor( COLOR_3DFACE ) ); m_CountersList.SetTextBkColor( ::GetSysColor( COLOR_3DFACE ) );
SetupListHeader(); FillTheList(); SortTheList();
VrfSetWindowText( m_NextDescription, IDS_GCNT_PAGE_NEXT_DESCR );
VERIFY( m_uTimerHandler = SetTimer( REFRESH_TIMER_ID, 5000, NULL ) );
return TRUE; // return TRUE unless you set the focus to a control
// EXCEPTION: OCX Property Pages should return FALSE
}
/////////////////////////////////////////////////////////////
VOID CGlobalCountPage::OnTimer(UINT nIDEvent) { if( nIDEvent == REFRESH_TIMER_ID ) { ASSERT_VALID( m_pParentSheet );
if( m_pParentSheet->GetActivePage() == this ) { //
// Refresh the displayed data
//
RefreshInfo(); } }
CPropertyPage::OnTimer(nIDEvent); }
/////////////////////////////////////////////////////////////////////////////
BOOL CGlobalCountPage::OnSetActive() { ASSERT_VALID( m_pParentSheet );
m_pParentSheet->SetWizardButtons( PSWIZB_BACK | PSWIZB_NEXT ); return CVerifierPropertyPage::OnSetActive(); }
/////////////////////////////////////////////////////////////////////////////
LRESULT CGlobalCountPage::OnWizardNext() { GoingToNextPageNotify( IDD_PERDRIVER_COUNTERS_PAGE );
return IDD_PERDRIVER_COUNTERS_PAGE; }
/////////////////////////////////////////////////////////////////////////////
void CGlobalCountPage::OnColumnclickGlobcList(NMHDR* pNMHDR, LRESULT* pResult) { NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR; if( 0 != pNMListView->iSubItem ) { //
// Clicked on the counter value column
//
if( m_nSortColumnIndex == pNMListView->iSubItem ) { //
// Change the current ascend/descend order for this column
//
m_bAscendSortValue = !m_bAscendSortValue; } } else { //
// Clicked on the counter name column
//
if( m_nSortColumnIndex == pNMListView->iSubItem ) { //
// Change the current ascend/descend order for this column
//
m_bAscendSortName = !m_bAscendSortName; } }
m_nSortColumnIndex = pNMListView->iSubItem;
SortTheList();
*pResult = 0; }
/////////////////////////////////////////////////////////////
LONG CGlobalCountPage::OnHelp( WPARAM wParam, LPARAM lParam ) { LONG lResult = 0; LPHELPINFO lpHelpInfo = (LPHELPINFO)lParam;
::WinHelp( (HWND) lpHelpInfo->hItemHandle, g_szVerifierHelpFile, HELP_WM_HELP, (DWORD_PTR) MyHelpIds );
return lResult; }
/////////////////////////////////////////////////////////////////////////////
void CGlobalCountPage::OnContextMenu(CWnd* pWnd, CPoint point) { ::WinHelp( pWnd->m_hWnd, g_szVerifierHelpFile, HELP_CONTEXTMENU, (DWORD_PTR) MyHelpIds ); }
|