mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 06:45:24 +00:00
7bffb70353
WinSpool : Implement missing API. Sync/Port from wine. Setting it to fast track for needed testing of the later GDI code. Leaving Fix-me debug prints on. Local tree has WinSpool_winetest turned on. So no debug noise during normal ReactOS operation. SpoolSS : Reordered header types. Have more Spl function to be added while SpoolSV is being coded to forward to LocalSpl. Based on wine and old research from the GDI code.
366 lines
15 KiB
C
366 lines
15 KiB
C
/*
|
|
* PROJECT: ReactOS Spooler API
|
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
|
* PURPOSE: Functions giving information about DEVMODE structures
|
|
* COPYRIGHT: Copyright 2016-2017 Colin Finck (colin@reactos.org)
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
|
|
typedef struct _MINIMUM_SIZE_TABLE
|
|
{
|
|
DWORD dwField;
|
|
WORD wSize;
|
|
}
|
|
MINIMUM_SIZE_TABLE, *PMINIMUM_SIZE_TABLE;
|
|
|
|
/**
|
|
* Minimum required DEVMODEA structure size based on the used fields. Must be in descending order!
|
|
*/
|
|
static MINIMUM_SIZE_TABLE MinimumSizeA[] = {
|
|
{ DM_PANNINGHEIGHT, FIELD_OFFSET(DEVMODEA, dmPanningHeight) + RTL_FIELD_SIZE(DEVMODEA, dmPanningHeight) },
|
|
{ DM_PANNINGWIDTH, FIELD_OFFSET(DEVMODEA, dmPanningWidth) + RTL_FIELD_SIZE(DEVMODEA, dmPanningWidth) },
|
|
{ DM_DITHERTYPE, FIELD_OFFSET(DEVMODEA, dmDitherType) + RTL_FIELD_SIZE(DEVMODEA, dmDitherType) },
|
|
{ DM_MEDIATYPE, FIELD_OFFSET(DEVMODEA, dmMediaType) + RTL_FIELD_SIZE(DEVMODEA, dmMediaType) },
|
|
{ DM_ICMINTENT, FIELD_OFFSET(DEVMODEA, dmICMIntent) + RTL_FIELD_SIZE(DEVMODEA, dmICMIntent) },
|
|
{ DM_ICMMETHOD, FIELD_OFFSET(DEVMODEA, dmICMMethod) + RTL_FIELD_SIZE(DEVMODEA, dmICMMethod) },
|
|
{ DM_DISPLAYFREQUENCY, FIELD_OFFSET(DEVMODEA, dmDisplayFrequency) + RTL_FIELD_SIZE(DEVMODEA, dmDisplayFrequency) },
|
|
{ DM_NUP, FIELD_OFFSET(DEVMODEA, dmNup) + RTL_FIELD_SIZE(DEVMODEA, dmNup) },
|
|
{ DM_DISPLAYFLAGS, FIELD_OFFSET(DEVMODEA, dmDisplayFlags) + RTL_FIELD_SIZE(DEVMODEA, dmDisplayFlags) },
|
|
{ DM_PELSHEIGHT, FIELD_OFFSET(DEVMODEA, dmPelsHeight) + RTL_FIELD_SIZE(DEVMODEA, dmPelsHeight) },
|
|
{ DM_PELSWIDTH, FIELD_OFFSET(DEVMODEA, dmPelsWidth) + RTL_FIELD_SIZE(DEVMODEA, dmPelsWidth) },
|
|
{ DM_BITSPERPEL, FIELD_OFFSET(DEVMODEA, dmBitsPerPel) + RTL_FIELD_SIZE(DEVMODEA, dmBitsPerPel) },
|
|
{ DM_LOGPIXELS, FIELD_OFFSET(DEVMODEA, dmLogPixels) + RTL_FIELD_SIZE(DEVMODEA, dmLogPixels) },
|
|
{ DM_FORMNAME, FIELD_OFFSET(DEVMODEA, dmFormName) + RTL_FIELD_SIZE(DEVMODEA, dmFormName) },
|
|
{ DM_COLLATE, FIELD_OFFSET(DEVMODEA, dmCollate) + RTL_FIELD_SIZE(DEVMODEA, dmCollate) },
|
|
{ DM_TTOPTION, FIELD_OFFSET(DEVMODEA, dmTTOption) + RTL_FIELD_SIZE(DEVMODEA, dmTTOption) },
|
|
{ DM_YRESOLUTION, FIELD_OFFSET(DEVMODEA, dmYResolution) + RTL_FIELD_SIZE(DEVMODEA, dmYResolution) },
|
|
{ DM_DUPLEX, FIELD_OFFSET(DEVMODEA, dmDuplex) + RTL_FIELD_SIZE(DEVMODEA, dmDuplex) },
|
|
{ DM_COLOR, FIELD_OFFSET(DEVMODEA, dmColor) + RTL_FIELD_SIZE(DEVMODEA, dmColor) },
|
|
{ DM_DISPLAYFIXEDOUTPUT, FIELD_OFFSET(DEVMODEA, dmDisplayFixedOutput) + RTL_FIELD_SIZE(DEVMODEA, dmDisplayFixedOutput) },
|
|
{ DM_DISPLAYORIENTATION, FIELD_OFFSET(DEVMODEA, dmDisplayOrientation) + RTL_FIELD_SIZE(DEVMODEA, dmDisplayOrientation) },
|
|
{ DM_POSITION, FIELD_OFFSET(DEVMODEA, dmPosition) + RTL_FIELD_SIZE(DEVMODEA, dmPosition) },
|
|
{ DM_PRINTQUALITY, FIELD_OFFSET(DEVMODEA, dmPrintQuality) + RTL_FIELD_SIZE(DEVMODEA, dmPrintQuality) },
|
|
{ DM_DEFAULTSOURCE, FIELD_OFFSET(DEVMODEA, dmDefaultSource) + RTL_FIELD_SIZE(DEVMODEA, dmDefaultSource) },
|
|
{ DM_COPIES, FIELD_OFFSET(DEVMODEA, dmCopies) + RTL_FIELD_SIZE(DEVMODEA, dmCopies) },
|
|
{ DM_SCALE, FIELD_OFFSET(DEVMODEA, dmScale) + RTL_FIELD_SIZE(DEVMODEA, dmScale) },
|
|
{ DM_PAPERWIDTH, FIELD_OFFSET(DEVMODEA, dmPaperWidth) + RTL_FIELD_SIZE(DEVMODEA, dmPaperWidth) },
|
|
{ DM_PAPERLENGTH, FIELD_OFFSET(DEVMODEA, dmPaperLength) + RTL_FIELD_SIZE(DEVMODEA, dmPaperLength) },
|
|
{ DM_PAPERSIZE, FIELD_OFFSET(DEVMODEA, dmPaperSize) + RTL_FIELD_SIZE(DEVMODEA, dmPaperSize) },
|
|
{ DM_ORIENTATION, FIELD_OFFSET(DEVMODEA, dmOrientation) + RTL_FIELD_SIZE(DEVMODEA, dmOrientation) },
|
|
{ 0, 0 }
|
|
};
|
|
|
|
/**
|
|
* Minimum required DEVMODEW structure size based on the used fields. Must be in descending order!
|
|
*/
|
|
static MINIMUM_SIZE_TABLE MinimumSizeW[] = {
|
|
{ DM_PANNINGHEIGHT, FIELD_OFFSET(DEVMODEW, dmPanningHeight) + RTL_FIELD_SIZE(DEVMODEW, dmPanningHeight) },
|
|
{ DM_PANNINGWIDTH, FIELD_OFFSET(DEVMODEW, dmPanningWidth) + RTL_FIELD_SIZE(DEVMODEW, dmPanningWidth) },
|
|
{ DM_DITHERTYPE, FIELD_OFFSET(DEVMODEW, dmDitherType) + RTL_FIELD_SIZE(DEVMODEW, dmDitherType) },
|
|
{ DM_MEDIATYPE, FIELD_OFFSET(DEVMODEW, dmMediaType) + RTL_FIELD_SIZE(DEVMODEW, dmMediaType) },
|
|
{ DM_ICMINTENT, FIELD_OFFSET(DEVMODEW, dmICMIntent) + RTL_FIELD_SIZE(DEVMODEW, dmICMIntent) },
|
|
{ DM_ICMMETHOD, FIELD_OFFSET(DEVMODEW, dmICMMethod) + RTL_FIELD_SIZE(DEVMODEW, dmICMMethod) },
|
|
{ DM_DISPLAYFREQUENCY, FIELD_OFFSET(DEVMODEW, dmDisplayFrequency) + RTL_FIELD_SIZE(DEVMODEW, dmDisplayFrequency) },
|
|
{ DM_NUP, FIELD_OFFSET(DEVMODEW, dmNup) + RTL_FIELD_SIZE(DEVMODEW, dmNup) },
|
|
{ DM_DISPLAYFLAGS, FIELD_OFFSET(DEVMODEW, dmDisplayFlags) + RTL_FIELD_SIZE(DEVMODEW, dmDisplayFlags) },
|
|
{ DM_PELSHEIGHT, FIELD_OFFSET(DEVMODEW, dmPelsHeight) + RTL_FIELD_SIZE(DEVMODEW, dmPelsHeight) },
|
|
{ DM_PELSWIDTH, FIELD_OFFSET(DEVMODEW, dmPelsWidth) + RTL_FIELD_SIZE(DEVMODEW, dmPelsWidth) },
|
|
{ DM_BITSPERPEL, FIELD_OFFSET(DEVMODEW, dmBitsPerPel) + RTL_FIELD_SIZE(DEVMODEW, dmBitsPerPel) },
|
|
{ DM_LOGPIXELS, FIELD_OFFSET(DEVMODEW, dmLogPixels) + RTL_FIELD_SIZE(DEVMODEW, dmLogPixels) },
|
|
{ DM_FORMNAME, FIELD_OFFSET(DEVMODEW, dmFormName) + RTL_FIELD_SIZE(DEVMODEW, dmFormName) },
|
|
{ DM_COLLATE, FIELD_OFFSET(DEVMODEW, dmCollate) + RTL_FIELD_SIZE(DEVMODEW, dmCollate) },
|
|
{ DM_TTOPTION, FIELD_OFFSET(DEVMODEW, dmTTOption) + RTL_FIELD_SIZE(DEVMODEW, dmTTOption) },
|
|
{ DM_YRESOLUTION, FIELD_OFFSET(DEVMODEW, dmYResolution) + RTL_FIELD_SIZE(DEVMODEW, dmYResolution) },
|
|
{ DM_DUPLEX, FIELD_OFFSET(DEVMODEW, dmDuplex) + RTL_FIELD_SIZE(DEVMODEW, dmDuplex) },
|
|
{ DM_COLOR, FIELD_OFFSET(DEVMODEW, dmColor) + RTL_FIELD_SIZE(DEVMODEW, dmColor) },
|
|
{ DM_DISPLAYFIXEDOUTPUT, FIELD_OFFSET(DEVMODEW, dmDisplayFixedOutput) + RTL_FIELD_SIZE(DEVMODEW, dmDisplayFixedOutput) },
|
|
{ DM_DISPLAYORIENTATION, FIELD_OFFSET(DEVMODEW, dmDisplayOrientation) + RTL_FIELD_SIZE(DEVMODEW, dmDisplayOrientation) },
|
|
{ DM_POSITION, FIELD_OFFSET(DEVMODEW, dmPosition) + RTL_FIELD_SIZE(DEVMODEW, dmPosition) },
|
|
{ DM_PRINTQUALITY, FIELD_OFFSET(DEVMODEW, dmPrintQuality) + RTL_FIELD_SIZE(DEVMODEW, dmPrintQuality) },
|
|
{ DM_DEFAULTSOURCE, FIELD_OFFSET(DEVMODEW, dmDefaultSource) + RTL_FIELD_SIZE(DEVMODEW, dmDefaultSource) },
|
|
{ DM_COPIES, FIELD_OFFSET(DEVMODEW, dmCopies) + RTL_FIELD_SIZE(DEVMODEW, dmCopies) },
|
|
{ DM_SCALE, FIELD_OFFSET(DEVMODEW, dmScale) + RTL_FIELD_SIZE(DEVMODEW, dmScale) },
|
|
{ DM_PAPERWIDTH, FIELD_OFFSET(DEVMODEW, dmPaperWidth) + RTL_FIELD_SIZE(DEVMODEW, dmPaperWidth) },
|
|
{ DM_PAPERLENGTH, FIELD_OFFSET(DEVMODEW, dmPaperLength) + RTL_FIELD_SIZE(DEVMODEW, dmPaperLength) },
|
|
{ DM_PAPERSIZE, FIELD_OFFSET(DEVMODEW, dmPaperSize) + RTL_FIELD_SIZE(DEVMODEW, dmPaperSize) },
|
|
{ DM_ORIENTATION, FIELD_OFFSET(DEVMODEW, dmOrientation) + RTL_FIELD_SIZE(DEVMODEW, dmOrientation) },
|
|
{ 0, 0 }
|
|
};
|
|
|
|
/**
|
|
* Replace the last character by a null terminator if the given ANSI string is not null-terminated.
|
|
*/
|
|
static __inline void
|
|
_FixStringA(PBYTE String, DWORD cbString)
|
|
{
|
|
const PBYTE pLastCharacter = &String[cbString / sizeof(BYTE) - 1];
|
|
PBYTE p = String;
|
|
|
|
while (*p)
|
|
{
|
|
if (p == pLastCharacter)
|
|
{
|
|
*p = 0;
|
|
break;
|
|
}
|
|
|
|
p++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Replace the last character by a null terminator if the given Unicode string is not null-terminated.
|
|
*/
|
|
static __inline void
|
|
_FixStringW(PWSTR String, DWORD cbString)
|
|
{
|
|
const PWSTR pLastCharacter = &String[cbString / sizeof(WCHAR) - 1];
|
|
PWSTR p = String;
|
|
|
|
while (*p)
|
|
{
|
|
if (p == pLastCharacter)
|
|
{
|
|
*p = 0;
|
|
break;
|
|
}
|
|
|
|
p++;
|
|
}
|
|
}
|
|
|
|
BOOL WINAPI
|
|
IsValidDevmodeA(PDEVMODEA pDevmode, size_t DevmodeSize)
|
|
{
|
|
PMINIMUM_SIZE_TABLE pTable = MinimumSizeA;
|
|
WORD wRequiredSize;
|
|
|
|
TRACE("IsValidDevmodeA(%p, %lu)\n", pDevmode, DevmodeSize);
|
|
|
|
// Check if a Devmode was given at all.
|
|
if (!pDevmode)
|
|
goto Failure;
|
|
|
|
// Verify that DevmodeSize is large enough to hold the public and private members of the structure.
|
|
if (DevmodeSize < pDevmode->dmSize + pDevmode->dmDriverExtra)
|
|
goto Failure;
|
|
|
|
// If the structure has private members, the public structure must be 32-bit packed.
|
|
if (pDevmode->dmDriverExtra && pDevmode->dmSize % 4)
|
|
goto Failure;
|
|
|
|
// Now determine the minimum possible dmSize based on the given fields in dmFields.
|
|
wRequiredSize = FIELD_OFFSET(DEVMODEA, dmFields) + RTL_FIELD_SIZE(DEVMODEA, dmFields);
|
|
|
|
while (pTable->dwField)
|
|
{
|
|
if (pDevmode->dmFields & pTable->dwField)
|
|
{
|
|
wRequiredSize = pTable->wSize;
|
|
break;
|
|
}
|
|
|
|
pTable++;
|
|
}
|
|
|
|
// Verify that the value in dmSize is big enough for the used fields.
|
|
if (pDevmode->dmSize < wRequiredSize)
|
|
goto Failure;
|
|
|
|
// Check if dmDeviceName and (if used) dmFormName are null-terminated.
|
|
// Fix this if they aren't.
|
|
_FixStringA(pDevmode->dmDeviceName, sizeof(pDevmode->dmDeviceName));
|
|
if (pDevmode->dmFields & DM_FORMNAME)
|
|
_FixStringA(pDevmode->dmFormName, sizeof(pDevmode->dmFormName));
|
|
|
|
// Return success without setting the error code.
|
|
return TRUE;
|
|
|
|
Failure:
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WINAPI
|
|
IsValidDevmodeW(PDEVMODEW pDevmode, size_t DevmodeSize)
|
|
{
|
|
PMINIMUM_SIZE_TABLE pTable = MinimumSizeW;
|
|
WORD wRequiredSize;
|
|
|
|
TRACE("IsValidDevmodeW(%p, %lu)\n", pDevmode, DevmodeSize);
|
|
|
|
// Check if a Devmode was given at all.
|
|
if (!pDevmode)
|
|
goto Failure;
|
|
|
|
// Verify that DevmodeSize is large enough to hold the public and private members of the structure.
|
|
if (DevmodeSize < pDevmode->dmSize + pDevmode->dmDriverExtra)
|
|
goto Failure;
|
|
|
|
// If the structure has private members, the public structure must be 32-bit packed.
|
|
if (pDevmode->dmDriverExtra && pDevmode->dmSize % 4)
|
|
goto Failure;
|
|
|
|
// Now determine the minimum possible dmSize based on the given fields in dmFields.
|
|
wRequiredSize = FIELD_OFFSET(DEVMODEW, dmFields) + RTL_FIELD_SIZE(DEVMODEW, dmFields);
|
|
|
|
while (pTable->dwField)
|
|
{
|
|
if (pDevmode->dmFields & pTable->dwField)
|
|
{
|
|
wRequiredSize = pTable->wSize;
|
|
break;
|
|
}
|
|
|
|
pTable++;
|
|
}
|
|
|
|
// Verify that the value in dmSize is big enough for the used fields.
|
|
if (pDevmode->dmSize < wRequiredSize)
|
|
goto Failure;
|
|
|
|
// Check if dmDeviceName and (if used) dmFormName are null-terminated.
|
|
// Fix this if they aren't.
|
|
_FixStringW(pDevmode->dmDeviceName, sizeof(pDevmode->dmDeviceName));
|
|
if (pDevmode->dmFields & DM_FORMNAME)
|
|
_FixStringW(pDevmode->dmFormName, sizeof(pDevmode->dmFormName));
|
|
|
|
// Return success without setting the error code.
|
|
return TRUE;
|
|
|
|
Failure:
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL WINAPI
|
|
IsValidDevmodeNoSizeW(PDEVMODEW pDevmode)
|
|
{
|
|
PMINIMUM_SIZE_TABLE pTable = MinimumSizeW;
|
|
WORD wRequiredSize;
|
|
|
|
TRACE("IsValidDevmodeNoSizeW(%p)\n", pDevmode);
|
|
|
|
// Check if a Devmode was given at all.
|
|
if (!pDevmode)
|
|
goto Failure;
|
|
|
|
// If the structure has private members, the public structure must be 32-bit packed.
|
|
if (pDevmode->dmDriverExtra && pDevmode->dmSize % 4)
|
|
goto Failure;
|
|
|
|
// Now determine the minimum possible dmSize based on the given fields in dmFields.
|
|
wRequiredSize = FIELD_OFFSET(DEVMODEW, dmFields) + RTL_FIELD_SIZE(DEVMODEW, dmFields);
|
|
|
|
while (pTable->dwField)
|
|
{
|
|
if (pDevmode->dmFields & pTable->dwField)
|
|
{
|
|
wRequiredSize = pTable->wSize;
|
|
break;
|
|
}
|
|
|
|
pTable++;
|
|
}
|
|
|
|
// Verify that the value in dmSize is big enough for the used fields.
|
|
if (pDevmode->dmSize < wRequiredSize)
|
|
goto Failure;
|
|
|
|
// Check if dmDeviceName and (if used) dmFormName are null-terminated.
|
|
// Fix this if they aren't.
|
|
_FixStringW(pDevmode->dmDeviceName, sizeof(pDevmode->dmDeviceName));
|
|
if (pDevmode->dmFields & DM_FORMNAME)
|
|
_FixStringW(pDevmode->dmFormName, sizeof(pDevmode->dmFormName));
|
|
|
|
// Return success without setting the error code.
|
|
return TRUE;
|
|
|
|
Failure:
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
return FALSE;
|
|
}
|
|
|
|
void RosConvertAnsiDevModeToUnicodeDevmode(PDEVMODEA pDevModeInput, PDEVMODEW *pDevModeOutput)
|
|
{
|
|
// FIXME: This function should become ConvertAnsiDevModeToUnicodeDevmode when its parameters are known!
|
|
|
|
// Check if a pDevModeInput and pDevModeOutput are both not NULL.
|
|
if (!pDevModeInput || !pDevModeOutput)
|
|
return;
|
|
|
|
*pDevModeOutput = GdiConvertToDevmodeW(pDevModeInput);
|
|
}
|
|
|
|
// Internal counterpart to GdiConvertToDevmodeW from gdi32
|
|
static __inline DEVMODEA*
|
|
_ConvertToDevmodeA(const DEVMODEW *dmW)
|
|
{
|
|
DEVMODEA *dmA;
|
|
WORD dmA_size, dmW_size;
|
|
size_t BytesToCopy;
|
|
|
|
dmW_size = dmW->dmSize;
|
|
|
|
/* this is the minimal dmSize that XP accepts */
|
|
if (dmW_size < FIELD_OFFSET(DEVMODEW, dmFields))
|
|
return NULL;
|
|
|
|
// Guard against callers that set dmSize incorrectly.
|
|
if (dmW_size > sizeof(DEVMODEW))
|
|
dmW_size = sizeof(DEVMODEW);
|
|
|
|
// dmA_size must become dmW_size without the additional 1 byte per character for each Unicode string (dmDeviceName and dmFormName).
|
|
dmA_size = dmW_size - CCHDEVICENAME;
|
|
if (dmW_size >= FIELD_OFFSET(DEVMODEW, dmFormName) + CCHFORMNAME * sizeof(WCHAR))
|
|
dmA_size -= CCHFORMNAME;
|
|
|
|
// Allocate the required bytes, that is dmSize for the ANSI DEVMODEA structure plus any extra bytes requested through dmDriverExtra.
|
|
dmA = HeapAlloc(GetProcessHeap(), 0, dmA_size + dmW->dmDriverExtra);
|
|
if (!dmA) return NULL;
|
|
|
|
// Every valid DEVMODEW has a dmDeviceName, which we convert to ANSI here.
|
|
WideCharToMultiByte(CP_ACP, 0, dmW->dmDeviceName, -1, (LPSTR)dmA->dmDeviceName, CCHDEVICENAME, NULL, NULL);
|
|
|
|
// Copy everything up to dmFormName or the remaining dmW_size, whatever is smaller.
|
|
BytesToCopy = min(FIELD_OFFSET(DEVMODEW, dmFormName) - FIELD_OFFSET(DEVMODEW, dmSpecVersion), dmW_size - CCHDEVICENAME * sizeof(WCHAR));
|
|
memcpy(&dmA->dmSpecVersion, &dmW->dmSpecVersion, BytesToCopy);
|
|
|
|
// Handle dmFormName if the input DEVMODEW is large enough to contain one.
|
|
if (dmW_size >= FIELD_OFFSET(DEVMODEW, dmFormName) + CCHFORMNAME * sizeof(WCHAR))
|
|
{
|
|
if (dmW->dmFields & DM_FORMNAME)
|
|
WideCharToMultiByte(CP_ACP, 0, dmW->dmFormName, -1, (LPSTR)dmA->dmFormName, CCHFORMNAME, NULL, NULL);
|
|
else
|
|
dmA->dmFormName[0] = 0;
|
|
|
|
// Copy the remaining fields.
|
|
if (dmW_size > FIELD_OFFSET(DEVMODEW, dmLogPixels))
|
|
memcpy(&dmA->dmLogPixels, &dmW->dmLogPixels, dmW_size - FIELD_OFFSET(DEVMODEW, dmLogPixels));
|
|
}
|
|
|
|
// Append dmDriverExtra if required.
|
|
if (dmW->dmDriverExtra)
|
|
memcpy((char *)dmA + dmA_size, (const char *)dmW + dmW_size, dmW->dmDriverExtra);
|
|
|
|
// Set the corrected dmSize and we are done.
|
|
dmA->dmSize = dmA_size;
|
|
|
|
return dmA;
|
|
}
|
|
|
|
void RosConvertUnicodeDevModeToAnsiDevmode(PDEVMODEW pDevModeInput, PDEVMODEA pDevModeOutput)
|
|
{
|
|
PDEVMODEA pTmp;
|
|
|
|
// FIXME: This function should become ConvertUnicodeDevModeToAnsiDevmode when its parameters are known!
|
|
|
|
// Check if a pDevModeInput and pDevModeOutput are both not NULL.
|
|
if (!pDevModeInput || !pDevModeOutput)
|
|
return;
|
|
|
|
pTmp = _ConvertToDevmodeA(pDevModeInput);
|
|
memcpy( pDevModeOutput, pTmp, pTmp->dmSize + pTmp->dmDriverExtra); // Copy into a Wide char (Larger) buffer.
|
|
HeapFree(hProcessHeap, 0, pTmp);
|
|
}
|