Make path names in cache files absolute (NB, cache format change) Stop

permitting cache files to be stored in font dirs. Bump cache magic.
    Don't include /fonts.cache-2 in cache hash construction.
reviewed by: Patrick Lam <plam@mit.edu>
This commit is contained in:
Keith Packard 2006-04-27 07:11:44 +00:00
parent 3b013a034a
commit c1c3ba06d5
7 changed files with 57 additions and 232 deletions

View File

@ -1,3 +1,33 @@
2006-04-27 Keith Packard <keithp@keithp.com>
reviewed by: Patrick Lam <plam@mit.edu>
* src/fccache.c: (FcDirCacheUnlink), (FcDirCacheHashName),
(FcDirCacheOpen), (FcDirCacheWrite):
* src/fcfreetype.c: (FcFreeTypeQuery):
* src/fcint.h:
* src/fclist.c: (FcListAppend):
* src/fcmatch.c: (FcFontRenderPrepare):
* src/fcpat.c: (FcPatternDestroy), (FcPatternBaseFreeze),
(FcPatternFreeze), (FcPatternGet), (FcPatternDuplicate),
(FcStrUnserialize):
Make path names in cache files absolute (NB, cache format change)
Stop permitting cache files to be stored in font dirs.
Bump cache magic.
Don't include /fonts.cache-2 in cache hash construction.
2006-04-26 Keith Packard <keithp@keithp.com>
* src/fccache.c: (FcDirCacheUnlink), (FcDirCacheOpen),
(FcDirCacheWrite):
* src/fcfreetype.c: (FcFreeTypeQuery):
* src/fcint.h:
* src/fclist.c: (FcListAppend):
* src/fcmatch.c: (FcFontRenderPrepare):
* src/fcpat.c: (FcPatternDestroy), (FcPatternBaseFreeze),
(FcPatternFreeze), (FcPatternGet), (FcPatternDuplicate),
(FcStrUnserialize):
2006-04-25 Patrick Lam <plam@mit.edu> 2006-04-25 Patrick Lam <plam@mit.edu>
* ChangeLog: * ChangeLog:
* README: * README:

View File

