#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
+#include <sys/select.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <arpa/inet.h>
printf("- Device flags: %x\n", device_flags);
printf("- Position: (%u,%u)\n", param1, param2);
- // debug
- rdp_send_bitmap_update(param1, param2);
-
break;
default:
printf("- Unknown type %x\n", message_type);
printf("Server is %u x %u\n", ntohs(si.width), ntohs(si.height));
- if (read(vnc_sock, buf, si.name_len) != si.name_len)
- error("short read on server name");
+ if (read(vnc_sock, buf, ntohl(si.name_len)) != ntohl(si.name_len))
+ error("short read on server name\n");
+
+ printf("Server name is '%*s' (%u bytes)\n", ntohl(si.name_len), buf, ntohl(si.name_len));
- printf("Server name is '%*s'\n", si.name_len, buf);
+ // we can only accept raw encoding
+ buf[0] = 2; // message type
+ buf[1] = 0; // padding
+ buf[2] = 0; // number of encodings
+ buf[3] = 1;
+ buf[4] = 0; // raw encoding
+ buf[5] = 0;
+ buf[6] = 0;
+ buf[7] = 0;
+
+ write(vnc_sock, buf, 8);
+
+ // request the entire framebuffer
+ buf[0] = 3; // message type
+ buf[1] = 0; // incremental
+ buf[2] = 0; // xpos
+ buf[3] = 0;
+ buf[4] = 0; // ypos
+ buf[5] = 0;
+ buf[6] = 0; // width
+ buf[7] = 640 % 0xff;
+ buf[8] = 0; // height
+ buf[9] = 480 % 0xff;
+
+ write(vnc_sock, buf, 10);
printf("Connected to VNC!\n");
return vnc_sock;
}
+struct vnc_rectangle {
+ unsigned short x, y, width, height;
+ unsigned int encoding;
+};
+
+void handle_vnc_fbupdate(int vnc_sock)
+{
+ struct vnc_rectangle rect;
+ unsigned char *data, *ptr;
+ unsigned short num_rect;
+ int ret, data_left, i;
+
+ if (read(vnc_sock, &num_rect, 2) != 2)
+ error("short read on num_rect\n");
+
+ for (i = 0; i < ntohs(num_rect); ++i) {
+ if (read(vnc_sock, &rect, sizeof(struct vnc_rectangle)) != sizeof(struct vnc_rectangle))
+ error("short read on vnc_rectangle\n");
+
+ printf("UPDATE: %ux%u at (%u,%u)\n", ntohs(rect.width), ntohs(rect.height),
+ ntohs(rect.x), ntohs(rect.y));
+
+ if (ntohl(rect.encoding) != 0)
+ error("unsupported encoding\n");
+
+ ptr = data = xmalloc(ntohs(rect.width) * ntohs(rect.height) * 3);
+ data_left = ntohs(rect.width) * ntohs(rect.height) * 3;
+
+ while (data_left > 0) {
+ ret = read(vnc_sock, ptr, data_left);
+ if (ret <= 0)
+ error("error on data read\n");
+
+ ptr += ret;
+ data_left -= ret;
+ }
+
+// rdp_send_bitmap_update(ntohs(rect.x), ntohs(rect.y), ntohs(rect.width), ntohs(rect.height), data);
+ xfree(data);
+ }
+}
+
int serve_client()
{
int vnc_sock = vnc_init();
+ int listen_on_vnc = 0;
if (!mcs_recv_connect_initial())
error("MCS_CONNECT_INITIAL recv failed");
uint8 type, data_pdu_type;
STREAM s;
- while ((s = rdp_recv(&type)) != NULL) {
- if (type != RDP_PDU_DATA) {
- printf("Unknown RDP packet of type %u\n", type);
- continue;
+ fd_set readfs;
+ FD_ZERO(&readfs);
+ FD_SET(tcp_get_socket(), &readfs);
+ FD_SET(vnc_sock, &readfs);
+
+ select(FD_SETSIZE, &readfs, NULL, NULL, NULL);
+
+ // activity on RDP socket?
+ if (FD_ISSET(tcp_get_socket(), &readfs)) {
+ if ((s = rdp_recv(&type)) != NULL) {
+ if (type != RDP_PDU_DATA) {
+ printf("Unknown RDP packet of type %u\n", type);
+ continue;
+ }
+
+ in_uint8s(s, 8); /* shareid, pad, streamid, length */
+ in_uint8(s, data_pdu_type);
+ in_uint8s(s, 3); /* compress_type, compress_len */
+
+ switch (data_pdu_type) {
+ case RDP_DATA_PDU_INPUT:
+ printf("Input PDU\n");
+ handle_input_pdu(s);
+ listen_on_vnc = 1;
+ break;
+ default:
+ printf("Unknown data PDU type %u\n", data_pdu_type);
+ };
}
+ }
- in_uint8s(s, 8); /* shareid, pad, streamid, length */
- in_uint8(s, data_pdu_type);
- in_uint8s(s, 3); /* compress_type, compress_len */
+ // activity on VNC socket?
+ if (FD_ISSET(vnc_sock, &readfs) && listen_on_vnc) {
+ unsigned char buf[256];
- switch (data_pdu_type) {
- case RDP_DATA_PDU_INPUT:
- printf("Input PDU\n");
- handle_input_pdu(s);
+ printf("Activity on VNC socket!\n");
+
+ if (read(vnc_sock, buf, 2) != 2)
+ error("short read on vnc_sock\n");
+
+ switch (buf[0]) {
+ case 0:
+ // frame buffer update!
+ handle_vnc_fbupdate(vnc_sock);
break;
default:
- printf("Unknown data PDU type %u\n", data_pdu_type);
- };
+ printf("Unknown VNC server message %x\n", buf[0]);
+// exit(1);
+ }
}
}
}