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.
 
 
 
 
 
 

532 lines
13 KiB

/*++
© 1998 Seagate Software, Inc. All rights reserved
Module Name:
RsRecall.cpp
Abstract:
Main module file - defines the overall COM server.
Author:
Rohde Wakefield [rohde] 04-Mar-1997
Revision History:
--*/
#include "stdafx.h"
#include "aclapi.h"
BEGIN_OBJECT_MAP( ObjectMap )
OBJECT_ENTRY( CLSID_CFsaRecallNotifyClient, CNotifyClient )
END_OBJECT_MAP()
const CString regString = TEXT( "reg" );
const CString unregString = TEXT( "unreg" );
HRESULT RegisterServer(void);
HRESULT UnregisterServer(void);
#ifdef _USRDLL
/////////////////////////////////////////////////////////////////////////////
// Setup to use if we are a DLL /////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
#define RecRegId IDR_CNotifyClientAppDll
/////////////////////////////////////////////////////////////////////////////
// Used to determine whether the DLL can be unloaded by OLE
STDAPI DllCanUnloadNow(void)
{
TRACEFNHR( "DllCanUnloadNow" );
AFX_MANAGE_STATE( AfxGetStaticModuleState( ) );
hrRet = (_Module.GetLockCount()==0) ? S_OK : S_FALSE;
return( hrRet );
}
/////////////////////////////////////////////////////////////////////////////
// Returns a class factory to create an object of the requested type
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID* ppv)
{
TRACEFNHR( "DllGetClassObject" );
AFX_MANAGE_STATE( AfxGetStaticModuleState( ) );
hrRet = _Module.GetClassObject(rclsid, riid, ppv);
return( hrRet );
}
/////////////////////////////////////////////////////////////////////////////
// DllRegisterServer - Adds entries to the system registry
STDAPI DllRegisterServer(void)
{
TRACEFNHR( "DllRegisterServer" );
AFX_MANAGE_STATE(AfxGetStaticModuleState());
// registers object, typelib and all interfaces in typelib
hrRet = RegisterServer( );
return( hrRet );
}
/////////////////////////////////////////////////////////////////////////////
// DllUnregisterServer - Removes entries from the system registry
STDAPI DllUnregisterServer(void)
{
TRACEFNHR( "DllUnregisterServer" );
AFX_MANAGE_STATE(AfxGetStaticModuleState());
hrRet = UnregisterServer( );
return( hrRet );
}
#else
/////////////////////////////////////////////////////////////////////////////
// Setup to use if we are a standalone app //////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
#define RecRegId IDR_CNotifyClientApp
class CRecallParse : public CCommandLineInfo {
virtual void ParseParam( LPCTSTR lpszParam, BOOL bFlag, BOOL bLast );
public:
BOOL m_RegAction;
CRecallParse( ) { m_RegAction = FALSE; };
};
void CRecallParse::ParseParam( LPCTSTR lpszParam, BOOL bFlag, BOOL bLast )
{
TRACEFN( "CRecallParse::ParseParam" );
CString cmdLine = lpszParam;
if( bFlag ) {
if( cmdLine.Left( unregString.GetLength( ) ) == unregString ) {
UnregisterServer( );
m_RegAction = TRUE;
} else if( cmdLine.Left( regString.GetLength( ) ) == regString ) {
RegisterServer( );
m_RegAction = TRUE;
}
}
}
#endif
/////////////////////////////////////////////////////////////////////////////
// RegisterServer - Adds entries to the system registry
HRESULT RegisterServer(void)
{
TRACEFNHR( "RegisterServer" );
try {
//
// Add the object entries
//
RecAffirmHr( _Module.RegisterServer( FALSE ) );
//
// Add server entries
//
RecAffirmHr( _Module.UpdateRegistryFromResource( RecRegId, TRUE ) );
//
// Set up access to be allowed by everyone (NULL DACL)
// Appears we need some owner and group, so use the current one.
//
CSecurityDescriptor secDesc;
PSECURITY_DESCRIPTOR pSDSelfRelative = 0;
RecAffirmHr( secDesc.InitializeFromProcessToken( ) );
DWORD secDescSize = 0;
MakeSelfRelativeSD( secDesc, pSDSelfRelative, &secDescSize );
pSDSelfRelative = (PSECURITY_DESCRIPTOR) new char[secDescSize];
if( MakeSelfRelativeSD( secDesc, pSDSelfRelative, &secDescSize ) ) {
CString keyName = TEXT( "AppID\\{D68BD5B2-D6AA-11d0-9EDA-00A02488FCDE}" );
CRegKey regKey;
regKey.Open( HKEY_CLASSES_ROOT, keyName, KEY_SET_VALUE );
RegSetValueEx( regKey.m_hKey, TEXT( "LaunchPermission" ), 0, REG_BINARY, (LPBYTE)pSDSelfRelative, secDescSize );
}
} RecCatch( hrRet );
return( hrRet );
}
/////////////////////////////////////////////////////////////////////////////
// UnregisterServer - Removes entries from the system registry
HRESULT UnregisterServer(void)
{
TRACEFNHR( "UnregisterServer" );
try {
RecAffirmHr( _Module.UnregisterServer() );
//
// Remove server entries
//
RecAffirmHr( _Module.UpdateRegistryFromResource( RecRegId, FALSE ) );
} RecCatch( hrRet );
return( hrRet );
}
/////////////////////////////////////////////////////////////////////////////
// CRecallApp
BEGIN_MESSAGE_MAP(CRecallApp, CWinApp)
//{{AFX_MSG_MAP(CRecallApp)
// NOTE - the ClassWizard will add and remove mapping macros here.
// DO NOT EDIT what you see in these blocks of generated code!
//}}AFX_MSG
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CRecallApp construction
CRecallApp::CRecallApp()
{
TRACEFN( "CRecallApp::CRecallApp" );
m_IdleCount = 0;
}
/////////////////////////////////////////////////////////////////////////////
// The one and only CRecallApp object
CRecallApp theApp;
/////////////////////////////////////////////////////////////////////////////
// CRecallApp initialization
BOOL CRecallApp::InitInstance()
{
TRACEFNHR( "CRecallApp::InitInstance" );
LPTSTR cmdLine = GetCommandLine( );
TRACE( cmdLine );
try {
_Module.Init( ObjectMap, m_hInstance );
m_dwMaxConcurrentNotes = MAX_CONCURRENT_RECALL_NOTES_DEFAULT;
InitCommonControls();
#ifndef _USRDLL
//
// Initialize the COM module (no point to continue if it fails)
//
hrRet = CoInitialize( 0 );
if (!SUCCEEDED(hrRet)) {
return FALSE;
}
//
// Parse the command line
//
CRecallParse parse;
ParseCommandLine( parse );
if( parse.m_RegAction ) {
return FALSE;
}
//
// This provides a NULL DACL which will allow access to everyone.
//
RecAffirmHr( CoInitializeSecurity( 0, -1, 0, 0, RPC_C_AUTHN_LEVEL_NONE, RPC_C_IMP_LEVEL_IDENTIFY, 0, EOAC_NONE, 0 ) );
//
// Register the Fsa callback object
//
RecAffirmHr( _Module.RegisterClassObjects( CLSCTX_LOCAL_SERVER | CLSCTX_REMOTE_SERVER, REGCLS_MULTIPLEUSE ) );
#endif
// m_Wnd.Create( 0, TEXT( "Remote Storage Recall Notification Wnd" ) );
// m_pMainWnd = &m_Wnd;
CRecallWnd *pWnd = new CRecallWnd; // will auto delete
RecAffirmPointer( pWnd );
pWnd->Create( 0, TEXT( "Remote Storage Recall Notification Wnd" ) );
m_pMainWnd = pWnd;
// Check to see if there is any custom setting in the Registry for max recall popups
// (ignore errors - just use default)
HKEY hRegKey;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, RSNTFY_REGISTRY_STRING, 0, KEY_QUERY_VALUE, &hRegKey) == ERROR_SUCCESS) {
DWORD dwType, dwValue;
DWORD cbData = sizeof(DWORD);
if (RegQueryValueEx(hRegKey, MAX_CONCURRENT_RECALL_NOTES, 0, &dwType, (BYTE*)&dwValue, &cbData) == ERROR_SUCCESS) {
if (REG_DWORD == dwType) {
// custom setting
m_dwMaxConcurrentNotes = dwValue;
}
}
RegCloseKey(hRegKey);
}
} RecCatch( hrRet );
return TRUE;
}
int CRecallApp::ExitInstance()
{
TRACEFN("CRecallApp::ExitInstance");
_Module.Term();
#ifndef _USRDLL
CoUninitialize();
#endif
return CWinApp::ExitInstance();
}
void CRecallApp::LockApp( )
{
TRACEFNLONG( "CRecallApp::LockApp" );
lRet = _Module.Lock( );
}
void CRecallApp::UnlockApp( )
{
TRACEFNLONG( "CRecallApp::UnlockApp" );
lRet = _Module.Unlock( );
// Don't call AfxPostQuitMessage when ref. count drops to zero
// The timer mechanism is responsible for terminating this application.
// Also, when the ref count drops to zero, COM terminates the process after some time.
}
HRESULT CRecallApp::AddRecall( IFsaRecallNotifyServer* pRecall )
{
TRACEFNHR( "CRecallApp::AddRecall" );
CRecallNote * pNote = 0;
try {
//
// Create a new note to show - only if we didn't pass the max number for concurrent notes
// Note: We return S_OK and not S_FALSE even if the recall popup is not displayed in order
// not to break the server (S_FALSE will cause unnecessary retries)
//
if (m_Recalls.GetCount() < (int)m_dwMaxConcurrentNotes) {
pNote = new CRecallNote( pRecall, CWnd::GetDesktopWindow( ) );
RecAffirmHr( pNote->m_hrCreate );
m_Recalls.AddTail( pNote );
} else {
TRACE( _T("Recall not added - reached max number of recalls"));
}
} RecCatchAndDo( hrRet,
if( 0 != pNote ) delete pNote;
);
return( hrRet );
}
//
// Note:
// No CS is used here because the RsNotify is initialized as a single threaded application
//
HRESULT CRecallApp::RemoveRecall( IFsaRecallNotifyServer* pRecall )
{
TRACEFNHR( "CRecallApp::RemoveRecall" );
hrRet = S_FALSE;
if( ( m_Recalls.IsEmpty() ) ) {
return( hrRet );
}
CRecallNote* pNote = 0;
POSITION pos = m_Recalls.GetHeadPosition( );
POSITION currentPos = 0;
//
// Look through the list and find this one
//
GUID recallId;
pRecall->GetIdentifier( &recallId );
while( pos != 0 ) {
currentPos = pos;
pNote = m_Recalls.GetNext( pos );
if( IsEqualGUID( recallId, pNote->m_RecallId ) ) {
if (pNote->m_bCancelled) {
//
// This means that somebody is already removing this recall
// The Remove may be called up to 3 times for the same recall in case
// of a recall cancellation
//
hrRet = S_OK;
pos = 0; // exit loop
} else {
pNote->m_bCancelled = TRUE;
//
// Remove and delete. Return OK.
//
m_Recalls.RemoveAt( currentPos );
pNote->DestroyWindow( );
pos = 0; // exit loop
hrRet = S_OK;
}
}
}
return( hrRet );
}
// CRecallApp::Tick - called every second (after an initial delay
// for startup) to keep track of idle time
void CRecallApp::Tick( )
{
TRACEFN( "CRecallApp::Tick");
// Check for pending recalls
if( m_Recalls.GetCount( ) ) {
// We have pending recalls, reset the idle count
TRACE( _T("m_Recalls.GetCount != 0") );
m_IdleCount = 0;
} else {
// We don't have pending recalls, increment the idle count
TRACE( _T("m_Recalls.GetCount == 0") );
m_IdleCount++;
if( m_IdleCount > RSRECALL_TIME_MAX_IDLE ) {
TRACE( _T("m_IdleCount > 0") );
// Nothing's happin', say "Goodbye"
m_pMainWnd->PostMessage( WM_CLOSE );
TRACE( _T("after PostMessage(WM_CLOSE)") );
}
}
}
/////////////////////////////////////////////////////////////////////////////
// CRecallWnd
CRecallWnd::CRecallWnd()
{
TRACEFN( "CRecallWnd::CRecallWnd" );
}
CRecallWnd::~CRecallWnd()
{
TRACEFN( "CRecallWnd::~CRecallWnd" );
}
BEGIN_MESSAGE_MAP(CRecallWnd, CWnd)
//{{AFX_MSG_MAP(CRecallWnd)
ON_WM_TIMER()
ON_WM_CREATE()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CRecallWnd message handlers
void CRecallWnd::OnTimer(UINT nIDEvent)
{
TRACEFN("CRecallWnd::OnTimer");
if (1 == nIDEvent) {
// Initial timer. Kill it and start one for every second
TRACE( _T("nIDEvent == 1") );
KillTimer( nIDEvent );
SetTimer( 2, 1000, NULL );
} else {
// One second timer. Notify the app object
RecApp->Tick();
}
CWnd::OnTimer( nIDEvent );
}
int CRecallWnd::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
TRACEFN("CRecallWnd::OnCreate" );
if (CWnd::OnCreate(lpCreateStruct) == -1)
return -1;
// Set the initial timer to allow time for startup
if (!SetTimer(1, RSRECALL_TIME_FOR_STARTUP * 1000, NULL))
return -1;
return 0;
}