reactos/modules/rostests/apitests/localspl/dll/main.c

209 lines
5.6 KiB
C
Raw Normal View History

/*
* PROJECT: ReactOS Local Spooler API Tests Injected DLL
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
* PURPOSE: Main functions
* COPYRIGHT: Copyright 2015-2017 Colin Finck (colin@reactos.org)
*/
#define __ROS_LONG64__
#define STANDALONE
#include <apitest.h>
#define WIN32_NO_STATUS
#include <io.h>
#include <windef.h>
#include <winbase.h>
#include <wingdi.h>
#include <winreg.h>
#include <winspool.h>
#include <winsplp.h>
#include "../localspl_apitest.h"
//#define NDEBUG
#include <debug.h>
// Test list
extern void func_fpEnumPrinters(void);
extern void func_fpGetPrintProcessorDirectory(void);
extern void func_fpSetJob(void);
const struct test winetest_testlist[] =
{
{ "fpEnumPrinters", func_fpEnumPrinters },
{ "fpGetPrintProcessorDirectory", func_fpGetPrintProcessorDirectory },
{ "fpSetJob", func_fpSetJob },
{ 0, 0 }
};
/**
* We don't link against winspool, so we don't have GetDefaultPrinterW.
* We can easily implement a similar function ourselves though.
*/
PWSTR
GetDefaultPrinterFromRegistry(VOID)
{
static const WCHAR wszWindowsKey[] = L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Windows";
static const WCHAR wszDeviceValue[] = L"Device";
DWORD cbNeeded;
DWORD dwErrorCode;
HKEY hWindowsKey = NULL;
PWSTR pwszDevice;
PWSTR pwszComma;
PWSTR pwszReturnValue = NULL;
// Open the registry key where the default printer for the current user is stored.
dwErrorCode = (DWORD)RegOpenKeyExW(HKEY_CURRENT_USER, wszWindowsKey, 0, KEY_READ, &hWindowsKey);
if (dwErrorCode != ERROR_SUCCESS)
{
skip("RegOpenKeyExW failed with status %u!\n", dwErrorCode);
goto Cleanup;
}
// Determine the size of the required buffer.
dwErrorCode = (DWORD)RegQueryValueExW(hWindowsKey, wszDeviceValue, NULL, NULL, NULL, &cbNeeded);
if (dwErrorCode != ERROR_SUCCESS)
{
skip("RegQueryValueExW failed with status %u!\n", dwErrorCode);
goto Cleanup;
}
// Allocate it.
pwszDevice = HeapAlloc(GetProcessHeap(), 0, cbNeeded);
if (!pwszDevice)
{
skip("HeapAlloc failed!\n");
goto Cleanup;
}
// Now get the actual value.
dwErrorCode = RegQueryValueExW(hWindowsKey, wszDeviceValue, NULL, NULL, (PBYTE)pwszDevice, &cbNeeded);
if (dwErrorCode != ERROR_SUCCESS)
{
skip("RegQueryValueExW failed with status %u!\n", dwErrorCode);
goto Cleanup;
}
// We get a string "<Printer Name>,winspool,<Port>:".
// Extract the printer name from it.
pwszComma = wcschr(pwszDevice, L',');
if (!pwszComma)
{
skip("Found no or invalid default printer: %S!\n", pwszDevice);
goto Cleanup;
}
// Return the default printer.
*pwszComma = 0;
pwszReturnValue = pwszDevice;
Cleanup:
if (hWindowsKey)
RegCloseKey(hWindowsKey);
return pwszReturnValue;
}
BOOL
GetLocalsplFuncs(LPPRINTPROVIDOR pp)
{
HMODULE hLocalspl;
PInitializePrintProvidor pfnInitializePrintProvidor;
// Get us a handle to the loaded localspl.dll.
hLocalspl = GetModuleHandleW(L"localspl");
if (!hLocalspl)
{
skip("GetModuleHandleW failed with error %u!\n", GetLastError());
return FALSE;
}
// Get a pointer to its InitializePrintProvidor function.
pfnInitializePrintProvidor = (PInitializePrintProvidor)GetProcAddress(hLocalspl, "InitializePrintProvidor");
if (!pfnInitializePrintProvidor)
{
skip("GetProcAddress failed with error %u!\n", GetLastError());
return FALSE;
}
// Get localspl's function pointers.
if (!pfnInitializePrintProvidor(pp, sizeof(PRINTPROVIDOR), NULL))
{
skip("pfnInitializePrintProvidor failed with error %u!\n", GetLastError());
return FALSE;
}
return TRUE;
}
PVOID
GetSpoolssFunc(const char* FunctionName)
{
HMODULE hSpoolss;
// Get us a handle to the loaded spoolss.dll.
hSpoolss = GetModuleHandleW(L"spoolss");
if (!hSpoolss)
{
skip("GetModuleHandleW failed with error %u!\n", GetLastError());
return FALSE;
}
return GetProcAddress(hSpoolss, FunctionName);
}
// Running the tests from the injected DLL and redirecting their output to the pipe.
BOOL WINAPI
DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
{
char szTestName[150];
DWORD cbRead;
FILE* fpStdout;
HANDLE hCommandPipe;
int iOldStdout;
// We only want to run our test once when the DLL is injected to the process.
if (fdwReason != DLL_PROCESS_ATTACH)
return TRUE;
// Read the test to run from the command pipe.
hCommandPipe = CreateFileW(COMMAND_PIPE_NAME, GENERIC_READ, 0, NULL, OPEN_EXISTING, 0, NULL);
if (hCommandPipe == INVALID_HANDLE_VALUE)
{
DPRINT("DLL: CreateFileW failed for the command pipe with error %lu!\n", GetLastError());
return FALSE;
}
if (!ReadFile(hCommandPipe, szTestName, sizeof(szTestName), &cbRead, NULL))
{
DPRINT("DLL: ReadFile failed for the command pipe with error %lu!\n", GetLastError());
return FALSE;
}
CloseHandle(hCommandPipe);
// Check if the test name is valid.
if (!find_test(szTestName))
{
DPRINT("DLL: Got invalid test name \"%s\"!\n", szTestName);
return FALSE;
}
// Backup our current stdout and set it to the output pipe.
iOldStdout = _dup(_fileno(stdout));
fpStdout = _wfreopen(OUTPUT_PIPE_NAME, L"w", stdout);
setbuf(stdout, NULL);
// Run the test.
run_test(szTestName);
// Restore stdout to the previous value.
fclose(fpStdout);
_dup2(iOldStdout, _fileno(stdout));
// Return FALSE so that our DLL is immediately unloaded.
return FALSE;
}