* (c) 2012 Michael Tinglof, Joe Balaz, Les Piech (Mercuri.ca)
*/
-/* [module]
- * import Util from "./util";
- * import Display from "./display";
- * import { Keyboard, Mouse } from "./input/devices"
- * import Websock from "./websock"
- * import Base64 from "./base64";
- * import DES from "./des";
- * import KeyTable from "./input/keysym";
- * import XtScancode from "./input/xtscancodes";
- * import Inflator from "./inflator.mod";
- */
+import * as Log from './util/logging.js';
+import _ from './util/localization.js';
+import { decodeUTF8 } from './util/strings.js';
+import { set_defaults, make_properties } from './util/properties.js';
+import Display from "./display.js";
+import Keyboard from "./input/keyboard.js";
+import Mouse from "./input/mouse.js";
+import Websock from "./websock.js";
+import Base64 from "./base64.js";
+import DES from "./des.js";
+import KeyTable from "./input/keysym.js";
+import XtScancode from "./input/xtscancodes.js";
+import Inflator from "./inflator.js";
+
/*jslint white: false, browser: true */
/*global window, Util, Display, Keyboard, Mouse, Websock, Websock_native, Base64, DES, KeyTable, Inflator, XtScancode */
-/* [module] export default */ function RFB(defaults) {
+export default function RFB(defaults) {
"use strict";
if (!defaults) {
defaults = {};
this._rfb_password = '';
this._rfb_path = '';
- this._rfb_state = 'disconnected';
+ this._rfb_connection_state = '';
+ this._rfb_init_state = '';
this._rfb_version = 0;
this._rfb_max_version = 3.8;
this._rfb_auth_scheme = '';
+ this._rfb_disconnect_reason = "";
this._rfb_tightvnc = false;
this._rfb_xvp_ver = 0;
['JPEG_quality_med', -26 ],
//['JPEG_quality_hi', -23 ],
//['compress_lo', -255 ],
- ['compress_hi', -247 ],
+ ['compress_hi', -254 ],
+ //['compress_max', -247 ],
['DesktopSize', -223 ],
['last_rect', -224 ],
this._sock = null; // Websock object
this._display = null; // Display object
+ this._flushing = false; // Display flushing state
this._keyboard = null; // Keyboard input handler object
this._mouse = null; // Mouse input handler object
this._disconnTimer = null; // disconnection timer
- this._msgTimer = null; // queued handle_msg timer
this._supportsFence = false;
zlib: [] // TIGHT zlib streams
};
- this._fb_Bpp = 4;
- this._fb_depth = 3;
this._fb_width = 0;
this._fb_height = 0;
this._fb_name = "";
this._qemuExtKeyEventSupported = false;
// set the default value on user-facing properties
- Util.set_defaults(this, defaults, {
+ set_defaults(this, defaults, {
'target': 'null', // VNC display rendering Canvas object
'focusContainer': document, // DOM element that captures keyboard input
'encrypt': false, // Use TLS/SSL/wss encryption
- 'true_color': true, // Request true color pixel data
'local_cursor': false, // Request locally rendered cursor
'shared': true, // Request shared mode
'view_only': false, // Disable client mouse/keyboard
'viewportDrag': false, // Move the viewport on mouse drags
// Callback functions
- 'onUpdateState': function () { }, // onUpdateState(rfb, state, oldstate, statusMsg): state update/change
+ 'onUpdateState': function () { }, // onUpdateState(rfb, state, oldstate): connection state change
'onNotification': function () { }, // onNotification(rfb, msg, level, options): notification for UI
+ 'onDisconnected': function () { }, // onDisconnected(rfb, reason): disconnection finished
'onPasswordRequired': function () { }, // onPasswordRequired(rfb, msg): VNC password is required
'onClipboard': function () { }, // onClipboard(rfb, text): RFB clipboard contents received
'onBell': function () { }, // onBell(rfb): RFB Bell message received
});
// main setup
- Util.Debug(">> RFB.constructor");
+ Log.Debug(">> RFB.constructor");
// populate encHandlers with bound versions
Object.keys(RFB.encodingHandlers).forEach(function (encName) {
// NB: nothing that needs explicit teardown should be done
// before this point, since this can throw an exception
try {
- this._display = new Display({target: this._target});
+ this._display = new Display({target: this._target,
+ onFlush: this._onFlush.bind(this)});
} catch (exc) {
- Util.Error("Display exception: " + exc);
+ Log.Error("Display exception: " + exc);
throw exc;
}
this._keyboard = new Keyboard({target: this._focusContainer,
- onKeyPress: this._handleKeyPress.bind(this)});
+ onKeyEvent: this._handleKeyEvent.bind(this)});
this._mouse = new Mouse({target: this._target,
onMouseButton: this._handleMouseButton.bind(this),
- onMouseMove: this._handleMouseMove.bind(this),
- notify: this._keyboard.sync.bind(this._keyboard)});
+ onMouseMove: this._handleMouseMove.bind(this)});
this._sock = new Websock();
this._sock.on('message', this._handle_message.bind(this));
this._sock.on('open', function () {
- if (this._rfb_state === 'connect') {
- this._updateState('ProtocolVersion', "Starting VNC handshake");
+ if ((this._rfb_connection_state === 'connecting') &&
+ (this._rfb_init_state === '')) {
+ this._rfb_init_state = 'ProtocolVersion';
+ Log.Debug("Starting VNC handshake");
} else {
- this._fail("Got unexpected WebSocket connection");
+ this._fail("Unexpected server connection");
}
}.bind(this));
this._sock.on('close', function (e) {
- Util.Warn("WebSocket on-close event");
+ Log.Warn("WebSocket on-close event");
var msg = "";
if (e.code) {
msg = " (code: " + e.code;
}
msg += ")";
}
- if (this._rfb_state === 'disconnect') {
- this._updateState('disconnected', 'VNC disconnected' + msg);
- } else if (this._rfb_state === 'ProtocolVersion') {
- this._fail('Failed to connect to server' + msg);
- } else if (this._rfb_state in {'failed': 1, 'disconnected': 1}) {
- Util.Error("Received onclose while disconnected" + msg);
- } else {
- this._fail("Server disconnected" + msg);
+ switch (this._rfb_connection_state) {
+ case 'disconnecting':
+ this._updateConnectionState('disconnected');
+ break;
+ case 'connecting':
+ this._fail('Failed to connect to server', msg);
+ break;
+ case 'connected':
+ // Handle disconnects that were initiated server-side
+ this._updateConnectionState('disconnecting');
+ this._updateConnectionState('disconnected');
+ break;
+ case 'disconnected':
+ this._fail("Unexpected server disconnect",
+ "Already disconnected: " + msg);
+ break;
+ default:
+ this._fail("Unexpected server disconnect",
+ "Not in any state yet: " + msg);
+ break;
}
this._sock.off('close');
}.bind(this));
this._sock.on('error', function (e) {
- Util.Warn("WebSocket on-error event");
+ Log.Warn("WebSocket on-error event");
});
this._init_vars();
+ this._cleanup();
var rmode = this._display.get_render_mode();
- Util.Info("Using native WebSockets");
- this._updateState('loaded', 'noVNC ready: native WebSockets, ' + rmode);
+ Log.Info("Using native WebSockets, render mode: " + rmode);
- Util.Debug("<< RFB.constructor");
+ Log.Debug("<< RFB.constructor");
};
-(function() {
- RFB.prototype = {
- // Public methods
- connect: function (host, port, password, path) {
- this._rfb_host = host;
- this._rfb_port = port;
- this._rfb_password = (password !== undefined) ? password : "";
- this._rfb_path = (path !== undefined) ? path : "";
-
- if (!this._rfb_host || !this._rfb_port) {
- return this._fail("Must set host and port");
- }
+RFB.prototype = {
+ // Public methods
+ connect: function (host, port, password, path) {
+ this._rfb_host = host;
+ this._rfb_port = port;
+ this._rfb_password = (password !== undefined) ? password : "";
+ this._rfb_path = (path !== undefined) ? path : "";
+
+ if (!this._rfb_host) {
+ return this._fail(
+ _("Must set host"));
+ }
- this._updateState('connect');
- return true;
- },
-
- disconnect: function () {
- this._updateState('disconnect', 'Disconnecting');
- this._sock.off('error');
- this._sock.off('message');
- this._sock.off('open');
- },
-
- sendPassword: function (passwd) {
- this._rfb_password = passwd;
- setTimeout(this._init_msg.bind(this), 0);
- },
-
- sendCtrlAltDel: function () {
- if (this._rfb_state !== 'normal' || this._view_only) { return false; }
- Util.Info("Sending Ctrl-Alt-Del");
-
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Control_L, 1);
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Alt_L, 1);
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Delete, 1);
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Delete, 0);
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Alt_L, 0);
- RFB.messages.keyEvent(this._sock, KeyTable.XK_Control_L, 0);
+ this._rfb_init_state = '';
+ this._updateConnectionState('connecting');
+ return true;
+ },
+
+ disconnect: function () {
+ this._updateConnectionState('disconnecting');
+ this._sock.off('error');
+ this._sock.off('message');
+ this._sock.off('open');
+ },
+
+ sendPassword: function (passwd) {
+ this._rfb_password = passwd;
+ setTimeout(this._init_msg.bind(this), 0);
+ },
+
+ sendCtrlAltDel: function () {
+ if (this._rfb_connection_state !== 'connected' || this._view_only) { return false; }
+ Log.Info("Sending Ctrl-Alt-Del");
+
+ this.sendKey(KeyTable.XK_Control_L, "ControlLeft", true);
+ this.sendKey(KeyTable.XK_Alt_L, "AltLeft", true);
+ this.sendKey(KeyTable.XK_Delete, "Delete", true);
+ this.sendKey(KeyTable.XK_Delete, "Delete", false);
+ this.sendKey(KeyTable.XK_Alt_L, "AltLeft", false);
+ this.sendKey(KeyTable.XK_Control_L, "ControlLeft", false);
+
+ return true;
+ },
+
+ xvpOp: function (ver, op) {
+ if (this._rfb_xvp_ver < ver) { return false; }
+ Log.Info("Sending XVP operation " + op + " (version " + ver + ")");
+ this._sock.send_string("\xFA\x00" + String.fromCharCode(ver) + String.fromCharCode(op));
+ return true;
+ },
+
+ xvpShutdown: function () {
+ return this.xvpOp(1, 2);
+ },
+
+ xvpReboot: function () {
+ return this.xvpOp(1, 3);
+ },
+
+ xvpReset: function () {
+ return this.xvpOp(1, 4);
+ },
+
+ // Send a key press. If 'down' is not specified then send a down key
+ // followed by an up key.
+ sendKey: function (keysym, code, down) {
+ if (this._rfb_connection_state !== 'connected' || this._view_only) { return false; }
+
+ if (down === undefined) {
+ this.sendKey(keysym, code, true);
+ this.sendKey(keysym, code, false);
return true;
- },
+ }
- xvpOp: function (ver, op) {
- if (this._rfb_xvp_ver < ver) { return false; }
- Util.Info("Sending XVP operation " + op + " (version " + ver + ")");
- this._sock.send_string("\xFA\x00" + String.fromCharCode(ver) + String.fromCharCode(op));
- return true;
- },
-
- xvpShutdown: function () {
- return this.xvpOp(1, 2);
- },
-
- xvpReboot: function () {
- return this.xvpOp(1, 3);
- },
-
- xvpReset: function () {
- return this.xvpOp(1, 4);
- },
-
- // Send a key press. If 'down' is not specified then send a down key
- // followed by an up key.
- sendKey: function (code, down) {
- if (this._rfb_state !== "normal" || this._view_only) { return false; }
- if (typeof down !== 'undefined') {
- Util.Info("Sending key code (" + (down ? "down" : "up") + "): " + code);
- RFB.messages.keyEvent(this._sock, code, down ? 1 : 0);
- } else {
- Util.Info("Sending key code (down + up): " + code);
- RFB.messages.keyEvent(this._sock, code, 1);
- RFB.messages.keyEvent(this._sock, code, 0);
+ var scancode = XtScancode[code];
+
+ if (this._qemuExtKeyEventSupported && scancode) {
+ // 0 is NoSymbol
+ keysym = keysym || 0;
+
+ Log.Info("Sending key (" + (down ? "down" : "up") + "): keysym " + keysym + ", scancode " + scancode);
+
+ RFB.messages.QEMUExtendedKeyEvent(this._sock, keysym, down, scancode);
+ } else {
+ if (!keysym) {
+ return false;
}
+ Log.Info("Sending keysym (" + (down ? "down" : "up") + "): " + keysym);
+ RFB.messages.keyEvent(this._sock, keysym, down ? 1 : 0);
+ }
+
+ return true;
+ },
+
+ clipboardPasteFrom: function (text) {
+ if (this._rfb_connection_state !== 'connected' || this._view_only) { return; }
+ RFB.messages.clientCutText(this._sock, text);
+ },
+
+ // Requests a change of remote desktop size. This message is an extension
+ // and may only be sent if we have received an ExtendedDesktopSize message
+ requestDesktopSize: function (width, height) {
+ if (this._rfb_connection_state !== 'connected' ||
+ this._view_only) {
+ return false;
+ }
+
+ if (this._supportsSetDesktopSize) {
+ RFB.messages.setDesktopSize(this._sock, width, height,
+ this._screen_id, this._screen_flags);
+ this._sock.flush();
return true;
- },
-
- clipboardPasteFrom: function (text) {
- if (this._rfb_state !== 'normal') { return; }
- RFB.messages.clientCutText(this._sock, text);
- },
-
- // Requests a change of remote desktop size. This message is an extension
- // and may only be sent if we have received an ExtendedDesktopSize message
- requestDesktopSize: function (width, height) {
- if (this._rfb_state !== "normal") { return; }
-
- if (this._supportsSetDesktopSize) {
- RFB.messages.setDesktopSize(this._sock, width, height,
- this._screen_id, this._screen_flags);
- this._sock.flush();
- }
- },
+ } else {
+ return false;
+ }
+ },
- // Private methods
+ // Private methods
- _connect: function () {
- Util.Debug(">> RFB.connect");
+ _connect: function () {
+ Log.Debug(">> RFB.connect");
+ this._init_vars();
- var uri;
- if (typeof UsingSocketIO !== 'undefined') {
- uri = 'http';
- } else {
- uri = this._encrypt ? 'wss' : 'ws';
- }
+ var uri;
+ if (typeof UsingSocketIO !== 'undefined') {
+ uri = 'http';
+ } else {
+ uri = this._encrypt ? 'wss' : 'ws';
+ }
- uri += '://' + this._rfb_host + ':' + this._rfb_port + '/' + this._rfb_path;
- Util.Info("connecting to " + uri);
+ uri += '://' + this._rfb_host;
+ if(this._rfb_port) {
+ uri += ':' + this._rfb_port;
+ }
+ uri += '/' + this._rfb_path;
- this._sock.open(uri, this._wsProtocols);
+ Log.Info("connecting to " + uri);
- Util.Debug("<< RFB.connect");
- },
-
- _init_vars: function () {
- // reset state
- this._FBU.rects = 0;
- this._FBU.subrects = 0; // RRE and HEXTILE
- this._FBU.lines = 0; // RAW
- this._FBU.tiles = 0; // HEXTILE
- this._FBU.zlibs = []; // TIGHT zlib encoders
- this._mouse_buttonMask = 0;
- this._mouse_arr = [];
- this._rfb_tightvnc = false;
-
- // Clear the per connection encoding stats
- var i;
- for (i = 0; i < this._encodings.length; i++) {
- this._encStats[this._encodings[i][1]][0] = 0;
+ try {
+ // WebSocket.onopen transitions to the RFB init states
+ this._sock.open(uri, this._wsProtocols);
+ } catch (e) {
+ if (e.name === 'SyntaxError') {
+ this._fail("Invalid host or port value given", e);
+ } else {
+ this._fail("Error while connecting", e);
}
+ }
- for (i = 0; i < 4; i++) {
- this._FBU.zlibs[i] = new Inflator.Inflate();
- }
- },
-
- _print_stats: function () {
- Util.Info("Encoding stats for this connection:");
- var i, s;
- for (i = 0; i < this._encodings.length; i++) {
- s = this._encStats[this._encodings[i][1]];
- if (s[0] + s[1] > 0) {
- Util.Info(" " + this._encodings[i][0] + ": " + s[0] + " rects");
- }
- }
+ Log.Debug("<< RFB.connect");
+ },
+
+ _disconnect: function () {
+ Log.Debug(">> RFB.disconnect");
+ this._cleanup();
+ this._sock.close();
+ this._print_stats();
+ Log.Debug("<< RFB.disconnect");
+ },
+
+ _init_vars: function () {
+ // reset state
+ this._FBU.rects = 0;
+ this._FBU.subrects = 0; // RRE and HEXTILE
+ this._FBU.lines = 0; // RAW
+ this._FBU.tiles = 0; // HEXTILE
+ this._FBU.zlibs = []; // TIGHT zlib encoders
+ this._mouse_buttonMask = 0;
+ this._mouse_arr = [];
+ this._rfb_tightvnc = false;
+
+ // Clear the per connection encoding stats
+ var i;
+ for (i = 0; i < this._encodings.length; i++) {
+ this._encStats[this._encodings[i][1]][0] = 0;
+ }
- Util.Info("Encoding stats since page load:");
- for (i = 0; i < this._encodings.length; i++) {
- s = this._encStats[this._encodings[i][1]];
- Util.Info(" " + this._encodings[i][0] + ": " + s[1] + " rects");
- }
- },
+ for (i = 0; i < 4; i++) {
+ this._FBU.zlibs[i] = new Inflator();
+ }
+ },
- _cleanupSocket: function (state) {
- if (this._msgTimer) {
- clearInterval(this._msgTimer);
- this._msgTimer = null;
+ _print_stats: function () {
+ Log.Info("Encoding stats for this connection:");
+ var i, s;
+ for (i = 0; i < this._encodings.length; i++) {
+ s = this._encStats[this._encodings[i][1]];
+ if (s[0] + s[1] > 0) {
+ Log.Info(" " + this._encodings[i][0] + ": " + s[0] + " rects");
}
+ }
- if (this._display && this._display.get_context()) {
- this._keyboard.ungrab();
- this._mouse.ungrab();
- if (state !== 'connect' && state !== 'loaded') {
- this._display.defaultCursor();
+ Log.Info("Encoding stats since page load:");
+ for (i = 0; i < this._encodings.length; i++) {
+ s = this._encStats[this._encodings[i][1]];
+ Log.Info(" " + this._encodings[i][0] + ": " + s[1] + " rects");
+ }
+ },
+
+ _cleanup: function () {
+ if (!this._view_only) { this._keyboard.ungrab(); }
+ if (!this._view_only) { this._mouse.ungrab(); }
+ this._display.defaultCursor();
+ if (Log.get_logging() !== 'debug') {
+ // Show noVNC logo on load and when disconnected, unless in
+ // debug mode
+ this._display.clear();
+ }
+ },
+
+ /*
+ * Connection states:
+ * connecting
+ * connected
+ * disconnecting
+ * disconnected - permanent state
+ */
+ _updateConnectionState: function (state) {
+ var oldstate = this._rfb_connection_state;
+
+ if (state === oldstate) {
+ Log.Debug("Already in state '" + state + "', ignoring");
+ return;
+ }
+
+ // The 'disconnected' state is permanent for each RFB object
+ if (oldstate === 'disconnected') {
+ Log.Error("Tried changing state of a disconnected RFB object");
+ return;
+ }
+
+ // Ensure proper transitions before doing anything
+ switch (state) {
+ case 'connected':
+ if (oldstate !== 'connecting') {
+ Log.Error("Bad transition to connected state, " +
+ "previous connection state: " + oldstate);
+ return;
}
- if (Util.get_logging() !== 'debug' || state === 'loaded') {
- // Show noVNC logo on load and when disconnected, unless in
- // debug mode
- this._display.clear();
+ break;
+
+ case 'disconnected':
+ if (oldstate !== 'disconnecting') {
+ Log.Error("Bad transition to disconnected state, " +
+ "previous connection state: " + oldstate);
+ return;
}
- }
+ break;
- this._sock.close();
- },
+ case 'connecting':
+ if (oldstate !== '') {
+ Log.Error("Bad transition to connecting state, " +
+ "previous connection state: " + oldstate);
+ return;
+ }
+ break;
- /*
- * Page states:
- * loaded - page load, equivalent to disconnected
- * disconnected - idle state
- * connect - starting to connect (to ProtocolVersion)
- * normal - connected
- * disconnect - starting to disconnect
- * failed - abnormal disconnect
- * fatal - failed to load page, or fatal error
- *
- * RFB protocol initialization states:
- * ProtocolVersion
- * Security
- * Authentication
- * SecurityResult
- * ClientInitialization - not triggered by server message
- * ServerInitialization (to normal)
- */
- _updateState: function (state, statusMsg) {
- var oldstate = this._rfb_state;
+ case 'disconnecting':
+ if (oldstate !== 'connected' && oldstate !== 'connecting') {
+ Log.Error("Bad transition to disconnecting state, " +
+ "previous connection state: " + oldstate);
+ return;
+ }
+ break;
- if (state === oldstate) {
- // Already here, ignore
- Util.Debug("Already in state '" + state + "', ignoring");
+ default:
+ Log.Error("Unknown connection state: " + state);
return;
- }
+ }
- /*
- * These are disconnected states. A previous connect may
- * asynchronously cause a connection so make sure we are closed.
- */
- if (state in {'disconnected': 1, 'loaded': 1, 'connect': 1,
- 'disconnect': 1, 'failed': 1, 'fatal': 1}) {
- this._cleanupSocket(state);
- }
+ // State change actions
- if (oldstate === 'fatal') {
- Util.Error('Fatal error, cannot continue');
- }
+ this._rfb_connection_state = state;
+ this._onUpdateState(this, state, oldstate);
- var cmsg = typeof(statusMsg) !== 'undefined' ? (" Msg: " + statusMsg) : "";
- var fullmsg = "New state '" + state + "', was '" + oldstate + "'." + cmsg;
- if (state === 'failed' || state === 'fatal') {
- Util.Error(cmsg);
- } else {
- Util.Warn(cmsg);
- }
+ var smsg = "New state '" + state + "', was '" + oldstate + "'.";
+ Log.Debug(smsg);
- if (oldstate === 'failed' && state === 'disconnected') {
- // do disconnect action, but stay in failed state
- this._rfb_state = 'failed';
- } else {
- this._rfb_state = state;
- }
+ if (this._disconnTimer && state !== 'disconnecting') {
+ Log.Debug("Clearing disconnect timer");
+ clearTimeout(this._disconnTimer);
+ this._disconnTimer = null;
- if (this._disconnTimer && this._rfb_state !== 'disconnect') {
- Util.Debug("Clearing disconnect timer");
- clearTimeout(this._disconnTimer);
- this._disconnTimer = null;
- this._sock.off('close'); // make sure we don't get a double event
- }
+ // make sure we don't get a double event
+ this._sock.off('close');
+ }
- switch (state) {
- case 'normal':
- if (oldstate === 'disconnected' || oldstate === 'failed') {
- Util.Error("Invalid transition from 'disconnected' or 'failed' to 'normal'");
- }
- break;
-
- case 'connect':
- this._init_vars();
- this._connect();
- // WebSocket.onopen transitions to 'ProtocolVersion'
- break;
-
- case 'disconnect':
- this._disconnTimer = setTimeout(function () {
- this._fail("Disconnect timeout");
- }.bind(this), this._disconnectTimeout * 1000);
-
- this._print_stats();
-
- // WebSocket.onclose transitions to 'disconnected'
- break;
-
- case 'failed':
- if (oldstate === 'disconnected') {
- Util.Error("Invalid transition from 'disconnected' to 'failed'");
- } else if (oldstate === 'normal') {
- Util.Error("Error while connected.");
- } else if (oldstate === 'init') {
- Util.Error("Error while initializing.");
- }
+ switch (state) {
+ case 'disconnected':
+ // Call onDisconnected callback after onUpdateState since
+ // we don't know if the UI only displays the latest message
+ if (this._rfb_disconnect_reason !== "") {
+ this._onDisconnected(this, this._rfb_disconnect_reason);
+ } else {
+ // No reason means clean disconnect
+ this._onDisconnected(this);
+ }
+ break;
- // Make sure we transition to disconnected
- setTimeout(function () {
- this._updateState('disconnected');
- }.bind(this), 50);
+ case 'connecting':
+ this._connect();
+ break;
- break;
+ case 'disconnecting':
+ this._disconnect();
- default:
- // No state change action to take
- }
+ this._disconnTimer = setTimeout(function () {
+ this._rfb_disconnect_reason = _("Disconnect timeout");
+ this._updateConnectionState('disconnected');
+ }.bind(this), this._disconnectTimeout * 1000);
+ break;
+ }
+ },
+
+ /* Print errors and disconnect
+ *
+ * The optional parameter 'details' is used for information that
+ * should be logged but not sent to the user interface.
+ */
+ _fail: function (msg, details) {
+ var fullmsg = msg;
+ if (typeof details !== 'undefined') {
+ fullmsg = msg + " (" + details + ")";
+ }
+ switch (this._rfb_connection_state) {
+ case 'disconnecting':
+ Log.Error("Failed when disconnecting: " + fullmsg);
+ break;
+ case 'connected':
+ Log.Error("Failed while connected: " + fullmsg);
+ break;
+ case 'connecting':
+ Log.Error("Failed when connecting: " + fullmsg);
+ break;
+ default:
+ Log.Error("RFB failure: " + fullmsg);
+ break;
+ }
+ this._rfb_disconnect_reason = msg; //This is sent to the UI
+
+ // Transition to disconnected without waiting for socket to close
+ this._updateConnectionState('disconnecting');
+ this._updateConnectionState('disconnected');
+
+ return false;
+ },
+
+ /*
+ * Send a notification to the UI. Valid levels are:
+ * 'normal'|'warn'|'error'
+ *
+ * NOTE: Options could be added in the future.
+ * NOTE: If this function is called multiple times, remember that the
+ * interface could be only showing the latest notification.
+ */
+ _notification: function(msg, level, options) {
+ switch (level) {
+ case 'normal':
+ case 'warn':
+ case 'error':
+ Log.Debug("Notification[" + level + "]:" + msg);
+ break;
+ default:
+ Log.Error("Invalid notification level: " + level);
+ return;
+ }
- if (oldstate === 'failed' && state === 'disconnected') {
- this._onUpdateState(this, state, oldstate);
- } else {
- this._onUpdateState(this, state, oldstate, statusMsg);
- }
- },
+ if (options) {
+ this._onNotification(this, msg, level, options);
+ } else {
+ this._onNotification(this, msg, level);
+ }
+ },
- _fail: function (msg) {
- this._updateState('failed', msg);
- return false;
- },
+ _handle_message: function () {
+ if (this._sock.rQlen() === 0) {
+ Log.Warn("handle_message called on an empty receive queue");
+ return;
+ }
- /*
- * Send a notification to the UI. Valid levels are:
- * 'normal'|'warn'|'error'
- *
- * NOTE: Options could be added in the future.
- * NOTE: If this function is called multiple times, remember that the
- * interface could be only showing the latest notification.
- */
- _notification: function(msg, level, options) {
- switch (level) {
- case 'normal':
- case 'warn':
- case 'error':
- Util.Debug("Notification[" + level + "]:" + msg);
- break;
- default:
- Util.Error("Invalid notification level: " + level);
- return;
- }
+ switch (this._rfb_connection_state) {
+ case 'disconnected':
+ Log.Error("Got data while disconnected");
+ break;
+ case 'connected':
+ while (true) {
+ if (this._flushing) {
+ break;
+ }
+ if (!this._normal_msg()) {
+ break;
+ }
+ if (this._sock.rQlen() === 0) {
+ break;
+ }
+ }
+ break;
+ default:
+ this._init_msg();
+ break;
+ }
+ },
- if (options) {
- this._onNotification(this, msg, level, options);
- } else {
- this._onNotification(this, msg, level);
- }
- },
+ _handleKeyEvent: function (keysym, code, down) {
+ this.sendKey(keysym, code, down);
+ },
- _handle_message: function () {
- if (this._sock.rQlen() === 0) {
- Util.Warn("handle_message called on an empty receive queue");
- return;
- }
+ _handleMouseButton: function (x, y, down, bmask) {
+ if (down) {
+ this._mouse_buttonMask |= bmask;
+ } else {
+ this._mouse_buttonMask &= ~bmask;
+ }
- switch (this._rfb_state) {
- case 'disconnected':
- case 'failed':
- Util.Error("Got data while disconnected");
- break;
- case 'normal':
- if (this._normal_msg() && this._sock.rQlen() > 0) {
- // true means we can continue processing
- // Give other events a chance to run
- if (this._msgTimer === null) {
- Util.Debug("More data to process, creating timer");
- this._msgTimer = setTimeout(function () {
- this._msgTimer = null;
- this._handle_message();
- }.bind(this), 0);
- } else {
- Util.Debug("More data to process, existing timer");
- }
- }
- break;
- default:
- this._init_msg();
- break;
- }
- },
+ if (this._viewportDrag) {
+ if (down && !this._viewportDragging) {
+ this._viewportDragging = true;
+ this._viewportDragPos = {'x': x, 'y': y};
- _handleKeyPress: function (keyevent) {
- if (this._view_only) { return; } // View only, skip keyboard, events
+ // Skip sending mouse events
+ return;
+ } else {
+ this._viewportDragging = false;
- var down = (keyevent.type == 'keydown');
- if (this._qemuExtKeyEventSupported) {
- var scancode = XtScancode[keyevent.code];
- if (scancode) {
- var keysym = keyevent.keysym;
- RFB.messages.QEMUExtendedKeyEvent(this._sock, keysym, down, scancode);
- } else {
- Util.Error('Unable to find a xt scancode for code = ' + keyevent.code);
+ // If the viewport didn't actually move, then treat as a mouse click event
+ // Send the button down event here, as the button up event is sent at the end of this function
+ if (!this._viewportHasMoved && !this._view_only) {
+ RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), bmask);
}
- } else {
- keysym = keyevent.keysym.keysym;
- RFB.messages.keyEvent(this._sock, keysym, down);
+ this._viewportHasMoved = false;
}
- },
+ }
- _handleMouseButton: function (x, y, down, bmask) {
- if (down) {
- this._mouse_buttonMask |= bmask;
- } else {
- this._mouse_buttonMask ^= bmask;
- }
+ if (this._view_only) { return; } // View only, skip mouse events
- if (this._viewportDrag) {
- if (down && !this._viewportDragging) {
- this._viewportDragging = true;
- this._viewportDragPos = {'x': x, 'y': y};
+ if (this._rfb_connection_state !== 'connected') { return; }
+ RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), this._mouse_buttonMask);
+ },
- // Skip sending mouse events
- return;
- } else {
- this._viewportDragging = false;
+ _handleMouseMove: function (x, y) {
+ if (this._viewportDragging) {
+ var deltaX = this._viewportDragPos.x - x;
+ var deltaY = this._viewportDragPos.y - y;
- // If the viewport didn't actually move, then treat as a mouse click event
- // Send the button down event here, as the button up event is sent at the end of this function
- if (!this._viewportHasMoved && !this._view_only) {
- RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), bmask);
- }
- this._viewportHasMoved = false;
- }
+ // The goal is to trigger on a certain physical width, the
+ // devicePixelRatio brings us a bit closer but is not optimal.
+ var dragThreshold = 10 * (window.devicePixelRatio || 1);
+
+ if (this._viewportHasMoved || (Math.abs(deltaX) > dragThreshold ||
+ Math.abs(deltaY) > dragThreshold)) {
+ this._viewportHasMoved = true;
+
+ this._viewportDragPos = {'x': x, 'y': y};
+ this._display.viewportChangePos(deltaX, deltaY);
}
- if (this._view_only) { return; } // View only, skip mouse events
+ // Skip sending mouse events
+ return;
+ }
- if (this._rfb_state !== "normal") { return; }
- RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), this._mouse_buttonMask);
- },
+ if (this._view_only) { return; } // View only, skip mouse events
- _handleMouseMove: function (x, y) {
- if (this._viewportDragging) {
- var deltaX = this._viewportDragPos.x - x;
- var deltaY = this._viewportDragPos.y - y;
+ if (this._rfb_connection_state !== 'connected') { return; }
+ RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), this._mouse_buttonMask);
+ },
- // The goal is to trigger on a certain physical width, the
- // devicePixelRatio brings us a bit closer but is not optimal.
- var dragThreshold = 10 * (window.devicePixelRatio || 1);
+ // Message Handlers
- if (this._viewportHasMoved || (Math.abs(deltaX) > dragThreshold ||
- Math.abs(deltaY) > dragThreshold)) {
- this._viewportHasMoved = true;
+ _negotiate_protocol_version: function () {
+ if (this._sock.rQlen() < 12) {
+ return this._fail("Error while negotiating with server",
+ "Incomplete protocol version");
+ }
- this._viewportDragPos = {'x': x, 'y': y};
- this._display.viewportChangePos(deltaX, deltaY);
- }
+ var sversion = this._sock.rQshiftStr(12).substr(4, 7);
+ Log.Info("Server ProtocolVersion: " + sversion);
+ var is_repeater = 0;
+ switch (sversion) {
+ case "000.000": // UltraVNC repeater
+ is_repeater = 1;
+ break;
+ case "003.003":
+ case "003.006": // UltraVNC
+ case "003.889": // Apple Remote Desktop
+ this._rfb_version = 3.3;
+ break;
+ case "003.007":
+ this._rfb_version = 3.7;
+ break;
+ case "003.008":
+ case "004.000": // Intel AMT KVM
+ case "004.001": // RealVNC 4.6
+ case "005.000": // RealVNC 5.3
+ this._rfb_version = 3.8;
+ break;
+ default:
+ return this._fail("Unsupported server",
+ "Invalid server version: " + sversion);
+ }
- // Skip sending mouse events
- return;
+ if (is_repeater) {
+ var repeaterID = this._repeaterID;
+ while (repeaterID.length < 250) {
+ repeaterID += "\0";
}
+ this._sock.send_string(repeaterID);
+ return true;
+ }
- if (this._view_only) { return; } // View only, skip mouse events
+ if (this._rfb_version > this._rfb_max_version) {
+ this._rfb_version = this._rfb_max_version;
+ }
- if (this._rfb_state !== "normal") { return; }
- RFB.messages.pointerEvent(this._sock, this._display.absX(x), this._display.absY(y), this._mouse_buttonMask);
- },
+ var cversion = "00" + parseInt(this._rfb_version, 10) +
+ ".00" + ((this._rfb_version * 10) % 10);
+ this._sock.send_string("RFB " + cversion + "\n");
+ Log.Debug('Sent ProtocolVersion: ' + cversion);
- // Message Handlers
+ this._rfb_init_state = 'Security';
+ },
- _negotiate_protocol_version: function () {
- if (this._sock.rQlen() < 12) {
- return this._fail("Incomplete protocol version");
+ _negotiate_security: function () {
+ // Polyfill since IE and PhantomJS doesn't have
+ // TypedArray.includes()
+ function includes(item, array) {
+ for (var i = 0; i < array.length; i++) {
+ if (array[i] === item) {
+ return true;
+ }
}
+ return false;
+ }
- var sversion = this._sock.rQshiftStr(12).substr(4, 7);
- Util.Info("Server ProtocolVersion: " + sversion);
- var is_repeater = 0;
- switch (sversion) {
- case "000.000": // UltraVNC repeater
- is_repeater = 1;
- break;
- case "003.003":
- case "003.006": // UltraVNC
- case "003.889": // Apple Remote Desktop
- this._rfb_version = 3.3;
- break;
- case "003.007":
- this._rfb_version = 3.7;
- break;
- case "003.008":
- case "004.000": // Intel AMT KVM
- case "004.001": // RealVNC 4.6
- case "005.000": // RealVNC 5.3
- this._rfb_version = 3.8;
- break;
- default:
- return this._fail("Invalid server version " + sversion);
+ if (this._rfb_version >= 3.7) {
+ // Server sends supported list, client decides
+ var num_types = this._sock.rQshift8();
+ if (this._sock.rQwait("security type", num_types, 1)) { return false; }
+
+ if (num_types === 0) {
+ var strlen = this._sock.rQshift32();
+ var reason = this._sock.rQshiftStr(strlen);
+ return this._fail("Error while negotiating with server",
+ "Security failure: " + reason);
+ }
+
+ var types = this._sock.rQshiftBytes(num_types);
+ Log.Debug("Server security types: " + types);
+
+ // Look for each auth in preferred order
+ this._rfb_auth_scheme = 0;
+ if (includes(1, types)) {
+ this._rfb_auth_scheme = 1; // None
+ } else if (includes(22, types)) {
+ this._rfb_auth_scheme = 22; // XVP
+ } else if (includes(16, types)) {
+ this._rfb_auth_scheme = 16; // Tight
+ } else if (includes(2, types)) {
+ this._rfb_auth_scheme = 2; // VNC Auth
+ } else {
+ return this._fail("Unsupported server",
+ "Unsupported security types: " + types);
}
- if (is_repeater) {
- var repeaterID = this._repeaterID;
- while (repeaterID.length < 250) {
- repeaterID += "\0";
- }
- this._sock.send_string(repeaterID);
- return true;
- }
+ this._sock.send([this._rfb_auth_scheme]);
+ } else {
+ // Server decides
+ if (this._sock.rQwait("security scheme", 4)) { return false; }
+ this._rfb_auth_scheme = this._sock.rQshift32();
+ }
- if (this._rfb_version > this._rfb_max_version) {
- this._rfb_version = this._rfb_max_version;
- }
+ this._rfb_init_state = 'Authentication';
+ Log.Debug('Authenticating using scheme: ' + this._rfb_auth_scheme);
- var cversion = "00" + parseInt(this._rfb_version, 10) +
- ".00" + ((this._rfb_version * 10) % 10);
- this._sock.send_string("RFB " + cversion + "\n");
- this._updateState('Security', 'Sent ProtocolVersion: ' + cversion);
- },
-
- _negotiate_security: function () {
- if (this._rfb_version >= 3.7) {
- // Server sends supported list, client decides
- var num_types = this._sock.rQshift8();
- if (this._sock.rQwait("security type", num_types, 1)) { return false; }
-
- if (num_types === 0) {
- var strlen = this._sock.rQshift32();
- var reason = this._sock.rQshiftStr(strlen);
- return this._fail("Security failure: " + reason);
- }
+ return this._init_msg(); // jump to authentication
+ },
- this._rfb_auth_scheme = 0;
- var types = this._sock.rQshiftBytes(num_types);
- Util.Debug("Server security types: " + types);
- for (var i = 0; i < types.length; i++) {
- if (types[i] > this._rfb_auth_scheme && (types[i] <= 16 || types[i] == 22)) {
- this._rfb_auth_scheme = types[i];
- }
- }
+ // authentication
+ _negotiate_xvp_auth: function () {
+ var xvp_sep = this._xvp_password_sep;
+ var xvp_auth = this._rfb_password.split(xvp_sep);
+ if (xvp_auth.length < 3) {
+ var msg = 'XVP credentials required (user' + xvp_sep +
+ 'target' + xvp_sep + 'password) -- got only ' + this._rfb_password;
+ this._onPasswordRequired(this, msg);
+ return false;
+ }
- if (this._rfb_auth_scheme === 0) {
- return this._fail("Unsupported security types: " + types);
- }
+ var xvp_auth_str = String.fromCharCode(xvp_auth[0].length) +
+ String.fromCharCode(xvp_auth[1].length) +
+ xvp_auth[0] +
+ xvp_auth[1];
+ this._sock.send_string(xvp_auth_str);
+ this._rfb_password = xvp_auth.slice(2).join(xvp_sep);
+ this._rfb_auth_scheme = 2;
+ return this._negotiate_authentication();
+ },
+
+ _negotiate_std_vnc_auth: function () {
+ if (this._rfb_password.length === 0) {
+ this._onPasswordRequired(this);
+ return false;
+ }
- this._sock.send([this._rfb_auth_scheme]);
- } else {
- // Server decides
- if (this._sock.rQwait("security scheme", 4)) { return false; }
- this._rfb_auth_scheme = this._sock.rQshift32();
- }
+ if (this._sock.rQwait("auth challenge", 16)) { return false; }
+
+ // TODO(directxman12): make genDES not require an Array
+ var challenge = Array.prototype.slice.call(this._sock.rQshiftBytes(16));
+ var response = RFB.genDES(this._rfb_password, challenge);
+ this._sock.send(response);
+ this._rfb_init_state = "SecurityResult";
+ return true;
+ },
+
+ _negotiate_tight_tunnels: function (numTunnels) {
+ var clientSupportedTunnelTypes = {
+ 0: { vendor: 'TGHT', signature: 'NOTUNNEL' }
+ };
+ var serverSupportedTunnelTypes = {};
+ // receive tunnel capabilities
+ for (var i = 0; i < numTunnels; i++) {
+ var cap_code = this._sock.rQshift32();
+ var cap_vendor = this._sock.rQshiftStr(4);
+ var cap_signature = this._sock.rQshiftStr(8);
+ serverSupportedTunnelTypes[cap_code] = { vendor: cap_vendor, signature: cap_signature };
+ }
- this._updateState('Authentication', 'Authenticating using scheme: ' + this._rfb_auth_scheme);
- return this._init_msg(); // jump to authentication
- },
-
- // authentication
- _negotiate_xvp_auth: function () {
- var xvp_sep = this._xvp_password_sep;
- var xvp_auth = this._rfb_password.split(xvp_sep);
- if (xvp_auth.length < 3) {
- var msg = 'XVP credentials required (user' + xvp_sep +
- 'target' + xvp_sep + 'password) -- got only ' + this._rfb_password;
- this._onPasswordRequired(this, msg);
- return false;
+ // choose the notunnel type
+ if (serverSupportedTunnelTypes[0]) {
+ if (serverSupportedTunnelTypes[0].vendor != clientSupportedTunnelTypes[0].vendor ||
+ serverSupportedTunnelTypes[0].signature != clientSupportedTunnelTypes[0].signature) {
+ return this._fail("Unsupported server",
+ "Client's tunnel type had the incorrect " +
+ "vendor or signature");
}
+ this._sock.send([0, 0, 0, 0]); // use NOTUNNEL
+ return false; // wait until we receive the sub auth count to continue
+ } else {
+ return this._fail("Unsupported server",
+ "Server wanted tunnels, but doesn't support " +
+ "the notunnel type");
+ }
+ },
- var xvp_auth_str = String.fromCharCode(xvp_auth[0].length) +
- String.fromCharCode(xvp_auth[1].length) +
- xvp_auth[0] +
- xvp_auth[1];
- this._sock.send_string(xvp_auth_str);
- this._rfb_password = xvp_auth.slice(2).join(xvp_sep);
- this._rfb_auth_scheme = 2;
- return this._negotiate_authentication();
- },
-
- _negotiate_std_vnc_auth: function () {
- if (this._rfb_password.length === 0) {
- this._onPasswordRequired(this);
- return false;
- }
+ _negotiate_tight_auth: function () {
+ if (!this._rfb_tightvnc) { // first pass, do the tunnel negotiation
+ if (this._sock.rQwait("num tunnels", 4)) { return false; }
+ var numTunnels = this._sock.rQshift32();
+ if (numTunnels > 0 && this._sock.rQwait("tunnel capabilities", 16 * numTunnels, 4)) { return false; }
- if (this._sock.rQwait("auth challenge", 16)) { return false; }
+ this._rfb_tightvnc = true;
- // TODO(directxman12): make genDES not require an Array
- var challenge = Array.prototype.slice.call(this._sock.rQshiftBytes(16));
- var response = RFB.genDES(this._rfb_password, challenge);
- this._sock.send(response);
- this._updateState("SecurityResult");
- return true;
- },
-
- _negotiate_tight_tunnels: function (numTunnels) {
- var clientSupportedTunnelTypes = {
- 0: { vendor: 'TGHT', signature: 'NOTUNNEL' }
- };
- var serverSupportedTunnelTypes = {};
- // receive tunnel capabilities
- for (var i = 0; i < numTunnels; i++) {
- var cap_code = this._sock.rQshift32();
- var cap_vendor = this._sock.rQshiftStr(4);
- var cap_signature = this._sock.rQshiftStr(8);
- serverSupportedTunnelTypes[cap_code] = { vendor: cap_vendor, signature: cap_signature };
+ if (numTunnels > 0) {
+ this._negotiate_tight_tunnels(numTunnels);
+ return false; // wait until we receive the sub auth to continue
}
+ }
- // choose the notunnel type
- if (serverSupportedTunnelTypes[0]) {
- if (serverSupportedTunnelTypes[0].vendor != clientSupportedTunnelTypes[0].vendor ||
- serverSupportedTunnelTypes[0].signature != clientSupportedTunnelTypes[0].signature) {
- return this._fail("Client's tunnel type had the incorrect vendor or signature");
- }
- this._sock.send([0, 0, 0, 0]); // use NOTUNNEL
- return false; // wait until we receive the sub auth count to continue
- } else {
- return this._fail("Server wanted tunnels, but doesn't support the notunnel type");
- }
- },
+ // second pass, do the sub-auth negotiation
+ if (this._sock.rQwait("sub auth count", 4)) { return false; }
+ var subAuthCount = this._sock.rQshift32();
+ if (subAuthCount === 0) { // empty sub-auth list received means 'no auth' subtype selected
+ this._rfb_init_state = 'SecurityResult';
+ return true;
+ }
- _negotiate_tight_auth: function () {
- if (!this._rfb_tightvnc) { // first pass, do the tunnel negotiation
- if (this._sock.rQwait("num tunnels", 4)) { return false; }
- var numTunnels = this._sock.rQshift32();
- if (numTunnels > 0 && this._sock.rQwait("tunnel capabilities", 16 * numTunnels, 4)) { return false; }
+ if (this._sock.rQwait("sub auth capabilities", 16 * subAuthCount, 4)) { return false; }
- this._rfb_tightvnc = true;
+ var clientSupportedTypes = {
+ 'STDVNOAUTH__': 1,
+ 'STDVVNCAUTH_': 2
+ };
- if (numTunnels > 0) {
- this._negotiate_tight_tunnels(numTunnels);
- return false; // wait until we receive the sub auth to continue
- }
- }
+ var serverSupportedTypes = [];
- // second pass, do the sub-auth negotiation
- if (this._sock.rQwait("sub auth count", 4)) { return false; }
- var subAuthCount = this._sock.rQshift32();
- if (subAuthCount === 0) { // empty sub-auth list received means 'no auth' subtype selected
- this._updateState('SecurityResult');
- return true;
+ for (var i = 0; i < subAuthCount; i++) {
+ var capNum = this._sock.rQshift32();
+ var capabilities = this._sock.rQshiftStr(12);
+ serverSupportedTypes.push(capabilities);
+ }
+
+ for (var authType in clientSupportedTypes) {
+ if (serverSupportedTypes.indexOf(authType) != -1) {
+ this._sock.send([0, 0, 0, clientSupportedTypes[authType]]);
+
+ switch (authType) {
+ case 'STDVNOAUTH__': // no auth
+ this._rfb_init_state = 'SecurityResult';
+ return true;
+ case 'STDVVNCAUTH_': // VNC auth
+ this._rfb_auth_scheme = 2;
+ return this._init_msg();
+ default:
+ return this._fail("Unsupported server",
+ "Unsupported tiny auth scheme: " +
+ authType);
+ }
}
+ }
- if (this._sock.rQwait("sub auth capabilities", 16 * subAuthCount, 4)) { return false; }
+ return this._fail("Unsupported server",
+ "No supported sub-auth types!");
+ },
+
+ _negotiate_authentication: function () {
+ switch (this._rfb_auth_scheme) {
+ case 0: // connection failed
+ if (this._sock.rQwait("auth reason", 4)) { return false; }
+ var strlen = this._sock.rQshift32();
+ var reason = this._sock.rQshiftStr(strlen);
+ return this._fail("Authentication failure", reason);
+
+ case 1: // no auth
+ if (this._rfb_version >= 3.8) {
+ this._rfb_init_state = 'SecurityResult';
+ return true;
+ }
+ this._rfb_init_state = 'ClientInitialisation';
+ return this._init_msg();
- var clientSupportedTypes = {
- 'STDVNOAUTH__': 1,
- 'STDVVNCAUTH_': 2
- };
+ case 22: // XVP auth
+ return this._negotiate_xvp_auth();
- var serverSupportedTypes = [];
+ case 2: // VNC authentication
+ return this._negotiate_std_vnc_auth();
- for (var i = 0; i < subAuthCount; i++) {
- var capNum = this._sock.rQshift32();
- var capabilities = this._sock.rQshiftStr(12);
- serverSupportedTypes.push(capabilities);
- }
+ case 16: // TightVNC Security Type
+ return this._negotiate_tight_auth();
- for (var authType in clientSupportedTypes) {
- if (serverSupportedTypes.indexOf(authType) != -1) {
- this._sock.send([0, 0, 0, clientSupportedTypes[authType]]);
-
- switch (authType) {
- case 'STDVNOAUTH__': // no auth
- this._updateState('SecurityResult');
- return true;
- case 'STDVVNCAUTH_': // VNC auth
- this._rfb_auth_scheme = 2;
- return this._init_msg();
- default:
- return this._fail("Unsupported tiny auth scheme: " + authType);
- }
+ default:
+ return this._fail("Unsupported server",
+ "Unsupported auth scheme: " +
+ this._rfb_auth_scheme);
+ }
+ },
+
+ _handle_security_result: function () {
+ if (this._sock.rQwait('VNC auth response ', 4)) { return false; }
+ switch (this._sock.rQshift32()) {
+ case 0: // OK
+ this._rfb_init_state = 'ClientInitialisation';
+ Log.Debug('Authentication OK');
+ return this._init_msg();
+ case 1: // failed
+ if (this._rfb_version >= 3.8) {
+ var length = this._sock.rQshift32();
+ if (this._sock.rQwait("SecurityResult reason", length, 8)) { return false; }
+ var reason = this._sock.rQshiftStr(length);
+ return this._fail("Authentication failure", reason);
+ } else {
+ return this._fail("Authentication failure");
}
- }
+ case 2:
+ return this._fail("Too many authentication attempts");
+ default:
+ return this._fail("Unsupported server",
+ "Unknown SecurityResult");
+ }
+ },
+
+ _negotiate_server_init: function () {
+ if (this._sock.rQwait("server initialization", 24)) { return false; }
+
+ /* Screen size */
+ this._fb_width = this._sock.rQshift16();
+ this._fb_height = this._sock.rQshift16();
+ this._destBuff = new Uint8Array(this._fb_width * this._fb_height * 4);
+
+ /* PIXEL_FORMAT */
+ var bpp = this._sock.rQshift8();
+ var depth = this._sock.rQshift8();
+ var big_endian = this._sock.rQshift8();
+ var true_color = this._sock.rQshift8();
+
+ var red_max = this._sock.rQshift16();
+ var green_max = this._sock.rQshift16();
+ var blue_max = this._sock.rQshift16();
+ var red_shift = this._sock.rQshift8();
+ var green_shift = this._sock.rQshift8();
+ var blue_shift = this._sock.rQshift8();
+ this._sock.rQskipBytes(3); // padding
+
+ // NB(directxman12): we don't want to call any callbacks or print messages until
+ // *after* we're past the point where we could backtrack
+
+ /* Connection name/title */
+ var name_length = this._sock.rQshift32();
+ if (this._sock.rQwait('server init name', name_length, 24)) { return false; }
+ this._fb_name = decodeUTF8(this._sock.rQshiftStr(name_length));
+
+ if (this._rfb_tightvnc) {
+ if (this._sock.rQwait('TightVNC extended server init header', 8, 24 + name_length)) { return false; }
+ // In TightVNC mode, ServerInit message is extended
+ var numServerMessages = this._sock.rQshift16();
+ var numClientMessages = this._sock.rQshift16();
+ var numEncodings = this._sock.rQshift16();
+ this._sock.rQskipBytes(2); // padding
+
+ var totalMessagesLength = (numServerMessages + numClientMessages + numEncodings) * 16;
+ if (this._sock.rQwait('TightVNC extended server init header', totalMessagesLength, 32 + name_length)) { return false; }
+
+ // we don't actually do anything with the capability information that TIGHT sends,
+ // so we just skip the all of this.
+
+ // TIGHT server message capabilities
+ this._sock.rQskipBytes(16 * numServerMessages);
+
+ // TIGHT client message capabilities
+ this._sock.rQskipBytes(16 * numClientMessages);
+
+ // TIGHT encoding capabilities
+ this._sock.rQskipBytes(16 * numEncodings);
+ }
- return this._fail("No supported sub-auth types!");
- },
+ // NB(directxman12): these are down here so that we don't run them multiple times
+ // if we backtrack
+ Log.Info("Screen: " + this._fb_width + "x" + this._fb_height +
+ ", bpp: " + bpp + ", depth: " + depth +
+ ", big_endian: " + big_endian +
+ ", true_color: " + true_color +
+ ", red_max: " + red_max +
+ ", green_max: " + green_max +
+ ", blue_max: " + blue_max +
+ ", red_shift: " + red_shift +
+ ", green_shift: " + green_shift +
+ ", blue_shift: " + blue_shift);
+
+ if (big_endian !== 0) {
+ Log.Warn("Server native endian is not little endian");
+ }
- _negotiate_authentication: function () {
- switch (this._rfb_auth_scheme) {
- case 0: // connection failed
- if (this._sock.rQwait("auth reason", 4)) { return false; }
- var strlen = this._sock.rQshift32();
- var reason = this._sock.rQshiftStr(strlen);
- return this._fail("Auth failure: " + reason);
+ if (red_shift !== 16) {
+ Log.Warn("Server native red-shift is not 16");
+ }
- case 1: // no auth
- if (this._rfb_version >= 3.8) {
- this._updateState('SecurityResult');
- return true;
- }
- this._updateState('ClientInitialisation', "No auth required");
- return this._init_msg();
+ if (blue_shift !== 0) {
+ Log.Warn("Server native blue-shift is not 0");
+ }
- case 22: // XVP auth
- return this._negotiate_xvp_auth();
+ // we're past the point where we could backtrack, so it's safe to call this
+ this._onDesktopName(this, this._fb_name);
- case 2: // VNC authentication
- return this._negotiate_std_vnc_auth();
+ this._display.resize(this._fb_width, this._fb_height);
+ this._onFBResize(this, this._fb_width, this._fb_height);
- case 16: // TightVNC Security Type
- return this._negotiate_tight_auth();
+ if (!this._view_only) { this._keyboard.grab(); }
+ if (!this._view_only) { this._mouse.grab(); }
- default:
- return this._fail("Unsupported auth scheme: " + this._rfb_auth_scheme);
- }
- },
-
- _handle_security_result: function () {
- if (this._sock.rQwait('VNC auth response ', 4)) { return false; }
- switch (this._sock.rQshift32()) {
- case 0: // OK
- this._updateState('ClientInitialisation', 'Authentication OK');
- return this._init_msg();
- case 1: // failed
- if (this._rfb_version >= 3.8) {
- var length = this._sock.rQshift32();
- if (this._sock.rQwait("SecurityResult reason", length, 8)) { return false; }
- var reason = this._sock.rQshiftStr(length);
- return this._fail(reason);
- } else {
- return this._fail("Authentication failure");
- }
- return false;
- case 2:
- return this._fail("Too many auth attempts");
- default:
- return this._fail("Unknown SecurityResult");
- }
- },
-
- _negotiate_server_init: function () {
- if (this._sock.rQwait("server initialization", 24)) { return false; }
-
- /* Screen size */
- this._fb_width = this._sock.rQshift16();
- this._fb_height = this._sock.rQshift16();
- this._destBuff = new Uint8Array(this._fb_width * this._fb_height * 4);
-
- /* PIXEL_FORMAT */
- var bpp = this._sock.rQshift8();
- var depth = this._sock.rQshift8();
- var big_endian = this._sock.rQshift8();
- var true_color = this._sock.rQshift8();
-
- var red_max = this._sock.rQshift16();
- var green_max = this._sock.rQshift16();
- var blue_max = this._sock.rQshift16();
- var red_shift = this._sock.rQshift8();
- var green_shift = this._sock.rQshift8();
- var blue_shift = this._sock.rQshift8();
- this._sock.rQskipBytes(3); // padding
-
- // NB(directxman12): we don't want to call any callbacks or print messages until
- // *after* we're past the point where we could backtrack
-
- /* Connection name/title */
- var name_length = this._sock.rQshift32();
- if (this._sock.rQwait('server init name', name_length, 24)) { return false; }
- this._fb_name = Util.decodeUTF8(this._sock.rQshiftStr(name_length));
-
- if (this._rfb_tightvnc) {
- if (this._sock.rQwait('TightVNC extended server init header', 8, 24 + name_length)) { return false; }
- // In TightVNC mode, ServerInit message is extended
- var numServerMessages = this._sock.rQshift16();
- var numClientMessages = this._sock.rQshift16();
- var numEncodings = this._sock.rQshift16();
- this._sock.rQskipBytes(2); // padding
-
- var totalMessagesLength = (numServerMessages + numClientMessages + numEncodings) * 16;
- if (this._sock.rQwait('TightVNC extended server init header', totalMessagesLength, 32 + name_length)) { return false; }
-
- // we don't actually do anything with the capability information that TIGHT sends,
- // so we just skip the all of this.
-
- // TIGHT server message capabilities
- this._sock.rQskipBytes(16 * numServerMessages);
-
- // TIGHT client message capabilities
- this._sock.rQskipBytes(16 * numClientMessages);
-
- // TIGHT encoding capabilities
- this._sock.rQskipBytes(16 * numEncodings);
- }
+ RFB.messages.pixelFormat(this._sock, 4, 3, true);
+ RFB.messages.clientEncodings(this._sock, this._encodings, this._local_cursor);
+ RFB.messages.fbUpdateRequest(this._sock, false, 0, 0, this._fb_width, this._fb_height);
- // NB(directxman12): these are down here so that we don't run them multiple times
- // if we backtrack
- Util.Info("Screen: " + this._fb_width + "x" + this._fb_height +
- ", bpp: " + bpp + ", depth: " + depth +
- ", big_endian: " + big_endian +
- ", true_color: " + true_color +
- ", red_max: " + red_max +
- ", green_max: " + green_max +
- ", blue_max: " + blue_max +
- ", red_shift: " + red_shift +
- ", green_shift: " + green_shift +
- ", blue_shift: " + blue_shift);
-
- if (big_endian !== 0) {
- Util.Warn("Server native endian is not little endian");
- }
+ this._timing.fbu_rt_start = (new Date()).getTime();
+ this._timing.pixels = 0;
- if (red_shift !== 16) {
- Util.Warn("Server native red-shift is not 16");
- }
+ this._updateConnectionState('connected');
+ return true;
+ },
- if (blue_shift !== 0) {
- Util.Warn("Server native blue-shift is not 0");
- }
+ /* RFB protocol initialization states:
+ * ProtocolVersion
+ * Security
+ * Authentication
+ * SecurityResult
+ * ClientInitialization - not triggered by server message
+ * ServerInitialization
+ */
+ _init_msg: function () {
+ switch (this._rfb_init_state) {
+ case 'ProtocolVersion':
+ return this._negotiate_protocol_version();
- // we're past the point where we could backtrack, so it's safe to call this
- this._onDesktopName(this, this._fb_name);
+ case 'Security':
+ return this._negotiate_security();
- if (this._true_color && this._fb_name === "Intel(r) AMT KVM") {
- Util.Warn("Intel AMT KVM only supports 8/16 bit depths. Disabling true color");
- this._true_color = false;
- }
+ case 'Authentication':
+ return this._negotiate_authentication();
- this._display.set_true_color(this._true_color);
- this._display.resize(this._fb_width, this._fb_height);
- this._onFBResize(this, this._fb_width, this._fb_height);
- this._keyboard.grab();
- this._mouse.grab();
+ case 'SecurityResult':
+ return this._handle_security_result();
- if (this._true_color) {
- this._fb_Bpp = 4;
- this._fb_depth = 3;
- } else {
- this._fb_Bpp = 1;
- this._fb_depth = 1;
- }
+ case 'ClientInitialisation':
+ this._sock.send([this._shared ? 1 : 0]); // ClientInitialisation
+ this._rfb_init_state = 'ServerInitialisation';
+ return true;
- RFB.messages.pixelFormat(this._sock, this._fb_Bpp, this._fb_depth, this._true_color);
- RFB.messages.clientEncodings(this._sock, this._encodings, this._local_cursor, this._true_color);
- RFB.messages.fbUpdateRequests(this._sock, false, this._display.getCleanDirtyReset(), this._fb_width, this._fb_height);
+ case 'ServerInitialisation':
+ return this._negotiate_server_init();
- this._timing.fbu_rt_start = (new Date()).getTime();
- this._timing.pixels = 0;
+ default:
+ return this._fail("Internal error", "Unknown init state: " +
+ this._rfb_init_state);
+ }
+ },
- if (this._encrypt) {
- this._updateState('normal', 'Connected (encrypted) to: ' + this._fb_name);
- } else {
- this._updateState('normal', 'Connected (unencrypted) to: ' + this._fb_name);
- }
- return true;
- },
+ _handle_set_colour_map_msg: function () {
+ Log.Debug("SetColorMapEntries");
- _init_msg: function () {
- switch (this._rfb_state) {
- case 'ProtocolVersion':
- return this._negotiate_protocol_version();
+ return this._fail("Protocol error", "Unexpected SetColorMapEntries message");
+ },
- case 'Security':
- return this._negotiate_security();
+ _handle_server_cut_text: function () {
+ Log.Debug("ServerCutText");
- case 'Authentication':
- return this._negotiate_authentication();
+ if (this._sock.rQwait("ServerCutText header", 7, 1)) { return false; }
+ this._sock.rQskipBytes(3); // Padding
+ var length = this._sock.rQshift32();
+ if (this._sock.rQwait("ServerCutText", length, 8)) { return false; }
- case 'SecurityResult':
- return this._handle_security_result();
+ var text = this._sock.rQshiftStr(length);
- case 'ClientInitialisation':
- this._sock.send([this._shared ? 1 : 0]); // ClientInitialisation
- this._updateState('ServerInitialisation', "Authentication OK");
- return true;
+ if (this._view_only) { return true; }
- case 'ServerInitialisation':
- return this._negotiate_server_init();
+ this._onClipboard(this, text);
- default:
- return this._fail("Unknown state: " + this._rfb_state);
- }
- },
+ return true;
+ },
- _handle_set_colour_map_msg: function () {
- Util.Debug("SetColorMapEntries");
- this._sock.rQskip8(); // Padding
+ _handle_server_fence_msg: function() {
+ if (this._sock.rQwait("ServerFence header", 8, 1)) { return false; }
+ this._sock.rQskipBytes(3); // Padding
+ var flags = this._sock.rQshift32();
+ var length = this._sock.rQshift8();
- var first_colour = this._sock.rQshift16();
- var num_colours = this._sock.rQshift16();
- if (this._sock.rQwait('SetColorMapEntries', num_colours * 6, 6)) { return false; }
+ if (this._sock.rQwait("ServerFence payload", length, 9)) { return false; }
- for (var c = 0; c < num_colours; c++) {
- var red = parseInt(this._sock.rQshift16() / 256, 10);
- var green = parseInt(this._sock.rQshift16() / 256, 10);
- var blue = parseInt(this._sock.rQshift16() / 256, 10);
- this._display.set_colourMap([blue, green, red], first_colour + c);
- }
- Util.Debug("colourMap: " + this._display.get_colourMap());
- Util.Info("Registered " + num_colours + " colourMap entries");
+ if (length > 64) {
+ Log.Warn("Bad payload length (" + length + ") in fence response");
+ length = 64;
+ }
- return true;
- },
+ var payload = this._sock.rQshiftStr(length);
+
+ this._supportsFence = true;
- _handle_server_cut_text: function () {
- Util.Debug("ServerCutText");
- if (this._sock.rQwait("ServerCutText header", 7, 1)) { return false; }
- this._sock.rQskipBytes(3); // Padding
- var length = this._sock.rQshift32();
- if (this._sock.rQwait("ServerCutText", length, 8)) { return false; }
+ /*
+ * Fence flags
+ *
+ * (1<<0) - BlockBefore
+ * (1<<1) - BlockAfter
+ * (1<<2) - SyncNext
+ * (1<<31) - Request
+ */
- var text = this._sock.rQshiftStr(length);
- this._onClipboard(this, text);
+ if (!(flags & (1<<31))) {
+ return this._fail("Internal error",
+ "Unexpected fence response");
+ }
- return true;
- },
+ // Filter out unsupported flags
+ // FIXME: support syncNext
+ flags &= (1<<0) | (1<<1);
+
+ // BlockBefore and BlockAfter are automatically handled by
+ // the fact that we process each incoming message
+ // synchronuosly.
+ RFB.messages.clientFence(this._sock, flags, payload);
+
+ return true;
+ },
+
+ _handle_xvp_msg: function () {
+ if (this._sock.rQwait("XVP version and message", 3, 1)) { return false; }
+ this._sock.rQskip8(); // Padding
+ var xvp_ver = this._sock.rQshift8();
+ var xvp_msg = this._sock.rQshift8();
+
+ switch (xvp_msg) {
+ case 0: // XVP_FAIL
+ Log.Error("Operation Failed");
+ this._notification("XVP Operation Failed", 'error');
+ break;
+ case 1: // XVP_INIT
+ this._rfb_xvp_ver = xvp_ver;
+ Log.Info("XVP extensions enabled (version " + this._rfb_xvp_ver + ")");
+ this._onXvpInit(this._rfb_xvp_ver);
+ break;
+ default:
+ this._fail("Unexpected server message",
+ "Illegal server XVP message " + xvp_msg);
+ break;
+ }
- _handle_server_fence_msg: function() {
- if (this._sock.rQwait("ServerFence header", 8, 1)) { return false; }
- this._sock.rQskipBytes(3); // Padding
- var flags = this._sock.rQshift32();
- var length = this._sock.rQshift8();
+ return true;
+ },
- if (this._sock.rQwait("ServerFence payload", length, 9)) { return false; }
+ _normal_msg: function () {
+ var msg_type;
- if (length > 64) {
- Util.Warn("Bad payload length (" + length + ") in fence response");
- length = 64;
- }
+ if (this._FBU.rects > 0) {
+ msg_type = 0;
+ } else {
+ msg_type = this._sock.rQshift8();
+ }
- var payload = this._sock.rQshiftStr(length);
+ switch (msg_type) {
+ case 0: // FramebufferUpdate
+ var ret = this._framebufferUpdate();
+ if (ret && !this._enabledContinuousUpdates) {
+ RFB.messages.fbUpdateRequest(this._sock, true, 0, 0,
+ this._fb_width, this._fb_height);
+ }
+ return ret;
- this._supportsFence = true;
+ case 1: // SetColorMapEntries
+ return this._handle_set_colour_map_msg();
- /*
- * Fence flags
- *
- * (1<<0) - BlockBefore
- * (1<<1) - BlockAfter
- * (1<<2) - SyncNext
- * (1<<31) - Request
- */
+ case 2: // Bell
+ Log.Debug("Bell");
+ this._onBell(this);
+ return true;
- if (!(flags & (1<<31))) {
- return this._fail("Unexpected fence response");
- }
+ case 3: // ServerCutText
+ return this._handle_server_cut_text();
+
+ case 150: // EndOfContinuousUpdates
+ var first = !(this._supportsContinuousUpdates);
+ this._supportsContinuousUpdates = true;
+ this._enabledContinuousUpdates = false;
+ if (first) {
+ this._enabledContinuousUpdates = true;
+ this._updateContinuousUpdates();
+ Log.Info("Enabling continuous updates.");
+ } else {
+ // FIXME: We need to send a framebufferupdaterequest here
+ // if we add support for turning off continuous updates
+ }
+ return true;
- // Filter out unsupported flags
- // FIXME: support syncNext
- flags &= (1<<0) | (1<<1);
+ case 248: // ServerFence
+ return this._handle_server_fence_msg();
- // BlockBefore and BlockAfter are automatically handled by
- // the fact that we process each incoming message
- // synchronuosly.
- RFB.messages.clientFence(this._sock, flags, payload);
+ case 250: // XVP
+ return this._handle_xvp_msg();
- return true;
- },
+ default:
+ this._fail("Unexpected server message", "Type:" + msg_type);
+ Log.Debug("sock.rQslice(0, 30): " + this._sock.rQslice(0, 30));
+ return true;
+ }
+ },
+
+ _onFlush: function() {
+ this._flushing = false;
+ // Resume processing
+ if (this._sock.rQlen() > 0) {
+ this._handle_message();
+ }
+ },
- _handle_xvp_msg: function () {
- if (this._sock.rQwait("XVP version and message", 3, 1)) { return false; }
+ _framebufferUpdate: function () {
+ var ret = true;
+ var now;
+
+ if (this._FBU.rects === 0) {
+ if (this._sock.rQwait("FBU header", 3, 1)) { return false; }
this._sock.rQskip8(); // Padding
- var xvp_ver = this._sock.rQshift8();
- var xvp_msg = this._sock.rQshift8();
-
- switch (xvp_msg) {
- case 0: // XVP_FAIL
- Util.Error("Operation Failed");
- this._notification("XVP Operation Failed", 'error');
- break;
- case 1: // XVP_INIT
- this._rfb_xvp_ver = xvp_ver;
- Util.Info("XVP extensions enabled (version " + this._rfb_xvp_ver + ")");
- this._onXvpInit(this._rfb_xvp_ver);
- break;
- default:
- this._fail("Disconnected: illegal server XVP message " + xvp_msg);
- break;
+ this._FBU.rects = this._sock.rQshift16();
+ this._FBU.bytes = 0;
+ this._timing.cur_fbu = 0;
+ if (this._timing.fbu_rt_start > 0) {
+ now = (new Date()).getTime();
+ Log.Info("First FBU latency: " + (now - this._timing.fbu_rt_start));
}
- return true;
- },
+ // Make sure the previous frame is fully rendered first
+ // to avoid building up an excessive queue
+ if (this._display.pending()) {
+ this._flushing = true;
+ this._display.flush();
+ return false;
+ }
+ }
- _normal_msg: function () {
- var msg_type;
+ while (this._FBU.rects > 0) {
+ if (this._rfb_connection_state !== 'connected') { return false; }
- if (this._FBU.rects > 0) {
- msg_type = 0;
- } else {
- msg_type = this._sock.rQshift8();
- }
+ if (this._sock.rQwait("FBU", this._FBU.bytes)) { return false; }
+ if (this._FBU.bytes === 0) {
+ if (this._sock.rQwait("rect header", 12)) { return false; }
+ /* New FramebufferUpdate */
- switch (msg_type) {
- case 0: // FramebufferUpdate
- var ret = this._framebufferUpdate();
- if (ret) {
- RFB.messages.fbUpdateRequests(this._sock,
- this._enabledContinuousUpdates,
- this._display.getCleanDirtyReset(),
- this._fb_width, this._fb_height);
- }
- return ret;
+ var hdr = this._sock.rQshiftBytes(12);
+ this._FBU.x = (hdr[0] << 8) + hdr[1];
+ this._FBU.y = (hdr[2] << 8) + hdr[3];
+ this._FBU.width = (hdr[4] << 8) + hdr[5];
+ this._FBU.height = (hdr[6] << 8) + hdr[7];
+ this._FBU.encoding = parseInt((hdr[8] << 24) + (hdr[9] << 16) +
+ (hdr[10] << 8) + hdr[11], 10);
- case 1: // SetColorMapEntries
- return this._handle_set_colour_map_msg();
+ this._onFBUReceive(this,
+ {'x': this._FBU.x, 'y': this._FBU.y,
+ 'width': this._FBU.width, 'height': this._FBU.height,
+ 'encoding': this._FBU.encoding,
+ 'encodingName': this._encNames[this._FBU.encoding]});
- case 2: // Bell
- Util.Debug("Bell");
- this._onBell(this);
- return true;
+ if (!this._encNames[this._FBU.encoding]) {
+ this._fail("Unexpected server message",
+ "Unsupported encoding " +
+ this._FBU.encoding);
+ return false;
+ }
+ }
- case 3: // ServerCutText
- return this._handle_server_cut_text();
-
- case 150: // EndOfContinuousUpdates
- var first = !(this._supportsContinuousUpdates);
- this._supportsContinuousUpdates = true;
- this._enabledContinuousUpdates = false;
- if (first) {
- this._enabledContinuousUpdates = true;
- this._updateContinuousUpdates();
- Util.Info("Enabling continuous updates.");
- } else {
- // FIXME: We need to send a framebufferupdaterequest here
- // if we add support for turning off continuous updates
- }
- return true;
+ this._timing.last_fbu = (new Date()).getTime();
- case 248: // ServerFence
- return this._handle_server_fence_msg();
+ ret = this._encHandlers[this._FBU.encoding]();
- case 250: // XVP
- return this._handle_xvp_msg();
+ now = (new Date()).getTime();
+ this._timing.cur_fbu += (now - this._timing.last_fbu);
- default:
- this._fail("Disconnected: illegal server message type " + msg_type);
- Util.Debug("sock.rQslice(0, 30): " + this._sock.rQslice(0, 30));
- return true;
+ if (ret) {
+ this._encStats[this._FBU.encoding][0]++;
+ this._encStats[this._FBU.encoding][1]++;
+ this._timing.pixels += this._FBU.width * this._FBU.height;
}
- },
-
- _framebufferUpdate: function () {
- var ret = true;
- var now;
-
- if (this._FBU.rects === 0) {
- if (this._sock.rQwait("FBU header", 3, 1)) { return false; }
- this._sock.rQskip8(); // Padding
- this._FBU.rects = this._sock.rQshift16();
- this._FBU.bytes = 0;
- this._timing.cur_fbu = 0;
+
+ if (this._timing.pixels >= (this._fb_width * this._fb_height)) {
+ if ((this._FBU.width === this._fb_width && this._FBU.height === this._fb_height) ||
+ this._timing.fbu_rt_start > 0) {
+ this._timing.full_fbu_total += this._timing.cur_fbu;
+ this._timing.full_fbu_cnt++;
+ Log.Info("Timing of full FBU, curr: " +
+ this._timing.cur_fbu + ", total: " +
+ this._timing.full_fbu_total + ", cnt: " +
+ this._timing.full_fbu_cnt + ", avg: " +
+ (this._timing.full_fbu_total / this._timing.full_fbu_cnt));
+ }
+
if (this._timing.fbu_rt_start > 0) {
- now = (new Date()).getTime();
- Util.Info("First FBU latency: " + (now - this._timing.fbu_rt_start));
+ var fbu_rt_diff = now - this._timing.fbu_rt_start;
+ this._timing.fbu_rt_total += fbu_rt_diff;
+ this._timing.fbu_rt_cnt++;
+ Log.Info("full FBU round-trip, cur: " +
+ fbu_rt_diff + ", total: " +
+ this._timing.fbu_rt_total + ", cnt: " +
+ this._timing.fbu_rt_cnt + ", avg: " +
+ (this._timing.fbu_rt_total / this._timing.fbu_rt_cnt));
+ this._timing.fbu_rt_start = 0;
}
}
- while (this._FBU.rects > 0) {
- if (this._rfb_state !== "normal") { return false; }
-
- if (this._sock.rQwait("FBU", this._FBU.bytes)) { return false; }
- if (this._FBU.bytes === 0) {
- if (this._sock.rQwait("rect header", 12)) { return false; }
- /* New FramebufferUpdate */
-
- var hdr = this._sock.rQshiftBytes(12);
- this._FBU.x = (hdr[0] << 8) + hdr[1];
- this._FBU.y = (hdr[2] << 8) + hdr[3];
- this._FBU.width = (hdr[4] << 8) + hdr[5];
- this._FBU.height = (hdr[6] << 8) + hdr[7];
- this._FBU.encoding = parseInt((hdr[8] << 24) + (hdr[9] << 16) +
- (hdr[10] << 8) + hdr[11], 10);
-
- this._onFBUReceive(this,
- {'x': this._FBU.x, 'y': this._FBU.y,
- 'width': this._FBU.width, 'height': this._FBU.height,
- 'encoding': this._FBU.encoding,
- 'encodingName': this._encNames[this._FBU.encoding]});
-
- if (!this._encNames[this._FBU.encoding]) {
- this._fail("Disconnected: unsupported encoding " +
- this._FBU.encoding);
- return false;
- }
- }
+ if (!ret) { return ret; } // need more data
+ }
- this._timing.last_fbu = (new Date()).getTime();
+ this._display.flip();
- ret = this._encHandlers[this._FBU.encoding]();
+ this._onFBUComplete(this,
+ {'x': this._FBU.x, 'y': this._FBU.y,
+ 'width': this._FBU.width, 'height': this._FBU.height,
+ 'encoding': this._FBU.encoding,
+ 'encodingName': this._encNames[this._FBU.encoding]});
- now = (new Date()).getTime();
- this._timing.cur_fbu += (now - this._timing.last_fbu);
+ return true; // We finished this FBU
+ },
- if (ret) {
- this._encStats[this._FBU.encoding][0]++;
- this._encStats[this._FBU.encoding][1]++;
- this._timing.pixels += this._FBU.width * this._FBU.height;
- }
+ _updateContinuousUpdates: function() {
+ if (!this._enabledContinuousUpdates) { return; }
- if (this._timing.pixels >= (this._fb_width * this._fb_height)) {
- if ((this._FBU.width === this._fb_width && this._FBU.height === this._fb_height) ||
- this._timing.fbu_rt_start > 0) {
- this._timing.full_fbu_total += this._timing.cur_fbu;
- this._timing.full_fbu_cnt++;
- Util.Info("Timing of full FBU, curr: " +
- this._timing.cur_fbu + ", total: " +
- this._timing.full_fbu_total + ", cnt: " +
- this._timing.full_fbu_cnt + ", avg: " +
- (this._timing.full_fbu_total / this._timing.full_fbu_cnt));
- }
+ RFB.messages.enableContinuousUpdates(this._sock, true, 0, 0,
+ this._fb_width, this._fb_height);
+ }
+};
- if (this._timing.fbu_rt_start > 0) {
- var fbu_rt_diff = now - this._timing.fbu_rt_start;
- this._timing.fbu_rt_total += fbu_rt_diff;
- this._timing.fbu_rt_cnt++;
- Util.Info("full FBU round-trip, cur: " +
- fbu_rt_diff + ", total: " +
- this._timing.fbu_rt_total + ", cnt: " +
- this._timing.fbu_rt_cnt + ", avg: " +
- (this._timing.fbu_rt_total / this._timing.fbu_rt_cnt));
- this._timing.fbu_rt_start = 0;
- }
- }
+make_properties(RFB, [
+ ['target', 'wo', 'dom'], // VNC display rendering Canvas object
+ ['focusContainer', 'wo', 'dom'], // DOM element that captures keyboard input
+ ['encrypt', 'rw', 'bool'], // Use TLS/SSL/wss encryption
+ ['local_cursor', 'rw', 'bool'], // Request locally rendered cursor
+ ['shared', 'rw', 'bool'], // Request shared mode
+ ['view_only', 'rw', 'bool'], // Disable client mouse/keyboard
+ ['xvp_password_sep', 'rw', 'str'], // Separator for XVP password fields
+ ['disconnectTimeout', 'rw', 'int'], // Time (s) to wait for disconnection
+ ['wsProtocols', 'rw', 'arr'], // Protocols to use in the WebSocket connection
+ ['repeaterID', 'rw', 'str'], // [UltraVNC] RepeaterID to connect to
+ ['viewportDrag', 'rw', 'bool'], // Move the viewport on mouse drags
+
+ // Callback functions
+ ['onUpdateState', 'rw', 'func'], // onUpdateState(rfb, state, oldstate): connection state change
+ ['onNotification', 'rw', 'func'], // onNotification(rfb, msg, level, options): notification for the UI
+ ['onDisconnected', 'rw', 'func'], // onDisconnected(rfb, reason): disconnection finished
+ ['onPasswordRequired', 'rw', 'func'], // onPasswordRequired(rfb, msg): VNC password is required
+ ['onClipboard', 'rw', 'func'], // onClipboard(rfb, text): RFB clipboard contents received
+ ['onBell', 'rw', 'func'], // onBell(rfb): RFB Bell message received
+ ['onFBUReceive', 'rw', 'func'], // onFBUReceive(rfb, fbu): RFB FBU received but not yet processed
+ ['onFBUComplete', 'rw', 'func'], // onFBUComplete(rfb, fbu): RFB FBU received and processed
+ ['onFBResize', 'rw', 'func'], // onFBResize(rfb, width, height): frame buffer resized
+ ['onDesktopName', 'rw', 'func'], // onDesktopName(rfb, name): desktop name received
+ ['onXvpInit', 'rw', 'func'] // onXvpInit(version): XVP extensions active for this connection
+]);
+
+RFB.prototype.set_local_cursor = function (cursor) {
+ if (!cursor || (cursor in {'0': 1, 'no': 1, 'false': 1})) {
+ this._local_cursor = false;
+ this._display.disableLocalCursor(); //Only show server-side cursor
+ } else {
+ if (this._display.get_cursor_uri()) {
+ this._local_cursor = true;
+ } else {
+ Log.Warn("Browser does not support local cursor");
+ this._display.disableLocalCursor();
+ }
+ }
- if (!ret) { return ret; } // need more data
- }
+ // Need to send an updated list of encodings if we are connected
+ if (this._rfb_connection_state === "connected") {
+ RFB.messages.clientEncodings(this._sock, this._encodings, cursor);
+ }
+};
- this._onFBUComplete(this,
- {'x': this._FBU.x, 'y': this._FBU.y,
- 'width': this._FBU.width, 'height': this._FBU.height,
- 'encoding': this._FBU.encoding,
- 'encodingName': this._encNames[this._FBU.encoding]});
+RFB.prototype.set_view_only = function (view_only) {
+ this._view_only = view_only;
+
+ if (this._rfb_connection_state === "connecting" ||
+ this._rfb_connection_state === "connected") {
+ if (view_only) {
+ this._keyboard.ungrab();
+ this._mouse.ungrab();
+ } else {
+ this._keyboard.grab();
+ this._mouse.grab();
+ }
+ }
+};
- return true; // We finished this FBU
- },
+RFB.prototype.get_display = function () { return this._display; };
+RFB.prototype.get_keyboard = function () { return this._keyboard; };
+RFB.prototype.get_mouse = function () { return this._mouse; };
- _updateContinuousUpdates: function() {
- if (!this._enabledContinuousUpdates) { return; }
+// Class Methods
+RFB.messages = {
+ keyEvent: function (sock, keysym, down) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- RFB.messages.enableContinuousUpdates(this._sock, true, 0, 0,
- this._fb_width, this._fb_height);
- }
- };
+ buff[offset] = 4; // msg-type
+ buff[offset + 1] = down;
- Util.make_properties(RFB, [
- ['target', 'wo', 'dom'], // VNC display rendering Canvas object
- ['focusContainer', 'wo', 'dom'], // DOM element that captures keyboard input
- ['encrypt', 'rw', 'bool'], // Use TLS/SSL/wss encryption
- ['true_color', 'rw', 'bool'], // Request true color pixel data
- ['local_cursor', 'rw', 'bool'], // Request locally rendered cursor
- ['shared', 'rw', 'bool'], // Request shared mode
- ['view_only', 'rw', 'bool'], // Disable client mouse/keyboard
- ['xvp_password_sep', 'rw', 'str'], // Separator for XVP password fields
- ['disconnectTimeout', 'rw', 'int'], // Time (s) to wait for disconnection
- ['wsProtocols', 'rw', 'arr'], // Protocols to use in the WebSocket connection
- ['repeaterID', 'rw', 'str'], // [UltraVNC] RepeaterID to connect to
- ['viewportDrag', 'rw', 'bool'], // Move the viewport on mouse drags
+ buff[offset + 2] = 0;
+ buff[offset + 3] = 0;
- // Callback functions
- ['onUpdateState', 'rw', 'func'], // onUpdateState(rfb, state, oldstate, statusMsg): RFB state update/change
- ['onNotification', 'rw', 'func'], // onNotification(rfb, msg, level, options): notification for the UI
- ['onPasswordRequired', 'rw', 'func'], // onPasswordRequired(rfb, msg): VNC password is required
- ['onClipboard', 'rw', 'func'], // onClipboard(rfb, text): RFB clipboard contents received
- ['onBell', 'rw', 'func'], // onBell(rfb): RFB Bell message received
- ['onFBUReceive', 'rw', 'func'], // onFBUReceive(rfb, fbu): RFB FBU received but not yet processed
- ['onFBUComplete', 'rw', 'func'], // onFBUComplete(rfb, fbu): RFB FBU received and processed
- ['onFBResize', 'rw', 'func'], // onFBResize(rfb, width, height): frame buffer resized
- ['onDesktopName', 'rw', 'func'], // onDesktopName(rfb, name): desktop name received
- ['onXvpInit', 'rw', 'func'] // onXvpInit(version): XVP extensions active for this connection
- ]);
-
- RFB.prototype.set_local_cursor = function (cursor) {
- if (!cursor || (cursor in {'0': 1, 'no': 1, 'false': 1})) {
- this._local_cursor = false;
- this._display.disableLocalCursor(); //Only show server-side cursor
- } else {
- if (this._display.get_cursor_uri()) {
- this._local_cursor = true;
- } else {
- Util.Warn("Browser does not support local cursor");
- this._display.disableLocalCursor();
+ buff[offset + 4] = (keysym >> 24);
+ buff[offset + 5] = (keysym >> 16);
+ buff[offset + 6] = (keysym >> 8);
+ buff[offset + 7] = keysym;
+
+ sock._sQlen += 8;
+ sock.flush();
+ },
+
+ QEMUExtendedKeyEvent: function (sock, keysym, down, keycode) {
+ function getRFBkeycode(xt_scancode) {
+ var upperByte = (keycode >> 8);
+ var lowerByte = (keycode & 0x00ff);
+ if (upperByte === 0xe0 && lowerByte < 0x7f) {
+ lowerByte = lowerByte | 0x80;
+ return lowerByte;
}
+ return xt_scancode;
}
- };
- RFB.prototype.get_display = function () { return this._display; };
- RFB.prototype.get_keyboard = function () { return this._keyboard; };
- RFB.prototype.get_mouse = function () { return this._mouse; };
-
- // Class Methods
- RFB.messages = {
- keyEvent: function (sock, keysym, down) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
-
- buff[offset] = 4; // msg-type
- buff[offset + 1] = down;
-
- buff[offset + 2] = 0;
- buff[offset + 3] = 0;
-
- buff[offset + 4] = (keysym >> 24);
- buff[offset + 5] = (keysym >> 16);
- buff[offset + 6] = (keysym >> 8);
- buff[offset + 7] = keysym;
-
- sock._sQlen += 8;
- sock.flush();
- },
-
- QEMUExtendedKeyEvent: function (sock, keysym, down, keycode) {
- function getRFBkeycode(xt_scancode) {
- var upperByte = (keycode >> 8);
- var lowerByte = (keycode & 0x00ff);
- if (upperByte === 0xe0 && lowerByte < 0x7f) {
- lowerByte = lowerByte | 0x80;
- return lowerByte;
- }
- return xt_scancode;
- }
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 255; // msg-type
+ buff[offset + 1] = 0; // sub msg-type
- buff[offset] = 255; // msg-type
- buff[offset + 1] = 0; // sub msg-type
+ buff[offset + 2] = (down >> 8);
+ buff[offset + 3] = down;
- buff[offset + 2] = (down >> 8);
- buff[offset + 3] = down;
+ buff[offset + 4] = (keysym >> 24);
+ buff[offset + 5] = (keysym >> 16);
+ buff[offset + 6] = (keysym >> 8);
+ buff[offset + 7] = keysym;
- buff[offset + 4] = (keysym >> 24);
- buff[offset + 5] = (keysym >> 16);
- buff[offset + 6] = (keysym >> 8);
- buff[offset + 7] = keysym;
+ var RFBkeycode = getRFBkeycode(keycode);
- var RFBkeycode = getRFBkeycode(keycode);
+ buff[offset + 8] = (RFBkeycode >> 24);
+ buff[offset + 9] = (RFBkeycode >> 16);
+ buff[offset + 10] = (RFBkeycode >> 8);
+ buff[offset + 11] = RFBkeycode;
- buff[offset + 8] = (RFBkeycode >> 24);
- buff[offset + 9] = (RFBkeycode >> 16);
- buff[offset + 10] = (RFBkeycode >> 8);
- buff[offset + 11] = RFBkeycode;
+ sock._sQlen += 12;
+ sock.flush();
+ },
- sock._sQlen += 12;
- sock.flush();
- },
+ pointerEvent: function (sock, x, y, mask) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- pointerEvent: function (sock, x, y, mask) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 5; // msg-type
- buff[offset] = 5; // msg-type
+ buff[offset + 1] = mask;
- buff[offset + 1] = mask;
+ buff[offset + 2] = x >> 8;
+ buff[offset + 3] = x;
- buff[offset + 2] = x >> 8;
- buff[offset + 3] = x;
+ buff[offset + 4] = y >> 8;
+ buff[offset + 5] = y;
- buff[offset + 4] = y >> 8;
- buff[offset + 5] = y;
+ sock._sQlen += 6;
+ sock.flush();
+ },
- sock._sQlen += 6;
- sock.flush();
- },
+ // TODO(directxman12): make this unicode compatible?
+ clientCutText: function (sock, text) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- // TODO(directxman12): make this unicode compatible?
- clientCutText: function (sock, text) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 6; // msg-type
- buff[offset] = 6; // msg-type
+ buff[offset + 1] = 0; // padding
+ buff[offset + 2] = 0; // padding
+ buff[offset + 3] = 0; // padding
- buff[offset + 1] = 0; // padding
- buff[offset + 2] = 0; // padding
- buff[offset + 3] = 0; // padding
+ var n = text.length;
- var n = text.length;
+ buff[offset + 4] = n >> 24;
+ buff[offset + 5] = n >> 16;
+ buff[offset + 6] = n >> 8;
+ buff[offset + 7] = n;
- buff[offset + 4] = n >> 24;
- buff[offset + 5] = n >> 16;
- buff[offset + 6] = n >> 8;
- buff[offset + 7] = n;
+ for (var i = 0; i < n; i++) {
+ buff[offset + 8 + i] = text.charCodeAt(i);
+ }
- for (var i = 0; i < n; i++) {
- buff[offset + 8 + i] = text.charCodeAt(i);
- }
+ sock._sQlen += 8 + n;
+ sock.flush();
+ },
+
+ setDesktopSize: function (sock, width, height, id, flags) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
+
+ buff[offset] = 251; // msg-type
+ buff[offset + 1] = 0; // padding
+ buff[offset + 2] = width >> 8; // width
+ buff[offset + 3] = width;
+ buff[offset + 4] = height >> 8; // height
+ buff[offset + 5] = height;
+
+ buff[offset + 6] = 1; // number-of-screens
+ buff[offset + 7] = 0; // padding
+
+ // screen array
+ buff[offset + 8] = id >> 24; // id
+ buff[offset + 9] = id >> 16;
+ buff[offset + 10] = id >> 8;
+ buff[offset + 11] = id;
+ buff[offset + 12] = 0; // x-position
+ buff[offset + 13] = 0;
+ buff[offset + 14] = 0; // y-position
+ buff[offset + 15] = 0;
+ buff[offset + 16] = width >> 8; // width
+ buff[offset + 17] = width;
+ buff[offset + 18] = height >> 8; // height
+ buff[offset + 19] = height;
+ buff[offset + 20] = flags >> 24; // flags
+ buff[offset + 21] = flags >> 16;
+ buff[offset + 22] = flags >> 8;
+ buff[offset + 23] = flags;
+
+ sock._sQlen += 24;
+ sock.flush();
+ },
+
+ clientFence: function (sock, flags, payload) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
+
+ buff[offset] = 248; // msg-type
+
+ buff[offset + 1] = 0; // padding
+ buff[offset + 2] = 0; // padding
+ buff[offset + 3] = 0; // padding
+
+ buff[offset + 4] = flags >> 24; // flags
+ buff[offset + 5] = flags >> 16;
+ buff[offset + 6] = flags >> 8;
+ buff[offset + 7] = flags;
+
+ var n = payload.length;
+
+ buff[offset + 8] = n; // length
+
+ for (var i = 0; i < n; i++) {
+ buff[offset + 9 + i] = payload.charCodeAt(i);
+ }
- sock._sQlen += 8 + n;
- sock.flush();
- },
-
- setDesktopSize: function (sock, width, height, id, flags) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
-
- buff[offset] = 251; // msg-type
- buff[offset + 1] = 0; // padding
- buff[offset + 2] = width >> 8; // width
- buff[offset + 3] = width;
- buff[offset + 4] = height >> 8; // height
- buff[offset + 5] = height;
-
- buff[offset + 6] = 1; // number-of-screens
- buff[offset + 7] = 0; // padding
-
- // screen array
- buff[offset + 8] = id >> 24; // id
- buff[offset + 9] = id >> 16;
- buff[offset + 10] = id >> 8;
- buff[offset + 11] = id;
- buff[offset + 12] = 0; // x-position
- buff[offset + 13] = 0;
- buff[offset + 14] = 0; // y-position
- buff[offset + 15] = 0;
- buff[offset + 16] = width >> 8; // width
- buff[offset + 17] = width;
- buff[offset + 18] = height >> 8; // height
- buff[offset + 19] = height;
- buff[offset + 20] = flags >> 24; // flags
- buff[offset + 21] = flags >> 16;
- buff[offset + 22] = flags >> 8;
- buff[offset + 23] = flags;
-
- sock._sQlen += 24;
- sock.flush();
- },
-
- clientFence: function (sock, flags, payload) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
-
- buff[offset] = 248; // msg-type
-
- buff[offset + 1] = 0; // padding
- buff[offset + 2] = 0; // padding
- buff[offset + 3] = 0; // padding
-
- buff[offset + 4] = flags >> 24; // flags
- buff[offset + 5] = flags >> 16;
- buff[offset + 6] = flags >> 8;
- buff[offset + 7] = flags;
-
- var n = payload.length;
-
- buff[offset + 8] = n; // length
-
- for (var i = 0; i < n; i++) {
- buff[offset + 9 + i] = payload.charCodeAt(i);
- }
+ sock._sQlen += 9 + n;
+ sock.flush();
+ },
- sock._sQlen += 9 + n;
- sock.flush();
- },
+ enableContinuousUpdates: function (sock, enable, x, y, width, height) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- enableContinuousUpdates: function (sock, enable, x, y, width, height) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 150; // msg-type
+ buff[offset + 1] = enable; // enable-flag
- buff[offset] = 150; // msg-type
- buff[offset + 1] = enable; // enable-flag
+ buff[offset + 2] = x >> 8; // x
+ buff[offset + 3] = x;
+ buff[offset + 4] = y >> 8; // y
+ buff[offset + 5] = y;
+ buff[offset + 6] = width >> 8; // width
+ buff[offset + 7] = width;
+ buff[offset + 8] = height >> 8; // height
+ buff[offset + 9] = height;
- buff[offset + 2] = x >> 8; // x
- buff[offset + 3] = x;
- buff[offset + 4] = y >> 8; // y
- buff[offset + 5] = y;
- buff[offset + 6] = width >> 8; // width
- buff[offset + 7] = width;
- buff[offset + 8] = height >> 8; // height
- buff[offset + 9] = height;
+ sock._sQlen += 10;
+ sock.flush();
+ },
- sock._sQlen += 10;
- sock.flush();
- },
+ pixelFormat: function (sock, bpp, depth, true_color) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- pixelFormat: function (sock, bpp, depth, true_color) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 0; // msg-type
- buff[offset] = 0; // msg-type
+ buff[offset + 1] = 0; // padding
+ buff[offset + 2] = 0; // padding
+ buff[offset + 3] = 0; // padding
- buff[offset + 1] = 0; // padding
- buff[offset + 2] = 0; // padding
- buff[offset + 3] = 0; // padding
+ buff[offset + 4] = bpp * 8; // bits-per-pixel
+ buff[offset + 5] = depth * 8; // depth
+ buff[offset + 6] = 0; // little-endian
+ buff[offset + 7] = true_color ? 1 : 0; // true-color
- buff[offset + 4] = bpp * 8; // bits-per-pixel
- buff[offset + 5] = depth * 8; // depth
- buff[offset + 6] = 0; // little-endian
- buff[offset + 7] = true_color ? 1 : 0; // true-color
+ buff[offset + 8] = 0; // red-max
+ buff[offset + 9] = 255; // red-max
- buff[offset + 8] = 0; // red-max
- buff[offset + 9] = 255; // red-max
+ buff[offset + 10] = 0; // green-max
+ buff[offset + 11] = 255; // green-max
- buff[offset + 10] = 0; // green-max
- buff[offset + 11] = 255; // green-max
+ buff[offset + 12] = 0; // blue-max
+ buff[offset + 13] = 255; // blue-max
- buff[offset + 12] = 0; // blue-max
- buff[offset + 13] = 255; // blue-max
+ buff[offset + 14] = 16; // red-shift
+ buff[offset + 15] = 8; // green-shift
+ buff[offset + 16] = 0; // blue-shift
- buff[offset + 14] = 16; // red-shift
- buff[offset + 15] = 8; // green-shift
- buff[offset + 16] = 0; // blue-shift
+ buff[offset + 17] = 0; // padding
+ buff[offset + 18] = 0; // padding
+ buff[offset + 19] = 0; // padding
- buff[offset + 17] = 0; // padding
- buff[offset + 18] = 0; // padding
- buff[offset + 19] = 0; // padding
+ sock._sQlen += 20;
+ sock.flush();
+ },
- sock._sQlen += 20;
- sock.flush();
- },
+ clientEncodings: function (sock, encodings, local_cursor) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- clientEncodings: function (sock, encodings, local_cursor, true_color) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset] = 2; // msg-type
+ buff[offset + 1] = 0; // padding
- buff[offset] = 2; // msg-type
- buff[offset + 1] = 0; // padding
+ // offset + 2 and offset + 3 are encoding count
- // offset + 2 and offset + 3 are encoding count
+ var i, j = offset + 4, cnt = 0;
+ for (i = 0; i < encodings.length; i++) {
+ if (encodings[i][0] === "Cursor" && !local_cursor) {
+ Log.Debug("Skipping Cursor pseudo-encoding");
+ } else {
+ var enc = encodings[i][1];
+ buff[j] = enc >> 24;
+ buff[j + 1] = enc >> 16;
+ buff[j + 2] = enc >> 8;
+ buff[j + 3] = enc;
- var i, j = offset + 4, cnt = 0;
- for (i = 0; i < encodings.length; i++) {
- if (encodings[i][0] === "Cursor" && !local_cursor) {
- Util.Debug("Skipping Cursor pseudo-encoding");
- } else if (encodings[i][0] === "TIGHT" && !true_color) {
- // TODO: remove this when we have tight+non-true-color
- Util.Warn("Skipping tight as it is only supported with true color");
- } else {
- var enc = encodings[i][1];
- buff[j] = enc >> 24;
- buff[j + 1] = enc >> 16;
- buff[j + 2] = enc >> 8;
- buff[j + 3] = enc;
-
- j += 4;
- cnt++;
- }
+ j += 4;
+ cnt++;
}
+ }
- buff[offset + 2] = cnt >> 8;
- buff[offset + 3] = cnt;
+ buff[offset + 2] = cnt >> 8;
+ buff[offset + 3] = cnt;
- sock._sQlen += j - offset;
- sock.flush();
- },
+ sock._sQlen += j - offset;
+ sock.flush();
+ },
- fbUpdateRequests: function (sock, onlyNonInc, cleanDirty, fb_width, fb_height) {
- var offsetIncrement = 0;
+ fbUpdateRequest: function (sock, incremental, x, y, w, h) {
+ var buff = sock._sQ;
+ var offset = sock._sQlen;
- var cb = cleanDirty.cleanBox;
- var w, h;
- if (!onlyNonInc && (cb.w > 0 && cb.h > 0)) {
- w = typeof cb.w === "undefined" ? fb_width : cb.w;
- h = typeof cb.h === "undefined" ? fb_height : cb.h;
- // Request incremental for clean box
- RFB.messages.fbUpdateRequest(sock, 1, cb.x, cb.y, w, h);
- }
+ if (typeof(x) === "undefined") { x = 0; }
+ if (typeof(y) === "undefined") { y = 0; }
- for (var i = 0; i < cleanDirty.dirtyBoxes.length; i++) {
- var db = cleanDirty.dirtyBoxes[i];
- // Force all (non-incremental) for dirty box
- w = typeof db.w === "undefined" ? fb_width : db.w;
- h = typeof db.h === "undefined" ? fb_height : db.h;
- RFB.messages.fbUpdateRequest(sock, 0, db.x, db.y, w, h);
- }
- },
+ buff[offset] = 3; // msg-type
+ buff[offset + 1] = incremental ? 1 : 0;
- fbUpdateRequest: function (sock, incremental, x, y, w, h) {
- var buff = sock._sQ;
- var offset = sock._sQlen;
+ buff[offset + 2] = (x >> 8) & 0xFF;
+ buff[offset + 3] = x & 0xFF;
- if (typeof(x) === "undefined") { x = 0; }
- if (typeof(y) === "undefined") { y = 0; }
+ buff[offset + 4] = (y >> 8) & 0xFF;
+ buff[offset + 5] = y & 0xFF;
- buff[offset] = 3; // msg-type
- buff[offset + 1] = incremental;
+ buff[offset + 6] = (w >> 8) & 0xFF;
+ buff[offset + 7] = w & 0xFF;
- buff[offset + 2] = (x >> 8) & 0xFF;
- buff[offset + 3] = x & 0xFF;
+ buff[offset + 8] = (h >> 8) & 0xFF;
+ buff[offset + 9] = h & 0xFF;
- buff[offset + 4] = (y >> 8) & 0xFF;
- buff[offset + 5] = y & 0xFF;
+ sock._sQlen += 10;
+ sock.flush();
+ }
+};
- buff[offset + 6] = (w >> 8) & 0xFF;
- buff[offset + 7] = w & 0xFF;
+RFB.genDES = function (password, challenge) {
+ var passwd = [];
+ for (var i = 0; i < password.length; i++) {
+ passwd.push(password.charCodeAt(i));
+ }
+ return (new DES(passwd)).encrypt(challenge);
+};
- buff[offset + 8] = (h >> 8) & 0xFF;
- buff[offset + 9] = h & 0xFF;
+RFB.encodingHandlers = {
+ RAW: function () {
+ if (this._FBU.lines === 0) {
+ this._FBU.lines = this._FBU.height;
+ }
- sock._sQlen += 10;
- sock.flush();
+ this._FBU.bytes = this._FBU.width * 4; // at least a line
+ if (this._sock.rQwait("RAW", this._FBU.bytes)) { return false; }
+ var cur_y = this._FBU.y + (this._FBU.height - this._FBU.lines);
+ var curr_height = Math.min(this._FBU.lines,
+ Math.floor(this._sock.rQlen() / (this._FBU.width * 4)));
+ this._display.blitImage(this._FBU.x, cur_y, this._FBU.width,
+ curr_height, this._sock.get_rQ(),
+ this._sock.get_rQi());
+ this._sock.rQskipBytes(this._FBU.width * curr_height * 4);
+ this._FBU.lines -= curr_height;
+
+ if (this._FBU.lines > 0) {
+ this._FBU.bytes = this._FBU.width * 4; // At least another line
+ } else {
+ this._FBU.rects--;
+ this._FBU.bytes = 0;
}
- };
- RFB.genDES = function (password, challenge) {
- var passwd = [];
- for (var i = 0; i < password.length; i++) {
- passwd.push(password.charCodeAt(i));
+ return true;
+ },
+
+ COPYRECT: function () {
+ this._FBU.bytes = 4;
+ if (this._sock.rQwait("COPYRECT", 4)) { return false; }
+ this._display.copyImage(this._sock.rQshift16(), this._sock.rQshift16(),
+ this._FBU.x, this._FBU.y, this._FBU.width,
+ this._FBU.height);
+
+ this._FBU.rects--;
+ this._FBU.bytes = 0;
+ return true;
+ },
+
+ RRE: function () {
+ var color;
+ if (this._FBU.subrects === 0) {
+ this._FBU.bytes = 4 + 4;
+ if (this._sock.rQwait("RRE", 4 + 4)) { return false; }
+ this._FBU.subrects = this._sock.rQshift32();
+ color = this._sock.rQshiftBytes(4); // Background
+ this._display.fillRect(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, color);
}
- return (new DES(passwd)).encrypt(challenge);
- };
- RFB.extract_data_uri = function (arr) {
- return ";base64," + Base64.encode(arr);
- };
+ while (this._FBU.subrects > 0 && this._sock.rQlen() >= (4 + 8)) {
+ color = this._sock.rQshiftBytes(4);
+ var x = this._sock.rQshift16();
+ var y = this._sock.rQshift16();
+ var width = this._sock.rQshift16();
+ var height = this._sock.rQshift16();
+ this._display.fillRect(this._FBU.x + x, this._FBU.y + y, width, height, color);
+ this._FBU.subrects--;
+ }
- RFB.encodingHandlers = {
- RAW: function () {
- if (this._FBU.lines === 0) {
- this._FBU.lines = this._FBU.height;
- }
+ if (this._FBU.subrects > 0) {
+ var chunk = Math.min(this._rre_chunk_sz, this._FBU.subrects);
+ this._FBU.bytes = (4 + 8) * chunk;
+ } else {
+ this._FBU.rects--;
+ this._FBU.bytes = 0;
+ }
- this._FBU.bytes = this._FBU.width * this._fb_Bpp; // at least a line
- if (this._sock.rQwait("RAW", this._FBU.bytes)) { return false; }
- var cur_y = this._FBU.y + (this._FBU.height - this._FBU.lines);
- var curr_height = Math.min(this._FBU.lines,
- Math.floor(this._sock.rQlen() / (this._FBU.width * this._fb_Bpp)));
- this._display.blitImage(this._FBU.x, cur_y, this._FBU.width,
- curr_height, this._sock.get_rQ(),
- this._sock.get_rQi());
- this._sock.rQskipBytes(this._FBU.width * curr_height * this._fb_Bpp);
- this._FBU.lines -= curr_height;
-
- if (this._FBU.lines > 0) {
- this._FBU.bytes = this._FBU.width * this._fb_Bpp; // At least another line
- } else {
- this._FBU.rects--;
- this._FBU.bytes = 0;
- }
+ return true;
+ },
- return true;
- },
+ HEXTILE: function () {
+ var rQ = this._sock.get_rQ();
+ var rQi = this._sock.get_rQi();
- COPYRECT: function () {
- this._FBU.bytes = 4;
- if (this._sock.rQwait("COPYRECT", 4)) { return false; }
- this._display.copyImage(this._sock.rQshift16(), this._sock.rQshift16(),
- this._FBU.x, this._FBU.y, this._FBU.width,
- this._FBU.height);
+ if (this._FBU.tiles === 0) {
+ this._FBU.tiles_x = Math.ceil(this._FBU.width / 16);
+ this._FBU.tiles_y = Math.ceil(this._FBU.height / 16);
+ this._FBU.total_tiles = this._FBU.tiles_x * this._FBU.tiles_y;
+ this._FBU.tiles = this._FBU.total_tiles;
+ }
- this._FBU.rects--;
- this._FBU.bytes = 0;
- return true;
- },
-
- RRE: function () {
- var color;
- if (this._FBU.subrects === 0) {
- this._FBU.bytes = 4 + this._fb_Bpp;
- if (this._sock.rQwait("RRE", 4 + this._fb_Bpp)) { return false; }
- this._FBU.subrects = this._sock.rQshift32();
- color = this._sock.rQshiftBytes(this._fb_Bpp); // Background
- this._display.fillRect(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, color);
+ while (this._FBU.tiles > 0) {
+ this._FBU.bytes = 1;
+ if (this._sock.rQwait("HEXTILE subencoding", this._FBU.bytes)) { return false; }
+ var subencoding = rQ[rQi]; // Peek
+ if (subencoding > 30) { // Raw
+ this._fail("Unexpected server message",
+ "Illegal hextile subencoding: " + subencoding);
+ return false;
}
- while (this._FBU.subrects > 0 && this._sock.rQlen() >= (this._fb_Bpp + 8)) {
- color = this._sock.rQshiftBytes(this._fb_Bpp);
- var x = this._sock.rQshift16();
- var y = this._sock.rQshift16();
- var width = this._sock.rQshift16();
- var height = this._sock.rQshift16();
- this._display.fillRect(this._FBU.x + x, this._FBU.y + y, width, height, color);
- this._FBU.subrects--;
- }
+ var subrects = 0;
+ var curr_tile = this._FBU.total_tiles - this._FBU.tiles;
+ var tile_x = curr_tile % this._FBU.tiles_x;
+ var tile_y = Math.floor(curr_tile / this._FBU.tiles_x);
+ var x = this._FBU.x + tile_x * 16;
+ var y = this._FBU.y + tile_y * 16;
+ var w = Math.min(16, (this._FBU.x + this._FBU.width) - x);
+ var h = Math.min(16, (this._FBU.y + this._FBU.height) - y);
- if (this._FBU.subrects > 0) {
- var chunk = Math.min(this._rre_chunk_sz, this._FBU.subrects);
- this._FBU.bytes = (this._fb_Bpp + 8) * chunk;
+ // Figure out how much we are expecting
+ if (subencoding & 0x01) { // Raw
+ this._FBU.bytes += w * h * 4;
} else {
- this._FBU.rects--;
- this._FBU.bytes = 0;
- }
-
- return true;
- },
-
- HEXTILE: function () {
- var rQ = this._sock.get_rQ();
- var rQi = this._sock.get_rQi();
-
- if (this._FBU.tiles === 0) {
- this._FBU.tiles_x = Math.ceil(this._FBU.width / 16);
- this._FBU.tiles_y = Math.ceil(this._FBU.height / 16);
- this._FBU.total_tiles = this._FBU.tiles_x * this._FBU.tiles_y;
- this._FBU.tiles = this._FBU.total_tiles;
+ if (subencoding & 0x02) { // Background
+ this._FBU.bytes += 4;
+ }
+ if (subencoding & 0x04) { // Foreground
+ this._FBU.bytes += 4;
+ }
+ if (subencoding & 0x08) { // AnySubrects
+ this._FBU.bytes++; // Since we aren't shifting it off
+ if (this._sock.rQwait("hextile subrects header", this._FBU.bytes)) { return false; }
+ subrects = rQ[rQi + this._FBU.bytes - 1]; // Peek
+ if (subencoding & 0x10) { // SubrectsColoured
+ this._FBU.bytes += subrects * (4 + 2);
+ } else {
+ this._FBU.bytes += subrects * 2;
+ }
+ }
}
- while (this._FBU.tiles > 0) {
- this._FBU.bytes = 1;
- if (this._sock.rQwait("HEXTILE subencoding", this._FBU.bytes)) { return false; }
- var subencoding = rQ[rQi]; // Peek
- if (subencoding > 30) { // Raw
- this._fail("Disconnected: illegal hextile subencoding " + subencoding);
- return false;
- }
+ if (this._sock.rQwait("hextile", this._FBU.bytes)) { return false; }
- var subrects = 0;
- var curr_tile = this._FBU.total_tiles - this._FBU.tiles;
- var tile_x = curr_tile % this._FBU.tiles_x;
- var tile_y = Math.floor(curr_tile / this._FBU.tiles_x);
- var x = this._FBU.x + tile_x * 16;
- var y = this._FBU.y + tile_y * 16;
- var w = Math.min(16, (this._FBU.x + this._FBU.width) - x);
- var h = Math.min(16, (this._FBU.y + this._FBU.height) - y);
-
- // Figure out how much we are expecting
- if (subencoding & 0x01) { // Raw
- this._FBU.bytes += w * h * this._fb_Bpp;
+ // We know the encoding and have a whole tile
+ this._FBU.subencoding = rQ[rQi];
+ rQi++;
+ if (this._FBU.subencoding === 0) {
+ if (this._FBU.lastsubencoding & 0x01) {
+ // Weird: ignore blanks are RAW
+ Log.Debug(" Ignoring blank after RAW");
} else {
- if (subencoding & 0x02) { // Background
- this._FBU.bytes += this._fb_Bpp;
- }
- if (subencoding & 0x04) { // Foreground
- this._FBU.bytes += this._fb_Bpp;
- }
- if (subencoding & 0x08) { // AnySubrects
- this._FBU.bytes++; // Since we aren't shifting it off
- if (this._sock.rQwait("hextile subrects header", this._FBU.bytes)) { return false; }
- subrects = rQ[rQi + this._FBU.bytes - 1]; // Peek
- if (subencoding & 0x10) { // SubrectsColoured
- this._FBU.bytes += subrects * (this._fb_Bpp + 2);
- } else {
- this._FBU.bytes += subrects * 2;
- }
- }
+ this._display.fillRect(x, y, w, h, this._FBU.background);
+ }
+ } else if (this._FBU.subencoding & 0x01) { // Raw
+ this._display.blitImage(x, y, w, h, rQ, rQi);
+ rQi += this._FBU.bytes - 1;
+ } else {
+ if (this._FBU.subencoding & 0x02) { // Background
+ this._FBU.background = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
+ rQi += 4;
+ }
+ if (this._FBU.subencoding & 0x04) { // Foreground
+ this._FBU.foreground = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
+ rQi += 4;
}
- if (this._sock.rQwait("hextile", this._FBU.bytes)) { return false; }
+ this._display.startTile(x, y, w, h, this._FBU.background);
+ if (this._FBU.subencoding & 0x08) { // AnySubrects
+ subrects = rQ[rQi];
+ rQi++;
- // We know the encoding and have a whole tile
- this._FBU.subencoding = rQ[rQi];
- rQi++;
- if (this._FBU.subencoding === 0) {
- if (this._FBU.lastsubencoding & 0x01) {
- // Weird: ignore blanks are RAW
- Util.Debug(" Ignoring blank after RAW");
- } else {
- this._display.fillRect(x, y, w, h, this._FBU.background);
- }
- } else if (this._FBU.subencoding & 0x01) { // Raw
- this._display.blitImage(x, y, w, h, rQ, rQi);
- rQi += this._FBU.bytes - 1;
- } else {
- if (this._FBU.subencoding & 0x02) { // Background
- if (this._fb_Bpp == 1) {
- this._FBU.background = rQ[rQi];
+ for (var s = 0; s < subrects; s++) {
+ var color;
+ if (this._FBU.subencoding & 0x10) { // SubrectsColoured
+ color = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
+ rQi += 4;
} else {
- // fb_Bpp is 4
- this._FBU.background = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
+ color = this._FBU.foreground;
}
- rQi += this._fb_Bpp;
- }
- if (this._FBU.subencoding & 0x04) { // Foreground
- if (this._fb_Bpp == 1) {
- this._FBU.foreground = rQ[rQi];
- } else {
- // this._fb_Bpp is 4
- this._FBU.foreground = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
- }
- rQi += this._fb_Bpp;
- }
+ var xy = rQ[rQi];
+ rQi++;
+ var sx = (xy >> 4);
+ var sy = (xy & 0x0f);
- this._display.startTile(x, y, w, h, this._FBU.background);
- if (this._FBU.subencoding & 0x08) { // AnySubrects
- subrects = rQ[rQi];
+ var wh = rQ[rQi];
rQi++;
+ var sw = (wh >> 4) + 1;
+ var sh = (wh & 0x0f) + 1;
- for (var s = 0; s < subrects; s++) {
- var color;
- if (this._FBU.subencoding & 0x10) { // SubrectsColoured
- if (this._fb_Bpp === 1) {
- color = rQ[rQi];
- } else {
- // _fb_Bpp is 4
- color = [rQ[rQi], rQ[rQi + 1], rQ[rQi + 2], rQ[rQi + 3]];
- }
- rQi += this._fb_Bpp;
- } else {
- color = this._FBU.foreground;
- }
- var xy = rQ[rQi];
- rQi++;
- var sx = (xy >> 4);
- var sy = (xy & 0x0f);
-
- var wh = rQ[rQi];
- rQi++;
- var sw = (wh >> 4) + 1;
- var sh = (wh & 0x0f) + 1;
-
- this._display.subTile(sx, sy, sw, sh, color);
- }
+ this._display.subTile(sx, sy, sw, sh, color);
}
- this._display.finishTile();
}
- this._sock.set_rQi(rQi);
- this._FBU.lastsubencoding = this._FBU.subencoding;
- this._FBU.bytes = 0;
- this._FBU.tiles--;
+ this._display.finishTile();
}
+ this._sock.set_rQi(rQi);
+ this._FBU.lastsubencoding = this._FBU.subencoding;
+ this._FBU.bytes = 0;
+ this._FBU.tiles--;
+ }
- if (this._FBU.tiles === 0) {
- this._FBU.rects--;
- }
+ if (this._FBU.tiles === 0) {
+ this._FBU.rects--;
+ }
- return true;
- },
+ return true;
+ },
- getTightCLength: function (arr) {
- var header = 1, data = 0;
- data += arr[0] & 0x7f;
- if (arr[0] & 0x80) {
+ getTightCLength: function (arr) {
+ var header = 1, data = 0;
+ data += arr[0] & 0x7f;
+ if (arr[0] & 0x80) {
+ header++;
+ data += (arr[1] & 0x7f) << 7;
+ if (arr[1] & 0x80) {
header++;
- data += (arr[1] & 0x7f) << 7;
- if (arr[1] & 0x80) {
- header++;
- data += arr[2] << 14;
- }
- }
- return [header, data];
- },
-
- display_tight: function (isTightPNG) {
- if (this._fb_depth === 1) {
- this._fail("Tight protocol handler only implements true color mode");
+ data += arr[2] << 14;
}
-
- this._FBU.bytes = 1; // compression-control byte
- if (this._sock.rQwait("TIGHT compression-control", this._FBU.bytes)) { return false; }
-
- var checksum = function (data) {
- var sum = 0;
- for (var i = 0; i < data.length; i++) {
- sum += data[i];
- if (sum > 65536) sum -= 65536;
- }
- return sum;
- };
-
- var resetStreams = 0;
- var streamId = -1;
- var decompress = function (data, expected) {
- for (var i = 0; i < 4; i++) {
- if ((resetStreams >> i) & 1) {
- this._FBU.zlibs[i].reset();
- Util.Info("Reset zlib stream " + i);
- }
+ }
+ return [header, data];
+ },
+
+ display_tight: function (isTightPNG) {
+ this._FBU.bytes = 1; // compression-control byte
+ if (this._sock.rQwait("TIGHT compression-control", this._FBU.bytes)) { return false; }
+
+ var checksum = function (data) {
+ var sum = 0;
+ for (var i = 0; i < data.length; i++) {
+ sum += data[i];
+ if (sum > 65536) sum -= 65536;
+ }
+ return sum;
+ };
+
+ var resetStreams = 0;
+ var streamId = -1;
+ var decompress = function (data, expected) {
+ for (var i = 0; i < 4; i++) {
+ if ((resetStreams >> i) & 1) {
+ this._FBU.zlibs[i].reset();
+ Log.Info("Reset zlib stream " + i);
}
+ }
- //var uncompressed = this._FBU.zlibs[streamId].uncompress(data, 0);
- var uncompressed = this._FBU.zlibs[streamId].inflate(data, true, expected);
- /*if (uncompressed.status !== 0) {
- Util.Error("Invalid data in zlib stream");
- }*/
-
- //return uncompressed.data;
- return uncompressed;
- }.bind(this);
-
- var indexedToRGBX2Color = function (data, palette, width, height) {
- // Convert indexed (palette based) image data to RGB
- // TODO: reduce number of calculations inside loop
- var dest = this._destBuff;
- var w = Math.floor((width + 7) / 8);
- var w1 = Math.floor(width / 8);
-
- /*for (var y = 0; y < height; y++) {
- var b, x, dp, sp;
- var yoffset = y * width;
- var ybitoffset = y * w;
- var xoffset, targetbyte;
- for (x = 0; x < w1; x++) {
- xoffset = yoffset + x * 8;
- targetbyte = data[ybitoffset + x];
- for (b = 7; b >= 0; b--) {
- dp = (xoffset + 7 - b) * 3;
- sp = (targetbyte >> b & 1) * 3;
- dest[dp] = palette[sp];
- dest[dp + 1] = palette[sp + 1];
- dest[dp + 2] = palette[sp + 2];
- }
- }
-
+ //var uncompressed = this._FBU.zlibs[streamId].uncompress(data, 0);
+ var uncompressed = this._FBU.zlibs[streamId].inflate(data, true, expected);
+ /*if (uncompressed.status !== 0) {
+ Log.Error("Invalid data in zlib stream");
+ }*/
+
+ //return uncompressed.data;
+ return uncompressed;
+ }.bind(this);
+
+ var indexedToRGBX2Color = function (data, palette, width, height) {
+ // Convert indexed (palette based) image data to RGB
+ // TODO: reduce number of calculations inside loop
+ var dest = this._destBuff;
+ var w = Math.floor((width + 7) / 8);
+ var w1 = Math.floor(width / 8);
+
+ /*for (var y = 0; y < height; y++) {
+ var b, x, dp, sp;
+ var yoffset = y * width;
+ var ybitoffset = y * w;
+ var xoffset, targetbyte;
+ for (x = 0; x < w1; x++) {
xoffset = yoffset + x * 8;
targetbyte = data[ybitoffset + x];
- for (b = 7; b >= 8 - width % 8; b--) {
+ for (b = 7; b >= 0; b--) {
dp = (xoffset + 7 - b) * 3;
sp = (targetbyte >> b & 1) * 3;
dest[dp] = palette[sp];
dest[dp + 1] = palette[sp + 1];
dest[dp + 2] = palette[sp + 2];
}
- }*/
-
- for (var y = 0; y < height; y++) {
- var b, x, dp, sp;
- for (x = 0; x < w1; x++) {
- for (b = 7; b >= 0; b--) {
- dp = (y * width + x * 8 + 7 - b) * 4;
- sp = (data[y * w + x] >> b & 1) * 3;
- dest[dp] = palette[sp];
- dest[dp + 1] = palette[sp + 1];
- dest[dp + 2] = palette[sp + 2];
- dest[dp + 3] = 255;
- }
- }
+ }
+
+ xoffset = yoffset + x * 8;
+ targetbyte = data[ybitoffset + x];
+ for (b = 7; b >= 8 - width % 8; b--) {
+ dp = (xoffset + 7 - b) * 3;
+ sp = (targetbyte >> b & 1) * 3;
+ dest[dp] = palette[sp];
+ dest[dp + 1] = palette[sp + 1];
+ dest[dp + 2] = palette[sp + 2];
+ }
+ }*/
- for (b = 7; b >= 8 - width % 8; b--) {
+ for (var y = 0; y < height; y++) {
+ var b, x, dp, sp;
+ for (x = 0; x < w1; x++) {
+ for (b = 7; b >= 0; b--) {
dp = (y * width + x * 8 + 7 - b) * 4;
sp = (data[y * w + x] >> b & 1) * 3;
dest[dp] = palette[sp];
}
}
- return dest;
- }.bind(this);
-
- var indexedToRGBX = function (data, palette, width, height) {
- // Convert indexed (palette based) image data to RGB
- var dest = this._destBuff;
- var total = width * height * 4;
- for (var i = 0, j = 0; i < total; i += 4, j++) {
- var sp = data[j] * 3;
- dest[i] = palette[sp];
- dest[i + 1] = palette[sp + 1];
- dest[i + 2] = palette[sp + 2];
- dest[i + 3] = 255;
+ for (b = 7; b >= 8 - width % 8; b--) {
+ dp = (y * width + x * 8 + 7 - b) * 4;
+ sp = (data[y * w + x] >> b & 1) * 3;
+ dest[dp] = palette[sp];
+ dest[dp + 1] = palette[sp + 1];
+ dest[dp + 2] = palette[sp + 2];
+ dest[dp + 3] = 255;
}
+ }
- return dest;
- }.bind(this);
-
- var rQi = this._sock.get_rQi();
- var rQ = this._sock.rQwhole();
- var cmode, data;
- var cl_header, cl_data;
-
- var handlePalette = function () {
- var numColors = rQ[rQi + 2] + 1;
- var paletteSize = numColors * this._fb_depth;
- this._FBU.bytes += paletteSize;
- if (this._sock.rQwait("TIGHT palette " + cmode, this._FBU.bytes)) { return false; }
-
- var bpp = (numColors <= 2) ? 1 : 8;
- var rowSize = Math.floor((this._FBU.width * bpp + 7) / 8);
- var raw = false;
- if (rowSize * this._FBU.height < 12) {
- raw = true;
- cl_header = 0;
- cl_data = rowSize * this._FBU.height;
- //clength = [0, rowSize * this._FBU.height];
- } else {
- // begin inline getTightCLength (returning two-item arrays is bad for performance with GC)
- var cl_offset = rQi + 3 + paletteSize;
- cl_header = 1;
- cl_data = 0;
- cl_data += rQ[cl_offset] & 0x7f;
- if (rQ[cl_offset] & 0x80) {
+ return dest;
+ }.bind(this);
+
+ var indexedToRGBX = function (data, palette, width, height) {
+ // Convert indexed (palette based) image data to RGB
+ var dest = this._destBuff;
+ var total = width * height * 4;
+ for (var i = 0, j = 0; i < total; i += 4, j++) {
+ var sp = data[j] * 3;
+ dest[i] = palette[sp];
+ dest[i + 1] = palette[sp + 1];
+ dest[i + 2] = palette[sp + 2];
+ dest[i + 3] = 255;
+ }
+
+ return dest;
+ }.bind(this);
+
+ var rQi = this._sock.get_rQi();
+ var rQ = this._sock.rQwhole();
+ var cmode, data;
+ var cl_header, cl_data;
+
+ var handlePalette = function () {
+ var numColors = rQ[rQi + 2] + 1;
+ var paletteSize = numColors * 3;
+ this._FBU.bytes += paletteSize;
+ if (this._sock.rQwait("TIGHT palette " + cmode, this._FBU.bytes)) { return false; }
+
+ var bpp = (numColors <= 2) ? 1 : 8;
+ var rowSize = Math.floor((this._FBU.width * bpp + 7) / 8);
+ var raw = false;
+ if (rowSize * this._FBU.height < 12) {
+ raw = true;
+ cl_header = 0;
+ cl_data = rowSize * this._FBU.height;
+ //clength = [0, rowSize * this._FBU.height];
+ } else {
+ // begin inline getTightCLength (returning two-item arrays is bad for performance with GC)
+ var cl_offset = rQi + 3 + paletteSize;
+ cl_header = 1;
+ cl_data = 0;
+ cl_data += rQ[cl_offset] & 0x7f;
+ if (rQ[cl_offset] & 0x80) {
+ cl_header++;
+ cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
+ if (rQ[cl_offset + 1] & 0x80) {
cl_header++;
- cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
- if (rQ[cl_offset + 1] & 0x80) {
- cl_header++;
- cl_data += rQ[cl_offset + 2] << 14;
- }
+ cl_data += rQ[cl_offset + 2] << 14;
}
- // end inline getTightCLength
}
+ // end inline getTightCLength
+ }
- this._FBU.bytes += cl_header + cl_data;
- if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
+ this._FBU.bytes += cl_header + cl_data;
+ if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
- // Shift ctl, filter id, num colors, palette entries, and clength off
- this._sock.rQskipBytes(3);
- //var palette = this._sock.rQshiftBytes(paletteSize);
- this._sock.rQshiftTo(this._paletteBuff, paletteSize);
- this._sock.rQskipBytes(cl_header);
+ // Shift ctl, filter id, num colors, palette entries, and clength off
+ this._sock.rQskipBytes(3);
+ //var palette = this._sock.rQshiftBytes(paletteSize);
+ this._sock.rQshiftTo(this._paletteBuff, paletteSize);
+ this._sock.rQskipBytes(cl_header);
- if (raw) {
- data = this._sock.rQshiftBytes(cl_data);
- } else {
- data = decompress(this._sock.rQshiftBytes(cl_data), rowSize * this._FBU.height);
- }
+ if (raw) {
+ data = this._sock.rQshiftBytes(cl_data);
+ } else {
+ data = decompress(this._sock.rQshiftBytes(cl_data), rowSize * this._FBU.height);
+ }
- // Convert indexed (palette based) image data to RGB
- var rgbx;
- if (numColors == 2) {
- rgbx = indexedToRGBX2Color(data, this._paletteBuff, this._FBU.width, this._FBU.height);
- this._display.blitRgbxImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, rgbx, 0, false);
- } else {
- rgbx = indexedToRGBX(data, this._paletteBuff, this._FBU.width, this._FBU.height);
- this._display.blitRgbxImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, rgbx, 0, false);
- }
+ // Convert indexed (palette based) image data to RGB
+ var rgbx;
+ if (numColors == 2) {
+ rgbx = indexedToRGBX2Color(data, this._paletteBuff, this._FBU.width, this._FBU.height);
+ this._display.blitRgbxImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, rgbx, 0, false);
+ } else {
+ rgbx = indexedToRGBX(data, this._paletteBuff, this._FBU.width, this._FBU.height);
+ this._display.blitRgbxImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, rgbx, 0, false);
+ }
- return true;
- }.bind(this);
-
- var handleCopy = function () {
- var raw = false;
- var uncompressedSize = this._FBU.width * this._FBU.height * this._fb_depth;
- if (uncompressedSize < 12) {
- raw = true;
- cl_header = 0;
- cl_data = uncompressedSize;
- } else {
- // begin inline getTightCLength (returning two-item arrays is for peformance with GC)
- var cl_offset = rQi + 1;
- cl_header = 1;
- cl_data = 0;
- cl_data += rQ[cl_offset] & 0x7f;
- if (rQ[cl_offset] & 0x80) {
+ return true;
+ }.bind(this);
+
+ var handleCopy = function () {
+ var raw = false;
+ var uncompressedSize = this._FBU.width * this._FBU.height * 3;
+ if (uncompressedSize < 12) {
+ raw = true;
+ cl_header = 0;
+ cl_data = uncompressedSize;
+ } else {
+ // begin inline getTightCLength (returning two-item arrays is for peformance with GC)
+ var cl_offset = rQi + 1;
+ cl_header = 1;
+ cl_data = 0;
+ cl_data += rQ[cl_offset] & 0x7f;
+ if (rQ[cl_offset] & 0x80) {
+ cl_header++;
+ cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
+ if (rQ[cl_offset + 1] & 0x80) {
cl_header++;
- cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
- if (rQ[cl_offset + 1] & 0x80) {
- cl_header++;
- cl_data += rQ[cl_offset + 2] << 14;
- }
+ cl_data += rQ[cl_offset + 2] << 14;
}
- // end inline getTightCLength
}
- this._FBU.bytes = 1 + cl_header + cl_data;
- if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
-
- // Shift ctl, clength off
- this._sock.rQshiftBytes(1 + cl_header);
+ // end inline getTightCLength
+ }
+ this._FBU.bytes = 1 + cl_header + cl_data;
+ if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
- if (raw) {
- data = this._sock.rQshiftBytes(cl_data);
- } else {
- data = decompress(this._sock.rQshiftBytes(cl_data), uncompressedSize);
- }
+ // Shift ctl, clength off
+ this._sock.rQshiftBytes(1 + cl_header);
- this._display.blitRgbImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, data, 0, false);
+ if (raw) {
+ data = this._sock.rQshiftBytes(cl_data);
+ } else {
+ data = decompress(this._sock.rQshiftBytes(cl_data), uncompressedSize);
+ }
- return true;
- }.bind(this);
+ this._display.blitRgbImage(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, data, 0, false);
- var ctl = this._sock.rQpeek8();
+ return true;
+ }.bind(this);
- // Keep tight reset bits
- resetStreams = ctl & 0xF;
+ var ctl = this._sock.rQpeek8();
- // Figure out filter
- ctl = ctl >> 4;
- streamId = ctl & 0x3;
+ // Keep tight reset bits
+ resetStreams = ctl & 0xF;
- if (ctl === 0x08) cmode = "fill";
- else if (ctl === 0x09) cmode = "jpeg";
- else if (ctl === 0x0A) cmode = "png";
- else if (ctl & 0x04) cmode = "filter";
- else if (ctl < 0x04) cmode = "copy";
- else return this._fail("Illegal tight compression received, ctl: " + ctl);
+ // Figure out filter
+ ctl = ctl >> 4;
+ streamId = ctl & 0x3;
- if (isTightPNG && (cmode === "filter" || cmode === "copy")) {
- return this._fail("filter/copy received in tightPNG mode");
- }
+ if (ctl === 0x08) cmode = "fill";
+ else if (ctl === 0x09) cmode = "jpeg";
+ else if (ctl === 0x0A) cmode = "png";
+ else if (ctl & 0x04) cmode = "filter";
+ else if (ctl < 0x04) cmode = "copy";
+ else return this._fail("Unexpected server message",
+ "Illegal tight compression received, " +
+ "ctl: " + ctl);
- switch (cmode) {
- // fill use fb_depth because TPIXELs drop the padding byte
- case "fill": // TPIXEL
- this._FBU.bytes += this._fb_depth;
- break;
- case "jpeg": // max clength
- this._FBU.bytes += 3;
- break;
- case "png": // max clength
- this._FBU.bytes += 3;
- break;
- case "filter": // filter id + num colors if palette
- this._FBU.bytes += 2;
- break;
- case "copy":
- break;
- }
+ if (isTightPNG && (cmode === "filter" || cmode === "copy")) {
+ return this._fail("Unexpected server message",
+ "filter/copy received in tightPNG mode");
+ }
- if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
+ switch (cmode) {
+ // fill use depth because TPIXELs drop the padding byte
+ case "fill": // TPIXEL
+ this._FBU.bytes += 3;
+ break;
+ case "jpeg": // max clength
+ this._FBU.bytes += 3;
+ break;
+ case "png": // max clength
+ this._FBU.bytes += 3;
+ break;
+ case "filter": // filter id + num colors if palette
+ this._FBU.bytes += 2;
+ break;
+ case "copy":
+ break;
+ }
- // Determine FBU.bytes
- switch (cmode) {
- case "fill":
- // skip ctl byte
- this._display.fillRect(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, [rQ[rQi + 3], rQ[rQi + 2], rQ[rQi + 1]], false);
- this._sock.rQskipBytes(4);
- break;
- case "png":
- case "jpeg":
- // begin inline getTightCLength (returning two-item arrays is for peformance with GC)
- var cl_offset = rQi + 1;
- cl_header = 1;
- cl_data = 0;
- cl_data += rQ[cl_offset] & 0x7f;
- if (rQ[cl_offset] & 0x80) {
+ if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
+
+ // Determine FBU.bytes
+ switch (cmode) {
+ case "fill":
+ // skip ctl byte
+ this._display.fillRect(this._FBU.x, this._FBU.y, this._FBU.width, this._FBU.height, [rQ[rQi + 3], rQ[rQi + 2], rQ[rQi + 1]], false);
+ this._sock.rQskipBytes(4);
+ break;
+ case "png":
+ case "jpeg":
+ // begin inline getTightCLength (returning two-item arrays is for peformance with GC)
+ var cl_offset = rQi + 1;
+ cl_header = 1;
+ cl_data = 0;
+ cl_data += rQ[cl_offset] & 0x7f;
+ if (rQ[cl_offset] & 0x80) {
+ cl_header++;
+ cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
+ if (rQ[cl_offset + 1] & 0x80) {
cl_header++;
- cl_data += (rQ[cl_offset + 1] & 0x7f) << 7;
- if (rQ[cl_offset + 1] & 0x80) {
- cl_header++;
- cl_data += rQ[cl_offset + 2] << 14;
- }
+ cl_data += rQ[cl_offset + 2] << 14;
}
- // end inline getTightCLength
- this._FBU.bytes = 1 + cl_header + cl_data; // ctl + clength size + jpeg-data
- if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
-
- // We have everything, render it
- this._sock.rQskipBytes(1 + cl_header); // shift off clt + compact length
- var img = new Image();
- img.src = "data: image/" + cmode +
- RFB.extract_data_uri(this._sock.rQshiftBytes(cl_data));
- this._display.renderQ_push({
- 'type': 'img',
- 'img': img,
- 'x': this._FBU.x,
- 'y': this._FBU.y
- });
- img = null;
- break;
- case "filter":
- var filterId = rQ[rQi + 1];
- if (filterId === 1) {
- if (!handlePalette()) { return false; }
- } else {
- // Filter 0, Copy could be valid here, but servers don't send it as an explicit filter
- // Filter 2, Gradient is valid but not use if jpeg is enabled
- this._fail("Unsupported tight subencoding received, filter: " + filterId);
- }
- break;
- case "copy":
- if (!handleCopy()) { return false; }
- break;
- }
-
-
- this._FBU.bytes = 0;
- this._FBU.rects--;
-
- return true;
- },
-
- TIGHT: function () { return this._encHandlers.display_tight(false); },
- TIGHT_PNG: function () { return this._encHandlers.display_tight(true); },
-
- last_rect: function () {
- this._FBU.rects = 0;
- return true;
- },
-
- handle_FB_resize: function () {
- this._fb_width = this._FBU.width;
- this._fb_height = this._FBU.height;
- this._destBuff = new Uint8Array(this._fb_width * this._fb_height * 4);
- this._display.resize(this._fb_width, this._fb_height);
- this._onFBResize(this, this._fb_width, this._fb_height);
- this._timing.fbu_rt_start = (new Date()).getTime();
- this._updateContinuousUpdates();
-
- this._FBU.bytes = 0;
- this._FBU.rects -= 1;
- return true;
- },
+ }
+ // end inline getTightCLength
+ this._FBU.bytes = 1 + cl_header + cl_data; // ctl + clength size + jpeg-data
+ if (this._sock.rQwait("TIGHT " + cmode, this._FBU.bytes)) { return false; }
- ExtendedDesktopSize: function () {
- this._FBU.bytes = 1;
- if (this._sock.rQwait("ExtendedDesktopSize", this._FBU.bytes)) { return false; }
-
- this._supportsSetDesktopSize = true;
- var number_of_screens = this._sock.rQpeek8();
-
- this._FBU.bytes = 4 + (number_of_screens * 16);
- if (this._sock.rQwait("ExtendedDesktopSize", this._FBU.bytes)) { return false; }
-
- this._sock.rQskipBytes(1); // number-of-screens
- this._sock.rQskipBytes(3); // padding
-
- for (var i = 0; i < number_of_screens; i += 1) {
- // Save the id and flags of the first screen
- if (i === 0) {
- this._screen_id = this._sock.rQshiftBytes(4); // id
- this._sock.rQskipBytes(2); // x-position
- this._sock.rQskipBytes(2); // y-position
- this._sock.rQskipBytes(2); // width
- this._sock.rQskipBytes(2); // height
- this._screen_flags = this._sock.rQshiftBytes(4); // flags
+ // We have everything, render it
+ this._sock.rQskipBytes(1 + cl_header); // shift off clt + compact length
+ data = this._sock.rQshiftBytes(cl_data);
+ this._display.imageRect(this._FBU.x, this._FBU.y, "image/" + cmode, data);
+ break;
+ case "filter":
+ var filterId = rQ[rQi + 1];
+ if (filterId === 1) {
+ if (!handlePalette()) { return false; }
} else {
- this._sock.rQskipBytes(16);
+ // Filter 0, Copy could be valid here, but servers don't send it as an explicit filter
+ // Filter 2, Gradient is valid but not use if jpeg is enabled
+ this._fail("Unexpected server message",
+ "Unsupported tight subencoding received, " +
+ "filter: " + filterId);
}
- }
+ break;
+ case "copy":
+ if (!handleCopy()) { return false; }
+ break;
+ }
- /*
- * The x-position indicates the reason for the change:
- *
- * 0 - server resized on its own
- * 1 - this client requested the resize
- * 2 - another client requested the resize
- */
-
- // We need to handle errors when we requested the resize.
- if (this._FBU.x === 1 && this._FBU.y !== 0) {
- var msg = "";
- // The y-position indicates the status code from the server
- switch (this._FBU.y) {
- case 1:
- msg = "Resize is administratively prohibited";
- break;
- case 2:
- msg = "Out of resources";
- break;
- case 3:
- msg = "Invalid screen layout";
- break;
- default:
- msg = "Unknown reason";
- break;
- }
- this._notification("Server did not accept the resize request: "
- + msg, 'normal');
- return true;
+
+ this._FBU.bytes = 0;
+ this._FBU.rects--;
+
+ return true;
+ },
+
+ TIGHT: function () { return this._encHandlers.display_tight(false); },
+ TIGHT_PNG: function () { return this._encHandlers.display_tight(true); },
+
+ last_rect: function () {
+ this._FBU.rects = 0;
+ return true;
+ },
+
+ handle_FB_resize: function () {
+ this._fb_width = this._FBU.width;
+ this._fb_height = this._FBU.height;
+ this._destBuff = new Uint8Array(this._fb_width * this._fb_height * 4);
+ this._display.resize(this._fb_width, this._fb_height);
+ this._onFBResize(this, this._fb_width, this._fb_height);
+ this._timing.fbu_rt_start = (new Date()).getTime();
+ this._updateContinuousUpdates();
+
+ this._FBU.bytes = 0;
+ this._FBU.rects -= 1;
+ return true;
+ },
+
+ ExtendedDesktopSize: function () {
+ this._FBU.bytes = 1;
+ if (this._sock.rQwait("ExtendedDesktopSize", this._FBU.bytes)) { return false; }
+
+ this._supportsSetDesktopSize = true;
+ var number_of_screens = this._sock.rQpeek8();
+
+ this._FBU.bytes = 4 + (number_of_screens * 16);
+ if (this._sock.rQwait("ExtendedDesktopSize", this._FBU.bytes)) { return false; }
+
+ this._sock.rQskipBytes(1); // number-of-screens
+ this._sock.rQskipBytes(3); // padding
+
+ for (var i = 0; i < number_of_screens; i += 1) {
+ // Save the id and flags of the first screen
+ if (i === 0) {
+ this._screen_id = this._sock.rQshiftBytes(4); // id
+ this._sock.rQskipBytes(2); // x-position
+ this._sock.rQskipBytes(2); // y-position
+ this._sock.rQskipBytes(2); // width
+ this._sock.rQskipBytes(2); // height
+ this._screen_flags = this._sock.rQshiftBytes(4); // flags
+ } else {
+ this._sock.rQskipBytes(16);
}
+ }
- this._encHandlers.handle_FB_resize();
- return true;
- },
+ /*
+ * The x-position indicates the reason for the change:
+ *
+ * 0 - server resized on its own
+ * 1 - this client requested the resize
+ * 2 - another client requested the resize
+ */
- DesktopSize: function () {
- this._encHandlers.handle_FB_resize();
+ // We need to handle errors when we requested the resize.
+ if (this._FBU.x === 1 && this._FBU.y !== 0) {
+ var msg = "";
+ // The y-position indicates the status code from the server
+ switch (this._FBU.y) {
+ case 1:
+ msg = "Resize is administratively prohibited";
+ break;
+ case 2:
+ msg = "Out of resources";
+ break;
+ case 3:
+ msg = "Invalid screen layout";
+ break;
+ default:
+ msg = "Unknown reason";
+ break;
+ }
+ this._notification("Server did not accept the resize request: "
+ + msg, 'normal');
return true;
- },
+ }
- Cursor: function () {
- Util.Debug(">> set_cursor");
- var x = this._FBU.x; // hotspot-x
- var y = this._FBU.y; // hotspot-y
- var w = this._FBU.width;
- var h = this._FBU.height;
+ this._encHandlers.handle_FB_resize();
+ return true;
+ },
- var pixelslength = w * h * this._fb_Bpp;
- var masklength = Math.floor((w + 7) / 8) * h;
+ DesktopSize: function () {
+ this._encHandlers.handle_FB_resize();
+ return true;
+ },
- this._FBU.bytes = pixelslength + masklength;
- if (this._sock.rQwait("cursor encoding", this._FBU.bytes)) { return false; }
+ Cursor: function () {
+ Log.Debug(">> set_cursor");
+ var x = this._FBU.x; // hotspot-x
+ var y = this._FBU.y; // hotspot-y
+ var w = this._FBU.width;
+ var h = this._FBU.height;
- this._display.changeCursor(this._sock.rQshiftBytes(pixelslength),
- this._sock.rQshiftBytes(masklength),
- x, y, w, h);
+ var pixelslength = w * h * 4;
+ var masklength = Math.floor((w + 7) / 8) * h;
- this._FBU.bytes = 0;
- this._FBU.rects--;
+ this._FBU.bytes = pixelslength + masklength;
+ if (this._sock.rQwait("cursor encoding", this._FBU.bytes)) { return false; }
- Util.Debug("<< set_cursor");
- return true;
- },
+ this._display.changeCursor(this._sock.rQshiftBytes(pixelslength),
+ this._sock.rQshiftBytes(masklength),
+ x, y, w, h);
- QEMUExtendedKeyEvent: function () {
- this._FBU.rects--;
+ this._FBU.bytes = 0;
+ this._FBU.rects--;
+ Log.Debug("<< set_cursor");
+ return true;
+ },
+
+ QEMUExtendedKeyEvent: function () {
+ this._FBU.rects--;
+
+ // Old Safari doesn't support creating keyboard events
+ try {
var keyboardEvent = document.createEvent("keyboardEvent");
if (keyboardEvent.code !== undefined) {
this._qemuExtKeyEventSupported = true;
- this._keyboard.setQEMUVNCKeyboardHandler();
}
- },
+ } catch (err) {
+ }
+ },
- JPEG_quality_lo: function () {
- Util.Error("Server sent jpeg_quality pseudo-encoding");
- },
+ JPEG_quality_lo: function () {
+ Log.Error("Server sent jpeg_quality pseudo-encoding");
+ },
- compress_lo: function () {
- Util.Error("Server sent compress level pseudo-encoding");
- }
- };
-})();
+ compress_lo: function () {
+ Log.Error("Server sent compress level pseudo-encoding");
+ }
+};