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.
971 lines
26 KiB
971 lines
26 KiB
/*++
|
|
|
|
Copyright (c) 1990 - 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
net.c
|
|
|
|
Abstract:
|
|
|
|
This module provides all the network stuuf for localspl
|
|
|
|
Author:
|
|
|
|
Dave Snipp (DaveSn) 15-Mar-1991
|
|
|
|
Notes:
|
|
|
|
We just need to get the winspool printer name associated with a given
|
|
queue name. The SHARE_INFO_2 structure has a shi2_path field that would
|
|
be nice to use, but NetShareGetInfo level 2 is privileged. So, by
|
|
DaveSn's arm twisting and agreement with windows/spooler/localspl/net.c,
|
|
we're going to use the shi1_remark field for this. This allows us to
|
|
do NetShareGetInfo level 1, which is not privileged.
|
|
|
|
This has been fixed by allowing OpenPrinter to succeed on share names.
|
|
If there is no comment, we put the printer name in as the remark
|
|
(for graceful upgrades from pre-PPC).
|
|
|
|
Revision History:
|
|
|
|
Jun 93 mattfe pIniSpooler
|
|
|
|
--*/
|
|
|
|
#define UNICODE 1
|
|
|
|
#define NOMINMAX
|
|
|
|
#include <precomp.h>
|
|
#pragma hdrstop
|
|
|
|
#include "clusspl.h"
|
|
|
|
#ifdef DBG_SHARE
|
|
#include <messages.h>
|
|
#endif
|
|
|
|
|
|
|
|
NET_API_STATUS (*pfnNetShareAdd)();
|
|
NET_API_STATUS (*pfnNetShareSetInfo)();
|
|
NET_API_STATUS (*pfnNetShareDel)();
|
|
NET_API_STATUS (*pfnNetServerEnum)();
|
|
NET_API_STATUS (*pfnNetWkstaUserGetInfo)();
|
|
NET_API_STATUS (*pfnNetApiBufferFree)();
|
|
NET_API_STATUS (*pfnNetAlertRaiseEx)();
|
|
NET_API_STATUS (*pfnNetShareGetInfo)(LPWSTR, LPWSTR, DWORD, LPBYTE *);
|
|
|
|
extern SHARE_INFO_2 PrintShareInfo;
|
|
extern SHARE_INFO_2 PrtProcsShareInfo;
|
|
|
|
BOOL
|
|
InitializeNet(
|
|
VOID
|
|
)
|
|
{
|
|
HANDLE hNetApi;
|
|
UINT uOldErrorMode = SetErrorMode(SEM_FAILCRITICALERRORS);
|
|
|
|
hNetApi = LoadLibrary(TEXT("netapi32.dll"));
|
|
|
|
SetErrorMode(uOldErrorMode);
|
|
if ( !hNetApi )
|
|
return FALSE;
|
|
|
|
pfnNetShareAdd = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetShareAdd");
|
|
pfnNetShareSetInfo = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetShareSetInfo");
|
|
pfnNetShareDel = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetShareDel");
|
|
pfnNetServerEnum = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetServerEnum");
|
|
pfnNetWkstaUserGetInfo = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetWkstaUserGetInfo");
|
|
pfnNetApiBufferFree = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetApiBufferFree");
|
|
pfnNetAlertRaiseEx = (NET_API_STATUS (*)()) GetProcAddress(hNetApi,"NetAlertRaiseEx");
|
|
pfnNetShareGetInfo = (NET_API_STATUS (*)(LPWSTR, LPWSTR, DWORD, LPBYTE *))GetProcAddress(hNetApi, "NetShareGetInfo");
|
|
|
|
if ( pfnNetShareAdd == NULL ||
|
|
pfnNetShareSetInfo == NULL ||
|
|
pfnNetShareDel == NULL ||
|
|
pfnNetServerEnum == NULL ||
|
|
pfnNetWkstaUserGetInfo == NULL ||
|
|
pfnNetApiBufferFree == NULL ||
|
|
pfnNetAlertRaiseEx == NULL ||
|
|
pfnNetShareGetInfo == NULL) {
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
BOOL
|
|
SetPrinterShareInfo(
|
|
PINIPRINTER pIniPrinter
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Sets the share information about a printer.
|
|
|
|
Note: This does not update the share path. We need to
|
|
delete and re-create the share in order to change the path.
|
|
|
|
Arguments:
|
|
|
|
pIniPrinter - Printer that needs to be updated.
|
|
|
|
Return Value:
|
|
|
|
TRUE - Success
|
|
FALSE - Failed
|
|
|
|
--*/
|
|
|
|
{
|
|
SHARE_INFO_502 ShareInfo;
|
|
HANDLE hToken;
|
|
PSECURITY_DESCRIPTOR pShareSecurityDescriptor = NULL;
|
|
DWORD ParmError, rc;
|
|
|
|
SplInSem();
|
|
|
|
pShareSecurityDescriptor = MapPrinterSDToShareSD(pIniPrinter->pSecurityDescriptor);
|
|
|
|
if ( !pShareSecurityDescriptor ) {
|
|
|
|
rc = ERROR_INVALID_SECURITY_DESCR;
|
|
goto Cleanup;
|
|
}
|
|
|
|
ZeroMemory((LPVOID)&ShareInfo, sizeof(ShareInfo));
|
|
|
|
ShareInfo.shi502_netname = pIniPrinter->pShareName;
|
|
ShareInfo.shi502_type = STYPE_PRINTQ;
|
|
ShareInfo.shi502_permissions = 0;
|
|
ShareInfo.shi502_max_uses = SHI_USES_UNLIMITED;
|
|
ShareInfo.shi502_current_uses = SHI_USES_UNLIMITED;
|
|
ShareInfo.shi502_passwd = NULL;
|
|
ShareInfo.shi502_security_descriptor = pShareSecurityDescriptor;
|
|
|
|
if ( pIniPrinter->pComment && pIniPrinter->pComment[0] ) {
|
|
|
|
ShareInfo.shi502_remark = pIniPrinter->pComment;
|
|
|
|
} else {
|
|
|
|
ShareInfo.shi502_remark = pIniPrinter->pName;
|
|
}
|
|
|
|
|
|
INCPRINTERREF(pIniPrinter);
|
|
LeaveSplSem();
|
|
|
|
SplOutSem(); // We *MUST* be out of our semaphore as the NetShareSet may
|
|
// come back and call spooler
|
|
|
|
hToken = RevertToPrinterSelf();
|
|
|
|
rc = (*pfnNetShareSetInfo)(NULL,
|
|
ShareInfo.shi502_netname,
|
|
502,
|
|
&ShareInfo,
|
|
&ParmError);
|
|
|
|
if ( rc ) {
|
|
|
|
if (rc == NERR_NetNameNotFound)
|
|
{
|
|
//
|
|
// This can happen deny all access to a shared printer, then
|
|
// restart the computer. The server service tries to validate
|
|
// it's share on startup, but since it has no access, it fails
|
|
// and deletes it (it also wants a handle to the printer). When
|
|
// you grant everyone access, we try to change the ACL on the
|
|
// SMB share, but since it was deleted, we fail. Recreate
|
|
// the share here.
|
|
//
|
|
EnterSplSem();
|
|
|
|
if (!ShareThisPrinter(pIniPrinter,
|
|
pIniPrinter->pShareName,
|
|
TRUE)) {
|
|
rc = GetLastError();
|
|
} else {
|
|
|
|
rc = ERROR_SUCCESS;
|
|
}
|
|
|
|
LeaveSplSem();
|
|
}
|
|
|
|
if (rc) {
|
|
|
|
DBGMSG(DBG_WARNING,
|
|
("NetShareSetInfo/ShareThisPrinter failed: Error %d, Parm %d\n",
|
|
rc, ParmError));
|
|
|
|
SetLastError(rc);
|
|
}
|
|
}
|
|
|
|
ImpersonatePrinterClient(hToken);
|
|
|
|
EnterSplSem();
|
|
DECPRINTERREF(pIniPrinter);
|
|
|
|
Cleanup:
|
|
SplInSem();
|
|
|
|
LocalFree(pShareSecurityDescriptor);
|
|
|
|
return rc == ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
BOOL
|
|
ShareThisPrinter(
|
|
PINIPRINTER pIniPrinter,
|
|
LPWSTR pShareName,
|
|
BOOL bShare
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Shares or UnShares a Printer.
|
|
|
|
Note: this really should be two functions, and the return value
|
|
is very confusing.
|
|
|
|
Note: no validation of sharename is done. This must be done by
|
|
callee, usually in ValidatePrinterInfo.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
Returns whether the printer is shared after this call.
|
|
|
|
TRUE - Shared
|
|
FALSE - Not Shared
|
|
|
|
--*/
|
|
{
|
|
SHARE_INFO_502 ShareInfo = {0};
|
|
DWORD rc;
|
|
DWORD ParmError;
|
|
PSECURITY_DESCRIPTOR pShareSecurityDescriptor = NULL;
|
|
PINISPOOLER pIniSpooler = pIniPrinter->pIniSpooler;
|
|
PSHARE_INFO_2 pShareInfo = (PSHARE_INFO_2)pIniSpooler->pDriversShareInfo;
|
|
LPTSTR pszPrinterNameWithToken = NULL;
|
|
HRESULT hr;
|
|
|
|
HANDLE hToken;
|
|
BOOL bReturn = FALSE;
|
|
BOOL bSame = FALSE;
|
|
LPWSTR pShareNameCopy = NULL;
|
|
|
|
SPLASSERT( pIniPrinter->pName );
|
|
SplInSem();
|
|
|
|
pShareNameCopy = AllocSplStr(pShareName);
|
|
|
|
if (!pShareNameCopy) {
|
|
goto Done;
|
|
}
|
|
|
|
if ( bShare ) {
|
|
|
|
if (!pfnNetShareAdd) {
|
|
SetLastError( ERROR_PROC_NOT_FOUND );
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// Share name validation has been moved into ValidatePrinterInfo.
|
|
//
|
|
|
|
if ((pShareSecurityDescriptor =
|
|
MapPrinterSDToShareSD(pIniPrinter->pSecurityDescriptor)) == NULL) {
|
|
SetLastError(ERROR_INVALID_SECURITY_DESCR);
|
|
goto Done;
|
|
}
|
|
|
|
ShareInfo.shi502_netname = pShareNameCopy; // hplaser
|
|
|
|
//
|
|
// If there is a comment, use it; otherwise set the remark
|
|
// to be the printer name.
|
|
//
|
|
// Note: if the printer name changes and we don't have a comment,
|
|
// we will reshare the printer to update the remark.
|
|
//
|
|
if( pIniPrinter->pComment && pIniPrinter->pComment[0] ){
|
|
|
|
ShareInfo.shi502_remark = pIniPrinter->pComment;
|
|
|
|
} else {
|
|
|
|
ShareInfo.shi502_remark = pIniPrinter->pName;
|
|
}
|
|
|
|
//
|
|
// Use the fully qualifed name, and make sure it exists in
|
|
// localspl by using LocalsplOnlyToken.
|
|
//
|
|
pszPrinterNameWithToken = pszGetPrinterName(
|
|
pIniPrinter,
|
|
pIniSpooler != pLocalIniSpooler,
|
|
pszLocalsplOnlyToken );
|
|
|
|
if( !pszPrinterNameWithToken ){
|
|
goto Done;
|
|
}
|
|
|
|
ShareInfo.shi502_path = pszPrinterNameWithToken;
|
|
ShareInfo.shi502_type = STYPE_PRINTQ;
|
|
ShareInfo.shi502_permissions = 0;
|
|
ShareInfo.shi502_max_uses = SHI_USES_UNLIMITED;
|
|
ShareInfo.shi502_current_uses = SHI_USES_UNLIMITED;
|
|
ShareInfo.shi502_passwd = NULL;
|
|
ShareInfo.shi502_security_descriptor = pShareSecurityDescriptor;
|
|
|
|
INCPRINTERREF(pIniPrinter);
|
|
LeaveSplSem();
|
|
|
|
//
|
|
// We *MUST* be out of our semaphore as the NetShareAdd is
|
|
// going to come round and call OpenPrinter.
|
|
//
|
|
SplOutSem();
|
|
|
|
// Go add the Print Share
|
|
|
|
hToken = RevertToPrinterSelf();
|
|
|
|
//
|
|
// Add a share for the spool\drivers directory:
|
|
//
|
|
if (rc = AddPrintShare(pIniSpooler))
|
|
{
|
|
if (rc != NERR_ServerNotStarted)
|
|
{
|
|
EnterSplSem();
|
|
DECPRINTERREF(pIniPrinter);
|
|
ImpersonatePrinterClient(hToken);
|
|
|
|
SetLastError(rc);
|
|
goto Done;
|
|
}
|
|
}
|
|
|
|
#if DBG
|
|
{
|
|
WCHAR UserName[256];
|
|
DWORD cbUserName=256;
|
|
|
|
if (MODULE_DEBUG & DBG_SECURITY)
|
|
GetUserName(UserName, &cbUserName);
|
|
|
|
DBGMSG( DBG_SECURITY, ( "Calling NetShareAdd in context %ws\n", UserName ) );
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// Add the printer share. Even if we failed above because Server Service is not started,
|
|
// we still want to try. At boot time, it's worthwhile, since the Server service might
|
|
// start accepting calls if it's done initializing.
|
|
//
|
|
rc = (*pfnNetShareAdd)(NULL, 502, (LPBYTE)&ShareInfo, &ParmError);
|
|
|
|
//
|
|
// Now take care of Web sharing. i.e. make sure wwwroot\sharename is created if the
|
|
// printer is either local or masqurading. We cannot allow sharing of RPC connections.
|
|
//
|
|
if (pIniSpooler->SpoolerFlags & SPL_TYPE_LOCAL)
|
|
{
|
|
WebShare( pShareNameCopy );
|
|
}
|
|
|
|
//
|
|
// If the return code is that the share already exists, then check to
|
|
// see whether this share is to the same device, if it is we just
|
|
// update the info on the share and return success.
|
|
//
|
|
if (rc == NERR_DuplicateShare) {
|
|
|
|
if (ERROR_SUCCESS == CheckShareSame(pIniPrinter, &ShareInfo, &bSame) && bSame) {
|
|
|
|
rc = ERROR_SUCCESS;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Allow remote connections by calling into spoolsv and register the named pipe protocol.
|
|
// The server keeps track of these calls, so doing it for each share is fine.
|
|
// This is policy driven. The policy value is read in spoolsv.
|
|
//
|
|
if ((rc == ERROR_SUCCESS || rc == NERR_ServerNotStarted) &&
|
|
FAILED(hr = AllowRemoteCalls()))
|
|
{
|
|
//
|
|
// If we fail enabling the RPC named pipe, then just unshare the printer
|
|
// and fail the call.
|
|
// A futher attempt to share the printer will try again to enable the RPC pipe.
|
|
//
|
|
if (rc == ERROR_SUCCESS)
|
|
{
|
|
(*pfnNetShareDel)(NULL, ShareInfo.shi502_netname, 0);
|
|
}
|
|
|
|
rc = HRESULT_CODE(hr);
|
|
}
|
|
|
|
ImpersonatePrinterClient(hToken);
|
|
|
|
EnterSplSem();
|
|
DECPRINTERREF(pIniPrinter);
|
|
|
|
if (ERROR_SUCCESS != rc) {
|
|
|
|
DBGMSG( DBG_WARNING,
|
|
( "NetShareAdd failed %lx, Parameter %d\n",
|
|
rc, ParmError ));
|
|
|
|
if ((rc == ERROR_INVALID_PARAMETER) &&
|
|
(ParmError == SHARE_NETNAME_PARMNUM)) {
|
|
|
|
rc = ERROR_INVALID_SHARENAME;
|
|
}
|
|
|
|
SetLastError(rc);
|
|
goto Done;
|
|
|
|
}
|
|
|
|
SPLASSERT( pIniPrinter != NULL);
|
|
SPLASSERT( pIniPrinter->signature == IP_SIGNATURE);
|
|
SPLASSERT( pIniPrinter->pIniSpooler != NULL);
|
|
SPLASSERT( pIniPrinter->pIniSpooler->signature == ISP_SIGNATURE );
|
|
|
|
CreateServerThread();
|
|
|
|
//
|
|
// The Printer is shared.
|
|
//
|
|
bReturn = TRUE;
|
|
|
|
} else {
|
|
|
|
if ( !pfnNetShareDel ) {
|
|
bReturn = TRUE;
|
|
goto Done;
|
|
|
|
}
|
|
|
|
INCPRINTERREF( pIniPrinter );
|
|
LeaveSplSem();
|
|
|
|
SplOutSem();
|
|
|
|
hToken = RevertToPrinterSelf();
|
|
|
|
rc = hToken ? ERROR_SUCCESS : GetLastError();
|
|
|
|
if ( rc == ERROR_SUCCESS ) {
|
|
|
|
rc = (*pfnNetShareDel)(NULL, pShareName, 0);
|
|
|
|
// Now take care of Web unsharing. i.e. make sure wwwroot\sharename is deleted.
|
|
if((pIniSpooler->SpoolerFlags & SPL_TYPE_LOCAL) && !(pIniSpooler->SpoolerFlags & SPL_TYPE_CLUSTER))
|
|
WebUnShare( pShareName );
|
|
}
|
|
|
|
if ( hToken ) {
|
|
|
|
if ( !ImpersonatePrinterClient(hToken) && rc == ERROR_SUCCESS ) {
|
|
rc = GetLastError();
|
|
}
|
|
}
|
|
|
|
|
|
EnterSplSem();
|
|
DECPRINTERREF(pIniPrinter);
|
|
|
|
// The share may have been deleted manually, so don't worry
|
|
// if we get NERR_NetNameNotFound:
|
|
|
|
if ( rc && ( rc != NERR_NetNameNotFound )){
|
|
|
|
DBGMSG(DBG_WARNING, ("NetShareDel failed %lx\n", rc));
|
|
SetLastError( rc );
|
|
bReturn = TRUE;
|
|
}
|
|
}
|
|
|
|
Done:
|
|
|
|
if( pShareSecurityDescriptor ){
|
|
LocalFree(pShareSecurityDescriptor);
|
|
}
|
|
|
|
FreeSplStr(pszPrinterNameWithToken);
|
|
FreeSplStr(pShareNameCopy);
|
|
|
|
return bReturn;
|
|
}
|
|
|
|
|
|
/*--
|
|
FillAlertStructWithJobStrings
|
|
|
|
Allocates memory which has to be freed by caller.
|
|
|
|
|
|
|
|
--*/
|
|
|
|
HRESULT
|
|
FillAlertStructWithJobStrings(
|
|
PINIJOB pIniJob,
|
|
PRINT_OTHER_INFO **pAlertInfo,
|
|
PDWORD size
|
|
)
|
|
{
|
|
HRESULT RetVal = E_FAIL;
|
|
DWORD cchSizeToAlloc = 0;
|
|
PBYTE pBuffer = NULL;
|
|
LPWSTR psz = NULL;
|
|
|
|
|
|
if (pIniJob && pAlertInfo && !(*pAlertInfo) && size)
|
|
{
|
|
//
|
|
// Do this in the splSem so that no one can jo a SetJob While we're not looking
|
|
//
|
|
EnterSplSem();
|
|
|
|
//
|
|
// We don't know how big these strings are going to be in future, so size
|
|
// them dynamically and alloc the structure to always be big enough.
|
|
//
|
|
cchSizeToAlloc = sizeof(PRINT_OTHER_INFO);
|
|
|
|
//
|
|
// We don't have to check some of these for existence, we know they exist.
|
|
//
|
|
cchSizeToAlloc += wcslen(pIniJob->pNotify) + 1 +
|
|
wcslen(pIniJob->pIniPrinter->pName) + 1 +
|
|
wcslen(pIniJob->pIniPrinter->pIniSpooler->pMachineName) + 1;
|
|
|
|
if ( pIniJob->pDocument )
|
|
{
|
|
cchSizeToAlloc += wcslen(pIniJob->pDocument) + 1;
|
|
}
|
|
else
|
|
{
|
|
cchSizeToAlloc += 2;
|
|
}
|
|
|
|
if (pIniJob->pIniPrinter->pIniSpooler->bEnableNetPopupToComputer &&
|
|
pIniJob->pMachineName)
|
|
{
|
|
cchSizeToAlloc += wcslen(pIniJob->pMachineName) + 1;
|
|
}
|
|
else
|
|
{
|
|
cchSizeToAlloc += wcslen(pIniJob->pNotify) + 1;
|
|
}
|
|
|
|
if ( pIniJob->pStatus )
|
|
{
|
|
cchSizeToAlloc += wcslen(pIniJob->pStatus) + 1;
|
|
}
|
|
else
|
|
{
|
|
cchSizeToAlloc += 2;
|
|
}
|
|
|
|
//
|
|
// Alloc the memory
|
|
//
|
|
pBuffer = AllocSplMem(cchSizeToAlloc * sizeof(WCHAR));
|
|
|
|
if ( pBuffer )
|
|
{
|
|
SIZE_T cchRemaining = cchSizeToAlloc;
|
|
|
|
psz = (LPWSTR)ALERT_VAR_DATA((PRINT_OTHER_INFO *)pBuffer);
|
|
|
|
//
|
|
// We know that pBuffer will at a minimum be as big as a PRINT_OTHER_INFO.
|
|
//
|
|
cchRemaining -= sizeof(PRINT_OTHER_INFO);
|
|
|
|
//
|
|
// Do the copying
|
|
//
|
|
|
|
//
|
|
// Computer Name
|
|
//
|
|
if(pIniJob->pIniPrinter->pIniSpooler->bEnableNetPopupToComputer &&
|
|
pIniJob->pMachineName ){
|
|
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pMachineName, &psz, &cchRemaining);
|
|
|
|
} else {
|
|
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pNotify, &psz, &cchRemaining);
|
|
}
|
|
|
|
|
|
//
|
|
// UserName
|
|
//
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pNotify, &psz, &cchRemaining);
|
|
|
|
//
|
|
// Document Name
|
|
//
|
|
if (pIniJob->pDocument)
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pDocument, &psz, &cchRemaining);
|
|
else
|
|
StrCchCopyMultipleStr(psz, cchRemaining, L"", &psz, &cchRemaining);
|
|
|
|
//
|
|
// Printer Name
|
|
//
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pIniPrinter->pName, &psz, &cchRemaining);
|
|
|
|
//
|
|
// Server Name
|
|
//
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pIniPrinter->pIniSpooler->pMachineName, &psz, &cchRemaining);
|
|
|
|
//
|
|
// Status_string
|
|
// We should pass in other status strings for the other status errors, too.
|
|
//
|
|
if (pIniJob->pStatus && (pIniJob->Status & (JOB_ERROR | JOB_OFFLINE | JOB_PAPEROUT)))
|
|
StrCchCopyMultipleStr(psz, cchRemaining, pIniJob->pStatus, &psz, &cchRemaining);
|
|
else
|
|
StrCchCopyMultipleStr(psz, cchRemaining, L"", &psz, &cchRemaining);
|
|
|
|
//
|
|
// Pass back the size and struct
|
|
//
|
|
*size = (DWORD)((PBYTE)psz - pBuffer);
|
|
*pAlertInfo = (PRINT_OTHER_INFO *)pBuffer;
|
|
RetVal = NOERROR;
|
|
|
|
}
|
|
else
|
|
{
|
|
RetVal = E_OUTOFMEMORY;
|
|
}
|
|
//
|
|
// Leave the Spooler Semaphore
|
|
//
|
|
LeaveSplSem();
|
|
|
|
}
|
|
else
|
|
{
|
|
RetVal = E_INVALIDARG;
|
|
}
|
|
|
|
return RetVal;
|
|
|
|
}
|
|
|
|
VOID
|
|
SendJobAlert(
|
|
PINIJOB pIniJob
|
|
)
|
|
{
|
|
PRINT_OTHER_INFO *pinfo = NULL;
|
|
DWORD RetVal = ERROR_SUCCESS;
|
|
DWORD Status;
|
|
FILETIME FileTime;
|
|
DWORD AlertSize = 0;
|
|
|
|
//
|
|
// Allow Hydra Sessions to be notified since they are remote
|
|
//
|
|
if( (USER_SHARED_DATA->SuiteMask & (1 << TerminalServer)) ) {
|
|
if ( !pIniJob->pNotify ||
|
|
!pIniJob->pNotify[0] ||
|
|
!pIniJob->pIniPrinter->pIniSpooler->bEnableNetPopups ) {
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
if ( !pIniJob->pNotify ||
|
|
!pIniJob->pNotify[0] ||
|
|
!(pIniJob->Status & JOB_REMOTE) ||
|
|
!pIniJob->pIniPrinter->pIniSpooler->bEnableNetPopups ) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if ( FAILED(RetVal = FillAlertStructWithJobStrings(pIniJob, &pinfo, &AlertSize)))
|
|
{
|
|
if ( pinfo )
|
|
FreeSplMem(pinfo);
|
|
|
|
return;
|
|
}
|
|
|
|
pinfo->alrtpr_jobid = pIniJob->JobId;
|
|
|
|
if (pIniJob->Status & (JOB_PRINTING | JOB_DESPOOLING | JOB_PRINTED | JOB_COMPLETE))
|
|
Status = PRJOB_QS_PRINTING;
|
|
else if (pIniJob->Status & JOB_PAUSED)
|
|
Status = PRJOB_QS_PAUSED;
|
|
else if (pIniJob->Status & JOB_SPOOLING)
|
|
Status = PRJOB_QS_SPOOLING;
|
|
else
|
|
Status = PRJOB_QS_QUEUED;
|
|
|
|
if (pIniJob->Status & (JOB_ERROR | JOB_OFFLINE | JOB_PAPEROUT)) {
|
|
|
|
Status |= PRJOB_ERROR;
|
|
|
|
if (pIniJob->Status & JOB_OFFLINE)
|
|
Status |= PRJOB_DESTOFFLINE;
|
|
|
|
if (pIniJob->Status & JOB_PAPEROUT)
|
|
Status |= PRJOB_DESTNOPAPER;
|
|
}
|
|
|
|
if (pIniJob->Status & JOB_PRINTED)
|
|
Status |= PRJOB_COMPLETE;
|
|
|
|
else if (pIniJob->Status & JOB_PENDING_DELETION)
|
|
Status |= PRJOB_DELETED;
|
|
|
|
pinfo->alrtpr_status = Status;
|
|
|
|
SystemTimeToFileTime( &pIniJob->Submitted, &FileTime );
|
|
|
|
// FileTimeToDosDateTime(&FileTime, &DosDate, &DosTime);
|
|
// pinfo->alrtpr_submitted = DosDate << 16 | DosTime;
|
|
|
|
RtlTimeToSecondsSince1970((PLARGE_INTEGER) &FileTime,
|
|
&pinfo->alrtpr_submitted);
|
|
|
|
pinfo->alrtpr_size = pIniJob->Size;
|
|
|
|
|
|
(*pfnNetAlertRaiseEx)(ALERT_PRINT_EVENT,
|
|
(PBYTE) pinfo,
|
|
AlertSize,
|
|
L"SPOOLER");
|
|
|
|
if ( pinfo )
|
|
FreeSplMem(pinfo);
|
|
}
|
|
|
|
DWORD
|
|
AddPrintShare(
|
|
PINISPOOLER pIniSpooler
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Adds the print$ share based on pIniSpooler.
|
|
|
|
Arguments:
|
|
|
|
pIniSpooler - Share path is based on this information. pDriversShareInfo
|
|
must be initialized before calling this routine.
|
|
|
|
Return Value:
|
|
|
|
TRUE - Success.
|
|
|
|
FALSE - Failed. LastError set.
|
|
|
|
--*/
|
|
{
|
|
DWORD rc;
|
|
DWORD ParmError;
|
|
SHARE_INFO_1501 ShareInfo1501 = {0};
|
|
PSHARE_INFO_2 pShareInfo = (PSHARE_INFO_2)pIniSpooler->pDriversShareInfo;
|
|
PSECURITY_DESCRIPTOR pSecurityDescriptor = NULL;
|
|
|
|
//
|
|
// Assert that the path is identical to the local one since there's
|
|
// only one print$ share. It should always be.
|
|
//
|
|
SPLASSERT( !lstrcmpi( pShareInfo->shi2_path,
|
|
((PSHARE_INFO_2)pLocalIniSpooler->pDriversShareInfo)->shi2_path ));
|
|
|
|
rc = (*pfnNetShareAdd)( NULL,
|
|
2,
|
|
(LPBYTE)pShareInfo,
|
|
&ParmError );
|
|
|
|
//
|
|
// If it already exists, assume it is set up correctly.
|
|
//
|
|
if( rc == NERR_DuplicateShare ){
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// If we didn't create the share, fail.
|
|
//
|
|
if( rc != ERROR_SUCCESS ){
|
|
|
|
DBGMSG( DBG_WARN,
|
|
( "AddPrintShare: Error %d, Parm %d\n", rc, ParmError));
|
|
|
|
return rc;
|
|
}
|
|
|
|
//
|
|
// Set security on the newly created share.
|
|
//
|
|
// Bug 54844
|
|
// If this fails, we've created the share but haven't put security
|
|
// on it. Then since it exists, we'll never try and set it again.
|
|
//
|
|
|
|
pSecurityDescriptor = CreateDriversShareSecurityDescriptor();
|
|
|
|
if( !pSecurityDescriptor ){
|
|
return GetLastError();
|
|
}
|
|
|
|
ShareInfo1501.shi1501_security_descriptor = pSecurityDescriptor;
|
|
|
|
rc = (*pfnNetShareSetInfo)( NULL,
|
|
pShareInfo->shi2_netname,
|
|
1501,
|
|
&ShareInfo1501,
|
|
&ParmError );
|
|
|
|
if( rc != ERROR_SUCCESS ){
|
|
|
|
DBGMSG( DBG_WARN,
|
|
( "NetShareSetInfo failed: Error %d, Parm %d\n",
|
|
rc, ParmError));
|
|
}
|
|
|
|
LocalFree(pSecurityDescriptor);
|
|
|
|
return rc;
|
|
}
|
|
|
|
/*++
|
|
|
|
Routine Name:
|
|
|
|
CheckShareSame
|
|
|
|
Routine Description:
|
|
|
|
This checks to see whether the given share name is the same on both the
|
|
local and remote machines.
|
|
|
|
Arguments:
|
|
|
|
pIniPrinter - The iniprinter for which we are adding the share.
|
|
pShareInfo502 - The share info that we are attempting to add the share
|
|
with.
|
|
pbSame - The return parameter is TRUE if the shares were the same
|
|
If the rc is not ERROR_SUCCESS, then the info could not
|
|
be set.
|
|
|
|
Return Value:
|
|
|
|
An error code.
|
|
|
|
--*/
|
|
DWORD
|
|
CheckShareSame(
|
|
IN PINIPRINTER pIniPrinter,
|
|
IN SHARE_INFO_502 *pShareInfo502,
|
|
OUT BOOL *pbSame
|
|
)
|
|
{
|
|
DWORD rc = ERROR_SUCCESS;
|
|
SHARE_INFO_2 *pShareInfoCompare = NULL;
|
|
BOOL bPathEquivalent = FALSE;
|
|
BOOL bSame = FALSE;
|
|
DWORD ParmError;
|
|
|
|
SplOutSem();
|
|
|
|
//
|
|
// Get the share info for the share, we should already have determined
|
|
// that this share exists.
|
|
//
|
|
rc = pfnNetShareGetInfo(NULL, pShareInfo502->shi502_netname, 2, (LPBYTE *)&pShareInfoCompare);
|
|
|
|
if (ERROR_SUCCESS == rc)
|
|
{
|
|
|
|
if (STYPE_PRINTQ == pShareInfoCompare->shi2_type)
|
|
{
|
|
//
|
|
// Check to see whether the paths are the same, in the upgrade case, the
|
|
// LocalSplOnly will be taken off, so, compare this too.
|
|
//
|
|
bSame = !_wcsicmp(pShareInfoCompare->shi2_path, pShareInfo502->shi502_path);
|
|
|
|
//
|
|
// If they are not the same, compare it to the name of the printer.
|
|
//
|
|
if (!bSame)
|
|
{
|
|
EnterSplSem();
|
|
|
|
bSame = bPathEquivalent = !_wcsicmp(pIniPrinter->pName, pShareInfoCompare->shi2_path);
|
|
|
|
LeaveSplSem();
|
|
}
|
|
}
|
|
}
|
|
|
|
*pbSame = bSame;
|
|
|
|
if (ERROR_SUCCESS == rc && bSame)
|
|
{
|
|
//
|
|
// If the paths are identical, we can just set the share info, otherwise
|
|
// we have to delete and recreate the share.
|
|
//
|
|
if (!bPathEquivalent)
|
|
{
|
|
//
|
|
// OK, they are the same, set the share info instead.
|
|
//
|
|
rc = (*pfnNetShareSetInfo)(NULL, pShareInfo502->shi502_netname, 502, pShareInfo502, &ParmError);
|
|
}
|
|
else
|
|
{
|
|
rc = (*pfnNetShareDel)(NULL, pShareInfo502->shi502_netname, 0);
|
|
|
|
if (ERROR_SUCCESS == rc)
|
|
{
|
|
rc = (*pfnNetShareAdd)(NULL, 502, (LPBYTE)pShareInfo502, &ParmError);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pShareInfoCompare)
|
|
{
|
|
pfnNetApiBufferFree(pShareInfoCompare);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|