750 lines
18 KiB
C++
750 lines
18 KiB
C++
//
|
|
// 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 );
|
|
}
|
|
|