Blame src/fcstr.c

Packit 352660
/*
Packit 352660
 * fontconfig/src/fcstr.c
Packit 352660
 *
Packit 352660
 * Copyright © 2000 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 <stdlib.h>
Packit 352660
#include <ctype.h>
Packit 352660
#include <string.h>
Packit 352660
Packit 352660
Packit 352660
/* Objects MT-safe for readonly access. */
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrCopy (const FcChar8 *s)
Packit 352660
{
Packit 352660
    return FcStrdup (s);
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrPlus (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    int	    s1l = strlen ((char *) s1);
Packit 352660
    int	    s2l = strlen ((char *) s2);
Packit 352660
    int	    l = s1l + s2l + 1;
Packit 352660
    FcChar8 *s = malloc (l);
Packit 352660
Packit 352660
    if (!s)
Packit 352660
	return 0;
Packit 352660
    memcpy (s, s1, s1l);
Packit 352660
    memcpy (s + s1l, s2, s2l + 1);
Packit 352660
    return s;
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrFree (FcChar8 *s)
Packit 352660
{
Packit 352660
    free (s);
Packit 352660
}
Packit 352660
Packit 352660
Packit 352660
#include "../fc-case/fccase.h"
Packit 352660
Packit 352660
#define FcCaseFoldUpperCount(cf) \
Packit 352660
    ((cf)->method == FC_CASE_FOLD_FULL ? 1 : (cf)->count)
Packit 352660
Packit 352660
typedef struct _FcCaseWalker {
Packit 352660
    const FcChar8   *read;
Packit 352660
    const FcChar8   *src;
Packit 352660
    FcChar8	    utf8[FC_MAX_CASE_FOLD_CHARS + 1];
Packit 352660
} FcCaseWalker;
Packit 352660
Packit 352660
static void
Packit 352660
FcStrCaseWalkerInit (const FcChar8 *src, FcCaseWalker *w)
Packit 352660
{
Packit 352660
    w->src = src;
Packit 352660
    w->read = 0;
Packit 352660
}
Packit 352660
Packit 352660
static FcChar8
Packit 352660
FcStrCaseWalkerLong (FcCaseWalker *w, FcChar8 r)
Packit 352660
{
Packit 352660
    FcChar32	ucs4;
Packit 352660
    int		slen;
Packit 352660
    int		len = strlen((char*)w->src);
Packit 352660
Packit 352660
    slen = FcUtf8ToUcs4 (w->src - 1, &ucs4, len + 1);
Packit 352660
    if (slen <= 0)
Packit 352660
	return r;
Packit 352660
    if (FC_MIN_FOLD_CHAR <= ucs4 && ucs4 <= FC_MAX_FOLD_CHAR)
Packit 352660
    {
Packit 352660
	int min = 0;
Packit 352660
	int max = FC_NUM_CASE_FOLD;
Packit 352660
Packit 352660
	while (min <= max)
Packit 352660
	{
Packit 352660
	    int		mid = (min + max) >> 1;
Packit 352660
	    FcChar32    low = fcCaseFold[mid].upper;
Packit 352660
	    FcChar32    high = low + FcCaseFoldUpperCount (&fcCaseFold[mid]);
Packit 352660
	
Packit 352660
	    if (high <= ucs4)
Packit 352660
		min = mid + 1;
Packit 352660
	    else if (ucs4 < low)
Packit 352660
		max = mid - 1;
Packit 352660
	    else
Packit 352660
	    {
Packit 352660
		const FcCaseFold    *fold = &fcCaseFold[mid];
Packit 352660
		int		    dlen;
Packit 352660
		
Packit 352660
		switch (fold->method) {
Packit 352660
		case  FC_CASE_FOLD_EVEN_ODD:
Packit 352660
		    if ((ucs4 & 1) != (fold->upper & 1))
Packit 352660
			return r;
Packit 352660
		    /* fall through ... */
Packit 352660
		default:
Packit 352660
		    dlen = FcUcs4ToUtf8 (ucs4 + fold->offset, w->utf8);
Packit 352660
		    break;
Packit 352660
		case FC_CASE_FOLD_FULL:
Packit 352660
		    dlen = fold->count;
Packit 352660
		    memcpy (w->utf8, fcCaseFoldChars + fold->offset, dlen);
Packit 352660
		    break;
Packit 352660
		}
Packit 352660
		
Packit 352660
		/* consume rest of src utf-8 bytes */
Packit 352660
		w->src += slen - 1;
Packit 352660
		
Packit 352660
		/* read from temp buffer */
Packit 352660
		w->utf8[dlen] = '\0';
Packit 352660
		w->read = w->utf8;
Packit 352660
		return *w->read++;
Packit 352660
	    }
Packit 352660
	}
Packit 352660
    }
Packit 352660
    return r;
Packit 352660
}
Packit 352660
Packit 352660
static FcChar8
Packit 352660
FcStrCaseWalkerNext (FcCaseWalker *w, const char *delims)
Packit 352660
{
Packit 352660
    FcChar8	r;
Packit 352660
Packit 352660
    if (w->read)
Packit 352660
    {
Packit 352660
	if ((r = *w->read++))
Packit 352660
	    return r;
Packit 352660
	w->read = 0;
Packit 352660
    }
Packit 352660
    do
Packit 352660
    {
Packit 352660
	r = *w->src++;
Packit 352660
    } while (r != 0 && delims && strchr (delims, r));
Packit 352660
Packit 352660
    if ((r & 0xc0) == 0xc0)
Packit 352660
	return FcStrCaseWalkerLong (w, r);
Packit 352660
    if ('A' <= r && r <= 'Z')
Packit 352660
        r = r - 'A' + 'a';
Packit 352660
    return r;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrDowncase (const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcCaseWalker    w;
Packit 352660
    int		    len = 0;
Packit 352660
    FcChar8	    *dst, *d;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s, &w);
Packit 352660
    while (FcStrCaseWalkerNext (&w, NULL))
Packit 352660
	len++;
Packit 352660
    d = dst = malloc (len + 1);
Packit 352660
    if (!d)
Packit 352660
	return 0;
Packit 352660
    FcStrCaseWalkerInit (s, &w);
Packit 352660
    while ((*d++ = FcStrCaseWalkerNext (&w, NULL)));
Packit 352660
    return dst;
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcStrCmpIgnoreCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
Packit 352660
    if (s1 == s2) return 0;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, NULL);
Packit 352660
	c2 = FcStrCaseWalkerNext (&w2, NULL);
Packit 352660
	if (!c1 || (c1 != c2))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return (int) c1 - (int) c2;
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcStrCmpIgnoreBlanksAndCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    return FcStrCmpIgnoreCaseAndDelims (s1, s2, (const FcChar8 *)" ");
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcStrCmpIgnoreCaseAndDelims (const FcChar8 *s1, const FcChar8 *s2, const FcChar8 *delims)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
Packit 352660
    if (s1 == s2) return 0;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, (const char *)delims);
Packit 352660
	c2 = FcStrCaseWalkerNext (&w2, (const char *)delims);
Packit 352660
	if (!c1 || (c1 != c2))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return (int) c1 - (int) c2;
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcStrCmp (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcChar8 c1, c2;
Packit 352660
Packit 352660
    if (s1 == s2)
Packit 352660
	return 0;
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = *s1++;
Packit 352660
	c2 = *s2++;
Packit 352660
	if (!c1 || c1 != c2)
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return (int) c1 - (int) c2;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Return a hash value for a string
Packit 352660
 */
Packit 352660
Packit 352660
FcChar32
Packit 352660
FcStrHashIgnoreCase (const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar32	    h = 0;
Packit 352660
    FcCaseWalker    w;
Packit 352660
    FcChar8	    c;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s, &w);
Packit 352660
    while ((c = FcStrCaseWalkerNext (&w, NULL)))
Packit 352660
	h = ((h << 3) ^ (h >> 3)) ^ c;
Packit 352660
    return h;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Is the head of s1 equal to s2?
Packit 352660
 */
Packit 352660
Packit 352660
static FcBool
Packit 352660
FcStrIsAtIgnoreBlanksAndCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, " ");
Packit 352660
	c2 = FcStrCaseWalkerNext (&w2, " ");
Packit 352660
	if (!c1 || (c1 != c2))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return c1 == c2 || !c2;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Does s1 contain an instance of s2 (ignoring blanks and case)?
Packit 352660
 */
Packit 352660
Packit 352660
const FcChar8 *
Packit 352660
FcStrContainsIgnoreBlanksAndCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    while (*s1)
Packit 352660
    {
Packit 352660
	if (FcStrIsAtIgnoreBlanksAndCase (s1, s2))
Packit 352660
	    return s1;
Packit 352660
	s1++;
Packit 352660
    }
Packit 352660
    return 0;
Packit 352660
}
Packit 352660
Packit 352660
static FcBool
Packit 352660
FcCharIsPunct (const FcChar8 c)
Packit 352660
{
Packit 352660
    if (c < '0')
Packit 352660
	return FcTrue;
Packit 352660
    if (c <= '9')
Packit 352660
	return FcFalse;
Packit 352660
    if (c < 'A')
Packit 352660
	return FcTrue;
Packit 352660
    if (c <= 'Z')
Packit 352660
	return FcFalse;
Packit 352660
    if (c < 'a')
Packit 352660
	return FcTrue;
Packit 352660
    if (c <= 'z')
Packit 352660
	return FcFalse;
Packit 352660
    if (c <= '~')
Packit 352660
	return FcTrue;
Packit 352660
    return FcFalse;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Is the head of s1 equal to s2?
Packit 352660
 */
Packit 352660
Packit 352660
static FcBool
Packit 352660
FcStrIsAtIgnoreCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, NULL);
Packit 352660
	c2 = FcStrCaseWalkerNext (&w2, NULL);
