* src/image.c (free_image):
* src/xfaces.c (free_realized_face): Handle scenarios where
free_frame_faces is called with the display connection cut.
* src/xterm.c (x_free_frame_resources): Call free_frame_faces
unconditionally, lest fontsets for this dead frame contaminate
Vfontset_list and produce crashes afterwards. (bug#66151)
(cherry picked from commit
9fc698479feef6fa660ff13e21619ea50bd404df)
c->images[img->id] = NULL;
#if !defined USE_CAIRO && defined HAVE_XRENDER
- if (img->picture)
- XRenderFreePicture (FRAME_X_DISPLAY (f), img->picture);
- if (img->mask_picture)
- XRenderFreePicture (FRAME_X_DISPLAY (f), img->mask_picture);
-#endif
+ /* FRAME_X_DISPLAY (f) could be NULL if this is being called from
+ the display IO error handler.*/
+
+ if (FRAME_X_DISPLAY (f))
+ {
+ if (img->picture)
+ XRenderFreePicture (FRAME_X_DISPLAY (f),
+ img->picture);
+ if (img->mask_picture)
+ XRenderFreePicture (FRAME_X_DISPLAY (f),
+ img->mask_picture);
+ }
+#endif /* !USE_CAIRO && HAVE_XRENDER */
+
+#ifdef HAVE_X_WINDOWS
+ if (FRAME_X_DISPLAY (f))
+#endif /* HAVE_X_WINDOWS */
+ /* Free resources, then free IMG. */
+ img->type->free_img (f, img);
- /* Free resources, then free IMG. */
- img->type->free_img (f, img);
xfree (img->face_font_family);
xfree (img);
}
/* Free fontset of FACE if it is ASCII face. */
if (face->fontset >= 0 && face == face->ascii_face)
free_face_fontset (f, face);
+
+#ifdef HAVE_X_WINDOWS
+ /* This function might be called with the frame's display
+ connection deleted, in which event the callbacks below
+ should not be executed, as they generate X requests. */
+ if (FRAME_X_DISPLAY (f))
+ return;
+#endif /* HAVE_X_WINDOWS */
+
if (face->gc)
{
block_input ();
xi_unlink_touch_points (f);
#endif
+ /* We must free faces before destroying windows because some
+ font-driver (e.g. xft) access a window while finishing a face.
+
+ This function must be called to remove this frame's fontsets from
+ Vfontset_list, and is itself responsible for not issuing X requests
+ if the connection has already been terminated. Otherwise, a future
+ call to a function that iterates over all existing fontsets might
+ crash, as they are not prepared to receive dead frames.
+ (bug#66151) */
+ free_frame_faces (f);
+
/* If a display connection is dead, don't try sending more
commands to the X server. */
if (dpyinfo->display)
if (f->pointer_invisible)
XTtoggle_invisible_pointer (f, 0);
- /* We must free faces before destroying windows because some
- font-driver (e.g. xft) access a window while finishing a
- face. */
- free_frame_faces (f);
tear_down_x_back_buffer (f);
if (f->output_data.x->icon_desc)