Windows2003-3790/base/mvdm/fax/wowfax/wowfax.c
2020-09-30 16:53:55 +02:00

511 lines
16 KiB
C

//************************************************************************
// Generic Win 3.1 fax printer driver support. 32-bit printer driver
// functions. Runs in Explorer.exe context.
//
// History:
// 02-jan-95 nandurir created.
// 01-feb-95 reedb Clean-up, support printer install and bug fixes.
// 14-mar-95 reedb Use GDI hooks to move most functionality to UI.
// 16-aug-95 reedb Move to kernel mode. Debug output and validate
// functions moved from FAXCOMM.C.
// 01-apr-02 cmjones convert back to usermode.
//
//************************************************************************
#include "wowfaxdd.h"
#include "winbase.h"
//************************************************************************
// Globals
//************************************************************************
DRVFN DrvFnTab[] = {
{INDEX_DrvEnablePDEV, (PFN)DrvEnablePDEV },
{INDEX_DrvDisablePDEV, (PFN)DrvDisablePDEV },
{INDEX_DrvCompletePDEV, (PFN)DrvCompletePDEV },
{INDEX_DrvEnableSurface, (PFN)DrvEnableSurface },
{INDEX_DrvDisableSurface, (PFN)DrvDisableSurface },
{INDEX_DrvStartDoc, (PFN)DrvStartDoc },
{INDEX_DrvStartPage, (PFN)DrvStartPage },
{INDEX_DrvSendPage, (PFN)DrvSendPage },
{INDEX_DrvEndDoc, (PFN)DrvEndDoc },
{INDEX_DrvDitherColor, (PFN)DrvDitherColor },
{INDEX_DrvEscape, (PFN)DrvEscape},
};
#define NO_DRVFN (sizeof(DrvFnTab) / sizeof(DrvFnTab[0]))
#if DBG
//************************************************************************
// faxlogprintf -
//
//
//************************************************************************
VOID faxlogprintf(PCHAR pszFmt, ...)
{
va_list ap;
CHAR buffer[512];
va_start(ap, pszFmt);
wvsprintfA(buffer, pszFmt, ap);
OutputDebugStringA(buffer);
va_end(ap);
}
#endif
//************************************************************************
// ValidateFaxDev - Validates the FAXDEV structure by checking the DWORD
// signature, which is a known fixed value.
//
//************************************************************************
BOOL ValidateFaxDev(LPFAXDEV lpFaxDev)
{
if (lpFaxDev) {
if (lpFaxDev->id == FAXDEV_ID) {
return TRUE;
}
LOGDEBUG(("ValidateFaxDev failed, bad id, lpFaxDev: %X\n", lpFaxDev));
}
else {
LOGDEBUG(("ValidateFaxDev failed, lpFaxDev: NULL\n"));
}
return FALSE;
}
//************************************************************************
// DllInitProc
//************************************************************************
BOOL DllInitProc(HMODULE hModule, DWORD Reason, PCONTEXT pContext)
{
UNREFERENCED_PARAMETER(hModule);
UNREFERENCED_PARAMETER(Reason);
UNREFERENCED_PARAMETER(pContext);
return TRUE;
}
//***************************************************************************
// DrvEnableDriver
//***************************************************************************
BOOL DrvEnableDriver(ULONG iEngineVersion, ULONG cb, DRVENABLEDATA *pded)
{
LOGDEBUG(("WOWFAX!DrvEnableDriver, iEngineVersion: %X, cb: %X, pded: %X\n", iEngineVersion, cb, pded));
if(!pded) {
return FALSE;
}
pded->iDriverVersion = DDI_DRIVER_VERSION_NT4;
if (cb < sizeof(DRVENABLEDATA)) {
LOGDEBUG(("WOWFAX!DrvEnableDriver, failed\n"));
return FALSE;
}
pded->c = NO_DRVFN;
pded->pdrvfn = DrvFnTab;
return TRUE;
}
//***************************************************************************
// DrvDisableDriver
//***************************************************************************
VOID DrvDisableDriver(VOID)
{
LOGDEBUG(("WOWFAX!DrvDisableDriver\n"));
return;
}
//***************************************************************************
// DrvEnablePDEV
//***************************************************************************
DHPDEV DrvEnablePDEV(DEVMODEW *pdevmode, // Driver data, Client FAXDEV
PWSTR pwstrPrtName, // Printer's name in CreateDC()
ULONG cPatterns, // Count of standard patterns
HSURF *phsurfPatterns, // Buffer for standard patterns
ULONG cjGdiInfo, // Size of buffer for GdiInfo
ULONG *pulGdiInfo, // Buffer for GDIINFO
ULONG cjDevInfo, // Number of bytes in devinfo
DEVINFO *pdevinfo, // Device info
HDEV hdev,
PWSTR pwstrDeviceName, // Device Name - "LaserJet II"
HANDLE hDriver // Printer handle for spooler access
)
{
LPFAXDEV lpCliFaxDev, lpSrvFaxDev = NULL;
LOGDEBUG(("WOWFAX!DrvEnablePDEV, pdevmode: %X, pwstrPrtName: %S\n", pdevmode, pwstrPrtName));
if (pdevmode) {
// Point to the end of the DEVMODE where the FAXDEV is located.
lpCliFaxDev = (LPFAXDEV) ((PBYTE)pdevmode + pdevmode->dmSize);
// Allocate a server side FAXDEV to be passed back to GDI. Copy the
// client side FAXDEV to the server side FAXDEV. Note all pointers in
// the client FAXDEV reference client side memory and cannot be
// dereferenced on the server side.
lpSrvFaxDev = (LPFAXDEV)EngAllocMem(FL_ZERO_MEMORY, sizeof(FAXDEV), FAXDEV_ID);
LOGDEBUG(("WOWFAX!DrvEnablePDEV, allocated lpSrvFaxDev: %X\n", lpSrvFaxDev));
if (InitPDEV(lpCliFaxDev, lpSrvFaxDev,
cPatterns, phsurfPatterns,
cjGdiInfo, pulGdiInfo,
cjDevInfo, pdevinfo)) {
lpSrvFaxDev->hDriver = hDriver;
return (DHPDEV)lpSrvFaxDev;
}
else {
LOGDEBUG(("WOWFAX!DrvEnablePDEV, failed\n"));
if (lpSrvFaxDev) {
EngFreeMem(lpSrvFaxDev);
lpSrvFaxDev = NULL;
}
}
}
return (DHPDEV)lpSrvFaxDev;
}
//***************************************************************************
// InitPDEV - Called by DrvEnablePDEV to initialize the server side PDEV/FAXDEV.
//***************************************************************************
BOOL InitPDEV(
LPFAXDEV lpCliFaxDev, // Pointer to the client side FAXDEV
LPFAXDEV lpSrvFaxDev, // Pointer to the server side FAXDEV
ULONG cPatterns, // Count of standard patterns
HSURF *phsurfPatterns, // Buffer for standard patterns
ULONG cjGdiInfo, // Size of buffer for GdiInfo
ULONG *pulGdiInfo, // Buffer for GDIINFO
ULONG cjDevInfo, // Number of bytes in devinfo
DEVINFO *pdevinfo // Device info
)
{
PGDIINFO pgdiinfo = (PGDIINFO)pulGdiInfo;
ULONG uColors[2];
if (!ValidateFaxDev(lpCliFaxDev)) {
return FALSE;
}
// lpSrvFaxDev hasn't been initialized yet, so just check pointer.
if (lpSrvFaxDev == NULL) {
LOGDEBUG(("WOWFAX!InitPDEV, failed, NULL lpSrvFaxDev parameter\n"));
return FALSE;
}
// Copy client FAXDEV to server.
RtlCopyMemory(lpSrvFaxDev, lpCliFaxDev, sizeof(FAXDEV));
// Copy GDIINFO from client FAXDEV to the GDI buffer for GDIINFO.
RtlCopyMemory(pgdiinfo, &(lpCliFaxDev->gdiinfo), sizeof(GDIINFO));
// Initialize the DEVINFO structure.
uColors[0] = RGB(0x00, 0x00, 0x00);
uColors[1] = RGB(0xff, 0xff, 0xff);
pdevinfo->hpalDefault = EngCreatePalette(PAL_INDEXED, 2, uColors, 0, 0, 0);
pdevinfo->iDitherFormat = BMF_1BPP;
// Make sure we don't journal.
pdevinfo->flGraphicsCaps |= GCAPS_DONTJOURNAL;
// Make sure we do dither.
pdevinfo->flGraphicsCaps |= GCAPS_HALFTONE | GCAPS_MONO_DITHER |
GCAPS_COLOR_DITHER;
// Copy the DEVINFO data to the server side FAXDEV.
RtlCopyMemory(&(lpSrvFaxDev->devinfo), pdevinfo, sizeof(DEVINFO));
return TRUE;
}
//***************************************************************************
// DrvCompletePDEV
//***************************************************************************
VOID DrvCompletePDEV(DHPDEV dhpdev, HDEV hdev)
{
LPFAXDEV lpSrvFaxDev = (LPFAXDEV) dhpdev;
LOGDEBUG(("WOWFAX!DrvCompletePDEV, dhpdev %X\n", dhpdev));
if (ValidateFaxDev(lpSrvFaxDev)) {
// Store the gdi handle.
lpSrvFaxDev->hdev = hdev;
}
else {
LOGDEBUG(("WOWFAX!DrvCompletePDEV, failed\n"));
}
return;
}
//***************************************************************************
// DrvDisablePDEV
//***************************************************************************
VOID DrvDisablePDEV(DHPDEV dhpdev)
{
LPFAXDEV lpSrvFaxDev = (LPFAXDEV) dhpdev;
LOGDEBUG(("WOWFAX!DrvDisablePDEV, dhpdev %X\n", dhpdev));
if (ValidateFaxDev(lpSrvFaxDev)) {
if (lpSrvFaxDev->devinfo.hpalDefault) {
EngDeletePalette(lpSrvFaxDev->devinfo.hpalDefault);
}
EngFreeMem(lpSrvFaxDev);
LOGDEBUG(("WOWFAX!DrvDisablePDEV, deallocated lpSrvFaxDev: %X\n", lpSrvFaxDev));
}
else {
LOGDEBUG(("WOWFAX!DrvDisablePDEV, failed\n"));
}
return;
}
//***************************************************************************
// DrvEnableSurface
//***************************************************************************
HSURF DrvEnableSurface(DHPDEV dhpdev)
{
LPFAXDEV lpFaxDev = (LPFAXDEV)dhpdev;
HBITMAP hbm = 0;
LOGDEBUG(("WOWFAX!DrvEnableSurface, lpFaxDev: %X\n", lpFaxDev));
if (ValidateFaxDev(lpFaxDev)) {
// GDI will allocate space for the bitmap bits. We'll use a DrvEscape
// to copy them to the client side.
hbm = EngCreateBitmap(lpFaxDev->gdiinfo.szlPhysSize,
lpFaxDev->bmWidthBytes,
lpFaxDev->bmFormat, BMF_TOPDOWN, NULL);
if (hbm) {
lpFaxDev->hbm = hbm;
EngAssociateSurface((HSURF)hbm, lpFaxDev->hdev, 0);
return (HSURF)hbm;
}
LOGDEBUG(("WOWFAX!DrvEnableSurface, EngCreateBitmap failed\n"));
}
return (HSURF)hbm;
}
//***************************************************************************
// DrvDisableSurface
//***************************************************************************
VOID DrvDisableSurface(
DHPDEV dhpdev
)
{
LPFAXDEV lpFaxDev = (LPFAXDEV)dhpdev;
LOGDEBUG(("WOWFAX!DrvDisableSurface, lpFaxDev: %X\n", lpFaxDev));
if (ValidateFaxDev(lpFaxDev)) {
if (lpFaxDev->hbm) {
EngDeleteSurface((HSURF)lpFaxDev->hbm);
lpFaxDev->hbm = 0;
return;
}
}
return;
}
//***************************************************************************
// DrvStartDoc
//***************************************************************************
BOOL DrvStartDoc(
SURFOBJ *pso,
PWSTR pwszDocName,
DWORD dwJobId
)
{
LOGDEBUG(("WOWFAX!DrvStartDoc, pso: %X, pwszDocName: %S, dwJobId: %X\n", pso, pwszDocName, dwJobId));
return TRUE;
}
//***************************************************************************
// DrvStartPage
//***************************************************************************
BOOL DrvStartPage(
SURFOBJ *pso
)
{
LPFAXDEV lpFaxDev;
BITMAP bm;
RECTL rc;
LOGDEBUG(("WOWFAX!DrvStartPage, pso: %X\n", pso));
// Calculate the size of the rectangle based on the input data
// in 'pso' - this will ensure that the entire bitmap is erased.
if (pso) {
lpFaxDev = (LPFAXDEV)pso->dhpdev;
if(ValidateFaxDev(lpFaxDev)) {
rc.left = 0;
rc.top = 0;
//rc.right = pso->lDelta * lpFaxDev->cPixPerByte;
//rc.bottom = pso->cjBits / pso->lDelta;
// These should be calc'd the same way that they were for
// the call to EngCreateBitmp() above. Otherwise, if the rect
// specified to EngEraseSurface() is larger than the bitmap rect,
// EngEraseSurface() will fail -- resulting in a black background.
rc.right = lpFaxDev->gdiinfo.szlPhysSize.cx;
rc.bottom = lpFaxDev->gdiinfo.szlPhysSize.cy;
EngEraseSurface(pso, &rc, COLOR_INDEX_WHITE);
return TRUE;
}
}
return FALSE;
}
//***************************************************************************
// DrvSendPage
//***************************************************************************
BOOL DrvSendPage(SURFOBJ *pso)
{
LOGDEBUG(("WOWFAX!DrvSendPage, pso %X\n", pso));
return TRUE;
}
//***************************************************************************
// DrvEndDoc
//***************************************************************************
BOOL DrvEndDoc(SURFOBJ *pso, FLONG fl)
{
LOGDEBUG(("WOWFAX!DrvEndDoc, pso %X\n", pso));
return TRUE;
}
ULONG
DrvDitherColor(
DHPDEV dhpdev,
ULONG iMode,
ULONG rgbColor,
ULONG *pulDither
)
{
return DCR_HALFTONE;
}
//***************************************************************************
// DrvEscape - Allows client side to get server side data.
//***************************************************************************
ULONG DrvEscape(
SURFOBJ *pso,
ULONG iEsc,
ULONG cjIn,
PVOID *pvIn,
ULONG cjOut,
PVOID *pvOut
)
{
LPFAXDEV lpSrvFaxDev;
ULONG ulRet = 0;
LOGDEBUG(("WOWFAX!DrvEscape, pso %X, iEsc: %X\n", pso, iEsc));
if (pso) {
lpSrvFaxDev = (LPFAXDEV)pso->dhpdev;
if (ValidateFaxDev(lpSrvFaxDev)) {
LOGDEBUG(("WOWFAX!DrvEscape, lpSrvFaxDev: %X\n", lpSrvFaxDev));
switch (iEsc) {
case DRV_ESC_GET_DEVMODE_PTR:
return (ULONG) lpSrvFaxDev->pdevmode;
case DRV_ESC_GET_FAXDEV_PTR:
return (ULONG) lpSrvFaxDev->lpClient;
case DRV_ESC_GET_SURF_INFO:
if (pvOut) {
if (cjOut == sizeof(LONG)) {
(LONG) *pvOut = pso->lDelta;
return (ULONG) pso->cjBits;
}
}
break;
case DRV_ESC_GET_BITMAP_BITS:
// Validate the buffer pointer and copy the bits.
if (pvOut) {
if (cjOut == pso->cjBits) {
RtlCopyMemory(pvOut, pso->pvBits, cjOut);
return cjOut;
}
LOGDEBUG(("WOWFAX!DrvEscape, bitmap size mismatch cjIn: %X, pso->cjBits: %X\n", cjIn, pso->cjBits));
}
break;
// default:
LOGDEBUG(("WOWFAX!DrvEscape, unknown escape: %X\n", iEsc));
} //switch
}
}
LOGDEBUG(("WOWFAX!DrvEscape, failed\n"));
return ulRet;
}
//***************************************************************************
// DrvQueryDriverInfo
// dwMode - Specify the information being queried
// pBuffer - Points to output buffer
// cbBuf - Size of output buffer in bytes
// pcbNeeded - Return the expected size of output buffer
//
//***************************************************************************
BOOL DrvQueryDriverInfo(
DWORD dwMode,
PVOID pBuffer,
DWORD cbBuf,
PDWORD pcbNeeded
)
{
UNREFERENCED_PARAMETER(pBuffer);
UNREFERENCED_PARAMETER(cbBuf);
UNREFERENCED_PARAMETER(pcbNeeded);
switch(dwMode) {
// this is the only one currently supported
case DRVQUERY_USERMODE:
return TRUE;
default:
LOGDEBUG(("DrvQueryDriverInfo: unexpected case: dwMode = %X\n",
dwMode));
return FALSE;
}
}