You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2991 lines
83 KiB
2991 lines
83 KiB
/*
|
|
Copyright 2005-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.
|
|
|
|
*/
|
|
|
|
#include "rdp.h"
|
|
#include "xrdp_rail.h"
|
|
#include "rdpglyph.h"
|
|
#include "rdprandr.h"
|
|
|
|
#include <signal.h>
|
|
#include <sys/ipc.h>
|
|
#include <sys/shm.h>
|
|
#include <sys/types.h>
|
|
|
|
#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)
|
|
|
|
static int g_use_shmem = 1; /* turns on or off */
|
|
static int g_shmemid = -1;
|
|
static char *g_shmemptr = 0;
|
|
static int g_shmem_lineBytes = 0;
|
|
static RegionPtr g_shm_reg = 0;
|
|
|
|
static int g_rect_id_ack = 0;
|
|
static int g_rect_id = 0;
|
|
|
|
static int g_listen_sck = 0;
|
|
static int g_sck = 0;
|
|
static int g_sck_closed = 0;
|
|
static int g_connected = 0;
|
|
static int g_dis_listen_sck = 0;
|
|
//static int g_dis_sck = 0;
|
|
//static int g_dis_sck_closed = 0;
|
|
//static int g_dis_connected = 0;
|
|
|
|
static int g_begin = 0;
|
|
static struct stream *g_out_s = 0;
|
|
static struct stream *g_in_s = 0;
|
|
static int g_button_mask = 0;
|
|
static int g_cursor_x = 0;
|
|
static int g_cursor_y = 0;
|
|
static OsTimerPtr g_timer = 0;
|
|
static int g_scheduled = 0;
|
|
static int g_count = 0;
|
|
static int g_rdpindex = -1;
|
|
|
|
extern DevPrivateKeyRec g_rdpWindowIndex; /* from rdpmain.c */
|
|
extern ScreenPtr g_pScreen; /* from rdpmain.c */
|
|
extern int g_Bpp; /* from rdpmain.c */
|
|
extern int g_Bpp_mask; /* from rdpmain.c */
|
|
extern rdpScreenInfoRec g_rdpScreen; /* from rdpmain.c */
|
|
extern int g_do_glyph_cache; /* from rdpmain.c */
|
|
extern int g_can_do_pix_to_pix; /* from rdpmain.c */
|
|
extern int g_use_rail; /* from rdpmain.c */
|
|
extern int g_do_composite; /* from rdpmain.c */
|
|
|
|
/* true is to use unix domain socket */
|
|
extern int g_use_uds; /* in rdpmain.c */
|
|
extern char g_uds_data[]; /* in rdpmain.c */
|
|
extern int g_do_dirty_ons; /* in rdpmain.c */
|
|
extern rdpPixmapRec g_screenPriv; /* in rdpmain.c */
|
|
extern int g_con_number; /* in rdpmain.c */
|
|
|
|
struct rdpup_os_bitmap
|
|
{
|
|
int used;
|
|
PixmapPtr pixmap;
|
|
rdpPixmapPtr priv;
|
|
int stamp;
|
|
};
|
|
|
|
#define USE_MAX_OS_BYTES 1
|
|
#define MAX_OS_BYTES (16 * 1024 * 1024)
|
|
static struct rdpup_os_bitmap *g_os_bitmaps = 0;
|
|
static int g_max_os_bitmaps = 0;
|
|
static int g_os_bitmap_stamp = 0;
|
|
static int g_os_bitmap_alloc_size = 0;
|
|
|
|
static int g_pixmap_byte_total = 0;
|
|
static int g_pixmap_num_used = 0;
|
|
|
|
struct rdpup_top_window
|
|
{
|
|
WindowPtr wnd;
|
|
struct rdpup_top_window *next;
|
|
};
|
|
|
|
/*
|
|
0 GXclear, 0
|
|
1 GXnor, DPon
|
|
2 GXandInverted, DPna
|
|
3 GXcopyInverted, Pn
|
|
4 GXandReverse, PDna
|
|
5 GXinvert, Dn
|
|
6 GXxor, DPx
|
|
7 GXnand, DPan
|
|
8 GXand, DPa
|
|
9 GXequiv, DPxn
|
|
a GXnoop, D
|
|
b GXorInverted, DPno
|
|
c GXcopy, P
|
|
d GXorReverse, PDno
|
|
e GXor, DPo
|
|
f GXset 1
|
|
*/
|
|
|
|
static int g_rdp_opcodes[16] =
|
|
{
|
|
0x00, /* GXclear 0x0 0 */
|
|
0x88, /* GXand 0x1 src AND dst */
|
|
0x44, /* GXandReverse 0x2 src AND NOT dst */
|
|
0xcc, /* GXcopy 0x3 src */
|
|
0x22, /* GXandInverted 0x4 NOT src AND dst */
|
|
0xaa, /* GXnoop 0x5 dst */
|
|
0x66, /* GXxor 0x6 src XOR dst */
|
|
0xee, /* GXor 0x7 src OR dst */
|
|
0x11, /* GXnor 0x8 NOT src AND NOT dst */
|
|
0x99, /* GXequiv 0x9 NOT src XOR dst */
|
|
0x55, /* GXinvert 0xa NOT dst */
|
|
0xdd, /* GXorReverse 0xb src OR NOT dst */
|
|
0x33, /* GXcopyInverted 0xc NOT src */
|
|
0xbb, /* GXorInverted 0xd NOT src OR dst */
|
|
0x77, /* GXnand 0xe NOT src OR NOT dst */
|
|
0xff /* GXset 0xf 1 */
|
|
};
|
|
|
|
static int g_do_kill_disconnected = 0; /* turn on or off */
|
|
static OsTimerPtr g_dis_timer = 0;
|
|
static int g_disconnect_scheduled = 0;
|
|
static CARD32 g_disconnect_timeout_s = 60; /* 60 seconds */
|
|
static CARD32 g_disconnect_time_ms = 0; /* time of disconnect in milliseconds */
|
|
|
|
static int g_do_multimon = 0; /* multimon - turn on or off */
|
|
|
|
/******************************************************************************/
|
|
static CARD32
|
|
rdpDeferredDisconnectCallback(OsTimerPtr timer, CARD32 now, pointer arg)
|
|
{
|
|
CARD32 lnow_ms;
|
|
|
|
LLOGLN(10, ("rdpDeferredDisconnectCallback"));
|
|
if (g_connected)
|
|
{
|
|
/* this should not happen */
|
|
LLOGLN(0, ("rdpDeferredDisconnectCallback: connected"));
|
|
if (g_dis_timer != 0)
|
|
{
|
|
LLOGLN(0, ("rdpDeferredDisconnectCallback: canceling g_dis_timer"));
|
|
TimerCancel(g_dis_timer);
|
|
TimerFree(g_dis_timer);
|
|
g_dis_timer = 0;
|
|
}
|
|
g_disconnect_scheduled = 0;
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(10, ("rdpDeferredDisconnectCallback: not connected"));
|
|
}
|
|
lnow_ms = GetTimeInMillis();
|
|
if (lnow_ms - g_disconnect_time_ms > g_disconnect_timeout_s * 1000)
|
|
{
|
|
LLOGLN(0, ("rdpDeferredDisconnectCallback: exit X11rdp"));
|
|
kill(getpid(), SIGTERM);
|
|
return 0;
|
|
}
|
|
g_dis_timer = TimerSet(g_dis_timer, 0, 1000 * 10,
|
|
rdpDeferredDisconnectCallback, 0);
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
static int
|
|
rdpup_disconnect(void)
|
|
{
|
|
int index;
|
|
|
|
LLOGLN(0, ("rdpup_disconnect:"));
|
|
if (g_do_kill_disconnected)
|
|
{
|
|
if (!g_disconnect_scheduled)
|
|
{
|
|
LLOGLN(0, ("rdpup_disconnect: starting g_dis_timer"));
|
|
g_dis_timer = TimerSet(g_dis_timer, 0, 1000 * 10,
|
|
rdpDeferredDisconnectCallback, 0);
|
|
g_disconnect_scheduled = 1;
|
|
}
|
|
g_disconnect_time_ms = GetTimeInMillis();
|
|
}
|
|
|
|
RemoveEnabledDevice(g_sck);
|
|
g_connected = 0;
|
|
g_tcp_close(g_sck);
|
|
g_sck = 0;
|
|
g_sck_closed = 1;
|
|
g_pixmap_byte_total = 0;
|
|
g_pixmap_num_used = 0;
|
|
g_rdpindex = -1;
|
|
|
|
if (g_max_os_bitmaps > 0)
|
|
{
|
|
for (index = 0; index < g_max_os_bitmaps; index++)
|
|
{
|
|
if (g_os_bitmaps[index].used)
|
|
{
|
|
if (g_os_bitmaps[index].priv != 0)
|
|
{
|
|
g_os_bitmaps[index].priv->status = 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
g_os_bitmap_alloc_size = 0;
|
|
|
|
g_max_os_bitmaps = 0;
|
|
g_free(g_os_bitmaps);
|
|
g_os_bitmaps = 0;
|
|
g_use_rail = 0;
|
|
g_do_glyph_cache = 0;
|
|
g_do_composite = 0;
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/* returns -1 on error */
|
|
int
|
|
rdpup_add_os_bitmap(PixmapPtr pixmap, rdpPixmapPtr priv)
|
|
{
|
|
int index;
|
|
int rv;
|
|
int oldest;
|
|
int oldest_index;
|
|
int this_bytes;
|
|
|
|
LLOGLN(10, ("rdpup_add_os_bitmap:"));
|
|
if (!g_connected)
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: test error 1"));
|
|
return -1;
|
|
}
|
|
|
|
if (g_os_bitmaps == 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: test error 2"));
|
|
return -1;
|
|
}
|
|
|
|
this_bytes = pixmap->devKind * pixmap->drawable.height;
|
|
if (this_bytes > MAX_OS_BYTES)
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: error, too big this_bytes %d "
|
|
"width %d height %d", this_bytes,
|
|
pixmap->drawable.height, pixmap->drawable.height));
|
|
return -1;
|
|
}
|
|
|
|
oldest = 0x7fffffff;
|
|
oldest_index = -1;
|
|
rv = -1;
|
|
index = 0;
|
|
|
|
while (index < g_max_os_bitmaps)
|
|
{
|
|
if (g_os_bitmaps[index].used == 0)
|
|
{
|
|
g_os_bitmaps[index].used = 1;
|
|
g_os_bitmaps[index].pixmap = pixmap;
|
|
g_os_bitmaps[index].priv = priv;
|
|
g_os_bitmaps[index].stamp = g_os_bitmap_stamp;
|
|
g_os_bitmap_stamp++;
|
|
g_pixmap_num_used++;
|
|
rv = index;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (g_os_bitmaps[index].stamp < oldest)
|
|
{
|
|
oldest = g_os_bitmaps[index].stamp;
|
|
oldest_index = index;
|
|
}
|
|
}
|
|
index++;
|
|
}
|
|
|
|
if (rv == -1)
|
|
{
|
|
if (oldest_index == -1)
|
|
{
|
|
LLOGLN(0, ("rdpup_add_os_bitmap: error"));
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: too many pixmaps removing "
|
|
"oldest_index %d", oldest_index));
|
|
rdpup_remove_os_bitmap(oldest_index);
|
|
rdpup_delete_os_surface(oldest_index);
|
|
g_os_bitmaps[oldest_index].used = 1;
|
|
g_os_bitmaps[oldest_index].pixmap = pixmap;
|
|
g_os_bitmaps[oldest_index].priv = priv;
|
|
g_os_bitmaps[oldest_index].stamp = g_os_bitmap_stamp;
|
|
g_os_bitmap_stamp++;
|
|
g_pixmap_num_used++;
|
|
rv = oldest_index;
|
|
}
|
|
}
|
|
|
|
if (rv < 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: test error 3"));
|
|
return rv;
|
|
}
|
|
|
|
g_os_bitmap_alloc_size += this_bytes;
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: this_bytes %d g_os_bitmap_alloc_size %d",
|
|
this_bytes, g_os_bitmap_alloc_size));
|
|
#if USE_MAX_OS_BYTES
|
|
while (g_os_bitmap_alloc_size > MAX_OS_BYTES)
|
|
{
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: must delete g_pixmap_num_used %d",
|
|
g_pixmap_num_used));
|
|
/* find oldest */
|
|
oldest = 0x7fffffff;
|
|
oldest_index = -1;
|
|
index = 0;
|
|
while (index < g_max_os_bitmaps)
|
|
{
|
|
if (g_os_bitmaps[index].used && (g_os_bitmaps[index].stamp < oldest))
|
|
{
|
|
oldest = g_os_bitmaps[index].stamp;
|
|
oldest_index = index;
|
|
}
|
|
index++;
|
|
}
|
|
if (oldest_index == -1)
|
|
{
|
|
LLOGLN(0, ("rdpup_add_os_bitmap: error 1"));
|
|
break;
|
|
}
|
|
if (oldest_index == rv)
|
|
{
|
|
LLOGLN(0, ("rdpup_add_os_bitmap: error 2"));
|
|
break;
|
|
}
|
|
rdpup_remove_os_bitmap(oldest_index);
|
|
rdpup_delete_os_surface(oldest_index);
|
|
}
|
|
#endif
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: new bitmap index %d", rv));
|
|
LLOGLN(10, ("rdpup_add_os_bitmap: g_pixmap_num_used %d "
|
|
"g_os_bitmap_stamp 0x%8.8x", g_pixmap_num_used, g_os_bitmap_stamp));
|
|
return rv;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
int
|
|
rdpup_remove_os_bitmap(int rdpindex)
|
|
{
|
|
PixmapPtr pixmap;
|
|
rdpPixmapPtr priv;
|
|
int this_bytes;
|
|
|
|
LLOGLN(10, ("rdpup_remove_os_bitmap: index %d stamp %d",
|
|
rdpindex, g_os_bitmaps[rdpindex].stamp));
|
|
|
|
if (g_os_bitmaps == 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_remove_os_bitmap: test error 1"));
|
|
return 1;
|
|
}
|
|
|
|
if ((rdpindex < 0) && (rdpindex >= g_max_os_bitmaps))
|
|
{
|
|
LLOGLN(10, ("rdpup_remove_os_bitmap: test error 2"));
|
|
return 1;
|
|
}
|
|
|
|
if (g_os_bitmaps[rdpindex].used)
|
|
{
|
|
pixmap = g_os_bitmaps[rdpindex].pixmap;
|
|
priv = g_os_bitmaps[rdpindex].priv;
|
|
draw_item_remove_all(priv);
|
|
this_bytes = pixmap->devKind * pixmap->drawable.height;
|
|
g_os_bitmap_alloc_size -= this_bytes;
|
|
LLOGLN(10, ("rdpup_remove_os_bitmap: this_bytes %d "
|
|
"g_os_bitmap_alloc_size %d", this_bytes,
|
|
g_os_bitmap_alloc_size));
|
|
g_os_bitmaps[rdpindex].used = 0;
|
|
g_os_bitmaps[rdpindex].pixmap = 0;
|
|
g_os_bitmaps[rdpindex].priv = 0;
|
|
g_pixmap_num_used--;
|
|
priv->status = 0;
|
|
priv->con_number = 0;
|
|
priv->use_count = 0;
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, ("rdpup_remove_os_bitmap: error"));
|
|
}
|
|
|
|
LLOGLN(10, ("rdpup_remove_os_bitmap: g_pixmap_num_used %d",
|
|
g_pixmap_num_used));
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
int
|
|
rdpup_update_os_use(int rdpindex)
|
|
{
|
|
LLOGLN(10, ("rdpup_update_use: index %d stamp %d",
|
|
rdpindex, g_os_bitmaps[rdpindex].stamp));
|
|
|
|
if (g_os_bitmaps == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
if ((rdpindex < 0) && (rdpindex >= g_max_os_bitmaps))
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
if (g_os_bitmaps[rdpindex].used)
|
|
{
|
|
g_os_bitmaps[rdpindex].stamp = g_os_bitmap_stamp;
|
|
g_os_bitmap_stamp++;
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, ("rdpup_update_use: error rdpindex %d", rdpindex));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* returns error */
|
|
static int
|
|
rdpup_send(char *data, int len)
|
|
{
|
|
int sent;
|
|
|
|
LLOGLN(10, ("rdpup_send - sending %d bytes", len));
|
|
|
|
if (g_sck_closed)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
while (len > 0)
|
|
{
|
|
sent = g_tcp_send(g_sck, data, len, 0);
|
|
|
|
if (sent == -1)
|
|
{
|
|
if (g_tcp_last_error_would_block(g_sck))
|
|
{
|
|
g_sleep(1);
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, ("rdpup_send: g_tcp_send failed(returned -1)"));
|
|
rdpup_disconnect();
|
|
return 1;
|
|
}
|
|
}
|
|
else if (sent == 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_send: g_tcp_send failed(returned zero)"));
|
|
rdpup_disconnect();
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
data += sent;
|
|
len -= sent;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_send_msg(struct stream *s)
|
|
{
|
|
int len;
|
|
int rv;
|
|
|
|
rv = 1;
|
|
|
|
if (s != 0)
|
|
{
|
|
len = (int)(s->end - s->data);
|
|
|
|
if (len > s->size)
|
|
{
|
|
rdpLog("overrun error len %d count %d\n", len, g_count);
|
|
}
|
|
|
|
s_pop_layer(s, iso_hdr);
|
|
out_uint16_le(s, 3);
|
|
out_uint16_le(s, g_count);
|
|
out_uint32_le(s, len - 8);
|
|
rv = rdpup_send(s->data, len);
|
|
}
|
|
|
|
if (rv != 0)
|
|
{
|
|
rdpLog("error in rdpup_send_msg\n");
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_send_pending(void)
|
|
{
|
|
int rv;
|
|
|
|
rv = 0;
|
|
if (g_connected && g_begin)
|
|
{
|
|
LLOGLN(10, ("end %d", g_count));
|
|
out_uint16_le(g_out_s, 2);
|
|
out_uint16_le(g_out_s, 4);
|
|
g_count++;
|
|
s_mark_end(g_out_s);
|
|
if (rdpup_send_msg(g_out_s) != 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_send_pending: rdpup_send_msg failed"));
|
|
rv = 1;
|
|
}
|
|
}
|
|
|
|
g_count = 0;
|
|
g_begin = 0;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static CARD32
|
|
rdpDeferredUpdateCallback(OsTimerPtr timer, CARD32 now, pointer arg)
|
|
{
|
|
LLOGLN(10, ("rdpDeferredUpdateCallback"));
|
|
|
|
if (g_do_dirty_ons)
|
|
{
|
|
if (g_rect_id == g_rect_id_ack)
|
|
{
|
|
rdpup_check_dirty_screen(&g_screenPriv);
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, ("rdpDeferredUpdateCallback: skipping"));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rdpup_send_pending();
|
|
}
|
|
|
|
g_scheduled = 0;
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpScheduleDeferredUpdate(void)
|
|
{
|
|
if (!g_scheduled)
|
|
{
|
|
g_scheduled = 1;
|
|
g_timer = TimerSet(g_timer, 0, 40, rdpDeferredUpdateCallback, 0);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* returns error */
|
|
static int
|
|
rdpup_recv(char *data, int len)
|
|
{
|
|
int rcvd;
|
|
|
|
if (g_sck_closed)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
while (len > 0)
|
|
{
|
|
rcvd = g_tcp_recv(g_sck, data, len, 0);
|
|
|
|
if (rcvd == -1)
|
|
{
|
|
if (g_tcp_last_error_would_block(g_sck))
|
|
{
|
|
g_sleep(1);
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, ("rdpup_recv: g_tcp_recv failed(returned -1)"));
|
|
rdpup_disconnect();
|
|
return 1;
|
|
}
|
|
}
|
|
else if (rcvd == 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_recv: g_tcp_recv failed(returned 0)"));
|
|
rdpup_disconnect();
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
data += rcvd;
|
|
len -= rcvd;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_recv_msg(struct stream *s)
|
|
{
|
|
int len;
|
|
int rv;
|
|
|
|
rv = 1;
|
|
|
|
if (s != 0)
|
|
{
|
|
init_stream(s, 4);
|
|
rv = rdpup_recv(s->data, 4);
|
|
|
|
if (rv == 0)
|
|
{
|
|
in_uint32_le(s, len);
|
|
|
|
if (len > 3)
|
|
{
|
|
init_stream(s, len);
|
|
rv = rdpup_recv(s->data, len - 4);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (rv != 0)
|
|
{
|
|
rdpLog("error in rdpup_recv_msg\n");
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/* wait 'millis' milliseconds for the socket to be able to receive */
|
|
/* returns boolean */
|
|
static int
|
|
sck_can_recv(int sck, int millis)
|
|
{
|
|
fd_set rfds;
|
|
struct timeval time;
|
|
int rv;
|
|
|
|
time.tv_sec = millis / 1000;
|
|
time.tv_usec = (millis * 1000) % 1000000;
|
|
FD_ZERO(&rfds);
|
|
|
|
if (sck > 0)
|
|
{
|
|
FD_SET(((unsigned int)sck), &rfds);
|
|
rv = select(sck + 1, &rfds, 0, 0, &time);
|
|
|
|
if (rv > 0)
|
|
{
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/*
|
|
this from miScreenInit
|
|
pScreen->mmWidth = (xsize * 254 + dpix * 5) / (dpix * 10);
|
|
pScreen->mmHeight = (ysize * 254 + dpiy * 5) / (dpiy * 10);
|
|
*/
|
|
static int
|
|
process_screen_size_msg(int width, int height, int bpp)
|
|
{
|
|
int mmwidth;
|
|
int mmheight;
|
|
int bytes;
|
|
Bool ok;
|
|
|
|
LLOGLN(0, ("process_screen_size_msg: set width %d height %d bpp %d",
|
|
width, height, bpp));
|
|
g_rdpScreen.rdp_width = width;
|
|
g_rdpScreen.rdp_height = height;
|
|
g_rdpScreen.rdp_bpp = bpp;
|
|
|
|
if (bpp < 15)
|
|
{
|
|
g_rdpScreen.rdp_Bpp = 1;
|
|
g_rdpScreen.rdp_Bpp_mask = 0xff;
|
|
}
|
|
else if (bpp == 15)
|
|
{
|
|
g_rdpScreen.rdp_Bpp = 2;
|
|
g_rdpScreen.rdp_Bpp_mask = 0x7fff;
|
|
}
|
|
else if (bpp == 16)
|
|
{
|
|
g_rdpScreen.rdp_Bpp = 2;
|
|
g_rdpScreen.rdp_Bpp_mask = 0xffff;
|
|
}
|
|
else if (bpp > 16)
|
|
{
|
|
g_rdpScreen.rdp_Bpp = 4;
|
|
g_rdpScreen.rdp_Bpp_mask = 0xffffff;
|
|
}
|
|
|
|
if (g_use_shmem)
|
|
{
|
|
if (g_shmemptr != 0)
|
|
{
|
|
shmdt(g_shmemptr);
|
|
g_shmemptr = 0;
|
|
}
|
|
bytes = g_rdpScreen.rdp_width * g_rdpScreen.rdp_height *
|
|
g_rdpScreen.rdp_Bpp;
|
|
g_shmemid = shmget(IPC_PRIVATE, bytes, IPC_CREAT | 0777);
|
|
if (g_shmemid != -1)
|
|
{
|
|
g_shmemptr = shmat(g_shmemid, 0, 0);
|
|
if (g_shmemptr == (void *) -1)
|
|
{
|
|
LLOGLN(0, ("process_screen_size_msg: shmat failed for %d "
|
|
"bytes g_shmemid %d", bytes, g_shmemid));
|
|
g_shmemptr = 0;
|
|
shmctl(g_shmemid, IPC_RMID, NULL);
|
|
g_shmemid = -1;
|
|
}
|
|
else
|
|
{
|
|
shmctl(g_shmemid, IPC_RMID, NULL);
|
|
}
|
|
LLOGLN(0, ("process_screen_size_msg: g_shmemid %d g_shmemptr %p",
|
|
g_shmemid, g_shmemptr));
|
|
g_shmem_lineBytes = g_rdpScreen.rdp_Bpp * g_rdpScreen.rdp_width;
|
|
if (g_shm_reg != 0)
|
|
{
|
|
RegionDestroy(g_shm_reg);
|
|
}
|
|
g_shm_reg = RegionCreate(NullBox, 0);
|
|
}
|
|
}
|
|
|
|
mmwidth = PixelToMM(width);
|
|
mmheight = PixelToMM(height);
|
|
|
|
if ((g_rdpScreen.width != width) || (g_rdpScreen.height != height))
|
|
{
|
|
LLOGLN(0, (" calling RRScreenSizeSet"));
|
|
ok = RRScreenSizeSet(g_pScreen, width, height, mmwidth, mmheight);
|
|
LLOGLN(0, (" RRScreenSizeSet ok=[%d]", ok));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
l_bound_by(int val, int low, int high)
|
|
{
|
|
if (val > high)
|
|
{
|
|
val = high;
|
|
}
|
|
|
|
if (val < low)
|
|
{
|
|
val = low;
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_send_caps(void)
|
|
{
|
|
struct stream *ls;
|
|
int len;
|
|
int rv;
|
|
int cap_count;
|
|
int cap_bytes;
|
|
|
|
make_stream(ls);
|
|
init_stream(ls, 8192);
|
|
s_push_layer(ls, iso_hdr, 8);
|
|
|
|
cap_count = 0;
|
|
cap_bytes = 0;
|
|
|
|
#if 0
|
|
out_uint16_le(ls, 0);
|
|
out_uint16_le(ls, 4);
|
|
cap_count++;
|
|
cap_bytes += 4;
|
|
|
|
out_uint16_le(ls, 1);
|
|
out_uint16_le(ls, 4);
|
|
cap_count++;
|
|
cap_bytes += 4;
|
|
#endif
|
|
|
|
s_mark_end(ls);
|
|
len = (int)(ls->end - ls->data);
|
|
s_pop_layer(ls, iso_hdr);
|
|
out_uint16_le(ls, 2); /* caps */
|
|
out_uint16_le(ls, cap_count); /* num caps */
|
|
out_uint32_le(ls, cap_bytes); /* caps len after header */
|
|
|
|
rv = rdpup_send(ls->data, len);
|
|
|
|
if (rv != 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_send_caps: rdpup_send failed"));
|
|
}
|
|
|
|
free_stream(ls);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
process_version_msg(int param1, int param2, int param3, int param4)
|
|
{
|
|
LLOGLN(0, ("process_version_msg: version %d %d %d %d", param1, param2,
|
|
param3, param4));
|
|
|
|
if ((param1 > 0) || (param2 > 0) || (param3 > 0) || (param4 > 0))
|
|
{
|
|
rdpup_send_caps();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_send_rail(void)
|
|
{
|
|
WindowPtr wnd;
|
|
rdpWindowRec *priv;
|
|
|
|
wnd = g_pScreen->root;
|
|
|
|
if (wnd != 0)
|
|
{
|
|
wnd = wnd->lastChild;
|
|
|
|
while (wnd != 0)
|
|
{
|
|
if (wnd->realized)
|
|
{
|
|
priv = GETWINPRIV(wnd);
|
|
priv->status = 1;
|
|
rdpup_create_window(wnd, priv);
|
|
}
|
|
|
|
wnd = wnd->prevSib;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define XR_BUTTON1 1
|
|
#define XR_BUTTON2 2
|
|
#define XR_BUTTON3 4
|
|
#define XR_BUTTON4 8
|
|
#define XR_BUTTON5 16
|
|
#define XR_BUTTON6 32
|
|
#define XR_BUTTON7 64
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpup_process_msg(struct stream *s)
|
|
{
|
|
int msg_type;
|
|
int msg;
|
|
int param1;
|
|
int param2;
|
|
int param3;
|
|
int param4;
|
|
int bytes;
|
|
int i1;
|
|
int flags;
|
|
int x;
|
|
int y;
|
|
int cx;
|
|
int cy;
|
|
int index;
|
|
RegionRec reg;
|
|
BoxRec box;
|
|
|
|
in_uint16_le(s, msg_type);
|
|
|
|
if (msg_type == 103)
|
|
{
|
|
in_uint32_le(s, msg);
|
|
in_uint32_le(s, param1);
|
|
in_uint32_le(s, param2);
|
|
in_uint32_le(s, param3);
|
|
in_uint32_le(s, param4);
|
|
LLOGLN(10, ("rdpup_process_msg - msg %d param1 %d param2 %d param3 %d "
|
|
"param4 %d", msg, param1, param2, param3, param4));
|
|
|
|
switch (msg)
|
|
{
|
|
case 15: /* key down */
|
|
case 16: /* key up */
|
|
KbdAddEvent(msg == 15, param1, param2, param3, param4);
|
|
break;
|
|
case 17: /* from RDP_INPUT_SYNCHRONIZE */
|
|
KbdSync(param1);
|
|
break;
|
|
case 100:
|
|
/* without the minus 2, strange things happen when dragging
|
|
past the width or height */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 101: /* left button up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON1);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 102: /* left button down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON1;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 103: /* right button up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON3);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 104: /* right button down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON3;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 105: /* middle button down */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON2);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 106: /* middle button up */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON2;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 107: /* button 4 up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON4);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 108: /* button 4 down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON4;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 109: /* button 5 up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON5);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 110: /* button 5 down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON5;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 111: /* button 6 up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON6);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 112: /* button 6 down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON6;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 113: /* button 7 up */
|
|
g_button_mask = g_button_mask & (~XR_BUTTON7);
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 114: /* button 7 down */
|
|
g_cursor_x = l_bound_by(param1, 0, g_rdpScreen.width - 2);
|
|
g_cursor_y = l_bound_by(param2, 0, g_rdpScreen.height - 2);
|
|
g_button_mask = g_button_mask | XR_BUTTON7;
|
|
PtrAddEvent(g_button_mask, g_cursor_x, g_cursor_y);
|
|
break;
|
|
case 200:
|
|
rdpup_begin_update();
|
|
rdpup_send_area(0, (param1 >> 16) & 0xffff, param1 & 0xffff,
|
|
(param2 >> 16) & 0xffff, param2 & 0xffff);
|
|
rdpup_end_update();
|
|
break;
|
|
case 300:
|
|
process_screen_size_msg(param1, param2, param3);
|
|
break;
|
|
case 301:
|
|
process_version_msg(param1, param2, param3, param4);
|
|
break;
|
|
}
|
|
}
|
|
else if (msg_type == 104)
|
|
{
|
|
in_uint32_le(s, bytes);
|
|
|
|
if (bytes > sizeof(g_rdpScreen.client_info))
|
|
{
|
|
bytes = sizeof(g_rdpScreen.client_info);
|
|
}
|
|
|
|
memcpy(&(g_rdpScreen.client_info), s->p - 4, bytes);
|
|
g_rdpScreen.client_info.size = bytes;
|
|
LLOGLN(0, ("rdpup_process_msg: got client info bytes %d", bytes));
|
|
LLOGLN(0, (" jpeg support %d", g_rdpScreen.client_info.jpeg));
|
|
i1 = g_rdpScreen.client_info.offscreen_support_level;
|
|
LLOGLN(0, (" offscreen support %d", i1));
|
|
i1 = g_rdpScreen.client_info.offscreen_cache_size;
|
|
LLOGLN(0, (" offscreen size %d", i1));
|
|
i1 = g_rdpScreen.client_info.offscreen_cache_entries;
|
|
LLOGLN(0, (" offscreen entries %d", i1));
|
|
|
|
if (g_rdpScreen.client_info.offscreen_support_level > 0)
|
|
{
|
|
if (g_rdpScreen.client_info.offscreen_cache_entries > 0)
|
|
{
|
|
g_max_os_bitmaps = g_rdpScreen.client_info.offscreen_cache_entries;
|
|
g_free(g_os_bitmaps);
|
|
g_os_bitmaps = (struct rdpup_os_bitmap *)
|
|
g_malloc(sizeof(struct rdpup_os_bitmap) * g_max_os_bitmaps, 1);
|
|
}
|
|
}
|
|
|
|
if (g_rdpScreen.client_info.rail_support_level > 0)
|
|
{
|
|
g_use_rail = 1;
|
|
#ifdef XRDP_WM_RDPUP
|
|
rdpup_send_rail();
|
|
#endif
|
|
}
|
|
if (g_rdpScreen.client_info.orders[0x1b]) /* 27 NEG_GLYPH_INDEX_INDEX */
|
|
{
|
|
LLOGLN(0, (" client supports glyph cache but server disabled"));
|
|
//g_do_glyph_cache = 1;
|
|
}
|
|
if (g_rdpScreen.client_info.order_flags_ex & 0x100)
|
|
{
|
|
g_do_composite = 1;
|
|
}
|
|
if (g_do_glyph_cache)
|
|
{
|
|
LLOGLN(0, (" using glyph cache"));
|
|
}
|
|
if (g_do_composite)
|
|
{
|
|
LLOGLN(0, (" using client composite"));
|
|
}
|
|
LLOGLN(10, ("order_flags_ex 0x%x", g_rdpScreen.client_info.order_flags_ex));
|
|
if (g_rdpScreen.client_info.offscreen_cache_entries == 2000)
|
|
{
|
|
LLOGLN(0, (" client can do offscreen to offscreen blits"));
|
|
g_can_do_pix_to_pix = 1;
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, (" client can not do offscreen to offscreen blits"));
|
|
g_can_do_pix_to_pix = 0;
|
|
}
|
|
if (g_rdpScreen.client_info.pointer_flags & 1)
|
|
{
|
|
LLOGLN(0, (" client can do new(color) cursor"));
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, (" client can not do new(color) cursor"));
|
|
}
|
|
|
|
if (g_rdpScreen.client_info.monitorCount > 0)
|
|
{
|
|
LLOGLN(0, (" client can do multimon"));
|
|
LLOGLN(0, (" client monitor data, monitorCount= %d", g_rdpScreen.client_info.monitorCount));
|
|
box.x1 = g_rdpScreen.client_info.minfo[0].left;
|
|
box.y1 = g_rdpScreen.client_info.minfo[0].top;
|
|
box.x2 = g_rdpScreen.client_info.minfo[0].right;
|
|
box.y2 = g_rdpScreen.client_info.minfo[0].bottom;
|
|
g_do_multimon = 1;
|
|
/* adjust monitor info so it's not negative */
|
|
for (index = 1; index < g_rdpScreen.client_info.monitorCount; index++)
|
|
{
|
|
box.x1 = min(box.x1, g_rdpScreen.client_info.minfo[index].left);
|
|
box.y1 = min(box.y1, g_rdpScreen.client_info.minfo[index].top);
|
|
box.x2 = max(box.x2, g_rdpScreen.client_info.minfo[index].right);
|
|
box.y2 = max(box.y2, g_rdpScreen.client_info.minfo[index].bottom);
|
|
}
|
|
for (index = 0; index < g_rdpScreen.client_info.monitorCount; index++)
|
|
{
|
|
g_rdpScreen.client_info.minfo[index].left -= box.x1;
|
|
g_rdpScreen.client_info.minfo[index].top -= box.y1;
|
|
g_rdpScreen.client_info.minfo[index].right -= box.x1;
|
|
g_rdpScreen.client_info.minfo[index].bottom -= box.y1;
|
|
LLOGLN(0, (" left %d top %d right %d bottom %d",
|
|
g_rdpScreen.client_info.minfo[index].left,
|
|
g_rdpScreen.client_info.minfo[index].top,
|
|
g_rdpScreen.client_info.minfo[index].right,
|
|
g_rdpScreen.client_info.minfo[index].bottom));
|
|
}
|
|
rdpRRSetRdpOutputs();
|
|
RRTellChanged(g_pScreen);
|
|
}
|
|
else
|
|
{
|
|
LLOGLN(0, (" client can not do multimon"));
|
|
g_do_multimon = 0;
|
|
rdpRRSetRdpOutputs();
|
|
RRTellChanged(g_pScreen);
|
|
}
|
|
|
|
rdpLoadLayout(&(g_rdpScreen.client_info));
|
|
|
|
}
|
|
else if (msg_type == 105)
|
|
{
|
|
LLOGLN(10, ("rdpup_process_msg: got msg 105"));
|
|
in_uint32_le(s, flags);
|
|
in_uint32_le(s, g_rect_id_ack);
|
|
in_uint32_le(s, x);
|
|
in_uint32_le(s, y);
|
|
in_uint32_le(s, cx);
|
|
in_uint32_le(s, cy);
|
|
LLOGLN(10, ("rdpup_process_msg: %d %d %d %d", x, y, cx ,cy));
|
|
LLOGLN(10, ("rdpup_process_msg: rect_id %d rect_id_ack %d", g_rect_id, g_rect_id_ack));
|
|
|
|
box.x1 = x;
|
|
box.y1 = y;
|
|
box.x2 = box.x1 + cx;
|
|
box.y2 = box.y1 + cy;
|
|
|
|
RegionInit(®, &box, 0);
|
|
LLOGLN(10, ("rdpup_process_msg: %d %d %d %d", box.x1, box.y1, box.x2, box.y2));
|
|
RegionSubtract(g_shm_reg, g_shm_reg, ®);
|
|
RegionUninit(®);
|
|
|
|
}
|
|
|
|
|
|
else
|
|
{
|
|
rdpLog("unknown message type in rdpup_process_msg %d\n", msg_type);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_get_screen_image_rect(struct image_data *id)
|
|
{
|
|
id->width = g_rdpScreen.width;
|
|
id->height = g_rdpScreen.height;
|
|
id->bpp = g_rdpScreen.rdp_bpp;
|
|
id->Bpp = g_rdpScreen.rdp_Bpp;
|
|
id->lineBytes = g_rdpScreen.paddedWidthInBytes;
|
|
id->pixels = g_rdpScreen.pfbMemory;
|
|
id->shmem_pixels = g_shmemptr;
|
|
id->shmem_id = g_shmemid;
|
|
id->shmem_offset = 0;
|
|
id->shmem_lineBytes = g_shmem_lineBytes;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_get_pixmap_image_rect(PixmapPtr pPixmap, struct image_data *id)
|
|
{
|
|
id->width = pPixmap->drawable.width;
|
|
id->height = pPixmap->drawable.height;
|
|
id->bpp = g_rdpScreen.rdp_bpp;
|
|
id->Bpp = g_rdpScreen.rdp_Bpp;
|
|
id->lineBytes = pPixmap->devKind;
|
|
id->pixels = (char *)(pPixmap->devPrivate.ptr);
|
|
id->shmem_pixels = 0;
|
|
id->shmem_id = 0;
|
|
id->shmem_offset = 0;
|
|
id->shmem_lineBytes = 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_init(void)
|
|
{
|
|
char text[256];
|
|
char *ptext;
|
|
int i;
|
|
const char *socket_dir;
|
|
|
|
socket_dir = g_socket_dir();
|
|
|
|
if (!g_directory_exist(socket_dir))
|
|
{
|
|
if (!g_create_dir(socket_dir))
|
|
{
|
|
LLOGLN(0, ("rdpup_init: g_create_dir(%s) failed", socket_dir));
|
|
return 0;
|
|
}
|
|
|
|
g_chmod_hex(socket_dir, 0x1777);
|
|
}
|
|
|
|
i = atoi(display);
|
|
|
|
if (i < 1)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (g_in_s == 0)
|
|
{
|
|
make_stream(g_in_s);
|
|
init_stream(g_in_s, 8192);
|
|
}
|
|
|
|
if (g_out_s == 0)
|
|
{
|
|
make_stream(g_out_s);
|
|
init_stream(g_out_s, 8192 * g_Bpp + 100);
|
|
}
|
|
|
|
if (g_use_uds)
|
|
{
|
|
g_sprintf(g_uds_data, "%s/xrdp_display_%s", socket_dir, display);
|
|
|
|
if (g_listen_sck == 0)
|
|
{
|
|
g_listen_sck = g_tcp_local_socket_stream();
|
|
|
|
if (g_tcp_local_bind(g_listen_sck, g_uds_data) != 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_init: g_tcp_local_bind failed"));
|
|
return 0;
|
|
}
|
|
|
|
g_tcp_listen(g_listen_sck);
|
|
AddEnabledDevice(g_listen_sck);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
g_sprintf(text, "62%2.2d", i);
|
|
|
|
if (g_listen_sck == 0)
|
|
{
|
|
g_listen_sck = g_tcp_socket();
|
|
|
|
if (g_tcp_bind(g_listen_sck, text) != 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
g_tcp_listen(g_listen_sck);
|
|
AddEnabledDevice(g_listen_sck);
|
|
}
|
|
}
|
|
|
|
g_dis_listen_sck = g_tcp_local_socket_dgram();
|
|
|
|
if (g_dis_listen_sck != 0)
|
|
{
|
|
g_sprintf(text, "%s/xrdp_disconnect_display_%s", socket_dir, display);
|
|
|
|
if (g_tcp_local_bind(g_dis_listen_sck, text) == 0)
|
|
{
|
|
AddEnabledDevice(g_dis_listen_sck);
|
|
}
|
|
else
|
|
{
|
|
rdpLog("g_tcp_local_bind failed [%s]\n", text);
|
|
}
|
|
}
|
|
|
|
ptext = getenv("XRDP_SESMAN_MAX_IDLE_TIME");
|
|
if (ptext != 0)
|
|
{
|
|
}
|
|
ptext = getenv("XRDP_SESMAN_MAX_DISC_TIME");
|
|
if (ptext != 0)
|
|
{
|
|
i = atoi(ptext);
|
|
if (i > 0)
|
|
{
|
|
g_do_kill_disconnected = 1;
|
|
g_disconnect_timeout_s = atoi(ptext);
|
|
}
|
|
}
|
|
ptext = getenv("XRDP_SESMAN_KILL_DISCONNECTED");
|
|
if (ptext != 0)
|
|
{
|
|
i = atoi(ptext);
|
|
if (i != 0)
|
|
{
|
|
g_do_kill_disconnected = 1;
|
|
}
|
|
}
|
|
|
|
if (g_do_kill_disconnected && (g_disconnect_timeout_s < 60))
|
|
{
|
|
g_disconnect_timeout_s = 60;
|
|
}
|
|
|
|
rdpLog("kill disconnected [%d] timeout [%d] sec\n", g_do_kill_disconnected,
|
|
g_disconnect_timeout_s);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_check(void)
|
|
{
|
|
int sel;
|
|
int new_sck;
|
|
char buf[8];
|
|
|
|
sel = g_tcp_select(g_listen_sck, g_sck, g_dis_listen_sck);
|
|
|
|
if (sel & 1)
|
|
{
|
|
new_sck = g_tcp_accept(g_listen_sck);
|
|
|
|
if (new_sck == -1)
|
|
{
|
|
}
|
|
else
|
|
{
|
|
if (g_sck != 0)
|
|
{
|
|
/* should maybe ask is user wants to allow here with timeout */
|
|
rdpLog("replacing connection, already got a connection\n");
|
|
rdpup_disconnect();
|
|
}
|
|
|
|
rdpLog("got a connection\n");
|
|
g_sck = new_sck;
|
|
g_tcp_set_non_blocking(g_sck);
|
|
g_tcp_set_no_delay(g_sck);
|
|
g_connected = 1;
|
|
g_sck_closed = 0;
|
|
g_begin = 0;
|
|
g_con_number++;
|
|
rdpGlyphInit();
|
|
AddEnabledDevice(g_sck);
|
|
|
|
if (g_dis_timer != 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_check: canceling g_dis_timer"));
|
|
TimerCancel(g_dis_timer);
|
|
TimerFree(g_dis_timer);
|
|
g_dis_timer = 0;
|
|
}
|
|
g_disconnect_scheduled = 0;
|
|
|
|
}
|
|
}
|
|
|
|
if (sel & 2)
|
|
{
|
|
if (rdpup_recv_msg(g_in_s) == 0)
|
|
{
|
|
rdpup_process_msg(g_in_s);
|
|
}
|
|
}
|
|
|
|
if (sel & 4)
|
|
{
|
|
if (g_tcp_recv(g_dis_listen_sck, buf, 4, 0) > 0)
|
|
{
|
|
if (g_sck != 0)
|
|
{
|
|
rdpLog("disconnecting session via user request\n");
|
|
rdpup_disconnect();
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_begin_update(void)
|
|
{
|
|
LLOGLN(10, ("rdpup_begin_update:"));
|
|
|
|
if (g_connected)
|
|
{
|
|
if (g_begin)
|
|
{
|
|
return 0;
|
|
}
|
|
init_stream(g_out_s, 0);
|
|
s_push_layer(g_out_s, iso_hdr, 8);
|
|
out_uint16_le(g_out_s, 1); /* begin update */
|
|
out_uint16_le(g_out_s, 4); /* size */
|
|
LLOGLN(10, ("begin %d", g_count));
|
|
g_begin = 1;
|
|
g_count = 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_end_update(void)
|
|
{
|
|
LLOGLN(10, ("rdpup_end_update"));
|
|
|
|
if (g_connected && g_begin)
|
|
{
|
|
if (g_do_dirty_ons)
|
|
{
|
|
/* in this mode, end update is only called in check dirty */
|
|
rdpup_send_pending();
|
|
}
|
|
else
|
|
{
|
|
rdpScheduleDeferredUpdate();
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_pre_check(int in_size)
|
|
{
|
|
int rv;
|
|
|
|
rv = 0;
|
|
if (!g_begin)
|
|
{
|
|
rdpup_begin_update();
|
|
}
|
|
|
|
if ((g_out_s->p - g_out_s->data) > (g_out_s->size - (in_size + 20)))
|
|
{
|
|
s_mark_end(g_out_s);
|
|
if (rdpup_send_msg(g_out_s) != 0)
|
|
{
|
|
LLOGLN(0, ("rdpup_pre_check: rdpup_send_msg failed"));
|
|
rv = 1;
|
|
}
|
|
g_count = 0;
|
|
init_stream(g_out_s, 0);
|
|
s_push_layer(g_out_s, iso_hdr, 8);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_fill_rect(short x, short y, int cx, int cy)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_fill_rect"));
|
|
rdpup_pre_check(12);
|
|
out_uint16_le(g_out_s, 3); /* fill rect */
|
|
out_uint16_le(g_out_s, 12); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_screen_blt(short x, short y, int cx, int cy, short srcx, short srcy)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_screen_blt x %d y %d cx %d cy %d srcx %d srcy %d",
|
|
x, y, cx, cy, srcx, srcy));
|
|
rdpup_pre_check(16);
|
|
out_uint16_le(g_out_s, 4); /* screen blt */
|
|
out_uint16_le(g_out_s, 16); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
out_uint16_le(g_out_s, srcx);
|
|
out_uint16_le(g_out_s, srcy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_clip(short x, short y, int cx, int cy)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_clip"));
|
|
rdpup_pre_check(12);
|
|
out_uint16_le(g_out_s, 10); /* set clip */
|
|
out_uint16_le(g_out_s, 12); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_reset_clip(void)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_reset_clip"));
|
|
rdpup_pre_check(4);
|
|
out_uint16_le(g_out_s, 11); /* reset clip */
|
|
out_uint16_le(g_out_s, 4); /* size */
|
|
g_count++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#define COLOR8(r, g, b) \
|
|
((((r) >> 5) << 0) | (((g) >> 5) << 3) | (((b) >> 6) << 6))
|
|
#define COLOR15(r, g, b) \
|
|
((((r) >> 3) << 10) | (((g) >> 3) << 5) | (((b) >> 3) << 0))
|
|
#define COLOR16(r, g, b) \
|
|
((((r) >> 3) << 11) | (((g) >> 2) << 5) | (((b) >> 3) << 0))
|
|
#define COLOR24(r, g, b) \
|
|
((((r) >> 0) << 0) | (((g) >> 0) << 8) | (((b) >> 0) << 16))
|
|
#define SPLITCOLOR32(r, g, b, c) \
|
|
{ \
|
|
r = ((c) >> 16) & 0xff; \
|
|
g = ((c) >> 8) & 0xff; \
|
|
b = (c) & 0xff; \
|
|
}
|
|
|
|
int
|
|
convert_pixel(int in_pixel)
|
|
{
|
|
int red;
|
|
int green;
|
|
int blue;
|
|
int rv;
|
|
|
|
rv = 0;
|
|
|
|
if (g_rdpScreen.depth == 24)
|
|
{
|
|
if (g_rdpScreen.rdp_bpp >= 24)
|
|
{
|
|
rv = in_pixel;
|
|
SPLITCOLOR32(red, green, blue, rv);
|
|
rv = COLOR24(red, green, blue);
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 16)
|
|
{
|
|
rv = in_pixel;
|
|
SPLITCOLOR32(red, green, blue, rv);
|
|
rv = COLOR16(red, green, blue);
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 15)
|
|
{
|
|
rv = in_pixel;
|
|
SPLITCOLOR32(red, green, blue, rv);
|
|
rv = COLOR15(red, green, blue);
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 8)
|
|
{
|
|
rv = in_pixel;
|
|
SPLITCOLOR32(red, green, blue, rv);
|
|
rv = COLOR8(red, green, blue);
|
|
}
|
|
}
|
|
else if (g_rdpScreen.depth == g_rdpScreen.rdp_bpp)
|
|
{
|
|
return in_pixel;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
int
|
|
convert_pixels(void *src, void *dst, int num_pixels)
|
|
{
|
|
unsigned int pixel;
|
|
unsigned int red;
|
|
unsigned int green;
|
|
unsigned int blue;
|
|
unsigned int *src32;
|
|
unsigned int *dst32;
|
|
unsigned short *dst16;
|
|
unsigned char *dst8;
|
|
int index;
|
|
|
|
if (g_rdpScreen.depth == g_rdpScreen.rdp_bpp)
|
|
{
|
|
memcpy(dst, src, num_pixels * g_Bpp);
|
|
return 0;
|
|
}
|
|
|
|
if (g_rdpScreen.depth == 24)
|
|
{
|
|
src32 = (unsigned int *)src;
|
|
|
|
if (g_rdpScreen.rdp_bpp >= 24)
|
|
{
|
|
dst32 = (unsigned int *)dst;
|
|
|
|
for (index = 0; index < num_pixels; index++)
|
|
{
|
|
pixel = *src32;
|
|
*dst32 = pixel;
|
|
dst32++;
|
|
src32++;
|
|
}
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 16)
|
|
{
|
|
dst16 = (unsigned short *)dst;
|
|
|
|
for (index = 0; index < num_pixels; index++)
|
|
{
|
|
pixel = *src32;
|
|
SPLITCOLOR32(red, green, blue, pixel);
|
|
pixel = COLOR16(red, green, blue);
|
|
*dst16 = pixel;
|
|
dst16++;
|
|
src32++;
|
|
}
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 15)
|
|
{
|
|
dst16 = (unsigned short *)dst;
|
|
|
|
for (index = 0; index < num_pixels; index++)
|
|
{
|
|
pixel = *src32;
|
|
SPLITCOLOR32(red, green, blue, pixel);
|
|
pixel = COLOR15(red, green, blue);
|
|
*dst16 = pixel;
|
|
dst16++;
|
|
src32++;
|
|
}
|
|
}
|
|
else if (g_rdpScreen.rdp_bpp == 8)
|
|
{
|
|
dst8 = (unsigned char *)dst;
|
|
|
|
for (index = 0; index < num_pixels; index++)
|
|
{
|
|
pixel = *src32;
|
|
SPLITCOLOR32(red, green, blue, pixel);
|
|
pixel = COLOR8(red, green, blue);
|
|
*dst8 = pixel;
|
|
dst8++;
|
|
src32++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
alpha_pixels(void* src, void* dst, int num_pixels)
|
|
{
|
|
unsigned int* src32;
|
|
unsigned char* dst8;
|
|
int index;
|
|
|
|
src32 = (unsigned int*)src;
|
|
dst8 = (unsigned char*)dst;
|
|
for (index = 0; index < num_pixels; index++)
|
|
{
|
|
*dst8 = (*src32) >> 24;
|
|
dst8++;
|
|
src32++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_fgcolor(int fgcolor)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_fgcolor"));
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 12); /* set fgcolor */
|
|
out_uint16_le(g_out_s, 8); /* size */
|
|
g_count++;
|
|
fgcolor = fgcolor & g_Bpp_mask;
|
|
fgcolor = convert_pixel(fgcolor) & g_rdpScreen.rdp_Bpp_mask;
|
|
out_uint32_le(g_out_s, fgcolor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_bgcolor(int bgcolor)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_bgcolor"));
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 13); /* set bg color */
|
|
out_uint16_le(g_out_s, 8); /* size */
|
|
g_count++;
|
|
bgcolor = bgcolor & g_Bpp_mask;
|
|
bgcolor = convert_pixel(bgcolor) & g_rdpScreen.rdp_Bpp_mask;
|
|
out_uint32_le(g_out_s, bgcolor);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_opcode(int opcode)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_opcode"));
|
|
rdpup_pre_check(6);
|
|
out_uint16_le(g_out_s, 14); /* set opcode */
|
|
out_uint16_le(g_out_s, 6); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, g_rdp_opcodes[opcode & 0xf]);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_pen(int style, int width)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_pen"));
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 17); /* set pen */
|
|
out_uint16_le(g_out_s, 8); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, style);
|
|
out_uint16_le(g_out_s, width);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_draw_line(short x1, short y1, short x2, short y2)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_draw_line"));
|
|
rdpup_pre_check(12);
|
|
out_uint16_le(g_out_s, 18); /* draw line */
|
|
out_uint16_le(g_out_s, 12); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, x1);
|
|
out_uint16_le(g_out_s, y1);
|
|
out_uint16_le(g_out_s, x2);
|
|
out_uint16_le(g_out_s, y2);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_cursor(short x, short y, char *cur_data, char *cur_mask)
|
|
{
|
|
int size;
|
|
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_cursor"));
|
|
size = 8 + 32 * (32 * 3) + 32 * (32 / 8);
|
|
rdpup_pre_check(size);
|
|
out_uint16_le(g_out_s, 19); /* set cursor */
|
|
out_uint16_le(g_out_s, size); /* size */
|
|
g_count++;
|
|
x = MAX(0, x);
|
|
x = MIN(31, x);
|
|
y = MAX(0, y);
|
|
y = MIN(31, y);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint8a(g_out_s, cur_data, 32 * (32 * 3));
|
|
out_uint8a(g_out_s, cur_mask, 32 * (32 / 8));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_set_cursor_ex(short x, short y, char *cur_data, char *cur_mask, int bpp)
|
|
{
|
|
int size;
|
|
int Bpp;
|
|
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_set_cursor_ex"));
|
|
Bpp = (bpp == 0) ? 3 : (bpp + 7) / 8;
|
|
size = 10 + 32 * (32 * Bpp) + 32 * (32 / 8);
|
|
rdpup_pre_check(size);
|
|
out_uint16_le(g_out_s, 51); /* set cursor ex */
|
|
out_uint16_le(g_out_s, size); /* size */
|
|
g_count++;
|
|
x = MAX(0, x);
|
|
x = MIN(31, x);
|
|
y = MAX(0, y);
|
|
y = MIN(31, y);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, bpp);
|
|
out_uint8a(g_out_s, cur_data, 32 * (32 * Bpp));
|
|
out_uint8a(g_out_s, cur_mask, 32 * (32 / 8));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_create_os_surface(int rdpindex, int width, int height)
|
|
{
|
|
LLOGLN(10, ("rdpup_create_os_surface:"));
|
|
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_create_os_surface width %d height %d", width, height));
|
|
rdpup_pre_check(12);
|
|
out_uint16_le(g_out_s, 20);
|
|
out_uint16_le(g_out_s, 12);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, rdpindex);
|
|
out_uint16_le(g_out_s, width);
|
|
out_uint16_le(g_out_s, height);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_create_os_surface_bpp(int rdpindex, int width, int height, int bpp)
|
|
{
|
|
LLOGLN(10, ("rdpup_create_os_surface_bpp:"));
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" width %d height %d bpp %d", width, height, bpp));
|
|
rdpup_pre_check(13);
|
|
out_uint16_le(g_out_s, 31);
|
|
out_uint16_le(g_out_s, 13);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, rdpindex);
|
|
out_uint16_le(g_out_s, width);
|
|
out_uint16_le(g_out_s, height);
|
|
out_uint8(g_out_s, bpp);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_switch_os_surface(int rdpindex)
|
|
{
|
|
LLOGLN(10, ("rdpup_switch_os_surface:"));
|
|
|
|
if (g_connected)
|
|
{
|
|
if (g_rdpindex == rdpindex)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
g_rdpindex = rdpindex;
|
|
LLOGLN(10, ("rdpup_switch_os_surface: rdpindex %d", rdpindex));
|
|
/* switch surface */
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 21);
|
|
out_uint16_le(g_out_s, 8);
|
|
out_uint32_le(g_out_s, rdpindex);
|
|
g_count++;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_delete_os_surface(int rdpindex)
|
|
{
|
|
LLOGLN(10, ("rdpup_delete_os_surface: rdpindex %d", rdpindex));
|
|
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, ("rdpup_delete_os_surface: rdpindex %d", rdpindex));
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 22);
|
|
out_uint16_le(g_out_s, 8);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, rdpindex);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
get_single_color(struct image_data *id, int x, int y, int w, int h)
|
|
{
|
|
int rv;
|
|
int i;
|
|
int j;
|
|
int p;
|
|
unsigned char *i8;
|
|
unsigned short *i16;
|
|
unsigned int *i32;
|
|
|
|
p = 0;
|
|
rv = -1;
|
|
|
|
if (g_Bpp == 1)
|
|
{
|
|
for (i = 0; i < h; i++)
|
|
{
|
|
i8 = (unsigned char *)(id->pixels +
|
|
((y + i) * id->lineBytes) + (x * g_Bpp));
|
|
|
|
if (i == 0)
|
|
{
|
|
p = *i8;
|
|
}
|
|
|
|
for (j = 0; j < w; j++)
|
|
{
|
|
if (i8[j] != p)
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
rv = p;
|
|
}
|
|
else if (g_Bpp == 2)
|
|
{
|
|
for (i = 0; i < h; i++)
|
|
{
|
|
i16 = (unsigned short *)(id->pixels +
|
|
((y + i) * id->lineBytes) + (x * g_Bpp));
|
|
|
|
if (i == 0)
|
|
{
|
|
p = *i16;
|
|
}
|
|
|
|
for (j = 0; j < w; j++)
|
|
{
|
|
if (i16[j] != p)
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
rv = p;
|
|
}
|
|
else if (g_Bpp == 4)
|
|
{
|
|
for (i = 0; i < h; i++)
|
|
{
|
|
i32 = (unsigned int *)(id->pixels +
|
|
((y + i) * id->lineBytes) + (x * g_Bpp));
|
|
|
|
if (i == 0)
|
|
{
|
|
p = *i32;
|
|
}
|
|
|
|
for (j = 0; j < w; j++)
|
|
{
|
|
if (i32[j] != p)
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
rv = p;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* split the bitmap up into 64 x 64 pixel areas
|
|
or send using shared memory */
|
|
void
|
|
rdpup_send_area(struct image_data *id, int x, int y, int w, int h)
|
|
{
|
|
char *s;
|
|
char *d;
|
|
int i;
|
|
int single_color;
|
|
int lx;
|
|
int ly;
|
|
int lh;
|
|
int lw;
|
|
int size;
|
|
int safety;
|
|
struct image_data lid;
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
LLOGLN(10, ("rdpup_send_area: id %p x %d y %d w %d h %d", id, x, y, w, h));
|
|
|
|
if (id == 0)
|
|
{
|
|
rdpup_get_screen_image_rect(&lid);
|
|
id = &lid;
|
|
}
|
|
|
|
if (x >= id->width)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (y >= id->height)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (x < 0)
|
|
{
|
|
w += x;
|
|
x = 0;
|
|
}
|
|
|
|
if (y < 0)
|
|
{
|
|
h += y;
|
|
y = 0;
|
|
}
|
|
|
|
if (w <= 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (h <= 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (x + w > id->width)
|
|
{
|
|
w = id->width - x;
|
|
}
|
|
|
|
if (y + h > id->height)
|
|
{
|
|
h = id->height - y;
|
|
}
|
|
|
|
LLOGLN(10, ("%d", w * h));
|
|
|
|
if (g_connected && g_begin)
|
|
{
|
|
LLOGLN(10, (" rdpup_send_area"));
|
|
|
|
if (id->shmem_pixels != 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_send_area: using shmem"));
|
|
box.x1 = x;
|
|
box.y1 = y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
LLOGLN(10, ("rdpup_send_area: 1 x %d y %d w %d h %d", x, y, w, h));
|
|
safety = 0;
|
|
while (RegionContainsRect(g_shm_reg, &box))
|
|
{
|
|
/* instead of rdpup_end_update, call rdpup_send_pending */
|
|
rdpup_send_pending();
|
|
rdpup_begin_update();
|
|
safety++;
|
|
if (safety > 100)
|
|
{
|
|
LLOGLN(0, ("rdpup_send_area: shmem timeout"));
|
|
break;
|
|
}
|
|
if (sck_can_recv(g_sck, 100))
|
|
{
|
|
if (rdpup_recv_msg(g_in_s) == 0)
|
|
{
|
|
rdpup_process_msg(g_in_s);
|
|
}
|
|
}
|
|
}
|
|
s = id->pixels;
|
|
s += y * id->lineBytes;
|
|
s += x * g_Bpp;
|
|
d = id->shmem_pixels + id->shmem_offset;
|
|
d += y * id->shmem_lineBytes;
|
|
d += x * g_rdpScreen.rdp_Bpp;
|
|
ly = y;
|
|
while (ly < y + h)
|
|
{
|
|
convert_pixels(s, d, w);
|
|
s += id->lineBytes;
|
|
d += id->shmem_lineBytes;
|
|
ly += 1;
|
|
}
|
|
size = 36;
|
|
rdpup_pre_check(size);
|
|
out_uint16_le(g_out_s, 60);
|
|
out_uint16_le(g_out_s, size);
|
|
g_count++;
|
|
LLOGLN(10, ("rdpup_send_area: 2 x %d y %d w %d h %d", x, y, w, h));
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, w);
|
|
out_uint16_le(g_out_s, h);
|
|
out_uint32_le(g_out_s, 0);
|
|
g_rect_id++;
|
|
out_uint32_le(g_out_s, g_rect_id);
|
|
out_uint32_le(g_out_s, id->shmem_id);
|
|
out_uint32_le(g_out_s, id->shmem_offset);
|
|
out_uint16_le(g_out_s, id->width);
|
|
out_uint16_le(g_out_s, id->height);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
RegionInit(®, &box, 0);
|
|
RegionUnion(g_shm_reg, g_shm_reg, ®);
|
|
RegionUninit(®);
|
|
return;
|
|
}
|
|
|
|
ly = y;
|
|
while ((ly < y + h) && g_connected)
|
|
{
|
|
lx = x;
|
|
|
|
while (lx < x + w)
|
|
{
|
|
lw = MIN(64, (x + w) - lx);
|
|
lh = MIN(64, (y + h) - ly);
|
|
single_color = get_single_color(id, lx, ly, lw, lh);
|
|
|
|
if (single_color != -1)
|
|
{
|
|
LLOGLN(10, ("%d sending single color", g_count));
|
|
rdpup_set_fgcolor(single_color);
|
|
rdpup_fill_rect(lx, ly, lw, lh);
|
|
}
|
|
else
|
|
{
|
|
size = lw * lh * id->Bpp + 24;
|
|
rdpup_pre_check(size);
|
|
out_uint16_le(g_out_s, 5);
|
|
out_uint16_le(g_out_s, size);
|
|
g_count++;
|
|
out_uint16_le(g_out_s, lx);
|
|
out_uint16_le(g_out_s, ly);
|
|
out_uint16_le(g_out_s, lw);
|
|
out_uint16_le(g_out_s, lh);
|
|
out_uint32_le(g_out_s, lw * lh * id->Bpp);
|
|
|
|
for (i = 0; i < lh; i++)
|
|
{
|
|
s = (id->pixels +
|
|
((ly + i) * id->lineBytes) + (lx * g_Bpp));
|
|
convert_pixels(s, g_out_s->p, lw);
|
|
g_out_s->p += lw * id->Bpp;
|
|
}
|
|
|
|
out_uint16_le(g_out_s, lw);
|
|
out_uint16_le(g_out_s, lh);
|
|
out_uint16_le(g_out_s, 0);
|
|
out_uint16_le(g_out_s, 0);
|
|
}
|
|
|
|
lx += 64;
|
|
}
|
|
|
|
ly += 64;
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* split the bitmap up into 64 x 64 pixel areas */
|
|
void
|
|
rdpup_send_alpha_area(struct image_data* id, int x, int y, int w, int h)
|
|
{
|
|
char* s;
|
|
int i;
|
|
int lx;
|
|
int ly;
|
|
int lh;
|
|
int lw;
|
|
int size;
|
|
struct image_data lid;
|
|
|
|
LLOGLN(10, ("rdpup_send_alpha_area: id %p x %d y %d w %d h %d",
|
|
id, x, y, w, h));
|
|
if (id == 0)
|
|
{
|
|
rdpup_get_screen_image_rect(&lid);
|
|
id = &lid;
|
|
}
|
|
|
|
if (x >= id->width)
|
|
{
|
|
return;
|
|
}
|
|
if (y >= id->height)
|
|
{
|
|
return;
|
|
}
|
|
if (x < 0)
|
|
{
|
|
w += x;
|
|
x = 0;
|
|
}
|
|
if (y < 0)
|
|
{
|
|
h += y;
|
|
y = 0;
|
|
}
|
|
if (w <= 0)
|
|
{
|
|
return;
|
|
}
|
|
if (h <= 0)
|
|
{
|
|
return;
|
|
}
|
|
if (x + w > id->width)
|
|
{
|
|
w = id->width - x;
|
|
}
|
|
if (y + h > id->height)
|
|
{
|
|
h = id->height - y;
|
|
}
|
|
LLOGLN(10, ("%d", w * h));
|
|
if (g_connected && g_begin)
|
|
{
|
|
LLOGLN(10, (" rdpup_send_area"));
|
|
ly = y;
|
|
while ((ly < y + h) && g_connected)
|
|
{
|
|
lx = x;
|
|
while ((lx < x + w) && g_connected)
|
|
{
|
|
lw = MIN(64, (x + w) - lx);
|
|
lh = MIN(64, (y + h) - ly);
|
|
size = lw * lh + 25;
|
|
rdpup_pre_check(size);
|
|
out_uint16_le(g_out_s, 32); /* server_paint_rect_bpp */
|
|
out_uint16_le(g_out_s, size);
|
|
g_count++;
|
|
out_uint16_le(g_out_s, lx);
|
|
out_uint16_le(g_out_s, ly);
|
|
out_uint16_le(g_out_s, lw);
|
|
out_uint16_le(g_out_s, lh);
|
|
out_uint32_le(g_out_s, lw * lh);
|
|
for (i = 0; i < lh; i++)
|
|
{
|
|
s = (id->pixels +
|
|
((ly + i) * id->lineBytes) + (lx * g_Bpp));
|
|
alpha_pixels(s, g_out_s->p, lw);
|
|
g_out_s->p += lw;
|
|
}
|
|
out_uint16_le(g_out_s, lw);
|
|
out_uint16_le(g_out_s, lh);
|
|
out_uint16_le(g_out_s, 0);
|
|
out_uint16_le(g_out_s, 0);
|
|
out_uint8(g_out_s, 8);
|
|
lx += 64;
|
|
}
|
|
ly += 64;
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_paint_rect_os(int x, int y, int cx, int cy,
|
|
int rdpindex, int srcx, int srcy)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
rdpup_pre_check(20);
|
|
out_uint16_le(g_out_s, 23);
|
|
out_uint16_le(g_out_s, 20);
|
|
g_count++;
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
out_uint32_le(g_out_s, rdpindex);
|
|
out_uint16_le(g_out_s, srcx);
|
|
out_uint16_le(g_out_s, srcy);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_set_hints(int hints, int mask)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
rdpup_pre_check(12);
|
|
out_uint16_le(g_out_s, 24);
|
|
out_uint16_le(g_out_s, 12);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, hints);
|
|
out_uint32_le(g_out_s, mask);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_create_window(WindowPtr pWindow, rdpWindowRec *priv)
|
|
{
|
|
int bytes;
|
|
int index;
|
|
int flags;
|
|
int num_window_rects;
|
|
int num_visibility_rects;
|
|
int title_bytes;
|
|
int style;
|
|
int ext_style;
|
|
int root_id;
|
|
char title[256];
|
|
|
|
LLOGLN(10, ("rdpup_create_window: id 0x%8.8x",
|
|
(int)(pWindow->drawable.id)));
|
|
|
|
if (g_connected)
|
|
{
|
|
root_id = pWindow->drawable.pScreen->root->drawable.id;
|
|
|
|
if (pWindow->overrideRedirect)
|
|
{
|
|
style = XR_STYLE_TOOLTIP;
|
|
ext_style = XR_EXT_STYLE_TOOLTIP;
|
|
}
|
|
else
|
|
{
|
|
style = XR_STYLE_NORMAL;
|
|
ext_style = XR_EXT_STYLE_NORMAL;
|
|
}
|
|
|
|
flags = WINDOW_ORDER_TYPE_WINDOW | WINDOW_ORDER_STATE_NEW;
|
|
strcpy(title, "title");
|
|
title_bytes = strlen(title);
|
|
|
|
num_window_rects = 1;
|
|
num_visibility_rects = 1;
|
|
|
|
/* calculate bytes */
|
|
bytes = (2 + 2) + (5 * 4) + (2 + title_bytes) + (12 * 4) +
|
|
(2 + num_window_rects * 8) + (4 + 4) +
|
|
(2 + num_visibility_rects * 8) + 4;
|
|
|
|
rdpup_pre_check(bytes);
|
|
out_uint16_le(g_out_s, 25);
|
|
out_uint16_le(g_out_s, bytes);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, pWindow->drawable.id); /* window_id */
|
|
out_uint32_le(g_out_s, pWindow->parent->drawable.id); /* owner_window_id */
|
|
flags |= WINDOW_ORDER_FIELD_OWNER;
|
|
out_uint32_le(g_out_s, style); /* style */
|
|
out_uint32_le(g_out_s, ext_style); /* extended_style */
|
|
flags |= WINDOW_ORDER_FIELD_STYLE;
|
|
out_uint32_le(g_out_s, 0x05); /* show_state */
|
|
flags |= WINDOW_ORDER_FIELD_SHOW;
|
|
out_uint16_le(g_out_s, title_bytes); /* title_info */
|
|
out_uint8a(g_out_s, title, title_bytes);
|
|
flags |= WINDOW_ORDER_FIELD_TITLE;
|
|
out_uint32_le(g_out_s, 0); /* client_offset_x */
|
|
out_uint32_le(g_out_s, 0); /* client_offset_y */
|
|
flags |= WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET;
|
|
out_uint32_le(g_out_s, pWindow->drawable.width); /* client_area_width */
|
|
out_uint32_le(g_out_s, pWindow->drawable.height); /* client_area_height */
|
|
flags |= WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE;
|
|
out_uint32_le(g_out_s, 0); /* rp_content */
|
|
out_uint32_le(g_out_s, root_id); /* root_parent_handle */
|
|
flags |= WINDOW_ORDER_FIELD_ROOT_PARENT;
|
|
out_uint32_le(g_out_s, pWindow->drawable.x); /* window_offset_x */
|
|
out_uint32_le(g_out_s, pWindow->drawable.y); /* window_offset_y */
|
|
flags |= WINDOW_ORDER_FIELD_WND_OFFSET;
|
|
out_uint32_le(g_out_s, 0); /* window_client_delta_x */
|
|
out_uint32_le(g_out_s, 0); /* window_client_delta_y */
|
|
flags |= WINDOW_ORDER_FIELD_WND_CLIENT_DELTA;
|
|
out_uint32_le(g_out_s, pWindow->drawable.width); /* window_width */
|
|
out_uint32_le(g_out_s, pWindow->drawable.height); /* window_height */
|
|
flags |= WINDOW_ORDER_FIELD_WND_SIZE;
|
|
out_uint16_le(g_out_s, num_window_rects); /* num_window_rects */
|
|
|
|
for (index = 0; index < num_window_rects; index++)
|
|
{
|
|
out_uint16_le(g_out_s, 0); /* left */
|
|
out_uint16_le(g_out_s, 0); /* top */
|
|
out_uint16_le(g_out_s, pWindow->drawable.width); /* right */
|
|
out_uint16_le(g_out_s, pWindow->drawable.height); /* bottom */
|
|
}
|
|
|
|
flags |= WINDOW_ORDER_FIELD_WND_RECTS;
|
|
out_uint32_le(g_out_s, pWindow->drawable.x); /* visible_offset_x */
|
|
out_uint32_le(g_out_s, pWindow->drawable.y); /* visible_offset_y */
|
|
flags |= WINDOW_ORDER_FIELD_VIS_OFFSET;
|
|
out_uint16_le(g_out_s, num_visibility_rects); /* num_visibility_rects */
|
|
|
|
for (index = 0; index < num_visibility_rects; index++)
|
|
{
|
|
out_uint16_le(g_out_s, 0); /* left */
|
|
out_uint16_le(g_out_s, 0); /* top */
|
|
out_uint16_le(g_out_s, pWindow->drawable.width); /* right */
|
|
out_uint16_le(g_out_s, pWindow->drawable.height); /* bottom */
|
|
}
|
|
|
|
flags |= WINDOW_ORDER_FIELD_VISIBILITY;
|
|
|
|
out_uint32_le(g_out_s, flags); /* flags */
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_delete_window(WindowPtr pWindow, rdpWindowRec *priv)
|
|
{
|
|
LLOGLN(10, ("rdpup_delete_window: id 0x%8.8x",
|
|
(int)(pWindow->drawable.id)));
|
|
|
|
if (g_connected)
|
|
{
|
|
rdpup_pre_check(8);
|
|
out_uint16_le(g_out_s, 26);
|
|
out_uint16_le(g_out_s, 8);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, pWindow->drawable.id); /* window_id */
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpup_show_window(WindowPtr pWindow, rdpWindowRec* priv, int showState)
|
|
{
|
|
LLOGLN(10, ("rdpup_show_window: id 0x%8.8x state 0x%x", pWindow->drawable.id,
|
|
showState));
|
|
if (g_connected)
|
|
{
|
|
int flags = WINDOW_ORDER_TYPE_WINDOW;
|
|
|
|
rdpup_pre_check(16);
|
|
out_uint16_le(g_out_s, 27);
|
|
out_uint16_le(g_out_s, 16);
|
|
g_count++;
|
|
out_uint32_le(g_out_s, pWindow->drawable.id);
|
|
flags |= WINDOW_ORDER_FIELD_SHOW;
|
|
out_uint32_le(g_out_s, flags);
|
|
out_uint32_le(g_out_s, showState);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_check_dirty(PixmapPtr pDirtyPixmap, rdpPixmapRec *pDirtyPriv)
|
|
{
|
|
int index;
|
|
int clip_index;
|
|
int count;
|
|
int num_clips;
|
|
BoxRec box;
|
|
xSegment *seg;
|
|
struct image_data id;
|
|
struct rdp_draw_item *di;
|
|
struct rdp_text* rtext;
|
|
struct rdp_text* trtext;
|
|
|
|
if (pDirtyPriv == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (pDirtyPriv->is_dirty == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
LLOGLN(10, ("rdpup_check_dirty: got dirty"));
|
|
rdpup_switch_os_surface(pDirtyPriv->rdpindex);
|
|
rdpup_get_pixmap_image_rect(pDirtyPixmap, &id);
|
|
rdpup_begin_update();
|
|
draw_item_pack(pDirtyPixmap, pDirtyPriv);
|
|
di = pDirtyPriv->draw_item_head;
|
|
|
|
while (di != 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_check_dirty: type %d", di->type));
|
|
|
|
switch (di->type)
|
|
{
|
|
case RDI_FILL:
|
|
rdpup_set_fgcolor(di->u.fill.fg_color);
|
|
rdpup_set_opcode(di->u.fill.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_FILL %d %d %d %d", box.x1, box.y1,
|
|
box.x2, box.y2));
|
|
rdpup_fill_rect(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_IMGLL:
|
|
rdpup_set_hints(1, 1);
|
|
rdpup_set_opcode(di->u.img.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_IMGLL %d %d %d %d", box.x1, box.y1,
|
|
box.x2, box.y2));
|
|
rdpup_send_area(&id, box.x1, box.y1, box.x2 - box.x1,
|
|
box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_set_hints(0, 1);
|
|
break;
|
|
case RDI_IMGLY:
|
|
rdpup_set_opcode(di->u.img.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_IMGLY %d %d %d %d", box.x1, box.y1,
|
|
box.x2, box.y2));
|
|
rdpup_send_area(&id, box.x1, box.y1, box.x2 - box.x1,
|
|
box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_LINE:
|
|
LLOGLN(10, (" RDI_LINE"));
|
|
num_clips = REGION_NUM_RECTS(di->reg);
|
|
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_set_fgcolor(di->u.line.fg_color);
|
|
rdpup_set_opcode(di->u.line.opcode);
|
|
rdpup_set_pen(0, di->u.line.width);
|
|
|
|
for (clip_index = num_clips - 1; clip_index >= 0; clip_index--)
|
|
{
|
|
box = REGION_RECTS(di->reg)[clip_index];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
|
|
for (index = 0; index < di->u.line.nseg; index++)
|
|
{
|
|
seg = di->u.line.segs + index;
|
|
LLOGLN(10, (" RDI_LINE %d %d %d %d", seg->x1, seg->y1,
|
|
seg->x2, seg->y2));
|
|
rdpup_draw_line(seg->x1, seg->y1, seg->x2, seg->y2);
|
|
}
|
|
}
|
|
}
|
|
|
|
rdpup_reset_clip();
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_SCRBLT:
|
|
LLOGLN(10, (" RDI_SCRBLT"));
|
|
break;
|
|
case RDI_TEXT:
|
|
LLOGLN(10, (" RDI_TEXT"));
|
|
num_clips = REGION_NUM_RECTS(di->reg);
|
|
if (num_clips > 0)
|
|
{
|
|
LLOGLN(10, (" num_clips %d", num_clips));
|
|
rdpup_set_fgcolor(di->u.text.fg_color);
|
|
rdpup_set_opcode(di->u.text.opcode);
|
|
rtext = di->u.text.rtext;
|
|
trtext = rtext;
|
|
while (trtext != 0)
|
|
{
|
|
rdp_text_chars_to_data(trtext);
|
|
for (clip_index = num_clips - 1; clip_index >= 0; clip_index--)
|
|
{
|
|
box = REGION_RECTS(di->reg)[clip_index];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
LLOGLN(10, (" %d %d %d %d", box.x1, box.y1, box.x2, box.y2));
|
|
box = RegionExtents(trtext->reg)[0];
|
|
rdpup_draw_text(trtext->font, trtext->flags, trtext->mixmode,
|
|
box.x1, box.y1, box.x2, box.y2,
|
|
//box.x1, box.y1, box.x2, box.y2,
|
|
0, 0, 0, 0,
|
|
trtext->x, trtext->y, trtext->data, trtext->data_bytes);
|
|
}
|
|
trtext = trtext->next;
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
}
|
|
|
|
di = di->next;
|
|
}
|
|
|
|
draw_item_remove_all(pDirtyPriv);
|
|
rdpup_end_update();
|
|
pDirtyPriv->is_dirty = 0;
|
|
rdpup_switch_os_surface(-1);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_check_dirty_screen(rdpPixmapRec *pDirtyPriv)
|
|
{
|
|
int index;
|
|
int clip_index;
|
|
int count;
|
|
int num_clips;
|
|
BoxRec box;
|
|
xSegment *seg;
|
|
struct image_data id;
|
|
struct rdp_draw_item *di;
|
|
|
|
if (pDirtyPriv == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
if (pDirtyPriv->is_dirty == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
LLOGLN(10, ("rdpup_check_dirty_screen: got dirty"));
|
|
rdpup_get_screen_image_rect(&id);
|
|
rdpup_begin_update();
|
|
draw_item_pack(0, pDirtyPriv);
|
|
di = pDirtyPriv->draw_item_head;
|
|
|
|
while (di != 0)
|
|
{
|
|
LLOGLN(10, ("rdpup_check_dirty_screen: type %d", di->type));
|
|
|
|
switch (di->type)
|
|
{
|
|
case RDI_FILL:
|
|
rdpup_set_fgcolor(di->u.fill.fg_color);
|
|
rdpup_set_opcode(di->u.fill.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_FILL %d %d %d %d", box.x1, box.y1,
|
|
box.x2, box.y2));
|
|
rdpup_fill_rect(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_IMGLL:
|
|
rdpup_set_hints(1, 1);
|
|
rdpup_set_opcode(di->u.img.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_IMGLL x %d y %d w %d h %d", box.x1, box.y1,
|
|
box.x2 - box.x1, box.y2 - box.y1));
|
|
rdpup_send_area(&id, box.x1, box.y1, box.x2 - box.x1,
|
|
box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_set_hints(0, 1);
|
|
break;
|
|
case RDI_IMGLY:
|
|
rdpup_set_opcode(di->u.img.opcode);
|
|
count = REGION_NUM_RECTS(di->reg);
|
|
|
|
for (index = 0; index < count; index++)
|
|
{
|
|
box = REGION_RECTS(di->reg)[index];
|
|
LLOGLN(10, (" RDI_IMGLY %d %d %d %d", box.x1, box.y1,
|
|
box.x2, box.y2));
|
|
rdpup_send_area(&id, box.x1, box.y1, box.x2 - box.x1,
|
|
box.y2 - box.y1);
|
|
}
|
|
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_LINE:
|
|
LLOGLN(10, (" RDI_LINE"));
|
|
num_clips = REGION_NUM_RECTS(di->reg);
|
|
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_set_fgcolor(di->u.line.fg_color);
|
|
rdpup_set_opcode(di->u.line.opcode);
|
|
rdpup_set_pen(0, di->u.line.width);
|
|
|
|
for (clip_index = num_clips - 1; clip_index >= 0; clip_index--)
|
|
{
|
|
box = REGION_RECTS(di->reg)[clip_index];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
|
|
for (index = 0; index < di->u.line.nseg; index++)
|
|
{
|
|
seg = di->u.line.segs + index;
|
|
LLOGLN(10, (" RDI_LINE %d %d %d %d", seg->x1, seg->y1,
|
|
seg->x2, seg->y2));
|
|
rdpup_draw_line(seg->x1, seg->y1, seg->x2, seg->y2);
|
|
}
|
|
}
|
|
}
|
|
|
|
rdpup_reset_clip();
|
|
rdpup_set_opcode(GXcopy);
|
|
break;
|
|
case RDI_SCRBLT:
|
|
LLOGLN(10, (" RDI_SCRBLT"));
|
|
break;
|
|
}
|
|
|
|
di = di->next;
|
|
}
|
|
|
|
draw_item_remove_all(pDirtyPriv);
|
|
rdpup_end_update();
|
|
pDirtyPriv->is_dirty = 0;
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_check_alpha_dirty(PixmapPtr pDirtyPixmap, rdpPixmapRec* pDirtyPriv)
|
|
{
|
|
struct image_data id;
|
|
|
|
LLOGLN(10, ("rdpup_check_alpha_dirty: width %d height %d",
|
|
pDirtyPixmap->drawable.width, pDirtyPixmap->drawable.height));
|
|
if (pDirtyPriv == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
LLOGLN(10, ("rdpup_check_alpha_dirty: is_alpha_dirty_not %d",
|
|
pDirtyPriv->is_alpha_dirty_not));
|
|
if (pDirtyPriv->is_alpha_dirty_not)
|
|
{
|
|
return 0;
|
|
}
|
|
pDirtyPriv->is_alpha_dirty_not = 1;
|
|
rdpup_switch_os_surface(pDirtyPriv->rdpindex);
|
|
rdpup_get_pixmap_image_rect(pDirtyPixmap, &id);
|
|
rdpup_begin_update();
|
|
rdpup_send_alpha_area(&id, 0, 0, pDirtyPixmap->drawable.width,
|
|
pDirtyPixmap->drawable.height);
|
|
rdpup_end_update();
|
|
rdpup_switch_os_surface(-1);
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_add_char(int font, int character, short x, short y, int cx, int cy,
|
|
char* bmpdata, int bmpdata_bytes)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_add_char"));
|
|
rdpup_pre_check(18 + bmpdata_bytes);
|
|
out_uint16_le(g_out_s, 28); /* add char */
|
|
out_uint16_le(g_out_s, 18 + bmpdata_bytes); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, font);
|
|
out_uint16_le(g_out_s, character);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
out_uint16_le(g_out_s, bmpdata_bytes);
|
|
out_uint8a(g_out_s, bmpdata, bmpdata_bytes);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_add_char_alpha(int font, int character, short x, short y, int cx, int cy,
|
|
char* bmpdata, int bmpdata_bytes)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_add_char_alpha"));
|
|
rdpup_pre_check(18 + bmpdata_bytes);
|
|
out_uint16_le(g_out_s, 29); /* add char alpha */
|
|
out_uint16_le(g_out_s, 18 + bmpdata_bytes); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, font);
|
|
out_uint16_le(g_out_s, character);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, cx);
|
|
out_uint16_le(g_out_s, cy);
|
|
out_uint16_le(g_out_s, bmpdata_bytes);
|
|
out_uint8a(g_out_s, bmpdata, bmpdata_bytes);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_draw_text(int font, int flags, int mixmode,
|
|
short clip_left, short clip_top,
|
|
short clip_right, short clip_bottom,
|
|
short box_left, short box_top,
|
|
short box_right, short box_bottom, short x, short y,
|
|
char* data, int data_bytes)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_draw_text"));
|
|
rdpup_pre_check(32 + data_bytes);
|
|
out_uint16_le(g_out_s, 30); /* draw text */
|
|
out_uint16_le(g_out_s, 32 + data_bytes); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, font);
|
|
out_uint16_le(g_out_s, flags);
|
|
out_uint16_le(g_out_s, mixmode);
|
|
out_uint16_le(g_out_s, clip_left);
|
|
out_uint16_le(g_out_s, clip_top);
|
|
out_uint16_le(g_out_s, clip_right);
|
|
out_uint16_le(g_out_s, clip_bottom);
|
|
out_uint16_le(g_out_s, box_left);
|
|
out_uint16_le(g_out_s, box_top);
|
|
out_uint16_le(g_out_s, box_right);
|
|
out_uint16_le(g_out_s, box_bottom);
|
|
out_uint16_le(g_out_s, x);
|
|
out_uint16_le(g_out_s, y);
|
|
out_uint16_le(g_out_s, data_bytes);
|
|
out_uint8a(g_out_s, data, data_bytes);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpup_composite(short srcidx, int srcformat, short srcwidth, CARD8 srcrepeat,
|
|
PictTransform* srctransform, CARD8 mskflags,
|
|
short mskidx, int mskformat, short mskwidth, CARD8 mskrepeat,
|
|
CARD8 op, short srcx, short srcy, short mskx, short msky,
|
|
short dstx, short dsty, short width, short height,
|
|
int dstformat)
|
|
{
|
|
if (g_connected)
|
|
{
|
|
LLOGLN(10, (" rdpup_composite"));
|
|
rdpup_pre_check(84);
|
|
out_uint16_le(g_out_s, 33);
|
|
out_uint16_le(g_out_s, 84); /* size */
|
|
g_count++;
|
|
out_uint16_le(g_out_s, srcidx);
|
|
out_uint32_le(g_out_s, srcformat);
|
|
out_uint16_le(g_out_s, srcwidth);
|
|
out_uint8(g_out_s, srcrepeat);
|
|
if (srctransform == 0)
|
|
{
|
|
out_uint8s(g_out_s, 10 * 4);
|
|
}
|
|
else
|
|
{
|
|
out_uint32_le(g_out_s, 1);
|
|
out_uint32_le(g_out_s, srctransform->matrix[0][0]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[0][1]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[0][2]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[1][0]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[1][1]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[1][2]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[2][0]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[2][1]);
|
|
out_uint32_le(g_out_s, srctransform->matrix[2][2]);
|
|
}
|
|
out_uint8(g_out_s, mskflags);
|
|
out_uint16_le(g_out_s, mskidx);
|
|
out_uint32_le(g_out_s, mskformat);
|
|
out_uint16_le(g_out_s, mskwidth);
|
|
out_uint8(g_out_s, mskrepeat);
|
|
out_uint8(g_out_s, op);
|
|
out_uint16_le(g_out_s, srcx);
|
|
out_uint16_le(g_out_s, srcy);
|
|
out_uint16_le(g_out_s, mskx);
|
|
out_uint16_le(g_out_s, msky);
|
|
out_uint16_le(g_out_s, dstx);
|
|
out_uint16_le(g_out_s, dsty);
|
|
out_uint16_le(g_out_s, width);
|
|
out_uint16_le(g_out_s, height);
|
|
out_uint32_le(g_out_s, dstformat);
|
|
}
|
|
return 0;
|
|
}
|