@ -48,7 +48,7 @@ static int
FcDirCacheOpen (const FcChar8 * dir); FcDirCacheOpen (const FcChar8 * dir);
static char * static char *
FcDirCacheHashName (char * cache_file, int collisions); FcDirCacheHashName (const FcChar8 * dir, int collisions);
static off_t static off_t
FcCacheSkipToArch (int fd, const char * arch); FcCacheSkipToArch (int fd, const char * arch);
@ -828,29 +828,21 @@ FcDirCacheHasCurrentArch (const FcChar8 *dir)
FcBool FcBool
FcDirCacheUnlink (const FcChar8 *dir, FcConfig *config) FcDirCacheUnlink (const FcChar8 *dir, FcConfig *config)
{ {
char *cache_file;
char *cache_hashed = 0; char *cache_hashed = 0;
int fd, collisions; int fd, collisions;
struct stat cache_stat; struct stat cache_stat;
char name_buf[FC_MAX_FILE_LEN]; char dir_buf[FC_MAX_FILE_LEN];
dir = FcConfigNormalizeFontDir (config, dir); dir = FcConfigNormalizeFontDir (config, dir);
cache_file = (char *)FcStrPlus (dir, (FcChar8 *) "/" FC_DIR_CACHE_FILE);
if (!cache_file)
return FcFalse;
/* First remove normal cache file. */ /* Remove any applicable hashed files. */
if (stat ((char *) cache_file, &cache_stat) == 0)
unlink ((char *)cache_file);
/* Next remove any applicable hashed files. */
fd = -1; collisions = 0; fd = -1; collisions = 0;
do do
{ {
if (cache_hashed) if (cache_hashed)
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
cache_hashed = FcDirCacheHashName (cache_file, collisions++); cache_hashed = FcDirCacheHashName (dir, collisions++);
if (!cache_hashed) if (!cache_hashed)
goto bail; goto bail;
@ -860,16 +852,15 @@ FcDirCacheUnlink (const FcChar8 *dir, FcConfig *config)
if (fd == -1) if (fd == -1)
{ {
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
FcStrFree ((FcChar8 *)cache_file);
return FcTrue; return FcTrue;
} }
if (!FcCacheReadString (fd, name_buf, sizeof (name_buf)) || !strlen(name_buf)) if (!FcCacheReadString (fd, dir_buf, sizeof (dir_buf)) || !strlen(dir_buf))
{ {
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
goto bail; goto bail;
} }
} while (strcmp (name_buf, cache_file) != 0); } while (strcmp ((char *) dir_buf, (char *) dir) != 0);
close (fd); close (fd);
@ -880,12 +871,10 @@ FcDirCacheUnlink (const FcChar8 *dir, FcConfig *config)
goto bail; goto bail;
} }
FcStrFree ((FcChar8 *)cache_file);
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
return FcTrue; return FcTrue;
bail: bail:
FcStrFree ((FcChar8 *)cache_file);
return FcFalse; return FcFalse;
} }
@ -1024,7 +1013,7 @@ static const char bin2hex[] = { '0', '1', '2', '3',
'c', 'd', 'e', 'f' }; 'c', 'd', 'e', 'f' };
static char * static char *
FcDirCacheHashName (char * cache_file, int collisions) FcDirCacheHashName (const FcChar8 * dir, int collisions)
{ {
unsigned char hash[16], hex_hash[33]; unsigned char hash[16], hex_hash[33];
char *cache_hashed; char *cache_hashed;
@ -1034,7 +1023,7 @@ FcDirCacheHashName (char * cache_file, int collisions)
struct MD5Context ctx; struct MD5Context ctx;
MD5Init (&ctx); MD5Init (&ctx);
MD5Update (&ctx, (unsigned char *)cache_file, strlen (cache_file)); MD5Update (&ctx, (unsigned char *)dir, strlen ((char *) dir));
for (i = 0; i < collisions; i++) for (i = 0; i < collisions; i++)
MD5Update (&ctx, &uscore, 1); MD5Update (&ctx, &uscore, 1);
@ -1066,60 +1055,40 @@ FcDirCacheOpen (const FcChar8 *dir)
{ {
FcBool found; FcBool found;
int fd = -1, collisions = 0; int fd = -1, collisions = 0;
char *cache_file, *cache_hashed; char *cache_hashed;
char name_buf[FC_MAX_FILE_LEN]; char dir_buf[FC_MAX_FILE_LEN];
struct stat dir_stat; struct stat dir_stat;
if (stat ((char *)dir, &dir_stat) == -1) if (stat ((char *)dir, &dir_stat) == -1)
return -1; return -1;
cache_file = (char *)FcStrPlus (dir, (FcChar8 *) "/" FC_DIR_CACHE_FILE);
if (!cache_file)
return -1;
found = FcFalse; found = FcFalse;
do do
{ {
struct stat c; struct stat c;
FcChar8 *name_buf_dir;
if (fd >= 0) if (fd >= 0)
close (fd); close (fd);
cache_hashed = FcDirCacheHashName (cache_file, collisions++); cache_hashed = FcDirCacheHashName (dir, collisions++);
if (!cache_hashed) if (!cache_hashed)
{
FcStrFree ((FcChar8 *)cache_file);
return -1; return -1;
}
fd = open(cache_hashed, O_RDONLY | O_BINARY); fd = open(cache_hashed, O_RDONLY | O_BINARY);
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
if (fd == -1) if (fd == -1)
break; break;
if (!FcCacheReadString (fd, name_buf, sizeof (name_buf)) || if (!FcCacheReadString (fd, dir_buf, sizeof (dir_buf)) ||
!strlen(name_buf)) !strlen(dir_buf))
break; break;
name_buf_dir = FcStrDirname ((FcChar8 *)name_buf); if (stat ((char *)dir_buf, &c) == -1)
if (stat ((char *)name_buf_dir, &c) == -1)
{
FcStrFree (name_buf_dir);
continue; continue;
}
FcStrFree (name_buf_dir);
found = (c.st_ino == dir_stat.st_ino) && (c.st_dev == dir_stat.st_dev); found = (c.st_ino == dir_stat.st_ino) && (c.st_dev == dir_stat.st_dev);
} while (!found); } while (!found);
if (!found || fd < 0)
{
if (fd >= 0)
close (fd);
fd = open(cache_file, O_RDONLY | O_BINARY);
}
FcStrFree ((FcChar8 *)cache_file);
return fd; return fd;
} }
@ -1311,13 +1280,12 @@ FcDirCacheProduce (FcFontSet *set, FcCache *metadata)
FcBool FcBool
FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir) FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
{ {
char *cache_file;
char *cache_hashed; char *cache_hashed;
int fd, fd_orig, i, dirs_count; int fd, fd_orig, i, dirs_count;
FcAtomic *atomic; FcAtomic *atomic;
FcCache metadata; FcCache metadata;
off_t current_arch_start = 0, truncate_to; off_t current_arch_start = 0, truncate_to;
char name_buf[FC_MAX_FILE_LEN]; char dir_buf[FC_MAX_FILE_LEN];
int collisions; int collisions;
char *current_arch_machine_name, * header; char *current_arch_machine_name, * header;
@ -1327,17 +1295,13 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
if (!dir) if (!dir)
return FcFalse; return FcFalse;
cache_file = (char *)FcStrPlus (dir, (FcChar8 *) "/" FC_DIR_CACHE_FILE);
if (!cache_file)
goto bail;
/* Ensure that we're not trampling a cache for some other dir. */ /* Ensure that we're not trampling a cache for some other dir. */
/* This is slightly different from FcDirCacheOpen, since it /* This is slightly different from FcDirCacheOpen, since it
* needs the filename, not the file descriptor. */ * needs the filename, not the file descriptor. */
fd = -1; collisions = 0; fd = -1; collisions = 0;
do do
{ {
cache_hashed = FcDirCacheHashName (cache_file, collisions++); cache_hashed = FcDirCacheHashName (dir, collisions++);
if (!cache_hashed) if (!cache_hashed)
goto bail0; goto bail0;
@ -1346,7 +1310,7 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
fd = open(cache_hashed, O_RDONLY | O_BINARY); fd = open(cache_hashed, O_RDONLY | O_BINARY);
if (fd == -1) if (fd == -1)
break; break;
if(!FcCacheReadString (fd, name_buf, sizeof (name_buf)) || !strlen(name_buf)) if(!FcCacheReadString (fd, dir_buf, sizeof (dir_buf)) || !strlen(dir_buf))
{ {
close (fd); close (fd);
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
@ -1354,7 +1318,7 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
} }
close (fd); close (fd);
if (strcmp (name_buf, cache_file) != 0) if (strcmp (dir_buf, (char *) dir) != 0)
{ {
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
continue; continue;
@ -1369,38 +1333,24 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
goto bail1; goto bail1;
if (FcDebug () & FC_DBG_CACHE) if (FcDebug () & FC_DBG_CACHE)
printf ("FcDirCacheWriteDir cache_file \"%s\"\n", cache_file); printf ("FcDirCacheWriteDir dir \"%s\" file \"%s\"\n",
dir, cache_hashed);
atomic = FcAtomicCreate ((FcChar8 *)cache_hashed); atomic = FcAtomicCreate ((FcChar8 *)cache_hashed);
if (!atomic) if (!atomic)
goto bail1; goto bail1;
if (!FcAtomicLock (atomic)) if (!FcAtomicLock (atomic))
{
/* Now try rewriting the original version of the file. */
FcAtomicDestroy (atomic);
atomic = FcAtomicCreate ((FcChar8 *)cache_file);
fd_orig = open (cache_file, O_RDONLY | O_BINARY);
if (fd_orig == -1)
fd_orig = open((char *)FcAtomicOrigFile (atomic), O_RDONLY | O_BINARY);
fd = open((char *)FcAtomicNewFile (atomic), O_RDWR | O_CREAT | O_BINARY, 0666);
if (fd == -1)
goto bail2; goto bail2;
}
/* In all cases, try opening the real location of the cache file first. */ /* open the original file to save relevant portions */
/* (even if that's not atomic.) */
fd_orig = open (cache_file, O_RDONLY | O_BINARY);
if (fd_orig == -1)
fd_orig = open((char *)FcAtomicOrigFile (atomic), O_RDONLY | O_BINARY); fd_orig = open((char *)FcAtomicOrigFile (atomic), O_RDONLY | O_BINARY);
fd = open((char *)FcAtomicNewFile (atomic), O_RDWR | O_CREAT | O_BINARY, 0666); fd = open((char *)FcAtomicNewFile (atomic), O_RDWR | O_CREAT | O_BINARY, 0666);
if (fd == -1) if (fd == -1)
goto bail3; goto bail3;
FcCacheWriteString (fd, cache_file); FcCacheWriteString (fd, (char *) dir);
current_arch_machine_name = FcCacheMachineSignature (); current_arch_machine_name = FcCacheMachineSignature ();
current_arch_start = 0; current_arch_start = 0;
@ -1489,7 +1439,6 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
if (!FcAtomicReplaceOrig(atomic)) if (!FcAtomicReplaceOrig(atomic))
goto bail3; goto bail3;
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
FcStrFree ((FcChar8 *)cache_file);
FcAtomicUnlock (atomic); FcAtomicUnlock (atomic);
FcAtomicDestroy (atomic); FcAtomicDestroy (atomic);
return FcTrue; return FcTrue;
@ -1505,11 +1454,8 @@ FcDirCacheWrite (FcFontSet *set, FcStrSet *dirs, const FcChar8 *dir)
bail1: bail1:
FcStrFree ((FcChar8 *)cache_hashed); FcStrFree ((FcChar8 *)cache_hashed);
bail0: bail0:
unlink ((char *)cache_file);
FcStrFree ((FcChar8 *)cache_file);
if (current_dir_block) if (current_dir_block)
free (current_dir_block); free (current_dir_block);
bail:
return FcFalse; return FcFalse;
} }

View File

@ -997,7 +997,6 @@ FcFreeTypeQuery (const FcChar8 *file,
FcChar8 *family = 0; FcChar8 *family = 0;
#endif #endif
FcChar8 *complex; FcChar8 *complex;
FcChar8 *file2;
const FcChar8 *foundry = 0; const FcChar8 *foundry = 0;
int spacing; int spacing;
TT_OS2 *os2; TT_OS2 *os2;
@ -1278,15 +1277,8 @@ FcFreeTypeQuery (const FcChar8 *file,
printf ("Saving unique fullname %s\n", full); printf ("Saving unique fullname %s\n", full);
} }
file2 = FcStrCopy (file); if (!FcPatternAddString (pat, FC_FILE, file))
if (!FcPatternAddString (pat, FC_FILE, FcStrBasename(file2)))
{
FcStrFree (file2);
goto bail1; goto bail1;
}
FcStrFree (file2);
FcPatternAddFullFname (pat, (const char *)FcStrCopy (file));
if (!FcPatternAddInteger (pat, FC_INDEX, id)) if (!FcPatternAddInteger (pat, FC_INDEX, id))
goto bail1; goto bail1;

View File

@ -333,7 +333,7 @@ typedef struct _FcCaseFold {
* cache which is then rewritten to the users home directory * cache which is then rewritten to the users home directory
*/ */
#define FC_CACHE_MAGIC 0xFC02FC03 #define FC_CACHE_MAGIC 0xFC02FC04
typedef struct _FcGlobalCacheDir FcGlobalCacheDir; typedef struct _FcGlobalCacheDir FcGlobalCacheDir;
@ -871,12 +871,6 @@ FcPatternFini (void);
FcBool FcBool
FcPatternAppend (FcPattern *p, FcPattern *s); FcPatternAppend (FcPattern *p, FcPattern *s);
void
FcPatternAddFullFname (const FcPattern *p, const char *fname);
void
FcPatternTransferFullFname (const FcPattern *new, const FcPattern *orig);
const FcChar8 * const FcChar8 *
FcStrStaticName (const FcChar8 *name); FcStrStaticName (const FcChar8 *name);

View File

@ -422,10 +422,6 @@ FcListAppend (FcListHashTable *table,
else else
defidx = 0; defidx = 0;
/* Also, copy over the full path info. */
if (!strcmp (os->objects[o], FC_FILE))
FcPatternTransferFullFname (bucket->pattern, font);
e = FcPatternFindElt (font, os->objects[o]); e = FcPatternFindElt (font, os->objects[o]);
if (e) if (e)
{ {

View File

@ -495,9 +495,6 @@ FcFontRenderPrepare (FcConfig *config,
FcValueCanonicalize(&FcValueListPtrU(pe->values)->value), FcTrue); FcValueCanonicalize(&FcValueListPtrU(pe->values)->value), FcTrue);
} }
if (FcPatternFindElt (font, FC_FILE))
FcPatternTransferFullFname (new, font);
FcConfigSubstituteWithPat (config, new, pat, FcMatchFont); FcConfigSubstituteWithPat (config, new, pat, FcMatchFont);
return new; return new;
} }

View File

@ -39,16 +39,6 @@ FcPatternEltPtrCreateDynamic (FcPatternElt * e);
static FcBool static FcBool
FcStrHashed (const FcChar8 *name); FcStrHashed (const FcChar8 *name);
static const char *
FcPatternFindFullFname (const FcPattern *p);
/* If you are trying to duplicate an FcPattern which will be used for
* rendering, be aware that (internally) you also have to use
* FcPatternTransferFullFname to transfer the associated filename. If
* you are copying the font (externally) using FcPatternGetString,
* then everything's fine; this caveat only applies if you're copying
* the bits individually. */
FcPattern * FcPattern *
FcPatternCreate (void) FcPatternCreate (void)
{ {
@ -311,12 +301,6 @@ FcPatternDestroy (FcPattern *p)
if (p->ref == FC_REF_CONSTANT || --p->ref > 0) if (p->ref == FC_REF_CONSTANT || --p->ref > 0)
return; return;
if (FcPatternFindFullFname (p))
{
FcStrFree ((FcChar8 *)FcPatternFindFullFname (p));
FcPatternAddFullFname (p, 0);
}
for (i = 0; i < p->num; i++) for (i = 0; i < p->num; i++)
FcValueListDestroy ((FcPatternEltU(p->elts)+i)->values); FcValueListDestroy ((FcPatternEltU(p->elts)+i)->values);
@ -602,9 +586,6 @@ FcPatternBaseFreeze (FcPattern *b)
(FcPatternEltU(b->elts)+i)->object; (FcPatternEltU(b->elts)+i)->object;
} }
if (FcPatternFindElt (b, FC_FILE))
FcPatternTransferFullFname (ep, b);
ent->hash = hash; ent->hash = hash;
ent->next = *bucket; ent->next = *bucket;
*bucket = ent; *bucket = ent;
@ -679,9 +660,6 @@ FcPatternFreeze (FcPattern *p)
} }
} }
if (FcPatternFindElt (p, FC_FILE))
FcPatternTransferFullFname (b, p);
/* /*
* Freeze base * Freeze base
*/ */
@ -1112,39 +1090,6 @@ FcPatternAddLangSet (FcPattern *p, const char *object, const FcLangSet *ls)
return FcPatternAdd (p, object, v, FcTrue); return FcPatternAdd (p, object, v, FcTrue);
} }
static FcResult
FcPatternGetFile (const FcPattern *p, const char *object, int id, FcChar8 ** s)
{
const char *fn, *fpath;
FcChar8 *fname;
int size;
fn = FcPatternFindFullFname(p);
if (fn)
{
*s = (FcChar8 *) fn;
return FcResultMatch;
}
if (!p->bank)
return FcResultMatch;
fpath = FcCacheFindBankDir (p->bank);
size = strlen((char *)fpath) + 1 + strlen ((char *)*s) + 1;
fname = malloc (size);
if (!fname)
return FcResultOutOfMemory;
FcMemAlloc (FC_MEM_STRING, size);
strcpy ((char *)fname, (char *)fpath);
strcat ((char *)fname, "/");
strcat ((char *)fname, (char *)*s);
FcPatternAddFullFname (p, (const char *)fname);
*s = (FcChar8 *)fname;
return FcResultMatch;
}
FcResult FcResult
FcPatternGet (const FcPattern *p, const char *object, int id, FcValue *v) FcPatternGet (const FcPattern *p, const char *object, int id, FcValue *v)
{ {
@ -1159,12 +1104,6 @@ FcPatternGet (const FcPattern *p, const char *object, int id, FcValue *v)
if (!id) if (!id)
{ {
*v = FcValueCanonicalize(&FcValueListPtrU(l)->value); *v = FcValueCanonicalize(&FcValueListPtrU(l)->value);
/* Pull the FC_FILE trick here too. */
if (v->type == FcTypeString &&
FcObjectToPtr(object) == FcObjectToPtr(FC_FILE))
return FcPatternGetFile (p, object, id, (FcChar8 **)&(v->u.s));
return FcResultMatch; return FcResultMatch;
} }
id--; id--;
@ -1332,7 +1271,6 @@ FcPatternDuplicate (const FcPattern *orig)
FcTrue)) FcTrue))
goto bail1; goto bail1;
} }
FcPatternTransferFullFname (new, orig);
return new; return new;
@ -2025,71 +1963,3 @@ FcStrUnserialize (FcCache * metadata, void *block_ptr)
return block_ptr; return block_ptr;
} }
/* we don't store these in the FcPattern itself because
* we don't want to serialize the directory names */
/* I suppose this should be cleaned upon termination, too... */
typedef struct _FcPatternDirMapping {
const FcPattern *p;
const char *fname;
} FcPatternDirMapping;
#define PATTERNDIR_HASH_SIZE 31
static struct patternDirBucket {
struct patternDirBucket *next;
FcPatternDirMapping m;
} FcPatternDirBuckets[PATTERNDIR_HASH_SIZE];
void
FcPatternAddFullFname (const FcPattern *p, const char *fname)
{
struct patternDirBucket *pb;
/* N.B. FcPatternHash fails, since it's contents-based, not
* address-based, and we're in the process of mutating the FcPattern. */
for (pb = &FcPatternDirBuckets
[((unsigned long)p / sizeof (FcPattern *)) % PATTERNDIR_HASH_SIZE];
pb->m.p != p && pb->next;
pb = pb->next)
;
if (pb->m.p == p)
{
pb->m.fname = fname;
return;
}
pb->next = malloc (sizeof (struct patternDirBucket));
if (!pb->next)
return;
FcMemAlloc (FC_MEM_CACHE, sizeof (struct patternDirBucket));
pb->next->next = 0;
pb->next->m.p = p;
pb->next->m.fname = fname;
}
static const char *
FcPatternFindFullFname (const FcPattern *p)
{
struct patternDirBucket *pb;
for (pb = &FcPatternDirBuckets
[((unsigned long)p / sizeof (FcPattern *)) % PATTERNDIR_HASH_SIZE];
pb; pb = pb->next)
if (pb->m.p == p)
return pb->m.fname;
return 0;
}
void
FcPatternTransferFullFname (const FcPattern *new, const FcPattern *orig)
{
FcChar8 * s;
FcPatternGetString (orig, FC_FILE, 0, &s);
FcPatternAddFullFname (new,
(char *)FcStrCopy
((FcChar8 *)FcPatternFindFullFname(orig)));
}