Packit 352660
	if (!c1 || (c1 != c2))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return c1 == c2 || !c2;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Does s1 contain an instance of s2 (ignoring blanks and case)?
Packit 352660
 */
Packit 352660
Packit 352660
const FcChar8 *
Packit 352660
FcStrContainsIgnoreCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    while (*s1)
Packit 352660
    {
Packit 352660
	if (FcStrIsAtIgnoreCase (s1, s2))
Packit 352660
	    return s1;
Packit 352660
	s1++;
Packit 352660
    }
Packit 352660
    return 0;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * Does s1 contain an instance of s2 on a word boundary (ignoring case)?
Packit 352660
 */
Packit 352660
Packit 352660
const FcChar8 *
Packit 352660
FcStrContainsWord (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcBool  wordStart = FcTrue;
Packit 352660
    int	    s1len = strlen ((char *) s1);
Packit 352660
    int	    s2len = strlen ((char *) s2);
Packit 352660
	
Packit 352660
    while (s1len >= s2len)
Packit 352660
    {
Packit 352660
	if (wordStart &&
Packit 352660
	    FcStrIsAtIgnoreCase (s1, s2) &&
Packit 352660
	    (s1len == s2len || FcCharIsPunct (s1[s2len])))
Packit 352660
	{
Packit 352660
	    return s1;
Packit 352660
	}
Packit 352660
	wordStart = FcFalse;
Packit 352660
	if (FcCharIsPunct (*s1))
Packit 352660
	    wordStart = FcTrue;
Packit 352660
	s1++;
Packit 352660
	s1len--;
Packit 352660
    }
Packit 352660
    return 0;
Packit 352660
}
Packit 352660
Packit 352660
/*
Packit 352660
 * returns the number of strings (ignoring delimitors and case) being matched
Packit 352660
 */
Packit 352660
Packit 352660
int
Packit 352660
FcStrMatchIgnoreCaseAndDelims (const FcChar8 *s1, const FcChar8 *s2, const FcChar8 *delims)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
Packit 352660
    if (s1 == s2) return 0;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, (const char *)delims);
Packit 352660
	c2 = FcStrCaseWalkerNext (&w2, (const char *)delims);
