upgrade to 2.14.2
[platform/upstream/fontconfig.git] / src / fcdefault.c
1 /*
2  * fontconfig/src/fcdefault.c
3  *
4  * Copyright © 2001 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 the author(s) not be used in
11  * advertising or publicity pertaining to distribution of the software without
12  * specific, written prior permission.  The authors make 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  * THE AUTHOR(S) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
17  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
18  * EVENT SHALL THE AUTHOR(S) 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 #include "fcint.h"
26 #include <limits.h>
27 #include <string.h>
28
29 /* MT-safe */
30
31 static const struct {
32     FcObject    field;
33     FcBool      value;
34 } FcBoolDefaults[] = {
35     { FC_HINTING_OBJECT,           FcTrue       },  /* !FT_LOAD_NO_HINTING */
36     { FC_VERTICAL_LAYOUT_OBJECT,   FcFalse      },  /* FC_LOAD_VERTICAL_LAYOUT */
37     { FC_AUTOHINT_OBJECT,          FcFalse      },  /* FC_LOAD_FORCE_AUTOHINT */
38     { FC_GLOBAL_ADVANCE_OBJECT,    FcTrue       },  /* !FC_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH */
39     { FC_EMBEDDED_BITMAP_OBJECT,   FcTrue       },  /* !FC_LOAD_NO_BITMAP */
40     { FC_DECORATIVE_OBJECT,        FcFalse      },
41     { FC_SYMBOL_OBJECT,            FcFalse      },
42     { FC_VARIABLE_OBJECT,          FcFalse      },
43 };
44
45 #define NUM_FC_BOOL_DEFAULTS    (int) (sizeof FcBoolDefaults / sizeof FcBoolDefaults[0])
46
47 FcStrSet *default_langs;
48
49 FcStrSet *
50 FcGetDefaultLangs (void)
51 {
52     FcStrSet *result;
53 retry:
54     result = (FcStrSet *) fc_atomic_ptr_get (&default_langs);
55     if (!result)
56     {
57         char *langs;
58
59         result = FcStrSetCreate ();
60
61         langs = getenv ("FC_LANG");
62         if (!langs || !langs[0])
63             langs = getenv ("LC_ALL");
64         if (!langs || !langs[0])
65         {
66             langs = getenv ("LC_CTYPE");
67             // On some macOS systems, LC_CTYPE is set to "UTF-8", which doesn't
68             // give any languge information. In this case, ignore LC_CTYPE and
69             // continue the search with LANG.
70             if (langs && (FcStrCmpIgnoreCase((const FcChar8 *) langs,
71                                              (const FcChar8 *)"UTF-8") == 0))
72             {
73                 langs = NULL;
74             }
75         }
76         if (!langs || !langs[0])
77             langs = getenv ("LANG");
78         if (langs && langs[0])
79         {
80             if (!FcStrSetAddLangs (result, langs))
81                 FcStrSetAdd (result, (const FcChar8 *) "en");
82         }
83         else
84             FcStrSetAdd (result, (const FcChar8 *) "en");
85
86         FcRefSetConst (&result->ref);
87         if (!fc_atomic_ptr_cmpexch (&default_langs, NULL, result)) {
88             FcRefInit (&result->ref, 1);
89             FcStrSetDestroy (result);
90             goto retry;
91         }
92     }
93
94     return result;
95 }
96
97 static FcChar8 *default_lang; /* MT-safe */
98
99 FcChar8 *
100 FcGetDefaultLang (void)
101 {
102     FcChar8 *lang;
103 retry:
104     lang = fc_atomic_ptr_get (&default_lang);
105     if (!lang)
106     {
107         FcStrSet *langs = FcGetDefaultLangs ();
108         lang = FcStrdup (langs->strs[0]);
109
110         if (!fc_atomic_ptr_cmpexch (&default_lang, NULL, lang)) {
111             free (lang);
112             goto retry;
113         }
114     }
115
116     return lang;
117 }
118
119 static FcChar8 *default_prgname;
120
121 FcChar8 *
122 FcGetPrgname (void)
123 {
124     FcChar8 *prgname;
125 retry:
126     prgname = fc_atomic_ptr_get (&default_prgname);
127     if (!prgname)
128     {
129 #ifdef _WIN32
130         char buf[MAX_PATH+1];
131
132         /* TODO This is ASCII-only; fix it. */
133         if (GetModuleFileNameA (GetModuleHandle (NULL), buf, sizeof (buf) / sizeof (buf[0])) > 0)
134         {
135             char *p;
136             unsigned int len;
137
138             p = strrchr (buf, '\\');
139             if (p)
140                 p++;
141             else
142                 p = buf;
143
144             len = strlen (p);
145
146             if (len > 4 && 0 == strcmp (p + len - 4, ".exe"))
147             {
148                 len -= 4;
149                 buf[len] = '\0';
150             }
151
152             prgname = FcStrdup (p);
153         }
154 #elif defined (HAVE_GETPROGNAME)
155         const char *q = getprogname ();
156         if (q)
157             prgname = FcStrdup (q);
158         else
159             prgname = FcStrdup ("");
160 #else
161 # if defined (HAVE_GETEXECNAME)
162         char *p = FcStrdup(getexecname ());
163 # elif defined (HAVE_READLINK)
164         size_t size = FC_PATH_MAX;
165         char *p = NULL;
166
167         while (1)
168         {
169             char *buf = malloc (size);
170             ssize_t len;
171
172             if (!buf)
173                 break;
174
175             len = readlink ("/proc/self/exe", buf, size - 1);
176             if (len < 0)
177             {
178                 free (buf);
179                 break;
180             }
181             if (len < size - 1)
182             {
183                 buf[len] = 0;
184                 p = buf;
185                 break;
186             }
187
188             free (buf);
189             size *= 2;
190         }
191 # else
192         char *p = NULL;
193 # endif
194         if (p)
195         {
196             char *r = strrchr (p, '/');
197             if (r)
198                 r++;
199             else
200                 r = p;
201
202             prgname = FcStrdup (r);
203         }
204
205         if (!prgname)
206             prgname = FcStrdup ("");
207
208         if (p)
209             free (p);
210 #endif
211
212         if (!fc_atomic_ptr_cmpexch (&default_prgname, NULL, prgname)) {
213             free (prgname);
214             goto retry;
215         }
216     }
217
218     if (prgname && !prgname[0])
219         return NULL;
220
221     return prgname;
222 }
223
224 static FcChar8 *default_desktop_name;
225
226 FcChar8 *
227 FcGetDesktopName (void)
228 {
229     FcChar8 *desktop_name;
230 retry:
231     desktop_name = fc_atomic_ptr_get (&default_desktop_name);
232     if (!desktop_name)
233     {
234         char *s = getenv ("XDG_CURRENT_DESKTOP");
235
236         if (!s)
237             desktop_name = FcStrdup ("");
238         else
239             desktop_name = FcStrdup (s);
240         if (!desktop_name)
241         {
242             fprintf (stderr, "Fontconfig error: out of memory in %s\n",
243                      __FUNCTION__);
244             return NULL;
245         }
246
247         if (!fc_atomic_ptr_cmpexch(&default_desktop_name, NULL, desktop_name))
248         {
249             free (desktop_name);
250             goto retry;
251         }
252     }
253     if (desktop_name && !desktop_name[0])
254         return NULL;
255
256     return desktop_name;
257 }
258
259 void
260 FcDefaultFini (void)
261 {
262     FcChar8  *lang;
263     FcStrSet *langs;
264     FcChar8  *prgname;
265     FcChar8  *desktop;
266
267     lang = fc_atomic_ptr_get (&default_lang);
268     if (lang && fc_atomic_ptr_cmpexch (&default_lang, lang, NULL))
269     {
270         free (lang);
271     }
272
273     langs = fc_atomic_ptr_get (&default_langs);
274     if (langs && fc_atomic_ptr_cmpexch (&default_langs, langs, NULL))
275     {
276         FcRefInit (&langs->ref, 1);
277         FcStrSetDestroy (langs);
278     }
279
280     prgname = fc_atomic_ptr_get (&default_prgname);
281     if (prgname && fc_atomic_ptr_cmpexch (&default_prgname, prgname, NULL))
282     {
283         free (prgname);
284     }
285
286     desktop = fc_atomic_ptr_get (&default_desktop_name);
287     if (desktop && fc_atomic_ptr_cmpexch(&default_desktop_name, desktop, NULL))
288     {
289         free (desktop);
290     }
291 }
292
293 void
294 FcDefaultSubstitute (FcPattern *pattern)
295 {
296     FcPatternIter iter;
297     FcValue v, namelang, v2;
298     int     i;
299     double      dpi, size, scale, pixelsize;
300
301     if (!FcPatternFindObjectIter (pattern, &iter, FC_WEIGHT_OBJECT))
302         FcPatternObjectAddInteger (pattern, FC_WEIGHT_OBJECT, FC_WEIGHT_NORMAL);
303
304     if (!FcPatternFindObjectIter (pattern, &iter, FC_SLANT_OBJECT))
305         FcPatternObjectAddInteger (pattern, FC_SLANT_OBJECT, FC_SLANT_ROMAN);
306
307     if (!FcPatternFindObjectIter (pattern, &iter, FC_WIDTH_OBJECT))
308         FcPatternObjectAddInteger (pattern, FC_WIDTH_OBJECT, FC_WIDTH_NORMAL);
309
310     for (i = 0; i < NUM_FC_BOOL_DEFAULTS; i++)
311         if (!FcPatternFindObjectIter (pattern, &iter, FcBoolDefaults[i].field))
312             FcPatternObjectAddBool (pattern, FcBoolDefaults[i].field, FcBoolDefaults[i].value);
313
314     if (FcPatternObjectGetDouble (pattern, FC_SIZE_OBJECT, 0, &size) != FcResultMatch)
315     {
316         FcRange *r;
317         double b, e;
318         if (FcPatternObjectGetRange (pattern, FC_SIZE_OBJECT, 0, &r) == FcResultMatch && FcRangeGetDouble (r, &b, &e))
319             size = (b + e) * .5;
320         else
321             size = 12.0L;
322     }
323     if (FcPatternObjectGetDouble (pattern, FC_SCALE_OBJECT, 0, &scale) != FcResultMatch)
324         scale = 1.0;
325     if (FcPatternObjectGetDouble (pattern, FC_DPI_OBJECT, 0, &dpi) != FcResultMatch)
326         dpi = 75.0;
327
328     if (!FcPatternFindObjectIter (pattern, &iter, FC_PIXEL_SIZE_OBJECT))
329     {
330         (void) FcPatternObjectDel (pattern, FC_SCALE_OBJECT);
331         FcPatternObjectAddDouble (pattern, FC_SCALE_OBJECT, scale);
332         pixelsize = size * scale;
333         (void) FcPatternObjectDel (pattern, FC_DPI_OBJECT);
334         FcPatternObjectAddDouble (pattern, FC_DPI_OBJECT, dpi);
335         pixelsize *= dpi / 72.0;
336         FcPatternObjectAddDouble (pattern, FC_PIXEL_SIZE_OBJECT, pixelsize);
337     }
338     else
339     {
340         FcPatternIterGetValue(pattern, &iter, 0, &v, NULL);
341         size = v.u.d;
342         size = size / dpi * 72.0 / scale;
343     }
344     (void) FcPatternObjectDel (pattern, FC_SIZE_OBJECT);
345     FcPatternObjectAddDouble (pattern, FC_SIZE_OBJECT, size);
346
347     if (!FcPatternFindObjectIter (pattern, &iter, FC_FONTVERSION_OBJECT))
348         FcPatternObjectAddInteger (pattern, FC_FONTVERSION_OBJECT, 0x7fffffff);
349
350     if (!FcPatternFindObjectIter (pattern, &iter, FC_HINT_STYLE_OBJECT))
351         FcPatternObjectAddInteger (pattern, FC_HINT_STYLE_OBJECT, FC_HINT_FULL);
352
353     if (!FcPatternFindObjectIter (pattern, &iter, FC_NAMELANG_OBJECT))
354         FcPatternObjectAddString (pattern, FC_NAMELANG_OBJECT, FcGetDefaultLang ());
355
356     /* shouldn't be failed. */
357     FcPatternObjectGet (pattern, FC_NAMELANG_OBJECT, 0, &namelang);
358     /* Add a fallback to ensure the english name when the requested language
359      * isn't available. this would helps for the fonts that have non-English
360      * name at the beginning.
361      */
362     /* Set "en-us" instead of "en" to avoid giving higher score to "en".
363      * This is a hack for the case that the orth is not like ll-cc, because,
364      * if no namelang isn't explicitly set, it will has something like ll-cc
365      * according to current locale. which may causes FcLangDifferentTerritory
366      * at FcLangCompare(). thus, the English name is selected so that
367      * exact matched "en" has higher score than ll-cc.
368      */
369     v2.type = FcTypeString;
370     v2.u.s = (FcChar8 *) "en-us";
371     if (!FcPatternFindObjectIter (pattern, &iter, FC_FAMILYLANG_OBJECT))
372     {
373         FcPatternObjectAdd (pattern, FC_FAMILYLANG_OBJECT, namelang, FcTrue);
374         FcPatternObjectAddWithBinding (pattern, FC_FAMILYLANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
375     }
376     if (!FcPatternFindObjectIter (pattern, &iter, FC_STYLELANG_OBJECT))
377     {
378         FcPatternObjectAdd (pattern, FC_STYLELANG_OBJECT, namelang, FcTrue);
379         FcPatternObjectAddWithBinding (pattern, FC_STYLELANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
380     }
381     if (!FcPatternFindObjectIter (pattern, &iter, FC_FULLNAMELANG_OBJECT))
382     {
383         FcPatternObjectAdd (pattern, FC_FULLNAMELANG_OBJECT, namelang, FcTrue);
384         FcPatternObjectAddWithBinding (pattern, FC_FULLNAMELANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
385     }
386
387     if (FcPatternObjectGet (pattern, FC_PRGNAME_OBJECT, 0, &v) == FcResultNoMatch)
388     {
389         FcChar8 *prgname = FcGetPrgname ();
390         if (prgname)
391             FcPatternObjectAddString (pattern, FC_PRGNAME_OBJECT, prgname);
392     }
393
394     if (FcPatternObjectGet (pattern, FC_DESKTOP_NAME_OBJECT, 0, &v) == FcResultNoMatch)
395     {
396         FcChar8 *desktop = FcGetDesktopName ();
397         if (desktop)
398             FcPatternObjectAddString (pattern, FC_DESKTOP_NAME_OBJECT, desktop);
399     }
400
401     if (!FcPatternFindObjectIter (pattern, &iter, FC_ORDER_OBJECT))
402         FcPatternObjectAddInteger (pattern, FC_ORDER_OBJECT, 0);
403 }
404 #define __fcdefault__
405 #include "fcaliastail.h"
406 #undef __fcdefault__