6c925893514cf763c48d5db5279c8cc4d9c0e76c
[libguac-client-rdp.git] / src / rdp_gdi.c
1
2 /* ***** BEGIN LICENSE BLOCK *****
3  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
4  *
5  * The contents of this file are subject to the Mozilla Public License Version
6  * 1.1 (the "License"); you may not use this file except in compliance with
7  * the License. You may obtain a copy of the License at
8  * http://www.mozilla.org/MPL/
9  *
10  * Software distributed under the License is distributed on an "AS IS" basis,
11  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12  * for the specific language governing rights and limitations under the
13  * License.
14  *
15  * The Original Code is libguac-client-rdp.
16  *
17  * The Initial Developer of the Original Code is
18  * Michael Jumper.
19  * Portions created by the Initial Developer are Copyright (C) 2011
20  * the Initial Developer. All Rights Reserved.
21  *
22  * Contributor(s):
23  * Matt Hortman
24  *
25  * Alternatively, the contents of this file may be used under the terms of
26  * either the GNU General Public License Version 2 or later (the "GPL"), or
27  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
28  * in which case the provisions of the GPL or the LGPL are applicable instead
29  * of those above. If you wish to allow use of your version of this file only
30  * under the terms of either the GPL or the LGPL, and not to allow others to
31  * use your version of this file under the terms of the MPL, indicate your
32  * decision by deleting the provisions above and replace them with the notice
33  * and other provisions required by the GPL or the LGPL. If you do not delete
34  * the provisions above, a recipient may use your version of this file under
35  * the terms of any one of the MPL, the GPL or the LGPL.
36  *
37  * ***** END LICENSE BLOCK ***** */
38
39 #include <freerdp/freerdp.h>
40
41 #include <guacamole/client.h>
42
43 #include "client.h"
44 #include "rdp_bitmap.h"
45
46 guac_transfer_function guac_rdp_rop3_transfer_function(guac_client* client,
47         int rop3) {
48
49     /* Translate supported ROP3 opcodes into composite modes */
50     switch (rop3) {
51
52         /* "SRCINVERT" (src ^ dest) */
53         case 0x66: return GUAC_TRANSFER_BINARY_XOR;
54
55         /* "SRCAND" (src & dest) */
56         case 0x88: return GUAC_TRANSFER_BINARY_AND;
57
58         /* "MERGEPAINT" (!src | dest)*/
59         case 0xBB: return GUAC_TRANSFER_BINARY_NSRC_OR;
60
61         /* "SRCCOPY" (src) */
62         case 0xCC: return GUAC_TRANSFER_BINARY_SRC;
63
64         /* "SRCPAINT" (src | dest) */
65         case 0xEE: return GUAC_TRANSFER_BINARY_OR;
66
67     }
68
69     /* Log warning if ROP3 opcode not supported */
70     guac_client_log_info (client, "guac_rdp_rop3_transfer_function: "
71             "UNSUPPORTED opcode = 0x%02X", rop3);
72
73     /* Default to BINARY_SRC */
74     return GUAC_TRANSFER_BINARY_SRC;
75
76 }
77
78 void guac_rdp_gdi_dstblt(rdpContext* context, DSTBLT_ORDER* dstblt) {
79
80     guac_client* client = ((rdp_freerdp_context*) context)->client;
81     const guac_layer* current_layer = ((rdp_guac_client_data*) client->data)->current_surface;
82
83     switch (dstblt->bRop) {
84
85         /* Blackness */
86         case 0:
87
88             /* Send black rectangle */
89             guac_protocol_send_rect(client->socket,
90                     GUAC_COMP_OVER, current_layer,
91                     dstblt->nLeftRect, dstblt->nTopRect,
92                     dstblt->nWidth, dstblt->nHeight,
93                     0, 0, 0, 255);
94             break;
95
96         /* Unsupported ROP3 */
97         default:
98             guac_client_log_info(client,
99                     "guac_rdp_gdi_dstblt(rop3=%i)", dstblt->bRop);
100
101     }
102
103
104
105 }
106
107 void guac_rdp_gdi_patblt(rdpContext* context, PATBLT_ORDER* patblt) {
108     guac_client* client = ((rdp_freerdp_context*) context)->client;
109     guac_client_log_info(client, "guac_rdp_gdi_patblt()");
110 }
111
112 void guac_rdp_gdi_scrblt(rdpContext* context, SCRBLT_ORDER* scrblt) {
113
114     guac_client* client = ((rdp_freerdp_context*) context)->client;
115     const guac_layer* current_layer = ((rdp_guac_client_data*) client->data)->current_surface;
116
117     /* Copy screen rect to current surface */
118     guac_protocol_send_copy(client->socket,
119             GUAC_DEFAULT_LAYER,
120             scrblt->nXSrc, scrblt->nYSrc, scrblt->nWidth, scrblt->nHeight,
121             GUAC_COMP_OVER, current_layer,
122             scrblt->nLeftRect, scrblt->nTopRect);
123
124 }
125
126 void guac_rdp_gdi_memblt(rdpContext* context, MEMBLT_ORDER* memblt) {
127
128     guac_client* client = ((rdp_freerdp_context*) context)->client;
129     const guac_layer* current_layer = ((rdp_guac_client_data*) client->data)->current_surface;
130     guac_socket* socket = client->socket;
131     guac_rdp_bitmap* bitmap = (guac_rdp_bitmap*) memblt->bitmap;
132
133     if (bitmap->layer != NULL) {
134
135         /* If operation is just SRC, simply copy */
136         if (memblt->bRop == 0xCC)
137             guac_protocol_send_copy(socket,
138                     bitmap->layer,
139                     memblt->nXSrc, memblt->nYSrc,
140                     memblt->nWidth, memblt->nHeight,
141                     GUAC_COMP_OVER,
142                     current_layer, memblt->nLeftRect, memblt->nTopRect);
143
144         /* Otherwise, use transfer */
145         else
146             guac_protocol_send_transfer(socket,
147                     bitmap->layer,
148                     memblt->nXSrc, memblt->nYSrc,
149                     memblt->nWidth, memblt->nHeight,
150                     guac_rdp_rop3_transfer_function(client, memblt->bRop),
151                     current_layer, memblt->nLeftRect, memblt->nTopRect);
152
153     }
154
155 }
156
157 void guac_rdp_gdi_opaquerect(rdpContext* context, OPAQUE_RECT_ORDER* opaque_rect) {
158
159     guac_client* client = ((rdp_freerdp_context*) context)->client;
160     uint32 color = freerdp_color_convert_var(opaque_rect->color,
161             context->instance->settings->color_depth, 32,
162             ((rdp_freerdp_context*) context)->clrconv);
163
164     const guac_layer* current_layer = ((rdp_guac_client_data*) client->data)->current_surface;
165
166     guac_protocol_send_rect(client->socket,
167             GUAC_COMP_OVER, current_layer,
168             opaque_rect->nLeftRect, opaque_rect->nTopRect,
169             opaque_rect->nWidth, opaque_rect->nHeight,
170             (color >> 16) & 0xFF,
171             (color >> 8 ) & 0xFF,
172             (color      ) & 0xFF,
173             255);
174
175 }
176
177 void guac_rdp_gdi_palette_update(rdpContext* context, PALETTE_UPDATE* palette) {
178
179     CLRCONV* clrconv = ((rdp_freerdp_context*) context)->clrconv;
180     clrconv->palette->count = palette->number;
181     clrconv->palette->entries = palette->entries;
182
183 }
184
185 void guac_rdp_gdi_set_bounds(rdpContext* context, rdpBounds* bounds) {
186
187     guac_client* client = ((rdp_freerdp_context*) context)->client;
188     const guac_layer* current_layer = ((rdp_guac_client_data*) client->data)->current_surface;
189
190     /* Set clip if specified */
191     if (bounds != NULL)
192         guac_protocol_send_clip(
193                 client->socket,
194                 current_layer,
195                 bounds->left,
196                 bounds->top,
197                 bounds->right - bounds->left + 1,
198                 bounds->bottom - bounds->top + 1);
199
200     /* Otherwise, reset clip */
201     else
202         guac_protocol_send_clip(
203                 client->socket,
204                 current_layer,
205                 0, 0,
206                 context->instance->settings->width,
207                 context->instance->settings->height);
208
209 }
210