summaryrefslogtreecommitdiffstats
path: root/xrdp/xrdp_wm.c
diff options
context:
space:
mode:
Diffstat (limited to 'xrdp/xrdp_wm.c')
-rw-r--r--xrdp/xrdp_wm.c195
1 files changed, 148 insertions, 47 deletions
diff --git a/xrdp/xrdp_wm.c b/xrdp/xrdp_wm.c
index f2ab6f64..63ba7eae 100644
--- a/xrdp/xrdp_wm.c
+++ b/xrdp/xrdp_wm.c
@@ -37,6 +37,8 @@ struct xrdp_wm* xrdp_wm_create(struct xrdp_process* owner)
self->painter = xrdp_painter_create(self);
self->rdp_layer = owner->rdp_layer;
self->cache = xrdp_cache_create(self, self->orders);
+ self->use_comp = 1;
+ self->op1 = 0;
return self;
}
@@ -44,15 +46,24 @@ struct xrdp_wm* xrdp_wm_create(struct xrdp_process* owner)
void xrdp_wm_delete(struct xrdp_wm* self)
{
if (self == 0)
+ {
return;
+ }
xrdp_cache_delete(self->cache);
xrdp_painter_delete(self->painter);
xrdp_bitmap_delete(self->screen);
/* free any modual stuff */
- if (self->mod != 0 && self->mod_exit != 0)
- self->mod_exit((int)self->mod);
+ if (self->mod != 0)
+ {
+ if (self->mod_exit != 0)
+ {
+ self->mod_exit((int)self->mod);
+ }
+ }
if (self->mod_handle != 0)
+ {
g_free_library(self->mod_handle);
+ }
/* free self */
g_free(self);
}
@@ -96,52 +107,140 @@ int xrdp_wm_send_bitmap(struct xrdp_wm* self, struct xrdp_bitmap* bitmap,
int lines_sending;
int Bpp;
int e;
+ int bufsize;
+ int total_bufsize;
+ int num_updates;
+ char* p_num_updates;
char* p;
char* q;
struct stream* s;
+ struct stream* temp_s;
- lines_sending = 0;
- make_stream(s);
- init_stream(s, 8192);
Bpp = (bitmap->bpp + 7) / 8;
- data_size = bitmap->width * bitmap->height * Bpp;
- line_size = bitmap->width * Bpp;
- total_lines = bitmap->height;
e = bitmap->width % 4;
if (e != 0)
e = 4 - e;
- i = 0;
- p = bitmap->data;
- if (line_size > 0 && total_lines > 0)
+ line_size = bitmap->width * Bpp;
+ make_stream(s);
+ init_stream(s, 8192);
+ if (self->use_comp)
{
- while (i < total_lines)
+ make_stream(temp_s);
+ init_stream(temp_s, 65536);
+ i = 0;
+ if (cy <= bitmap->height)
+ i = cy;
+ while (i > 0)
{
- lines_sending = 4096 / (line_size + e * Bpp);
- if (i + lines_sending > total_lines)
- lines_sending = total_lines - i;
- p = p + line_size * lines_sending;
+ total_bufsize = 0;
+ num_updates = 0;
xrdp_rdp_init_data(self->rdp_layer, s);
out_uint16_le(s, RDP_UPDATE_BITMAP);
- out_uint16_le(s, 1); /* num updates */
- out_uint16_le(s, x);
- out_uint16_le(s, y + i);
- out_uint16_le(s, (x + cx) - 1);
- out_uint16_le(s, (y + i + lines_sending) - 1);
- out_uint16_le(s, bitmap->width + e);
- out_uint16_le(s, lines_sending);
- out_uint16_le(s, bitmap->bpp); /* bpp */
- out_uint16_le(s, 0); /* compress */
- out_uint16_le(s, (line_size + e * Bpp) * lines_sending); /* bufsize */
- q = p;
- for (j = 0; j < lines_sending; j++)
+ p_num_updates = s->p;
+ out_uint8s(s, 2); /* num_updates set later */
+ do
{
- q = q - line_size;
- out_uint8a(s, q, line_size)
- out_uint8s(s, e * Bpp);
- }
- s_mark_end(s);
+ if (self->op1)
+ {
+ s_push_layer(s, channel_hdr, 18);
+ }
+ else
+ {
+ s_push_layer(s, channel_hdr, 26);
+ }
+ p = s->p;
+ lines_sending = xrdp_bitmap_compress(bitmap->data, bitmap->width,
+ bitmap->height,
+ s, bitmap->bpp,
+ 4096 - total_bufsize,
+ i - 1, temp_s);
+ if (lines_sending == 0)
+ break;
+ num_updates++;
+ bufsize = s->p - p;
+ total_bufsize += bufsize;
+ i = i - lines_sending;
+ s_mark_end(s);
+ s_pop_layer(s, channel_hdr);
+ out_uint16_le(s, x); /* left */
+ out_uint16_le(s, y + i); /* top */
+ out_uint16_le(s, (x + cx) - 1); /* right */
+ out_uint16_le(s, (y + i + lines_sending) - 1); /* bottom */
+ out_uint16_le(s, bitmap->width + e); /* width */
+ out_uint16_le(s, lines_sending); /* height */
+ out_uint16_le(s, bitmap->bpp); /* bpp */
+ if (self->op1)
+ {
+ out_uint16_le(s, 0x401); /* compress */
+ out_uint16_le(s, bufsize); /* compressed size */
+ j = (bitmap->width + e) * Bpp;
+ j = j * lines_sending;
+ }
+ else
+ {
+ out_uint16_le(s, 0x1); /* compress */
+ j = bufsize + 8;
+ out_uint16_le(s, j);
+ out_uint8s(s, 2); /* pad */
+ out_uint16_le(s, bufsize); /* compressed size */
+ j = (bitmap->width + e) * Bpp;
+ out_uint16_le(s, j); /* line size */
+ j = j * lines_sending;
+ out_uint16_le(s, j); /* final size */
+ }
+ if (j > 32768)
+ g_printf("error, decompressed size too big, its %d\n\r", j);
+ if (bufsize > 8192)
+ g_printf("error, compressed size too big, its %d\n\r", bufsize);
+ s->p = s->end;
+ } while (total_bufsize < 4096 && i > 0);
+ p_num_updates[0] = num_updates;
+ p_num_updates[1] = num_updates >> 8;
xrdp_rdp_send_data(self->rdp_layer, s, RDP_DATA_PDU_UPDATE);
- i = i + lines_sending;
+ if (total_bufsize > 8192)
+ g_printf("error, total compressed size too big, its %d\n\r",
+ total_bufsize);
+ }
+ free_stream(temp_s);
+ }
+ else
+ {
+ lines_sending = 0;
+ data_size = bitmap->width * bitmap->height * Bpp;
+ total_lines = bitmap->height;
+ i = 0;
+ p = bitmap->data;
+ if (line_size > 0 && total_lines > 0)
+ {
+ while (i < total_lines)
+ {
+ lines_sending = 4096 / (line_size + e * Bpp);
+ if (i + lines_sending > total_lines)
+ lines_sending = total_lines - i;
+ p = p + line_size * lines_sending;
+ xrdp_rdp_init_data(self->rdp_layer, s);
+ out_uint16_le(s, RDP_UPDATE_BITMAP);
+ out_uint16_le(s, 1); /* num updates */
+ out_uint16_le(s, x);
+ out_uint16_le(s, y + i);
+ out_uint16_le(s, (x + cx) - 1);
+ out_uint16_le(s, (y + i + lines_sending) - 1);
+ out_uint16_le(s, bitmap->width + e);
+ out_uint16_le(s, lines_sending);
+ out_uint16_le(s, bitmap->bpp); /* bpp */
+ out_uint16_le(s, 0); /* compress */
+ out_uint16_le(s, (line_size + e * Bpp) * lines_sending); /* bufsize */
+ q = p;
+ for (j = 0; j < lines_sending; j++)
+ {
+ q = q - line_size;
+ out_uint8a(s, q, line_size)
+ out_uint8s(s, e * Bpp);
+ }
+ s_mark_end(s);
+ xrdp_rdp_send_data(self->rdp_layer, s, RDP_DATA_PDU_UPDATE);
+ i = i + lines_sending;
+ }
}
}
free_stream(s);
@@ -664,7 +763,9 @@ int xrdp_wm_mouse_move(struct xrdp_wm* self, int x, int y)
if (self->mod != 0) /* if screen is mod controled */
{
if (self->mod->mod_event != 0)
- self->mod->mod_event((int)self->mod, WM_MOUSEMOVE, x, y);
+ {
+ self->mod->mod_event(self->mod, WM_MOUSEMOVE, x, y);
+ }
}
}
if (self->button_down != 0)
@@ -769,29 +870,29 @@ int xrdp_wm_mouse_click(struct xrdp_wm* self, int x, int y, int but, int down)
if (self->mod->mod_event != 0)
{
if (but == 1 && down)
- self->mod->mod_event((int)self->mod, WM_LBUTTONDOWN, x, y);
+ self->mod->mod_event(self->mod, WM_LBUTTONDOWN, x, y);
else if (but == 1 && !down)
- self->mod->mod_event((int)self->mod, WM_LBUTTONUP, x, y);
+ self->mod->mod_event(self->mod, WM_LBUTTONUP, x, y);
if (but == 2 && down)
- self->mod->mod_event((int)self->mod, WM_RBUTTONDOWN, x, y);
+ self->mod->mod_event(self->mod, WM_RBUTTONDOWN, x, y);
else if (but == 2 && !down)
- self->mod->mod_event((int)self->mod, WM_RBUTTONUP, x, y);
+ self->mod->mod_event(self->mod, WM_RBUTTONUP, x, y);
if (but == 3 && down)
- self->mod->mod_event((int)self->mod, WM_BUTTON3DOWN, x, y);
+ self->mod->mod_event(self->mod, WM_BUTTON3DOWN, x, y);
else if (but == 3 && !down)
- self->mod->mod_event((int)self->mod, WM_BUTTON3UP, x, y);
+ self->mod->mod_event(self->mod, WM_BUTTON3UP, x, y);
if (but == 4)
{
- self->mod->mod_event((int)self->mod, WM_BUTTON4DOWN,
+ self->mod->mod_event(self->mod, WM_BUTTON4DOWN,
self->mouse_x, self->mouse_y);
- self->mod->mod_event((int)self->mod, WM_BUTTON4UP,
+ self->mod->mod_event(self->mod, WM_BUTTON4UP,
self->mouse_x, self->mouse_y);
}
if (but == 5)
{
- self->mod->mod_event((int)self->mod, WM_BUTTON5DOWN,
+ self->mod->mod_event(self->mod, WM_BUTTON5DOWN,
self->mouse_x, self->mouse_y);
- self->mod->mod_event((int)self->mod, WM_BUTTON5UP,
+ self->mod->mod_event(self->mod, WM_BUTTON5UP,
self->mouse_x, self->mouse_y);
}
}
@@ -928,9 +1029,9 @@ int xrdp_wm_key(struct xrdp_wm* self, int device_flags, int scan_code)
self->num_lock,
self->scroll_lock);
if (c != 0)
- self->mod->mod_event((int)self->mod, msg, c, 0xffff);
+ self->mod->mod_event(self->mod, msg, c, 0xffff);
else
- self->mod->mod_event((int)self->mod, msg, scan_code, device_flags);
+ self->mod->mod_event(self->mod, msg, scan_code, device_flags);
}
}
else if (self->focused_window != 0)