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.

5841 lines
167 KiB

  1. #include "precomp.h"
  2. //
  3. // OE.C
  4. // Order Encoder, display driver side
  5. //
  6. // Copyright(c) Microsoft 1997-
  7. //
  8. //
  9. // Number of entries in the font alias table.
  10. //
  11. #define NUM_ALIAS_FONTS 3
  12. //
  13. // Define entries in the Font Alias table. This table is used to convert
  14. // non-existant fonts (used by certain widely used applications) into
  15. // something we can use as a local font.
  16. //
  17. // The font names that we alias are:
  18. //
  19. // "Helv"
  20. // This is used by Excel. It is mapped directly onto "MS Sans Serif".
  21. //
  22. // "MS Dialog"
  23. // This is used by Word. It is the same as an 8pt bold MS Sans Serif.
  24. // We actually map it to a "MS Sans Serif" font that is one pel narrower
  25. // than the metrics specify (because all matching is done on non-bold
  26. // fonts) - hence the 1 value in the charWidthAdjustment field.
  27. //
  28. // "MS Dialog Light"
  29. // Added as part of the Win95 performance enhancements...Presumably for
  30. // MS-Word...
  31. //
  32. //
  33. FONT_ALIAS_TABLE fontAliasTable[NUM_ALIAS_FONTS] =
  34. {
  35. { "Helv", "MS Sans Serif", 0 },
  36. { "MS Dialog", "MS Sans Serif", 1 },
  37. { "MS Dialog Light", "MS Sans Serif", 0 }
  38. };
  39. //
  40. // FUNCTION: OE_SendAsOrder see oe.h
  41. //
  42. BOOL OE_SendAsOrder(DWORD order)
  43. {
  44. BOOL rc = FALSE;
  45. DebugEntry(OE_SendAsOrder);
  46. //
  47. // Only check the order if we are allowed to send orders in the first
  48. // place!
  49. //
  50. if (g_oeSendOrders)
  51. {
  52. TRACE_OUT(("Orders enabled"));
  53. //
  54. // We are sending some orders, so check individual flags.
  55. //
  56. rc = (BOOL)g_oeOrderSupported[HIWORD(order)];
  57. TRACE_OUT(("Send order %lx HIWORD %hu", order, HIWORD(order)));
  58. }
  59. DebugExitDWORD(OE_SendAsOrder, rc);
  60. return(rc);
  61. }
  62. //
  63. // OE_RectIntersectsSDA - see oe.h
  64. //
  65. BOOL OE_RectIntersectsSDA(LPRECT pRect)
  66. {
  67. RECT rectVD;
  68. BOOL fIntersection = FALSE;
  69. UINT i;
  70. DebugEntry(OE_RectIntersectsSDA);
  71. //
  72. // Copy the supplied rectangle, converting to inclusive Virtual
  73. // Desktop coords.
  74. //
  75. rectVD.left = pRect->left;
  76. rectVD.top = pRect->top;
  77. rectVD.right = pRect->right - 1;
  78. rectVD.bottom = pRect->bottom - 1;
  79. //
  80. // Loop through each of the bounding rectangles checking for
  81. // an intersection with the supplied rectangle.
  82. //
  83. for (i = 0; i <= BA_NUM_RECTS; i++)
  84. {
  85. if ( (g_baBounds[i].InUse) &&
  86. (g_baBounds[i].Coord.left <= rectVD.right) &&
  87. (g_baBounds[i].Coord.top <= rectVD.bottom) &&
  88. (g_baBounds[i].Coord.right >= rectVD.left) &&
  89. (g_baBounds[i].Coord.bottom >= rectVD.top) )
  90. {
  91. TRACE_OUT(("Rect(%d,%d)(%d,%d) intersects SDA(%d,%d)(%d,%d)",
  92. rectVD.left, rectVD.top,
  93. rectVD.right, rectVD.bottom,
  94. g_baBounds[i].Coord.left, g_baBounds[i].Coord.top,
  95. g_baBounds[i].Coord.right, g_baBounds[i].Coord.bottom));
  96. fIntersection = TRUE;
  97. break;
  98. }
  99. }
  100. DebugExitDWORD(OE_RectIntersectsSDA, fIntersection);
  101. return(fIntersection);
  102. }
  103. //
  104. // DrvBitBlt - see NT DDK documentation.
  105. //
  106. BOOL DrvBitBlt( SURFOBJ *psoDst,
  107. SURFOBJ *psoSrc,
  108. SURFOBJ *psoMask,
  109. CLIPOBJ *pco,
  110. XLATEOBJ *pxlo,
  111. RECTL *prclDst,
  112. POINTL *pptlSrc,
  113. POINTL *pptlMask,
  114. BRUSHOBJ *pbo,
  115. POINTL *pptlBrush,
  116. ROP4 rop4 )
  117. {
  118. LPOSI_PDEV ppdev = (LPOSI_PDEV)psoDst->dhpdev;
  119. BOOL rc = TRUE;
  120. UINT orderType = 0;
  121. BYTE rop3;
  122. LPINT_ORDER pOrder = NULL;
  123. LPDSTBLT_ORDER pDstBlt;
  124. LPSCRBLT_ORDER pScrBlt;
  125. LPMEMBLT_ORDER pMemBlt;
  126. LPMEM3BLT_ORDER pMem3Blt;
  127. BOOL fSendOrder = FALSE;
  128. BOOL fAccumulate = FALSE;
  129. UINT fOrderFlags = OF_SPOILABLE;
  130. RECT bounds;
  131. RECT intersectRect;
  132. POINT origin;
  133. POE_BRUSH_DATA pCurrentBrush;
  134. MEMBLT_ORDER_EXTRA_INFO memBltExtraInfo;
  135. DebugEntry(DrvBitBlt);
  136. //
  137. // DO THIS _BEFORE_ TAKING LOCKS
  138. //
  139. if (!g_oeViewers)
  140. goto NO_LOCK_EXIT;
  141. OE_SHM_START_WRITING;
  142. //
  143. // Get the bounding rectangle for the operation.
  144. //
  145. RECT_FROM_RECTL(bounds, (*prclDst));
  146. //
  147. // Check if we are accumulating data for this function
  148. //
  149. fAccumulate = OEAccumulateOutput(psoDst, pco, &bounds);
  150. if (!fAccumulate)
  151. {
  152. DC_QUIT;
  153. }
  154. //
  155. // Convert the data to virtual coordinates.
  156. //
  157. OELRtoVirtual(&bounds, 1);
  158. //
  159. // Check if this 4-way ROP simplifies to a 3-way ROP. A 4-way ROP
  160. // contains two 3-way ROPS, one for each setting of a mask bit - the
  161. // high ROP3 corresponds to a value of zero in the mask bit.
  162. //
  163. // If the two 3-way ROPs are the same, we know the 4-way ROP is a 3-way
  164. // ROP.
  165. //
  166. if (ROP3_LOW_FROM_ROP4(rop4) == ROP3_HIGH_FROM_ROP4(rop4))
  167. {
  168. //
  169. // Take the high byte as the 3-way ROP.
  170. //
  171. rop3 = ROP3_HIGH_FROM_ROP4(rop4);
  172. TRACE_OUT(( "4-way ROP %04x is really 3-way %02x", rop4, rop3));
  173. }
  174. else
  175. {
  176. TRACE_OUT(( "4-way ROP %08x", rop4));
  177. DC_QUIT;
  178. }
  179. //
  180. // Determine the command type. It can be one of the following.
  181. //
  182. // DSTBLT - A destination only BLT (no source, or pattern)
  183. // PATBLT - a pattern BLT (no source)
  184. // SCRBLT - a screen to screen BLT
  185. // MEMBLT - a memory to screen BLT (no pattern)
  186. // MEM3BLT - a memory to screen 3-way BLT
  187. //
  188. //
  189. // Check for destination only BLTs (ie. independent of source bits).
  190. //
  191. if ((psoSrc == NULL) || ROP3_NO_SOURCE(rop3))
  192. {
  193. //
  194. // Check for a pattern or true destination BLT.
  195. //
  196. if (ROP3_NO_PATTERN(rop3))
  197. {
  198. TRACE_OUT(( "DSTBLT"));
  199. orderType = ORD_DSTBLT;
  200. }
  201. else
  202. {
  203. TRACE_OUT(( "PATBLT"));
  204. orderType = ORD_PATBLT;
  205. }
  206. }
  207. else
  208. {
  209. //
  210. // We have a source BLT, check whether we have screen or memory
  211. // BLTs.
  212. //
  213. if (psoSrc->hsurf != ppdev->hsurfScreen)
  214. {
  215. if (psoDst->hsurf != ppdev->hsurfScreen)
  216. {
  217. ERROR_OUT(( "MEM to MEM blt!"));
  218. }
  219. else
  220. {
  221. //
  222. // We have a memory to screen BLT, check which type.
  223. //
  224. if ((ppdev->cBitsPerPel == 4) && (rop3 != 0xcc))
  225. {
  226. //
  227. // No order -- the result depends on the palette
  228. // which is dicy in VGA
  229. //
  230. TRACE_OUT(("No order on VGA for rop 0x%02x", rop3));
  231. DC_QUIT;
  232. }
  233. if (ROP3_NO_PATTERN(rop3))
  234. {
  235. TRACE_OUT(( "MEMBLT"));
  236. orderType = ORD_MEMBLT;
  237. }
  238. else
  239. {
  240. TRACE_OUT(( "MEM3BLT"));
  241. orderType = ORD_MEM3BLT;
  242. }
  243. }
  244. }
  245. else
  246. {
  247. if (psoDst->hsurf != ppdev->hsurfScreen)
  248. {
  249. TRACE_OUT(( "SCR to MEM blt!"));
  250. }
  251. else
  252. {
  253. //
  254. // We only support destination only screen BLTs (ie. no
  255. // patterns allowed).
  256. //
  257. if (ROP3_NO_PATTERN(rop3))
  258. {
  259. TRACE_OUT(( "SCRBLT"));
  260. orderType = ORD_SCRBLT;
  261. }
  262. else
  263. {
  264. TRACE_OUT(( "Unsupported screen ROP %x", rop3));
  265. }
  266. }
  267. }
  268. }
  269. //
  270. // Check if we have a supported order.
  271. //
  272. if (orderType == 0)
  273. {
  274. TRACE_OUT(( "Unsupported BLT"));
  275. fAccumulate = FALSE;
  276. DC_QUIT;
  277. }
  278. //
  279. // Check if we are allowed to send this order (determined by the
  280. // negotiated capabilities of all the machines in the conference).
  281. //
  282. if (!OE_SendAsOrder(orderType))
  283. {
  284. TRACE_OUT(( "Order %d not allowed", orderType));
  285. DC_QUIT;
  286. }
  287. //
  288. // Check if we are allowed to send the ROP.
  289. //
  290. if (!OESendRop3AsOrder(rop3))
  291. {
  292. TRACE_OUT(( "Cannot send ROP %d", rop3));
  293. DC_QUIT;
  294. }
  295. //
  296. // Check for overcomplicated clipping.
  297. //
  298. if (OEClippingIsComplex(pco))
  299. {
  300. TRACE_OUT(( "Clipping is too complex"));
  301. DC_QUIT;
  302. }
  303. //
  304. // If this is a Memblt, do an initial check on whether it is cachable
  305. //
  306. if ((orderType == ORD_MEMBLT) || (orderType == ORD_MEM3BLT))
  307. {
  308. //
  309. // We have to fill in a structure containing extra into
  310. // specifically for a MEM(3)BLT order.
  311. //
  312. memBltExtraInfo.pSource = psoSrc;
  313. memBltExtraInfo.pDest = psoDst;
  314. memBltExtraInfo.pXlateObj = pxlo;
  315. if (!SBC_DDIsMemScreenBltCachable(&memBltExtraInfo))
  316. {
  317. TRACE_OUT(( "MemBlt is not cachable"));
  318. DC_QUIT;
  319. }
  320. //
  321. // It is cachable. Before we get SBC to do the caching, we have to
  322. // allow it to queue a color table (if required).
  323. //
  324. if (!SBC_DDMaybeQueueColorTable(ppdev))
  325. {
  326. TRACE_OUT(( "Unable to queue color table for MemBlt"));
  327. DC_QUIT;
  328. }
  329. }
  330. //
  331. // We have a recognised order - do the specific checks for each order.
  332. //
  333. switch (orderType)
  334. {
  335. case ORD_DSTBLT:
  336. //
  337. // Allocate the memory for the order.
  338. //
  339. pOrder = OA_DDAllocOrderMem(sizeof(DSTBLT_ORDER),0);
  340. if (pOrder == NULL)
  341. {
  342. TRACE_OUT(( "Failed to alloc order"));
  343. DC_QUIT;
  344. }
  345. pDstBlt = (LPDSTBLT_ORDER)pOrder->abOrderData;
  346. //
  347. // Set the spoiler flag if the rop is opaque.
  348. //
  349. if (ROP3_IS_OPAQUE(rop3))
  350. {
  351. fOrderFlags |= OF_SPOILER;
  352. }
  353. //
  354. // Store the order type.
  355. //
  356. pDstBlt->type = LOWORD(orderType);
  357. //
  358. // Virtual desktop co-ordinates.
  359. //
  360. pDstBlt->nLeftRect = bounds.left;
  361. pDstBlt->nTopRect = bounds.top;
  362. pDstBlt->nWidth = bounds.right - bounds.left + 1;
  363. pDstBlt->nHeight = bounds.bottom - bounds.top + 1;
  364. pDstBlt->bRop = rop3;
  365. TRACE_OUT(( "DstBlt X %d Y %d w %d h %d rop %02X",
  366. pDstBlt->nLeftRect,
  367. pDstBlt->nTopRect,
  368. pDstBlt->nWidth,
  369. pDstBlt->nHeight,
  370. pDstBlt->bRop));
  371. break;
  372. case ORD_PATBLT:
  373. if ( !OEEncodePatBlt(ppdev,
  374. pbo,
  375. pptlBrush,
  376. rop3,
  377. &bounds,
  378. &pOrder) )
  379. {
  380. //
  381. // Something went wrong with the encoding, so skip to the
  382. // end to add this operation to the SDA.
  383. //
  384. DC_QUIT;
  385. }
  386. fOrderFlags = pOrder->OrderHeader.Common.fOrderFlags;
  387. break;
  388. case ORD_SCRBLT:
  389. //
  390. // Check for a SCRBLT as a result of a Desktop Scroll. We must
  391. // ignore these as they will stuff the remote desktop.
  392. //
  393. // The check is simple - if the virtual position of the source
  394. // is the same as the virual position of the target for a
  395. // SRCCOPY type SCRBLT, we have a hit...
  396. //
  397. POINT_FROM_POINTL(origin, (*pptlSrc));
  398. //
  399. // Allocate the memory for the order.
  400. //
  401. pOrder = OA_DDAllocOrderMem(sizeof(SCRBLT_ORDER),0);
  402. if (pOrder == NULL)
  403. {
  404. TRACE_OUT(( "Failed to alloc order"));
  405. DC_QUIT;
  406. }
  407. pScrBlt = (LPSCRBLT_ORDER)pOrder->abOrderData;
  408. //
  409. // Store the order type.
  410. //
  411. pScrBlt->type = LOWORD(orderType);
  412. //
  413. // All data which is sent over the wire must be in virtual
  414. // desktop co-ordinates. OELRtoVirtual has already converted
  415. // bounds to an inclusive rectangle in virtual co-ordinates.
  416. //
  417. pScrBlt->nLeftRect = bounds.left;
  418. pScrBlt->nTopRect = bounds.top;
  419. pScrBlt->nWidth = bounds.right - bounds.left + 1;
  420. pScrBlt->nHeight = bounds.bottom - bounds.top + 1;
  421. pScrBlt->bRop = rop3;
  422. //
  423. // Source point on the screen.
  424. //
  425. OELPtoVirtual(&origin, 1);
  426. pScrBlt->nXSrc = origin.x;
  427. pScrBlt->nYSrc = origin.y;
  428. //
  429. // Screen to screen blts are Blocking orders (i.e. they
  430. // prevent any previous orders from being spoilt).
  431. //
  432. // We do not mark Screen to Screen blts as SPOILER orders. If
  433. // the ROP is opaque we could spoil the destination rect, but
  434. // only the area that does not overlap with the src rectangle.
  435. // The most common use of Screen to Screen blts is scrolling,
  436. // where the src and dst rects almost completely overlap,
  437. // giving only a small "spoiler" region. The spoiler region
  438. // could also be complex (more that 1 rect).
  439. //
  440. // Consequently, the potential gains of trying to spoil using
  441. // these orders are small compared to the complexity of the
  442. // code required.
  443. //
  444. //
  445. fOrderFlags |= OF_BLOCKER;
  446. //
  447. // If the blt is screen to screen and the source overlaps the
  448. // destination and the clipping is not simple (> 1 rect) then
  449. // we do not want to send this as an order.
  450. //
  451. // (This is because we would need some complex code to
  452. // calculate the order in which to blt through each of the clip
  453. // rects. As this case is pretty rare, it seems reasonable to
  454. // just send it as Screen Data).
  455. //
  456. if (!OEClippingIsSimple(pco))
  457. {
  458. //
  459. // Calculate the overlapping rectangle.
  460. //
  461. intersectRect.left = max(pScrBlt->nLeftRect, pScrBlt->nXSrc);
  462. intersectRect.right = min(
  463. pScrBlt->nLeftRect + pScrBlt->nWidth-1,
  464. pScrBlt->nXSrc + pScrBlt->nWidth-1 );
  465. intersectRect.top = max(pScrBlt->nTopRect, pScrBlt->nYSrc);
  466. intersectRect.bottom = min(
  467. pScrBlt->nTopRect + pScrBlt->nHeight-1,
  468. pScrBlt->nYSrc + pScrBlt->nHeight-1 );
  469. //
  470. // Check for a src / dst overlap. If they overlap, the
  471. // intersection is a well-ordered non-trivial rectangle.
  472. //
  473. if ( (intersectRect.left <= intersectRect.right ) &&
  474. (intersectRect.top <= intersectRect.bottom) )
  475. {
  476. //
  477. // The src & dest overlap. Free up the order memory
  478. // and skip out now. The destination rectangle will be
  479. // added to the Screen Data Area.
  480. //
  481. OA_DDFreeOrderMem(pOrder);
  482. DC_QUIT;
  483. }
  484. }
  485. TRACE_OUT(( "ScrBlt x %d y %d w %d h %d sx %d sy %d rop %02X",
  486. pScrBlt->nLeftRect,
  487. pScrBlt->nTopRect,
  488. pScrBlt->nWidth,
  489. pScrBlt->nHeight,
  490. pScrBlt->nXSrc,
  491. pScrBlt->nYSrc,
  492. pScrBlt->bRop));
  493. break;
  494. case ORD_MEMBLT:
  495. //
  496. // Allocate the memory for the order - don't use OA as we are
  497. // only going to tile this order immediately. Instead, we have
  498. // a static buffer to receive the template order data.
  499. //
  500. pOrder = (LPINT_ORDER)g_oeTmpOrderBuffer;
  501. pMemBlt = (LPMEMBLT_ORDER)pOrder->abOrderData;
  502. pOrder->OrderHeader.Common.cbOrderDataLength
  503. = sizeof(MEMBLT_R2_ORDER);
  504. //
  505. // Store the order type.
  506. //
  507. pMemBlt->type = LOWORD(orderType);
  508. //
  509. // Any data which is sent over the wire must be in virtual
  510. // desktop co-ordinates. The bounding rectangle has already
  511. // been converted by OELRtoScreen.
  512. //
  513. pMemBlt->nLeftRect = bounds.left;
  514. pMemBlt->nTopRect = bounds.top;
  515. pMemBlt->nWidth = bounds.right - bounds.left + 1;
  516. pMemBlt->nHeight = bounds.bottom - bounds.top + 1;
  517. pMemBlt->bRop = rop3;
  518. //
  519. // We need to store the source bitmap origin. This is a memory
  520. // object, so screen/virtual conversions are unnecessary.
  521. //
  522. pMemBlt->nXSrc = pptlSrc->x;
  523. pMemBlt->nYSrc = pptlSrc->y;
  524. //
  525. // Mark the order as opaque if necessary.
  526. //
  527. if (ROP3_IS_OPAQUE(rop3))
  528. {
  529. fOrderFlags |= OF_SPOILER;
  530. }
  531. //
  532. // Store the src bitmap handle in the order.
  533. //
  534. pMemBlt->cacheId = 0;
  535. TRACE_OUT(( "MemBlt dx %d dy %d w %d h %d sx %d sy %d rop %04X",
  536. pMemBlt->nLeftRect,
  537. pMemBlt->nTopRect,
  538. pMemBlt->nWidth,
  539. pMemBlt->nHeight,
  540. pMemBlt->nXSrc,
  541. pMemBlt->nYSrc,
  542. pMemBlt->bRop));
  543. break;
  544. case ORD_MEM3BLT:
  545. //
  546. // Check that the brush pattern is simple.
  547. //
  548. if (!OECheckBrushIsSimple(ppdev, pbo, &pCurrentBrush))
  549. {
  550. TRACE_OUT(( "Brush is not simple"));
  551. orderType = 0;
  552. DC_QUIT;
  553. }
  554. //
  555. // Allocate the memory for the order - don't use OA as we are
  556. // only going to tile this order immediately. Instead, we have
  557. // a static buffer to receive the template order data.
  558. //
  559. pOrder = (LPINT_ORDER)g_oeTmpOrderBuffer;
  560. pMem3Blt = (LPMEM3BLT_ORDER)pOrder->abOrderData;
  561. pOrder->OrderHeader.Common.cbOrderDataLength
  562. = sizeof(MEM3BLT_R2_ORDER);
  563. //
  564. // Store the order type.
  565. //
  566. pMem3Blt->type = LOWORD(orderType);
  567. //
  568. // All data which is sent over the wire must be in virtual
  569. // desktop co-ordinates. OELRtoVirtual has already done this
  570. // conversion for us.
  571. //
  572. pMem3Blt->nLeftRect = bounds.left;
  573. pMem3Blt->nTopRect = bounds.top;
  574. pMem3Blt->nWidth = bounds.right - bounds.left + 1;
  575. pMem3Blt->nHeight = bounds.bottom - bounds.top + 1;
  576. pMem3Blt->bRop = rop3;
  577. //
  578. // We need to store the source bitmap origin. This is a memory
  579. // object, so screen/virtual conversions are unnecessary.
  580. //
  581. pMem3Blt->nXSrc = pptlSrc->x;
  582. pMem3Blt->nYSrc = pptlSrc->y;
  583. //
  584. // Mark the order as opaque if necessary.
  585. //
  586. if (ROP3_IS_OPAQUE(rop3))
  587. {
  588. fOrderFlags |= OF_SPOILER;
  589. }
  590. //
  591. // Store the src bitmap handle in the order.
  592. //
  593. pMem3Blt->cacheId = 0;
  594. //
  595. // Set up the information required for the pattern.
  596. //
  597. pMem3Blt->BackColor = pCurrentBrush->back;
  598. pMem3Blt->ForeColor = pCurrentBrush->fore;
  599. //
  600. // The protocol brush origin is the point on the screen where
  601. // we want the brush to start being drawn from (tiling where
  602. // necessary). This must be in virtual coordinates.
  603. //
  604. pMem3Blt->BrushOrgX = pptlBrush->x;
  605. pMem3Blt->BrushOrgY = pptlBrush->y;
  606. OELPtoVirtual((LPPOINT)&pMem3Blt->BrushOrgX, 1);
  607. //
  608. // Extra brush data from the data when we realised the brush.
  609. //
  610. pMem3Blt->BrushStyle = pCurrentBrush->style;
  611. pMem3Blt->BrushHatch = pCurrentBrush->style;
  612. RtlCopyMemory(pMem3Blt->BrushExtra,
  613. pCurrentBrush->brushData,
  614. sizeof(pMem3Blt->BrushExtra));
  615. TRACE_OUT(( "Mem3Blt brush %02X %02X dx %d dy %d w %d h %d "
  616. "sx %d sy %d rop %04X",
  617. pMem3Blt->BrushStyle,
  618. pMem3Blt->BrushHatch,
  619. pMem3Blt->nLeftRect,
  620. pMem3Blt->nTopRect,
  621. pMem3Blt->nWidth,
  622. pMem3Blt->nHeight,
  623. pMem3Blt->nXSrc,
  624. pMem3Blt->nYSrc,
  625. pMem3Blt->bRop));
  626. break;
  627. default:
  628. ERROR_OUT(( "New unsupported order %08lx", orderType));
  629. orderType = 0;
  630. break;
  631. }
  632. //
  633. // We have generated an order so make sure we send it.
  634. //
  635. if (orderType != 0)
  636. {
  637. fSendOrder = TRUE;
  638. }
  639. DC_EXIT_POINT:
  640. //
  641. // If we did not send an order, we must accumulate the output in the
  642. // Screen Data Area.
  643. //
  644. if (fSendOrder)
  645. {
  646. //
  647. // Check if the ROP has a dependency on the destination.
  648. //
  649. if (!ROP3_NO_TARGET(rop3))
  650. {
  651. TRACE_OUT(( "ROP has a target dependency"));
  652. fOrderFlags |= OF_DESTROP;
  653. }
  654. //
  655. // Store the general order data. The bounding rectagle
  656. // co-ordinates must be virtual desktop. OELRtoVirtual has already
  657. // converted rect for us.
  658. //
  659. pOrder->OrderHeader.Common.fOrderFlags = (TSHR_UINT16)fOrderFlags;
  660. TSHR_RECT16_FROM_RECT(&pOrder->OrderHeader.Common.rcsDst, bounds);
  661. //
  662. // Add the order to the cache. Note that we have the new tiled
  663. // processing for MEMBLT and MEM3BLT orders.
  664. //
  665. if ((orderType == ORD_MEMBLT) || (orderType == ORD_MEM3BLT))
  666. {
  667. OETileBitBltOrder(pOrder, &memBltExtraInfo, pco);
  668. }
  669. else
  670. {
  671. OEClipAndAddOrder(pOrder, NULL, pco);
  672. }
  673. }
  674. else
  675. {
  676. if (fAccumulate)
  677. {
  678. OEClipAndAddScreenData(&bounds, pco);
  679. }
  680. }
  681. OE_SHM_STOP_WRITING;
  682. NO_LOCK_EXIT:
  683. DebugExitDWORD(DrvBitBlt, rc);
  684. return(rc);
  685. }
  686. //
  687. // DrvStretchBlt - see NT DDK documentation.
  688. //
  689. BOOL DrvStretchBlt(SURFOBJ *psoDst,
  690. SURFOBJ *psoSrc,
  691. SURFOBJ *psoMask,
  692. CLIPOBJ *pco,
  693. XLATEOBJ *pxlo,
  694. COLORADJUSTMENT *pca,
  695. POINTL *pptlHTOrg,
  696. RECTL *prclDst,
  697. RECTL *prclSrc,
  698. POINTL *pptlMask,
  699. ULONG iMode)
  700. {
  701. BOOL rc = TRUE;
  702. RECT rectSrc;
  703. RECT rectDst;
  704. BOOL fAccumulate = FALSE;
  705. POINTL ptlSrc;
  706. BOOL usedBitBlt = FALSE;
  707. DebugEntry(DrvStretchBlt);
  708. //
  709. // DO THIS _BEFORE_ TAKING LOCK
  710. //
  711. if (!g_oeViewers)
  712. goto NO_LOCK_EXIT;
  713. OE_SHM_START_WRITING;
  714. //
  715. // Get the source and destination rectangles
  716. //
  717. RECT_FROM_RECTL(rectSrc, (*prclSrc));
  718. RECT_FROM_RECTL(rectDst, (*prclDst));
  719. //
  720. // Check if we are accumulating data for this function
  721. //
  722. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  723. if (!fAccumulate)
  724. {
  725. DC_QUIT;
  726. }
  727. //
  728. // Check that we have a valid ROP code. The NT DDK states that the ROP
  729. // code for the StretchBlt is implicit in the mask specification. If a
  730. // mask is specified, we have an implicit ROP4 of 0xCCAA, otherwise the
  731. // code is 0xCCCC.
  732. //
  733. // Our BitBlt code only encodes orders for ROP3s, so we must throw any
  734. // StretchBlts with a mask.
  735. //
  736. if (psoMask != NULL)
  737. {
  738. TRACE_OUT(( "Mask specified"));
  739. DC_QUIT;
  740. }
  741. //
  742. // Check for overcomplicated clipping.
  743. //
  744. if (OEClippingIsComplex(pco))
  745. {
  746. TRACE_OUT(( "Clipping is too complex"));
  747. DC_QUIT;
  748. }
  749. //
  750. // Rectangles are now well-ordered, check if we have a degenerate (ie.
  751. // no stretch) case.
  752. //
  753. if ( (rectSrc.right - rectSrc.left == rectDst.right - rectDst.left) &&
  754. (rectSrc.bottom - rectSrc.top == rectDst.bottom - rectDst.top ) )
  755. {
  756. //
  757. // This can be passed on to the BitBlt code.
  758. //
  759. usedBitBlt = TRUE;
  760. ptlSrc.x = prclSrc->left;
  761. ptlSrc.y = prclSrc->top;
  762. rc = DrvBitBlt(psoDst,
  763. psoSrc,
  764. psoMask,
  765. pco,
  766. pxlo,
  767. prclDst,
  768. &ptlSrc,
  769. pptlMask,
  770. NULL,
  771. NULL,
  772. 0xCCCC);
  773. //
  774. // We have stored this object in the BitBlt, so don't store the
  775. // data again.
  776. //
  777. fAccumulate = FALSE;
  778. }
  779. DC_EXIT_POINT:
  780. if (fAccumulate)
  781. {
  782. //
  783. // Convert the data to virtual coordinates.
  784. //
  785. OELRtoVirtual(&rectDst, 1);
  786. //
  787. // Update the screen data area
  788. //
  789. OEClipAndAddScreenData(&rectDst, pco);
  790. }
  791. OE_SHM_STOP_WRITING;
  792. NO_LOCK_EXIT:
  793. DebugExitDWORD(DrvStretchBlt, rc);
  794. return(rc);
  795. }
  796. //
  797. // DrvCopyBits - see NT DDK documentation.
  798. //
  799. BOOL DrvCopyBits(SURFOBJ *psoDst,
  800. SURFOBJ *psoSrc,
  801. CLIPOBJ *pco,
  802. XLATEOBJ *pxlo,
  803. RECTL *prclDst,
  804. POINTL *pptlSrc)
  805. {
  806. //
  807. // CopyBits is a fast path for the NT display drivers. In our case it
  808. // can always be processed as a BITBLT.
  809. //
  810. return(DrvBitBlt( psoDst,
  811. psoSrc,
  812. NULL,
  813. pco,
  814. pxlo,
  815. prclDst,
  816. pptlSrc,
  817. NULL,
  818. NULL,
  819. NULL,
  820. 0xCCCC));
  821. }
  822. //
  823. // DrvTextOut - see NT DDK documentation.
  824. //
  825. BOOL DrvTextOut(SURFOBJ *pso,
  826. STROBJ *pstro,
  827. FONTOBJ *pfo,
  828. CLIPOBJ *pco,
  829. RECTL *prclExtra,
  830. RECTL *prclOpaque,
  831. BRUSHOBJ *pboFore,
  832. BRUSHOBJ *pboOpaque,
  833. POINTL *pptlOrg,
  834. MIX mix)
  835. {
  836. LPOSI_PDEV ppdev = (LPOSI_PDEV)pso->dhpdev;
  837. BOOL rc = TRUE;
  838. RECT rectDst;
  839. RECT rectText;
  840. LPINT_ORDER pOrder;
  841. LPINT_ORDER pOpaqueOrder;
  842. LPTEXTOUT_ORDER pTextOut;
  843. LPEXTTEXTOUT_ORDER pExtTextOut;
  844. BOOL fSendOrder = FALSE;
  845. BOOL fAccumulate = FALSE;
  846. char ansiString[ORD_MAX_STRING_LEN_WITHOUT_DELTAS+2];
  847. ULONG ansiLen;
  848. ULONG tempLen;
  849. UINT orderType = 0;
  850. ULONG maxLength;
  851. LPSTR lpVariable;
  852. BOOL fMoreData;
  853. ULONG count;
  854. ULONG i;
  855. GLYPHPOS* pGlyphData;
  856. int currentDelta;
  857. LPVARIABLE_DELTAX lpDeltaPos;
  858. UINT fontFlags;
  859. UINT fontAscender;
  860. UINT fontHeight;
  861. UINT fontWidth;
  862. UINT fontWeight;
  863. UINT fontIndex;
  864. POINTL lastPtl;
  865. LPCOMMON_TEXTORDER pCommon;
  866. POINT startPoint;
  867. BOOL sendDeltaX = FALSE;
  868. DebugEntry(DrvTextOut);
  869. //
  870. // DO THIS _BEFORE_ TAKING LOCKS
  871. //
  872. if (!g_oeViewers)
  873. goto NO_LOCK_EXIT;
  874. OE_SHM_START_WRITING;
  875. //
  876. // Get bounding rectangle and convert to a RECT.
  877. //
  878. if (prclOpaque != NULL)
  879. {
  880. RECT_FROM_RECTL(rectDst, (*prclOpaque));
  881. }
  882. else
  883. {
  884. RECT_FROM_RECTL(rectDst, pstro->rclBkGround);
  885. TRACE_OUT(( "Using STROBJ bgd for size"));
  886. }
  887. //
  888. // Check if we are accumulating data for this function
  889. //
  890. fAccumulate = OEAccumulateOutput(pso, pco, &rectDst);
  891. if (!fAccumulate)
  892. {
  893. DC_QUIT;
  894. }
  895. //
  896. // Convert to virtual coordinates
  897. //
  898. OELRtoVirtual(&rectDst, 1);
  899. //
  900. // Determine which order we will generate
  901. //
  902. if ( ((pstro->flAccel & SO_FLAG_DEFAULT_PLACEMENT) != 0) &&
  903. (prclOpaque == NULL) )
  904. {
  905. orderType = ORD_TEXTOUT;
  906. maxLength = ORD_MAX_STRING_LEN_WITHOUT_DELTAS;
  907. }
  908. else
  909. {
  910. orderType = ORD_EXTTEXTOUT;
  911. maxLength = ORD_MAX_STRING_LEN_WITH_DELTAS;
  912. }
  913. //
  914. // Check if we are allowed to send this order (determined by the
  915. // negotiated capabilities of all the machines in the conference).
  916. //
  917. if (!OE_SendAsOrder(orderType))
  918. {
  919. TRACE_OUT(( "Text order %x not allowed", orderType));
  920. DC_QUIT;
  921. }
  922. //
  923. // Check for a valid brush for the test operation.
  924. //
  925. if (pboFore->iSolidColor == -1)
  926. {
  927. TRACE_OUT(( "Bad brush for text fg"));
  928. DC_QUIT;
  929. }
  930. if (pboOpaque->iSolidColor == -1)
  931. {
  932. TRACE_OUT(( "Bad brush for text bg"));
  933. DC_QUIT;
  934. }
  935. //
  936. // Check that we don't have any modifiers rects on the font
  937. //
  938. if (prclExtra != NULL)
  939. {
  940. TRACE_OUT(( "Unsupported extra rects"));
  941. DC_QUIT;
  942. }
  943. //
  944. // Check that text orientation is OK.
  945. //
  946. if (pstro->flAccel & OE_BAD_TEXT_MASK)
  947. {
  948. TRACE_OUT(("DrvTextOut - unsupported flAccel 0x%08x", pstro->flAccel));
  949. DC_QUIT;
  950. }
  951. //
  952. // Check we have a valid string.
  953. //
  954. if (pstro->pwszOrg == NULL)
  955. {
  956. TRACE_OUT(( "No string - opaque %x", prclOpaque));
  957. DC_QUIT;
  958. }
  959. //
  960. // Check for overcomplicated clipping.
  961. //
  962. if (OEClippingIsComplex(pco))
  963. {
  964. TRACE_OUT(( "Clipping is too complex"));
  965. DC_QUIT;
  966. }
  967. //
  968. // Convert the string to an ANSI representation.
  969. //
  970. RtlFillMemory(ansiString, sizeof(ansiString), 0);
  971. EngUnicodeToMultiByteN(ansiString,
  972. maxLength,
  973. &ansiLen,
  974. pstro->pwszOrg,
  975. pstro->cGlyphs * sizeof(WCHAR));
  976. //
  977. // The conversion claims it never fails, but we have seen results that
  978. // are completely different on the remote box. So we convert the ANSI
  979. // string back to UNICODE and check that we still have what we started
  980. // with.
  981. //
  982. EngMultiByteToUnicodeN(g_oeTempString,
  983. sizeof(g_oeTempString),
  984. &tempLen,
  985. ansiString,
  986. ansiLen);
  987. //
  988. // Check we don't have too much data, or that the translation failed to
  989. // give the correct data. This happens when we try to translate
  990. // UNICODE text.
  991. //
  992. if ( (tempLen != pstro->cGlyphs * sizeof(WCHAR)) ||
  993. (memcmp(pstro->pwszOrg, g_oeTempString, tempLen) != 0) )
  994. {
  995. TRACE_OUT(( "String not translated"));
  996. DC_QUIT;
  997. }
  998. //
  999. // Check that the font is valid.
  1000. //
  1001. if (!OECheckFontIsSupported(pfo, ansiString, ansiLen,
  1002. &fontHeight,
  1003. &fontAscender,
  1004. &fontWidth,
  1005. &fontWeight,
  1006. &fontFlags,
  1007. &fontIndex,
  1008. &sendDeltaX))
  1009. {
  1010. TRACE_OUT(( "Unsupported font for '%s'", ansiString));
  1011. //
  1012. // Check if there is an opaque rectangle. If so it is worth
  1013. // splitting this out. Word can output an entire line comprising a
  1014. // single character followed by background, eg bullets, where the
  1015. // line is blanked by drawing the bullet character at the start of
  1016. // the line followed by a large - >1000 pixel - opaque rect.
  1017. // Splitting the opaque rect from the text means we can send a
  1018. // small area of SD for the unmatched font char while encoding the
  1019. // large opaque rectangle.
  1020. //
  1021. if ( (prclOpaque != NULL) &&
  1022. (pstro->cGlyphs == 1) &&
  1023. (pstro->flAccel & SO_HORIZONTAL) &&
  1024. OE_SendAsOrder(ORD_PATBLT))
  1025. {
  1026. //
  1027. // There is an opaque rectangle and a single char.
  1028. // Encode the opaque rectangle. First get a copy of the target
  1029. // rect so we can use it later (and flip it into screen
  1030. // coordinates).
  1031. //
  1032. TRACE_OUT(( "Have 1 char + opaque rect"));
  1033. rectText.left = rectDst.left;
  1034. rectText.top = rectDst.top;
  1035. rectText.right = rectDst.right + 1;
  1036. rectText.bottom = rectDst.bottom + 1;
  1037. //
  1038. // Call into the PATBLT encoding function.
  1039. //
  1040. if ( !OEEncodePatBlt(ppdev,
  1041. pboOpaque,
  1042. pptlOrg,
  1043. OE_COPYPEN_ROP,
  1044. &rectDst,
  1045. &pOpaqueOrder) )
  1046. {
  1047. //
  1048. // Something went wrong with the encoding, so skip to the
  1049. // end to add this operation to the SDA.
  1050. //
  1051. TRACE_OUT(( "Failed to encode opaque rect"));
  1052. DC_QUIT;
  1053. }
  1054. //
  1055. // Store the general order data. The bounding rectagle
  1056. // co-ordinates must be virtual desktop. OELRtoVirtual has
  1057. // already converted rect for us.
  1058. //
  1059. TSHR_RECT16_FROM_RECT(&pOpaqueOrder->OrderHeader.Common.rcsDst, rectDst);
  1060. //
  1061. // Add the order to the cache.
  1062. //
  1063. OEClipAndAddOrder(pOpaqueOrder, NULL, pco);
  1064. //
  1065. // Calculate the bounds of the text. Get the glyph positions
  1066. // for the left and right, and assume the top and bottom equate
  1067. // approximately to the opaque rectangle.
  1068. //
  1069. if ( pstro->pgp == NULL)
  1070. {
  1071. //
  1072. // The string object doesn't contain the GLYPHPOS info, so
  1073. // enumerate the glyphs.
  1074. //
  1075. TRACE_OUT(( "Enumerate glyphs"));
  1076. STROBJ_vEnumStart(pstro);
  1077. STROBJ_bEnum(pstro, &count, &pGlyphData);
  1078. }
  1079. else
  1080. {
  1081. //
  1082. // The string object already contains the GLYPHPOS info, so
  1083. // just grab the pointer to it.
  1084. //
  1085. pGlyphData = pstro->pgp;
  1086. }
  1087. rectDst = rectText;
  1088. rectDst.left = max(rectDst.left, pGlyphData[0].ptl.x);
  1089. if ( pstro->ulCharInc == 0 )
  1090. {
  1091. //
  1092. // No character increment for this string object. Just use
  1093. // the maximum glyph width to calculate the right bounding
  1094. // edge.
  1095. //
  1096. TRACE_OUT(( "no charinc glyph %d trg %d left %d maxX %d",
  1097. pGlyphData[0].ptl.x,
  1098. rectDst.right,
  1099. rectDst.left,
  1100. pfo->cxMax));
  1101. rectDst.right = min(rectDst.right, (int)(pGlyphData[0].ptl.x +
  1102. pfo->cxMax - 1));
  1103. }
  1104. else
  1105. {
  1106. //
  1107. // The string object has a character increment, so use it
  1108. // to determine the right bounding edge.
  1109. //
  1110. TRACE_OUT(( "charinc %x glyph %d trg %d left %d",
  1111. pstro->ulCharInc,
  1112. pGlyphData[0].ptl.x,
  1113. rectDst.right,
  1114. rectDst.left));
  1115. rectDst.right = min(rectDst.right, (int)(pGlyphData[0].ptl.x +
  1116. pstro->ulCharInc - 1));
  1117. }
  1118. //
  1119. // Flip the target rectangle back to virtual coordinates.
  1120. //
  1121. rectDst.right -= 1;
  1122. rectDst.bottom -= 1;
  1123. }
  1124. //
  1125. // Skip to the end to add to the SDA.
  1126. //
  1127. DC_QUIT;
  1128. }
  1129. //
  1130. // It is possible that the font matching blows our previous decision to
  1131. // generate a TextOut order and we need to generate an ExtTextOut order
  1132. // instead. We need to reverify our parameters if this is the case.
  1133. //
  1134. if ((sendDeltaX) && (orderType != ORD_EXTTEXTOUT))
  1135. {
  1136. TRACE_OUT(( "Text order must be EXTTEXTOUT"));
  1137. //
  1138. // Set up for ExtTexOut orders.
  1139. //
  1140. orderType = ORD_EXTTEXTOUT;
  1141. maxLength = ORD_MAX_STRING_LEN_WITH_DELTAS;
  1142. //
  1143. // Check if we are allowed to send this order (determined by the
  1144. // negotiated capabilities of all the machines in the conference).
  1145. //
  1146. if (!OE_SendAsOrder(orderType))
  1147. {
  1148. TRACE_OUT(( "Text order %x not allowed", orderType));
  1149. DC_QUIT;
  1150. }
  1151. //
  1152. // Make sure we haven't blown the order size.
  1153. //
  1154. if (pstro->cGlyphs > maxLength)
  1155. {
  1156. TRACE_OUT(( "Text limit blown", pstro->cGlyphs));
  1157. DC_QUIT;
  1158. }
  1159. }
  1160. //
  1161. // Get the proper start position for the text.
  1162. //
  1163. if ( pstro->pgp == NULL)
  1164. {
  1165. STROBJ_vEnumStart(pstro);
  1166. STROBJ_bEnum(pstro, &count, &pGlyphData);
  1167. if (count == 0)
  1168. {
  1169. WARNING_OUT(( "No glyphs"));
  1170. DC_QUIT;
  1171. }
  1172. }
  1173. else
  1174. {
  1175. pGlyphData = pstro->pgp;
  1176. }
  1177. startPoint.x = pGlyphData[0].ptl.x;
  1178. //
  1179. // Check if we should be using baseline alignment for the y
  1180. // coordinate. If we should be, the value in the glyph data is
  1181. // correct. If not, we the y coordinate is for the top of the
  1182. // text, and we have to calculate it.
  1183. //
  1184. if (g_oeBaselineTextEnabled)
  1185. {
  1186. startPoint.y = pGlyphData[0].ptl.y;
  1187. fontFlags |= NF_BASELINE;
  1188. }
  1189. else
  1190. {
  1191. startPoint.y = pGlyphData[0].ptl.y - fontAscender;
  1192. }
  1193. //
  1194. // Allocate memory for the order
  1195. //
  1196. switch (orderType)
  1197. {
  1198. case ORD_TEXTOUT:
  1199. {
  1200. //
  1201. // Allocate the memory
  1202. //
  1203. pOrder = OA_DDAllocOrderMem((UINT)( sizeof(TEXTOUT_ORDER)
  1204. - ORD_MAX_STRING_LEN_WITHOUT_DELTAS
  1205. + ansiLen ),
  1206. 0);
  1207. if (pOrder == NULL)
  1208. {
  1209. TRACE_OUT(( "Failed to alloc order"));
  1210. DC_QUIT;
  1211. }
  1212. pTextOut = (LPTEXTOUT_ORDER)pOrder->abOrderData;
  1213. //
  1214. // Set up the order type.
  1215. //
  1216. pTextOut->type = ORD_TEXTOUT_TYPE;
  1217. //
  1218. // Get a pointer to the fields which are common to both TextOut
  1219. // and ExtTextOut
  1220. //
  1221. pCommon = &pTextOut->common;
  1222. }
  1223. break;
  1224. case ORD_EXTTEXTOUT:
  1225. {
  1226. //
  1227. // BOGUS LAURABU
  1228. // This allocates space for a deltax array whether or not one is needed
  1229. //
  1230. //
  1231. // Allocate the memory
  1232. //
  1233. pOrder = OA_DDAllocOrderMem((UINT)( sizeof(EXTTEXTOUT_ORDER)
  1234. - ORD_MAX_STRING_LEN_WITHOUT_DELTAS
  1235. - (ORD_MAX_STRING_LEN_WITH_DELTAS
  1236. * sizeof(TSHR_INT32))
  1237. + ansiLen * (sizeof(TSHR_INT32) + 1)
  1238. + 4), // Allow for internal padding
  1239. 0);
  1240. if (pOrder == NULL)
  1241. {
  1242. TRACE_OUT(( "Failed to alloc order"));
  1243. DC_QUIT;
  1244. }
  1245. pExtTextOut = (LPEXTTEXTOUT_ORDER)pOrder->abOrderData;
  1246. //
  1247. // Set up the order type.
  1248. //
  1249. pExtTextOut->type = ORD_EXTTEXTOUT_TYPE;
  1250. //
  1251. // Get a pointer to the fields which are common to both TextOut
  1252. // and ExtTextOut
  1253. //
  1254. pCommon = &pExtTextOut->common;
  1255. }
  1256. break;
  1257. default:
  1258. {
  1259. ERROR_OUT(( "Unknown order %x", orderType));
  1260. DC_QUIT;
  1261. }
  1262. break;
  1263. }
  1264. //
  1265. // Fill in the fields which are common to both TextOut and ExtTextOut
  1266. //
  1267. // Convert to virtual coordinates
  1268. //
  1269. OELPtoVirtual(&startPoint, 1);
  1270. //
  1271. // The x and y values are available in virtual coords from the bounds
  1272. // rectangle.
  1273. //
  1274. pCommon->nXStart = startPoint.x;
  1275. pCommon->nYStart = startPoint.y;
  1276. //
  1277. // Get the text colours.
  1278. //
  1279. OEConvertColor(ppdev,
  1280. &pCommon->BackColor,
  1281. pboOpaque->iSolidColor,
  1282. NULL);
  1283. OEConvertColor(ppdev,
  1284. &pCommon->ForeColor,
  1285. pboFore->iSolidColor,
  1286. NULL);
  1287. //
  1288. // The transparency of the operation is determined by whether we have
  1289. // an opaque rectangle or not.
  1290. //
  1291. pCommon->BackMode = (prclOpaque == NULL) ? TRANSPARENT : OPAQUE;
  1292. //
  1293. // NT has a character extra spacing, not a generic for every character
  1294. // spacing. So, we always set this value to 0.
  1295. //
  1296. pCommon->CharExtra = 0;
  1297. //
  1298. // NT does not provide a break of any sorts.
  1299. //
  1300. pCommon->BreakExtra = 0;
  1301. pCommon->BreakCount = 0;
  1302. //
  1303. // Copy the font details
  1304. //
  1305. pCommon->FontHeight = fontHeight;
  1306. pCommon->FontWidth = fontWidth;
  1307. pCommon->FontWeight = fontWeight;
  1308. pCommon->FontFlags = fontFlags;
  1309. pCommon->FontIndex = fontIndex;
  1310. //
  1311. // Now fill in the order specific data
  1312. //
  1313. switch (orderType)
  1314. {
  1315. case ORD_TEXTOUT:
  1316. //
  1317. // Copy across the text string.
  1318. //
  1319. pTextOut->variableString.len = (BYTE)ansiLen;
  1320. RtlCopyMemory(pTextOut->variableString.string,
  1321. ansiString,
  1322. ansiLen);
  1323. //
  1324. // Make sure we send the order
  1325. //
  1326. fSendOrder = TRUE;
  1327. TRACE_OUT(( "TEXTOUT: X %u Y %u bm %u FC %02X%02X%02X "
  1328. "BC %02X%02X%02X",
  1329. pTextOut->common.nXStart,
  1330. pTextOut->common.nYStart,
  1331. pTextOut->common.BackMode,
  1332. pTextOut->common.ForeColor.red,
  1333. pTextOut->common.ForeColor.green,
  1334. pTextOut->common.ForeColor.blue,
  1335. pTextOut->common.BackColor.red,
  1336. pTextOut->common.BackColor.green,
  1337. pTextOut->common.BackColor.blue));
  1338. TRACE_OUT(( "Font: fx %u fy %u fw %u ff %04x fh %u len %u",
  1339. pTextOut->common.FontWidth,
  1340. pTextOut->common.FontHeight,
  1341. pTextOut->common.FontWeight,
  1342. pTextOut->common.FontFlags,
  1343. pTextOut->common.FontIndex,
  1344. ansiLen));
  1345. TRACE_OUT(( "String '%s'", ansiString));
  1346. break;
  1347. case ORD_EXTTEXTOUT:
  1348. //
  1349. // Since our text is only ever fully contained within the
  1350. // opaque rectangle, we only set the opaque flag (and ignore
  1351. // the clipping).
  1352. //
  1353. pExtTextOut->fuOptions = (prclOpaque == NULL) ? 0 : ETO_OPAQUE;
  1354. //
  1355. // Set up the bounding rectangle for the operation.
  1356. // EXT_TEXT_OUT orders use TSHR_RECT32s, hence we can't directly
  1357. // assign rectDst to it.
  1358. //
  1359. pExtTextOut->rectangle.left = rectDst.left;
  1360. pExtTextOut->rectangle.top = rectDst.top;
  1361. pExtTextOut->rectangle.right = rectDst.right;
  1362. pExtTextOut->rectangle.bottom = rectDst.bottom;
  1363. //
  1364. // Copy across the text string.
  1365. //
  1366. pExtTextOut->variableString.len = ansiLen;
  1367. RtlCopyMemory(pExtTextOut->variableString.string,
  1368. ansiString,
  1369. ansiLen);
  1370. //
  1371. // WHOOP WHOOP WHOOP - Prepare to shut your eyes...
  1372. //
  1373. // Although we have a defined fixed length structure for
  1374. // storing ExtTextOut orders, we must not send the full
  1375. // structure over the network as the text will only be, say, 10
  1376. // characters while the structure contains room for 127.
  1377. //
  1378. // Hence we pack the structure now to remove all the blank data
  1379. // BUT we must maintain the natural alignment of the variables.
  1380. //
  1381. // So we know the length of the string which we can use to
  1382. // start the new delta structure at the next 4-byte boundary.
  1383. //
  1384. lpVariable = ((LPBYTE)(&pExtTextOut->variableString))
  1385. + ansiLen
  1386. + sizeof(pExtTextOut->variableString.len);
  1387. lpVariable = (LPSTR)
  1388. DC_ROUND_UP_4((UINT_PTR)lpVariable);
  1389. lpDeltaPos = (LPVARIABLE_DELTAX)lpVariable;
  1390. //
  1391. // Do we need a delta array, or are the chars at their default
  1392. // positions.
  1393. //
  1394. if ( sendDeltaX ||
  1395. ((pstro->flAccel & SO_FLAG_DEFAULT_PLACEMENT) == 0) )
  1396. {
  1397. //
  1398. // Store the length of the position deltas.
  1399. //
  1400. lpDeltaPos->len = ansiLen * sizeof(TSHR_INT32);
  1401. //
  1402. // Set up the position deltas.
  1403. //
  1404. STROBJ_vEnumStart(pstro);
  1405. fMoreData = TRUE;
  1406. currentDelta = 0;
  1407. while (fMoreData)
  1408. {
  1409. //
  1410. // Get the next set of glyph data
  1411. //
  1412. fMoreData = STROBJ_bEnum(pstro, &count, &pGlyphData);
  1413. for (i = 0; i < count; i++)
  1414. {
  1415. //
  1416. // The first time through we must set up the first
  1417. // glyph position.
  1418. //
  1419. if ((currentDelta == 0) && (i == 0))
  1420. {
  1421. lastPtl.x = pGlyphData[0].ptl.x;
  1422. lastPtl.y = pGlyphData[0].ptl.y;
  1423. TRACE_OUT(( "First Pos %d", lastPtl.x));
  1424. }
  1425. else
  1426. {
  1427. //
  1428. // For subsequent entries, we need to add the
  1429. // delta on the X position to the array.
  1430. //
  1431. if (pstro->ulCharInc == 0)
  1432. {
  1433. lpDeltaPos->deltaX[currentDelta]
  1434. = pGlyphData[i].ptl.x
  1435. - lastPtl.x;
  1436. //
  1437. // Check for delta Y's - which we can't
  1438. // encode
  1439. //
  1440. if (pGlyphData[i].ptl.y - lastPtl.y)
  1441. {
  1442. WARNING_OUT(( "New Y %d",
  1443. pGlyphData[i].ptl.y));
  1444. OA_DDFreeOrderMem(pOrder);
  1445. DC_QUIT;
  1446. }
  1447. //
  1448. // Store the last position for the next
  1449. // time round.
  1450. //
  1451. lastPtl.x = pGlyphData[i].ptl.x;
  1452. lastPtl.y = pGlyphData[i].ptl.y;
  1453. TRACE_OUT(( "Next Pos %d %d", i, lastPtl.x));
  1454. }
  1455. else
  1456. {
  1457. lpDeltaPos->deltaX[currentDelta]
  1458. = pstro->ulCharInc;
  1459. }
  1460. currentDelta++;
  1461. }
  1462. }
  1463. }
  1464. //
  1465. // For the last entry, we need to set up the data by hand
  1466. // (there are only n-1 deltas for n chars)
  1467. //
  1468. // This is done for compatibility with Windows 95 which
  1469. // requires the last delta to be the delta to the place
  1470. // where the next char would be if there were n+1 chars in
  1471. // the string.
  1472. //
  1473. if (pstro->ulCharInc == 0)
  1474. {
  1475. //
  1476. // No characters left - fudge a value of the width of
  1477. // the last character.
  1478. //
  1479. lpDeltaPos->deltaX[currentDelta] =
  1480. pGlyphData[count-1].pgdf->pgb->sizlBitmap.cx;
  1481. }
  1482. else
  1483. {
  1484. //
  1485. // All chars are evenly spaced, so just stick the value
  1486. // in.
  1487. //
  1488. lpDeltaPos->deltaX[currentDelta] = pstro->ulCharInc;
  1489. }
  1490. //
  1491. // WHOOP WHOOP WHOOP - You can open your eyes now...
  1492. //
  1493. //
  1494. // We must indicate the presence of this field to the
  1495. // receiver.
  1496. //
  1497. pExtTextOut->fuOptions |= ETO_LPDX;
  1498. }
  1499. else
  1500. {
  1501. //
  1502. // Mark the delta array as empty.
  1503. //
  1504. lpDeltaPos->len = 0;
  1505. }
  1506. //
  1507. // WHOOP WHOOP WHOOP - You can open your eyes now...
  1508. //
  1509. //
  1510. // Make sure we send the order
  1511. //
  1512. fSendOrder = TRUE;
  1513. TRACE_OUT(( "EXTTEXTOUT: X %u Y %u bm %u FC %02X%02X%02X "
  1514. "BC %02X%02X%02X",
  1515. pExtTextOut->common.nXStart,
  1516. pExtTextOut->common.nYStart,
  1517. pExtTextOut->common.BackMode,
  1518. pExtTextOut->common.ForeColor.red,
  1519. pExtTextOut->common.ForeColor.green,
  1520. pExtTextOut->common.ForeColor.blue,
  1521. pExtTextOut->common.BackColor.red,
  1522. pExtTextOut->common.BackColor.green,
  1523. pExtTextOut->common.BackColor.blue));
  1524. TRACE_OUT(( "Extra: Opt %x X1 %d Y1 %d X2 %d Y2 %d",
  1525. pExtTextOut->fuOptions,
  1526. pExtTextOut->rectangle.left,
  1527. pExtTextOut->rectangle.top,
  1528. pExtTextOut->rectangle.right,
  1529. pExtTextOut->rectangle.bottom));
  1530. TRACE_OUT(( "Font: fx %u fy %u fw %u ff %04x fh %u len %u",
  1531. pExtTextOut->common.FontWidth,
  1532. pExtTextOut->common.FontHeight,
  1533. pExtTextOut->common.FontWeight,
  1534. pExtTextOut->common.FontFlags,
  1535. pExtTextOut->common.FontIndex,
  1536. ansiLen));
  1537. TRACE_OUT(( "String '%s'", ansiString));
  1538. break;
  1539. default:
  1540. ERROR_OUT(( "Unknown order %x", orderType));
  1541. break;
  1542. }
  1543. DC_EXIT_POINT:
  1544. //
  1545. // If we did not send an order, we must accumulate the output in the
  1546. // Screen Data Area.
  1547. //
  1548. if (fSendOrder)
  1549. {
  1550. //
  1551. // Store the general order data. The bounding rectangle position
  1552. // must be in virtual desktop co-ordinates. OELRtoVirtual has
  1553. // already done this.
  1554. //
  1555. pOrder->OrderHeader.Common.fOrderFlags = OF_SPOILABLE;
  1556. TSHR_RECT16_FROM_RECT(&pOrder->OrderHeader.Common.rcsDst, rectDst);
  1557. //
  1558. // Add the order to the cache.
  1559. //
  1560. OEClipAndAddOrder(pOrder, NULL, pco);
  1561. }
  1562. else
  1563. {
  1564. if (fAccumulate)
  1565. {
  1566. OEClipAndAddScreenData(&rectDst, pco);
  1567. }
  1568. }
  1569. OE_SHM_STOP_WRITING;
  1570. NO_LOCK_EXIT:
  1571. DebugExitDWORD(DrvTextOut, rc);
  1572. return(rc);
  1573. }
  1574. //
  1575. // DrvLineTo - see NT DDK documentation.
  1576. //
  1577. BOOL DrvLineTo(SURFOBJ *pso,
  1578. CLIPOBJ *pco,
  1579. BRUSHOBJ *pbo,
  1580. LONG x1,
  1581. LONG y1,
  1582. LONG x2,
  1583. LONG y2,
  1584. RECTL *prclBounds,
  1585. MIX mix)
  1586. {
  1587. LPOSI_PDEV ppdev = (LPOSI_PDEV)pso->dhpdev;
  1588. BOOL rc = TRUE;
  1589. RECT rectDst;
  1590. POINT startPoint;
  1591. POINT endPoint;
  1592. BOOL fAccumulate = FALSE;
  1593. DebugEntry(DrvLineTo);
  1594. //
  1595. // DO THIS _BEFORE_ TAKING LOCK
  1596. //
  1597. if (!g_oeViewers)
  1598. goto NO_LOCK_EXIT;
  1599. OE_SHM_START_WRITING;
  1600. //
  1601. // Get bounding rectangle and convert to a RECT.
  1602. //
  1603. RECT_FROM_RECTL(rectDst, (*prclBounds));
  1604. //
  1605. // Check if we are accumulating data for this function
  1606. //
  1607. fAccumulate = OEAccumulateOutput(pso, pco, &rectDst);
  1608. if (!fAccumulate)
  1609. {
  1610. DC_QUIT;
  1611. }
  1612. //
  1613. // Convert the data to virtual coordinates.
  1614. //
  1615. OELRtoVirtual(&rectDst, 1);
  1616. //
  1617. // Check if we are allowed to send this order (determined by the
  1618. // negotiated capabilities of all the machines in the conference).
  1619. //
  1620. if (!OE_SendAsOrder(ORD_LINETO))
  1621. {
  1622. TRACE_OUT(( "LineTo order not allowed"));
  1623. DC_QUIT;
  1624. }
  1625. //
  1626. // Check for a valid brush for the test operation.
  1627. //
  1628. if (pbo->iSolidColor == -1)
  1629. {
  1630. TRACE_OUT(( "Bad brush for line"));
  1631. DC_QUIT;
  1632. }
  1633. //
  1634. // Check for overcomplicated clipping.
  1635. //
  1636. if (OEClippingIsComplex(pco))
  1637. {
  1638. TRACE_OUT(( "Clipping is too complex"));
  1639. DC_QUIT;
  1640. }
  1641. //
  1642. // Set up data for order
  1643. //
  1644. startPoint.x = x1;
  1645. startPoint.y = y1;
  1646. endPoint.x = x2;
  1647. endPoint.y = y2;
  1648. //
  1649. // Store that order.
  1650. //
  1651. if (!OEAddLine(ppdev,
  1652. &startPoint,
  1653. &endPoint,
  1654. &rectDst,
  1655. mix & 0x1F,
  1656. 1,
  1657. pbo->iSolidColor,
  1658. pco))
  1659. {
  1660. TRACE_OUT(( "Failed to add order - use SDA"));
  1661. DC_QUIT;
  1662. }
  1663. //
  1664. // We have stored this object, so don't store the data in the SDA
  1665. // again.
  1666. //
  1667. fAccumulate = FALSE;
  1668. DC_EXIT_POINT:
  1669. if (fAccumulate)
  1670. {
  1671. OEClipAndAddScreenData(&rectDst, pco);
  1672. }
  1673. OE_SHM_STOP_WRITING;
  1674. NO_LOCK_EXIT:
  1675. DebugExitDWORD(DrvLineTo, rc);
  1676. return(rc);
  1677. }
  1678. //
  1679. // DrvStrokePath - see NT DDK documentation.
  1680. //
  1681. BOOL DrvStrokePath(SURFOBJ *pso,
  1682. PATHOBJ *ppo,
  1683. CLIPOBJ *pco,
  1684. XFORMOBJ *pxo,
  1685. BRUSHOBJ *pbo,
  1686. POINTL *pptlBrushOrg,
  1687. LINEATTRS *plineattrs,
  1688. MIX mix)
  1689. {
  1690. LPOSI_PDEV ppdev = (LPOSI_PDEV)pso->dhpdev;
  1691. BOOL rc = TRUE;
  1692. RECTFX rectfxTrg;
  1693. RECT rectDst;
  1694. BOOL fMore = TRUE;
  1695. PATHDATA pathData;
  1696. POINT startPoint;
  1697. POINT nextPoint;
  1698. POINT endPoint;
  1699. BOOL fAccumulate = FALSE;
  1700. UINT i;
  1701. DebugEntry(DrvStrokePath);
  1702. //
  1703. // DO THIS _BEFORE_ TAKING LOCK
  1704. //
  1705. if (!g_oeViewers)
  1706. goto NO_LOCK_EXIT;
  1707. OE_SHM_START_WRITING;
  1708. //
  1709. // Get bounding rectangle and convert to a RECT.
  1710. //
  1711. PATHOBJ_vGetBounds(ppo, &rectfxTrg);
  1712. RECT_FROM_RECTFX(rectDst, rectfxTrg);
  1713. //
  1714. // Check if we are accumulating data for this function
  1715. //
  1716. fAccumulate = OEAccumulateOutput(pso, pco, &rectDst);
  1717. if (!fAccumulate)
  1718. {
  1719. DC_QUIT;
  1720. }
  1721. //
  1722. // Check if we are allowed to send this order (determined by the
  1723. // negotiated capabilities of all the machines in the conference).
  1724. //
  1725. if (!OE_SendAsOrder(ORD_LINETO))
  1726. {
  1727. TRACE_OUT(( "LineTo order not allowed"));
  1728. DC_QUIT;
  1729. }
  1730. //
  1731. // Check for a valid brush for the test operation.
  1732. //
  1733. if (pbo->iSolidColor == -1)
  1734. {
  1735. TRACE_OUT(( "Bad brush for line"));
  1736. DC_QUIT;
  1737. }
  1738. //
  1739. // Check for overcomplicated clipping.
  1740. //
  1741. if (OEClippingIsComplex(pco))
  1742. {
  1743. TRACE_OUT(( "Clipping is too complex"));
  1744. DC_QUIT;
  1745. }
  1746. //
  1747. // See if we can optimize the path...
  1748. //
  1749. // We cannot send:
  1750. //
  1751. // - curvy lines (i.e. beziers)
  1752. // - lines with funny end patterns (i.e. geometric lines)
  1753. // - non Windows standard patterns
  1754. //
  1755. if ( ((ppo->fl & PO_BEZIERS) == 0) &&
  1756. ((plineattrs->fl & LA_GEOMETRIC) == 0) &&
  1757. (plineattrs->pstyle == NULL) )
  1758. {
  1759. //
  1760. // This is a set of solid cosmetic (i.e. no fancy end styles)
  1761. // lines. Let's send those orders.
  1762. //
  1763. // NT stores all paths as a set of independent sub-paths. Each
  1764. // sub-path can start at a new point that is NOT linked to the
  1765. // previous sub-path.
  1766. //
  1767. // Paths used for this function (as opposed to DrvFillPath or
  1768. // DrvStrokeAndFillPath) do not need to be closed.
  1769. //
  1770. PATHOBJ_vEnumStart(ppo);
  1771. while (fMore)
  1772. {
  1773. //
  1774. // Get the next set of lines
  1775. //
  1776. fMore = PATHOBJ_bEnum(ppo, &pathData);
  1777. TRACE_OUT(( "PTS: %lu FLAG: %08lx",
  1778. pathData.count,
  1779. pathData.flags));
  1780. //
  1781. // If this is the start of a path, remember the point in case
  1782. // we need to close the path at the end.
  1783. //
  1784. if (pathData.flags & PD_BEGINSUBPATH)
  1785. {
  1786. POINT_FROM_POINTFIX(startPoint, pathData.pptfx[0]);
  1787. POINT_FROM_POINTFIX(nextPoint, pathData.pptfx[0]);
  1788. }
  1789. //
  1790. // Generate orders for each line in the path.
  1791. //
  1792. for (i = 0; i < pathData.count; i++)
  1793. {
  1794. POINT_FROM_POINTFIX(endPoint, pathData.pptfx[i]);
  1795. if ( (nextPoint.x != endPoint.x) ||
  1796. (nextPoint.y != endPoint.y) )
  1797. {
  1798. if (!OEAddLine(ppdev,
  1799. &nextPoint,
  1800. &endPoint,
  1801. &rectDst,
  1802. mix & 0x1f,
  1803. plineattrs->elWidth.l,
  1804. pbo->iSolidColor,
  1805. pco))
  1806. {
  1807. DC_QUIT;
  1808. }
  1809. }
  1810. nextPoint.x = endPoint.x;
  1811. nextPoint.y = endPoint.y;
  1812. }
  1813. //
  1814. // Close the path if necessary.
  1815. //
  1816. if ((pathData.flags & PD_CLOSEFIGURE) != 0)
  1817. {
  1818. if (!OEAddLine(ppdev,
  1819. &endPoint,
  1820. &startPoint,
  1821. &rectDst,
  1822. mix & 0x1f,
  1823. plineattrs->elWidth.l,
  1824. pbo->iSolidColor,
  1825. pco))
  1826. {
  1827. DC_QUIT;
  1828. }
  1829. }
  1830. }
  1831. //
  1832. // We have processed the entire thing as orders - don't send screen
  1833. // data.
  1834. //
  1835. fAccumulate = FALSE;
  1836. }
  1837. DC_EXIT_POINT:
  1838. if (fAccumulate)
  1839. {
  1840. //
  1841. // Convert the bounds to virtual coordinates.
  1842. //
  1843. OELRtoVirtual(&rectDst, 1);
  1844. TRACE_OUT(( "SDA: (%d,%d)(%d,%d)",
  1845. rectDst.left,
  1846. rectDst.top,
  1847. rectDst.right,
  1848. rectDst.bottom));
  1849. //
  1850. // Update the Screen Data Area
  1851. //
  1852. OEClipAndAddScreenData(&rectDst, pco);
  1853. }
  1854. OE_SHM_STOP_WRITING;
  1855. NO_LOCK_EXIT:
  1856. DebugExitDWORD(DrvStrokePath, rc);
  1857. return(rc);
  1858. }
  1859. //
  1860. // DrvFillPath - see NT DDK documentation.
  1861. //
  1862. BOOL DrvFillPath(SURFOBJ *pso,
  1863. PATHOBJ *ppo,
  1864. CLIPOBJ *pco,
  1865. BRUSHOBJ *pbo,
  1866. POINTL *pptlBrushOrg,
  1867. MIX mix,
  1868. FLONG flOptions)
  1869. {
  1870. BOOL rc = TRUE;
  1871. RECTFX rectfxTrg;
  1872. RECT rectDst;
  1873. DebugEntry(DrvFillPath);
  1874. //
  1875. // DO THIS _BEFORE_ TAKING LOCK
  1876. //
  1877. if (!g_oeViewers)
  1878. goto NO_LOCK_EXIT;
  1879. OE_SHM_START_WRITING;
  1880. //
  1881. // Get bounding rectangle and convert to a RECT.
  1882. //
  1883. PATHOBJ_vGetBounds(ppo, &rectfxTrg);
  1884. RECT_FROM_RECTFX(rectDst, rectfxTrg);
  1885. //
  1886. // Check if we are accumulating data for this function
  1887. //
  1888. if (!OEAccumulateOutput(pso, pco, &rectDst))
  1889. {
  1890. DC_QUIT;
  1891. }
  1892. //
  1893. // Convert the bounds to virtual coordinates.
  1894. //
  1895. OELRtoVirtual(&rectDst, 1);
  1896. TRACE_OUT(( "SDA: (%d,%d)(%d,%d)",
  1897. rectDst.left,
  1898. rectDst.top,
  1899. rectDst.right,
  1900. rectDst.bottom));
  1901. //
  1902. // Update the Screen Data Area
  1903. //
  1904. OEClipAndAddScreenData(&rectDst, pco);
  1905. DC_EXIT_POINT:
  1906. OE_SHM_STOP_WRITING;
  1907. NO_LOCK_EXIT:
  1908. DebugExitDWORD(DrvFillPath, rc);
  1909. return(rc);
  1910. }
  1911. //
  1912. // DrvStrokeAndFillPath - see NT DDK documentation.
  1913. //
  1914. BOOL DrvStrokeAndFillPath(SURFOBJ *pso,
  1915. PATHOBJ *ppo,
  1916. CLIPOBJ *pco,
  1917. XFORMOBJ *pxo,
  1918. BRUSHOBJ *pboStroke,
  1919. LINEATTRS *plineattrs,
  1920. BRUSHOBJ *pboFill,
  1921. POINTL *pptlBrushOrg,
  1922. MIX mixFill,
  1923. FLONG flOptions)
  1924. {
  1925. BOOL rc = TRUE;
  1926. RECTFX rectfxTrg;
  1927. RECT rectDst;
  1928. DebugEntry(DrvStrokeAndFillPath);
  1929. //
  1930. // DO THIS _BEFORE_ TAKING LOCK
  1931. //
  1932. if (!g_oeViewers)
  1933. goto NO_LOCK_EXIT;
  1934. OE_SHM_START_WRITING;
  1935. //
  1936. // Get bounding rectangle and convert to a RECT.
  1937. //
  1938. PATHOBJ_vGetBounds(ppo, &rectfxTrg);
  1939. RECT_FROM_RECTFX(rectDst, rectfxTrg);
  1940. //
  1941. // Check if we are accumulating data for this function
  1942. //
  1943. if (!OEAccumulateOutput(pso, pco, &rectDst))
  1944. {
  1945. DC_QUIT;
  1946. }
  1947. //
  1948. // Convert the bounds to virtual coordinates.
  1949. //
  1950. OELRtoVirtual(&rectDst, 1);
  1951. //
  1952. // Update the Screen Data Area
  1953. //
  1954. OEClipAndAddScreenData(&rectDst, pco);
  1955. DC_EXIT_POINT:
  1956. OE_SHM_STOP_WRITING;
  1957. NO_LOCK_EXIT:
  1958. DebugExitDWORD(DrvStrokeAndFillPath, rc);
  1959. return(rc);
  1960. }
  1961. //
  1962. // DrvPaint - see NT DDK documentation.
  1963. //
  1964. BOOL DrvPaint(SURFOBJ *pso,
  1965. CLIPOBJ *pco,
  1966. BRUSHOBJ *pbo,
  1967. POINTL *pptlBrushOrg,
  1968. MIX mix)
  1969. {
  1970. BOOL rc = TRUE;
  1971. RECT rectDst;
  1972. BOOL fAccumulate = FALSE;
  1973. ROP4 rop4;
  1974. DebugEntry(DrvPaint);
  1975. //
  1976. // DO THIS _BEFORE_ TAKING LOCK
  1977. //
  1978. if (!g_oeViewers)
  1979. goto NO_LOCK_EXIT;
  1980. OE_SHM_START_WRITING;
  1981. //
  1982. // Get bounding rectangle and convert to a RECT.
  1983. //
  1984. RECT_FROM_RECTL(rectDst, pco->rclBounds);
  1985. //
  1986. // Check if we are accumulating data for this function
  1987. //
  1988. fAccumulate = OEAccumulateOutput(pso, pco, &rectDst);
  1989. if (!fAccumulate)
  1990. {
  1991. DC_QUIT;
  1992. }
  1993. //
  1994. // Convert to virtual coordinates.
  1995. //
  1996. OELRtoVirtual(&rectDst, 1);
  1997. //
  1998. // Check for overcomplicated clipping.
  1999. //
  2000. if (OEClippingIsComplex(pco))
  2001. {
  2002. TRACE_OUT(( "Clipping is too complex"));
  2003. DC_QUIT;
  2004. }
  2005. //
  2006. // The low byte of the mix represents a ROP2. We need a ROP4 for
  2007. // BitBlt, so convert the mix as follows.
  2008. //
  2009. // Remember the definitions of 2, 3 & 4 way ROP codes.
  2010. //
  2011. // Msk Pat Src Dst
  2012. //
  2013. // 1 1 1 1 �������͸�����ͻ ROP2 uses P & D only
  2014. // 1 1 1 0 � �
  2015. // 1 1 0 1 Ŀ � � ROP3 uses P, S & D
  2016. // 1 1 0 0 �ROP2-1�ROP3 �ROP4
  2017. // 1 0 1 1 �(see � � ROP4 uses M, P, S & D
  2018. // 1 0 1 0 �� note)� �
  2019. // 1 0 0 1 � �
  2020. // 1 0 0 0 �������; �
  2021. // 0 1 1 1 �
  2022. // 0 1 1 0 � NOTE: Windows defines its
  2023. // 0 1 0 1 � ROP2 codes as the bitwise
  2024. // 0 1 0 0 � value calculated here
  2025. // 0 0 1 1 � plus one. All other ROP
  2026. // 0 0 1 0 � codes are the straight
  2027. // 0 0 0 1 � bitwise value.
  2028. // 0 0 0 0 ��������������ͼ
  2029. //
  2030. // Or, algorithmically...
  2031. //
  2032. // ROP3 = (ROP2 & 0x3) | ((ROP2 & 0xC) << 4) | (ROP2 << 2)
  2033. //
  2034. // ROP4 = (ROP3 << 8) | ROP3
  2035. //
  2036. mix = (mix & 0x1F) - 1;
  2037. rop4 = (mix & 0x3) | ((mix & 0xC) << 4) | (mix << 2);
  2038. rop4 = (rop4 << 8) | rop4;
  2039. //
  2040. // This can be passed on to the BitBlt code.
  2041. //
  2042. rc = DrvBitBlt( pso,
  2043. NULL,
  2044. NULL,
  2045. pco,
  2046. NULL,
  2047. &pco->rclBounds,
  2048. NULL,
  2049. NULL,
  2050. pbo,
  2051. pptlBrushOrg,
  2052. rop4 );
  2053. //
  2054. // We have stored this object in the BitBlt, so don't store the data
  2055. // again.
  2056. //
  2057. fAccumulate = FALSE;
  2058. DC_EXIT_POINT:
  2059. if (fAccumulate)
  2060. {
  2061. OEClipAndAddScreenData(&rectDst, pco);
  2062. }
  2063. OE_SHM_STOP_WRITING;
  2064. NO_LOCK_EXIT:
  2065. DebugExitDWORD(DrvPaint, rc);
  2066. return(rc);
  2067. }
  2068. //
  2069. // OE_DDProcessRequest - see oe.h
  2070. //
  2071. ULONG OE_DDProcessRequest
  2072. (
  2073. SURFOBJ* pso,
  2074. UINT cjIn,
  2075. void * pvIn,
  2076. UINT cjOut,
  2077. void * pvOut
  2078. )
  2079. {
  2080. BOOL rc = TRUE;
  2081. LPOSI_ESCAPE_HEADER pHeader;
  2082. DebugEntry(OE_DDProcessRequest);
  2083. //
  2084. // Get the request number.
  2085. //
  2086. pHeader = pvIn;
  2087. switch (pHeader->escapeFn)
  2088. {
  2089. case OE_ESC_NEW_FONTS:
  2090. {
  2091. if ((cjIn != sizeof(OE_NEW_FONTS)) ||
  2092. (cjOut != sizeof(OE_NEW_FONTS)))
  2093. {
  2094. ERROR_OUT(("OE_DDProcessRequest: Invalid sizes %d, %d for OE_ESC_NEW_FONTS",
  2095. cjIn, cjOut));
  2096. rc = FALSE;
  2097. DC_QUIT;
  2098. }
  2099. //
  2100. // Get new local font data
  2101. //
  2102. OEDDSetNewFonts(pvIn);
  2103. }
  2104. break;
  2105. case OE_ESC_NEW_CAPABILITIES:
  2106. {
  2107. if ((cjIn != sizeof(OE_NEW_CAPABILITIES)) ||
  2108. (cjOut != sizeof(OE_NEW_CAPABILITIES)))
  2109. {
  2110. ERROR_OUT(("OE_DDProcessRequest: Invalid sizes %d, %d for OE_ESC_NEW_CAPABILITIES",
  2111. cjIn, cjOut));
  2112. rc = FALSE;
  2113. DC_QUIT;
  2114. }
  2115. //
  2116. // The capabilities have changed - take the new copy.
  2117. //
  2118. OEDDSetNewCapabilities(pvIn);
  2119. }
  2120. break;
  2121. default:
  2122. {
  2123. ERROR_OUT(("Unrecognised OE escape"));
  2124. rc = FALSE;
  2125. }
  2126. break;
  2127. }
  2128. DC_EXIT_POINT:
  2129. DebugExitDWORD(OE_DDProcessRequest, rc);
  2130. return((ULONG)rc);
  2131. }
  2132. //
  2133. // OE_DDTerm()
  2134. // This cleans up objects used
  2135. //
  2136. void OE_DDTerm(void)
  2137. {
  2138. DebugEntry(OE_DDTerm);
  2139. //
  2140. // Free font list
  2141. //
  2142. if (g_poeLocalFonts)
  2143. {
  2144. TRACE_OUT(("OE_DDLocalHosting: freeing font block since we're done sharing"));
  2145. EngFreeMem(g_poeLocalFonts);
  2146. g_poeLocalFonts = NULL;
  2147. g_oeNumFonts = 0;
  2148. }
  2149. DebugExitVOID(OE_DDTerm);
  2150. }
  2151. //
  2152. // DrvRealizeBrush - see NT DDK documentation.
  2153. //
  2154. BOOL DrvRealizeBrush(BRUSHOBJ *pbo,
  2155. SURFOBJ *psoTarget,
  2156. SURFOBJ *psoPattern,
  2157. SURFOBJ *psoMask,
  2158. XLATEOBJ *pxlo,
  2159. ULONG iHatch)
  2160. {
  2161. LPOSI_PDEV ppdev = (LPOSI_PDEV)psoTarget->dhpdev;
  2162. BOOL rc = TRUE;
  2163. LPBYTE pData;
  2164. BYTE brushBits[8];
  2165. UINT color1;
  2166. UINT color2;
  2167. int i;
  2168. int j;
  2169. BOOL monochromeBrush = TRUE;
  2170. DebugEntry(DrvRealizeBrush);
  2171. //
  2172. // This function only sets up local data, so shared memory protection
  2173. // is not required.
  2174. //
  2175. //
  2176. // Since this function is called only when we call BRUSHOBJ_pvGetRBrush
  2177. // and we don't do any processing until we are in a share we don't need
  2178. // an explicit check for hosting here ('coz that happened before the
  2179. // call to realise the brush).
  2180. //
  2181. //
  2182. // A valid brush satisfies either of the following criteria.
  2183. //
  2184. // 1) It is a standard hatch brush (as passed by DrvEnablePDEV)
  2185. // 2) It is an 8x8 monochrome bitmap
  2186. //
  2187. //
  2188. // Check for a Windows standard hatch
  2189. //
  2190. if (iHatch < HS_DDI_MAX)
  2191. {
  2192. TRACE_OUT(( "Standard hatch %lu", iHatch));
  2193. rc = OEStoreBrush(ppdev,
  2194. pbo,
  2195. BS_HATCHED,
  2196. NULL,
  2197. pxlo,
  2198. (BYTE)iHatch,
  2199. 0,
  2200. 1);
  2201. DC_QUIT;
  2202. }
  2203. //
  2204. // If the driver has been passed a dither color brush we can support
  2205. // this by sending a solid color brush definition
  2206. //
  2207. if ((iHatch & RB_DITHERCOLOR) != 0)
  2208. {
  2209. TRACE_OUT(( "Standard hatch %lu", iHatch));
  2210. rc = OEStoreBrush(ppdev,
  2211. pbo,
  2212. BS_SOLID,
  2213. NULL,
  2214. NULL,
  2215. (BYTE)iHatch,
  2216. iHatch & 0xFFFFFF,
  2217. 0);
  2218. DC_QUIT;
  2219. }
  2220. //
  2221. // Check for a simple 8x8 brush
  2222. //
  2223. if ( (psoPattern->sizlBitmap.cx == 8) &&
  2224. (psoPattern->sizlBitmap.cy == 8) )
  2225. {
  2226. //
  2227. // Check for 2 colours only in the bitmap.
  2228. //
  2229. // NOTE: There's a flag (BMF_TOPDOWN) in psoPattern->fjBitmap
  2230. // that's supposed to indicate whether the bitmap is top-down or
  2231. // bottom-up, but it is not always set up correctly. In fact, the
  2232. // bitmaps are always the wrong way up for our protocol, so we have
  2233. // to flip them regardless of the flag. Hence the row numbers are
  2234. // reversed ('i' loops) in all the conversions below.
  2235. //
  2236. pData = psoPattern->pvScan0;
  2237. switch (psoPattern->iBitmapFormat)
  2238. {
  2239. case BMF_1BPP:
  2240. {
  2241. //
  2242. // 1 bpp MUST be 2 colours maximum.
  2243. //
  2244. color1 = 1;
  2245. color2 = 0;
  2246. for (i = 7; i >= 0; i--)
  2247. {
  2248. brushBits[i] = *pData;
  2249. pData += psoPattern->lDelta;
  2250. }
  2251. }
  2252. break;
  2253. case BMF_4BPP:
  2254. {
  2255. //
  2256. // See if it is really a 2 colour brush. Start off with
  2257. // both colours the same.
  2258. //
  2259. color1 = pData[0] & 15;
  2260. color2 = color1;
  2261. //
  2262. // Iterate through each row of the bitmap.
  2263. //
  2264. for (i = 7; (i >= 0) && (monochromeBrush); i--)
  2265. {
  2266. brushBits[i] = 0;
  2267. //
  2268. // Check each pixel in the row: 4bpp->2 pixels per byte
  2269. //
  2270. for (j = 0; (j < 4) && (monochromeBrush); j++)
  2271. {
  2272. //
  2273. // Check the 1st pixel color
  2274. //
  2275. if ( (color1 != (UINT)(pData[j] & 0x0F)) &&
  2276. (color2 != (UINT)(pData[j] & 0x0F)) )
  2277. {
  2278. if (color1 == color2)
  2279. {
  2280. color2 = (pData[j] & 0x0F);
  2281. }
  2282. else
  2283. {
  2284. monochromeBrush = FALSE;
  2285. }
  2286. }
  2287. //
  2288. // Check the 2nd pixel color
  2289. //
  2290. if ( (color1 != (UINT)((pData[j] & 0xF0) >> 4)) &&
  2291. (color2 != (UINT)((pData[j] & 0xF0) >> 4)) )
  2292. {
  2293. if (color1 == color2)
  2294. {
  2295. color2 = (pData[j] & 0xF0) >> 4;
  2296. }
  2297. else
  2298. {
  2299. monochromeBrush = FALSE;
  2300. }
  2301. }
  2302. //
  2303. // Set up the brush data. High bit is leftmost.
  2304. //
  2305. if ((UINT)(pData[j] & 0x0F) == color1)
  2306. {
  2307. brushBits[i] |= 0x40 >> (j * 2);
  2308. }
  2309. if ((UINT)(pData[j] & 0xF0) >> 4 == color1)
  2310. {
  2311. brushBits[i] |= 0x80 >> (j * 2);
  2312. }
  2313. }
  2314. //
  2315. // Get start of next row.
  2316. //
  2317. pData += psoPattern->lDelta;
  2318. }
  2319. }
  2320. break;
  2321. case BMF_8BPP:
  2322. {
  2323. //
  2324. // See if it is really a 2 colour brush. Start off with
  2325. // both colours the same.
  2326. //
  2327. color1 = pData[0];
  2328. color2 = color1;
  2329. //
  2330. // Iterate through each row of the bitmap.
  2331. //
  2332. for (i = 7; (i >= 0) && (monochromeBrush); i--)
  2333. {
  2334. brushBits[i] = 0;
  2335. //
  2336. // Check each pixel in the row: 8bpp->1 pixel per byte
  2337. //
  2338. for (j = 0; (j < 8) && (monochromeBrush); j++)
  2339. {
  2340. //
  2341. // Check each pixel.
  2342. //
  2343. if ( (color1 != pData[j]) &&
  2344. (color2 != pData[j]) )
  2345. {
  2346. if (color1 == color2)
  2347. {
  2348. color2 = pData[j];
  2349. }
  2350. else
  2351. {
  2352. monochromeBrush = FALSE;
  2353. }
  2354. }
  2355. //
  2356. // Update the brush data. High bit is leftmost.
  2357. //
  2358. if (pData[j] == color1)
  2359. {
  2360. brushBits[i] |= 0x80 >> j;
  2361. }
  2362. }
  2363. //
  2364. // Get start of next row.
  2365. //
  2366. pData += psoPattern->lDelta;
  2367. }
  2368. }
  2369. break;
  2370. default:
  2371. {
  2372. //
  2373. // Unsupported colour depth.
  2374. //
  2375. monochromeBrush = FALSE;
  2376. }
  2377. break;
  2378. }
  2379. }
  2380. else
  2381. {
  2382. //
  2383. // The brush is the wrong size or requires dithering and so cannot
  2384. // be sent over the wire.
  2385. //
  2386. monochromeBrush = FALSE;
  2387. }
  2388. //
  2389. // Store that brush.
  2390. //
  2391. if (monochromeBrush)
  2392. {
  2393. //
  2394. // Store the brush - note that we have a monochrome brush where the
  2395. // color bit is set up so that 0 = color2 and 1 = color1. This
  2396. // actually corresponds to 0 = fg and 1 = bg for the protocol
  2397. // colors.
  2398. //
  2399. TRACE_OUT(( "Storing brush: type %d bg %x fg %x",
  2400. psoPattern->iBitmapFormat,
  2401. color1,
  2402. color2));
  2403. rc = OEStoreBrush(ppdev,
  2404. pbo,
  2405. BS_PATTERN,
  2406. brushBits,
  2407. pxlo,
  2408. 0,
  2409. color2,
  2410. color1);
  2411. }
  2412. else
  2413. {
  2414. TRACE_OUT(( "Rejected brush h %08lx s (%ld, %ld) fmt %lu",
  2415. iHatch,
  2416. psoPattern != NULL ? psoPattern->sizlBitmap.cx : 0,
  2417. psoPattern != NULL ? psoPattern->sizlBitmap.cy : 0,
  2418. psoPattern != NULL ? psoPattern->iBitmapFormat : 0));
  2419. rc = OEStoreBrush(ppdev, pbo, BS_NULL, NULL, pxlo, 0, 0, 0);
  2420. }
  2421. DC_EXIT_POINT:
  2422. DebugExitDWORD(DrvRealizeBrush, rc);
  2423. return(rc);
  2424. }
  2425. //
  2426. // DrvSaveScreenBits - see NT DDK documentation.
  2427. //
  2428. ULONG_PTR DrvSaveScreenBits(SURFOBJ *pso,
  2429. ULONG iMode,
  2430. ULONG_PTR ident,
  2431. RECTL *prcl)
  2432. {
  2433. BOOL rc;
  2434. UINT ourMode;
  2435. RECT rectDst;
  2436. DebugEntry(DrvSaveScreenBits);
  2437. TRACE_OUT(("DrvSaveScreenBits: %s",
  2438. ((iMode == SS_SAVE) ? "SAVE" :
  2439. ((iMode == SS_RESTORE) ? "RESTORE" : "DISCARD"))));
  2440. TRACE_OUT((" rect {%04ld, %04ld, %04ld, %04ld}",
  2441. prcl->left, prcl->top, prcl->right, prcl->bottom));
  2442. //
  2443. // Default is TRUE, let SaveBits happen if we don't care. Which we don't
  2444. // if we have no shared memory (NetMeeting isn't running), no window list
  2445. // (no shared apps), or the operation isn't intersecting a window we
  2446. // care about.
  2447. //
  2448. // Note that if we return TRUE on a save, and FALSE on a restore later
  2449. // (because we are now sharing that area for example), USER+GRE handle
  2450. // that. So it's ok.
  2451. //
  2452. rc = TRUE;
  2453. //
  2454. // DO THIS _BEFORE_ TAKING LOCK
  2455. //
  2456. if (!g_oeViewers)
  2457. goto NO_LOCK_EXIT;
  2458. //
  2459. // If we have no shared memory (NetMeeting isn't running), this will bail
  2460. // out immediately.
  2461. //
  2462. OE_SHM_START_WRITING;
  2463. //
  2464. // Get the bounding rectangle for the operation. NOTE that this is
  2465. // meaningless for SS_FREE.
  2466. //
  2467. RECT_FROM_RECTL(rectDst, (*prcl));
  2468. if (iMode != SS_FREE)
  2469. {
  2470. //
  2471. // Check if we are accumulating data for this area, ONLY FOR
  2472. // SAVEs. We may get notified after a window is gone to
  2473. // restore or discard bits we had saved.
  2474. //
  2475. if (!OEAccumulateOutputRect(pso, &rectDst))
  2476. {
  2477. TRACE_OUT(("DrvSaveScreenBits: save/restore in area we don't care about"));
  2478. DC_QUIT;
  2479. }
  2480. }
  2481. //
  2482. // Convert the NT orders to our generic save/restore types.
  2483. //
  2484. switch (iMode)
  2485. {
  2486. case SS_SAVE:
  2487. {
  2488. ourMode = ONBOARD_SAVE;
  2489. }
  2490. break;
  2491. case SS_RESTORE:
  2492. {
  2493. ourMode = ONBOARD_RESTORE;
  2494. }
  2495. break;
  2496. case SS_FREE:
  2497. {
  2498. ourMode = ONBOARD_DISCARD;
  2499. }
  2500. break;
  2501. default:
  2502. {
  2503. ERROR_OUT(( "Unknown type %lu", iMode));
  2504. DC_QUIT;
  2505. }
  2506. }
  2507. //
  2508. // Call through to the SSI handler.
  2509. //
  2510. rc = SSI_SaveScreenBitmap(&rectDst, ourMode);
  2511. DC_EXIT_POINT:
  2512. OE_SHM_STOP_WRITING;
  2513. NO_LOCK_EXIT:
  2514. TRACE_OUT(("DrvSaveScreenBits returning %d", rc));
  2515. DebugExitDWORD(DrvSaveScreenBits, rc);
  2516. return(rc);
  2517. }
  2518. //
  2519. // Function: OEUnicodeStrlen
  2520. //
  2521. // Description: Get the length of a unicode string in bytes.
  2522. //
  2523. // Parameters: pString - Unicode string to be read
  2524. //
  2525. // Returns: Length of the Unicode string in bytes
  2526. //
  2527. int OEUnicodeStrlen(PWSTR pString)
  2528. {
  2529. int i;
  2530. for (i = 0; pString[i] != 0; i++)
  2531. ;
  2532. return((i + 1) * sizeof(WCHAR));
  2533. }
  2534. //
  2535. // Function: OEExpandColor
  2536. //
  2537. // Description: Converts a generic bitwise representation of an RGB color
  2538. // index into an 8-bit color index as used by the line
  2539. // protocol.
  2540. //
  2541. //
  2542. void OEExpandColor
  2543. (
  2544. LPBYTE lpField,
  2545. ULONG srcColor,
  2546. ULONG mask
  2547. )
  2548. {
  2549. ULONG colorTmp;
  2550. DebugEntry(OEExpandColor);
  2551. //
  2552. // Different example bit masks:
  2553. //
  2554. // Normal 24-bit:
  2555. // 0x000000FF (red)
  2556. // 0x0000FF00 (green)
  2557. // 0x00FF0000 (blue)
  2558. //
  2559. // True color 32-bits:
  2560. // 0xFF000000 (red)
  2561. // 0x00FF0000 (green)
  2562. // 0x0000FF00 (blue)
  2563. //
  2564. // 5-5-5 16-bits
  2565. // 0x0000001F (red)
  2566. // 0x000003E0 (green)
  2567. // 0x00007C00 (blue)
  2568. //
  2569. // 5-6-5 16-bits
  2570. // 0x0000001F (red)
  2571. // 0x000007E0 (green)
  2572. // 0x0000F800 (blue)
  2573. //
  2574. //
  2575. // Convert the color using the following algorithm.
  2576. //
  2577. // <new color> = <old color> * <new bpp mask> / <old bpp mask>
  2578. //
  2579. // where:
  2580. //
  2581. // new bpp mask = mask for all bits at new setting (0xFF for 8bpp)
  2582. //
  2583. // This way maximal (eg. 0x1F) and minimal (eg. 0x00) settings are
  2584. // converted into the correct 8-bit maximum and minimum.
  2585. //
  2586. // Rearranging the above equation we get:
  2587. //
  2588. // <new color> = (<old color> & <old bpp mask>) * 0xFF / <old bpp mask>
  2589. //
  2590. // where:
  2591. //
  2592. // <old bpp mask> = mask for the color
  2593. //
  2594. //
  2595. // LAURABU BOGUS:
  2596. // We need to avoid overflow caused by the multiply. NOTE: in theory
  2597. // we should use a double, but that's painfully slow. So for now hack
  2598. // it. If the HIBYTE is set, just right shift 24 bits.
  2599. //
  2600. colorTmp = srcColor & mask;
  2601. if (colorTmp & 0xFF000000)
  2602. colorTmp >>= 24;
  2603. else
  2604. colorTmp = (colorTmp * 0xFF) / mask;
  2605. *lpField = (BYTE)colorTmp;
  2606. TRACE_OUT(( "0x%lX -> 0x%X", srcColor, *lpField));
  2607. DebugExitVOID(OEExpandColor);
  2608. }
  2609. //
  2610. // Function: OEConvertColor
  2611. //
  2612. // Description: Convert a color from the NT Display Driver into a TSHR_COLOR
  2613. //
  2614. // Parameters: pDCColor - (returned) color in protocol format
  2615. // osColor - color from the NT display driver
  2616. // pxlo - XLATEOBJ for the color to be converted
  2617. // (NULL if no translation is required)
  2618. //
  2619. // Returns: (none)
  2620. //
  2621. void OEConvertColor(LPOSI_PDEV ppdev, LPTSHR_COLOR pTshrColor,
  2622. ULONG osColor,
  2623. XLATEOBJ* pxlo)
  2624. {
  2625. ULONG realIndex;
  2626. DebugEntry(OEConvertColor);
  2627. //
  2628. // Make sure we have a default setting.
  2629. //
  2630. RtlFillMemory(pTshrColor, sizeof(TSHR_COLOR), 0);
  2631. //
  2632. // Check if color translation is required.
  2633. //
  2634. if ((pxlo != NULL) && (pxlo->flXlate != XO_TRIVIAL))
  2635. {
  2636. //
  2637. // Convert from BMP to device color.
  2638. //
  2639. realIndex = XLATEOBJ_iXlate(pxlo, osColor);
  2640. if (realIndex == -1)
  2641. {
  2642. ERROR_OUT(( "Failed to convert color 0x%lx", osColor));
  2643. DC_QUIT;
  2644. }
  2645. }
  2646. else
  2647. {
  2648. //
  2649. // Use the OS color without translation
  2650. //
  2651. realIndex = osColor;
  2652. }
  2653. TRACE_OUT(( "Device color 0x%lX", realIndex));
  2654. //
  2655. // We now have the device specific version of the color. Time to
  2656. // convert it into a 24-bit RGB color as used by the line protocol.
  2657. //
  2658. switch (ppdev->iBitmapFormat)
  2659. {
  2660. case BMF_1BPP:
  2661. case BMF_4BPP:
  2662. case BMF_4RLE:
  2663. case BMF_8BPP:
  2664. case BMF_8RLE:
  2665. //
  2666. // Palette type device - use the device color as an index into
  2667. // our palette array.
  2668. //
  2669. pTshrColor->red = (BYTE)ppdev->pPal[realIndex].peRed;
  2670. pTshrColor->green= (BYTE)ppdev->pPal[realIndex].peGreen;
  2671. pTshrColor->blue = (BYTE)ppdev->pPal[realIndex].peBlue;
  2672. break;
  2673. case BMF_16BPP:
  2674. case BMF_24BPP:
  2675. case BMF_32BPP:
  2676. //
  2677. // Generic colour masks (could be eg. 5-6-5 for 16 or 8-8-8
  2678. // for 24 bits per pel). We must mask off the other bits and
  2679. // shift down to bit 0.
  2680. //
  2681. OEExpandColor(&(pTshrColor->red),
  2682. realIndex,
  2683. ppdev->flRed);
  2684. OEExpandColor(&(pTshrColor->green),
  2685. realIndex,
  2686. ppdev->flGreen);
  2687. OEExpandColor(&(pTshrColor->blue),
  2688. realIndex,
  2689. ppdev->flBlue);
  2690. break;
  2691. default:
  2692. ERROR_OUT(( "Unrecognised BMP color depth %lu",
  2693. ppdev->iBitmapFormat));
  2694. break;
  2695. }
  2696. TRACE_OUT(( "Red %x green %x blue %x", pTshrColor->red,
  2697. pTshrColor->green,
  2698. pTshrColor->blue));
  2699. DC_EXIT_POINT:
  2700. DebugExitVOID(OEConvertColor);
  2701. }
  2702. //
  2703. // Function: OEStoreBrush
  2704. //
  2705. // Description: Store the brush data required for pattern realted orders.
  2706. // This function is called by DrvRealiseBrush when it has data
  2707. // to be stored about a brush.
  2708. //
  2709. // Parameters: pbo - BRUSHOBJ of the brush to be stored
  2710. // style - Style of the brush (as defined in the DC-Share
  2711. // protocol)
  2712. // pBits - Pointer to the bits which are used to define
  2713. // a BS_PATTERN brush.
  2714. // pxlo - XLATEOBJ for the brush.
  2715. // hatch - Standard Windows hatch pattern index for a
  2716. // BS_HATCHED brush.
  2717. // color1 - index into XLATEOBJ for bit set color
  2718. // OR exact 24bpp color to use (pxlo == NULL)
  2719. // color2 - index into XLATEOBJ for bit clear color
  2720. // OR exact 24bpp color to use (pxlo == NULL)
  2721. //
  2722. // Returns: (none)
  2723. //
  2724. BOOL OEStoreBrush(LPOSI_PDEV ppdev,
  2725. BRUSHOBJ* pbo,
  2726. BYTE style,
  2727. LPBYTE pBits,
  2728. XLATEOBJ* pxlo,
  2729. BYTE hatch,
  2730. UINT color1,
  2731. UINT color2)
  2732. {
  2733. BOOL rc = FALSE;
  2734. int i;
  2735. LPBYTE pData;
  2736. ULONG* pColorTable;
  2737. POE_BRUSH_DATA pBrush;
  2738. DebugEntry(OEStoreBrush);
  2739. //
  2740. // Allocate the space for the brush data.
  2741. //
  2742. pBrush = (POE_BRUSH_DATA)BRUSHOBJ_pvAllocRbrush(pbo,
  2743. sizeof(OE_BRUSH_DATA));
  2744. if (pBrush == NULL)
  2745. {
  2746. ERROR_OUT(( "No memory"));
  2747. DC_QUIT;
  2748. }
  2749. //
  2750. // Reset the brush definition
  2751. //
  2752. RtlFillMemory(pBrush, sizeof(OE_BRUSH_DATA), 0);
  2753. //
  2754. // Set the new brush data.
  2755. //
  2756. pBrush->style = style;
  2757. pBrush->hatch = hatch;
  2758. TRACE_OUT(( " Style: %d Hatch: %d", style, hatch));
  2759. //
  2760. // For pattern brushes, copy the brush specific data.
  2761. //
  2762. if (style == BS_PATTERN)
  2763. {
  2764. //
  2765. // Copy the brush bits. Since this is an 8x8 mono bitmap, we can
  2766. // copy the first byte of the brush data for each scan line.
  2767. //
  2768. // NOTE however that the brush structures sent over the wire
  2769. // re-use the hatching variable as the first byte of the brush data.
  2770. //
  2771. pData = pBits;
  2772. pBrush->hatch = *pData;
  2773. TRACE_OUT(( " Hatch: %d", *pData));
  2774. pData++;
  2775. for (i = 0; i < 7; i++)
  2776. {
  2777. pBrush->brushData[i] = pData[i];
  2778. TRACE_OUT(( " Data[%d]: %d", i, pData[i]));
  2779. }
  2780. //
  2781. // Get pointer to the bitmap color table.
  2782. //
  2783. pColorTable = pxlo->pulXlate;
  2784. if (pColorTable == NULL)
  2785. {
  2786. pColorTable = XLATEOBJ_piVector(pxlo);
  2787. }
  2788. }
  2789. //
  2790. // Store the foreground and background colours for the brush.
  2791. //
  2792. if (pxlo != NULL)
  2793. {
  2794. //
  2795. // Conversion required.
  2796. //
  2797. OEConvertColor(ppdev,
  2798. &pBrush->fore,
  2799. color1,
  2800. pxlo);
  2801. OEConvertColor(ppdev,
  2802. &pBrush->back,
  2803. color2,
  2804. pxlo);
  2805. }
  2806. else
  2807. {
  2808. //
  2809. // We have been passed an exact 24bpp color - this only happens for
  2810. // solid brushes so we don't need to convert color2.
  2811. //
  2812. pBrush->fore.red = (BYTE) (color1 & 0x0000FF);
  2813. pBrush->fore.green = (BYTE)((color1 & 0x00FF00) >> 8);
  2814. pBrush->fore.blue = (BYTE)((color1 & 0xFF0000) >> 16);
  2815. }
  2816. rc = TRUE;
  2817. DC_EXIT_POINT:
  2818. DebugExitDWORD(OEStoreBrush, rc);
  2819. return(rc);
  2820. }
  2821. //
  2822. // Function: OECheckBrushIsSimple
  2823. //
  2824. // Description: Check that the brush is a 'simple' object we can transfer
  2825. // over the DC-Share protocol.
  2826. //
  2827. // Parameters: pbo - BRUSHOBJ of the brush to be checked.
  2828. //
  2829. // Returns: TRUE - brush can be sent as DC-Share order
  2830. // FALSE - brush is too complicated.
  2831. //
  2832. BOOL OECheckBrushIsSimple(LPOSI_PDEV ppdev,
  2833. BRUSHOBJ* pbo,
  2834. POE_BRUSH_DATA* ppBrush)
  2835. {
  2836. BOOL rc = FALSE;
  2837. POE_BRUSH_DATA pBrush = NULL;
  2838. DebugEntry(OECheckBrushIsSimple);
  2839. //
  2840. // A 'simple' brush satisfies any of the following.
  2841. //
  2842. // 1) It is a solid color.
  2843. // 2) It is a valid brush as stored by DrvRealizeBrush.
  2844. //
  2845. //
  2846. // Check for a simple solid colour.
  2847. //
  2848. if (pbo->iSolidColor != -1)
  2849. {
  2850. //
  2851. // Use the reserved brush definition to set up the solid colour.
  2852. //
  2853. TRACE_OUT(( "Simple solid colour %08lx", pbo->iSolidColor));
  2854. pBrush = &g_oeBrushData;
  2855. //
  2856. // Set up the specific data for this brush.
  2857. //
  2858. OEConvertColor(ppdev, &pBrush->fore, pbo->iSolidColor, NULL);
  2859. pBrush->back.red = 0;
  2860. pBrush->back.green = 0;
  2861. pBrush->back.blue = 0;
  2862. pBrush->style = BS_SOLID;
  2863. pBrush->hatch = 0;
  2864. RtlFillMemory(pBrush->brushData, 7, 0);
  2865. //
  2866. // We have a valid brush - return true.
  2867. //
  2868. rc = TRUE;
  2869. DC_QUIT;
  2870. }
  2871. //
  2872. // Check brush definition (which was stored when we realized the
  2873. // brush).
  2874. //
  2875. pBrush = (POE_BRUSH_DATA)pbo->pvRbrush;
  2876. if (pBrush == NULL)
  2877. {
  2878. pBrush = (POE_BRUSH_DATA)BRUSHOBJ_pvGetRbrush(pbo);
  2879. if (pBrush == NULL)
  2880. {
  2881. //
  2882. // We can get NULL returned from BRUSHOBJ_pvGetRbrush when the
  2883. // brush is NULL or in low-memory situations (when the brush
  2884. // realization may fail).
  2885. //
  2886. TRACE_OUT(( "NULL returned from BRUSHOBJ_pvGetRbrush"));
  2887. DC_QUIT;
  2888. }
  2889. }
  2890. //
  2891. // Check it is an encodable brush.
  2892. //
  2893. if (pBrush->style == BS_NULL)
  2894. {
  2895. TRACE_OUT(( "Complex brush"));
  2896. DC_QUIT;
  2897. }
  2898. //
  2899. // Evrything passed - let's use this brush.
  2900. //
  2901. rc = TRUE;
  2902. DC_EXIT_POINT:
  2903. //
  2904. // Return the brush definition
  2905. //
  2906. *ppBrush = pBrush;
  2907. TRACE_OUT(( "Returning %d - 0x%08lx", rc, pBrush));
  2908. DebugExitDWORD(OECheckBrushIsSimple, rc);
  2909. return(rc);
  2910. }
  2911. //
  2912. // Function: OEClippingIsSimple
  2913. //
  2914. // Description: Check to see if the clipping on the graphics object is
  2915. // trivial
  2916. //
  2917. // Parameters: pco - CLIPOBJ of the graphics object to be checked.
  2918. //
  2919. // Returns: TRUE - Clipping is trivial
  2920. // FALSE - Clipping is complex
  2921. //
  2922. BOOL OEClippingIsSimple(CLIPOBJ* pco)
  2923. {
  2924. BOOL rc = TRUE;
  2925. DebugEntry(OEClippingIsSimple);
  2926. //
  2927. // Check for a valid clip object
  2928. //
  2929. if (pco == NULL)
  2930. {
  2931. TRACE_OUT(( "No clipobj"));
  2932. DC_QUIT;
  2933. }
  2934. //
  2935. // Check for complexity of clipping
  2936. //
  2937. switch (pco->iDComplexity)
  2938. {
  2939. case DC_TRIVIAL:
  2940. case DC_RECT:
  2941. //
  2942. // Trivial (ignore clipping) or simple (one square) clipping -
  2943. // no worries.
  2944. //
  2945. TRACE_OUT(( "Simple clipping"));
  2946. DC_QUIT;
  2947. default:
  2948. TRACE_OUT(( "Clipping is complex"));
  2949. break;
  2950. }
  2951. //
  2952. // Failed all tests - must be too complicated.
  2953. //
  2954. rc = FALSE;
  2955. DC_EXIT_POINT:
  2956. DebugExitDWORD(OEClippingIsSimple, rc);
  2957. return(rc);
  2958. }
  2959. //
  2960. // Function: OEClippingIsComplex
  2961. //
  2962. // Description: Check to see if the clipping on the graphics object is too
  2963. // complicated to be sent as an order or multiple orders.
  2964. //
  2965. // Parameters: pco - CLIPOBJ of the graphics object to be checked.
  2966. //
  2967. // Returns: TRUE - Clipping is too complicated
  2968. // FALSE - Clipping is sufficiently simple to send as orders
  2969. //
  2970. BOOL OEClippingIsComplex(CLIPOBJ* pco)
  2971. {
  2972. BOOL rc = FALSE;
  2973. BOOL fMoreRects;
  2974. OE_ENUMRECTS clip;
  2975. UINT numRects = 0;
  2976. DebugEntry(OEClippingIsComplex);
  2977. //
  2978. // If the any of the following are true, the clipping is not too
  2979. // complicated.
  2980. //
  2981. // 1) The clip object does not exist.
  2982. // 2) The clipping is trivial (the object exists, but there are no
  2983. // clipping rectangles).
  2984. // 3) The clipping is a single rectangle.
  2985. // 4) The object enumerates to less than 'n' rectangles.
  2986. //
  2987. //
  2988. // Check for a valid clip object
  2989. //
  2990. if (pco == NULL)
  2991. {
  2992. TRACE_OUT(( "No clipobj"));
  2993. DC_QUIT;
  2994. }
  2995. //
  2996. // Check for complexity of clipping
  2997. //
  2998. switch (pco->iDComplexity)
  2999. {
  3000. case DC_TRIVIAL:
  3001. case DC_RECT:
  3002. //
  3003. // Trivial or simple clipping - no worries.
  3004. //
  3005. TRACE_OUT(( "Simple clipping"));
  3006. DC_QUIT;
  3007. case DC_COMPLEX:
  3008. //
  3009. // Lots of rectangles - make sure that it is less than the
  3010. // acceptable limit.
  3011. // The documentation for this function incorrectly states that
  3012. // the returned value is the total number of rectangles
  3013. // comprising the clip region. In fact, -1 is always returned,
  3014. // even when the final parameter is non-zero. This means we
  3015. // have to enumerate to get the number of rects.
  3016. //
  3017. CLIPOBJ_cEnumStart(pco,
  3018. FALSE,
  3019. CT_RECTANGLES,
  3020. CD_ANY,
  3021. 0);
  3022. //
  3023. // MSDN: It is possible for CLIPOBJ_bEnum to return TRUE with
  3024. // the number of clipping rectangles equal to zero. In such
  3025. // cases, the driver should call CLIPOBJ_bEnum again without
  3026. // taking any action. Get as many rectangles as we permit for
  3027. // order encoding - this loop should execute once only.
  3028. // If the number of rects equals COMPLEX_CLIP_RECT_COUNT the
  3029. // 1st invocation of CLIPOBJ_bEnum returns that there are more
  3030. // rects and a second call returns there are no more without
  3031. // returning any in addition to those returned on the first
  3032. // call. Our buffer has space for COMPLEX_CLIP_RECT_COUNT+1
  3033. // rects so we should never have to execute the loop more than
  3034. // once.
  3035. //
  3036. do
  3037. {
  3038. fMoreRects = CLIPOBJ_bEnum(pco,
  3039. sizeof(clip),
  3040. (ULONG *)&clip.rects);
  3041. numRects += clip.rects.c;
  3042. } while ( fMoreRects && (numRects <= COMPLEX_CLIP_RECT_COUNT) );
  3043. //
  3044. // If there are no more rectangles in the clip region then the
  3045. // clipping complexity is within our limits for order encoding.
  3046. //
  3047. if ( numRects <= COMPLEX_CLIP_RECT_COUNT )
  3048. {
  3049. TRACE_OUT(( "Acceptable clipping %u", numRects));
  3050. DC_QUIT;
  3051. }
  3052. break;
  3053. default:
  3054. ERROR_OUT(( "Unknown clipping"));
  3055. break;
  3056. }
  3057. //
  3058. // Failed all tests - must be too complicated.
  3059. //
  3060. TRACE_OUT(( "Complex clipping"));
  3061. rc = TRUE;
  3062. DC_EXIT_POINT:
  3063. DebugExitDWORD(OEClippingIsComplex, rc);
  3064. return(rc);
  3065. }
  3066. //
  3067. // Function: OEAccumulateOutput
  3068. //
  3069. // Description: Check to see if we should accumulate this output for
  3070. // sending to the remote machine.
  3071. //
  3072. // Parameters: pso - Pointer to the target surface
  3073. // pco - Pointer to the clip object (may be NULL)
  3074. // pRect - Pointer to the bounding rectangle of the operation
  3075. //
  3076. // Returns: TRUE - We should accumulate the output
  3077. // FALSE - ignore the output
  3078. //
  3079. BOOL OEAccumulateOutput(SURFOBJ* pso, CLIPOBJ *pco, LPRECT pRect)
  3080. {
  3081. BOOL rc = FALSE;
  3082. LPOSI_PDEV ppdev = ((LPOSI_PDEV)pso->dhpdev);
  3083. DebugEntry(OEAccumulateOutput);
  3084. //
  3085. // Validate we have valid parameters to access the surface.
  3086. //
  3087. if (ppdev == NULL)
  3088. {
  3089. TRACE_OUT(( "NULL PDEV"));
  3090. DC_QUIT;
  3091. }
  3092. //
  3093. // Check for the screen surface, which will be a bitmap in the hosting
  3094. // only code.
  3095. //
  3096. if (ppdev->hsurfScreen != pso->hsurf)
  3097. {
  3098. TRACE_OUT(( "Dest is not our surface"));
  3099. DC_QUIT;
  3100. }
  3101. if (pso->dhsurf == NULL)
  3102. {
  3103. ERROR_OUT(( "NULL hSurf"));
  3104. DC_QUIT;
  3105. }
  3106. rc = TRUE;
  3107. DC_EXIT_POINT:
  3108. DebugExitBOOL(OEAccumulateOutput, rc);
  3109. return(rc);
  3110. }
  3111. //
  3112. // Function: OEAccumulateOutputRect
  3113. //
  3114. // Description: Check to see if we should accumulate the given output rect
  3115. // for sending to the remote machine.
  3116. //
  3117. // Most drawing functions will use OEAccumulateOutput, which
  3118. // just checks for a single point within the hosted area.
  3119. // This function checks for any part of the given rectangle
  3120. // intersecting with the hosted area. It is currently only
  3121. // used by DrvSaveScreenBitmap - operations which may not
  3122. // lie completetely within the hosted area.
  3123. //
  3124. // Parameters: pso - Pointer to the target surface
  3125. // pRect - Pointer to the bounding rectangle of the operation
  3126. //
  3127. // Returns: TRUE - We should accumulate the output
  3128. // FALSE - ignore the output
  3129. //
  3130. BOOL OEAccumulateOutputRect( SURFOBJ* pso, LPRECT pRect)
  3131. {
  3132. BOOL rc = FALSE;
  3133. LPOSI_PDEV ppdev = ((LPOSI_PDEV)pso->dhpdev);
  3134. DebugEntry(OEAccumulateOutputRect);
  3135. //
  3136. // Validate we have valid parameters to access the surface.
  3137. //
  3138. if (ppdev == NULL)
  3139. {
  3140. TRACE_OUT(( "NULL PDEV"));
  3141. DC_QUIT;
  3142. }
  3143. //
  3144. // Check for the screen surface, which will be a bitmap in the hosting
  3145. // only code.
  3146. //
  3147. if (ppdev->hsurfScreen != pso->hsurf)
  3148. {
  3149. TRACE_OUT(( "Dest is not our surface"));
  3150. DC_QUIT;
  3151. }
  3152. if (pso->dhsurf == NULL)
  3153. {
  3154. ERROR_OUT(( "NULL hSurf"));
  3155. DC_QUIT;
  3156. }
  3157. //
  3158. // Check if we are accumulating this window.
  3159. //
  3160. rc = TRUE;
  3161. DC_EXIT_POINT:
  3162. TRACE_OUT(("OEAccumulateOutputRect: rect {%d, %d, %d, %d} is %sshared",
  3163. pRect->left, pRect->top, pRect->right, pRect->bottom,
  3164. (rc ? "" : "NOT ")));
  3165. DebugExitBOOL(OEAccumulateOutputRect, rc);
  3166. return(rc);
  3167. }
  3168. //
  3169. // Function: OESendRop3AsOrder
  3170. //
  3171. // Description: Check if we are allowed to send this 3-way ROP. A ROP may
  3172. // be disallowed if it relies on the destination data.
  3173. //
  3174. // Parameters: rop3 - the 3-way ROP to be checked.
  3175. //
  3176. // Returns: TRUE - We are allowed to send this ROP
  3177. // FALSE - We can't send this ROP
  3178. //
  3179. BOOL OESendRop3AsOrder(BYTE rop3)
  3180. {
  3181. BOOL rc = TRUE;
  3182. DebugEntry(OESendRop3AsOrder);
  3183. //
  3184. // Rop 0x5F is used by MSDN to highlight search keywords. This XORs
  3185. // a pattern with the destination, producing markedly different (and
  3186. // sometimes unreadable) shadow output. We special-case no-encoding for
  3187. // it.
  3188. //
  3189. if (rop3 == 0x5F)
  3190. {
  3191. TRACE_OUT(("Rop3 0x5F never encoded"));
  3192. rc = FALSE;
  3193. }
  3194. DebugExitBOOL(OESendRop3AsOrder, rc);
  3195. return(rc);
  3196. }
  3197. //
  3198. // Function: OECheckFontIsSupported
  3199. //
  3200. // Description: Check if we are allowed to send this font. Fonts are
  3201. // disallowed while they are being negotiated on a new entry
  3202. // to the share.
  3203. //
  3204. // Parameters: pfo - (IN) the font to be checked
  3205. // pFontText - (IN) text message to be sent
  3206. // textLen - (IN) length of text message
  3207. // pFontHeight - (OUT) font height in points
  3208. // pFontAscender - (OUT) font ascender in points
  3209. // pFontWidth - (OUT) ave font width in points
  3210. // pFontWeight - (OUT) font weight
  3211. // pFontFlags - (OUT) font style flags
  3212. // pFontIndex - (OUT) font table index
  3213. // pSendDeltaX - (OUT) Do we need to send delta X coords?
  3214. //
  3215. // Returns: TRUE - We are allowed to send this font
  3216. // FALSE - We can't send this font
  3217. //
  3218. BOOL OECheckFontIsSupported
  3219. (
  3220. FONTOBJ* pfo,
  3221. LPSTR pFontText,
  3222. UINT textLen,
  3223. LPUINT pFontHeight,
  3224. LPUINT pFontAscender,
  3225. LPUINT pFontWidth,
  3226. LPUINT pFontWeight,
  3227. LPUINT pFontFlags,
  3228. LPUINT pFontIndex,
  3229. LPBOOL pSendDeltaX
  3230. )
  3231. {
  3232. BOOL rc = FALSE;
  3233. PIFIMETRICS pFontMetrics;
  3234. UINT codePage;
  3235. UINT i;
  3236. UINT iLocal;
  3237. UINT matchQuality;
  3238. UINT charWidthAdjustment = 0;
  3239. char fontName[FH_FACESIZE];
  3240. ULONG fontNameLen;
  3241. PWSTR pUnicodeString;
  3242. XFORMOBJ* pxform;
  3243. POINTL xformSize[3];
  3244. int compareResult;
  3245. FLOATOBJ_XFORM xformFloatData;
  3246. DebugEntry(OECheckFontIsSupported);
  3247. //
  3248. // Set up default return values
  3249. //
  3250. *pSendDeltaX = FALSE;
  3251. //
  3252. // Check that we have a valid list of font data from the remotes.
  3253. //
  3254. if (!g_oeTextEnabled)
  3255. {
  3256. TRACE_OUT(( "Fonts unavailable"));
  3257. DC_QUIT;
  3258. }
  3259. //
  3260. // Check for valid font attributes
  3261. //
  3262. pFontMetrics = FONTOBJ_pifi(pfo);
  3263. if (pFontMetrics->fsSelection & FM_SEL_OUTLINED)
  3264. {
  3265. TRACE_OUT(( "Unsupported font style"));
  3266. DC_QUIT;
  3267. }
  3268. //
  3269. // The current protocol cannot apply a general 2-D transform to text
  3270. // orders, so we must reject any weird ones such as:
  3271. //
  3272. // - rotations
  3273. // - X or Y shears
  3274. // - X or Y reflections
  3275. // - scaling with a negative value.
  3276. //
  3277. // Or put another way, we only allow:
  3278. //
  3279. // - the identity transformation
  3280. // - scaling with a positive value.
  3281. //
  3282. pxform = FONTOBJ_pxoGetXform(pfo);
  3283. if (pxform != NULL)
  3284. {
  3285. //
  3286. // Get the details of the transformation. Note we can ignore the
  3287. // translation vector as it does not affect the font sizing /
  3288. // orientation, so we are only interested in the matrix values...
  3289. //
  3290. //
  3291. // NOTE: Do NOT use floating point explicitly!
  3292. // Can't do float ops in ring 0 with normal lib for x86.
  3293. // Use FLOATOBJs instead and corresponding Eng services.
  3294. // On alpha, these are macros and are way fast in any case.
  3295. //
  3296. if (XFORMOBJ_iGetFloatObjXform(pxform, &xformFloatData) != DDI_ERROR)
  3297. {
  3298. //
  3299. // Rotations and shears will have cross dependencies on the x
  3300. // and y components.
  3301. //
  3302. if ( (!FLOATOBJ_EqualLong(&xformFloatData.eM12, 0)) ||
  3303. (!FLOATOBJ_EqualLong(&xformFloatData.eM21, 0)) )
  3304. {
  3305. TRACE_OUT(( "Rejected rotn/shear"));
  3306. DC_QUIT;
  3307. }
  3308. //
  3309. // Reflections and scaling operations with negative scale
  3310. // factors will have negative values on the leading diagonal of
  3311. // the matrix.
  3312. //
  3313. if ( (FLOATOBJ_LessThanLong(&xformFloatData.eM11, 0)) ||
  3314. (FLOATOBJ_LessThanLong(&xformFloatData.eM22, 0)) )
  3315. {
  3316. TRACE_OUT(( "Rejected refln/-ive"));
  3317. DC_QUIT;
  3318. }
  3319. }
  3320. }
  3321. //
  3322. // Get the current font code page for font matching.
  3323. //
  3324. switch (pFontMetrics->jWinCharSet)
  3325. {
  3326. case ANSI_CHARSET:
  3327. TRACE_OUT(( "ANSI font"));
  3328. codePage = NF_CP_WIN_ANSI;
  3329. break;
  3330. case OEM_CHARSET:
  3331. TRACE_OUT(( "OEM font"));
  3332. codePage = NF_CP_WIN_OEM;
  3333. break;
  3334. case SYMBOL_CHARSET:
  3335. TRACE_OUT(("Symbol font"));
  3336. codePage = NF_CP_WIN_SYMBOL;
  3337. break;
  3338. default:
  3339. TRACE_OUT(( "Unknown CP %d", pFontMetrics->jWinCharSet));
  3340. codePage = NF_CP_UNKNOWN;
  3341. break;
  3342. }
  3343. //
  3344. // Get the name of the font.
  3345. //
  3346. pUnicodeString = (PWSTR)( (LPBYTE)pFontMetrics +
  3347. pFontMetrics->dpwszFamilyName );
  3348. EngUnicodeToMultiByteN(fontName,
  3349. sizeof(fontName),
  3350. &fontNameLen,
  3351. pUnicodeString,
  3352. OEUnicodeStrlen(pUnicodeString));
  3353. //
  3354. // Search our Font Alias Table for the current family name. If we find
  3355. // it, replace it with the alias name from the table.
  3356. //
  3357. for (i = 0; i < NUM_ALIAS_FONTS; i++)
  3358. {
  3359. if (!strcmp((LPSTR)fontName,
  3360. (LPSTR)(fontAliasTable[i].pszOriginalFontName)))
  3361. {
  3362. TRACE_OUT(( "Alias name: %s -> %s",
  3363. (LPSTR)fontName,
  3364. (LPSTR)(fontAliasTable[i].pszAliasFontName)));
  3365. strcpy((LPSTR)fontName,
  3366. (LPSTR)(fontAliasTable[i].pszAliasFontName));
  3367. charWidthAdjustment = fontAliasTable[i].charWidthAdjustment;
  3368. break;
  3369. }
  3370. }
  3371. TRACE_OUT(( "Font name: '%s'", fontName));
  3372. //
  3373. // We have a font name to match with those we know to be available
  3374. // remotely. Try to jump straight to the first entry in the local font
  3375. // table starting with the same character as this font. If this index
  3376. // slot is empty (i.e. has a value of USHRT_MAX) then the loop will
  3377. // immediately exit
  3378. //
  3379. TRACE_OUT(( "Looking for matching fonts"));
  3380. for (iLocal = g_oeLocalFontIndex[(BYTE)fontName[0]];
  3381. iLocal < g_oeNumFonts;
  3382. iLocal++)
  3383. {
  3384. TRACE_OUT(( "Trying font number %hd", iLocal));
  3385. //
  3386. // If this font is not supported remotely then skip it.
  3387. //
  3388. ASSERT(g_poeLocalFonts);
  3389. matchQuality = g_poeLocalFonts[iLocal].SupportCode;
  3390. if (matchQuality == FH_SC_NO_MATCH)
  3391. {
  3392. continue;
  3393. }
  3394. //
  3395. // See if we've got a facename match
  3396. //
  3397. compareResult =
  3398. strcmp(g_poeLocalFonts[iLocal].Details.nfFaceName, fontName);
  3399. if (compareResult < 0)
  3400. {
  3401. //
  3402. // We haven't found a match yet, but we haven't gone far enough
  3403. // into this list.
  3404. //
  3405. continue;
  3406. }
  3407. else if (compareResult > 0)
  3408. {
  3409. //
  3410. // We're past the part of the local font array that's applicable.
  3411. // We didn't find a match, it must not exist.
  3412. //
  3413. break;
  3414. }
  3415. //
  3416. // The font names match. Now see if the other attributes do...
  3417. //
  3418. //
  3419. // This is looking promising - a font with the right name is
  3420. // supported on the remote system.
  3421. //
  3422. // Start building up the details in the global variables while
  3423. // making further checks...
  3424. //
  3425. *pFontFlags = 0;
  3426. *pFontIndex = iLocal;
  3427. *pFontWeight = pFontMetrics->usWinWeight;
  3428. //
  3429. // Check for a fixed pitch font.
  3430. //
  3431. if ((pFontMetrics->jWinPitchAndFamily & FIXED_PITCH) != 0)
  3432. {
  3433. *pFontFlags |= NF_FIXED_PITCH;
  3434. }
  3435. //
  3436. // Is it a TrueType font?
  3437. //
  3438. if ((pfo->flFontType & TRUETYPE_FONTTYPE) != 0)
  3439. {
  3440. *pFontFlags |= NF_TRUE_TYPE;
  3441. }
  3442. //
  3443. // Get the basic width and height.
  3444. //
  3445. xformSize[0].y = 0;
  3446. xformSize[0].x = 0;
  3447. xformSize[1].y = pFontMetrics->fwdUnitsPerEm;
  3448. xformSize[1].x = pFontMetrics->fwdAveCharWidth;
  3449. xformSize[2].y = pFontMetrics->fwdWinAscender;
  3450. xformSize[2].x = 0;
  3451. //
  3452. // We now need to convert these sizes if the GDI has provided a
  3453. // transform object.
  3454. //
  3455. if (pxform != NULL)
  3456. {
  3457. if (!XFORMOBJ_bApplyXform(pxform,
  3458. XF_LTOL,
  3459. 3,
  3460. &xformSize,
  3461. &xformSize))
  3462. {
  3463. ERROR_OUT(( "Xform failed"));
  3464. continue;
  3465. }
  3466. }
  3467. //
  3468. // Calculate the font width and height.
  3469. //
  3470. *pFontHeight = (UINT)(xformSize[1].y - xformSize[0].y);
  3471. *pFontWidth = (UINT)(xformSize[1].x - xformSize[0].x
  3472. - charWidthAdjustment);
  3473. TRACE_OUT(( "Device font size %hdx%hd", *pFontWidth, *pFontHeight));
  3474. //
  3475. // Get the offset to the start of the text cell.
  3476. //
  3477. *pFontAscender = (UINT)(xformSize[2].y - xformSize[0].y);
  3478. //
  3479. // Check that we have a matching pair - where we require that the
  3480. // fonts (ie the one being used by the application and the one
  3481. // we've matched with the remote system) are the same pitch (ie
  3482. // variable or fixed) and use the same technology (ie TrueType or
  3483. // not).
  3484. //
  3485. if ((g_poeLocalFonts[iLocal].Details.nfFontFlags & NF_FIXED_PITCH) !=
  3486. ((TSHR_UINT16)(*pFontFlags) & NF_FIXED_PITCH))
  3487. {
  3488. TRACE_OUT(( "Fixed pitch mismatch"));
  3489. continue;
  3490. }
  3491. if ((g_poeLocalFonts[iLocal].Details.nfFontFlags & NF_TRUE_TYPE) !=
  3492. ((TSHR_UINT16)*pFontFlags & NF_TRUE_TYPE))
  3493. {
  3494. TRACE_OUT(( "True type mismatch"));
  3495. continue;
  3496. }
  3497. //
  3498. // We have a pair of fonts with the same attributes - either both
  3499. // fixed pitch or both variable pitch - and using the same font
  3500. // technology.
  3501. //
  3502. // If the font is fixed pitch then we must also check that this
  3503. // particular size matches.
  3504. //
  3505. // If the font is not fixed pitch (scalable) then we assume that it
  3506. // is matchable.
  3507. //
  3508. if (g_poeLocalFonts[iLocal].Details.nfFontFlags & NF_FIXED_SIZE)
  3509. {
  3510. //
  3511. // The font is fixed size, so we must check that this
  3512. // particular size is matchable.
  3513. //
  3514. if ( (*pFontHeight != g_poeLocalFonts[iLocal].Details.nfAveHeight) ||
  3515. (*pFontWidth != g_poeLocalFonts[iLocal].Details.nfAveWidth) )
  3516. {
  3517. //
  3518. // The sizes differ, so we must fail this match.
  3519. //
  3520. TRACE_OUT(( "Size mismatch"));
  3521. continue;
  3522. }
  3523. }
  3524. //
  3525. // Hey! We've got a matched pair!
  3526. //
  3527. rc = TRUE;
  3528. TRACE_OUT(( "Found match at local font %hd", iLocal));
  3529. break;
  3530. }
  3531. if (rc != TRUE)
  3532. {
  3533. TRACE_OUT(( "Couldn't find matching font in table"));
  3534. DC_QUIT;
  3535. }
  3536. //
  3537. // Build up the rest of the font flags. We have already put the pitch
  3538. // flag in place.
  3539. //
  3540. if ( ((pFontMetrics->fsSelection & FM_SEL_ITALIC) != 0) ||
  3541. ((pfo->flFontType & FO_SIM_ITALIC) != 0) )
  3542. {
  3543. TRACE_OUT(( "Italic"));
  3544. *pFontFlags |= NF_ITALIC;
  3545. }
  3546. if ((pFontMetrics->fsSelection & FM_SEL_UNDERSCORE) != 0)
  3547. {
  3548. TRACE_OUT(( "Underline"));
  3549. *pFontFlags |= NF_UNDERLINE;
  3550. }
  3551. if ((pFontMetrics->fsSelection & FM_SEL_STRIKEOUT) != 0)
  3552. {
  3553. TRACE_OUT(( "Strikeout"));
  3554. *pFontFlags |= NF_STRIKEOUT;
  3555. }
  3556. //
  3557. // It is possible to have a font made bold by Windows, i.e. the
  3558. // standard font definition is not bold, but windows manipulates the
  3559. // font data to create a bold effect. This is marked by the
  3560. // FO_SIM_BOLD flag.
  3561. //
  3562. // In this case we need to ensure that the font flags are marked as
  3563. // bold according to the weight.
  3564. //
  3565. if ( ((pfo->flFontType & FO_SIM_BOLD) != 0) &&
  3566. ( pFontMetrics->usWinWeight < FW_BOLD) )
  3567. {
  3568. TRACE_OUT(( "Upgrading weight for a bold font"));
  3569. *pFontWeight = FW_BOLD;
  3570. }
  3571. //
  3572. // If the font is an exact match, or if it is an approximate match for
  3573. // its entire range (0x00 to 0xFF) then send it happily. If not...only
  3574. // send chars within the range 0x20->0x7F ("true ASCII").
  3575. //
  3576. ASSERT(g_poeLocalFonts);
  3577. if (codePage != g_poeLocalFonts[iLocal].Details.nfCodePage)
  3578. {
  3579. TRACE_OUT(( "Using different CP: downgrade to APPROX_ASC"));
  3580. matchQuality = FH_SC_APPROX_ASCII_MATCH;
  3581. }
  3582. //
  3583. // If we don't have an exact match, check the individual characters.
  3584. //
  3585. if ( (matchQuality != FH_SC_EXACT_MATCH ) &&
  3586. (matchQuality != FH_SC_APPROX_MATCH) )
  3587. {
  3588. //
  3589. // The approximate match is only valid if we use a font that
  3590. // supports the ANSI character set.
  3591. //
  3592. if ((pFontMetrics->jWinCharSet & ANSI_CHARSET) != 0)
  3593. {
  3594. TRACE_OUT(( "Cannot do match without ANSI support"));
  3595. DC_QUIT;
  3596. }
  3597. //
  3598. // This font is not a good match across its entire range. Check
  3599. // that all chars are within the desired range.
  3600. //
  3601. for (i = 0; i < textLen; i++)
  3602. {
  3603. if ( (pFontText[i] == 0) ||
  3604. ( (pFontText[i] >= NF_ASCII_FIRST) &&
  3605. (pFontText[i] <= NF_ASCII_LAST) ) )
  3606. {
  3607. continue;
  3608. }
  3609. //
  3610. // Can only get here by finding a char outside our acceptable
  3611. // range.
  3612. //
  3613. TRACE_OUT(( "found non ASCII char %x", pFontText[i]));
  3614. DC_QUIT;
  3615. }
  3616. }
  3617. //
  3618. // We have a valid font. Now sort out delta X issues.
  3619. //
  3620. //
  3621. // If we do not need to send delta X arrays then exit now.
  3622. //
  3623. if (!(g_oeFontCaps & CAPS_FONT_NEED_X_ALWAYS))
  3624. {
  3625. if (!(g_oeFontCaps & CAPS_FONT_NEED_X_SOMETIMES))
  3626. {
  3627. //
  3628. // CAPS_FONT_NEED_X_SOMETIMES and CAPS_FONT_NEED_X_ALWAYS are
  3629. // both not set so we can exit now. (We do not need a delta X
  3630. // array).
  3631. //
  3632. TRACE_OUT(( "Capabilities eliminated delta X"));
  3633. DC_QUIT;
  3634. }
  3635. //
  3636. // CAPS_FONT_NEED_X_SOMETIMES is set and CAPS_FONT_NEED_X_ALWAYS is
  3637. // not set. In this case whether we need a delta X is determined
  3638. // by whether the font is an exact match or an approximate match
  3639. // (because of either approximation of name, signature, or aspect
  3640. // ratio). We can only find this out after we have extracted the
  3641. // font handle from the existing order.
  3642. //
  3643. }
  3644. //
  3645. // If the string is a single character (or less) then we can just
  3646. // return.
  3647. //
  3648. if (textLen <= 1)
  3649. {
  3650. TRACE_OUT(( "String only %lu long", textLen));
  3651. DC_QUIT;
  3652. }
  3653. //
  3654. // Capabilities allow us to ignore delta X position if we have an exact
  3655. // match.
  3656. //
  3657. if ((matchQuality & FH_SC_EXACT) != 0)
  3658. {
  3659. //
  3660. // Exit immediately, providing that there is no override to always
  3661. // send increments.
  3662. //
  3663. if (!(g_oeFontCaps & CAPS_FONT_NEED_X_ALWAYS))
  3664. {
  3665. TRACE_OUT(( "Font has exact match"));
  3666. DC_QUIT;
  3667. }
  3668. }
  3669. //
  3670. // We have passed all the checks - we must send a delta X array.
  3671. //
  3672. TRACE_OUT(( "Must send delta X"));
  3673. *pSendDeltaX = TRUE;
  3674. DC_EXIT_POINT:
  3675. DebugExitDWORD(OECheckFontIsSupported, rc);
  3676. return(rc);
  3677. }
  3678. //
  3679. // Function: OELPtoVirtual
  3680. //
  3681. // Description: Adjusts window coordinates to virtual desktop coordinates.
  3682. // Clips the result to [+32766, -32768].
  3683. //
  3684. // Parameters: pPoints - Array of points to be converted
  3685. // cPoints - Number of points to be converted
  3686. //
  3687. // Returns: (none)
  3688. //
  3689. void OELPtoVirtual
  3690. (
  3691. LPPOINT aPts,
  3692. UINT cPts
  3693. )
  3694. {
  3695. int l;
  3696. TSHR_INT16 s;
  3697. DebugEntry(OELPtoVirtual);
  3698. //
  3699. // Convert to screen coordinates
  3700. //
  3701. while (cPts > 0)
  3702. {
  3703. //
  3704. // Look for int16 overflow in the X coordinate
  3705. //
  3706. l = aPts->x;
  3707. s = (TSHR_INT16)l;
  3708. if (l == (int)s)
  3709. {
  3710. aPts->x = s;
  3711. }
  3712. else
  3713. {
  3714. //
  3715. // HIWORD(l) will be 1 for positive overflow, 0xFFFF for
  3716. // negative overflow. Therefore we will get 0x7FFE or 0x8000
  3717. // (+32766 or -32768).
  3718. //
  3719. aPts->x = 0x7FFF - HIWORD(l);
  3720. TRACE_OUT(("adjusted X from %ld to %d", l, aPts->x));
  3721. }
  3722. //
  3723. // Look for int16 overflow in the Y coordinate
  3724. //
  3725. l = aPts->y;
  3726. s = (TSHR_INT16)l;
  3727. if (l == (int)s)
  3728. {
  3729. aPts->y = s;
  3730. }
  3731. else
  3732. {
  3733. //
  3734. // HIWORD(l) will be 1 for positive overflow, 0xFFFF for
  3735. // negative overflow. Therefore we will get 0x7FFE or 0x8000
  3736. // (+32766 or -32768).
  3737. //
  3738. aPts->y = 0x7FFF - HIWORD(l);
  3739. TRACE_OUT(("adjusted Y from %ld to %d", l, aPts->y));
  3740. }
  3741. //
  3742. // Move on to the next point
  3743. //
  3744. --cPts;
  3745. ++aPts;
  3746. }
  3747. DebugExitVOID(OELPtoVirtual);
  3748. }
  3749. //
  3750. // Function: OELRtoVirtual
  3751. //
  3752. // Description: Adjusts RECT in window coordinates to virtual coordinates.
  3753. // Clips the result to [+32766, -32768].
  3754. //
  3755. // Parameters: pRects - Array of rects to be converted
  3756. // numRects - Number of rects to be converted
  3757. //
  3758. // Returns: (none)
  3759. //
  3760. // NB. This function takes a Windows rectangle (exclusive coords) and
  3761. // returns a DC-Share rectangle (inclusive coords).
  3762. //
  3763. void OELRtoVirtual
  3764. (
  3765. LPRECT aRects,
  3766. UINT cRects
  3767. )
  3768. {
  3769. DebugEntry(OELRtoVirtual);
  3770. //
  3771. // Convert the points to screen coords, clipping to INT16s
  3772. //
  3773. OELPtoVirtual((LPPOINT)aRects, 2 * cRects);
  3774. //
  3775. // Make each rectangle inclusive
  3776. //
  3777. while (cRects > 0)
  3778. {
  3779. aRects->right--;
  3780. aRects->bottom--;
  3781. //
  3782. // Move on to the next rect
  3783. //
  3784. cRects--;
  3785. aRects++;
  3786. }
  3787. DebugExitVOID(OELRtoVirtual);
  3788. }
  3789. //
  3790. // Function: OEClipAndAddOrder
  3791. //
  3792. // Description: Adds the order to the order buffer, splitting it up into
  3793. // multiple orders if the clipping is complicated. If we fail
  3794. // to send the full order, we accumulate it in the SDA instead
  3795. //
  3796. // Parameters: pOrder - Order to be stored.
  3797. // pExtraInfo - Pointer to extra data associated with the
  3798. // order. This data depends on the order type,
  3799. // and may be NULL.
  3800. // pco - Clipping object for the area
  3801. //
  3802. // Returns: (none)
  3803. //
  3804. void OEClipAndAddOrder(LPINT_ORDER pOrder,
  3805. void * pExtraInfo,
  3806. CLIPOBJ* pco)
  3807. {
  3808. BOOL fOrderClipped;
  3809. BOOL fMoreRects;
  3810. RECT clippedRect;
  3811. RECT orderRect;
  3812. LPINT_ORDER pNewOrder;
  3813. LPINT_ORDER pLastOrder = NULL;
  3814. OE_ENUMRECTS clip;
  3815. UINT i;
  3816. UINT numRects = 0;
  3817. DebugEntry(OEClipAndAddOrder);
  3818. //
  3819. // Convert the order rectangle passed in (in virtual co-ordinates) back
  3820. // to screen co-ordinates. It is going to be clipped against clip
  3821. // rectangles returned to us in screen co-ordinates.
  3822. //
  3823. // Note that we also convert to exclusive coords here to make
  3824. // comparison with the exclusive Windows coords easier.
  3825. //
  3826. orderRect.left = pOrder->OrderHeader.Common.rcsDst.left;
  3827. orderRect.top = pOrder->OrderHeader.Common.rcsDst.top;
  3828. orderRect.right = pOrder->OrderHeader.Common.rcsDst.right + 1;
  3829. orderRect.bottom = pOrder->OrderHeader.Common.rcsDst.bottom + 1;
  3830. fOrderClipped = FALSE;
  3831. TRACE_OUT(( "orderRect: (%d,%d)(%d,%d)",
  3832. orderRect.left,
  3833. orderRect.top,
  3834. orderRect.right,
  3835. orderRect.bottom));
  3836. //
  3837. // Check if we have a clipping object at all.
  3838. //
  3839. if ((pco == NULL) || (pco->iDComplexity == DC_TRIVIAL))
  3840. {
  3841. //
  3842. // No clipping object - just use the bounds
  3843. //
  3844. clippedRect = orderRect;
  3845. fOrderClipped = TRUE;
  3846. pLastOrder = pOrder;
  3847. }
  3848. else if (pco->iDComplexity == DC_RECT)
  3849. {
  3850. //
  3851. // One clipping rectangle - use it directly.
  3852. //
  3853. RECT_FROM_RECTL(clippedRect, pco->rclBounds);
  3854. clippedRect.left = max(clippedRect.left, orderRect.left);
  3855. clippedRect.bottom = min(clippedRect.bottom, orderRect.bottom);
  3856. clippedRect.right = min(clippedRect.right, orderRect.right);
  3857. clippedRect.top = max(clippedRect.top, orderRect.top);
  3858. fOrderClipped = TRUE;
  3859. pLastOrder = pOrder;
  3860. }
  3861. else
  3862. {
  3863. //
  3864. // OA can only cope as long as the orders are added in the same
  3865. // order that they were allocated, so we need to do a little
  3866. // shuffling here.
  3867. //
  3868. // We always keep one order outstanding (pLastOrder) and a flag to
  3869. // indicate if it is valid (fOrderClipped). The first time we find
  3870. // a valid clipping rectangle, we set up pLastOrder and
  3871. // fOrderClipped. If we find we need to allocate a new order, we
  3872. // request the memory for the new order (pNewOrder), add pLastOrder
  3873. // and store pNewOrder in pLastOrder.
  3874. //
  3875. // Once we have finished enumerating the clipping rectangles, if
  3876. // pLastOrder is valid, we add it in.
  3877. //
  3878. // Also, while we are adding all these orders, OA must not purge
  3879. // the order heap otherwise we'll be left holding an invalid
  3880. // pointer.
  3881. //
  3882. pNewOrder = pOrder;
  3883. g_oaPurgeAllowed = FALSE;
  3884. //
  3885. // Multiple clipping rectangles - Enumerate all the rectangles
  3886. // involved in this drawing operation.
  3887. // The documentation for this function incorrectly states that
  3888. // the returned value is the total number of rectangles
  3889. // comprising the clip region. In fact, -1 is always returned,
  3890. // even when the final parameter is non-zero.
  3891. //
  3892. CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES, CD_ANY, 0);
  3893. //
  3894. // Get the clip rectangles. We fetch these into the clip buffer
  3895. // which is big enough to get all the clip rectangles we expect + 1.
  3896. // If the order runs across this number of clip rects or more then
  3897. // we will already have decided to send it as screen data.
  3898. // The clip rectangle fetching is contained within a loop because,
  3899. // while we expect to call CLIPOBJ_bEnum once only, it is possible
  3900. // for this functio to return zero rects and report that there are
  3901. // more to fetch (according to MSDN).
  3902. //
  3903. do
  3904. {
  3905. fMoreRects = CLIPOBJ_bEnum(pco,
  3906. sizeof(clip),
  3907. (ULONG *)&clip.rects);
  3908. //
  3909. // The clipping object can decide that there are no more
  3910. // rectangles and that this query has returned no rectangles,
  3911. // so we must check for any valid data in the returned
  3912. // rectangle list.
  3913. //
  3914. if (clip.rects.c == 0)
  3915. {
  3916. //
  3917. // We didn't get any rects this time so go round again - if
  3918. // we're finished, the loop termination condition will take
  3919. // us out. CLIPOBJ_bEnum can return a count of zero when
  3920. // there are still more rects.
  3921. //
  3922. TRACE_OUT(( "No rects this time, more %u", fMoreRects));
  3923. continue;
  3924. }
  3925. //
  3926. // To get to here we expect to have fetched all the rects and
  3927. // no more. Do a quick check.
  3928. //
  3929. numRects += clip.rects.c;
  3930. ASSERT( (numRects <= COMPLEX_CLIP_RECT_COUNT) );
  3931. //
  3932. // Process each clip rectangle by clipping the drawing order to
  3933. // it.
  3934. //
  3935. for ( i = 0; i < clip.rects.c; i++ )
  3936. {
  3937. TRACE_OUT(( " (%d,%d)(%d,%d)",
  3938. clip.rects.arcl[i].left,
  3939. clip.rects.arcl[i].top,
  3940. clip.rects.arcl[i].right,
  3941. clip.rects.arcl[i].bottom));
  3942. //
  3943. // Check for an intersection
  3944. //
  3945. if ( (clip.rects.arcl[i].left >= orderRect.right) ||
  3946. (clip.rects.arcl[i].bottom <= orderRect.top) ||
  3947. (clip.rects.arcl[i].right <= orderRect.left) ||
  3948. (clip.rects.arcl[i].top >= orderRect.bottom) )
  3949. {
  3950. //
  3951. // No intersection, move on to next clip rect.
  3952. //
  3953. continue;
  3954. }
  3955. //
  3956. // There is an intersection, so we may need to add a new
  3957. // order to the buffer to cater for this rectangle.
  3958. //
  3959. if (fOrderClipped)
  3960. {
  3961. //
  3962. // The order has already been clipped once, so it
  3963. // actually intersects more than one clip rect, ie
  3964. // fOrderClipped is always FALSE for at least the first
  3965. // clip rectangle in the clip.rects buffer. We cope
  3966. // with this by duplicating the order and clipping it
  3967. // again.
  3968. //
  3969. pNewOrder = OA_DDAllocOrderMem(
  3970. pLastOrder->OrderHeader.Common.cbOrderDataLength, 0);
  3971. if (pNewOrder == NULL)
  3972. {
  3973. WARNING_OUT(( "Order memory allocation failed" ));
  3974. goto CLIP_ORDER_FAILED;
  3975. }
  3976. //
  3977. // Copy the header & data from the original order to
  3978. // the new order (making sure that we don't overwrite
  3979. // the list information at the start of the header).
  3980. //
  3981. memcpy((LPBYTE)pNewOrder
  3982. + FIELD_SIZE(INT_ORDER, OrderHeader.list),
  3983. (LPBYTE)pLastOrder
  3984. + FIELD_SIZE(INT_ORDER, OrderHeader.list),
  3985. pLastOrder->OrderHeader.Common.cbOrderDataLength
  3986. + sizeof(INT_ORDER_HEADER)
  3987. - FIELD_SIZE(INT_ORDER, OrderHeader.list));
  3988. //
  3989. // Set the destination (clip) rectangle (in virtual
  3990. // desktop coordinates).
  3991. //
  3992. TSHR_RECT16_FROM_RECT(
  3993. &pLastOrder->OrderHeader.Common.rcsDst,
  3994. clippedRect);
  3995. pLastOrder->OrderHeader.Common.rcsDst.right -= 1;
  3996. pLastOrder->OrderHeader.Common.rcsDst.bottom -= 1;
  3997. TRACE_OUT(( "Adding duplicate order (%d,%d) (%d,%d)",
  3998. pLastOrder->OrderHeader.Common.rcsDst.left,
  3999. pLastOrder->OrderHeader.Common.rcsDst.top,
  4000. pLastOrder->OrderHeader.Common.rcsDst.right,
  4001. pLastOrder->OrderHeader.Common.rcsDst.bottom));
  4002. //
  4003. // Add the order to the Order List.
  4004. //
  4005. OA_DDAddOrder(pLastOrder, pExtraInfo);
  4006. }
  4007. //
  4008. // Update the clipping rectangle for the order to be sent.
  4009. //
  4010. clippedRect.left = max(clip.rects.arcl[i].left,
  4011. orderRect.left);
  4012. clippedRect.bottom= min(clip.rects.arcl[i].bottom,
  4013. orderRect.bottom);
  4014. clippedRect.right = min(clip.rects.arcl[i].right,
  4015. orderRect.right);
  4016. clippedRect.top = max(clip.rects.arcl[i].top,
  4017. orderRect.top);
  4018. fOrderClipped = TRUE;
  4019. pLastOrder = pNewOrder;
  4020. }
  4021. } while (fMoreRects);
  4022. }
  4023. //
  4024. // Check whether the clipping has removed the order entirely.
  4025. //
  4026. if (fOrderClipped)
  4027. {
  4028. TSHR_RECT16_FROM_RECT(&pLastOrder->OrderHeader.Common.rcsDst,
  4029. clippedRect);
  4030. pLastOrder->OrderHeader.Common.rcsDst.right -= 1;
  4031. pLastOrder->OrderHeader.Common.rcsDst.bottom -= 1;
  4032. TRACE_OUT(( "Adding order (%d,%d) (%d,%d)",
  4033. pLastOrder->OrderHeader.Common.rcsDst.left,
  4034. pLastOrder->OrderHeader.Common.rcsDst.top,
  4035. pLastOrder->OrderHeader.Common.rcsDst.right,
  4036. pLastOrder->OrderHeader.Common.rcsDst.bottom));
  4037. //
  4038. // Add the order to the Order List.
  4039. //
  4040. OA_DDAddOrder(pLastOrder, pExtraInfo);
  4041. }
  4042. else
  4043. {
  4044. TRACE_OUT(( "Order clipped completely"));
  4045. OA_DDFreeOrderMem(pOrder);
  4046. }
  4047. DC_QUIT;
  4048. CLIP_ORDER_FAILED:
  4049. //
  4050. // Allocation of memory for a duplicate order failed. Just add the
  4051. // original order's destination rect into the SDA and free the order.
  4052. //
  4053. // The order rectangle is already in inclusive virtual coordinates.
  4054. //
  4055. TRACE_OUT(( "Order add failed, add to SDA"));
  4056. RECT_FROM_TSHR_RECT16(&orderRect,pLastOrder->OrderHeader.Common.rcsDst);
  4057. OA_DDFreeOrderMem(pLastOrder);
  4058. BA_AddScreenData(&orderRect);
  4059. DC_EXIT_POINT:
  4060. //
  4061. // Make sure that we always re-enable heap purging.
  4062. //
  4063. g_oaPurgeAllowed = TRUE;
  4064. DebugExitVOID(OEClipAndAddOrder);
  4065. }
  4066. //
  4067. // Function: OEClipAndAddScreenData
  4068. //
  4069. // Description: Determines if we need to accumulate any screen data for the
  4070. // specified area. If so, it is added to the SDA.
  4071. //
  4072. // Parameters: pRect - Bounding rectangle of area to be accumulated
  4073. // pco - Clipping object for the area
  4074. //
  4075. // Returns: (none)
  4076. //
  4077. void OEClipAndAddScreenData(LPRECT pRect, CLIPOBJ* pco)
  4078. {
  4079. RECT SDACandidate;
  4080. BOOL fMoreRects;
  4081. RECT clippedRect;
  4082. OE_ENUMRECTS clip;
  4083. UINT i;
  4084. DebugEntry(OEClipAndAddScreenData);
  4085. //
  4086. // Convert the order rectangle passed in (in virtual co-ordinates) back
  4087. // to screen co-ordinates. It is going to be clipped against clip
  4088. // rectangles returned to us in screen co-ordinates.
  4089. //
  4090. // Note that we also convert to exclusive coords here to make
  4091. // comparison with the exclusive Windows coords easier.
  4092. //
  4093. SDACandidate.left = pRect->left;
  4094. SDACandidate.top = pRect->top;
  4095. SDACandidate.right = pRect->right + 1;
  4096. SDACandidate.bottom = pRect->bottom + 1;
  4097. TRACE_OUT(( "SDACandidate: (%d,%d)(%d,%d)",
  4098. SDACandidate.left,
  4099. SDACandidate.top,
  4100. SDACandidate.right,
  4101. SDACandidate.bottom));
  4102. //
  4103. // Check if we have a clipping object at all.
  4104. //
  4105. if ((pco == NULL) || (pco->iDComplexity == DC_TRIVIAL))
  4106. {
  4107. //
  4108. // Convert the clipped rect into Virtual Desktop coords.
  4109. //
  4110. clippedRect = SDACandidate;
  4111. clippedRect.right -= 1;
  4112. clippedRect.bottom -= 1;
  4113. //
  4114. // Add the clipped rect into the SDA.
  4115. //
  4116. TRACE_OUT(( "Adding SDA (%d,%d)(%d,%d)", clippedRect.left,
  4117. clippedRect.top,
  4118. clippedRect.right,
  4119. clippedRect.bottom));
  4120. BA_AddScreenData(&clippedRect);
  4121. }
  4122. else if (pco->iDComplexity == DC_RECT)
  4123. {
  4124. //
  4125. // One clipping rectangle - use it directly, converting into
  4126. // Virtual Desktop coords. Make sure the rectangle is valid before
  4127. // adding to the SDA.
  4128. //
  4129. RECT_FROM_RECTL(clippedRect, pco->rclBounds);
  4130. clippedRect.left = max(clippedRect.left, SDACandidate.left);
  4131. clippedRect.right = min(clippedRect.right, SDACandidate.right) + -1;
  4132. if ( clippedRect.left <= clippedRect.right )
  4133. {
  4134. clippedRect.bottom = min(clippedRect.bottom,
  4135. SDACandidate.bottom) + -1;
  4136. clippedRect.top = max(clippedRect.top, SDACandidate.top);
  4137. if ( clippedRect.bottom >= clippedRect.top )
  4138. {
  4139. //
  4140. // Add the clipped rect into the SDA.
  4141. //
  4142. TRACE_OUT(( "Adding SDA RECT (%d,%d)(%d,%d)",
  4143. clippedRect.left,
  4144. clippedRect.top,
  4145. clippedRect.right,
  4146. clippedRect.bottom));
  4147. BA_AddScreenData(&clippedRect);
  4148. }
  4149. }
  4150. }
  4151. else
  4152. {
  4153. //
  4154. // Enumerate all the rectangles involved in this drawing operation.
  4155. // The documentation for this function incorrectly states that
  4156. // the returned value is the total number of rectangles
  4157. // comprising the clip region. In fact, -1 is always returned,
  4158. // even when the final parameter is non-zero.
  4159. //
  4160. CLIPOBJ_cEnumStart(pco, FALSE, CT_RECTANGLES, CD_ANY, 0);
  4161. do
  4162. {
  4163. //
  4164. // Get the next batch of clipping rectangles
  4165. //
  4166. fMoreRects = CLIPOBJ_bEnum(pco,
  4167. sizeof(clip),
  4168. (ULONG *)&clip.rects);
  4169. for ( i = 0; i < clip.rects.c; i++ )
  4170. {
  4171. TRACE_OUT(( " (%d,%d)(%d,%d)",
  4172. clip.rects.arcl[i].left,
  4173. clip.rects.arcl[i].top,
  4174. clip.rects.arcl[i].right,
  4175. clip.rects.arcl[i].bottom));
  4176. //
  4177. // Intersect the SDA rect with the clip rect, checking for
  4178. // no intersection.
  4179. //
  4180. clippedRect.left = max( clip.rects.arcl[i].left,
  4181. SDACandidate.left );
  4182. clippedRect.right = min( clip.rects.arcl[i].right,
  4183. SDACandidate.right );
  4184. if (clippedRect.left >= clippedRect.right)
  4185. {
  4186. //
  4187. // No horizontal intersection.
  4188. //
  4189. continue;
  4190. }
  4191. clippedRect.bottom = min( clip.rects.arcl[i].bottom,
  4192. SDACandidate.bottom );
  4193. clippedRect.top = max( clip.rects.arcl[i].top,
  4194. SDACandidate.top );
  4195. if (clippedRect.top >= clippedRect.bottom)
  4196. {
  4197. //
  4198. // No vertical intersection.
  4199. //
  4200. continue;
  4201. }
  4202. //
  4203. // Convert the clipped rect into Virtual Desktop coords.
  4204. //
  4205. clippedRect.right -= 1;
  4206. clippedRect.bottom -= 1;
  4207. //
  4208. // Add the clipped rect into the SDA.
  4209. //
  4210. TRACE_OUT(( "Adding SDA (%d,%d)(%d,%d)",
  4211. clippedRect.left,
  4212. clippedRect.top,
  4213. clippedRect.right,
  4214. clippedRect.bottom));
  4215. BA_AddScreenData(&clippedRect);
  4216. }
  4217. } while (fMoreRects);
  4218. }
  4219. DebugExitVOID(OEClipAndAddScreenData);
  4220. }
  4221. //
  4222. // FUNCTION: OEDDSetNewFonts
  4223. //
  4224. // DESCRIPTION:
  4225. //
  4226. // Set the new font handling information to be used by the display driver.
  4227. //
  4228. // RETURNS:
  4229. //
  4230. // NONE
  4231. //
  4232. //
  4233. void OEDDSetNewFonts(LPOE_NEW_FONTS pRequest)
  4234. {
  4235. UINT cbNewSize;
  4236. DebugEntry(OEDDSetNewFonts);
  4237. TRACE_OUT(( "New fonts %d", pRequest->countFonts));
  4238. //
  4239. // Initialize new number of fonts to zero in case an error happens.
  4240. // We don't want to use stale font info if so.
  4241. //
  4242. g_oeNumFonts = 0;
  4243. g_oeFontCaps = pRequest->fontCaps;
  4244. //
  4245. // Free our previous font block if we had one.
  4246. //
  4247. if (g_poeLocalFonts)
  4248. {
  4249. EngFreeMem(g_poeLocalFonts);
  4250. g_poeLocalFonts = NULL;
  4251. }
  4252. //
  4253. // Alloc a new one, the size of the new font block.
  4254. //
  4255. cbNewSize = pRequest->countFonts * sizeof(LOCALFONT);
  4256. g_poeLocalFonts = EngAllocMem(0, cbNewSize, OSI_ALLOC_TAG);
  4257. if (! g_poeLocalFonts)
  4258. {
  4259. ERROR_OUT(("OEDDSetNewFonts: can't allocate space for font info"));
  4260. DC_QUIT;
  4261. }
  4262. //
  4263. // OK, if we're here, this is going to succeed. Copy the info over.
  4264. //
  4265. g_oeNumFonts = pRequest->countFonts;
  4266. memcpy(g_poeLocalFonts, pRequest->fontData, cbNewSize);
  4267. memcpy(g_oeLocalFontIndex, pRequest->fontIndex,
  4268. sizeof(g_oeLocalFontIndex[0]) * FH_LOCAL_INDEX_SIZE);
  4269. DC_EXIT_POINT:
  4270. DebugExitVOID(OEDDSetNewFonts);
  4271. }
  4272. //
  4273. // FUNCTION: OEDDSetNewCapabilities
  4274. //
  4275. // DESCRIPTION:
  4276. //
  4277. // Set the new OE related capabilities
  4278. //
  4279. // RETURNS:
  4280. //
  4281. // NONE
  4282. //
  4283. // PARAMETERS:
  4284. //
  4285. // pDataIn - pointer to the input buffer
  4286. //
  4287. //
  4288. void OEDDSetNewCapabilities(LPOE_NEW_CAPABILITIES pCapabilities)
  4289. {
  4290. DebugEntry(OEDDSetNewCapabilities);
  4291. //
  4292. // Copy the data from the Share Core.
  4293. //
  4294. g_oeBaselineTextEnabled = pCapabilities->baselineTextEnabled;
  4295. g_oeSendOrders = pCapabilities->sendOrders;
  4296. g_oeTextEnabled = pCapabilities->textEnabled;
  4297. //
  4298. // The share core has passed down a pointer to it's copy of the order
  4299. // support array. We take a copy for the kernel here.
  4300. //
  4301. memcpy(g_oeOrderSupported,
  4302. pCapabilities->orderSupported,
  4303. sizeof(g_oeOrderSupported));
  4304. TRACE_OUT(( "OE caps: BLT %c Orders %c Text %c",
  4305. g_oeBaselineTextEnabled ? 'Y': 'N',
  4306. g_oeSendOrders ? 'Y': 'N',
  4307. g_oeTextEnabled ? 'Y': 'N'));
  4308. DebugExitVOID(OEDDSetNewCapabilities);
  4309. }
  4310. //
  4311. // Function: OETileBitBltOrder
  4312. //
  4313. // Description: Divides a single large BitBlt order into a series of small,
  4314. // "tiled" BitBlt orders, each of which is added to the order
  4315. // queue.
  4316. //
  4317. // Parameters: pOrder - Template order to be tiled
  4318. // pExtraInfo - Structure containing pointers to the source
  4319. // and destination surface objects, and a pointer
  4320. // to the color translation object for the Blt
  4321. // pco - Clipping object for the operation
  4322. //
  4323. // Returns: TRUE - Stored in orders (and possibly some SDA)
  4324. // FALSE- Stored in SDA (or contained bad data)
  4325. //
  4326. //
  4327. void OETileBitBltOrder
  4328. (
  4329. LPINT_ORDER pOrder,
  4330. LPMEMBLT_ORDER_EXTRA_INFO pExtraInfo,
  4331. CLIPOBJ* pco
  4332. )
  4333. {
  4334. UINT tileWidth;
  4335. UINT tileHeight;
  4336. int srcLeft;
  4337. int srcTop;
  4338. int srcRight;
  4339. int srcBottom;
  4340. int xFirstTile;
  4341. int yFirstTile;
  4342. int xTile;
  4343. int yTile;
  4344. UINT type;
  4345. int bmpWidth, bmpHeight;
  4346. RECT destRect;
  4347. DebugEntry(OETileBitBltOrder);
  4348. //
  4349. // Extract the src bitmap handle from the Order - if the order is not a
  4350. // memory to screen blit, we get out now.
  4351. //
  4352. type = ((LPMEMBLT_ORDER)pOrder->abOrderData)->type;
  4353. switch (type)
  4354. {
  4355. case ORD_MEMBLT_TYPE:
  4356. {
  4357. srcLeft = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nXSrc;
  4358. srcTop = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nYSrc;
  4359. srcRight = srcLeft +
  4360. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nWidth;
  4361. srcBottom = srcTop +
  4362. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nHeight;
  4363. destRect.left = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nLeftRect;
  4364. destRect.top = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nTopRect;
  4365. destRect.right = destRect.left +
  4366. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nWidth;
  4367. destRect.bottom= destRect.top +
  4368. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nHeight;
  4369. }
  4370. break;
  4371. case ORD_MEM3BLT_TYPE:
  4372. {
  4373. srcLeft = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nXSrc;
  4374. srcTop = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nYSrc;
  4375. srcRight = srcLeft +
  4376. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nWidth;
  4377. srcBottom = srcTop +
  4378. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nHeight;
  4379. destRect.left = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nLeftRect;
  4380. destRect.top = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nTopRect;
  4381. destRect.right= destRect.left +
  4382. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nWidth;
  4383. destRect.bottom = destRect.top +
  4384. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nHeight;
  4385. }
  4386. break;
  4387. default:
  4388. {
  4389. ERROR_OUT(( "Invalid order type %u", type));
  4390. }
  4391. break;
  4392. }
  4393. //
  4394. // Fetch the bitmap details.
  4395. //
  4396. bmpWidth = (int)pExtraInfo->pSource->sizlBitmap.cx;
  4397. bmpHeight = (int)pExtraInfo->pSource->sizlBitmap.cy;
  4398. if (!SBC_DDQueryBitmapTileSize(bmpWidth, bmpHeight, &tileWidth, &tileHeight))
  4399. {
  4400. //
  4401. // This could happen if some 2.x user joins the share.
  4402. //
  4403. TRACE_OUT(("Bitmap is not tileable"));
  4404. OEClipAndAddScreenData(&destRect, pco);
  4405. }
  4406. else
  4407. {
  4408. //
  4409. // Tile the order. If an individual tile fails to get queued as an
  4410. // order, OEAddTiledBitBltOrder() will add it as screen data. Hence
  4411. // no return value to be checked.
  4412. //
  4413. xFirstTile = srcLeft - (srcLeft % tileWidth);
  4414. yFirstTile = srcTop - (srcTop % tileHeight);
  4415. for (yTile = yFirstTile; yTile < srcBottom; yTile += tileHeight)
  4416. {
  4417. for (xTile = xFirstTile; xTile < srcRight; xTile += tileWidth)
  4418. {
  4419. OEAddTiledBitBltOrder(pOrder, pExtraInfo, pco, xTile, yTile,
  4420. tileWidth, tileHeight);
  4421. }
  4422. }
  4423. }
  4424. DebugExitVOID(OETileBitBltOrder);
  4425. }
  4426. //
  4427. // Function: OEAddTiledBitBltOrder
  4428. //
  4429. // Description: Takes an unmodified "large" BitBlt and a tile rectangle,
  4430. // makes a copy of the order and modifies the copied order's
  4431. // src/dest so it applies to the source tile only. The order
  4432. // is added to the order queue. If the allocation of the
  4433. // "tiled" order fails, the destination rect is added to SDA
  4434. //
  4435. // Parameters: pOrder - Template order to be added
  4436. // pExtraInfo - Pointer to the extra BitBlt info
  4437. // pco - Clipping object for the BitBlt
  4438. // xTile - X position of the tile
  4439. // yTile - Y position of the tile
  4440. // tileWidth - tile width
  4441. // tileHeight - tile height
  4442. //
  4443. // Returns: none
  4444. //
  4445. //
  4446. void OEAddTiledBitBltOrder(
  4447. LPINT_ORDER pOrder,
  4448. LPMEMBLT_ORDER_EXTRA_INFO pExtraInfo,
  4449. CLIPOBJ* pco,
  4450. int xTile,
  4451. int yTile,
  4452. UINT tileWidth,
  4453. UINT tileHeight)
  4454. {
  4455. LPINT_ORDER pTileOrder;
  4456. LPINT pXSrc = NULL;
  4457. LPINT pYSrc = NULL;
  4458. LPINT pLeft = NULL;
  4459. LPINT pTop = NULL;
  4460. LPINT pWidth = NULL;
  4461. LPINT pHeight = NULL;
  4462. RECT srcRect;
  4463. RECT destRect;
  4464. UINT type;
  4465. DebugEntry(OETileAndAddBitBltOrder);
  4466. //
  4467. // This is a trusted interface - assume the type is correct
  4468. //
  4469. type = ((LPMEMBLT_ORDER)pOrder->abOrderData)->type;
  4470. ASSERT(((type == ORD_MEMBLT_TYPE) || (type == ORD_MEM3BLT_TYPE)));
  4471. //
  4472. // Do processing which depends on the type of bit blt being tiled:
  4473. // - save existing src and dest rects
  4474. // - make a copy of the order (which will be the tile order)
  4475. // - save pointers to the fields in the tile order which we're likely
  4476. // to change.
  4477. //
  4478. if (type == ORD_MEMBLT_TYPE)
  4479. {
  4480. srcRect.left = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nXSrc;
  4481. srcRect.top = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nYSrc;
  4482. srcRect.right = srcRect.left +
  4483. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nWidth;
  4484. srcRect.bottom = srcRect.top +
  4485. ((LPMEMBLT_ORDER)pOrder->abOrderData)->nHeight;
  4486. destRect.left = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nLeftRect;
  4487. destRect.top = ((LPMEMBLT_ORDER)pOrder->abOrderData)->nTopRect;
  4488. //
  4489. // We must allocate enough space for the maximum size order that
  4490. // SBC may use (i.e. an R2 order). We default to filling in the
  4491. // data as an R1 order.
  4492. //
  4493. pTileOrder = OA_DDAllocOrderMem(sizeof(MEMBLT_R2_ORDER),0);
  4494. if (pTileOrder == NULL)
  4495. {
  4496. TRACE_OUT(( "No space for tile order"));
  4497. DC_QUIT;
  4498. }
  4499. //
  4500. // We must not mess up the linked list data in the orders.
  4501. //
  4502. RtlCopyMemory(((LPBYTE)pTileOrder) +
  4503. FIELD_SIZE(INT_ORDER, OrderHeader.list),
  4504. ((LPBYTE)pOrder) +
  4505. FIELD_SIZE(INT_ORDER, OrderHeader.list),
  4506. sizeof(INT_ORDER_HEADER)
  4507. + sizeof(MEMBLT_R2_ORDER)
  4508. - FIELD_SIZE(INT_ORDER, OrderHeader.list));
  4509. pXSrc = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nXSrc;
  4510. pYSrc = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nYSrc;
  4511. pWidth = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nWidth;
  4512. pHeight = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nHeight;
  4513. pLeft = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nLeftRect;
  4514. pTop = &((LPMEMBLT_ORDER)pTileOrder->abOrderData)->nTopRect;
  4515. }
  4516. else
  4517. {
  4518. srcRect.left = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nXSrc;
  4519. srcRect.top = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nYSrc;
  4520. srcRect.right = srcRect.left +
  4521. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nWidth;
  4522. srcRect.bottom = srcRect.top +
  4523. ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nHeight;
  4524. destRect.left = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nLeftRect;
  4525. destRect.top = ((LPMEM3BLT_ORDER)pOrder->abOrderData)->nTopRect;
  4526. //
  4527. // We must allocate enough space for the maximum size order that
  4528. // SBC may use (i.e. an R2 order). We default to filling in the
  4529. // data as an R1 order.
  4530. //
  4531. pTileOrder = OA_DDAllocOrderMem(sizeof(MEM3BLT_R2_ORDER),0);
  4532. if (pTileOrder == NULL)
  4533. {
  4534. TRACE_OUT(( "No space for tile order"));
  4535. DC_QUIT;
  4536. }
  4537. //
  4538. // We must not mess up the linked list data in the orders.
  4539. //
  4540. RtlCopyMemory(((LPBYTE)pTileOrder) +
  4541. FIELD_SIZE(INT_ORDER, OrderHeader.list),
  4542. ((LPBYTE)pOrder) +
  4543. FIELD_SIZE(INT_ORDER, OrderHeader.list),
  4544. sizeof(INT_ORDER_HEADER)
  4545. + sizeof(MEM3BLT_R2_ORDER)
  4546. - FIELD_SIZE(INT_ORDER, OrderHeader.list));
  4547. pXSrc = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nXSrc;
  4548. pYSrc = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nYSrc;
  4549. pWidth = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nWidth;
  4550. pHeight = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nHeight;
  4551. pLeft = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nLeftRect;
  4552. pTop = &((LPMEM3BLT_ORDER)pTileOrder->abOrderData)->nTopRect;
  4553. }
  4554. TRACE_OUT(( "Tiling order, orig srcLeft=%hd, srcTop=%hd, srcRight=%hd, "
  4555. "srcBottom=%hd, destX=%hd, destY=%hd; "
  4556. "xTile=%hd, yTile=%hd, tileW=%hd, tileH=%hd",
  4557. srcRect.left, srcRect.top, srcRect.right, srcRect.bottom,
  4558. destRect.left, destRect.top,
  4559. xTile, yTile, tileWidth, tileHeight));
  4560. DC_EXIT_POINT:
  4561. //
  4562. // NOTE: ALL THE POINTERS MAY BE NULL AT THIS POINT - DO NOT USE THEM
  4563. // UNTIL YOU VERIFY PTILEORDER IS NON-NULL.
  4564. //
  4565. // Intersect source and tile rects, and set up destination rect
  4566. // accordingly - we need to do this even if we failed to copy the
  4567. // order, because the tiled source rect will have to be added to the
  4568. // screen data area.
  4569. //
  4570. if (xTile > srcRect.left)
  4571. {
  4572. destRect.left += (xTile - srcRect.left);
  4573. srcRect.left = xTile;
  4574. }
  4575. if (yTile > srcRect.top)
  4576. {
  4577. destRect.top += (yTile - srcRect.top);
  4578. srcRect.top = yTile;
  4579. }
  4580. srcRect.right = min((UINT)srcRect.right, xTile + tileWidth);
  4581. srcRect.bottom = min((UINT)srcRect.bottom, yTile + tileHeight);
  4582. destRect.right = destRect.left + (srcRect.right - srcRect.left);
  4583. destRect.bottom = destRect.top + (srcRect.bottom - srcRect.top);
  4584. //
  4585. // If the order was successfully copied above, then modify the order
  4586. // to contain the tiled coordinates, and add it to the order list.
  4587. // Otherwise, send the dest rect as screen data.
  4588. //
  4589. if (pTileOrder != NULL)
  4590. {
  4591. TRACE_OUT(( "Tile order originally: srcX=%hd, srcY=%hd, destX=%hd, "
  4592. "destY=%hd, w=%hd, h=%hd",
  4593. *pXSrc, *pYSrc, *pLeft, *pTop, *pWidth, *pHeight));
  4594. *pXSrc = srcRect.left;
  4595. *pYSrc = srcRect.top;
  4596. *pLeft = destRect.left;
  4597. *pTop = destRect.top;
  4598. *pWidth = srcRect.right - srcRect.left;
  4599. *pHeight = srcRect.bottom - srcRect.top;
  4600. pTileOrder->OrderHeader.Common.rcsDst.left = (TSHR_INT16)destRect.left;
  4601. pTileOrder->OrderHeader.Common.rcsDst.right = (TSHR_INT16)destRect.right;
  4602. pTileOrder->OrderHeader.Common.rcsDst.top = (TSHR_INT16)destRect.top;
  4603. pTileOrder->OrderHeader.Common.rcsDst.bottom =
  4604. (TSHR_INT16)destRect.bottom;
  4605. TRACE_OUT(( "Adding order srcX=%hd, srcY=%hd, destX=%hd, destY=%hd,"
  4606. " w=%hd, h=%hd",
  4607. *pXSrc, *pYSrc, *pLeft, *pTop, *pWidth, *pHeight));
  4608. OEClipAndAddOrder(pTileOrder, pExtraInfo, pco);
  4609. }
  4610. else
  4611. {
  4612. TRACE_OUT(( "Failed to allocate order - sending as screen data"));
  4613. OEClipAndAddScreenData(&destRect, pco);
  4614. }
  4615. DebugExitVOID(OETileAndAddBitBltOrder);
  4616. }
  4617. // NAME: OEAddLine
  4618. //
  4619. // PURPOSE:
  4620. //
  4621. // Add a LineTo order to the order heap.
  4622. //
  4623. // RETURNS:
  4624. //
  4625. // TRUE - Attempted to add to heap
  4626. // FALSE - No room left to allocate an order
  4627. //
  4628. // PARAMS:
  4629. //
  4630. // ppdev - display driver PDEV
  4631. // startPoint - start point of line
  4632. // endPoint - end point of line
  4633. // rectDst - bounding rectangle
  4634. // rop2 - ROP2 to use with line
  4635. // width - width of line to add
  4636. // color - color of line to add
  4637. // pco - clipping object for drawing operation
  4638. //
  4639. BOOL OEAddLine(LPOSI_PDEV ppdev,
  4640. LPPOINT startPoint,
  4641. LPPOINT endPoint,
  4642. LPRECT rectDst,
  4643. UINT rop2,
  4644. UINT width,
  4645. UINT color,
  4646. CLIPOBJ* pco)
  4647. {
  4648. BOOL rc = FALSE;
  4649. LPLINETO_ORDER pLineTo;
  4650. LPINT_ORDER pOrder;
  4651. DebugEntry(OEAddLine);
  4652. //
  4653. // Allocate the memory for the order.
  4654. //
  4655. pOrder = OA_DDAllocOrderMem(sizeof(LINETO_ORDER),0);
  4656. if (pOrder == NULL)
  4657. {
  4658. TRACE_OUT(( "Failed to alloc order"));
  4659. DC_QUIT;
  4660. }
  4661. pLineTo = (LPLINETO_ORDER)pOrder->abOrderData;
  4662. //
  4663. // Mark this order type.
  4664. //
  4665. pLineTo->type = ORD_LINETO_TYPE;
  4666. //
  4667. // Store the line end coordinates.
  4668. //
  4669. pLineTo->nXStart = startPoint->x;
  4670. pLineTo->nYStart = startPoint->y;
  4671. pLineTo->nXEnd = endPoint->x;
  4672. pLineTo->nYEnd = endPoint->y;
  4673. //
  4674. // We must convert these values to virtual coords.
  4675. //
  4676. OELPtoVirtual((LPPOINT)&pLineTo->nXStart, 2);
  4677. //
  4678. // Always do solid lines, so it does not matter what we specify as the
  4679. // back color.
  4680. //
  4681. RtlFillMemory(&pLineTo->BackColor,
  4682. sizeof(pLineTo->BackColor),
  4683. 0);
  4684. //
  4685. // We only draw solid lines with no option as to what we do to the
  4686. // background, so this is always transparent.
  4687. //
  4688. pLineTo->BackMode = TRANSPARENT;
  4689. //
  4690. // Get the ROP value.
  4691. //
  4692. pLineTo->ROP2 = rop2;
  4693. //
  4694. // The NT Display Driver is only called to accelerate simple solid
  4695. // lines. So we only support pen styles of PS_SOLID.
  4696. //
  4697. pLineTo->PenStyle = PS_SOLID;
  4698. //
  4699. // Get the pen width.
  4700. //
  4701. pLineTo->PenWidth = width;
  4702. //
  4703. // Set up the color.
  4704. //
  4705. OEConvertColor(ppdev,
  4706. &pLineTo->PenColor,
  4707. color,
  4708. NULL);
  4709. TRACE_OUT(( "LineTo BC %02x%02x%02x BM %04X rop2 %02X "
  4710. "pen %04X %04X %02x%02x%02x x1 %d y1 %d x2 %d y2 %d",
  4711. pLineTo->BackColor.red,
  4712. pLineTo->BackColor.green,
  4713. pLineTo->BackColor.blue,
  4714. pLineTo->BackMode,
  4715. pLineTo->ROP2,
  4716. pLineTo->PenStyle,
  4717. pLineTo->PenWidth,
  4718. pLineTo->PenColor.red,
  4719. pLineTo->PenColor.green,
  4720. pLineTo->PenColor.blue,
  4721. pLineTo->nXStart,
  4722. pLineTo->nYStart,
  4723. pLineTo->nXEnd,
  4724. pLineTo->nYEnd));
  4725. //
  4726. // Store the general order data. The bounding rectangle must be in to
  4727. // virtual desktop co-ordinates. OELRtoVirtual has already done this.
  4728. //
  4729. pOrder->OrderHeader.Common.fOrderFlags = OF_SPOILABLE;
  4730. pOrder->OrderHeader.Common.rcsDst.left = (TSHR_INT16)rectDst->left;
  4731. pOrder->OrderHeader.Common.rcsDst.right = (TSHR_INT16)rectDst->right;
  4732. pOrder->OrderHeader.Common.rcsDst.top = (TSHR_INT16)rectDst->top;
  4733. pOrder->OrderHeader.Common.rcsDst.bottom = (TSHR_INT16)rectDst->bottom;
  4734. //
  4735. // Store that order!
  4736. //
  4737. OEClipAndAddOrder(pOrder, NULL, pco);
  4738. rc = TRUE;
  4739. DC_EXIT_POINT:
  4740. DebugExitDWORD(OEAddLine, rc);
  4741. return(rc);
  4742. }
  4743. // NAME: OEEncodePatBlt
  4744. //
  4745. // PURPOSE:
  4746. //
  4747. // Attempts to encode a PatBlt order. This function allocates the memory
  4748. // for the encoded order (pointer returned in ppOrder). If the function
  4749. // completes successfully, it is the caller's responsibility to free this
  4750. // memory.
  4751. //
  4752. // RETURNS:
  4753. //
  4754. // TRUE - Order encoded
  4755. // FALSE - Order not encoded (so add to SDA)
  4756. //
  4757. // PARAMS:
  4758. //
  4759. // ppdev - display driver PDEV
  4760. // pbo - brush object for the blt
  4761. // pptlBrush - brush origin
  4762. // rop3 - 3-way rop to use
  4763. // pBounds - bounding rectangle
  4764. // ppOrder - the encoded order
  4765. //
  4766. BOOL OEEncodePatBlt(LPOSI_PDEV ppdev,
  4767. BRUSHOBJ *pbo,
  4768. POINTL *pptlBrush,
  4769. BYTE rop3,
  4770. LPRECT pBounds,
  4771. LPINT_ORDER *ppOrder)
  4772. {
  4773. BOOL rc = FALSE;
  4774. POE_BRUSH_DATA pCurrentBrush;
  4775. LPPATBLT_ORDER pPatBlt;
  4776. UINT orderFlags = OF_SPOILABLE;
  4777. DebugEntry(OEEncodePatBlt);
  4778. //
  4779. // Check for a simple brush pattern.
  4780. //
  4781. if ( OECheckBrushIsSimple(ppdev, pbo, &pCurrentBrush) )
  4782. {
  4783. //
  4784. // Allocate the memory for the order.
  4785. //
  4786. *ppOrder = OA_DDAllocOrderMem(sizeof(PATBLT_ORDER),0);
  4787. if (*ppOrder != NULL)
  4788. {
  4789. pPatBlt = (LPPATBLT_ORDER)((*ppOrder)->abOrderData);
  4790. //
  4791. // Set the opaque flag if the rop is opaque.
  4792. //
  4793. if (ROP3_IS_OPAQUE(rop3))
  4794. {
  4795. orderFlags |= OF_SPOILER;
  4796. }
  4797. //
  4798. // Set up order type.
  4799. //
  4800. pPatBlt->type = LOWORD(ORD_PATBLT);
  4801. //
  4802. // Virtual desktop co-ordinates.
  4803. //
  4804. pPatBlt->nLeftRect = pBounds->left;
  4805. pPatBlt->nTopRect = pBounds->top;
  4806. pPatBlt->nWidth = pBounds->right - pBounds->left + 1;
  4807. pPatBlt->nHeight = pBounds->bottom - pBounds->top + 1;
  4808. pPatBlt->bRop = rop3;
  4809. //
  4810. // Pattern colours.
  4811. //
  4812. pPatBlt->BackColor = pCurrentBrush->back;
  4813. pPatBlt->ForeColor = pCurrentBrush->fore;
  4814. //
  4815. // The protocol brush origin is the point on the screen where
  4816. // we want the brush to start being drawn from (tiling where
  4817. // necessary). This must be in virtual coordinates.
  4818. //
  4819. pPatBlt->BrushOrgX = pptlBrush->x;
  4820. pPatBlt->BrushOrgY = pptlBrush->y;
  4821. OELPtoVirtual((LPPOINT)&pPatBlt->BrushOrgX, 1);
  4822. //
  4823. // Extra brush data from the data when we realised the brush.
  4824. //
  4825. pPatBlt->BrushStyle = pCurrentBrush->style;
  4826. pPatBlt->BrushHatch = pCurrentBrush->hatch;
  4827. RtlCopyMemory(pPatBlt->BrushExtra,
  4828. pCurrentBrush->brushData,
  4829. sizeof(pPatBlt->BrushExtra));
  4830. TRACE_OUT(( "PatBlt BC %02x%02x%02x FC %02x%02x%02x "
  4831. "Brush %02X %02X X %d Y %d w %d h %d rop %02X",
  4832. pPatBlt->BackColor.red,
  4833. pPatBlt->BackColor.green,
  4834. pPatBlt->BackColor.blue,
  4835. pPatBlt->ForeColor.red,
  4836. pPatBlt->ForeColor.green,
  4837. pPatBlt->ForeColor.blue,
  4838. pPatBlt->BrushStyle,
  4839. pPatBlt->BrushHatch,
  4840. pPatBlt->nLeftRect,
  4841. pPatBlt->nTopRect,
  4842. pPatBlt->nWidth,
  4843. pPatBlt->nHeight,
  4844. pPatBlt->bRop));
  4845. //
  4846. // Copy any order flags into the encoded order structure.
  4847. //
  4848. (*ppOrder)->OrderHeader.Common.fOrderFlags = (TSHR_UINT16)orderFlags;
  4849. rc = TRUE;
  4850. }
  4851. else
  4852. {
  4853. TRACE_OUT(( "Failed to alloc order"));
  4854. }
  4855. }
  4856. else
  4857. {
  4858. TRACE_OUT(( "Brush is not simple"));
  4859. }
  4860. DebugExitDWORD(OEEncodePatBlt, rc);
  4861. return(rc);
  4862. }
  4863. //
  4864. // DrvTransparentBlt()
  4865. // NEW FOR NT5
  4866. //
  4867. BOOL DrvTransparentBlt
  4868. (
  4869. SURFOBJ * psoDst,
  4870. SURFOBJ * psoSrc,
  4871. CLIPOBJ * pco,
  4872. XLATEOBJ * pxlo,
  4873. RECTL * prclDst,
  4874. RECTL * prclSrc,
  4875. ULONG iTransColor,
  4876. ULONG ulReserved
  4877. )
  4878. {
  4879. BOOL rc = TRUE;
  4880. RECT rectSrc;
  4881. RECT rectDst;
  4882. BOOL fAccumulate = FALSE;
  4883. DebugEntry(DrvTransparentBlt);
  4884. //
  4885. // DO THIS _BEFORE_ TAKING LOCK
  4886. //
  4887. if (!g_oeViewers)
  4888. goto NO_LOCK_EXIT;
  4889. OE_SHM_START_WRITING;
  4890. //
  4891. // Get bounding rectangle and convert to a RECT.
  4892. //
  4893. RECT_FROM_RECTL(rectSrc, (*prclSrc));
  4894. RECT_FROM_RECTL(rectDst, (*prclDst));
  4895. //
  4896. // Check if we are accumulating data for this function
  4897. //
  4898. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  4899. if (!fAccumulate)
  4900. {
  4901. DC_QUIT;
  4902. }
  4903. //
  4904. // Convert to virtual coordinates.
  4905. //
  4906. OELRtoVirtual(&rectDst, 1);
  4907. DC_EXIT_POINT:
  4908. if (fAccumulate)
  4909. {
  4910. OEClipAndAddScreenData(&rectDst, pco);
  4911. }
  4912. OE_SHM_STOP_WRITING;
  4913. NO_LOCK_EXIT:
  4914. DebugExitBOOL(DrvTransparentBlt, rc);
  4915. return(rc);
  4916. }
  4917. //
  4918. // DrvAlphaBlend()
  4919. // NEW FOR NT5
  4920. //
  4921. BOOL DrvAlphaBlend
  4922. (
  4923. SURFOBJ * psoDst,
  4924. SURFOBJ * psoSrc,
  4925. CLIPOBJ * pco,
  4926. XLATEOBJ * pxlo,
  4927. RECTL * prclDst,
  4928. RECTL * prclSrc,
  4929. BLENDOBJ * pBlendObj
  4930. )
  4931. {
  4932. BOOL rc = TRUE;
  4933. RECT rectSrc;
  4934. RECT rectDst;
  4935. BOOL fAccumulate = FALSE;
  4936. DebugEntry(DrvAlphaBlend);
  4937. //
  4938. // DO THIS _BEFORE_ TAKING LOCK
  4939. //
  4940. if (!g_oeViewers)
  4941. goto NO_LOCK_EXIT;
  4942. OE_SHM_START_WRITING;
  4943. //
  4944. // Get bounding rectangle and convert to a RECT.
  4945. //
  4946. RECT_FROM_RECTL(rectSrc, (*prclSrc));
  4947. RECT_FROM_RECTL(rectDst, (*prclDst));
  4948. //
  4949. // Check if we are accumulating data for this function
  4950. //
  4951. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  4952. if (!fAccumulate)
  4953. {
  4954. DC_QUIT;
  4955. }
  4956. //
  4957. // Convert to virtual coordinates.
  4958. //
  4959. OELRtoVirtual(&rectDst, 1);
  4960. DC_EXIT_POINT:
  4961. if (fAccumulate)
  4962. {
  4963. OEClipAndAddScreenData(&rectDst, pco);
  4964. }
  4965. OE_SHM_STOP_WRITING;
  4966. NO_LOCK_EXIT:
  4967. DebugExitBOOL(DrvAlphaBlend, rc);
  4968. return(rc);
  4969. }
  4970. //
  4971. // DrvPlgBlt()
  4972. // NEW FOR NT5
  4973. //
  4974. BOOL DrvPlgBlt
  4975. (
  4976. SURFOBJ * psoDst,
  4977. SURFOBJ * psoSrc,
  4978. SURFOBJ * psoMsk,
  4979. CLIPOBJ * pco,
  4980. XLATEOBJ * pxlo,
  4981. COLORADJUSTMENT * pca,
  4982. POINTL * pptlBrushOrg,
  4983. POINTFIX * pptfx,
  4984. RECTL * prclDst,
  4985. POINTL * pptlSrc,
  4986. ULONG iMode
  4987. )
  4988. {
  4989. BOOL rc = TRUE;
  4990. RECT rectDst;
  4991. BOOL fAccumulate = FALSE;
  4992. DebugEntry(DrvPlgBlt);
  4993. //
  4994. // DO THIS _BEFORE_ TAKING LOCK
  4995. //
  4996. if (!g_oeViewers)
  4997. goto NO_LOCK_EXIT;
  4998. OE_SHM_START_WRITING;
  4999. //
  5000. // Get bounding rectangle and convert to a RECT.
  5001. //
  5002. RECT_FROM_RECTL(rectDst, (*prclDst));
  5003. //
  5004. // Check if we are accumulating data for this function
  5005. //
  5006. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  5007. if (!fAccumulate)
  5008. {
  5009. DC_QUIT;
  5010. }
  5011. //
  5012. // Convert to virtual coordinates.
  5013. //
  5014. OELRtoVirtual(&rectDst, 1);
  5015. DC_EXIT_POINT:
  5016. if (fAccumulate)
  5017. {
  5018. OEClipAndAddScreenData(&rectDst, pco);
  5019. }
  5020. OE_SHM_STOP_WRITING;
  5021. NO_LOCK_EXIT:
  5022. DebugExitBOOL(DrvPlgBlt, rc);
  5023. return(rc);
  5024. }
  5025. //
  5026. // DrvStretchBltROP()
  5027. // NEW FOR NT5
  5028. //
  5029. BOOL DrvStretchBltROP
  5030. (
  5031. SURFOBJ * psoDst,
  5032. SURFOBJ * psoSrc,
  5033. SURFOBJ * psoMask,
  5034. CLIPOBJ * pco,
  5035. XLATEOBJ * pxlo,
  5036. COLORADJUSTMENT * pca,
  5037. POINTL * pptlHTOrg,
  5038. RECTL * prclDst,
  5039. RECTL * prclSrc,
  5040. POINTL * pptlMask,
  5041. ULONG iMode,
  5042. BRUSHOBJ * pbo,
  5043. DWORD rop4
  5044. )
  5045. {
  5046. BOOL rc = TRUE;
  5047. RECT rectSrc;
  5048. RECT rectDst;
  5049. BOOL fAccumulate = FALSE;
  5050. DebugEntry(DrvStretchBltROP);
  5051. //
  5052. // DO THIS _BEFORE_ TAKING LOCK
  5053. //
  5054. if (!g_oeViewers)
  5055. goto NO_LOCK_EXIT;
  5056. OE_SHM_START_WRITING;
  5057. //
  5058. // Get bounding rectangle and convert to a RECT.
  5059. //
  5060. RECT_FROM_RECTL(rectSrc, (*prclSrc));
  5061. RECT_FROM_RECTL(rectDst, (*prclDst));
  5062. //
  5063. // Check if we are accumulating data for this function
  5064. //
  5065. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  5066. if (!fAccumulate)
  5067. {
  5068. DC_QUIT;
  5069. }
  5070. //
  5071. // Convert to virtual coordinates.
  5072. //
  5073. OELRtoVirtual(&rectDst, 1);
  5074. DC_EXIT_POINT:
  5075. if (fAccumulate)
  5076. {
  5077. OEClipAndAddScreenData(&rectDst, pco);
  5078. }
  5079. OE_SHM_STOP_WRITING;
  5080. NO_LOCK_EXIT:
  5081. DebugExitBOOL(DrvStretchBltROP, rc);
  5082. return(rc);
  5083. }
  5084. //
  5085. // DrvGradientFill()
  5086. // NEW FOR NT5
  5087. //
  5088. BOOL DrvGradientFill
  5089. (
  5090. SURFOBJ * psoDst,
  5091. CLIPOBJ * pco,
  5092. XLATEOBJ * pxlo,
  5093. TRIVERTEX * pVertex,
  5094. ULONG nVertex,
  5095. PVOID pMesh,
  5096. ULONG nMesh,
  5097. RECTL * prclExtents,
  5098. POINTL * pptlDitherOrg,
  5099. ULONG ulMode
  5100. )
  5101. {
  5102. BOOL rc = TRUE;
  5103. RECT rectDst;
  5104. BOOL fAccumulate = FALSE;
  5105. DebugEntry(DrvGradientFill);
  5106. //
  5107. // DO THIS _BEFORE_ TAKING LOCK
  5108. //
  5109. if (!g_oeViewers)
  5110. goto NO_LOCK_EXIT;
  5111. OE_SHM_START_WRITING;
  5112. //
  5113. // Get bounding rectangle and convert to a RECT.
  5114. //
  5115. RECT_FROM_RECTL(rectDst, pco->rclBounds);
  5116. //
  5117. // Check if we are accumulating data for this function
  5118. //
  5119. fAccumulate = OEAccumulateOutput(psoDst, pco, &rectDst);
  5120. if (!fAccumulate)
  5121. {
  5122. DC_QUIT;
  5123. }
  5124. //
  5125. // Convert to virtual coordinates.
  5126. //
  5127. OELRtoVirtual(&rectDst, 1);
  5128. DC_EXIT_POINT:
  5129. if (fAccumulate)
  5130. {
  5131. OEClipAndAddScreenData(&rectDst, pco);
  5132. }
  5133. OE_SHM_STOP_WRITING;
  5134. NO_LOCK_EXIT:
  5135. DebugExitBOOL(DrvGradientFill, rc);
  5136. return(rc);
  5137. }
  5138.