| 1 | /* $Id: printer.cpp,v 1.14 2004/05/04 13:49:24 sandervl Exp $ */
|
|---|
| 2 |
|
|---|
| 3 | /*
|
|---|
| 4 | * GDI32 printer apis
|
|---|
| 5 | *
|
|---|
| 6 | * Copyright 2001 Sander van Leeuwen (sandervl@xs4all.nl)
|
|---|
| 7 | * Copyright 2002-2003 InnoTek Systemberatung GmbH (sandervl@innotek.de)
|
|---|
| 8 | *
|
|---|
| 9 | * Parts based on Wine code (StartDocW)
|
|---|
| 10 | *
|
|---|
| 11 | * Copyright 1996 John Harvey
|
|---|
| 12 | * Copyright 1998 Huw Davies
|
|---|
| 13 | * Copyright 1998 Andreas Mohr
|
|---|
| 14 | * Copyright 1999 Klaas van Gend
|
|---|
| 15 | *
|
|---|
| 16 | * Project Odin Software License can be found in LICENSE.TXT
|
|---|
| 17 | *
|
|---|
| 18 | */
|
|---|
| 19 | #include <os2win.h>
|
|---|
| 20 | #include <stdarg.h>
|
|---|
| 21 | #include <stdio.h>
|
|---|
| 22 | #include <string.h>
|
|---|
| 23 | #include <dcdata.h>
|
|---|
| 24 | #include <dbglog.h>
|
|---|
| 25 | #include <heapstring.h>
|
|---|
| 26 | #include "oslibgpi.h"
|
|---|
| 27 |
|
|---|
| 28 | #define DBG_LOCALLOG DBG_printer
|
|---|
| 29 | #include "dbglocal.h"
|
|---|
| 30 |
|
|---|
| 31 | // Private WGSS escape extensions
|
|---|
| 32 | #define PRIVATE_QUERYORIENTATION 0x99990001
|
|---|
| 33 | #define PRIVATE_QUERYCOPIES 0x99990002
|
|---|
| 34 | #define PRIVATE_QUERYCOLLATE 0x99990003
|
|---|
| 35 |
|
|---|
| 36 | static char *lpszPassThrough = NULL;
|
|---|
| 37 | static int cbPassThrough = 0;
|
|---|
| 38 |
|
|---|
| 39 | //NOTE: We need to insert this postscript statement into the stream or else
|
|---|
| 40 | // the output will be completely inverted (x & y)
|
|---|
| 41 | static char szSetupStringPortrait[] = "%%BeginSetup\n[{\n%%BeginColorModelSetup\n<< /ProcessColorModel /DeviceCMYK >> setpagedevice\n%%EndColorModelSetup\n} stopped cleartomark\n%%EndSetup\n";
|
|---|
| 42 | static char szSetupStringLandscape[] = "%%BeginSetup\n[{\n%%BeginColorModelSetup\n<< /ProcessColorModel /DeviceCMYK >> setpagedevice\n%%EndColorModelSetup\n} stopped cleartomark\n-90 rotate\n%%EndSetup\n";
|
|---|
| 43 |
|
|---|
| 44 |
|
|---|
| 45 | static BOOL fPostscriptPassthrough = TRUE;
|
|---|
| 46 |
|
|---|
| 47 | //******************************************************************************
|
|---|
| 48 | //******************************************************************************
|
|---|
| 49 | BOOL WIN32API ODIN_QueryPostscriptPassthrough()
|
|---|
| 50 | {
|
|---|
| 51 | return fPostscriptPassthrough;
|
|---|
| 52 | }
|
|---|
| 53 | //******************************************************************************
|
|---|
| 54 | //******************************************************************************
|
|---|
| 55 | void WIN32API ODIN_SetPostscriptPassthrough(BOOL fEnabled)
|
|---|
| 56 | {
|
|---|
| 57 | fPostscriptPassthrough = fEnabled;
|
|---|
| 58 | }
|
|---|
| 59 | //******************************************************************************
|
|---|
| 60 | //******************************************************************************
|
|---|
| 61 | int WIN32API Escape( HDC hdc, int nEscape, int cbInput, LPCSTR lpvInData, PVOID lpvOutData)
|
|---|
| 62 | {
|
|---|
| 63 | int rc;
|
|---|
| 64 | const char *lpszEscape = NULL;
|
|---|
| 65 |
|
|---|
| 66 | switch(nEscape)
|
|---|
| 67 | {
|
|---|
| 68 | case SET_BOUNDS:
|
|---|
| 69 | {
|
|---|
| 70 | RECT *r = (RECT *)lpvInData;
|
|---|
| 71 | if(cbInput != sizeof(RECT)) {
|
|---|
| 72 | dprintf(("WARNING: cbInput != sizeof(RECT) (=%d) for SET_BOUNDS", cbInput));
|
|---|
| 73 | return 0;
|
|---|
| 74 | }
|
|---|
| 75 | dprintf(("SET_BOUNDS (%d,%d) (%d,%d)", r->left, r->top, r->right, r->bottom));
|
|---|
| 76 | return 0;
|
|---|
| 77 | }
|
|---|
| 78 | case POSTSCRIPT_PASSTHROUGH:
|
|---|
| 79 | if(!lpszEscape) lpszEscape = "POSTSCRIPT_PASSTHROUGH";
|
|---|
| 80 | nEscape = PASSTHROUGH;
|
|---|
| 81 | case POSTSCRIPT_DATA:
|
|---|
| 82 | if(!lpszEscape) lpszEscape = "POSTSCRIPT_DATA";
|
|---|
| 83 | case PASSTHROUGH:
|
|---|
| 84 | {
|
|---|
| 85 | if(!lpszEscape) lpszEscape = "PASSTHROUGH";
|
|---|
| 86 | dprintf(("Postscript %s data of size %d", lpszEscape, *(WORD *)lpvInData));
|
|---|
| 87 |
|
|---|
| 88 | if(fPostscriptPassthrough == FALSE)
|
|---|
| 89 | {
|
|---|
| 90 | return 0;
|
|---|
| 91 | }
|
|---|
| 92 | if(lpszPassThrough) {
|
|---|
| 93 | O32_Escape(hdc, nEscape, cbPassThrough, lpszPassThrough, NULL);
|
|---|
| 94 | free(lpszPassThrough);
|
|---|
| 95 | lpszPassThrough = NULL;
|
|---|
| 96 | }
|
|---|
| 97 | pDCData pHps = (pDCData)OSLibGpiQueryDCData((HPS)hdc);
|
|---|
| 98 | if(pHps && (pHps->Reserved & DC_FLAG_SEND_POSTSCRIPT_SETUP_STRING))
|
|---|
| 99 | {
|
|---|
| 100 | DWORD dmOrientation = 0, hdrsize, dmCollate, dmCopies;
|
|---|
| 101 | char *pvSetupData = NULL;
|
|---|
| 102 |
|
|---|
| 103 | O32_Escape(hdc, PRIVATE_QUERYORIENTATION, 0, NULL, &dmOrientation);
|
|---|
| 104 | if(dmOrientation == DMORIENT_LANDSCAPE) {
|
|---|
| 105 | hdrsize = sizeof(szSetupStringLandscape)-1;
|
|---|
| 106 | pvSetupData = (char *)alloca(hdrsize+sizeof(WORD));
|
|---|
| 107 | }
|
|---|
| 108 | else {//portrait
|
|---|
| 109 | hdrsize = sizeof(szSetupStringPortrait)-1;
|
|---|
| 110 | pvSetupData = (char *)alloca(hdrsize+sizeof(WORD));
|
|---|
| 111 | }
|
|---|
| 112 | if(pvSetupData) {
|
|---|
| 113 | *(WORD *)pvSetupData = hdrsize;
|
|---|
| 114 | memcpy(pvSetupData+sizeof(WORD), (hdrsize == sizeof(szSetupStringPortrait)-1) ? szSetupStringPortrait : szSetupStringLandscape, hdrsize);
|
|---|
| 115 |
|
|---|
| 116 | O32_Escape(hdc, nEscape, *(WORD *)pvSetupData, pvSetupData, NULL);
|
|---|
| 117 |
|
|---|
| 118 | dprintf(("Send Postscript setup string %d (%x):\n%s", dmOrientation, pHps->Reserved, pvSetupData+sizeof(WORD)));
|
|---|
| 119 | }
|
|---|
| 120 | pHps->Reserved &= ~DC_FLAG_SEND_POSTSCRIPT_SETUP_STRING;
|
|---|
| 121 | }
|
|---|
| 122 |
|
|---|
| 123 | rc = O32_Escape(hdc, nEscape, cbInput, lpvInData, lpvOutData);
|
|---|
| 124 | if(rc == 1) rc = *(WORD *)lpvInData;
|
|---|
| 125 | else rc = 0;
|
|---|
| 126 | return rc;
|
|---|
| 127 | }
|
|---|
| 128 |
|
|---|
| 129 | case PRIVATE_QUERYORIENTATION:
|
|---|
| 130 | DebugInt3();
|
|---|
| 131 | return 0;
|
|---|
| 132 |
|
|---|
| 133 | case SPCLPASSTHROUGH2:
|
|---|
| 134 | {
|
|---|
| 135 | int rawsize = *(WORD *)(lpvInData+4);
|
|---|
| 136 |
|
|---|
| 137 | dprintf(("SPCLPASSTHROUGH2: pretend success"));
|
|---|
| 138 | dprintf(("SPCLPASSTHROUGH2: virt mem %x size %x", *(DWORD *)lpvInData, rawsize));
|
|---|
| 139 | if(lpszPassThrough == NULL) {
|
|---|
| 140 | lpszPassThrough = (char *)malloc(rawsize+sizeof(WORD));
|
|---|
| 141 | if(lpszPassThrough == NULL) {
|
|---|
| 142 | DebugInt3();
|
|---|
| 143 | return 0;
|
|---|
| 144 | }
|
|---|
| 145 | memcpy(lpszPassThrough+sizeof(WORD), (char *)lpvInData+6, rawsize);
|
|---|
| 146 | cbPassThrough = rawsize;
|
|---|
| 147 | *(WORD *)lpszPassThrough = (WORD)cbPassThrough;
|
|---|
| 148 | }
|
|---|
| 149 | else {
|
|---|
| 150 | char *tmp = lpszPassThrough;
|
|---|
| 151 |
|
|---|
| 152 | lpszPassThrough = (char *)malloc(cbPassThrough+rawsize+sizeof(WORD));
|
|---|
| 153 | if(lpszPassThrough == NULL) {
|
|---|
| 154 | DebugInt3();
|
|---|
| 155 | return 0;
|
|---|
| 156 | }
|
|---|
| 157 | memcpy(lpszPassThrough+sizeof(WORD), tmp+sizeof(WORD), cbPassThrough);
|
|---|
| 158 | free(tmp);
|
|---|
| 159 | memcpy(lpszPassThrough+sizeof(WORD)+cbPassThrough, (char *)lpvInData+6, rawsize);
|
|---|
| 160 | cbPassThrough += rawsize;
|
|---|
| 161 | *(WORD *)lpszPassThrough = (WORD)cbPassThrough;
|
|---|
| 162 | }
|
|---|
| 163 | return 1;
|
|---|
| 164 | }
|
|---|
| 165 |
|
|---|
| 166 | case DOWNLOADFACE:
|
|---|
| 167 | dprintf(("DOWNLOADFACE: pretend success"));
|
|---|
| 168 | return 1;
|
|---|
| 169 |
|
|---|
| 170 | case POSTSCRIPT_IGNORE:
|
|---|
| 171 | {
|
|---|
| 172 | BOOL ret = FALSE;
|
|---|
| 173 | dprintf(("POSTSCRIPT_IGNORE %d", *(WORD *)lpvInData));
|
|---|
| 174 | return ret;
|
|---|
| 175 | }
|
|---|
| 176 | case EPSPRINTING:
|
|---|
| 177 | {
|
|---|
| 178 | UINT epsprint = *(UINT*)lpvInData;
|
|---|
| 179 | dprintf(("EPSPRINTING support %sable.\n",epsprint?"en":"dis"));
|
|---|
| 180 | return 1;
|
|---|
| 181 | }
|
|---|
| 182 |
|
|---|
| 183 | case QUERYESCSUPPORT:
|
|---|
| 184 | {
|
|---|
| 185 | DWORD nEscapeSup = *(DWORD *)lpvInData;
|
|---|
| 186 | BOOL fdprintf = FALSE;
|
|---|
| 187 |
|
|---|
| 188 | switch(nEscapeSup) {
|
|---|
| 189 | case SET_BOUNDS:
|
|---|
| 190 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: SET_BOUNDS"));
|
|---|
| 191 | fdprintf = TRUE;
|
|---|
| 192 | /* case SPCLPASSTHROUGH2:
|
|---|
| 193 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: SPCLPASSTHROUGH2"));
|
|---|
| 194 | fdprintf = TRUE;
|
|---|
| 195 | case DOWNLOADHEADER:
|
|---|
| 196 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: DOWNLOADHEADER"));
|
|---|
| 197 | fdprintf = TRUE;
|
|---|
| 198 | case DOWNLOADFACE:
|
|---|
| 199 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: DOWNLOADFACE"));
|
|---|
| 200 | fdprintf = TRUE;
|
|---|
| 201 | */
|
|---|
| 202 | case POSTSCRIPT_IGNORE:
|
|---|
| 203 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: POSTSCRIPT_IGNORE"));
|
|---|
| 204 | fdprintf = TRUE;
|
|---|
| 205 | case EPSPRINTING:
|
|---|
| 206 | {
|
|---|
| 207 | if(!fdprintf) dprintf(("QUERYESCSUPPORT: EPSPRINTING"));
|
|---|
| 208 | fdprintf = TRUE;
|
|---|
| 209 |
|
|---|
| 210 | nEscapeSup = POSTSCRIPT_PASSTHROUGH;
|
|---|
| 211 | return Escape(hdc, QUERYESCSUPPORT, sizeof(nEscapeSup), (LPCSTR)&nEscapeSup, NULL);
|
|---|
| 212 | }
|
|---|
| 213 | case POSTSCRIPT_PASSTHROUGH:
|
|---|
| 214 | case POSTSCRIPT_DATA:
|
|---|
| 215 | case PASSTHROUGH:
|
|---|
| 216 | if(fPostscriptPassthrough == FALSE)
|
|---|
| 217 | {
|
|---|
| 218 | return 0;
|
|---|
| 219 | }
|
|---|
| 220 | break;
|
|---|
| 221 |
|
|---|
| 222 | default:
|
|---|
| 223 | break;
|
|---|
| 224 | }
|
|---|
| 225 | }
|
|---|
| 226 | default:
|
|---|
| 227 | if(cbInput && lpvInData) {
|
|---|
| 228 | ULONG *tmp = (ULONG *)lpvInData;
|
|---|
| 229 | for(int i=0;i<min(16,cbInput/4);i++) {
|
|---|
| 230 | dprintf(("GDI32: Escape par %d: %x", i, *tmp++));
|
|---|
| 231 | }
|
|---|
| 232 | }
|
|---|
| 233 | break;
|
|---|
| 234 | }
|
|---|
| 235 |
|
|---|
| 236 | rc = O32_Escape(hdc, nEscape, cbInput, lpvInData, lpvOutData);
|
|---|
| 237 | if(rc == 0) {
|
|---|
| 238 | dprintf(("GDI32: Escape %x %d %d %x %x returned %d (WARNING: might not be implemented!!) ", hdc, nEscape, cbInput, lpvInData, lpvOutData, rc));
|
|---|
| 239 | }
|
|---|
| 240 | else dprintf(("GDI32: Escape %x %d %d %x %x returned %d ", hdc, nEscape, cbInput, lpvInData, lpvOutData, rc));
|
|---|
| 241 |
|
|---|
| 242 | return rc;
|
|---|
| 243 | }
|
|---|
| 244 | //******************************************************************************
|
|---|
| 245 | //******************************************************************************
|
|---|
| 246 | INT WIN32API ExtEscape(HDC hdc, INT nEscape, INT cbInput, LPCSTR lpszInData,
|
|---|
| 247 | INT cbOutput, LPSTR lpszOutData)
|
|---|
| 248 | {
|
|---|
| 249 | dprintf(("GDI32: ExtEscape, %x %x %d %x %d %x partly implemented", hdc, nEscape, cbInput, lpszInData, cbOutput, lpszOutData));
|
|---|
| 250 |
|
|---|
| 251 | switch(nEscape) {
|
|---|
| 252 | case DOWNLOADHEADER:
|
|---|
| 253 | {
|
|---|
| 254 | dprintf(("DOWNLOADHEADER: hardcoded result"));
|
|---|
| 255 | if(lpszOutData && cbOutput >= 8) {
|
|---|
| 256 | strcpy(lpszOutData, "PM_1.2");
|
|---|
| 257 | return 1;
|
|---|
| 258 | }
|
|---|
| 259 | dprintf(("Not enough room for proc name"));
|
|---|
| 260 | return 0;
|
|---|
| 261 | }
|
|---|
| 262 | }
|
|---|
| 263 | return Escape(hdc, nEscape, cbInput, lpszInData, lpszOutData);
|
|---|
| 264 | }
|
|---|
| 265 | //******************************************************************************
|
|---|
| 266 | //******************************************************************************
|
|---|
| 267 | INT WIN32API StartDocA(HDC hdc, const DOCINFOA *lpDocInfo)
|
|---|
| 268 | {
|
|---|
| 269 | INT ret;
|
|---|
| 270 |
|
|---|
| 271 | if(lpDocInfo) {
|
|---|
| 272 | dprintf(("GDI32: StartDocA %x %x (%s %s %s %x)", hdc, lpDocInfo, lpDocInfo->lpszDocName, lpDocInfo->lpszOutput, lpDocInfo->lpszDatatype, lpDocInfo->fwType));
|
|---|
| 273 | }
|
|---|
| 274 | else dprintf(("GDI32: StartDocA %x %x", hdc, lpDocInfo));
|
|---|
| 275 |
|
|---|
| 276 | dprintf(("lpDocInfo->cbSize %x", lpDocInfo->cbSize));
|
|---|
| 277 | dprintf(("lpDocInfo->lpszDocName %s", lpDocInfo->lpszDocName));
|
|---|
| 278 | dprintf(("lpDocInfo->lpszOutput %s", lpDocInfo->lpszOutput));
|
|---|
| 279 | dprintf(("lpDocInfo->lpszDatatype %s", lpDocInfo->lpszDatatype));
|
|---|
| 280 | dprintf(("lpDocInfo->fwType %x", lpDocInfo->fwType));
|
|---|
| 281 |
|
|---|
| 282 | ret = O32_StartDoc(hdc, (LPDOCINFOA)lpDocInfo);
|
|---|
| 283 |
|
|---|
| 284 | if(ret != 0) {
|
|---|
| 285 | pDCData pHps = (pDCData)OSLibGpiQueryDCData((HPS)hdc);
|
|---|
| 286 | if(pHps)
|
|---|
| 287 | {
|
|---|
| 288 | pHps->Reserved |= DC_FLAG_SEND_POSTSCRIPT_SETUP_STRING;
|
|---|
| 289 | }
|
|---|
| 290 | else DebugInt3();
|
|---|
| 291 | }
|
|---|
| 292 |
|
|---|
| 293 | return ret;
|
|---|
| 294 | }
|
|---|
| 295 | //******************************************************************************
|
|---|
| 296 | //******************************************************************************
|
|---|
| 297 | INT WIN32API StartDocW(HDC hdc, const DOCINFOW* doc)
|
|---|
| 298 | {
|
|---|
| 299 | DOCINFOA docA;
|
|---|
| 300 | INT ret;
|
|---|
| 301 |
|
|---|
| 302 | docA.cbSize = doc->cbSize;
|
|---|
| 303 | docA.lpszDocName = doc->lpszDocName ?
|
|---|
| 304 | HEAP_strdupWtoA( GetProcessHeap(), 0, doc->lpszDocName ) : NULL;
|
|---|
| 305 | docA.lpszOutput = doc->lpszOutput ?
|
|---|
| 306 | HEAP_strdupWtoA( GetProcessHeap(), 0, doc->lpszOutput ) : NULL;
|
|---|
| 307 | docA.lpszDatatype = doc->lpszDatatype ?
|
|---|
| 308 | HEAP_strdupWtoA( GetProcessHeap(), 0, doc->lpszDatatype ) : NULL;
|
|---|
| 309 | docA.fwType = doc->fwType;
|
|---|
| 310 |
|
|---|
| 311 | ret = StartDocA(hdc, &docA);
|
|---|
| 312 |
|
|---|
| 313 | if(docA.lpszDocName)
|
|---|
| 314 | HeapFree( GetProcessHeap(), 0, (LPSTR)docA.lpszDocName );
|
|---|
| 315 | if(docA.lpszOutput)
|
|---|
| 316 | HeapFree( GetProcessHeap(), 0, (LPSTR)docA.lpszOutput );
|
|---|
| 317 | if(docA.lpszDatatype)
|
|---|
| 318 | HeapFree( GetProcessHeap(), 0, (LPSTR)docA.lpszDatatype );
|
|---|
| 319 |
|
|---|
| 320 | return ret;
|
|---|
| 321 | }
|
|---|
| 322 | //******************************************************************************
|
|---|
| 323 | //******************************************************************************
|
|---|
| 324 | int WIN32API EndDoc( HDC hdc)
|
|---|
| 325 | {
|
|---|
| 326 | dprintf(("GDI32: EndDoc %x", hdc));
|
|---|
| 327 | return O32_EndDoc(hdc);
|
|---|
| 328 | }
|
|---|
| 329 | //******************************************************************************
|
|---|
| 330 | //******************************************************************************
|
|---|
| 331 | int WIN32API AbortDoc(HDC hdc)
|
|---|
| 332 | {
|
|---|
| 333 | dprintf(("GDI32: AbortDoc %x", hdc));
|
|---|
| 334 | return O32_AbortDoc(hdc);
|
|---|
| 335 | }
|
|---|
| 336 | //******************************************************************************
|
|---|
| 337 | //******************************************************************************
|
|---|
| 338 | int WIN32API StartPage( HDC hdc)
|
|---|
| 339 | {
|
|---|
| 340 | dprintf(("GDI32: StartPage %x", hdc));
|
|---|
| 341 | return O32_StartPage(hdc);
|
|---|
| 342 | }
|
|---|
| 343 | //******************************************************************************
|
|---|
| 344 | //******************************************************************************
|
|---|
| 345 | int WIN32API EndPage( HDC hdc)
|
|---|
| 346 | {
|
|---|
| 347 | dprintf(("GDI32: EndPage %x", hdc));
|
|---|
| 348 | return O32_EndPage(hdc);
|
|---|
| 349 | }
|
|---|
| 350 | //******************************************************************************
|
|---|
| 351 | //******************************************************************************
|
|---|
| 352 | int WIN32API SetAbortProc(HDC hdc, ABORTPROC lpAbortProc)
|
|---|
| 353 | {
|
|---|
| 354 | dprintf(("GDI32: SetAbortProc %x %x - stub (1)", hdc, lpAbortProc));
|
|---|
| 355 | return(1);
|
|---|
| 356 | }
|
|---|
| 357 | //******************************************************************************
|
|---|
| 358 | //******************************************************************************
|
|---|