mirror of https://github.com/xqemu/xqemu.git
vnc: fix use-after-free in vnc_update_client_sync
Spotted by Coverity: 876 static int vnc_update_client_sync(VncState *vs, int has_dirty) 877 { (1) Event freed_arg: "vnc_update_client(VncState *, int)" frees "vs". [details] Also see events: [deref_arg] 878 int ret = vnc_update_client(vs, has_dirty); (2) Event deref_arg: Calling "vnc_jobs_join(VncState *)" dereferences freed pointer "vs". [details] Also see events: [freed_arg] 879 vnc_jobs_join(vs); 880 return ret; 881 } Remove vnc_update_client_sync wrapper, replace it with an additional argument to vnc_update_client, so we can so the sync properly in vnc_update_client (i.e. skip it in case of a client disconnect). Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Reviewed-by: Markus Armbruster <armbru@redhat.com>
This commit is contained in:
parent
e3c1adf16e
commit
38ee14f4f3
21
ui/vnc.c
21
ui/vnc.c
|
@ -417,8 +417,7 @@ out_error:
|
||||||
3) resolutions > 1024
|
3) resolutions > 1024
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int vnc_update_client(VncState *vs, int has_dirty);
|
static int vnc_update_client(VncState *vs, int has_dirty, bool sync);
|
||||||
static int vnc_update_client_sync(VncState *vs, int has_dirty);
|
|
||||||
static void vnc_disconnect_start(VncState *vs);
|
static void vnc_disconnect_start(VncState *vs);
|
||||||
|
|
||||||
static void vnc_colordepth(VncState *vs);
|
static void vnc_colordepth(VncState *vs);
|
||||||
|
@ -751,7 +750,7 @@ static void vnc_dpy_copy(DisplayChangeListener *dcl,
|
||||||
QTAILQ_FOREACH_SAFE(vs, &vd->clients, next, vn) {
|
QTAILQ_FOREACH_SAFE(vs, &vd->clients, next, vn) {
|
||||||
if (vnc_has_feature(vs, VNC_FEATURE_COPYRECT)) {
|
if (vnc_has_feature(vs, VNC_FEATURE_COPYRECT)) {
|
||||||
vs->force_update = 1;
|
vs->force_update = 1;
|
||||||
vnc_update_client_sync(vs, 1);
|
vnc_update_client(vs, 1, true);
|
||||||
/* vs might be free()ed here */
|
/* vs might be free()ed here */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -874,14 +873,7 @@ static int find_and_clear_dirty_height(struct VncState *vs,
|
||||||
return h;
|
return h;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int vnc_update_client_sync(VncState *vs, int has_dirty)
|
static int vnc_update_client(VncState *vs, int has_dirty, bool sync)
|
||||||
{
|
|
||||||
int ret = vnc_update_client(vs, has_dirty);
|
|
||||||
vnc_jobs_join(vs);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int vnc_update_client(VncState *vs, int has_dirty)
|
|
||||||
{
|
{
|
||||||
if (vs->need_update && vs->csock != -1) {
|
if (vs->need_update && vs->csock != -1) {
|
||||||
VncDisplay *vd = vs->vd;
|
VncDisplay *vd = vs->vd;
|
||||||
|
@ -940,8 +932,11 @@ static int vnc_update_client(VncState *vs, int has_dirty)
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (vs->csock == -1)
|
if (vs->csock == -1) {
|
||||||
vnc_disconnect_finish(vs);
|
vnc_disconnect_finish(vs);
|
||||||
|
} else if (sync) {
|
||||||
|
vnc_jobs_join(vs);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -2734,7 +2729,7 @@ static void vnc_refresh(DisplayChangeListener *dcl)
|
||||||
vnc_unlock_display(vd);
|
vnc_unlock_display(vd);
|
||||||
|
|
||||||
QTAILQ_FOREACH_SAFE(vs, &vd->clients, next, vn) {
|
QTAILQ_FOREACH_SAFE(vs, &vd->clients, next, vn) {
|
||||||
rects += vnc_update_client(vs, has_dirty);
|
rects += vnc_update_client(vs, has_dirty, false);
|
||||||
/* vs might be free()ed here */
|
/* vs might be free()ed here */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue