mirror of
https://github.com/leahneukirchen/cwm.git
synced 2023-08-10 21:13:12 +03:00
Alter the r1.35 of event.c race fix. Remove the forward looking event
queue check (removing the need for a server grab/ungrab) - if the client is going away, let it fall all the way through to a DestroyNotify event. There's no longer a need for us to manually destroy a client ourselves (removing yet another server grab/ungrab). Instead, when the UnmapNotify event is synthetic, simply set the state to Withdrawn (as per ICCCM), else Iconic (in our case 'hidden'). Verified with test case from the 2009 race which was the original reason for r1.35 of event.c.
This commit is contained in:
parent
bda68b0924
commit
2937b2066f
2
calmwm.h
2
calmwm.h
@ -370,7 +370,7 @@ struct client_ctx *client_current(void);
|
|||||||
void client_cycle(struct screen_ctx *, int);
|
void client_cycle(struct screen_ctx *, int);
|
||||||
void client_cycle_leave(struct screen_ctx *,
|
void client_cycle_leave(struct screen_ctx *,
|
||||||
struct client_ctx *);
|
struct client_ctx *);
|
||||||
void client_delete(struct client_ctx *, int);
|
void client_delete(struct client_ctx *);
|
||||||
void client_draw_border(struct client_ctx *);
|
void client_draw_border(struct client_ctx *);
|
||||||
struct client_ctx *client_find(Window);
|
struct client_ctx *client_find(Window);
|
||||||
void client_freeze(struct client_ctx *);
|
void client_freeze(struct client_ctx *);
|
||||||
|
11
client.c
11
client.c
@ -149,20 +149,11 @@ client_init(Window win, struct screen_ctx *sc, int mapped)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
client_delete(struct client_ctx *cc, int destroy)
|
client_delete(struct client_ctx *cc)
|
||||||
{
|
{
|
||||||
struct screen_ctx *sc = cc->sc;
|
struct screen_ctx *sc = cc->sc;
|
||||||
struct winname *wn;
|
struct winname *wn;
|
||||||
|
|
||||||
if (destroy) {
|
|
||||||
XGrabServer(X_Dpy);
|
|
||||||
cc->state = WithdrawnState;
|
|
||||||
xu_set_wm_state(cc->win, cc->state);
|
|
||||||
XRemoveFromSaveSet(X_Dpy, cc->win);
|
|
||||||
XSync(X_Dpy, False);
|
|
||||||
XUngrabServer(X_Dpy);
|
|
||||||
}
|
|
||||||
|
|
||||||
TAILQ_REMOVE(&sc->mruq, cc, mru_entry);
|
TAILQ_REMOVE(&sc->mruq, cc, mru_entry);
|
||||||
TAILQ_REMOVE(&Clientq, cc, entry);
|
TAILQ_REMOVE(&Clientq, cc, entry);
|
||||||
|
|
||||||
|
20
xevents.c
20
xevents.c
@ -97,27 +97,15 @@ static void
|
|||||||
xev_handle_unmapnotify(XEvent *ee)
|
xev_handle_unmapnotify(XEvent *ee)
|
||||||
{
|
{
|
||||||
XUnmapEvent *e = &ee->xunmap;
|
XUnmapEvent *e = &ee->xunmap;
|
||||||
XEvent ev;
|
|
||||||
struct client_ctx *cc;
|
struct client_ctx *cc;
|
||||||
|
|
||||||
/* XXX, we need a recursive locking wrapper around grab server */
|
|
||||||
XGrabServer(X_Dpy);
|
|
||||||
if ((cc = client_find(e->window)) != NULL) {
|
if ((cc = client_find(e->window)) != NULL) {
|
||||||
/*
|
if (e->send_event) {
|
||||||
* If it's going to die anyway, nuke it.
|
cc->state = WithdrawnState;
|
||||||
*
|
xu_set_wm_state(cc->win, cc->state);
|
||||||
* Else, if it's a synthetic event delete state, since they
|
|
||||||
* want it to be withdrawn. ICCM recommends you withdraw on
|
|
||||||
* this even if we haven't alredy been told to iconify, to
|
|
||||||
* deal with legacy clients.
|
|
||||||
*/
|
|
||||||
if (XCheckTypedWindowEvent(X_Dpy, cc->win,
|
|
||||||
DestroyNotify, &ev) || e->send_event != 0) {
|
|
||||||
client_delete(cc, 1);
|
|
||||||
} else
|
} else
|
||||||
client_hide(cc);
|
client_hide(cc);
|
||||||
}
|
}
|
||||||
XUngrabServer(X_Dpy);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -127,7 +115,7 @@ xev_handle_destroynotify(XEvent *ee)
|
|||||||
struct client_ctx *cc;
|
struct client_ctx *cc;
|
||||||
|
|
||||||
if ((cc = client_find(e->window)) != NULL)
|
if ((cc = client_find(e->window)) != NULL)
|
||||||
client_delete(cc, 0);
|
client_delete(cc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
Loading…
Reference in New Issue
Block a user