2020-09-30 17:12:29 +02:00

1661 lines
37 KiB
C

/*++
Copyright (c) 1990-1994 Microsoft Corporation
All rights reserved
Module Name:
Change.c
Abstract:
Handles implementation for WaitForPrinterChange and related apis.
FindFirstPrinterChangeNotification
RouterFindNextPrinterChangeNotification
FindClosePrinterChangeNotification
Used by providors:
ReplyPrinterChangeNotification [Function Call]
CallRouterFindFirstPrinterChangeNotification [Function Call]
Author:
Albert Ting (AlbertT) 18-Jan-94
Environment:
User Mode -Win32
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
#include <ntfytab.h>
#define PRINTER_NOTIFY_DEFAULT_POLLTIME 10000
CRITICAL_SECTION RouterNotifySection;
LPWSTR pszSelfMachine;
PCHANGEINFO pChangeInfoHead;
HANDLE hEventPoll;
BOOL
SetupReplyNotification(
PCHANGE pChange,
DWORD fdwOptions,
DWORD fdwStatus,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PPRINTER_NOTIFY_INIT pPrinterNotifyInit);
BOOL
FindFirstPrinterChangeNotificationWorker(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
DWORD dwPID,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PHANDLE phEvent,
BOOL bSpooler);
DWORD
FindClosePrinterChangeNotificationWorker(
HANDLE hPrinter);
BOOL
SetupChange(
PPRINTHANDLE pPrintHandle,
HANDLE hPrinterRemote,
LPWSTR pszLocalMachine,
PDWORD pfdwStatus,
DWORD fdwOptions,
DWORD fdwFilterFlags,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PPRINTER_NOTIFY_INIT* pPrinterNotifyInit);
VOID
FailChange(
PPRINTHANDLE pPrintHandle);
BOOL
RouterRefreshPrinterChangeNotification(
HANDLE hPrinter,
DWORD dwColor,
PVOID pPrinterNotifyRefresh,
PPRINTER_NOTIFY_INFO* ppInfo);
BOOL
WPCInit()
{
WCHAR szName[MAX_PATH];
DWORD i;
//
// Create non-signaled, autoreset event.
//
hEventPoll = CreateEvent(NULL, FALSE, FALSE, NULL);
if (!hEventPoll)
return FALSE;
szName[0] = szName[1] = L'\\';
i = MAX_PATH-2;
if (!GetComputerName(szName+2, &i)) {
DBGMSG(DBG_ERROR, ("GetComputerName failed.\n"));
return FALSE;
}
pszSelfMachine = AllocSplStr(szName);
if (!pszSelfMachine)
return FALSE;
InitializeCriticalSection(&RouterNotifySection);
return TRUE;
}
VOID
WPCDestroy()
{
#if 0
FreeSplStr(pszSelfMachine);
DeleteCriticalSection(&RouterNotifySection);
CloseHandle(hEventPoll);
#endif
}
BOOL
RouterFindFirstPrinterChangeNotification(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
DWORD dwPID,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PHANDLE phEvent)
{
return FindFirstPrinterChangeNotificationWorker(
hPrinter,
fdwFilterFlags,
fdwOptions,
dwPID,
pPrinterNotifyOptions,
phEvent,
FALSE);
}
BOOL
FindFirstPrinterChangeNotificationWorker(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
DWORD dwPID,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PHANDLE phEvent,
BOOL bSpooler)
/*++
Routine Description:
Sets up notification (coming from client side, winspool.drv).
Create an event and duplicate it into the clients address
space so we can communicate with it.
Arguments:
hPrinter - Printer to watch
fdwFilterFlags - Type of notification to set up (filter)
fdwOptions - user specified option (GROUPING, etc.)
dwPID - PID of the client process (needed to dup handle)
phEvent - hEvent to pass back to the client.
bSpooler - Indicates if called from spooler. If TRUE, then we don't
need to duplicate the event.
Return Value:
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
HANDLE hProcess;
DWORD fdwStatus = 0;
PCHANGE pChange = NULL;
PPRINTER_NOTIFY_INIT pPrinterNotifyInit = NULL;
BOOL bReturn;
EnterRouterSem();
//
// Clear this out
//
*phEvent = NULL;
if (!SetupChange(pPrintHandle,
pPrintHandle,
pszSelfMachine,
&fdwStatus,
fdwOptions,
fdwFilterFlags,
pPrinterNotifyOptions,
&pPrinterNotifyInit)) {
LeaveRouterSem();
return FALSE;
}
//
// !! LATER !!
//
// When Delegation is supported:
//
// Create the event with security access based on the impersonation
// token so that we can filter out bogus notifications from
// random people. (Save the token in the pSpool in localspl, then
// impersonate before RPCing back here. Then we can check if
// we have access to the event.)
//
//
// Create the event here that we trigger on notifications.
// We will duplicate this event into the target client process.
//
pPrintHandle->pChange->hEvent = CreateEvent(
NULL,
TRUE,
FALSE,
NULL);
if (!pPrintHandle->pChange->hEvent) {
goto Fail;
}
if (bSpooler) {
//
// Client is in the spooler process.
//
*phEvent = pPrintHandle->pChange->hEvent;
} else {
//
// Client is local.
//
//
// Success, create pair
//
hProcess = OpenProcess(PROCESS_DUP_HANDLE,
FALSE,
dwPID);
if (!hProcess) {
goto Fail;
}
bReturn = DuplicateHandle(GetCurrentProcess(),
pPrintHandle->pChange->hEvent,
hProcess,
phEvent,
EVENT_ALL_ACCESS,
TRUE,
0);
CloseHandle(hProcess);
if (!bReturn) {
goto Fail;
}
}
bReturn = SetupReplyNotification(pPrintHandle->pChange,
fdwStatus,
fdwOptions,
pPrinterNotifyOptions,
pPrinterNotifyInit);
if (bReturn) {
pPrintHandle->pChange->eStatus &= ~STATUS_CHANGE_FORMING;
pPrintHandle->pChange->eStatus |=
STATUS_CHANGE_VALID|STATUS_CHANGE_CLIENT;
} else {
Fail:
if (pPrintHandle->pChange->hEvent)
CloseHandle(pPrintHandle->pChange->hEvent);
//
// Local case must close handle twice since we may have
// duplicated the event.
//
if (!bSpooler && *phEvent)
CloseHandle(*phEvent);
FailChange(pPrintHandle);
bReturn = FALSE;
}
LeaveRouterSem();
return bReturn;
}
BOOL
RemoteFindFirstPrinterChangeNotification(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
LPWSTR pszLocalMachine,
HANDLE hPrinterRemote,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions)
/*++
Routine Description:
Handles FFPCN coming from other machines. Providers can use
the ProvidorRemoteFFPCN call to initiate the RPC which this function
handles. This code ships the call to the print provider. Note
that we don't create any events here since the client is on
a remote machine.
Arguments:
hPrinter - printer to watch
fdwFilterFlags - type of notification to watch
fdwOptions -- options on watch
pszLocalMachine - name of local machine that requested the watch
hPrinterRemote - pointer to pSpool valid in the local machines router
Return Value:
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
BOOL bReturn;
DWORD fdwStatus = 0;
PCHANGE pChange = NULL;
PPRINTER_NOTIFY_INIT pPrinterNotifyInit = NULL;
LPWSTR pszLocalMachineCopy;
pszLocalMachineCopy = AllocSplStr(pszLocalMachine);
if (!pszLocalMachineCopy)
return FALSE;
EnterRouterSem();
if (!SetupChange(pPrintHandle,
hPrinterRemote,
pszLocalMachineCopy,
&fdwStatus,
fdwOptions,
fdwFilterFlags,
pPrinterNotifyOptions,
&pPrinterNotifyInit)) {
LeaveRouterSem();
return FALSE;
}
bReturn = SetupReplyNotification(pPrintHandle->pChange,
fdwStatus,
fdwOptions,
pPrinterNotifyOptions,
pPrinterNotifyInit);
if (bReturn) {
pPrintHandle->pChange->eStatus = STATUS_CHANGE_VALID;
} else {
FailChange(pPrintHandle);
}
LeaveRouterSem();
return bReturn;
}
BOOL
RouterFindNextPrinterChangeNotification(
HANDLE hPrinter,
DWORD fdwFlags,
LPDWORD pfdwChangeFlags,
PPRINTER_NOTIFY_OPTIONS pOptions,
PPRINTER_NOTIFY_INFO* ppInfo)
/*++
Routine Description:
Return information about notification that just occurred and
reset to look for more notifications.
Arguments:
hPrinter - printer to reset event handle
fdwFlags - flags (PRINTER_NOTIFY_NEXT_INFO)
pfdwChange - return result of changes
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions
pReplyContainer - Reply info to pass out.
Return Value:
BOOL
** NOTE **
Always assume client process is on the same machine. The client
machine router always handles this call.
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
PCHANGE pChange = pPrintHandle->pChange;
BOOL bReturn = FALSE;
if (ppInfo) {
*ppInfo = NULL;
}
//
// Currently only REFRESH option is defined.
//
if( pOptions && ( pOptions->Flags & ~PRINTER_NOTIFY_OPTIONS_REFRESH )){
SetLastError( ERROR_INVALID_PARAMETER );
return FALSE;
}
EnterRouterSem();
if (pPrintHandle->signature != PRINTHANDLE_SIGNATURE ||
!pChange ||
!(pChange->eStatus & (STATUS_CHANGE_VALID|STATUS_CHANGE_CLIENT))) {
SetLastError(ERROR_INVALID_HANDLE);
goto Done;
}
//
// For now, we collapse all notifications into 1.
//
pChange->dwCount = 0;
//
// Tell the user what changes occurred,
// then clear it out.
//
*pfdwChangeFlags = pChange->fdwChangeFlags;
pChange->fdwChangeFlags = 0;
ResetEvent(pChange->hEvent);
if (pOptions && pOptions->Flags & PRINTER_NOTIFY_OPTIONS_REFRESH) {
//
// Increment color.
//
pPrintHandle->pChange->dwColor++;
LeaveRouterSem();
bReturn = RouterRefreshPrinterChangeNotification(
hPrinter,
pPrintHandle->pChange->dwColor,
pOptions,
ppInfo);
return bReturn;
}
//
// If they want data && (we have data || if we have flags that we wish
// to send to the user), copy the data.
//
if( ppInfo &&
(fdwFlags & PRINTER_NOTIFY_NEXT_INFO) &&
NotifyNeeded( pChange )){
*ppInfo = pChange->ChangeInfo.pPrinterNotifyInfo;
pChange->ChangeInfo.pPrinterNotifyInfo = NULL;
//
// If we need to notify because of a DISCARD, but we don't
// have a pPrinterNotifyInfo, then allocate one and return it.
//
if( !*ppInfo ){
DBGMSG( DBG_TRACE,
( "RFNPCN: Discard with no pPrinterNotifyInfo: pChange %x\n",
pChange ));
//
// We need to return some information back to the client, so
// allocate a block that has zero items. The header will be
// marked as DISCARDED, which tells the user to refresh.
//
*ppInfo = RouterAllocPrinterNotifyInfo( 0 );
if( !*ppInfo ){
//
// Failed to alloc memory; fail the call.
//
bReturn = FALSE;
goto Done;
}
//
// Mark the newly allocated information as DISCARDED.
//
(*ppInfo)->Flags = PRINTER_NOTIFY_INFO_DISCARDED;
}
}
bReturn = TRUE;
Done:
LeaveRouterSem();
return bReturn;
}
BOOL
FindClosePrinterChangeNotification(
HANDLE hPrinter)
{
DWORD dwError;
LPPRINTHANDLE pPrintHandle = (LPPRINTHANDLE)hPrinter;
EnterRouterSem();
if (pPrintHandle->signature != PRINTHANDLE_SIGNATURE ||
!pPrintHandle->pChange ||
!(pPrintHandle->pChange->eStatus & STATUS_CHANGE_VALID)) {
DBGMSG(DBG_ERROR, ("FCPCNW: Invalid handle 0x%x\n", pPrintHandle));
dwError = ERROR_INVALID_HANDLE;
} else {
dwError = FindClosePrinterChangeNotificationWorker(hPrinter);
}
LeaveRouterSem();
if (dwError) {
SetLastError(dwError);
return FALSE;
}
return TRUE;
}
DWORD
FindClosePrinterChangeNotificationWorker(
HANDLE hPrinter)
/*++
Routine Description:
Close a notification.
Arguments:
hPrinter -- printer that we want to close
Return Value:
ERROR code
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
BOOL bLocal = FALSE;
HANDLE hNotifyRemote = NULL;
DWORD dwError = ERROR_SUCCESS;
DBGMSG(DBG_NOTIFY, ("FCPCN: Closing 0x%x ->pChange 0x%x\n",
pPrintHandle, pPrintHandle->pChange));
RouterInSem();
//
// If the notification exists, shut it down (this is the
// local case). If we are called remotely, we don't need to
// do this, since hEvent wasn't created.
//
if (pPrintHandle->pChange->eStatus & STATUS_CHANGE_CLIENT) {
CloseHandle(pPrintHandle->pChange->hEvent);
bLocal = TRUE;
}
//
// Remember what the hNotifyRemote is, in case we want to delete it.
//
hNotifyRemote = pPrintHandle->pChange->hNotifyRemote;
//
// Failure to free implies we're using it now. In this case,
// don't try and free the hNotifyRemote.
//
if (!FreeChange(pPrintHandle->pChange)) {
hNotifyRemote = NULL;
}
//
// If local, don't allow new reply-s to be set up.
//
if (bLocal) {
RemoveReplyClient(pPrintHandle,
REPLY_TYPE_NOTIFICATION);
}
//
// We must zero this out to prevent other threads from
// attempting to close this context handle (client side)
// at the same time we are closing it.
//
pPrintHandle->pChange = NULL;
if (!bLocal) {
//
// Remote case, shut down the notification handle if
// there is one here. (If there is a double hop, only
// the second hop will have a notification reply. Currently
// only 1 hop is support during registration, however.)
//
LeaveRouterSem();
CloseReplyRemote(hNotifyRemote);
EnterRouterSem();
}
LeaveRouterSem();
RouterOutSem();
if (!(*pPrintHandle->pProvidor->PrintProvidor.
fpFindClosePrinterChangeNotification) (pPrintHandle->hPrinter)) {
dwError = GetLastError();
}
EnterRouterSem();
return dwError;
}
BOOL
SetupReplyNotification(
PCHANGE pChange,
DWORD fdwStatus,
DWORD fdwOptions,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PPRINTER_NOTIFY_INIT pPrinterNotifyInit)
{
DWORD dwReturn = ERROR_SUCCESS;
RouterInSem();
if (!pChange) {
dwReturn = ERROR_INVALID_PARAMETER;
goto Fail;
}
SPLASSERT(pChange->eStatus & STATUS_CHANGE_FORMING);
if (fdwStatus & PRINTER_NOTIFY_STATUS_ENDPOINT) {
//
// For remote notification, we must setup a reply.
//
if (_wcsicmp(pChange->pszLocalMachine, pszSelfMachine)) {
LeaveRouterSem();
dwReturn = OpenReplyRemote(pChange->pszLocalMachine,
&pChange->hNotifyRemote,
pChange->hPrinterRemote,
REPLY_TYPE_NOTIFICATION,
0,
NULL);
EnterRouterSem();
if (dwReturn)
goto Fail;
}
//
// The user can specify different status flags for the
// notfication. Process them here.
//
if (fdwStatus & PRINTER_NOTIFY_STATUS_POLL) {
//
// If there wasn't an error, then our reply notification
// handle is valid, and we should do the polling.
//
pChange->ChangeInfo.dwPollTime =
(pPrinterNotifyInit &&
pPrinterNotifyInit->PollTime) ?
pPrinterNotifyInit->PollTime :
PRINTER_NOTIFY_DEFAULT_POLLTIME;
pChange->ChangeInfo.dwPollTimeLeft = pChange->ChangeInfo.dwPollTime;
//
// Don't cause a poll the first time it's added.
//
pChange->ChangeInfo.bResetPollTime = TRUE;
LinkAdd(&pChange->ChangeInfo.Link, (PLINK*)&pChangeInfoHead);
SetEvent(hEventPoll);
}
pChange->ChangeInfo.fdwStatus = fdwStatus;
} else {
pChange->hPrinterRemote = NULL;
}
if (pPrinterNotifyOptions) {
pChange->eStatus |= STATUS_CHANGE_INFO;
}
Fail:
if (dwReturn) {
SetLastError(dwReturn);
return FALSE;
}
return TRUE;
}
//
// The Reply* functions handle calls from the server back to the client,
// indicating that something changed.
//
BOOL
ReplyPrinterChangeNotificationWorker(
HANDLE hPrinter,
DWORD dwColor,
DWORD fdwChangeFlags,
PDWORD pdwResult,
PPRINTER_NOTIFY_INFO pPrinterNotifyInfo
)
/*++
Routine Description:
Notifies the client that something happened. If local, simply
set the event. If remote, call ThreadNotify which spawns a thread
to RPC to the client router. (This call from the server -> client
requires that the spooler pipe use the NULL session, since we
are in local system context and RPC calls (with no impersonation)
use the NULL session.)
Arguments:
hPrinter -- printer that changed
dwColor -- color time stamp of data
fdwChangeFlags -- flags that changed
pdwResult -- result flags (OPTIONAL)
pPrinterNotifyInfo -- Notify info data. Note that if this is NULL,
we don't set the DISCARDED flag. This is different
than PartialRPN,
Return Value:
BOOL TRUE = success
FALSE = fail
--*/
{
LPPRINTHANDLE pPrintHandle = (LPPRINTHANDLE)hPrinter;
PCHANGE pChange;
BOOL bReturn = FALSE;
EnterRouterSem();
if (!pPrintHandle ||
pPrintHandle->signature != PRINTHANDLE_SIGNATURE ||
!pPrintHandle->pChange ||
!(pPrintHandle->pChange->eStatus & STATUS_CHANGE_VALID)) {
SetLastError(ERROR_INVALID_HANDLE);
goto Done;
}
pChange = pPrintHandle->pChange;
if (pdwResult) {
*pdwResult = 0;
}
if (pChange->eStatus & STATUS_CHANGE_DISCARDED) {
DBGMSG(DBG_WARNING,
("RPCNW: Discarded x%x, eStatus = 0x%x, pInfo = 0x%x\n",
pChange,
pChange->eStatus,
pChange->ChangeInfo.pPrinterNotifyInfo));
if (pdwResult && pPrinterNotifyInfo) {
*pdwResult |= (PRINTER_NOTIFY_INFO_DISCARDED |
PRINTER_NOTIFY_INFO_DISCARDNOTED);
}
//
// If the discard has already been noted, then we don't need
// to notify the client. If it hasn't been noted, we need to
// trigger a notification, since the the client needs to refresh.
//
if (pChange->eStatus & STATUS_CHANGE_DISCARDNOTED) {
bReturn = TRUE;
goto Done;
}
}
if (pChange->eStatus & STATUS_CHANGE_INFO && pPrinterNotifyInfo) {
*pdwResult |= AppendPrinterNotifyInfo(pPrintHandle,
dwColor,
pPrinterNotifyInfo);
//
// AppendPrinterNotifyInfo will set the color mismatch
// bit if the notification info is stale (doesn't match color).
//
if (*pdwResult & PRINTER_NOTIFY_INFO_COLORMISMATCH) {
DBGMSG(DBG_WARNING, ("RPCN: Color mismatch; discarding\n"));
bReturn = TRUE;
goto Done;
}
}
//
// Store up the changes for querying later.
//
pChange->fdwChangeFlags |= fdwChangeFlags;
//
// Ensure that there is a valid notification waiting.
// This is an optimization that avoids the case where the print
// providor calls PartialRPCN several times, then is about to
// call ReplyPCN. Before it does this, we process the
// the notification (either the client picks it up, or it rpcs
// out to a remote router). Suddenly we have no notification
// data, so return instead of sending nothing.
//
if (!NotifyNeeded(pChange)) {
bReturn = TRUE;
goto Done;
}
//
// Notify Here
//
// If this is the local machine, then just set the event and update.
//
if (pChange->eStatus & STATUS_CHANGE_CLIENT) {
if (!pChange->hEvent ||
pChange->hEvent == INVALID_HANDLE_VALUE) {
DBGMSG(DBG_WARNING, ("ReplyNotify invalid event\n"));
SetLastError(ERROR_INVALID_HANDLE);
goto Done;
}
if (!SetEvent(pChange->hEvent)) {
//
// SetEvent failed!
//
DBGMSG(DBG_ERROR, ("ReplyNotify SetEvent Failed (ignore it!): Error %d.\n", GetLastError()));
goto Done;
}
//
// Keep count of notifications so that we return the correct
// number of FNPCNs.
//
pChange->dwCount++;
DBGMSG(DBG_NOTIFY, (">>>> Local trigger 0x%x\n", fdwChangeFlags));
bReturn = TRUE;
} else {
//
// Remote case.
//
// Note: pPrintHandle is invalid, since hNotify is valid only in the
// client router address space.
//
DBGMSG(DBG_NOTIFY, ("*** Trigger remote event *** 0x%x\n",
pPrintHandle));
bReturn = ThreadNotify(pPrintHandle);
}
Done:
LeaveRouterSem();
return bReturn;
}
BOOL
FreeChange(
PCHANGE pChange)
/*++
Routine Description:
Frees the change structure.
Arguments:
Return Value:
TRUE = Deleted
FALSE = deferred.
NOTE: Assumes in Critical Section
--*/
{
RouterInSem();
//
// Remove ourselves from the list if the providor wanted us
// to send out polling notifications.
//
if (pChange->ChangeInfo.fdwStatus & PRINTER_NOTIFY_STATUS_POLL)
LinkDelete(&pChange->ChangeInfo.Link, (PLINK*)&pChangeInfoHead);
//
// pPrintHandle should never refer to the pChange again. This
// ensures that the FreePrinterHandle only frees the pChange once.
//
if (pChange->ChangeInfo.pPrintHandle) {
pChange->ChangeInfo.pPrintHandle->pChange = NULL;
pChange->ChangeInfo.pPrintHandle = NULL;
}
if (pChange->cRef || pChange->eStatus & STATUS_CHANGE_FORMING) {
pChange->eStatus |= STATUS_CHANGE_CLOSING;
DBGMSG(DBG_NOTIFY, ("FreeChange: 0x%x in use: cRef = %d\n",
pChange,
pChange->cRef));
return FALSE;
}
//
// If the pszLocalMachine is ourselves, then don't free it,
// since there's a single instance locally.
//
if (pChange->pszLocalMachine != pszSelfMachine && pChange->pszLocalMachine)
FreeSplStr(pChange->pszLocalMachine);
if (pChange->ChangeInfo.pPrinterNotifyInfo) {
RouterFreePrinterNotifyInfo(pChange->ChangeInfo.pPrinterNotifyInfo);
}
DBGMSG(DBG_NOTIFY, ("FreeChange: 0x%x ->pPrintHandle 0x%x\n",
pChange, pChange->ChangeInfo.pPrintHandle));
FreeSplMem(pChange);
return TRUE;
}
VOID
FreePrinterHandle(
PPRINTHANDLE pPrintHandle)
{
EnterRouterSem();
//
// If on wait list, force wait list to free it.
//
if (pPrintHandle->pChange) {
FreeChange(pPrintHandle->pChange);
}
//
// Inform all notifys on this printer handle that they are gone.
//
FreePrinterHandleNotifys(pPrintHandle);
LeaveRouterSem();
DBGMSG(DBG_NOTIFY, ("FreePrinterHandle: 0x%x, pChange = 0x%x, pNotify = 0x%x\n",
pPrintHandle, pPrintHandle->pNotify,
pPrintHandle->pChange));
FreeSplMem(pPrintHandle);
}
VOID
HandlePollNotifications(
VOID)
/*++
Routine Description:
This handles the pulsing of notifications for any providor that wants
to do polling. It never returns.
Arguments:
VOID
Return Value:
VOID (also never returns)
NOTE: This thread should never exit, since hpmon uses this thread
for initialization. If this thread exists, certain services
this thread initializes quit.
--*/
{
PCHANGEINFO pChangeInfo;
DWORD dwSleepTime = INFINITE;
DWORD dwTimeElapsed;
DWORD dwPreSleepTicks;
while (TRUE) {
dwPreSleepTicks = GetTickCount();
if (WaitForSingleObject(hEventPoll, dwSleepTime) == WAIT_TIMEOUT) {
dwTimeElapsed = dwSleepTime;
} else {
dwTimeElapsed = GetTickCount() - dwPreSleepTicks;
}
EnterRouterSem();
//
// Initialize sleep time to INFINITY.
//
dwSleepTime = INFINITE;
for (pChangeInfo = pChangeInfoHead;
pChangeInfo;
pChangeInfo = (PCHANGEINFO)pChangeInfo->Link.pNext) {
//
// If first time or a notification came in,
// we just want to reset the time.
//
if (pChangeInfo->bResetPollTime) {
pChangeInfo->dwPollTimeLeft = pChangeInfo->dwPollTime;
pChangeInfo->bResetPollTime = FALSE;
} else if (pChangeInfo->dwPollTimeLeft <= dwTimeElapsed) {
//
// Cause a notification.
//
ReplyPrinterChangeNotificationWorker(
pChangeInfo->pPrintHandle,
0,
pChangeInfo->fdwFilterFlags,
NULL,
NULL);
pChangeInfo->dwPollTimeLeft = pChangeInfo->dwPollTime;
} else {
//
// They've slept dwTimeElapsed, so take that off of
// their dwPollTimeLeft.
//
pChangeInfo->dwPollTimeLeft -= dwTimeElapsed;
}
//
// Now compute what is the least amout of time we wish
// to sleep before the next guy should be woken up.
//
if (dwSleepTime > pChangeInfo->dwPollTimeLeft)
dwSleepTime = pChangeInfo->dwPollTimeLeft;
}
LeaveRouterSem();
}
}
BOOL
SetupChange(
PPRINTHANDLE pPrintHandle,
HANDLE hPrinterRemote,
LPWSTR pszLocalMachine,
PDWORD pfdwStatus,
DWORD fdwOptions,
DWORD fdwFilterFlags,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PPRINTER_NOTIFY_INIT* ppPrinterNotifyInit)
/*++
Routine Description:
Sets up the pChange structure. Validates the handle,
then attempts to alloc it.
Arguments:
Return Value:
--*/
{
PCHANGE pChange;
BOOL bReturn;
RouterInSem();
if (!pPrintHandle || pPrintHandle->signature != PRINTHANDLE_SIGNATURE) {
SetLastError(ERROR_INVALID_HANDLE);
return FALSE;
}
if (pPrintHandle->pChange) {
DBGMSG(DBG_ERROR, ("FFPCN: Already watching printer handle.\n"));
//
// Error: already watched
//
SetLastError(ERROR_ALREADY_WAITING);
return FALSE;
}
pChange = AllocSplMem(sizeof(CHANGE));
DBGMSG(DBG_NOTIFY, ("FFPCN pChange allocated 0x%x\n", pChange));
if (!pChange) {
//
// Failed to allocate memory, quit.
//
return FALSE;
}
pPrintHandle->pChange = pChange;
pChange->signature = CHANGEHANDLE_SIGNATURE;
pChange->eStatus = STATUS_CHANGE_FORMING;
pChange->ChangeInfo.pPrintHandle = pPrintHandle;
pChange->ChangeInfo.fdwOptions = fdwOptions;
//
// Don't watch for Failed Connections.
//
pChange->ChangeInfo.fdwFilterFlags =
fdwFilterFlags & ~PRINTER_CHANGE_FAILED_CONNECTION_PRINTER;
pChange->hPrinterRemote = hPrinterRemote;
pChange->pszLocalMachine = pszLocalMachine;
//
// As soon as we leave the critical section, pPrintHandle
// may vanish! If this is the case, out pChange->eStatus STATUS_CHANGE_CLOSING
// bit will be set.
//
LeaveRouterSem();
//
// Once we leave the critical section, we are may try and
// alter the notification. To guard against this, we always
// check eValid.
//
bReturn = (*pPrintHandle->pProvidor->PrintProvidor.
fpFindFirstPrinterChangeNotification) (pPrintHandle->hPrinter,
fdwFilterFlags,
fdwOptions,
(HANDLE)pPrintHandle,
pfdwStatus,
pPrinterNotifyOptions,
ppPrinterNotifyInit);
EnterRouterSem();
//
// On fail exit.
//
if (!bReturn) {
pPrintHandle->pChange = NULL;
if (pChange) {
//
// We no longer need to be saved, so change
// eStatus to be 0.
//
pChange->eStatus = 0;
DBGMSG(DBG_NOTIFY, ("FFPCN: Error %d, pChange deleting 0x%x %d\n",
GetLastError(),
pChange));
FreeChange(pChange);
}
return FALSE;
}
return TRUE;
}
VOID
FailChange(
PPRINTHANDLE pPrintHandle)
{
LeaveRouterSem();
//
// Shut it down since we failed.
//
(*pPrintHandle->pProvidor->PrintProvidor.
fpFindClosePrinterChangeNotification) (pPrintHandle->hPrinter);
EnterRouterSem();
//
// We no longer need to be saved, so change
// eStatus to be 0.
//
pPrintHandle->pChange->eStatus = 0;
DBGMSG(DBG_NOTIFY, ("FFPCN: Error %d, pChange deleting 0x%x %d\n",
GetLastError(),
pPrintHandle->pChange));
FreeChange(pPrintHandle->pChange);
}
/*------------------------------------------------------------------------
Entry points for providors.
------------------------------------------------------------------------*/
BOOL
ProvidorFindFirstPrinterChangeNotification(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
HANDLE hChange,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PPRINTER_NOTIFY_INIT* ppPrinterNotifyInit)
/*++
Routine Description:
Handles any FFPCN that originates from a providor.
Localspl does this when it wants to put a notification on a port.
Arguments:
Return Value:
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
BOOL bReturnValue;
DWORD fdwStatus = 0;
bReturnValue = (*pPrintHandle->pProvidor->PrintProvidor.
fpFindFirstPrinterChangeNotification) (pPrintHandle->hPrinter,
fdwFilterFlags,
fdwOptions,
hChange,
&fdwStatus,
pPrinterNotifyOptions,
ppPrinterNotifyInit);
if (bReturnValue) {
//
// !! LATER !! Check return value of SetupReply...
//
EnterRouterSem();
SetupReplyNotification(((PPRINTHANDLE)hChange)->pChange,
fdwStatus,
fdwOptions,
pPrinterNotifyOptions,
ppPrinterNotifyInit ?
*ppPrinterNotifyInit :
NULL);
LeaveRouterSem();
}
return bReturnValue;
}
DWORD
CallRouterFindFirstPrinterChangeNotification(
HANDLE hPrinterRPC,
DWORD fdwFilterFlags,
DWORD fdwOptions,
HANDLE hPrinterLocal,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions)
/*++
Routine Description:
This is called by providers if they want to pass a notification
along to another machine. This notification must originate from
this machine but needs to be passed to another spooler.
Arguments:
hPrinter - context handle to use for communication
fdwFilterFlags - watch items
fdwOptions - watch options
hPrinterLocal - pPrinter structure valid in this address space,
and also is the sink for the notifications.
Return Value:
Error code
--*/
{
DWORD ReturnValue = 0;
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinterLocal;
EnterRouterSem();
BeginReplyClient(pPrintHandle,
REPLY_TYPE_NOTIFICATION);
LeaveRouterSem();
RpcTryExcept {
ReturnValue = RpcRemoteFindFirstPrinterChangeNotificationEx(
hPrinterRPC,
fdwFilterFlags,
fdwOptions,
pPrintHandle->pChange->pszLocalMachine,
(DWORD)hPrinterLocal,
(PRPC_V2_NOTIFY_OPTIONS)pPrinterNotifyOptions);
} RpcExcept(1) {
ReturnValue = RpcExceptionCode();
} RpcEndExcept
//
// Talking to downlevel (Daytona) box.
//
if (ReturnValue == RPC_S_PROCNUM_OUT_OF_RANGE) {
if (!pPrinterNotifyOptions) {
RpcTryExcept {
ReturnValue = RpcRemoteFindFirstPrinterChangeNotification(
hPrinterRPC,
fdwFilterFlags,
fdwOptions,
pPrintHandle->pChange->pszLocalMachine,
(DWORD)hPrinterLocal,
0,
NULL);
} RpcExcept(1) {
ReturnValue = RpcExceptionCode();
} RpcEndExcept
}
}
EnterRouterSem();
EndReplyClient(pPrintHandle,
REPLY_TYPE_NOTIFICATION);
LeaveRouterSem();
return ReturnValue;
}
BOOL
ProvidorFindClosePrinterChangeNotification(
HANDLE hPrinter)
/*++
Routine Description:
Handles any FCPCN that originates from a providor.
Localspl does this when it wants to put a notification on a port.
Arguments:
Return Value:
NOTE: Assumes a client notification was setup already.
--*/
{
LPPRINTHANDLE pPrintHandle=(LPPRINTHANDLE)hPrinter;
return (*pPrintHandle->pProvidor->PrintProvidor.
fpFindClosePrinterChangeNotification) (pPrintHandle->hPrinter);
}
/*------------------------------------------------------------------------
Entry points for spooler components.
------------------------------------------------------------------------*/
BOOL
SpoolerFindFirstPrinterChangeNotification(
HANDLE hPrinter,
DWORD fdwFilterFlags,
DWORD fdwOptions,
PHANDLE phEvent,
PPRINTER_NOTIFY_OPTIONS pPrinterNotifyOptions,
PVOID pvReserved)
{
return FindFirstPrinterChangeNotificationWorker(
hPrinter,
fdwFilterFlags,
fdwOptions,
0,
pPrinterNotifyOptions,
phEvent,
TRUE);
}
BOOL
SpoolerFindNextPrinterChangeNotification(
HANDLE hPrinter,
LPDWORD pfdwChange,
LPVOID pPrinterNotifyOptions,
LPVOID *ppPrinterNotifyInfo)
/*++
Routine Description:
Jump routine for FindNext for internal spooler components to use.
Arguments:
Return Value:
--*/
{
BOOL bReturn;
DWORD fdwFlags = 0;
if (ppPrinterNotifyInfo) {
fdwFlags = PRINTER_NOTIFY_NEXT_INFO;
}
bReturn = RouterFindNextPrinterChangeNotification(
hPrinter,
fdwFlags,
pfdwChange,
pPrinterNotifyOptions,
(PPRINTER_NOTIFY_INFO*)ppPrinterNotifyInfo);
return bReturn;
}
VOID
SpoolerFreePrinterNotifyInfo(
PPRINTER_NOTIFY_INFO pInfo)
{
RouterFreePrinterNotifyInfo(pInfo);
}
BOOL
SpoolerFindClosePrinterChangeNotification(
HANDLE hPrinter)
/*++
Routine Description:
Jump routine for FindClose for internal spooler components to use.
Arguments:
Return Value:
--*/
{
return FindClosePrinterChangeNotification(hPrinter);
}
#if DBG
VOID
EnterRouterSem()
{
EnterCriticalSection(&RouterNotifySection);
}
VOID
LeaveRouterSem(
VOID)
{
if ((DWORD)RouterNotifySection.OwningThread != GetCurrentThreadId()) {
DBGMSG(DBG_ERROR, ("Not in router semaphore\n"));
}
LeaveCriticalSection(&RouterNotifySection);
}
VOID
RouterInSem(
VOID
)
{
if ((DWORD)RouterNotifySection.OwningThread != GetCurrentThreadId()) {
DBGMSG(DBG_ERROR, ("Not in spooler semaphore\n"));
}
}
VOID
RouterOutSem(
VOID
)
{
if ((DWORD)RouterNotifySection.OwningThread == GetCurrentThreadId()) {
DBGMSG(DBG_ERROR, ("Inside spooler semaphore !!\n"));
SPLASSERT((DWORD)RouterNotifySection.OwningThread != GetCurrentThreadId());
}
}
#endif /* DBG */