ui: add barrier client.

ui: bugfixes for vnc & egl.
 -----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2.0.22 (GNU/Linux)
 
 iQIcBAABAgAGBQJdgymRAAoJEEy22O7T6HE4jaYQAMF32j4qv9E7vwkCoKVv/OdS
 Ztf80rd0dy7RYrJEjP7BgzZqdVXNjSR5g5Z8SqYV5C92RrUYhOUkXyHVBimdynZS
 bm1w4Nxj+ndI8R9OBsZlts0MQw0zfusVf890n0id5l/tw95nBjOKIeEhMG4bdHHe
 2F3z/y6InLUdMuShLMeZFfXqvUyXQtuyDzc2hob+gBre/Q0LQOVV1b6bTg3uYjg/
 SFRFQkwB5Q7ecH0PUIhFj9yuxbs8FsuxbBB+bkdXidYetIYiBF5GEWvAT6ddjkra
 RxcmFNeZmJDGHM4B4uMZ/fEf0yuFxwaow4BNeFkbeogH3Diy2zNz11Iz7TOiDHiU
 4g22pxRSwuaWG6BUlpTPmWW6QkdQ+5e/jQVcN2BECJliIdSGUiEw6+7iE3g7BDok
 258JEKILGpG6fqy97xnee2+crbUu+BQgEBlj9op9aHBZKCKAFQL1UMT8gegr+jn8
 u32TWJux0ZwEHCHE0tNXIB28UN3+FTw9JnG2TvXXgtr6IC8ov9gjkBYMmogxZRgl
 7FBpKpJGCb/HmEOxUEFCCXBGEYIfKROlfMTwlO7n2ygvh//Ut00B+bAjrqpW74uc
 edLZPllhYF9PKVj8+UpdbpmLt6Lr2snCU7Zk7JyvOYbatyBHTOtQ26CoRUGh7SS2
 7rHmRy3MriLWwzMNQDmI
 =/V1o
 -----END PGP SIGNATURE-----

Merge remote-tracking branch 'remotes/kraxel/tags/ui-20190919-pull-request' into staging

ui: add barrier client.
ui: bugfixes for vnc & egl.

# gpg: Signature made Thu 19 Sep 2019 08:09:05 BST
# gpg:                using RSA key 4CB6D8EED3E87138
# gpg: Good signature from "Gerd Hoffmann (work) <kraxel@redhat.com>" [full]
# gpg:                 aka "Gerd Hoffmann <gerd@kraxel.org>" [full]
# gpg:                 aka "Gerd Hoffmann (private) <kraxel@gmail.com>" [full]
# Primary key fingerprint: A032 8CFF B93A 17A7 9901  FE7D 4CB6 D8EE D3E8 7138

* remotes/kraxel/tags/ui-20190919-pull-request:
  vnc: fix memory leak when vnc disconnect
  ui: add an embedded Barrier client
  vnc: fix websocket field in events
  ui/egl: fix framebuffer reads

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2019-09-19 17:16:07 +01:00
commit a77d20bafc
13 changed files with 1409 additions and 169 deletions

370
docs/barrier.txt Normal file
View File

@ -0,0 +1,370 @@
QEMU Barrier Client
* About
Barrier is a KVM (Keyboard-Video-Mouse) software forked from Symless's
synergy 1.9 codebase.
See https://github.com/debauchee/barrier
* QEMU usage
Generally, mouse and keyboard are grabbed through the QEMU video
interface emulation.
But when we want to use a video graphic adapter via a PCI passthrough
there is no way to provide the keyboard and mouse inputs to the VM
except by plugging a second set of mouse and keyboard to the host
or by installing a KVM software in the guest OS.
The QEMU Barrier client avoids this by implementing directly the Barrier
protocol into QEMU.
This protocol is enabled by adding an input-barrier object to QEMU.
Syntax: input-barrier,id=<object-id>,name=<guest display name>
[,server=<barrier server address>][,port=<barrier server port>]
[,x-origin=<x-origin>][,y-origin=<y-origin>]
[,width=<width>][,height=<height>]
The object can be added on the QEMU command line, for instance with:
... -object input-barrier,id=barrier0,name=VM-1 ...
where VM-1 is the name the display configured int the Barrier server
on the host providing the mouse and the keyboard events.
by default <barrier server address> is "localhost", port is 24800,
<x-origin> and <y-origin> are set to 0, <width> and <height> to
1920 and 1080.
If Barrier server is stopped QEMU needs to be reconnected manually,
by removing and re-adding the input-barrier object, for instance
with the help of the HMP monitor:
(qemu) object_del barrier0
(qemu) object_add input-barrier,id=barrier0,name=VM-1
* Message format
Message format between the server and client is in two parts:
1- the payload length is a 32bit integer in network endianness,
2- the payload
The payload starts with a 4byte string (without NUL) which is the
command. The first command between the server and the client
is the only command not encoded on 4 bytes ("Barrier").
The remaining part of the payload is decoded according to the command.
* Protocol Description (from barrier/src/lib/barrier/protocol_types.h)
- barrierCmdHello "Barrier"
Direction: server -> client
Parameters: { int16_t minor, int16_t major }
Description:
Say hello to client
minor = protocol major version number supported by server
major = protocol minor version number supported by server
- barrierCmdHelloBack "Barrier"
Direction: client ->server
Parameters: { int16_t minor, int16_t major, char *name}
Description:
Respond to hello from server
minor = protocol major version number supported by client
major = protocol minor version number supported by client
name = client name
- barrierCmdDInfo "DINF"
Direction: client ->server
Parameters: { int16_t x_origin, int16_t y_origin, int16_t width, int16_t height, int16_t x, int16_t y}
Description:
The client screen must send this message in response to the
barrierCmdQInfo message. It must also send this message when the
screen's resolution changes. In this case, the client screen should
ignore any barrierCmdDMouseMove messages until it receives a
barrierCmdCInfoAck in order to prevent attempts to move the mouse off
the new screen area.
- barrierCmdCNoop "CNOP"
Direction: client -> server
Parameters: None
Description:
No operation
- barrierCmdCClose "CBYE"
Direction: server -> client
Parameters: None
Description:
Close connection
- barrierCmdCEnter "CINN"
Direction: server -> client
Parameters: { int16_t x, int16_t y, int32_t seq, int16_t modifier }
Description:
Enter screen.
x,y = entering screen absolute coordinates
seq = sequence number, which is used to order messages between
screens. the secondary screen must return this number
with some messages
modifier = modifier key mask. this will have bits set for each
toggle modifier key that is activated on entry to the
screen. the secondary screen should adjust its toggle
modifiers to reflect that state.
- barrierCmdCLeave "COUT"
Direction: server -> client
Parameters: None
Description:
Leaving screen. the secondary screen should send clipboard data in
response to this message for those clipboards that it has grabbed
(i.e. has sent a barrierCmdCClipboard for and has not received a
barrierCmdCClipboard for with a greater sequence number) and that
were grabbed or have changed since the last leave.
- barrierCmdCClipboard "CCLP"
Direction: server -> client
Parameters: { int8_t id, int32_t seq }
Description:
Grab clipboard. Sent by screen when some other app on that screen
grabs a clipboard.
id = the clipboard identifier
seq = sequence number. Client must use the sequence number passed in
the most recent barrierCmdCEnter. the server always sends 0.
- barrierCmdCScreenSaver "CSEC"
Direction: server -> client
Parameters: { int8_t started }
Description:
Screensaver change.
started = Screensaver on primary has started (1) or closed (0)
- barrierCmdCResetOptions "CROP"
Direction: server -> client
Parameters: None
Description:
Reset options. Client should reset all of its options to their
defaults.
- barrierCmdCInfoAck "CIAK"
Direction: server -> client
Parameters: None
Description:
Resolution change acknowledgment. Sent by server in response to a
client screen's barrierCmdDInfo. This is sent for every
barrierCmdDInfo, whether or not the server had sent a barrierCmdQInfo.
- barrierCmdCKeepAlive "CALV"
Direction: server -> client
Parameters: None
Description:
Keep connection alive. Sent by the server periodically to verify
that connections are still up and running. clients must reply in
kind on receipt. if the server gets an error sending the message or
does not receive a reply within a reasonable time then the server
disconnects the client. if the client doesn't receive these (or any
message) periodically then it should disconnect from the server. the
appropriate interval is defined by an option.
- barrierCmdDKeyDown "DKDN"
Direction: server -> client
Parameters: { int16_t keyid, int16_t modifier [,int16_t button] }
Description:
Key pressed.
keyid = X11 key id
modified = modified mask
button = X11 Xkb keycode (optional)
- barrierCmdDKeyRepeat "DKRP"
Direction: server -> client
Parameters: { int16_t keyid, int16_t modifier, int16_t repeat [,int16_t button] }
Description:
Key auto-repeat.
keyid = X11 key id
modified = modified mask
repeat = number of repeats
button = X11 Xkb keycode (optional)
- barrierCmdDKeyUp "DKUP"
Direction: server -> client
Parameters: { int16_t keyid, int16_t modifier [,int16_t button] }
Description:
Key released.
keyid = X11 key id
modified = modified mask
button = X11 Xkb keycode (optional)
- barrierCmdDMouseDown "DMDN"
Direction: server -> client
Parameters: { int8_t button }
Description:
Mouse button pressed.
button = button id
- barrierCmdDMouseUp "DMUP"
Direction: server -> client
Parameters: { int8_t button }
Description:
Mouse button release.
button = button id
- barrierCmdDMouseMove "DMMV"
Direction: server -> client
Parameters: { int16_t x, int16_t y }
Description:
Absolute mouse moved.
x,y = absolute screen coordinates
- barrierCmdDMouseRelMove "DMRM"
Direction: server -> client
Parameters: { int16_t x, int16_t y }
Description:
Relative mouse moved.
x,y = r relative screen coordinates
- barrierCmdDMouseWheel "DMWM"
Direction: server -> client
Parameters: { int16_t x , int16_t y } or { int16_t y }
Description:
Mouse scroll. The delta should be +120 for one tick forward (away
from the user) or right and -120 for one tick backward (toward the
user) or left.
x = x delta
y = y delta
- barrierCmdDClipboard "DCLP"
Direction: server -> client
Parameters: { int8_t id, int32_t seq, int8_t mark, char *data }
Description:
Clipboard data.
id = clipboard id
seq = sequence number. The sequence number is 0 when sent by the
server. Client screens should use the/ sequence number from
the most recent barrierCmdCEnter.
- barrierCmdDSetOptions "DSOP"
Direction: server -> client
Parameters: { int32 t nb, { int32_t id, int32_t val }[] }
Description:
Set options. Client should set the given option/value pairs.
nb = numbers of { id, val } entries
id = option id
val = option new value
- barrierCmdDFileTransfer "DFTR"
Direction: server -> client
Parameters: { int8_t mark, char *content }
Description:
Transfer file data.
mark = 0 means the content followed is the file size
1 means the content followed is the chunk data
2 means the file transfer is finished
- barrierCmdDDragInfo "DDRG" int16_t char *
Direction: server -> client
Parameters: { int16_t nb, char *content }
Description:
Drag information.
nb = number of dragging objects
content = object's directory
- barrierCmdQInfo "QINF"
Direction: server -> client
Parameters: None
Description:
Query screen info
Client should reply with a barrierCmdDInfo
- barrierCmdEIncompatible "EICV"
Direction: server -> client
Parameters: { int16_t nb, major *minor }
Description:
Incompatible version.
major = major version
minor = minor version
- barrierCmdEBusy "EBSY"
Direction: server -> client
Parameters: None
Description:
Name provided when connecting is already in use.
- barrierCmdEUnknown "EUNK"
Direction: server -> client
Parameters: None
Description:
Unknown client. Name provided when connecting is not in primary's
screen configuration map.
- barrierCmdEBad "EBAD"
Direction: server -> client
Parameters: None
Description:
Protocol violation. Server should disconnect after sending this
message.
* TO DO
- Enable SSL
- Manage SetOptions/ResetOptions commands

