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.
451 lines
9.5 KiB
451 lines
9.5 KiB
//=================================================================
|
|
//
|
|
// WinMsgEvent.cpp --
|
|
//
|
|
// Copyright © Microsoft Corporation. All rights reserved.
|
|
//
|
|
//=================================================================
|
|
|
|
#include "precomp.h"
|
|
#include <lockwrap.h>
|
|
#include "WinMsgEvent.h"
|
|
|
|
// initialize class globals
|
|
CCritSec CWinMsgEvent::mg_csMapLock ;
|
|
CCritSec CWinMsgEvent::mg_csWindowLock ;
|
|
CAutoEvent CWinMsgEvent::mg_aeCreateWindow ;
|
|
CWinMsgEvent::Sink_Map CWinMsgEvent::mg_oSinkMap ;
|
|
HANDLE CWinMsgEvent::mg_hThreadPumpHandle = NULL;
|
|
HWND CWinMsgEvent::mg_hWnd = NULL;
|
|
|
|
#define EVENT_MAP_LOCK_ CLockWrapper t_oAcs( mg_csMapLock ) ;
|
|
#define WINDOW_LOCK_ CLockWrapper t_oAcs( mg_csWindowLock ) ;
|
|
|
|
// per object call
|
|
CWinMsgEvent::CWinMsgEvent()
|
|
{}
|
|
|
|
// per object call
|
|
CWinMsgEvent::~CWinMsgEvent()
|
|
{
|
|
UnRegisterAllMessages() ;
|
|
|
|
// clear the WM_ENDSESSION handler
|
|
if( mg_oSinkMap.end() == mg_oSinkMap.find( WM_ENDSESSION ) )
|
|
{
|
|
// Note: If WM_ENDSESSION was never IN the map, this
|
|
// call will return zero (failed). However, it won't
|
|
// do anything bad, so just ignore it.
|
|
SetConsoleCtrlHandler( &CtrlHandlerRoutine, FALSE ) ;
|
|
}
|
|
}
|
|
|
|
|
|
// per object call
|
|
void CWinMsgEvent::RegisterForMessage(
|
|
|
|
IN UINT a_message
|
|
)
|
|
{
|
|
BOOL t_bFound = FALSE ;
|
|
BOOL t_bCreateWindow = FALSE ;
|
|
|
|
{ EVENT_MAP_LOCK_
|
|
|
|
if( mg_oSinkMap.empty() )
|
|
{
|
|
t_bCreateWindow = TRUE ;
|
|
}
|
|
else // lookup for message/object duplicate
|
|
{
|
|
CWinMsgEvent::Sink_Map::iterator t_SinkIter ;
|
|
t_SinkIter = mg_oSinkMap.find( a_message ) ;
|
|
|
|
while( t_SinkIter != mg_oSinkMap.end() )
|
|
{
|
|
if( a_message == t_SinkIter->first )
|
|
{
|
|
if( this == t_SinkIter->second )
|
|
{
|
|
t_bFound = TRUE ;
|
|
break ;
|
|
}
|
|
++t_SinkIter ;
|
|
}
|
|
else
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( !t_bFound )
|
|
{
|
|
// Set up a handler to simulate this message
|
|
// as we won't get it running under local system account.
|
|
if( WM_ENDSESSION == a_message &&
|
|
mg_oSinkMap.end() == mg_oSinkMap.find( WM_ENDSESSION ) )
|
|
{
|
|
SetConsoleCtrlHandler( &CtrlHandlerRoutine, TRUE ) ;
|
|
}
|
|
|
|
// map the desired message for this object instance
|
|
mg_oSinkMap.insert(
|
|
|
|
pair<UINT const, CWinMsgEvent*>
|
|
( a_message,
|
|
this ) ) ;
|
|
}
|
|
}
|
|
|
|
if( t_bCreateWindow )
|
|
{
|
|
CreateMsgProvider() ;
|
|
}
|
|
}
|
|
|
|
// per object call
|
|
bool CWinMsgEvent::UnRegisterMessage(
|
|
|
|
IN UINT a_message
|
|
)
|
|
{
|
|
bool t_bRet = false ;
|
|
BOOL t_bDestroyWindow = FALSE ;
|
|
|
|
{ EVENT_MAP_LOCK_
|
|
|
|
CWinMsgEvent::Sink_Map::iterator t_SinkIter ;
|
|
t_SinkIter = mg_oSinkMap.find( a_message ) ;
|
|
|
|
while( t_SinkIter != mg_oSinkMap.end() )
|
|
{
|
|
if( a_message == t_SinkIter->first )
|
|
{
|
|
if( this == t_SinkIter->second )
|
|
{
|
|
t_SinkIter = mg_oSinkMap.erase( t_SinkIter ) ;
|
|
t_bRet = true ;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
t_SinkIter++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
|
|
if( mg_oSinkMap.empty() )
|
|
{
|
|
t_bDestroyWindow = TRUE ;
|
|
}
|
|
}
|
|
|
|
if( t_bDestroyWindow )
|
|
{
|
|
DestroyMsgWindow() ;
|
|
}
|
|
|
|
return t_bRet ;
|
|
}
|
|
|
|
// per object call
|
|
void CWinMsgEvent::UnRegisterAllMessages()
|
|
{
|
|
BOOL t_bDestroyWindow = FALSE ;
|
|
|
|
{ // Used for scoping the lock
|
|
|
|
EVENT_MAP_LOCK_
|
|
|
|
CWinMsgEvent::Sink_Map::iterator t_SinkIter ;
|
|
t_SinkIter = mg_oSinkMap.begin() ;
|
|
|
|
while( t_SinkIter != mg_oSinkMap.end() )
|
|
{
|
|
if( this == t_SinkIter->second )
|
|
{
|
|
t_SinkIter = mg_oSinkMap.erase( t_SinkIter ) ;
|
|
}
|
|
else
|
|
{
|
|
t_SinkIter++;
|
|
}
|
|
}
|
|
|
|
if( mg_oSinkMap.empty() )
|
|
{
|
|
t_bDestroyWindow = TRUE;
|
|
}
|
|
}
|
|
|
|
if( t_bDestroyWindow )
|
|
{
|
|
DestroyMsgWindow() ;
|
|
}
|
|
}
|
|
|
|
|
|
// global private
|
|
void CWinMsgEvent::CreateMsgProvider()
|
|
{
|
|
WINDOW_LOCK_
|
|
|
|
if( NULL == mg_hThreadPumpHandle )
|
|
{
|
|
DWORD t_dwThreadID ;
|
|
|
|
// Create a thread that will spin off a windowed msg pump
|
|
mg_hThreadPumpHandle = CreateThread(
|
|
NULL, // pointer to security attributes
|
|
0L, // initial thread stack size
|
|
dwThreadProc, // pointer to thread function
|
|
0L, // argument for new thread
|
|
0L, // creation flags
|
|
&t_dwThreadID ) ;
|
|
|
|
// wait for async window create
|
|
mg_aeCreateWindow.Wait( INFINITE );
|
|
|
|
if( !mg_hWnd )
|
|
{
|
|
CloseHandle( mg_hThreadPumpHandle ) ;
|
|
mg_hThreadPumpHandle = NULL ;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
void CWinMsgEvent::DestroyMsgWindow()
|
|
{
|
|
WINDOW_LOCK_
|
|
|
|
HANDLE t_hThreadPumpHandle = mg_hThreadPumpHandle ;
|
|
HWND t_hWnd = mg_hWnd ;
|
|
|
|
// clear globals
|
|
mg_hThreadPumpHandle = NULL ;
|
|
mg_hWnd = NULL ;
|
|
|
|
if( t_hWnd )
|
|
{
|
|
SendMessage( t_hWnd, WM_CLOSE, 0, 0 ) ;
|
|
}
|
|
|
|
if( t_hThreadPumpHandle )
|
|
{
|
|
WaitForSingleObject(
|
|
|
|
t_hThreadPumpHandle,
|
|
20000
|
|
);
|
|
|
|
CloseHandle( t_hThreadPumpHandle ) ;
|
|
}
|
|
}
|
|
|
|
BOOL WINAPI CWinMsgEvent::CtrlHandlerRoutine(DWORD dwCtrlType)
|
|
{
|
|
HWND t_hWnd = NULL ;
|
|
UINT t_message = 0 ;
|
|
WPARAM t_wParam = 0 ;
|
|
LPARAM t_lParam = 0 ;
|
|
|
|
// simulate the message
|
|
if( CTRL_LOGOFF_EVENT == dwCtrlType )
|
|
{
|
|
t_message = WM_ENDSESSION ;
|
|
t_wParam = TRUE ; // session ending
|
|
t_lParam = ENDSESSION_LOGOFF ; // Logoff event
|
|
}
|
|
else if( CTRL_SHUTDOWN_EVENT == dwCtrlType )
|
|
{
|
|
t_message = WM_ENDSESSION ;
|
|
t_wParam = TRUE ; // session ending
|
|
t_lParam = 0 ; // Shutdown event
|
|
}
|
|
|
|
if( t_message )
|
|
{
|
|
//
|
|
MsgWndProc( t_hWnd,
|
|
t_message,
|
|
t_wParam,
|
|
t_lParam ) ;
|
|
}
|
|
|
|
return FALSE; // Pass event on to next handler.
|
|
}
|
|
|
|
// worker thread pump, global private
|
|
DWORD WINAPI CWinMsgEvent::dwThreadProc( LPVOID a_lpParameter )
|
|
{
|
|
DWORD t_dwRet = FALSE ;
|
|
|
|
if( CreateMsgWindow() )
|
|
{
|
|
WindowsDispatch() ;
|
|
|
|
t_dwRet = TRUE ;
|
|
}
|
|
|
|
UnregisterClass( MSGWINDOWNAME, GetModuleHandle(NULL) ) ;
|
|
|
|
return t_dwRet ;
|
|
}
|
|
|
|
// global private
|
|
HWND CWinMsgEvent::CreateMsgWindow()
|
|
{
|
|
DWORD t_Err = 0;
|
|
HMODULE t_hMod = GetModuleHandle(NULL);
|
|
|
|
if (t_hMod != NULL)
|
|
{
|
|
WNDCLASS wndclass ;
|
|
wndclass.style = 0 ;
|
|
wndclass.lpfnWndProc = MsgWndProc ;
|
|
wndclass.cbClsExtra = 0 ;
|
|
wndclass.cbWndExtra = 0 ;
|
|
wndclass.hInstance = t_hMod ;
|
|
wndclass.hIcon = NULL ;
|
|
wndclass.hCursor = NULL ;
|
|
wndclass.hbrBackground = NULL ;
|
|
wndclass.lpszMenuName = NULL ;
|
|
wndclass.lpszClassName = MSGWINDOWNAME ;
|
|
|
|
RegisterClass( &wndclass ) ;
|
|
|
|
mg_hWnd = CreateWindowEx( WS_EX_TOPMOST,
|
|
MSGWINDOWNAME,
|
|
TEXT("WinMsgEventProv"),
|
|
WS_OVERLAPPED,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
CW_USEDEFAULT,
|
|
NULL,
|
|
NULL,
|
|
t_hMod,
|
|
NULL ) ;
|
|
}
|
|
else
|
|
{
|
|
t_Err = GetLastError();
|
|
}
|
|
|
|
mg_aeCreateWindow.Signal();
|
|
return mg_hWnd ;
|
|
}
|
|
|
|
// global private
|
|
void CWinMsgEvent::WindowsDispatch()
|
|
{
|
|
BOOL t_GetMessage ;
|
|
MSG t_lpMsg ;
|
|
|
|
while ( ( t_GetMessage = GetMessage ( &t_lpMsg , NULL , 0 , 0 ) ) == TRUE )
|
|
{
|
|
TranslateMessage ( &t_lpMsg ) ;
|
|
DispatchMessage ( &t_lpMsg ) ;
|
|
}
|
|
}
|
|
|
|
// global private
|
|
LRESULT CALLBACK CWinMsgEvent::MsgWndProc(
|
|
|
|
IN HWND a_hWnd,
|
|
IN UINT a_message,
|
|
IN WPARAM a_wParam,
|
|
IN LPARAM a_lParam
|
|
)
|
|
{
|
|
LRESULT t_lResult = TRUE ;
|
|
E_ReturnAction t_eReturnAction = e_DefProc ;
|
|
|
|
switch ( a_message )
|
|
{
|
|
default:
|
|
{
|
|
// Run through the message map
|
|
// If registered requestor(s) are found dispatch it...
|
|
|
|
EVENT_MAP_LOCK_
|
|
|
|
CWinMsgEvent::Sink_Map::iterator t_SinkIter ;
|
|
t_SinkIter = mg_oSinkMap.find( a_message ) ;
|
|
|
|
while( t_SinkIter != mg_oSinkMap.end() )
|
|
{
|
|
if( a_message == t_SinkIter->first )
|
|
{
|
|
// signal
|
|
t_SinkIter->second->WinMsgEvent(
|
|
|
|
a_hWnd,
|
|
a_message,
|
|
a_wParam,
|
|
a_lParam,
|
|
t_eReturnAction,
|
|
t_lResult ) ;
|
|
++t_SinkIter ;
|
|
}
|
|
else
|
|
{
|
|
break ;
|
|
}
|
|
}
|
|
// special return processing ---
|
|
//
|
|
// The default is to defer to DefWindowProc.
|
|
// However, multiple sinks can exist for a message.
|
|
// Each may require special return processing.
|
|
//
|
|
// Example: WM_POWERBROADCAST submessage PBT_APMQUERYSUSPEND requires
|
|
// the returning of TRUE to indicate interest in additional Power
|
|
// Event messages. This a passive request ( asking for additional info )
|
|
// but another sink registered for this message may have a different opinion.
|
|
// Trivial perhaps, but other message processing may be different; placing
|
|
// the requestor at odds with the intent of another.
|
|
|
|
// Behavior here: All sinks are called with the
|
|
// updated t_eReturnAction from the last sink call.
|
|
// If a sink suspects it would have to act diffently based on specific
|
|
// knowledge of message usage the sink will have to instead spin off
|
|
// its own window to handle the special return and not make use of this
|
|
// generalized class.
|
|
//
|
|
if( e_DefProc == t_eReturnAction )
|
|
{
|
|
t_lResult = DefWindowProc( a_hWnd, a_message, a_wParam, a_lParam ) ;
|
|
}
|
|
break ;
|
|
}
|
|
|
|
case WM_CLOSE:
|
|
{
|
|
if ( mg_hWnd != NULL)
|
|
{
|
|
t_lResult = 0;
|
|
}
|
|
else
|
|
{
|
|
t_lResult = DefWindowProc( a_hWnd, a_message, a_wParam, a_lParam ) ;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_DESTROY:
|
|
{
|
|
PostMessage ( a_hWnd, WM_QUIT, 0, 0 ) ;
|
|
t_lResult = 0;
|
|
}
|
|
break ;
|
|
}
|
|
|
|
return t_lResult;
|
|
}
|