1083 lines
28 KiB
C
1083 lines
28 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* X11 GDI
|
|
*
|
|
* Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xutil.h>
|
|
|
|
#include <freerdp/gdi/gdi.h>
|
|
#include <freerdp/codec/rfx.h>
|
|
#include <freerdp/codec/nsc.h>
|
|
#include <freerdp/constants.h>
|
|
#include <freerdp/codec/color.h>
|
|
#include <freerdp/codec/bitmap.h>
|
|
|
|
#include "xf_gdi.h"
|
|
|
|
static UINT8 GDI_BS_HATCHED_PATTERNS[] =
|
|
{
|
|
0xFF, 0xFF, 0xFF, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, /* HS_HORIZONTAL */
|
|
0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7, 0xF7, /* HS_VERTICAL */
|
|
0xFE, 0xFD, 0xFB, 0xF7, 0xEF, 0xDF, 0xBF, 0x7F, /* HS_FDIAGONAL */
|
|
0x7F, 0xBF, 0xDF, 0xEF, 0xF7, 0xFB, 0xFD, 0xFE, /* HS_BDIAGONAL */
|
|
0xF7, 0xF7, 0xF7, 0x00, 0xF7, 0xF7, 0xF7, 0xF7, /* HS_CROSS */
|
|
0x7E, 0xBD, 0xDB, 0xE7, 0xE7, 0xDB, 0xBD, 0x7E /* HS_DIACROSS */
|
|
};
|
|
|
|
static const BYTE xf_rop2_table[] =
|
|
{
|
|
0,
|
|
GXclear, /* 0 */
|
|
GXnor, /* DPon */
|
|
GXandInverted, /* DPna */
|
|
GXcopyInverted, /* Pn */
|
|
GXandReverse, /* PDna */
|
|
GXinvert, /* Dn */
|
|
GXxor, /* DPx */
|
|
GXnand, /* DPan */
|
|
GXand, /* DPa */
|
|
GXequiv, /* DPxn */
|
|
GXnoop, /* D */
|
|
GXorInverted, /* DPno */
|
|
GXcopy, /* P */
|
|
GXorReverse, /* PDno */
|
|
GXor, /* DPo */
|
|
GXset /* 1 */
|
|
};
|
|
|
|
BOOL xf_set_rop2(xfContext* xfc, int rop2)
|
|
{
|
|
if ((rop2 < 0x01) || (rop2 > 0x10))
|
|
{
|
|
fprintf(stderr, "Unsupported ROP2: %d\n", rop2);
|
|
return FALSE;
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, xf_rop2_table[rop2]);
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL xf_set_rop3(xfContext* xfc, int rop3)
|
|
{
|
|
int function = -1;
|
|
|
|
switch (rop3)
|
|
{
|
|
case GDI_BLACKNESS:
|
|
function = GXclear;
|
|
break;
|
|
|
|
case GDI_DPon:
|
|
function = GXnor;
|
|
break;
|
|
|
|
case GDI_DPna:
|
|
function = GXandInverted;
|
|
break;
|
|
|
|
case GDI_Pn:
|
|
function = GXcopyInverted;
|
|
break;
|
|
|
|
case GDI_NOTSRCERASE:
|
|
function = GXnor;
|
|
break;
|
|
|
|
case GDI_DSna:
|
|
function = GXandInverted;
|
|
break;
|
|
|
|
case GDI_NOTSRCCOPY:
|
|
function = GXcopyInverted;
|
|
break;
|
|
|
|
case GDI_SRCERASE:
|
|
function = GXandReverse;
|
|
break;
|
|
|
|
case GDI_PDna:
|
|
function = GXandReverse;
|
|
break;
|
|
|
|
case GDI_DSTINVERT:
|
|
function = GXinvert;
|
|
break;
|
|
|
|
case GDI_PATINVERT:
|
|
function = GXxor;
|
|
break;
|
|
|
|
case GDI_DPan:
|
|
function = GXnand;
|
|
break;
|
|
|
|
case GDI_SRCINVERT:
|
|
function = GXxor;
|
|
break;
|
|
|
|
case GDI_DSan:
|
|
function = GXnand;
|
|
break;
|
|
|
|
case GDI_SRCAND:
|
|
function = GXand;
|
|
break;
|
|
|
|
case GDI_DSxn:
|
|
function = GXequiv;
|
|
break;
|
|
|
|
case GDI_DPa:
|
|
function = GXand;
|
|
break;
|
|
|
|
case GDI_PDxn:
|
|
function = GXequiv;
|
|
break;
|
|
|
|
case GDI_D:
|
|
function = GXnoop;
|
|
break;
|
|
|
|
case GDI_DPno:
|
|
function = GXorInverted;
|
|
break;
|
|
|
|
case GDI_MERGEPAINT:
|
|
function = GXorInverted;
|
|
break;
|
|
|
|
case GDI_SRCCOPY:
|
|
function = GXcopy;
|
|
break;
|
|
|
|
case GDI_SDno:
|
|
function = GXorReverse;
|
|
break;
|
|
|
|
case GDI_SRCPAINT:
|
|
function = GXor;
|
|
break;
|
|
|
|
case GDI_PATCOPY:
|
|
function = GXcopy;
|
|
break;
|
|
|
|
case GDI_PDno:
|
|
function = GXorReverse;
|
|
break;
|
|
|
|
case GDI_DPo:
|
|
function = GXor;
|
|
break;
|
|
|
|
case GDI_WHITENESS:
|
|
function = GXset;
|
|
break;
|
|
|
|
case GDI_PSDPxax:
|
|
function = GXand;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (function < 0)
|
|
{
|
|
fprintf(stderr, "Unsupported ROP3: 0x%08X\n", rop3);
|
|
XSetFunction(xfc->display, xfc->gc, GXclear);
|
|
return FALSE;
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, function);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Pixmap xf_brush_new(xfContext* xfc, int width, int height, int bpp, BYTE* data)
|
|
{
|
|
Pixmap bitmap;
|
|
BYTE* cdata;
|
|
XImage* image;
|
|
|
|
bitmap = XCreatePixmap(xfc->display, xfc->drawable, width, height, xfc->depth);
|
|
|
|
if (data != NULL)
|
|
{
|
|
GC gc;
|
|
|
|
cdata = freerdp_image_convert(data, NULL, width, height, bpp, xfc->bpp, xfc->clrconv);
|
|
|
|
image = XCreateImage(xfc->display, xfc->visual, xfc->depth,
|
|
ZPixmap, 0, (char*) cdata, width, height, xfc->scanline_pad, 0);
|
|
|
|
gc = XCreateGC(xfc->display, xfc->drawable, 0, NULL);
|
|
XPutImage(xfc->display, bitmap, gc, image, 0, 0, 0, 0, width, height);
|
|
XFree(image);
|
|
|
|
if (cdata != data)
|
|
free(cdata);
|
|
|
|
XFreeGC(xfc->display, gc);
|
|
}
|
|
|
|
return bitmap;
|
|
}
|
|
|
|
Pixmap xf_mono_bitmap_new(xfContext* xfc, int width, int height, BYTE* data)
|
|
{
|
|
int scanline;
|
|
XImage* image;
|
|
Pixmap bitmap;
|
|
|
|
scanline = (width + 7) / 8;
|
|
|
|
bitmap = XCreatePixmap(xfc->display, xfc->drawable, width, height, 1);
|
|
|
|
image = XCreateImage(xfc->display, xfc->visual, 1,
|
|
ZPixmap, 0, (char*) data, width, height, 8, scanline);
|
|
|
|
XPutImage(xfc->display, bitmap, xfc->gc_mono, image, 0, 0, 0, 0, width, height);
|
|
XFree(image);
|
|
|
|
return bitmap;
|
|
}
|
|
|
|
void xf_gdi_palette_update(rdpContext* context, PALETTE_UPDATE* palette)
|
|
{
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
CopyMemory(xfc->clrconv->palette, palette, sizeof(rdpPalette));
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_set_bounds(rdpContext* context, rdpBounds* bounds)
|
|
{
|
|
XRectangle clip;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
if (bounds != NULL)
|
|
{
|
|
clip.x = bounds->left;
|
|
clip.y = bounds->top;
|
|
clip.width = bounds->right - bounds->left + 1;
|
|
clip.height = bounds->bottom - bounds->top + 1;
|
|
XSetClipRectangles(xfc->display, xfc->gc, 0, 0, &clip, 1, YXBanded);
|
|
}
|
|
else
|
|
{
|
|
XSetClipMask(xfc->display, xfc->gc, None);
|
|
}
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_dstblt(rdpContext* context, DSTBLT_ORDER* dstblt)
|
|
{
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
xf_set_rop3(xfc, gdi_rop3_code(dstblt->bRop));
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
dstblt->nLeftRect, dstblt->nTopRect,
|
|
dstblt->nWidth, dstblt->nHeight);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, dstblt->nLeftRect, dstblt->nTopRect, dstblt->nWidth, dstblt->nHeight);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_patblt(rdpContext* context, PATBLT_ORDER* patblt)
|
|
{
|
|
Pixmap pattern;
|
|
rdpBrush* brush;
|
|
UINT32 foreColor;
|
|
UINT32 backColor;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
brush = &patblt->brush;
|
|
xf_set_rop3(xfc, gdi_rop3_code(patblt->bRop));
|
|
|
|
foreColor = freerdp_color_convert_var(patblt->foreColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
backColor = freerdp_color_convert_var(patblt->backColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
if (brush->style == GDI_BS_SOLID)
|
|
{
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, foreColor);
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
patblt->nLeftRect, patblt->nTopRect, patblt->nWidth, patblt->nHeight);
|
|
}
|
|
else if (brush->style == GDI_BS_HATCHED)
|
|
{
|
|
pattern = xf_mono_bitmap_new(xfc, 8, 8, GDI_BS_HATCHED_PATTERNS + 8 * brush->hatch);
|
|
|
|
XSetForeground(xfc->display, xfc->gc, backColor);
|
|
XSetBackground(xfc->display, xfc->gc, foreColor);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillOpaqueStippled);
|
|
XSetStipple(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
patblt->nLeftRect, patblt->nTopRect, patblt->nWidth, patblt->nHeight);
|
|
|
|
XFreePixmap(xfc->display, pattern);
|
|
}
|
|
else if (brush->style == GDI_BS_PATTERN)
|
|
{
|
|
if (brush->bpp > 1)
|
|
{
|
|
pattern = xf_brush_new(xfc, 8, 8, brush->bpp, brush->data);
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillTiled);
|
|
XSetTile(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
patblt->nLeftRect, patblt->nTopRect, patblt->nWidth, patblt->nHeight);
|
|
|
|
XSetTile(xfc->display, xfc->gc, xfc->primary);
|
|
|
|
XFreePixmap(xfc->display, pattern);
|
|
}
|
|
else
|
|
{
|
|
pattern = xf_mono_bitmap_new(xfc, 8, 8, brush->data);
|
|
|
|
XSetForeground(xfc->display, xfc->gc, backColor);
|
|
XSetBackground(xfc->display, xfc->gc, foreColor);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillOpaqueStippled);
|
|
XSetStipple(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
patblt->nLeftRect, patblt->nTopRect, patblt->nWidth, patblt->nHeight);
|
|
|
|
XFreePixmap(xfc->display, pattern);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "unimplemented brush style:%d\n", brush->style);
|
|
}
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, patblt->nLeftRect, patblt->nTopRect, patblt->nWidth, patblt->nHeight);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_scrblt(rdpContext* context, SCRBLT_ORDER* scrblt)
|
|
{
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
xf_set_rop3(xfc, gdi_rop3_code(scrblt->bRop));
|
|
|
|
XCopyArea(xfc->display, xfc->primary, xfc->drawing, xfc->gc, scrblt->nXSrc, scrblt->nYSrc,
|
|
scrblt->nWidth, scrblt->nHeight, scrblt->nLeftRect, scrblt->nTopRect);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, scrblt->nLeftRect, scrblt->nTopRect, scrblt->nWidth, scrblt->nHeight);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_opaque_rect(rdpContext* context, OPAQUE_RECT_ORDER* opaque_rect)
|
|
{
|
|
UINT32 color;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
color = freerdp_color_convert_var(opaque_rect->color, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, color);
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
opaque_rect->nLeftRect, opaque_rect->nTopRect,
|
|
opaque_rect->nWidth, opaque_rect->nHeight);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, opaque_rect->nLeftRect, opaque_rect->nTopRect,
|
|
opaque_rect->nWidth, opaque_rect->nHeight);
|
|
}
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_multi_opaque_rect(rdpContext* context, MULTI_OPAQUE_RECT_ORDER* multi_opaque_rect)
|
|
{
|
|
int i;
|
|
UINT32 color;
|
|
DELTA_RECT* rectangle;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
color = freerdp_color_convert_var(multi_opaque_rect->color, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, color);
|
|
|
|
for (i = 1; i < multi_opaque_rect->numRectangles + 1; i++)
|
|
{
|
|
rectangle = &multi_opaque_rect->rectangles[i];
|
|
|
|
XFillRectangle(xfc->display, xfc->drawing, xfc->gc,
|
|
rectangle->left, rectangle->top,
|
|
rectangle->width, rectangle->height);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, rectangle->left, rectangle->top, rectangle->width, rectangle->height);
|
|
}
|
|
}
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_draw_nine_grid(rdpContext* context, DRAW_NINE_GRID_ORDER* draw_nine_grid)
|
|
{
|
|
fprintf(stderr, "DrawNineGrid\n");
|
|
}
|
|
|
|
void xf_gdi_line_to(rdpContext* context, LINE_TO_ORDER* line_to)
|
|
{
|
|
UINT32 color;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
xf_set_rop2(xfc, line_to->bRop2);
|
|
color = freerdp_color_convert_var(line_to->penColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, color);
|
|
|
|
XDrawLine(xfc->display, xfc->drawing, xfc->gc,
|
|
line_to->nXStart, line_to->nYStart, line_to->nXEnd, line_to->nYEnd);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
int width, height;
|
|
|
|
width = line_to->nXStart - line_to->nXEnd;
|
|
height = line_to->nYStart - line_to->nYEnd;
|
|
|
|
if (width < 0)
|
|
width *= (-1);
|
|
|
|
if (height < 0)
|
|
height *= (-1);
|
|
|
|
gdi_InvalidateRegion(xfc->hdc, line_to->nXStart, line_to->nYStart, width, height);
|
|
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_polyline(rdpContext* context, POLYLINE_ORDER* polyline)
|
|
{
|
|
int i;
|
|
int x, y;
|
|
int x1, y1;
|
|
int x2, y2;
|
|
int npoints;
|
|
UINT32 color;
|
|
XPoint* points;
|
|
int width, height;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
xf_set_rop2(xfc, polyline->bRop2);
|
|
color = freerdp_color_convert_var(polyline->penColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, color);
|
|
|
|
npoints = polyline->numPoints + 1;
|
|
points = malloc(sizeof(XPoint) * npoints);
|
|
|
|
points[0].x = polyline->xStart;
|
|
points[0].y = polyline->yStart;
|
|
|
|
for (i = 0; i < polyline->numPoints; i++)
|
|
{
|
|
points[i + 1].x = polyline->points[i].x;
|
|
points[i + 1].y = polyline->points[i].y;
|
|
}
|
|
|
|
XDrawLines(xfc->display, xfc->drawing, xfc->gc, points, npoints, CoordModePrevious);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
x1 = points[0].x;
|
|
y1 = points[0].y;
|
|
|
|
for (i = 1; i < npoints; i++)
|
|
{
|
|
x2 = points[i].x + x1;
|
|
y2 = points[i].y + y1;
|
|
|
|
x = (x2 < x1) ? x2 : x1;
|
|
width = (x2 > x1) ? x2 - x1 : x1 - x2;
|
|
|
|
y = (y2 < y1) ? y2 : y1;
|
|
height = (y2 > y1) ? y2 - y1 : y1 - y2;
|
|
|
|
x1 = x2;
|
|
y1 = y2;
|
|
|
|
gdi_InvalidateRegion(xfc->hdc, x, y, width, height);
|
|
}
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
free(points);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_memblt(rdpContext* context, MEMBLT_ORDER* memblt)
|
|
{
|
|
xfBitmap* bitmap;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
bitmap = (xfBitmap*) memblt->bitmap;
|
|
xf_set_rop3(xfc, gdi_rop3_code(memblt->bRop));
|
|
|
|
XCopyArea(xfc->display, bitmap->pixmap, xfc->drawing, xfc->gc,
|
|
memblt->nXSrc, memblt->nYSrc, memblt->nWidth, memblt->nHeight,
|
|
memblt->nLeftRect, memblt->nTopRect);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, memblt->nLeftRect, memblt->nTopRect, memblt->nWidth, memblt->nHeight);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_mem3blt(rdpContext* context, MEM3BLT_ORDER* mem3blt)
|
|
{
|
|
rdpBrush* brush;
|
|
xfBitmap* bitmap;
|
|
UINT32 foreColor;
|
|
UINT32 backColor;
|
|
Pixmap pattern = 0;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
brush = &mem3blt->brush;
|
|
bitmap = (xfBitmap*) mem3blt->bitmap;
|
|
xf_set_rop3(xfc, gdi_rop3_code(mem3blt->bRop));
|
|
foreColor = freerdp_color_convert_var(mem3blt->foreColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
backColor = freerdp_color_convert_var(mem3blt->backColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
if (brush->style == GDI_BS_PATTERN)
|
|
{
|
|
if (brush->bpp > 1)
|
|
{
|
|
pattern = xf_brush_new(xfc, 8, 8, brush->bpp, brush->data);
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillTiled);
|
|
XSetTile(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
}
|
|
else
|
|
{
|
|
pattern = xf_mono_bitmap_new(xfc, 8, 8, brush->data);
|
|
|
|
XSetForeground(xfc->display, xfc->gc, backColor);
|
|
XSetBackground(xfc->display, xfc->gc, foreColor);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillOpaqueStippled);
|
|
XSetStipple(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
}
|
|
}
|
|
else if (brush->style == GDI_BS_SOLID)
|
|
{
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, backColor);
|
|
XSetBackground(xfc->display, xfc->gc, foreColor);
|
|
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Mem3Blt unimplemented brush style:%d\n", brush->style);
|
|
}
|
|
|
|
XCopyArea(xfc->display, bitmap->pixmap, xfc->drawing, xfc->gc,
|
|
mem3blt->nXSrc, mem3blt->nYSrc, mem3blt->nWidth, mem3blt->nHeight,
|
|
mem3blt->nLeftRect, mem3blt->nTopRect);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, mem3blt->nLeftRect, mem3blt->nTopRect, mem3blt->nWidth, mem3blt->nHeight);
|
|
}
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetTSOrigin(xfc->display, xfc->gc, 0, 0);
|
|
|
|
if (pattern != 0)
|
|
XFreePixmap(xfc->display, pattern);
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_polygon_sc(rdpContext* context, POLYGON_SC_ORDER* polygon_sc)
|
|
{
|
|
int i, npoints;
|
|
XPoint* points;
|
|
UINT32 brush_color;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
xf_set_rop2(xfc, polygon_sc->bRop2);
|
|
brush_color = freerdp_color_convert_var(polygon_sc->brushColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
npoints = polygon_sc->numPoints + 1;
|
|
points = malloc(sizeof(XPoint) * npoints);
|
|
|
|
points[0].x = polygon_sc->xStart;
|
|
points[0].y = polygon_sc->yStart;
|
|
|
|
for (i = 0; i < polygon_sc->numPoints; i++)
|
|
{
|
|
points[i + 1].x = polygon_sc->points[i].x;
|
|
points[i + 1].y = polygon_sc->points[i].y;
|
|
}
|
|
|
|
switch (polygon_sc->fillMode)
|
|
{
|
|
case 1: /* alternate */
|
|
XSetFillRule(xfc->display, xfc->gc, EvenOddRule);
|
|
break;
|
|
|
|
case 2: /* winding */
|
|
XSetFillRule(xfc->display, xfc->gc, WindingRule);
|
|
break;
|
|
|
|
default:
|
|
fprintf(stderr, "PolygonSC unknown fillMode: %d\n", polygon_sc->fillMode);
|
|
break;
|
|
}
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetForeground(xfc->display, xfc->gc, brush_color);
|
|
|
|
XFillPolygon(xfc->display, xfc->drawing, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
XFillPolygon(xfc->display, xfc->drawable, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
free(points);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_polygon_cb(rdpContext* context, POLYGON_CB_ORDER* polygon_cb)
|
|
{
|
|
int i, npoints;
|
|
XPoint* points;
|
|
Pixmap pattern;
|
|
rdpBrush* brush;
|
|
UINT32 foreColor;
|
|
UINT32 backColor;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
brush = &(polygon_cb->brush);
|
|
xf_set_rop2(xfc, polygon_cb->bRop2);
|
|
foreColor = freerdp_color_convert_var(polygon_cb->foreColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
backColor = freerdp_color_convert_var(polygon_cb->backColor, context->settings->ColorDepth, xfc->bpp, xfc->clrconv);
|
|
|
|
npoints = polygon_cb->numPoints + 1;
|
|
points = malloc(sizeof(XPoint) * npoints);
|
|
|
|
points[0].x = polygon_cb->xStart;
|
|
points[0].y = polygon_cb->yStart;
|
|
|
|
for (i = 0; i < polygon_cb->numPoints; i++)
|
|
{
|
|
points[i + 1].x = polygon_cb->points[i].x;
|
|
points[i + 1].y = polygon_cb->points[i].y;
|
|
}
|
|
|
|
switch (polygon_cb->fillMode)
|
|
{
|
|
case 1: /* alternate */
|
|
XSetFillRule(xfc->display, xfc->gc, EvenOddRule);
|
|
break;
|
|
|
|
case 2: /* winding */
|
|
XSetFillRule(xfc->display, xfc->gc, WindingRule);
|
|
break;
|
|
|
|
default:
|
|
fprintf(stderr, "PolygonCB unknown fillMode: %d\n", polygon_cb->fillMode);
|
|
break;
|
|
}
|
|
|
|
if (brush->style == GDI_BS_PATTERN)
|
|
{
|
|
if (brush->bpp > 1)
|
|
{
|
|
pattern = xf_brush_new(xfc, 8, 8, brush->bpp, brush->data);
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillTiled);
|
|
XSetTile(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
|
|
XFillPolygon(xfc->display, xfc->drawing, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
XFillPolygon(xfc->display, xfc->drawable, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
}
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetTSOrigin(xfc->display, xfc->gc, 0, 0);
|
|
XFreePixmap(xfc->display, pattern);
|
|
}
|
|
else
|
|
{
|
|
pattern = xf_mono_bitmap_new(xfc, 8, 8, brush->data);
|
|
|
|
XSetForeground(xfc->display, xfc->gc, backColor);
|
|
XSetBackground(xfc->display, xfc->gc, foreColor);
|
|
|
|
if (polygon_cb->backMode == BACKMODE_TRANSPARENT)
|
|
XSetFillStyle(xfc->display, xfc->gc, FillStippled);
|
|
else if (polygon_cb->backMode == BACKMODE_OPAQUE)
|
|
XSetFillStyle(xfc->display, xfc->gc, FillOpaqueStippled);
|
|
|
|
XSetStipple(xfc->display, xfc->gc, pattern);
|
|
XSetTSOrigin(xfc->display, xfc->gc, brush->x, brush->y);
|
|
|
|
XFillPolygon(xfc->display, xfc->drawing, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
|
|
if (xfc->drawing == xfc->primary)
|
|
{
|
|
XFillPolygon(xfc->display, xfc->drawable, xfc->gc,
|
|
points, npoints, Complex, CoordModePrevious);
|
|
}
|
|
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
XSetTSOrigin(xfc->display, xfc->gc, 0, 0);
|
|
XFreePixmap(xfc->display, pattern);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "PolygonCB unimplemented brush style:%d\n", brush->style);
|
|
}
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
free(points);
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_ellipse_sc(rdpContext* context, ELLIPSE_SC_ORDER* ellipse_sc)
|
|
{
|
|
fprintf(stderr, "EllipseSC\n");
|
|
}
|
|
|
|
void xf_gdi_ellipse_cb(rdpContext* context, ELLIPSE_CB_ORDER* ellipse_cb)
|
|
{
|
|
fprintf(stderr, "EllipseCB\n");
|
|
}
|
|
|
|
void xf_gdi_surface_frame_marker(rdpContext* context, SURFACE_FRAME_MARKER* surface_frame_marker)
|
|
{
|
|
rdpSettings* settings;
|
|
xfContext* xfc = (xfContext*) context;
|
|
|
|
settings = xfc->instance->settings;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
switch (surface_frame_marker->frameAction)
|
|
{
|
|
case SURFACECMD_FRAMEACTION_BEGIN:
|
|
xfc->frame_begin = TRUE;
|
|
xfc->frame_x1 = 0;
|
|
xfc->frame_y1 = 0;
|
|
xfc->frame_x2 = 0;
|
|
xfc->frame_y2 = 0;
|
|
break;
|
|
|
|
case SURFACECMD_FRAMEACTION_END:
|
|
xfc->frame_begin = FALSE;
|
|
if ((xfc->frame_x2 > xfc->frame_x1) && (xfc->frame_y2 > xfc->frame_y1))
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, xfc->frame_x1, xfc->frame_y1,
|
|
xfc->frame_x2 - xfc->frame_x1, xfc->frame_y2 - xfc->frame_y1);
|
|
}
|
|
if (settings->FrameAcknowledge > 0)
|
|
{
|
|
IFCALL(xfc->instance->update->SurfaceFrameAcknowledge, context, surface_frame_marker->frameId);
|
|
}
|
|
break;
|
|
}
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
static void xf_gdi_surface_update_frame(xfContext* xfc, UINT16 tx, UINT16 ty, UINT16 width, UINT16 height)
|
|
{
|
|
if (!xfc->remote_app)
|
|
{
|
|
if (xfc->frame_begin)
|
|
{
|
|
if (xfc->frame_x2 > xfc->frame_x1 && xfc->frame_y2 > xfc->frame_y1)
|
|
{
|
|
xfc->frame_x1 = MIN(xfc->frame_x1, tx);
|
|
xfc->frame_y1 = MIN(xfc->frame_y1, ty);
|
|
xfc->frame_x2 = MAX(xfc->frame_x2, tx + width);
|
|
xfc->frame_y2 = MAX(xfc->frame_y2, ty + height);
|
|
}
|
|
else
|
|
{
|
|
xfc->frame_x1 = tx;
|
|
xfc->frame_y1 = ty;
|
|
xfc->frame_x2 = tx + width;
|
|
xfc->frame_y2 = ty + height;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, tx, ty, width, height);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
gdi_InvalidateRegion(xfc->hdc, tx, ty, width, height);
|
|
}
|
|
}
|
|
|
|
void xf_gdi_surface_bits(rdpContext* context, SURFACE_BITS_COMMAND* surface_bits_command)
|
|
{
|
|
int i, tx, ty;
|
|
XImage* image;
|
|
RFX_MESSAGE* message;
|
|
xfContext* xfc = (xfContext*) context;
|
|
RFX_CONTEXT* rfx_context = (RFX_CONTEXT*) xfc->rfx_context;
|
|
NSC_CONTEXT* nsc_context = (NSC_CONTEXT*) xfc->nsc_context;
|
|
|
|
xf_lock_x11(xfc, FALSE);
|
|
|
|
if (surface_bits_command->codecID == RDP_CODEC_ID_REMOTEFX)
|
|
{
|
|
message = rfx_process_message(rfx_context,
|
|
surface_bits_command->bitmapData, surface_bits_command->bitmapDataLength);
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
|
|
XSetClipRectangles(xfc->display, xfc->gc,
|
|
surface_bits_command->destLeft, surface_bits_command->destTop,
|
|
(XRectangle*) message->rects, message->numRects, YXBanded);
|
|
|
|
/* Draw the tiles to primary surface, each is 64x64. */
|
|
for (i = 0; i < message->numTiles; i++)
|
|
{
|
|
image = XCreateImage(xfc->display, xfc->visual, 24, ZPixmap, 0,
|
|
(char*) message->tiles[i]->data, 64, 64, 32, 0);
|
|
|
|
tx = message->tiles[i]->x + surface_bits_command->destLeft;
|
|
ty = message->tiles[i]->y + surface_bits_command->destTop;
|
|
|
|
XPutImage(xfc->display, xfc->primary, xfc->gc, image, 0, 0, tx, ty, 64, 64);
|
|
XFree(image);
|
|
}
|
|
|
|
/* Copy the updated region from backstore to the window. */
|
|
for (i = 0; i < message->numRects; i++)
|
|
{
|
|
tx = message->rects[i].x + surface_bits_command->destLeft;
|
|
ty = message->rects[i].y + surface_bits_command->destTop;
|
|
|
|
xf_gdi_surface_update_frame(xfc, tx, ty, message->rects[i].width, message->rects[i].height);
|
|
}
|
|
|
|
XSetClipMask(xfc->display, xfc->gc, None);
|
|
rfx_message_free(rfx_context, message);
|
|
}
|
|
else if (surface_bits_command->codecID == RDP_CODEC_ID_NSCODEC)
|
|
{
|
|
nsc_process_message(nsc_context, surface_bits_command->bpp, surface_bits_command->width, surface_bits_command->height,
|
|
surface_bits_command->bitmapData, surface_bits_command->bitmapDataLength);
|
|
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
|
|
xfc->bmp_codec_nsc = (BYTE*) realloc(xfc->bmp_codec_nsc,
|
|
surface_bits_command->width * surface_bits_command->height * 4);
|
|
|
|
freerdp_image_flip(nsc_context->BitmapData, xfc->bmp_codec_nsc,
|
|
surface_bits_command->width, surface_bits_command->height, 32);
|
|
|
|
image = XCreateImage(xfc->display, xfc->visual, 24, ZPixmap, 0,
|
|
(char*) xfc->bmp_codec_nsc, surface_bits_command->width, surface_bits_command->height, 32, 0);
|
|
|
|
XPutImage(xfc->display, xfc->primary, xfc->gc, image, 0, 0,
|
|
surface_bits_command->destLeft, surface_bits_command->destTop,
|
|
surface_bits_command->width, surface_bits_command->height);
|
|
XFree(image);
|
|
|
|
xf_gdi_surface_update_frame(xfc,
|
|
surface_bits_command->destLeft, surface_bits_command->destTop,
|
|
surface_bits_command->width, surface_bits_command->height);
|
|
|
|
XSetClipMask(xfc->display, xfc->gc, None);
|
|
}
|
|
else if (surface_bits_command->codecID == RDP_CODEC_ID_NONE)
|
|
{
|
|
XSetFunction(xfc->display, xfc->gc, GXcopy);
|
|
XSetFillStyle(xfc->display, xfc->gc, FillSolid);
|
|
|
|
/* Validate that the data received is large enough */
|
|
if ((surface_bits_command->width * surface_bits_command->height * surface_bits_command->bpp / 8) <= (surface_bits_command->bitmapDataLength))
|
|
{
|
|
xfc->bmp_codec_none = (BYTE*) realloc(xfc->bmp_codec_none,
|
|
surface_bits_command->width * surface_bits_command->height * 4);
|
|
|
|
freerdp_image_flip(surface_bits_command->bitmapData, xfc->bmp_codec_none,
|
|
surface_bits_command->width, surface_bits_command->height, 32);
|
|
|
|
image = XCreateImage(xfc->display, xfc->visual, 24, ZPixmap, 0,
|
|
(char*) xfc->bmp_codec_none, surface_bits_command->width, surface_bits_command->height, 32, 0);
|
|
|
|
XPutImage(xfc->display, xfc->primary, xfc->gc, image, 0, 0,
|
|
surface_bits_command->destLeft, surface_bits_command->destTop,
|
|
surface_bits_command->width, surface_bits_command->height);
|
|
XFree(image);
|
|
|
|
xf_gdi_surface_update_frame(xfc,
|
|
surface_bits_command->destLeft, surface_bits_command->destTop,
|
|
surface_bits_command->width, surface_bits_command->height);
|
|
|
|
XSetClipMask(xfc->display, xfc->gc, None);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Invalid bitmap size - data is %d bytes for %dx%d\n update", surface_bits_command->bitmapDataLength, surface_bits_command->width, surface_bits_command->height);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Unsupported codecID %d\n", surface_bits_command->codecID);
|
|
}
|
|
|
|
xf_unlock_x11(xfc, FALSE);
|
|
}
|
|
|
|
void xf_gdi_register_update_callbacks(rdpUpdate* update)
|
|
{
|
|
rdpPrimaryUpdate* primary = update->primary;
|
|
|
|
update->Palette = xf_gdi_palette_update;
|
|
update->SetBounds = xf_gdi_set_bounds;
|
|
|
|
primary->DstBlt = xf_gdi_dstblt;
|
|
primary->PatBlt = xf_gdi_patblt;
|
|
primary->ScrBlt = xf_gdi_scrblt;
|
|
primary->OpaqueRect = xf_gdi_opaque_rect;
|
|
primary->DrawNineGrid = NULL;
|
|
primary->MultiDstBlt = NULL;
|
|
primary->MultiPatBlt = NULL;
|
|
primary->MultiScrBlt = NULL;
|
|
primary->MultiOpaqueRect = xf_gdi_multi_opaque_rect;
|
|
primary->MultiDrawNineGrid = NULL;
|
|
primary->LineTo = xf_gdi_line_to;
|
|
primary->Polyline = xf_gdi_polyline;
|
|
primary->MemBlt = xf_gdi_memblt;
|
|
primary->Mem3Blt = xf_gdi_mem3blt;
|
|
primary->SaveBitmap = NULL;
|
|
primary->GlyphIndex = NULL;
|
|
primary->FastIndex = NULL;
|
|
primary->FastGlyph = NULL;
|
|
primary->PolygonSC = xf_gdi_polygon_sc;
|
|
primary->PolygonCB = xf_gdi_polygon_cb;
|
|
primary->EllipseSC = xf_gdi_ellipse_sc;
|
|
primary->EllipseCB = xf_gdi_ellipse_cb;
|
|
|
|
update->SurfaceBits = xf_gdi_surface_bits;
|
|
update->SurfaceFrameMarker = xf_gdi_surface_frame_marker;
|
|
}
|
|
|