Blame src/fcdefault.c

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__