diff options
Diffstat (limited to 'app/sanity.c')
-rw-r--r-- | app/sanity.c | 738 |
1 files changed, 738 insertions, 0 deletions
diff --git a/app/sanity.c b/app/sanity.c new file mode 100644 index 0000000..727482c --- /dev/null +++ b/app/sanity.c @@ -0,0 +1,738 @@ +/* GIMP - The GNU Image Manipulation Program + * Copyright (C) 1995 Spencer Kimball and Peter Mattis + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +#include "config.h" + +#include <cairo.h> +#include <fontconfig/fontconfig.h> +#include <pango/pango.h> +#include <pango/pangoft2.h> +#include <gdk-pixbuf/gdk-pixbuf.h> +#include <lcms2.h> +#include <gexiv2/gexiv2.h> +#include <gegl.h> + +#include "libgimpbase/gimpbase.h" + +#include "sanity.h" + +#include "gimp-intl.h" + + +/* early-stage tests */ +static gchar * sanity_check_gimp (void); +static gchar * sanity_check_glib (void); +static gchar * sanity_check_cairo (void); +static gchar * sanity_check_pango (void); +static gchar * sanity_check_fontconfig (void); +static gchar * sanity_check_freetype (void); +static gchar * sanity_check_gdk_pixbuf (void); +static gchar * sanity_check_lcms (void); +static gchar * sanity_check_gexiv2 (void); +static gchar * sanity_check_babl (void); +static gchar * sanity_check_gegl (void); +static gchar * sanity_check_filename_encoding (void); + +/* late-stage tests */ +static gchar * sanity_check_gegl_ops (void); + + +/* public functions */ + +/* early-stage sanity check, performed before the call to app_run(). */ +const gchar * +sanity_check_early (void) +{ + gchar *abort_message = NULL; + + if (! abort_message) + abort_message = sanity_check_gimp (); + + if (! abort_message) + abort_message = sanity_check_glib (); + + if (! abort_message) + abort_message = sanity_check_cairo (); + + if (! abort_message) + abort_message = sanity_check_pango (); + + if (! abort_message) + abort_message = sanity_check_fontconfig (); + + if (! abort_message) + abort_message = sanity_check_freetype (); + + if (! abort_message) + abort_message = sanity_check_gdk_pixbuf (); + + if (! abort_message) + abort_message = sanity_check_lcms (); + + if (! abort_message) + abort_message = sanity_check_gexiv2 (); + + if (! abort_message) + abort_message = sanity_check_babl (); + + if (! abort_message) + abort_message = sanity_check_gegl (); + + if (! abort_message) + abort_message = sanity_check_filename_encoding (); + + return abort_message; +} + +/* late-stage sanity check, performed during app_run(), after the user + * configuration has been loaded. + */ +const gchar * +sanity_check_late (void) +{ + gchar *abort_message = NULL; + + /* the gegl ops test initializes all gegl ops; in particular, it initializes + * all the strings used by their properties, which appear in the ui. it + * must be run after we've called language_init(), potentially overriding + * LANGUAGE according to the user config, or else all affected strings would + * use the translation corresponding to the system locale, regardless. + */ + if (! abort_message) + abort_message = sanity_check_gegl_ops (); + + return abort_message; +} + + +/* private functions */ + + +/* early-stage tests */ + +static gboolean +sanity_check_version (guint major_version, guint required_major, + guint minor_version, guint required_minor, + guint micro_version, guint required_micro) +{ + if (major_version > required_major) + return TRUE; + + if (major_version < required_major) + return FALSE; + + if (minor_version > required_minor) + return TRUE; + + if (minor_version < required_minor) + return FALSE; + + if (micro_version >= required_micro) + return TRUE; + + return FALSE; +} + +static gchar * +sanity_check_gimp (void) +{ + if (GIMP_MAJOR_VERSION != gimp_major_version || + GIMP_MINOR_VERSION != gimp_minor_version || + GIMP_MICRO_VERSION != gimp_micro_version) + { + return g_strdup_printf + ("Libgimp version mismatch!\n\n" + "The GIMP binary cannot run with a libgimp version\n" + "other than its own. This is GIMP %d.%d.%d, but the\n" + "libgimp version is %d.%d.%d.\n\n" + "Maybe you have GIMP versions in both /usr and /usr/local ?", + GIMP_MAJOR_VERSION, GIMP_MINOR_VERSION, GIMP_MICRO_VERSION, + gimp_major_version, gimp_minor_version, gimp_micro_version); + } + + return NULL; +} + +static gchar * +sanity_check_glib (void) +{ +#define GLIB_REQUIRED_MAJOR 2 +#define GLIB_REQUIRED_MINOR 56 +#define GLIB_REQUIRED_MICRO 2 + + const gchar *mismatch = glib_check_version (GLIB_REQUIRED_MAJOR, + GLIB_REQUIRED_MINOR, + GLIB_REQUIRED_MICRO); + + if (mismatch) + { + return g_strdup_printf + ("%s\n\n" + "GIMP requires GLib version %d.%d.%d or later.\n" + "Installed GLib version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older GLib version.\n\n" + "Please upgrade to GLib version %d.%d.%d or later.", + mismatch, + GLIB_REQUIRED_MAJOR, GLIB_REQUIRED_MINOR, GLIB_REQUIRED_MICRO, + glib_major_version, glib_minor_version, glib_micro_version, + GLIB_REQUIRED_MAJOR, GLIB_REQUIRED_MINOR, GLIB_REQUIRED_MICRO); + } + +#undef GLIB_REQUIRED_MAJOR +#undef GLIB_REQUIRED_MINOR +#undef GLIB_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_cairo (void) +{ +#define CAIRO_REQUIRED_MAJOR 1 +#define CAIRO_REQUIRED_MINOR 12 +#define CAIRO_REQUIRED_MICRO 2 + + if (cairo_version () < CAIRO_VERSION_ENCODE (CAIRO_REQUIRED_MAJOR, + CAIRO_REQUIRED_MINOR, + CAIRO_REQUIRED_MICRO)) + { + return g_strdup_printf + ("The Cairo version being used is too old!\n\n" + "GIMP requires Cairo version %d.%d.%d or later.\n" + "Installed Cairo version is %s.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older Cairo version.\n\n" + "Please upgrade to Cairo version %d.%d.%d or later.", + CAIRO_REQUIRED_MAJOR, CAIRO_REQUIRED_MINOR, CAIRO_REQUIRED_MICRO, + cairo_version_string (), + CAIRO_REQUIRED_MAJOR, CAIRO_REQUIRED_MINOR, CAIRO_REQUIRED_MICRO); + } + +#undef CAIRO_REQUIRED_MAJOR +#undef CAIRO_REQUIRED_MINOR +#undef CAIRO_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_pango (void) +{ +#define PANGO_REQUIRED_MAJOR 1 +#define PANGO_REQUIRED_MINOR 29 +#define PANGO_REQUIRED_MICRO 4 + + const gchar *mismatch = pango_version_check (PANGO_REQUIRED_MAJOR, + PANGO_REQUIRED_MINOR, + PANGO_REQUIRED_MICRO); + + if (mismatch) + { + const gint pango_major_version = pango_version () / 100 / 100; + const gint pango_minor_version = pango_version () / 100 % 100; + const gint pango_micro_version = pango_version () % 100; + + return g_strdup_printf + ("%s\n\n" + "GIMP requires Pango version %d.%d.%d or later.\n" + "Installed Pango version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older Pango version.\n\n" + "Please upgrade to Pango version %d.%d.%d or later.", + mismatch, + PANGO_REQUIRED_MAJOR, PANGO_REQUIRED_MINOR, PANGO_REQUIRED_MICRO, + pango_major_version, pango_minor_version, pango_micro_version, + PANGO_REQUIRED_MAJOR, PANGO_REQUIRED_MINOR, PANGO_REQUIRED_MICRO); + } + +#undef PANGO_REQUIRED_MAJOR +#undef PANGO_REQUIRED_MINOR +#undef PANGO_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_fontconfig (void) +{ + const gint fc_version = FcGetVersion (); + +#define FC_REQUIRED_MAJOR 2 +#define FC_REQUIRED_MINOR 2 +#define FC_REQUIRED_MICRO 0 + + if (fc_version < ((FC_REQUIRED_MAJOR * 10000) + + (FC_REQUIRED_MINOR * 100) + + (FC_REQUIRED_MICRO * 1))) + { + const gint fc_major_version = fc_version / 100 / 100; + const gint fc_minor_version = fc_version / 100 % 100; + const gint fc_micro_version = fc_version % 100; + + return g_strdup_printf + ("The Fontconfig version being used is too old!\n\n" + "GIMP requires Fontconfig version %d.%d.%d or later.\n" + "The Fontconfig version loaded by GIMP is %d.%d.%d.\n\n" + "This may be caused by another instance of libfontconfig.so.1\n" + "being installed in the system, probably in /usr/X11R6/lib.\n" + "Please correct the situation or report it to someone who can.", + FC_REQUIRED_MAJOR, FC_REQUIRED_MINOR, FC_REQUIRED_MICRO, + fc_major_version, fc_minor_version, fc_micro_version); + } + +#undef FC_REQUIRED_MAJOR +#undef FC_REQUIRED_MINOR +#undef FC_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_freetype (void) +{ + FT_Library ft_library; + FT_Int ft_major_version; + FT_Int ft_minor_version; + FT_Int ft_micro_version; + FT_Int ft_version; + +#define FT_REQUIRED_MAJOR 2 +#define FT_REQUIRED_MINOR 1 +#define FT_REQUIRED_MICRO 7 + + if (FT_Init_FreeType (&ft_library) != 0) + g_error ("FT_Init_FreeType() failed"); + + FT_Library_Version (ft_library, + &ft_major_version, + &ft_minor_version, + &ft_micro_version); + + if (FT_Done_FreeType (ft_library) != 0) + g_error ("FT_Done_FreeType() failed"); + + ft_version = (ft_major_version * 10000 + + ft_minor_version * 100 + + ft_micro_version * 1); + + if (ft_version < ((FT_REQUIRED_MAJOR * 10000) + + (FT_REQUIRED_MINOR * 100) + + (FT_REQUIRED_MICRO * 1))) + { + return g_strdup_printf + ("FreeType version too old!\n\n" + "GIMP requires FreeType version %d.%d.%d or later.\n" + "Installed FreeType version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older FreeType version.\n\n" + "Please upgrade to FreeType version %d.%d.%d or later.", + FT_REQUIRED_MAJOR, FT_REQUIRED_MINOR, FT_REQUIRED_MICRO, + ft_major_version, ft_minor_version, ft_micro_version, + FT_REQUIRED_MAJOR, FT_REQUIRED_MINOR, FT_REQUIRED_MICRO); + } + +#undef FT_REQUIRED_MAJOR +#undef FT_REQUIRED_MINOR +#undef FT_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_gdk_pixbuf (void) +{ +#define GDK_PIXBUF_REQUIRED_MAJOR 2 +#define GDK_PIXBUF_REQUIRED_MINOR 30 +#define GDK_PIXBUF_REQUIRED_MICRO 8 + + if (! sanity_check_version (gdk_pixbuf_major_version, GDK_PIXBUF_REQUIRED_MAJOR, + gdk_pixbuf_minor_version, GDK_PIXBUF_REQUIRED_MINOR, + gdk_pixbuf_micro_version, GDK_PIXBUF_REQUIRED_MICRO)) + { + return g_strdup_printf + ("GdkPixbuf version too old!\n\n" + "GIMP requires GdkPixbuf version %d.%d.%d or later.\n" + "Installed GdkPixbuf version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older GdkPixbuf version.\n\n" + "Please upgrade to GdkPixbuf version %d.%d.%d or later.", + GDK_PIXBUF_REQUIRED_MAJOR, GDK_PIXBUF_REQUIRED_MINOR, GDK_PIXBUF_REQUIRED_MICRO, + gdk_pixbuf_major_version, gdk_pixbuf_minor_version, gdk_pixbuf_micro_version, + GDK_PIXBUF_REQUIRED_MAJOR, GDK_PIXBUF_REQUIRED_MINOR, GDK_PIXBUF_REQUIRED_MICRO); + } + +#undef GDK_PIXBUF_REQUIRED_MAJOR +#undef GDK_PIXBUF_REQUIRED_MINOR +#undef GDK_PIXBUF_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_lcms (void) +{ +#define LCMS_REQUIRED_MAJOR 2 +#define LCMS_REQUIRED_MINOR 8 + + gint lcms_version = cmsGetEncodedCMMversion (); + + if (lcms_version < (LCMS_REQUIRED_MAJOR * 1000 + + LCMS_REQUIRED_MINOR * 10)) + { + const gint lcms_major_version = lcms_version / 1000; + const gint lcms_minor_version = lcms_version % 1000 / 10; + + return g_strdup_printf + ("Liblcms2 version too old!\n\n" + "GIMP requires LittleCMS version %d.%d or later.\n" + "Installed LittleCMS version is %d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older LittleCMS version.\n\n" + "Please upgrade to LittleCMS version %d.%d or later.", + LCMS_REQUIRED_MAJOR, LCMS_REQUIRED_MINOR, + lcms_major_version, lcms_minor_version, + LCMS_REQUIRED_MAJOR, LCMS_REQUIRED_MINOR); + } + +#undef LCMS_REQUIRED_MAJOR +#undef LCMS_REQUIRED_MINOR + + return NULL; +} + +static gchar * +sanity_check_gexiv2 (void) +{ +#ifdef GEXIV2_MAJOR_VERSION + +#define GEXIV2_REQUIRED_MAJOR 0 +#define GEXIV2_REQUIRED_MINOR 10 +#define GEXIV2_REQUIRED_MICRO 6 + + gint gexiv2_version = gexiv2_get_version (); + + if (gexiv2_version < (GEXIV2_REQUIRED_MAJOR * 100 * 100 + + GEXIV2_REQUIRED_MINOR * 100 + + GEXIV2_REQUIRED_MICRO)) + { + const gint gexiv2_major_version = gexiv2_version / 100 / 100; + const gint gexiv2_minor_version = gexiv2_version / 100 % 100; + const gint gexiv2_micro_version = gexiv2_version % 100; + + return g_strdup_printf + ("gexiv2 version too old!\n\n" + "GIMP requires gexiv2 version %d.%d.%d or later.\n" + "Installed gexiv2 version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older gexiv2 version.\n\n" + "Please upgrade to gexiv2 version %d.%d.%d or later.", + GEXIV2_REQUIRED_MAJOR, GEXIV2_REQUIRED_MINOR, GEXIV2_REQUIRED_MICRO, + gexiv2_major_version, gexiv2_minor_version, gexiv2_micro_version, + GEXIV2_REQUIRED_MAJOR, GEXIV2_REQUIRED_MINOR, GEXIV2_REQUIRED_MICRO); + } + +#undef GEXIV2_REQUIRED_MAJOR +#undef GEXIV2_REQUIRED_MINOR +#undef GEXIV2_REQUIRED_MICRO + +#endif + + return NULL; +} + +static gchar * +sanity_check_babl (void) +{ + gint babl_major_version; + gint babl_minor_version; + gint babl_micro_version; + +#define BABL_REQUIRED_MAJOR 0 +#define BABL_REQUIRED_MINOR 1 +#define BABL_REQUIRED_MICRO 78 + + babl_get_version (&babl_major_version, + &babl_minor_version, + &babl_micro_version); + + if (! sanity_check_version (babl_major_version, BABL_REQUIRED_MAJOR, + babl_minor_version, BABL_REQUIRED_MINOR, + babl_micro_version, BABL_REQUIRED_MICRO)) + { + return g_strdup_printf + ("BABL version too old!\n\n" + "GIMP requires BABL version %d.%d.%d or later.\n" + "Installed BABL version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older BABL version.\n\n" + "Please upgrade to BABL version %d.%d.%d or later.", + BABL_REQUIRED_MAJOR, BABL_REQUIRED_MINOR, BABL_REQUIRED_MICRO, + babl_major_version, babl_minor_version, babl_micro_version, + BABL_REQUIRED_MAJOR, BABL_REQUIRED_MINOR, BABL_REQUIRED_MICRO); + } + +#undef BABL_REQUIRED_MAJOR +#undef BABL_REQUIRED_MINOR +#undef BABL_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_gegl (void) +{ + gint gegl_major_version; + gint gegl_minor_version; + gint gegl_micro_version; + +#define GEGL_REQUIRED_MAJOR 0 +#define GEGL_REQUIRED_MINOR 4 +#define GEGL_REQUIRED_MICRO 38 + + gegl_get_version (&gegl_major_version, + &gegl_minor_version, + &gegl_micro_version); + + if (! sanity_check_version (gegl_major_version, GEGL_REQUIRED_MAJOR, + gegl_minor_version, GEGL_REQUIRED_MINOR, + gegl_micro_version, GEGL_REQUIRED_MICRO)) + { + return g_strdup_printf + ("GEGL version too old!\n\n" + "GIMP requires GEGL version %d.%d.%d or later.\n" + "Installed GEGL version is %d.%d.%d.\n\n" + "Somehow you or your software packager managed\n" + "to install GIMP with an older GEGL version.\n\n" + "Please upgrade to GEGL version %d.%d.%d or later.", + GEGL_REQUIRED_MAJOR, GEGL_REQUIRED_MINOR, GEGL_REQUIRED_MICRO, + gegl_major_version, gegl_minor_version, gegl_micro_version, + GEGL_REQUIRED_MAJOR, GEGL_REQUIRED_MINOR, GEGL_REQUIRED_MICRO); + } + +#undef GEGL_REQUIRED_MAJOR +#undef GEGL_REQUIRED_MINOR +#undef GEGL_REQUIRED_MICRO + + return NULL; +} + +static gchar * +sanity_check_filename_encoding (void) +{ + gchar *result; + GError *error = NULL; + + result = g_filename_to_utf8 ("", -1, NULL, NULL, &error); + + if (! result) + { + gchar *msg = + g_strdup_printf + (_("The configured filename encoding cannot be converted to UTF-8: " + "%s\n\n" + "Please check the value of the environment variable " + "G_FILENAME_ENCODING."), + error->message); + + g_error_free (error); + + return msg; + } + + g_free (result); + + result = g_filename_to_utf8 (gimp_directory (), -1, NULL, NULL, &error); + + if (! result) + { + gchar *msg = + g_strdup_printf + (_("The name of the directory holding the GIMP user configuration " + "cannot be converted to UTF-8: " + "%s\n\n" + "Your filesystem probably stores files in an encoding " + "other than UTF-8 and you didn't tell GLib about this. " + "Please set the environment variable G_FILENAME_ENCODING."), + error->message); + + g_error_free (error); + + return msg; + } + + g_free (result); + + return NULL; +} + + +/* late-stage tests */ + +static gchar * +sanity_check_gegl_ops (void) +{ + static const gchar *required_ops[] = + { + "gegl:alien-map", + "gegl:bayer-matrix", + "gegl:bloom", + "gegl:buffer-sink", + "gegl:buffer-source", + "gegl:c2g", + "gegl:cache", + "gegl:cartoon", + "gegl:cell-noise", + "gegl:checkerboard", + "gegl:color", + "gegl:color-enhance", + "gegl:color-exchange", + "gegl:color-rotate", + "gegl:color-temperature", + "gegl:color-to-alpha", + "gegl:component-extract", + "gegl:convolution-matrix", + "gegl:copy-buffer", + "gegl:crop", + "gegl:cubism", + "gegl:deinterlace", + "gegl:difference-of-gaussians", + "gegl:diffraction-patterns", + "gegl:displace", + "gegl:distance-transform", + "gegl:dither", + "gegl:dropshadow", + "gegl:edge", + "gegl:edge-laplace", + "gegl:edge-neon", + "gegl:edge-sobel", + "gegl:emboss", + "gegl:engrave", + "gegl:exposure", + "gegl:fattal02", + "gegl:focus-blur", + "gegl:fractal-trace", + "gegl:gaussian-blur", + "gegl:gaussian-blur-selective", + "gegl:gegl", + "gegl:grid", + "gegl:high-pass", + "gegl:hue-chroma", + "gegl:illusion", + "gegl:image-gradient", + "gegl:invert-gamma", + "gegl:invert-linear", + "gegl:lens-blur", + "gegl:lens-distortion", + "gegl:lens-flare", + "gegl:linear-sinusoid", + "gegl:long-shadow", + "gegl:mantiuk06", + "gegl:map-absolute", + "gegl:map-relative", + "gegl:matting-global", + "gegl:maze", + "gegl:mean-curvature-blur", + "gegl:median-blur", + "gegl:mirrors", + "gegl:mono-mixer", + "gegl:mosaic", + "gegl:motion-blur-circular", + "gegl:motion-blur-linear", + "gegl:motion-blur-zoom", + "gegl:newsprint", + "gegl:noise-cie-lch", + "gegl:noise-hsv", + "gegl:noise-hurl", + "gegl:noise-pick", + "gegl:noise-rgb", + "gegl:noise-slur", + "gegl:noise-solid", + "gegl:noise-spread", + "gegl:normal-map", + "gegl:npd", + "gegl:oilify", + "gegl:opacity", + "gegl:over", + "gegl:panorama-projection", + "gegl:perlin-noise", + "gegl:photocopy", + "gegl:pixelize", + "gegl:polar-coordinates", + "gegl:recursive-transform", + "gegl:red-eye-removal", + "gegl:reinhard05", + "gegl:rgb-clip", + "gegl:ripple", + "gegl:saturation", + "gegl:scale-ratio", + "gegl:seamless-clone", + "gegl:sepia", + "gegl:shadows-highlights", + "gegl:shift", + "gegl:simplex-noise", + "gegl:shift", + "gegl:sinus", + "gegl:slic", + "gegl:snn-mean", + "gegl:softglow", + "gegl:spherize", + "gegl:spiral", + "gegl:stereographic-projection", + "gegl:stretch-contrast", + "gegl:stretch-contrast-hsv", + "gegl:stress", + "gegl:supernova", + "gegl:threshold", + "gegl:tile", + "gegl:tile-paper", + "gegl:tile-glass", + "gegl:tile-seamless", + "gegl:transform", + "gegl:translate", + "gegl:unsharp-mask", + "gegl:value-invert", + "gegl:value-propagate", + "gegl:variable-blur", + "gegl:video-degradation", + "gegl:vignette", + "gegl:warp", + "gegl:waterpixels", + "gegl:wavelet-blur", + "gegl:waves", + "gegl:whirl-pinch", + "gegl:write-buffer" + }; + + gint i; + + for (i = 0; i < G_N_ELEMENTS (required_ops); i++) + { + if (! gegl_has_operation (required_ops[i])) + { + return g_strdup_printf + ("GEGL operation missing!\n\n" + "GIMP requires the GEGL operation \"%s\".\n" + "This operation cannot be found. Check your\n" + "GEGL install and ensure it has been compiled\n" + "with any dependencies required for GIMP.", + required_ops [i]); + } + } + + return NULL; +} |