mirror of
https://github.com/kasmtech/KasmVNC.git
synced 2025-01-09 15:38:14 +01:00
25b8e64adb
This change adds support for the VMware Mouse Position pseudo-encoding[1], which is used to notify VNC clients when X11 clients call `XWarpPointer()`[2]. This function is called by SDL (and other similar libraries) when they detect that the server does not support native relative motion, like some RFB clients. With this, RFB clients can choose to adjust the local cursor position under certain circumstances to match what the server has set. For instance, if pointer lock has been enabled on the client's machine and the cursor is not being drawn locally, the local position of the cursor is irrelevant, so the RFB client can use what the server sends as the canonical absolute position of the cursor. This ultimately enables the possibility of games (especially FPS games) to behave how users expect (if the clients implement the corresponding change). Part of: #619 1: https://github.com/rfbproto/rfbproto/blob/master/rfbproto.rst#vmware-cursor-position-pseudo-encoding 2: https://tronche.com/gui/x/xlib/input/XWarpPointer.html 3: https://hg.libsdl.org/SDL/file/28e3b60e2131/src/events/SDL_mouse.c#l804
2453 lines
67 KiB
C
2453 lines
67 KiB
C
/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved.
|
|
* Copyright 2009-2017 Pierre Ossman for Cendio AB
|
|
*
|
|
* This is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This software is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
|
|
* USA.
|
|
*/
|
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
#include <dix-config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "vncHooks.h"
|
|
#include "vncExtInit.h"
|
|
|
|
#include "xorg-version.h"
|
|
|
|
#include "scrnintstr.h"
|
|
#include "windowstr.h"
|
|
#include "cursorstr.h"
|
|
#include "gcstruct.h"
|
|
#include "regionstr.h"
|
|
#include "dixfontstr.h"
|
|
#include "colormapst.h"
|
|
#ifdef RENDER
|
|
#include "picturestr.h"
|
|
#endif
|
|
#include "randrstr.h"
|
|
|
|
#define DBGPRINT(x) //(fprintf x)
|
|
|
|
// MAX_RECTS_PER_OP is the maximum number of rectangles we generate from
|
|
// operations like Polylines and PolySegment. If the operation is more complex
|
|
// than this, we simply use the bounding box. Ideally it would be a
|
|
// command-line option, but that would involve an extra malloc each time, so we
|
|
// fix it here.
|
|
#define MAX_RECTS_PER_OP 5
|
|
|
|
// vncHooksScreenRec and vncHooksGCRec contain pointers to the original
|
|
// functions which we "wrap" in order to hook the screen changes. The screen
|
|
// functions are each wrapped individually, while the GC "funcs" and "ops" are
|
|
// wrapped as a unit.
|
|
|
|
typedef struct _vncHooksScreenRec {
|
|
int ignoreHooks;
|
|
|
|
CloseScreenProcPtr CloseScreen;
|
|
CreateGCProcPtr CreateGC;
|
|
CopyWindowProcPtr CopyWindow;
|
|
ClearToBackgroundProcPtr ClearToBackground;
|
|
#if XORG < 110
|
|
RestoreAreasProcPtr RestoreAreas;
|
|
#endif
|
|
DisplayCursorProcPtr DisplayCursor;
|
|
#if XORG >= 119
|
|
CursorWarpedToProcPtr CursorWarpedTo;
|
|
#endif
|
|
ScreenBlockHandlerProcPtr BlockHandler;
|
|
#ifdef RENDER
|
|
CompositeProcPtr Composite;
|
|
GlyphsProcPtr Glyphs;
|
|
CompositeRectsProcPtr CompositeRects;
|
|
TrapezoidsProcPtr Trapezoids;
|
|
TrianglesProcPtr Triangles;
|
|
#if PICTURE_SCREEN_VERSION >= 2
|
|
TriStripProcPtr TriStrip;
|
|
TriFanProcPtr TriFan;
|
|
#endif
|
|
#endif
|
|
RRSetConfigProcPtr rrSetConfig;
|
|
RRScreenSetSizeProcPtr rrScreenSetSize;
|
|
RRCrtcSetProcPtr rrCrtcSet;
|
|
} vncHooksScreenRec, *vncHooksScreenPtr;
|
|
|
|
typedef struct _vncHooksGCRec {
|
|
#if XORG >= 116
|
|
const GCFuncs *funcs;
|
|
const GCOps *ops;
|
|
#else
|
|
GCFuncs *funcs;
|
|
GCOps *ops;
|
|
#endif
|
|
} vncHooksGCRec, *vncHooksGCPtr;
|
|
|
|
#define wrap(priv, real, mem, func) {\
|
|
priv->mem = real->mem; \
|
|
real->mem = func; \
|
|
}
|
|
|
|
#define unwrap(priv, real, mem) {\
|
|
real->mem = priv->mem; \
|
|
}
|
|
|
|
#if XORG < 19
|
|
static int vncHooksScreenPrivateKeyIndex;
|
|
static int vncHooksGCPrivateKeyIndex;
|
|
static DevPrivateKey vncHooksScreenPrivateKey = &vncHooksScreenPrivateKeyIndex;
|
|
static DevPrivateKey vncHooksGCPrivateKey = &vncHooksGCPrivateKeyIndex;
|
|
#else
|
|
static DevPrivateKeyRec vncHooksScreenKeyRec;
|
|
static DevPrivateKeyRec vncHooksGCKeyRec;
|
|
#define vncHooksScreenPrivateKey (&vncHooksScreenKeyRec)
|
|
#define vncHooksGCPrivateKey (&vncHooksGCKeyRec)
|
|
#endif
|
|
|
|
#define vncHooksScreenPrivate(pScreen) \
|
|
(vncHooksScreenPtr) dixLookupPrivate(&(pScreen)->devPrivates, \
|
|
vncHooksScreenPrivateKey)
|
|
#define vncHooksGCPrivate(pGC) \
|
|
(vncHooksGCPtr) dixLookupPrivate(&(pGC)->devPrivates, \
|
|
vncHooksGCPrivateKey)
|
|
|
|
// screen functions
|
|
|
|
#if XORG <= 112
|
|
static Bool vncHooksCloseScreen(int i, ScreenPtr pScreen);
|
|
#else
|
|
static Bool vncHooksCloseScreen(ScreenPtr pScreen);
|
|
#endif
|
|
static Bool vncHooksCreateGC(GCPtr pGC);
|
|
static void vncHooksCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg,
|
|
RegionPtr pOldRegion);
|
|
static void vncHooksClearToBackground(WindowPtr pWin, int x, int y, int w,
|
|
int h, Bool generateExposures);
|
|
#if XORG < 110
|
|
static RegionPtr vncHooksRestoreAreas(WindowPtr pWin, RegionPtr prgnExposed);
|
|
#endif
|
|
static Bool vncHooksDisplayCursor(DeviceIntPtr pDev,
|
|
ScreenPtr pScreen, CursorPtr cursor);
|
|
#if XORG >= 119
|
|
static void vncHooksCursorWarpedTo(DeviceIntPtr pDev,
|
|
ScreenPtr pScreen_, ClientPtr pClient,
|
|
WindowPtr pWindow, SpritePtr pSprite,
|
|
int x, int y);
|
|
#endif
|
|
#if XORG <= 112
|
|
static void vncHooksBlockHandler(int i, pointer blockData, pointer pTimeout,
|
|
pointer pReadmask);
|
|
#elif XORG <= 118
|
|
static void vncHooksBlockHandler(ScreenPtr pScreen, void * pTimeout,
|
|
void * pReadmask);
|
|
#else
|
|
static void vncHooksBlockHandler(ScreenPtr pScreen, void * pTimeout);
|
|
#endif
|
|
#ifdef RENDER
|
|
static void vncHooksComposite(CARD8 op, PicturePtr pSrc, PicturePtr pMask,
|
|
PicturePtr pDst, INT16 xSrc, INT16 ySrc, INT16 xMask,
|
|
INT16 yMask, INT16 xDst, INT16 yDst, CARD16 width, CARD16 height);
|
|
static void vncHooksGlyphs(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc, int nlists,
|
|
GlyphListPtr lists, GlyphPtr * glyphs);
|
|
static void vncHooksCompositeRects(CARD8 op, PicturePtr pDst,
|
|
xRenderColor * color, int nRect, xRectangle *rects);
|
|
static void vncHooksTrapezoids(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int ntrap, xTrapezoid * traps);
|
|
static void vncHooksTriangles(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int ntri, xTriangle * tris);
|
|
#if PICTURE_SCREEN_VERSION >= 2
|
|
static void vncHooksTriStrip(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int npoint, xPointFixed * points);
|
|
static void vncHooksTriFan(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int npoint, xPointFixed * points);
|
|
#endif
|
|
#endif
|
|
static Bool vncHooksRandRSetConfig(ScreenPtr pScreen, Rotation rotation,
|
|
int rate, RRScreenSizePtr pSize);
|
|
static Bool vncHooksRandRScreenSetSize(ScreenPtr pScreen,
|
|
CARD16 width, CARD16 height,
|
|
CARD32 mmWidth, CARD32 mmHeight);
|
|
static Bool vncHooksRandRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc,
|
|
RRModePtr mode, int x, int y,
|
|
Rotation rotation, int numOutputs,
|
|
RROutputPtr *outputs);
|
|
|
|
// GC "funcs"
|
|
|
|
static void vncHooksValidateGC(GCPtr pGC, unsigned long changes,
|
|
DrawablePtr pDrawable);
|
|
static void vncHooksChangeGC(GCPtr pGC, unsigned long mask);
|
|
static void vncHooksCopyGC(GCPtr src, unsigned long mask, GCPtr dst);
|
|
static void vncHooksDestroyGC(GCPtr pGC);
|
|
static void vncHooksChangeClip(GCPtr pGC, int type, void * pValue,int nrects);
|
|
static void vncHooksDestroyClip(GCPtr pGC);
|
|
static void vncHooksCopyClip(GCPtr dst, GCPtr src);
|
|
|
|
static GCFuncs vncHooksGCFuncs = {
|
|
vncHooksValidateGC, vncHooksChangeGC, vncHooksCopyGC, vncHooksDestroyGC,
|
|
vncHooksChangeClip, vncHooksDestroyClip, vncHooksCopyClip,
|
|
};
|
|
|
|
// GC "ops"
|
|
|
|
static void vncHooksFillSpans(DrawablePtr pDrawable, GCPtr pGC, int nInit,
|
|
DDXPointPtr pptInit, int *pwidthInit,
|
|
int fSorted);
|
|
static void vncHooksSetSpans(DrawablePtr pDrawable, GCPtr pGC, char *psrc,
|
|
DDXPointPtr ppt, int *pwidth, int nspans,
|
|
int fSorted);
|
|
static void vncHooksPutImage(DrawablePtr pDrawable, GCPtr pGC, int depth,
|
|
int x, int y, int w, int h, int leftPad,
|
|
int format, char *pBits);
|
|
static RegionPtr vncHooksCopyArea(DrawablePtr pSrc, DrawablePtr pDst,
|
|
GCPtr pGC, int srcx, int srcy, int w, int h,
|
|
int dstx, int dsty);
|
|
static RegionPtr vncHooksCopyPlane(DrawablePtr pSrc, DrawablePtr pDst,
|
|
GCPtr pGC, int srcx, int srcy, int w, int h,
|
|
int dstx, int dsty, unsigned long plane);
|
|
static void vncHooksPolyPoint(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, xPoint *pts);
|
|
static void vncHooksPolylines(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, DDXPointPtr ppts);
|
|
static void vncHooksPolySegment(DrawablePtr pDrawable, GCPtr pGC, int nseg,
|
|
xSegment *segs);
|
|
static void vncHooksPolyRectangle(DrawablePtr pDrawable, GCPtr pGC, int nrects,
|
|
xRectangle *rects);
|
|
static void vncHooksPolyArc(DrawablePtr pDrawable, GCPtr pGC, int narcs,
|
|
xArc *arcs);
|
|
static void vncHooksFillPolygon(DrawablePtr pDrawable, GCPtr pGC, int shape,
|
|
int mode, int count, DDXPointPtr pts);
|
|
static void vncHooksPolyFillRect(DrawablePtr pDrawable, GCPtr pGC, int nrects,
|
|
xRectangle *rects);
|
|
static void vncHooksPolyFillArc(DrawablePtr pDrawable, GCPtr pGC, int narcs,
|
|
xArc *arcs);
|
|
static int vncHooksPolyText8(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, char *chars);
|
|
static int vncHooksPolyText16(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, unsigned short *chars);
|
|
static void vncHooksImageText8(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, char *chars);
|
|
static void vncHooksImageText16(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, unsigned short *chars);
|
|
static void vncHooksImageGlyphBlt(DrawablePtr pDrawable, GCPtr pGC, int x,
|
|
int y, unsigned int nglyph,
|
|
CharInfoPtr *ppci, void * pglyphBase);
|
|
static void vncHooksPolyGlyphBlt(DrawablePtr pDrawable, GCPtr pGC, int x,
|
|
int y, unsigned int nglyph,
|
|
CharInfoPtr *ppci, void * pglyphBase);
|
|
static void vncHooksPushPixels(GCPtr pGC, PixmapPtr pBitMap,
|
|
DrawablePtr pDrawable, int w, int h, int x,
|
|
int y);
|
|
|
|
static GCOps vncHooksGCOps = {
|
|
vncHooksFillSpans, vncHooksSetSpans, vncHooksPutImage, vncHooksCopyArea,
|
|
vncHooksCopyPlane, vncHooksPolyPoint, vncHooksPolylines, vncHooksPolySegment,
|
|
vncHooksPolyRectangle, vncHooksPolyArc, vncHooksFillPolygon,
|
|
vncHooksPolyFillRect, vncHooksPolyFillArc, vncHooksPolyText8,
|
|
vncHooksPolyText16, vncHooksImageText8, vncHooksImageText16,
|
|
vncHooksImageGlyphBlt, vncHooksPolyGlyphBlt, vncHooksPushPixels
|
|
};
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// vncHooksInit() is called at initialisation time and every time the server
|
|
// resets. It is called once for each screen, but the indexes are only
|
|
// allocated once for each server generation.
|
|
|
|
int vncHooksInit(int scrIdx)
|
|
{
|
|
ScreenPtr pScreen;
|
|
vncHooksScreenPtr vncHooksScreen;
|
|
|
|
#ifdef RENDER
|
|
PictureScreenPtr ps;
|
|
#endif
|
|
rrScrPrivPtr rp;
|
|
|
|
pScreen = screenInfo.screens[scrIdx];
|
|
|
|
if (sizeof(BoxRec) != sizeof(struct UpdateRect)) {
|
|
ErrorF("vncHooksInit: Incompatible BoxRec size\n");
|
|
return FALSE;
|
|
}
|
|
|
|
#if XORG < 19
|
|
if (!dixRequestPrivate(vncHooksScreenPrivateKey, sizeof(vncHooksScreenRec))) {
|
|
ErrorF("vncHooksInit: Allocation of vncHooksScreen failed\n");
|
|
return FALSE;
|
|
}
|
|
if (!dixRequestPrivate(vncHooksGCPrivateKey, sizeof(vncHooksGCRec))) {
|
|
ErrorF("vncHooksInit: Allocation of vncHooksGCRec failed\n");
|
|
return FALSE;
|
|
}
|
|
|
|
#else
|
|
if (!dixRegisterPrivateKey(&vncHooksScreenKeyRec, PRIVATE_SCREEN,
|
|
sizeof(vncHooksScreenRec))) {
|
|
ErrorF("vncHooksInit: Allocation of vncHooksScreen failed\n");
|
|
return FALSE;
|
|
}
|
|
if (!dixRegisterPrivateKey(&vncHooksGCKeyRec, PRIVATE_GC,
|
|
sizeof(vncHooksGCRec))) {
|
|
ErrorF("vncHooksInit: Allocation of vncHooksGCRec failed\n");
|
|
return FALSE;
|
|
}
|
|
|
|
#endif
|
|
|
|
vncHooksScreen = vncHooksScreenPrivate(pScreen);
|
|
|
|
vncHooksScreen->ignoreHooks = 0;
|
|
|
|
wrap(vncHooksScreen, pScreen, CloseScreen, vncHooksCloseScreen);
|
|
wrap(vncHooksScreen, pScreen, CreateGC, vncHooksCreateGC);
|
|
wrap(vncHooksScreen, pScreen, CopyWindow, vncHooksCopyWindow);
|
|
wrap(vncHooksScreen, pScreen, ClearToBackground, vncHooksClearToBackground);
|
|
#if XORG < 110
|
|
wrap(vncHooksScreen, pScreen, RestoreAreas, vncHooksRestoreAreas);
|
|
#endif
|
|
wrap(vncHooksScreen, pScreen, DisplayCursor, vncHooksDisplayCursor);
|
|
#if XORG >= 119
|
|
wrap(vncHooksScreen, pScreen, CursorWarpedTo, vncHooksCursorWarpedTo);
|
|
#endif
|
|
wrap(vncHooksScreen, pScreen, BlockHandler, vncHooksBlockHandler);
|
|
#ifdef RENDER
|
|
ps = GetPictureScreenIfSet(pScreen);
|
|
if (ps) {
|
|
wrap(vncHooksScreen, ps, Composite, vncHooksComposite);
|
|
wrap(vncHooksScreen, ps, Glyphs, vncHooksGlyphs);
|
|
wrap(vncHooksScreen, ps, CompositeRects, vncHooksCompositeRects);
|
|
wrap(vncHooksScreen, ps, Trapezoids, vncHooksTrapezoids);
|
|
wrap(vncHooksScreen, ps, Triangles, vncHooksTriangles);
|
|
#if PICTURE_SCREEN_VERSION >= 2
|
|
wrap(vncHooksScreen, ps, TriStrip, vncHooksTriStrip);
|
|
wrap(vncHooksScreen, ps, TriFan, vncHooksTriFan);
|
|
#endif
|
|
}
|
|
#endif
|
|
rp = rrGetScrPriv(pScreen);
|
|
if (rp) {
|
|
/* Some RandR callbacks are optional */
|
|
if (rp->rrSetConfig)
|
|
wrap(vncHooksScreen, rp, rrSetConfig, vncHooksRandRSetConfig);
|
|
if (rp->rrScreenSetSize)
|
|
wrap(vncHooksScreen, rp, rrScreenSetSize, vncHooksRandRScreenSetSize);
|
|
if (rp->rrCrtcSet)
|
|
wrap(vncHooksScreen, rp, rrCrtcSet, vncHooksRandRCrtcSet);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// vncGetScreenImage() grabs a chunk of data from the main screen into the
|
|
// provided buffer. It lives here rather than in XorgGlue.c because it
|
|
// temporarily pauses the hooks.
|
|
|
|
void vncGetScreenImage(int scrIdx, int x, int y, int width, int height,
|
|
char *buffer, int strideBytes)
|
|
{
|
|
ScreenPtr pScreen = screenInfo.screens[scrIdx];
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen);
|
|
|
|
int i;
|
|
|
|
vncHooksScreen->ignoreHooks++;
|
|
|
|
// We do one line at a time since GetImage() cannot handle stride
|
|
for (i = y; i < y + height; i++) {
|
|
DrawablePtr pDrawable;
|
|
#if XORG < 19
|
|
pDrawable = (DrawablePtr) WindowTable[scrIdx];
|
|
#else
|
|
pDrawable = (DrawablePtr) pScreen->root;
|
|
#endif
|
|
|
|
(*pScreen->GetImage) (pDrawable, x, i, width, 1,
|
|
ZPixmap, (unsigned long)~0L, buffer);
|
|
|
|
buffer += strideBytes;
|
|
}
|
|
|
|
vncHooksScreen->ignoreHooks--;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Helper functions
|
|
//
|
|
|
|
static inline void add_changed(ScreenPtr pScreen, RegionPtr reg)
|
|
{
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen);
|
|
if (vncHooksScreen->ignoreHooks)
|
|
return;
|
|
if (RegionNil(reg))
|
|
return;
|
|
vncAddChanged(pScreen->myNum,
|
|
(const struct UpdateRect*)RegionExtents(reg),
|
|
RegionNumRects(reg),
|
|
(const struct UpdateRect*)RegionRects(reg));
|
|
}
|
|
|
|
static inline void add_copied(ScreenPtr pScreen, RegionPtr dst,
|
|
int dx, int dy)
|
|
{
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen);
|
|
if (vncHooksScreen->ignoreHooks)
|
|
return;
|
|
if (RegionNil(dst))
|
|
return;
|
|
vncAddCopied(pScreen->myNum,
|
|
(const struct UpdateRect*)RegionExtents(dst),
|
|
RegionNumRects(dst),
|
|
(const struct UpdateRect*)RegionRects(dst), dx, dy);
|
|
}
|
|
|
|
static inline Bool is_visible(DrawablePtr drawable)
|
|
{
|
|
PixmapPtr scrPixmap;
|
|
|
|
scrPixmap = drawable->pScreen->GetScreenPixmap(drawable->pScreen);
|
|
|
|
if (drawable->type == DRAWABLE_WINDOW) {
|
|
WindowPtr window;
|
|
PixmapPtr winPixmap;
|
|
|
|
window = (WindowPtr)drawable;
|
|
winPixmap = drawable->pScreen->GetWindowPixmap(window);
|
|
|
|
if (!window->viewable)
|
|
return FALSE;
|
|
|
|
if (winPixmap != scrPixmap)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
if (drawable != &scrPixmap->drawable)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// screen functions
|
|
//
|
|
|
|
// Unwrap and rewrap helpers
|
|
|
|
#define SCREEN_PROLOGUE(scrn,field) \
|
|
ScreenPtr pScreen = scrn; \
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); \
|
|
unwrap(vncHooksScreen, pScreen, field); \
|
|
DBGPRINT((stderr,"vncHooks" #field " called\n"));
|
|
|
|
#define SCREEN_EPILOGUE(field) \
|
|
wrap(vncHooksScreen, pScreen, field, vncHooks##field); \
|
|
|
|
|
|
// CloseScreen - unwrap the screen functions and call the original CloseScreen
|
|
// function
|
|
|
|
#if XORG <= 112
|
|
static Bool vncHooksCloseScreen(int i, ScreenPtr pScreen_)
|
|
#else
|
|
static Bool vncHooksCloseScreen(ScreenPtr pScreen_)
|
|
#endif
|
|
{
|
|
#ifdef RENDER
|
|
PictureScreenPtr ps;
|
|
#endif
|
|
rrScrPrivPtr rp;
|
|
|
|
SCREEN_PROLOGUE(pScreen_, CloseScreen);
|
|
|
|
unwrap(vncHooksScreen, pScreen, CreateGC);
|
|
unwrap(vncHooksScreen, pScreen, CopyWindow);
|
|
unwrap(vncHooksScreen, pScreen, ClearToBackground);
|
|
#if XORG < 110
|
|
unwrap(vncHooksScreen, pScreen, RestoreAreas);
|
|
#endif
|
|
unwrap(vncHooksScreen, pScreen, DisplayCursor);
|
|
unwrap(vncHooksScreen, pScreen, BlockHandler);
|
|
#ifdef RENDER
|
|
ps = GetPictureScreenIfSet(pScreen);
|
|
if (ps) {
|
|
unwrap(vncHooksScreen, ps, Composite);
|
|
unwrap(vncHooksScreen, ps, Glyphs);
|
|
unwrap(vncHooksScreen, ps, CompositeRects);
|
|
unwrap(vncHooksScreen, ps, Trapezoids);
|
|
unwrap(vncHooksScreen, ps, Triangles);
|
|
#if PICTURE_SCREEN_VERSION >= 2
|
|
unwrap(vncHooksScreen, ps, TriStrip);
|
|
unwrap(vncHooksScreen, ps, TriFan);
|
|
#endif
|
|
}
|
|
#endif
|
|
rp = rrGetScrPriv(pScreen);
|
|
if (rp) {
|
|
unwrap(vncHooksScreen, rp, rrSetConfig);
|
|
unwrap(vncHooksScreen, rp, rrScreenSetSize);
|
|
unwrap(vncHooksScreen, rp, rrCrtcSet);
|
|
}
|
|
|
|
DBGPRINT((stderr,"vncHooksCloseScreen: unwrapped screen functions\n"));
|
|
|
|
#if XORG <= 112
|
|
return (*pScreen->CloseScreen)(i, pScreen);
|
|
#else
|
|
return (*pScreen->CloseScreen)(pScreen);
|
|
#endif
|
|
}
|
|
|
|
// CreateGC - wrap the "GC funcs"
|
|
|
|
static Bool vncHooksCreateGC(GCPtr pGC)
|
|
{
|
|
vncHooksGCPtr vncHooksGC = vncHooksGCPrivate(pGC);
|
|
Bool ret;
|
|
|
|
SCREEN_PROLOGUE(pGC->pScreen, CreateGC);
|
|
|
|
ret = (*pScreen->CreateGC) (pGC);
|
|
|
|
vncHooksGC->ops = NULL;
|
|
vncHooksGC->funcs = pGC->funcs;
|
|
pGC->funcs = &vncHooksGCFuncs;
|
|
|
|
SCREEN_EPILOGUE(CreateGC);
|
|
|
|
return ret;
|
|
}
|
|
|
|
// CopyWindow - destination of the copy is the old region, clipped by
|
|
// borderClip, translated by the delta. This call only does the copy - it
|
|
// doesn't affect any other bits.
|
|
|
|
static void vncHooksCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg,
|
|
RegionPtr pOldRegion)
|
|
{
|
|
int dx, dy;
|
|
BoxRec screen_box;
|
|
RegionRec copied, screen_rgn;
|
|
|
|
SCREEN_PROLOGUE(pWin->drawable.pScreen, CopyWindow);
|
|
|
|
RegionNull(&copied);
|
|
RegionCopy(&copied, pOldRegion);
|
|
|
|
screen_box.x1 = 0;
|
|
screen_box.y1 = 0;
|
|
screen_box.x2 = pScreen->width;
|
|
screen_box.y2 = pScreen->height;
|
|
|
|
RegionInitBoxes(&screen_rgn, &screen_box, 1);
|
|
|
|
dx = pWin->drawable.x - ptOldOrg.x;
|
|
dy = pWin->drawable.y - ptOldOrg.y;
|
|
|
|
// RFB tracks copies in terms of destination rectangle, not source.
|
|
// We also need to copy with changes to the Window's clipping region.
|
|
// Finally, make sure we don't get copies to or from regions outside
|
|
// the framebuffer.
|
|
RegionIntersect(&copied, &copied, &screen_rgn);
|
|
RegionTranslate(&copied, dx, dy);
|
|
RegionIntersect(&copied, &copied, &screen_rgn);
|
|
RegionIntersect(&copied, &copied, &pWin->borderClip);
|
|
|
|
(*pScreen->CopyWindow) (pWin, ptOldOrg, pOldRegion);
|
|
|
|
add_copied(pScreen, &copied, dx, dy);
|
|
|
|
RegionUninit(&copied);
|
|
RegionUninit(&screen_rgn);
|
|
|
|
SCREEN_EPILOGUE(CopyWindow);
|
|
}
|
|
|
|
// ClearToBackground - changed region is the given rectangle, clipped by
|
|
// clipList, but only if generateExposures is false.
|
|
|
|
static void vncHooksClearToBackground(WindowPtr pWin, int x, int y, int w,
|
|
int h, Bool generateExposures)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
SCREEN_PROLOGUE(pWin->drawable.pScreen, ClearToBackground);
|
|
|
|
box.x1 = x + pWin->drawable.x;
|
|
box.y1 = y + pWin->drawable.y;
|
|
box.x2 = w ? (box.x1 + w) : (pWin->drawable.x + pWin->drawable.width);
|
|
box.y2 = h ? (box.y1 + h) : (pWin->drawable.y + pWin->drawable.height);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, &pWin->clipList);
|
|
|
|
(*pScreen->ClearToBackground) (pWin, x, y, w, h, generateExposures);
|
|
|
|
if (!generateExposures) {
|
|
add_changed(pScreen, ®);
|
|
}
|
|
|
|
RegionUninit(®);
|
|
|
|
SCREEN_EPILOGUE(ClearToBackground);
|
|
}
|
|
|
|
#if XORG < 110
|
|
// RestoreAreas - changed region is the given region
|
|
|
|
static RegionPtr vncHooksRestoreAreas(WindowPtr pWin, RegionPtr pRegion)
|
|
{
|
|
RegionRec reg;
|
|
|
|
SCREEN_PROLOGUE(pWin->drawable.pScreen, RestoreAreas);
|
|
|
|
REGION_NULL(pScreen, ®);
|
|
REGION_COPY(pScreen, ®, pRegion);
|
|
|
|
RegionPtr result = (*pScreen->RestoreAreas) (pWin, pRegion);
|
|
|
|
add_changed(pScreen, ®);
|
|
|
|
REGION_UNINIT(pScreen, ®);
|
|
|
|
SCREEN_EPILOGUE(RestoreAreas);
|
|
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
// DisplayCursor - get the cursor shape
|
|
|
|
static Bool vncHooksDisplayCursor(DeviceIntPtr pDev,
|
|
ScreenPtr pScreen_, CursorPtr cursor)
|
|
{
|
|
Bool ret;
|
|
|
|
SCREEN_PROLOGUE(pScreen_, DisplayCursor);
|
|
|
|
ret = (*pScreen->DisplayCursor) (pDev, pScreen, cursor);
|
|
|
|
/*
|
|
* XXX DIX calls this function with NULL argument to remove cursor sprite from
|
|
* screen. Should we handle this in setCursor as well?
|
|
*/
|
|
if (cursor != NullCursor) {
|
|
int width, height;
|
|
int hotX, hotY;
|
|
|
|
unsigned char *rgbaData;
|
|
|
|
width = cursor->bits->width;
|
|
height = cursor->bits->height;
|
|
|
|
hotX = cursor->bits->xhot;
|
|
hotY = cursor->bits->yhot;
|
|
|
|
rgbaData = malloc(width * height * 4);
|
|
if (rgbaData == NULL)
|
|
goto out;
|
|
|
|
#ifdef ARGB_CURSOR
|
|
if (cursor->bits->argb) {
|
|
unsigned char *out;
|
|
CARD32 *in;
|
|
int i;
|
|
|
|
in = cursor->bits->argb;
|
|
out = rgbaData;
|
|
for (i = 0; i < width*height; i++) {
|
|
out[0] = (*in >> 16) & 0xff;
|
|
out[1] = (*in >> 8) & 0xff;
|
|
out[2] = (*in >> 0) & 0xff;
|
|
out[3] = (*in >> 24) & 0xff;
|
|
out += 4;
|
|
in++;
|
|
}
|
|
} else {
|
|
#endif
|
|
unsigned char *out;
|
|
int xMaskBytesPerRow;
|
|
|
|
xMaskBytesPerRow = BitmapBytePad(width);
|
|
|
|
out = rgbaData;
|
|
for (int y = 0; y < height; y++) {
|
|
for (int x = 0; x < width; x++) {
|
|
int byte = y * xMaskBytesPerRow + x / 8;
|
|
#if (BITMAP_BIT_ORDER == MSBFirst)
|
|
int bit = 7 - x % 8;
|
|
#else
|
|
int bit = x % 8;
|
|
#endif
|
|
|
|
if (cursor->bits->source[byte] & (1 << bit)) {
|
|
out[0] = cursor->foreRed;
|
|
out[1] = cursor->foreGreen;
|
|
out[2] = cursor->foreBlue;
|
|
} else {
|
|
out[0] = cursor->backRed;
|
|
out[1] = cursor->backGreen;
|
|
out[2] = cursor->backBlue;
|
|
}
|
|
|
|
if (cursor->bits->mask[byte] & (1 << bit))
|
|
out[3] = 0xff;
|
|
else
|
|
out[3] = 0x00;
|
|
|
|
out += 4;
|
|
}
|
|
}
|
|
#ifdef ARGB_CURSOR
|
|
}
|
|
#endif
|
|
|
|
vncSetCursor(width, height, hotX, hotY, rgbaData);
|
|
|
|
free(rgbaData);
|
|
}
|
|
|
|
out:
|
|
SCREEN_EPILOGUE(DisplayCursor);
|
|
|
|
return ret;
|
|
}
|
|
|
|
// CursorWarpedTo - notify that the cursor was warped
|
|
|
|
#if XORG >= 119
|
|
static void vncHooksCursorWarpedTo(DeviceIntPtr pDev,
|
|
ScreenPtr pScreen_, ClientPtr pClient,
|
|
WindowPtr pWindow, SpritePtr pSprite,
|
|
int x, int y)
|
|
{
|
|
SCREEN_PROLOGUE(pScreen_, CursorWarpedTo);
|
|
vncSetCursorPos(pScreen->myNum, x, y);
|
|
SCREEN_EPILOGUE(CursorWarpedTo);
|
|
}
|
|
#endif
|
|
|
|
// BlockHandler - ignore any changes during the block handler - it's likely
|
|
// these are just drawing the cursor.
|
|
|
|
#if XORG <= 112
|
|
static void vncHooksBlockHandler(int i, pointer blockData, pointer pTimeout,
|
|
pointer pReadmask)
|
|
#elif XORG <= 118
|
|
static void vncHooksBlockHandler(ScreenPtr pScreen_, void * pTimeout,
|
|
void * pReadmask)
|
|
#else
|
|
static void vncHooksBlockHandler(ScreenPtr pScreen_, void * pTimeout)
|
|
#endif
|
|
{
|
|
#if XORG <= 112
|
|
SCREEN_PROLOGUE(screenInfo.screens[i], BlockHandler);
|
|
#else
|
|
SCREEN_PROLOGUE(pScreen_, BlockHandler);
|
|
#endif
|
|
|
|
vncHooksScreen->ignoreHooks++;
|
|
|
|
#if XORG <= 112
|
|
(*pScreen->BlockHandler) (i, blockData, pTimeout, pReadmask);
|
|
#elif XORG <= 118
|
|
(*pScreen->BlockHandler) (pScreen, pTimeout, pReadmask);
|
|
#else
|
|
(*pScreen->BlockHandler) (pScreen, pTimeout);
|
|
#endif
|
|
|
|
vncHooksScreen->ignoreHooks--;
|
|
|
|
SCREEN_EPILOGUE(BlockHandler);
|
|
}
|
|
|
|
#ifdef RENDER
|
|
|
|
// Unwrap and rewrap helpers
|
|
|
|
#define RENDER_PROLOGUE(scrn,field) \
|
|
ScreenPtr pScreen = scrn; \
|
|
PictureScreenPtr ps = GetPictureScreen(pScreen); \
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); \
|
|
unwrap(vncHooksScreen, ps, field); \
|
|
DBGPRINT((stderr,"vncHooks" #field " called\n"));
|
|
|
|
#define RENDER_EPILOGUE(field) \
|
|
wrap(vncHooksScreen, ps, field, vncHooks##field); \
|
|
|
|
// Composite - The core of XRENDER
|
|
|
|
static void vncHooksComposite(CARD8 op, PicturePtr pSrc, PicturePtr pMask,
|
|
PicturePtr pDst, INT16 xSrc, INT16 ySrc, INT16 xMask,
|
|
INT16 yMask, INT16 xDst, INT16 yDst, CARD16 width, CARD16 height)
|
|
{
|
|
RegionRec changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, Composite);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec box;
|
|
RegionRec fbreg;
|
|
|
|
box.x1 = max(pDst->pDrawable->x + xDst, 0);
|
|
box.y1 = max(pDst->pDrawable->y + yDst, 0);
|
|
box.x2 = box.x1 + width;
|
|
box.y2 = box.y1 + height;
|
|
RegionInitBoxes(&changed, &box, 1);
|
|
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = pScreen->width;
|
|
box.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &box, 1);
|
|
|
|
RegionIntersect(&changed, &changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
RegionNull(&changed);
|
|
}
|
|
|
|
|
|
(*ps->Composite)(op, pSrc, pMask, pDst, xSrc, ySrc,
|
|
xMask, yMask, xDst, yDst, width, height);
|
|
|
|
add_changed(pScreen, &changed);
|
|
|
|
RegionUninit(&changed);
|
|
|
|
RENDER_EPILOGUE(Composite);
|
|
}
|
|
|
|
static int
|
|
GlyphCount(int nlist, GlyphListPtr list, GlyphPtr * glyphs)
|
|
{
|
|
int count;
|
|
|
|
count = 0;
|
|
while (nlist--) {
|
|
count += list->len;
|
|
list++;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
static RegionPtr
|
|
GlyphsToRegion(ScreenPtr pScreen, int nlist, GlyphListPtr list, GlyphPtr *glyphs)
|
|
{
|
|
int n;
|
|
GlyphPtr glyph;
|
|
int x, y;
|
|
|
|
int nrects = GlyphCount(nlist, list, glyphs);
|
|
xRectangle rects[nrects];
|
|
xRectanglePtr rect;
|
|
|
|
x = 0;
|
|
y = 0;
|
|
rect = &rects[0];
|
|
while (nlist--) {
|
|
x += list->xOff;
|
|
y += list->yOff;
|
|
n = list->len;
|
|
list++;
|
|
while (n--) {
|
|
glyph = *glyphs++;
|
|
rect->x = x - glyph->info.x;
|
|
rect->y = y - glyph->info.y;
|
|
rect->width = glyph->info.width;
|
|
rect->height = glyph->info.height;
|
|
x += glyph->info.xOff;
|
|
y += glyph->info.yOff;
|
|
rect++;
|
|
}
|
|
}
|
|
|
|
return RECTS_TO_REGION(pScreen, nrects, rects, CT_NONE);
|
|
}
|
|
|
|
// Glyphs - Glyph specific version of Composite (caches and whatnot)
|
|
|
|
static void vncHooksGlyphs(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc, int nlists,
|
|
GlyphListPtr lists, GlyphPtr * glyphs)
|
|
{
|
|
RegionPtr changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, Glyphs);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec fbbox;
|
|
RegionRec fbreg;
|
|
|
|
changed = GlyphsToRegion(pScreen, nlists, lists, glyphs);
|
|
RegionTranslate(changed,
|
|
pDst->pDrawable->x, pDst->pDrawable->y);
|
|
|
|
fbbox.x1 = 0;
|
|
fbbox.y1 = 0;
|
|
fbbox.x2 = pScreen->width;
|
|
fbbox.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &fbbox, 1);
|
|
|
|
RegionIntersect(changed, changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
changed = RegionCreate(NullBox, 0);
|
|
}
|
|
|
|
(*ps->Glyphs)(op, pSrc, pDst, maskFormat, xSrc, ySrc, nlists, lists, glyphs);
|
|
|
|
add_changed(pScreen, changed);
|
|
|
|
RegionDestroy(changed);
|
|
|
|
RENDER_EPILOGUE(Glyphs);
|
|
}
|
|
|
|
static void vncHooksCompositeRects(CARD8 op, PicturePtr pDst,
|
|
xRenderColor * color, int nRect, xRectangle *rects)
|
|
{
|
|
RegionPtr changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, CompositeRects);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
changed = RECTS_TO_REGION(pScreen, nRect, rects, CT_NONE);
|
|
} else {
|
|
changed = RegionCreate(NullBox, 0);
|
|
}
|
|
|
|
(*ps->CompositeRects)(op, pDst, color, nRect, rects);
|
|
|
|
add_changed(pScreen, changed);
|
|
|
|
RegionDestroy(changed);
|
|
|
|
RENDER_EPILOGUE(CompositeRects);
|
|
}
|
|
|
|
static inline short FixedToShort(xFixed fixed)
|
|
{
|
|
return (fixed + 0x8000) >> 16;
|
|
}
|
|
|
|
static void vncHooksTrapezoids(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int ntrap, xTrapezoid * traps)
|
|
{
|
|
RegionRec changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, Trapezoids);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec box;
|
|
RegionRec fbreg;
|
|
|
|
// FIXME: We do a very crude bounding box around everything.
|
|
// Might not be worth optimizing since this call is rarely
|
|
// used.
|
|
box.x1 = SHRT_MAX;
|
|
box.y1 = SHRT_MAX;
|
|
box.x2 = 0;
|
|
box.y2 = 0;
|
|
for (int i = 0;i < ntrap;i++) {
|
|
if (FixedToShort(traps[i].left.p1.x) < box.x1)
|
|
box.x1 = FixedToShort(traps[i].left.p1.x);
|
|
if (FixedToShort(traps[i].left.p2.x) < box.x1)
|
|
box.x1 = FixedToShort(traps[i].left.p2.x);
|
|
if (FixedToShort(traps[i].top) < box.y1)
|
|
box.y1 = FixedToShort(traps[i].top);
|
|
if (FixedToShort(traps[i].right.p1.x) > box.x2)
|
|
box.x2 = FixedToShort(traps[i].right.p1.x);
|
|
if (FixedToShort(traps[i].right.p2.x) > box.x2)
|
|
box.x2 = FixedToShort(traps[i].right.p2.x);
|
|
if (FixedToShort(traps[i].bottom) > box.y2)
|
|
box.y2 = FixedToShort(traps[i].bottom);
|
|
}
|
|
|
|
box.x1 += pDst->pDrawable->x;
|
|
box.y1 += pDst->pDrawable->y;
|
|
box.x2 += pDst->pDrawable->x;
|
|
box.y2 += pDst->pDrawable->y;
|
|
RegionInitBoxes(&changed, &box, 1);
|
|
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = pScreen->width;
|
|
box.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &box, 1);
|
|
|
|
RegionIntersect(&changed, &changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
RegionNull(&changed);
|
|
}
|
|
|
|
(*ps->Trapezoids)(op, pSrc, pDst, maskFormat, xSrc, ySrc, ntrap, traps);
|
|
|
|
add_changed(pScreen, &changed);
|
|
|
|
RegionUninit(&changed);
|
|
|
|
RENDER_EPILOGUE(Trapezoids);
|
|
}
|
|
|
|
static void vncHooksTriangles(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int ntri, xTriangle * tris)
|
|
{
|
|
RegionRec changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, Triangles);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec box;
|
|
RegionRec fbreg;
|
|
|
|
// FIXME: We do a very crude bounding box around everything.
|
|
// Might not be worth optimizing since this call is rarely
|
|
// used.
|
|
box.x1 = SHRT_MAX;
|
|
box.y1 = SHRT_MAX;
|
|
box.x2 = 0;
|
|
box.y2 = 0;
|
|
for (int i = 0;i < ntri;i++) {
|
|
xFixed left, right, top, bottom;
|
|
|
|
left = min(min(tris[i].p1.x, tris[i].p2.x), tris[i].p3.x);
|
|
right = max(max(tris[i].p1.x, tris[i].p2.x), tris[i].p3.x);
|
|
top = min(min(tris[i].p1.y, tris[i].p2.y), tris[i].p3.y);
|
|
bottom = max(max(tris[i].p1.y, tris[i].p2.y), tris[i].p3.y);
|
|
|
|
if (FixedToShort(left) < box.x1)
|
|
box.x1 = FixedToShort(left);
|
|
if (FixedToShort(top) < box.y1)
|
|
box.y1 = FixedToShort(top);
|
|
if (FixedToShort(right) > box.x2)
|
|
box.x2 = FixedToShort(right);
|
|
if (FixedToShort(bottom) > box.y2)
|
|
box.y2 = FixedToShort(bottom);
|
|
}
|
|
|
|
box.x1 += pDst->pDrawable->x;
|
|
box.y1 += pDst->pDrawable->y;
|
|
box.x2 += pDst->pDrawable->x;
|
|
box.y2 += pDst->pDrawable->y;
|
|
RegionInitBoxes(&changed, &box, 1);
|
|
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = pScreen->width;
|
|
box.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &box, 1);
|
|
|
|
RegionIntersect(&changed, &changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
RegionNull(&changed);
|
|
}
|
|
|
|
(*ps->Triangles)(op, pSrc, pDst, maskFormat, xSrc, ySrc, ntri, tris);
|
|
|
|
add_changed(pScreen, &changed);
|
|
|
|
RegionUninit(&changed);
|
|
|
|
RENDER_EPILOGUE(Triangles);
|
|
}
|
|
|
|
#if PICTURE_SCREEN_VERSION >= 2
|
|
|
|
static void vncHooksTriStrip(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int npoint, xPointFixed * points)
|
|
{
|
|
RegionRec changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, TriStrip);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec box;
|
|
RegionRec fbreg;
|
|
|
|
// FIXME: We do a very crude bounding box around everything.
|
|
// Might not be worth optimizing since this call is rarely
|
|
// used.
|
|
box.x1 = SHRT_MAX;
|
|
box.y1 = SHRT_MAX;
|
|
box.x2 = 0;
|
|
box.y2 = 0;
|
|
for (int i = 0;i < npoint;i++) {
|
|
if (FixedToShort(points[i].x) < box.x1)
|
|
box.x1 = FixedToShort(points[i].x);
|
|
if (FixedToShort(points[i].y) < box.y1)
|
|
box.y1 = FixedToShort(points[i].y);
|
|
if (FixedToShort(points[i].x) > box.x2)
|
|
box.x2 = FixedToShort(points[i].x);
|
|
if (FixedToShort(points[i].y) > box.y2)
|
|
box.y2 = FixedToShort(points[i].y);
|
|
}
|
|
|
|
box.x1 += pDst->pDrawable->x;
|
|
box.y1 += pDst->pDrawable->y;
|
|
box.x2 += pDst->pDrawable->x;
|
|
box.y2 += pDst->pDrawable->y;
|
|
RegionInitBoxes(&changed, &box, 1);
|
|
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = pScreen->width;
|
|
box.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &box, 1);
|
|
|
|
RegionIntersect(&changed, &changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
RegionNull(&changed);
|
|
}
|
|
|
|
(*ps->TriStrip)(op, pSrc, pDst, maskFormat, xSrc, ySrc, npoint, points);
|
|
|
|
add_changed(pScreen, &changed);
|
|
|
|
RegionUninit(&changed);
|
|
|
|
RENDER_EPILOGUE(TriStrip);
|
|
}
|
|
|
|
static void vncHooksTriFan(CARD8 op, PicturePtr pSrc, PicturePtr pDst,
|
|
PictFormatPtr maskFormat, INT16 xSrc, INT16 ySrc,
|
|
int npoint, xPointFixed * points)
|
|
{
|
|
RegionRec changed;
|
|
|
|
RENDER_PROLOGUE(pDst->pDrawable->pScreen, TriFan);
|
|
|
|
if (is_visible(pDst->pDrawable)) {
|
|
BoxRec box;
|
|
RegionRec fbreg;
|
|
|
|
// FIXME: We do a very crude bounding box around everything.
|
|
// Might not be worth optimizing since this call is rarely
|
|
// used.
|
|
box.x1 = SHRT_MAX;
|
|
box.y1 = SHRT_MAX;
|
|
box.x2 = 0;
|
|
box.y2 = 0;
|
|
for (int i = 0;i < npoint;i++) {
|
|
if (FixedToShort(points[i].x) < box.x1)
|
|
box.x1 = FixedToShort(points[i].x);
|
|
if (FixedToShort(points[i].y) < box.y1)
|
|
box.y1 = FixedToShort(points[i].y);
|
|
if (FixedToShort(points[i].x) > box.x2)
|
|
box.x2 = FixedToShort(points[i].x);
|
|
if (FixedToShort(points[i].y) > box.y2)
|
|
box.y2 = FixedToShort(points[i].y);
|
|
}
|
|
|
|
box.x1 += pDst->pDrawable->x;
|
|
box.y1 += pDst->pDrawable->y;
|
|
box.x2 += pDst->pDrawable->x;
|
|
box.y2 += pDst->pDrawable->y;
|
|
RegionInitBoxes(&changed, &box, 1);
|
|
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = pScreen->width;
|
|
box.y2 = pScreen->height;
|
|
RegionInitBoxes(&fbreg, &box, 1);
|
|
|
|
RegionIntersect(&changed, &changed, &fbreg);
|
|
|
|
RegionUninit(&fbreg);
|
|
} else {
|
|
RegionNull(&changed);
|
|
}
|
|
|
|
(*ps->TriFan)(op, pSrc, pDst, maskFormat, xSrc, ySrc, npoint, points);
|
|
|
|
add_changed(pScreen, &changed);
|
|
|
|
RegionUninit(&changed);
|
|
|
|
RENDER_EPILOGUE(TriFan);
|
|
}
|
|
|
|
#endif /* PICTURE_SCREEN_VERSION */
|
|
|
|
#endif /* RENDER */
|
|
|
|
// Unwrap and rewrap helpers
|
|
|
|
#define RANDR_PROLOGUE(field) \
|
|
rrScrPrivPtr rp = rrGetScrPriv(pScreen); \
|
|
vncHooksScreenPtr vncHooksScreen = vncHooksScreenPrivate(pScreen); \
|
|
unwrap(vncHooksScreen, rp, rr##field); \
|
|
DBGPRINT((stderr,"vncHooksRandR" #field " called\n"));
|
|
|
|
#define RANDR_EPILOGUE(field) \
|
|
wrap(vncHooksScreen, rp, rr##field, vncHooksRandR##field); \
|
|
|
|
// RandRSetConfig - follow any framebuffer changes
|
|
|
|
static Bool vncHooksRandRSetConfig(ScreenPtr pScreen, Rotation rotation,
|
|
int rate, RRScreenSizePtr pSize)
|
|
{
|
|
Bool ret;
|
|
|
|
RANDR_PROLOGUE(SetConfig);
|
|
|
|
vncPreScreenResize(pScreen->myNum);
|
|
ret = (*rp->rrSetConfig)(pScreen, rotation, rate, pSize);
|
|
vncPostScreenResize(pScreen->myNum, ret, pScreen->width, pScreen->height);
|
|
|
|
RANDR_EPILOGUE(SetConfig);
|
|
|
|
if (!ret)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool vncHooksRandRScreenSetSize(ScreenPtr pScreen,
|
|
CARD16 width, CARD16 height,
|
|
CARD32 mmWidth, CARD32 mmHeight)
|
|
{
|
|
Bool ret;
|
|
|
|
RANDR_PROLOGUE(ScreenSetSize);
|
|
|
|
vncPreScreenResize(pScreen->myNum);
|
|
ret = (*rp->rrScreenSetSize)(pScreen, width, height, mmWidth, mmHeight);
|
|
vncPostScreenResize(pScreen->myNum, ret, pScreen->width, pScreen->height);
|
|
|
|
RANDR_EPILOGUE(ScreenSetSize);
|
|
|
|
if (!ret)
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool vncHooksRandRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc,
|
|
RRModePtr mode, int x, int y,
|
|
Rotation rotation, int num_outputs,
|
|
RROutputPtr *outputs)
|
|
{
|
|
Bool ret;
|
|
|
|
RANDR_PROLOGUE(CrtcSet);
|
|
|
|
ret = (*rp->rrCrtcSet)(pScreen, crtc, mode, x, y, rotation,
|
|
num_outputs, outputs);
|
|
|
|
RANDR_EPILOGUE(CrtcSet);
|
|
|
|
if (!ret)
|
|
return FALSE;
|
|
|
|
vncRefreshScreenLayout(pScreen->myNum);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GC "funcs"
|
|
//
|
|
|
|
// Unwrap and rewrap helpers
|
|
|
|
#define GC_FUNC_PROLOGUE(pGC, name)\
|
|
vncHooksGCPtr pGCPriv = vncHooksGCPrivate(pGC);\
|
|
unwrap(pGCPriv, pGC, funcs);\
|
|
if (pGCPriv->ops) unwrap(pGCPriv, pGC, ops)\
|
|
DBGPRINT((stderr,"vncHooks" #name " called\n"))
|
|
|
|
#define GC_FUNC_EPILOGUE(pGC)\
|
|
wrap(pGCPriv, pGC, funcs, &vncHooksGCFuncs);\
|
|
if (pGCPriv->ops) wrap(pGCPriv, pGC, ops, &vncHooksGCOps)
|
|
|
|
// ValidateGC - wrap the "ops" if the drawable is on screen
|
|
|
|
static void vncHooksValidateGC(GCPtr pGC, unsigned long changes,
|
|
DrawablePtr pDrawable)
|
|
{
|
|
GC_FUNC_PROLOGUE(pGC, ValidateGC);
|
|
(*pGC->funcs->ValidateGC) (pGC, changes, pDrawable);
|
|
if (is_visible(pDrawable)) {
|
|
pGCPriv->ops = pGC->ops;
|
|
DBGPRINT((stderr,"vncHooksValidateGC: wrapped GC ops\n"));
|
|
} else {
|
|
pGCPriv->ops = NULL;
|
|
}
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
// Other GC funcs - just unwrap and call on
|
|
|
|
static void vncHooksChangeGC(GCPtr pGC, unsigned long mask) {
|
|
GC_FUNC_PROLOGUE(pGC, ChangeGC);
|
|
(*pGC->funcs->ChangeGC) (pGC, mask);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
static void vncHooksCopyGC(GCPtr src, unsigned long mask, GCPtr dst) {
|
|
GC_FUNC_PROLOGUE(dst, CopyGC);
|
|
(*dst->funcs->CopyGC) (src, mask, dst);
|
|
GC_FUNC_EPILOGUE(dst);
|
|
}
|
|
static void vncHooksDestroyGC(GCPtr pGC) {
|
|
GC_FUNC_PROLOGUE(pGC, DestroyGC);
|
|
(*pGC->funcs->DestroyGC) (pGC);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
static void vncHooksChangeClip(GCPtr pGC, int type, void * pValue, int nrects)
|
|
{
|
|
GC_FUNC_PROLOGUE(pGC, ChangeClip);
|
|
(*pGC->funcs->ChangeClip) (pGC, type, pValue, nrects);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
static void vncHooksDestroyClip(GCPtr pGC) {
|
|
GC_FUNC_PROLOGUE(pGC, DestroyClip);
|
|
(*pGC->funcs->DestroyClip) (pGC);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
static void vncHooksCopyClip(GCPtr dst, GCPtr src) {
|
|
GC_FUNC_PROLOGUE(dst, CopyClip);
|
|
(*dst->funcs->CopyClip) (dst, src);
|
|
GC_FUNC_EPILOGUE(dst);
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GC "ops"
|
|
//
|
|
|
|
// Unwrap and rewrap helpers
|
|
|
|
#if XORG >= 116
|
|
#define GC_OP_PROLOGUE(pGC, name)\
|
|
vncHooksGCPtr pGCPriv = vncHooksGCPrivate(pGC);\
|
|
const GCFuncs *oldFuncs = pGC->funcs;\
|
|
unwrap(pGCPriv, pGC, funcs);\
|
|
unwrap(pGCPriv, pGC, ops);\
|
|
DBGPRINT((stderr,"vncHooks" #name " called\n"))
|
|
#else
|
|
#define GC_OP_PROLOGUE(pGC, name)\
|
|
vncHooksGCPtr pGCPriv = vncHooksGCPrivate(pGC);\
|
|
GCFuncs *oldFuncs = pGC->funcs;\
|
|
unwrap(pGCPriv, pGC, funcs);\
|
|
unwrap(pGCPriv, pGC, ops);\
|
|
DBGPRINT((stderr,"vncHooks" #name " called\n"))
|
|
#endif
|
|
|
|
#define GC_OP_EPILOGUE(pGC)\
|
|
wrap(pGCPriv, pGC, funcs, oldFuncs); \
|
|
wrap(pGCPriv, pGC, ops, &vncHooksGCOps)
|
|
|
|
// FillSpans - assume the entire clip region is damaged. This is pessimistic,
|
|
// but I believe this function is rarely used so it doesn't matter.
|
|
|
|
static void vncHooksFillSpans(DrawablePtr pDrawable, GCPtr pGC, int nInit,
|
|
DDXPointPtr pptInit, int *pwidthInit,
|
|
int fSorted)
|
|
{
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, FillSpans);
|
|
|
|
RegionNull(®);
|
|
RegionCopy(®, pGC->pCompositeClip);
|
|
|
|
if (pDrawable->type == DRAWABLE_WINDOW)
|
|
RegionIntersect(®, ®, &((WindowPtr)pDrawable)->borderClip);
|
|
|
|
(*pGC->ops->FillSpans) (pDrawable, pGC, nInit, pptInit, pwidthInit, fSorted);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// SetSpans - assume the entire clip region is damaged. This is pessimistic,
|
|
// but I believe this function is rarely used so it doesn't matter.
|
|
|
|
static void vncHooksSetSpans(DrawablePtr pDrawable, GCPtr pGC, char *psrc,
|
|
DDXPointPtr ppt, int *pwidth, int nspans,
|
|
int fSorted)
|
|
{
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, SetSpans);
|
|
|
|
RegionNull(®);
|
|
RegionCopy(®, pGC->pCompositeClip);
|
|
|
|
if (pDrawable->type == DRAWABLE_WINDOW)
|
|
RegionIntersect(®, ®, &((WindowPtr)pDrawable)->borderClip);
|
|
|
|
(*pGC->ops->SetSpans) (pDrawable, pGC, psrc, ppt, pwidth, nspans, fSorted);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PutImage - changed region is the given rectangle, clipped by pCompositeClip
|
|
|
|
static void vncHooksPutImage(DrawablePtr pDrawable, GCPtr pGC, int depth,
|
|
int x, int y, int w, int h, int leftPad,
|
|
int format, char *pBits)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PutImage);
|
|
|
|
box.x1 = x + pDrawable->x;
|
|
box.y1 = y + pDrawable->y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PutImage) (pDrawable, pGC, depth, x, y, w, h, leftPad, format,
|
|
pBits);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// CopyArea - destination of the copy is the dest rectangle, clipped by
|
|
// pCompositeClip. Any parts of the destination which cannot be copied from
|
|
// the source (could be all of it) go into the changed region.
|
|
|
|
static RegionPtr vncHooksCopyArea(DrawablePtr pSrc, DrawablePtr pDst,
|
|
GCPtr pGC, int srcx, int srcy, int w, int h,
|
|
int dstx, int dsty)
|
|
{
|
|
RegionRec dst, src, changed;
|
|
|
|
RegionPtr ret;
|
|
|
|
GC_OP_PROLOGUE(pGC, CopyArea);
|
|
|
|
// Apparently this happens now and then...
|
|
if ((w == 0) || (h == 0))
|
|
RegionNull(&dst);
|
|
else {
|
|
BoxRec box;
|
|
|
|
box.x1 = dstx + pDst->x;
|
|
box.y1 = dsty + pDst->y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
|
|
RegionInitBoxes(&dst, &box, 1);
|
|
}
|
|
|
|
RegionIntersect(&dst, &dst, pGC->pCompositeClip);
|
|
|
|
// The source of the data has to be something that's on screen.
|
|
if (is_visible(pSrc)) {
|
|
BoxRec box;
|
|
|
|
box.x1 = srcx + pSrc->x;
|
|
box.y1 = srcy + pSrc->y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
|
|
RegionInitBoxes(&src, &box, 1);
|
|
|
|
if ((pSrc->type == DRAWABLE_WINDOW) &&
|
|
RegionNotEmpty(&((WindowPtr)pSrc)->clipList)) {
|
|
RegionIntersect(&src, &src, &((WindowPtr)pSrc)->clipList);
|
|
}
|
|
|
|
RegionTranslate(&src,
|
|
dstx + pDst->x - srcx - pSrc->x,
|
|
dsty + pDst->y - srcy - pSrc->y);
|
|
} else {
|
|
RegionNull(&src);
|
|
}
|
|
|
|
RegionNull(&changed);
|
|
|
|
RegionSubtract(&changed, &dst, &src);
|
|
RegionIntersect(&dst, &dst, &src);
|
|
|
|
ret = (*pGC->ops->CopyArea) (pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty);
|
|
|
|
add_copied(pGC->pScreen, &dst,
|
|
dstx + pDst->x - srcx - pSrc->x,
|
|
dsty + pDst->y - srcy - pSrc->y);
|
|
|
|
add_changed(pGC->pScreen, &changed);
|
|
|
|
RegionUninit(&dst);
|
|
RegionUninit(&src);
|
|
RegionUninit(&changed);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
// CopyPlane - changed region is the destination rectangle, clipped by
|
|
// pCompositeClip
|
|
|
|
static RegionPtr vncHooksCopyPlane(DrawablePtr pSrc, DrawablePtr pDst,
|
|
GCPtr pGC, int srcx, int srcy, int w, int h,
|
|
int dstx, int dsty, unsigned long plane)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
RegionPtr ret;
|
|
|
|
GC_OP_PROLOGUE(pGC, CopyPlane);
|
|
|
|
box.x1 = dstx + pDst->x;
|
|
box.y1 = dsty + pDst->y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
ret = (*pGC->ops->CopyPlane) (pSrc, pDst, pGC, srcx, srcy, w, h,
|
|
dstx, dsty, plane);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
|
|
return ret;
|
|
}
|
|
|
|
// PolyPoint - changed region is the bounding rect, clipped by pCompositeClip
|
|
|
|
static void vncHooksPolyPoint(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, xPoint *pts)
|
|
{
|
|
int minX, minY, maxX, maxY;
|
|
int i;
|
|
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyPoint);
|
|
|
|
if (npt == 0) {
|
|
(*pGC->ops->PolyPoint) (pDrawable, pGC, mode, npt, pts);
|
|
goto out;
|
|
}
|
|
|
|
minX = pts[0].x;
|
|
maxX = pts[0].x;
|
|
minY = pts[0].y;
|
|
maxY = pts[0].y;
|
|
|
|
if (mode == CoordModePrevious) {
|
|
int x = pts[0].x;
|
|
int y = pts[0].y;
|
|
|
|
for (i = 1; i < npt; i++) {
|
|
x += pts[i].x;
|
|
y += pts[i].y;
|
|
if (x < minX) minX = x;
|
|
if (x > maxX) maxX = x;
|
|
if (y < minY) minY = y;
|
|
if (y > maxY) maxY = y;
|
|
}
|
|
} else {
|
|
for (i = 1; i < npt; i++) {
|
|
if (pts[i].x < minX) minX = pts[i].x;
|
|
if (pts[i].x > maxX) maxX = pts[i].x;
|
|
if (pts[i].y < minY) minY = pts[i].y;
|
|
if (pts[i].y > maxY) maxY = pts[i].y;
|
|
}
|
|
}
|
|
|
|
box.x1 = minX + pDrawable->x;
|
|
box.y1 = minY + pDrawable->y;
|
|
box.x2 = maxX + 1 + pDrawable->x;
|
|
box.y2 = maxY + 1 + pDrawable->y;
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyPoint) (pDrawable, pGC, mode, npt, pts);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// Polylines - changed region is the union of the bounding rects of each line,
|
|
// clipped by pCompositeClip. If there are more than MAX_RECTS_PER_OP lines,
|
|
// just use the bounding rect of all the lines.
|
|
|
|
static void vncHooksPolylines(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, DDXPointPtr ppts)
|
|
{
|
|
int nRegRects;
|
|
xRectangle regRects[MAX_RECTS_PER_OP];
|
|
|
|
int lw;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, Polylines);
|
|
|
|
if (npt == 0) {
|
|
(*pGC->ops->Polylines) (pDrawable, pGC, mode, npt, ppts);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = npt - 1;
|
|
|
|
lw = pGC->lineWidth;
|
|
if (lw == 0)
|
|
lw = 1;
|
|
|
|
if (npt == 1)
|
|
{
|
|
// a single point
|
|
nRegRects = 1;
|
|
regRects[0].x = pDrawable->x + ppts[0].x - lw;
|
|
regRects[0].y = pDrawable->y + ppts[0].y - lw;
|
|
regRects[0].width = 2*lw;
|
|
regRects[0].height = 2*lw;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* mitered joins can project quite a way from
|
|
* the line end; the 11 degree miter limit limits
|
|
* this extension to lw / (2 * tan(11/2)), rounded up
|
|
* and converted to int yields 6 * lw
|
|
*/
|
|
|
|
int extra;
|
|
|
|
int prevX, prevY, curX, curY;
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
|
|
int i;
|
|
|
|
extra = lw / 2;
|
|
if (pGC->joinStyle == JoinMiter) {
|
|
extra = 6 * lw;
|
|
}
|
|
|
|
prevX = ppts[0].x + pDrawable->x;
|
|
prevY = ppts[0].y + pDrawable->y;
|
|
minX = maxX = prevX;
|
|
minY = maxY = prevY;
|
|
|
|
for (i = 0; i < nRegRects; i++) {
|
|
if (mode == CoordModeOrigin) {
|
|
curX = pDrawable->x + ppts[i+1].x;
|
|
curY = pDrawable->y + ppts[i+1].y;
|
|
} else {
|
|
curX = prevX + ppts[i+1].x;
|
|
curY = prevY + ppts[i+1].y;
|
|
}
|
|
|
|
if (prevX > curX) {
|
|
rectX1 = curX - extra;
|
|
rectX2 = prevX + extra + 1;
|
|
} else {
|
|
rectX1 = prevX - extra;
|
|
rectX2 = curX + extra + 1;
|
|
}
|
|
|
|
if (prevY > curY) {
|
|
rectY1 = curY - extra;
|
|
rectY2 = prevY + extra + 1;
|
|
} else {
|
|
rectY1 = prevY - extra;
|
|
rectY2 = curY + extra + 1;
|
|
}
|
|
|
|
if (nRegRects <= MAX_RECTS_PER_OP) {
|
|
regRects[i].x = rectX1;
|
|
regRects[i].y = rectY1;
|
|
regRects[i].width = rectX2 - rectX1;
|
|
regRects[i].height = rectY2 - rectY1;
|
|
} else {
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
|
|
prevX = curX;
|
|
prevY = curY;
|
|
}
|
|
|
|
if (nRegRects > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->Polylines) (pDrawable, pGC, mode, npt, ppts);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolySegment - changed region is the union of the bounding rects of each
|
|
// segment, clipped by pCompositeClip. If there are more than MAX_RECTS_PER_OP
|
|
// segments, just use the bounding rect of all the segments.
|
|
|
|
static void vncHooksPolySegment(DrawablePtr pDrawable, GCPtr pGC, int nseg,
|
|
xSegment *segs)
|
|
{
|
|
xRectangle regRects[MAX_RECTS_PER_OP];
|
|
int nRegRects;
|
|
|
|
int lw, extra;
|
|
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
|
|
int i;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolySegment);
|
|
|
|
if (nseg == 0) {
|
|
(*pGC->ops->PolySegment) (pDrawable, pGC, nseg, segs);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = nseg;
|
|
|
|
lw = pGC->lineWidth;
|
|
extra = lw / 2;
|
|
|
|
minX = maxX = segs[0].x1;
|
|
minY = maxY = segs[0].y1;
|
|
|
|
for (i = 0; i < nseg; i++) {
|
|
if (segs[i].x1 > segs[i].x2) {
|
|
rectX1 = pDrawable->x + segs[i].x2 - extra;
|
|
rectX2 = pDrawable->x + segs[i].x1 + extra + 1;
|
|
} else {
|
|
rectX1 = pDrawable->x + segs[i].x1 - extra;
|
|
rectX2 = pDrawable->x + segs[i].x2 + extra + 1;
|
|
}
|
|
|
|
if (segs[i].y1 > segs[i].y2) {
|
|
rectY1 = pDrawable->y + segs[i].y2 - extra;
|
|
rectY2 = pDrawable->y + segs[i].y1 + extra + 1;
|
|
} else {
|
|
rectY1 = pDrawable->y + segs[i].y1 - extra;
|
|
rectY2 = pDrawable->y + segs[i].y2 + extra + 1;
|
|
}
|
|
|
|
if (nseg <= MAX_RECTS_PER_OP) {
|
|
regRects[i].x = rectX1;
|
|
regRects[i].y = rectY1;
|
|
regRects[i].width = rectX2 - rectX1;
|
|
regRects[i].height = rectY2 - rectY1;
|
|
} else {
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
}
|
|
|
|
if (nseg > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolySegment) (pDrawable, pGC, nseg, segs);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolyRectangle - changed region is the union of the bounding rects around
|
|
// each side of the outline rectangles, clipped by pCompositeClip. If there
|
|
// are more than MAX_RECTS_PER_OP rectangles, just use the bounding rect of all
|
|
// the rectangles.
|
|
|
|
static void vncHooksPolyRectangle(DrawablePtr pDrawable, GCPtr pGC, int nrects,
|
|
xRectangle *rects)
|
|
{
|
|
xRectangle regRects[MAX_RECTS_PER_OP*4];
|
|
int nRegRects;
|
|
|
|
int lw, extra;
|
|
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
|
|
int i;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyRectangle);
|
|
|
|
if (nrects == 0) {
|
|
(*pGC->ops->PolyRectangle) (pDrawable, pGC, nrects, rects);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = nrects * 4;
|
|
|
|
lw = pGC->lineWidth;
|
|
extra = lw / 2;
|
|
|
|
minX = maxX = rects[0].x;
|
|
minY = maxY = rects[0].y;
|
|
|
|
for (i = 0; i < nrects; i++) {
|
|
if (nrects <= MAX_RECTS_PER_OP) {
|
|
regRects[i*4].x = rects[i].x - extra + pDrawable->x;
|
|
regRects[i*4].y = rects[i].y - extra + pDrawable->y;
|
|
regRects[i*4].width = rects[i].width + 1 + 2 * extra;
|
|
regRects[i*4].height = 1 + 2 * extra;
|
|
|
|
regRects[i*4+1].x = rects[i].x - extra + pDrawable->x;
|
|
regRects[i*4+1].y = rects[i].y - extra + pDrawable->y;
|
|
regRects[i*4+1].width = 1 + 2 * extra;
|
|
regRects[i*4+1].height = rects[i].height + 1 + 2 * extra;
|
|
|
|
regRects[i*4+2].x = rects[i].x + rects[i].width - extra + pDrawable->x;
|
|
regRects[i*4+2].y = rects[i].y - extra + pDrawable->y;
|
|
regRects[i*4+2].width = 1 + 2 * extra;
|
|
regRects[i*4+2].height = rects[i].height + 1 + 2 * extra;
|
|
|
|
regRects[i*4+3].x = rects[i].x - extra + pDrawable->x;
|
|
regRects[i*4+3].y = rects[i].y + rects[i].height - extra + pDrawable->y;
|
|
regRects[i*4+3].width = rects[i].width + 1 + 2 * extra;
|
|
regRects[i*4+3].height = 1 + 2 * extra;
|
|
} else {
|
|
rectX1 = pDrawable->x + rects[i].x - extra;
|
|
rectY1 = pDrawable->y + rects[i].y - extra;
|
|
rectX2 = pDrawable->x + rects[i].x + rects[i].width + extra+1;
|
|
rectY2 = pDrawable->y + rects[i].y + rects[i].height + extra+1;
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
}
|
|
|
|
if (nrects > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyRectangle) (pDrawable, pGC, nrects, rects);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolyArc - changed region is the union of bounding rects around each arc,
|
|
// clipped by pCompositeClip. If there are more than MAX_RECTS_PER_OP
|
|
// arcs, just use the bounding rect of all the arcs.
|
|
|
|
static void vncHooksPolyArc(DrawablePtr pDrawable, GCPtr pGC, int narcs,
|
|
xArc *arcs)
|
|
{
|
|
xRectangle regRects[MAX_RECTS_PER_OP];
|
|
int nRegRects;
|
|
|
|
int lw, extra;
|
|
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
|
|
int i;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyArc);
|
|
|
|
if (narcs == 0) {
|
|
(*pGC->ops->PolyArc) (pDrawable, pGC, narcs, arcs);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = narcs;
|
|
|
|
lw = pGC->lineWidth;
|
|
if (lw == 0)
|
|
lw = 1;
|
|
extra = lw / 2;
|
|
|
|
minX = maxX = arcs[0].x;
|
|
minY = maxY = arcs[0].y;
|
|
|
|
for (i = 0; i < narcs; i++) {
|
|
if (narcs <= MAX_RECTS_PER_OP) {
|
|
regRects[i].x = arcs[i].x - extra + pDrawable->x;
|
|
regRects[i].y = arcs[i].y - extra + pDrawable->y;
|
|
regRects[i].width = arcs[i].width + lw;
|
|
regRects[i].height = arcs[i].height + lw;
|
|
} else {
|
|
rectX1 = pDrawable->x + arcs[i].x - extra;
|
|
rectY1 = pDrawable->y + arcs[i].y - extra;
|
|
rectX2 = pDrawable->x + arcs[i].x + arcs[i].width + lw;
|
|
rectY2 = pDrawable->y + arcs[i].y + arcs[i].height + lw;
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
}
|
|
|
|
if (narcs > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyArc) (pDrawable, pGC, narcs, arcs);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
|
|
// FillPolygon - changed region is the bounding rect around the polygon,
|
|
// clipped by pCompositeClip
|
|
|
|
static void vncHooksFillPolygon(DrawablePtr pDrawable, GCPtr pGC, int shape,
|
|
int mode, int count, DDXPointPtr pts)
|
|
{
|
|
int minX, minY, maxX, maxY;
|
|
int i;
|
|
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, FillPolygon);
|
|
|
|
if (count == 0) {
|
|
(*pGC->ops->FillPolygon) (pDrawable, pGC, shape, mode, count, pts);
|
|
goto out;
|
|
}
|
|
|
|
minX = pts[0].x;
|
|
maxX = pts[0].x;
|
|
minY = pts[0].y;
|
|
maxY = pts[0].y;
|
|
|
|
if (mode == CoordModePrevious) {
|
|
int x = pts[0].x;
|
|
int y = pts[0].y;
|
|
|
|
for (i = 1; i < count; i++) {
|
|
x += pts[i].x;
|
|
y += pts[i].y;
|
|
if (x < minX) minX = x;
|
|
if (x > maxX) maxX = x;
|
|
if (y < minY) minY = y;
|
|
if (y > maxY) maxY = y;
|
|
}
|
|
} else {
|
|
for (i = 1; i < count; i++) {
|
|
if (pts[i].x < minX) minX = pts[i].x;
|
|
if (pts[i].x > maxX) maxX = pts[i].x;
|
|
if (pts[i].y < minY) minY = pts[i].y;
|
|
if (pts[i].y > maxY) maxY = pts[i].y;
|
|
}
|
|
}
|
|
|
|
box.x1 = minX + pDrawable->x;
|
|
box.y1 = minY + pDrawable->y;
|
|
box.x2 = maxX + 1 + pDrawable->x;
|
|
box.y2 = maxY + 1 + pDrawable->y;
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->FillPolygon) (pDrawable, pGC, shape, mode, count, pts);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolyFillRect - changed region is the union of the rectangles, clipped by
|
|
// pCompositeClip. If there are more than MAX_RECTS_PER_OP rectangles, just
|
|
// use the bounding rect of all the rectangles.
|
|
|
|
static void vncHooksPolyFillRect(DrawablePtr pDrawable, GCPtr pGC, int nrects,
|
|
xRectangle *rects)
|
|
{
|
|
xRectangle regRects[MAX_RECTS_PER_OP];
|
|
int nRegRects;
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
int i;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyFillRect);
|
|
|
|
if (nrects == 0) {
|
|
(*pGC->ops->PolyFillRect) (pDrawable, pGC, nrects, rects);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = nrects;
|
|
minX = maxX = rects[0].x;
|
|
minY = maxY = rects[0].y;
|
|
|
|
for (i = 0; i < nrects; i++) {
|
|
if (nrects <= MAX_RECTS_PER_OP) {
|
|
regRects[i].x = rects[i].x + pDrawable->x;
|
|
regRects[i].y = rects[i].y + pDrawable->y;
|
|
regRects[i].width = rects[i].width;
|
|
regRects[i].height = rects[i].height;
|
|
} else {
|
|
rectX1 = pDrawable->x + rects[i].x;
|
|
rectY1 = pDrawable->y + rects[i].y;
|
|
rectX2 = pDrawable->x + rects[i].x + rects[i].width;
|
|
rectY2 = pDrawable->y + rects[i].y + rects[i].height;
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
}
|
|
|
|
if (nrects > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyFillRect) (pDrawable, pGC, nrects, rects);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolyFillArc - changed region is the union of bounding rects around each arc,
|
|
// clipped by pCompositeClip. If there are more than MAX_RECTS_PER_OP arcs,
|
|
// just use the bounding rect of all the arcs.
|
|
|
|
static void vncHooksPolyFillArc(DrawablePtr pDrawable, GCPtr pGC, int narcs,
|
|
xArc *arcs)
|
|
{
|
|
xRectangle regRects[MAX_RECTS_PER_OP];
|
|
int nRegRects;
|
|
|
|
int lw, extra;
|
|
|
|
int rectX1, rectY1, rectX2, rectY2;
|
|
int minX, minY, maxX, maxY;
|
|
|
|
int i;
|
|
|
|
RegionPtr reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyFillArc);
|
|
|
|
if (narcs == 0) {
|
|
(*pGC->ops->PolyFillArc) (pDrawable, pGC, narcs, arcs);
|
|
goto out;
|
|
}
|
|
|
|
nRegRects = narcs;
|
|
|
|
lw = pGC->lineWidth;
|
|
if (lw == 0)
|
|
lw = 1;
|
|
extra = lw / 2;
|
|
|
|
minX = maxX = arcs[0].x;
|
|
minY = maxY = arcs[0].y;
|
|
|
|
for (i = 0; i < narcs; i++) {
|
|
if (narcs <= MAX_RECTS_PER_OP) {
|
|
regRects[i].x = arcs[i].x - extra + pDrawable->x;
|
|
regRects[i].y = arcs[i].y - extra + pDrawable->y;
|
|
regRects[i].width = arcs[i].width + lw;
|
|
regRects[i].height = arcs[i].height + lw;
|
|
} else {
|
|
rectX1 = pDrawable->x + arcs[i].x - extra;
|
|
rectY1 = pDrawable->y + arcs[i].y - extra;
|
|
rectX2 = pDrawable->x + arcs[i].x + arcs[i].width + lw;
|
|
rectY2 = pDrawable->y + arcs[i].y + arcs[i].height + lw;
|
|
if (rectX1 < minX) minX = rectX1;
|
|
if (rectY1 < minY) minY = rectY1;
|
|
if (rectX2 > maxX) maxX = rectX2;
|
|
if (rectY2 > maxY) maxY = rectY2;
|
|
}
|
|
}
|
|
|
|
if (narcs > MAX_RECTS_PER_OP) {
|
|
regRects[0].x = minX;
|
|
regRects[0].y = minY;
|
|
regRects[0].width = maxX - minX;
|
|
regRects[0].height = maxY - minY;
|
|
nRegRects = 1;
|
|
}
|
|
|
|
reg = RECTS_TO_REGION(pGC->pScreen, nRegRects, regRects, CT_NONE);
|
|
RegionIntersect(reg, reg, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyFillArc) (pDrawable, pGC, narcs, arcs);
|
|
|
|
add_changed(pGC->pScreen, reg);
|
|
|
|
RegionDestroy(reg);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// GetTextBoundingRect - calculate a bounding rectangle around n chars of a
|
|
// font. Not particularly accurate, but good enough.
|
|
|
|
static void GetTextBoundingRect(DrawablePtr pDrawable, FontPtr font, int x,
|
|
int y, int nchars, BoxPtr box)
|
|
{
|
|
int ascent = max(FONTASCENT(font), FONTMAXBOUNDS(font, ascent));
|
|
int descent = max(FONTDESCENT(font), FONTMAXBOUNDS(font, descent));
|
|
int charWidth = max(FONTMAXBOUNDS(font,rightSideBearing),
|
|
FONTMAXBOUNDS(font,characterWidth));
|
|
|
|
box->x1 = pDrawable->x + x;
|
|
box->y1 = pDrawable->y + y - ascent;
|
|
box->x2 = box->x1 + charWidth * nchars;
|
|
box->y2 = box->y1 + ascent + descent;
|
|
|
|
if (FONTMINBOUNDS(font,leftSideBearing) < 0)
|
|
box->x1 += FONTMINBOUNDS(font,leftSideBearing);
|
|
}
|
|
|
|
// PolyText8 - changed region is bounding rect around count chars, clipped by
|
|
// pCompositeClip
|
|
|
|
static int vncHooksPolyText8(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, char *chars)
|
|
{
|
|
int ret;
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyText8);
|
|
|
|
if (count == 0) {
|
|
ret = (*pGC->ops->PolyText8) (pDrawable, pGC, x, y, count, chars);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, count, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
ret = (*pGC->ops->PolyText8) (pDrawable, pGC, x, y, count, chars);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
|
|
return ret;
|
|
}
|
|
|
|
// PolyText16 - changed region is bounding rect around count chars, clipped by
|
|
// pCompositeClip
|
|
|
|
static int vncHooksPolyText16(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, unsigned short *chars)
|
|
{
|
|
int ret;
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyText16);
|
|
|
|
if (count == 0) {
|
|
ret = (*pGC->ops->PolyText16) (pDrawable, pGC, x, y, count, chars);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, count, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
ret = (*pGC->ops->PolyText16) (pDrawable, pGC, x, y, count, chars);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
|
|
return ret;
|
|
}
|
|
|
|
// ImageText8 - changed region is bounding rect around count chars, clipped by
|
|
// pCompositeClip
|
|
|
|
static void vncHooksImageText8(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, char *chars)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, ImageText8);
|
|
|
|
if (count == 0) {
|
|
(*pGC->ops->ImageText8) (pDrawable, pGC, x, y, count, chars);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, count, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->ImageText8) (pDrawable, pGC, x, y, count, chars);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// ImageText16 - changed region is bounding rect around count chars, clipped by
|
|
// pCompositeClip
|
|
|
|
static void vncHooksImageText16(DrawablePtr pDrawable, GCPtr pGC, int x, int y,
|
|
int count, unsigned short *chars)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, ImageText16);
|
|
|
|
if (count == 0) {
|
|
(*pGC->ops->ImageText16) (pDrawable, pGC, x, y, count, chars);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, count, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->ImageText16) (pDrawable, pGC, x, y, count, chars);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// ImageGlyphBlt - changed region is bounding rect around nglyph chars, clipped
|
|
// by pCompositeClip
|
|
|
|
static void vncHooksImageGlyphBlt(DrawablePtr pDrawable, GCPtr pGC, int x,
|
|
int y, unsigned int nglyph,
|
|
CharInfoPtr *ppci, void * pglyphBase)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, ImageGlyphBlt);
|
|
|
|
if (nglyph == 0) {
|
|
(*pGC->ops->ImageGlyphBlt) (pDrawable, pGC, x, y, nglyph, ppci,pglyphBase);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, nglyph, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->ImageGlyphBlt) (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PolyGlyphBlt - changed region is bounding rect around nglyph chars, clipped
|
|
// by pCompositeClip
|
|
|
|
static void vncHooksPolyGlyphBlt(DrawablePtr pDrawable, GCPtr pGC, int x,
|
|
int y, unsigned int nglyph,
|
|
CharInfoPtr *ppci, void * pglyphBase)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PolyGlyphBlt);
|
|
|
|
if (nglyph == 0) {
|
|
(*pGC->ops->PolyGlyphBlt) (pDrawable, pGC, x, y, nglyph, ppci,pglyphBase);
|
|
goto out;
|
|
}
|
|
|
|
GetTextBoundingRect(pDrawable, pGC->font, x, y, nglyph, &box);
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PolyGlyphBlt) (pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
out:
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
// PushPixels - changed region is the given rectangle, clipped by
|
|
// pCompositeClip
|
|
|
|
static void vncHooksPushPixels(GCPtr pGC, PixmapPtr pBitMap,
|
|
DrawablePtr pDrawable, int w, int h, int x,
|
|
int y)
|
|
{
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
GC_OP_PROLOGUE(pGC, PushPixels);
|
|
|
|
box.x1 = x + pDrawable->x;
|
|
box.y1 = y + pDrawable->y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
|
|
RegionInitBoxes(®, &box, 1);
|
|
RegionIntersect(®, ®, pGC->pCompositeClip);
|
|
|
|
(*pGC->ops->PushPixels) (pGC, pBitMap, pDrawable, w, h, x, y);
|
|
|
|
add_changed(pGC->pScreen, ®);
|
|
|
|
RegionUninit(®);
|
|
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|