/* * fontconfig/test/test-issue107.c * * Copyright © 2000 Keith Packard * Copyright © 2018 Akira TAGOH * * Permission to use, copy, modify, distribute, and sell this software and its * documentation for any purpose is hereby granted without fee, provided that * the above copyright notice appear in all copies and that both that * copyright notice and this permission notice appear in supporting * documentation, and that the name of the author(s) not be used in * advertising or publicity pertaining to distribution of the software without * specific, written prior permission. The authors make no * representations about the suitability of this software for any purpose. It * is provided "as is" without express or implied warranty. * * THE AUTHOR(S) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO * EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY SPECIAL, INDIRECT OR * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR * PERFORMANCE OF THIS SOFTWARE. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #ifdef HAVE_UNISTD_H #include #endif #ifndef HAVE_STRUCT_DIRENT_D_TYPE #include #include #endif #include #ifdef _WIN32 # define FC_DIR_SEPARATOR '\\' # define FC_DIR_SEPARATOR_S "\\" #else # define FC_DIR_SEPARATOR '/' # define FC_DIR_SEPARATOR_S "/" #endif #ifdef _WIN32 #include #define mkdir(path,mode) _mkdir(path) int setenv(const char *name, const char *value, int o) { size_t len = strlen(name) + strlen(value) + 1; char *s = malloc(len+1); int ret; snprintf(s, len, "%s=%s", name, value); ret = _putenv(s); free(s); return ret; } #endif extern FcChar8 *FcConfigRealFilename (FcConfig *, FcChar8 *); #ifdef HAVE_MKDTEMP #define fc_mkdtemp mkdtemp #else char * fc_mkdtemp (char *template) { if (!mktemp (template) || mkdir (template, 0700)) return NULL; return template; } #endif FcBool mkdir_p (const char *dir) { char *parent; FcBool ret; if (strlen (dir) == 0) return FcFalse; parent = (char *) FcStrDirname ((const FcChar8 *) dir); if (!parent) return FcFalse; if (access (parent, F_OK) == 0) ret = mkdir (dir, 0755) == 0 && chmod (dir, 0755) == 0; else if (access (parent, F_OK) == -1) ret = mkdir_p (parent) && (mkdir (dir, 0755) == 0) && chmod (dir, 0755) == 0; else ret = FcFalse; free (parent); return ret; } FcBool unlink_dirs (const char *dir) { DIR *d = opendir (dir); struct dirent *e; size_t len = strlen (dir); char *n = NULL; FcBool ret = FcTrue; #ifndef HAVE_STRUCT_DIRENT_D_TYPE struct stat statb; #endif if (!d) return FcFalse; while ((e = readdir (d)) != NULL) { size_t l; if (strcmp (e->d_name, ".") == 0 || strcmp (e->d_name, "..") == 0) continue; l = strlen (e->d_name) + 1; if (n) free (n); n = malloc (l + len + 1); if (!n) { ret = FcFalse; break; } strcpy (n, dir); n[len] = FC_DIR_SEPARATOR; strcpy (&n[len + 1], e->d_name); #ifdef HAVE_STRUCT_DIRENT_D_TYPE if (e->d_type == DT_DIR) #else if (stat (n, &statb) == -1) { fprintf (stderr, "E: %s\n", n); ret = FcFalse; break; } if (S_ISDIR (statb.st_mode)) #endif { if (!unlink_dirs (n)) { fprintf (stderr, "E: %s\n", n); ret = FcFalse; break; } } else { if (unlink (n) == -1) { fprintf (stderr, "E: %s\n", n); ret = FcFalse; break; } } } if (n) free (n); closedir (d); if (rmdir (dir) == -1) { fprintf (stderr, "E: %s\n", dir); return FcFalse; } return ret; } int main(void) { FcConfig *cfg = FcConfigCreate (); char *basedir = NULL, template[512] = "/tmp/fc107-XXXXXX"; char *sysroot = NULL, systempl[512] = "/tmp/fc107-XXXXXX"; FcChar8 *d = NULL, *dd = NULL; FcCache *c = NULL; const FcChar8 *doc = (const FcChar8 *) "" "\n" " %s\n" "\n" ""; int retval = 0; size_t len; retval++; basedir = fc_mkdtemp (template); if (!basedir) { fprintf (stderr, "%s: %s\n", template, strerror (errno)); goto bail; } retval++; sysroot = fc_mkdtemp (systempl); if (!sysroot) { fprintf (stderr, "%s: %s\n", systempl, strerror (errno)); goto bail; } retval++; fprintf (stderr, "D: Creating %s\n", basedir); mkdir_p (basedir); len = strlen ((const char *) doc) + strlen (basedir) + 1; dd = malloc (len); snprintf ((char *) dd, len, (char *) doc, basedir); if (!FcConfigParseAndLoadFromMemory (cfg, dd, FcFalse)) { fprintf (stderr, "%s: Unable to load a config\n", basedir); goto bail; } sleep (1); c = FcDirCacheRead ((const FcChar8 *) basedir, FcFalse, cfg); FcDirCacheUnload (c); sleep (1); retval++; if (!FcConfigUptoDate (cfg)) { fprintf (stderr, "updated. need to reload.\n"); goto bail; } setenv ("FONTCONFIG_SYSROOT", sysroot, 1); fprintf (stderr, "D: Creating %s\n", sysroot); mkdir_p (sysroot); retval++; d = FcStrBuildFilename ((const FcChar8 *) sysroot, basedir, NULL); fprintf (stderr, "D: Creating %s\n", d); mkdir_p ((const char *) d); free (d); retval++; free (dd); len = strlen ((const char *) doc) + strlen (basedir) + 1; dd = malloc (len); snprintf ((char *) dd, len, (char *) doc, basedir); if (!FcConfigParseAndLoadFromMemory (cfg, dd, FcFalse)) { fprintf (stderr, "%s: Unable to load a config\n", basedir); goto bail; } sleep (1); c = FcDirCacheRead ((const FcChar8 *) basedir, FcFalse, cfg); FcDirCacheUnload (c); sleep (1); retval++; if (!FcConfigUptoDate (cfg)) { fprintf (stderr, "updated. need to reload (sysroot)\n"); goto bail; } retval = 0; bail: fprintf (stderr, "Cleaning up\n"); if (basedir) unlink_dirs (basedir); if (sysroot) unlink_dirs (sysroot); if (dd) free (dd); FcConfigDestroy (cfg); return retval; }