2 * Copyright © 2000 Keith Packard
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that
7 * copyright notice and this permission notice appear in supporting
8 * documentation, and that the name of the author(s) not be used in
9 * advertising or publicity pertaining to distribution of the software without
10 * specific, written prior permission. The authors make no
11 * representations about the suitability of this software for any purpose. It
12 * is provided "as is" without express or implied warranty.
14 * THE AUTHOR(S) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16 * EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
20 * PERFORMANCE OF THIS SOFTWARE.
26 /* Objects MT-safe for readonly access. */
29 FcPatternCreate (void)
33 p = (FcPattern *) malloc (sizeof (FcPattern));
38 p->elts_offset = FcPtrToOffset (p, NULL);
39 FcRefInit (&p->ref, 1);
44 FcValueDestroy (FcValue v)
46 switch ((int) v.type) {
51 FcMatrixFree ((FcMatrix *) v.u.m);
54 FcCharSetDestroy ((FcCharSet *) v.u.c);
57 FcLangSetDestroy ((FcLangSet *) v.u.l);
65 FcValueCanonicalize (const FcValue *v)
69 switch ((int) v->type)
72 new.u.s = FcValueString(v);
73 new.type = FcTypeString;
76 new.u.c = FcValueCharSet(v);
77 new.type = FcTypeCharSet;
80 new.u.l = FcValueLangSet(v);
81 new.type = FcTypeLangSet;
91 FcValueSave (FcValue v)
93 switch ((int) v.type) {
95 v.u.s = FcStrdup (v.u.s);
100 v.u.m = FcMatrixCopy (v.u.m);
105 v.u.c = FcCharSetCopy ((FcCharSet *) v.u.c);
110 v.u.l = FcLangSetCopy (v.u.l);
121 FcValueListCreate (void)
123 return calloc (1, sizeof (FcValueList));
127 FcValueListDestroy (FcValueListPtr l)
132 switch ((int) l->value.type) {
134 FcFree (l->value.u.s);
137 FcMatrixFree ((FcMatrix *)l->value.u.m);
141 ((FcCharSet *) (l->value.u.c));
145 ((FcLangSet *) (l->value.u.l));
150 next = FcValueListNext(l);
156 FcValueListPrepend (FcValueListPtr vallist,
158 FcValueBinding binding)
162 if (value.type == FcTypeVoid)
164 new = FcValueListCreate ();
168 new->value = FcValueSave (value);
169 new->binding = binding;
176 FcValueListAppend (FcValueListPtr vallist,
178 FcValueBinding binding)
180 FcValueListPtr new, last;
182 if (value.type == FcTypeVoid)
184 new = FcValueListCreate ();
188 new->value = FcValueSave (value);
189 new->binding = binding;
194 for (last = vallist; FcValueListNext (last); last = FcValueListNext (last));
205 FcValueListDuplicate(FcValueListPtr orig)
207 FcValueListPtr new = NULL, l, t = NULL;
210 for (l = orig; l != NULL; l = FcValueListNext (l))
214 t = new = FcValueListCreate();
218 t->next = FcValueListCreate();
219 t = FcValueListNext (t);
221 v = FcValueCanonicalize (&l->value);
222 t->value = FcValueSave (v);
223 t->binding = l->binding;
231 FcValueEqual (FcValue va, FcValue vb)
233 if (va.type != vb.type)
235 if (va.type == FcTypeInteger)
237 va.type = FcTypeDouble;
240 if (vb.type == FcTypeInteger)
242 vb.type = FcTypeDouble;
245 if (va.type != vb.type)
250 return FcFalse; /* don't know how to compare this object */
254 return va.u.i == vb.u.i;
256 return va.u.d == vb.u.d;
258 return FcStrCmpIgnoreCase (va.u.s, vb.u.s) == 0;
260 return va.u.b == vb.u.b;
262 return FcMatrixEqual (va.u.m, vb.u.m);
264 return FcCharSetEqual (va.u.c, vb.u.c);
266 return va.u.f == vb.u.f;
268 return FcLangSetEqual (va.u.l, vb.u.l);
274 FcDoubleHash (double d)
284 FcStringHash (const FcChar8 *s)
291 h = ((h << 1) | (h >> 31)) ^ c;
296 FcValueHash (const FcValue *v)
303 return (FcChar32) v->u.i;
305 return FcDoubleHash (v->u.d);
307 return FcStringHash (FcValueString(v));
309 return (FcChar32) v->u.b;
311 return (FcDoubleHash (v->u.m->xx) ^
312 FcDoubleHash (v->u.m->xy) ^
313 FcDoubleHash (v->u.m->yx) ^
314 FcDoubleHash (v->u.m->yy));
316 return (FcChar32) FcValueCharSet(v)->num;
318 return FcStringHash ((const FcChar8 *) ((FT_Face) v->u.f)->family_name) ^
319 FcStringHash ((const FcChar8 *) ((FT_Face) v->u.f)->style_name);
321 return FcLangSetHash (FcValueLangSet(v));
327 FcValueListEqual (FcValueListPtr la, FcValueListPtr lb)
334 if (!FcValueEqual (la->value, lb->value))
336 la = FcValueListNext(la);
337 lb = FcValueListNext(lb);
345 FcValueListHash (FcValueListPtr l)
349 for (; l; l = FcValueListNext(l))
351 hash = ((hash << 1) | (hash >> 31)) ^ FcValueHash (&l->value);
357 FcPatternDestroy (FcPattern *p)
365 if (FcRefIsConst (&p->ref))
367 FcCacheObjectDereference (p);
371 if (FcRefDec (&p->ref) != 1)
374 elts = FcPatternElts (p);
375 for (i = 0; i < p->num; i++)
376 FcValueListDestroy (FcPatternEltValues(&elts[i]));
383 FcPatternObjectPosition (const FcPattern *p, FcObject object)
385 int low, high, mid, c;
386 FcPatternElt *elts = FcPatternElts(p);
394 mid = (low + high) >> 1;
395 c = elts[mid].object - object;
409 FcPatternObjectFindElt (const FcPattern *p, FcObject object)
411 int i = FcPatternObjectPosition (p, object);
414 return &FcPatternElts(p)[i];
418 FcPatternObjectInsertElt (FcPattern *p, FcObject object)
423 i = FcPatternObjectPosition (p, object);
428 /* reallocate array */
429 if (p->num + 1 >= p->size)
431 int s = p->size + 16;
434 FcPatternElt *e0 = FcPatternElts(p);
435 e = (FcPatternElt *) realloc (e0, s * sizeof (FcPatternElt));
436 if (!e) /* maybe it was mmapped */
438 e = malloc(s * sizeof (FcPatternElt));
440 memcpy(e, e0, p->num * sizeof (FcPatternElt));
444 e = (FcPatternElt *) malloc (s * sizeof (FcPatternElt));
447 p->elts_offset = FcPtrToOffset (p, e);
450 e[p->size].object = 0;
451 e[p->size].values = NULL;
456 e = FcPatternElts(p);
460 sizeof (FcPatternElt) *
466 e[i].object = object;
470 return FcPatternElts(p) + i;
474 FcPatternEqual (const FcPattern *pa, const FcPattern *pb)
477 FcPatternElt *pae, *pbe;
482 if (pa->num != pb->num)
484 pae = FcPatternElts(pa);
485 pbe = FcPatternElts(pb);
486 for (i = 0; i < pa->num; i++)
488 if (pae[i].object != pbe[i].object)
490 if (!FcValueListEqual (FcPatternEltValues(&pae[i]),
491 FcPatternEltValues(&pbe[i])))
498 FcPatternHash (const FcPattern *p)
502 FcPatternElt *pe = FcPatternElts(p);
504 for (i = 0; i < p->num; i++)
506 h = (((h << 1) | (h >> 31)) ^
508 FcValueListHash (FcPatternEltValues(&pe[i])));
514 FcPatternEqualSubset (const FcPattern *pai, const FcPattern *pbi, const FcObjectSet *os)
516 FcPatternElt *ea, *eb;
519 for (i = 0; i < os->nobject; i++)
521 FcObject object = FcObjectFromName (os->objects[i]);
522 ea = FcPatternObjectFindElt (pai, object);
523 eb = FcPatternObjectFindElt (pbi, object);
528 if (!FcValueListEqual (FcPatternEltValues(ea), FcPatternEltValues(eb)))
541 FcPatternObjectListAdd (FcPattern *p,
547 FcValueListPtr l, *prev;
549 if (FcRefIsConst (&p->ref))
553 * Make sure the stored type is valid for built-in objects
555 for (l = list; l != NULL; l = FcValueListNext (l))
557 if (!FcObjectValidType (object, l->value.type))
560 "Fontconfig warning: FcPattern object %s does not accept value", FcObjectName (object));
561 FcValuePrintFile (stderr, l->value);
562 fprintf (stderr, "\n");
567 e = FcPatternObjectInsertElt (p, object);
573 for (prev = &e->values; *prev; prev = &(*prev)->next)
579 for (prev = &list; *prev; prev = &(*prev)->next)
592 FcPatternObjectAddWithBinding (FcPattern *p,
595 FcValueBinding binding,
599 FcValueListPtr new, *prev;
601 if (FcRefIsConst (&p->ref))
604 new = FcValueListCreate ();
608 value = FcValueSave (value);
609 if (value.type == FcTypeVoid)
613 * Make sure the stored type is valid for built-in objects
615 if (!FcObjectValidType (object, value.type))
618 "Fontconfig warning: FcPattern object %s does not accept value",
619 FcObjectName (object));
620 FcValuePrintFile (stderr, value);
621 fprintf (stderr, "\n");
626 new->binding = binding;
629 e = FcPatternObjectInsertElt (p, object);
635 for (prev = &e->values; *prev; prev = &(*prev)->next)
641 new->next = e->values;
648 FcValueDestroy (value);
656 FcPatternObjectAdd (FcPattern *p, FcObject object, FcValue value, FcBool append)
658 return FcPatternObjectAddWithBinding (p, object,
659 value, FcValueBindingStrong, append);
663 FcPatternAdd (FcPattern *p, const char *object, FcValue value, FcBool append)
665 return FcPatternObjectAddWithBinding (p, FcObjectFromName (object),
666 value, FcValueBindingStrong, append);
670 FcPatternAddWeak (FcPattern *p, const char *object, FcValue value, FcBool append)
672 return FcPatternObjectAddWithBinding (p, FcObjectFromName (object),
673 value, FcValueBindingWeak, append);
677 FcPatternObjectDel (FcPattern *p, FcObject object)
681 e = FcPatternObjectFindElt (p, object);
686 FcValueListDestroy (e->values);
688 /* shuffle existing ones down */
690 (FcPatternElts(p) + p->num - (e + 1)) *
691 sizeof (FcPatternElt));
693 e = FcPatternElts(p) + p->num;
700 FcPatternDel (FcPattern *p, const char *object)
702 return FcPatternObjectDel (p, FcObjectFromName (object));
706 FcPatternRemove (FcPattern *p, const char *object, int id)
709 FcValueListPtr *prev, l;
711 e = FcPatternObjectFindElt (p, FcObjectFromName (object));
714 for (prev = &e->values; (l = *prev); prev = &l->next)
720 FcValueListDestroy (l);
722 FcPatternDel (p, object);
731 FcPatternObjectAddInteger (FcPattern *p, FcObject object, int i)
735 v.type = FcTypeInteger;
737 return FcPatternObjectAdd (p, object, v, FcTrue);
741 FcPatternAddInteger (FcPattern *p, const char *object, int i)
743 return FcPatternObjectAddInteger (p, FcObjectFromName (object), i);
747 FcPatternObjectAddDouble (FcPattern *p, FcObject object, double d)
751 v.type = FcTypeDouble;
753 return FcPatternObjectAdd (p, object, v, FcTrue);
758 FcPatternAddDouble (FcPattern *p, const char *object, double d)
760 return FcPatternObjectAddDouble (p, FcObjectFromName (object), d);
764 FcPatternObjectAddString (FcPattern *p, FcObject object, const FcChar8 *s)
772 return FcPatternObjectAdd (p, object, v, FcTrue);
775 v.type = FcTypeString;
777 return FcPatternObjectAdd (p, object, v, FcTrue);
781 FcPatternAddString (FcPattern *p, const char *object, const FcChar8 *s)
783 return FcPatternObjectAddString (p, FcObjectFromName (object), s);
787 FcPatternAddMatrix (FcPattern *p, const char *object, const FcMatrix *s)
791 v.type = FcTypeMatrix;
793 return FcPatternAdd (p, object, v, FcTrue);
798 FcPatternObjectAddBool (FcPattern *p, FcObject object, FcBool b)
804 return FcPatternObjectAdd (p, object, v, FcTrue);
808 FcPatternAddBool (FcPattern *p, const char *object, FcBool b)
810 return FcPatternObjectAddBool (p, FcObjectFromName (object), b);
814 FcPatternAddCharSet (FcPattern *p, const char *object, const FcCharSet *c)
818 v.type = FcTypeCharSet;
819 v.u.c = (FcCharSet *)c;
820 return FcPatternAdd (p, object, v, FcTrue);
824 FcPatternAddFTFace (FcPattern *p, const char *object, const FT_Face f)
828 v.type = FcTypeFTFace;
830 return FcPatternAdd (p, object, v, FcTrue);
834 FcPatternAddLangSet (FcPattern *p, const char *object, const FcLangSet *ls)
838 v.type = FcTypeLangSet;
839 v.u.l = (FcLangSet *)ls;
840 return FcPatternAdd (p, object, v, FcTrue);
844 FcPatternObjectGet (const FcPattern *p, FcObject object, int id, FcValue *v)
850 return FcResultNoMatch;
851 e = FcPatternObjectFindElt (p, object);
853 return FcResultNoMatch;
854 for (l = FcPatternEltValues(e); l; l = FcValueListNext(l))
858 *v = FcValueCanonicalize(&l->value);
859 return FcResultMatch;
867 FcPatternGet (const FcPattern *p, const char *object, int id, FcValue *v)
869 return FcPatternObjectGet (p, FcObjectFromName (object), id, v);
873 FcPatternObjectGetInteger (const FcPattern *p, FcObject object, int id, int *i)
878 r = FcPatternObjectGet (p, object, id, &v);
879 if (r != FcResultMatch)
881 switch ((int) v.type) {
889 return FcResultTypeMismatch;
891 return FcResultMatch;
895 FcPatternGetInteger (const FcPattern *p, const char *object, int id, int *i)
897 return FcPatternObjectGetInteger (p, FcObjectFromName (object), id, i);
902 FcPatternObjectGetDouble (const FcPattern *p, FcObject object, int id, double *d)
907 r = FcPatternObjectGet (p, object, id, &v);
908 if (r != FcResultMatch)
910 switch ((int) v.type) {
918 return FcResultTypeMismatch;
920 return FcResultMatch;
924 FcPatternGetDouble (const FcPattern *p, const char *object, int id, double *d)
926 return FcPatternObjectGetDouble (p, FcObjectFromName (object), id, d);
930 FcPatternObjectGetString (const FcPattern *p, FcObject object, int id, FcChar8 ** s)
935 r = FcPatternObjectGet (p, object, id, &v);
936 if (r != FcResultMatch)
938 if (v.type != FcTypeString)
939 return FcResultTypeMismatch;
941 *s = (FcChar8 *) v.u.s;
942 return FcResultMatch;
946 FcPatternGetString (const FcPattern *p, const char *object, int id, FcChar8 ** s)
948 return FcPatternObjectGetString (p, FcObjectFromName (object), id, s);
952 FcPatternGetMatrix(const FcPattern *p, const char *object, int id, FcMatrix **m)
957 r = FcPatternGet (p, object, id, &v);
958 if (r != FcResultMatch)
960 if (v.type != FcTypeMatrix)
961 return FcResultTypeMismatch;
962 *m = (FcMatrix *)v.u.m;
963 return FcResultMatch;
968 FcPatternGetBool(const FcPattern *p, const char *object, int id, FcBool *b)
973 r = FcPatternGet (p, object, id, &v);
974 if (r != FcResultMatch)
976 if (v.type != FcTypeBool)
977 return FcResultTypeMismatch;
979 return FcResultMatch;
983 FcPatternGetCharSet(const FcPattern *p, const char *object, int id, FcCharSet **c)
988 r = FcPatternGet (p, object, id, &v);
989 if (r != FcResultMatch)
991 if (v.type != FcTypeCharSet)
992 return FcResultTypeMismatch;
993 *c = (FcCharSet *)v.u.c;
994 return FcResultMatch;
998 FcPatternGetFTFace(const FcPattern *p, const char *object, int id, FT_Face *f)
1003 r = FcPatternGet (p, object, id, &v);
1004 if (r != FcResultMatch)
1006 if (v.type != FcTypeFTFace)
1007 return FcResultTypeMismatch;
1008 *f = (FT_Face) v.u.f;
1009 return FcResultMatch;
1013 FcPatternGetLangSet(const FcPattern *p, const char *object, int id, FcLangSet **ls)
1018 r = FcPatternGet (p, object, id, &v);
1019 if (r != FcResultMatch)
1021 if (v.type != FcTypeLangSet)
1022 return FcResultTypeMismatch;
1023 *ls = (FcLangSet *)v.u.l;
1024 return FcResultMatch;
1028 FcPatternDuplicate (const FcPattern *orig)
1035 new = FcPatternCreate ();
1039 e = FcPatternElts(orig);
1041 for (i = 0; i < orig->num; i++)
1043 for (l = FcPatternEltValues(e + i); l; l = FcValueListNext(l))
1045 if (!FcPatternObjectAddWithBinding (new, e[i].object,
1046 FcValueCanonicalize(&l->value),
1057 FcPatternDestroy (new);
1063 FcPatternReference (FcPattern *p)
1065 if (!FcRefIsConst (&p->ref))
1068 FcCacheObjectReference (p);
1072 FcPatternVaBuild (FcPattern *p, va_list va)
1076 FcPatternVapBuild (ret, p, va);
1081 FcPatternBuild (FcPattern *p, ...)
1086 FcPatternVapBuild (p, p, va);
1092 * Add all of the elements in 's' to 'p'
1095 FcPatternAppend (FcPattern *p, FcPattern *s)
1101 for (i = 0; i < s->num; i++)
1103 e = FcPatternElts(s)+i;
1104 for (v = FcPatternEltValues(e); v; v = FcValueListNext(v))
1106 if (!FcPatternObjectAddWithBinding (p, e->object,
1107 FcValueCanonicalize(&v->value),
1108 v->binding, FcTrue))
1116 FcPatternFilter (FcPattern *p, const FcObjectSet *os)
1124 return FcPatternDuplicate (p);
1126 ret = FcPatternCreate ();
1130 for (i = 0; i < os->nobject; i++)
1132 FcObject object = FcObjectFromName (os->objects[i]);
1133 e = FcPatternObjectFindElt (p, object);
1136 for (v = FcPatternEltValues(e); v; v = FcValueListNext(v))
1138 if (!FcPatternObjectAddWithBinding (ret, e->object,
1139 FcValueCanonicalize(&v->value),
1140 v->binding, FcTrue))
1148 FcPatternDestroy (ret);
1154 FcPatternSerializeAlloc (FcSerialize *serialize, const FcPattern *pat)
1157 FcPatternElt *elts = FcPatternElts(pat);
1159 if (!FcSerializeAlloc (serialize, pat, sizeof (FcPattern)))
1161 if (!FcSerializeAlloc (serialize, elts, pat->num * sizeof (FcPatternElt)))
1163 for (i = 0; i < pat->num; i++)
1164 if (!FcValueListSerializeAlloc (serialize, FcPatternEltValues(elts+i)))
1170 FcPatternSerialize (FcSerialize *serialize, const FcPattern *pat)
1172 FcPattern *pat_serialized;
1173 FcPatternElt *elts = FcPatternElts (pat);
1174 FcPatternElt *elts_serialized;
1175 FcValueList *values_serialized;
1178 pat_serialized = FcSerializePtr (serialize, pat);
1179 if (!pat_serialized)
1181 *pat_serialized = *pat;
1182 pat_serialized->size = pat->num;
1183 FcRefSetConst (&pat_serialized->ref);
1185 elts_serialized = FcSerializePtr (serialize, elts);
1186 if (!elts_serialized)
1189 pat_serialized->elts_offset = FcPtrToOffset (pat_serialized,
1192 for (i = 0; i < pat->num; i++)
1194 values_serialized = FcValueListSerialize (serialize, FcPatternEltValues (elts+i));
1195 if (!values_serialized)
1197 elts_serialized[i].object = elts[i].object;
1198 elts_serialized[i].values = FcPtrToEncodedOffset (&elts_serialized[i],
1202 if (FcDebug() & FC_DBG_CACHEV) {
1203 printf ("Raw pattern:\n");
1204 FcPatternPrint (pat);
1205 printf ("Serialized pattern:\n");
1206 FcPatternPrint (pat_serialized);
1209 return pat_serialized;
1213 FcValueListSerializeAlloc (FcSerialize *serialize, const FcValueList *vl)
1217 if (!FcSerializeAlloc (serialize, vl, sizeof (FcValueList)))
1219 switch ((int) vl->value.type) {
1221 if (!FcStrSerializeAlloc (serialize, vl->value.u.s))
1225 if (!FcCharSetSerializeAlloc (serialize, vl->value.u.c))
1229 if (!FcLangSetSerializeAlloc (serialize, vl->value.u.l))
1241 FcValueListSerialize (FcSerialize *serialize, const FcValueList *vl)
1243 FcValueList *vl_serialized;
1244 FcChar8 *s_serialized;
1245 FcCharSet *c_serialized;
1246 FcLangSet *l_serialized;
1247 FcValueList *head_serialized = NULL;
1248 FcValueList *prev_serialized = NULL;
1252 vl_serialized = FcSerializePtr (serialize, vl);
1256 if (prev_serialized)
1257 prev_serialized->next = FcPtrToEncodedOffset (prev_serialized,
1261 head_serialized = vl_serialized;
1263 vl_serialized->next = NULL;
1264 vl_serialized->value.type = vl->value.type;
1265 switch ((int) vl->value.type) {
1267 vl_serialized->value.u.i = vl->value.u.i;
1270 vl_serialized->value.u.d = vl->value.u.d;
1273 s_serialized = FcStrSerialize (serialize, vl->value.u.s);
1276 vl_serialized->value.u.s = FcPtrToEncodedOffset (&vl_serialized->value,
1281 vl_serialized->value.u.b = vl->value.u.b;
1287 c_serialized = FcCharSetSerialize (serialize, vl->value.u.c);
1290 vl_serialized->value.u.c = FcPtrToEncodedOffset (&vl_serialized->value,
1298 l_serialized = FcLangSetSerialize (serialize, vl->value.u.l);
1301 vl_serialized->value.u.l = FcPtrToEncodedOffset (&vl_serialized->value,
1308 prev_serialized = vl_serialized;
1311 return head_serialized;
1314 #include "fcaliastail.h"
1315 #include "fcftaliastail.h"