2025-08-08 20:00:36 +02:00

4801 lines
137 KiB
C

/**************************************************************************/
/* */
/* Copyright (c) 2001, 2011 NoMachine (http://www.nomachine.com) */
/* Copyright (c) 2008-2017 Oleksandr Shneyder <o.shneyder@phoca-gmbh.de> */
/* Copyright (c) 2011-2022 Mike Gabriel <mike.gabriel@das-netzwerkteam.de>*/
/* Copyright (c) 2014-2019 Mihai Moldovan <ionic@ionic.de> */
/* Copyright (c) 2014-2022 Ulrich Sibiller <uli42@gmx.de> */
/* Copyright (c) 2015-2016 Qindel Group (http://www.qindel.com) */
/* */
/* NXAGENT, NX protocol compression and NX extensions to this software */
/* are copyright of the aforementioned persons and companies. */
/* */
/* Redistribution and use of the present software is allowed according */
/* to terms specified in the file LICENSE which comes in the source */
/* distribution. */
/* */
/* All rights reserved. */
/* */
/* NOTE: This software has received contributions from various other */
/* contributors, only the core maintainers and supporters are listed as */
/* copyright holders. Please contact us, if you feel you should be listed */
/* as copyright holder, as well. */
/* */
/**************************************************************************/
/*
Copyright 1993 by Davor Matic
Permission to use, copy, modify, distribute, and sell this software
and its documentation for any purpose is hereby granted without fee,
provided that the above copyright notice appear in all copies and that
both that copyright notice and this permission notice appear in
supporting documentation. Davor Matic makes no representations about
the suitability of this software for any purpose. It is provided "as
is" without express or implied warranty.
*/
#include "scrnintstr.h"
#include "dix.h"
#include "dixstruct.h"
#include "mi.h"
#include "micmap.h"
#include "colormapst.h"
#include "resource.h"
#include "mipointer.h"
#include "../../fb/fb.h"
#include "../../randr/randrstr.h"
#include "inputstr.h"
#include "mivalidate.h"
#include "misc.h"
#include "Agent.h"
#include "Display.h"
#include "Screen.h"
#include "Extensions.h"
#include "Atoms.h"
#include "GCs.h"
#include "GCOps.h"
#include "Image.h"
#include "Drawable.h"
#include "Font.h"
#include "Colormap.h"
#include "Cursor.h"
#include "Visual.h"
#include "Events.h"
#include "Init.h"
#include "Args.h"
#include "Client.h"
#include "Options.h"
#include "Splash.h"
#include "Render.h"
#include "Trap.h"
#include "Keyboard.h"
#include "Pointer.h"
#include "Reconnect.h"
#include "Composite.h"
#include <nx/Shadow.h>
#include "Utils.h"
#include <nx-X11/Xlib.h>
#include "X11/include/Xinerama_nxagent.h"
#define GC XlibGC
#define Font XlibFont
#define KeySym XlibKeySym
#define XID XlibXID
#include <nx-X11/Xlibint.h>
#undef GC
#undef Font
#undef KeySym
#undef XID
#include "Xatom.h"
#include "Xproto.h"
#include "compext/Compext.h"
#include "mibstorest.h"
/*
* Set here the required log level.
*/
#define PANIC
#define WARNING
#undef TEST
#undef DEBUG
#undef WATCH
#undef DUMP
/*
* Display a pixmap on an shadow
* display used for debug.
*/
#ifdef DUMP
void nxagentShowPixmap(PixmapPtr pPixmap, int x, int y, int width, int height);
void nxagentFbRestoreArea(PixmapPtr pPixmap, WindowPtr pWin, int xSrc, int ySrc, int width,
int height, int xDst, int yDst)
#endif
#ifdef WATCH
#include "unistd.h"
#endif
extern Pixmap nxagentIconPixmap;
extern Pixmap nxagentIconShape;
extern Bool useXpmIcon;
extern Bool nxagentReportWindowIds;
#ifdef NXAGENT_TIMESTAMP
extern unsigned long startTime;
#endif
Window nxagentDefaultWindows[MAXSCREENS];
Window nxagentInputWindows[MAXSCREENS];
Window nxagentScreenSaverWindows[MAXSCREENS];
#ifdef NXAGENT_ONSTART
XlibAtom nxagentReadyAtom;
#endif
ScreenPtr nxagentDefaultScreen = NULL;
int nxagentArgc = 0;
char **nxagentArgv = NULL;
#ifdef NXAGENT_ARTSD
char mcop_atom[] = "MCOPGLOBALS";
Atom mcop_local_atom = None;
unsigned char fromHexNibble(char c);
#endif
Window nxagentIconWindow = None;
Window nxagentFullscreenWindow = None;
#ifdef VIEWPORT_FRAME
WindowPtr nxagentViewportFrameLeft;
WindowPtr nxagentViewportFrameRight;
WindowPtr nxagentViewportFrameAbove;
WindowPtr nxagentViewportFrameBelow;
#endif /* #ifdef VIEWPORT_FRAME */
Bool nxagentCreateScreenResources(ScreenPtr pScreen);
void nxagentPrintAgentGeometry(char *hdrMessage, char *prefix);
/*
* These variables are for shadowing feature.
*/
Bool nxagentShadowResize = False;
WindowPtr nxagentShadowWindowPtr = NULL;
static XID accessPixmapID;
static Window accessWindowID;
static int imageByteOrder;
static unsigned char nxagentMasterDepth;
static unsigned char nxagentCheckDepth = 0;
static unsigned int nxagentBppShadow;
static unsigned int nxagentBppMaster;
int nxagentShadowXConnectionNumber;
GCPtr nxagentShadowGCPtr = NULL;
PixmapPtr nxagentShadowPixmapPtr = NULL;
char * nxagentShadowBuffer;
unsigned char nxagentShadowDepth;
int nxagentShadowWidth;
int nxagentShadowHeight;
Display * nxagentShadowDisplay;
short nxagentShadowUid = -1;
void nxagentShadowAdaptDepth(unsigned int, unsigned int, unsigned int, char **);
RegionRec nxagentShadowUpdateRegion;
#define NXAGENT_DEFAULT_DPI 96
#define NXAGENT_AUTO_DPI -1
#ifndef NXAGENT_RANDR_MODE_PREFIX
#define NXAGENT_RANDR_MODE_PREFIX nx_
#endif
extern Bool nxagentAutoDPI;
extern char *nxagentKeyboard;
/*
* From randr/randr.c. This was originally static
* but we need it here.
*/
int TellChanged(WindowPtr pWin, void * value);
int nxagentBitsPerPixel(int depth)
{
if (depth == 1) return 1;
else if (depth <= 8) return 8;
else if (depth <= 16) return 16;
else return 32;
}
void nxagentSetScreenInfo(ScreenInfo *scrInfo)
{
/*
* Setup global screen info parameters. In the Xnest
* server this stuff is done after having opened the
* real display as Xnest lets the screen reflect the
* order of the remote end. Agent will instead set
* the order according to local endianness and swap
* data whenever it is appropriate.
*
* From a standard implementation:
*
* scrInfo->imageByteOrder = IMAGE_BYTE_ORDER;
* scrInfo->bitmapScanlinePad = BITMAP_SCANLINE_PAD;
* scrInfo->bitmapScanlineUnit = BITMAP_SCANLINE_UNIT;
* scrInfo->bitmapBitOrder = BITMAP_BIT_ORDER;
*
* From Xnest implementation:
*
* scrInfo -> imageByteOrder = ImageByteOrder(nxagentDisplay);
* scrInfo -> bitmapScanlineUnit = BitmapUnit(nxagentDisplay);
* scrInfo -> bitmapScanlinePad = BitmapPad(nxagentDisplay);
* scrInfo -> bitmapBitOrder = BitmapBitOrder(nxagentDisplay);
*/
scrInfo -> imageByteOrder = IMAGE_BYTE_ORDER;
scrInfo -> bitmapScanlinePad = BITMAP_SCANLINE_PAD;
scrInfo -> bitmapScanlineUnit = BITMAP_SCANLINE_UNIT;
scrInfo -> bitmapBitOrder = BITMAP_BIT_ORDER;
#ifdef TEST
fprintf(stderr, "%s: Server image order is [%d] bitmap order is [%d].\n", __func__,
scrInfo -> imageByteOrder, scrInfo -> bitmapBitOrder);
fprintf(stderr, "%s: Server scanline unit is [%d] scanline pad is [%d].\n", __func__,
scrInfo -> bitmapScanlineUnit, scrInfo -> bitmapScanlinePad);
#endif
}
void nxagentSetPixmapFormats(ScreenInfo *scrInfo)
{
/*
* Formats are created with no care of which are supported
* on the real display. Creating only formats supported
* by the remote end makes troublesome handling migration
* of session from a display to another.
*/
scrInfo -> numPixmapFormats = nxagentNumPixmapFormats;
for (int i = 0; i < nxagentNumPixmapFormats; i++)
{
scrInfo -> formats[i].depth = nxagentPixmapFormats[i].depth;
scrInfo -> formats[i].bitsPerPixel = nxagentPixmapFormats[i].bits_per_pixel;
scrInfo -> formats[i].scanlinePad = nxagentPixmapFormats[i].scanline_pad;
#ifdef TEST
fprintf(stderr, "%s: Set format at index [%d] to depth [%d] "
"bits per pixel [%d] scanline pad [%d].\n", __func__, i,
scrInfo -> formats[i].depth, scrInfo -> formats[i].bitsPerPixel,
scrInfo -> formats[i].scanlinePad);
#endif
}
}
/* check if possible_parent is parent of candidate */
Bool nxagentIsParentOf(Display *d, XlibWindow possible_parent, XlibWindow candidate)
{
XlibWindow parent, root, *children = NULL;
unsigned int num_children;
if (XQueryTree(d, candidate, &root, &parent, &children, &num_children))
{
SAFE_XFree(children);
#ifdef TEST
fprintf(stderr, "%s: parent of full screen window [%p] root [%p] possible_parent [%p] candidate [%p]\n", __func__, parent, root, possible_parent, candidate);
#endif
return (parent == possible_parent);
}
else
{
return False;
}
}
/*
* Pressing the minimize keystroke while in fullscreen mode will call
* this function. It will unmap the fullscreen window and iconify the
* previously created icon window immediately. The window manager may
* decide how to show an iconified window. kwin e.g. shows it in the
* task bar.
*/
void nxagentMinimizeFromFullScreen(ScreenPtr pScreen)
{
if (nxagentOption(Fullscreen) && !nxagentOption(AllScreens))
{
nxagentSwitchFullscreen(pScreen, False);
return;
}
XUnmapWindow(nxagentDisplay, nxagentFullscreenWindow);
XIconifyWindow(nxagentDisplay, nxagentIconWindow,
DefaultScreen(nxagentDisplay));
}
/*
* This is the opposite function to nxagentMinimizeFromFullscreen. It
* will map the fullscreen window and unmap the icon window. It is
* only called if fullscreen mode was active when the minimize
* keystroke was pressed.
* Some window managers tend to do 'interesting' things with the
* icon window, which we try to counterfeit here.
*/
void nxagentMaximizeToFullScreen(ScreenPtr pScreen)
{
if (nxagentOption(Fullscreen) && !nxagentOption(AllScreens))
{
nxagentSwitchFullscreen(pScreen, True);
return;
}
/*
XUnmapWindow(nxagentDisplay, nxagentIconWindow);
*/
Window root = RootWindow(nxagentDisplay, DefaultScreen(nxagentDisplay));
/*
FIXME: We'll check for ReparentNotify and LeaveNotify events after
XReparentWindow() in order to avoid the session window being
iconified. We could avoid the session window being iconified
when a LeaveNotify event is received, so this check would be
unnecessary.
*/
/* only reparent if necessary. FIXME: also check if the desired coordinates match */
if (!nxagentIsParentOf(nxagentDisplay, root, nxagentFullscreenWindow))
{
XReparentWindow(nxagentDisplay, nxagentFullscreenWindow,
root, 0, 0);
for (int i = 0; i < 100 && nxagentWMIsRunning; i++)
{
XEvent e;
#ifdef TEST
fprintf(stderr, "%s: WARNING! Going to wait for the ReparentNotify event [%d].\n", __func__, i);
#endif
if (XCheckTypedWindowEvent(nxagentDisplay, nxagentFullscreenWindow, ReparentNotify, &e))
{
break;
}
XSync(nxagentDisplay, 0);
nxagentWaitEvents(nxagentDisplay, 50);
}
}
else
{
#ifdef TEST
fprintf(stderr, "%s: FullscreenWindow already is child of root window - skipping reparenting,\n", __func__);
#endif
}
XMapRaised(nxagentDisplay, nxagentFullscreenWindow);
XIconifyWindow(nxagentDisplay, nxagentIconWindow,
DefaultScreen(nxagentDisplay));
/* swallow all LeaveNotify events for the FullscreenWindow;
Normally this does not swallow anything these days, but when
using fvwm you see one of these events here. */
while (1)
{
XEvent e;
if (!XCheckTypedWindowEvent(nxagentDisplay, nxagentFullscreenWindow, LeaveNotify, &e))
break;
#ifdef TEST
fprintf(stderr, "%s: swallowing LeaveNotify event\n", __func__);
#endif
}
/*
XMapWindow(nxagentDisplay, nxagentIconWindow);
*/
}
Window nxagentCreateIconWindow(void)
{
/*
* Create icon window.
*/
XSetWindowAttributes attributes = {
.override_redirect = False,
.colormap = DefaultColormap(nxagentDisplay, DefaultScreen(nxagentDisplay)),
.background_pixmap = nxagentScreenSaverPixmap,
};
unsigned long valuemask = CWOverrideRedirect | CWBackPixmap | CWColormap;
#ifdef TEST
fprintf(stderr, "%s: Going to create new icon window.\n", __func__);
#endif
Window w = XCreateWindow(nxagentDisplay, DefaultRootWindow(nxagentDisplay),
0, 0, 1, 1, 0,
DefaultDepth(nxagentDisplay, DefaultScreen(nxagentDisplay)),
InputOutput,
DefaultVisual(nxagentDisplay, DefaultScreen(nxagentDisplay)),
valuemask, &attributes);
if (nxagentReportWindowIds)
{
fprintf(stderr, "NXAGENT_WINDOW_ID: ICON_WINDOW,WID:[0x%x]\n", w);
}
#ifdef TEST
fprintf(stderr, "%s: Created new icon window with id [0x%x].\n", __func__,
w);
#endif
/*
* Set hints to the window manager for the icon window.
*/
XSizeHints* sizeHints = XAllocSizeHints();
if (sizeHints)
{
sizeHints->flags = PMinSize | PMaxSize;
sizeHints->min_width = sizeHints->max_width = 1;
sizeHints->min_height = sizeHints->max_height = 1;
}
XWMHints* wmHints = XAllocWMHints();;
if (wmHints)
{
wmHints->flags = StateHint | IconPixmapHint;
wmHints->initial_state = IconicState;
wmHints->icon_pixmap = nxagentIconPixmap;
if (useXpmIcon)
{
wmHints->flags |= IconMaskHint;
wmHints->icon_mask = nxagentIconShape;
}
}
char *winname = NULL;
#ifdef DEBUG
if (-1 == asprintf(&winname, "%s Icon", nxagentWindowName))
{
/* If memory allocation wasn't possible, or some other error
occurs, asprintf() will return -1, and the content of
winname is undefined. */
winname = NULL;
}
#endif
Xutf8SetWMProperties(nxagentDisplay, w,
winname ? winname : nxagentWindowName,
winname ? winname : nxagentWindowName,
NULL , 0 , sizeHints, wmHints, NULL);
SAFE_free(winname);
SAFE_XFree(sizeHints);
SAFE_XFree(wmHints);
/*
* Enable events from the icon window.
*/
Mask mask = nxagentGetDefaultEventMask();
XSelectInput(nxagentDisplay, w, (mask & ~(KeyPressMask |
KeyReleaseMask)) | StructureNotifyMask);
/*
* Notify to client if user closes icon window.
*/
if (nxagentWMIsRunning && !nxagentOption(Rootless))
{
XlibAtom deleteWMAtom = nxagentAtoms[2]; /* WM_DELETE_WINDOW */
XSetWMProtocols(nxagentDisplay, w, &deleteWMAtom, 1);
}
return w;
}
Bool nxagentMagicPixelZone(int x, int y)
{
return (x >= nxagentOption(Width) - 1 && y < 1);
}
void nxagentSetScreenSaverTime(void)
{
#ifdef TEST
fprintf(stderr, "%s: ScreenSaverTime was [%lu], ScreenSaverInterval was [%lu].\n", __func__,
(long unsigned int)ScreenSaverTime, (long unsigned int)ScreenSaverInterval);
#endif
ScreenSaverInterval = ScreenSaverTime;
#ifdef TEST
fprintf(stderr, "%s: ScreenSaverTime now is [%lu], ScreenSaverInterval now is [%lu].\n", __func__,
(long unsigned int)ScreenSaverTime, (long unsigned int)ScreenSaverInterval);
#endif
}
/*
* This is the called when the "hardware" should take care of the
* blanking.
*
* "what" can be one if these:
* SCREEN_SAVER_ON Turns on the screen saver; disables video
* SCREEN_SAVER_OFF Turns off the screen saver; enables video
* SCREEN_SAVER_FORCER Updates time of last screen saver mode change
* SCREEN_SAVER_CYCLE Cycle to new pattern
*
* Returns True if the 'what' action was successful and False otherwise.
*
* Returning False the SaveScreens() function (which calls this one)
* tries to build a screen-saver creating a new window. In some cases
* we do not want this so we return True. If we want the dix to take
* care fo blanking we return False.
*/
static Bool nxagentSaveScreen(ScreenPtr pScreen, int what)
{
#ifdef TEST
fprintf(stderr, "%s: Called for screen at [%p] with parameter [%s]. ScreenSaverTime [%d]\n", __func__,
(void *) pScreen,
what == SCREEN_SAVER_ON ? "SCREEN_SAVER_ON" :
what == SCREEN_SAVER_OFF ? "SCREEN_SAVER_OFF" :
what == SCREEN_SAVER_FORCER ? "SCREEN_SAVER_FORCER" :
what == SCREEN_SAVER_CYCLE ? "SCREEN_SAVER_CYCLE" :
"UNKNOWN",
ScreenSaverTime);
#endif
if (what == SCREEN_SAVER_OFF)
{
return False;
}
/*
* The lastDeviceEventTime is updated every time a device event is
* received, and it is used by WaitForSomething() to know when the
* SaveScreens() function should be called. This solution doesn't
* take care of a pointer button not released, so we have to handle
* this case by ourselves.
*/
/*
FIXME: Do we need to check the key grab if the
autorepeat feature is disabled?
*/
if (inputInfo.pointer -> button -> buttonsDown > 0)
{
#ifdef TEST
fprintf(stderr, "%s: Ignoring timeout, there is a pointer button down.\n", __func__);
#endif
/*
* Returning False the SaveScreens() function (which calls this one)
* tries to build a screen-saver creating a new window. We don't
* want this, so we return True here.
*/
return True;
}
return False;
}
Bool nxagentCreateScreenResources(ScreenPtr pScreen)
{
CreatePixmapProcPtr savedCreatePixmap = pScreen->CreatePixmap;
ModifyPixmapHeaderProcPtr savedModifyPixmapHeader = pScreen->ModifyPixmapHeader;
pScreen->CreatePixmap = fbCreatePixmap;
pScreen->ModifyPixmapHeader = miModifyPixmapHeader;
Bool ret = miCreateScreenResources(pScreen);
pScreen->CreatePixmap = savedCreatePixmap;
pScreen->ModifyPixmapHeader = savedModifyPixmapHeader;
return ret;
}
static Bool nxagentCursorOffScreen(ScreenPtr *pPtrScreen, int *x, int *y)
{
return False;
}
static void nxagentCrossScreen(ScreenPtr pScreen, Bool entering)
{
}
static miPointerScreenFuncRec nxagentPointerCursorFuncs =
{
nxagentCursorOffScreen,
nxagentCrossScreen,
miPointerWarpCursor
};
#ifdef VIEWPORT_FRAME
void nxagentInitViewportFrame(ScreenPtr pScreen, WindowPtr pRootWin)
{
int error = Success;
VisualID visual = 0;
XID xid;
if (nxagentOption(Rootless))
{
return;
}
for (int i = 0; i < pScreen -> numDepths; i++)
{
if (pScreen -> allowedDepths[i].depth == pRootWin -> drawable.depth)
{
visual = pScreen -> allowedDepths[i].vids[0];
break;
}
}
/*
* It is not necessary create the windows on the real X server. But this
* windows are not visible. Create them it is not a great effort, and avoids
* many errors.
*
* nxagentScreenTrap = True;
*/
xid = FakeClientID(serverClient -> index);
#ifdef TEST
fprintf(stderr, "%s: XID = [%lx]\n", __func__, xid);
#endif
nxagentViewportFrameLeft = CreateWindow(xid, pRootWin, -NXAGENT_FRAME_WIDTH, 0, NXAGENT_FRAME_WIDTH,
pRootWin -> drawable.height,
0, InputOutput, 0, NULL,
pRootWin -> drawable.depth,
serverClient, visual, &error);
AddResource(xid, RT_WINDOW, (void *) nxagentViewportFrameLeft);
if (error != Success)
{
#ifdef WARNING
fprintf(stderr, "%s: Error creating nxagentViewportFrameLeft.\n", __func__);
#endif
error = Success;
}
xid = FakeClientID(serverClient -> index);
#ifdef TEST
fprintf(stderr, "%s: XID = [%lx]\n", __func__, xid);
#endif
nxagentViewportFrameRight = CreateWindow(xid, pRootWin, pRootWin -> drawable.width, 0,
NXAGENT_FRAME_WIDTH,
pRootWin -> drawable.height,
0, InputOutput, 0, NULL,
pRootWin -> drawable.depth,
serverClient, visual,
&error);
AddResource(xid, RT_WINDOW, (void *) nxagentViewportFrameRight);
if (error != Success)
{
#ifdef WARNING
fprintf(stderr, "%s: Error creating nxagentViewportFrameRight.\n", __func__);
#endif
error = Success;
}
xid = FakeClientID(serverClient -> index);
#ifdef TEST
fprintf(stderr, "%s: XID = [%lx]\n", __func__, xid);
#endif
nxagentViewportFrameAbove = CreateWindow(xid, pRootWin, 0, -NXAGENT_FRAME_WIDTH,
pRootWin -> drawable.width,
NXAGENT_FRAME_WIDTH, 0,
InputOutput, 0, NULL,
pRootWin -> drawable.depth,
serverClient, visual,
&error);
AddResource(xid, RT_WINDOW, (void *) nxagentViewportFrameAbove);
if (error != Success)
{
#ifdef WARNING
fprintf(stderr, "%s: Error creating nxagentViewportFrameAbove.\n", __func__);
#endif
error = Success;
}
xid = FakeClientID(serverClient -> index);
#ifdef TEST
fprintf(stderr, "%s: XID = [%lx]\n", __func__, xid);
#endif
nxagentViewportFrameBelow = CreateWindow(xid, pRootWin, 0,
pRootWin -> drawable.height,
pRootWin -> drawable.width,
NXAGENT_FRAME_WIDTH, 0,
InputOutput, 0, NULL,
pRootWin -> drawable.depth,
serverClient, visual, &error);
AddResource(xid, RT_WINDOW, (void *) nxagentViewportFrameBelow);
if (error != Success)
{
#ifdef WARNING
fprintf(stderr, "%s: Error creating nxagentViewportFrameBelow.\n", __func__);
#endif
}
nxagentViewportFrameLeft -> overrideRedirect = 1;
nxagentViewportFrameRight -> overrideRedirect = 1;
nxagentViewportFrameAbove -> overrideRedirect = 1;
nxagentViewportFrameBelow -> overrideRedirect = 1;
MapWindow(nxagentViewportFrameLeft, serverClient);
MapWindow(nxagentViewportFrameRight, serverClient);
MapWindow(nxagentViewportFrameAbove, serverClient);
MapWindow(nxagentViewportFrameBelow, serverClient);
/*
* nxagentScreenTrap = False;
*/
}
#endif /* #ifdef VIEWPORT_FRAME */
void nxagentPrintAgentGeometry(char *hdrMessage, char *prefix)
{
#ifdef WARNING
if (prefix == NULL)
{
prefix = "";
}
if (hdrMessage)
{
fprintf(stderr, "%s /--------------- %s -----------------.\n", prefix, hdrMessage);
}
fprintf(stderr, "%s | Root window at offset (%d,%d) size (%d,%d).\n", prefix,
nxagentOption(RootX), nxagentOption(RootY),
nxagentOption(RootWidth), nxagentOption(RootHeight));
fprintf(stderr, "%s | Default window at offset (%d,%d) size (%d,%d) border size %d.\n", prefix,
nxagentOption(X), nxagentOption(Y), nxagentOption(Width), nxagentOption(Height),
nxagentOption(BorderWidth));
fprintf(stderr, "%s | Span between root window and default window is (%d,%d).\n", prefix,
nxagentOption(ViewportXSpan), nxagentOption(ViewportYSpan));
fprintf(stderr, "%s | Default window in window mode has offset (%d,%d) and size (%d,%d).\n", prefix,
nxagentOption(SavedX), nxagentOption(SavedY), nxagentOption(SavedWidth), nxagentOption(SavedHeight));
fprintf(stderr, "%s | Fullscreen is %s.\n", prefix,
nxagentOption(Fullscreen) ? "ON" : "OFF");
fprintf(stderr, "%s | AllScreens is %s.\n", prefix,
nxagentOption(AllScreens) ? "ON" : "OFF");
fprintf(stderr, "%s | Desktop resize mode is %s.\n", prefix,
nxagentOption(DesktopResize) ? "ON" : "OFF");
fprintf(stderr, "%s | Resize desktop at startup is %s.\n", prefix,
nxagentResizeDesktopAtStartup ? "ON" : "OFF");
if (hdrMessage)
{
fprintf(stderr, "%s \\--------------- %s -----------------.\n", prefix, hdrMessage);
}
#endif
}
static int nxagentColorOffset(unsigned long mask)
{
int count;
for (count = 0; !(mask & 1) && count < 32; count++)
{
mask >>= 1;
}
return count;
}
void freeDepths(DepthPtr depths, int num)
{
for (int i = 0; i < num; i++)
{
#ifdef DEBUG
fprintf(stderr, "%s: freeing depth [%d] index [%d] vids [%p]\n", __func__, depths[i].depth, i, (void*) depths[i].vids);
#endif
SAFE_free(depths[i].vids);
}
SAFE_free(depths);
}
Bool nxagentOpenScreen(ScreenPtr pScreen, int argc, char *argv[])
{
Bool resetAgentPosition = False;
#ifdef TEST
fprintf(stderr, "%s: Called for screen index [%d].\n", __func__,
pScreen->myNum);
#endif
if (nxagentRenderEnable && !nxagentReconnectTrap)
{
PictureScreenPrivateIndex = -1;
}
nxagentDefaultScreen = pScreen;
nxagentQueryAtoms(pScreen);
#ifdef NXAGENT_ONSTART
nxagentReadyAtom = nxagentAtoms[3]; /* WM_NX_READY */
#endif
/*
* Force geometry parameter to user geometry except if in rootless mode.
*/
if (nxagentResizeDesktopAtStartup && !nxagentOption(Rootless))
{
if (nxagentUserGeometry.flag & XValue)
{
nxagentChangeOption(X, nxagentUserGeometry.X);
}
if (nxagentUserGeometry.flag & YValue)
{
nxagentChangeOption(Y, nxagentUserGeometry.Y);
}
if (nxagentUserGeometry.flag & WidthValue)
{
int uw = nxagentUserGeometry.Width;
nxagentChangeOption(Width, uw);
nxagentChangeOption(RootWidth, uw);
nxagentChangeOption(SavedWidth, min(nxagentOption(SavedWidth), uw));
}
if (nxagentUserGeometry.flag & HeightValue)
{
int uh = nxagentUserGeometry.Height;
nxagentChangeOption(Height, uh);
nxagentChangeOption(RootHeight, uh);
nxagentChangeOption(SavedHeight, min(nxagentOption(SavedHeight), uh));
}
}
/*
* This is first time the screen is initialized.
* Filling the geometry parameter from user geometry.
* we do not need this in rootless mode since we ignore
* user geometry then.
*/
if (!nxagentReconnectTrap && !nxagentOption(Rootless))
{
if (nxagentUserGeometry.flag & XValue)
{
nxagentChangeOption(X, nxagentUserGeometry.X);
}
if (nxagentUserGeometry.flag & YValue)
{
nxagentChangeOption(Y, nxagentUserGeometry.Y);
}
if (nxagentUserGeometry.flag & WidthValue)
{
nxagentChangeOption(RootWidth, nxagentUserGeometry.Width);
}
if (nxagentUserGeometry.flag & HeightValue)
{
nxagentChangeOption(RootHeight, nxagentUserGeometry.Height);
}
}
else if (nxagentWMIsRunning && !nxagentOption(Rootless) && !nxagentOption(Fullscreen))
{
/*
* At reconnection, try to estimate the shift due to WM reparenting.
*/
if (nxagentOption(X) >= 6)
{
nxagentChangeOption(X, nxagentOption(X) - 6);
}
if (nxagentOption(Y) >= 25)
{
nxagentChangeOption(Y, nxagentOption(Y) - 25);
}
}
/*
* Determine the size of the root window. It is the maximum size of
* the screen if we are either in rootless or in fullscreen mode.
*/
if (!nxagentOption(Rootless) && !nxagentWMIsRunning)
{
#ifdef TEST
fprintf(stderr, "%s: Forcing fullscreen mode with no window manager running.\n", __func__);
#endif
nxagentChangeOption(Fullscreen, True);
nxagentChangeOption(AllScreens, True);
if (nxagentOption(ClientOs) == ClientOsWinnt &&
(!nxagentReconnectTrap || nxagentResizeDesktopAtStartup))
{
NXSetExposeParameters(nxagentDisplay, 0, 0, 0);
}
}
if (nxagentOption(Fullscreen) &&
nxagentWMIsRunning &&
nxagentReconnectTrap &&
!nxagentResizeDesktopAtStartup &&
nxagentXServerGeometryChanged())
{
#ifdef TEST
fprintf(stderr, "%s: Forcing window mode with server geometry changed.\n", __func__);
#endif
nxagentChangeOption(Fullscreen, False);
nxagentChangeOption(AllScreens, False);
nxagentFullscreenWindow = None;
resetAgentPosition = True;
}
nxagentChangeOption(BorderWidth, 0);
/* get the screen size of the real X server once */
int w = WidthOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
int h = HeightOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
if (nxagentOption(Fullscreen))
{
nxagentChangeOption(X, 0);
nxagentChangeOption(Y, 0);
/*
* Fullscreen will use the window manager on the remote X
* server. We need to size the window appropriately because the
* window manager stores the size and restores it when fullscreen
* mode is left. In AllScreens mode the window manager is
* instructed to not draw any window decorations. Here we need the
* full width to cover the whole screen.
*/
if (nxagentOption(AllScreens))
{
nxagentChangeOption(Width, w);
nxagentChangeOption(Height, h);
}
else
{
nxagentChangeOption(Width, w * 3 / 4);
nxagentChangeOption(Height, h * 3 / 4);
}
/* first time screen initialization or resize during reconnect */
if (!nxagentReconnectTrap || nxagentResizeDesktopAtStartup)
{
if (nxagentOption(RootWidth) >= w)
{
nxagentChangeOption(SavedWidth, w * 3 / 4);
}
else
{
nxagentChangeOption(SavedWidth, nxagentOption(RootWidth));
}
if (nxagentOption(RootHeight) >= h)
{
nxagentChangeOption(SavedHeight, h * 3 / 4);
}
else
{
nxagentChangeOption(SavedHeight, nxagentOption(RootHeight));
}
nxagentChangeOption(RootWidth, w);
nxagentChangeOption(RootHeight, h);
nxagentChangeOption(RootX, 0);
nxagentChangeOption(RootY, 0);
}
else
{
/* center */
nxagentChangeOption(RootX, (w - nxagentOption(RootWidth)) / 2);
nxagentChangeOption(RootY, (h - nxagentOption(RootHeight)) / 2);
}
}
else if (nxagentOption(Rootless))
{
nxagentChangeOption(X, 0);
nxagentChangeOption(Y, 0);
nxagentChangeOption(RootX, 0);
nxagentChangeOption(RootY, 0);
nxagentChangeOption(RootWidth, w);
nxagentChangeOption(RootHeight, h);
nxagentChangeOption(Width, w);
nxagentChangeOption(Height, h);
}
else /* window mode */
{
/*
* screen is initialized for the first time.
*/
if (!nxagentReconnectTrap)
{
nxagentChangeOption(RootX, 0);
nxagentChangeOption(RootY, 0);
nxagentChangeOption(Width, nxagentOption(RootWidth));
nxagentChangeOption(Height, nxagentOption(RootHeight));
}
/*
* Ensure that the agent window won't be bigger
* than the root window.
*/
nxagentChangeOption(Width, min(nxagentOption(Width), nxagentOption(RootWidth)));
nxagentChangeOption(Height, min(nxagentOption(Height), nxagentOption(RootHeight)));
/*
* Be sure that the agent window won't be bigger
* than the X server root window.
*/
if (nxagentOption(Width) > w)
{
nxagentChangeOption(Width, w * 3 / 4);
}
if (nxagentOption(Height) > h)
{
nxagentChangeOption(Height, h * 3 / 4);
}
/*
* Forcing the agent window geometry to be equal to the root
* window geometry the first time the screen is initialized if the
* geometry hasn't been explicitly set in the option file and if
* the root window isn't bigger than the X server root window..
*/
if (!nxagentReconnectTrap)
{
if ((nxagentOption(RootWidth) < w) &&
!(nxagentUserGeometry.flag & WidthValue))
{
nxagentChangeOption(Width, nxagentOption(RootWidth));
}
if ((nxagentOption(RootHeight) < h) &&
!(nxagentUserGeometry.flag & HeightValue))
{
nxagentChangeOption(Height, nxagentOption(RootHeight));
}
}
if (resetAgentPosition)
{
/* center */
nxagentChangeOption(X, (w - nxagentOption(Width)) / 2);
nxagentChangeOption(Y, (h - nxagentOption(Height)) / 2);
}
nxagentChangeOption(SavedWidth, nxagentOption(RootWidth));
nxagentChangeOption(SavedHeight, nxagentOption(RootHeight));
}
nxagentChangeOption(SavedRootWidth, nxagentOption(RootWidth));
nxagentChangeOption(SavedRootHeight, nxagentOption(RootHeight));
nxagentChangeOption(ViewportXSpan, nxagentOption(Width) - nxagentOption(RootWidth));
nxagentChangeOption(ViewportYSpan, nxagentOption(Height) - nxagentOption(RootHeight));
if (!nxagentReconnectTrap)
{
if (nxagentOption(Persistent))
{
nxagentArgc = argc;
nxagentArgv = argv;
}
#ifdef NXAGENT_TIMESTAMP
fprintf(stderr, "Screen: going to open screen, time is [%d] milliseconds.\n",
GetTimeInMillis() - startTime);
#endif
/*
* Initialize all our privates.
*/
if (AllocateWindowPrivate(pScreen, nxagentWindowPrivateIndex, sizeof(nxagentPrivWindowRec)) == 0 ||
AllocateGCPrivate(pScreen, nxagentGCPrivateIndex, sizeof(nxagentPrivGC)) == 0 ||
AllocateClientPrivate(nxagentClientPrivateIndex, sizeof(PrivClientRec)) == 0 ||
AllocatePixmapPrivate(pScreen, nxagentPixmapPrivateIndex, sizeof(nxagentPrivPixmapRec)) == 0)
{
return False;
}
/*
* Initialize the depths.
*/
DepthPtr depths = (DepthPtr) malloc(nxagentNumDepths * sizeof(DepthRec));
if (!depths)
{
#ifdef WARNING
fprintf(stderr, "WARNING: Could not allocate depths array\n");
#endif
/* FIXME: free data allocated above */
return False;
}
for (int i = 0; i < nxagentNumDepths; i++)
{
depths[i].depth = nxagentDepths[i];
depths[i].numVids = 0;
depths[i].vids = (VisualID *) malloc(MAXVISUALSPERDEPTH * sizeof(VisualID));
}
/*
* Initialize the visuals.
*/
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "Debug: Setting up visuals. Original array has size "
"[%d].\n", nxagentNumVisuals);
#endif
int numVisuals = 0;
int numDepths = nxagentNumDepths;
VisualPtr visuals = (VisualPtr) malloc(nxagentNumVisuals * sizeof(VisualRec));
if (!visuals)
{
#ifdef WARNING
fprintf(stderr, "WARNING: Could not allocate visual array\n");
#endif
/* FIXME: free data allocated above */
return False;
}
int defaultVisualIndex = 0;
for (int i = 0; i < nxagentNumVisuals; i++)
{
visuals[numVisuals].vid = FakeClientID(0);
visuals[numVisuals].class = nxagentVisuals[i].class;
visuals[numVisuals].bitsPerRGBValue = nxagentVisuals[i].bits_per_rgb;
visuals[numVisuals].ColormapEntries = nxagentVisuals[i].colormap_size;
visuals[numVisuals].nplanes = nxagentVisuals[i].depth;
visuals[numVisuals].redMask = nxagentVisuals[i].red_mask;
visuals[numVisuals].greenMask = nxagentVisuals[i].green_mask;
visuals[numVisuals].blueMask = nxagentVisuals[i].blue_mask;
visuals[numVisuals].offsetRed = nxagentColorOffset(nxagentVisuals[i].red_mask);
visuals[numVisuals].offsetGreen = nxagentColorOffset(nxagentVisuals[i].green_mask);
visuals[numVisuals].offsetBlue = nxagentColorOffset(nxagentVisuals[i].blue_mask);
/*
* Check for and remove the duplicates.
*/
if (i == nxagentDefaultVisualIndex)
{
defaultVisualIndex = numVisuals;
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "Debug: Set default visual index [%d].\n" ,
defaultVisualIndex);
#endif
}
else
{
int j;
for (j = 0; j < numVisuals; j++)
{
if (visuals[numVisuals].class == visuals[j].class &&
visuals[numVisuals].bitsPerRGBValue == visuals[j].bitsPerRGBValue &&
visuals[numVisuals].ColormapEntries == visuals[j].ColormapEntries &&
visuals[numVisuals].nplanes == visuals[j].nplanes &&
visuals[numVisuals].redMask == visuals[j].redMask &&
visuals[numVisuals].greenMask == visuals[j].greenMask &&
visuals[numVisuals].blueMask == visuals[j].blueMask &&
visuals[numVisuals].offsetRed == visuals[j].offsetRed &&
visuals[numVisuals].offsetGreen == visuals[j].offsetGreen &&
visuals[numVisuals].offsetBlue == visuals[j].offsetBlue)
break;
}
if (j < numVisuals)
continue;
}
int depthIndex = UNDEFINED;
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "Debug: Added visual [%lu].\n" ,
(long unsigned int)visuals[numVisuals].vid);
#endif
for (int j = 0; j < numDepths; j++)
{
if (depths[j].depth == nxagentVisuals[i].depth)
{
depthIndex = j;
break;
}
}
if (depthIndex == UNDEFINED)
{
#ifdef WARNING
fprintf(stderr, "%s: WARNING! Can't find a matching depth for visual depth [%d].\n", __func__,
nxagentVisuals[i].depth);
#endif
depthIndex = numDepths;
depths[depthIndex].depth = nxagentVisuals[i].depth;
depths[depthIndex].numVids = 0;
depths[depthIndex].vids = (VisualID *) malloc(MAXVISUALSPERDEPTH * sizeof(VisualID));
numDepths++;
}
if (depths[depthIndex].numVids >= MAXVISUALSPERDEPTH)
{
FatalError("Visual table overflow");
}
depths[depthIndex].vids[depths[depthIndex].numVids] = visuals[numVisuals].vid;
depths[depthIndex].numVids++;
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "Debug: Registered visual [%lu] for depth [%d (%d)].\n" ,
(long unsigned int)visuals[numVisuals].vid, depthIndex,
depths[depthIndex].depth);
#endif
numVisuals++;
}
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "Debug: Setting default visual [%d (%lu)].\n",
defaultVisualIndex, (long unsigned int)visuals[defaultVisualIndex].vid);
fprintf(stderr, "Debug: Setting root depth [%d].\n",
visuals[defaultVisualIndex].nplanes);
#endif
VisualID defaultVisual = visuals[defaultVisualIndex].vid;
int rootDepth = visuals[defaultVisualIndex].nplanes;
nxagentInitAlphaVisual();
int bitsPerPixel = nxagentBitsPerPixel(rootDepth);
int sizeInBytes;
if (bitsPerPixel == 1)
{
sizeInBytes = PixmapBytePad(nxagentOption(RootWidth), rootDepth) * nxagentOption(RootHeight);
}
else
{
sizeInBytes = PixmapBytePad(nxagentOption(RootWidth), rootDepth) * nxagentOption(RootHeight) * bitsPerPixel/8;
}
#ifdef TEST
fprintf(stderr, "%s: Frame buffer allocated. rootDepth "
"[%d] bitsPerPixel [%d] sizeInBytes [%d]\n", __func__, rootDepth, bitsPerPixel, sizeInBytes);
#endif
void * pFrameBufferBits = (char *) malloc(sizeInBytes);
if (!pFrameBufferBits)
{
freeDepths(depths, numDepths);
SAFE_free(visuals);
return FALSE;
}
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "%s: Before fbScreenInit numVisuals [%d] numDepths [%d] "
"rootDepth [%d] defaultVisual [%lu].\n", __func__, numVisuals, numDepths,
rootDepth, (long unsigned int)defaultVisual);
#endif
if (monitorResolution < 1)
{
if (nxagentAutoDPI)
{
monitorResolution = NXAGENT_AUTO_DPI;
}
else
{
monitorResolution = NXAGENT_DEFAULT_DPI;
}
}
if (!fbScreenInit(pScreen, pFrameBufferBits, nxagentOption(RootWidth), nxagentOption(RootHeight),
monitorResolution, monitorResolution, PixmapBytePad(nxagentOption(RootWidth), rootDepth), bitsPerPixel))
{
freeDepths(depths, numDepths);
SAFE_free(visuals);
return FALSE;
}
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "%s: After fbScreenInit numVisuals [%d] numDepths [%d] "
"rootDepth [%d] defaultVisual [%lu].\n", __func__, numVisuals, numDepths,
rootDepth, (long unsigned int)defaultVisual);
#endif
/*
* Complete the initialization of the GLX extension. This will add
* the GLX visuals and will modify numVisuals and numDepths.
*/
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "%s: Before GLX numVisuals [%d] numDepths [%d] "
"rootDepth [%d] defaultVisual [%lu].\n", __func__, numVisuals, numDepths,
rootDepth, (long unsigned int)defaultVisual);
#endif
nxagentInitGlxExtension(&visuals, &depths, &numVisuals, &numDepths,
&rootDepth, &defaultVisual);
#if defined(DEBUG) || defined(DEBUG_COLORMAP)
fprintf(stderr, "%s: After GLX numVisuals [%d] numDepths [%d] "
"rootDepth [%d] defaultVisual [%lu].\n", __func__, numVisuals, numDepths,
rootDepth, (long unsigned int)defaultVisual);
#endif
/*
* Replace the visuals and depths initialized by fbScreenInit with
* our own.
*/
freeDepths(pScreen->allowedDepths, pScreen->numDepths);
pScreen -> allowedDepths = depths;
pScreen -> numDepths = numDepths;
pScreen -> rootDepth = rootDepth;
SAFE_free(pScreen -> visuals);
pScreen -> visuals = visuals;
pScreen -> numVisuals = numVisuals;
pScreen -> rootVisual = defaultVisual;
/*
* Set up the internal structures used for tracking the proxy
* resources associated to the unpack and split operations.
*/
nxagentInitSplitResources();
nxagentInitUnpackResources();
#ifdef WATCH
fprintf(stderr, "%s: Watchpoint 7.\n", __func__);
/*
Reply Total Cached Bits In Bits Out Bits/Reply Ratio
------- ----- ------ ------- -------- ---------- -----
N/A
*/
sleep(30);
#endif
if (nxagentParentWindow != 0)
{
/*
* This would cause a GetWindowAttributes
* and a GetGeometry (asynchronous) reply.
*/
XWindowAttributes gattributes;
XGetWindowAttributes(nxagentDisplay, nxagentParentWindow, &gattributes);
nxagentChangeOption(Width, gattributes.width);
nxagentChangeOption(Height, gattributes.height);
}
if (nxagentOption(Fullscreen))
{
/*
* We need to disable the host's screensaver or it will
* otherwise grab the screen even if it is under agent's
* control.
*/
XSetScreenSaver(nxagentDisplay, 0, 0, DefaultExposures, DefaultBlanking);
}
if (nxagentTrue24)
{
fbGetScreenPrivate(pScreen) -> win32bpp = visuals[nxagentDefaultVisualIndex].nplanes;
fbGetScreenPrivate(pScreen) -> pix32bpp = visuals[nxagentDefaultVisualIndex].nplanes;
}
else
{
fbGetScreenPrivate(pScreen) -> win32bpp = 32;
fbGetScreenPrivate(pScreen) -> pix32bpp = 32;
}
/*
* We call miScreenInit with NULL in place of the screen area if we
* don't want to initialize the frame buffer.
*
* if (!miScreenInit(pScreen, NULL, nxagentOption(RootWidth),
* nxagentOption(RootHeight), 1, 1, nxagentOption(RootWidth),
* visuals[nxagentDefaultVisualIndex].nplanes, / * Root depth. * /
* numDepths, depths,
* visuals[nxagentDefaultVisualIndex].vid,* Root visual. * /
* numVisuals, visuals))
* return FALSE;
*/
if (monitorResolution < 0)
{
pScreen->mmWidth = nxagentOption(RootWidth) * DisplayWidthMM(nxagentDisplay,
DefaultScreen(nxagentDisplay)) / DisplayWidth(nxagentDisplay,
DefaultScreen(nxagentDisplay));
pScreen->mmHeight = nxagentOption(RootHeight) * DisplayHeightMM(nxagentDisplay,
DefaultScreen(nxagentDisplay)) / DisplayHeight(nxagentDisplay,
DefaultScreen(nxagentDisplay));
}
pScreen->defColormap = (Colormap) FakeClientID(0);
pScreen->minInstalledCmaps = MINCMAPS;
pScreen->maxInstalledCmaps = MAXCMAPS;
pScreen->whitePixel = nxagentWhitePixel;
pScreen->blackPixel = nxagentBlackPixel;
/* rgf */
/* GCperDepth */
/* PixmapPerDepth */
/* WindowPrivateLen */
/* WindowPrivateSizes */
/* totalWindowSize */
/* GCPrivateLen */
/* GCPrivateSizes */
/* totalGCSize */
/*
* Random screen procedures.
*/
pScreen->CloseScreen = nxagentCloseScreen;
pScreen->QueryBestSize = nxagentQueryBestSize;
pScreen->SaveScreen = nxagentSaveScreen;
pScreen->GetImage = nxagentGetImage;
pScreen->GetSpans = nxagentGetSpans;
pScreen->PointerNonInterestBox = (void (*)()) 0;
pScreen->SourceValidate = (void (*)()) 0;
pScreen->CreateScreenResources = nxagentCreateScreenResources;
/*
* Window Procedures.
*
* Note that the following functions are not
* replaced with nxagent counterparts:
*
* ValidateTreeProcPtr ValidateTree;
* ClearToBackgroundProcPtr ClearToBackground;
*
* Note also that the ConfigureWindow procedure
* has not a pointer in the screen structure.
*/
pScreen->CreateWindow = nxagentCreateWindow;
pScreen->DestroyWindow = nxagentDestroyWindow;
pScreen->PositionWindow = nxagentPositionWindow;
pScreen->ChangeWindowAttributes = nxagentChangeWindowAttributes;
pScreen->RealizeWindow = nxagentRealizeWindow;
pScreen->UnrealizeWindow = nxagentUnrealizeWindow;
pScreen->PostValidateTree = nxagentPostValidateTree;
pScreen->WindowExposures = nxagentWindowExposures;
pScreen->PaintWindowBackground = nxagentPaintWindowBackground;
pScreen->PaintWindowBorder = nxagentPaintWindowBorder;
pScreen->CopyWindow = nxagentCopyWindow;
pScreen->ClipNotify = nxagentClipNotify;
pScreen->RestackWindow = nxagentRestackWindow;
pScreen->ReparentWindow = nxagentReparentWindow;
/*
* Pixmap procedures.
*/
pScreen->CreatePixmap = nxagentCreatePixmap;
pScreen->DestroyPixmap = nxagentDestroyPixmap;
/*
* This is originally miModifyPixmapHeader() from miscrinit.c. It
* is used to recycle the scratch pixmap for this screen.
*/
pScreen->ModifyPixmapHeader = nxagentModifyPixmapHeader;
/*
* Font procedures.
*/
pScreen->RealizeFont = nxagentRealizeFont;
pScreen->UnrealizeFont = nxagentUnrealizeFont;
/*
* GC procedures.
*/
pScreen->CreateGC = nxagentCreateGC;
pScreen->BitmapToRegion = nxagentPixmapToRegion;
/*
* Colormap procedures.
*/
pScreen->CreateColormap = nxagentCreateColormap;
pScreen->DestroyColormap = nxagentDestroyColormap;
pScreen->InstallColormap = nxagentInstallColormap;
pScreen->UninstallColormap = nxagentUninstallColormap;
pScreen->ListInstalledColormaps = nxagentListInstalledColormaps;
pScreen->StoreColors = nxagentStoreColors;
pScreen->ResolveColor = nxagentResolveColor;
/*
* Backing store procedures.
*/
pScreen->SaveDoomedAreas = (void (*)()) 0;
pScreen->RestoreAreas = (RegionPtr (*)()) 0;
pScreen->ExposeCopy = (void (*)()) 0;
pScreen->TranslateBackingStore = (RegionPtr (*)()) 0;
pScreen->ClearBackingStore = (RegionPtr (*)()) 0;
pScreen->DrawGuarantee = (void (*)()) 0;
if (enableBackingStore)
{
#ifdef TEST
fprintf(stderr, "%s: Going to initialize backing store.\n", __func__);
#endif
pScreen -> BackingStoreFuncs.SaveAreas = nxagentSaveAreas;
pScreen -> BackingStoreFuncs.RestoreAreas = nxagentRestoreAreas;
pScreen -> BackingStoreFuncs.SetClipmaskRgn = 0;
pScreen -> BackingStoreFuncs.GetImagePixmap = 0;
pScreen -> BackingStoreFuncs.GetSpansPixmap = 0;
miInitializeBackingStore(pScreen);
}
/*
* OS layer procedures.
*/
pScreen->BlockHandler = (ScreenBlockHandlerProcPtr) NoopDDA;
pScreen->WakeupHandler = (ScreenWakeupHandlerProcPtr) NoopDDA;
pScreen->blockData = NULL;
pScreen->wakeupData = NULL;
#ifdef RENDER
/*
* Initialize picture support. This have to be placed here because
* miDCInitialize calls DamageSetup, that should wrap the picture
* screen functions. So PictureInit has to be called before.
*/
if (nxagentRenderEnable && !nxagentReconnectTrap)
{
if (!nxagentPictureInit(pScreen, 0, 0))
{
nxagentRenderEnable = False;
return FALSE;
}
if (nxagentAlphaEnabled)
{
fprintf(stderr, "Info: Using alpha channel in render extension.\n");
}
}
#endif /* RENDER */
/*
* From misprite.c: called from device-dependent screen
* initialization proc after all of the function pointers
* have been stored in the screen structure.
*/
miDCInitialize(pScreen, &nxagentPointerCursorFuncs);
/*
* Cursor Procedures.
*/
pScreen->ConstrainCursor = nxagentConstrainCursor;
pScreen->CursorLimits = nxagentCursorLimits;
pScreen->DisplayCursor = nxagentDisplayCursor;
pScreen->RealizeCursor = nxagentRealizeCursor;
pScreen->UnrealizeCursor = nxagentUnrealizeCursor;
pScreen->RecolorCursor = nxagentRecolorCursor;
nxagentSetCursorPositionW = pScreen->SetCursorPosition;
pScreen->SetCursorPosition = nxagentSetCursorPosition;
#define POSITION_OFFSET (pScreen->myNum * (nxagentOption(Width) + \
nxagentOption(Height)) / 32)
/*
* Complete the initialization of the RANDR extension.
*/
nxagentInitRandRExtension(pScreen);
}
#ifdef TEST
nxagentPrintAgentGeometry(NULL, "nxagentOpenScreen:");
#endif
if (nxagentDoFullGeneration ||
nxagentReconnectTrap)
{
unsigned long valuemask = CWBackPixel | CWEventMask | CWColormap;
XSetWindowAttributes attributes = {
.background_pixel = nxagentBlackPixel,
.event_mask = nxagentGetDefaultEventMask(),
.colormap = nxagentDefaultVisualColormap(nxagentDefaultVisual(pScreen))
};
if (nxagentOption(AllScreens))
{
valuemask |= CWOverrideRedirect;
attributes.override_redirect = True;
}
if (nxagentOption(Fullscreen))
{
if (nxagentReconnectTrap)
{
/*
* We need to disable the host's screensaver or
* it will otherwise grab the screen even if it
* is under agent's control.
*/
XSetScreenSaver(nxagentDisplay, 0, 0, DefaultExposures, DefaultBlanking);
}
}
/*
* This would be used when running agent
* embedded into another X window.
*/
if (nxagentParentWindow != 0)
{
nxagentDefaultWindows[pScreen->myNum] = nxagentParentWindow;
XSelectInput(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum], nxagentGetDefaultEventMask());
}
else
{
/*
* Create any top-level window as a child of the real root of
* the remote display. See also the InitRootWindow() procedure
* and the function handling the splash screen.
*/
#ifdef TEST
fprintf(stderr, "%s: Going to create new default window.\n", __func__);
#endif
nxagentDefaultWindows[pScreen->myNum] =
XCreateWindow(nxagentDisplay,
DefaultRootWindow(nxagentDisplay),
nxagentOption(X) + POSITION_OFFSET,
nxagentOption(Y) + POSITION_OFFSET,
nxagentOption(Width),
nxagentOption(Height),
nxagentOption(BorderWidth),
pScreen->rootDepth,
InputOutput,
nxagentDefaultVisual(pScreen),
valuemask, &attributes);
if (nxagentReportWindowIds)
{
fprintf(stderr, "NXAGENT_WINDOW_ID: SCREEN_WINDOW:[%d],WID:[0x%x]\n", pScreen->myNum, nxagentDefaultWindows[pScreen->myNum]);
}
#ifdef TEST
fprintf(stderr, "%s: Created new default window for screen [%d] with id [0x%x].\n", __func__,
pScreen->myNum, nxagentDefaultWindows[pScreen->myNum]);
#endif
if (!nxagentOption(Rootless))
{
XSetWindowAttributes inpattributes = {.event_mask = PointerMotionMask};
nxagentInputWindows[pScreen->myNum] =
XCreateWindow(nxagentDisplay,
nxagentDefaultWindows[pScreen->myNum],
0, 0,
nxagentOption(Width),
nxagentOption(Height),
0, 0, InputOnly,
nxagentDefaultVisual(pScreen),
CWEventMask, &inpattributes);
if (nxagentReportWindowIds)
{
fprintf(stderr, "NXAGENT_WINDOW_ID: INPUT_WINDOW:[%d],WID:[0x%x]\n", pScreen->myNum, nxagentInputWindows[pScreen->myNum]);
}
#ifdef DEBUG
{
char *winname = NULL;
if (-1 != asprintf(&winname, "%s Input", nxagentWindowName))
{
Xutf8SetWMProperties(nxagentDisplay, nxagentInputWindows[pScreen->myNum],
winname, winname, NULL , 0 , NULL, NULL, NULL);
SAFE_free(winname);
}
}
#endif
#ifdef TEST
fprintf(stderr, "%s: Created new input window for screen [%d] with id [0x%x].\n", __func__,
pScreen->myNum, nxagentInputWindows[pScreen->myNum]);
#endif
}
#ifdef X2GO
/*
* Setting WM_CLASS to "X2GoAgent" when running in X2Go Agent mode
* we need it to properly display all window parameters by some WMs
* (for example on Maemo)
*/
#endif
{
#ifdef TEST
fprintf(stderr, "%s: Setting WM_CLASS and WM_NAME for window with id [%ld].\n", __func__,
(long int)nxagentDefaultWindows[pScreen->myNum]);
#endif
XClassHint hint;
#ifdef X2GO
if (nxagentX2go)
{
hint.res_name = strdup("X2GoAgent");
hint.res_class = strdup("X2GoAgent");
}
else
#endif
{
hint.res_name = strdup("NXAgent");
hint.res_class = strdup("NXAgent");
}
XSetClassHint(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum], &hint);
SAFE_free(hint.res_name);
SAFE_free(hint.res_class);
}
if (nxagentOption(Fullscreen))
{
nxagentFullscreenWindow = nxagentDefaultWindows[pScreen->myNum];
}
}
if (nxagentOption(Fullscreen))
{
#ifdef NXAGENT_ONSTART
/*
* FIXME: Do we still need to set this property?
*/
if (nxagentAtoms[8] != 0)
{
XChangeProperty(nxagentDisplay,
nxagentDefaultWindows[pScreen->myNum],
nxagentAtoms[8], /* NX_AGENT_SIGNATURE */
XA_STRING,
8,
PropModeReplace,
(unsigned char*) "X-AGENT",
strlen("X-AGENT"));
}
#endif
XSelectInput(nxagentDisplay, nxagentFullscreenWindow, nxagentGetDefaultEventMask());
}
XSizeHints* sizeHints = XAllocSizeHints();
if (sizeHints)
{
sizeHints->flags = PPosition | PMinSize | PMaxSize;
sizeHints->x = nxagentOption(X) + POSITION_OFFSET;
sizeHints->y = nxagentOption(Y) + POSITION_OFFSET;
sizeHints->min_width = MIN_NXAGENT_WIDTH;
sizeHints->min_height = MIN_NXAGENT_HEIGHT;
sizeHints->width = nxagentOption(RootWidth);
sizeHints->height = nxagentOption(RootHeight);
if (nxagentOption(DesktopResize) || nxagentOption(Fullscreen))
{
sizeHints->max_width = WidthOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
sizeHints->max_height = HeightOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
}
else
{
sizeHints->max_width = nxagentOption(RootWidth);
sizeHints->max_height = nxagentOption(RootHeight);
}
if (nxagentUserGeometry.flag & XValue || nxagentUserGeometry.flag & YValue)
sizeHints->flags |= USPosition;
if (nxagentUserGeometry.flag & WidthValue || nxagentUserGeometry.flag & HeightValue)
sizeHints->flags |= USSize;
}
XWMHints* wmHints = XAllocWMHints();
if (wmHints)
{
wmHints->flags = IconPixmapHint;
wmHints->icon_pixmap = nxagentIconPixmap;
if (useXpmIcon)
{
wmHints->flags |= IconMaskHint;
wmHints->icon_mask = nxagentIconShape;
}
}
char *winname = NULL;
#ifdef DEBUG
if (-1 == asprintf(&winname, "%s Default[%d]", nxagentWindowName, pScreen->myNum))
{
/* If memory allocation wasn't possible, or some other error
occurs, these functions will return -1, and the contents of
winname are undefined. */
winname = NULL;
}
#endif
Xutf8SetWMProperties(nxagentDisplay,
nxagentDefaultWindows[pScreen->myNum],
winname ? winname : nxagentWindowName,
winname ? winname : nxagentWindowName,
argv , argc , sizeHints, wmHints, NULL);
SAFE_free(winname);
SAFE_XFree(sizeHints);
SAFE_XFree(wmHints);
/*
* Clear the window but let it unmapped. We'll map it
* at the time the we'll initialize our screen root
* and only if we are not running in rootless mode.
*/
XClearWindow(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum]);
if (nxagentOption(AllScreens))
{
if (nxagentReconnectTrap)
{
XGrabKeyboard(nxagentDisplay, nxagentFullscreenWindow, True, GrabModeAsync,
GrabModeAsync, CurrentTime);
}
nxagentIconWindow = nxagentCreateIconWindow();
}
else
{
nxagentIconWindow = None;
}
/*
* When we don't have window manager we grab keyboard
* to let nxagent get keyboard events.
*/
if (!nxagentWMIsRunning && !nxagentOption(Fullscreen))
{
#ifdef TEST
fprintf(stderr, "%s: No window manager, we call XGrabKeyboard.\n", __func__);
#endif
XGrabKeyboard(nxagentDisplay, RootWindow (nxagentDisplay, 0), True, GrabModeAsync,
GrabModeAsync, CurrentTime);
}
}
if (!nxagentCreateDefaultColormap(pScreen))
{
#ifdef PANIC
fprintf(stderr, "%s: Failed to create default colormap for screen.\n", __func__);
#endif
return False;
}
/*
* The purpose of this check is to verify if there is a window
* manager running. Unfortunately due to the way we manage the
* intern atoms call, the atom will always exist.
*/
if (nxagentWMIsRunning)
{
XlibAtom deleteWMatom = nxagentAtoms[2]; /* WM_DELETE_WINDOW */
#ifdef TEST
fprintf(stderr, "%s: Found WM, delete window atom [%ld].\n", __func__,
deleteWMatom);
#endif
/* FIXME: This doing the same thing in both cases. The
comments do not seem accurate (anymore?) */
if (!nxagentOption(Rootless))
{
/*
* Set the WM_DELETE_WINDOW protocol for the main agent
* window and, if we are in fullscreen mode, include the
* icon window.
*/
XSetWMProtocols(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum], &deleteWMatom, 1);
}
else
{
/*
* We need to register the ICCCM WM_DELETE_WINDOW
* protocol for any top-level window or the agent
* will be killed if any window is closed.
*/
XSetWMProtocols(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum], &deleteWMatom, 1);
#ifdef TEST
fprintf(stderr, "Warning: Not setting the WM_DELETE_WINDOW protocol.\n");
#endif
}
}
else
{
/*
* We should always enable the configuration of the remote X
* server's devices if we are running fullscreen and there is no
* WM running.
*/
if (nxagentOption(Fullscreen))
{
#ifdef TEST
fprintf(stderr, "%s: WARNING! Forcing propagation of device control changes.\n", __func__);
#endif
nxagentChangeOption(DeviceControl, True);
}
}
/*
* Inform the user whether the agent's clients will be able to
* change the real X server's keyboard and pointer settings.
*/
if (!nxagentOption(DeviceControl))
{
fprintf(stderr, "Info: Not using local device configuration changes.\n");
}
else
{
fprintf(stderr, "Info: Using local device configuration changes.\n");
}
#ifdef RENDER
/*
* if (nxagentRenderEnable && !nxagentReconnectTrap)
* {
* if (!nxagentPictureInit(pScreen, 0, 0))
* {
* nxagentRenderEnable = False;
*
* return FALSE;
* }
*
* if (nxagentAlphaEnabled)
* {
* fprintf(stderr, "Info: Using alpha channel in render extension.\n");
* }
* }
*/
#endif /* RENDER */
/*
* Check if the composite extension is supported on the remote
* display and prepare the agent for its use.
*/
nxagentCompositeExtensionInit();
/* We use this to get informed about RandR changes on the real display.
FIXME: It would probably be better to use an RRScreenChangeNotifyEvent here. */
XSelectInput(nxagentDisplay, DefaultRootWindow(nxagentDisplay), StructureNotifyMask);
#ifdef NXAGENT_TIMESTAMP
fprintf(stderr, "Screen: open screen finished, time is [%d] milliseconds.\n",
GetTimeInMillis() - startTime);
#endif
#ifdef WATCH
fprintf(stderr, "%s: Watchpoint 8.\n", __func__);
/*
Reply Total Cached Bits In Bits Out Bits/Reply Ratio
------- ----- ------ ------- -------- ---------- -----
#1 U 2 1 5344 bits (1 KB) -> 2344 bits (0 KB) -> 2672/1 -> 1172/1 = 2.280:1
#16 11 2816 bits (0 KB) -> 197 bits (0 KB) -> 256/1 -> 18/1 = 14.294:1
#91 1 16640 bits (2 KB) -> 12314 bits (2 KB) -> 16640/1 -> 12314/1 = 1.351:1
#98 2 512 bits (0 KB) -> 57 bits (0 KB) -> 256/1 -> 28/1 = 8.982:1
*/
sleep(30);
#endif
return True;
}
Bool nxagentCloseScreen(ScreenPtr pScreen)
{
#ifdef DEBUG
fprintf(stderr, "running %s()\n", __func__);
#endif
/*
* We have called fbScreenInit() in nxagenOpenScreen, which in turn
* called fbOpenScreen. But we are not using the data as created by
* fbOpenScreen but have freed it and replaced by our own. So we free
* our own stuff here and take care that fbCloseScreen will not free
* them again.
*/
freeDepths(pScreen->allowedDepths, pScreen->numDepths);
pScreen->allowedDepths = NULL;
pScreen->numDepths = 0;
/*
* Free the frame buffer.
*/
SAFE_free(((PixmapPtr)pScreen -> devPrivate) -> devPrivate.ptr);
SAFE_free(pScreen->devPrivate);
SAFE_free(pScreen->visuals);
fbCloseScreen(pScreen);
/*
* Reset the geometry and alpha information
* used by proxy to unpack the packed images.
*/
nxagentResetVisualCache();
nxagentResetAlphaCache();
nxagentReleaseAllSplits();
/*
* The assumption is that all X resources will be
* destroyed upon closing the display connection.
* There is no need to generate extra protocol.
*/
return True;
}
/*
* This function comes from the xfree86 Xserver.
*/
static void nxagentSetRootClip (ScreenPtr pScreen, Bool enable)
{
WindowPtr pWin = pScreen->root;
Bool WasViewable = (Bool)(pWin->viewable);
Bool anyMarked = FALSE;
RegionPtr pOldClip = NULL;
#ifdef DO_SAVE_UNDERS
Bool dosave = FALSE;
#endif
WindowPtr pLayerWin;
if (WasViewable)
{
for (WindowPtr pChild = pWin->firstChild; pChild; pChild = pChild->nextSib)
{
(void) (*pScreen->MarkOverlappedWindows)(pChild,
pChild,
&pLayerWin);
}
(*pScreen->MarkWindow) (pWin);
anyMarked = TRUE;
if (pWin->valdata)
{
if (HasBorder (pWin))
{
RegionPtr borderVisible = RegionCreate(NullBox, 1);
RegionSubtract(borderVisible,
&pWin->borderClip, &pWin->winSize);
pWin->valdata->before.borderVisible = borderVisible;
}
pWin->valdata->before.resized = TRUE;
}
}
/*
* Use REGION_BREAK to avoid optimizations in ValidateTree
* that assume the root borderClip can't change. Well, normally
* it doesn't...)
*/
if (enable)
{
BoxRec box = {.x1 = 0, .y1 = 0, .x2 = pScreen->width, .y2 = pScreen->height};
RegionInit(&pWin->winSize, &box, 1);
RegionInit(&pWin->borderSize, &box, 1);
if (WasViewable)
RegionReset(&pWin->borderClip, &box);
pWin->drawable.width = pScreen->width;
pWin->drawable.height = pScreen->height;
RegionBreak(&pWin->clipList);
}
else
{
RegionEmpty(&pWin->borderClip);
RegionBreak(&pWin->clipList);
}
ResizeChildrenWinSize (pWin, 0, 0, 0, 0);
if (WasViewable)
{
if (pWin->backStorage)
{
pOldClip = RegionCreate(NullBox, 1);
RegionCopy(pOldClip, &pWin->clipList);
}
if (pWin->firstChild)
{
anyMarked |= (*pScreen->MarkOverlappedWindows)(pWin->firstChild,
pWin->firstChild,
(WindowPtr *)NULL);
}
else
{
(*pScreen->MarkWindow) (pWin);
anyMarked = TRUE;
}
#ifdef DO_SAVE_UNDERS
if (DO_SAVE_UNDERS(pWin))
{
dosave = (*pScreen->ChangeSaveUnder)(pLayerWin, pLayerWin);
}
#endif /* DO_SAVE_UNDERS */
if (anyMarked)
(*pScreen->ValidateTree)(pWin, NullWindow, VTOther);
}
if (pWin->backStorage && pOldClip &&
((pWin->backingStore == Always) || WasViewable))
{
if (!WasViewable)
pOldClip = &pWin->clipList; /* a convenient empty region */
RegionPtr bsExposed = (*pScreen->TranslateBackingStore)
(pWin, 0, 0, pOldClip,
pWin->drawable.x, pWin->drawable.y);
if (WasViewable)
RegionDestroy(pOldClip);
if (bsExposed)
{
RegionPtr valExposed = NullRegion;
if (pWin->valdata)
valExposed = &pWin->valdata->after.exposed;
(*pScreen->WindowExposures) (pWin, valExposed, bsExposed);
if (valExposed)
RegionEmpty(valExposed);
RegionDestroy(bsExposed);
}
}
if (WasViewable)
{
if (anyMarked)
(*pScreen->HandleExposures)(pWin);
#ifdef DO_SAVE_UNDERS
if (dosave)
(*pScreen->PostChangeSaveUnder)(pLayerWin, pLayerWin);
#endif /* DO_SAVE_UNDERS */
if (anyMarked && pScreen->PostValidateTree)
(*pScreen->PostValidateTree)(pWin, NullWindow, VTOther);
}
if (pWin->realized)
WindowsRestructured ();
FlushAllOutput ();
}
Bool nxagentResizeScreen(ScreenPtr pScreen, int width, int height,
int mmWidth, int mmHeight, Bool doresize)
{
#ifdef DEBUG
fprintf(stderr, "%s: called with w [%d] (%dmm) h [%d] (%dmm) doresize [%d]\n",
__func__, width, mmWidth, height, mmHeight, doresize);
#endif
#ifdef TEST
nxagentPrintAgentGeometry("Before Resize Screen", "nxagentResizeScreen:");
#endif
/*
* Change screen properties.
*/
int oldWidth = pScreen -> width;
int oldHeight = pScreen -> height;
int oldMmWidth = pScreen -> mmWidth;
int oldMmHeight = pScreen -> mmHeight;
pScreen -> width = width;
pScreen -> height = height;
/*
* Compute screen dimensions if they aren't provided.
*/
if (mmWidth == 0)
{
if (monitorResolution < 0)
{
mmWidth = width * DisplayWidthMM(nxagentDisplay, DefaultScreen(nxagentDisplay)) /
DisplayWidth(nxagentDisplay, DefaultScreen(nxagentDisplay));
}
else
{
mmWidth = (width * 254 + monitorResolution * 5) / (monitorResolution * 10);
}
if (mmWidth < 1)
{
mmWidth = 1;
}
}
if (mmHeight == 0)
{
if (monitorResolution < 0)
{
mmHeight = height * DisplayHeightMM(nxagentDisplay, DefaultScreen(nxagentDisplay)) /
DisplayHeight(nxagentDisplay, DefaultScreen(nxagentDisplay));
}
else
{
mmHeight = (height * 254 + monitorResolution * 5) / (monitorResolution * 10);
}
if (mmHeight < 1)
{
mmHeight = 1;
}
}
pScreen -> mmWidth = mmWidth;
pScreen -> mmHeight = mmHeight;
#ifdef DEBUG
fprintf(stderr, "%s: old w [%d] (%dmm) h [%d] (%dmm) new w [%d] (%dmm) h [%d] (%dmm)\n",
__func__, oldWidth, oldMmWidth, oldHeight, oldMmHeight, width, mmWidth, height, mmHeight);
#endif
PixmapPtr pPixmap = fbGetScreenPixmap(pScreen);
char *fbBits = realloc(pPixmap -> devPrivate.ptr, PixmapBytePad(width, pScreen->rootDepth) *
height * BitsPerPixel(pScreen->rootDepth) / 8);
if (fbBits == NULL)
{
goto nxagentResizeScreenError;
}
if (!miModifyPixmapHeader(pPixmap, width, height,
pScreen->rootDepth, BitsPerPixel(pScreen->rootDepth),
PixmapBytePad(width,
pScreen->rootDepth), fbBits))
{
/*
FIXME: We should try to restore the previously
reallocated frame buffer pixmap.
*/
goto nxagentResizeScreenError;
}
#ifdef DEBUG
fprintf(stderr, "%s: setting RootWidth/Height to [%d]x[%d]\n", __func__, width, height);
#endif
nxagentChangeOption(RootWidth, width);
nxagentChangeOption(RootHeight, height);
if (nxagentOption(Fullscreen))
{
nxagentChangeOption(RootX, (nxagentOption(Width) -
nxagentOption(RootWidth)) / 2);
nxagentChangeOption(RootY, (nxagentOption(Height) -
nxagentOption(RootHeight)) / 2);
}
else
{
nxagentChangeOption(RootX, 0);
nxagentChangeOption(RootY, 0);
}
nxagentChangeOption(ViewportXSpan, nxagentOption(Width) - nxagentOption(RootWidth));
nxagentChangeOption(ViewportYSpan, nxagentOption(Height) - nxagentOption(RootHeight));
/*
* Change agent window size and size hints.
*/
if (!(nxagentOption(Fullscreen) || nxagentOption(AllScreens)))
{
nxagentSetWMNormalHints(pScreen->myNum, width, height);
if (doresize)
{
#ifdef DEBUG
fprintf(stderr, "%s: resizing DefaultWindow to [%d]x[%d]\n", __func__, width, height);
#endif
XResizeWindow(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum], width, height);
}
if (!nxagentOption(Rootless))
{
#ifdef DEBUG
fprintf(stderr, "%s: resizing InputWindow to [%d]x[%d]\n", __func__, width, height);
#endif
XResizeWindow(nxagentDisplay, nxagentInputWindows[pScreen -> myNum], width, height);
}
}
/*
* Set properties for the agent root window.
*/
BoxRec box = {.x1 = 0, .y1 = 0, .x2 = width, .y2 = height};
pScreen->root -> drawable.width = width;
pScreen->root -> drawable.height = height;
pScreen->root -> drawable.x = 0;
pScreen->root -> drawable.y = 0;
RegionInit(&pScreen->root -> borderSize, &box, 1);
RegionInit(&pScreen->root -> winSize, &box, 1);
RegionInit(&pScreen->root -> clipList, &box, 1);
RegionInit(&pScreen->root -> borderClip, &box, 1);
#ifdef DEBUG
fprintf(stderr, "%s: calling PositionWindow() [0,0] for root Window\n", __func__);
#endif
(*pScreen -> PositionWindow)(pScreen->root, 0, 0);
nxagentSetRootClip(pScreen, 1);
#ifdef DEBUG
fprintf(stderr, "%s: moving root window to [%d,%d]\n", __func__, nxagentOption(RootX), nxagentOption(RootY));
#endif
XMoveWindow(nxagentDisplay, nxagentWindow(screenInfo.screens[0]->root),
nxagentOption(RootX), nxagentOption(RootY));
nxagentMoveViewport(pScreen, 0, 0);
/*
* Update pointer bounds.
*/
ScreenRestructured(pScreen);
#ifdef TEST
nxagentPrintAgentGeometry("After Resize Screen", "nxagentResizeScreen:");
#endif
nxagentSetPrintGeometry(pScreen -> myNum);
return 1;
nxagentResizeScreenError:
#ifdef DEBUG
fprintf(stderr, "%s: nxagentResizeScreenError\n", __func__);
#endif
pScreen -> width = oldWidth;
pScreen -> height = oldHeight;
pScreen -> mmWidth = oldMmWidth;
pScreen -> mmHeight = oldMmHeight;
return 0;
}
void nxagentShadowSetRatio(float floatXRatio, float floatYRatio)
{
if (floatXRatio == 0)
{
floatXRatio = 1.0;
}
if (floatYRatio == 0)
{
floatYRatio = 1.0;
}
int intXRatio = floatXRatio * (1 << PRECISION);
int intYRatio = floatYRatio * (1 << PRECISION);
nxagentChangeOption(FloatXRatio, floatXRatio);
nxagentChangeOption(FloatYRatio, floatYRatio);
nxagentChangeOption(XRatio, intXRatio);
nxagentChangeOption(YRatio, intYRatio);
#ifdef TEST
fprintf(stderr, "Info: Using X ratio [%f] Y ratio [%f].\n",
nxagentOption(FloatXRatio), nxagentOption(FloatYRatio));
#endif
}
void nxagentShadowSetWindowsSize(void)
{
XResizeWindow(nxagentDisplay, nxagentDefaultWindows[0],
nxagentOption(Width), nxagentOption(Height));
XMoveResizeWindow(nxagentDisplay, nxagentInputWindows[0], 0, 0,
nxagentOption(Width), nxagentOption(Height));
}
void nxagentShadowSetWindowOptions(void)
{
nxagentChangeOption(RootWidth, nxagentScale(nxagentShadowWidth, nxagentOption(XRatio)));
nxagentChangeOption(RootHeight, nxagentScale(nxagentShadowHeight, nxagentOption(YRatio)));
nxagentChangeOption(SavedRootWidth, nxagentOption(RootWidth));
nxagentChangeOption(SavedRootHeight, nxagentOption(RootHeight));
nxagentChangeOption(RootX, (nxagentOption(Width) - nxagentOption(RootWidth)) >> 1);
nxagentChangeOption(RootY, (nxagentOption(Height) - nxagentOption(RootHeight)) >> 1);
}
int nxagentShadowInit(ScreenPtr pScreen, WindowPtr pWin)
{
char *layout = NULL;
#ifdef TEST
fprintf(stderr, "Info: Init shadow session. nxagentDisplayName [%s] "
"nxagentDisplay [%p] nxagentShadowDisplayName [%s].\n",
nxagentDisplayName, (void *) nxagentDisplay,
nxagentShadowDisplayName);
#endif
if (nxagentKeyboard != NULL)
{
int i;
for (i = 0; nxagentKeyboard[i] != '/' && nxagentKeyboard[i] != '\0'; i++);
if(nxagentKeyboard[i] == '\0' || nxagentKeyboard[i + 1] == '\0' || i == 0)
{
#ifdef WARNING
fprintf(stderr,"WARNING! Wrong keyboard type: %s.\n", nxagentKeyboard);
#endif
}
else
{
layout = strdup(&nxagentKeyboard[i + 1]);
}
}
#ifdef DEBUG
fprintf(stderr, "%s: Setting the master uid [%d].\n", __func__,
nxagentShadowUid);
#endif
if (nxagentShadowUid != -1)
{
NXShadowSetDisplayUid(nxagentShadowUid);
}
if (nxagentOption(UseDamage))
{
NXShadowDisableDamage();
}
if (NXShadowCreate(nxagentDisplay, layout, nxagentShadowDisplayName,
(void *) &nxagentShadowDisplay) != 1)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Failed to initialize shadow "
"display [%s].\n", __func__, nxagentShadowDisplayName);
#endif
return -1;
}
/*
* The shadow nxagent sets the _NX_SHADOW property on the master X
* server root window in order to notify its presence.
*/
XlibAtom nxagentShadowAtom = XInternAtom(nxagentShadowDisplay, "_NX_SHADOW", False);
XChangeProperty(nxagentShadowDisplay, DefaultRootWindow(nxagentShadowDisplay),
nxagentShadowAtom, XA_STRING, 8, PropModeReplace, NULL, 0);
if (NXShadowAddUpdaterDisplay(nxagentDisplay, &nxagentShadowWidth,
&nxagentShadowHeight, &nxagentMasterDepth) == 0)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Failed to add display [%s].\n", __func__,
nxagentDisplayName);
#endif
return -1;
}
if (nxagentOption(Fullscreen))
{
nxagentShadowSetRatio(WidthOfScreen(DefaultScreenOfDisplay(nxagentDisplay)) * 1.0 / nxagentShadowWidth,
HeightOfScreen(DefaultScreenOfDisplay(nxagentDisplay)) * 1.0 / nxagentShadowHeight);
}
else if (nxagentUserGeometry.flag != 0)
{
nxagentShadowSetRatio(nxagentOption(RootWidth) * 1.0 / nxagentShadowWidth,
nxagentOption(RootHeight) * 1.0 / nxagentShadowHeight);
}
if (DefaultVisualOfScreen(DefaultScreenOfDisplay(nxagentDisplay)) ->
class != TrueColor)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! The visual class of the remote "
"X server is not TrueColor.\n", __func__);
#endif
return -1;
}
if (DefaultVisualOfScreen(DefaultScreenOfDisplay(nxagentShadowDisplay)) ->
class != TrueColor)
{
#ifdef PANIC
const char *className;
switch (DefaultVisualOfScreen(DefaultScreenOfDisplay(nxagentShadowDisplay)) -> class)
{
case StaticGray: { className = "StaticGray"; break; }
case StaticColor: { className = "StaticColor"; break; }
case PseudoColor: { className = "PseudoColor"; break; }
case DirectColor: { className = "DirectColor"; break; }
case GrayScale: { className = "GrayScale"; break; }
default: { className = ""; break; }
}
fprintf(stderr, "%s: PANIC! Cannot shadow the display. "
"%s visual class is not supported. Only TrueColor visuals "
"are supported.\n", __func__, className);
#endif /* #endif PANIC */
return -1;
}
nxagentShadowDepth = pScreen -> rootDepth;
switch (nxagentMasterDepth)
{
case 32:
case 24:
{
if (nxagentShadowDepth == 16)
{
nxagentCheckDepth = 1;
}
else if (nxagentShadowDepth == 8)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Unable to shadow a %d bit "
"display with a 8 bit screen depth.\n", __func__, nxagentMasterDepth);
#endif
return -1;
}
nxagentBppMaster = 4;
break;
}
case 16:
{
if (nxagentShadowDepth > 16)
{
nxagentCheckDepth = 1;
}
else if (nxagentShadowDepth == 8)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Unable to shadow a 16 bit "
"display with a 8 bit screen depth.\n", __func__);
#endif
return -1;
}
nxagentBppMaster = 2;
break;
}
case 8:
{
if (nxagentShadowDepth != 8)
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Unable to shadow a 8 bit "
"display with a %d bit screen depth.\n", __func__, nxagentShadowDepth);
#endif
return -1;
}
nxagentBppMaster = 1;
break;
}
default:
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! The depth is not 32, 24, 16 or 8 bit.\n", __func__);
#endif
return -1;
}
}
if (nxagentShadowDepth >= 24)
{
nxagentBppShadow = 4;
}
else if (nxagentShadowDepth == 16)
{
nxagentBppShadow = 2;
}
else if (nxagentShadowDepth == 8)
{
nxagentBppShadow = 1;
}
imageByteOrder = nxagentShadowDisplay -> byte_order;
nxagentShadowXConnectionNumber = XConnectionNumber(nxagentShadowDisplay);
#ifdef TEST
fprintf(stderr, "%s: Adding the X connection [%d] "
"to the device set.\n", __func__, nxagentShadowXConnectionNumber);
#endif
SetNotifyFd(nxagentShadowXConnectionNumber, nxagentNotifyConnection, X_NOTIFY_READ, NULL);
accessPixmapID = FakeClientID(serverClient -> index);
AddResource(accessPixmapID, RT_PIXMAP, (void *)nxagentShadowPixmapPtr);
accessWindowID = FakeClientID(serverClient -> index);
AddResource(accessWindowID, RT_WINDOW, (void *)nxagentShadowWindowPtr);
nxagentResizeScreen(pScreen, nxagentShadowWidth, nxagentShadowHeight, pScreen -> mmWidth, pScreen -> mmHeight, True);
nxagentShadowCreateMainWindow(pScreen, pWin, nxagentShadowWidth, nxagentShadowHeight);
if (nxagentRemoteMajor <= 3)
{
nxagentShadowSetWindowsSize();
nxagentSetWMNormalHints(0, nxagentOption(Width), nxagentOption(Height));
}
XMapWindow(nxagentDisplay, nxagentDefaultWindows[0]);
/*
* Clean up the main window.
*/
XGCValues value = {
.foreground = 0x00000000,
.background = 0x00000000,
.plane_mask = 0xffffffff,
.fill_style = FillSolid,
};
XlibGC gc = XCreateGC(nxagentDisplay, nxagentPixmap(nxagentShadowPixmapPtr), GCBackground |
GCForeground | GCFillStyle | GCPlaneMask, &value);
XFillRectangle(nxagentDisplay, nxagentPixmap(nxagentShadowPixmapPtr), gc, 0, 0,
nxagentShadowWidth, nxagentShadowHeight);
XFreeGC(nxagentDisplay, gc);
RegionInit(&nxagentShadowUpdateRegion, (BoxRec*)NULL, 1);
return 0;
}
int nxagentShadowCreateMainWindow(ScreenPtr pScreen, WindowPtr pWin, int width, int height)
{
nxagentShadowWidth = width;
nxagentShadowHeight = height;
NXShadowUpdateBuffer((void *)&nxagentShadowBuffer);
#ifdef TEST
fprintf(stderr, "%s: Update frame buffer [%p].\n", __func__, nxagentShadowBuffer);
#endif
nxagentShadowSetWindowOptions();
if (nxagentShadowPixmapPtr != NULL)
{
nxagentDestroyPixmap(nxagentShadowPixmapPtr);
}
if (nxagentShadowWindowPtr != NULL)
{
DeleteWindow(nxagentShadowWindowPtr, accessWindowID);
}
nxagentShadowPixmapPtr = nxagentCreatePixmap(pScreen, nxagentShadowWidth, nxagentShadowHeight, nxagentShadowDepth, 0);
if (nxagentShadowPixmapPtr)
{
ChangeResourceValue(accessPixmapID, RT_PIXMAP, (void *) nxagentShadowPixmapPtr);
nxagentShadowPixmapPtr -> drawable.id = accessPixmapID;
#ifdef TEST
fprintf(stderr, "%s: nxagentShadowPixmapPtr [%p] PixmapM -> drawable.id [%u].\n", __func__,
(void *)nxagentShadowPixmapPtr, nxagentShadowPixmapPtr -> drawable.id);
fprintf(stderr, "%s: Create pixmap with width [%d] height [%d] depth [%d].\n", __func__,
nxagentShadowWidth, nxagentShadowHeight, (int)nxagentShadowDepth);
#endif
}
else
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Failed to create pixmap with width [%d] height [%d] depth [%d].\n", __func__,
nxagentShadowWidth, nxagentShadowHeight, (int)nxagentShadowDepth);
#endif
}
XFreePixmap(nxagentDisplay, nxagentPixmap(nxagentVirtualPixmap(nxagentShadowPixmapPtr)));
XID xid = XCreatePixmap(nxagentDisplay, nxagentDefaultWindows[0],
nxagentScale(nxagentShadowWidth, nxagentOption(XRatio)),
nxagentScale(nxagentShadowHeight, nxagentOption(YRatio)), nxagentShadowDepth);
nxagentPixmap(nxagentVirtualPixmap(nxagentShadowPixmapPtr)) = xid;
nxagentPixmap(nxagentRealPixmap(nxagentShadowPixmapPtr)) = xid;
if (nxagentShadowGCPtr != NULL)
{
FreeScratchGC(nxagentShadowGCPtr);
}
/*
* FIXME: Should use CreateGC.
*/
nxagentShadowGCPtr = GetScratchGC(nxagentShadowPixmapPtr -> drawable.depth, nxagentShadowPixmapPtr -> drawable.pScreen);
if (nxagentShadowGCPtr)
{
#ifdef TEST
fprintf(stderr, "%s: Created GC with pGC[%p]\n", __func__, (void *) nxagentShadowGCPtr);
#endif
ValidateGC((DrawablePtr)nxagentShadowPixmapPtr, nxagentShadowGCPtr);
}
else
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Failed to create GC.", __func__);
#endif
}
Mask mask = CWBackPixmap | CWEventMask | CWCursor;
Mask maskb = nxagentGetDefaultEventMask() | ResizeRedirectMask | ExposureMask;
XID values[4];
XID *vlist = values;
*vlist++ = (XID)nxagentShadowPixmapPtr -> drawable.id;
*vlist++ = (XID)maskb;
*vlist = (XID)None;
int error;
nxagentShadowWindowPtr = CreateWindow(accessWindowID, pWin, 0, 0, nxagentShadowWidth,
nxagentShadowHeight, 0, InputOutput, mask, (XID *)values,
nxagentShadowDepth, serverClient, CopyFromParent, &error);
mask = CWWidth | CWHeight;
XWindowChanges changes = {
.width = nxagentScale(nxagentShadowWidth, nxagentOption(XRatio)),
.height = nxagentScale(nxagentShadowHeight, nxagentOption(YRatio))
};
XConfigureWindow(nxagentDisplay, nxagentWindow(nxagentShadowWindowPtr), mask, &changes);
nxagentMoveViewport(pScreen, 0, 0);
if (nxagentShadowWindowPtr && !error)
{
#ifdef TEST
fprintf(stderr, "%s: Create window with nxagentShadowWindowPtr [%p]"
"nxagentShadowWindowPtr -> drawable.id [%u].\n", __func__, (void *) nxagentShadowWindowPtr,
nxagentShadowWindowPtr -> drawable.id);
fprintf(stderr, "%s: parent nxagentShadowWindowPtr [%p] parent -> drawable.id [%u].\n", __func__,
(void *)nxagentShadowWindowPtr->parent, nxagentShadowWindowPtr -> parent -> drawable.id);
#endif
ChangeResourceValue(accessWindowID, RT_WINDOW, (void *) nxagentShadowWindowPtr);
}
else
{
#ifdef PANIC
fprintf(stderr, "%s: PANIC! Failed to create window.\n", __func__);
#endif
}
XMapWindow(nxagentDisplay, nxagentWindow(nxagentShadowWindowPtr));
MapWindow(nxagentShadowWindowPtr, serverClient);
mask = CWX | CWY | CWWidth | CWHeight;
changes.x = nxagentOption(RootX);
changes.y = nxagentOption(RootY);
changes.width = nxagentScale(nxagentShadowWidth, nxagentOption(XRatio));
changes.height = nxagentScale(nxagentShadowHeight, nxagentOption(YRatio));
XConfigureWindow(nxagentDisplay, nxagentWindow(pWin), mask, &changes);
#ifdef TEST
fprintf(stderr, "%s: Completed mapping of Access window.\n", __func__);
#endif
return 0;
}
int nxagentShadowSendUpdates(int *suspended)
{
*suspended = 0;
if (RegionNil(&nxagentShadowUpdateRegion))
{
return 0;
}
nxagentMarkCorruptedRegion((DrawablePtr)nxagentShadowPixmapPtr, &nxagentShadowUpdateRegion);
RegionEmpty(&nxagentShadowUpdateRegion);
return 1;
}
int nxagentShadowPoll(PixmapPtr shadowPixmapPtr, GCPtr shadowGCPtr,
unsigned char shadowDepth, int shadowWidth,
int shadowHeight, char *shadowBuffer, int *changed, int *suspended)
{
RegionRec updateRegion;
RegionRec tempRegion;
RegionNull(&updateRegion);
RegionNull(&tempRegion);
int result = NXShadowHasChanged(nxagentUserInput, NULL, suspended);
*changed = result;
if (result == 1)
{
char *tBuffer = NULL;
char *iBuffer, *ptBox;
nxagentRemoveSplashWindow();
long numRects;
NXShadowExportChanges(&numRects, &ptBox);
BoxRec *pBox = (BoxRec *)ptBox;
#ifdef TEST
fprintf(stderr, "%s: nRects[%ld], pBox[%p] depth[%d].\n", __func__, numRects, (void *) pBox, shadowDepth);
#endif
for (int n = 0; n < numRects; n++)
{
/*
* The BoxRec struct defined in the Xserver has a different
* variable order in comparison with the BoxRec struct in the Xlib.
* the second and third field are inverted.
*/
int x = pBox[n].x1;
int y = pBox[n].x2;
int y2 = pBox[n].y2;
unsigned int width = pBox[n].y1 - pBox[n].x1;/* y1 = x2 */
unsigned int height = y2 - pBox[n].x2; /* x2 = y1 */
if((x + width) > shadowWidth || (y + height) > shadowHeight)
{
/*
* Out of bounds. Maybe a resize of the master session is going on.
*/
continue;
}
int line = PixmapBytePad(width, nxagentMasterDepth);
#ifdef DEBUG
fprintf(stderr, "%s: Rectangle Number[%d] - x[%d]y[%d]W[%u]H[%u].\n", __func__, n+1, x, y, width, height);
#endif
unsigned int length = nxagentImageLength(width, height, ZPixmap, 0, nxagentMasterDepth);
SAFE_free(tBuffer);
tBuffer = malloc(length);
if (tBuffer == NULL)
{
#ifdef PANIC
fprintf(stderr, "%s: malloc failed.\n", __func__);
#endif
return -1;
}
iBuffer = tBuffer;
for (int c = 0; c + y < y2; c++)
{
memcpy(tBuffer, shadowBuffer + x * nxagentBppMaster +
(y + c) * shadowWidth * nxagentBppMaster, line);
tBuffer += line;
}
tBuffer = iBuffer;
if (nxagentCheckDepth == 1)
{
nxagentShadowAdaptDepth(width, height, line, &tBuffer);
}
fbPutImage(nxagentVirtualDrawable((DrawablePtr)shadowPixmapPtr), shadowGCPtr,
shadowDepth, x, y, width, height, 0, ZPixmap, tBuffer);
BoxRec box = {.x1 = x, .x2 = x + width, .y1 = y, .y2 = y + height};
RegionInit(&tempRegion, &box, 1);
RegionAppend(&updateRegion, &tempRegion);
RegionUninit(&tempRegion);
int overlap;
RegionValidate(&updateRegion, &overlap);
RegionUnion(&nxagentShadowUpdateRegion, &nxagentShadowUpdateRegion, &updateRegion);
}
SAFE_free(tBuffer);
RegionUninit(&updateRegion);
}
else if (result == -1)
{
#ifdef TEST
fprintf(stderr, "%s: polling failed!\n", __func__);
#endif
usleep(50 * 1000);
return -1;
}
return 0;
}
void nxagentShadowAdaptDepth(unsigned int width, unsigned int height,
unsigned int lineMaster, char **buffer)
{
unsigned char * tBuffer = (unsigned char *) *buffer;
unsigned int length = nxagentImageLength(width, height, ZPixmap, 0, nxagentShadowDepth);
unsigned char *cBuffer = malloc(length);
unsigned char *icBuffer = cBuffer;
if (!cBuffer)
{
#ifdef WARNING
fprintf(stderr, "WARNING: could not allocate cBuffer\n");
#endif
return;
}
Visual *pVisual = nxagentImageVisual((DrawablePtr) nxagentShadowPixmapPtr, nxagentShadowDepth);
if (pVisual == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: WARNING! Visual not found. Using default visual.\n", __func__);
#endif
pVisual = nxagentVisuals[nxagentDefaultVisualIndex].visual;
}
#ifdef TEST
fprintf(stderr, "%s: Shadow redMask [%lu] greenMask[%lu] blueMask[%lu].\n", __func__,
pVisual -> red_mask, pVisual -> green_mask, pVisual -> blue_mask);
#endif
unsigned long redMask = nxagentShadowDisplay -> screens[0].root_visual[0].red_mask;
unsigned long greenMask = nxagentShadowDisplay -> screens[0].root_visual[0].green_mask;
unsigned long blueMask = nxagentShadowDisplay -> screens[0].root_visual[0].blue_mask;
#ifdef TEST
fprintf(stderr, "%s: Master redMask [%lu] greenMask[%lu] blueMask[%lu].\n", __func__,
redMask, greenMask, blueMask);
#endif
switch(nxagentMasterDepth)
{
/*
* The Shadow agent has 24 bit depth.
*/
case 16:
{
unsigned int pad = lineMaster - nxagentBppMaster * width;
#ifdef TEST
fprintf(stderr, "%s: line [%d] width[%d] pad[%d].\n", __func__, lineMaster, width, pad);
#endif
while (height > 0)
{
for (unsigned int c = 0; c < width ; c++)
{
unsigned char red;
unsigned char green;
unsigned char blue;
unsigned short color16;
if (imageByteOrder == LSBFirst)
{
color16 = *tBuffer++;
color16 |= (*tBuffer << 8);
}
else
{
color16 = (*tBuffer++) << 8;
color16 |= *tBuffer;
}
blue = ((short) blueMask & color16) << 3;
blue |= 0x3;
if (greenMask == 0x7e0)
{
/*
* bit mask 5-6-5
*/
green = ((short) greenMask & color16) >> 3;
green |= 0x2;
red = ((short) redMask & color16) >> 8;
red |= 0x3;
}
else
{
/*
* bit mask 5-5-5
*/
green = ((short) greenMask & color16) >> 2;
green |= 0x3;
red = ((short) redMask & color16) >> 7;
red |= 0x3;
}
tBuffer++;
if (nxagentDisplay -> byte_order == LSBFirst)
{
*cBuffer++ = blue;
*cBuffer++ = green;
*cBuffer++ = red;
cBuffer++;
}
else
{
cBuffer++;
*cBuffer++ = red;
*cBuffer++ = green;
*cBuffer++ = blue;
}
}
tBuffer += pad;
height--;
}
break;
}
/*
* The Shadow agent has 16 bit depth.
*/
case 24:
{
unsigned int lineShadow = PixmapBytePad(width, nxagentShadowDepth);
unsigned int pad = lineShadow - nxagentBppShadow * width;
#ifdef TEST
fprintf(stderr, "%s: line [%d] width[%d] pad[%d].\n", __func__, lineShadow, width, pad);
#endif
while (height > 0)
{
for (unsigned int c = 0; c < width; c++)
{
unsigned short color16;
unsigned int color32;
if (imageByteOrder == LSBFirst)
{
color32 = *tBuffer++;
color32 |= (*tBuffer++ << 8);
color32 |= (*tBuffer++ << 16);
tBuffer++;
}
else
{
tBuffer++;
color32 = (*tBuffer++ << 16);
color32 |= (*tBuffer++ << 8);
color32 |= *tBuffer++;
}
color16 = (color32 & (pVisual -> blue_mask << 3)) >> 3;
if (pVisual -> green_mask == 0x7e0)
{
/*
* bit mask 5-6-5
*/
color16 |= (color32 & (pVisual -> green_mask << 5)) >> 5;
color16 |= (color32 & (pVisual -> red_mask << 8)) >> 8;
}
else
{
/*
* bit mask 5-5-5
*/
color16 |= (color32 & (pVisual -> green_mask << 6)) >> 6;
color16 |= (color32 & (pVisual -> red_mask << 9)) >> 9;
}
if (nxagentDisplay -> byte_order == LSBFirst)
{
*cBuffer++ = color16 & 0xff;
*cBuffer++ = (color16 & 0xff00) >> 8;
}
else
{
*cBuffer++ = (color16 & 0xff00) >> 8;
*cBuffer++ = color16 & 0xff;
}
}
cBuffer += pad;
height--;
}
break;
}
}
cBuffer = (unsigned char *) *buffer;
*buffer = (char *) icBuffer;
SAFE_free(cBuffer);
}
#ifdef NXAGENT_ARTSD
unsigned char fromHexNibble(char c)
{
int uc = (unsigned char)c;
if(uc >= '0' && uc <= '9') return uc - (unsigned char)'0';
if(uc >= 'a' && uc <= 'f') return uc + 10 - (unsigned char)'a';
if(uc >= 'A' && uc <= 'F') return uc + 10 - (unsigned char)'A';
return 16; /*error*/
}
void nxagentPropagateArtsdProperties(ScreenPtr pScreen, char *port)
{
char tchar[] = " ";
/*
FIXME: The port information is not used at the moment and produces a
warning on recent gcc versions. Do we need such information
to run the audio forwarding?
char *chport;
char hex[] = "0123456789abcdef";
*/
Window rootWin = DefaultRootWindow(nxagentDisplay);
XlibAtom propAtom = nxagentAtoms[4]; /* MCOPGLOBALS */
XlibAtom atomReturnType;
int iReturnFormat;
unsigned long ulReturnItems;
unsigned long ulReturnBytesLeft;
unsigned char *pszReturnData = NULL;
/*
* Get at most 64KB of data.
*/
int iReturn = XGetWindowProperty(nxagentDisplay,
rootWin,
propAtom,
0,
65536 / 4,
False,
XA_STRING,
&atomReturnType,
&iReturnFormat,
&ulReturnItems,
&ulReturnBytesLeft,
&pszReturnData);
if (iReturn == Success && atomReturnType != None &&
ulReturnItems > 0 && pszReturnData != NULL)
{
#ifdef TEST
fprintf(stderr, "%s: Got [%ld] elements of format [%d] with [%ld] bytes left.\n", __func__,
ulReturnItems, iReturnFormat, ulReturnBytesLeft);
#endif
#ifdef WARNING
if (ulReturnBytesLeft > 0)
{
fprintf(stderr, "%s: WARNING! Could not get the whole ARTSD property data.\n", __func__);
}
#endif
char *local_buf = (char *) malloc(strlen((char*)pszReturnData) + 100);
if (local_buf)
{
memset(local_buf, 0, strlen((char *) pszReturnData));
int i, in;
for (i = 0, in = 0; pszReturnData[i] != '\0'; i++)
{
local_buf[in] = pszReturnData[i];
if (pszReturnData[i] == ':')
{
i++;
while (pszReturnData[i] != '\n')
{
unsigned char h = fromHexNibble(pszReturnData[i]);
i++;
if (pszReturnData[i] == '\0')
continue;
unsigned char l = fromHexNibble(pszReturnData[i]);
i++;
if (h >= 16 || l >= 16)
continue;
/*
* FIXME: The array tchar[] was used uninitialized. It's
* not clear to me the original purpose of the piece of
* code using it. To be removed in future versions.
*/
tchar[0] = tchar[1];
tchar[1] = tchar[2];
tchar[2] = tchar[3];
tchar[3] = (h << 4) + l;
tchar[4] ='\0';
if (strncmp(tchar, "tcp:", 4) == 0)
{
local_buf[in - 7] = '1';
local_buf[in - 6] = strlen(port) + 47;
in++;
local_buf[in] = pszReturnData[i - 2];
in++;
local_buf[in] = pszReturnData[i - 1];
/* "localhost:" */
strcat(local_buf,"6c6f63616c686f73743a");
in += 20;
/*
FIXME: The port information is not used at the moment and produces a
warning on recent gcc versions. Do we need such information
to run the audio forwarding?
chport = &port[0];
while(*chport != '\0')
{
in++;
local_buf[in] = hex[(*chport >> 4) & 0xf];
in++;
local_buf[in] = hex[*chport & 0xf];
*chport++;
}
*/
strcat(local_buf,"00");
in += 2;
while (pszReturnData[i] != '\n')
{
i++;
}
}
else
{
in++;
local_buf[in] = pszReturnData[i - 2];
in++;
local_buf[in] = pszReturnData[i - 1];
}
}
in++;
local_buf[in] = pszReturnData[i];
}
in++;
}
local_buf[in] = 0;
if (strlen(local_buf))
{
mcop_local_atom = MakeAtom(mcop_atom, strlen(mcop_atom), 1);
ChangeWindowProperty(pScreen->root,
mcop_local_atom,
XA_STRING,
iReturnFormat, PropModeReplace,
strlen(local_buf), local_buf, 1);
}
SAFE_free(local_buf);
}
}
}
#endif
Bool nxagentReconnectScreen(void *p0)
{
PixmapPtr pPixmap = (PixmapPtr)nxagentDefaultScreen->devPrivate;
#if defined(NXAGENT_RECONNECT_DEBUG) || defined(NXAGENT_RECONNECT_SCREEN_DEBUG)
fprintf(stderr, "%s\n", __func__);
#endif
if (!nxagentOpenScreen(nxagentDefaultScreen, nxagentArgc, nxagentArgv))
{
return False;
}
nxagentPixmap(pPixmap) = XCreatePixmap(nxagentDisplay,
nxagentDefaultWindows[nxagentDefaultScreen->myNum],
pPixmap -> drawable.width,
pPixmap -> drawable.height,
pPixmap -> drawable.depth);
#ifdef NXAGENT_RECONNECT_SCREEN_DEBUG
fprintf(stderr, "%s: recreated %p - ID %lx\n", __func__,
pPixmap,
nxagentPixmap( pPixmap ));
#endif
CARD16 w = 16;
CARD16 h = 16;
(*nxagentDefaultScreen->QueryBestSize)(StippleShape, &w, &h, nxagentDefaultScreen);
if (!(nxagentPixmap(nxagentDefaultScreen->PixmapPerDepth[0]) =
XCreatePixmap(nxagentDisplay,
nxagentDefaultDrawables[1],
w,
h,
1)));
Mask mask = nxagentGetDefaultEventMask();
mask |= NXAGENT_KEYBOARD_EVENT_MASK | NXAGENT_POINTER_EVENT_MASK;
nxagentSetDefaultEventMask(mask);
XSelectInput(nxagentDisplay, nxagentDefaultWindows[0], mask);
/*
* Turn off the screen-saver
*/
SaveScreens(SCREEN_SAVER_OFF, ScreenSaverActive);
/*
* reset the time to the next auto-disconnection.
*/
lastDeviceEventTime.milliseconds = GetTimeInMillis();
nxagentSetTimeoutTimer(0);
return True;
}
/* intersect two rectangles */
Bool intersect(int ax1, int ay1, unsigned int aw, unsigned int ah,
int bx1, int by1, unsigned int bw, unsigned int bh,
int *x, int *y, unsigned int *w, unsigned int *h)
{
int ax2 = ax1 + aw;
int ay2 = ay1 + ah;
int bx2 = bx1 + bw;
int by2 = by1 + bh;
/* thanks to http://silentmatt.com/rectangle-intersection */
/* check if there's any intersection at all */
if (ax2 < bx1 || bx2 < ax1 || ay2 < by1 || by2 < ay1) {
#ifdef DEBUG
fprintf(stderr, "%s: the given rectangles do not intersect at all\n", __func__);
#endif
return FALSE;
}
int tx1 = max(ax1, bx1);
int ty1 = max(ay1, by1);
int tx2 = min(ax2, bx2);
int ty2 = min(ay2, by2);
int ix = tx1 - ax1;
int iy = ty1 - ay1;
unsigned int iw = tx2 - tx1;
unsigned int ih = ty2 - ty1;
/* check if the resulting rectangle is feasible */
if (iw <= 0 || ih <= 0) {
#ifdef DEBUG
fprintf(stderr, "%s: intersection rectangle not feasible\n", __func__);
#endif
return FALSE;
}
*x = ix;
*y = iy;
*w = iw;
*h = ih;
#ifdef DEBUG
fprintf(stderr, "%s: intersection is: ([%d],[%d]) [ %d x %d ]\n", __func__, *x, *y, *w, *h);
#endif
return TRUE;
}
#ifndef NXAGENT_RANDR_XINERAMA_CLIPPING
/* intersect two rectangles, return aw/ah for w/h if resulting
rectangle is (partly) outside of bounding box */
Bool intersect_bb(int ax1, int ay1, unsigned int aw, unsigned int ah,
int bx1, int by1, unsigned int bw, unsigned int bh,
int bbx1, int bby1, int bbx2, int bby2,
int *x, int *y, unsigned int *w, unsigned int *h)
{
#ifdef DEBUG
fprintf(stderr, "%s: session window: ([%d],[%d]) [ %d x %d ]\n", __func__, ax1, ay1, aw, ah);
fprintf(stderr, "%s: crtc: ([%d],[%d]) [ %d x %d ]\n", __func__, bx1, by1, bw, bh);
fprintf(stderr, "%s: bounding box: ([%d],[%d]) [ %d x %d ]\n", __func__, bbx1, bby1, bbx2-bbx1, bby2-bby1);
#endif
Bool result = intersect(ax1, ay1, aw, ah, bx1, by1, bw, bh, x, y, w, h);
if (result == TRUE)
{
/*
* ###### The X-Coordinate ######
*/
/* check if outside-left of bounding box */
if (bx1 == bbx1 && ax1 < bbx1)
{
*w += bbx1 - ax1;
*x = 0;
#ifdef DEBUG
fprintf(stderr, "%s: session box is outside-left of the bounding box - width gets adapted to [%d]\n", __func__, *w);
#endif
}
/* check if outside-right of bounding box */
if (bx1 + bw == bbx2 && ax1 + aw > bbx2)
{
*w += ax1 + aw - bbx2;
#ifdef DEBUG
fprintf(stderr, "%s: session box is outside-right of the bounding box - width gets adapted to [%d]\n", __func__, *w);
#endif
}
/*
* ###### The Y-Coordinate ######
*/
/* check if outside-above of bounding box */
if (by1 == bby1 && ay1 < bby1)
{
*h += bby1 - ay1;
*y = 0;
#ifdef DEBUG
fprintf(stderr, "%s: session box is outside-above of the bounding box - height gets adapted to [%d]\n", __func__, *h);
#endif
}
/* check if outside-below of bounding box */
if (by1 + bh == bby2 && ay1 + ah > bby2)
{
*h += ay1 + ah - bby2;
#ifdef DEBUG
fprintf(stderr, "%s: session box is outside-below of the bounding box - height gets adapted to [%d]\n", __func__, *h);
#endif
}
}
return result;
}
#endif
RRModePtr nxagentRRCustomMode = NULL;
/*
This is basically the code that was used on screen resize before
xinerama was implemented. We need it as fallback if the user
disables xinerama
*/
void nxagentAdjustCustomMode(ScreenPtr pScreen)
{
rrScrPrivPtr pScrPriv = rrGetScrPriv(pScreen);
if (pScrPriv)
{
RROutputPtr output = RRFirstOutput(pScreen);
if (output && output -> crtc)
{
const int refresh = 60;
int width = nxagentOption(Width);
int height = nxagentOption(Height);
RRCrtcPtr crtc = output -> crtc;
for (int c = 0; c < pScrPriv -> numCrtcs; c++)
{
RRCrtcSet(pScrPriv -> crtcs[c], NULL, 0, 0, RR_Rotate_0, 0, NULL);
}
char name[100];
sprintf(name, "%dx%d", width, height);
xRRModeInfo modeInfo = {
.width = width,
.height = height,
.hTotal = width,
.vTotal = height,
.dotClock = ((CARD32) width * (CARD32) height *
(CARD32) refresh),
.nameLength = strlen(name)
};
if (nxagentRRCustomMode != NULL)
{
RROutputDeleteUserMode(output, nxagentRRCustomMode);
FreeResource(nxagentRRCustomMode -> mode.id, 0);
if (crtc != NULL && crtc -> mode == nxagentRRCustomMode)
{
RRCrtcSet(crtc, NULL, 0, 0, RR_Rotate_0, 0, NULL);
}
#ifdef TEST
fprintf(stderr, "%s: Going to destroy mode [%p] with refcnt [%d].\n",
__func__, (void *)nxagentRRCustomMode, nxagentRRCustomMode->refcnt);
#endif
RRModeDestroy(nxagentRRCustomMode);
}
nxagentRRCustomMode = RRModeGet(&modeInfo, name);
RROutputAddUserMode(output, nxagentRRCustomMode);
RRCrtcSet(crtc, nxagentRRCustomMode, 0, 0, RR_Rotate_0, 1, &output);
RROutputChanged(output, 1);
}
pScrPriv -> lastSetTime = currentTime;
pScrPriv->changed = 1;
pScrPriv->configChanged = 1;
} /* if (pScrPriv) */
RRScreenSizeNotify(pScreen);
}
int nxagentChangeScreenConfig(int screen, int width, int height, Bool doresize)
{
#ifdef DEBUG
fprintf(stderr, "%s: called for screen [%d], width [%d] height [%d] doresize [%d]\n", __func__, screen, width, height, doresize);
#endif
#ifdef TEST
fprintf(stderr, "%s: screenInfo.screens[%d]->root [%p]\n", __func__, screen, (void *) screenInfo.screens[screen]);
#endif
if (screenInfo.screens[screen]->root == NULL)
{
return 0;
}
UpdateCurrentTime();
#ifdef DEBUG
if (nxagentGrabServerInfo.grabstate == SERVER_GRABBED)
fprintf(stderr, "%s: grabstate [SERVER_GRABBED], client [%p]\n", __func__, (void *) nxagentGrabServerInfo.client);
else if (nxagentGrabServerInfo.grabstate == SERVER_UNGRABBED)
fprintf(stderr, "%s: grabstate [SERVER_UNGRABBED], client [%p]\n", __func__, (void *) nxagentGrabServerInfo.client);
else if (nxagentGrabServerInfo.grabstate == CLIENT_PERVIOUS)
fprintf(stderr, "%s: grabstate [CLIENT_PERVIOUS], client [%p]\n", __func__, (void *) nxagentGrabServerInfo.client);
else if (nxagentGrabServerInfo.grabstate == CLIENT_IMPERVIOUS)
fprintf(stderr, "%s: grabstate [CLIENT_IMPERVIOUS], client [%p]\n", __func__, (void *) nxagentGrabServerInfo.client);
else
fprintf(stderr, "%s: grabstate [UNKNOWN], client [%p]\n", __func__, (void *) nxagentGrabServerInfo.client);
#endif
if (nxagentGrabServerInfo.grabstate == SERVER_GRABBED)
{
/*
* If any client grabbed the server it won't expect screen
* configuration changes until it releases the grab. That could
* lead to an X error because available modes are changed
* in the meantime.
*/
#ifdef TEST
fprintf(stderr, "%s: Cancel with grabbed server (grab held by [%p]).\n", __func__, (void *) nxagentGrabServerInfo.client);
#endif
return 0;
}
ScreenPtr pScreen = screenInfo.screens[screen] -> root -> drawable.pScreen;
#ifdef TEST
fprintf(stderr, "%s: Changing config to %d x %d\n", __func__, width, height);
#endif
int r = nxagentResizeScreen(pScreen, width, height, 0, 0, doresize);
if (r != 0)
{
if (nxagentOption(Xinerama) && (noRRXineramaExtension == FALSE))
{
nxagentAdjustRandRXinerama(pScreen);
}
else
{
#ifdef DEBUG
fprintf(stderr, "%s: Xinerama is disabled\n", __func__);
#endif
nxagentAdjustCustomMode(pScreen);
}
}
#ifdef DEBUG
fprintf(stderr, "%s: current geometry: %d,%d %dx%d\n", __func__, nxagentOption(X), nxagentOption(Y), nxagentOption(Width), nxagentOption(Height));
fprintf(stderr, "%s: returning [%d]\n", __func__, r);
#endif
return r;
}
/*
Destroy an output after removing it from any crtc that might reference it
*/
void nxagentDropOutput(RROutputPtr o)
{
RRCrtcPtr c = o->crtc;
if (c)
{
for (int i = 0; i < c->numOutputs; i++)
{
if (c->outputs[i] == o)
{
#ifdef DEBUG
fprintf(stderr, "%s: output [%s] is in use by crtc [%p], removing it from there\n", __func__, o->name, c);
#endif
RRCrtcSet(c, NULL, 0, 0, RR_Rotate_0, 0, NULL);
}
}
}
#ifdef DEBUG
fprintf(stderr, "%s: destroying output [%s]\n", __func__, o->name);
#endif
RROutputDestroy(o);
}
int nxagentAdjustRandRXinerama(ScreenPtr pScreen)
{
int width = nxagentOption(Width);
int height = nxagentOption(Height);
rrScrPrivPtr pScrPriv = rrGetScrPriv(pScreen);
if (pScrPriv)
{
int number = 0;
XineramaScreenInfo *screeninfo = XineramaQueryScreens(nxagentDisplay, &number);
if (number)
{
#ifdef DEBUG
fprintf(stderr, "%s: XineramaQueryScreens() returned [%d] screens:\n", __func__, number);
for (int i = 0; i < number; i++) {
fprintf(stderr, "%s: screen_number [%d] x_org [%d] y_org [%d] width [%d] height [%d]\n", __func__, screeninfo[i].screen_number, screeninfo[i].x_org, screeninfo[i].y_org, screeninfo[i].width, screeninfo[i].height);
}
#endif
}
else
{
#ifdef DEBUG
fprintf(stderr, "%s: XineramaQueryScreens() failed - continuing without Xinerama\n", __func__);
#endif
}
/*
* if there's no xinerama on the real server or xinerama is
* disabled in nxagent we only report one big screen. Clients
* still see xinerama enabled but it will report only one (big)
* screen. This is consistent with the way rrxinerama always
* behaved. The single PanoramiX/Xinerama extension however
* disables xinerama if only one screen exists.
*/
if (number == 0)
{
#ifdef DEBUG
fprintf(stderr, "%s: faking xinerama\n", __func__);
#endif
number = 1;
SAFE_free(screeninfo);
if (!(screeninfo = malloc(sizeof(XineramaScreenInfo))))
{
return FALSE;
}
/* fake a xinerama screeninfo that covers the whole screen */
screeninfo->screen_number = 0;
screeninfo->x_org = nxagentOption(X);
screeninfo->y_org = nxagentOption(Y);
screeninfo->width = nxagentOption(Width);
screeninfo->height = nxagentOption(Height);
}
#ifdef DEBUG
fprintf(stderr, "%s: numCrtcs [%d], numOutputs [%d]\n", __func__, pScrPriv->numCrtcs, pScrPriv->numOutputs);
{
/*
* Convert old RANDR 1.0 data (if any) to current structure. This
* is needed once at the first run of this function. If we don't
* do this here it will be done implicitly later and add mode(s) to
* our crtc(s)!
*/
Bool rrgetinfo = RRGetInfo(pScreen, FALSE);
fprintf(stderr, "%s: RRGetInfo returned [%d]\n", __func__, rrgetinfo);
}
#else
/* we are not interested in the return code */
RRGetInfo(pScreen, FALSE);
#endif
#ifndef NXAGENT_RANDR_XINERAMA_CLIPPING
/* calculate bounding box (outer edges) */
int bbx2, bbx1, bby1, bby2;
bbx2 = bby2 = 0;
bbx1 = bby1 = INT_MAX;
for (int i = 0; i < number; i++)
{
bbx2 = max(bbx2, screeninfo[i].x_org + screeninfo[i].width);
bby2 = max(bby2, screeninfo[i].y_org + screeninfo[i].height);
bbx1 = min(bbx1, screeninfo[i].x_org);
bby1 = min(bby1, screeninfo[i].y_org);
}
#ifdef DEBUG
fprintf(stderr, "%s: bounding box: left [%d] right [%d] top [%d] bottom [%d]\n", __func__, bbx1, bbx2, bby1, bby2);
#endif
#endif
#ifdef DEBUG
fprintf(stderr, "%s: numCrtcs [%d], numOutputs [%d]\n", __func__, pScrPriv->numCrtcs, pScrPriv->numOutputs);
#endif
/* adjust the number of CRTCs to match the number of reported
xinerama screens on the real server */
while (number != pScrPriv->numCrtcs)
{
if (number < pScrPriv->numCrtcs)
{
#ifdef DEBUG
fprintf(stderr, "%s: destroying crtc\n", __func__);
#endif
/* first reset the crtc to free possible outputs, then destroy the crtc */
RRCrtcSet(pScrPriv->crtcs[pScrPriv->numCrtcs - 1], NULL, 0, 0, RR_Rotate_0, 0, NULL);
RRCrtcDestroy(pScrPriv->crtcs[pScrPriv->numCrtcs - 1]);
}
else
{
#ifdef DEBUG
fprintf(stderr, "%s: adding crtc\n", __func__);
#endif
RRCrtcCreate(pScreen, NULL);
}
}
#ifdef DEBUG
fprintf(stderr, "%s: numCrtcs [%d], numOutputs [%d]\n", __func__, pScrPriv->numCrtcs, pScrPriv->numOutputs);
#endif
/* set gamma. Currently the only reason for doing this is
preventing the xrandr command from complaining about missing
gamma. */
for (int i = 0; i < pScrPriv->numCrtcs; i++)
{
if (pScrPriv->crtcs[i]->gammaSize == 0)
{
CARD16 gamma = 0;
RRCrtcGammaSetSize(pScrPriv->crtcs[i], 1);
RRCrtcGammaSet(pScrPriv->crtcs[i], &gamma, &gamma, &gamma);
RRCrtcGammaNotify(pScrPriv->crtcs[i]);
}
}
/* delete superfluous non-NX outputs */
for (int i = pScrPriv->numOutputs - 1; i >= 0; i--)
if (strncmp(pScrPriv->outputs[i]->name, "NX", 2))
nxagentDropOutput(pScrPriv->outputs[i]);
/* at this stage only NX outputs are left - we delete the superfluous ones */
for (int i = pScrPriv->numOutputs - 1; i >= number; i--)
nxagentDropOutput(pScrPriv->outputs[i]);
/* add and init outputs */
for (int i = 0; i < number; i++)
{
RROutputPtr output;
if (i >= pScrPriv->numOutputs)
{
char name[100];
sprintf(name, "NX%d", i+1);
output = RROutputCreate(pScreen, name, strlen(name), NULL);
/* will be done later
RROutputSetConnection(output, RR_Disconnected);
*/
#ifdef DEBUG
fprintf(stderr, "%s: created new output [%s]\n", __func__, name);
#endif
}
else
{
output = pScrPriv->outputs[i];
}
#ifdef DEBUG
fprintf(stderr, "%s: adjusting output [%s]\n", __func__, output->name);
#endif
RROutputSetCrtcs(output, &(pScrPriv->crtcs[i]), 1);
/* FIXME: Isn't there a function for setting this? */
output->crtc = pScrPriv->crtcs[i];
/* FIXME: get SubPixelOrder from real X server */
RROutputSetSubpixelOrder(output, SubPixelUnknown);
/* FIXME: What is the correct physical size here? */
RROutputSetPhysicalSize(output, 0, 0);
}
for (int i = 0; i < pScrPriv->numOutputs; i++)
{
int new_x = 0;
int new_y = 0;
unsigned int new_w = 0;
unsigned int new_h = 0;
/* if there's no intersection disconnect the output */
#ifdef NXAGENT_RANDR_XINERAMA_CLIPPING
Bool disable_output = !intersect(nxagentOption(X), nxagentOption(Y),
width, height,
screeninfo[i].x_org, screeninfo[i].y_org,
screeninfo[i].width, screeninfo[i].height,
&new_x, &new_y, &new_w, &new_h);
#else
Bool disable_output = !intersect_bb(nxagentOption(X), nxagentOption(Y),
width, height,
screeninfo[i].x_org, screeninfo[i].y_org,
screeninfo[i].width, screeninfo[i].height,
bbx1, bby1, bbx2, bby2,
&new_x, &new_y, &new_w, &new_h);
#endif
/* save previous mode */
RRModePtr prevmode = pScrPriv->crtcs[i]->mode;
#ifdef DEBUG
if (prevmode)
{
fprintf(stderr, "%s: output [%d] name [%s]: prevmode [%s] ([%p]) refcnt [%d]\n", __func__, i, pScrPriv->outputs[i]->name, prevmode->name, (void *)prevmode, prevmode->refcnt);
}
else
{
fprintf(stderr, "%s: output [%d] name [%s]: no prevmode\n", __func__, i, pScrPriv->outputs[i]->name);
}
#endif
RROutputSetCrtcs(pScrPriv->outputs[i], &(pScrPriv->crtcs[i]), 1);
if (disable_output)
{
#ifdef DEBUG
fprintf(stderr, "%s: output [%d] name [%s]: no (valid) intersection - disconnecting\n", __func__, i, pScrPriv->outputs[i]->name);
#endif
RROutputSetConnection(pScrPriv->outputs[i], RR_Disconnected);
/*
* Tests revealed that some window managers (e.g. LXDE) also
* take disconnected outputs into account when calculating
* stuff like wallpaper tile size and maximum window
* size. This is problematic when a disconnected output is
* smaller than any of the connected ones. Solution: unset the
* mode of the output's crtc. This also leads to xinerama not
* showing the disconnected head anymore.
*/
if (prevmode)
{
#ifdef DEBUG
fprintf(stderr, "%s: removing mode from output [%d] name [%s]\n", __func__, i, pScrPriv->outputs[i]->name);
#endif
RROutputSetModes(pScrPriv->outputs[i], NULL, 0, 0);
#ifdef DEBUG
fprintf(stderr, "%s: removing mode from ctrc [%d]\n", __func__, i);
#endif
RRCrtcSet(pScrPriv->crtcs[i], NULL, 0, 0, RR_Rotate_0, 1, &(pScrPriv->outputs[i]));
}
}
else
{
#ifdef DEBUG
fprintf(stderr, "%s: output [%d] name [%s]: intersection is x [%d] y [%d] width [%d] height [%d]\n", __func__, i, pScrPriv->outputs[i]->name, new_x, new_y, new_w, new_h);
#endif
RROutputSetConnection(pScrPriv->outputs[i], RR_Connected);
/* avoid collisions with pre-existing default modes by using a
separate namespace. If we'd simply use XxY we could not
distinguish between pre-existing modes which should stay
and our own modes that should be removed after use. */
char name[100];
sprintf(name, "%s%dx%d", QUOTE(NXAGENT_RANDR_MODE_PREFIX), new_w, new_h);
const int refresh = 60;
xRRModeInfo modeInfo = {
.width = new_w,
.height = new_h,
.hTotal = new_w,
.vTotal = new_h,
.dotClock = ((CARD32) new_w * (CARD32) new_h * (CARD32) refresh),
.nameLength = strlen(name)
};
RRModePtr mymode = RRModeGet(&modeInfo, name);
#ifdef DEBUG
if (mymode)
{
fprintf(stderr, "%s: output [%d] name [%s]: mode [%s] ([%p]) created/received, refcnt [%d]\n", __func__, i, pScrPriv->outputs[i]->name, name, (void *) mymode, mymode->refcnt);
}
else
{
/* FIXME: what is the correct behaviour in this case? */
fprintf(stderr, "%s: output [%d] name [%s]: mode [%s] creation failed!\n", __func__, i, pScrPriv->outputs[i]->name, name);
}
#endif
if (prevmode && mymode == prevmode)
{
#ifdef DEBUG
fprintf(stderr, "%s: mymode [%s] ([%p]) == prevmode [%s] ([%p])\n", __func__, mymode->name, (void *) mymode, prevmode->name, (void *)prevmode);
#endif
/* if they are the same RRModeGet() has increased the
refcnt by 1. We decrease it again by calling only
RRModeDestroy() and forget about prevmode */
RRModeDestroy(mymode);
}
else
{
#ifdef DEBUG
fprintf(stderr, "%s: setting mode [%s] ([%p]) refcnt [%d] for output %d [%s]\n", __func__, mymode->name, (void *) mymode, mymode->refcnt, i, pScrPriv->outputs[i]->name);
#endif
RROutputSetModes(pScrPriv->outputs[i], &mymode, 1, 0);
}
#ifdef DEBUG
fprintf(stderr, "%s: setting mode [%s] ([%p]) refcnt [%d] for crtc %d\n", __func__, mymode->name, (void *) mymode, mymode->refcnt, i);
#endif
RRCrtcSet(pScrPriv->crtcs[i], mymode, new_x, new_y, RR_Rotate_0, 1, &(pScrPriv->outputs[i]));
} /* if disable_output */
/* throw away the mode if otherwise unused. We do not need it
anymore. We call FreeResource() to ensure the system will not
try to free it again on shutdown */
if (prevmode && prevmode->refcnt == 1)
{
#ifdef DEBUG
fprintf(stderr, "%s: destroying prevmode [%s]\n", __func__, prevmode->name);
#endif
FreeResource(prevmode->mode.id, 0);
}
RROutputChanged(pScrPriv->outputs[i], TRUE);
RRCrtcChanged(pScrPriv->crtcs[i], TRUE);
}
/* release allocated memory */
SAFE_free(screeninfo);
#ifdef DEBUG
for (int i = 0; i < pScrPriv->numCrtcs; i++)
{
RRModePtr mode = pScrPriv->crtcs[i]->mode;
if (mode) {
fprintf(stderr, "%s: crtc [%d] ([%p]) has mode [%s] ([%p]), refcnt [%d] and [%d] outputs:\n", __func__, i, (void *) pScrPriv->crtcs[i], pScrPriv->crtcs[i]->mode->name, (void *)pScrPriv->crtcs[i]->mode, pScrPriv->crtcs[i]->mode->refcnt, pScrPriv->crtcs[i]->numOutputs);
}
else
{
fprintf(stderr, "%s: crtc [%d] ([%p]) has no mode and [%d] outputs:\n", __func__, i, (void *) pScrPriv->crtcs[i], pScrPriv->crtcs[i]->numOutputs);
}
if (pScrPriv->crtcs[i]->numOutputs > 0)
for (int j = 0; j < pScrPriv->crtcs[i]->numOutputs; j++)
fprintf(stderr, "%s: output [%d] name [%s]->crtc=[%p]\n", __func__, j, pScrPriv->crtcs[i]->outputs[j]->name, (void *)pScrPriv->crtcs[i]->outputs[j]->crtc);
}
#endif
pScrPriv -> lastSetTime = currentTime;
pScrPriv->changed = TRUE;
pScrPriv->configChanged = TRUE;
}
/*
* Adjust screen size according the newly set modes.
* Not calling this function leads to the initial screen size left in place,
* which is not what we want in case the window is resizable.
*/
RRScreenSizeNotify(pScreen);
/* FIXME: adjust maximum screen size according to remote randr/xinerama setup */
#ifdef DEBUG
fprintf(stderr, "%s: Min %dx%d, Max %dx%d \n", __func__, pScrPriv->minWidth, pScrPriv->minHeight, pScrPriv->maxWidth, pScrPriv->maxHeight);
#endif
return TRUE;
}
void nxagentSaveAreas(PixmapPtr pPixmap, RegionPtr prgnSave, int xorg, int yorg, WindowPtr pWin)
{
miBSWindowPtr pBackingStore = (miBSWindowPtr) pWin -> backStorage;
PixmapPtr pVirtualPixmap = nxagentVirtualPixmap(pPixmap);
nxagentPrivPixmapPtr pPrivPixmap = nxagentPixmapPriv(pPixmap);
pPrivPixmap -> isBackingPixmap = 1;
fbCopyWindowProc(&pWin -> drawable, &pVirtualPixmap -> drawable, 0, RegionRects(prgnSave),
RegionNumRects(prgnSave), xorg, yorg, FALSE, FALSE, 0, 0);
XGCValues values = {.subwindow_mode = IncludeInferiors};
XlibGC gc = XCreateGC(nxagentDisplay, nxagentWindow(screenInfo.screens[0]->root), GCSubwindowMode, &values);
/*
* Initialize to the corrupted region.
* Coordinates are relative to the window.
*/
RegionRec cleanRegion;
RegionInit(&cleanRegion, NullBox, 1);
RegionCopy(&cleanRegion, nxagentCorruptedRegion((DrawablePtr) pWin));
/*
* Subtract the corrupted region from the saved region.
*/
RegionSubtract(&pBackingStore -> SavedRegion, &pBackingStore -> SavedRegion, &cleanRegion);
/*
* Translate the corrupted region. Coordinates
* are relative to the backing store pixmap.
*/
RegionTranslate(&cleanRegion, -pBackingStore -> x, -pBackingStore -> y);
/*
* Compute the clean region to be saved: subtract
* the corrupted region from the region to be saved.
*/
RegionSubtract(&cleanRegion, prgnSave, &cleanRegion);
int nRects = RegionNumRects(&cleanRegion);
int size = nRects * sizeof(XRectangle);
XRectangle *pRects = (XRectangle *) malloc(size);
if (!pRects)
{
#ifdef WARNING
fprintf(stderr, "Could not allocate pRects\n");
#endif
return;
}
BoxPtr pBox = RegionRects(&cleanRegion);
for (int i = nRects; i-- > 0;)
{
pRects[i].x = pBox[i].x1;
pRects[i].y = pBox[i].y1;
pRects[i].width = pBox[i].x2 - pBox[i].x1;
pRects[i].height = pBox[i].y2 - pBox[i].y1;
}
XSetClipRectangles(nxagentDisplay, gc, 0, 0, pRects, nRects, Unsorted);
SAFE_free(pRects);
BoxRec extents = *RegionExtents(&cleanRegion);
RegionUninit(&cleanRegion);
int xDst = extents.x1;
int yDst = extents.y1;
/*
* Left here the wrong solution. The window could be not
* configured yet on the real X, whilst the x and y in the
* WindowRec are the new coordinates. The right solution
* is the other, as it is independent from the window
* coordinates.
*
* xSrc = xDst + xorg - pWin -> drawable.x;
* ySrc = yDst + yorg - pWin -> drawable.y;
*/
int xSrc = xDst + pBackingStore -> x;
int ySrc = yDst + pBackingStore -> y;
int w = extents.x2 - extents.x1;
int h = extents.y2 - extents.y1;
XCopyArea(nxagentDisplay, nxagentWindow(pWin), nxagentPixmap(pPixmap), gc,
xSrc, ySrc, w, h, xDst, yDst);
nxagentAddItemBSPixmapList(nxagentPixmap(pPixmap), pPixmap, pWin,
pBackingStore -> x, pBackingStore -> y);
#ifdef TEST
fprintf(stderr,"%s: Added pixmap [%p] with id [%d] on window [%p] to BSPixmapList.\n", __func__,
(void *) pPixmap, nxagentPixmap(pPixmap), (void *) pWin);
#endif
XFreeGC(nxagentDisplay, gc);
return;
}
void nxagentRestoreAreas(PixmapPtr pPixmap, RegionPtr prgnRestore, int xorg,
int yorg, WindowPtr pWin)
{
/*
* Limit the area to restore to the
* root window size.
*/
RegionIntersect(prgnRestore, prgnRestore,
&pWin -> drawable.pScreen -> root -> winSize);
miBSWindowPtr pBackingStore = (miBSWindowPtr) pWin -> backStorage;
PixmapPtr pVirtualPixmap = nxagentVirtualPixmap(pPixmap);
fbCopyWindowProc(&pVirtualPixmap -> drawable, &pWin -> drawable, 0, RegionRects(prgnRestore),
RegionNumRects(prgnRestore), -xorg, -yorg, FALSE, FALSE, 0, 0);
XGCValues values = {.subwindow_mode = ClipByChildren};
XlibGC gc = XCreateGC(nxagentDisplay, nxagentWindow(screenInfo.screens[0]->root), GCSubwindowMode, &values);
/*
* Translate the reference point to the origin of the window.
*/
RegionTranslate(prgnRestore,
-pWin -> drawable.x - pWin -> borderWidth,
-pWin -> drawable.y - pWin -> borderWidth);
RegionPtr clipRegion = prgnRestore;
if (nxagentDrawableStatus((DrawablePtr) pPixmap) == NotSynchronized)
{
clipRegion = RegionCreate(NullBox, 1);
RegionCopy(clipRegion,
nxagentCorruptedRegion((DrawablePtr) pPixmap));
/*
* Translate the reference point to the origin of the window.
*/
RegionTranslate(clipRegion,
pBackingStore -> x, pBackingStore -> y);
RegionIntersect(clipRegion, prgnRestore, clipRegion);
/*
* Subtract the corrupted region from the saved areas.
* miBSRestoreAreas will return the exposure region.
*/
RegionSubtract(&pBackingStore->SavedRegion,
&pBackingStore->SavedRegion, clipRegion);
/*
* Store the corrupted region to send expose later.
*/
if (nxagentRemoteExposeRegion != NULL)
{
RegionTranslate(clipRegion, pWin -> drawable.x, pWin -> drawable.y);
RegionUnion(nxagentRemoteExposeRegion, nxagentRemoteExposeRegion, clipRegion);
RegionTranslate(clipRegion, -pWin -> drawable.x, -pWin -> drawable.y);
}
/*
* Compute the region to be restored.
*/
RegionSubtract(clipRegion, prgnRestore, clipRegion);
}
int nRects = RegionNumRects(clipRegion);
int size = nRects * sizeof(XRectangle);
XRectangle *pRects = (XRectangle *) malloc(size);
if (!pRects)
{
#ifdef WARNING
fprintf(stderr, "Could not allocate pRects\n");
#endif
return;
}
BoxPtr pBox = RegionRects(clipRegion);
for (int i = nRects; i-- > 0;)
{
pRects[i].x = pBox[i].x1;
pRects[i].y = pBox[i].y1;
pRects[i].width = pBox[i].x2 - pBox[i].x1;
pRects[i].height = pBox[i].y2 - pBox[i].y1;
}
XSetClipRectangles(nxagentDisplay, gc, 0, 0, pRects, nRects, Unsorted);
SAFE_free(pRects);
BoxRec extents = *RegionExtents(clipRegion);
int xDst = extents.x1;
int yDst = extents.y1;
int xSrc = xDst - xorg + pWin -> drawable.x;
int ySrc = yDst - yorg + pWin -> drawable.y;
int w = extents.x2 - extents.x1;
int h = extents.y2 - extents.y1;
nxagentFlushConfigureWindow();
XCopyArea(nxagentDisplay, nxagentPixmap(pPixmap), nxagentWindow(pWin), gc,
xSrc, ySrc, w, h, xDst, yDst);
XFreeGC(nxagentDisplay, gc);
if (clipRegion != NULL && clipRegion != prgnRestore)
{
RegionDestroy(clipRegion);
}
/*
* Restore the reference point to the origin of the screen.
*/
RegionTranslate(prgnRestore,
pWin -> drawable.x - pWin -> borderWidth,
pWin -> drawable.y + pWin -> borderWidth);
return;
}
void nxagentSetWMNormalHints(int screen, int width, int height)
{
XSizeHints* sizeHints = XAllocSizeHints();
if (!sizeHints)
return;
/*
* Change agent window size and size hints.
*/
sizeHints->flags = PPosition | PMinSize | PMaxSize;
sizeHints->x = nxagentOption(X);
sizeHints->y = nxagentOption(Y);
sizeHints->min_width = MIN_NXAGENT_WIDTH;
sizeHints->min_height = MIN_NXAGENT_HEIGHT;
sizeHints->width = width;
sizeHints->height = height;
if (nxagentOption(DesktopResize))
{
sizeHints->max_width = WidthOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
sizeHints->max_height = HeightOfScreen(DefaultScreenOfDisplay(nxagentDisplay));
}
else
{
sizeHints->max_width = nxagentOption(RootWidth);
sizeHints->max_height = nxagentOption(RootHeight);
}
if (nxagentUserGeometry.flag & XValue || nxagentUserGeometry.flag & YValue)
{
sizeHints->flags |= USPosition;
}
if (nxagentUserGeometry.flag & WidthValue || nxagentUserGeometry.flag & HeightValue)
{
sizeHints->flags |= USSize;
}
XSetWMNormalHints(nxagentDisplay, nxagentDefaultWindows[screen], sizeHints);
SAFE_XFree(sizeHints);
}
/*
set maxsize in WMNormalSizeHints
Note: this will _drop_ all existing hints since XSetWMNormalHints()
replaces any existing property
*/
void nxagentSetWMNormalHintsMaxsize(ScreenPtr pScreen, int maxwidth, int maxheight)
{
XSizeHints* sizeHints = XAllocSizeHints();
if (sizeHints)
{
sizeHints->flags = PMaxSize;
sizeHints->max_width = maxwidth;
sizeHints->max_height = maxheight;
XSetWMNormalHints(nxagentDisplay, nxagentDefaultWindows[pScreen->myNum],
sizeHints);
SAFE_XFree(sizeHints);
}
}
void nxagentShadowAdaptToRatio(void)
{
ScreenPtr pScreen = screenInfo.screens[0];
nxagentShadowSetRatio(nxagentOption(Width) * 1.0 / nxagentShadowWidth,
nxagentOption(Height) * 1.0 / nxagentShadowHeight);
nxagentShadowCreateMainWindow(pScreen, screenInfo.screens[0]->root, nxagentShadowWidth, nxagentShadowHeight);
nxagentSetWMNormalHintsMaxsize(pScreen,
WidthOfScreen(DefaultScreenOfDisplay(nxagentDisplay)),
HeightOfScreen(DefaultScreenOfDisplay(nxagentDisplay)));
BoxRec box = {
.x1 = 0,
.y1 = 0,
.x2 = nxagentShadowPixmapPtr -> drawable.width,
.y2 = nxagentShadowPixmapPtr -> drawable.height
};
RegionRec region;
RegionInit(&region, &box, 1);
nxagentMarkCorruptedRegion((DrawablePtr)nxagentShadowPixmapPtr, &region);
RegionUninit(&region);
}
void nxagentPrintGeometry(void)
{
for (int i = 0; i < screenInfo.numScreens; i++)
{
if (nxagentPrintGeometryFlags & (1 << i))
{
fprintf(stderr, "Info: Screen [%d] resized to geometry [%dx%d] "
"fullscreen [%d].\n", i, screenInfo.screens[i] -> width,
screenInfo.screens[i] -> height,
nxagentOption(Fullscreen));
}
}
nxagentPrintGeometryFlags = 0;
}
#ifdef DUMP
void nxagentShowPixmap(PixmapPtr pPixmap, int x, int y, int width, int height)
{
static int init = 1;
static Display *shadow = NULL;
static Window win = 0;
int depth = pPixmap -> drawable.depth;
/*
int pixmapWidth = pPixmap -> drawable.width;
int pixmapHeight = pPixmap -> drawable.height;
*/
unsigned int format = (depth == 1) ? XYPixmap : ZPixmap;
if (init)
{
shadow = XOpenDisplay("localhost:0");
if (shadow == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: WARNING! Shadow display not opened.\n", __func__);
#endif
return;
}
init = False;
}
if (win == 0)
{
win = XCreateSimpleWindow(shadow, DefaultRootWindow(shadow), 0, 0,
width, height, 0, 0xFFCC33, 0xFF);
XSelectInput(shadow, win, StructureNotifyMask);
XMapWindow(shadow, win);
for(;;)
{
XEvent e;
XNextEvent(shadow, &e);
if (e.type == MapNotify)
{
break;
}
}
}
else
{
XResizeWindow(nxagentDisplay, win, width, height);
XRaiseWindow(nxagentDisplay, win);
}
int length = nxagentImageLength(width, height, format, 0, depth);
char *data = malloc(length);
if (data == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: malloc failed.\n", __func__);
#endif
return;
}
/*
FIXME
XImage *image = XGetImage(nxagentDisplay, nxagentPixmap(pPixmap), x, y,
width, height, AllPlanes, format);
*/
XImage *image = XGetImage(nxagentDisplay, RootWindow(nxagentDisplay, 0), 0, 0,
width, height, AllPlanes, format);
if (image == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: XGetImage failed.\n", __func__);
#endif
SAFE_free(data);
return;
}
fbGetImage((DrawablePtr)pPixmap, 0, 0,
width, height, format, AllPlanes, data);
memcpy(image -> data, data, length);
XGCValues value = {
.foreground = 0xffffff,
.background = 0x000000,
.plane_mask = 0xffffff,
.fill_style = FillSolid
};
XlibGC gc = XCreateGC(shadow, win, GCBackground |
GCForeground | GCFillStyle | GCPlaneMask, &value);
XSync(shadow, 0);
NXCleanImage(image);
XPutImage(shadow, win, gc, image, 0, 0, 0, 0, width, height);
XFlush(shadow);
XFreeGC(shadow, gc);
if (image != NULL)
{
XDestroyImage(image);
}
SAFE_free(data);
/*
FIXME
if (win != NULL)
{
XDestroyWindow(shadow, win);
}
*/
}
void nxagentFbRestoreArea(PixmapPtr pPixmap, WindowPtr pWin, int xSrc, int ySrc, int width,
int height, int xDst, int yDst)
{
Display *shadow = nxagentDisplay;
int depth = pPixmap -> drawable.depth;
/*
int pixmapWidth = pPixmap -> drawable.width;
int pixmapHeight = pPixmap -> drawable.height;
*/
unsigned int format = (depth == 1) ? XYPixmap : ZPixmap;
int length = nxagentImageLength(width, height, format, 0, depth);
char *data = malloc(length);
if (data == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: malloc failed.\n", __func__);
#endif
return;
}
XImage *image = NULL;
/*
XImage *image = XGetImage(nxagentDisplay, nxagentPixmap(pPixmap), xSrc, ySrc,
width, height, AllPlanes, format);
*/
if (image == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: XGetImage failed.\n", __func__);
#endif
SAFE_free(data);
return;
}
fbGetImage((DrawablePtr)pPixmap, xSrc, ySrc,
width, height, format, AllPlanes, data);
/*
FIXME
*/
Visual *pVisual = nxagentImageVisual((DrawablePtr) pPixmap, depth);
if (pVisual == NULL)
{
#ifdef WARNING
fprintf(stderr, "%s: WARNING! Visual not found. Using default visual.\n", __func__);
#endif
pVisual = nxagentVisuals[nxagentDefaultVisualIndex].visual;
}
image = XCreateImage(nxagentDisplay, pVisual,
depth, format, 0, (char *) data,
width, height, BitmapPad(nxagentDisplay),
nxagentImagePad(width, format, 0, depth));
/*
FIXME
memcpy(image -> data, data, length);
*/
fprintf(stderr, "%s: Cleaning %d bytes of image.\n", __func__, length);
XGCValues value = {
.foreground = 0xffffff,
.background = 0x000000,
.plane_mask = 0xffffff,
.fill_style = FillSolid,
.function = GXcopy
};
XlibGC gc = XCreateGC(shadow, nxagentWindow(screenInfo.screens[0]->root), GCBackground |
GCForeground | GCFillStyle | GCPlaneMask | GCFunction, &value);
NXCleanImage(image);
XPutImage(shadow, nxagentWindow(pWin), gc, image, 0, 0, xDst, yDst, width, height);
/*
FIXME
*/
XFlush(shadow);
XFreeGC(shadow, gc);
if (image)
{
XDestroyImage(image);
}
/*
FIXME
SAFE_free(data);
*/
}
#endif