View File

@ -25,7 +25,7 @@ void egl_fb_setup_for_tex(egl_fb *fb, int width, int height,
GLuint texture, bool delete); GLuint texture, bool delete);
void egl_fb_setup_new_tex(egl_fb *fb, int width, int height); void egl_fb_setup_new_tex(egl_fb *fb, int width, int height);
void egl_fb_blit(egl_fb *dst, egl_fb *src, bool flip); void egl_fb_blit(egl_fb *dst, egl_fb *src, bool flip);
void egl_fb_read(void *dst, egl_fb *src); void egl_fb_read(DisplaySurface *dst, egl_fb *src);
void egl_texture_blit(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip); void egl_texture_blit(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip);
void egl_texture_blend(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip, void egl_texture_blend(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip,

View File

@ -9,6 +9,7 @@ vnc-obj-y += vnc-jobs.o
common-obj-y += keymaps.o console.o cursor.o qemu-pixman.o common-obj-y += keymaps.o console.o cursor.o qemu-pixman.o
common-obj-y += input.o input-keymap.o input-legacy.o kbd-state.o common-obj-y += input.o input-keymap.o input-legacy.o kbd-state.o
common-obj-y += input-barrier.o
common-obj-$(CONFIG_LINUX) += input-linux.o common-obj-$(CONFIG_LINUX) += input-linux.o
common-obj-$(CONFIG_SPICE) += spice-core.o spice-input.o spice-display.o common-obj-$(CONFIG_SPICE) += spice-core.o spice-input.o spice-display.o
common-obj-$(CONFIG_COCOA) += cocoa.o common-obj-$(CONFIG_COCOA) += cocoa.o

View File

@ -133,8 +133,6 @@ static void egl_scanout_flush(DisplayChangeListener *dcl,
if (!edpy->guest_fb.texture || !edpy->ds) { if (!edpy->guest_fb.texture || !edpy->ds) {
return; return;
} }
assert(surface_width(edpy->ds) == edpy->guest_fb.width);
assert(surface_height(edpy->ds) == edpy->guest_fb.height);
assert(surface_format(edpy->ds) == PIXMAN_x8r8g8b8); assert(surface_format(edpy->ds) == PIXMAN_x8r8g8b8);
if (edpy->cursor_fb.texture) { if (edpy->cursor_fb.texture) {
@ -149,7 +147,7 @@ static void egl_scanout_flush(DisplayChangeListener *dcl,
egl_fb_blit(&edpy->blit_fb, &edpy->guest_fb, edpy->y_0_top); egl_fb_blit(&edpy->blit_fb, &edpy->guest_fb, edpy->y_0_top);
} }
egl_fb_read(surface_data(edpy->ds), &edpy->blit_fb); egl_fb_read(edpy->ds, &edpy->blit_fb);
dpy_gfx_update(edpy->dcl.con, x, y, w, h); dpy_gfx_update(edpy->dcl.con, x, y, w, h);
} }

View File

@ -102,12 +102,12 @@ void egl_fb_blit(egl_fb *dst, egl_fb *src, bool flip)
GL_COLOR_BUFFER_BIT, GL_LINEAR); GL_COLOR_BUFFER_BIT, GL_LINEAR);
} }
void egl_fb_read(void *dst, egl_fb *src) void egl_fb_read(DisplaySurface *dst, egl_fb *src)
{ {
glBindFramebuffer(GL_READ_FRAMEBUFFER, src->framebuffer); glBindFramebuffer(GL_READ_FRAMEBUFFER, src->framebuffer);
glReadBuffer(GL_COLOR_ATTACHMENT0_EXT); glReadBuffer(GL_COLOR_ATTACHMENT0_EXT);
glReadPixels(0, 0, src->width, src->height, glReadPixels(0, 0, surface_width(dst), surface_height(dst),
GL_BGRA, GL_UNSIGNED_BYTE, dst); GL_BGRA, GL_UNSIGNED_BYTE, surface_data(dst));
} }
void egl_texture_blit(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip) void egl_texture_blit(QemuGLShader *gls, egl_fb *dst, egl_fb *src, bool flip)

750
ui/input-barrier.c Normal file
View File

