[librsvg] (#276) - Guard against getting a cairo_t in an error state at the toplevel API
- From: Federico Mena Quintero <federico src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [librsvg] (#276) - Guard against getting a cairo_t in an error state at the toplevel API
- Date: Fri, 1 Jun 2018 15:14:47 +0000 (UTC)
commit 2d8ecd19288d2c6b728f3d91baf04493490fbe9c
Author: Federico Mena Quintero <federico gnome org>
Date: Fri Jun 1 10:13:58 2018 -0500
(#276) - Guard against getting a cairo_t in an error state at the toplevel API
The cairo-dock program was passing a cairo_t in an error state to
rsvg_handle_render_cairo(), and so we failed deep in the innards of
librsvg when cairo-rs validates the cairo::Context's status.
Cairo-dock was doing something like
surf = cairo_image_surface_create (... an invalid size ...);
cr = cairo_create (surf);
rsvg_handle_render_cairo (handle, cr); // we now catch the error right here
We catch invalid cr's, emit a warning, and return FALSE from
rsvg_handle_render_cairo*().
https://gitlab.gnome.org/GNOME/librsvg/issues/276
librsvg/rsvg-handle.c | 11 +++++++++++
tests/api.c | 27 +++++++++++++++++++++++++++
2 files changed, 38 insertions(+)
---
diff --git a/librsvg/rsvg-handle.c b/librsvg/rsvg-handle.c
index 7c7e288c..cb4537d8 100644
--- a/librsvg/rsvg-handle.c
+++ b/librsvg/rsvg-handle.c
@@ -790,16 +790,27 @@ rsvg_handle_render_cairo_sub (RsvgHandle * handle, cairo_t * cr, const char *id)
{
RsvgDrawingCtx *draw;
RsvgNode *drawsub = NULL;
+ cairo_status_t status;
g_return_val_if_fail (handle != NULL, FALSE);
if (handle->priv->hstate != RSVG_HANDLE_STATE_CLOSED_OK)
return FALSE;
+ status = cairo_status (cr);
+
+ if (status != CAIRO_STATUS_SUCCESS) {
+ g_warning ("cannot render on a cairo_t with a failure status (status=%d, %s)",
+ (int) status,
+ cairo_status_to_string (status));
+ return FALSE;
+ }
+
if (id && *id)
drawsub = rsvg_defs_lookup (handle->priv->defs, id);
if (drawsub == NULL && id != NULL) {
+ g_warning ("element id=\"%s\" does not exist", id);
/* todo: there's no way to signal that @id doesn't exist */
return FALSE;
}
diff --git a/tests/api.c b/tests/api.c
index 018cdee8..75b5d7af 100644
--- a/tests/api.c
+++ b/tests/api.c
@@ -465,6 +465,32 @@ dimensions_and_position (void)
g_object_unref (handle);
}
+static void
+detects_cairo_context_in_error (void)
+{
+ if (g_test_subprocess ()) {
+ char *filename = get_test_filename ("example.svg");
+ GError *error = NULL;
+
+ RsvgHandle *handle = rsvg_handle_new_from_file (filename, &error);
+ g_assert (handle != NULL);
+ g_assert (error == NULL);
+
+ /* this is wrong; it is to simulate creating a surface and a cairo_t in error */
+ cairo_surface_t *surf = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, -1, -1);
+ cairo_t *cr = cairo_create (surf);
+
+ /* rsvg_handle_render_cairo() should return FALSE when it gets a cr in an error state */
+ g_assert (!rsvg_handle_render_cairo (handle, cr));
+
+ return;
+ }
+
+ g_test_trap_subprocess (NULL, 0, 0);
+ g_test_trap_assert_failed ();
+ g_test_trap_assert_stderr ("*WARNING*cannot render on a cairo_t with a failure status*");
+}
+
int
main (int argc, char **argv)
{
@@ -491,6 +517,7 @@ main (int argc, char **argv)
g_test_add_func ("/api/handle_get_pixbuf", handle_get_pixbuf);
g_test_add_func ("/api/handle_get_pixbuf_sub", handle_get_pixbuf_sub);
g_test_add_func ("/api/dimensions_and_position", dimensions_and_position);
+ g_test_add_func ("/api/detects_cairo_context_in_error", detects_cairo_context_in_error);
return g_test_run ();
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]