wok-current rev 17271
busybox: add fbvnc
author | Pascal Bellard <pascal.bellard@slitaz.org> |
---|---|
date | Fri Oct 24 16:26:40 2014 +0200 (2014-10-24) |
parents | 7c22b4192384 |
children | d03507a03d90 |
files | busybox/receipt busybox/stuff/busybox-1.22-fbvnc.u busybox/stuff/busybox-1.22.config busybox/stuff/busybox-1.22.config-ssfs busybox/stuff/busybox-1.22.config-static |
line diff
1.1 --- a/busybox/receipt Fri Oct 24 12:46:53 2014 +0200 1.2 +++ b/busybox/receipt Fri Oct 24 16:26:40 2014 +0200 1.3 @@ -44,7 +44,7 @@ 1.4 diet.u 1.5 losetup.u 1.6 fatattr.u 1.7 -dpkg_deb-xz.u 1.8 +fbvnc.u 1.9 EOT 1.10 cp $stuff/$PACKAGE-${VERSION%.*}.config .config 1.11 }
2.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 2.2 +++ b/busybox/stuff/busybox-1.22-fbvnc.u Fri Oct 24 16:26:40 2014 +0200 2.3 @@ -0,0 +1,681 @@ 2.4 + text data bss dec hex filename 2.5 + 3118 0 0 3118 c2e util-linux/fbvnc.o 2.6 +--- /dev/null 2.7 ++++ busybox/util-linux/fbvnc.c 2.8 +@@ -0,0 +1,551 @@ 2.9 ++/* vi: set sw=4 ts=4: */ 2.10 ++/* 2.11 ++ * A small linux framebuffer VNC viewer 2.12 ++ * 2.13 ++ * pascal.bellard@ads-lu.com 2.14 ++ * 2.15 ++ * Based on Ali Gholami Rudi's fbvnc.c 2.16 ++ * http://repo.or.cz/w/fbvnc.git 2.17 ++ * 2.18 ++ * Licensed under GPLv2 or later, see file LICENSE in this source tree. 2.19 ++ */ 2.20 ++ 2.21 ++//applet:IF_FBVNC(APPLET(fbvnc, BB_DIR_BIN, BB_SUID_DROP)) 2.22 ++ 2.23 ++//kbuild:lib-$(CONFIG_FBVNC) += fbvnc.o 2.24 ++ 2.25 ++//config:config FBVNC 2.26 ++//config: bool "fbvnc" 2.27 ++//config: default n 2.28 ++//config: depends on PLATFORM_LINUX 2.29 ++//config: help 2.30 ++//config: A linux framebuffer VNC viewer. 2.31 ++ 2.32 ++//usage:#define fbvnc_trivial_usage 2.33 ++//usage: "[VNC_SERVER] [PORT]" 2.34 ++//usage:#define fbvnc_full_usage "\n\n" 2.35 ++//usage: "A linux framebuffer VNC viewer." 2.36 ++//usage: "\nTo exit, move mouse to upper left corner and press ESC." 2.37 ++ 2.38 ++#include "libbb.h" 2.39 ++#include "vnc.h" 2.40 ++ 2.41 ++/* Stuff stolen from the kernel's fb.h */ 2.42 ++#define FB_ACTIVATE_ALL 64 2.43 ++enum { 2.44 ++ FBIOGET_VSCREENINFO = 0x4600, 2.45 ++ FBIOPUT_VSCREENINFO = 0x4601, 2.46 ++ FBIOGET_FSCREENINFO = 0x4602, 2.47 ++ FBIOGETCMAP = 0x4604, 2.48 ++ FBIOPUTCMAP = 0x4605 2.49 ++}; 2.50 ++ 2.51 ++struct fb_bitfield { 2.52 ++ uint32_t offset; /* beginning of bitfield */ 2.53 ++ uint32_t length; /* length of bitfield */ 2.54 ++ uint32_t msb_right; /* !=0: Most significant bit is right */ 2.55 ++}; 2.56 ++struct fb_var_screeninfo { 2.57 ++ uint32_t xres; /* visible resolution */ 2.58 ++ uint32_t yres; 2.59 ++ uint32_t xres_virtual; /* virtual resolution */ 2.60 ++ uint32_t yres_virtual; 2.61 ++ uint32_t xoffset; /* offset from virtual to visible */ 2.62 ++ uint32_t yoffset; /* resolution */ 2.63 ++ 2.64 ++ uint32_t bits_per_pixel; 2.65 ++ uint32_t grayscale; /* !=0 Graylevels instead of colors */ 2.66 ++ 2.67 ++ struct fb_bitfield red; /* bitfield in fb mem if true color, */ 2.68 ++ struct fb_bitfield green; /* else only length is significant */ 2.69 ++ struct fb_bitfield blue; 2.70 ++ struct fb_bitfield transp; /* transparency */ 2.71 ++ 2.72 ++ uint32_t nonstd; /* !=0 Non standard pixel format */ 2.73 ++ 2.74 ++ uint32_t activate; /* see FB_ACTIVATE_x */ 2.75 ++ 2.76 ++ uint32_t height; /* height of picture in mm */ 2.77 ++ uint32_t width; /* width of picture in mm */ 2.78 ++ 2.79 ++ uint32_t accel_flags; /* acceleration flags (hints) */ 2.80 ++ 2.81 ++ /* Timing: All values in pixclocks, except pixclock (of course) */ 2.82 ++ uint32_t pixclock; /* pixel clock in ps (pico seconds) */ 2.83 ++ uint32_t left_margin; /* time from sync to picture */ 2.84 ++ uint32_t right_margin; /* time from picture to sync */ 2.85 ++ uint32_t upper_margin; /* time from sync to picture */ 2.86 ++ uint32_t lower_margin; 2.87 ++ uint32_t hsync_len; /* length of horizontal sync */ 2.88 ++ uint32_t vsync_len; /* length of vertical sync */ 2.89 ++ uint32_t sync; /* see FB_SYNC_x */ 2.90 ++ uint32_t vmode; /* see FB_VMODE_x */ 2.91 ++ uint32_t reserved[6]; /* Reserved for future compatibility */ 2.92 ++}; 2.93 ++ 2.94 ++#define DEFAULTFBDEV FB_0 2.95 ++ 2.96 ++struct fb_fix_screeninfo { 2.97 ++ char id[16]; /* identification string eg "TT Builtin" */ 2.98 ++ unsigned long smem_start; /* Start of frame buffer mem */ 2.99 ++ /* (physical address) */ 2.100 ++ uint32_t smem_len; /* Length of frame buffer mem */ 2.101 ++ uint32_t type; /* see FB_TYPE_* */ 2.102 ++ uint32_t type_aux; /* Interleave for interleaved Planes */ 2.103 ++ uint32_t visual; /* see FB_VISUAL_* */ 2.104 ++ uint16_t xpanstep; /* zero if no hardware panning */ 2.105 ++ uint16_t ypanstep; /* zero if no hardware panning */ 2.106 ++ uint16_t ywrapstep; /* zero if no hardware ywrap */ 2.107 ++ uint32_t line_length; /* length of a line in bytes */ 2.108 ++ unsigned long mmio_start; /* Start of Memory Mapped I/O */ 2.109 ++ /* (physical address) */ 2.110 ++ uint32_t mmio_len; /* Length of Memory Mapped I/O */ 2.111 ++ uint32_t accel; /* Indicate to driver which */ 2.112 ++ /* specific chip/card we have */ 2.113 ++ uint16_t reserved[3]; /* Reserved for future compatibility */ 2.114 ++}; 2.115 ++ 2.116 ++struct fb_cmap { 2.117 ++ uint32_t start; /* First entry */ 2.118 ++ uint32_t len; /* Number of entries */ 2.119 ++ uint16_t *red; /* Red values */ 2.120 ++ uint16_t *green; 2.121 ++ uint16_t *blue; 2.122 ++ uint16_t *transp; /* transparency, can be NULL */ 2.123 ++}; 2.124 ++ 2.125 ++#define FB_VISUAL_TRUECOLOR 2 /* True color */ 2.126 ++ 2.127 ++#define COLORLEVELS (1 << 8) 2.128 ++ 2.129 ++struct scroll_data { 2.130 ++ int size; 2.131 ++ int srv_size; 2.132 ++ int offset; 2.133 ++ int pos; 2.134 ++}; 2.135 ++ 2.136 ++struct globals { 2.137 ++ struct termios term_orig; 2.138 ++ struct pollfd ufds[3]; 2.139 ++#define kbd_fd ufds[0].fd 2.140 ++#define vnc_fd ufds[1].fd 2.141 ++#define rat_fd ufds[2].fd 2.142 ++ struct scroll_data scroll[2]; 2.143 ++#define cols scroll[0].size 2.144 ++#define srv_cols scroll[0].srv_size 2.145 ++#define oc scroll[0].offset 2.146 ++#define mc scroll[0].pos 2.147 ++#define rows scroll[1].size 2.148 ++#define srv_rows scroll[1].srv_size 2.149 ++#define or scroll[1].offset 2.150 ++#define mr scroll[1].pos 2.151 ++ int fb_fd; 2.152 ++ void *fb_ptr; 2.153 ++ int bpp; 2.154 ++ int nr, ng, nb; 2.155 ++ struct fb_var_screeninfo vinfo; 2.156 ++ struct fb_fix_screeninfo finfo; 2.157 ++ unsigned short red[COLORLEVELS], green[COLORLEVELS], blue[COLORLEVELS]; 2.158 ++}; 2.159 ++ 2.160 ++#define G (*ptr_to_globals) 2.161 ++#define INIT_G() do { \ 2.162 ++ SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \ 2.163 ++} while (0) 2.164 ++ 2.165 ++static int fb_len(void) 2.166 ++{ 2.167 ++ return G.finfo.line_length * G.vinfo.yres_virtual; 2.168 ++} 2.169 ++ 2.170 ++static void fb_ioctl_cmap(int fct, struct fb_cmap *cmap) 2.171 ++{ 2.172 ++ if (G.finfo.visual == FB_VISUAL_TRUECOLOR) 2.173 ++ return; 2.174 ++ cmap->start = 0; 2.175 ++ cmap->len = MAX(G.nr, MAX(G.ng, G.nb)); 2.176 ++ cmap->transp = NULL; 2.177 ++ xioctl(G.fb_fd, fct, cmap); 2.178 ++} 2.179 ++ 2.180 ++static void fb_cmap_save(int save) 2.181 ++{ 2.182 ++ struct fb_cmap cmap; 2.183 ++ 2.184 ++ cmap.red = G.red; 2.185 ++ cmap.green = G.green; 2.186 ++ cmap.blue = G.blue; 2.187 ++ fb_ioctl_cmap(save ? FBIOGETCMAP : FBIOPUTCMAP, &cmap); 2.188 ++} 2.189 ++ 2.190 ++static void fb_build_cmap(unsigned short *color, int n) 2.191 ++{ 2.192 ++ int i, inc = 65535 / (n - 1); 2.193 ++ 2.194 ++ for (i = 0; n--; i += inc) 2.195 ++ *color++ = i; 2.196 ++} 2.197 ++ 2.198 ++static void fb_cmap(void) 2.199 ++{ 2.200 ++ unsigned short red[COLORLEVELS], green[COLORLEVELS], blue[COLORLEVELS]; 2.201 ++ struct fb_cmap cmap; 2.202 ++ 2.203 ++ fb_build_cmap(cmap.red = red, G.nr); 2.204 ++ fb_build_cmap(cmap.green = green, G.ng); 2.205 ++ fb_build_cmap(cmap.blue = blue, G.nb); 2.206 ++ fb_ioctl_cmap(FBIOPUTCMAP, &cmap); 2.207 ++} 2.208 ++ 2.209 ++static void fb_init(void) 2.210 ++{ 2.211 ++ G.fb_fd = xopen(DEFAULTFBDEV, O_RDWR); 2.212 ++ xioctl(G.fb_fd, FBIOGET_VSCREENINFO, &G.vinfo); 2.213 ++ xioctl(G.fb_fd, FBIOGET_FSCREENINFO, &G.finfo); 2.214 ++ close_on_exec_on(G.fb_fd); 2.215 ++ G.fb_ptr = mmap(NULL, fb_len(), PROT_READ | PROT_WRITE, MAP_SHARED, G.fb_fd, 0); 2.216 ++ if (G.fb_ptr == MAP_FAILED) 2.217 ++ bb_perror_msg_and_die("mmap"); 2.218 ++ G.bpp = (G.vinfo.bits_per_pixel + 7) >> 3; 2.219 ++ G.nr = 1 << G.vinfo.red.length; 2.220 ++ G.nb = 1 << G.vinfo.blue.length; 2.221 ++ G.ng = 1 << G.vinfo.green.length; 2.222 ++ fb_cmap_save(1); 2.223 ++ fb_cmap(); 2.224 ++} 2.225 ++ 2.226 ++static void fb_free(void) 2.227 ++{ 2.228 ++ fb_cmap_save(0); 2.229 ++ munmap(G.fb_ptr, fb_len()); 2.230 ++ close(G.fb_fd); 2.231 ++} 2.232 ++ 2.233 ++#define fb_rows vinfo.yres 2.234 ++#define fb_cols vinfo.xres 2.235 ++ 2.236 ++static void fb_set(int r, int c, void *mem, int len) 2.237 ++{ 2.238 ++ memcpy(G.fb_ptr + (r + G.vinfo.yoffset) * G.finfo.line_length + 2.239 ++ (c + G.vinfo.xoffset) * G.bpp, mem, len * G.bpp); 2.240 ++} 2.241 ++ 2.242 ++#define line_buffer bb_common_bufsiz1 2.243 ++#define MAXPIX (sizeof(line_buffer)/sizeof(uint32_t)) 2.244 ++ 2.245 ++static void skip(int len) 2.246 ++{ 2.247 ++ int n; 2.248 ++ while (len > 0 && (n = read(G.vnc_fd, line_buffer, 2.249 ++ MIN(len, sizeof(line_buffer)))) > 0) 2.250 ++ len -= n; 2.251 ++} 2.252 ++ 2.253 ++static void vnc_init(void) 2.254 ++{ 2.255 ++ struct vnc_client_init clientinit; 2.256 ++ struct vnc_server_init serverinit; 2.257 ++ struct vnc_client_pixelfmt pixfmt_cmd; 2.258 ++ int connstat = VNC_CONN_FAILED; 2.259 ++ 2.260 ++ write(G.vnc_fd, "RFB 003.003\n", 12); 2.261 ++ skip(12); 2.262 ++ 2.263 ++ xread(G.vnc_fd, &connstat, sizeof(connstat)); 2.264 ++ 2.265 ++ if (ntohl(connstat) != VNC_CONN_NOAUTH) 2.266 ++ bb_perror_msg_and_die("vnc auth"); 2.267 ++ 2.268 ++ clientinit.shared = 1; 2.269 ++ write(G.vnc_fd, &clientinit, sizeof(clientinit)); 2.270 ++ read(G.vnc_fd, &serverinit, sizeof(serverinit)); 2.271 ++ 2.272 ++ fb_init(); 2.273 ++ G.srv_cols = ntohs(serverinit.w); 2.274 ++ G.srv_rows = ntohs(serverinit.h); 2.275 ++ G.cols = MIN(G.srv_cols, G.fb_cols); 2.276 ++ G.rows = MIN(G.srv_rows, G.fb_rows); 2.277 ++ G.mr = G.rows / 2; 2.278 ++ G.mc = G.cols / 2; 2.279 ++ 2.280 ++ skip(ntohl(serverinit.len)); 2.281 ++ pixfmt_cmd.type = VNC_CLIENT_PIXFMT; 2.282 ++ pixfmt_cmd.format.bigendian = 0; 2.283 ++ pixfmt_cmd.format.truecolor = 1; 2.284 ++ pixfmt_cmd.format.bpp = 2.285 ++ pixfmt_cmd.format.depth = G.bpp << 3; 2.286 ++ pixfmt_cmd.format.rmax = htons(G.nr - 1); 2.287 ++ pixfmt_cmd.format.gmax = htons(G.ng - 1); 2.288 ++ pixfmt_cmd.format.bmax = htons(G.nb - 1); 2.289 ++ pixfmt_cmd.format.rshl = G.vinfo.red.offset; 2.290 ++ pixfmt_cmd.format.gshl = G.vinfo.green.offset; 2.291 ++ pixfmt_cmd.format.bshl = G.vinfo.blue.offset; 2.292 ++ write(G.vnc_fd, &pixfmt_cmd, sizeof(pixfmt_cmd)); 2.293 ++} 2.294 ++ 2.295 ++static void vnc_refresh(int inc) 2.296 ++{ 2.297 ++ struct vnc_client_fbup fbup_req; 2.298 ++ fbup_req.type = VNC_CLIENT_FBUP; 2.299 ++ fbup_req.inc = inc; 2.300 ++ fbup_req.x = htons(G.oc); 2.301 ++ fbup_req.y = htons(G.or); 2.302 ++ fbup_req.w = htons(G.oc + G.cols); 2.303 ++ fbup_req.h = htons(G.or + G.rows); 2.304 ++ write(G.vnc_fd, &fbup_req, sizeof(fbup_req)); 2.305 ++} 2.306 ++ 2.307 ++static void cleanup(void) 2.308 ++{ 2.309 ++ const char *reset = "\x1b[?25h" "\x1b[2J\x1b[H"; 2.310 ++ fb_free(); 2.311 ++ tcsetattr_stdin_TCSANOW(&G.term_orig); 2.312 ++ write(STDOUT_FILENO, reset, strlen(reset)); 2.313 ++ if (ENABLE_FEATURE_CLEAN_UP) { 2.314 ++ close(G.vnc_fd); 2.315 ++ close(G.rat_fd); 2.316 ++ } 2.317 ++} 2.318 ++ 2.319 ++static void killed(int code) NORETURN; 2.320 ++static void killed(int code) 2.321 ++{ 2.322 ++ cleanup(); 2.323 ++ if (code > EXIT_FAILURE) 2.324 ++ kill_myself_with_sig(code); 2.325 ++ exit(code); 2.326 ++} 2.327 ++ 2.328 ++static void vnc_event(void) 2.329 ++{ 2.330 ++ struct vnc_rect uprect; 2.331 ++ union { 2.332 ++ struct vnc_server_fbup fbup; 2.333 ++ struct vnc_server_cuttext cuttext; 2.334 ++ struct vnc_server_colormap colormap; 2.335 ++ } msg; 2.336 ++ int n; 2.337 ++ 2.338 ++ switch (xread_char(G.vnc_fd)) { 2.339 ++ case VNC_SERVER_FBUP: 2.340 ++ xread(G.vnc_fd, &msg.fbup.pad, sizeof(msg.fbup) - 1); 2.341 ++ n = ntohs(msg.fbup.n); 2.342 ++ while (n--) { 2.343 ++ int x, y, w, h, l, i; 2.344 ++ xread(G.vnc_fd, &uprect, sizeof(uprect)); 2.345 ++ if (uprect.enc != 0) 2.346 ++ killed(1); 2.347 ++ i = 0; 2.348 ++ x = ntohs(uprect.x) - G.oc; 2.349 ++ y = ntohs(uprect.y) - G.or; 2.350 ++ w = ntohs(uprect.w); 2.351 ++ h = ntohs(uprect.h); 2.352 ++ l = MIN(w, G.cols - x); 2.353 ++ if (x < 0) { 2.354 ++ l = MIN(w + x, G.cols); 2.355 ++ i = -x; 2.356 ++ x = 0; 2.357 ++ } 2.358 ++ for (; h--; y++) { 2.359 ++ int a, b, c = i; 2.360 ++ for (a = b = 0; w > b; b += a, c = 0) { 2.361 ++ int len; 2.362 ++ a = MIN(w - b, MAXPIX); 2.363 ++ len = MIN(a, l - b) - c; 2.364 ++ xread(G.vnc_fd, line_buffer, a * G.bpp); 2.365 ++ if (y >= 0 && y < G.rows && len > 0) 2.366 ++ fb_set(y, x + b, 2.367 ++ line_buffer + (c * G.bpp), 2.368 ++ len); 2.369 ++ } 2.370 ++ } 2.371 ++ } 2.372 ++ break; 2.373 ++ case VNC_SERVER_BELL: 2.374 ++ break; 2.375 ++ case VNC_SERVER_CUTTEXT: 2.376 ++ xread(G.vnc_fd, &msg.cuttext.pad1, sizeof(msg.cuttext) - 1); 2.377 ++ skip(ntohl(msg.cuttext.len)); 2.378 ++ break; 2.379 ++ case VNC_SERVER_COLORMAP: 2.380 ++ xread(G.vnc_fd, &msg.colormap.pad, sizeof(msg.colormap) - 1); 2.381 ++ skip(ntohs(msg.colormap.n) * 3 * 2); 2.382 ++ break; 2.383 ++ default: 2.384 ++ killed(1); 2.385 ++ } 2.386 ++} 2.387 ++ 2.388 ++static int update_scroll(struct scroll_data *s) 2.389 ++{ 2.390 ++ int shift = s->size / 5; 2.391 ++ int max = s->srv_size - s->size; 2.392 ++ int status = 0; 2.393 ++ if (s->pos < s->offset) { 2.394 ++ if ((s->offset -= shift) < 0) 2.395 ++ s->offset = 0; 2.396 ++ } 2.397 ++ else if (s->pos >= s->offset + s->size && s->offset < max) { 2.398 ++ if ((s->offset += shift) > max) 2.399 ++ s->offset = max; 2.400 ++ } 2.401 ++ else status++; 2.402 ++ s->pos = MAX(s->offset, MIN(s->offset + s->size - 1, s->pos)); 2.403 ++ return status; 2.404 ++} 2.405 ++ 2.406 ++static void rat_event(void) 2.407 ++{ 2.408 ++ signed char ie[3]; 2.409 ++ struct vnc_client_ratevent me = {VNC_CLIENT_RATEVENT}; 2.410 ++ int mask = 0; 2.411 ++ int refresh; 2.412 ++ 2.413 ++ xread(G.rat_fd, &ie, sizeof(ie)); 2.414 ++ G.mc += ie[1]; 2.415 ++ G.mr -= ie[2]; 2.416 ++ refresh = 2 - update_scroll(&G.scroll[0]) - update_scroll(&G.scroll[1]); 2.417 ++ if (ie[0] & 0x01) 2.418 ++ mask |= VNC_BUTTON1_MASK; 2.419 ++ if (ie[0] & 0x04) 2.420 ++ mask |= VNC_BUTTON2_MASK; 2.421 ++ if (ie[0] & 0x02) 2.422 ++ mask |= VNC_BUTTON3_MASK; 2.423 ++ me.y = htons(G.mr); 2.424 ++ me.x = htons(G.mc); 2.425 ++ me.mask = mask; 2.426 ++ write(G.vnc_fd, &me, sizeof(me)); 2.427 ++ if (refresh) 2.428 ++ vnc_refresh(0); 2.429 ++} 2.430 ++ 2.431 ++static int press(int key, int down) 2.432 ++{ 2.433 ++ struct vnc_client_keyevent ke = {VNC_CLIENT_KEYEVENT}; 2.434 ++ ke.key = htonl(key); 2.435 ++ ke.down = down; 2.436 ++ return write(G.vnc_fd, &ke, sizeof(ke)); 2.437 ++} 2.438 ++ 2.439 ++static void kbd_event(void) 2.440 ++{ 2.441 ++ char key[1024]; 2.442 ++ int i, nr; 2.443 ++ 2.444 ++ if ((nr = read(0, key, sizeof(key))) <= 0 ) 2.445 ++ killed(1); 2.446 ++ for (i = 0; i < nr; i++) { 2.447 ++ int k = -1; 2.448 ++ int mod[4]; 2.449 ++ int nmod = 0; 2.450 ++ switch (key[i]) { 2.451 ++ case 0x08: 2.452 ++ case 0x7f: 2.453 ++ k = 0xff08; 2.454 ++ break; 2.455 ++ case 0x09: 2.456 ++ k = 0xff09; 2.457 ++ break; 2.458 ++ case 0x1b: 2.459 ++ if (G.oc + G.mc + G.or + G.mr == 0) 2.460 ++ killed(0); 2.461 ++ if (i + 2 < nr && key[i + 1] == '[') { 2.462 ++ if (key[i + 2] == 'A') 2.463 ++ k = 0xff52; 2.464 ++ if (key[i + 2] == 'B') 2.465 ++ k = 0xff54; 2.466 ++ if (key[i + 2] == 'C') 2.467 ++ k = 0xff53; 2.468 ++ if (key[i + 2] == 'D') 2.469 ++ k = 0xff51; 2.470 ++ if (key[i + 2] == 'H') 2.471 ++ k = 0xff50; 2.472 ++ if (k > 0) { 2.473 ++ i += 2; 2.474 ++ break; 2.475 ++ } 2.476 ++ } 2.477 ++ k = 0xff1b; 2.478 ++ if (i + 1 < nr) { 2.479 ++ mod[nmod++] = 0xffe9; 2.480 ++ k = key[++i]; 2.481 ++ } 2.482 ++ break; 2.483 ++ case 0x0d: 2.484 ++ k = 0xff0d; 2.485 ++ break; 2.486 ++ case 0x0c: /* ^L: redraw */ 2.487 ++ vnc_refresh(0); 2.488 ++ default: 2.489 ++ k = (unsigned char) key[i]; 2.490 ++ } 2.491 ++ if ((k >= 'A' && k <= 'Z') || strchr(":\"<>?{}|+_()*&^%$#@!~", k)) 2.492 ++ mod[nmod++] = 0xffe1; 2.493 ++ if (k >= 1 && k <= 26) { 2.494 ++ k += 'a' - 1; 2.495 ++ mod[nmod++] = 0xffe3; 2.496 ++ } 2.497 ++ if (k > 0) { 2.498 ++ int j; 2.499 ++ mod[nmod] = k; 2.500 ++ for (j = 0; j <= nmod; j++) 2.501 ++ press(mod[j], 1); 2.502 ++ press(k, 0); 2.503 ++ for (j = 0; j < nmod; j++) 2.504 ++ press(mod[j], 0); 2.505 ++ } 2.506 ++ } 2.507 ++} 2.508 ++ 2.509 ++static void term_setup(void) 2.510 ++{ 2.511 ++ struct termios termios; 2.512 ++ const char *init = "\x1b[?25l" "\x1b[2J\x1b[H" "** fbvnc **"; 2.513 ++ 2.514 ++ write(STDOUT_FILENO, init, strlen(init)); 2.515 ++ tcgetattr (STDIN_FILENO, &termios); 2.516 ++ G.term_orig = termios; 2.517 ++ cfmakeraw(&termios); 2.518 ++ tcsetattr_stdin_TCSANOW(&termios); 2.519 ++} 2.520 ++ 2.521 ++int fbvnc_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; 2.522 ++int fbvnc_main(int argc, char **argv) 2.523 ++{ 2.524 ++ char *host = (char *) "127.0.0.1"; 2.525 ++ int port, pending = 0; 2.526 ++ 2.527 ++ INIT_G(); 2.528 ++ if (argc >= 2) 2.529 ++ host = argv[1]; 2.530 ++ port = bb_lookup_port((argc >= 3) ? argv[2] : "vnc", "tcp", 5900); 2.531 ++ G.vnc_fd = create_and_connect_stream_or_die(host, port); 2.532 ++ vnc_init(); 2.533 ++ G.rat_fd = open("/dev/input/mice", O_RDONLY); 2.534 ++ term_setup(); 2.535 ++ atexit(cleanup); 2.536 ++ bb_signals(BB_FATAL_SIGS, killed); 2.537 ++ 2.538 ++ G.ufds[0].events = 2.539 ++ G.ufds[1].events = 2.540 ++ G.ufds[2].events = POLLIN; 2.541 ++ vnc_refresh(0); 2.542 ++ while (1) { 2.543 ++ int status = poll(G.ufds, 3, 500); 2.544 ++ if (status == -1 && errno != EINTR) 2.545 ++ killed(1); 2.546 ++ if (!status) 2.547 ++ continue; 2.548 ++ if (G.ufds[0].revents & POLLIN) 2.549 ++ kbd_event(); 2.550 ++ if (G.ufds[1].revents & POLLIN) { 2.551 ++ vnc_event(); 2.552 ++ pending = 0; 2.553 ++ } 2.554 ++ if (G.ufds[2].revents & POLLIN) 2.555 ++ rat_event(); 2.556 ++ if (!pending++) 2.557 ++ vnc_refresh(1); 2.558 ++ } 2.559 ++} 2.560 +--- /dev/null 2.561 ++++ busybox/util-linux/vnc.h 2.562 +@@ -0,0 +1,122 @@ 2.563 ++#define VNC_CONN_FAILED 0 2.564 ++#define VNC_CONN_NOAUTH 1 2.565 ++#define VNC_CONN_AUTH 2 2.566 ++ 2.567 ++#define VNC_AUTH_OK 0 2.568 ++#define VNC_AUTH_FAILED 1 2.569 ++#define VNC_AUTH_TOOMANY 2 2.570 ++ 2.571 ++#define VNC_SERVER_FBUP 0 2.572 ++#define VNC_SERVER_COLORMAP 1 2.573 ++#define VNC_SERVER_BELL 2 2.574 ++#define VNC_SERVER_CUTTEXT 3 2.575 ++ 2.576 ++#define VNC_CLIENT_PIXFMT 0 2.577 ++#define VNC_CLIENT_COLORMAP 1 2.578 ++#define VNC_CLIENT_SETENC 2 2.579 ++#define VNC_CLIENT_FBUP 3 2.580 ++#define VNC_CLIENT_KEYEVENT 4 2.581 ++#define VNC_CLIENT_RATEVENT 5 2.582 ++#define VNC_CLIENT_CUTTEXT 6 2.583 ++ 2.584 ++#define VNC_ENC_RAW 0 2.585 ++#define VNC_ENC_COPYRECT 1 2.586 ++#define VNC_ENC_RRE 2 2.587 ++#define VNC_ENC_CORRE 4 2.588 ++#define VNC_ENC_HEXTILE 5 2.589 ++ 2.590 ++#define VNC_BUTTON1_MASK 0x1 2.591 ++#define VNC_BUTTON2_MASK 0x2 2.592 ++#define VNC_BUTTON3_MASK 0x4 2.593 ++ 2.594 ++typedef unsigned char u8; 2.595 ++typedef unsigned short u16; 2.596 ++typedef unsigned int u32; 2.597 ++ 2.598 ++struct vnc_pixelfmt { 2.599 ++ u8 bpp; 2.600 ++ u8 depth; 2.601 ++ u8 bigendian; 2.602 ++ u8 truecolor; 2.603 ++ u16 rmax; 2.604 ++ u16 gmax; 2.605 ++ u16 bmax; 2.606 ++ u8 rshl; 2.607 ++ u8 gshl; 2.608 ++ u8 bshl; 2.609 ++ 2.610 ++ u8 pad1; 2.611 ++ u16 pad2; 2.612 ++}; 2.613 ++ 2.614 ++struct vnc_client_init { 2.615 ++ u8 shared; 2.616 ++}; 2.617 ++ 2.618 ++struct vnc_server_init { 2.619 ++ u16 w; 2.620 ++ u16 h; 2.621 ++ struct vnc_pixelfmt fmt; 2.622 ++ u32 len; 2.623 ++ /* char name[len]; */ 2.624 ++}; 2.625 ++ 2.626 ++struct vnc_rect { 2.627 ++ u16 x, y; 2.628 ++ u16 w, h; 2.629 ++ u32 enc; 2.630 ++ /* rect bytes */ 2.631 ++}; 2.632 ++ 2.633 ++struct vnc_server_fbup { 2.634 ++ u8 type; 2.635 ++ u8 pad; 2.636 ++ u16 n; 2.637 ++ /* struct vnc_rect rects[n]; */ 2.638 ++}; 2.639 ++ 2.640 ++struct vnc_server_cuttext { 2.641 ++ u8 type; 2.642 ++ u8 pad1; 2.643 ++ u16 pad2; 2.644 ++ u32 len; 2.645 ++ /* char text[length] */ 2.646 ++}; 2.647 ++ 2.648 ++struct vnc_server_colormap { 2.649 ++ u8 type; 2.650 ++ u8 pad; 2.651 ++ u16 first; 2.652 ++ u16 n; 2.653 ++ /* u8 colors[n * 3 * 2]; */ 2.654 ++}; 2.655 ++ 2.656 ++struct vnc_client_pixelfmt { 2.657 ++ u8 type; 2.658 ++ u8 pad1; 2.659 ++ u16 pad2; 2.660 ++ struct vnc_pixelfmt format; 2.661 ++}; 2.662 ++ 2.663 ++struct vnc_client_fbup { 2.664 ++ u8 type; 2.665 ++ u8 inc; 2.666 ++ u16 x; 2.667 ++ u16 y; 2.668 ++ u16 w; 2.669 ++ u16 h; 2.670 ++}; 2.671 ++ 2.672 ++struct vnc_client_keyevent { 2.673 ++ u8 type; 2.674 ++ u8 down; 2.675 ++ u16 pad; 2.676 ++ u32 key; 2.677 ++}; 2.678 ++ 2.679 ++struct vnc_client_ratevent { 2.680 ++ u8 type; 2.681 ++ u8 mask; 2.682 ++ u16 x; 2.683 ++ u16 y; 2.684 ++};
3.1 --- a/busybox/stuff/busybox-1.22.config Fri Oct 24 12:46:53 2014 +0200 3.2 +++ b/busybox/stuff/busybox-1.22.config Fri Oct 24 16:26:40 2014 +0200 3.3 @@ -527,6 +527,7 @@ 3.4 # Linux System Utilities 3.5 # 3.6 CONFIG_BLOCKDEV=y 3.7 +CONFIG_FBVNC=y 3.8 CONFIG_FSTRIM=y 3.9 # CONFIG_MDEV is not set 3.10 # CONFIG_FEATURE_MDEV_CONF is not set
4.1 --- a/busybox/stuff/busybox-1.22.config-ssfs Fri Oct 24 12:46:53 2014 +0200 4.2 +++ b/busybox/stuff/busybox-1.22.config-ssfs Fri Oct 24 16:26:40 2014 +0200 4.3 @@ -527,6 +527,7 @@ 4.4 # Linux System Utilities 4.5 # 4.6 # CONFIG_BLOCKDEV is not set 4.7 +# CONFIG_FBVNC is not set 4.8 # CONFIG_FSTRIM is not set 4.9 # CONFIG_MDEV is not set 4.10 # CONFIG_FEATURE_MDEV_CONF is not set
5.1 --- a/busybox/stuff/busybox-1.22.config-static Fri Oct 24 12:46:53 2014 +0200 5.2 +++ b/busybox/stuff/busybox-1.22.config-static Fri Oct 24 16:26:40 2014 +0200 5.3 @@ -527,6 +527,7 @@ 5.4 # Linux System Utilities 5.5 # 5.6 # CONFIG_BLOCKDEV is not set 5.7 +# CONFIG_FBVNC is not set 5.8 # CONFIG_FSTRIM is not set 5.9 # CONFIG_MDEV is not set 5.10 # CONFIG_FEATURE_MDEV_CONF is not set