@ -0,0 +1,750 @@
/*
* SPDX-License-Identifier: GPL-2.0-or-later
*
* This work is licensed under the terms of the GNU GPL, version 2 or later.
* See the COPYING file in the top-level directory.
*/
#include "qemu/osdep.h"
#include "sysemu/sysemu.h"
#include "qemu/main-loop.h"
#include "qemu/sockets.h"
#include "qapi/error.h"
#include "qom/object_interfaces.h"
#include "io/channel-socket.h"
#include "ui/input.h"
#include "ui/vnc_keysym.h" /* use name2keysym from VNC as we use X11 values */
#include "qemu/cutils.h"
#include "qapi/qmp/qerror.h"
#include "input-barrier.h"
#define TYPE_INPUT_BARRIER "input-barrier"
#define INPUT_BARRIER(obj) \
OBJECT_CHECK(InputBarrier, (obj), TYPE_INPUT_BARRIER)
#define INPUT_BARRIER_GET_CLASS(obj) \
OBJECT_GET_CLASS(InputBarrierClass, (obj), TYPE_INPUT_BARRIER)
#define INPUT_BARRIER_CLASS(klass) \
OBJECT_CLASS_CHECK(InputBarrierClass, (klass), TYPE_INPUT_BARRIER)
typedef struct InputBarrier InputBarrier;
typedef struct InputBarrierClass InputBarrierClass;
#define MAX_HELLO_LENGTH 1024
struct InputBarrier {
Object parent;
QIOChannelSocket *sioc;
guint ioc_tag;
/* display properties */
gchar *name;
int16_t x_origin, y_origin;
int16_t width, height;
/* keyboard/mouse server */
SocketAddress saddr;
char buffer[MAX_HELLO_LENGTH];
};
struct InputBarrierClass {
ObjectClass parent_class;
};
static const char *cmd_names[] = {
[barrierCmdCNoop] = "CNOP",
[barrierCmdCClose] = "CBYE",
[barrierCmdCEnter] = "CINN",
[barrierCmdCLeave] = "COUT",
[barrierCmdCClipboard] = "CCLP",
[barrierCmdCScreenSaver] = "CSEC",
[barrierCmdCResetOptions] = "CROP",
[barrierCmdCInfoAck] = "CIAK",
[barrierCmdCKeepAlive] = "CALV",
[barrierCmdDKeyDown] = "DKDN",
[barrierCmdDKeyRepeat] = "DKRP",
[barrierCmdDKeyUp] = "DKUP",
[barrierCmdDMouseDown] = "DMDN",
[barrierCmdDMouseUp] = "DMUP",
[barrierCmdDMouseMove] = "DMMV",
[barrierCmdDMouseRelMove] = "DMRM",
[barrierCmdDMouseWheel] = "DMWM",
[barrierCmdDClipboard] = "DCLP",
[barrierCmdDInfo] = "DINF",
[barrierCmdDSetOptions] = "DSOP",
[barrierCmdDFileTransfer] = "DFTR",
[barrierCmdDDragInfo] = "DDRG",
[barrierCmdQInfo] = "QINF",
[barrierCmdEIncompatible] = "EICV",
[barrierCmdEBusy] = "EBSY",
[barrierCmdEUnknown] = "EUNK",
[barrierCmdEBad] = "EBAD",
[barrierCmdHello] = "Barrier",
[barrierCmdHelloBack] = "Barrier",
};
static kbd_layout_t *kbd_layout;
static int input_barrier_to_qcode(uint16_t keyid, uint16_t keycode)
{
/* keycode is optional, if it is not provided use keyid */
if (keycode && keycode <= qemu_input_map_xorgkbd_to_qcode_len) {
return qemu_input_map_xorgkbd_to_qcode[keycode];
}
if (keyid >= 0xE000 && keyid <= 0xEFFF) {
keyid += 0x1000;
}
/* keyid is the X11 key id */
if (kbd_layout) {
keycode = keysym2scancode(kbd_layout, keyid, NULL, false);
return qemu_input_key_number_to_qcode(keycode);
}
return qemu_input_map_x11_to_qcode[keyid];
}
static int input_barrier_to_mouse(uint8_t buttonid)
{
switch (buttonid) {
case barrierButtonLeft:
return INPUT_BUTTON_LEFT;
case barrierButtonMiddle:
return INPUT_BUTTON_MIDDLE;
case barrierButtonRight:
return INPUT_BUTTON_RIGHT;
case barrierButtonExtra0:
return INPUT_BUTTON_SIDE;
}
return buttonid;
}
#define read_char(x, p, l) \
do { \
int size = sizeof(char); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
x = *(char *)p; \
p += size; \
l -= size; \
} while (0)
#define read_short(x, p, l) \
do { \
int size = sizeof(short); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
x = ntohs(*(short *)p); \
p += size; \
l -= size; \
} while (0)
#define write_short(p, x, l) \
do { \
int size = sizeof(short); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
*(short *)p = htons(x); \
p += size; \
l -= size; \
} while (0)
#define read_int(x, p, l) \
do { \
int size = sizeof(int); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
x = ntohl(*(int *)p); \
p += size; \
l -= size; \
} while (0)
#define write_int(p, x, l) \
do { \
int size = sizeof(int); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
*(int *)p = htonl(x); \
p += size; \
l -= size; \
} while (0)
#define write_cmd(p, c, l) \
do { \
int size = strlen(cmd_names[c]); \
if (l < size) { \
return G_SOURCE_REMOVE; \
} \
memcpy(p, cmd_names[c], size); \
p += size; \
l -= size; \
} while (0)
#define write_string(p, s, l) \
do { \
int size = strlen(s); \
if (l < size + sizeof(int)) { \
return G_SOURCE_REMOVE; \
} \
*(int *)p = htonl(size); \
p += sizeof(size); \
l -= sizeof(size); \
memcpy(p, s, size); \
p += size; \
l -= size; \
} while (0)
static gboolean readcmd(InputBarrier *ib, struct barrierMsg *msg)
{
int ret, len, i;
enum barrierCmd cmd;
char *p;
ret = qio_channel_read(QIO_CHANNEL(ib->sioc), (char *)&len, sizeof(len),
NULL);
if (ret < 0) {
return G_SOURCE_REMOVE;
}
len = ntohl(len);
if (len > MAX_HELLO_LENGTH) {
return G_SOURCE_REMOVE;
}
ret = qio_channel_read(QIO_CHANNEL(ib->sioc), ib->buffer, len, NULL);
if (ret < 0) {
return G_SOURCE_REMOVE;
}
p = ib->buffer;
if (len >= strlen(cmd_names[barrierCmdHello]) &&
memcmp(p, cmd_names[barrierCmdHello],
strlen(cmd_names[barrierCmdHello])) == 0) {
cmd = barrierCmdHello;
p += strlen(cmd_names[barrierCmdHello]);
len -= strlen(cmd_names[barrierCmdHello]);
} else {
for (cmd = 0; cmd < barrierCmdHello; cmd++) {
if (memcmp(ib->buffer, cmd_names[cmd], 4) == 0) {
break;
}
}
if (cmd == barrierCmdHello) {
return G_SOURCE_REMOVE;
}
p += 4;
len -= 4;
}
msg->cmd = cmd;
switch (cmd) {
/* connection */
case barrierCmdHello:
read_short(msg->version.major, p, len);
read_short(msg->version.minor, p, len);
break;
case barrierCmdDSetOptions:
read_int(msg->set.nb, p, len);
msg->set.nb /= 2;
if (msg->set.nb > BARRIER_MAX_OPTIONS) {
msg->set.nb = BARRIER_MAX_OPTIONS;
}
i = 0;
while (len && i < msg->set.nb) {
read_int(msg->set.option[i].id, p, len);
/* it's a string, restore endianness */
msg->set.option[i].id = htonl(msg->set.option[i].id);
msg->set.option[i].nul = 0;
read_int(msg->set.option[i].value, p, len);
i++;
}
break;
case barrierCmdQInfo:
break;
/* mouse */
case barrierCmdDMouseMove:
case barrierCmdDMouseRelMove:
read_short(msg->mousepos.x, p, len);
read_short(msg->mousepos.y, p, len);
break;
case barrierCmdDMouseDown:
case barrierCmdDMouseUp:
read_char(msg->mousebutton.buttonid, p, len);
break;
case barrierCmdDMouseWheel:
read_short(msg->mousepos.y, p, len);
msg->mousepos.x = 0;
if (len) {
msg->mousepos.x = msg->mousepos.y;
read_short(msg->mousepos.y, p, len);
}
break;
/* keyboard */
case barrierCmdDKeyDown:
case barrierCmdDKeyUp:
read_short(msg->key.keyid, p, len);
read_short(msg->key.modifier, p, len);
msg->key.button = 0;
if (len) {
read_short(msg->key.button, p, len);
}
break;
case barrierCmdDKeyRepeat:
read_short(msg->repeat.keyid, p, len);
read_short(msg->repeat.modifier, p, len);
read_short(msg->repeat.repeat, p, len);
msg->repeat.button = 0;
if (len) {
read_short(msg->repeat.button, p, len);
}
break;
case barrierCmdCInfoAck:
case barrierCmdCResetOptions:
case barrierCmdCEnter:
case barrierCmdDClipboard:
case barrierCmdCKeepAlive:
case barrierCmdCLeave:
case barrierCmdCClose:
break;
/* Invalid from the server */
case barrierCmdHelloBack:
case barrierCmdCNoop:
case barrierCmdDInfo:
break;
/* Error codes */
case barrierCmdEIncompatible:
read_short(msg->version.major, p, len);
read_short(msg->version.minor, p, len);
break;
case barrierCmdEBusy:
case barrierCmdEUnknown:
case barrierCmdEBad:
break;
default:
return G_SOURCE_REMOVE;
}
return G_SOURCE_CONTINUE;
}
static gboolean writecmd(InputBarrier *ib, struct barrierMsg *msg)
{
char *p;
int ret, i;
int avail, len;
p = ib->buffer;
avail = MAX_HELLO_LENGTH;
/* reserve space to store the length */
p += sizeof(int);
avail -= sizeof(int);
switch (msg->cmd) {
case barrierCmdHello:
if (msg->version.major < BARRIER_VERSION_MAJOR ||
(msg->version.major == BARRIER_VERSION_MAJOR &&
msg->version.minor < BARRIER_VERSION_MINOR)) {
ib->ioc_tag = 0;
return G_SOURCE_REMOVE;
}
write_cmd(p, barrierCmdHelloBack, avail);
write_short(p, BARRIER_VERSION_MAJOR, avail);
write_short(p, BARRIER_VERSION_MINOR, avail);
write_string(p, ib->name, avail);
break;
case barrierCmdCClose:
ib->ioc_tag = 0;
return G_SOURCE_REMOVE;
case barrierCmdQInfo:
write_cmd(p, barrierCmdDInfo, avail);
write_short(p, ib->x_origin, avail);
write_short(p, ib->y_origin, avail);
write_short(p, ib->width, avail);
write_short(p, ib->height, avail);
write_short(p, 0, avail); /* warpsize (obsolete) */
write_short(p, 0, avail); /* mouse x */
write_short(p, 0, avail); /* mouse y */
break;
case barrierCmdCInfoAck:
break;
case barrierCmdCResetOptions:
/* TODO: reset options */
break;
case barrierCmdDSetOptions:
/* TODO: set options */
break;
case barrierCmdCEnter:
break;
case barrierCmdDClipboard:
break;
case barrierCmdCKeepAlive:
write_cmd(p, barrierCmdCKeepAlive, avail);
break;
case barrierCmdCLeave:
break;
/* mouse */
case barrierCmdDMouseMove:
qemu_input_queue_abs(NULL, INPUT_AXIS_X, msg->mousepos.x,
ib->x_origin, ib->width);
qemu_input_queue_abs(NULL, INPUT_AXIS_Y, msg->mousepos.y,
ib->y_origin, ib->height);
qemu_input_event_sync();
break;
case barrierCmdDMouseRelMove:
qemu_input_queue_rel(NULL, INPUT_AXIS_X, msg->mousepos.x);
qemu_input_queue_rel(NULL, INPUT_AXIS_Y, msg->mousepos.y);
qemu_input_event_sync();
break;
case barrierCmdDMouseDown:
qemu_input_queue_btn(NULL,
input_barrier_to_mouse(msg->mousebutton.buttonid),
true);
qemu_input_event_sync();
break;
case barrierCmdDMouseUp:
qemu_input_queue_btn(NULL,
input_barrier_to_mouse(msg->mousebutton.buttonid),
false);
qemu_input_event_sync();
break;
case barrierCmdDMouseWheel:
qemu_input_queue_btn(NULL, (msg->mousepos.y > 0) ? INPUT_BUTTON_WHEEL_UP
: INPUT_BUTTON_WHEEL_DOWN, true);
qemu_input_event_sync();
qemu_input_queue_btn(NULL, (msg->mousepos.y > 0) ? INPUT_BUTTON_WHEEL_UP
: INPUT_BUTTON_WHEEL_DOWN, false);
qemu_input_event_sync();
break;
/* keyboard */
case barrierCmdDKeyDown:
qemu_input_event_send_key_qcode(NULL,
input_barrier_to_qcode(msg->key.keyid, msg->key.button),
true);
break;
case barrierCmdDKeyRepeat:
for (i = 0; i < msg->repeat.repeat; i++) {
qemu_input_event_send_key_qcode(NULL,
input_barrier_to_qcode(msg->repeat.keyid, msg->repeat.button),
false);
qemu_input_event_send_key_qcode(NULL,
input_barrier_to_qcode(msg->repeat.keyid, msg->repeat.button),
true);
}
break;
case barrierCmdDKeyUp:
qemu_input_event_send_key_qcode(NULL,
input_barrier_to_qcode(msg->key.keyid, msg->key.button),
false);
break;
default:
write_cmd(p, barrierCmdEUnknown, avail);
break;;
}
len = MAX_HELLO_LENGTH - avail - sizeof(int);
if (len) {
p = ib->buffer;
avail = sizeof(len);
write_int(p, len, avail);
ret = qio_channel_write(QIO_CHANNEL(ib->sioc), ib->buffer,
len + sizeof(len), NULL);
if (ret < 0) {
ib->ioc_tag = 0;
return G_SOURCE_REMOVE;
}
}
return G_SOURCE_CONTINUE;
}
static gboolean input_barrier_event(QIOChannel *ioc G_GNUC_UNUSED,
GIOCondition condition, void *opaque)
{
InputBarrier *ib = opaque;
int ret;
struct barrierMsg msg;
ret = readcmd(ib, &msg);
if (ret == G_SOURCE_REMOVE) {
ib->ioc_tag = 0;
return G_SOURCE_REMOVE;
}
return writecmd(ib, &msg);
}
static void input_barrier_complete(UserCreatable *uc, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(uc);
Error *local_err = NULL;
if (!ib->name) {
error_setg(errp, QERR_MISSING_PARAMETER, "name");
return;
}
/*
* Connect to the primary
* Primary is the server where the keyboard and the mouse
* are connected and forwarded to the secondary (the client)
*/
ib->sioc = qio_channel_socket_new();
qio_channel_set_name(QIO_CHANNEL(ib->sioc), "barrier-client");
qio_channel_socket_connect_sync(ib->sioc, &ib->saddr, &local_err);
if (local_err) {
error_propagate(errp, local_err);
return;
}
qio_channel_set_delay(QIO_CHANNEL(ib->sioc), false);
ib->ioc_tag = qio_channel_add_watch(QIO_CHANNEL(ib->sioc), G_IO_IN,
input_barrier_event, ib, NULL);
}
static void input_barrier_instance_finalize(Object *obj)
{
InputBarrier *ib = INPUT_BARRIER(obj);
if (ib->ioc_tag) {
g_source_remove(ib->ioc_tag);
ib->ioc_tag = 0;
}
if (ib->sioc) {
qio_channel_close(QIO_CHANNEL(ib->sioc), NULL);
object_unref(OBJECT(ib->sioc));
}
g_free(ib->name);
g_free(ib->saddr.u.inet.host);
g_free(ib->saddr.u.inet.port);
}
static char *input_barrier_get_name(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup(ib->name);
}
static void input_barrier_set_name(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
if (ib->name) {
error_setg(errp, "name property already set");
return;
}
ib->name = g_strdup(value);
}
static char *input_barrier_get_server(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup(ib->saddr.u.inet.host);
}
static void input_barrier_set_server(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
g_free(ib->saddr.u.inet.host);
ib->saddr.u.inet.host = g_strdup(value);
}
static char *input_barrier_get_port(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup(ib->saddr.u.inet.port);
}
static void input_barrier_set_port(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
g_free(ib->saddr.u.inet.port);
ib->saddr.u.inet.port = g_strdup(value);
}
static void input_barrier_set_x_origin(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
int result, err;
err = qemu_strtoi(value, NULL, 0, &result);
if (err < 0 || result < 0 || result > SHRT_MAX) {
error_setg(errp,
"x-origin property must be in the range [0..%d]", SHRT_MAX);
return;
}
ib->x_origin = result;
}
static char *input_barrier_get_x_origin(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup_printf("%d", ib->x_origin);
}
static void input_barrier_set_y_origin(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
int result, err;
err = qemu_strtoi(value, NULL, 0, &result);
if (err < 0 || result < 0 || result > SHRT_MAX) {
error_setg(errp,
"y-origin property must be in the range [0..%d]", SHRT_MAX);
return;
}
ib->y_origin = result;
}
static char *input_barrier_get_y_origin(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup_printf("%d", ib->y_origin);
}
static void input_barrier_set_width(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
int result, err;
err = qemu_strtoi(value, NULL, 0, &result);
if (err < 0 || result < 0 || result > SHRT_MAX) {
error_setg(errp,
"width property must be in the range [0..%d]", SHRT_MAX);
return;
}
ib->width = result;
}
static char *input_barrier_get_width(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup_printf("%d", ib->width);
}
static void input_barrier_set_height(Object *obj, const char *value,
Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
int result, err;
err = qemu_strtoi(value, NULL, 0, &result);
if (err < 0 || result < 0 || result > SHRT_MAX) {
error_setg(errp,
"height property must be in the range [0..%d]", SHRT_MAX);
return;
}
ib->height = result;
}
static char *input_barrier_get_height(Object *obj, Error **errp)
{
InputBarrier *ib = INPUT_BARRIER(obj);
return g_strdup_printf("%d", ib->height);
}
static void input_barrier_instance_init(Object *obj)
{
InputBarrier *ib = INPUT_BARRIER(obj);
ib->saddr.type = SOCKET_ADDRESS_TYPE_INET;
ib->saddr.u.inet.host = g_strdup("localhost");
ib->saddr.u.inet.port = g_strdup("24800");
ib->x_origin = 0;
ib->y_origin = 0;
ib->width = 1920;
ib->height = 1080;
object_property_add_str(obj, "name",
input_barrier_get_name,
input_barrier_set_name, NULL);
object_property_add_str(obj, "server",
input_barrier_get_server,
input_barrier_set_server, NULL);
object_property_add_str(obj, "port",
input_barrier_get_port,
input_barrier_set_port, NULL);
object_property_add_str(obj, "x-origin",
input_barrier_get_x_origin,
input_barrier_set_x_origin, NULL);
object_property_add_str(obj, "y-origin",
input_barrier_get_y_origin,
input_barrier_set_y_origin, NULL);
object_property_add_str(obj, "width",
input_barrier_get_width,
input_barrier_set_width, NULL);
object_property_add_str(obj, "height",
input_barrier_get_height,
input_barrier_set_height, NULL);
}
static void input_barrier_class_init(ObjectClass *oc, void *data)
{
UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
ucc->complete = input_barrier_complete;
/* always use generic keymaps */
if (keyboard_layout) {
/* We use X11 key id, so use VNC name2keysym */
kbd_layout = init_keyboard_layout(name2keysym, keyboard_layout,
&error_fatal);
}
}
static const TypeInfo input_barrier_info = {
.name = TYPE_INPUT_BARRIER,
.parent = TYPE_OBJECT,
.class_size = sizeof(InputBarrierClass),
.class_init = input_barrier_class_init,
.instance_size = sizeof(InputBarrier),
.instance_init = input_barrier_instance_init,
.instance_finalize = input_barrier_instance_finalize,
.interfaces = (InterfaceInfo[]) {
{ TYPE_USER_CREATABLE },
{ }
}
};
static void register_types(void)
{
type_register_static(&input_barrier_info);
}
type_init(register_types);

