windows-nt/Source/XPSP1/NT/printscan/ui/wiaacmgr/wiaacmgr.cpp
2020-09-26 16:20:57 +08:00

581 lines
19 KiB
C++

// wiaacmgr.cpp : Implementation of WinMain
// Note: Proxy/Stub Information
// To build a separate proxy/stub DLL,
// run nmake -f wiaacmgrps.mk in the project directory.
#include "precomp.h"
#include "resource.h"
#include "wiaacmgr.h"
#include <shpriv.h>
#include <shlguid.h>
#include <stdarg.h>
#include "wiaacmgr_i.c"
#include "acqmgr.h"
#include "eventprompt.h"
#include "runwiz.h"
#include <initguid.h>
HINSTANCE g_hInstance;
#if defined(DBG_GENERATE_PRETEND_EVENT)
extern "C" int WINAPI _tWinMain( HINSTANCE hInstance, HINSTANCE /*hPrevInstance*/, LPTSTR lpCmdLine, int /*nShowCmd*/)
{
WIA_DEBUG_CREATE( hInstance );
SHFusionInitializeFromModuleID( hInstance, 123 );
g_hInstance = hInstance;
HRESULT hr = CoInitialize(NULL);
if (SUCCEEDED(hr))
{
IWiaDevMgr *pIWiaDevMgr = NULL;
hr = CoCreateInstance( CLSID_WiaDevMgr, NULL, CLSCTX_LOCAL_SERVER, IID_IWiaDevMgr, (void**)&pIWiaDevMgr );
if (SUCCEEDED(hr))
{
BSTR bstrDeviceID;
hr = pIWiaDevMgr->SelectDeviceDlgID( NULL, 0, 0, &bstrDeviceID );
if (hr == S_OK)
{
CEventParameters EventParameters;
EventParameters.EventGUID = WIA_EVENT_DEVICE_CONNECTED;
EventParameters.strEventDescription = L"";
EventParameters.strDeviceID = static_cast<LPCWSTR>(bstrDeviceID);
EventParameters.strDeviceDescription = L"";
EventParameters.ulEventType = 0;
EventParameters.ulReserved = 0;
EventParameters.hwndParent = NULL;
HANDLE hThread = CAcquisitionThread::Create( EventParameters );
if (hThread)
WaitForSingleObject(hThread,INFINITE);
SysFreeString(bstrDeviceID);
}
pIWiaDevMgr->Release();
}
CoUninitialize();
}
SHFusionUninitialize();
WIA_REPORT_LEAKS();
WIA_DEBUG_DESTROY();
return 0;
}
#else // !defined(DBG_GENERATE_PRETEND_EVENT)
const DWORD dwTimeOut = 5000; // time for EXE to be idle before shutting down
const DWORD dwPause = 1000; // time to wait for threads to finish up
// Passed to CreateThread to monitor the shutdown event
static DWORD WINAPI MonitorProc(void* pv)
{
CExeModule* p = (CExeModule*)pv;
p->MonitorShutdown();
return 0;
}
LONG CExeModule::Unlock()
{
LONG l = CComModule::Unlock();
if (l == 0)
{
bActivity = true;
SetEvent(hEventShutdown); // tell monitor that we transitioned to zero
}
return l;
}
//Monitors the shutdown event
void CExeModule::MonitorShutdown()
{
while (1)
{
WaitForSingleObject(hEventShutdown, INFINITE);
DWORD dwWait=0;
do
{
bActivity = false;
dwWait = WaitForSingleObject(hEventShutdown, dwTimeOut);
} while (dwWait == WAIT_OBJECT_0);
// timed out
if (!bActivity && m_nLockCnt == 0) // if no activity let's really bail
{
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
CoSuspendClassObjects();
if (!bActivity && m_nLockCnt == 0)
#endif
break;
}
}
CloseHandle(hEventShutdown);
PostThreadMessage(dwThreadID, WM_QUIT, 0, 0);
}
bool CExeModule::StartMonitor()
{
hEventShutdown = CreateEvent(NULL, false, false, NULL);
if (hEventShutdown == NULL)
return false;
DWORD dwThreadID;
HANDLE h = CreateThread(NULL, 0, MonitorProc, this, 0, &dwThreadID);
return (h != NULL);
}
CExeModule _Module;
BEGIN_OBJECT_MAP(ObjectMap)
OBJECT_ENTRY(CLSID_AcquisitionManager, CAcquisitionManager)
OBJECT_ENTRY(CLSID_MinimalTransfer, CMinimalTransfer)
OBJECT_ENTRY(WIA_EVENT_HANDLER_PROMPT, CEventPrompt)
OBJECT_ENTRY(CLSID_StiEventHandler, CStiEventHandler)
END_OBJECT_MAP()
LPCTSTR FindOneOf(LPCTSTR p1, LPCTSTR p2)
{
while (p1 != NULL && *p1 != NULL)
{
LPCTSTR p = p2;
while (p != NULL && *p != NULL)
{
if (*p1 == *p)
return CharNext(p1);
p = CharNext(p);
}
p1 = CharNext(p1);
}
return NULL;
}
//
// This MUST be removed before we ship. To remove it, remove
// the define ENABLE_SETUP_LOGGING from the sources file
//
#if defined(ENABLE_SETUP_LOGGING)
void _WizardPrintLogFileMessage( LPCTSTR pszFormat, ... )
{
//
// The name of the log file
//
static const TCHAR c_szLogFileName[] = TEXT("wiaacmgr.log");
//
// The path to the log file
//
static TCHAR szLogFilePathName[MAX_PATH] = {0};
//
// If we don't have a pathname, create it
//
if (!lstrlen(szLogFilePathName))
{
if (GetWindowsDirectory( szLogFilePathName, ARRAYSIZE(szLogFilePathName) ))
{
lstrcat( szLogFilePathName, TEXT("\\") );
lstrcat( szLogFilePathName, c_szLogFileName );
}
}
//
// If we still don't have a pathname, return
//
if (!lstrlen(szLogFilePathName))
{
return;
}
//
// Add the date and time to the output string
//
TCHAR szMessage[1500] = {0};
GetDateFormat( LOCALE_SYSTEM_DEFAULT, 0, NULL, TEXT("MM'/'dd'/'yy' '"), szMessage+lstrlen(szMessage), ARRAYSIZE(szMessage)-lstrlen(szMessage) );
GetTimeFormat( LOCALE_SYSTEM_DEFAULT, 0, NULL, TEXT("HH':'mm':'ss' '"), szMessage+lstrlen(szMessage), ARRAYSIZE(szMessage)-lstrlen(szMessage) );
//
// Add the process ID and thread ID to the output string
//
wsprintf( szMessage+lstrlen(szMessage), TEXT("%08X %08X : "), GetCurrentProcessId(), GetCurrentThreadId() );
//
// Format the output string
//
va_list pArgs;
va_start( pArgs, pszFormat );
wvsprintf( szMessage+lstrlen(szMessage), pszFormat, pArgs );
va_end( pArgs );
lstrcat( szMessage, TEXT("\r\n" ) );
//
// Open the file
//
HANDLE hFile = CreateFile( szLogFilePathName, GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL );
if (hFile)
{
//
// Go to the end of the file
//
if (INVALID_SET_FILE_POINTER != SetFilePointer(hFile,0,NULL,FILE_END))
{
//
// Write out the string
//
DWORD dwWritten;
WriteFile(hFile,szMessage,lstrlen(szMessage)*sizeof(TCHAR),&dwWritten,NULL);
}
//
// Close the file
//
CloseHandle(hFile);
}
}
#define LOG_MESSAGE(x) _WizardPrintLogFileMessage x
#else
#define LOG_MESSAGE(x)
#endif ENABLE_SETUP_LOGGING
static HRESULT RegisterForWiaEvents( LPCWSTR pszDevice, const CLSID &clsidHandler, const IID &iidEvent, int nName, int nDescription, int nIcon, bool bDefault, bool bRegister )
{
WIA_PUSH_FUNCTION((TEXT("RegisterForWiaEvents( device: %ws, default: %d, register: %d )"), pszDevice, bDefault, bRegister ));
WIA_PRINTGUID((clsidHandler,TEXT("Handler:")));
WIA_PRINTGUID((iidEvent,TEXT("Event:")));
CComPtr<IWiaDevMgr> pWiaDevMgr;
HRESULT hr = CoCreateInstance( CLSID_WiaDevMgr, NULL, CLSCTX_LOCAL_SERVER, IID_IWiaDevMgr, (void**)&pWiaDevMgr );
LOG_MESSAGE((TEXT(" 0x%08X CoCreateInstance( CLSID_WiaDevMgr )"), hr ));
if (SUCCEEDED(hr))
{
CSimpleBStr bstrDeviceId(pszDevice);
CSimpleBStr bstrName(CSimpleString(nName,_Module.m_hInst));
CSimpleBStr bstrDescription(CSimpleString(nDescription,_Module.m_hInst));
CSimpleBStr bstrIcon(CSimpleString(nIcon,_Module.m_hInst));
WIA_TRACE((TEXT("device: %ws"), pszDevice ));
WIA_TRACE((TEXT("name : %ws"), bstrName.BString() ));
WIA_TRACE((TEXT("desc : %ws"), bstrDescription.BString() ));
WIA_TRACE((TEXT("icon : %ws"), bstrIcon.BString() ));
if (bRegister)
{
if (bstrName.BString() && bstrDescription.BString() && bstrIcon.BString())
{
hr = pWiaDevMgr->RegisterEventCallbackCLSID(
WIA_REGISTER_EVENT_CALLBACK,
pszDevice ? bstrDeviceId.BString() : NULL,
&iidEvent,
&clsidHandler,
bstrName,
bstrDescription,
bstrIcon
);
LOG_MESSAGE((TEXT(" 0x%08X RegisterEventCallbackCLSID( WIA_REGISTER_EVENT_CALLBACK, \"%ws\" )"), hr, bstrDeviceId.BString() ));
if (SUCCEEDED(hr) && bDefault)
{
hr = pWiaDevMgr->RegisterEventCallbackCLSID(
WIA_SET_DEFAULT_HANDLER,
pszDevice ? bstrDeviceId.BString() : NULL,
&iidEvent,
&clsidHandler,
bstrName,
bstrDescription,
bstrIcon
);
LOG_MESSAGE((TEXT(" 0x%08X RegisterEventCallbackCLSID( WIA_SET_DEFAULT_HANDLER, \"%ws\" )"), hr, bstrDeviceId.BString() ));
if (FAILED(hr))
{
WIA_PRINTHRESULT((hr,TEXT("WIA_SET_DEFAULT_HANDLER failed")));
}
}
else if (FAILED(hr))
{
WIA_PRINTHRESULT((hr,TEXT("WIA_REGISTER_EVENT_CALLBACK failed")));
}
}
}
else
{
hr = pWiaDevMgr->RegisterEventCallbackCLSID(
WIA_UNREGISTER_EVENT_CALLBACK,
pszDevice ? bstrDeviceId.BString() : NULL,
&iidEvent,
&clsidHandler,
bstrName,
bstrDescription,
bstrIcon
);
LOG_MESSAGE((TEXT(" 0x%08X RegisterEventCallbackCLSID( WIA_UNREGISTER_EVENT_CALLBACK, \"%ws\" )"), hr, bstrDeviceId.BString() ));
if (FAILED(hr))
{
WIA_PRINTHRESULT((hr,TEXT("WIA_SET_DEFAULT_HANDLER failed")));
}
}
}
if (FAILED(hr))
{
WIA_PRINTHRESULT((hr,TEXT("Unable to register for the event!")));
}
return hr;
}
struct CRegistryEntry
{
HKEY hKeyParent;
LPCTSTR pszKey;
LPCTSTR pszValueName;
DWORD dwType;
LPARAM nValue;
DWORD dwSize;
bool bOverwrite;
};
bool SetRegistryValues( CRegistryEntry *pRegistryEntries, int nEntryCount )
{
if (!pRegistryEntries)
{
return FALSE;
}
//
// Assume success
//
bool bResult = true;
//
// Create the registry entries
//
for (int i=0;i<nEntryCount && bResult;i++)
{
//
// Create or open the key
//
HKEY hKey = NULL;
if (ERROR_SUCCESS == RegCreateKeyEx( pRegistryEntries[i].hKeyParent, pRegistryEntries[i].pszKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ|KEY_WRITE, NULL, &hKey, NULL ))
{
//
// If we have a value
//
if (pRegistryEntries[i].dwType && pRegistryEntries[i].pszValueName)
{
//
// If we are preventing existing values from being overwritten, check to see if the value exists
//
if (!pRegistryEntries[i].bOverwrite)
{
//
// Is the value present?
//
DWORD dwType = 0, dwSize = 0;
if (ERROR_SUCCESS == RegQueryValueEx( hKey, pRegistryEntries[i].pszValueName, NULL, &dwType, NULL, &dwSize ))
{
//
// The value exists, so continue
//
continue;
}
}
if (ERROR_SUCCESS != RegSetValueEx( hKey, pRegistryEntries[i].pszValueName, NULL, pRegistryEntries[i].dwType, reinterpret_cast<CONST BYTE *>(pRegistryEntries[i].nValue), pRegistryEntries[i].dwSize ))
{
bResult = false;
}
}
}
}
return bResult;
}
/////////////////////////////////////////////////////////////////////////////
//
extern "C" int WINAPI _tWinMain(HINSTANCE hInstance,
HINSTANCE /*hPrevInstance*/, LPTSTR lpCmdLine, int /*nShowCmd*/)
{
//
// Save the global hInstance
//
g_hInstance = hInstance;
LOG_MESSAGE((TEXT("Entering main: %s"), GetCommandLine()));
//
// Create the debugger
//
WIA_DEBUG_CREATE( hInstance );
//
// this line necessary for _ATL_MIN_CRT
//
lpCmdLine = GetCommandLine();
//
// Initialize fusion
//
SHFusionInitializeFromModuleID( hInstance, 123 );
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
#else
HRESULT hRes = CoInitialize(NULL);
#endif
int nRet = 0;
if (SUCCEEDED(hRes))
{
_Module.Init(ObjectMap, hInstance, &LIBID_WIAACMGRLib);
_Module.dwThreadID = GetCurrentThreadId();
TCHAR szTokens[] = _T("-/");
//
// Assume we'll be running as a server
//
BOOL bRun = TRUE;
//
// Assume we won't be selecting a device and running the wizard
//
BOOL bRunWizard = FALSE;
//
// If there are no switches, we are not going to run the server
//
int nSwitchCount = 0;
LPCTSTR lpszToken = FindOneOf( lpCmdLine, szTokens );
while (lpszToken != NULL)
{
//
// One more switch. If we don't have any, we are going to do the selection dialog instead.
//
nSwitchCount++;
if (lstrcmpi(lpszToken, _T("RegServer"))==0)
{
//
// Register the server
//
LOG_MESSAGE((TEXT("Begin handling /RegServer")));
hRes = _Module.UpdateRegistryFromResource(IDR_ACQUISITIONMANAGER, TRUE);
LOG_MESSAGE((TEXT(" 0x%08X _Module.UpdateRegistryFromResource(IDR_ACQUISITIONMANAGER)"), hRes ));
_Module.UpdateRegistryFromResource(IDR_MINIMALTRANSFER,TRUE);
LOG_MESSAGE((TEXT(" 0x%08X _Module.UpdateRegistryFromResource(IDR_MINIMALTRANSFER)"), hRes ));
_Module.UpdateRegistryFromResource(IDR_STIEVENTHANDLER,TRUE);
LOG_MESSAGE((TEXT(" 0x%08X _Module.UpdateRegistryFromResource(IDR_STIEVENTHANDLER)"), hRes ));
nRet = _Module.RegisterServer(TRUE);
LOG_MESSAGE((TEXT(" 0x%08X _Module.RegisterServer(TRUE)"), nRet ));
hRes = RegisterForWiaEvents( NULL, CLSID_AcquisitionManager, WIA_EVENT_DEVICE_CONNECTED, IDS_DOWNLOADMANAGER_NAME, IDS_DOWNLOADMANAGER_DESC, IDS_DOWNLOADMANAGER_ICON, false, true );
LOG_MESSAGE((TEXT(" 0x%08X RegisterForWiaEvents( WIA_EVENT_DEVICE_CONNECTED )"), hRes ));
#if defined(TESTING_STI_EVENT_HANDLER)
hRes = RegisterForWiaEvents( NULL, CLSID_StiEventHandler, WIA_EVENT_DEVICE_CONNECTED, IDS_DOWNLOADMANAGER_NAME, IDS_DOWNLOADMANAGER_DESC, IDS_DOWNLOADMANAGER_ICON, false, true );
MessageBox( NULL, TEXT("Test-only code"), TEXT("DEBUG"), 0 );
#endif
RegisterForWiaEvents( NULL, CLSID_AcquisitionManager, GUID_ScanImage, IDS_DOWNLOADMANAGER_NAME, IDS_DOWNLOADMANAGER_DESC, IDS_DOWNLOADMANAGER_ICON, false, true );
LOG_MESSAGE((TEXT(" 0x%08X RegisterForWiaEvents( GUID_ScanImage )"), hRes ));
bRun = FALSE;
LOG_MESSAGE((TEXT("End handling /RegServer\r\n")));
break;
}
if (lstrcmpi(lpszToken, _T("UnregServer"))==0)
{
_Module.UpdateRegistryFromResource(IDR_ACQUISITIONMANAGER, FALSE);
_Module.UpdateRegistryFromResource(IDR_MINIMALTRANSFER,FALSE);
nRet = _Module.UnregisterServer(TRUE);
RegisterForWiaEvents( NULL, CLSID_AcquisitionManager, WIA_EVENT_DEVICE_CONNECTED, IDS_DOWNLOADMANAGER_NAME, IDS_DOWNLOADMANAGER_DESC, IDS_DOWNLOADMANAGER_ICON, false, false );
RegisterForWiaEvents( NULL, CLSID_AcquisitionManager, GUID_ScanImage, IDS_DOWNLOADMANAGER_NAME, IDS_DOWNLOADMANAGER_DESC, IDS_DOWNLOADMANAGER_ICON, false, false );
bRun = FALSE;
break;
}
if (CSimpleString(lpszToken).ToUpper().Left(12)==CSimpleString(TEXT("SELECTDEVICE")))
{
bRunWizard = TRUE;
bRun = FALSE;
break;
}
if (CSimpleString(lpszToken).ToUpper().Left(10)==CSimpleString(TEXT("REGCONNECT")))
{
lpszToken = FindOneOf(lpszToken, TEXT(" "));
WIA_TRACE((TEXT("Handling RegConnect for %s"), lpszToken ));
hRes = RegisterForWiaEvents( CSimpleStringConvert::WideString(CSimpleString(lpszToken)), CLSID_MinimalTransfer, WIA_EVENT_DEVICE_CONNECTED, IDS_MINIMALTRANSFER_NAME, IDS_MINIMALTRANSFER_DESC, IDS_MINIMALTRANSFER_ICON, true, true );
bRun = FALSE;
break;
}
if (CSimpleString(lpszToken).ToUpper().Left(12)==CSimpleString(TEXT("UNREGCONNECT")))
{
lpszToken = FindOneOf(lpszToken, TEXT(" "));
WIA_TRACE((TEXT("Handling RegUnconnect for %s"), lpszToken ));
hRes = RegisterForWiaEvents( CSimpleStringConvert::WideString(CSimpleString(lpszToken)), CLSID_MinimalTransfer, WIA_EVENT_DEVICE_CONNECTED, IDS_MINIMALTRANSFER_NAME, IDS_MINIMALTRANSFER_DESC, IDS_MINIMALTRANSFER_ICON, false, false );
bRun = FALSE;
break;
}
lpszToken = FindOneOf(lpszToken, szTokens);
}
//
// if /SelectDevice was specified, or no arguments were specified, we want to start the wizard
//
if (bRunWizard || !nSwitchCount)
{
HRESULT hr = RunWiaWizard::RunWizard( NULL, NULL, TEXT("WiaWizardSingleInstanceDeviceSelection") );
if (FAILED(hr))
{
MessageBox( NULL, CSimpleString( IDS_NO_DEVICE_TEXT, g_hInstance ), CSimpleString( IDS_ERROR_TITLE, g_hInstance ), MB_ICONHAND );
}
}
//
// Otherwise run embedded
//
else if (bRun)
{
_Module.StartMonitor();
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED);
_ASSERTE(SUCCEEDED(hRes));
hRes = CoResumeClassObjects();
#else
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER, REGCLS_MULTIPLEUSE);
#endif
_ASSERTE(SUCCEEDED(hRes));
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
_Module.RevokeClassObjects();
}
_Module.Term();
CoUninitialize();
}
//
// Uninitialize fusion
//
SHFusionUninitialize();
WIA_REPORT_LEAKS();
WIA_DEBUG_DESTROY();
LOG_MESSAGE((TEXT("Exiting main\r\n")));
return nRet;
}
#endif // DBG_GENERATE_PRETEND_EVENT