Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

3674 lines
101 KiB

  1. /*++
  2. *
  3. * WOW v1.0
  4. *
  5. * Copyright (c) 1991, Microsoft Corporation
  6. *
  7. * WGDI.C
  8. * WOW32 16-bit GDI API support
  9. *
  10. * History:
  11. * 07-Mar-1991 Jeff Parsons (jeffpar)
  12. * Created.
  13. *
  14. * 09-Apr-1991 NigelT
  15. * Various defines are used here to remove calls to Win32
  16. * features which don't work yet.
  17. *
  18. * 06-June-1992 Chandan Chauhan (ChandanC)
  19. * Fixed BITMAP and DeviceIndependentBitmap (DIB) issues
  20. *
  21. * 22-May-1995 Craig Jones (a-craigj)
  22. * METAFILE NOTE: several 32-bit API's will return TRUE when GDI is in
  23. * "metafile" mode -- however, the POINT struct does not get
  24. * updated by GDI32 even if the API returns successfully so
  25. * we just return TRUE or FALSE as the point coors like W3.1.
  26. *
  27. --*/
  28. #include "precomp.h"
  29. #pragma hdrstop
  30. #include "wowgdip.h"
  31. #include "wdib.h"
  32. #include "stddef.h" // these three are needed to include the
  33. #include "wingdip.h"
  34. // definition of EXTTEXTMETRICS in wingdip.h
  35. // [bodind]
  36. MODNAME(wgdi.c);
  37. #define SETGDIXFORM 4113
  38. #define RESETPAGE 4114
  39. // This must be removed from POSTBETA for sure. ChandanC 3/22/94.
  40. #define IGNORESTARTPGAE 0x7FFFFFFF
  41. #define ADD_MSTT 0x7FFFFFFD
  42. #ifdef FE_SB
  43. // WOWCF_FE_FLW2_PRINTING_PS, Japanese Lotus Freelance printing with PostScript.
  44. // Between Escape( BEGIN_PATH ) and Escape( END_PATH ), select brush object
  45. // with WHITE_BRUSH
  46. BOOL fCmptFLW = FALSE;
  47. #endif // FE_SB
  48. LPDEVMODE GetDefaultDevMode32(LPSTR pszDriver); // Implemented in wspool.c
  49. // Hack for apps which try to be their own printer driver & send form feeds to
  50. // the printer in Escape(PassThrough) calls. This mechanism prevents an
  51. // additional page being spit out of the printer when the app calls EndDoc()
  52. // because GDI32 EndDoc() does an implicit form feed.
  53. typedef struct _FormFeedHack {
  54. struct _FormFeedHack UNALIGNED *next;
  55. HAND16 hTask16;
  56. HDC hdc;
  57. LPBYTE lpBytes;
  58. int cbBytes;
  59. } FORMFEEDHACK;
  60. typedef FORMFEEDHACK UNALIGNED *PFORMFEEDHACK;
  61. PFORMFEEDHACK gpFormFeedHackList = NULL; // start of global formfeed Hack list
  62. LONG HandleFormFeedHack(HDC hdc, LPBYTE lpdata, int cb);
  63. LPBYTE SendFrontEndOfDataStream(HDC hdc, LPBYTE lpData, int *cb, LONG *ul);
  64. void FreeFormFeedHackNode(PFORMFEEDHACK pNode);
  65. void FreeTaskFormFeedHacks(HAND16 hTask16);
  66. void SendFormFeedHack(HDC hdc);
  67. PFORMFEEDHACK FindFormFeedHackNode(HDC hdc);
  68. PFORMFEEDHACK CreateFormFeedHackNode(HDC hdc, int cb, LPBYTE lpData);
  69. void RemoveFormFeedHack(HDC hdc);
  70. ULONG FASTCALL WG32CreateBitmap(PVDMFRAME pFrame)
  71. {
  72. ULONG ul;
  73. register PCREATEBITMAP16 parg16;
  74. LPBYTE lpBitsOriginal;
  75. GETARGPTR(pFrame, sizeof(CREATEBITMAP16), parg16);
  76. GETOPTPTR(parg16->f5, 0, lpBitsOriginal);
  77. ul = GETHBITMAP16(CreateBitmap(INT32(parg16->f1),
  78. INT32(parg16->f2),
  79. LOBYTE(parg16->f3),
  80. LOBYTE(parg16->f4),
  81. lpBitsOriginal));
  82. WOW32APIWARN(ul, "CreateBitmap");
  83. FREEOPTPTR(lpBitsOriginal);
  84. FREEARGPTR(parg16);
  85. RETURN(ul);
  86. }
  87. ULONG FASTCALL WG32CreateBitmapIndirect(PVDMFRAME pFrame)
  88. {
  89. ULONG ul;
  90. register PCREATEBITMAPINDIRECT16 parg16;
  91. PBITMAP16 pbm16;
  92. BITMAP bm;
  93. LPBYTE lp = NULL;
  94. GETARGPTR(pFrame, sizeof(CREATEBITMAPINDIRECT16), parg16);
  95. GETVDMPTR(parg16->f1, sizeof(BITMAP16), pbm16);
  96. GETOPTPTR(pbm16->bmBits, 0, lp);
  97. bm.bmType = (LONG) FETCHSHORT(pbm16->bmType);
  98. bm.bmWidth = (LONG) FETCHSHORT(pbm16->bmWidth);
  99. bm.bmHeight = (LONG) FETCHSHORT(pbm16->bmHeight);
  100. bm.bmWidthBytes = (LONG) FETCHSHORT(pbm16->bmWidthBytes);
  101. bm.bmPlanes = (WORD) pbm16->bmPlanes;
  102. bm.bmBitsPixel = (WORD) pbm16->bmBitsPixel;
  103. bm.bmBits = lp;
  104. ul = GETHBITMAP16(CreateBitmapIndirect(&bm));
  105. WOW32APIWARN(ul, "CreateBitmapIndirect");
  106. FREEOPTPTR(lp);
  107. FREEARGPTR(parg16);
  108. RETURN(ul);
  109. }
  110. ULONG FASTCALL WG32CreateBrushIndirect(PVDMFRAME pFrame)
  111. {
  112. ULONG ul;
  113. LOGBRUSH t1;
  114. HAND16 hMem16;
  115. HANDLE hMem32 = NULL;
  116. LPBYTE lpMem16, lpMem32;
  117. INT cb;
  118. VPVOID vp;
  119. register PCREATEBRUSHINDIRECT16 parg16;
  120. GETARGPTR(pFrame, sizeof(CREATEBRUSHINDIRECT16), parg16);
  121. GETLOGBRUSH16(parg16->f1, &t1);
  122. // some apps don't properly set the style. Make sure it is a valid w3.1 style
  123. if (t1.lbStyle > BS_DIBPATTERN)
  124. t1.lbStyle = BS_SOLID;
  125. if (t1.lbStyle == BS_PATTERN) {
  126. t1.lbStyle = BS_PATTERN8X8;
  127. }
  128. else if (t1.lbStyle == BS_DIBPATTERN) {
  129. hMem16 = (WORD) t1.lbHatch;
  130. if (hMem16) {
  131. vp = RealLockResource16(hMem16, &cb);
  132. if (vp) {
  133. GETMISCPTR(vp, lpMem16);
  134. hMem32 = WOWGLOBALALLOC(GMEM_MOVEABLE, cb);
  135. WOW32ASSERT(hMem32);
  136. if (hMem32) {
  137. lpMem32 = GlobalLock(hMem32);
  138. RtlCopyMemory(lpMem32, lpMem16, cb);
  139. GlobalUnlock(hMem32);
  140. }
  141. GlobalUnlock16(hMem16);
  142. FREEMISCPTR(lpMem16);
  143. }
  144. }
  145. t1.lbHatch = (LONG)hMem32;
  146. }
  147. else if (t1.lbStyle == BS_SOLID)
  148. {
  149. t1.lbColor = COLOR32(t1.lbColor);
  150. }
  151. ul = GETHBRUSH16(CreateBrushIndirect(&t1));
  152. if (hMem32)
  153. {
  154. WOWGLOBALFREE(hMem32);
  155. }
  156. WOW32APIWARN(ul, "CreateBrushIndirect");
  157. FREEARGPTR(parg16);
  158. RETURN(ul);
  159. }
  160. ULONG FASTCALL WG32CreateCompatibleDC(PVDMFRAME pFrame)
  161. {
  162. ULONG ul;
  163. HDC hdc;
  164. register PCREATECOMPATIBLEDC16 parg16;
  165. GETARGPTR(pFrame, sizeof(CREATECOMPATIBLEDC16), parg16);
  166. if ( parg16->f1 ) {
  167. hdc = HDC32(parg16->f1);
  168. if ( hdc == NULL ) {
  169. FREEARGPTR(parg16);
  170. return(0);
  171. }
  172. } else {
  173. hdc = NULL;
  174. }
  175. ul = GETHDC16(CreateCompatibleDC(hdc));
  176. //
  177. // Some apps such as MSWORKS and MS PUBLISHER use some wizard code that accepts
  178. // a hDC or a hWnd as a parameter and attempt to figure out what type of handle
  179. // it is by using the IsWindow() call. Since both handles come from different
  180. // handle spaces they may end up the same value and this wizard code will end
  181. // up writing to the DC for a random window. By ORing in a 1 we ensure that the
  182. // handle types will never share the same value since all hWnds are even. Note
  183. // that this hack is also made in WU32GetDC().
  184. //
  185. // Note that there are some apps that use the lower 2 bits of the hDC for their
  186. // own purposes.
  187. if (ul && CURRENTPTD()->dwWOWCompatFlags & WOWCF_UNIQUEHDCHWND) {
  188. ul = ul | 1;
  189. }
  190. WOW32APIWARN(ul, "CreateCompatibleDC");
  191. FREEARGPTR(parg16);
  192. RETURN(ul);
  193. }
  194. ULONG FASTCALL WG32CreateDC(PVDMFRAME pFrame)
  195. {
  196. ULONG ul = 0;
  197. PSZ psz1 = NULL;
  198. PSZ psz2 = NULL;
  199. PSZ psz3 = NULL;
  200. PSZ pszDib;
  201. LPDEVMODE t4 = NULL;
  202. register PCREATEDC16 parg16;
  203. GETARGPTR(pFrame, sizeof(CREATEDC16), parg16);
  204. if(parg16->f1) {
  205. if(!(psz1 = malloc_w_strcpy_vp16to32(parg16->f1, FALSE, 0)))
  206. goto ExitPath;
  207. }
  208. if(parg16->f2) {
  209. if(!(psz2 = malloc_w_strcpy_vp16to32(parg16->f2, FALSE, 0)))
  210. goto ExitPath;
  211. }
  212. if(parg16->f3) {
  213. if(!(psz3 = malloc_w_strcpy_vp16to32(parg16->f3, FALSE, 0)))
  214. goto ExitPath;
  215. }
  216. // Note: parg16->f4 will usually be a lpDevMode, but it is documented
  217. // that it can also be a lpBitMapInfo if the driver name is "dib.drv"
  218. // test for "dib.drv". Director 4.0 uses "dirdib.drv"
  219. if (psz1 && ((pszDib = WOW32_strstr (psz1, "DIB")) ||
  220. (pszDib = WOW32_strstr (psz1, "dib")))) {
  221. if (WOW32_stricmp (pszDib, "DIB") == 0 ||
  222. WOW32_stricmp (pszDib, "DIB.DRV") == 0) {
  223. ul = GETHDC16(W32HandleDibDrv ((PVPVOID)parg16->f4));
  224. // Note: flat 16:16 ptrs should be considered invalid after this call
  225. }
  226. }
  227. // handle normal non-dib.drv case
  228. else {
  229. if (FETCHDWORD(parg16->f4) == 0L) {
  230. t4 = GetDefaultDevMode32(psz2);
  231. }
  232. else {
  233. t4 = ThunkDevMode16to32(parg16->f4);
  234. }
  235. // this can callback into a 16-bit fax driver!
  236. ul = GETHDC16(CreateDC(psz1, psz2, psz3, t4));
  237. // Note: flat 16:16 ptrs should be considered invalid after this call
  238. FREEARGPTR(parg16);
  239. }
  240. ExitPath:
  241. if(psz1)
  242. free_w(psz1);
  243. if(psz2)
  244. free_w(psz2);
  245. if(psz3)
  246. free_w(psz3);
  247. WOW32APIWARN(ul, "CreateDC");
  248. FREEDEVMODE32(t4);
  249. FREEARGPTR(parg16);
  250. RETURN(ul);
  251. }
  252. ULONG FASTCALL WG32CreateDIBPatternBrush(PVDMFRAME pFrame)
  253. {
  254. ULONG ul = 0;
  255. LPBYTE lpb16;
  256. LOGBRUSH logbr;
  257. register PCREATEDIBPATTERNBRUSH16 parg16;
  258. GETARGPTR(pFrame, sizeof(CREATEDIBPATTERNBRUSH16), parg16);
  259. GETMISCPTR(parg16->f1, lpb16);
  260. logbr.lbStyle = BS_DIBPATTERN8X8;
  261. logbr.lbColor = WORD32(parg16->f2);
  262. logbr.lbHatch = (LONG) lpb16;
  263. ul = GETHBRUSH16(CreateBrushIndirect(&logbr));
  264. WOW32APIWARN(ul, "CreateDIBPatternBrush");
  265. FREEARGPTR(parg16);
  266. RETURN(ul);
  267. }
  268. ULONG FASTCALL WG32CreateDIBitmap(PVDMFRAME pFrame)
  269. {
  270. ULONG ul=0;
  271. LPBYTE lpib4;
  272. BITMAPINFOHEADER bmxh2;
  273. STACKBMI32 bmi32;
  274. LPBITMAPINFOHEADER lpbmih32;
  275. LPBITMAPINFO lpbmi32;
  276. register PCREATEDIBITMAP16 parg16;
  277. GETARGPTR(pFrame, sizeof(CREATEDIBITMAP16), parg16);
  278. GETMISCPTR(parg16->f4, lpib4);
  279. lpbmih32 = CopyBMIH16ToBMIH32((PVPVOID) FETCHDWORD(parg16->f2), &bmxh2);
  280. if (lpbmih32) {
  281. lpbmi32 = CopyBMI16ToBMI32((PVPVOID)FETCHDWORD(parg16->f5),
  282. (LPBITMAPINFO)&bmi32,
  283. FETCHWORD(parg16->f6));
  284. // see if we need to adjust the image sze for RLE bitmaps
  285. if(lpbmi32 && lpib4 && (DWORD32(parg16->f3) == CBM_INIT)) {
  286. if((lpbmi32->bmiHeader.biCompression == BI_RLE4) ||
  287. (lpbmi32->bmiHeader.biCompression == BI_RLE8)) {
  288. if(lpbmi32->bmiHeader.biSizeImage == 0) {
  289. lpbmi32->bmiHeader.biSizeImage =
  290. Get_RLE_Compression_Size(lpbmi32->bmiHeader.biCompression,
  291. lpib4,
  292. parg16->f4);
  293. }
  294. }
  295. }
  296. ul = GETHBITMAP16(CreateDIBitmap(HDC32(parg16->f1),
  297. lpbmih32,
  298. DWORD32(parg16->f3),
  299. lpib4,
  300. lpbmi32,
  301. WORD32(parg16->f6) ));
  302. }
  303. WOW32APIWARN(ul, "CreateDIBitmap");
  304. FREEMISCPTR(lpib4);
  305. FREEARGPTR(parg16);
  306. return(ul);
  307. }
  308. ULONG FASTCALL WG32CreateEllipticRgnIndirect(PVDMFRAME pFrame)
  309. {
  310. ULONG ul;
  311. RECT t1;
  312. register PCREATEELLIPTICRGNINDIRECT16 parg16;
  313. GETARGPTR(pFrame, sizeof(CREATEELLIPTICRGNINDIRECT16), parg16);
  314. WOW32VERIFY(GETRECT16(parg16->f1, &t1));
  315. ul = GETHRGN16(CreateEllipticRgnIndirect(&t1));
  316. WOW32APIWARN(ul, "CreateEllipticRgnIndirect");
  317. FREEARGPTR(parg16);
  318. RETURN(ul);
  319. }
  320. ULONG FASTCALL WG32CreateIC(PVDMFRAME pFrame)
  321. {
  322. ULONG ul = 0;
  323. PSZ psz1 = NULL;
  324. PSZ psz2 = NULL;
  325. PSZ psz2t = NULL;
  326. PSZ psz3 = NULL;
  327. LPDEVMODE t4 = NULL;
  328. register PCREATEIC16 parg16;
  329. INT len;
  330. PCHAR pch;
  331. CHAR achDevice[256];
  332. DWORD dw;
  333. GETARGPTR(pFrame, sizeof(CREATEIC16), parg16);
  334. if(parg16->f1) {
  335. if(!(psz1 = malloc_w_strcpy_vp16to32(parg16->f1, FALSE, 0)))
  336. goto ExitPath;
  337. }
  338. if(parg16->f2) {
  339. if(!(psz2 = malloc_w_strcpy_vp16to32(parg16->f2, FALSE, 0)))
  340. goto ExitPath;
  341. }
  342. if(parg16->f3) {
  343. if(!(psz3 = malloc_w_strcpy_vp16to32(parg16->f3, FALSE, 0)))
  344. goto ExitPath;
  345. }
  346. if (FETCHDWORD(parg16->f4) != 0L) {
  347. t4 = ThunkDevMode16to32(parg16->f4);
  348. }
  349. else {
  350. t4 = GetDefaultDevMode32(psz2);
  351. }
  352. // invalidate all flat ptrs to 16:16 memory now!
  353. FREEARGPTR(parg16);
  354. psz2t = psz2;
  355. //
  356. // HACK Alert!
  357. //
  358. // NBI's Legacy comes with a pscript.drv that resides in [AppPath]\cbt
  359. // they call CreateIC() specifying the the path to this file as the driver.
  360. // the app GP faults if the CreateIC returns 0. once they've loaded this
  361. // printer driver, on successive calls to CreateIC() they simply use
  362. // "PSCRIPT" as the driver name and use "PostScript Printer", "FILE:" as
  363. // the other parms.
  364. // let's recognize these driver names and try to replace it with the
  365. // system default printer. if there's no printer installed, a GP fault
  366. // is unavoidable. the app appears to use this pscript.drv only during
  367. // the tutorial, so we're not providing life-support for an app that's
  368. // clinically dead.
  369. //
  370. //
  371. // check for a driver name that ends with "PSCRIPT" and if so,
  372. // check that the device name is "PostScript Printer".
  373. // on NT the driver name should always be "winspool", although it's
  374. // completely ignored.
  375. //
  376. // PageMaker 5.0a calls this with ("pscript","Postscript printer","LPT1:",0)
  377. // when opening calibrat.pt5
  378. len = psz1 ? strlen(psz1) : 0;
  379. if (len >= 7) {
  380. #if 0
  381. static CHAR achPS[] = "PostScript Printer";
  382. #endif
  383. if (!WOW32_stricmp(psz1+len-7, "pscript")
  384. #if 0
  385. // let's see who else thinks they're using a pscript driver
  386. //
  387. && (RtlCompareMemory(achPS, psz2, sizeof(achPS)) == sizeof(achPS))
  388. #endif
  389. ) {
  390. LOGDEBUG(LOG_ALWAYS,("WOW32: CreateIC - detected request for Pscript driver\n"));
  391. dw = GetProfileString("windows", "device", "", achDevice,
  392. sizeof(achDevice));
  393. if (dw) {
  394. psz2t = achDevice;
  395. pch = WOW32_strchr(achDevice, ',');
  396. if (pch) {
  397. *pch = '\0';
  398. }
  399. }
  400. }
  401. }
  402. // this can callback into a 16-bit fax driver!
  403. ul = GETHDC16(CreateIC(psz1, psz2t, psz3, t4));
  404. // Note: flat 16:16 ptrs should be considered invalid after this call
  405. FREEARGPTR(parg16);
  406. ExitPath:
  407. if(psz1)
  408. free_w(psz1);
  409. if(psz2)
  410. free_w(psz2);
  411. if(psz3)
  412. free_w(psz3);
  413. WOW32APIWARN(ul, "CreateIC");
  414. FREEDEVMODE32(t4);
  415. RETURN(ul);
  416. }
  417. ULONG FASTCALL WG32CreatePatternBrush(PVDMFRAME pFrame)
  418. {
  419. ULONG ul;
  420. LOGBRUSH logbr;
  421. register PCREATEPATTERNBRUSH16 parg16;
  422. GETARGPTR(pFrame, sizeof(CREATEPATTERNBRUSH16), parg16);
  423. logbr.lbStyle = BS_PATTERN8X8;
  424. logbr.lbColor = 0;
  425. logbr.lbHatch = (LONG)HBITMAP32(parg16->f1);
  426. ul = GETHBRUSH16(CreateBrushIndirect(&logbr));
  427. WOW32APIWARN(ul, "CreatePatternBrush");
  428. FREEARGPTR(parg16);
  429. RETURN(ul);
  430. }
  431. ULONG FASTCALL WG32CreatePenIndirect(PVDMFRAME pFrame)
  432. {
  433. ULONG ul;
  434. LOGPEN t1;
  435. register PCREATEPENINDIRECT16 parg16;
  436. GETARGPTR(pFrame, sizeof(CREATEPENINDIRECT16), parg16);
  437. GETLOGPEN16(parg16->f1, &t1);
  438. ul = GETHPEN16(CreatePenIndirect(&t1));
  439. WOW32APIWARN(ul, "CreatePenIndirect");
  440. FREEARGPTR(parg16);
  441. RETURN(ul);
  442. }
  443. ULONG FASTCALL WG32CreatePolyPolygonRgn(PVDMFRAME pFrame)
  444. {
  445. ULONG ul;
  446. LPPOINT pPoints;
  447. PINT pPolyCnt;
  448. UINT cpts = 0;
  449. INT ii;
  450. register PCREATEPOLYPOLYGONRGN16 parg16;
  451. INT cInt16;
  452. INT BufferT[256]; // comfortably large array
  453. GETARGPTR(pFrame, sizeof(CREATEPOLYPOLYGONRGN16), parg16);
  454. cInt16 = INT32(parg16->f3);
  455. pPolyCnt = STACKORHEAPALLOC(cInt16 * sizeof(INT), sizeof(BufferT), BufferT);
  456. if (!pPolyCnt) {
  457. FREEARGPTR(parg16);
  458. RETURN(0);
  459. }
  460. getintarray16(parg16->f2, cInt16, pPolyCnt);
  461. for (ii=0; ii < cInt16; ii++)
  462. cpts += pPolyCnt[ii];
  463. pPoints = STACKORHEAPALLOC(cpts * sizeof(POINT),
  464. sizeof(BufferT) - cInt16 * sizeof(INT),
  465. BufferT + cInt16);
  466. getpoint16(parg16->f1, cpts, pPoints);
  467. ul = GETHRGN16(CreatePolyPolygonRgn(pPoints,
  468. pPolyCnt,
  469. INT32(parg16->f3),
  470. INT32(parg16->f4)));
  471. WOW32APIWARN(ul, "CreatePolyPolygonRgn");
  472. STACKORHEAPFREE(pPoints, BufferT + cInt16);
  473. STACKORHEAPFREE(pPolyCnt, BufferT);
  474. FREEARGPTR(parg16);
  475. RETURN(ul);
  476. }
  477. ULONG FASTCALL WG32CreatePolygonRgn(PVDMFRAME pFrame)
  478. {
  479. ULONG ul;
  480. LPPOINT t1;
  481. register PCREATEPOLYGONRGN16 parg16;
  482. POINT BufferT[128];
  483. GETARGPTR(pFrame, sizeof(CREATEPOLYGONRGN16), parg16);
  484. t1 = STACKORHEAPALLOC(parg16->f2 * sizeof(POINT), sizeof(BufferT), BufferT);
  485. getpoint16(parg16->f1, parg16->f2, t1);
  486. ul = GETHRGN16(CreatePolygonRgn(t1, INT32(parg16->f2), INT32(parg16->f3)));
  487. WOW32APIWARN(ul, "CreatePolygonRgn");
  488. STACKORHEAPFREE(t1, BufferT);
  489. FREEARGPTR(parg16);
  490. RETURN(ul);
  491. }
  492. ULONG FASTCALL WG32CreateRectRgnIndirect(PVDMFRAME pFrame)
  493. {
  494. ULONG ul;
  495. RECT t1;
  496. register PCREATERECTRGNINDIRECT16 parg16;
  497. GETARGPTR(pFrame, sizeof(CREATERECTRGNINDIRECT16), parg16);
  498. WOW32VERIFY(GETRECT16(parg16->f1, &t1));
  499. ul = GETHRGN16(CreateRectRgnIndirect(&t1));
  500. WOW32APIWARN(ul, "CreateRectRgnIndirect");
  501. FREEARGPTR(parg16);
  502. RETURN(ul);
  503. }
  504. ULONG FASTCALL WG32DPtoLP(PVDMFRAME pFrame)
  505. {
  506. ULONG ul=0;
  507. LPPOINT t2;
  508. register PDPTOLP16 parg16;
  509. POINT BufferT[128];
  510. GETARGPTR(pFrame, sizeof(DPTOLP16), parg16);
  511. t2 = STACKORHEAPALLOC(parg16->f3 * sizeof(POINT), sizeof(BufferT), BufferT);
  512. if (t2) {
  513. getpoint16(parg16->f2, parg16->f3, t2);
  514. ul = GETBOOL16(DPtoLP(HDC32(parg16->f1), t2, INT32(parg16->f3)));
  515. PUTPOINTARRAY16(parg16->f2, parg16->f3, t2);
  516. STACKORHEAPFREE(t2, BufferT);
  517. }
  518. FREEARGPTR(parg16);
  519. RETURN(ul);
  520. }
  521. ULONG FASTCALL WG32DeleteDC(PVDMFRAME pFrame)
  522. {
  523. ULONG ul;
  524. register PDELETEDC16 parg16;
  525. GETARGPTR(pFrame, sizeof(DELETEDC16), parg16);
  526. if ((ul = W32CheckAndFreeDibInfo (HDC32(parg16->f1))) == FALSE) {
  527. ul = GETBOOL16(DeleteDC(HDC32(parg16->f1)));
  528. }
  529. FREEARGPTR(parg16);
  530. RETURN(ul);
  531. }
  532. ULONG FASTCALL WG32DeleteObject(PVDMFRAME pFrame)
  533. {
  534. ULONG ul;
  535. register PDELETEOBJECT16 parg16;
  536. HGDIOBJ hGdiObj;
  537. GETARGPTR(pFrame, sizeof(DELETEOBJECT16), parg16);
  538. if ((pDibSectionInfoHead == NULL) ||
  539. (ul = W32CheckAndFreeDibSectionInfo (HBITMAP32(parg16->f1))) == FALSE) {
  540. hGdiObj = HOBJ32(parg16->f1);
  541. ul = GETBOOL16(DeleteObject(hGdiObj));
  542. if (!ul) {
  543. //
  544. // Most apps probably don't care what the return value from
  545. // DeleteObject is, but in any case, in this regard, NT has
  546. // different logic from win31/win95. For example, it appears
  547. // that win95 always returns TRUE when a palette object is
  548. // passed to DeleteObject, even if the palette was not deleted
  549. // (because it was already selected).
  550. //
  551. // Here we try to decide if we should change the return value
  552. // from FALSE to TRUE.
  553. //
  554. // ChessMaster 3000 tries to delete a Palette object that
  555. // is still selected. If it fails, then it puts up a popup.
  556. //
  557. switch(GetObjectType(hGdiObj)) {
  558. case OBJ_PAL:
  559. case OBJ_PEN:
  560. case OBJ_BRUSH:
  561. ul = TRUE;
  562. break;
  563. }
  564. }
  565. }
  566. FREEARGPTR(parg16);
  567. RETURN(ul);
  568. }
  569. // WARNING: This function may cause 16-bit memory movement
  570. INT W32EnumObjFunc(LPSTR lpLogObject, PENUMOBJDATA pEnumObjData)
  571. {
  572. PARM16 Parm16;
  573. INT iReturn;
  574. WOW32ASSERT(pEnumObjData);
  575. switch (pEnumObjData->ObjType) {
  576. case OBJ_BRUSH:
  577. PUTLOGBRUSH16(pEnumObjData->vpObjData, sizeof(LOGBRUSH), (LPLOGBRUSH)lpLogObject);
  578. break;
  579. case OBJ_PEN:
  580. PUTLOGPEN16(pEnumObjData->vpObjData, sizeof(LOGPEN), (LPLOGPEN)lpLogObject);
  581. break;
  582. default:
  583. LOGDEBUG(LOG_ALWAYS,("WOW32 ERROR -- Illegal type %d passes to EnumObj\n",pEnumObjData->ObjType));
  584. return 0;
  585. } // end switch
  586. STOREDWORD(Parm16.EnumObjProc.vpLogObject, pEnumObjData->vpObjData);
  587. STOREDWORD(Parm16.EnumObjProc.vpData, pEnumObjData->dwUserParam);
  588. CallBack16(RET_ENUMOBJPROC, &Parm16, pEnumObjData->vpfnEnumObjProc, (PVPVOID)&iReturn);
  589. return (SHORT)iReturn;
  590. }
  591. ULONG FASTCALL WG32EnumObjects(PVDMFRAME pFrame)
  592. {
  593. ULONG ul = 0;
  594. register PENUMOBJECTS16 parg16;
  595. ENUMOBJDATA EnumObjData;
  596. GETARGPTR(pFrame, sizeof(ENUMOBJECTS16), parg16);
  597. EnumObjData.ObjType = INT32(parg16->f2);
  598. switch(EnumObjData.ObjType) {
  599. case OBJ_BRUSH:
  600. EnumObjData.vpObjData = malloc16(sizeof(LOGBRUSH16));
  601. break;
  602. case OBJ_PEN:
  603. EnumObjData.vpObjData = malloc16(sizeof(LOGPEN16));
  604. break;
  605. default:
  606. LOGDEBUG(LOG_ALWAYS,("WOW32 ERROR -- Illegal type %d passes to EnumObj\n",EnumObjData.ObjType));
  607. EnumObjData.vpObjData = (VPVOID)0;
  608. }
  609. // malloc16 may have caused 16-bit memory movement - invalidate flat ptrs
  610. FREEVDMPTR(pFrame);
  611. FREEARGPTR(parg16);
  612. GETFRAMEPTR(((PTD)CURRENTPTD())->vpStack, pFrame);
  613. GETARGPTR(pFrame, sizeof(ENUMOBJECTS16), parg16);
  614. if( EnumObjData.vpObjData ) {
  615. EnumObjData.vpfnEnumObjProc = DWORD32(parg16->f3);
  616. EnumObjData.dwUserParam = DWORD32(parg16->f4);
  617. ul = (ULONG)(GETINT16(EnumObjects(HDC32(parg16->f1),
  618. (int)INT32(parg16->f2),
  619. (GOBJENUMPROC)W32EnumObjFunc,
  620. (LPARAM)&EnumObjData)));
  621. // 16-bit memory may have moved - invalidate flat pointers
  622. FREEARGPTR(parg16);
  623. FREEVDMPTR(pFrame);
  624. free16(EnumObjData.vpObjData);
  625. }
  626. FREEARGPTR(parg16);
  627. RETURN(ul);
  628. }
  629. /**************************************************************************\
  630. *
  631. \**************************************************************************/
  632. typedef struct _ESCKERNPAIR {
  633. union
  634. {
  635. BYTE each[2];
  636. WORD both;
  637. } kpPair;
  638. SHORT KernAmount;
  639. } ESCKERNPAIR;
  640. /******************************Public*Routine******************************\
  641. * iGetKerningPairsEsc32
  642. *
  643. * History:
  644. * Tue 16-Mar-1993 11:08:36 by Kirk Olynyk [kirko]
  645. * Wrote it.
  646. \**************************************************************************/
  647. int
  648. iGetKerningPairsEsc32(
  649. HDC hdc,
  650. ESCKERNPAIR *pekp
  651. )
  652. {
  653. int i,j;
  654. int n;
  655. ESCKERNPAIR *pekpT, *pekpTOut;
  656. KERNINGPAIR *pkpT;
  657. KERNINGPAIR *pkp = (KERNINGPAIR*) NULL;
  658. if
  659. (
  660. (n = GetKerningPairs(hdc,0,NULL)) &&
  661. (pkp = malloc_w((n * sizeof(KERNINGPAIR)))) &&
  662. (n = (int) GetKerningPairs(hdc, n, pkp))
  663. )
  664. {
  665. n = min(n,512);
  666. //
  667. // load the low byte of each word, Win 3.1 doesn't seem to care about
  668. // the high byte
  669. //
  670. pekpT = pekp;
  671. pekpTOut = pekp + n;
  672. pkpT = pkp;
  673. while (pekpT < pekpTOut)
  674. {
  675. pekpT->kpPair.each[0] = (BYTE) pkpT->wFirst;
  676. pekpT->kpPair.each[1] = (BYTE) pkpT->wSecond;
  677. pekpT->KernAmount = (SHORT) pkpT->iKernAmount;
  678. pekpT += 1;
  679. pkpT += 1;
  680. }
  681. //
  682. // bubble sort word formed by byte pair
  683. //
  684. for (i = 0; i < n - 1; i++)
  685. {
  686. for (j = n-1; j > i; --j)
  687. {
  688. if (pekp[j-1].kpPair.both > pekp[j].kpPair.both)
  689. {
  690. ESCKERNPAIR ekp;
  691. ekp = pekp[j];
  692. pekp[j] = pekp[j-1];
  693. pekp[j] = ekp;
  694. }
  695. }
  696. }
  697. }
  698. if (pkp)
  699. {
  700. free_w(pkp);
  701. }
  702. return(n);
  703. }
  704. ULONG FASTCALL WG32Escape(PVDMFRAME pFrame)
  705. {
  706. ULONG ul=0;
  707. register PESCAPE16 parg16;
  708. PVOID pin = NULL;
  709. int iMapMode;
  710. CHAR buf[32];
  711. GETARGPTR(pFrame, sizeof(ESCAPE16), parg16);
  712. GETOPTPTR(parg16->f4, 0, pin);
  713. switch (INT32(parg16->f2)) {
  714. case GETPHYSPAGESIZE:
  715. case GETPRINTINGOFFSET:
  716. case GETSCALINGFACTOR:
  717. { POINT pt;
  718. ul = GETINT16(Escape(HDC32(parg16->f1),
  719. INT32(parg16->f2),
  720. INT32(parg16->f3),
  721. pin,
  722. (LPSTR)&pt));
  723. if (!ul)
  724. {
  725. // if these fail, they are almost certainly doing it on a display dc.
  726. // We will fill the return value in with reasonable values for those
  727. // apps (micrographx draw) that ignore our return values.
  728. // we still return failure.
  729. switch (INT32(parg16->f2))
  730. {
  731. case GETPHYSPAGESIZE:
  732. pt.x = GetDeviceCaps(HDC32(parg16->f1),HORZRES);
  733. pt.y = GetDeviceCaps(HDC32(parg16->f1),VERTRES);
  734. break;
  735. case GETPRINTINGOFFSET:
  736. pt.x = 0;
  737. pt.y = 0;
  738. break;
  739. default:
  740. break;
  741. }
  742. }
  743. PUTPOINT16(parg16->f5, &pt);
  744. }
  745. break;
  746. case GETCOLORTABLE:
  747. {
  748. PDWORD pdw;
  749. DWORD dw;
  750. INT i;
  751. if (pin) {
  752. i = (INT) FETCHSHORT(*(PSHORT)pin);
  753. } else {
  754. ul = (ULONG)-1;
  755. break;
  756. }
  757. ul = GETINT16(Escape(HDC32(parg16->f1),
  758. INT32(parg16->f2),
  759. sizeof(INT),
  760. (LPCSTR)&i,
  761. &dw));
  762. GETVDMPTR(parg16->f5, sizeof(DWORD), pdw);
  763. STOREDWORD ((*pdw), dw);
  764. FREEVDMPTR(pdw);
  765. }
  766. break;
  767. case NEXTBAND:
  768. { RECT rt;
  769. ul = GETINT16(Escape(HDC32(parg16->f1),
  770. INT32(parg16->f2),
  771. INT32(parg16->f3),
  772. pin,
  773. (LPSTR)&rt));
  774. PUTRECT16(parg16->f5, &rt);
  775. }
  776. break;
  777. case QUERYESCSUPPORT:
  778. {
  779. INT i;
  780. i = (INT) FETCHWORD((*(PWORD)pin));
  781. switch (i) {
  782. // For Escapes return FALSE for MGX Draw and TRUE for all other apps.
  783. // ChandanC, 27/5/93.
  784. //
  785. case OPENCHANNEL: // 4110
  786. case DOWNLOADHEADER: // 4111
  787. case CLOSECHANNEL: // 4112
  788. case SETGDIXFORM: // 4113
  789. case RESETPAGE: // 4114
  790. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  791. ul = 0;
  792. }
  793. else {
  794. ul = 1;
  795. }
  796. break;
  797. case POSTSCRIPT_PASSTHROUGH: // 4115
  798. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  799. ul = 0;
  800. }
  801. else {
  802. LOGDEBUG(3,("Querying support for escape %x\n",i));
  803. ul = GETINT16(Escape(HDC32(parg16->f1),
  804. INT32(parg16->f2),
  805. sizeof(int),
  806. (PVOID)&i,
  807. NULL));
  808. }
  809. break;
  810. case ENCAPSULATED_POSTSCRIPT: // 4116
  811. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  812. ul = 0;
  813. }
  814. else {
  815. LOGDEBUG(3,("Querying support for escape %x\n",i));
  816. ul = GETINT16(DrawEscape (HDC32(parg16->f1),
  817. INT32(parg16->f2),
  818. sizeof(int),
  819. (PVOID)&i));
  820. }
  821. break;
  822. case GETEXTENDEDTEXTMETRICS:
  823. case GETPAIRKERNTABLE:
  824. case FLUSHOUTPUT:
  825. ul = 1;
  826. break;
  827. case SETCOPYCOUNT:
  828. case GETCOLORTABLE:
  829. case GETPHYSPAGESIZE:
  830. case GETPRINTINGOFFSET:
  831. case GETSCALINGFACTOR:
  832. case NEXTBAND:
  833. case SETABORTPROC:
  834. case BEGIN_PATH:
  835. case END_PATH:
  836. case CLIP_TO_PATH:
  837. LOGDEBUG(3,("Querying support for escape %x\n",i));
  838. ul = GETINT16(Escape(HDC32(parg16->f1),
  839. INT32(parg16->f2),
  840. sizeof(int),
  841. (PVOID)&i,
  842. NULL));
  843. break;
  844. case POSTSCRIPT_DATA:
  845. case POSTSCRIPT_IGNORE:
  846. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  847. ul = 0;
  848. }
  849. else {
  850. LOGDEBUG(3,("Querying support for escape %x\n",i));
  851. ul = GETINT16(Escape(HDC32(parg16->f1),
  852. INT32(parg16->f2),
  853. sizeof(int),
  854. (PVOID)&i,
  855. NULL));
  856. }
  857. break;
  858. case GETTECHNOLOGY:
  859. case PASSTHROUGH:
  860. case DOWNLOADFACE:
  861. case GETFACENAME:
  862. case GETDEVICEUNITS:
  863. case EPSPRINTING:
  864. LOGDEBUG(3,("Querying support for escape %x\n",i));
  865. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  866. INT32(parg16->f2),
  867. sizeof(int),
  868. (PVOID)&i,
  869. 0,
  870. NULL));
  871. break;
  872. case DRAWPATTERNRECT:
  873. LOGDEBUG(3,("Querying support for escape %x\n",i));
  874. // some apps just don't do DRAWPATTERNRECT correctly (Excel
  875. // 6.0c, Word 6.0c Access 2.0 #122856). Others can't handle
  876. // the new 32bit DRAWPATTERNRECT struct (AmiPro 3.1
  877. // #107210). We just tell them that it isn't supported and
  878. // force them to figure out the effect on their own.
  879. if(CURRENTPTD()->dwWOWCompatFlagsEx &
  880. WOWCFEX_SAYNO2DRAWPATTERNRECT) {
  881. ul = 0;
  882. break;
  883. }
  884. else {
  885. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  886. INT32(parg16->f2),
  887. sizeof(int),
  888. (PVOID)&i,
  889. 0,
  890. NULL));
  891. break;
  892. }
  893. default:
  894. LOGDEBUG(3,("Querying support for escape %x\n",i));
  895. ul = GETINT16(Escape(HDC32(parg16->f1),
  896. INT32(parg16->f2),
  897. sizeof(int),
  898. (PVOID)&i,
  899. NULL));
  900. break;
  901. }
  902. }
  903. break;
  904. case SETABORTPROC:
  905. ((PTDB)SEGPTR(pFrame->wTDB, 0))->TDB_vpfnAbortProc =
  906. FETCHDWORD(parg16->f4);
  907. ul = GETINT16(Escape(HDC32(parg16->f1),
  908. INT32(parg16->f2),
  909. 0,
  910. (LPCSTR)W32AbortProc,
  911. NULL));
  912. break;
  913. case GETDEVICEUNITS:
  914. {
  915. PVOID pout;
  916. LONG out[4];
  917. ul = GETINT16(Escape(HDC32(parg16->f1),
  918. INT32(parg16->f2),
  919. 0,
  920. NULL,
  921. (LPSTR)out));
  922. if (ul == 1) {
  923. GETOPTPTR(parg16->f5, 0, pout);
  924. RtlCopyMemory(pout, out, sizeof(out));
  925. FREEOPTPTR(pout);
  926. }
  927. }
  928. break;
  929. case GETPAIRKERNTABLE:
  930. {
  931. PVOID pout;
  932. GETOPTPTR(parg16->f5, 0, pout);
  933. ul = GETINT16(iGetKerningPairsEsc32(
  934. HDC32(parg16->f1),
  935. (ESCKERNPAIR*) pout));
  936. FREEOPTPTR(pout);
  937. }
  938. break;
  939. case GETEXTENDEDTEXTMETRICS:
  940. {
  941. PVOID pout;
  942. EXTTEXTMETRIC etm;
  943. if ( (ul = GETINT16(GetETM(HDC32(parg16->f1),
  944. &etm))) != 0 )
  945. {
  946. GETOPTPTR(parg16->f5, 0, pout);
  947. RtlCopyMemory(pout, &etm, sizeof(EXTTEXTMETRIC));
  948. FREEOPTPTR(pout);
  949. }
  950. }
  951. break;
  952. case OPENCHANNEL: // 4110
  953. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  954. ul = 0;
  955. } else {
  956. DOCINFO16 *pout;
  957. DOCINFO DocInfo;
  958. DocInfo.cbSize = sizeof(DocInfo);
  959. DocInfo.lpszDatatype = "RAW";
  960. DocInfo.fwType = 0;
  961. GETOPTPTR(parg16->f5, 0, pout);
  962. if (pout) {
  963. GETOPTPTR(pout->lpszDocName, 0, DocInfo.lpszDocName);
  964. GETOPTPTR(pout->lpszOutput, 0, DocInfo.lpszOutput);
  965. ul = StartDoc(HDC32(parg16->f1), &DocInfo);
  966. FREEOPTPTR(DocInfo.lpszDocName);
  967. FREEOPTPTR(DocInfo.lpszOutput);
  968. } else {
  969. //
  970. // Fifth parameter null, use old (startdoc) format
  971. //
  972. GETOPTPTR(parg16->f4, 0, DocInfo.lpszDocName);
  973. DocInfo.lpszOutput = NULL;
  974. ul = StartDoc(HDC32(parg16->f1), &DocInfo);
  975. FREEOPTPTR(DocInfo.lpszDocName);
  976. }
  977. FREEOPTPTR(pout);
  978. }
  979. break;
  980. case DOWNLOADHEADER: // 4111
  981. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  982. ul = 0;
  983. }
  984. else {
  985. PBYTE pout;
  986. char ach[64];
  987. GETOPTPTR(parg16->f5, 0, pout);
  988. if (pout) {
  989. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  990. INT32(parg16->f2),
  991. 0,
  992. NULL,
  993. sizeof(ach),
  994. ach));
  995. strcpy (pout, ach);
  996. }
  997. else {
  998. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  999. INT32(parg16->f2),
  1000. 0,
  1001. NULL,
  1002. 0,
  1003. NULL));
  1004. }
  1005. FREEOPTPTR(pout);
  1006. }
  1007. break;
  1008. case CLOSECHANNEL: // 4112
  1009. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1010. ul = 0;
  1011. }
  1012. else {
  1013. // send any buffered data streams to the printer before EndDoc
  1014. if(CURRENTPTD()->dwWOWCompatFlagsEx & WOWCFEX_FORMFEEDHACK) {
  1015. SendFormFeedHack(HDC32(parg16->f1));
  1016. }
  1017. ul = EndDoc(HDC32(parg16->f1));
  1018. }
  1019. break;
  1020. // This Escape is defined for PageMaker. It is SETGDIXFORM.
  1021. // ChandanC, 24/5/93.
  1022. //
  1023. case SETGDIXFORM: // 4113
  1024. case RESETPAGE: // 4114
  1025. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1026. ul = 0;
  1027. }
  1028. else {
  1029. ul = 1;
  1030. }
  1031. break;
  1032. case POSTSCRIPT_PASSTHROUGH: // 4115
  1033. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1034. ul = 0;
  1035. }
  1036. else {
  1037. ul = GETINT16(Escape(HDC32(parg16->f1),
  1038. INT32(parg16->f2),
  1039. (FETCHWORD(*(PWORD)pin) + 2),
  1040. (LPCSTR)pin,
  1041. NULL));
  1042. }
  1043. break;
  1044. case ENCAPSULATED_POSTSCRIPT: // 4116
  1045. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1046. ul = 0;
  1047. }
  1048. else {
  1049. DWORD cb;
  1050. PVOID lpInData32 = NULL;
  1051. lpInData32 = pin;
  1052. cb = FETCHDWORD (*(PDWORD) pin);
  1053. if ((DWORD) pin & 3) {
  1054. if (lpInData32 = (PVOID) malloc_w (cb)) {
  1055. RtlCopyMemory(lpInData32, pin, cb);
  1056. }
  1057. else {
  1058. ul = 0;
  1059. break;
  1060. }
  1061. }
  1062. ul = GETINT16(DrawEscape(HDC32(parg16->f1),
  1063. INT32(parg16->f2),
  1064. cb,
  1065. lpInData32));
  1066. if (((DWORD) pin & 3) && (lpInData32)) {
  1067. free_w (lpInData32);
  1068. }
  1069. }
  1070. break;
  1071. case POSTSCRIPT_DATA:
  1072. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1073. ul = 0;
  1074. LOGDEBUG (LOG_ALWAYS, ("MicroGraphax app using POSTSCRIPT_DATA, contact PingW/ChandanC\n"));
  1075. WOW32ASSERT(FALSE);
  1076. }
  1077. else {
  1078. //
  1079. // XPress needs IGNORESTARTPAGE escape.
  1080. // PingW, ChandanC 3/22/94
  1081. //
  1082. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_NEEDIGNORESTARTPAGE) {
  1083. int l;
  1084. char szBuf[40];
  1085. if ((l = ExtEscape(HDC32(parg16->f1),
  1086. GETTECHNOLOGY,
  1087. 0,
  1088. NULL,
  1089. sizeof(szBuf),
  1090. szBuf)) > 0) {
  1091. if (!WOW32_stricmp(szBuf, szPostscript)) {
  1092. l = ExtEscape(HDC32(parg16->f1),
  1093. IGNORESTARTPGAE,
  1094. 0,
  1095. NULL,
  1096. 0,
  1097. NULL);
  1098. }
  1099. }
  1100. }
  1101. ul = GETINT16(Escape(HDC32(parg16->f1),
  1102. INT32(parg16->f2),
  1103. (FETCHWORD(*(PWORD)pin) + 2),
  1104. (LPCSTR)pin,
  1105. NULL));
  1106. }
  1107. break;
  1108. case POSTSCRIPT_IGNORE:
  1109. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_MGX_ESCAPES) {
  1110. ul = 0;
  1111. LOGDEBUG (LOG_ALWAYS, ("MicroGraphax app using POSTSCRIPT_IGNORE, contact PingW/ChandanC\n"));
  1112. WOW32ASSERT(FALSE);
  1113. }
  1114. else {
  1115. ul = GETINT16(Escape(HDC32(parg16->f1),
  1116. INT32(parg16->f2),
  1117. INT32(parg16->f3),
  1118. (LPCSTR)pin,
  1119. NULL));
  1120. }
  1121. break;
  1122. case BEGIN_PATH:
  1123. // Some apps set the empty clip region before doing the path escapes,
  1124. // We need to undo that so the drawing APIs between begin and endpath
  1125. // go through to drivers.
  1126. SelectClipRgn(HDC32(parg16->f1),NULL);
  1127. // fall through to escape call
  1128. case END_PATH:
  1129. #ifdef FE_SB
  1130. // WOWCF_FE_FLW2_PRINTING_PS,
  1131. // Japanese Lotus Freelance printing with PostScript.
  1132. // Between Escape( BEGIN_PATH ) and Escape( END_PATH ),
  1133. // select brush object with WHITE_BRUSH
  1134. if (GetSystemDefaultLangID() == 0x411 &&
  1135. CURRENTPTD()->dwWOWCompatFlagsFE & WOWCF_FE_FLW2_PRINTING_PS ) {
  1136. if( INT32(parg16->f2) == BEGIN_PATH )
  1137. fCmptFLW = TRUE;
  1138. else
  1139. fCmptFLW = FALSE;
  1140. }
  1141. #endif // FE_SB
  1142. // fall through to escape call
  1143. case CLIP_TO_PATH:
  1144. ul = GETINT16(Escape(HDC32(parg16->f1),
  1145. INT32(parg16->f2),
  1146. INT32(parg16->f3),
  1147. (LPCSTR)pin,
  1148. NULL));
  1149. break;
  1150. case PASSTHROUGH:
  1151. // if this is a form feed hack app...
  1152. if(CURRENTPTD()->dwWOWCompatFlagsEx & WOWCFEX_FORMFEEDHACK) {
  1153. ul = HandleFormFeedHack(HDC32(parg16->f1),
  1154. pin,
  1155. FETCHWORD(*(PWORD)pin)); // cb only
  1156. }
  1157. else {
  1158. ul = GETINT16(Escape(HDC32(parg16->f1),
  1159. INT32(parg16->f2),
  1160. (FETCHWORD(*(PWORD)pin) + 2),
  1161. (LPCSTR)pin,
  1162. NULL));
  1163. }
  1164. break;
  1165. case FLUSHOUTPUT:
  1166. ul = TRUE;
  1167. break;
  1168. case DOWNLOADFACE:
  1169. {
  1170. WORD InData;
  1171. PWORD lpInData = NULL;
  1172. // PM5 forgot to set there map mode so we do it.
  1173. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_FORCETWIPSESCAPE)
  1174. iMapMode = SetMapMode(HDC32(parg16->f1),MM_TWIPS);
  1175. if (pin) {
  1176. InData = FETCHWORD(*(PWORD)pin);
  1177. lpInData = &InData;
  1178. }
  1179. ul = GETINT16(Escape(HDC32(parg16->f1),
  1180. INT32(parg16->f2),
  1181. sizeof(USHORT),
  1182. (LPCSTR)lpInData,
  1183. NULL));
  1184. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_FORCETWIPSESCAPE)
  1185. SetMapMode(HDC32(parg16->f1),iMapMode);
  1186. }
  1187. break;
  1188. case GETFACENAME:
  1189. {
  1190. PSZ pout;
  1191. CHAR ach[60];
  1192. // PM5 forgot to set there map mode so we do it.
  1193. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_FORCETWIPSESCAPE)
  1194. iMapMode = SetMapMode(HDC32(parg16->f1),MM_TWIPS);
  1195. GETOPTPTR(parg16->f5, 0, pout);
  1196. // This HACK is for FH4.0 only. If you have any questions
  1197. // talk to PingW or ChandanC.
  1198. // July 21st 1994.
  1199. //
  1200. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_ADD_MSTT) {
  1201. ExtEscape(HDC32(parg16->f1),
  1202. ADD_MSTT,
  1203. 0,
  1204. NULL,
  1205. 60,
  1206. ach);
  1207. }
  1208. // pass in 60 as a magic number. Just copy out the valid string.
  1209. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  1210. INT32(parg16->f2),
  1211. INT32(parg16->f3),
  1212. pin,
  1213. 60,
  1214. ach));
  1215. #ifdef FE_SB
  1216. ach[59] = '\0';
  1217. #endif // !FE_SB
  1218. strcpy (pout, ach);
  1219. FREEOPTPTR(pout);
  1220. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_FORCETWIPSESCAPE)
  1221. SetMapMode(HDC32(parg16->f1),iMapMode);
  1222. }
  1223. break;
  1224. case EPSPRINTING:
  1225. {
  1226. WORD InData;
  1227. PWORD lpInData = NULL;
  1228. if (pin)
  1229. {
  1230. InData = FETCHWORD(*(PWORD)pin);
  1231. lpInData = &InData;
  1232. }
  1233. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  1234. INT32(parg16->f2),
  1235. sizeof(BOOL),
  1236. (LPCSTR)lpInData,
  1237. 0,
  1238. NULL));
  1239. }
  1240. break;
  1241. case GETTECHNOLOGY:
  1242. {
  1243. PVOID pout;
  1244. buf[0] = '\0';
  1245. GETOPTPTR(parg16->f5, 0, pout);
  1246. if (!(CURRENTPTD()->dwWOWCompatFlags & WOWCF_FORCENOPOSTSCRIPT))
  1247. {
  1248. ul = GETINT16(ExtEscape(HDC32(parg16->f1),
  1249. INT32(parg16->f2),
  1250. INT32(parg16->f3),
  1251. pin,
  1252. sizeof(buf),
  1253. buf));
  1254. #ifdef FE_SB
  1255. // #636 When function is not supported,
  1256. // wow may destroy app's stack
  1257. if ((int)ul < 0)
  1258. buf[0] = '\0';
  1259. #endif // FE_SB
  1260. }
  1261. if (pout)
  1262. strcpy (pout, buf);
  1263. FREEOPTPTR(pout);
  1264. }
  1265. break;
  1266. case SETCOPYCOUNT:
  1267. {
  1268. int cCopiesOut, cCopiesIn=1;
  1269. if (pin)
  1270. cCopiesIn = *(UNALIGNED SHORT *)pin;
  1271. ul = GETINT16(Escape(HDC32(parg16->f1),
  1272. INT32(parg16->f2),
  1273. pin ? sizeof(cCopiesIn) : 0,
  1274. pin ? &cCopiesIn : pin,
  1275. parg16->f5 ? &cCopiesOut : NULL));
  1276. if ( parg16->f5 ) {
  1277. if ( (INT)ul > 0 ) {
  1278. PUTINT16(parg16->f5, cCopiesOut);
  1279. } else {
  1280. // Pagemaker v4 needs the output value
  1281. PUTINT16(parg16->f5, 1);
  1282. }
  1283. }
  1284. }
  1285. break;
  1286. case STARTDOC:
  1287. {
  1288. PVOID pout;
  1289. GETOPTPTR(parg16->f5, 0, pout);
  1290. //
  1291. // Win32 StartDoc depends on having the correct current directory
  1292. // when printing to FILE: (which pops up for a filename).
  1293. //
  1294. UpdateDosCurrentDirectory(DIR_DOS_TO_NT);
  1295. ul = GETINT16(Escape(HDC32(parg16->f1),
  1296. INT32(parg16->f2),
  1297. INT32(parg16->f3),
  1298. pin,
  1299. pout));
  1300. //
  1301. // PhotoShop needs a StartPage when it does StartDoc Escape.
  1302. // PingW, ChandanC 3/22/94
  1303. //
  1304. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_NEEDSTARTPAGE) {
  1305. int l;
  1306. char szBuf[80];
  1307. //
  1308. // It should be done ONLY for POSTSCRIPT drivers
  1309. //
  1310. if ((l = ExtEscape(HDC32(parg16->f1),
  1311. GETTECHNOLOGY,
  1312. 0,
  1313. NULL,
  1314. sizeof(szBuf),
  1315. szBuf)) > 0) {
  1316. if (!WOW32_stricmp(szBuf, szPostscript)) {
  1317. l = StartPage(HDC32(parg16->f1));
  1318. }
  1319. }
  1320. }
  1321. FREEOPTPTR(pout);
  1322. }
  1323. break;
  1324. // Win'95 really messed us up here. They changed the DRAWPATTERNRECT
  1325. // structure instead of defining a new DRAWPATTERNRECT32 struct.
  1326. // On NT, all metafiles are 32-bit. So if an app is building a metafile,
  1327. // it will only know how to do 16-bit DRAWPATTERNRECT (DPR). We have
  1328. // to convert it to a 32-bit DPR before passing it to GDI. On the other
  1329. // hand, if an app is reading a metafile (either memory or disk file) it
  1330. // will be 32-bit if it was created by calls to NT's metafile API, or it
  1331. // could be 16-bit if it is something the app created or a file shipped
  1332. // with the app. If the app handles it correctly by using the size
  1333. // field in the metafile record, then we can fix them up correctly.
  1334. case DRAWPATTERNRECT:
  1335. {
  1336. DRAWPATRECT dpr32in;
  1337. if(pin) {
  1338. // if the app specifies that it is a 32-bit sized DPR struct
  1339. if(INT32(parg16->f3) == sizeof(DRAWPATRECT)) {
  1340. dpr32in.ptPosition.x =
  1341. (LONG)FETCHDWORD(((PDRAWPATRECT)pin)->ptPosition.x);
  1342. dpr32in.ptPosition.y =
  1343. (LONG)FETCHDWORD(((PDRAWPATRECT)pin)->ptPosition.y);
  1344. dpr32in.ptSize.x =
  1345. (LONG)FETCHDWORD(((PDRAWPATRECT)pin)->ptSize.x);
  1346. dpr32in.ptSize.y =
  1347. (LONG)FETCHDWORD(((PDRAWPATRECT)pin)->ptSize.y);
  1348. dpr32in.wStyle =
  1349. FETCHWORD(((PDRAWPATRECT)pin)->wStyle);
  1350. dpr32in.wPattern =
  1351. FETCHWORD(((PDRAWPATRECT)pin)->wPattern);
  1352. }
  1353. // else any other size, all we can do is assume that they
  1354. // are passing it as a 16-bit DRAWPATTERNRECT
  1355. else {
  1356. dpr32in.ptPosition.x =
  1357. (LONG)FETCHWORD(((PDPR16)pin)->ptPosition.x);
  1358. dpr32in.ptPosition.y =
  1359. (LONG)FETCHWORD(((PDPR16)pin)->ptPosition.y);
  1360. dpr32in.ptSize.x =
  1361. (LONG)FETCHWORD(((PDPR16)pin)->ptSize.x);
  1362. dpr32in.ptSize.y =
  1363. (LONG)FETCHWORD(((PDPR16)pin)->ptSize.y);
  1364. dpr32in.wStyle = FETCHWORD(((PDPR16)pin)->wStyle);
  1365. dpr32in.wPattern = FETCHWORD(((PDPR16)pin)->wPattern);
  1366. }
  1367. ul = GETINT16(Escape(HDC32(parg16->f1),
  1368. DRAWPATTERNRECT,
  1369. sizeof(DRAWPATRECT),
  1370. (PVOID)&dpr32in,
  1371. NULL));
  1372. }
  1373. }
  1374. break;
  1375. default:
  1376. {
  1377. PVOID pout;
  1378. GETOPTPTR(parg16->f5, 0, pout);
  1379. ul = GETINT16(Escape(HDC32(parg16->f1),
  1380. INT32(parg16->f2),
  1381. INT32(parg16->f3),
  1382. pin,
  1383. pout));
  1384. FREEOPTPTR(pout);
  1385. }
  1386. break;
  1387. } // end switch
  1388. FREEOPTPTR(pin);
  1389. FREEARGPTR(parg16);
  1390. RETURN(ul);
  1391. }
  1392. ULONG FASTCALL WG32GetBitmapDimension(PVDMFRAME pFrame)
  1393. {
  1394. ULONG ul;
  1395. SIZE size2;
  1396. register PGETBITMAPDIMENSION16 parg16;
  1397. GETARGPTR(pFrame, sizeof(GETBITMAPDIMENSION16), parg16);
  1398. ul = 0;
  1399. if (GetBitmapDimensionEx(HBITMAP32(parg16->f1), &size2)) {
  1400. ul = (WORD)size2.cx | (size2.cy << 16);
  1401. }
  1402. else {
  1403. WOW32APIWARN (ul, "GetBitmapDimension");
  1404. }
  1405. FREEARGPTR(parg16);
  1406. RETURN(ul);
  1407. }
  1408. ULONG FASTCALL WG32GetBrushOrg(PVDMFRAME pFrame)
  1409. {
  1410. ULONG ul;
  1411. POINT pt;
  1412. POINT pt2;
  1413. register PGETBRUSHORG16 parg16;
  1414. GETARGPTR(pFrame, sizeof(GETBRUSHORG16), parg16);
  1415. // for windows compatability, we must first add the DCorg
  1416. // since windows brushorg is relative to the screen where as NT
  1417. // is relative to the window. In the future, this should call
  1418. // a private gdi entry point to avoid an extra c/s hit. (erick)
  1419. ul = 0;
  1420. if (GetDCOrgEx(HDC32(parg16->f1),&pt))
  1421. {
  1422. if (GetBrushOrgEx(HDC32(parg16->f1), &pt2)) {
  1423. ul = (WORD)(pt2.x + pt.x) | ((pt2.y + pt.y) << 16);
  1424. }
  1425. }
  1426. FREEARGPTR(parg16);
  1427. RETURN(ul);
  1428. }
  1429. ULONG FASTCALL WG32GetClipBox(PVDMFRAME pFrame)
  1430. {
  1431. ULONG ul;
  1432. RECT t2;
  1433. register PGETCLIPBOX16 parg16;
  1434. HDC hdc;
  1435. GETARGPTR(pFrame, sizeof(GETCLIPBOX16), parg16);
  1436. hdc = HDC32(parg16->f1);
  1437. ul = GETINT16(GetClipBox(hdc,&t2));
  1438. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_SIMPLEREGION) {
  1439. ul = SIMPLEREGION;
  1440. }
  1441. PUTRECT16(parg16->f2, &t2);
  1442. FREEARGPTR(parg16);
  1443. RETURN(ul);
  1444. }
  1445. ULONG FASTCALL WG32GetCurrentPosition(PVDMFRAME pFrame)
  1446. {
  1447. ULONG ul;
  1448. POINT pt;
  1449. register PGETCURRENTPOSITION16 parg16;
  1450. GETARGPTR(pFrame, sizeof(GETCURRENTPOSITION16), parg16);
  1451. ul = 0;
  1452. if (GetCurrentPositionEx(HDC32(parg16->f1), &pt)) {
  1453. ul = (WORD)pt.x | (pt.y << 16);
  1454. }
  1455. FREEARGPTR(parg16);
  1456. RETURN(ul);
  1457. }
  1458. ULONG FASTCALL WG32GetDCOrg(PVDMFRAME pFrame)
  1459. {
  1460. ULONG ul;
  1461. POINT pt;
  1462. register PGETDCORG16 parg16;
  1463. GETARGPTR(pFrame, sizeof(GETDCORG16), parg16);
  1464. ul = 0;
  1465. if ( GetDCOrgEx(HDC32(parg16->f1),&pt) ) {
  1466. ul = (WORD)pt.x | (pt.y << 16);
  1467. }
  1468. LOGDEBUG(6,("GetDCOrg for hdc %x returns %lx\n",parg16->f1,ul));
  1469. FREEARGPTR(parg16);
  1470. RETURN(ul);
  1471. }
  1472. // PowerPoint 2&3 call this 2 times for each slide in the slide previewer
  1473. // the lpvBits == NULL on the first call (presumably to find the size to
  1474. // allocate) -- in which case Win3.x returns 1
  1475. ULONG FASTCALL WG32GetDIBits(PVDMFRAME pFrame)
  1476. {
  1477. INT nbmiSize;
  1478. ULONG ul = 0L;
  1479. PBYTE pb5;
  1480. PBYTE pb6;
  1481. STACKBMI32 bmi32;
  1482. LPBITMAPINFO lpbmi32;
  1483. register PGETDIBITS16 parg16;
  1484. GETARGPTR(pFrame, sizeof(GETDIBITS16), parg16);
  1485. GETMISCPTR(parg16->f5, pb5);
  1486. GETMISCPTR(parg16->f6, pb6);
  1487. // copy just the BITMAPINFOHEADER portion of the BITMAPINFO struct
  1488. if(lpbmi32=(LPBITMAPINFO)CopyBMIH16ToBMIH32((PVPVOID)FETCHDWORD(parg16->f6),
  1489. (LPBITMAPINFOHEADER)&bmi32)) {
  1490. // gdi32 will adjust key fields of the BITMAPINFOHEADER & copy the
  1491. // color table into the 32-bit BITMAPINFO struct
  1492. if( ul = GETINT16(GetDIBits(HDC32(parg16->f1),
  1493. HBITMAP32(parg16->f2),
  1494. WORD32(parg16->f3),
  1495. WORD32(parg16->f4),
  1496. pb5,
  1497. lpbmi32,
  1498. WORD32(parg16->f7))) ) {
  1499. // if lpvBits, then they want the bits of the bitmap too
  1500. if(pb5) {
  1501. ul = WORD32(parg16->f4); // return # scanlines requested
  1502. FLUSHVDMPTR(parg16->f5, SIZE_BOGUS, pb5);
  1503. }
  1504. // else tell app that BITMAPINFO structure filled in only
  1505. else {
  1506. ul = 1L;
  1507. }
  1508. // copy the updated BITMAPINFO struct back into the 16-bit version
  1509. nbmiSize = GetBMI32Size(lpbmi32, WORD32(parg16->f7));
  1510. RtlCopyMemory(pb6, lpbmi32, nbmiSize);
  1511. FLUSHVDMPTR(parg16->f6, nbmiSize, pb6);
  1512. }
  1513. }
  1514. FREEMISCPTR(pb5);
  1515. FREEMISCPTR(pb6);
  1516. FREEARGPTR(parg16);
  1517. RETURN(ul);
  1518. }
  1519. ULONG FASTCALL WG32GetDeviceCaps(PVDMFRAME pFrame)
  1520. {
  1521. ULONG ul;
  1522. register PGETDEVICECAPS16 parg16;
  1523. GETARGPTR(pFrame, sizeof(GETDEVICECAPS16), parg16);
  1524. if (INT32(parg16->f2) == COLORRES) {
  1525. ul = 18;
  1526. }
  1527. else {
  1528. ul = GetDeviceCaps(HDC32(parg16->f1), INT32(parg16->f2));
  1529. if (ul == (ULONG)-1) {
  1530. switch (parg16->f2) {
  1531. case NUMBRUSHES:
  1532. case NUMPENS:
  1533. case NUMCOLORS:
  1534. ul = 2048;
  1535. break;
  1536. default:
  1537. break;
  1538. }
  1539. } else if(ul > 32767) {
  1540. // 16-bit apps can't handle 16M colors in a 16-bit INT
  1541. // most just check if the return is <= 2 to see if they are
  1542. // going to print mono-chrome or color.
  1543. if(parg16->f2 == NUMCOLORS) {
  1544. ul = 32767;
  1545. }
  1546. }
  1547. // if the 4plane conversion flag is set, tell them we are 4 planes 1bpp
  1548. // instead of 1plane 4bpp.
  1549. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_4PLANECONVERSION) {
  1550. if (INT32(parg16->f2) == BITSPIXEL) {
  1551. if ((ul == 4) && (GetDeviceCaps(HDC32(parg16->f1),PLANES) == 1))
  1552. ul = 1;
  1553. }
  1554. else if (INT32(parg16->f2) == PLANES) {
  1555. if ((ul == 1) && (GetDeviceCaps(HDC32(parg16->f1),BITSPIXEL) == 4))
  1556. ul = 4;
  1557. }
  1558. }
  1559. if ( (POLYGONALCAPS == parg16->f2) && (CURRENTPTD()->dwWOWCompatFlags & WOWCF_NOPC_RECTANGLE)) {
  1560. ul &= !PC_RECTANGLE; // Quattro Pro 1.0 for Windows doesn't handle this bit well.
  1561. }
  1562. if ( RASTERCAPS == INT32(parg16->f2) )
  1563. {
  1564. //
  1565. // bjm 10/10/97
  1566. // This is always on in Win31 and Win95 (supporting BITMAPS is
  1567. // pretty much a requirement for drivers) so Win32 killed
  1568. // the bit and will never return it.
  1569. // So, let's make sure it's always on.
  1570. //
  1571. ul |= 0x8000;
  1572. }
  1573. }
  1574. ul = GETINT16(ul);
  1575. FREEARGPTR(parg16);
  1576. RETURN(ul);
  1577. }
  1578. ULONG FASTCALL WG32GetEnvironment(PVDMFRAME pFrame)
  1579. {
  1580. // not a Win32 function
  1581. //
  1582. // if lpEnviron==NULL then the user is querying the size of device
  1583. // data. WinProj doesn't check the return value, calling the driver
  1584. // with an undersized buffer, trashing the global heap.
  1585. // WinFax passes a hard coded 0xA9 == 0x44+0x69 == sizeof(win3.0 DevMode) +
  1586. // known WinFax.DRV->dmDriverExtra. Beware also that WinFax calls this
  1587. // whenever an app calls any API that requires a DevMode.
  1588. INT len;
  1589. ULONG ul=0;
  1590. register PGETENVIRONMENT16 parg16;
  1591. PSZ psz;
  1592. PSZ psz1 = NULL;
  1593. VPDEVMODE31 vpdm2;
  1594. CHAR *pszDriver = NULL;
  1595. UINT cbT = 0;
  1596. WORD nMaxBytes;
  1597. GETARGPTR(pFrame, sizeof(GETENVIRONMENT16), parg16);
  1598. // save off the 16-bit params now since this could callback into a 16-bit
  1599. // fax driver & cause 16-bit memory to move.
  1600. GETPSZPTR(parg16->f1, psz);
  1601. if(psz) {
  1602. len = lstrlen(psz) + 1;
  1603. psz1 = malloc_w(len);
  1604. if(psz1) {
  1605. strcpy(psz1, psz);
  1606. pszDriver = malloc_w(max(len,40));
  1607. }
  1608. }
  1609. FREEPSZPTR(psz);
  1610. vpdm2 = FETCHDWORD(parg16->f2);
  1611. nMaxBytes = FETCHWORD(parg16->f3);
  1612. FREEARGPTR(parg16);
  1613. // invalidate all flat ptrs to 16:16 memory now
  1614. // this implies that psz1 may also be bad
  1615. if(!pszDriver) {
  1616. goto exitpath;
  1617. }
  1618. if (!(spoolerapis[WOW_EXTDEVICEMODE].lpfn)) {
  1619. if (!LoadLibraryAndGetProcAddresses("WINSPOOL.DRV", spoolerapis, WOW_SPOOLERAPI_COUNT)) {
  1620. goto exitpath;
  1621. }
  1622. }
  1623. // get required size for output buffer
  1624. // When WinFax calls this api, calls to GetDriverName for pszDriver ==
  1625. // "FaxModem" seem to fail -- this is a good thing if the app called
  1626. // ExtDeviceMode() because we would get into an infinite loop here. WinFax
  1627. // just fills in a DevMode with default values if this api fails.
  1628. if (GetDriverName(psz1, pszDriver)) {
  1629. ul = (*spoolerapis[WOW_EXTDEVICEMODE].lpfn)(NULL,
  1630. NULL,
  1631. NULL,
  1632. pszDriver,
  1633. psz1,
  1634. NULL,
  1635. NULL,
  1636. 0);
  1637. LOGDEBUG(6,("WOW::GetEnvironment returning ul = %d, for Device = %s, Port = %s \n", ul, pszDriver, psz1));
  1638. // adjust the size for our DEVMODE handling (see note in wstruc.c)
  1639. // (it won't hurt to allocate too much)
  1640. if(ul) {
  1641. ul += sizeof(WOWDM31);
  1642. cbT = (UINT)ul;
  1643. }
  1644. // if they also want us to fill in their environment structure...
  1645. if ((vpdm2 != 0) && (ul != 0)) {
  1646. LPDEVMODE lpdmOutput;
  1647. if (lpdmOutput = malloc_w(ul)) {
  1648. // this might be calling into a 16-bit fax driver!!
  1649. ul = (*spoolerapis[WOW_EXTDEVICEMODE].lpfn)(NULL,
  1650. NULL,
  1651. lpdmOutput,
  1652. pszDriver,
  1653. psz1,
  1654. NULL,
  1655. NULL,
  1656. DM_OUT_BUFFER);
  1657. // if a WinFax call to GetDriverName() succeeds & gets to here
  1658. // we may need to hack on the lpdmOutput->dmSize==0x40
  1659. // (Win3.0 size) to account for the hard coded buffer size of
  1660. // 0xa9 the app passes. So far I haven't seen WinFax get past
  1661. // GetDriverName() call & it still seems to work OK.
  1662. if (ul > 0L) {
  1663. // Use the min of nMaxBytes & what we calculated
  1664. ThunkDevMode32to16(vpdm2, lpdmOutput, min(nMaxBytes, cbT));
  1665. }
  1666. free_w(lpdmOutput);
  1667. LOGDEBUG(6,("WOW::GetEnvironment getting DEVMODE structure, ul = %d, for Device = %s, Port = %s \n", ul, pszDriver, psz1));
  1668. }
  1669. }
  1670. }
  1671. exitpath:
  1672. if(psz1) {
  1673. free_w(psz1);
  1674. }
  1675. if(pszDriver) {
  1676. free_w(pszDriver);
  1677. }
  1678. RETURN(ul);
  1679. }
  1680. ULONG FASTCALL WG32GetObject(PVDMFRAME pFrame)
  1681. {
  1682. ULONG ul;
  1683. HANDLE hgdi;
  1684. register PGETOBJECT16 parg16;
  1685. GETARGPTR(pFrame, sizeof(GETOBJECT16), parg16);
  1686. hgdi = HOBJ32(parg16->f1);
  1687. switch (GetObjectType(hgdi)) {
  1688. case OBJ_BITMAP:
  1689. {
  1690. BITMAP bm;
  1691. ul = GETINT16(GetObject(hgdi, sizeof(BITMAP), (LPSTR)&bm));
  1692. if (ul) {
  1693. PUTBITMAP16(parg16->f3, parg16->f2, &bm);
  1694. if ( ul > sizeof(BITMAP16) ) {
  1695. ul = sizeof(BITMAP16);
  1696. }
  1697. }
  1698. }
  1699. break;
  1700. case OBJ_BRUSH:
  1701. {
  1702. LOGBRUSH lb;
  1703. ul = GETINT16(GetObject(hgdi, sizeof(LOGBRUSH), (LPSTR)&lb));
  1704. if (ul) {
  1705. PUTLOGBRUSH16(parg16->f3, parg16->f2, &lb);
  1706. if (ul > sizeof(LOGBRUSH16)) {
  1707. ul = sizeof(LOGBRUSH16);
  1708. }
  1709. }
  1710. }
  1711. break;
  1712. case OBJ_PEN:
  1713. {
  1714. LOGPEN lp;
  1715. ul = GETINT16(GetObject(hgdi, sizeof(LOGPEN), (LPSTR)&lp));
  1716. if (ul) {
  1717. PUTLOGPEN16(parg16->f3, parg16->f2, &lp);
  1718. if (ul > sizeof(LOGPEN16)) {
  1719. ul = sizeof(LOGPEN16);
  1720. }
  1721. }
  1722. }
  1723. break;
  1724. case OBJ_FONT:
  1725. {
  1726. LOGFONT lf;
  1727. ul = GETINT16(GetObject(hgdi, sizeof(LOGFONT), (LPSTR)&lf));
  1728. if (ul) {
  1729. PUTLOGFONT16(parg16->f3, parg16->f2, &lf);
  1730. if (ul > sizeof(LOGFONT16)) {
  1731. ul = sizeof(LOGFONT16);
  1732. }
  1733. }
  1734. }
  1735. break;
  1736. case OBJ_PAL:
  1737. {
  1738. PSHORT16 lpT;
  1739. SHORT sT;
  1740. ul = GETINT16(GetObject(hgdi, sizeof(SHORT), (LPSTR)&sT));
  1741. if (ul && (FETCHWORD(parg16->f2) >= sizeof(WORD))) {
  1742. GETVDMPTR(FETCHDWORD(parg16->f3), sizeof(WORD), lpT);
  1743. if (lpT) {
  1744. STOREWORD(lpT[0], sT);
  1745. }
  1746. FREEVDMPTR(lpT);
  1747. }
  1748. }
  1749. break;
  1750. default:
  1751. {
  1752. PBYTE pb3;
  1753. LOGDEBUG(LOG_ALWAYS,(" HACK: GetObject handle unknown, contact ChandanC\n"));
  1754. GETVDMPTR(parg16->f3, parg16->f2, pb3);
  1755. ul = GETINT16(GetObject(hgdi, INT32(parg16->f2), pb3));
  1756. FLUSHVDMPTR(parg16->f3, parg16->f2, pb3);
  1757. FREEVDMPTR(pb3);
  1758. }
  1759. } // switch
  1760. WOW32APIWARN(ul, "GetObject");
  1761. FREEARGPTR(parg16);
  1762. RETURN(ul);
  1763. }
  1764. ULONG FASTCALL WG32GetRgnBox(PVDMFRAME pFrame)
  1765. {
  1766. ULONG ul;
  1767. RECT t2;
  1768. register PGETRGNBOX16 parg16;
  1769. GETARGPTR(pFrame, sizeof(GETRGNBOX16), parg16);
  1770. ul = GETINT16(GetRgnBox(HRGN32(parg16->f1), &t2));
  1771. PUTRECT16(parg16->f2, &t2);
  1772. FREEARGPTR(parg16);
  1773. RETURN(ul);
  1774. }
  1775. ULONG FASTCALL WG32GetViewportExt(PVDMFRAME pFrame)
  1776. {
  1777. ULONG ul;
  1778. SIZE size;
  1779. register PGETVIEWPORTEXT16 parg16;
  1780. GETARGPTR(pFrame, sizeof(GETVIEWPORTEXT16), parg16);
  1781. ul = 0;
  1782. if (GetViewportExtEx(HDC32(parg16->f1), &size)) {
  1783. //
  1784. // win31 returns 1 rather than 0 unless there was an error
  1785. //
  1786. if (!(ul = (WORD)size.cx | (size.cy << 16)))
  1787. ul = 1;
  1788. }
  1789. FREEARGPTR(parg16);
  1790. RETURN(ul);
  1791. }
  1792. ULONG FASTCALL WG32GetViewportOrg(PVDMFRAME pFrame)
  1793. {
  1794. ULONG ul;
  1795. POINT pt;
  1796. register PGETVIEWPORTORG16 parg16;
  1797. GETARGPTR(pFrame, sizeof(GETVIEWPORTORG16), parg16);
  1798. ul = 0;
  1799. if (GetViewportOrgEx(HDC32(parg16->f1), &pt)) {
  1800. ul = (WORD)pt.x | (pt.y << 16);
  1801. }
  1802. FREEARGPTR(parg16);
  1803. RETURN(ul);
  1804. }
  1805. ULONG FASTCALL WG32GetWindowExt(PVDMFRAME pFrame)
  1806. {
  1807. ULONG ul;
  1808. SIZE size;
  1809. register PGETWINDOWEXT16 parg16;
  1810. GETARGPTR(pFrame, sizeof(GETWINDOWEXT16), parg16);
  1811. ul = 0;
  1812. if (GetWindowExtEx(HDC32(parg16->f1), &size)) {
  1813. if (!(ul = (WORD)size.cx | (size.cy << 16))) // see above
  1814. ul = 1;
  1815. }
  1816. FREEARGPTR(parg16);
  1817. RETURN(ul);
  1818. }
  1819. ULONG FASTCALL WG32GetWindowOrg(PVDMFRAME pFrame)
  1820. {
  1821. ULONG ul;
  1822. POINT pt;
  1823. register PGETWINDOWORG16 parg16;
  1824. GETARGPTR(pFrame, sizeof(GETWINDOWORG16), parg16);
  1825. ul = 0;
  1826. if (GetWindowOrgEx(HDC32(parg16->f1), &pt)) {
  1827. ul = (WORD)pt.x | (pt.y << 16);
  1828. }
  1829. FREEARGPTR(parg16);
  1830. RETURN(ul);
  1831. }
  1832. ULONG FASTCALL WG32LPtoDP(PVDMFRAME pFrame)
  1833. {
  1834. ULONG ul=0;
  1835. LPPOINT t2;
  1836. register PLPTODP16 parg16;
  1837. POINT BufferT[128];
  1838. GETARGPTR(pFrame, sizeof(LPTODP16), parg16);
  1839. t2 = STACKORHEAPALLOC(parg16->f3 * sizeof(POINT), sizeof(BufferT), BufferT);
  1840. if (t2) {
  1841. getpoint16(parg16->f2, parg16->f3, t2);
  1842. ul = GETBOOL16(LPtoDP(HDC32(parg16->f1), t2, INT32(parg16->f3)));
  1843. PUTPOINTARRAY16(parg16->f2, parg16->f3, t2);
  1844. STACKORHEAPFREE(t2, BufferT);
  1845. }
  1846. FREEARGPTR(parg16);
  1847. RETURN(ul);
  1848. }
  1849. void W32LineDDAFunc(int x, int y, PLINEDDADATA pDDAData)
  1850. {
  1851. PARM16 Parm16;
  1852. WOW32ASSERT(pDDAData);
  1853. Parm16.LineDDAProc.vpData = (VPVOID)pDDAData->dwUserDDAParam;
  1854. Parm16.LineDDAProc.x = (SHORT)x;
  1855. Parm16.LineDDAProc.y = (SHORT)y;
  1856. Parm16.LineDDAProc.vpData = (VPVOID)pDDAData->dwUserDDAParam;
  1857. CallBack16(RET_LINEDDAPROC, &Parm16, pDDAData->vpfnLineDDAProc, NULL);
  1858. return;
  1859. }
  1860. ULONG FASTCALL WG32LineDDA(PVDMFRAME pFrame)
  1861. {
  1862. LINEDDADATA DDAData;
  1863. register PLINEDDA16 parg16;
  1864. GETARGPTR(pFrame, sizeof(LINEDDA16), parg16);
  1865. DDAData.vpfnLineDDAProc = DWORD32(parg16->f5);
  1866. DDAData.dwUserDDAParam = DWORD32(parg16->f6);
  1867. LineDDA(INT32(parg16->f1),
  1868. INT32(parg16->f2),
  1869. INT32(parg16->f3),
  1870. INT32(parg16->f4),
  1871. (LINEDDAPROC)W32LineDDAFunc,
  1872. (LPARAM)&DDAData);
  1873. // 16-bit memory may have moved - invalidate flat pointers now
  1874. FREEVDMPTR(pFrame);
  1875. FREEARGPTR(parg16);
  1876. RETURN(1L);
  1877. }
  1878. ULONG FASTCALL WG32MoveTo(PVDMFRAME pFrame)
  1879. {
  1880. ULONG ul;
  1881. POINT pt;
  1882. register PMOVETO16 parg16;
  1883. GETARGPTR(pFrame, sizeof(MOVETO16), parg16);
  1884. ul = 0;
  1885. pt.x = 1L; // see "METAFILE NOTE"
  1886. pt.y = 0L;
  1887. if (MoveToEx(HDC32(parg16->f1),
  1888. INT32(parg16->f2),
  1889. INT32(parg16->f3),
  1890. &pt)) {
  1891. ul = (WORD)pt.x | (pt.y << 16);
  1892. }
  1893. FREEARGPTR(parg16);
  1894. RETURN(ul);
  1895. }
  1896. ULONG FASTCALL WG32OffsetViewportOrg(PVDMFRAME pFrame)
  1897. {
  1898. ULONG ul;
  1899. POINT pt;
  1900. register POFFSETVIEWPORTORG16 parg16;
  1901. GETARGPTR(pFrame, sizeof(OFFSETVIEWPORTORG16), parg16);
  1902. ul = 0;
  1903. pt.x = 1L; // see "METAFILE NOTE"
  1904. pt.y = 0L;
  1905. if (OffsetViewportOrgEx(HDC32(parg16->f1),
  1906. INT32(parg16->f2),
  1907. INT32(parg16->f3),
  1908. &pt)) {
  1909. ul = (WORD)pt.x | (pt.y << 16);
  1910. }
  1911. FREEARGPTR(parg16);
  1912. RETURN(ul);
  1913. }
  1914. ULONG FASTCALL WG32OffsetWindowOrg(PVDMFRAME pFrame)
  1915. {
  1916. ULONG ul;
  1917. POINT pt;
  1918. register POFFSETWINDOWORG16 parg16;
  1919. GETARGPTR(pFrame, sizeof(OFFSETWINDOWORG16), parg16);
  1920. ul = 0;
  1921. pt.x = 1L; // see "METAFILE NOTE"
  1922. pt.y = 0L;
  1923. if (OffsetWindowOrgEx(HDC32(parg16->f1),
  1924. INT32(parg16->f2),
  1925. INT32(parg16->f3),
  1926. &pt)) {
  1927. ul = (WORD)pt.x | (pt.y << 16);
  1928. }
  1929. FREEARGPTR(parg16);
  1930. RETURN(ul);
  1931. }
  1932. ULONG FASTCALL WG32PolyPolygon(PVDMFRAME pFrame)
  1933. {
  1934. ULONG ul=0;
  1935. LPPOINT pPoints;
  1936. PINT pPolyCnt;
  1937. UINT cpts = 0;
  1938. INT ii;
  1939. register PPOLYPOLYGON16 parg16;
  1940. INT cInt16;
  1941. INT BufferT[256]; // comfortably large array
  1942. GETARGPTR(pFrame, sizeof(POLYPOLYGON16), parg16);
  1943. cInt16 = INT32(parg16->f4);
  1944. pPolyCnt = STACKORHEAPALLOC(cInt16 * sizeof(INT), sizeof(BufferT), BufferT);
  1945. if (!pPolyCnt) {
  1946. FREEARGPTR(parg16);
  1947. RETURN(0);
  1948. }
  1949. getintarray16(parg16->f3, cInt16, pPolyCnt);
  1950. for (ii=0; ii < cInt16; ii++)
  1951. cpts += pPolyCnt[ii];
  1952. pPoints = STACKORHEAPALLOC(cpts * sizeof(POINT),
  1953. sizeof(BufferT) - cInt16 * sizeof(INT),
  1954. BufferT + cInt16);
  1955. if (pPoints) {
  1956. getpoint16(parg16->f2, cpts, pPoints);
  1957. ul = GETBOOL16(PolyPolygon(HDC32(parg16->f1),
  1958. pPoints,
  1959. pPolyCnt,
  1960. INT32(parg16->f4)));
  1961. STACKORHEAPFREE(pPoints, BufferT + cInt16);
  1962. }
  1963. STACKORHEAPFREE(pPolyCnt, BufferT);
  1964. FREEARGPTR(parg16);
  1965. RETURN(ul);
  1966. }
  1967. ULONG FASTCALL WG32PolyPolylineWOW(PVDMFRAME pFrame)
  1968. {
  1969. ULONG ul;
  1970. register PPOLYPOLYLINEWOW16 parg16;
  1971. LPPOINT pptArray;
  1972. LPDWORD pcntArray;
  1973. DWORD cnt;
  1974. GETARGPTR(pFrame, sizeof(POLYPOLYLINEWOW16), parg16);
  1975. cnt = FETCHDWORD(parg16->f4);
  1976. GETVDMPTR(parg16->f2, sizeof(POINT)*cnt, pptArray);
  1977. GETVDMPTR(parg16->f3, sizeof(DWORD)*cnt, pcntArray);
  1978. ul = GETBOOL16(PolyPolyline(HDC32(parg16->f1),
  1979. pptArray,
  1980. pcntArray,
  1981. cnt));
  1982. FREEVDMPTR(pptArray);
  1983. FREEVDMPTR(pcntArray);
  1984. FREEARGPTR(parg16);
  1985. RETURN(ul);
  1986. }
  1987. ULONG FASTCALL WG32Polygon(PVDMFRAME pFrame)
  1988. {
  1989. ULONG ul=0;
  1990. LPPOINT p2;
  1991. register PPOLYGON16 parg16;
  1992. POINT BufferT[128];
  1993. #ifdef FE_SB // for Japanese Lotus Freelance
  1994. HBRUSH hbr = 0;
  1995. #endif // FE_SB
  1996. GETARGPTR(pFrame, sizeof(POLYGON16), parg16);
  1997. p2 = STACKORHEAPALLOC(parg16->f3 * sizeof(POINT), sizeof(BufferT), BufferT);
  1998. if (p2) {
  1999. getpoint16(parg16->f2, parg16->f3, p2);
  2000. #ifdef FE_SB
  2001. // WOWCF_FE_FLW2_PRINTING_PS, Japanese Lotus Freelance
  2002. // printing with PostScript.
  2003. // Between Escape( BEGIN_PATH ) and Escape( END_PATH ),
  2004. // select brush object with WHITE_BRUSH
  2005. if (GetSystemDefaultLangID() == 0x411 && fCmptFLW &&
  2006. CURRENTPTD()->dwWOWCompatFlagsFE & WOWCF_FE_FLW2_PRINTING_PS) {
  2007. hbr = SelectObject( HDC32(parg16->f1), GetStockObject( WHITE_BRUSH ));
  2008. }
  2009. #endif // FE_SB
  2010. ul = GETBOOL16(Polygon(HDC32(parg16->f1), p2, INT32(parg16->f3)));
  2011. #ifdef FE_SB
  2012. if (hbr) {
  2013. SelectObject( HDC32(parg16->f1), hbr );
  2014. }
  2015. #endif // FE_SB
  2016. STACKORHEAPFREE(p2, BufferT);
  2017. }
  2018. FREEARGPTR(parg16);
  2019. RETURN(ul);
  2020. }
  2021. ULONG FASTCALL WG32Polyline(PVDMFRAME pFrame)
  2022. {
  2023. ULONG ul=0;
  2024. PPOINT t2;
  2025. register PPOLYLINE16 parg16;
  2026. POINT BufferT[128];
  2027. GETARGPTR(pFrame, sizeof(POLYLINE16), parg16);
  2028. t2 = STACKORHEAPALLOC(parg16->f3 * sizeof(POINT), sizeof(BufferT), BufferT);
  2029. if (t2) {
  2030. getpoint16(parg16->f2, parg16->f3, t2);
  2031. ul = GETBOOL16(Polyline(HDC32(parg16->f1), t2, INT32(parg16->f3)));
  2032. STACKORHEAPFREE(t2, BufferT);
  2033. }
  2034. FREEARGPTR(parg16);
  2035. RETURN(ul);
  2036. }
  2037. ULONG FASTCALL WG32RectInRegion(PVDMFRAME pFrame)
  2038. {
  2039. ULONG ul;
  2040. RECT t2;
  2041. register PRECTINREGION16 parg16;
  2042. GETARGPTR(pFrame, sizeof(RECTINREGION16), parg16);
  2043. WOW32VERIFY(GETRECT16(parg16->f2, &t2));
  2044. ul = GETBOOL16(RectInRegion(HRGN32(parg16->f1), &t2));
  2045. FREEARGPTR(parg16);
  2046. RETURN(ul);
  2047. }
  2048. ULONG FASTCALL WG32RectVisible(PVDMFRAME pFrame)
  2049. {
  2050. ULONG ul;
  2051. RECT t2;
  2052. register PRECTVISIBLE16 parg16;
  2053. GETARGPTR(pFrame, sizeof(RECTVISIBLE16), parg16);
  2054. WOW32VERIFY(GETRECT16(parg16->f2, &t2));
  2055. ul = GETBOOL16(RectVisible(HDC32(parg16->f1), &t2));
  2056. FREEARGPTR(parg16);
  2057. RETURN(ul);
  2058. }
  2059. ULONG FASTCALL WG32ScaleViewportExt(PVDMFRAME pFrame)
  2060. {
  2061. ULONG ul;
  2062. SIZE size;
  2063. register PSCALEVIEWPORTEXT16 parg16;
  2064. GETARGPTR(pFrame, sizeof(SCALEVIEWPORTEXT16), parg16);
  2065. ul = 0;
  2066. if (ScaleViewportExtEx(HDC32(parg16->f1),
  2067. INT32(parg16->f2),
  2068. INT32(parg16->f3),
  2069. INT32(parg16->f4),
  2070. INT32(parg16->f5),
  2071. &size)) {
  2072. if (!(ul = (WORD)size.cx | (size.cy << 16))) // see above
  2073. ul = 1;
  2074. }
  2075. FREEARGPTR(parg16);
  2076. RETURN(ul);
  2077. }
  2078. ULONG FASTCALL WG32ScaleWindowExt(PVDMFRAME pFrame)
  2079. {
  2080. ULONG ul;
  2081. SIZE size;
  2082. register PSCALEWINDOWEXT16 parg16;
  2083. GETARGPTR(pFrame, sizeof(SCALEWINDOWEXT16), parg16);
  2084. ul = 0;
  2085. if (ScaleWindowExtEx(HDC32(parg16->f1),
  2086. INT32(parg16->f2),
  2087. INT32(parg16->f3),
  2088. INT32(parg16->f4),
  2089. INT32(parg16->f5),
  2090. &size)) {
  2091. if (!(ul = (WORD)size.cx | (size.cy << 16))) // see above
  2092. ul = 1;
  2093. }
  2094. FREEARGPTR(parg16);
  2095. RETURN(ul);
  2096. }
  2097. /******************************Public*Routine******************************\
  2098. * PBYTE pjCvtPlaneToPacked4
  2099. *
  2100. * Convert a 4plane, 1bpp bitmap into a 1plane, 4bpp bitmap.
  2101. * This functions returns a pointer that must later be freed with LocalFree().
  2102. *
  2103. * This has been added for PhotoShop 16 color vga compatability.
  2104. *
  2105. * History:
  2106. * 28-May-1993 -by- Eric Kutter [erick]
  2107. * Wrote it.
  2108. \**************************************************************************/
  2109. PBYTE pjCvtPlaneToPacked4(
  2110. BITMAP *pbm,
  2111. PBYTE pjSrc,
  2112. DWORD *pcjSrc)
  2113. {
  2114. PBYTE pjDstRet;
  2115. PBYTE pjDst;
  2116. PBYTE pjPlane[4]; // pointer to first byte of current scan for each plane
  2117. DWORD cjWidth; // width of the destination in bytes
  2118. DWORD cjSrcWidth; // width of the source scan in bytes inc. all planes
  2119. DWORD cy; // number of scans
  2120. BYTE shift; // shift value,
  2121. DWORD i,x,y;
  2122. // just grab the width of the dest out of the BITMAP
  2123. cjWidth = pbm->bmWidthBytes;
  2124. // the src should be word aligned for each plane with 4 planes
  2125. cjSrcWidth = ((pbm->bmWidth + 15) & ~15) / 8 * 4;
  2126. if (!cjSrcWidth) {
  2127. WOW32ASSERT ( 0 != cjSrcWidth );
  2128. return(NULL);
  2129. }
  2130. // compute the height, the smaller of the bm height and the source height
  2131. cy = min((DWORD)pbm->bmHeight,(DWORD)(*pcjSrc / cjSrcWidth));
  2132. // allocate the new chunk of memory
  2133. *pcjSrc = cy * cjWidth;
  2134. pjDst = LocalAlloc(LMEM_FIXED,*pcjSrc);
  2135. if (pjDst == NULL)
  2136. return(NULL);
  2137. pjDstRet = pjDst;
  2138. // intialize the beginings of the planes
  2139. for (i = 0; i < 4; ++i)
  2140. pjPlane[i] = pjSrc + (cjSrcWidth / 4) * i;
  2141. // loop through the scans
  2142. for (y = 0; y < cy; ++y)
  2143. {
  2144. shift = 7;
  2145. // loop through the bytes within a scan
  2146. for (x = 0; x < cjWidth; ++x)
  2147. {
  2148. // bit 7 -> nibble 1
  2149. // bit 6 -> nibble 0
  2150. // bit 5 -> nibble 3
  2151. // bit 4 -> nibble 2
  2152. // . . .
  2153. *pjDst = (((pjPlane[0][x/4] >> (shift-1)) & 1) << 0 ) | // 0x01
  2154. (((pjPlane[1][x/4] >> (shift-1)) & 1) << 1 ) | // 0x02
  2155. (((pjPlane[2][x/4] >> (shift-1)) & 1) << 2 ) | // 0x04
  2156. (((pjPlane[3][x/4] >> (shift-1)) & 1) << 3 ) | // 0x08
  2157. (((pjPlane[0][x/4] >> (shift-0)) & 1) << 4 ) | // 0x10
  2158. (((pjPlane[1][x/4] >> (shift-0)) & 1) << 5 ) | // 0x20
  2159. (((pjPlane[2][x/4] >> (shift-0)) & 1) << 6 ) | // 0x40
  2160. (((pjPlane[3][x/4] >> (shift-0)) & 1) << 7 ); // 0x80
  2161. pjDst++;
  2162. shift = (shift - 2) & 7;
  2163. }
  2164. pjPlane[0] += cjSrcWidth;
  2165. pjPlane[1] += cjSrcWidth;
  2166. pjPlane[2] += cjSrcWidth;
  2167. pjPlane[3] += cjSrcWidth;
  2168. }
  2169. return(pjDstRet);
  2170. }
  2171. ULONG FASTCALL WG32SetBitmapBits(PVDMFRAME pFrame)
  2172. {
  2173. ULONG ul = 0;
  2174. PBYTE pb3;
  2175. register PSETBITMAPBITS16 parg16;
  2176. HBITMAP hbm;
  2177. DWORD cj;
  2178. BITMAP bm;
  2179. BOOL fValidObj;
  2180. GETARGPTR(pFrame, sizeof(SETBITMAPBITS16), parg16);
  2181. GETOPTPTR(parg16->f3, 0, pb3);
  2182. hbm = HBITMAP32(parg16->f1);
  2183. cj = DWORD32(parg16->f2);
  2184. fValidObj = (GetObject(hbm,sizeof(BITMAP),&bm) == sizeof(BITMAP));
  2185. if (CURRENTPTD()->dwWOWCompatFlags & WOWCF_4PLANECONVERSION) {
  2186. // Get the size of the destination bitmap
  2187. if (fValidObj &&
  2188. (bm.bmPlanes == 1) &&
  2189. (bm.bmBitsPixel == 4))
  2190. {
  2191. PBYTE pjCvt = pjCvtPlaneToPacked4(&bm,pb3,&cj);
  2192. if (pjCvt)
  2193. ul = SetBitmapBits(hbm,cj,pjCvt);
  2194. else
  2195. ul = 0;
  2196. LocalFree(pjCvt);
  2197. hbm = 0;
  2198. }
  2199. }
  2200. else {
  2201. cj = min(cj, (DWORD)(bm.bmWidthBytes * bm.bmHeight));
  2202. }
  2203. if (hbm != 0)
  2204. ul = GETLONG16(SetBitmapBits(hbm,cj,pb3));
  2205. WOW32APIWARN (ul, "SetBitmapBits");
  2206. FREEMISCPTR(pb3);
  2207. FREEARGPTR(parg16);
  2208. RETURN(ul);
  2209. }
  2210. ULONG FASTCALL WG32SetBitmapDimension(PVDMFRAME pFrame)
  2211. {
  2212. ULONG ul;
  2213. SIZE size4;
  2214. register PSETBITMAPDIMENSION16 parg16;
  2215. GETARGPTR(pFrame, sizeof(SETBITMAPDIMENSION16), parg16);
  2216. ul = 0;
  2217. if (SetBitmapDimensionEx(HBITMAP32(parg16->f1),
  2218. INT32(parg16->f2),
  2219. INT32(parg16->f3),
  2220. &size4)) {
  2221. ul = (WORD)size4.cx | (size4.cy << 16);
  2222. }
  2223. FREEARGPTR(parg16);
  2224. RETURN(ul);
  2225. }
  2226. ULONG FASTCALL WG32SetBrushOrg(PVDMFRAME pFrame)
  2227. {
  2228. ULONG ul = 0;
  2229. POINT pt;
  2230. POINT pt2;
  2231. register PSETBRUSHORG16 parg16;
  2232. GETARGPTR(pFrame, sizeof(SETBRUSHORG16), parg16);
  2233. // for windows compatability, we must first subtract off the DCorg
  2234. // since windows brushorg is relative to the screen where as NT
  2235. // is relative to the window. In the future, this should call
  2236. // a private gdi entry point to avoid an extra c/s hit. (erick)
  2237. if (GetDCOrgEx(HDC32(parg16->f1),&pt))
  2238. {
  2239. ul = 0;
  2240. pt2.x = 1L; // see "METAFILE NOTE"
  2241. pt2.y = 0L;
  2242. if (SetBrushOrgEx(HDC32(parg16->f1),
  2243. INT32(parg16->f2) - pt.x,
  2244. INT32(parg16->f3) - pt.y,
  2245. &pt2)) {
  2246. // add the origin back on so the app gets a consistent return value.
  2247. // view...all from micrografx designer doesn't work unless this returns
  2248. // the right thing.
  2249. ul = (WORD)(pt2.x + pt.x) | ((pt2.y + pt.y) << 16);
  2250. }
  2251. }
  2252. FREEARGPTR(parg16);
  2253. RETURN(ul);
  2254. }
  2255. ULONG FASTCALL WG32SetDIBits(PVDMFRAME pFrame)
  2256. {
  2257. ULONG ul = 0L;
  2258. PBYTE pb5;
  2259. STACKBMI32 bmi32;
  2260. LPBITMAPINFO lpbmi32;
  2261. register PSETDIBITS16 parg16;
  2262. GETARGPTR(pFrame, sizeof(SETDIBITS16), parg16);
  2263. GETMISCPTR(parg16->f5, pb5);
  2264. lpbmi32 = CopyBMI16ToBMI32((PVPVOID)FETCHDWORD(parg16->f6),
  2265. (LPBITMAPINFO)&bmi32,
  2266. FETCHWORD(parg16->f7));
  2267. // see if we need to adjust the image sze for RLE bitmaps
  2268. if(lpbmi32 && pb5) {
  2269. if((lpbmi32->bmiHeader.biCompression == BI_RLE4) ||
  2270. (lpbmi32->bmiHeader.biCompression == BI_RLE8)) {
  2271. if(lpbmi32->bmiHeader.biSizeImage == 0) {
  2272. lpbmi32->bmiHeader.biSizeImage =
  2273. Get_RLE_Compression_Size(lpbmi32->bmiHeader.biCompression,
  2274. pb5,
  2275. parg16->f5);
  2276. }
  2277. }
  2278. }
  2279. ul = GETINT16(SetDIBits(HDC32(parg16->f1),
  2280. HBITMAP32(parg16->f2),
  2281. WORD32(parg16->f3),
  2282. WORD32(parg16->f4),
  2283. pb5,
  2284. lpbmi32,
  2285. WORD32(parg16->f7)));
  2286. WOW32APIWARN (ul, "WG32SetDIBits\n");
  2287. FREEMISCPTR(pb5);
  2288. FREEARGPTR(parg16);
  2289. RETURN(ul);
  2290. }
  2291. ULONG FASTCALL WG32SetDIBitsToDevice(PVDMFRAME pFrame)
  2292. {
  2293. ULONG ul = 0L;
  2294. PSZ p10;
  2295. STACKBMI32 bmi32;
  2296. LPBITMAPINFO lpbmi32;
  2297. register PSETDIBITSTODEVICE16 parg16;
  2298. GETARGPTR(pFrame, sizeof(SETDIBITSTODEVICE16), parg16);
  2299. GETMISCPTR(parg16->f10, p10);
  2300. lpbmi32 = CopyBMI16ToBMI32((PVPVOID)FETCHDWORD(parg16->f11),
  2301. (LPBITMAPINFO)&bmi32,
  2302. FETCHWORD(parg16->f12));
  2303. // these are doc'd as WORD in Win3.0, doc'd as INT in Win3.1
  2304. WOW32ASSERTMSG(((INT)parg16->f4 >= 0),("WOW:signed val - CMJones\n"));
  2305. WOW32ASSERTMSG(((INT)parg16->f5 >= 0),("WOW:signed val - CMJones\n"));
  2306. WOW32ASSERTMSG(((INT)parg16->f8 >= 0),("WOW:signed val - CMJones\n"));
  2307. WOW32ASSERTMSG(((INT)parg16->f9 >= 0),("WOW:signed val - CMJones\n"));
  2308. // see if we need to adjust the image sze for RLE bitmaps
  2309. if(lpbmi32 && p10) {
  2310. if((lpbmi32->bmiHeader.biCompression == BI_RLE4) ||
  2311. (lpbmi32->bmiHeader.biCompression == BI_RLE8)) {
  2312. if(lpbmi32->bmiHeader.biSizeImage == 0) {
  2313. lpbmi32->bmiHeader.biSizeImage =
  2314. Get_RLE_Compression_Size(lpbmi32->bmiHeader.biCompression,
  2315. p10,
  2316. parg16->f10);
  2317. }
  2318. }
  2319. }
  2320. ul = GETINT16(SetDIBitsToDevice(HDC32(parg16->f1),
  2321. INT32(parg16->f2),
  2322. INT32(parg16->f3),
  2323. WORD32(parg16->f4),
  2324. WORD32(parg16->f5),
  2325. INT32(parg16->f6),
  2326. INT32(parg16->f7),
  2327. WORD32(parg16->f8),
  2328. WORD32(parg16->f9),
  2329. p10,
  2330. lpbmi32,
  2331. WORD32(parg16->f12)));
  2332. WOW32APIWARN (ul, "WG32SetDIBitsToDevice\n");
  2333. FREEMISCPTR(p10);
  2334. FREEARGPTR(parg16);
  2335. RETURN(ul);
  2336. }
  2337. ULONG FASTCALL WG32SetViewportExt(PVDMFRAME pFrame)
  2338. {
  2339. ULONG ul;
  2340. SIZE size;
  2341. register PSETVIEWPORTEXT16 parg16;
  2342. GETARGPTR(pFrame, sizeof(SETVIEWPORTEXT16), parg16);
  2343. ul = 0;
  2344. if (SetViewportExtEx(HDC32(parg16->f1),
  2345. INT32(parg16->f2),
  2346. INT32(parg16->f3),
  2347. &size)) {
  2348. if (!(ul = (WORD)size.cx | (size.cy << 16))) // see above
  2349. ul = 1;
  2350. }
  2351. FREEARGPTR(parg16);
  2352. RETURN(ul);
  2353. }
  2354. ULONG FASTCALL WG32SetViewportOrg(PVDMFRAME pFrame)
  2355. {
  2356. ULONG ul;
  2357. POINT pt;
  2358. register PSETVIEWPORTORG16 parg16;
  2359. GETARGPTR(pFrame, sizeof(SETVIEWPORTORG16), parg16);
  2360. ul = 0;
  2361. pt.x = 1L; // see "METAFILE NOTE"
  2362. pt.y = 0L;
  2363. if (SetViewportOrgEx(HDC32(parg16->f1),
  2364. INT32(parg16->f2),
  2365. INT32(parg16->f3),
  2366. &pt)) {
  2367. ul = (WORD)pt.x | (pt.y << 16);
  2368. }
  2369. FREEARGPTR(parg16);
  2370. RETURN(ul);
  2371. }
  2372. ULONG FASTCALL WG32SetWindowExt(PVDMFRAME pFrame)
  2373. {
  2374. ULONG ul;
  2375. SIZE size;
  2376. register PSETWINDOWEXT16 parg16;
  2377. GETARGPTR(pFrame, sizeof(SETWINDOWEXT16), parg16);
  2378. ul = 0;
  2379. if (SetWindowExtEx(HDC32(parg16->f1),
  2380. INT32(parg16->f2),
  2381. INT32(parg16->f3),
  2382. &size)) {
  2383. if (!(ul = (WORD)size.cx | (size.cy << 16))) // see above
  2384. ul = 1;
  2385. }
  2386. FREEARGPTR(parg16);
  2387. RETURN(ul);
  2388. }
  2389. ULONG FASTCALL WG32SetWindowOrg(PVDMFRAME pFrame)
  2390. {
  2391. ULONG ul;
  2392. POINT pt;
  2393. register PSETWINDOWORG16 parg16;
  2394. GETARGPTR(pFrame, sizeof(SETWINDOWORG16), parg16);
  2395. ul = 0;
  2396. pt.x = 1L; // see "METAFILE NOTE"
  2397. pt.y = 0L;
  2398. if (SetWindowOrgEx(HDC32(parg16->f1),
  2399. INT32(parg16->f2),
  2400. INT32(parg16->f3),
  2401. &pt)) {
  2402. ul = (WORD)pt.x | (pt.y << 16);
  2403. }
  2404. FREEARGPTR(parg16);
  2405. RETURN(ul);
  2406. }
  2407. ULONG FASTCALL WG32StretchDIBits(PVDMFRAME pFrame)
  2408. {
  2409. ULONG ul = 0L;
  2410. PBYTE pb10;
  2411. STACKBMI32 bmi32;
  2412. LPBITMAPINFO lpbmi32;
  2413. register PSTRETCHDIBITS16 parg16;
  2414. GETARGPTR(pFrame, sizeof(STRETCHDIBITS16), parg16);
  2415. GETMISCPTR(parg16->f10, pb10);
  2416. lpbmi32 = CopyBMI16ToBMI32((PVPVOID)FETCHDWORD(parg16->f11),
  2417. (LPBITMAPINFO)&bmi32,
  2418. FETCHWORD(parg16->f12));
  2419. // see if we need to adjust the image sze for RLE bitmaps
  2420. if(lpbmi32 && pb10) {
  2421. if((lpbmi32->bmiHeader.biCompression == BI_RLE4) ||
  2422. (lpbmi32->bmiHeader.biCompression == BI_RLE8)) {
  2423. if(lpbmi32->bmiHeader.biSizeImage == 0) {
  2424. lpbmi32->bmiHeader.biSizeImage =
  2425. Get_RLE_Compression_Size(lpbmi32->bmiHeader.biCompression,
  2426. pb10,
  2427. parg16->f10);
  2428. }
  2429. }
  2430. }
  2431. ul = GETINT16(StretchDIBits(HDC32(parg16->f1),
  2432. INT32(parg16->f2),
  2433. INT32(parg16->f3),
  2434. INT32(parg16->f4),
  2435. INT32(parg16->f5),
  2436. INT32(parg16->f6),
  2437. INT32(parg16->f7),
  2438. INT32(parg16->f8),
  2439. INT32(parg16->f9),
  2440. pb10,
  2441. lpbmi32,
  2442. (DWORD)FETCHWORD(parg16->f12),
  2443. DWORD32(parg16->f13)));
  2444. WOW32APIWARN (ul, "WG32StretchDIBits\n");
  2445. FREEMISCPTR(pb10);
  2446. FREEARGPTR(parg16);
  2447. RETURN(ul);
  2448. }
  2449. //
  2450. // This routine calls back the apps SetAbortProc routine.
  2451. //
  2452. LONG W32AbortProc(HDC hPr, int code)
  2453. {
  2454. LONG lReturn;
  2455. PARM16 Parm16;
  2456. register PTD ptd;
  2457. DWORD AbortProcT;
  2458. ptd = CURRENTPTD();
  2459. WOW32ASSERT(ptd->htask16);
  2460. AbortProcT = ((PTDB)SEGPTR(ptd->htask16, 0))->TDB_vpfnAbortProc;
  2461. if (AbortProcT) {
  2462. Parm16.SetAbortProc.hPr = GETHDC16(hPr);
  2463. Parm16.SetAbortProc.code = (SHORT) code;
  2464. CallBack16(RET_SETABORTPROC,
  2465. &Parm16,
  2466. AbortProcT,
  2467. (PVPVOID)&lReturn);
  2468. lReturn = (LONG)LOWORD(lReturn); // Returns a BOOL
  2469. }
  2470. else {
  2471. lReturn = (LONG)TRUE;
  2472. }
  2473. return (lReturn);
  2474. }
  2475. // note: cb is the number of data bytes in lpData NOT including the USHORT byte
  2476. // count at the start of the data stream. In other words, lpData contains
  2477. // cb + sizeof(USHORT) bytes.
  2478. LONG HandleFormFeedHack(HDC hdc, LPBYTE lpdata, int cb)
  2479. {
  2480. int cbBytes;
  2481. LONG ul;
  2482. PFORMFEEDHACK pCur;
  2483. // look for a node with a pointer to a data stream buffer from the previous
  2484. // call to Escape(,,PASSTHROUGH,,)...
  2485. pCur = FindFormFeedHackNode(hdc);
  2486. // if we found one, it's time to send the data stream to the printer...
  2487. if(pCur) {
  2488. // ...time to send it to the printer
  2489. ul = GETINT16(Escape(hdc,
  2490. PASSTHROUGH,
  2491. pCur->cbBytes + sizeof(USHORT),
  2492. pCur->lpBytes,
  2493. NULL));
  2494. // free the current node
  2495. FreeFormFeedHackNode(pCur);
  2496. // if there was a problem we're done
  2497. if(ul <= 0) {
  2498. return(ul);
  2499. }
  2500. }
  2501. // send everything up to the last form feed in the new data stream
  2502. cbBytes = cb;
  2503. lpdata = SendFrontEndOfDataStream(hdc, lpdata, &cbBytes, &ul);
  2504. // if there was a problem
  2505. // OR if the entire data stream got sent since it didn't contain a formfeed
  2506. // -- we're done
  2507. if(lpdata == NULL) {
  2508. return(ul); // this will contain error code OR number of bytes sent
  2509. }
  2510. // else create a node for this data stream
  2511. else {
  2512. pCur = CreateFormFeedHackNode(hdc, cbBytes, lpdata);
  2513. // if we can't allocate a new node...
  2514. if(pCur == NULL) {
  2515. // Things are in pretty bad shape if we get to here...
  2516. // We need to write the byte count at the front of the data stream.
  2517. // Remember lpdata had a word size byte count at the front of it
  2518. // when it was sent to us.
  2519. // if any bytes got sent via SendFrontEndOfDataStream()...
  2520. if(cbBytes < cb) {
  2521. // ...first we need to word align this for Escape32()...
  2522. if((DWORD)lpdata & 0x00000001) {
  2523. lpdata--;
  2524. *lpdata = '\0'; // stick a harmless char in the stream
  2525. cbBytes++; // ...and account for it
  2526. }
  2527. // ...adjust the data stream ptr to accomodate the byte count...
  2528. lpdata -= sizeof(USHORT);
  2529. }
  2530. // ...write in the byte count...
  2531. *(UNALIGNED USHORT *)lpdata = (USHORT)cbBytes;
  2532. // ...and send the remainder of the data stream to the printer.
  2533. // If an extra page gets sent to the printer, too bad.
  2534. ul = GETINT16(Escape(hdc,
  2535. PASSTHROUGH,
  2536. cbBytes + sizeof(USHORT),
  2537. lpdata,
  2538. NULL));
  2539. // if the was an error, return it to the app
  2540. if(ul <= 0) {
  2541. return(ul);
  2542. }
  2543. // else we managed to get everything sent to the printer OK
  2544. else {
  2545. return(cb); // return the number of bytes the app sent
  2546. }
  2547. }
  2548. }
  2549. // return the number of bytes the app requested to send
  2550. return(cb);
  2551. }
  2552. LPBYTE SendFrontEndOfDataStream(HDC hdc, LPBYTE lpData, int *cb, LONG *ul)
  2553. {
  2554. int diff;
  2555. LPBYTE lpByte, lpStart;
  2556. // if there's no data or a bad cb, just send it so we can get the error code
  2557. if((lpData == NULL) || (*cb <= 0)) {
  2558. *ul = GETINT16(Escape(hdc,
  2559. PASSTHROUGH,
  2560. *cb + sizeof(USHORT),
  2561. lpData,
  2562. NULL));
  2563. return(NULL);
  2564. }
  2565. // find the start of the actual data after the byte count
  2566. lpStart = lpData + sizeof(USHORT);
  2567. // look for a formfeed char at or near the end of the data stream
  2568. lpByte = lpStart + ((*cb - 1) * sizeof(BYTE));
  2569. while(lpByte >= lpStart) {
  2570. // if we have found the odious formfeed char....
  2571. if((UCHAR)(*lpByte) == 0x0c) {
  2572. diff = lpByte - lpStart;
  2573. // send everything in the stream up to (but not incl) the formfeed
  2574. if(diff) {
  2575. // adjust the byte count in the data stream
  2576. *(UNALIGNED USHORT *)lpData = (USHORT)diff;
  2577. // send it to the printer
  2578. *ul = GETINT16(Escape(hdc,
  2579. PASSTHROUGH,
  2580. diff + sizeof(USHORT),
  2581. lpData,
  2582. NULL));
  2583. // if there was a problem, return it to the app
  2584. if(*ul <= 0) {
  2585. return(NULL);
  2586. }
  2587. }
  2588. // else formfeed is the first char in the data stream
  2589. else {
  2590. *ul = *cb; // just lie and say we sent it all
  2591. }
  2592. // adjust the remaining number of bytes
  2593. *cb -= diff;
  2594. // return ptr to the formfeed char as new start of data stream
  2595. return(lpByte);
  2596. }
  2597. lpByte--;
  2598. }
  2599. // if there are no formfeed's in the data stream just send the whole thing
  2600. *ul = GETINT16(Escape(hdc,
  2601. PASSTHROUGH,
  2602. *cb + sizeof(USHORT),
  2603. lpData,
  2604. NULL));
  2605. return(NULL); // specify we sent the whole thing
  2606. }
  2607. // note: this assumes that if there is a node, there is a list
  2608. void FreeFormFeedHackNode(PFORMFEEDHACK pNode)
  2609. {
  2610. PFORMFEEDHACK pCur, pPrev, pListStart;
  2611. pPrev = NULL;
  2612. pCur = pListStart = gpFormFeedHackList;
  2613. // if there is a node, there must be a node list
  2614. WOW32ASSERT(pCur);
  2615. if(pNode) {
  2616. while(pCur) {
  2617. if(pCur == pNode) {
  2618. if(pNode->lpBytes) {
  2619. free_w(pNode->lpBytes);
  2620. }
  2621. if(pPrev) {
  2622. pPrev->next = pCur->next;
  2623. }
  2624. else {
  2625. pListStart = pCur->next;
  2626. }
  2627. free_w(pNode);
  2628. break;
  2629. }
  2630. else {
  2631. pPrev = pCur;
  2632. pCur = pCur->next;
  2633. }
  2634. }
  2635. }
  2636. gpFormFeedHackList = pListStart;
  2637. }
  2638. void FreeTaskFormFeedHacks(HAND16 h16)
  2639. {
  2640. PFORMFEEDHACK pNext, pCur;
  2641. pCur = gpFormFeedHackList;
  2642. while(pCur) {
  2643. if(pCur->hTask16 == h16) {
  2644. // we already told the app we sent this so give it one last try
  2645. Escape(pCur->hdc,
  2646. PASSTHROUGH,
  2647. pCur->cbBytes + sizeof(USHORT),
  2648. pCur->lpBytes,
  2649. NULL);
  2650. pNext = pCur->next;
  2651. if(pCur->lpBytes) {
  2652. free_w(pCur->lpBytes);
  2653. }
  2654. if(pCur == gpFormFeedHackList) {
  2655. gpFormFeedHackList = pNext;
  2656. }
  2657. free_w(pCur);
  2658. pCur = pNext;
  2659. }
  2660. }
  2661. }
  2662. // this should only be called by Escape(,,ENDDOC,,)
  2663. void SendFormFeedHack(HDC hdc)
  2664. {
  2665. int cb;
  2666. LPBYTE pBytes = NULL;
  2667. PFORMFEEDHACK pCur;
  2668. pCur = gpFormFeedHackList;
  2669. while(pCur) {
  2670. if(pCur->hdc == hdc) {
  2671. if(pCur->lpBytes) {
  2672. cb = pCur->cbBytes;
  2673. // point to actual data after byte count
  2674. pBytes = pCur->lpBytes + sizeof(USHORT);
  2675. // strip the form feed from the buffered data stream...
  2676. if((UCHAR)(*pBytes) == 0x0c) {
  2677. *pBytes = '\0';
  2678. pBytes++;
  2679. cb--;
  2680. }
  2681. // strip the carriage ret from the buffered data stream...
  2682. // (some apps put a carriage return after the last formfeed)
  2683. if((UCHAR)(*pBytes) == 0x0d) {
  2684. *pBytes = '\0';
  2685. cb--;
  2686. }
  2687. // ...and send it to the printer
  2688. if(cb > 0) {
  2689. Escape(hdc,
  2690. PASSTHROUGH,
  2691. cb + sizeof(USHORT),
  2692. pCur->lpBytes,
  2693. NULL);
  2694. }
  2695. }
  2696. // free this node from the hack list now
  2697. FreeFormFeedHackNode(pCur);
  2698. break;
  2699. }
  2700. pCur = pCur->next;
  2701. }
  2702. }
  2703. PFORMFEEDHACK FindFormFeedHackNode(HDC hdc)
  2704. {
  2705. PFORMFEEDHACK pCur;
  2706. pCur = gpFormFeedHackList;
  2707. while(pCur) {
  2708. if(pCur->hdc == hdc) {
  2709. return(pCur);
  2710. }
  2711. pCur = pCur->next;
  2712. }
  2713. return(NULL);
  2714. }
  2715. // this will only get called if PART of the data stream got sent to the printer
  2716. PFORMFEEDHACK CreateFormFeedHackNode(HDC hdc, int cb, LPBYTE lpData)
  2717. {
  2718. LPBYTE pBytes;
  2719. PFORMFEEDHACK pNode;
  2720. // allocate a new node
  2721. pNode = malloc_w(sizeof(FORMFEEDHACK));
  2722. // if we were able to get one...
  2723. if(pNode) {
  2724. // ...allocate a buffer for the data stream
  2725. pBytes = malloc_w(cb + sizeof(USHORT));
  2726. // if we were able to get one...
  2727. if(pBytes) {
  2728. // ...fill in the node...
  2729. pNode->hdc = hdc;
  2730. pNode->lpBytes = pBytes;
  2731. pNode->cbBytes = cb;
  2732. pNode->hTask16 = CURRENTPTD()->htask16;
  2733. // ...and stick the new node at the front of the node list
  2734. pNode->next = gpFormFeedHackList;
  2735. gpFormFeedHackList = pNode;
  2736. // add the new size to the front of the data stream
  2737. *(UNALIGNED USHORT *)pBytes = (USHORT)cb;
  2738. pBytes += sizeof(USHORT);
  2739. // copy the the data stream into the node buffer
  2740. RtlCopyMemory(pBytes, lpData, cb);
  2741. return(pNode);
  2742. }
  2743. // else if we couldn't get a data stream buffer...
  2744. else {
  2745. free_w(pNode);
  2746. }
  2747. }
  2748. return(NULL); // return NULL if either allocate failed
  2749. }
  2750. // this should only be called by Escape(,,AbortDOC,,) and AbortDoc()
  2751. void RemoveFormFeedHack(HDC hdc)
  2752. {
  2753. PFORMFEEDHACK pNode;
  2754. pNode = FindFormFeedHackNode(hdc);
  2755. if(pNode) {
  2756. FreeFormFeedHackNode(pNode);
  2757. }
  2758. }
  2759. #ifdef FE_SB //GetFontAssocStatus, pisuih, 10/5/94'
  2760. int GetFontAssocStatus(HDC hdc); //Modified by bklee. 02/01/95
  2761. ULONG FASTCALL WG32GetFontAssocStatus(PVDMFRAME pFrame)
  2762. {
  2763. ULONG ul;
  2764. register PGETFONTASSOCSTATUS16 parg16;
  2765. GETARGPTR(pFrame, sizeof(GETFONTASSOCSTATUS16), parg16);
  2766. ul = GetFontAssocStatus(HDC32(parg16->f1));
  2767. FREEARGPTR(parg16);
  2768. RETURN (ul);
  2769. }
  2770. #endif //FE_SB
  2771. /*+++
  2772. This returns the number of bytes in RLE4 and RLE8 compressed bitmaps.
  2773. Code below fixes problems in Sounditoutland with RLE-encoded
  2774. bitmaps which have biSizeImage == 0. On Win 3.1 they work since
  2775. gdi is happy with decoding some piece of memory. NT GDI however
  2776. needs to know the size of bits passed. We remedy this by calculating
  2777. size using RET_GETDIBSIZE (GetSelectorLimit). GDI won't copy the
  2778. memory, it will just use size as indication of accessibility
  2779. Applications: "Sound It Out Land", QuarkXpress, KidPhonics
  2780. ---*/
  2781. ULONG Get_RLE_Compression_Size(DWORD RLE_Type, PBYTE pStart, VPVOID vpBytes)
  2782. {
  2783. BOOL bDone = FALSE;
  2784. PBYTE pBytes = pStart;
  2785. PARM16 Parm16;
  2786. ULONG SelectorLimit;
  2787. LONG lSize;
  2788. if(pBytes && vpBytes) {
  2789. Parm16.WndProc.wParam = HIWORD(vpBytes);
  2790. // get # bytes allocated to the selector (this even works for huge)
  2791. CallBack16(RET_GETDIBSIZE, &Parm16, 0, (PVPVOID)&SelectorLimit);
  2792. // is the selector valid?
  2793. if(SelectorLimit != 0 && SelectorLimit != 0xffffffff) {
  2794. // max byte buffer = memory block size - starting offset
  2795. lSize = (LONG)SelectorLimit - LOWORD(vpBytes) + 1;
  2796. } else {
  2797. LOGDEBUG(LOG_ALWAYS, ("WOW:Get_RLE_Compression_Size: Selector [ptr:%x] is invalid\n", (DWORD)vpBytes));
  2798. return(0);
  2799. }
  2800. while (!bDone) {
  2801. // if absolute mode
  2802. if (*pBytes == 0) {
  2803. switch (pBytes[1]) {
  2804. case 0: // end of line
  2805. pBytes += 2;
  2806. break;
  2807. case 1: // end of bitmap
  2808. pBytes += 2;
  2809. bDone = TRUE;
  2810. break;
  2811. case 2: // offset
  2812. pBytes += 4;
  2813. break;
  2814. default:
  2815. // align the bytes to word boundries
  2816. if(RLE_Type == BI_RLE4) {
  2817. pBytes += ((2 + ((pBytes[1] + 3) / 2)) & ~1);
  2818. } else {
  2819. pBytes += ((2 + pBytes[1] + 1) & ~1);
  2820. }
  2821. break;
  2822. }
  2823. } else { // else encoded mode
  2824. pBytes += 2;
  2825. }
  2826. // are we past the end of the selector?
  2827. if ( lSize < (pBytes - pStart + 1) ) {
  2828. LOGDEBUG(LOG_ALWAYS, ("WOW:Get_RLE_Compression_Size:Bad RLE size: %x < %x\n", lSize, (pBytes - pStart)));
  2829. return(lSize);
  2830. }
  2831. }
  2832. }
  2833. return((ULONG)(pBytes - pStart));
  2834. }