Packit 352660
	if (!c1 || (c1 != c2))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return w1.src - s1 - 1;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrGlobMatch (const FcChar8 *glob,
Packit 352660
		const FcChar8 *string)
Packit 352660
{
Packit 352660
    FcChar8	c;
Packit 352660
Packit 352660
    while ((c = *glob++))
Packit 352660
    {
Packit 352660
	switch (c) {
Packit 352660
	case '*':
Packit 352660
	    /* short circuit common case */
Packit 352660
	    if (!*glob)
Packit 352660
		return FcTrue;
Packit 352660
	    /* short circuit another common case */
Packit 352660
	    if (strchr ((char *) glob, '*') == 0)
Packit 352660
	    {
Packit 352660
		size_t l1, l2;
Packit 352660
Packit 352660
		l1 = strlen ((char *) string);
Packit 352660
		l2 = strlen ((char *) glob);
Packit 352660
		if (l1 < l2)
Packit 352660
		    return FcFalse;
Packit 352660
		string += (l1 - l2);
Packit 352660
	    }
Packit 352660
	    while (*string)
Packit 352660
	    {
Packit 352660
		if (FcStrGlobMatch (glob, string))
Packit 352660
		    return FcTrue;
Packit 352660
		string++;
Packit 352660
	    }
Packit 352660
	    return FcFalse;
Packit 352660
	case '?':
Packit 352660
	    if (*string++ == '\0')
Packit 352660
		return FcFalse;
Packit 352660
	    break;
Packit 352660
	default:
Packit 352660
	    if (*string++ != c)
Packit 352660
		return FcFalse;
Packit 352660
	    break;
Packit 352660
	}
Packit 352660
    }
Packit 352660
    return *string == '\0';
Packit 352660
}
Packit 352660
Packit 352660
const FcChar8 *
Packit 352660
FcStrStrIgnoreCase (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcCaseWalker    w1, w2;
Packit 352660
    FcChar8	    c1, c2;
Packit 352660
    const FcChar8   *cur;
Packit 352660
Packit 352660
    if (!s1 || !s2)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    if (s1 == s2)
Packit 352660
	return s1;
Packit 352660
Packit 352660
    FcStrCaseWalkerInit (s1, &w1;;
Packit 352660
    FcStrCaseWalkerInit (s2, &w2;;
Packit 352660
Packit 352660
    c2 = FcStrCaseWalkerNext (&w2, NULL);
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	cur = w1.src;
Packit 352660
	c1 = FcStrCaseWalkerNext (&w1, NULL);
Packit 352660
	if (!c1)
Packit 352660
	    break;
Packit 352660
	if (c1 == c2)
Packit 352660
	{
Packit 352660
	    FcCaseWalker    w1t = w1;
Packit 352660
	    FcCaseWalker    w2t = w2;
Packit 352660
	    FcChar8	    c1t, c2t;
Packit 352660
Packit 352660
	    for (;;)
Packit 352660
	    {
Packit 352660
		c1t = FcStrCaseWalkerNext (&w1t, NULL);
Packit 352660
		c2t = FcStrCaseWalkerNext (&w2t, NULL);
Packit 352660
Packit 352660
		if (!c2t)
Packit 352660
		    return cur;
Packit 352660
		if (c2t != c1t)
Packit 352660
		    break;
Packit 352660
	    }
Packit 352660
	}
Packit 352660
    }
Packit 352660
    return 0;
Packit 352660
}
Packit 352660
Packit 352660
const FcChar8 *
Packit 352660
FcStrStr (const FcChar8 *s1, const FcChar8 *s2)
Packit 352660
{
Packit 352660
    FcChar8 c1, c2;
Packit 352660
    const FcChar8 * p = s1;
Packit 352660
    const FcChar8 * b = s2;
Packit 352660
Packit 352660
    if (!s1 || !s2)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    if (s1 == s2)
Packit 352660
	return s1;
Packit 352660
Packit 352660
again:
Packit 352660
    c2 = *s2++;
Packit 352660
Packit 352660
    if (!c2)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	p = s1;
Packit 352660
	c1 = *s1++;
Packit 352660
	if (!c1 || c1 == c2)
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
Packit 352660
    if (c1 != c2)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    for (;;)
Packit 352660
    {
Packit 352660
	c1 = *s1;
Packit 352660
	c2 = *s2;
Packit 352660
	if (c1 && c2 && c1 != c2)
Packit 352660
	{
Packit 352660
	    s1 = p + 1;
Packit 352660
	    s2 = b;
Packit 352660
	    goto again;
Packit 352660
	}
Packit 352660
	if (!c2)
Packit 352660
	    return p;
Packit 352660
	if (!c1)
Packit 352660
	    return 0;
Packit 352660
	++ s1;
Packit 352660
	++ s2;
Packit 352660
    }
Packit 352660
    /* never reached. */
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcUtf8ToUcs4 (const FcChar8 *src_orig,
Packit 352660
	      FcChar32	    *dst,
Packit 352660
	      int	    len)
Packit 352660
{
Packit 352660
    const FcChar8   *src = src_orig;
Packit 352660
    FcChar8	    s;
Packit 352660
    int		    extra;
Packit 352660
    FcChar32	    result;
Packit 352660
Packit 352660
    if (len == 0)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    s = *src++;
Packit 352660
    len--;
Packit 352660
Packit 352660
    if (!(s & 0x80))
Packit 352660
    {
Packit 352660
	result = s;
Packit 352660
	extra = 0;
Packit 352660
    }
Packit 352660
    else if (!(s & 0x40))
Packit 352660
    {
Packit 352660
	return -1;
Packit 352660
    }
Packit 352660
    else if (!(s & 0x20))
Packit 352660
    {
Packit 352660
	result = s & 0x1f;
Packit 352660
	extra = 1;
Packit 352660
    }
Packit 352660
    else if (!(s & 0x10))
Packit 352660
    {
Packit 352660
	result = s & 0xf;
Packit 352660
	extra = 2;
Packit 352660
    }
Packit 352660
    else if (!(s & 0x08))
Packit 352660
    {
Packit 352660
	result = s & 0x07;
Packit 352660
	extra = 3;
Packit 352660
    }
Packit 352660
    else if (!(s & 0x04))
Packit 352660
    {
Packit 352660
	result = s & 0x03;
Packit 352660
	extra = 4;
Packit 352660
    }
Packit 352660
    else if ( ! (s & 0x02))
Packit 352660
    {
Packit 352660
	result = s & 0x01;
Packit 352660
	extra = 5;
Packit 352660
    }
Packit 352660
    else
Packit 352660
    {
Packit 352660
	return -1;
Packit 352660
    }
Packit 352660
    if (extra > len)
Packit 352660
	return -1;
Packit 352660
Packit 352660
    while (extra--)
Packit 352660
    {
Packit 352660
	result <<= 6;
Packit 352660
	s = *src++;
Packit 352660
	
Packit 352660
	if ((s & 0xc0) != 0x80)
Packit 352660
	    return -1;
Packit 352660
	
Packit 352660
	result |= s & 0x3f;
Packit 352660
    }
Packit 352660
    *dst = result;
Packit 352660
    return src - src_orig;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcUtf8Len (const FcChar8    *string,
Packit 352660
	   int		    len,
Packit 352660
	   int		    *nchar,
Packit 352660
	   int		    *wchar)
Packit 352660
{
Packit 352660
    int		n;
Packit 352660
    int		clen;
Packit 352660
    FcChar32	c;
Packit 352660
    FcChar32	max;
Packit 352660
Packit 352660
    n = 0;
Packit 352660
    max = 0;
Packit 352660
    while (len)
Packit 352660
    {
Packit 352660
	clen = FcUtf8ToUcs4 (string, &c, len);
Packit 352660
	if (clen <= 0)	/* malformed UTF8 string */
Packit 352660
	    return FcFalse;
Packit 352660
	if (c > max)
Packit 352660
	    max = c;
Packit 352660
	string += clen;
Packit 352660
	len -= clen;
Packit 352660
	n++;
Packit 352660
    }
Packit 352660
    *nchar = n;
Packit 352660
    if (max >= 0x10000)
Packit 352660
	*wchar = 4;
Packit 352660
    else if (max > 0x100)
Packit 352660
	*wchar = 2;
Packit 352660
    else
Packit 352660
	*wchar = 1;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
int
Packit 352660
FcUcs4ToUtf8 (FcChar32	ucs4,
Packit 352660
	      FcChar8	dest[FC_UTF8_MAX_LEN])
Packit 352660
{
Packit 352660
    int	bits;
Packit 352660
    FcChar8 *d = dest;
Packit 352660
Packit 352660
    if      (ucs4 <       0x80) {  *d++=  ucs4;                         bits= -6; }
Packit 352660
    else if (ucs4 <      0x800) {  *d++= ((ucs4 >>  6) & 0x1F) | 0xC0;  bits=  0; }
Packit 352660
    else if (ucs4 <    0x10000) {  *d++= ((ucs4 >> 12) & 0x0F) | 0xE0;  bits=  6; }
Packit 352660
    else if (ucs4 <   0x200000) {  *d++= ((ucs4 >> 18) & 0x07) | 0xF0;  bits= 12; }
Packit 352660
    else if (ucs4 <  0x4000000) {  *d++= ((ucs4 >> 24) & 0x03) | 0xF8;  bits= 18; }
Packit 352660
    else if (ucs4 < 0x80000000) {  *d++= ((ucs4 >> 30) & 0x01) | 0xFC;  bits= 24; }
Packit 352660
    else return 0;
Packit 352660
Packit 352660
    for ( ; bits >= 0; bits-= 6) {
Packit 352660
	*d++= ((ucs4 >> bits) & 0x3F) | 0x80;
Packit 352660
    }
Packit 352660
    return d - dest;
Packit 352660
}
Packit 352660
Packit 352660
#define GetUtf16(src,endian) \
Packit 352660
    ((FcChar16) ((src)[endian == FcEndianBig ? 0 : 1] << 8) | \
Packit 352660
     (FcChar16) ((src)[endian == FcEndianBig ? 1 : 0]))
Packit 352660
Packit 352660
int
Packit 352660
FcUtf16ToUcs4 (const FcChar8	*src_orig,
Packit 352660
	       FcEndian		endian,
Packit 352660
	       FcChar32		*dst,
Packit 352660
	       int		len)	/* in bytes */
Packit 352660
{
Packit 352660
    const FcChar8   *src = src_orig;
Packit 352660
    FcChar16	    a, b;
Packit 352660
    FcChar32	    result;
Packit 352660
Packit 352660
    if (len < 2)
Packit 352660
	return 0;
Packit 352660
Packit 352660
    a = GetUtf16 (src, endian); src += 2; len -= 2;
Packit 352660
Packit 352660
    /*
Packit 352660
     * Check for surrogate
Packit 352660
     */
Packit 352660
    if ((a & 0xfc00) == 0xd800)
Packit 352660
    {
Packit 352660
	if (len < 2)
Packit 352660
	    return 0;
Packit 352660
	b = GetUtf16 (src, endian); src += 2; len -= 2;
Packit 352660
	/*
Packit 352660
	 * Check for invalid surrogate sequence
Packit 352660
	 */
Packit 352660
	if ((b & 0xfc00) != 0xdc00)
Packit 352660
	    return 0;
Packit 352660
	result = ((((FcChar32) a & 0x3ff) << 10) |
Packit 352660
		  ((FcChar32) b & 0x3ff)) + 0x10000;
Packit 352660
    }
Packit 352660
    else
Packit 352660
	result = a;
Packit 352660
    *dst = result;
Packit 352660
    return src - src_orig;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcUtf16Len (const FcChar8   *string,
Packit 352660
	    FcEndian	    endian,
Packit 352660
	    int		    len,	/* in bytes */
Packit 352660
	    int		    *nchar,
Packit 352660
	    int		    *wchar)
Packit 352660
{
Packit 352660
    int		n;
Packit 352660
    int		clen;
Packit 352660
    FcChar32	c;
Packit 352660
    FcChar32	max;
Packit 352660
Packit 352660
    n = 0;
Packit 352660
    max = 0;
Packit 352660
    while (len)
Packit 352660
    {
Packit 352660
	clen = FcUtf16ToUcs4 (string, endian, &c, len);
Packit 352660
	if (clen <= 0)	/* malformed UTF8 string */
Packit 352660
	    return FcFalse;
Packit 352660
	if (c > max)
Packit 352660
	    max = c;
Packit 352660
	string += clen;
Packit 352660
	len -= clen;
Packit 352660
	n++;
Packit 352660
    }
Packit 352660
    *nchar = n;
Packit 352660
    if (max >= 0x10000)
Packit 352660
	*wchar = 4;
Packit 352660
    else if (max > 0x100)
Packit 352660
	*wchar = 2;
Packit 352660
    else
Packit 352660
	*wchar = 1;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrBufInit (FcStrBuf *buf, FcChar8 *init, int size)
Packit 352660
{
Packit 352660
    if (init)
Packit 352660
    {
Packit 352660
	buf->buf = init;
Packit 352660
	buf->size = size;
Packit 352660
    } else
Packit 352660
    {
Packit 352660
	buf->buf = buf->buf_static;
Packit 352660
	buf->size = sizeof (buf->buf_static);
Packit 352660
    }
Packit 352660
    buf->allocated = FcFalse;
Packit 352660
    buf->failed = FcFalse;
Packit 352660
    buf->len = 0;
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrBufDestroy (FcStrBuf *buf)
Packit 352660
{
Packit 352660
    if (buf->allocated)
Packit 352660
    {
Packit 352660
	free (buf->buf);
Packit 352660
	FcStrBufInit (buf, 0, 0);
Packit 352660
    }
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrBufDone (FcStrBuf *buf)
Packit 352660
{
Packit 352660
    FcChar8 *ret;
Packit 352660
Packit 352660
    if (buf->failed)
Packit 352660
	ret = NULL;
Packit 352660
    else
Packit 352660
	ret = malloc (buf->len + 1);
Packit 352660
    if (ret)
Packit 352660
    {
Packit 352660
	memcpy (ret, buf->buf, buf->len);
Packit 352660
	ret[buf->len] = '\0';
Packit 352660
    }
Packit 352660
    FcStrBufDestroy (buf);
Packit 352660
    return ret;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrBufDoneStatic (FcStrBuf *buf)
Packit 352660
{
Packit 352660
    FcStrBufChar (buf, '\0');
Packit 352660
Packit 352660
    if (buf->failed)
Packit 352660
	return NULL;
Packit 352660
Packit 352660
    return buf->buf;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrBufChar (FcStrBuf *buf, FcChar8 c)
Packit 352660
{
Packit 352660
    if (buf->len == buf->size)
Packit 352660
    {
Packit 352660
	FcChar8	    *new;
Packit 352660
	int	    size;
Packit 352660
Packit 352660
	if (buf->failed)
Packit 352660
	    return FcFalse;
Packit 352660
Packit 352660
	if (buf->allocated)
Packit 352660
	{
Packit 352660
	    size = buf->size * 2;
Packit 352660
	    new = realloc (buf->buf, size);
Packit 352660
	}
Packit 352660
	else
Packit 352660
	{
Packit 352660
	    size = buf->size + 64;
Packit 352660
	    new = malloc (size);
Packit 352660
	    if (new)
Packit 352660
	    {
Packit 352660
		buf->allocated = FcTrue;
Packit 352660
		memcpy (new, buf->buf, buf->len);
Packit 352660
	    }
Packit 352660
	}
Packit 352660
	if (!new)
Packit 352660
	{
Packit 352660
	    buf->failed = FcTrue;
Packit 352660
	    return FcFalse;
Packit 352660
	}
Packit 352660
	buf->size = size;
Packit 352660
	buf->buf = new;
Packit 352660
    }
Packit 352660
    buf->buf[buf->len++] = c;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrBufString (FcStrBuf *buf, const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar8 c;
Packit 352660
    while ((c = *s++))
Packit 352660
	if (!FcStrBufChar (buf, c))
Packit 352660
	    return FcFalse;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrBufData (FcStrBuf *buf, const FcChar8 *s, int len)
Packit 352660
{
Packit 352660
    while (len-- > 0)
Packit 352660
	if (!FcStrBufChar (buf, *s++))
Packit 352660
	    return FcFalse;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrUsesHome (const FcChar8 *s)
Packit 352660
{
Packit 352660
    return *s == '~';
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrIsAbsoluteFilename (const FcChar8 *s)
Packit 352660
{
Packit 352660
#ifdef _WIN32
Packit 352660
    if (*s == '\\' ||
Packit 352660
	(isalpha (*s) && s[1] == ':' && (s[2] == '/' || s[2] == '\\')))
Packit 352660
	return FcTrue;
Packit 352660
#endif
Packit 352660
    return *s == '/';
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrBuildFilename (const FcChar8 *path,
Packit 352660
		    ...)
Packit 352660
{
Packit 352660
    va_list ap;
Packit 352660
    FcStrSet *sset;
Packit 352660
    FcStrList *list;
Packit 352660
    FcChar8 *s, *ret = NULL, *p;
Packit 352660
    size_t len = 0;
Packit 352660
Packit 352660
    if (!path)
Packit 352660
	return NULL;
Packit 352660
Packit 352660
    sset = FcStrSetCreateEx (FCSS_ALLOW_DUPLICATES | FCSS_GROW_BY_64);
Packit 352660
    if (!sset)
Packit 352660
	return NULL;
Packit 352660
Packit 352660
    if (!FcStrSetAdd (sset, path))
Packit 352660
	goto bail0;
Packit 352660
Packit 352660
    va_start (ap, path);
Packit 352660
    while (1)
Packit 352660
    {
Packit 352660
	s = (FcChar8 *)va_arg (ap, FcChar8 *);
Packit 352660
	if (!s)
Packit 352660
	    break;
Packit 352660
	if (!FcStrSetAdd (sset, s))
Packit 352660
	    goto bail1;
Packit 352660
    }
Packit 352660
    list = FcStrListCreate (sset);
Packit 352660
    while ((s = FcStrListNext (list)))
Packit 352660
    {
Packit 352660
	len += strlen ((const char *)s) + 1;
Packit 352660
    }
Packit 352660
    list->n = 0;
Packit 352660
    ret = malloc (sizeof (FcChar8) * (len + 1));
Packit 352660
    if (!ret)
Packit 352660
	goto bail2;
Packit 352660
    p = ret;
Packit 352660
    while ((s = FcStrListNext (list)))
Packit 352660
    {
Packit 352660
	if (p != ret)
Packit 352660
	{
Packit 352660
	    p[0] = FC_DIR_SEPARATOR;
Packit 352660
	    p++;
Packit 352660
	}
Packit 352660
	len = strlen ((const char *)s);
Packit 352660
	memcpy (p, s, len);
Packit 352660
	p += len;
Packit 352660
    }
Packit 352660
    *p = 0;
Packit 352660
Packit 352660
bail2:
Packit 352660
    FcStrListDone (list);
Packit 352660
bail1:
Packit 352660
    va_end (ap);
Packit 352660
bail0:
Packit 352660
    FcStrSetDestroy (sset);
Packit 352660
Packit 352660
    return ret;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrCopyFilename (const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar8 *new;
Packit 352660
Packit 352660
    if (*s == '~')
Packit 352660
    {
Packit 352660
	FcChar8	*home = FcConfigHome ();
Packit 352660
	FcChar8	*full;
Packit 352660
	int	size;
Packit 352660
	if (!home)
Packit 352660
	    return NULL;
Packit 352660
	size = strlen ((char *) home) + strlen ((char *) s);
Packit 352660
	full = (FcChar8 *) malloc (size + 1);
Packit 352660
	if (!full)
Packit 352660
	    return NULL;
Packit 352660
	strcpy ((char *) full, (char *) home);
Packit 352660
	strcat ((char *) full, (char *) s + 1);
Packit 352660
	new = FcStrCanonFilename (full);
Packit 352660
	free (full);
Packit 352660
    }
Packit 352660
    else
Packit 352660
	new = FcStrCanonFilename (s);
Packit 352660
Packit 352660
    return new;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrLastSlash (const FcChar8  *path)
Packit 352660
{
Packit 352660
    FcChar8	    *slash;
Packit 352660
Packit 352660
    slash = (FcChar8 *) strrchr ((const char *) path, '/');
Packit 352660
#ifdef _WIN32
Packit 352660
    {
Packit 352660
        FcChar8     *backslash;
Packit 352660
Packit 352660
	backslash = (FcChar8 *) strrchr ((const char *) path, '\\');
Packit 352660
	if (!slash || (backslash && backslash > slash))
Packit 352660
	    slash = backslash;
Packit 352660
    }
Packit 352660
#endif
Packit 352660
Packit 352660
    return slash;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrDirname (const FcChar8 *file)
Packit 352660
{
Packit 352660
    FcChar8 *slash;
Packit 352660
    FcChar8 *dir;
Packit 352660
Packit 352660
    slash = FcStrLastSlash (file);
Packit 352660
    if (!slash)
Packit 352660
	return FcStrCopy ((FcChar8 *) ".");
Packit 352660
    dir = malloc ((slash - file) + 1);
Packit 352660
    if (!dir)
Packit 352660
	return 0;
Packit 352660
    strncpy ((char *) dir, (const char *) file, slash - file);
Packit 352660
    dir[slash - file] = '\0';
Packit 352660
    return dir;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrBasename (const FcChar8 *file)
Packit 352660
{
Packit 352660
    FcChar8 *slash;
Packit 352660
Packit 352660
    slash = FcStrLastSlash (file);
Packit 352660
    if (!slash)
Packit 352660
	return FcStrCopy (file);
Packit 352660
    return FcStrCopy (slash + 1);
Packit 352660
}
Packit 352660
Packit 352660
static FcChar8 *
Packit 352660
FcStrCanonAbsoluteFilename (const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar8 *file;
Packit 352660
    FcChar8 *f;
Packit 352660
    const FcChar8 *slash;
Packit 352660
    int size;
Packit 352660
Packit 352660
    size = strlen ((char *) s) + 1;
Packit 352660
    file = malloc (size);
Packit 352660
    if (!file)
Packit 352660
	return NULL;
Packit 352660
    slash = NULL;
Packit 352660
    f = file;
Packit 352660
#ifdef _WIN32
Packit 352660
    if (*s == '/' && *(s+1) == '/') /* Network path, do not squash // */
Packit 352660
	*f++ = *s++;
Packit 352660
#endif
Packit 352660
    for (;;) {
Packit 352660
	if (*s == '/' || *s == '\0')
Packit 352660
	{
Packit 352660
	    if (slash)
Packit 352660
	    {
Packit 352660
		switch (s - slash) {
Packit 352660
		case 1:
Packit 352660
		    f -= 1;	/* squash // and trim final / from file */
Packit 352660
		    break;
Packit 352660
		case 2:
Packit 352660
		    if (!strncmp ((char *) slash, "/.", 2))
Packit 352660
		    {
Packit 352660
			f -= 2;	/* trim /. from file */
Packit 352660
		    }
Packit 352660
		    break;
Packit 352660
		case 3:
Packit 352660
		    if (!strncmp ((char *) slash, "/..", 3))
Packit 352660
		    {
Packit 352660
			f -= 3;	/* trim /.. from file */
Packit 352660
			while (f > file) {
Packit 352660
			    if (*--f == '/')
Packit 352660
				break;
Packit 352660
			}
Packit 352660
		    }
Packit 352660
		    break;
Packit 352660
		}
Packit 352660
	    }
Packit 352660
	    slash = s;
Packit 352660
	}
Packit 352660
	if (!(*f++ = *s++))
Packit 352660
	    break;
Packit 352660
    }
Packit 352660
    return file;
Packit 352660
}
Packit 352660
Packit 352660
#ifdef _WIN32
Packit 352660
/*
Packit 352660
 * Convert '\\' to '/' , remove double '/'
Packit 352660
 */
Packit 352660
static void
Packit 352660
FcConvertDosPath (char *str)
Packit 352660
{
Packit 352660
  size_t len = strlen (str);
Packit 352660
  char *p = str;
Packit 352660
  char *dest = str;
Packit 352660
  char *end = str + len;
Packit 352660
  char last = 0;
Packit 352660
Packit 352660
  if (*p == '\\')
Packit 352660
    {
Packit 352660
      *p = '/';
Packit 352660
      p++;
Packit 352660
      dest++;
Packit 352660
    }
Packit 352660
  while (p < end)
Packit 352660
    {
Packit 352660
      if (*p == '\\')
Packit 352660
	*p = '/';
Packit 352660
Packit 352660
      if (*p != '/'
Packit 352660
	  || last != '/')
Packit 352660
	{
Packit 352660
	  *dest++ = *p;
Packit 352660
	}
Packit 352660
Packit 352660
      last = *p;
Packit 352660
      p++;
Packit 352660
    }
Packit 352660
Packit 352660
  *dest = 0;
Packit 352660
}
Packit 352660
#endif
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrCanonFilename (const FcChar8 *s)
Packit 352660
{
Packit 352660
#ifdef _WIN32
Packit 352660
    FcChar8 full[FC_MAX_FILE_LEN + 2];
Packit 352660
    int size = GetFullPathName ((LPCSTR) s, sizeof (full) -1,
Packit 352660
				(LPSTR) full, NULL);
Packit 352660
Packit 352660
    if (size == 0)
Packit 352660
	perror ("GetFullPathName");
Packit 352660
Packit 352660
    FcConvertDosPath ((char *) full);
Packit 352660
    return FcStrCanonAbsoluteFilename (full);
Packit 352660
#else
Packit 352660
    if (s[0] == '/')
Packit 352660
	return FcStrCanonAbsoluteFilename (s);
Packit 352660
    else
Packit 352660
    {
Packit 352660
	FcChar8	*full;
Packit 352660
	FcChar8 *file;
Packit 352660
Packit 352660
	FcChar8	cwd[FC_MAX_FILE_LEN + 2];
Packit 352660
	if (getcwd ((char *) cwd, FC_MAX_FILE_LEN) == NULL)
Packit 352660
	    return NULL;
Packit 352660
	full = FcStrBuildFilename (cwd, s, NULL);
Packit 352660
	file = FcStrCanonAbsoluteFilename (full);
Packit 352660
	FcStrFree (full);
Packit 352660
	return file;
Packit 352660
    }
Packit 352660
#endif
Packit 352660
}
Packit 352660
Packit 352660
Packit 352660
FcStrSet *
Packit 352660
FcStrSetCreate (void)
Packit 352660
{
Packit 352660
    return FcStrSetCreateEx (FCSS_DEFAULT);
Packit 352660
}
Packit 352660
Packit 352660
FcStrSet *
Packit 352660
FcStrSetCreateEx (unsigned int control)
Packit 352660
{
Packit 352660
    FcStrSet	*set = malloc (sizeof (FcStrSet));
Packit 352660
    if (!set)
Packit 352660
	return 0;
Packit 352660
    FcRefInit (&set->ref, 1);
Packit 352660
    set->num = 0;
Packit 352660
    set->size = 0;
Packit 352660
    set->strs = 0;
Packit 352660
    set->control = control;
Packit 352660
    return set;
Packit 352660
}
Packit 352660
Packit 352660
static FcBool
Packit 352660
_FcStrSetGrow (FcStrSet *set, int growElements)
Packit 352660
{
Packit 352660
    /* accommodate an additional NULL entry at the end of the array */
Packit 352660
    FcChar8 **strs = malloc ((set->size + growElements + 1) * sizeof (FcChar8 *));
Packit 352660
    if (!strs)
Packit 352660
        return FcFalse;
Packit 352660
    if (set->num)
Packit 352660
        memcpy (strs, set->strs, set->num * sizeof (FcChar8 *));
Packit 352660
    if (set->strs)
Packit 352660
        free (set->strs);
Packit 352660
    set->size = set->size + growElements;
Packit 352660
    set->strs = strs;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
static FcBool
Packit 352660
_FcStrSetAppend (FcStrSet *set, FcChar8 *s)
Packit 352660
{
Packit 352660
    if (!FcStrSetHasControlBit (set, FCSS_ALLOW_DUPLICATES))
Packit 352660
    {
Packit 352660
        if (FcStrSetMember (set, s))
Packit 352660
        {
Packit 352660
            FcStrFree (s);
Packit 352660
            return FcTrue;
Packit 352660
        }
Packit 352660
    }
Packit 352660
    if (set->num == set->size)
Packit 352660
    {
Packit 352660
        int growElements = FcStrSetHasControlBit (set, FCSS_GROW_BY_64) ? 64 : 1;
Packit 352660
        if (!_FcStrSetGrow(set, growElements))
Packit 352660
            return FcFalse;
Packit 352660
    }
Packit 352660
    set->strs[set->num++] = s;
Packit 352660
    set->strs[set->num] = 0;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetMember (FcStrSet *set, const FcChar8 *s)
Packit 352660
{
Packit 352660
    int	i;
Packit 352660
Packit 352660
    for (i = 0; i < set->num; i++)
Packit 352660
	if (!FcStrCmp (set->strs[i], s))
Packit 352660
	    return FcTrue;
Packit 352660
    return FcFalse;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetEqual (FcStrSet *sa, FcStrSet *sb)
Packit 352660
{
Packit 352660
    int	i;
Packit 352660
    if (sa->num != sb->num)
Packit 352660
	return FcFalse;
Packit 352660
    for (i = 0; i < sa->num; i++)
Packit 352660
	if (!FcStrSetMember (sb, sa->strs[i]))
Packit 352660
	    return FcFalse;
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetAdd (FcStrSet *set, const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar8 *new = FcStrCopy (s);
Packit 352660
    if (!new)
Packit 352660
	return FcFalse;
Packit 352660
    if (!_FcStrSetAppend (set, new))
Packit 352660
    {
Packit 352660
	FcStrFree (new);
Packit 352660
	return FcFalse;
Packit 352660
    }
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetAddFilename (FcStrSet *set, const FcChar8 *s)
Packit 352660
{
Packit 352660
    FcChar8 *new = FcStrCopyFilename (s);
Packit 352660
    if (!new)
Packit 352660
	return FcFalse;
Packit 352660
    if (!_FcStrSetAppend (set, new))
Packit 352660
    {
Packit 352660
	FcStrFree (new);
Packit 352660
	return FcFalse;
Packit 352660
    }
Packit 352660
    return FcTrue;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetAddLangs (FcStrSet *strs, const char *languages)
Packit 352660
{
Packit 352660
    const char *p = languages, *next;
Packit 352660
    FcChar8 lang[128] = {0}, *normalized_lang;
Packit 352660
    size_t len;
Packit 352660
    FcBool ret = FcFalse;
Packit 352660
Packit 352660
    if (!languages)
Packit 352660
	return FcFalse;
Packit 352660
Packit 352660
    while ((next = strchr (p, ':')))
Packit 352660
    {
Packit 352660
	len = next - p;
Packit 352660
	len = FC_MIN (len, 127);
Packit 352660
	strncpy ((char *) lang, p, len);
Packit 352660
	lang[len] = 0;
Packit 352660
	/* ignore an empty item */
Packit 352660
	if (*lang)
Packit 352660
	{
Packit 352660
	    normalized_lang = FcLangNormalize ((const FcChar8 *) lang);
Packit 352660
	    if (normalized_lang)
Packit 352660
	    {
Packit 352660
		FcStrSetAdd (strs, normalized_lang);
Packit 352660
		FcStrFree (normalized_lang);
Packit 352660
		ret = FcTrue;
Packit 352660
	    }
Packit 352660
	}
Packit 352660
	p = next + 1;
Packit 352660
    }
Packit 352660
    if (*p)
Packit 352660
    {
Packit 352660
	normalized_lang = FcLangNormalize ((const FcChar8 *) p);
Packit 352660
	if (normalized_lang)
Packit 352660
	{
Packit 352660
	    FcStrSetAdd (strs, normalized_lang);
Packit 352660
	    FcStrFree (normalized_lang);
Packit 352660
	    ret = FcTrue;
Packit 352660
	}
Packit 352660
    }
Packit 352660
Packit 352660
    return ret;
Packit 352660
}
Packit 352660
Packit 352660
FcBool
Packit 352660
FcStrSetDel (FcStrSet *set, const FcChar8 *s)
Packit 352660
{
Packit 352660
    int	i;
Packit 352660
Packit 352660
    for (i = 0; i < set->num; i++)
Packit 352660
	if (!FcStrCmp (set->strs[i], s))
Packit 352660
	{
Packit 352660
	    FcStrFree (set->strs[i]);
Packit 352660
	    /*
Packit 352660
	     * copy remaining string pointers and trailing
Packit 352660
	     * NULL
Packit 352660
	     */
Packit 352660
	    memmove (&set->strs[i], &set->strs[i+1],
Packit 352660
		     (set->num - i) * sizeof (FcChar8 *));
Packit 352660
	    set->num--;
Packit 352660
	    return FcTrue;
Packit 352660
	}
Packit 352660
    return FcFalse;
Packit 352660
}
Packit 352660
Packit 352660
/* TODO Make public */
Packit 352660
static FcStrSet *
Packit 352660
FcStrSetReference (FcStrSet *set)
Packit 352660
{
Packit 352660
    if (FcRefIsConst (&set->ref))
Packit 352660
	return set;
Packit 352660
Packit 352660
    FcRefInc (&set->ref);
Packit 352660
    return set;
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrSetDestroy (FcStrSet *set)
Packit 352660
{
Packit 352660
    int	i;
Packit 352660
Packit 352660
    /* We rely on this in FcGetDefaultLangs for caching. */
Packit 352660
    if (FcRefIsConst (&set->ref))
Packit 352660
	return;
Packit 352660
Packit 352660
    if (FcRefDec (&set->ref) != 1)
Packit 352660
	return;
Packit 352660
Packit 352660
    for (i = 0; i < set->num; i++)
Packit 352660
	FcStrFree (set->strs[i]);
Packit 352660
    if (set->strs)
Packit 352660
	free (set->strs);
Packit 352660
    free (set);
Packit 352660
}
Packit 352660
Packit 352660
FcStrList *
Packit 352660
FcStrListCreate (FcStrSet *set)
Packit 352660
{
Packit 352660
    FcStrList	*list;
Packit 352660
Packit 352660
    list = malloc (sizeof (FcStrList));
Packit 352660
    if (!list)
Packit 352660
	return 0;
Packit 352660
    list->set = set;
Packit 352660
    FcStrSetReference (set);
Packit 352660
    list->n = 0;
Packit 352660
    return list;
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrListFirst (FcStrList *list)
Packit 352660
{
Packit 352660
    list->n = 0;
Packit 352660
}
Packit 352660
Packit 352660
FcChar8 *
Packit 352660
FcStrListNext (FcStrList *list)
Packit 352660
{
Packit 352660
    if (list->n >= list->set->num)
Packit 352660
	return 0;
Packit 352660
    return list->set->strs[list->n++];
Packit 352660
}
Packit 352660
Packit 352660
void
Packit 352660
FcStrListDone (FcStrList *list)
Packit 352660
{
Packit 352660
    FcStrSetDestroy (list->set);
Packit 352660
    free (list);
Packit 352660
}
Packit 352660
Packit 352660
#define __fcstr__
Packit 352660
#include "fcaliastail.h"
Packit 352660
#undef __fcstr__