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.
1623 lines
48 KiB
1623 lines
48 KiB
/*++
|
|
|
|
|
|
Copyright (c) 1990 - 1996 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
port.c
|
|
|
|
Abstract:
|
|
|
|
This module contains functions to control port threads
|
|
|
|
PrintDocumentThruPrintProcessor
|
|
CreatePortThread
|
|
DestroyPortThread
|
|
PortThread
|
|
|
|
Author:
|
|
|
|
Dave Snipp (DaveSn) 15-Mar-1991
|
|
|
|
Revision History:
|
|
|
|
KrishnaG 3-Feb-1991 - moved all monitor based functions to monitor.c
|
|
Matthew Felton (mattfe) Feb 1994 Added OpenMonitorPort CloseMonitorPort
|
|
|
|
--*/
|
|
|
|
#include <precomp.h>
|
|
#pragma hdrstop
|
|
|
|
#include "clusspl.h"
|
|
#include "filepool.hxx"
|
|
|
|
WCHAR *szFilePort = L"FILE:";
|
|
|
|
|
|
VOID
|
|
PrintDocumentThruPrintProcessor(
|
|
PINIPORT pIniPort,
|
|
PPRINTPROCESSOROPENDATA pOpenData
|
|
);
|
|
|
|
|
|
// ShutdownPorts
|
|
//
|
|
// Called when the DLL_PROCESS_DETATCH is called
|
|
// Close all portthreads
|
|
// Close all monitorports
|
|
|
|
VOID
|
|
ShutdownPorts(
|
|
PINISPOOLER pIniSpooler
|
|
)
|
|
{
|
|
PINIPORT pIniPort;
|
|
|
|
if (!pIniSpooler || (pIniSpooler == INVALID_HANDLE_VALUE))
|
|
{
|
|
return;
|
|
}
|
|
|
|
EnterSplSem();
|
|
SplInSem();
|
|
|
|
pIniPort = pIniSpooler->pIniPort;
|
|
|
|
while(pIniPort) {
|
|
|
|
DestroyPortThread(pIniPort, TRUE);
|
|
|
|
//
|
|
// Don't close monitor port since DLL_ATTACH may have been called
|
|
//
|
|
// CloseMonitorPort(pIniPort);
|
|
|
|
RemoveDeviceName(pIniPort);
|
|
pIniPort = pIniPort->pNext;
|
|
}
|
|
|
|
LeaveSplSem();
|
|
|
|
return;
|
|
}
|
|
|
|
BOOL
|
|
CreatePortThread(
|
|
PINIPORT pIniPort
|
|
)
|
|
{
|
|
DWORD ThreadId;
|
|
BOOL bReturnValue = FALSE;
|
|
|
|
SplInSem();
|
|
|
|
SPLASSERT (( pIniPort != NULL) &&
|
|
( pIniPort->signature == IPO_SIGNATURE));
|
|
|
|
// Don't bother creating a thread for ports that don't have a monitor:
|
|
|
|
if (!(pIniPort->Status & PP_MONITOR))
|
|
return TRUE;
|
|
|
|
|
|
if ( pIniPort->Status & PP_THREADRUNNING)
|
|
return TRUE;
|
|
|
|
|
|
try {
|
|
|
|
pIniPort->Semaphore = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
|
|
if ( pIniPort->Semaphore == NULL )
|
|
leave;
|
|
|
|
pIniPort->Ready = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
if ( pIniPort->Ready == NULL ) {
|
|
leave;
|
|
}
|
|
|
|
pIniPort->Status |= PP_RUNTHREAD;
|
|
|
|
pIniPort->hPortThread = CreateThread(NULL, INITIAL_STACK_COMMIT,
|
|
(LPTHREAD_START_ROUTINE)PortThread,
|
|
pIniPort,
|
|
0, &ThreadId);
|
|
|
|
|
|
if( pIniPort->hPortThread == NULL ) {
|
|
|
|
pIniPort->Status &= ~PP_RUNTHREAD;
|
|
leave;
|
|
}
|
|
|
|
if ( !SetThreadPriority(pIniPort->hPortThread, dwPortThreadPriority) ) {
|
|
DBGMSG(DBG_WARNING, ("CreatePortThread - Setting thread priority failed %d\n", GetLastError()));
|
|
}
|
|
|
|
LeaveSplSem();
|
|
|
|
// Make CreatePortThread Synchronous
|
|
|
|
WaitForSingleObject( pIniPort->Ready, INFINITE );
|
|
|
|
EnterSplSem();
|
|
SplInSem();
|
|
|
|
pIniPort->Status |= PP_THREADRUNNING;
|
|
|
|
bReturnValue = TRUE;
|
|
|
|
} finally {
|
|
|
|
if ( !bReturnValue ) {
|
|
|
|
if ( pIniPort->Semaphore != NULL ) {
|
|
|
|
CloseHandle( pIniPort->Semaphore );
|
|
pIniPort->Semaphore = NULL;
|
|
}
|
|
|
|
if ( pIniPort->Ready != NULL ) {
|
|
|
|
CloseHandle( pIniPort->Ready );
|
|
pIniPort->Ready = NULL;
|
|
SetEvent( pIniPort->hPortThreadRunning );
|
|
}
|
|
}
|
|
}
|
|
return bReturnValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
BOOL
|
|
DestroyPortThread(
|
|
PINIPORT pIniPort,
|
|
BOOL bShutdown
|
|
)
|
|
{
|
|
SplInSem();
|
|
|
|
// PortThread checks for PP_RUNTHREAD
|
|
// and exits if it is not set.
|
|
|
|
pIniPort->Status &= ~PP_RUNTHREAD;
|
|
|
|
if (pIniPort->Semaphore && !SetEvent(pIniPort->Semaphore)) {
|
|
return FALSE;
|
|
}
|
|
|
|
if( pIniPort->hPortThread != NULL) {
|
|
|
|
INCPORTREF(pIniPort);
|
|
LeaveSplSem();
|
|
|
|
if ( WaitForSingleObject( pIniPort->hPortThread, INFINITE) == WAIT_FAILED ) {
|
|
|
|
EnterSplSem();
|
|
DECPORTREF(pIniPort);
|
|
return FALSE;
|
|
}
|
|
|
|
EnterSplSem();
|
|
DECPORTREF(pIniPort);
|
|
}
|
|
|
|
if (pIniPort->hPortThread != NULL) {
|
|
|
|
CloseHandle(pIniPort->hPortThread);
|
|
pIniPort->hPortThread = NULL;
|
|
|
|
}
|
|
|
|
//
|
|
// The port may have been changed while the printer was printing.
|
|
// Thus when the port thread finally goes away now is the time to
|
|
// close the monitor. However we can't call the monitor during shutdown
|
|
// since DLL_DETACH may already have been issued to the monitor dll
|
|
//
|
|
if ( !pIniPort->cPrinters && !bShutdown)
|
|
CloseMonitorPort(pIniPort);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
VOID
|
|
RemoveIniPortFromIniJob(
|
|
PINIJOB pIniJob,
|
|
PINIPORT pIniPort
|
|
)
|
|
{
|
|
PINISPOOLER pIniSpooler = pIniJob->pIniPrinter->pIniSpooler;
|
|
|
|
NOTIFYVECTOR NotifyVector;
|
|
|
|
SplInSem();
|
|
|
|
//
|
|
// Increment the refcount since deleting the job may delete the
|
|
// pIniJob, which would delete the pIniSpooler.
|
|
//
|
|
INCSPOOLERREF( pIniSpooler );
|
|
|
|
SPLASSERT(pIniJob &&
|
|
pIniJob->signature == IJ_SIGNATURE &&
|
|
pIniJob->pIniPort);
|
|
|
|
SPLASSERT( pIniJob->pIniPort == pIniPort );
|
|
|
|
pIniPort->cJobs--;
|
|
|
|
pIniJob->pIniPort = NULL;
|
|
|
|
SPLASSERT( pIniJob->Status & JOB_DESPOOLING );
|
|
|
|
// Chained Jobs
|
|
// For a Chained Master Job do not remove JOB_DESPOOLING
|
|
// since we don't want the scheduler to reschedule this
|
|
// to another port
|
|
|
|
if ( pIniPort->pIniJob != pIniJob ) {
|
|
|
|
// Normal Path
|
|
// When NOT a chained job.
|
|
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~JOB_DESPOOLING);
|
|
|
|
COPYNV(NotifyVector, NVJobStatus);
|
|
NotifyVector[JOB_NOTIFY_TYPE] |= BIT(I_JOB_PORT_NAME) |
|
|
BIT(I_JOB_PAGES_PRINTED) |
|
|
BIT(I_JOB_BYTES_PRINTED);
|
|
|
|
SetPrinterChange( pIniJob->pIniPrinter,
|
|
pIniJob,
|
|
NotifyVector,
|
|
PRINTER_CHANGE_SET_JOB,
|
|
pIniSpooler);
|
|
}
|
|
|
|
// RestartJob() doesn't remove JOB_PRINTED or JOB_BLOCKED_DEVQ
|
|
// or JOB_DESPOOLING or JOB_COMPLETE if the despooling bit is on
|
|
// this is to avoid problems where we have completed "Printing"
|
|
// the job via a print processor and now the port thread is logging
|
|
// the job printed and sending an alert message.
|
|
|
|
|
|
if ( pIniJob->Status & JOB_RESTART )
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~( JOB_PRINTED | JOB_BLOCKED_DEVQ | JOB_COMPLETE));
|
|
|
|
DeleteJobCheck(pIniJob);
|
|
|
|
//
|
|
// pIniJob may be gone at this point.
|
|
//
|
|
|
|
//
|
|
// If we're at zero then set hEventNoPrintingJobs if it exists.
|
|
//
|
|
if( !pIniSpooler->cFullPrintingJobs &&
|
|
pIniSpooler->hEventNoPrintingJobs ){
|
|
|
|
SetEvent( pIniSpooler->hEventNoPrintingJobs );
|
|
}
|
|
|
|
//
|
|
// Matches INCSPOOLERREF at beginning of this function.
|
|
//
|
|
DECSPOOLERREF( pIniSpooler );
|
|
}
|
|
|
|
DWORD
|
|
PortThread(
|
|
PINIPORT pIniPort
|
|
)
|
|
{
|
|
DWORD rc;
|
|
PRINTPROCESSOROPENDATA OpenData;
|
|
PINIJOB pIniJob;
|
|
DWORD NextJobId = 0;
|
|
DWORD Position;
|
|
DWORD dwDevQueryPrint = 0;
|
|
DWORD dwJobDirect = 0;
|
|
DWORD dwDevQueryPrintStatus = 0;
|
|
WCHAR ErrorString[MAX_PATH];
|
|
BOOL bRawDatatype;
|
|
|
|
//
|
|
// Power management. While we have port threads, we don't want the
|
|
// system to go to sleep. Note that if we have a hung job, we will
|
|
// not go to sleep.
|
|
//
|
|
SetThreadExecutionState( ES_SYSTEM_REQUIRED | ES_CONTINUOUS );
|
|
|
|
EnterSplSem();
|
|
|
|
INCSPOOLERREF( pIniPort->pIniSpooler );
|
|
|
|
SPLASSERT( pIniPort->signature == IPO_SIGNATURE );
|
|
|
|
if ( pIniPort->Status & PP_MONITOR ) {
|
|
|
|
if ( pIniPort->Status & PP_FILE )
|
|
{
|
|
rc = StatusFromHResult(OpenMonitorPort(pIniPort,
|
|
NULL,
|
|
NULL));
|
|
if (rc == ERROR_SUCCESS)
|
|
{
|
|
ReleaseMonitorPort(pIniPort);
|
|
}
|
|
|
|
DBGMSG(DBG_TRACE, (" After opening the file pseudo monitor port %d\n", rc));
|
|
|
|
} else {
|
|
// LPRMON returns NULL ( fails and expect us to open it again
|
|
// inside PrintingDirectlyToPort, so for now remove this assert
|
|
// since OpenMonitorPort was added to PrintingDirectlyToPort
|
|
// SPLASSERT( pIniPort->hPort != NULL );
|
|
}
|
|
}
|
|
|
|
SetEvent( pIniPort->Ready );
|
|
ResetEvent( pIniPort->hPortThreadRunning );
|
|
|
|
while (TRUE) {
|
|
|
|
SplInSem();
|
|
SPLASSERT( pIniPort->signature == IPO_SIGNATURE );
|
|
|
|
DBGMSG(DBG_TRACE, ("Re-entering the Port Loop -- will blow away any Current Job\n"));
|
|
|
|
pIniPort->Status |= PP_WAITING;
|
|
//
|
|
// Signal Ready event, to tell CreatePortThread that the Loop has successfully started.
|
|
// hPortThreadRunning is signalled in the loop for each port it goes through. (and once
|
|
// before leaving the port thread).
|
|
//
|
|
|
|
CHECK_SCHEDULER();
|
|
|
|
DBGMSG( DBG_PORT, ("Port %ws: WaitForSingleObject( %x )\n",
|
|
pIniPort->pName, pIniPort->Semaphore ) );
|
|
|
|
LeaveSplSem();
|
|
SplOutSem();
|
|
|
|
//
|
|
// Any modification to the pIniPort structure by other threads
|
|
// can be done only at this point.
|
|
//
|
|
|
|
rc = WaitForSingleObject( pIniPort->Semaphore, INFINITE );
|
|
|
|
EnterSplSem();
|
|
SplInSem();
|
|
|
|
SPLASSERT( pIniPort->signature == IPO_SIGNATURE );
|
|
|
|
DBGMSG( DBG_PORT, ("Port %ws: WaitForSingleObject( %x ) returned\n",
|
|
pIniPort->pName, pIniPort->Semaphore));
|
|
|
|
if ( !( pIniPort->Status & PP_RUNTHREAD ) ) {
|
|
|
|
DBGMSG(DBG_TRACE, ("Thread for Port %ws Closing Down\n", pIniPort->pName));
|
|
|
|
pIniPort->Status &= ~(PP_THREADRUNNING | PP_WAITING);
|
|
CloseHandle( pIniPort->Semaphore );
|
|
pIniPort->Semaphore = NULL;
|
|
CloseHandle( pIniPort->Ready );
|
|
pIniPort->Ready = NULL;
|
|
SetEvent(pIniPort->hPortThreadRunning);
|
|
|
|
|
|
if ( pIniPort->Status & PP_FILE )
|
|
{
|
|
rc = StatusFromHResult(CloseMonitorPort(pIniPort));
|
|
DBGMSG(DBG_TRACE, (" After closing the file pseudo monitor port\n %d\n"));
|
|
|
|
DBGMSG( DBG_WARN,
|
|
( "PortThread: DecSpoolerRef %x\n",
|
|
pIniPort->pIniSpooler ));
|
|
}
|
|
|
|
DECSPOOLERREF( pIniPort->pIniSpooler );
|
|
|
|
LeaveSplSem();
|
|
SplOutSem();
|
|
|
|
//
|
|
// Power management. We are done.
|
|
//
|
|
SetThreadExecutionState(ES_CONTINUOUS);
|
|
|
|
ExitThread (FALSE);
|
|
}
|
|
|
|
//
|
|
// Bad assumption -- that at this point we definitely have a Job
|
|
//
|
|
|
|
if ( ( pIniJob = pIniPort->pIniJob ) &&
|
|
pIniPort->pIniJob->pIniPrintProc ) {
|
|
|
|
SPLASSERT( pIniJob->signature == IJ_SIGNATURE );
|
|
SPLASSERT( pIniJob->Status & JOB_DESPOOLING );
|
|
//
|
|
// WMI Trace Events
|
|
//
|
|
INCJOBREF(pIniJob);
|
|
LeaveSplSem();
|
|
LogWmiTraceEvent(pIniJob->JobId, EVENT_TRACE_TYPE_SPL_PRINTJOB, NULL);
|
|
EnterSplSem();
|
|
DECJOBREF(pIniJob);
|
|
|
|
|
|
|
|
DBGMSG(DBG_PORT, ("Port %ws: received job\n", pIniPort->pName));
|
|
|
|
SPLASSERT(pIniJob->cRef != 0);
|
|
DBGMSG(DBG_PORT, ("PortThread(1):cRef = %d\n", pIniJob->cRef));
|
|
|
|
//
|
|
// !! HACK !!
|
|
//
|
|
// If the datatype is 1.008 but the print proc doesn't support it,
|
|
// then change it to 1.003 just for the print proc.
|
|
//
|
|
// This happens for the lexmark print processor. They support
|
|
// NT EMF 1.003, but not 1.008. They just call GdiPlayEMF, so
|
|
// they really can support 1.008 since they don't look at the
|
|
// data. However, since they don't advertise this, they can't
|
|
// print.
|
|
//
|
|
// We work around this by switching the datatype back to 1.003.
|
|
//
|
|
if (!_wcsicmp(pIniJob->pDatatype, gszNT5EMF) &&
|
|
!CheckDataTypes(pIniJob->pIniPrintProc, gszNT5EMF))
|
|
{
|
|
OpenData.pDatatype = AllocSplStr(gszNT4EMF);
|
|
}
|
|
else
|
|
{
|
|
OpenData.pDatatype = AllocSplStr(pIniJob->pDatatype);
|
|
}
|
|
|
|
OpenData.pDevMode = AllocDevMode(pIniJob->pDevMode);
|
|
OpenData.pParameters = AllocSplStr(pIniJob->pParameters);
|
|
OpenData.JobId = pIniJob->JobId;
|
|
OpenData.pDocumentName = AllocSplStr(pIniJob->pDocument);
|
|
OpenData.pOutputFile = AllocSplStr(pIniJob->pOutputFile);
|
|
|
|
//
|
|
// Check if we have RAW Printing
|
|
//
|
|
bRawDatatype = ValidRawDatatype(pIniJob->pDatatype);
|
|
|
|
OpenData.pPrinterName = pszGetPrinterName(
|
|
pIniJob->pIniPrinter,
|
|
pIniPort->pIniSpooler != pLocalIniSpooler,
|
|
NULL );
|
|
|
|
//
|
|
// If all of the allocations succeeded, then call into the print
|
|
// processor, otherwise, we discard the job for now. Some of this
|
|
// data can be NULL (pParameters, pOutputFile).
|
|
//
|
|
if (OpenData.pDatatype && OpenData.pPrinterName &&
|
|
(OpenData.pDevMode || !pIniJob->pDevMode) &&
|
|
(OpenData.pParameters || !pIniJob->pParameters) &&
|
|
(OpenData.pDocumentName || !pIniJob->pDocument) &&
|
|
(OpenData.pOutputFile || !pIniJob->pOutputFile) ) {
|
|
|
|
|
|
dwDevQueryPrint = pIniJob->pIniPrinter->Attributes & PRINTER_ATTRIBUTE_ENABLE_DEVQ;
|
|
|
|
if ((pIniJob->Status & JOB_DIRECT) ||
|
|
((pIniJob->Status & JOB_DOWNLEVEL) &&
|
|
ValidRawDatatype(pIniJob->pDatatype))) {
|
|
|
|
dwJobDirect = 1;
|
|
|
|
}
|
|
|
|
//
|
|
// If we are restarting to print a document
|
|
// clear its counters and remove the restart flag
|
|
//
|
|
if ( pIniJob->Status & JOB_RESTART ) {
|
|
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~(JOB_RESTART | JOB_INTERRUPTED));
|
|
|
|
pIniJob->cbPrinted = 0;
|
|
pIniJob->cPagesPrinted = 0;
|
|
|
|
//
|
|
// Only use dwReboots if not RAW.
|
|
//
|
|
if (!bRawDatatype)
|
|
{
|
|
//
|
|
// Solves bug 229913;
|
|
// Decrement number of reboots if job is restarted.
|
|
// ReadShadowJob checks the number of reboots and delete the job if too many
|
|
//
|
|
if( pIniJob->dwReboots ){
|
|
pIniJob->dwReboots--;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// Job is being restarted, so clear all errors?
|
|
//
|
|
ClearJobError( pIniJob );
|
|
pIniJob->dwAlert = 0;
|
|
|
|
//
|
|
// Only use dwReboots if not RAW.
|
|
//
|
|
if (!bRawDatatype)
|
|
{
|
|
pIniJob->dwReboots++;
|
|
WriteShadowJob(pIniJob, TRUE);
|
|
}
|
|
|
|
LeaveSplSem();
|
|
SplOutSem();
|
|
|
|
if ( ( dwDevQueryPrintStatus = CallDevQueryPrint(OpenData.pPrinterName,
|
|
OpenData.pDevMode,
|
|
ErrorString,
|
|
MAX_PATH,
|
|
dwDevQueryPrint,
|
|
dwJobDirect) ) ) {
|
|
|
|
PrintDocumentThruPrintProcessor( pIniPort, &OpenData );
|
|
|
|
}
|
|
|
|
SplOutSem();
|
|
EnterSplSem();
|
|
}
|
|
else {
|
|
|
|
DiscardJobFromPortThread(pIniJob, TRUE);
|
|
}
|
|
|
|
//
|
|
// Decrement number of EMF jobs rendering and update available memory
|
|
//
|
|
RemoveFromJobList(pIniJob, JOB_SCHEDULE_LIST);
|
|
|
|
SPLASSERT( pIniPort->signature == IPO_SIGNATURE );
|
|
SPLASSERT( pIniPort->pIniJob != NULL );
|
|
SPLASSERT( pIniJob == pIniPort->pIniJob);
|
|
SPLASSERT( pIniJob->signature == IJ_SIGNATURE );
|
|
|
|
//
|
|
// Chained Jobs
|
|
// If we have a chain of jobs, we now need to find the next job in the chain
|
|
// and make sure its printed to the same port.
|
|
//
|
|
|
|
if (!( pIniJob->Status & ( JOB_PENDING_DELETION | JOB_RESTART )) &&
|
|
( pIniJob->pCurrentIniJob != NULL ) &&
|
|
( pIniJob->pCurrentIniJob->NextJobId != 0 )) {
|
|
|
|
// Follow the Chained Job to the Next Job
|
|
// Look at scheduler to see where it picks up this job and assigns it back
|
|
// to this port thread.
|
|
|
|
pIniJob->pCurrentIniJob = FindJob( pIniJob->pIniPrinter, pIniJob->pCurrentIniJob->NextJobId, &Position );
|
|
|
|
if ( pIniJob->pCurrentIniJob == NULL ) {
|
|
|
|
pIniPort->pIniJob = NULL;
|
|
|
|
DBGMSG( DBG_WARNING, ("PortThread didn't find NextJob\n"));
|
|
|
|
} else {
|
|
|
|
SPLASSERT( pIniJob->pCurrentIniJob->signature == IJ_SIGNATURE );
|
|
|
|
DBGMSG( DBG_WARNING, ("PortThread completed JobId %d, NextJobId %d\n", pIniJob->JobId,
|
|
pIniJob->pCurrentIniJob->JobId ));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
// Nothing More in Chain
|
|
//
|
|
|
|
pIniJob->pCurrentIniJob = NULL;
|
|
pIniPort->pIniJob = NULL;
|
|
}
|
|
|
|
if( !pIniJob->pCurrentIniJob ){
|
|
|
|
//
|
|
// Decrement the pIniSpooler job count. We only decrement
|
|
// at the end of a chain since we don't increment in the
|
|
// middle of a chained job.
|
|
//
|
|
--pIniJob->pIniPrinter->pIniSpooler->cFullPrintingJobs;
|
|
}
|
|
|
|
DBGMSG(DBG_PORT, ("PortThread job has now printed - status:0x%0x\n", pIniJob->Status));
|
|
|
|
FreeDevMode(OpenData.pDevMode);
|
|
FreeSplStr(OpenData.pDatatype);
|
|
FreeSplStr(OpenData.pParameters);
|
|
FreeSplStr(OpenData.pDocumentName);
|
|
FreeSplStr(OpenData.pOutputFile);
|
|
FreeSplStr(OpenData.pPrinterName);
|
|
|
|
pIniJob->Time = GetTickCount() - pIniJob->Time;
|
|
|
|
if (!dwDevQueryPrintStatus) {
|
|
|
|
DBGMSG(DBG_PORT, ("PortThread Job has not printed because of DevQueryPrint failed\n"));
|
|
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_BLOCKED_DEVQ);
|
|
SPLASSERT( !(pIniJob->Status & JOB_PRINTED));
|
|
pIniJob->Time = 0;
|
|
|
|
FreeSplStr( pIniJob->pStatus );
|
|
pIniJob->pStatus = AllocSplStr(ErrorString);
|
|
|
|
SetPrinterChange(pIniJob->pIniPrinter,
|
|
pIniJob,
|
|
NVJobStatusAndString,
|
|
PRINTER_CHANGE_SET_JOB,
|
|
pIniJob->pIniPrinter->pIniSpooler );
|
|
|
|
} else if ( !( pIniJob->Status & JOB_TIMEOUT ) ) {
|
|
|
|
|
|
//
|
|
// Only Log the event and send a popup if the last in the chain
|
|
//
|
|
|
|
if ( !(pIniJob->Status & JOB_RESTART) &&
|
|
pIniJob->pCurrentIniJob == NULL ) {
|
|
|
|
//
|
|
// A job can be in JOB_COMPLETE state when it was sent ot printer
|
|
// but the last page isn't ejected yet. A job completely sent to printer
|
|
// can be either in JOB_COMPLETE or JOB_PRINTED state.
|
|
// Monitors that doesn't support TEOJ will set the job as JOB_PRINTED
|
|
// right after the job was sent to printer and we don't want to set it on
|
|
// JOB_COMPLETE.
|
|
// For BIDI Monitors we'll come down here possibly before
|
|
// the monitor sets the job as JOB_PRINTED. We set the job as JOB_COMPLETE
|
|
// so that the scheduler will ignore it.
|
|
//
|
|
if (!(pIniJob->Status & (JOB_ERROR | JOB_PAPEROUT | JOB_OFFLINE)) &&
|
|
!(pIniJob->Status & JOB_PRINTED))
|
|
{
|
|
if (pIniJob->cPages == 0 &&
|
|
(pIniJob->Size == 0 || pIniJob->dwValidSize == 0) &&
|
|
!(pIniJob->Status & JOB_TYPE_ADDJOB))
|
|
{
|
|
//
|
|
// Set empty document to Printed as the monitor won't do it.
|
|
// Make exception for job submitted with AddJob. The monitor
|
|
// is still in charge for doing it.
|
|
//
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_PRINTED);
|
|
}
|
|
else
|
|
{
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_COMPLETE);
|
|
}
|
|
}
|
|
|
|
|
|
// For Remote NT Jobs cPagesPrinted and cTotalPagesPrinted
|
|
// are NOT updated since we are getting RAW data. So we
|
|
// use the cPages field instead.
|
|
|
|
if (pIniJob->cPagesPrinted == 0) {
|
|
pIniJob->cPagesPrinted = pIniJob->cPages;
|
|
pIniJob->pIniPrinter->cTotalPagesPrinted += pIniJob->cPages;
|
|
}
|
|
|
|
INCJOBREF(pIniJob);
|
|
LeaveSplSem();
|
|
|
|
LogJobPrinted(pIniJob);
|
|
|
|
EnterSplSem();
|
|
DECJOBREF(pIniJob);
|
|
}
|
|
|
|
}
|
|
|
|
SplInSem();
|
|
|
|
DBGMSG(DBG_PORT, ("PortThread(2):cRef = %d\n", pIniJob->cRef));
|
|
|
|
// Hi End Print Shops like to keep around jobs after they have
|
|
// completed. They do this so they can print a proof it and then
|
|
// print it again for the final run. Spooling the job again may take
|
|
// several hours which they want to avoid.
|
|
// Even if KEEPPRINTEDJOBS is set they can still manually delete
|
|
// the job via printman.
|
|
|
|
if (( pIniJob->pIniPrinter->Attributes & PRINTER_ATTRIBUTE_KEEPPRINTEDJOBS ) ||
|
|
( pIniJob->Status & JOB_TIMEOUT ) ) {
|
|
|
|
//
|
|
// WMI Trace Events.
|
|
//
|
|
// Treat a keep and restart as a delete and spool new job.
|
|
WMI_SPOOL_DATA WmiData;
|
|
DWORD CreateInfo;
|
|
if (GetFileCreationInfo(pIniJob->hFileItem, &CreateInfo) != S_OK) {
|
|
// Assume all file created.
|
|
CreateInfo = FP_ALL_FILES_CREATED;
|
|
}
|
|
SplWmiCopyEndJobData(&WmiData, pIniJob, CreateInfo);
|
|
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~JOB_PENDING_DELETION);
|
|
pIniJob->cbPrinted = 0;
|
|
|
|
//
|
|
// Set the job as JOB_COMPLETE if not already set as JOB_PRINTED
|
|
// by Monitor.
|
|
// Monitors that doesn't support TEOJ will set the job as JOB_PRINTED
|
|
// right after the job was sent to printer and we don't want to set it on
|
|
// JOB_COMPLETE.
|
|
// For BIDI Monitors we'll come down here possibly before
|
|
// the monitor sets the job as JOB_PRINTED. We set the job as JOB_COMPLETE
|
|
// so that the scheduler will ignore it.
|
|
//
|
|
if (!(pIniJob->Status & JOB_PRINTED)) {
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_COMPLETE);
|
|
}
|
|
|
|
//
|
|
// Only use dwReboots if not RAW.
|
|
//
|
|
if (!bRawDatatype)
|
|
{
|
|
--pIniJob->dwReboots;
|
|
}
|
|
|
|
//
|
|
// We need to update the shadow file regardless the job type.
|
|
// There is the job status that we need to update.
|
|
//
|
|
INCJOBREF(pIniJob);
|
|
|
|
//
|
|
// WriteShadowJob leaves the CS, So make sure that the ref on the
|
|
// pIniJob is kept high.
|
|
//
|
|
WriteShadowJob(pIniJob, TRUE);
|
|
|
|
LeaveSplSem();
|
|
//
|
|
// WMI Trace Events.
|
|
//
|
|
// The job is done. If it is restarted you get a new spool job event.
|
|
LogWmiTraceEvent(pIniJob->JobId, EVENT_TRACE_TYPE_SPL_DELETEJOB,
|
|
&WmiData);
|
|
|
|
EnterSplSem();
|
|
DECJOBREF(pIniJob);
|
|
|
|
|
|
SPLASSERT( pIniPort->signature == IPO_SIGNATURE );
|
|
SPLASSERT( pIniJob->signature == IJ_SIGNATURE );
|
|
|
|
}
|
|
|
|
SplInSem();
|
|
|
|
SPLASSERT( pIniJob->cRef != 0 );
|
|
DECJOBREF(pIniJob);
|
|
|
|
RemoveIniPortFromIniJob(pIniJob, pIniPort);
|
|
|
|
//
|
|
// N.B. The pIniJob may be gone at this point.
|
|
//
|
|
|
|
} else {
|
|
|
|
//
|
|
// !! VERIFY !!
|
|
//
|
|
SPLASSERT(pIniJob != NULL);
|
|
|
|
if (pIniJob != NULL) {
|
|
|
|
DBGMSG(DBG_PORT, ("Port %ws: deleting job\n", pIniPort->pName));
|
|
|
|
// SPLASSERT( pIniJob->Time != 0 );
|
|
pIniJob->Time = GetTickCount() - pIniJob->Time;
|
|
//pIniJob->Status |= JOB_PRINTED;
|
|
|
|
if ( pIniJob->hFileItem == INVALID_HANDLE_VALUE )
|
|
{
|
|
CloseHandle( pIniJob->hWriteFile );
|
|
}
|
|
pIniJob->hWriteFile = INVALID_HANDLE_VALUE;
|
|
|
|
DBGMSG(DBG_PORT, ("Port %ws - calling DeleteJob because PrintProcessor wasn't available\n"));
|
|
RemoveIniPortFromIniJob(pIniJob, pIniPort);
|
|
|
|
DeleteJob(pIniJob,BROADCAST);
|
|
|
|
//
|
|
// N.B. The pIniJob may be gone at this point.
|
|
//
|
|
}
|
|
}
|
|
|
|
//SetCurrentSid(NULL);
|
|
DBGMSG(DBG_PORT,("Returning back to pickup a new job or to delete the PortThread\n"));
|
|
|
|
}
|
|
|
|
SetEvent( pIniPort->hPortThreadRunning );
|
|
|
|
SPLASSERT( FALSE );
|
|
return 0;
|
|
}
|
|
|
|
VOID
|
|
PrintDocumentThruPrintProcessor(
|
|
PINIPORT pIniPort,
|
|
PPRINTPROCESSOROPENDATA pOpenData
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print the document associated with pIniPort on the print
|
|
processor.
|
|
|
|
Status of pIniPort->Status = PP_RUNTHREAD
|
|
PP_THREADRUNNING
|
|
PP_MONITOR
|
|
~PP_WAITING
|
|
|
|
NOTE: If PrintProc->Open is called and succeeds, PrintProc->Close
|
|
must be called to cleanup.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
PINIJOB pIniJob = pIniPort->pIniJob;
|
|
WCHAR szSpecialPortorPrinterName[MAX_UNC_PRINTER_NAME + MAX_PATH + PRINTER_NAME_SUFFIX_MAX];
|
|
BOOL bJobError = FALSE;
|
|
NOTIFYVECTOR NotifyVector;
|
|
LPTSTR pszModify;
|
|
UINT cchLen;
|
|
BOOL bFailJob = FALSE;
|
|
BOOL bRemoteGuest = FALSE;
|
|
BOOL bSpecialCaseDriver = FALSE;
|
|
DWORD Error;
|
|
|
|
//
|
|
// Check if printing principal is remote guest. Remote guest does not have enough
|
|
// permissions to print EMF. The EMF playback code in GDI32 fails for certain EMF records.
|
|
// Because of this, we create an impersonation token based on the process token.
|
|
//
|
|
if ((bSpecialCaseDriver = IsSpecialDriver(pIniJob->pIniDriver, pIniJob->pIniPrintProc, pIniJob->pIniPrinter->pIniSpooler)) ||
|
|
(Error = PrincipalIsRemoteGuest(pIniJob->hToken, &bRemoteGuest)) == ERROR_SUCCESS)
|
|
{
|
|
if (bRemoteGuest || bSpecialCaseDriver)
|
|
{
|
|
Error = ImpersonateSelf(SecurityImpersonation) ? ERROR_SUCCESS : GetLastError();
|
|
}
|
|
else
|
|
{
|
|
Error = SetCurrentSid(pIniJob->hToken) ? ERROR_SUCCESS : GetLastError();
|
|
}
|
|
}
|
|
|
|
if (Error != ERROR_SUCCESS)
|
|
{
|
|
LogPrintProcError(Error, pIniJob);
|
|
|
|
bFailJob = TRUE;
|
|
|
|
goto Complete;
|
|
}
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor pIniPort %x pOpenData %x\n", pIniPort, pOpenData));
|
|
|
|
COPYNV(NotifyVector, NVJobStatus);
|
|
|
|
cchLen = lstrlen( pIniJob->pIniPrinter->pIniSpooler->pMachineName );
|
|
|
|
//
|
|
// Do a length check. PRINTER_NAME_SUFFIX_MAX holds the extra 4 separator
|
|
// characters and the NULL terminator.
|
|
// We need to be able to fit both the port name and the printer name into
|
|
// the buffer, plus the suffix and internal characters.
|
|
//
|
|
if (lstrlen(pIniPort->pName) + cchLen + PRINTER_NAME_SUFFIX_MAX > COUNTOF(szSpecialPortorPrinterName) ||
|
|
lstrlen(pIniJob->pIniPrinter->pName) + cchLen + PRINTER_NAME_SUFFIX_MAX > COUNTOF(szSpecialPortorPrinterName)) {
|
|
|
|
//
|
|
// We should log an event, but this is a very rare event, only
|
|
// in the print api tests.
|
|
//
|
|
bFailJob = TRUE;
|
|
goto Complete;
|
|
}
|
|
|
|
//
|
|
// For clustered spoolers, make sure it is fully qualified.
|
|
// pszModify points to the string immediately after the server
|
|
// name that can be modified.
|
|
//
|
|
// Always modify the string at pszModify, but pass in
|
|
// szSpecialPortorPrinterName.
|
|
//
|
|
StrNCatBuff(szSpecialPortorPrinterName, COUNTOF(szSpecialPortorPrinterName), pIniJob->pIniPrinter->pIniSpooler->pMachineName, L"\\", NULL);
|
|
|
|
pszModify = &szSpecialPortorPrinterName[cchLen+1];
|
|
|
|
//
|
|
// \\Server\
|
|
// ^---------------------- szSpecialPortorPrinterName
|
|
// ^------------- pszModify
|
|
//
|
|
// Append the rest of the string at pszModify:
|
|
//
|
|
// \\Server\PortName, Port
|
|
// \\Server\PrinterName, Job 33
|
|
//
|
|
|
|
//
|
|
// Now create the port name, so that we can do the
|
|
// secret open printer. the printer name will be
|
|
// "FILE:, Port" and this will open a PRINTER_HANDLE_PORT
|
|
// If we fail, then if the app thread may be waiting for
|
|
// the pIniJob->StartDocComplete to be set, which would
|
|
// ordinarily be done in the StartDocPrinter of the port.
|
|
// We will do this little courtesy,
|
|
//
|
|
StringCchPrintf(pszModify, COUNTOF(szSpecialPortorPrinterName) - (pszModify - szSpecialPortorPrinterName), L"%ws, Port", pIniPort->pName );
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor Attempting PrintProcessor Open on %ws\n", szSpecialPortorPrinterName ));
|
|
|
|
if (!(pIniPort->hProc = (HANDLE)(*pIniJob->pIniPrintProc->Open)
|
|
(szSpecialPortorPrinterName, pOpenData))) {
|
|
|
|
|
|
DBGMSG( DBG_WARNING, ("PrintDocumentThruPrintProcessor Failed Open error %d\n", GetLastError() ));
|
|
|
|
bFailJob = TRUE;
|
|
goto Complete;
|
|
}
|
|
|
|
//
|
|
// For Jobs, turn this off even if it's not clustered.
|
|
//
|
|
if( !( pIniJob->pIniPrinter->pIniSpooler->SpoolerFlags & SPL_TYPE_CLUSTER )){
|
|
|
|
pszModify = szSpecialPortorPrinterName;
|
|
|
|
//
|
|
//
|
|
// ^---------------------- szSpecialPortorPrinterName
|
|
// ^---------------------- pszModify
|
|
//
|
|
// PortName, Port
|
|
// PrinterName, Job 33
|
|
//
|
|
}
|
|
|
|
EnterSplSem();
|
|
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_PRINTING);
|
|
pIniJob->Time = GetTickCount();
|
|
|
|
NotifyVector[JOB_NOTIFY_TYPE] |= BIT(I_JOB_PORT_NAME);
|
|
|
|
SetPrinterChange(pIniJob->pIniPrinter,
|
|
pIniJob,
|
|
NotifyVector,
|
|
PRINTER_CHANGE_SET_JOB,
|
|
pIniJob->pIniPrinter->pIniSpooler);
|
|
|
|
|
|
LeaveSplSem();
|
|
|
|
//
|
|
// Create Special Name "PrinterName, Job xxx"
|
|
//
|
|
StringCchPrintf(pszModify, COUNTOF(szSpecialPortorPrinterName) - (pszModify - szSpecialPortorPrinterName), L"%ws, Job %d", pIniJob->pIniPrinter->pName, pIniJob->JobId);
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor calling Print hProc %x file %ws\n",
|
|
pIniPort->hProc, szSpecialPortorPrinterName ));
|
|
|
|
if (!(*pIniJob->pIniPrintProc->Print)(pIniPort->hProc, szSpecialPortorPrinterName)) {
|
|
|
|
Error = GetLastError();
|
|
|
|
if (ERROR_PRINT_CANCELLED != Error)
|
|
{
|
|
LogPrintProcError(Error, pIniJob);
|
|
}
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor Print hProc %x Error %d\n", pIniPort->hProc, Error));
|
|
|
|
|
|
EnterSplSem();
|
|
|
|
if ( pIniJob->StartDocComplete ) {
|
|
SetEvent( pIniJob->StartDocComplete );
|
|
}
|
|
|
|
bJobError = TRUE;
|
|
|
|
LeaveSplSem();
|
|
|
|
} else {
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor Print hProc %x %ws Success\n",
|
|
pIniPort->hProc, szSpecialPortorPrinterName ));
|
|
}
|
|
|
|
//
|
|
// Now close the print processor.
|
|
//
|
|
|
|
EnterSplSem();
|
|
|
|
SPLASSERT( pIniPort->hProc != NULL );
|
|
|
|
DBGMSG( DBG_TRACE, ("PrintDocumentThruPrintProcessor calling Close hProc %x\n", pIniPort->hProc ));
|
|
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~JOB_PRINTING);
|
|
|
|
LeaveSplSem();
|
|
|
|
//
|
|
// JOB_PP_CLOSE is used to prevent the print processor from recursively
|
|
// calling back into itself. This happens for some third party print processor.
|
|
// Race conditions don't apply for this flag since 2 threads don't access it
|
|
// simultaneously.
|
|
//
|
|
|
|
if (!(pIniJob->Status & JOB_PP_CLOSE))
|
|
{
|
|
|
|
EnterCriticalSection(&pIniJob->pIniPrintProc->CriticalSection);
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_PP_CLOSE);
|
|
|
|
if (!(*pIniJob->pIniPrintProc->Close)(pIniPort->hProc))
|
|
{
|
|
DBGMSG( DBG_WARNING, ("PrintDocumentThruPrintProcessor failed Close hProc %x Error %d\n",
|
|
pIniPort->hProc, GetLastError() ));
|
|
}
|
|
|
|
pIniPort->hProc = NULL;
|
|
InterlockedAnd((LONG*)&(pIniJob->Status), ~JOB_PP_CLOSE);
|
|
LeaveCriticalSection(&pIniJob->pIniPrintProc->CriticalSection);
|
|
|
|
//
|
|
// WMI Trace Events
|
|
//
|
|
if (pIniJob->pDevMode)
|
|
{
|
|
WMI_SPOOL_DATA Data;
|
|
SplWmiCopyRenderedData(&Data, pIniJob->pDevMode);
|
|
LogWmiTraceEvent(pIniJob->JobId, EVENT_TRACE_TYPE_SPL_JOBRENDERED, &Data);
|
|
}
|
|
}
|
|
|
|
Complete:
|
|
|
|
if (bFailJob || bJobError)
|
|
{
|
|
DiscardJobFromPortThread(pIniJob, bFailJob);
|
|
}
|
|
|
|
//
|
|
// RevertToSelf and SetCurrentSid have identical behavior. If one call ImpersonateSelf
|
|
// and the SetCurrentSid instead of RevertToSelf, that's still fine.
|
|
//
|
|
if (bRemoteGuest || bSpecialCaseDriver)
|
|
{
|
|
RevertToSelf();
|
|
}
|
|
else
|
|
{
|
|
SetCurrentSid(NULL);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
UpdatePortStatusForAllPrinters(
|
|
PINIPORT pIniPort
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This routine is called when an IniPorts status changed so that we go
|
|
through each printer connected to the port and update their port status
|
|
|
|
Arguments:
|
|
pIniPort - Port whose status chanegd
|
|
|
|
Return Value:
|
|
Nothing
|
|
|
|
--*/
|
|
{
|
|
PINIPRINTER pIniPrinter;
|
|
PINIPORT pIniPrinterPort;
|
|
DWORD dwIndex1, dwIndex2, dwPortStatus, dwSeverity;
|
|
|
|
for ( dwIndex1 = 0 ; dwIndex1 < pIniPort->cPrinters ; ++dwIndex1 ) {
|
|
|
|
pIniPrinter = pIniPort->ppIniPrinter[dwIndex1];
|
|
dwSeverity = 0;
|
|
dwPortStatus = 0;
|
|
|
|
//
|
|
// Pick the most severe status associated with all ports
|
|
//
|
|
for ( dwIndex2 = 0 ; dwIndex2 < pIniPrinter->cPorts ; ++dwIndex2 ) {
|
|
|
|
pIniPrinterPort = pIniPrinter->ppIniPorts[dwIndex2];
|
|
|
|
if ( pIniPrinterPort->Status & PP_ERROR ) {
|
|
|
|
dwSeverity = PP_ERROR;
|
|
dwPortStatus = PortToPrinterStatus(pIniPrinterPort->PrinterStatus);
|
|
break; // no need to go thru rest of the ports for this printer
|
|
} else if ( pIniPrinterPort->Status & PP_WARNING ) {
|
|
|
|
if ( dwSeverity != PP_WARNING ) {
|
|
|
|
dwSeverity = PP_WARNING;
|
|
dwPortStatus = PortToPrinterStatus(pIniPrinterPort->PrinterStatus);
|
|
}
|
|
} else if ( pIniPrinterPort->Status & PP_INFORMATIONAL ) {
|
|
|
|
if ( dwSeverity == 0 ) {
|
|
|
|
dwSeverity = PP_INFORMATIONAL;
|
|
dwPortStatus = PortToPrinterStatus(pIniPrinterPort->PrinterStatus);
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( pIniPrinter->PortStatus != dwPortStatus ) {
|
|
|
|
pIniPrinter->PortStatus = dwPortStatus;
|
|
SetPrinterChange(pIniPrinter,
|
|
NULL,
|
|
NVPrinterStatus,
|
|
PRINTER_CHANGE_SET_PRINTER,
|
|
pIniPrinter->pIniSpooler);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//
|
|
// Table is by port status values in winspool.h
|
|
//
|
|
DWORD PortToPrinterStatusMappings[] = {
|
|
|
|
0,
|
|
PRINTER_STATUS_OFFLINE,
|
|
PRINTER_STATUS_PAPER_JAM,
|
|
PRINTER_STATUS_PAPER_OUT,
|
|
PRINTER_STATUS_OUTPUT_BIN_FULL,
|
|
PRINTER_STATUS_PAPER_PROBLEM,
|
|
PRINTER_STATUS_NO_TONER,
|
|
PRINTER_STATUS_DOOR_OPEN,
|
|
PRINTER_STATUS_USER_INTERVENTION,
|
|
PRINTER_STATUS_OUT_OF_MEMORY,
|
|
PRINTER_STATUS_TONER_LOW,
|
|
PRINTER_STATUS_WARMING_UP,
|
|
PRINTER_STATUS_POWER_SAVE,
|
|
};
|
|
|
|
|
|
BOOL
|
|
LocalSetPort(
|
|
LPWSTR pszName,
|
|
LPWSTR pszPortName,
|
|
DWORD dwLevel,
|
|
LPBYTE pPortInfo
|
|
)
|
|
{
|
|
PINIPORT pIniPort;
|
|
PPORT_INFO_3 pPortInfo3 = (PPORT_INFO_3) pPortInfo;
|
|
DWORD dwLastError = ERROR_SUCCESS;
|
|
DWORD dwNewStatus, dwOldStatus;
|
|
BOOL bJobStatusChanged = FALSE;
|
|
WCHAR szPort[MAX_PATH + 9];
|
|
LPWSTR pszComma;
|
|
PINISPOOLER pIniSpooler = FindSpoolerByNameIncRef( pszName, NULL );
|
|
BOOL SemEntered = FALSE;
|
|
|
|
if( !pIniSpooler )
|
|
{
|
|
dwLastError = ERROR_INVALID_NAME;
|
|
goto Cleanup;
|
|
}
|
|
|
|
|
|
if ( !MyName(pszName, pIniSpooler) ) {
|
|
|
|
dwLastError = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// The monitor needs to be able to set or clear the error for the port. The monitor
|
|
// is loaded by the spooler. If the monitor doesn't link to winspool.drv, then the
|
|
// call to SetPort comes directly, i.e. not via RPC. In this case we do not want
|
|
// to check for admin privileges. We allow any user to set the port status.
|
|
//
|
|
if (!ValidateObjectAccess(SPOOLER_OBJECT_SERVER,
|
|
IsCallViaRPC() ? SERVER_ACCESS_ADMINISTER : SERVER_ACCESS_ENUMERATE,
|
|
NULL,
|
|
NULL,
|
|
pIniSpooler )) {
|
|
|
|
dwLastError = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
if( !pszPortName ){
|
|
|
|
dwLastError = ERROR_UNKNOWN_PORT ;
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Some ports will come in as "port,1234abcd" so truncate the
|
|
// suffix.
|
|
//
|
|
StringCchCopy(szPort, COUNTOF(szPort), pszPortName);
|
|
|
|
pszComma = wcschr( szPort, TEXT( ',' ));
|
|
if( pszComma ){
|
|
*pszComma = 0;
|
|
}
|
|
|
|
SemEntered = TRUE;
|
|
|
|
EnterSplSem();
|
|
pIniPort = FindPort(szPort, pIniSpooler);
|
|
|
|
if ( !pIniPort ) {
|
|
|
|
dwLastError = ERROR_UNKNOWN_PORT;
|
|
goto Cleanup;
|
|
}
|
|
|
|
if ( dwLevel != 3 ) {
|
|
|
|
dwLastError = ERROR_INVALID_LEVEL;
|
|
goto Cleanup;
|
|
}
|
|
|
|
if ( !pPortInfo ) {
|
|
|
|
dwLastError = ERROR_INVALID_PARAMETER;
|
|
goto Cleanup;
|
|
}
|
|
|
|
switch (pPortInfo3->dwSeverity) {
|
|
case 0:
|
|
if ( pPortInfo3->dwStatus || pPortInfo3->pszStatus ) {
|
|
|
|
dwLastError = ERROR_INVALID_PARAMETER;
|
|
goto Cleanup;
|
|
}
|
|
dwNewStatus = 0;
|
|
break;
|
|
|
|
case PORT_STATUS_TYPE_ERROR:
|
|
dwNewStatus = PP_ERROR;
|
|
break;
|
|
|
|
case PORT_STATUS_TYPE_WARNING:
|
|
dwNewStatus = PP_WARNING;
|
|
break;
|
|
|
|
case PORT_STATUS_TYPE_INFO:
|
|
dwNewStatus = PP_INFORMATIONAL;
|
|
break;
|
|
|
|
default:
|
|
dwLastError = ERROR_INVALID_PARAMETER;
|
|
goto Cleanup;
|
|
}
|
|
|
|
dwOldStatus = pIniPort->Status;
|
|
|
|
//
|
|
// Clear old status
|
|
//
|
|
pIniPort->PrinterStatus = 0;
|
|
pIniPort->Status &= ~(PP_ERROR | PP_WARNING | PP_INFORMATIONAL);
|
|
|
|
if ( pIniPort->pszStatus ) {
|
|
|
|
//
|
|
// If the job currently has the same status as port free it
|
|
//
|
|
if ( pIniPort->pIniJob &&
|
|
pIniPort->pIniJob->pStatus &&
|
|
!wcscmp(pIniPort->pIniJob->pStatus, pIniPort->pszStatus) ) {
|
|
|
|
FreeSplStr(pIniPort->pIniJob->pStatus);
|
|
pIniPort->pIniJob->pStatus = NULL;
|
|
bJobStatusChanged = TRUE;
|
|
}
|
|
|
|
FreeSplStr(pIniPort->pszStatus);
|
|
pIniPort->pszStatus = NULL;
|
|
}
|
|
|
|
//
|
|
// If string field is used for status use it, else look at dwStatus
|
|
//
|
|
if ( pPortInfo3->pszStatus && *pPortInfo3->pszStatus ) {
|
|
|
|
pIniPort->pszStatus = AllocSplStr(pPortInfo3->pszStatus);
|
|
if ( !pIniPort->pszStatus ) {
|
|
dwLastError = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
if ( pIniPort->pIniJob && !pIniPort->pIniJob->pStatus ) {
|
|
|
|
pIniPort->pIniJob->pStatus = AllocSplStr(pIniPort->pszStatus);
|
|
bJobStatusChanged = TRUE;
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
// If we add new entries to winspool.h they should be added here too
|
|
//
|
|
if ( pPortInfo3->dwStatus >=
|
|
sizeof(PortToPrinterStatusMappings)/sizeof(PortToPrinterStatusMappings[0]) ) {
|
|
|
|
dwLastError = ERROR_INVALID_PARAMETER;
|
|
goto Cleanup;
|
|
}
|
|
pIniPort->PrinterStatus = pPortInfo3->dwStatus;
|
|
}
|
|
|
|
if( bJobStatusChanged ){
|
|
|
|
SetPrinterChange( pIniPort->pIniJob->pIniPrinter,
|
|
pIniPort->pIniJob,
|
|
NVJobStatusString,
|
|
PRINTER_CHANGE_SET_JOB,
|
|
pIniPort->pIniJob->pIniPrinter->pIniSpooler );
|
|
}
|
|
|
|
pIniPort->Status |= dwNewStatus;
|
|
|
|
UpdatePortStatusForAllPrinters(pIniPort);
|
|
if ( (dwOldStatus & PP_ERROR) &&
|
|
!(dwNewStatus & PP_ERROR) ) {
|
|
|
|
//
|
|
// if it is a transition to an non - error state , set event to unlock LocalWritePrinter
|
|
//
|
|
pIniPort->ErrorTime = 0;
|
|
if( pIniPort->hErrorEvent != NULL ){
|
|
|
|
SetEvent(pIniPort->hErrorEvent);
|
|
}
|
|
|
|
CHECK_SCHEDULER();
|
|
}
|
|
|
|
if ( !(dwOldStatus & PP_ERROR) &&
|
|
!(dwNewStatus & PP_ERROR) ) {
|
|
|
|
//
|
|
// when non-error state persists(after two calls with an non - error state) ,
|
|
// close the hErrorEvent handle
|
|
//
|
|
if( pIniPort->hErrorEvent != NULL ){
|
|
|
|
CloseHandle(pIniPort->hErrorEvent);
|
|
pIniPort->hErrorEvent = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
if ( !(dwOldStatus & PP_ERROR) &&
|
|
(dwNewStatus & PP_ERROR) &&
|
|
(pIniPort->cJobs) &&
|
|
(pIniSpooler->bRestartJobOnPoolEnabled) &&
|
|
( pPortInfo3->dwStatus == PORT_STATUS_OFFLINE ||
|
|
pPortInfo3->dwStatus == PORT_STATUS_PAPER_JAM ||
|
|
pPortInfo3->dwStatus == PORT_STATUS_PAPER_OUT ||
|
|
pPortInfo3->dwStatus == PORT_STATUS_DOOR_OPEN ||
|
|
pPortInfo3->dwStatus == PORT_STATUS_PAPER_PROBLEM ||
|
|
pPortInfo3->dwStatus == PORT_STATUS_NO_TONER)) {
|
|
|
|
//
|
|
// If it is a transition to an error state and port has an job assigned, create event as non-signalled or Reset event it
|
|
// LocalWritePrinter will get stuck if the port is in error state and this event is reset
|
|
//
|
|
if( pIniPort->ErrorTime == 0 ){
|
|
|
|
pIniPort->ErrorTime = GetTickCount();
|
|
|
|
if( pIniPort->hErrorEvent == NULL ){
|
|
|
|
pIniPort->hErrorEvent = CreateEvent(NULL,
|
|
EVENT_RESET_MANUAL,
|
|
EVENT_INITIAL_STATE_NOT_SIGNALED,
|
|
NULL );
|
|
}else{
|
|
|
|
ResetEvent( pIniPort->hErrorEvent );
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if ( (dwOldStatus & PP_ERROR) &&
|
|
(dwNewStatus & PP_ERROR) &&
|
|
(pIniPort->cJobs) &&
|
|
(pIniPort->hErrorEvent != NULL) ) {
|
|
|
|
//
|
|
// When error state persists , check the time since error occured.
|
|
//
|
|
if( (GetTickCount() - pIniPort->ErrorTime) > pIniSpooler->dwRestartJobOnPoolTimeout * 1000 ){
|
|
|
|
//
|
|
// If time out and printer is a pool ( more than one port assigned ),
|
|
// clear job error and restart the job.
|
|
//
|
|
if( (pIniPort->pIniJob) &&
|
|
(pIniPort->pIniJob->pIniPrinter) &&
|
|
(pIniPort->pIniJob->pIniPrinter->cPorts > 1) ){
|
|
|
|
//
|
|
// Don't restart the job if it is already deleted or restarted.
|
|
//
|
|
BOOL bWasRestarted = pIniPort->pIniJob->Status & (JOB_PENDING_DELETION | JOB_RESTART);
|
|
|
|
if( !bWasRestarted ){
|
|
|
|
ClearJobError( pIniPort->pIniJob );
|
|
|
|
RestartJob( pIniPort->pIniJob );
|
|
}
|
|
|
|
SetEvent( pIniPort->hErrorEvent );
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Cleanup:
|
|
if(pIniSpooler)
|
|
{
|
|
FindSpoolerByNameDecRef( pIniSpooler );
|
|
}
|
|
|
|
if(SemEntered)
|
|
{
|
|
LeaveSplSem();
|
|
SplOutSem();
|
|
}
|
|
|
|
if(dwLastError != ERROR_SUCCESS)
|
|
{
|
|
SetLastError(dwLastError);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
VOID
|
|
SetPortErrorEvent(
|
|
PINIPORT pIniPort
|
|
)
|
|
{
|
|
SplInSem();
|
|
if(pIniPort && pIniPort->hErrorEvent) {
|
|
SetEvent(pIniPort->hErrorEvent);
|
|
}
|
|
}
|
|
|
|
|
|
/*++
|
|
|
|
Routine Name:
|
|
|
|
DiscardJob
|
|
|
|
Description:
|
|
|
|
This routine discards the given job when an irrecoverable error occurs
|
|
in the port thread.
|
|
|
|
Arguments:
|
|
|
|
pIniJob - The job we are wanting to discard.
|
|
bStartDocComplete - If TRUE, we should set the StartDocComplete event.
|
|
|
|
Returns:
|
|
|
|
Nothing.
|
|
|
|
--*/
|
|
VOID
|
|
DiscardJobFromPortThread(
|
|
IN PINIJOB pIniJob,
|
|
IN BOOL bStartDocComplete
|
|
)
|
|
{
|
|
EnterSplSem();
|
|
|
|
//
|
|
// App might be waiting for the StartDoc to Complete
|
|
//
|
|
if (pIniJob->StartDocComplete && bStartDocComplete) {
|
|
|
|
SetEvent(pIniJob->StartDocComplete);
|
|
}
|
|
|
|
//
|
|
// If the job had an error, mark it pending deletion. The port monitor
|
|
// may not do this if EndDocPort was never called.
|
|
//
|
|
if (!(pIniJob->Status & JOB_RESTART)){
|
|
|
|
InterlockedOr((LONG*)&(pIniJob->Status), JOB_PENDING_DELETION);
|
|
|
|
//
|
|
// Release any thread waiting on LocalSetPort
|
|
//
|
|
SetPortErrorEvent(pIniJob->pIniPort);
|
|
|
|
//
|
|
// Release any thread waiting on SeekPrinter
|
|
//
|
|
SeekPrinterSetEvent(pIniJob, NULL, TRUE);
|
|
}
|
|
|
|
LeaveSplSem();
|
|
}
|
|
|