112
ui/input-barrier.h Normal file
View File

@ -0,0 +1,112 @@
/*
* SPDX-License-Identifier: GPL-2.0-or-later
*
* This work is licensed under the terms of the GNU GPL, version 2 or later.
* See the COPYING file in the top-level directory.
*/
#ifndef UI_INPUT_BARRIER_H
#define UI_INPUT_BARRIER_H
/* Barrier protocol */
#define BARRIER_VERSION_MAJOR 1
#define BARRIER_VERSION_MINOR 6
enum barrierCmd {
barrierCmdCNoop,
barrierCmdCClose,
barrierCmdCEnter,
barrierCmdCLeave,
barrierCmdCClipboard,
barrierCmdCScreenSaver,
barrierCmdCResetOptions,
barrierCmdCInfoAck,
barrierCmdCKeepAlive,
barrierCmdDKeyDown,
barrierCmdDKeyRepeat,
barrierCmdDKeyUp,
barrierCmdDMouseDown,
barrierCmdDMouseUp,
barrierCmdDMouseMove,
barrierCmdDMouseRelMove,
barrierCmdDMouseWheel,
barrierCmdDClipboard,
barrierCmdDInfo,
barrierCmdDSetOptions,
barrierCmdDFileTransfer,
barrierCmdDDragInfo,
barrierCmdQInfo,
barrierCmdEIncompatible,
barrierCmdEBusy,
barrierCmdEUnknown,
barrierCmdEBad,
/* connection sequence */
barrierCmdHello,
barrierCmdHelloBack,
};
enum {
barrierButtonNone,
barrierButtonLeft,
barrierButtonMiddle,
barrierButtonRight,
barrierButtonExtra0
};
struct barrierVersion {
int16_t major;
int16_t minor;
};
struct barrierMouseButton {
int8_t buttonid;
};
struct barrierEnter {
int16_t x;
int16_t y;
int32_t seqn;
int16_t modifier;
};
struct barrierMousePos {
int16_t x;
int16_t y;
};
struct barrierKey {
int16_t keyid;
int16_t modifier;
int16_t button;
};
struct barrierRepeat {
int16_t keyid;
int16_t modifier;
int16_t repeat;
int16_t button;
};
#define BARRIER_MAX_OPTIONS 32
struct barrierSet {
int nb;
struct {
int id;
char nul;
int value;
} option[BARRIER_MAX_OPTIONS];
};
struct barrierMsg {
enum barrierCmd cmd;
union {
struct barrierVersion version;
struct barrierMouseButton mousebutton;
struct barrierMousePos mousepos;
struct barrierEnter enter;
struct barrierKey key;
struct barrierRepeat repeat;
struct barrierSet set;
};
};
#endif

View File

