]> git.wh0rd.org - fontconfig.git/blobdiff - src/fcdir.c
Bug 44826 - <alias> must contain only a single <family>
[fontconfig.git] / src / fcdir.c
index a49cb54bbcf309e30f1047a082bdbb8e5dc49e12..8a2b97625066eeb05eeffc014cebb8a749056441 100644 (file)
 /*
- * $XFree86: $
+ * fontconfig/src/fcdir.c
  *
- * Copyright © 2000 Keith Packard, member of The XFree86 Project, Inc.
+ * Copyright © 2000 Keith Packard
  *
  * 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 Keith Packard not be used in
+ * 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.  Keith Packard makes no
+ * 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.
  *
- * KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
+ * THE AUTHOR(S) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
- * EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
+ * 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.
  */
 
-#include <sys/types.h>
-#include <dirent.h>
-#include <stdlib.h>
-#include <string.h>
 #include "fcint.h"
-
-#define FC_INVALID_FONT_FILE "."
+#include <dirent.h>
 
 FcBool
-FcFileScan (FcFontSet  *set,
-           FcFileCache *cache,
-           FcBlanks    *blanks,
-           const char  *file,
-           FcBool      force)
+FcFileIsDir (const FcChar8 *file)
+{
+    struct stat            statb;
+
+    if (FcStat (file, &statb) != 0)
+       return FcFalse;
+    return S_ISDIR(statb.st_mode);
+}
+
+static FcBool
+FcFileScanFontConfig (FcFontSet                *set,
+                     FcBlanks          *blanks,
+                     const FcChar8     *file,
+                     FcConfig          *config)
 {
-    int                    id;
-    char           *name;
-    FcPattern      *font;
-    FcBool         ret = FcTrue;
-    int                    count;
-    
+    FcPattern  *font;
+    FcBool     ret = FcTrue;
+    int                id;
+    int                count = 0;
+
     id = 0;
     do
     {
-       if (!force && cache)
-           name = FcFileCacheFind (cache, file, id, &count);
-       else
-           name = 0;
-       if (name)
+       font = 0;
+       /*
+        * Nothing in the cache, scan the file
+        */
+       if (FcDebug () & FC_DBG_SCAN)
        {
-           /* "." means the file doesn't contain a font */
-           if (strcmp (name, FC_INVALID_FONT_FILE) != 0)
-           {
-               font = FcNameParse (name);
-               if (font)
-                   FcPatternAddString (font, FC_FILE, file);
-           }
-           else
-               font = 0;
+           printf ("\tScanning file %s...", file);
+           fflush (stdout);
        }
-       else
+       font = FcFreeTypeQuery (file, id, blanks, &count);
+       if (FcDebug () & FC_DBG_SCAN)
+           printf ("done\n");
+
+       /*
+        * Edit pattern with user-defined rules
+        */
+       if (font && config && !FcConfigSubstitute (config, font, FcMatchScan))
        {
-           if (FcDebug () & FC_DBG_SCAN)
-           {
-               printf ("\tScanning file %s...", file);
-               fflush (stdout);
-           }
-           font = FcFreeTypeQuery (file, id, blanks, &count);
-           if (FcDebug () & FC_DBG_SCAN)
-               printf ("done\n");
-           if (!force && cache)
-           {
-               if (font)
-               {
-                   FcChar8     *unparse;
-
-                   unparse = FcNameUnparse (font);
-                   if (unparse)
-                   {
-                       (void) FcFileCacheUpdate (cache, file, id, unparse);
-                       free (unparse);
-                   }
-               }
-               else
-               {
-                   /* negative cache files not containing fonts */
-                   FcFileCacheUpdate (cache, file, id, FC_INVALID_FONT_FILE);
-               }
-           }
+           FcPatternDestroy (font);
+           font = NULL;
+           ret = FcFalse;
        }
+
+       /*
+        * Add the font
+        */
        if (font)
        {
+           if (FcDebug() & FC_DBG_SCANV)
+           {
+               printf ("Final font pattern:\n");
+               FcPatternPrint (font);
+           }
            if (!FcFontSetAdd (set, font))
            {
                FcPatternDestroy (font);
-               font = 0;
+               font = NULL;
                ret = FcFalse;
            }
        }
+       else if (font)
+           FcPatternDestroy (font);
        id++;
     } while (font && ret && id < count);
     return ret;
 }
 
 FcBool
