|
Packit |
352660 |
/*
|
|
Packit |
352660 |
* fontconfig/src/fcdefault.c
|
|
Packit |
352660 |
*
|
|
Packit |
352660 |
* Copyright © 2001 Keith Packard
|
|
Packit |
352660 |
*
|
|
Packit |
352660 |
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
Packit |
352660 |
* documentation for any purpose is hereby granted without fee, provided that
|
|
Packit |
352660 |
* the above copyright notice appear in all copies and that both that
|
|
Packit |
352660 |
* copyright notice and this permission notice appear in supporting
|
|
Packit |
352660 |
* documentation, and that the name of the author(s) not be used in
|
|
Packit |
352660 |
* advertising or publicity pertaining to distribution of the software without
|
|
Packit |
352660 |
* specific, written prior permission. The authors make no
|
|
Packit |
352660 |
* representations about the suitability of this software for any purpose. It
|
|
Packit |
352660 |
* is provided "as is" without express or implied warranty.
|
|
Packit |
352660 |
*
|
|
Packit |
352660 |
* THE AUTHOR(S) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
Packit |
352660 |
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
|
Packit |
352660 |
* EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
|
Packit |
352660 |
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
|
Packit |
352660 |
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
Packit |
352660 |
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
Packit |
352660 |
* PERFORMANCE OF THIS SOFTWARE.
|
|
Packit |
352660 |
*/
|
|
Packit |
352660 |
|
|
Packit |
352660 |
#include "fcint.h"
|
|
Packit |
352660 |
#include <limits.h>
|
|
Packit |
352660 |
#include <string.h>
|
|
Packit |
352660 |
|
|
Packit |
352660 |
/* MT-safe */
|
|
Packit |
352660 |
|
|
Packit |
352660 |
static const struct {
|
|
Packit |
352660 |
FcObject field;
|
|
Packit |
352660 |
FcBool value;
|
|
Packit |
352660 |
} FcBoolDefaults[] = {
|
|
Packit |
352660 |
{ FC_HINTING_OBJECT, FcTrue }, /* !FT_LOAD_NO_HINTING */
|
|
Packit |
352660 |
{ FC_VERTICAL_LAYOUT_OBJECT, FcFalse }, /* FC_LOAD_VERTICAL_LAYOUT */
|
|
Packit |
352660 |
{ FC_AUTOHINT_OBJECT, FcFalse }, /* FC_LOAD_FORCE_AUTOHINT */
|
|
Packit |
352660 |
{ FC_GLOBAL_ADVANCE_OBJECT, FcTrue }, /* !FC_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH */
|
|
Packit |
352660 |
{ FC_EMBEDDED_BITMAP_OBJECT, FcTrue }, /* !FC_LOAD_NO_BITMAP */
|
|
Packit |
352660 |
{ FC_DECORATIVE_OBJECT, FcFalse },
|
|
Packit |
352660 |
{ FC_SYMBOL_OBJECT, FcFalse },
|
|
Packit |
352660 |
{ FC_VARIABLE_OBJECT, FcFalse },
|
|
Packit |
352660 |
};
|
|
Packit |
352660 |
|
|
Packit |
352660 |
#define NUM_FC_BOOL_DEFAULTS (int) (sizeof FcBoolDefaults / sizeof FcBoolDefaults[0])
|
|
Packit |
352660 |
|
|
Packit |
352660 |
FcStrSet *default_langs;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
FcStrSet *
|
|
Packit |
352660 |
FcGetDefaultLangs (void)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcStrSet *result;
|
|
Packit |
352660 |
retry:
|
|
Packit |
352660 |
result = (FcStrSet *) fc_atomic_ptr_get (&default_langs);
|
|
Packit |
352660 |
if (!result)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
char *langs;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
result = FcStrSetCreate ();
|
|
Packit |
352660 |
|
|
Packit |
352660 |
langs = getenv ("FC_LANG");
|
|
Packit |
352660 |
if (!langs || !langs[0])
|
|
Packit |
352660 |
langs = getenv ("LC_ALL");
|
|
Packit |
352660 |
if (!langs || !langs[0])
|
|
Packit |
352660 |
langs = getenv ("LC_CTYPE");
|
|
Packit |
352660 |
if (!langs || !langs[0])
|
|
Packit |
352660 |
langs = getenv ("LANG");
|
|
Packit |
352660 |
if (langs && langs[0])
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
if (!FcStrSetAddLangs (result, langs))
|
|
Packit |
352660 |
FcStrSetAdd (result, (const FcChar8 *) "en");
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
FcStrSetAdd (result, (const FcChar8 *) "en");
|
|
Packit |
352660 |
|
|
Packit |
352660 |
FcRefSetConst (&result->ref);
|
|
Packit |
352660 |
if (!fc_atomic_ptr_cmpexch (&default_langs, NULL, result)) {
|
|
Packit |
352660 |
FcRefInit (&result->ref, 1);
|
|
Packit |
352660 |
FcStrSetDestroy (result);
|
|
Packit |
352660 |
goto retry;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
return result;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
static FcChar8 *default_lang; /* MT-safe */
|
|
Packit |
352660 |
|
|
Packit |
352660 |
FcChar8 *
|
|
Packit |
352660 |
FcGetDefaultLang (void)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcChar8 *lang;
|
|
Packit |
352660 |
retry:
|
|
Packit |
352660 |
lang = fc_atomic_ptr_get (&default_lang);
|
|
Packit |
352660 |
if (!lang)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcStrSet *langs = FcGetDefaultLangs ();
|
|
Packit |
352660 |
lang = FcStrdup (langs->strs[0]);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!fc_atomic_ptr_cmpexch (&default_lang, NULL, lang)) {
|
|
Packit |
352660 |
free (lang);
|
|
Packit |
352660 |
goto retry;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
return lang;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
static FcChar8 *default_prgname;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
FcChar8 *
|
|
Packit |
352660 |
FcGetPrgname (void)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcChar8 *prgname;
|
|
Packit |
352660 |
retry:
|
|
Packit |
352660 |
prgname = fc_atomic_ptr_get (&default_prgname);
|
|
Packit |
352660 |
if (!prgname)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
#ifdef _WIN32
|
|
Packit |
352660 |
char buf[MAX_PATH+1];
|
|
Packit |
352660 |
|
|
Packit |
352660 |
/* TODO This is ASCII-only; fix it. */
|
|
Packit |
352660 |
if (GetModuleFileNameA (GetModuleHandle (NULL), buf, sizeof (buf) / sizeof (buf[0])) > 0)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
char *p;
|
|
Packit |
352660 |
unsigned int len;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
p = strrchr (buf, '\\');
|
|
Packit |
352660 |
if (p)
|
|
Packit |
352660 |
p++;
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
p = buf;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
len = strlen (p);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (len > 4 && 0 == strcmp (p + len - 4, ".exe"))
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
len -= 4;
|
|
Packit |
352660 |
buf[len] = '\0';
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
prgname = FcStrdup (p);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
#elif defined (HAVE_GETPROGNAME)
|
|
Packit |
352660 |
const char *q = getprogname ();
|
|
Packit |
352660 |
if (q)
|
|
Packit |
352660 |
prgname = FcStrdup (q);
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
prgname = FcStrdup ("");
|
|
Packit |
352660 |
#else
|
|
Packit |
352660 |
# if defined (HAVE_GETEXECNAME)
|
|
Packit |
352660 |
char *p = FcStrdup(getexecname ());
|
|
Packit |
352660 |
# elif defined (HAVE_READLINK)
|
|
Packit |
352660 |
size_t size = FC_PATH_MAX;
|
|
Packit |
352660 |
char *p = NULL;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
while (1)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
char *buf = malloc (size);
|
|
Packit |
352660 |
ssize_t len;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!buf)
|
|
Packit |
352660 |
break;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
len = readlink ("/proc/self/exe", buf, size - 1);
|
|
Packit |
352660 |
if (len < 0)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
free (buf);
|
|
Packit |
352660 |
break;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
if (len < size - 1)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
buf[len] = 0;
|
|
Packit |
352660 |
p = buf;
|
|
Packit |
352660 |
break;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
free (buf);
|
|
Packit |
352660 |
size *= 2;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
# else
|
|
Packit |
352660 |
char *p = NULL;
|
|
Packit |
352660 |
# endif
|
|
Packit |
352660 |
if (p)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
char *r = strrchr (p, '/');
|
|
Packit |
352660 |
if (r)
|
|
Packit |
352660 |
r++;
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
r = p;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
prgname = FcStrdup (r);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!prgname)
|
|
Packit |
352660 |
prgname = FcStrdup ("");
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (p)
|
|
Packit |
352660 |
free (p);
|
|
Packit |
352660 |
#endif
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!fc_atomic_ptr_cmpexch (&default_prgname, NULL, prgname)) {
|
|
Packit |
352660 |
free (prgname);
|
|
Packit |
352660 |
goto retry;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (prgname && !prgname[0])
|
|
Packit |
352660 |
return NULL;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
return prgname;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
void
|
|
Packit |
352660 |
FcDefaultFini (void)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcChar8 *lang;
|
|
Packit |
352660 |
FcStrSet *langs;
|
|
Packit |
352660 |
FcChar8 *prgname;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
lang = fc_atomic_ptr_get (&default_lang);
|
|
Packit |
352660 |
if (lang && fc_atomic_ptr_cmpexch (&default_lang, lang, NULL)) {
|
|
Packit |
352660 |
free (lang);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
langs = fc_atomic_ptr_get (&default_langs);
|
|
Packit |
352660 |
if (langs && fc_atomic_ptr_cmpexch (&default_langs, langs, NULL)) {
|
|
Packit |
352660 |
FcRefInit (&langs->ref, 1);
|
|
Packit |
352660 |
FcStrSetDestroy (langs);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
prgname = fc_atomic_ptr_get (&default_prgname);
|
|
Packit |
352660 |
if (prgname && fc_atomic_ptr_cmpexch (&default_prgname, prgname, NULL)) {
|
|
Packit |
352660 |
free (prgname);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
void
|
|
Packit |
352660 |
FcDefaultSubstitute (FcPattern *pattern)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcPatternIter iter;
|
|
Packit |
352660 |
FcValue v, namelang, v2;
|
|
Packit |
352660 |
int i;
|
|
Packit |
352660 |
double dpi, size, scale, pixelsize;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_WEIGHT_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddInteger (pattern, FC_WEIGHT_OBJECT, FC_WEIGHT_NORMAL);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_SLANT_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddInteger (pattern, FC_SLANT_OBJECT, FC_SLANT_ROMAN);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_WIDTH_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddInteger (pattern, FC_WIDTH_OBJECT, FC_WIDTH_NORMAL);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
for (i = 0; i < NUM_FC_BOOL_DEFAULTS; i++)
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FcBoolDefaults[i].field))
|
|
Packit |
352660 |
FcPatternObjectAddBool (pattern, FcBoolDefaults[i].field, FcBoolDefaults[i].value);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (FcPatternObjectGetDouble (pattern, FC_SIZE_OBJECT, 0, &size) != FcResultMatch)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcRange *r;
|
|
Packit |
352660 |
double b, e;
|
|
Packit |
352660 |
if (FcPatternObjectGetRange (pattern, FC_SIZE_OBJECT, 0, &r) == FcResultMatch && FcRangeGetDouble (r, &b, &e))
|
|
Packit |
352660 |
size = (b + e) * .5;
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
size = 12.0L;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
if (FcPatternObjectGetDouble (pattern, FC_SCALE_OBJECT, 0, &scale) != FcResultMatch)
|
|
Packit |
352660 |
scale = 1.0;
|
|
Packit |
352660 |
if (FcPatternObjectGetDouble (pattern, FC_DPI_OBJECT, 0, &dpi) != FcResultMatch)
|
|
Packit |
352660 |
dpi = 75.0;
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_PIXEL_SIZE_OBJECT))
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
(void) FcPatternObjectDel (pattern, FC_SCALE_OBJECT);
|
|
Packit |
352660 |
FcPatternObjectAddDouble (pattern, FC_SCALE_OBJECT, scale);
|
|
Packit |
352660 |
pixelsize = size * scale;
|
|
Packit |
352660 |
(void) FcPatternObjectDel (pattern, FC_DPI_OBJECT);
|
|
Packit |
352660 |
FcPatternObjectAddDouble (pattern, FC_DPI_OBJECT, dpi);
|
|
Packit |
352660 |
pixelsize *= dpi / 72.0;
|
|
Packit |
352660 |
FcPatternObjectAddDouble (pattern, FC_PIXEL_SIZE_OBJECT, pixelsize);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
else
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcPatternIterGetValue(pattern, &iter, 0, &v, NULL);
|
|
Packit |
352660 |
size = v.u.d;
|
|
Packit |
352660 |
size = size / dpi * 72.0 / scale;
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
(void) FcPatternObjectDel (pattern, FC_SIZE_OBJECT);
|
|
Packit |
352660 |
FcPatternObjectAddDouble (pattern, FC_SIZE_OBJECT, size);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_FONTVERSION_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddInteger (pattern, FC_FONTVERSION_OBJECT, 0x7fffffff);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_HINT_STYLE_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddInteger (pattern, FC_HINT_STYLE_OBJECT, FC_HINT_FULL);
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_NAMELANG_OBJECT))
|
|
Packit |
352660 |
FcPatternObjectAddString (pattern, FC_NAMELANG_OBJECT, FcGetDefaultLang ());
|
|
Packit |
352660 |
|
|
Packit |
352660 |
/* shouldn't be failed. */
|
|
Packit |
352660 |
FcPatternObjectGet (pattern, FC_NAMELANG_OBJECT, 0, &namelang);
|
|
Packit |
352660 |
/* Add a fallback to ensure the english name when the requested language
|
|
Packit |
352660 |
* isn't available. this would helps for the fonts that have non-English
|
|
Packit |
352660 |
* name at the beginning.
|
|
Packit |
352660 |
*/
|
|
Packit |
352660 |
/* Set "en-us" instead of "en" to avoid giving higher score to "en".
|
|
Packit |
352660 |
* This is a hack for the case that the orth is not like ll-cc, because,
|
|
Packit |
352660 |
* if no namelang isn't explicitly set, it will has something like ll-cc
|
|
Packit |
352660 |
* according to current locale. which may causes FcLangDifferentTerritory
|
|
Packit |
352660 |
* at FcLangCompare(). thus, the English name is selected so that
|
|
Packit |
352660 |
* exact matched "en" has higher score than ll-cc.
|
|
Packit |
352660 |
*/
|
|
Packit |
352660 |
v2.type = FcTypeString;
|
|
Packit |
352660 |
v2.u.s = (FcChar8 *) "en-us";
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_FAMILYLANG_OBJECT))
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcPatternObjectAdd (pattern, FC_FAMILYLANG_OBJECT, namelang, FcTrue);
|
|
Packit |
352660 |
FcPatternObjectAddWithBinding (pattern, FC_FAMILYLANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_STYLELANG_OBJECT))
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcPatternObjectAdd (pattern, FC_STYLELANG_OBJECT, namelang, FcTrue);
|
|
Packit |
352660 |
FcPatternObjectAddWithBinding (pattern, FC_STYLELANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
if (!FcPatternFindObjectIter (pattern, &iter, FC_FULLNAMELANG_OBJECT))
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcPatternObjectAdd (pattern, FC_FULLNAMELANG_OBJECT, namelang, FcTrue);
|
|
Packit |
352660 |
FcPatternObjectAddWithBinding (pattern, FC_FULLNAMELANG_OBJECT, v2, FcValueBindingWeak, FcTrue);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
|
|
Packit |
352660 |
if (FcPatternObjectGet (pattern, FC_PRGNAME_OBJECT, 0, &v) == FcResultNoMatch)
|
|
Packit |
352660 |
{
|
|
Packit |
352660 |
FcChar8 *prgname = FcGetPrgname ();
|
|
Packit |
352660 |
if (prgname)
|
|
Packit |
352660 |
FcPatternObjectAddString (pattern, FC_PRGNAME_OBJECT, prgname);
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
}
|
|
Packit |
352660 |
#define __fcdefault__
|
|
Packit |
352660 |
#include "fcaliastail.h"
|
|
Packit |
352660 |
#undef __fcdefault__
|