@ -116,7 +116,7 @@ static int send_png_rect(VncState *vs, int x, int y, int w, int h,
static bool tight_can_send_png_rect(VncState *vs, int w, int h) static bool tight_can_send_png_rect(VncState *vs, int w, int h)
{ {
if (vs->tight.type != VNC_ENCODING_TIGHT_PNG) { if (vs->tight->type != VNC_ENCODING_TIGHT_PNG) {
return false; return false;
} }
@ -144,7 +144,7 @@ tight_detect_smooth_image24(VncState *vs, int w, int h)
int pixels = 0; int pixels = 0;
int pix, left[3]; int pix, left[3];
unsigned int errors; unsigned int errors;
unsigned char *buf = vs->tight.tight.buffer; unsigned char *buf = vs->tight->tight.buffer;
/* /*
* If client is big-endian, color samples begin from the second * If client is big-endian, color samples begin from the second
@ -215,7 +215,7 @@ tight_detect_smooth_image24(VncState *vs, int w, int h)
int pixels = 0; \ int pixels = 0; \
int sample, sum, left[3]; \ int sample, sum, left[3]; \
unsigned int errors; \ unsigned int errors; \
unsigned char *buf = vs->tight.tight.buffer; \ unsigned char *buf = vs->tight->tight.buffer; \
\ \
endian = 0; /* FIXME */ \ endian = 0; /* FIXME */ \
\ \
@ -296,8 +296,8 @@ static int
tight_detect_smooth_image(VncState *vs, int w, int h) tight_detect_smooth_image(VncState *vs, int w, int h)
{ {
unsigned int errors; unsigned int errors;
int compression = vs->tight.compression; int compression = vs->tight->compression;
int quality = vs->tight.quality; int quality = vs->tight->quality;
if (!vs->vd->lossy) { if (!vs->vd->lossy) {
return 0; return 0;
@ -309,7 +309,7 @@ tight_detect_smooth_image(VncState *vs, int w, int h)
return 0; return 0;
} }
if (vs->tight.quality != (uint8_t)-1) { if (vs->tight->quality != (uint8_t)-1) {
if (w * h < VNC_TIGHT_JPEG_MIN_RECT_SIZE) { if (w * h < VNC_TIGHT_JPEG_MIN_RECT_SIZE) {
return 0; return 0;
} }
@ -320,9 +320,9 @@ tight_detect_smooth_image(VncState *vs, int w, int h)
} }
if (vs->client_pf.bytes_per_pixel == 4) { if (vs->client_pf.bytes_per_pixel == 4) {
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
errors = tight_detect_smooth_image24(vs, w, h); errors = tight_detect_smooth_image24(vs, w, h);
if (vs->tight.quality != (uint8_t)-1) { if (vs->tight->quality != (uint8_t)-1) {
return (errors < tight_conf[quality].jpeg_threshold24); return (errors < tight_conf[quality].jpeg_threshold24);
} }
return (errors < tight_conf[compression].gradient_threshold24); return (errors < tight_conf[compression].gradient_threshold24);
@ -352,7 +352,7 @@ tight_detect_smooth_image(VncState *vs, int w, int h)
uint##bpp##_t c0, c1, ci; \ uint##bpp##_t c0, c1, ci; \
int i, n0, n1; \ int i, n0, n1; \
\ \
data = (uint##bpp##_t *)vs->tight.tight.buffer; \ data = (uint##bpp##_t *)vs->tight->tight.buffer; \
\ \
c0 = data[0]; \ c0 = data[0]; \
i = 1; \ i = 1; \
@ -423,9 +423,9 @@ static int tight_fill_palette(VncState *vs, int x, int y,
{ {
int max; int max;
max = count / tight_conf[vs->tight.compression].idx_max_colors_divisor; max = count / tight_conf[vs->tight->compression].idx_max_colors_divisor;
if (max < 2 && if (max < 2 &&
count >= tight_conf[vs->tight.compression].mono_min_rect_size) { count >= tight_conf[vs->tight->compression].mono_min_rect_size) {
max = 2; max = 2;
} }
if (max >= 256) { if (max >= 256) {
@ -558,7 +558,7 @@ tight_filter_gradient24(VncState *vs, uint8_t *buf, int w, int h)
int x, y, c; int x, y, c;
buf32 = (uint32_t *)buf; buf32 = (uint32_t *)buf;
memset(vs->tight.gradient.buffer, 0, w * 3 * sizeof(int)); memset(vs->tight->gradient.buffer, 0, w * 3 * sizeof(int));
if (1 /* FIXME */) { if (1 /* FIXME */) {
shift[0] = vs->client_pf.rshift; shift[0] = vs->client_pf.rshift;
@ -575,7 +575,7 @@ tight_filter_gradient24(VncState *vs, uint8_t *buf, int w, int h)
upper[c] = 0; upper[c] = 0;
here[c] = 0; here[c] = 0;
} }
prev = (int *)vs->tight.gradient.buffer; prev = (int *)vs->tight->gradient.buffer;
for (x = 0; x < w; x++) { for (x = 0; x < w; x++) {
pix32 = *buf32++; pix32 = *buf32++;
for (c = 0; c < 3; c++) { for (c = 0; c < 3; c++) {
@ -615,7 +615,7 @@ tight_filter_gradient24(VncState *vs, uint8_t *buf, int w, int h)
int prediction; \ int prediction; \
int x, y, c; \ int x, y, c; \
\ \
memset (vs->tight.gradient.buffer, 0, w * 3 * sizeof(int)); \ memset(vs->tight->gradient.buffer, 0, w * 3 * sizeof(int)); \
\ \
endian = 0; /* FIXME */ \ endian = 0; /* FIXME */ \
\ \
@ -631,7 +631,7 @@ tight_filter_gradient24(VncState *vs, uint8_t *buf, int w, int h)
upper[c] = 0; \ upper[c] = 0; \
here[c] = 0; \ here[c] = 0; \
} \ } \
prev = (int *)vs->tight.gradient.buffer; \ prev = (int *)vs->tight->gradient.buffer; \
for (x = 0; x < w; x++) { \ for (x = 0; x < w; x++) { \
pix = *buf; \ pix = *buf; \
if (endian) { \ if (endian) { \
@ -785,7 +785,7 @@ static void extend_solid_area(VncState *vs, int x, int y, int w, int h,
static int tight_init_stream(VncState *vs, int stream_id, static int tight_init_stream(VncState *vs, int stream_id,
int level, int strategy) int level, int strategy)
{ {
z_streamp zstream = &vs->tight.stream[stream_id]; z_streamp zstream = &vs->tight->stream[stream_id];
if (zstream->opaque == NULL) { if (zstream->opaque == NULL) {
int err; int err;
@ -803,15 +803,15 @@ static int tight_init_stream(VncState *vs, int stream_id,
return -1; return -1;
} }
vs->tight.levels[stream_id] = level; vs->tight->levels[stream_id] = level;
zstream->opaque = vs; zstream->opaque = vs;
} }
if (vs->tight.levels[stream_id] != level) { if (vs->tight->levels[stream_id] != level) {
if (deflateParams(zstream, level, strategy) != Z_OK) { if (deflateParams(zstream, level, strategy) != Z_OK) {
return -1; return -1;
} }
vs->tight.levels[stream_id] = level; vs->tight->levels[stream_id] = level;
} }
return 0; return 0;
} }
@ -839,11 +839,11 @@ static void tight_send_compact_size(VncState *vs, size_t len)
static int tight_compress_data(VncState *vs, int stream_id, size_t bytes, static int tight_compress_data(VncState *vs, int stream_id, size_t bytes,
int level, int strategy) int level, int strategy)
{ {
z_streamp zstream = &vs->tight.stream[stream_id]; z_streamp zstream = &vs->tight->stream[stream_id];
int previous_out; int previous_out;
if (bytes < VNC_TIGHT_MIN_TO_COMPRESS) { if (bytes < VNC_TIGHT_MIN_TO_COMPRESS) {
vnc_write(vs, vs->tight.tight.buffer, vs->tight.tight.offset); vnc_write(vs, vs->tight->tight.buffer, vs->tight->tight.offset);
return bytes; return bytes;
} }
@ -852,13 +852,13 @@ static int tight_compress_data(VncState *vs, int stream_id, size_t bytes,
} }
/* reserve memory in output buffer */ /* reserve memory in output buffer */
buffer_reserve(&vs->tight.zlib, bytes + 64); buffer_reserve(&vs->tight->zlib, bytes + 64);
/* set pointers */ /* set pointers */
zstream->next_in = vs->tight.tight.buffer; zstream->next_in = vs->tight->tight.buffer;
zstream->avail_in = vs->tight.tight.offset; zstream->avail_in = vs->tight->tight.offset;
zstream->next_out = vs->tight.zlib.buffer + vs->tight.zlib.offset; zstream->next_out = vs->tight->zlib.buffer + vs->tight->zlib.offset;
zstream->avail_out = vs->tight.zlib.capacity - vs->tight.zlib.offset; zstream->avail_out = vs->tight->zlib.capacity - vs->tight->zlib.offset;
previous_out = zstream->avail_out; previous_out = zstream->avail_out;
zstream->data_type = Z_BINARY; zstream->data_type = Z_BINARY;
@ -868,14 +868,14 @@ static int tight_compress_data(VncState *vs, int stream_id, size_t bytes,
return -1; return -1;
} }
vs->tight.zlib.offset = vs->tight.zlib.capacity - zstream->avail_out; vs->tight->zlib.offset = vs->tight->zlib.capacity - zstream->avail_out;
/* ...how much data has actually been produced by deflate() */ /* ...how much data has actually been produced by deflate() */
bytes = previous_out - zstream->avail_out; bytes = previous_out - zstream->avail_out;
tight_send_compact_size(vs, bytes); tight_send_compact_size(vs, bytes);
vnc_write(vs, vs->tight.zlib.buffer, bytes); vnc_write(vs, vs->tight->zlib.buffer, bytes);
buffer_reset(&vs->tight.zlib); buffer_reset(&vs->tight->zlib);
return bytes; return bytes;
} }
@ -927,16 +927,17 @@ static int send_full_color_rect(VncState *vs, int x, int y, int w, int h)
vnc_write_u8(vs, stream << 4); /* no flushing, no filter */ vnc_write_u8(vs, stream << 4); /* no flushing, no filter */
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
tight_pack24(vs, vs->tight.tight.buffer, w * h, &vs->tight.tight.offset); tight_pack24(vs, vs->tight->tight.buffer, w * h,
&vs->tight->tight.offset);
bytes = 3; bytes = 3;
} else { } else {
bytes = vs->client_pf.bytes_per_pixel; bytes = vs->client_pf.bytes_per_pixel;
} }
bytes = tight_compress_data(vs, stream, w * h * bytes, bytes = tight_compress_data(vs, stream, w * h * bytes,
tight_conf[vs->tight.compression].raw_zlib_level, tight_conf[vs->tight->compression].raw_zlib_level,
Z_DEFAULT_STRATEGY); Z_DEFAULT_STRATEGY);
return (bytes >= 0); return (bytes >= 0);
} }
@ -947,14 +948,14 @@ static int send_solid_rect(VncState *vs)
vnc_write_u8(vs, VNC_TIGHT_FILL << 4); /* no flushing, no filter */ vnc_write_u8(vs, VNC_TIGHT_FILL << 4); /* no flushing, no filter */
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
tight_pack24(vs, vs->tight.tight.buffer, 1, &vs->tight.tight.offset); tight_pack24(vs, vs->tight->tight.buffer, 1, &vs->tight->tight.offset);
bytes = 3; bytes = 3;
} else { } else {
bytes = vs->client_pf.bytes_per_pixel; bytes = vs->client_pf.bytes_per_pixel;
} }
vnc_write(vs, vs->tight.tight.buffer, bytes); vnc_write(vs, vs->tight->tight.buffer, bytes);
return 1; return 1;
} }
@ -963,7 +964,7 @@ static int send_mono_rect(VncState *vs, int x, int y,
{ {
ssize_t bytes; ssize_t bytes;
int stream = 1; int stream = 1;
int level = tight_conf[vs->tight.compression].mono_zlib_level; int level = tight_conf[vs->tight->compression].mono_zlib_level;
#ifdef CONFIG_VNC_PNG #ifdef CONFIG_VNC_PNG
if (tight_can_send_png_rect(vs, w, h)) { if (tight_can_send_png_rect(vs, w, h)) {
@ -991,26 +992,26 @@ static int send_mono_rect(VncState *vs, int x, int y,
uint32_t buf[2] = {bg, fg}; uint32_t buf[2] = {bg, fg};
size_t ret = sizeof (buf); size_t ret = sizeof (buf);
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
tight_pack24(vs, (unsigned char*)buf, 2, &ret); tight_pack24(vs, (unsigned char*)buf, 2, &ret);
} }
vnc_write(vs, buf, ret); vnc_write(vs, buf, ret);
tight_encode_mono_rect32(vs->tight.tight.buffer, w, h, bg, fg); tight_encode_mono_rect32(vs->tight->tight.buffer, w, h, bg, fg);
break; break;
} }
case 2: case 2:
vnc_write(vs, &bg, 2); vnc_write(vs, &bg, 2);
vnc_write(vs, &fg, 2); vnc_write(vs, &fg, 2);
tight_encode_mono_rect16(vs->tight.tight.buffer, w, h, bg, fg); tight_encode_mono_rect16(vs->tight->tight.buffer, w, h, bg, fg);
break; break;
default: default:
vnc_write_u8(vs, bg); vnc_write_u8(vs, bg);
vnc_write_u8(vs, fg); vnc_write_u8(vs, fg);
tight_encode_mono_rect8(vs->tight.tight.buffer, w, h, bg, fg); tight_encode_mono_rect8(vs->tight->tight.buffer, w, h, bg, fg);
break; break;
} }
vs->tight.tight.offset = bytes; vs->tight->tight.offset = bytes;
bytes = tight_compress_data(vs, stream, bytes, level, Z_DEFAULT_STRATEGY); bytes = tight_compress_data(vs, stream, bytes, level, Z_DEFAULT_STRATEGY);
return (bytes >= 0); return (bytes >= 0);
@ -1040,7 +1041,7 @@ static void write_palette(int idx, uint32_t color, void *opaque)
static bool send_gradient_rect(VncState *vs, int x, int y, int w, int h) static bool send_gradient_rect(VncState *vs, int x, int y, int w, int h)
{ {
int stream = 3; int stream = 3;
int level = tight_conf[vs->tight.compression].gradient_zlib_level; int level = tight_conf[vs->tight->compression].gradient_zlib_level;
ssize_t bytes; ssize_t bytes;
if (vs->client_pf.bytes_per_pixel == 1) { if (vs->client_pf.bytes_per_pixel == 1) {
@ -1050,23 +1051,23 @@ static bool send_gradient_rect(VncState *vs, int x, int y, int w, int h)
vnc_write_u8(vs, (stream | VNC_TIGHT_EXPLICIT_FILTER) << 4); vnc_write_u8(vs, (stream | VNC_TIGHT_EXPLICIT_FILTER) << 4);
vnc_write_u8(vs, VNC_TIGHT_FILTER_GRADIENT); vnc_write_u8(vs, VNC_TIGHT_FILTER_GRADIENT);
buffer_reserve(&vs->tight.gradient, w * 3 * sizeof (int)); buffer_reserve(&vs->tight->gradient, w * 3 * sizeof(int));
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
tight_filter_gradient24(vs, vs->tight.tight.buffer, w, h); tight_filter_gradient24(vs, vs->tight->tight.buffer, w, h);
bytes = 3; bytes = 3;
} else if (vs->client_pf.bytes_per_pixel == 4) { } else if (vs->client_pf.bytes_per_pixel == 4) {
tight_filter_gradient32(vs, (uint32_t *)vs->tight.tight.buffer, w, h); tight_filter_gradient32(vs, (uint32_t *)vs->tight->tight.buffer, w, h);
bytes = 4; bytes = 4;
} else { } else {
tight_filter_gradient16(vs, (uint16_t *)vs->tight.tight.buffer, w, h); tight_filter_gradient16(vs, (uint16_t *)vs->tight->tight.buffer, w, h);
bytes = 2; bytes = 2;
} }
buffer_reset(&vs->tight.gradient); buffer_reset(&vs->tight->gradient);
bytes = w * h * bytes; bytes = w * h * bytes;
vs->tight.tight.offset = bytes; vs->tight->tight.offset = bytes;
bytes = tight_compress_data(vs, stream, bytes, bytes = tight_compress_data(vs, stream, bytes,
level, Z_FILTERED); level, Z_FILTERED);
@ -1077,7 +1078,7 @@ static int send_palette_rect(VncState *vs, int x, int y,
int w, int h, VncPalette *palette) int w, int h, VncPalette *palette)
{ {
int stream = 2; int stream = 2;
int level = tight_conf[vs->tight.compression].idx_zlib_level; int level = tight_conf[vs->tight->compression].idx_zlib_level;
int colors; int colors;
ssize_t bytes; ssize_t bytes;
@ -1104,12 +1105,12 @@ static int send_palette_rect(VncState *vs, int x, int y,
palette_iter(palette, write_palette, &priv); palette_iter(palette, write_palette, &priv);
vnc_write(vs, header, sizeof(header)); vnc_write(vs, header, sizeof(header));
if (vs->tight.pixel24) { if (vs->tight->pixel24) {
tight_pack24(vs, vs->output.buffer + old_offset, colors, &offset); tight_pack24(vs, vs->output.buffer + old_offset, colors, &offset);
vs->output.offset = old_offset + offset; vs->output.offset = old_offset + offset;
} }
tight_encode_indexed_rect32(vs->tight.tight.buffer, w * h, palette); tight_encode_indexed_rect32(vs->tight->tight.buffer, w * h, palette);
break; break;
} }
case 2: case 2:
@ -1119,7 +1120,7 @@ static int send_palette_rect(VncState *vs, int x, int y,
palette_iter(palette, write_palette, &priv); palette_iter(palette, write_palette, &priv);
vnc_write(vs, header, sizeof(header)); vnc_write(vs, header, sizeof(header));
tight_encode_indexed_rect16(vs->tight.tight.buffer, w * h, palette); tight_encode_indexed_rect16(vs->tight->tight.buffer, w * h, palette);
break; break;
} }
default: default:
@ -1127,7 +1128,7 @@ static int send_palette_rect(VncState *vs, int x, int y,
break; break;
} }
bytes = w * h; bytes = w * h;
vs->tight.tight.offset = bytes; vs->tight->tight.offset = bytes;
bytes = tight_compress_data(vs, stream, bytes, bytes = tight_compress_data(vs, stream, bytes,
level, Z_DEFAULT_STRATEGY); level, Z_DEFAULT_STRATEGY);
@ -1146,7 +1147,7 @@ static int send_palette_rect(VncState *vs, int x, int y,
static void jpeg_init_destination(j_compress_ptr cinfo) static void jpeg_init_destination(j_compress_ptr cinfo)
{ {
VncState *vs = cinfo->client_data; VncState *vs = cinfo->client_data;
Buffer *buffer = &vs->tight.jpeg; Buffer *buffer = &vs->tight->jpeg;
cinfo->dest->next_output_byte = (JOCTET *)buffer->buffer + buffer->offset; cinfo->dest->next_output_byte = (JOCTET *)buffer->buffer + buffer->offset;
cinfo->dest->free_in_buffer = (size_t)(buffer->capacity - buffer->offset); cinfo->dest->free_in_buffer = (size_t)(buffer->capacity - buffer->offset);
@ -1156,7 +1157,7 @@ static void jpeg_init_destination(j_compress_ptr cinfo)
static boolean jpeg_empty_output_buffer(j_compress_ptr cinfo) static boolean jpeg_empty_output_buffer(j_compress_ptr cinfo)
{ {
VncState *vs = cinfo->client_data; VncState *vs = cinfo->client_data;
Buffer *buffer = &vs->tight.jpeg; Buffer *buffer = &vs->tight->jpeg;
buffer->offset = buffer->capacity; buffer->offset = buffer->capacity;
buffer_reserve(buffer, 2048); buffer_reserve(buffer, 2048);
@ -1168,7 +1169,7 @@ static boolean jpeg_empty_output_buffer(j_compress_ptr cinfo)
static void jpeg_term_destination(j_compress_ptr cinfo) static void jpeg_term_destination(j_compress_ptr cinfo)
{ {
VncState *vs = cinfo->client_data; VncState *vs = cinfo->client_data;
Buffer *buffer = &vs->tight.jpeg; Buffer *buffer = &vs->tight->jpeg;
buffer->offset = buffer->capacity - cinfo->dest->free_in_buffer; buffer->offset = buffer->capacity - cinfo->dest->free_in_buffer;
} }
@ -1187,7 +1188,7 @@ static int send_jpeg_rect(VncState *vs, int x, int y, int w, int h, int quality)
return send_full_color_rect(vs, x, y, w, h); return send_full_color_rect(vs, x, y, w, h);
} }
buffer_reserve(&vs->tight.jpeg, 2048); buffer_reserve(&vs->tight->jpeg, 2048);
cinfo.err = jpeg_std_error(&jerr); cinfo.err = jpeg_std_error(&jerr);
jpeg_create_compress(&cinfo); jpeg_create_compress(&cinfo);
@ -1222,9 +1223,9 @@ static int send_jpeg_rect(VncState *vs, int x, int y, int w, int h, int quality)
vnc_write_u8(vs, VNC_TIGHT_JPEG << 4); vnc_write_u8(vs, VNC_TIGHT_JPEG << 4);
tight_send_compact_size(vs, vs->tight.jpeg.offset); tight_send_compact_size(vs, vs->tight->jpeg.offset);
vnc_write(vs, vs->tight.jpeg.buffer, vs->tight.jpeg.offset); vnc_write(vs, vs->tight->jpeg.buffer, vs->tight->jpeg.offset);
buffer_reset(&vs->tight.jpeg); buffer_reset(&vs->tight->jpeg);
return 1; return 1;
} }
@ -1240,7 +1241,7 @@ static void write_png_palette(int idx, uint32_t pix, void *opaque)
VncState *vs = priv->vs; VncState *vs = priv->vs;
png_colorp color = &priv->png_palette[idx]; png_colorp color = &priv->png_palette[idx];
if (vs->tight.pixel24) if (vs->tight->pixel24)
{ {
color->red = (pix >> vs->client_pf.rshift) & vs->client_pf.rmax; color->red = (pix >> vs->client_pf.rshift) & vs->client_pf.rmax;
color->green = (pix >> vs->client_pf.gshift) & vs->client_pf.gmax; color->green = (pix >> vs->client_pf.gshift) & vs->client_pf.gmax;
@ -1267,10 +1268,10 @@ static void png_write_data(png_structp png_ptr, png_bytep data,
{ {
VncState *vs = png_get_io_ptr(png_ptr); VncState *vs = png_get_io_ptr(png_ptr);
buffer_reserve(&vs->tight.png, vs->tight.png.offset + length); buffer_reserve(&vs->tight->png, vs->tight->png.offset + length);
memcpy(vs->tight.png.buffer + vs->tight.png.offset, data, length); memcpy(vs->tight->png.buffer + vs->tight->png.offset, data, length);
vs->tight.png.offset += length; vs->tight->png.offset += length;
} }
static void png_flush_data(png_structp png_ptr) static void png_flush_data(png_structp png_ptr)
@ -1295,8 +1296,8 @@ static int send_png_rect(VncState *vs, int x, int y, int w, int h,
png_infop info_ptr; png_infop info_ptr;
png_colorp png_palette = NULL; png_colorp png_palette = NULL;
pixman_image_t *linebuf; pixman_image_t *linebuf;
int level = tight_png_conf[vs->tight.compression].png_zlib_level; int level = tight_png_conf[vs->tight->compression].png_zlib_level;
int filters = tight_png_conf[vs->tight.compression].png_filters; int filters = tight_png_conf[vs->tight->compression].png_filters;
uint8_t *buf; uint8_t *buf;
int dy; int dy;
@ -1340,21 +1341,23 @@ static int send_png_rect(VncState *vs, int x, int y, int w, int h,
png_set_PLTE(png_ptr, info_ptr, png_palette, palette_size(palette)); png_set_PLTE(png_ptr, info_ptr, png_palette, palette_size(palette));
if (vs->client_pf.bytes_per_pixel == 4) { if (vs->client_pf.bytes_per_pixel == 4) {
tight_encode_indexed_rect32(vs->tight.tight.buffer, w * h, palette); tight_encode_indexed_rect32(vs->tight->tight.buffer, w * h,
palette);
} else { } else {
tight_encode_indexed_rect16(vs->tight.tight.buffer, w * h, palette); tight_encode_indexed_rect16(vs->tight->tight.buffer, w * h,
palette);
} }
} }
png_write_info(png_ptr, info_ptr); png_write_info(png_ptr, info_ptr);
buffer_reserve(&vs->tight.png, 2048); buffer_reserve(&vs->tight->png, 2048);
linebuf = qemu_pixman_linebuf_create(PIXMAN_BE_r8g8b8, w); linebuf = qemu_pixman_linebuf_create(PIXMAN_BE_r8g8b8, w);
buf = (uint8_t *)pixman_image_get_data(linebuf); buf = (uint8_t *)pixman_image_get_data(linebuf);
for (dy = 0; dy < h; dy++) for (dy = 0; dy < h; dy++)
{ {
if (color_type == PNG_COLOR_TYPE_PALETTE) { if (color_type == PNG_COLOR_TYPE_PALETTE) {
memcpy(buf, vs->tight.tight.buffer + (dy * w), w); memcpy(buf, vs->tight->tight.buffer + (dy * w), w);
} else { } else {
qemu_pixman_linebuf_fill(linebuf, vs->vd->server, w, x, y + dy); qemu_pixman_linebuf_fill(linebuf, vs->vd->server, w, x, y + dy);
} }
@ -1372,27 +1375,27 @@ static int send_png_rect(VncState *vs, int x, int y, int w, int h,
vnc_write_u8(vs, VNC_TIGHT_PNG << 4); vnc_write_u8(vs, VNC_TIGHT_PNG << 4);
tight_send_compact_size(vs, vs->tight.png.offset); tight_send_compact_size(vs, vs->tight->png.offset);
vnc_write(vs, vs->tight.png.buffer, vs->tight.png.offset); vnc_write(vs, vs->tight->png.buffer, vs->tight->png.offset);
buffer_reset(&vs->tight.png); buffer_reset(&vs->tight->png);
return 1; return 1;
} }
#endif /* CONFIG_VNC_PNG */ #endif /* CONFIG_VNC_PNG */
static void vnc_tight_start(VncState *vs) static void vnc_tight_start(VncState *vs)
{ {
buffer_reset(&vs->tight.tight); buffer_reset(&vs->tight->tight);
// make the output buffer be the zlib buffer, so we can compress it later // make the output buffer be the zlib buffer, so we can compress it later
vs->tight.tmp = vs->output; vs->tight->tmp = vs->output;
vs->output = vs->tight.tight; vs->output = vs->tight->tight;
} }
static void vnc_tight_stop(VncState *vs) static void vnc_tight_stop(VncState *vs)
{ {
// switch back to normal output/zlib buffers // switch back to normal output/zlib buffers
vs->tight.tight = vs->output; vs->tight->tight = vs->output;
vs->output = vs->tight.tmp; vs->output = vs->tight->tmp;
} }
static int send_sub_rect_nojpeg(VncState *vs, int x, int y, int w, int h, static int send_sub_rect_nojpeg(VncState *vs, int x, int y, int w, int h,
@ -1426,9 +1429,9 @@ static int send_sub_rect_jpeg(VncState *vs, int x, int y, int w, int h,
int ret; int ret;
if (colors == 0) { if (colors == 0) {
if (force || (tight_jpeg_conf[vs->tight.quality].jpeg_full && if (force || (tight_jpeg_conf[vs->tight->quality].jpeg_full &&
tight_detect_smooth_image(vs, w, h))) { tight_detect_smooth_image(vs, w, h))) {
int quality = tight_conf[vs->tight.quality].jpeg_quality; int quality = tight_conf[vs->tight->quality].jpeg_quality;
ret = send_jpeg_rect(vs, x, y, w, h, quality); ret = send_jpeg_rect(vs, x, y, w, h, quality);
} else { } else {
@ -1440,9 +1443,9 @@ static int send_sub_rect_jpeg(VncState *vs, int x, int y, int w, int h,
ret = send_mono_rect(vs, x, y, w, h, bg, fg); ret = send_mono_rect(vs, x, y, w, h, bg, fg);
} else if (colors <= 256) { } else if (colors <= 256) {
if (force || (colors > 96 && if (force || (colors > 96 &&
tight_jpeg_conf[vs->tight.quality].jpeg_idx && tight_jpeg_conf[vs->tight->quality].jpeg_idx &&
tight_detect_smooth_image(vs, w, h))) { tight_detect_smooth_image(vs, w, h))) {
int quality = tight_conf[vs->tight.quality].jpeg_quality; int quality = tight_conf[vs->tight->quality].jpeg_quality;
ret = send_jpeg_rect(vs, x, y, w, h, quality); ret = send_jpeg_rect(vs, x, y, w, h, quality);
} else { } else {
@ -1480,20 +1483,20 @@ static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
qemu_thread_atexit_add(&vnc_tight_cleanup_notifier); qemu_thread_atexit_add(&vnc_tight_cleanup_notifier);
} }
vnc_framebuffer_update(vs, x, y, w, h, vs->tight.type); vnc_framebuffer_update(vs, x, y, w, h, vs->tight->type);
vnc_tight_start(vs); vnc_tight_start(vs);
vnc_raw_send_framebuffer_update(vs, x, y, w, h); vnc_raw_send_framebuffer_update(vs, x, y, w, h);
vnc_tight_stop(vs); vnc_tight_stop(vs);
#ifdef CONFIG_VNC_JPEG #ifdef CONFIG_VNC_JPEG
if (!vs->vd->non_adaptive && vs->tight.quality != (uint8_t)-1) { if (!vs->vd->non_adaptive && vs->tight->quality != (uint8_t)-1) {
double freq = vnc_update_freq(vs, x, y, w, h); double freq = vnc_update_freq(vs, x, y, w, h);
if (freq < tight_jpeg_conf[vs->tight.quality].jpeg_freq_min) { if (freq < tight_jpeg_conf[vs->tight->quality].jpeg_freq_min) {
allow_jpeg = false; allow_jpeg = false;
} }
if (freq >= tight_jpeg_conf[vs->tight.quality].jpeg_freq_threshold) { if (freq >= tight_jpeg_conf[vs->tight->quality].jpeg_freq_threshold) {
force_jpeg = true; force_jpeg = true;
vnc_sent_lossy_rect(vs, x, y, w, h); vnc_sent_lossy_rect(vs, x, y, w, h);
} }
@ -1503,7 +1506,7 @@ static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
colors = tight_fill_palette(vs, x, y, w * h, &bg, &fg, color_count_palette); colors = tight_fill_palette(vs, x, y, w * h, &bg, &fg, color_count_palette);
#ifdef CONFIG_VNC_JPEG #ifdef CONFIG_VNC_JPEG
if (allow_jpeg && vs->tight.quality != (uint8_t)-1) { if (allow_jpeg && vs->tight->quality != (uint8_t)-1) {
ret = send_sub_rect_jpeg(vs, x, y, w, h, bg, fg, colors, ret = send_sub_rect_jpeg(vs, x, y, w, h, bg, fg, colors,
color_count_palette, force_jpeg); color_count_palette, force_jpeg);
} else { } else {
@ -1520,7 +1523,7 @@ static int send_sub_rect(VncState *vs, int x, int y, int w, int h)
static int send_sub_rect_solid(VncState *vs, int x, int y, int w, int h) static int send_sub_rect_solid(VncState *vs, int x, int y, int w, int h)
{ {
vnc_framebuffer_update(vs, x, y, w, h, vs->tight.type); vnc_framebuffer_update(vs, x, y, w, h, vs->tight->type);
vnc_tight_start(vs); vnc_tight_start(vs);
vnc_raw_send_framebuffer_update(vs, x, y, w, h); vnc_raw_send_framebuffer_update(vs, x, y, w, h);
@ -1538,8 +1541,8 @@ static int send_rect_simple(VncState *vs, int x, int y, int w, int h,
int rw, rh; int rw, rh;
int n = 0; int n = 0;
max_size = tight_conf[vs->tight.compression].max_rect_size; max_size = tight_conf[vs->tight->compression].max_rect_size;
max_width = tight_conf[vs->tight.compression].max_rect_width; max_width = tight_conf[vs->tight->compression].max_rect_width;
if (split && (w > max_width || w * h > max_size)) { if (split && (w > max_width || w * h > max_size)) {
max_sub_width = (w > max_width) ? max_width : w; max_sub_width = (w > max_width) ? max_width : w;
@ -1648,16 +1651,16 @@ static int tight_send_framebuffer_update(VncState *vs, int x, int y,
if (vs->client_pf.bytes_per_pixel == 4 && vs->client_pf.rmax == 0xFF && if (vs->client_pf.bytes_per_pixel == 4 && vs->client_pf.rmax == 0xFF &&
vs->client_pf.bmax == 0xFF && vs->client_pf.gmax == 0xFF) { vs->client_pf.bmax == 0xFF && vs->client_pf.gmax == 0xFF) {
vs->tight.pixel24 = true; vs->tight->pixel24 = true;
} else { } else {
vs->tight.pixel24 = false; vs->tight->pixel24 = false;
} }
#ifdef CONFIG_VNC_JPEG #ifdef CONFIG_VNC_JPEG
if (vs->tight.quality != (uint8_t)-1) { if (vs->tight->quality != (uint8_t)-1) {
double freq = vnc_update_freq(vs, x, y, w, h); double freq = vnc_update_freq(vs, x, y, w, h);
if (freq > tight_jpeg_conf[vs->tight.quality].jpeg_freq_threshold) { if (freq > tight_jpeg_conf[vs->tight->quality].jpeg_freq_threshold) {
return send_rect_simple(vs, x, y, w, h, false); return send_rect_simple(vs, x, y, w, h, false);
} }
} }
@ -1669,8 +1672,8 @@ static int tight_send_framebuffer_update(VncState *vs, int x, int y,
/* Calculate maximum number of rows in one non-solid rectangle. */ /* Calculate maximum number of rows in one non-solid rectangle. */
max_rows = tight_conf[vs->tight.compression].max_rect_size; max_rows = tight_conf[vs->tight->compression].max_rect_size;
max_rows /= MIN(tight_conf[vs->tight.compression].max_rect_width, w); max_rows /= MIN(tight_conf[vs->tight->compression].max_rect_width, w);
return find_large_solid_color_rect(vs, x, y, w, h, max_rows); return find_large_solid_color_rect(vs, x, y, w, h, max_rows);
} }
@ -1678,33 +1681,33 @@ static int tight_send_framebuffer_update(VncState *vs, int x, int y,
int vnc_tight_send_framebuffer_update(VncState *vs, int x, int y, int vnc_tight_send_framebuffer_update(VncState *vs, int x, int y,
int w, int h) int w, int h)
{ {
vs->tight.type = VNC_ENCODING_TIGHT; vs->tight->type = VNC_ENCODING_TIGHT;
return tight_send_framebuffer_update(vs, x, y, w, h); return tight_send_framebuffer_update(vs, x, y, w, h);
} }
int vnc_tight_png_send_framebuffer_update(VncState *vs, int x, int y, int vnc_tight_png_send_framebuffer_update(VncState *vs, int x, int y,
int w, int h) int w, int h)
{ {
vs->tight.type = VNC_ENCODING_TIGHT_PNG; vs->tight->type = VNC_ENCODING_TIGHT_PNG;
return tight_send_framebuffer_update(vs, x, y, w, h); return tight_send_framebuffer_update(vs, x, y, w, h);
} }
void vnc_tight_clear(VncState *vs) void vnc_tight_clear(VncState *vs)
{ {
int i; int i;
for (i=0; i<ARRAY_SIZE(vs->tight.stream); i++) { for (i = 0; i < ARRAY_SIZE(vs->tight->stream); i++) {
if (vs->tight.stream[i].opaque) { if (vs->tight->stream[i].opaque) {
deflateEnd(&vs->tight.stream[i]); deflateEnd(&vs->tight->stream[i]);
} }
} }
buffer_free(&vs->tight.tight); buffer_free(&vs->tight->tight);
buffer_free(&vs->tight.zlib); buffer_free(&vs->tight->zlib);
buffer_free(&vs->tight.gradient); buffer_free(&vs->tight->gradient);
#ifdef CONFIG_VNC_JPEG #ifdef CONFIG_VNC_JPEG
buffer_free(&vs->tight.jpeg); buffer_free(&vs->tight->jpeg);
#endif #endif
#ifdef CONFIG_VNC_PNG #ifdef CONFIG_VNC_PNG
buffer_free(&vs->tight.png); buffer_free(&vs->tight->png);
#endif #endif
} }

View File

@ -76,7 +76,8 @@ static int vnc_zlib_stop(VncState *vs)
zstream->zalloc = vnc_zlib_zalloc; zstream->zalloc = vnc_zlib_zalloc;
zstream->zfree = vnc_zlib_zfree; zstream->zfree = vnc_zlib_zfree;
err = deflateInit2(zstream, vs->tight.compression, Z_DEFLATED, MAX_WBITS, err = deflateInit2(zstream, vs->tight->compression, Z_DEFLATED,
MAX_WBITS,
MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY); MAX_MEM_LEVEL, Z_DEFAULT_STRATEGY);
if (err != Z_OK) { if (err != Z_OK) {
@ -84,16 +85,16 @@ static int vnc_zlib_stop(VncState *vs)
return -1; return -1;
} }
vs->zlib.level = vs->tight.compression; vs->zlib.level = vs->tight->compression;
zstream->opaque = vs; zstream->opaque = vs;
} }
if (vs->tight.compression != vs->zlib.level) { if (vs->tight->compression != vs->zlib.level) {
if (deflateParams(zstream, vs->tight.compression, if (deflateParams(zstream, vs->tight->compression,
Z_DEFAULT_STRATEGY) != Z_OK) { Z_DEFAULT_STRATEGY) != Z_OK) {
return -1; return -1;
} }
vs->zlib.level = vs->tight.compression; vs->zlib.level = vs->tight->compression;
} }
// reserve memory in output buffer // reserve memory in output buffer

View File

@ -37,18 +37,18 @@ static const int bits_per_packed_pixel[] = {
static void vnc_zrle_start(VncState *vs) static void vnc_zrle_start(VncState *vs)
{ {
buffer_reset(&vs->zrle.zrle); buffer_reset(&vs->zrle->zrle);
/* make the output buffer be the zlib buffer, so we can compress it later */ /* make the output buffer be the zlib buffer, so we can compress it later */
vs->zrle.tmp = vs->output; vs->zrle->tmp = vs->output;
vs->output = vs->zrle.zrle; vs->output = vs->zrle->zrle;
} }
static void vnc_zrle_stop(VncState *vs) static void vnc_zrle_stop(VncState *vs)
{ {
/* switch back to normal output/zlib buffers */ /* switch back to normal output/zlib buffers */
vs->zrle.zrle = vs->output; vs->zrle->zrle = vs->output;
vs->output = vs->zrle.tmp; vs->output = vs->zrle->tmp;
} }
static void *zrle_convert_fb(VncState *vs, int x, int y, int w, int h, static void *zrle_convert_fb(VncState *vs, int x, int y, int w, int h,
@ -56,24 +56,24 @@ static void *zrle_convert_fb(VncState *vs, int x, int y, int w, int h,
{ {
Buffer tmp; Buffer tmp;
buffer_reset(&vs->zrle.fb); buffer_reset(&vs->zrle->fb);
buffer_reserve(&vs->zrle.fb, w * h * bpp + bpp); buffer_reserve(&vs->zrle->fb, w * h * bpp + bpp);
tmp = vs->output; tmp = vs->output;
vs->output = vs->zrle.fb; vs->output = vs->zrle->fb;
vnc_raw_send_framebuffer_update(vs, x, y, w, h); vnc_raw_send_framebuffer_update(vs, x, y, w, h);
vs->zrle.fb = vs->output; vs->zrle->fb = vs->output;
vs->output = tmp; vs->output = tmp;
return vs->zrle.fb.buffer; return vs->zrle->fb.buffer;
} }
static int zrle_compress_data(VncState *vs, int level) static int zrle_compress_data(VncState *vs, int level)
{ {
z_streamp zstream = &vs->zrle.stream; z_streamp zstream = &vs->zrle->stream;
buffer_reset(&vs->zrle.zlib); buffer_reset(&vs->zrle->zlib);
if (zstream->opaque != vs) { if (zstream->opaque != vs) {
int err; int err;
@ -93,13 +93,13 @@ static int zrle_compress_data(VncState *vs, int level)
} }
/* reserve memory in output buffer */ /* reserve memory in output buffer */
buffer_reserve(&vs->zrle.zlib, vs->zrle.zrle.offset + 64); buffer_reserve(&vs->zrle->zlib, vs->zrle->zrle.offset + 64);
/* set pointers */ /* set pointers */
zstream->next_in = vs->zrle.zrle.buffer; zstream->next_in = vs->zrle->zrle.buffer;
zstream->avail_in = vs->zrle.zrle.offset; zstream->avail_in = vs->zrle->zrle.offset;
zstream->next_out = vs->zrle.zlib.buffer + vs->zrle.zlib.offset; zstream->next_out = vs->zrle->zlib.buffer + vs->zrle->zlib.offset;
zstream->avail_out = vs->zrle.zlib.capacity - vs->zrle.zlib.offset; zstream->avail_out = vs->zrle->zlib.capacity - vs->zrle->zlib.offset;
zstream->data_type = Z_BINARY; zstream->data_type = Z_BINARY;
/* start encoding */ /* start encoding */
@ -108,8 +108,8 @@ static int zrle_compress_data(VncState *vs, int level)
return -1; return -1;
} }
vs->zrle.zlib.offset = vs->zrle.zlib.capacity - zstream->avail_out; vs->zrle->zlib.offset = vs->zrle->zlib.capacity - zstream->avail_out;
return vs->zrle.zlib.offset; return vs->zrle->zlib.offset;
} }
/* Try to work out whether to use RLE and/or a palette. We do this by /* Try to work out whether to use RLE and/or a palette. We do this by
@ -259,14 +259,14 @@ static int zrle_send_framebuffer_update(VncState *vs, int x, int y,
size_t bytes; size_t bytes;
int zywrle_level; int zywrle_level;
if (vs->zrle.type == VNC_ENCODING_ZYWRLE) { if (vs->zrle->type == VNC_ENCODING_ZYWRLE) {
if (!vs->vd->lossy || vs->tight.quality == (uint8_t)-1 if (!vs->vd->lossy || vs->tight->quality == (uint8_t)-1
|| vs->tight.quality == 9) { || vs->tight->quality == 9) {
zywrle_level = 0; zywrle_level = 0;
vs->zrle.type = VNC_ENCODING_ZRLE; vs->zrle->type = VNC_ENCODING_ZRLE;
} else if (vs->tight.quality < 3) { } else if (vs->tight->quality < 3) {
zywrle_level = 3; zywrle_level = 3;
} else if (vs->tight.quality < 6) { } else if (vs->tight->quality < 6) {
zywrle_level = 2; zywrle_level = 2;
} else { } else {
zywrle_level = 1; zywrle_level = 1;
@ -337,30 +337,30 @@ static int zrle_send_framebuffer_update(VncState *vs, int x, int y,
vnc_zrle_stop(vs); vnc_zrle_stop(vs);
bytes = zrle_compress_data(vs, Z_DEFAULT_COMPRESSION); bytes = zrle_compress_data(vs, Z_DEFAULT_COMPRESSION);
vnc_framebuffer_update(vs, x, y, w, h, vs->zrle.type); vnc_framebuffer_update(vs, x, y, w, h, vs->zrle->type);
vnc_write_u32(vs, bytes); vnc_write_u32(vs, bytes);
vnc_write(vs, vs->zrle.zlib.buffer, vs->zrle.zlib.offset); vnc_write(vs, vs->zrle->zlib.buffer, vs->zrle->zlib.offset);
return 1; return 1;
} }
int vnc_zrle_send_framebuffer_update(VncState *vs, int x, int y, int w, int h) int vnc_zrle_send_framebuffer_update(VncState *vs, int x, int y, int w, int h)
{ {
vs->zrle.type = VNC_ENCODING_ZRLE; vs->zrle->type = VNC_ENCODING_ZRLE;
return zrle_send_framebuffer_update(vs, x, y, w, h); return zrle_send_framebuffer_update(vs, x, y, w, h);
} }
int vnc_zywrle_send_framebuffer_update(VncState *vs, int x, int y, int w, int h) int vnc_zywrle_send_framebuffer_update(VncState *vs, int x, int y, int w, int h)
{ {
vs->zrle.type = VNC_ENCODING_ZYWRLE; vs->zrle->type = VNC_ENCODING_ZYWRLE;
return zrle_send_framebuffer_update(vs, x, y, w, h); return zrle_send_framebuffer_update(vs, x, y, w, h);
} }
void vnc_zrle_clear(VncState *vs) void vnc_zrle_clear(VncState *vs)
{ {
if (vs->zrle.stream.opaque) { if (vs->zrle->stream.opaque) {
deflateEnd(&vs->zrle.stream); deflateEnd(&vs->zrle->stream);
} }
buffer_free(&vs->zrle.zrle); buffer_free(&vs->zrle->zrle);
buffer_free(&vs->zrle.fb); buffer_free(&vs->zrle->fb);
buffer_free(&vs->zrle.zlib); buffer_free(&vs->zrle->zlib);
} }

View File

@ -96,7 +96,7 @@ static void ZRLE_ENCODE(VncState *vs, int x, int y, int w, int h,
static void ZRLE_ENCODE_TILE(VncState *vs, ZRLE_PIXEL *data, int w, int h, static void ZRLE_ENCODE_TILE(VncState *vs, ZRLE_PIXEL *data, int w, int h,
int zywrle_level) int zywrle_level)
{ {
VncPalette *palette = &vs->zrle.palette; VncPalette *palette = &vs->zrle->palette;
int runs = 0; int runs = 0;
int single_pixels = 0; int single_pixels = 0;

View File

@ -278,6 +278,7 @@ static void vnc_client_cache_addr(VncState *client)
vnc_init_basic_info_from_remote_addr(client->sioc, vnc_init_basic_info_from_remote_addr(client->sioc,
qapi_VncClientInfo_base(client->info), qapi_VncClientInfo_base(client->info),
&err); &err);
client->info->websocket = client->websocket;
if (err) { if (err) {
qapi_free_VncClientInfo(client->info); qapi_free_VncClientInfo(client->info);
client->info = NULL; client->info = NULL;
@ -1306,6 +1307,8 @@ void vnc_disconnect_finish(VncState *vs)
object_unref(OBJECT(vs->sioc)); object_unref(OBJECT(vs->sioc));
vs->sioc = NULL; vs->sioc = NULL;
vs->magic = 0; vs->magic = 0;
g_free(vs->zrle);
g_free(vs->tight);
g_free(vs); g_free(vs);
} }
@ -2057,8 +2060,8 @@ static void set_encodings(VncState *vs, int32_t *encodings, size_t n_encodings)
vs->features = 0; vs->features = 0;
vs->vnc_encoding = 0; vs->vnc_encoding = 0;
vs->tight.compression = 9; vs->tight->compression = 9;
vs->tight.quality = -1; /* Lossless by default */ vs->tight->quality = -1; /* Lossless by default */
vs->absolute = -1; vs->absolute = -1;
/* /*
@ -2126,11 +2129,11 @@ static void set_encodings(VncState *vs, int32_t *encodings, size_t n_encodings)
vs->features |= VNC_FEATURE_LED_STATE_MASK; vs->features |= VNC_FEATURE_LED_STATE_MASK;
break; break;
case VNC_ENCODING_COMPRESSLEVEL0 ... VNC_ENCODING_COMPRESSLEVEL0 + 9: case VNC_ENCODING_COMPRESSLEVEL0 ... VNC_ENCODING_COMPRESSLEVEL0 + 9:
vs->tight.compression = (enc & 0x0F); vs->tight->compression = (enc & 0x0F);
break; break;
case VNC_ENCODING_QUALITYLEVEL0 ... VNC_ENCODING_QUALITYLEVEL0 + 9: case VNC_ENCODING_QUALITYLEVEL0 ... VNC_ENCODING_QUALITYLEVEL0 + 9:
if (vs->vd->lossy) { if (vs->vd->lossy) {
vs->tight.quality = (enc & 0x0F); vs->tight->quality = (enc & 0x0F);
} }
break; break;
default: default:
@ -3033,6 +3036,8 @@ static void vnc_connect(VncDisplay *vd, QIOChannelSocket *sioc,
int i; int i;
trace_vnc_client_connect(vs, sioc); trace_vnc_client_connect(vs, sioc);
vs->zrle = g_new0(VncZrle, 1);
vs->tight = g_new0(VncTight, 1);
vs->magic = VNC_MAGIC; vs->magic = VNC_MAGIC;
vs->sioc = sioc; vs->sioc = sioc;
object_ref(OBJECT(vs->sioc)); object_ref(OBJECT(vs->sioc));
@ -3044,19 +3049,19 @@ static void vnc_connect(VncDisplay *vd, QIOChannelSocket *sioc,
buffer_init(&vs->output, "vnc-output/%p", sioc); buffer_init(&vs->output, "vnc-output/%p", sioc);
buffer_init(&vs->jobs_buffer, "vnc-jobs_buffer/%p", sioc); buffer_init(&vs->jobs_buffer, "vnc-jobs_buffer/%p", sioc);
buffer_init(&vs->tight.tight, "vnc-tight/%p", sioc); buffer_init(&vs->tight->tight, "vnc-tight/%p", sioc);
buffer_init(&vs->tight.zlib, "vnc-tight-zlib/%p", sioc); buffer_init(&vs->tight->zlib, "vnc-tight-zlib/%p", sioc);
buffer_init(&vs->tight.gradient, "vnc-tight-gradient/%p", sioc); buffer_init(&vs->tight->gradient, "vnc-tight-gradient/%p", sioc);
#ifdef CONFIG_VNC_JPEG #ifdef CONFIG_VNC_JPEG
buffer_init(&vs->tight.jpeg, "vnc-tight-jpeg/%p", sioc); buffer_init(&vs->tight->jpeg, "vnc-tight-jpeg/%p", sioc);
#endif #endif
#ifdef CONFIG_VNC_PNG #ifdef CONFIG_VNC_PNG
buffer_init(&vs->tight.png, "vnc-tight-png/%p", sioc); buffer_init(&vs->tight->png, "vnc-tight-png/%p", sioc);
#endif #endif
buffer_init(&vs->zlib.zlib, "vnc-zlib/%p", sioc); buffer_init(&vs->zlib.zlib, "vnc-zlib/%p", sioc);
buffer_init(&vs->zrle.zrle, "vnc-zrle/%p", sioc); buffer_init(&vs->zrle->zrle, "vnc-zrle/%p", sioc);
buffer_init(&vs->zrle.fb, "vnc-zrle-fb/%p", sioc); buffer_init(&vs->zrle->fb, "vnc-zrle-fb/%p", sioc);
buffer_init(&vs->zrle.zlib, "vnc-zrle-zlib/%p", sioc); buffer_init(&vs->zrle->zlib, "vnc-zrle-zlib/%p", sioc);
if (skipauth) { if (skipauth) {
vs->auth = VNC_AUTH_NONE; vs->auth = VNC_AUTH_NONE;

View File

@ -338,10 +338,10 @@ struct VncState
/* Encoding specific, if you add something here, don't forget to /* Encoding specific, if you add something here, don't forget to
* update vnc_async_encoding_start() * update vnc_async_encoding_start()
*/ */
VncTight tight; VncTight *tight;
VncZlib zlib; VncZlib zlib;
VncHextile hextile; VncHextile hextile;
VncZrle zrle; VncZrle *zrle;
VncZywrle zywrle; VncZywrle zywrle;
Notifier mouse_mode_notifier; Notifier mouse_mode_notifier;