-FcDirScan (FcFontSet   *set,
-          FcFileCache  *cache,
-          FcBlanks     *blanks,
-          const char   *dir,
-          FcBool       force)
+FcFileScanConfig (FcFontSet    *set,
+                 FcStrSet      *dirs,
+                 FcBlanks      *blanks,
+                 const FcChar8 *file,
+                 FcConfig      *config)
 {
-    DIR                    *d;
-    struct dirent   *e;
-    char           *file;
-    char           *base;
-    FcBool         ret = FcTrue;
-
-    file = (char *) malloc (strlen (dir) + 1 + 256 + 1);
-    if (!file)
-       return FcFalse;
+    if (FcFileIsDir (file))
+       return FcStrSetAdd (dirs, file);
+    else
+       return FcFileScanFontConfig (set, blanks, file, config);
+}
+
+FcBool
+FcFileScan (FcFontSet      *set,
+           FcStrSet        *dirs,
+           FcFileCache     *cache, /* XXX unused */
+           FcBlanks        *blanks,
+           const FcChar8   *file,
+           FcBool          force)
+{
+    return FcFileScanConfig (set, dirs, blanks, file, FcConfigGetCurrent ());
+}
+
+/*
+ * Strcmp helper that takes pointers to pointers, copied from qsort(3) manpage
+ */
+static int
+cmpstringp(const void *p1, const void *p2)
+{
+    return strcmp(* (char **) p1, * (char **) p2);
+}
+
+FcBool
+FcDirScanConfig (FcFontSet     *set,
+                FcStrSet       *dirs,
+                FcBlanks       *blanks,
+                const FcChar8  *dir,
+                FcBool         force, /* XXX unused */
+                FcConfig       *config)
+{
+    DIR                        *d;
+    struct dirent      *e;
+    FcStrSet           *files;
+    FcChar8            *file;
+    FcChar8            *base;
+    FcBool             ret = FcTrue;
+    int                        i;
 
-    strcpy (file, dir);
-    strcat (file, "/");
-    base = file + strlen (file);
     if (!force)
-    {
-       strcpy (base, FC_DIR_CACHE_FILE);
-       
-       if (FcFileCacheReadDir (set, file))
-       {
-           free (file);
-           return FcTrue;
-       }
+       return FcFalse;
+
+    if (!set && !dirs)
+       return FcTrue;
+
+    if (!blanks)
+       blanks = FcConfigGetBlanks (config);
+
+    /* freed below */
+    file = (FcChar8 *) malloc (strlen ((char *) dir) + 1 + FC_MAX_FILE_LEN + 1);
+    if (!file) {
+       ret = FcFalse;
+       goto bail;
     }
-    
-    d = opendir (dir);
+
+    strcpy ((char *) file, (char *) dir);
+    strcat ((char *) file, "/");
+    base = file + strlen ((char *) file);
+
+    if (FcDebug () & FC_DBG_SCAN)
+       printf ("\tScanning dir %s\n", dir);
+       
+    d = opendir ((char *) dir);
     if (!d)
     {
-       free (file);
-       return FcFalse;
+       /* Don't complain about missing directories */
+       if (errno != ENOENT)
+           ret = FcFalse;
+       goto bail;
     }
-    while (ret && (e = readdir (d)))
+
+    files = FcStrSetCreate ();
+    if (!files)
+    {
+       ret = FcFalse;
+       goto bail1;
+    }
+    while ((e = readdir (d)))
     {
-       if (e->d_name[0] != '.')
+       if (e->d_name[0] != '.' && strlen (e->d_name) < FC_MAX_FILE_LEN)
        {
-           strcpy (base, e->d_name);
-           FcFileScan (set, cache, blanks, file, force);
+           strcpy ((char *) base, (char *) e->d_name);
+           if (!FcStrSetAdd (files, file)) {
+               ret = FcFalse;
+               goto bail2;
+           }
        }
     }
-    free (file);
+
+    /*
+     * Sort files to make things prettier
+     */
+    qsort(files->strs, files->num, sizeof(FcChar8 *), cmpstringp);
+
+    /*
+     * Scan file files to build font patterns
+     */
+    for (i = 0; i < files->num; i++)
+       FcFileScanConfig (set, dirs, blanks, files->strs[i], config);
+
+bail2:
+    FcStrSetDestroy (files);
+bail1:
     closedir (d);
+bail:
     return ret;
 }
 
 FcBool
-FcDirSave (FcFontSet *set, const char *dir)
+FcDirScan (FcFontSet       *set,
+          FcStrSet         *dirs,
+          FcFileCache      *cache, /* XXX unused */
+          FcBlanks         *blanks,
+          const FcChar8    *dir,
+          FcBool           force /* XXX unused */)
 {
-    char           *file;
-    char           *base;
-    FcBool         ret;
-    
-    file = (char *) malloc (strlen (dir) + 1 + 256 + 1);
-    if (!file)
+    if (cache || !force)
        return FcFalse;
 
-    strcpy (file, dir);
-    strcat (file, "/");
-    base = file + strlen (file);
-    strcpy (base, FC_DIR_CACHE_FILE);
-    ret = FcFileCacheWriteDir (set, file);
-    free (file);
-    return ret;
+    return FcDirScanConfig (set, dirs, blanks, dir, force, FcConfigGetCurrent ());
+}
+
+/*
+ * Scan the specified directory and construct a cache of its contents
+ */
+FcCache *
+FcDirCacheScan (const FcChar8 *dir, FcConfig *config)
+{
+    FcStrSet           *dirs;
+    FcBool             ret = FcTrue;
+    FcFontSet          *set;
+    FcCache            *cache = NULL;
+    struct stat                dir_stat;
+
+    if (FcDebug () & FC_DBG_FONTSET)
+       printf ("cache scan dir %s\n", dir);
+
+    if (FcStat (dir, &dir_stat) < 0)
+    {
+       if (errno != ENOENT)
+           ret = FcFalse;
+       goto bail;
+    }
+
+    set = FcFontSetCreate();
+    if (!set)
+    {
+       ret = FcFalse;
+       goto bail;
+    }
+
+    dirs = FcStrSetCreate ();
+    if (!dirs)
+    {
+       ret = FcFalse;
+       goto bail1;
+    }
+
+    /*
+     * Scan the dir
+     */
+    if (!FcDirScanConfig (set, dirs, NULL, dir, FcTrue, config))
+    {
+       ret = FcFalse;
+       goto bail2;
+    }
+
+    /*
+     * Build the cache object
+     */
+    cache = FcDirCacheBuild (set, dir, &dir_stat, dirs);
+    if (!cache)
+    {
+       ret = FcFalse;
+       goto bail2;
+    }
+
+    /*
+     * Write out the cache file, ignoring any troubles
+     */
+    FcDirCacheWrite (cache, config);
+
+ bail2:
+    FcStrSetDestroy (dirs);
+ bail1:
+    FcFontSetDestroy (set);
+ bail:
+    return cache;
+}
+
+/*
+ * Read (or construct) the cache for a directory
+ */
+FcCache *
+FcDirCacheRead (const FcChar8 *dir, FcBool force, FcConfig *config)
+{
+    FcCache            *cache = NULL;
+
+    /* Try to use existing cache file */
+    if (!force)
+       cache = FcDirCacheLoad (dir, config, NULL);
+
+    /* Not using existing cache file, construct new cache */
+    if (!cache)
+       cache = FcDirCacheScan (dir, config);
+
+    return cache;
 }
 
+FcBool
+FcDirSave (FcFontSet *set, FcStrSet * dirs, const FcChar8 *dir)
+{
+    return FcFalse; /* XXX deprecated */
+}
+#define __fcdir__
+#include "fcaliastail.h"
+#undef __fcdir__