mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
255 lines
8.1 KiB
C
255 lines
8.1 KiB
C
|
/*
|
||
|
* PROJECT: ReactOS Spooler Router
|
||
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
||
|
* PURPOSE: Functions related to Printer Configuration Data
|
||
|
* COPYRIGHT: Copyright 2020 ReactOS
|
||
|
*/
|
||
|
|
||
|
#include "precomp.h"
|
||
|
|
||
|
BOOL WINAPI
|
||
|
AddPrinterDriverExW(PWSTR pName, DWORD Level, PBYTE pDriverInfo, DWORD dwFileCopyFlags)
|
||
|
{
|
||
|
BOOL bReturnValue;
|
||
|
DWORD dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
PLIST_ENTRY pEntry;
|
||
|
PSPOOLSS_PRINT_PROVIDER pPrintProvider;
|
||
|
|
||
|
// Loop through all Print Providers.
|
||
|
for (pEntry = PrintProviderList.Flink; pEntry != &PrintProviderList; pEntry = pEntry->Flink)
|
||
|
{
|
||
|
pPrintProvider = CONTAINING_RECORD(pEntry, SPOOLSS_PRINT_PROVIDER, Entry);
|
||
|
|
||
|
bReturnValue = pPrintProvider->PrintProvider.fpAddPrinterDriverEx(pName, Level, pDriverInfo, dwFileCopyFlags);
|
||
|
|
||
|
if (bReturnValue == ROUTER_SUCCESS)
|
||
|
{
|
||
|
dwErrorCode = ERROR_SUCCESS;
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
else if (bReturnValue == ROUTER_STOP_ROUTING)
|
||
|
{
|
||
|
ERR("A Print Provider returned ROUTER_STOP_ROUTING for Printer \"%S\"!\n", pName);
|
||
|
dwErrorCode = GetLastError();
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Cleanup:
|
||
|
// ERROR_INVALID_NAME by the Print Provider is translated to ERROR_INVALID_PRINTER_NAME here, but not in other APIs as far as I know.
|
||
|
if (dwErrorCode == ERROR_INVALID_NAME)
|
||
|
dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
|
||
|
SetLastError(dwErrorCode);
|
||
|
return (dwErrorCode == ERROR_SUCCESS);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
AddPrinterDriverW(PWSTR pName, DWORD Level, PBYTE pDriverInfo)
|
||
|
{
|
||
|
TRACE("AddPrinterDriverW(%S, %lu, %p)\n", pName, Level, pDriverInfo);
|
||
|
return AddPrinterDriverExW(pName, Level, pDriverInfo, APD_COPY_NEW_FILES);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
DeletePrinterDriverExW(PWSTR pName, PWSTR pEnvironment, PWSTR pDriverName, DWORD dwDeleteFlag, DWORD dwVersionFlag)
|
||
|
{
|
||
|
BOOL bReturnValue;
|
||
|
DWORD dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
PLIST_ENTRY pEntry;
|
||
|
PSPOOLSS_PRINT_PROVIDER pPrintProvider;
|
||
|
|
||
|
// Loop through all Print Providers.
|
||
|
for (pEntry = PrintProviderList.Flink; pEntry != &PrintProviderList; pEntry = pEntry->Flink)
|
||
|
{
|
||
|
pPrintProvider = CONTAINING_RECORD(pEntry, SPOOLSS_PRINT_PROVIDER, Entry);
|
||
|
|
||
|
bReturnValue = pPrintProvider->PrintProvider.fpDeletePrinterDriverEx(pName, pEnvironment, pDriverName, dwDeleteFlag, dwVersionFlag);
|
||
|
|
||
|
if (bReturnValue == ROUTER_SUCCESS)
|
||
|
{
|
||
|
dwErrorCode = ERROR_SUCCESS;
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
else if (bReturnValue == ROUTER_STOP_ROUTING)
|
||
|
{
|
||
|
ERR("A Print Provider returned ROUTER_STOP_ROUTING for Printer \"%S\"!\n", pName);
|
||
|
dwErrorCode = GetLastError();
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Cleanup:
|
||
|
// ERROR_INVALID_NAME by the Print Provider is translated to ERROR_INVALID_PRINTER_NAME here, but not in other APIs as far as I know.
|
||
|
if (dwErrorCode == ERROR_INVALID_NAME)
|
||
|
dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
|
||
|
SetLastError(dwErrorCode);
|
||
|
return (dwErrorCode == ERROR_SUCCESS);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
DeletePrinterDriverW(PWSTR pName, PWSTR pEnvironment, PWSTR pDriverName)
|
||
|
{
|
||
|
TRACE("DeletePrinterDriverW(%S, %S, %S)\n", pName, pEnvironment, pDriverName);
|
||
|
return DeletePrinterDriverExW(pName, pEnvironment, pDriverName, 0, 0);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
EnumPrinterDriversW(PWSTR pName, PWSTR pEnvironment, DWORD Level, PBYTE pDriverInfo, DWORD cbBuf, PDWORD pcbNeeded, PDWORD pcReturned)
|
||
|
{
|
||
|
DWORD cbCallBuffer;
|
||
|
DWORD cbNeeded;
|
||
|
DWORD dwErrorCode = MAXDWORD;
|
||
|
DWORD dwReturned;
|
||
|
PBYTE pCallBuffer;
|
||
|
BOOL Ret = FALSE;
|
||
|
PSPOOLSS_PRINT_PROVIDER pPrintProvider;
|
||
|
PLIST_ENTRY pEntry;
|
||
|
|
||
|
// Begin counting.
|
||
|
*pcbNeeded = 0;
|
||
|
*pcReturned = 0;
|
||
|
|
||
|
if ( cbBuf && !pDriverInfo )
|
||
|
{
|
||
|
dwErrorCode = ERROR_INVALID_USER_BUFFER;
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
|
||
|
// At the beginning, we have the full buffer available.
|
||
|
cbCallBuffer = cbBuf;
|
||
|
pCallBuffer = pDriverInfo;
|
||
|
|
||
|
// Loop through all Print Providers.
|
||
|
for (pEntry = PrintProviderList.Flink; pEntry != &PrintProviderList; pEntry = pEntry->Flink)
|
||
|
{
|
||
|
pPrintProvider = CONTAINING_RECORD(pEntry, SPOOLSS_PRINT_PROVIDER, Entry);
|
||
|
|
||
|
// Call the EnumPrinters function of this Print Provider.
|
||
|
cbNeeded = 0;
|
||
|
dwReturned = 0;
|
||
|
|
||
|
Ret = pPrintProvider->PrintProvider.fpEnumPrinterDrivers( pName, pEnvironment, Level, pCallBuffer, cbCallBuffer, &cbNeeded, &dwReturned);
|
||
|
|
||
|
if ( !Ret )
|
||
|
{
|
||
|
dwErrorCode = GetLastError();
|
||
|
}
|
||
|
|
||
|
// Add the returned counts to the total values.
|
||
|
*pcbNeeded += cbNeeded;
|
||
|
*pcReturned += dwReturned;
|
||
|
|
||
|
// Reduce the available buffer size for the next call without risking an underflow.
|
||
|
if (cbNeeded < cbCallBuffer)
|
||
|
cbCallBuffer -= cbNeeded;
|
||
|
else
|
||
|
cbCallBuffer = 0;
|
||
|
|
||
|
// Advance the buffer if the caller provided it.
|
||
|
if (pCallBuffer)
|
||
|
pCallBuffer += cbNeeded;
|
||
|
|
||
|
// dwErrorCode shall not be overwritten if a previous EnumPrinters call already succeeded.
|
||
|
if (dwErrorCode != ERROR_SUCCESS)
|
||
|
dwErrorCode = GetLastError();
|
||
|
}
|
||
|
|
||
|
Cleanup:
|
||
|
SetLastError(dwErrorCode);
|
||
|
return (dwErrorCode == ERROR_SUCCESS);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
GetPrinterDriverW(HANDLE hPrinter, PWSTR pEnvironment, DWORD Level, PBYTE pDriverInfo, DWORD cbBuf, PDWORD pcbNeeded)
|
||
|
{
|
||
|
PSPOOLSS_PRINTER_HANDLE pHandle = (PSPOOLSS_PRINTER_HANDLE)hPrinter;
|
||
|
|
||
|
// Sanity checks.
|
||
|
if (!pHandle)
|
||
|
{
|
||
|
SetLastError(ERROR_INVALID_PARAMETER);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return pHandle->pPrintProvider->PrintProvider.fpGetPrinterDriver(pHandle->hPrinter, pEnvironment, Level, pDriverInfo, cbBuf, pcbNeeded);
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL WINAPI
|
||
|
GetPrinterDriverExW(
|
||
|
HANDLE hPrinter,
|
||
|
LPWSTR pEnvironment,
|
||
|
DWORD Level,
|
||
|
LPBYTE pDriverInfo,
|
||
|
DWORD cbBuf,
|
||
|
LPDWORD pcbNeeded,
|
||
|
DWORD dwClientMajorVersion,
|
||
|
DWORD dwClientMinorVersion,
|
||
|
PDWORD pdwServerMajorVersion,
|
||
|
PDWORD pdwServerMinorVersion )
|
||
|
{
|
||
|
PSPOOLSS_PRINTER_HANDLE pHandle = (PSPOOLSS_PRINTER_HANDLE)hPrinter;
|
||
|
|
||
|
FIXME("GetPrinterDriverExW(%p, %lu, %lu, %p, %lu, %p, %lu, %lu, %p, %p)\n", hPrinter, pEnvironment, Level, pDriverInfo, cbBuf, pcbNeeded, dwClientMajorVersion, dwClientMinorVersion, pdwServerMajorVersion, pdwServerMinorVersion);
|
||
|
|
||
|
// Sanity checks.
|
||
|
if (!pHandle)
|
||
|
{
|
||
|
SetLastError(ERROR_INVALID_PARAMETER);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
if ( cbBuf && !pDriverInfo )
|
||
|
{
|
||
|
SetLastError(ERROR_INVALID_USER_BUFFER);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return pHandle->pPrintProvider->PrintProvider.fpGetPrinterDriverEx(pHandle->hPrinter, pEnvironment, Level, pDriverInfo, cbBuf, pcbNeeded, dwClientMajorVersion, dwClientMinorVersion, pdwServerMajorVersion, pdwServerMinorVersion);
|
||
|
}
|
||
|
|
||
|
BOOL WINAPI
|
||
|
GetPrinterDriverDirectoryW(PWSTR pName, PWSTR pEnvironment, DWORD Level, PBYTE pDriverDirectory, DWORD cbBuf, PDWORD pcbNeeded)
|
||
|
{
|
||
|
BOOL bReturnValue;
|
||
|
DWORD dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
PLIST_ENTRY pEntry;
|
||
|
PSPOOLSS_PRINT_PROVIDER pPrintProvider;
|
||
|
|
||
|
if ( cbBuf && !pDriverDirectory )
|
||
|
{
|
||
|
SetLastError(ERROR_INVALID_USER_BUFFER);
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
// Loop through all Print Providers.
|
||
|
for (pEntry = PrintProviderList.Flink; pEntry != &PrintProviderList; pEntry = pEntry->Flink)
|
||
|
{
|
||
|
pPrintProvider = CONTAINING_RECORD(pEntry, SPOOLSS_PRINT_PROVIDER, Entry);
|
||
|
|
||
|
bReturnValue = pPrintProvider->PrintProvider.fpGetPrinterDriverDirectory(pName, pEnvironment, Level, pDriverDirectory, cbBuf, pcbNeeded);
|
||
|
|
||
|
if (bReturnValue == ROUTER_SUCCESS)
|
||
|
{
|
||
|
dwErrorCode = ERROR_SUCCESS;
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
else if (bReturnValue == ROUTER_STOP_ROUTING)
|
||
|
{
|
||
|
ERR("A Print Provider returned ROUTER_STOP_ROUTING for Printer \"%S\"!\n", pName);
|
||
|
dwErrorCode = GetLastError();
|
||
|
goto Cleanup;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Cleanup:
|
||
|
// ERROR_INVALID_NAME by the Print Provider is translated to ERROR_INVALID_PRINTER_NAME here, but not in other APIs as far as I know.
|
||
|
if (dwErrorCode == ERROR_INVALID_NAME)
|
||
|
dwErrorCode = ERROR_INVALID_PRINTER_NAME;
|
||
|
|
||
|
SetLastError(dwErrorCode);
|
||
|
return (dwErrorCode == ERROR_SUCCESS);
|
||
|
}
|