/* This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. xrdp: A Remote Desktop Protocol server. Copyright (C) Jay Sorg 2005-2009 librdp main file */ #include "rdp.h" /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_start(struct mod* mod, int w, int h, int bpp) { DEBUG(("in lib_mod_start")); mod->width = w; mod->height = h; mod->rdp_bpp = bpp; mod->xrdp_bpp = bpp; mod->keylayout = 0x409; g_strncpy(mod->port, "3389", 255); /* default */ DEBUG(("out lib_mod_start")); return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_connect(struct mod* mod) { DEBUG(("in lib_mod_connect")); /* clear screen */ mod->server_begin_update(mod); mod->server_set_fgcolor(mod, 0); mod->server_fill_rect(mod, 0, 0, mod->width, mod->height); mod->server_end_update(mod); /* connect */ if (rdp_rdp_connect(mod->rdp_layer, mod->ip, mod->port) == 0) { mod->sck = mod->rdp_layer->sec_layer->mcs_layer->iso_layer->tcp_layer->sck; g_tcp_set_non_blocking(mod->sck); g_tcp_set_no_delay(mod->sck); mod->sck_obj = g_create_wait_obj_from_socket(mod->sck, 0); DEBUG(("out lib_mod_connect")); return 0; } DEBUG(("out lib_mod_connect error")); return 1; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_event(struct mod* mod, int msg, long param1, long param2, long param3, long param4) { struct stream* s; if (!mod->up_and_running) { return 0; } DEBUG(("in lib_mod_event")); make_stream(s); init_stream(s, 8192 * 2); switch (msg) { case 15: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_SCANCODE, param2, param1, 0); break; case 16: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_SCANCODE, param2, param1, 0); break; case 17: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_SYNCHRONIZE, param4, param3, 0); break; case 100: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_MOVE, param1, param2); break; case 101: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON1, param1, param2); break; case 102: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON1 | MOUSE_FLAG_DOWN, param1, param2); break; case 103: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON2, param1, param2); break; case 104: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON2 | MOUSE_FLAG_DOWN, param1, param2); break; case 105: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON3, param1, param2); break; case 106: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON3 | MOUSE_FLAG_DOWN, param1, param2); break; case 107: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON4, param1, param2); break; case 108: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON4 | MOUSE_FLAG_DOWN, param1, param2); break; case 109: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON5, param1, param2); break; case 110: rdp_rdp_send_input(mod->rdp_layer, s, 0, RDP_INPUT_MOUSE, MOUSE_FLAG_BUTTON5 | MOUSE_FLAG_DOWN, param1, param2); break; case 200: rdp_rdp_send_invalidate(mod->rdp_layer, s, (param1 >> 16) & 0xffff, param1 & 0xffff, (param2 >> 16) & 0xffff, param2 & 0xffff); break; } free_stream(s); DEBUG(("out lib_mod_event")); return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_signal(struct mod* mod) { int type; int cont; struct stream* s; DEBUG(("in lib_mod_signal")); if (mod->in_s == 0) { make_stream(mod->in_s); } s = mod->in_s; init_stream(s, 8192 * 2); cont = 1; while (cont) { type = 0; if (rdp_rdp_recv(mod->rdp_layer, s, &type) != 0) { DEBUG(("out lib_mod_signal error rdp_rdp_recv failed")); return 1; } DEBUG(("lib_mod_signal type %d", type)); switch (type) { case RDP_PDU_DATA: rdp_rdp_process_data_pdu(mod->rdp_layer, s); break; case RDP_PDU_DEMAND_ACTIVE: rdp_rdp_process_demand_active(mod->rdp_layer, s); mod->up_and_running = 1; break; case RDP_PDU_DEACTIVATE: mod->up_and_running = 0; break; case RDP_PDU_REDIRECT: break; case 0: break; default: break; } cont = s->next_packet < s->end; } DEBUG(("out lib_mod_signal")); return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_end(struct mod* mod) { rdp_rdp_delete(mod->rdp_layer); mod->rdp_layer = 0; free_stream(mod->in_s); mod->in_s = 0; if (mod->sck_obj != 0) { g_delete_wait_obj_from_socket(mod->sck_obj); mod->sck_obj = 0; } if (mod->sck != 0) { g_tcp_close(mod->sck); mod->sck = 0; } return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_set_param(struct mod* mod, char* name, char* value) { if (g_strncasecmp(name, "ip", 255) == 0) { g_strncpy(mod->ip, value, 255); } else if (g_strncasecmp(name, "port", 255) == 0) { g_strncpy(mod->port, value, 255); } else if (g_strncasecmp(name, "username", 255) == 0) { g_strncpy(mod->username, value, 255); } else if (g_strncasecmp(name, "password", 255) == 0) { g_strncpy(mod->password, value, 255); } else if (g_strncasecmp(name, "hostname", 255) == 0) { g_strncpy(mod->hostname, value, 255); } else if (g_strncasecmp(name, "keylayout", 255) == 0) { mod->keylayout = g_atoi(value); } return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_get_wait_objs(struct mod* mod, tbus* read_objs, int* rcount, tbus* write_objs, int* wcount, int* timeout) { int i; i = *rcount; if (mod != 0) { if (mod->sck_obj != 0) { read_objs[i++] = mod->sck_obj; } } *rcount = i; return 0; } /******************************************************************************/ /* return error */ int DEFAULT_CC lib_mod_check_wait_objs(struct mod* mod) { int rv; rv = 0; if (mod != 0) { if (mod->sck_obj != 0) { if (g_is_wait_obj_set(mod->sck_obj)) { rv = lib_mod_signal(mod); } } } return rv; } /******************************************************************************/ struct mod* EXPORT_CC mod_init(void) { struct mod* mod; DEBUG(("in mod_init")); mod = (struct mod*)g_malloc(sizeof(struct mod), 1); mod->size = sizeof(struct mod); mod->handle = (long)mod; mod->mod_connect = lib_mod_connect; mod->mod_start = lib_mod_start; mod->mod_event = lib_mod_event; mod->mod_signal = lib_mod_signal; mod->mod_end = lib_mod_end; mod->mod_set_param = lib_mod_set_param; mod->mod_get_wait_objs = lib_mod_get_wait_objs; mod->mod_check_wait_objs = lib_mod_check_wait_objs; mod->rdp_layer = rdp_rdp_create(mod); DEBUG(("out mod_init")); return mod; } /******************************************************************************/ int EXPORT_CC mod_exit(struct mod* mod) { DEBUG(("in mod_exit")); g_free(mod); DEBUG(("out mod_exit")); return 0; }