(root)/
fontconfig-2.14.2/
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__