]> git.wh0rd.org - fontconfig.git/blob - fc-cat/fc-cat.c
Fix fc-cat again. Sigh.
[fontconfig.git] / fc-cat / fc-cat.c
1 /*
2 * $RCSId: xc/lib/fontconfig/fc-cache/fc-cache.c,v 1.8tsi Exp $
3 *
4 * Copyright © 2002 Keith Packard
5 *
6 * Permission to use, copy, modify, distribute, and sell this software and its
7 * documentation for any purpose is hereby granted without fee, provided that
8 * the above copyright notice appear in all copies and that both that
9 * copyright notice and this permission notice appear in supporting
10 * documentation, and that the name of Keith Packard not be used in
11 * advertising or publicity pertaining to distribution of the software without
12 * specific, written prior permission. Keith Packard makes no
13 * representations about the suitability of this software for any purpose. It
14 * is provided "as is" without express or implied warranty.
15 *
16 * KEITH PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
17 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
18 * EVENT SHALL KEITH PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR
19 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
20 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
21 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
22 * PERFORMANCE OF THIS SOFTWARE.
23 */
24
25 #ifdef HAVE_CONFIG_H
26 #include <config.h>
27 #else
28 #ifdef linux
29 #define HAVE_GETOPT_LONG 1
30 #endif
31 #define HAVE_GETOPT 1
32 #endif
33
34 #include <fontconfig/fontconfig.h>
35 #include <../src/fccache.c>
36 #include <stdio.h>
37 #include <stdlib.h>
38 #include <unistd.h>
39 #include <sys/types.h>
40 #include <sys/stat.h>
41 #include <errno.h>
42
43 #ifndef HAVE_GETOPT
44 #define HAVE_GETOPT 0
45 #endif
46 #ifndef HAVE_GETOPT_LONG
47 #define HAVE_GETOPT_LONG 0
48 #endif
49
50 #if HAVE_GETOPT_LONG
51 #undef _GNU_SOURCE
52 #define _GNU_SOURCE
53 #include <getopt.h>
54 const struct option longopts[] = {
55 {"version", 0, 0, 'V'},
56 {"verbose", 0, 0, 'v'},
57 {"help", 0, 0, '?'},
58 {NULL,0,0,0},
59 };
60 #else
61 #if HAVE_GETOPT
62 extern char *optarg;
63 extern int optind, opterr, optopt;
64 #endif
65 #endif
66
67 /*
68 * POSIX has broken stdio so that getc must do thread-safe locking,
69 * this is a serious performance problem for applications doing large
70 * amounts of IO with getc (as is done here). If available, use
71 * the getc_unlocked varient instead.
72 */
73
74 #if defined(getc_unlocked) || defined(_IO_getc_unlocked)
75 #define GETC(f) getc_unlocked(f)
76 #define PUTC(c,f) putc_unlocked(c,f)
77 #else
78 #define GETC(f) getc(f)
79 #define PUTC(c,f) putc(c,f)
80 #endif
81
82 FcBool
83 FcCachePrintSet (FcFontSet *set, FcStrSet *dirs, char *base_name);
84
85 static FcBool
86 FcCacheWriteChars (FILE *f, const FcChar8 *chars)
87 {
88 FcChar8 c;
89 while ((c = *chars++))
90 {
91 switch (c) {
92 case '"':
93 case '\\':
94 if (PUTC ('\\', f) == EOF)
95 return FcFalse;
96 /* fall through */
97 default:
98 if (PUTC (c, f) == EOF)
99 return FcFalse;
100 }
101 }
102 return FcTrue;
103 }
104
105 static FcBool
106 FcCacheWriteUlong (FILE *f, unsigned long t)
107 {
108 int pow;
109 unsigned long temp, digit;
110
111 temp = t;
112 pow = 1;
113 while (temp >= 10)
114 {
115 temp /= 10;
116 pow *= 10;
117 }
118 temp = t;
119 while (pow)
120 {
121 digit = temp / pow;
122 if (PUTC ((char) digit + '0', f) == EOF)
123 return FcFalse;
124 temp = temp - pow * digit;
125 pow = pow / 10;
126 }
127 return FcTrue;
128 }
129
130 static FcBool
131 FcCacheWriteInt (FILE *f, int i)
132 {
133 return FcCacheWriteUlong (f, (unsigned long) i);
134 }
135
136 static FcBool
137 FcCacheWriteStringOld (FILE *f, const FcChar8 *string)
138 {
139
140 if (PUTC ('"', f) == EOF)
141 return FcFalse;
142 if (!FcCacheWriteChars (f, string))
143 return FcFalse;
144 if (PUTC ('"', f) == EOF)
145 return FcFalse;
146 return FcTrue;
147 }
148
149 static void
150 usage (char *program)
151 {
152 #if HAVE_GETOPT_LONG
153 fprintf (stderr, "usage: %s [-V?] [--version] [--help] <fonts.cache-2>\n",
154 program);
155 #else
156 fprintf (stderr, "usage: %s [-fsvV?] <fonts.cache-2>\n",
157 program);
158 #endif
159 fprintf (stderr, "Reads font information caches in <fonts.cache-2>\n");
160 fprintf (stderr, "\n");
161 #if HAVE_GETOPT_LONG
162 fprintf (stderr, " -V, --version display font config version and exit\n");
163 fprintf (stderr, " -?, --help display this help and exit\n");
164 #else
165 fprintf (stderr, " -V (version) display font config version and exit\n");
166 fprintf (stderr, " -? (help) display this help and exit\n");
167 #endif
168 exit (1);
169 }
170
171 static int
172 FcCacheFileOpen (char *cache_file, off_t *size)
173 {
174 int fd;
175 struct stat file_stat;
176
177 fd = open(cache_file, O_RDONLY | O_BINARY);
178 if (fd < 0)
179 return -1;
180
181 if (fstat (fd, &file_stat) < 0) {
182 close (fd);
183 return -1;
184 }
185 *size = file_stat.st_size;
186 return fd;
187 }
188
189 /*
190 * return the path from the directory containing 'cache' to 'file'
191 */
192
193 static const FcChar8 *
194 FcFileBaseName (const char *cache, const FcChar8 *file)
195 {
196 const FcChar8 *cache_slash;
197 int cache_len = strlen (cache);
198
199 if (!strncmp (cache, file, cache_len) && file[cache_len] == '/')
200 return file + cache_len + 1;
201 return file;
202 }
203
204 FcBool
205 FcCachePrintSet (FcFontSet *set, FcStrSet *dirs, char *base_name)
206 {
207 FcPattern *font;
208 FcChar8 *name, *dir;
209 const FcChar8 *file, *base;
210 int ret;
211 int n;
212 int id;
213 FcStrList *list;
214
215 list = FcStrListCreate (dirs);
216 if (!list)
217 goto bail2;
218
219 while ((dir = FcStrListNext (list)))
220 {
221 base = FcFileBaseName (base_name, dir);
222 if (!FcCacheWriteStringOld (stdout, base))
223 goto bail3;
224 if (PUTC (' ', stdout) == EOF)
225 goto bail3;
226 if (!FcCacheWriteInt (stdout, 0))
227 goto bail3;
228 if (PUTC (' ', stdout) == EOF)
229 goto bail3;
230 if (!FcCacheWriteStringOld (stdout, FC_FONT_FILE_DIR))
231 goto bail3;
232 if (PUTC ('\n', stdout) == EOF)
233 goto bail3;
234 }
235
236 for (n = 0; n < set->nfont; n++)
237 {
238 FcPattern **fonts = FcFontSetFonts (set);
239 FcPattern *encoded_font = fonts[n];
240 FcPattern *font = FcEncodedOffsetToPtr (set, encoded_font, FcPattern);
241
242 if (FcPatternGetString (font, FC_FILE, 0, (FcChar8 **) &file) != FcResultMatch)
243 goto bail3;
244 base = FcFileBaseName (base_name, file);
245 if (FcPatternGetInteger (font, FC_INDEX, 0, &id) != FcResultMatch)
246 goto bail3;
247 if (!FcCacheWriteStringOld (stdout, base))
248 goto bail3;
249 if (PUTC (' ', stdout) == EOF)
250 goto bail3;
251 if (!FcCacheWriteInt (stdout, id))
252 goto bail3;
253 if (PUTC (' ', stdout) == EOF)
254 goto bail3;
255 name = FcNameUnparse (font);
256 if (!name)
257 goto bail3;
258 ret = FcCacheWriteStringOld (stdout, name);
259 FcStrFree (name);
260 if (!ret)
261 goto bail3;
262 if (PUTC ('\n', stdout) == EOF)
263 goto bail3;
264 }
265
266 FcStrListDone (list);
267
268 return FcTrue;
269
270 bail3:
271 FcStrListDone (list);
272 bail2:
273 return FcFalse;
274 }
275
276 FcCache *
277 FcCacheFileMap (const FcChar8 *file)
278 {
279 FcCache *cache;
280 int fd;
281 struct stat file_stat;
282
283 fd = open (file, O_RDONLY | O_BINARY);
284 if (fd < 0)
285 return NULL;
286 if (fstat (fd, &file_stat) < 0) {
287 close (fd);
288 return NULL;
289 }
290 if (FcCacheLoad (fd, file_stat.st_size, &cache)) {
291 close (fd);
292 return cache;
293 }
294 close (fd);
295 return NULL;
296 }
297
298 int
299 main (int argc, char **argv)
300 {
301 int i;
302 int ret = 0;
303 FcFontSet *fs;
304 FcStrSet *dirs;
305 FcCache *cache;
306 FcConfig *config;
307 int verbose = 0;
308 #if HAVE_GETOPT_LONG || HAVE_GETOPT
309 int c;
310
311 #if HAVE_GETOPT_LONG
312 while ((c = getopt_long (argc, argv, "Vv?", longopts, NULL)) != -1)
313 #else
314 while ((c = getopt (argc, argv, "Vv?")) != -1)
315 #endif
316 {
317 switch (c) {
318 case 'V':
319 fprintf (stderr, "fontconfig version %d.%d.%d\n",
320 FC_MAJOR, FC_MINOR, FC_REVISION);
321 exit (0);
322 case 'v':
323 verbose++;
324 break;
325 default:
326 usage (argv[0]);
327 }
328 }
329 i = optind;
330 #else
331 i = 1;
332 #endif
333
334 config = FcInitLoadConfig ();
335 if (!config)
336 {
337 fprintf (stderr, "%s: Can't init font config library\n", argv[0]);
338 return 1;
339 }
340 FcConfigSetCurrent (config);
341
342 if (i >= argc)
343 usage (argv[0]);
344
345 for (; i < argc; i++)
346 {
347 int j;
348 off_t size;
349 intptr_t *cache_dirs;
350
351 if (FcFileIsDir ((const FcChar8 *)argv[i]))
352 cache = FcDirCacheMap ((const FcChar8 *) argv[i], config);
353 else
354 cache = FcCacheFileMap (argv[i]);
355 if (!cache)
356 {
357 perror (argv[i]);
358 ret++;
359 continue;
360 }
361
362 dirs = FcStrSetCreate ();
363 fs = FcCacheSet (cache);
364 cache_dirs = FcCacheDirs (cache);
365 for (j = 0; j < cache->dirs_count; j++)
366 FcStrSetAdd (dirs, FcOffsetToPtr (cache_dirs,
367 cache_dirs[j],
368 FcChar8));
369
370 if (verbose)
371 printf ("Name: %s\nDirectory: %s\n", argv[i], FcCacheDir(cache));
372 FcCachePrintSet (fs, dirs, FcCacheDir (cache));
373
374 FcStrSetDestroy (dirs);
375
376 FcDirCacheUnmap (cache);
377 }
378
379 return 0;
380 }