451 lines
13 KiB
C
451 lines
13 KiB
C
/*
|
|
Copyright 2011-2013 Jay Sorg
|
|
|
|
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.
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
RandR extension implementation
|
|
|
|
*/
|
|
|
|
#include "rdp.h"
|
|
#include "rdprandr.h"
|
|
|
|
#if 1
|
|
#define DEBUG_OUT(arg)
|
|
#else
|
|
#define DEBUG_OUT(arg) ErrorF arg
|
|
#endif
|
|
|
|
extern rdpScreenInfoRec g_rdpScreen; /* from rdpmain.c */
|
|
extern DeviceIntPtr g_pointer; /* in rdpmain.c */
|
|
extern DeviceIntPtr g_keyboard; /* in rdpmain.c */
|
|
extern ScreenPtr g_pScreen; /* in rdpmain.c */
|
|
extern WindowPtr g_invalidate_window; /* in rdpmain.c */
|
|
|
|
static XID g_wid = 0;
|
|
|
|
static int g_panning = 0;
|
|
|
|
#define LOG_LEVEL 1
|
|
#define LLOG(_level, _args) \
|
|
do { if (_level < LOG_LEVEL) { ErrorF _args ; } } while (0)
|
|
#define LLOGLN(_level, _args) \
|
|
do { if (_level < LOG_LEVEL) { ErrorF _args ; ErrorF("\n"); } } while (0)
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRRegisterSize(ScreenPtr pScreen, int width, int height)
|
|
{
|
|
int mmwidth;
|
|
int mmheight;
|
|
RRScreenSizePtr pSize;
|
|
|
|
ErrorF("rdpRRRegisterSize: width %d height %d\n", width, height);
|
|
mmwidth = PixelToMM(width);
|
|
mmheight = PixelToMM(height);
|
|
pSize = RRRegisterSize(pScreen, width, height, mmwidth, mmheight);
|
|
/* Tell RandR what the current config is */
|
|
RRSetCurrentConfig(pScreen, RR_Rotate_0, 0, pSize);
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRSetConfig(ScreenPtr pScreen, Rotation rotateKind, int rate,
|
|
RRScreenSizePtr pSize)
|
|
{
|
|
ErrorF("rdpRRSetConfig:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRGetInfo(ScreenPtr pScreen, Rotation *pRotations)
|
|
{
|
|
ErrorF("rdpRRGetInfo:\n");
|
|
*pRotations = RR_Rotate_0;
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* for lack of a better way, a window is created that covers the area and
|
|
when its deleted, it's invalidated */
|
|
static int
|
|
rdpInvalidateArea(ScreenPtr pScreen, int x, int y, int cx, int cy)
|
|
{
|
|
WindowPtr pWin;
|
|
int result;
|
|
int attri;
|
|
XID attributes[4];
|
|
Mask mask;
|
|
|
|
DEBUG_OUT(("rdpInvalidateArea:\n"));
|
|
mask = 0;
|
|
attri = 0;
|
|
attributes[attri++] = pScreen->blackPixel;
|
|
mask |= CWBackPixel;
|
|
attributes[attri++] = xTrue;
|
|
mask |= CWOverrideRedirect;
|
|
|
|
if (g_wid == 0)
|
|
{
|
|
g_wid = FakeClientID(0);
|
|
}
|
|
|
|
pWin = CreateWindow(g_wid, pScreen->root,
|
|
x, y, cx, cy, 0, InputOutput, mask,
|
|
attributes, 0, serverClient,
|
|
wVisual(pScreen->root), &result);
|
|
|
|
if (result == 0)
|
|
{
|
|
g_invalidate_window = pWin;
|
|
MapWindow(pWin, serverClient);
|
|
DeleteWindow(pWin, None);
|
|
g_invalidate_window = pWin;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRScreenSetSize(ScreenPtr pScreen, CARD16 width, CARD16 height,
|
|
CARD32 mmWidth, CARD32 mmHeight)
|
|
{
|
|
PixmapPtr screenPixmap;
|
|
BoxRec box;
|
|
|
|
ErrorF("rdpRRScreenSetSize: width %d height %d mmWidth %d mmHeight %d\n",
|
|
width, height, (int)mmWidth, (int)mmHeight);
|
|
|
|
if ((width < 1) || (height < 1))
|
|
{
|
|
ErrorF(" error width %d height %d\n", width, height);
|
|
return FALSE;
|
|
}
|
|
|
|
g_rdpScreen.width = width;
|
|
g_rdpScreen.height = height;
|
|
g_rdpScreen.paddedWidthInBytes =
|
|
PixmapBytePad(g_rdpScreen.width, g_rdpScreen.depth);
|
|
g_rdpScreen.sizeInBytes =
|
|
g_rdpScreen.paddedWidthInBytes * g_rdpScreen.height;
|
|
pScreen->width = width;
|
|
pScreen->height = height;
|
|
pScreen->mmWidth = mmWidth;
|
|
pScreen->mmHeight = mmHeight;
|
|
|
|
screenPixmap = pScreen->GetScreenPixmap(pScreen);
|
|
|
|
if (screenPixmap != 0)
|
|
{
|
|
ErrorF(" resizing screenPixmap [%p] to %dx%d, currently at %dx%d\n",
|
|
(void *)screenPixmap, width, height,
|
|
screenPixmap->drawable.width, screenPixmap->drawable.height);
|
|
if (g_rdpScreen.sizeInBytes > g_rdpScreen.sizeInBytesAlloc)
|
|
{
|
|
g_free(g_rdpScreen.pfbMemory);
|
|
g_rdpScreen.pfbMemory = (char*)g_malloc(g_rdpScreen.sizeInBytes, 1);
|
|
g_rdpScreen.sizeInBytesAlloc = g_rdpScreen.sizeInBytes;
|
|
ErrorF("new buffer size %d\n", g_rdpScreen.sizeInBytes);
|
|
}
|
|
pScreen->ModifyPixmapHeader(screenPixmap, width, height,
|
|
g_rdpScreen.depth, g_rdpScreen.bitsPerPixel,
|
|
g_rdpScreen.paddedWidthInBytes,
|
|
g_rdpScreen.pfbMemory);
|
|
ErrorF(" pixmap resized to %dx%d\n",
|
|
screenPixmap->drawable.width, screenPixmap->drawable.height);
|
|
}
|
|
|
|
DEBUG_OUT((" root window %p\n", (void *)pScreen->root));
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = width;
|
|
box.y2 = height;
|
|
RegionInit(&pScreen->root->winSize, &box, 1);
|
|
RegionInit(&pScreen->root->borderSize, &box, 1);
|
|
RegionReset(&pScreen->root->borderClip, &box);
|
|
RegionBreak(&pScreen->root->clipList);
|
|
pScreen->root->drawable.width = width;
|
|
pScreen->root->drawable.height = height;
|
|
ResizeChildrenWinSize(pScreen->root, 0, 0, 0, 0);
|
|
RRGetInfo(pScreen, 1);
|
|
RRScreenSizeNotify(pScreen);
|
|
rdpInvalidateArea(g_pScreen, 0, 0, g_rdpScreen.width, g_rdpScreen.height);
|
|
ErrorF(" screen resized to %dx%d\n",
|
|
pScreen->width, pScreen->height);
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRCrtcSet(ScreenPtr pScreen, RRCrtcPtr crtc, RRModePtr mode,
|
|
int x, int y, Rotation rotation, int numOutputs,
|
|
RROutputPtr *outputs)
|
|
{
|
|
ErrorF("rdpRRCrtcSet:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRCrtcSetGamma(ScreenPtr pScreen, RRCrtcPtr crtc)
|
|
{
|
|
ErrorF("rdpRRCrtcSetGamma:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRCrtcGetGamma(ScreenPtr pScreen, RRCrtcPtr crtc)
|
|
{
|
|
ErrorF("rdpRRCrtcGetGamma:\n");
|
|
crtc->gammaSize = 1;
|
|
if (crtc->gammaRed == NULL)
|
|
{
|
|
crtc->gammaRed = g_malloc(32, 1);
|
|
}
|
|
if (crtc->gammaBlue == NULL)
|
|
{
|
|
crtc->gammaBlue = g_malloc(32, 1);
|
|
}
|
|
if (crtc->gammaGreen == NULL)
|
|
{
|
|
crtc->gammaGreen = g_malloc(32, 1);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRROutputSetProperty(ScreenPtr pScreen, RROutputPtr output, Atom property,
|
|
RRPropertyValuePtr value)
|
|
{
|
|
ErrorF("rdpRROutputSetProperty:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRROutputValidateMode(ScreenPtr pScreen, RROutputPtr output,
|
|
RRModePtr mode)
|
|
{
|
|
ErrorF("rdpRROutputValidateMode:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpRRModeDestroy(ScreenPtr pScreen, RRModePtr mode)
|
|
{
|
|
ErrorF("rdpRRModeDestroy:\n");
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRROutputGetProperty(ScreenPtr pScreen, RROutputPtr output, Atom property)
|
|
{
|
|
ErrorF("rdpRROutputGetProperty:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRGetPanning(ScreenPtr pScrn, RRCrtcPtr crtc, BoxPtr totalArea,
|
|
BoxPtr trackingArea, INT16 *border)
|
|
{
|
|
ErrorF("rdpRRGetPanning:\n");
|
|
|
|
if (!g_panning)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (totalArea != 0)
|
|
{
|
|
totalArea->x1 = 0;
|
|
totalArea->y1 = 0;
|
|
totalArea->x2 = g_rdpScreen.width;
|
|
totalArea->y2 = g_rdpScreen.height;
|
|
}
|
|
|
|
if (trackingArea != 0)
|
|
{
|
|
trackingArea->x1 = 0;
|
|
trackingArea->y1 = 0;
|
|
trackingArea->x2 = g_rdpScreen.width;
|
|
trackingArea->y2 = g_rdpScreen.height;
|
|
}
|
|
|
|
if (border != 0)
|
|
{
|
|
border[0] = 0;
|
|
border[1] = 0;
|
|
border[2] = 0;
|
|
border[3] = 0;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpRRSetPanning(ScreenPtr pScrn, RRCrtcPtr crtc, BoxPtr totalArea,
|
|
BoxPtr trackingArea, INT16 *border)
|
|
{
|
|
ErrorF("rdpRRSetPanning:\n");
|
|
return TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static RROutputPtr
|
|
rdpRRAddOutput(const char *aname, int x, int y, int width, int height)
|
|
{
|
|
RRModePtr mode;
|
|
RRCrtcPtr crtc;
|
|
RROutputPtr output;
|
|
xRRModeInfo modeInfo;
|
|
char name[64];
|
|
|
|
sprintf (name, "%dx%d", width, height);
|
|
memset (&modeInfo, 0, sizeof(modeInfo));
|
|
modeInfo.width = width;
|
|
modeInfo.height = height;
|
|
modeInfo.nameLength = strlen(name);
|
|
mode = RRModeGet(&modeInfo, name);
|
|
if (mode == 0)
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RRModeGet failed"));
|
|
return 0;
|
|
}
|
|
|
|
crtc = RRCrtcCreate(g_pScreen, NULL);
|
|
if (crtc == 0)
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RRCrtcCreate failed"));
|
|
RRModeDestroy(mode);
|
|
return 0;
|
|
}
|
|
output = RROutputCreate(g_pScreen, aname, strlen(aname), NULL);
|
|
if (output == 0)
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RROutputCreate failed"));
|
|
RRCrtcDestroy(crtc);
|
|
RRModeDestroy(mode);
|
|
return 0;
|
|
}
|
|
if (!RROutputSetClones(output, NULL, 0))
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RROutputSetClones failed"));
|
|
}
|
|
if (!RROutputSetModes(output, &mode, 1, 0))
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RROutputSetModes failed"));
|
|
}
|
|
if (!RROutputSetCrtcs(output, &crtc, 1))
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RROutputSetCrtcs failed"));
|
|
}
|
|
if (!RROutputSetConnection(output, RR_Connected))
|
|
{
|
|
LLOGLN(0, ("rdpRRAddOutput: RROutputSetConnection failed"));
|
|
}
|
|
RRCrtcNotify(crtc, mode, x, y, RR_Rotate_0, NULL, 1, &output);
|
|
|
|
return output;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
RRSetPrimaryOutput(rrScrPrivPtr pScrPriv, RROutputPtr output)
|
|
{
|
|
if (pScrPriv->primaryOutput == output)
|
|
{
|
|
return;
|
|
}
|
|
/* clear the old primary */
|
|
if (pScrPriv->primaryOutput)
|
|
{
|
|
RROutputChanged(pScrPriv->primaryOutput, 0);
|
|
pScrPriv->primaryOutput = NULL;
|
|
}
|
|
/* set the new primary */
|
|
if (output)
|
|
{
|
|
pScrPriv->primaryOutput = output;
|
|
RROutputChanged(output, 0);
|
|
}
|
|
pScrPriv->layoutChanged = TRUE;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpRRSetRdpOutputs(void)
|
|
{
|
|
rrScrPrivPtr pRRScrPriv;
|
|
int index;
|
|
int width;
|
|
int height;
|
|
char text[256];
|
|
RROutputPtr output;
|
|
|
|
pRRScrPriv = rrGetScrPriv(g_pScreen);
|
|
|
|
LLOGLN(0, ("rdpRRSetRdpOutputs: numCrtcs %d", pRRScrPriv->numCrtcs));
|
|
while (pRRScrPriv->numCrtcs > 0)
|
|
{
|
|
RRCrtcDestroy(pRRScrPriv->crtcs[0]);
|
|
}
|
|
LLOGLN(0, ("rdpRRSetRdpOutputs: numOutputs %d", pRRScrPriv->numOutputs));
|
|
while (pRRScrPriv->numOutputs > 0)
|
|
{
|
|
RROutputDestroy(pRRScrPriv->outputs[0]);
|
|
}
|
|
|
|
if (g_rdpScreen.client_info.monitorCount == 0)
|
|
{
|
|
rdpRRAddOutput("rdp0", 0, 0, g_rdpScreen.width, g_rdpScreen.height);
|
|
}
|
|
else
|
|
{
|
|
for (index = 0; index < g_rdpScreen.client_info.monitorCount; index++)
|
|
{
|
|
snprintf(text, 255, "rdp%d", index);
|
|
width = g_rdpScreen.client_info.minfo[index].right - g_rdpScreen.client_info.minfo[index].left + 1;
|
|
height = g_rdpScreen.client_info.minfo[index].bottom - g_rdpScreen.client_info.minfo[index].top + 1;
|
|
output = rdpRRAddOutput(text,
|
|
g_rdpScreen.client_info.minfo[index].left,
|
|
g_rdpScreen.client_info.minfo[index].top,
|
|
width, height);
|
|
if ((output != 0) && (g_rdpScreen.client_info.minfo[index].is_primary))
|
|
{
|
|
RRSetPrimaryOutput(pRRScrPriv, output);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (index = 0; index < pRRScrPriv->numOutputs; index++)
|
|
{
|
|
RROutputSetCrtcs(pRRScrPriv->outputs[index], pRRScrPriv->crtcs,
|
|
pRRScrPriv->numCrtcs);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|