Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/src/java.desktop/unix/native/common/awt/fontpath.c
41153 views
1
/*
2
* Copyright (c) 1998, 2020, Oracle and/or its affiliates. All rights reserved.
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
*
5
* This code is free software; you can redistribute it and/or modify it
6
* under the terms of the GNU General Public License version 2 only, as
7
* published by the Free Software Foundation. Oracle designates this
8
* particular file as subject to the "Classpath" exception as provided
9
* by Oracle in the LICENSE file that accompanied this code.
10
*
11
* This code is distributed in the hope that it will be useful, but WITHOUT
12
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14
* version 2 for more details (a copy is included in the LICENSE file that
15
* accompanied this code).
16
*
17
* You should have received a copy of the GNU General Public License version
18
* 2 along with this work; if not, write to the Free Software Foundation,
19
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20
*
21
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22
* or visit www.oracle.com if you need additional information or have any
23
* questions.
24
*/
25
26
#if defined(__linux__)
27
#include <string.h>
28
#endif /* __linux__ */
29
#include <stdio.h>
30
#include <stdlib.h>
31
#include <strings.h>
32
#include <sys/types.h>
33
#include <sys/stat.h>
34
#include <sys/mman.h>
35
#include <fcntl.h>
36
#include <unistd.h>
37
38
#include <jni.h>
39
#include <jni_util.h>
40
#include <jvm_md.h>
41
#include <sizecalc.h>
42
#ifndef HEADLESS
43
#include <X11/Xlib.h>
44
#include <awt.h>
45
#else
46
/* locks ought to be included from awt.h */
47
#define AWT_LOCK()
48
#define AWT_UNLOCK()
49
#endif /* !HEADLESS */
50
51
#if defined(__linux__) && !defined(MAP_FAILED)
52
#define MAP_FAILED ((caddr_t)-1)
53
#endif
54
55
#ifndef HEADLESS
56
extern Display *awt_display;
57
#endif /* !HEADLESS */
58
59
#define FONTCONFIG_DLL_VERSIONED VERSIONED_JNI_LIB_NAME("fontconfig", "1")
60
#define FONTCONFIG_DLL JNI_LIB_NAME("fontconfig")
61
62
#define MAXFDIRS 512 /* Max number of directories that contain fonts */
63
64
#if defined( __linux__)
65
/* All the known interesting locations we have discovered on
66
* various flavors of Linux
67
*/
68
static char *fullLinuxFontPath[] = {
69
"/usr/X11R6/lib/X11/fonts/TrueType", /* RH 7.1+ */
70
"/usr/X11R6/lib/X11/fonts/truetype", /* SuSE */
71
"/usr/X11R6/lib/X11/fonts/tt",
72
"/usr/X11R6/lib/X11/fonts/TTF",
73
"/usr/X11R6/lib/X11/fonts/OTF", /* RH 9.0 (but empty!) */
74
"/usr/share/fonts/ja/TrueType", /* RH 7.2+ */
75
"/usr/share/fonts/truetype",
76
"/usr/share/fonts/ko/TrueType", /* RH 9.0 */
77
"/usr/share/fonts/zh_CN/TrueType", /* RH 9.0 */
78
"/usr/share/fonts/zh_TW/TrueType", /* RH 9.0 */
79
"/var/lib/defoma/x-ttcidfont-conf.d/dirs/TrueType", /* Debian */
80
"/usr/X11R6/lib/X11/fonts/Type1",
81
"/usr/share/fonts/default/Type1", /* RH 9.0 */
82
NULL, /* terminates the list */
83
};
84
#elif defined(_AIX)
85
static char *fullAixFontPath[] = {
86
"/usr/lpp/X11/lib/X11/fonts/Type1", /* from X11.fnt.iso_T1 */
87
"/usr/lpp/X11/lib/X11/fonts/TrueType", /* from X11.fnt.ucs.ttf */
88
NULL, /* terminates the list */
89
};
90
#endif
91
92
static char **getFontConfigLocations();
93
94
typedef struct {
95
const char *name[MAXFDIRS];
96
int num;
97
} fDirRecord, *fDirRecordPtr;
98
99
#ifndef HEADLESS
100
101
/*
102
* Returns True if display is local, False of it's remote.
103
*/
104
jboolean isDisplayLocal(JNIEnv *env) {
105
static jboolean isLocal = False;
106
static jboolean isLocalSet = False;
107
jboolean ret;
108
109
if (! isLocalSet) {
110
jclass geCls = (*env)->FindClass(env, "java/awt/GraphicsEnvironment");
111
CHECK_NULL_RETURN(geCls, JNI_FALSE);
112
jmethodID getLocalGE = (*env)->GetStaticMethodID(env, geCls,
113
"getLocalGraphicsEnvironment",
114
"()Ljava/awt/GraphicsEnvironment;");
115
CHECK_NULL_RETURN(getLocalGE, JNI_FALSE);
116
jobject ge = (*env)->CallStaticObjectMethod(env, geCls, getLocalGE);
117
JNU_CHECK_EXCEPTION_RETURN(env, JNI_FALSE);
118
119
jclass sgeCls = (*env)->FindClass(env,
120
"sun/java2d/SunGraphicsEnvironment");
121
CHECK_NULL_RETURN(sgeCls, JNI_FALSE);
122
if ((*env)->IsInstanceOf(env, ge, sgeCls)) {
123
jmethodID isDisplayLocal = (*env)->GetMethodID(env, sgeCls,
124
"isDisplayLocal",
125
"()Z");
126
JNU_CHECK_EXCEPTION_RETURN(env, JNI_FALSE);
127
isLocal = (*env)->CallBooleanMethod(env, ge, isDisplayLocal);
128
JNU_CHECK_EXCEPTION_RETURN(env, JNI_FALSE);
129
} else {
130
isLocal = True;
131
}
132
isLocalSet = True;
133
}
134
135
return isLocal;
136
}
137
138
static void AddFontsToX11FontPath ( fDirRecord *fDirP )
139
{
140
char *onePath;
141
int index, nPaths;
142
int origNumPaths, length;
143
int origIndex;
144
int totalDirCount;
145
char **origFontPath;
146
char **tempFontPath;
147
int doNotAppend;
148
int *appendDirList;
149
char **newFontPath;
150
int err, compareLength;
151
char fontDirPath[512];
152
int dirFile;
153
154
doNotAppend = 0;
155
156
if ( fDirP->num == 0 ) return;
157
158
appendDirList = SAFE_SIZE_ARRAY_ALLOC(malloc, fDirP->num, sizeof ( int ));
159
if ( appendDirList == NULL ) {
160
return; /* if it fails we cannot do much */
161
}
162
163
origFontPath = XGetFontPath ( awt_display, &nPaths );
164
165
totalDirCount = nPaths;
166
origNumPaths = nPaths;
167
tempFontPath = origFontPath;
168
169
170
for (index = 0; index < fDirP->num; index++ ) {
171
172
doNotAppend = 0;
173
174
tempFontPath = origFontPath;
175
for ( origIndex = 0; origIndex < nPaths; origIndex++ ) {
176
177
onePath = *tempFontPath;
178
179
compareLength = strlen ( onePath );
180
if ( onePath[compareLength -1] == '/' )
181
compareLength--;
182
183
/* there is a slash at the end of every solaris X11 font path name */
184
if ( strncmp ( onePath, fDirP->name[index], compareLength ) == 0 ) {
185
doNotAppend = 1;
186
break;
187
}
188
tempFontPath++;
189
}
190
191
appendDirList[index] = 0;
192
if ( doNotAppend == 0 ) {
193
snprintf(fontDirPath, sizeof(fontDirPath), "%s/fonts.dir", fDirP->name[index]);
194
fontDirPath[sizeof(fontDirPath) - 1] = '\0';
195
dirFile = open ( fontDirPath, O_RDONLY, 0 );
196
if ( dirFile == -1 ) {
197
doNotAppend = 1;
198
} else {
199
close ( dirFile );
200
totalDirCount++;
201
appendDirList[index] = 1;
202
}
203
}
204
205
}
206
207
/* if no changes are required do not bother to do a setfontpath */
208
if ( totalDirCount == nPaths ) {
209
free ( ( void *) appendDirList );
210
XFreeFontPath ( origFontPath );
211
return;
212
}
213
214
215
newFontPath = SAFE_SIZE_ARRAY_ALLOC(malloc, totalDirCount, sizeof(char *));
216
/* if it fails free things and get out */
217
if ( newFontPath == NULL ) {
218
free ( ( void *) appendDirList );
219
XFreeFontPath ( origFontPath );
220
return;
221
}
222
223
for ( origIndex = 0; origIndex < nPaths; origIndex++ ) {
224
onePath = origFontPath[origIndex];
225
newFontPath[origIndex] = onePath;
226
}
227
228
/* now add the other font paths */
229
230
for (index = 0; index < fDirP->num; index++ ) {
231
232
if ( appendDirList[index] == 1 ) {
233
234
/* printf ( "Appending %s\n", fDirP->name[index] ); */
235
236
onePath = SAFE_SIZE_ARRAY_ALLOC(malloc, strlen (fDirP->name[index]) + 2, sizeof( char ) );
237
if (onePath == NULL) {
238
free ( ( void *) appendDirList );
239
240
for ( index = origIndex; index < nPaths; index++ ) {
241
free( newFontPath[index] );
242
}
243
244
free( ( void *) newFontPath);
245
XFreeFontPath ( origFontPath );
246
return;
247
}
248
strcpy ( onePath, fDirP->name[index] );
249
strcat ( onePath, "/" );
250
newFontPath[nPaths++] = onePath;
251
/* printf ( "The path to be appended is %s\n", onePath ); */
252
}
253
}
254
255
/* printf ( "The dir count = %d\n", totalDirCount ); */
256
free ( ( void *) appendDirList );
257
258
XSetFontPath ( awt_display, newFontPath, totalDirCount );
259
260
for ( index = origNumPaths; index < totalDirCount; index++ ) {
261
free( newFontPath[index] );
262
}
263
264
free ( (void *) newFontPath );
265
XFreeFontPath ( origFontPath );
266
return;
267
}
268
#endif /* !HEADLESS */
269
270
271
#ifndef HEADLESS
272
static char **getX11FontPath ()
273
{
274
char **x11Path, **fontdirs;
275
int i, pos, slen, nPaths, numDirs;
276
277
x11Path = XGetFontPath (awt_display, &nPaths);
278
279
/* This isn't ever going to be perfect: the font path may contain
280
* much we aren't interested in, but the cost should be moderate
281
* Exclude all directories that contain the strings "Speedo","/F3/",
282
* "75dpi", "100dpi", "misc" or "bitmap", or don't begin with a "/",
283
* the last of which should exclude font servers.
284
* Also exclude the user specific ".gnome*" directories which
285
* aren't going to contain the system fonts we need.
286
* Hopefully we are left only with Type1 and TrueType directories.
287
* It doesn't matter much if there are extraneous directories, it'll just
288
* cost us a little wasted effort upstream.
289
*/
290
fontdirs = (char**)calloc(nPaths+1, sizeof(char*));
291
if (fontdirs == NULL) {
292
return NULL;
293
}
294
pos = 0;
295
for (i=0; i < nPaths; i++) {
296
if (x11Path[i][0] != '/') {
297
continue;
298
}
299
if (strstr(x11Path[i], "/75dpi") != NULL) {
300
continue;
301
}
302
if (strstr(x11Path[i], "/100dpi") != NULL) {
303
continue;
304
}
305
if (strstr(x11Path[i], "/misc") != NULL) {
306
continue;
307
}
308
if (strstr(x11Path[i], "/Speedo") != NULL) {
309
continue;
310
}
311
if (strstr(x11Path[i], ".gnome") != NULL) {
312
continue;
313
}
314
fontdirs[pos] = strdup(x11Path[i]);
315
slen = strlen(fontdirs[pos]);
316
if (slen > 0 && fontdirs[pos][slen-1] == '/') {
317
fontdirs[pos][slen-1] = '\0'; /* null out trailing "/" */
318
}
319
pos++;
320
}
321
322
XFreeFontPath(x11Path);
323
if (pos == 0) {
324
free(fontdirs);
325
fontdirs = NULL;
326
}
327
return fontdirs;
328
}
329
330
331
#endif /* !HEADLESS */
332
333
#if defined(__linux__)
334
/* from awt_LoadLibrary.c */
335
JNIEXPORT jboolean JNICALL AWTIsHeadless();
336
#endif
337
338
/* This eliminates duplicates, at a non-linear but acceptable cost
339
* since the lists are expected to be reasonably short, and then
340
* deletes references to non-existent directories, and returns
341
* a single path consisting of unique font directories.
342
*/
343
static char* mergePaths(char **p1, char **p2, char **p3, jboolean noType1) {
344
345
int len1=0, len2=0, len3=0, totalLen=0, numDirs=0,
346
currLen, i, j, found, pathLen=0;
347
char **ptr, **fontdirs;
348
char *fontPath = NULL;
349
350
if (p1 != NULL) {
351
ptr = p1;
352
while (*ptr++ != NULL) len1++;
353
}
354
if (p2 != NULL) {
355
ptr = p2;
356
357
while (*ptr++ != NULL) len2++;
358
}
359
if (p3 != NULL) {
360
ptr = p3;
361
while (*ptr++ != NULL) len3++;
362
}
363
totalLen = len1+len2+len3;
364
fontdirs = (char**)calloc(totalLen, sizeof(char*));
365
if (fontdirs == NULL) {
366
return NULL;
367
}
368
369
for (i=0; i < len1; i++) {
370
if (noType1 && strstr(p1[i], "Type1") != NULL) {
371
continue;
372
}
373
fontdirs[numDirs++] = p1[i];
374
}
375
376
currLen = numDirs; /* only compare against previous path dirs */
377
for (i=0; i < len2; i++) {
378
if (noType1 && strstr(p2[i], "Type1") != NULL) {
379
continue;
380
}
381
found = 0;
382
for (j=0; j < currLen; j++) {
383
if (strcmp(fontdirs[j], p2[i]) == 0) {
384
found = 1;
385
break;
386
}
387
}
388
if (!found) {
389
fontdirs[numDirs++] = p2[i];
390
}
391
}
392
393
currLen = numDirs; /* only compare against previous path dirs */
394
for (i=0; i < len3; i++) {
395
if (noType1 && strstr(p3[i], "Type1") != NULL) {
396
continue;
397
}
398
found = 0;
399
for (j=0; j < currLen; j++) {
400
if (strcmp(fontdirs[j], p3[i]) == 0) {
401
found = 1;
402
break;
403
}
404
}
405
if (!found) {
406
fontdirs[numDirs++] = p3[i];
407
}
408
}
409
410
/* Now fontdirs contains unique dirs and numDirs records how many.
411
* What we don't know is if they all exist. On reflection I think
412
* this isn't an issue, so for now I will return all these locations,
413
* converted to one string */
414
for (i=0; i<numDirs; i++) {
415
pathLen += (strlen(fontdirs[i]) + 1);
416
}
417
if (pathLen > 0 && (fontPath = malloc(pathLen))) {
418
*fontPath = '\0';
419
for (i = 0; i<numDirs; i++) {
420
if (i != 0) {
421
strcat(fontPath, ":");
422
}
423
strcat(fontPath, fontdirs[i]);
424
}
425
}
426
free (fontdirs);
427
428
return fontPath;
429
}
430
431
/*
432
* The goal of this function is to find all "system" fonts which
433
* are needed by the JRE to display text in supported locales etc, and
434
* to support APIs which allow users to enumerate all system fonts and use
435
* them from their Java applications.
436
* The preferred mechanism is now using the new "fontconfig" library
437
* This exists on newer versions of Linux and Solaris (S10 and above)
438
* The library is dynamically located. The results are merged with
439
* a set of "known" locations and with the X11 font path, if running in
440
* a local X11 environment.
441
* The hardwired paths are built into the JDK binary so as new font locations
442
* are created on a host plaform for them to be located by the JRE they will
443
* need to be added ito the host's font configuration database, typically
444
* /etc/fonts/local.conf, and to ensure that directory contains a fonts.dir
445
* NB: Fontconfig also depends heavily for performance on the host O/S
446
* maintaining up to date caches.
447
* This is consistent with the requirements of the desktop environments
448
* on these OSes.
449
* This also frees us from X11 APIs as JRE is required to function in
450
* a "headless" mode where there is no Xserver.
451
*/
452
static char *getPlatformFontPathChars(JNIEnv *env, jboolean noType1, jboolean isX11) {
453
454
char **fcdirs = NULL, **x11dirs = NULL, **knowndirs = NULL, *path = NULL;
455
456
/* As of 1.5 we try to use fontconfig on both Solaris and Linux.
457
* If its not available NULL is returned.
458
*/
459
fcdirs = getFontConfigLocations();
460
461
#if defined(__linux__)
462
knowndirs = fullLinuxFontPath;
463
#elif defined(_AIX)
464
knowndirs = fullAixFontPath;
465
#endif
466
/* REMIND: this code requires to be executed when the GraphicsEnvironment
467
* is already initialised. That is always true, but if it were not so,
468
* this code could throw an exception and the fontpath would fail to
469
* be initialised.
470
*/
471
#ifndef HEADLESS
472
if (isX11) { // The following only works in an x11 environment.
473
#if defined(__linux__)
474
/* There's no headless build on linux ... */
475
if (!AWTIsHeadless()) { /* .. so need to call a function to check */
476
#endif
477
/* Using the X11 font path to locate font files is now a fallback
478
* useful only if fontconfig failed, or is incomplete. So we could
479
* remove this code completely and the consequences should be rare
480
* and non-fatal. If this happens, then the calling Java code can
481
* be modified to no longer require that the AWT lock (the X11GE)
482
* be initialised prior to calling this code.
483
*/
484
AWT_LOCK();
485
if (isDisplayLocal(env)) {
486
x11dirs = getX11FontPath();
487
}
488
AWT_UNLOCK();
489
#if defined(__linux__)
490
}
491
#endif
492
}
493
#endif /* !HEADLESS */
494
path = mergePaths(fcdirs, x11dirs, knowndirs, noType1);
495
if (fcdirs != NULL) {
496
char **p = fcdirs;
497
while (*p != NULL) free(*p++);
498
free(fcdirs);
499
}
500
501
if (x11dirs != NULL) {
502
char **p = x11dirs;
503
while (*p != NULL) free(*p++);
504
free(x11dirs);
505
}
506
507
return path;
508
}
509
510
JNIEXPORT jstring JNICALL Java_sun_awt_FcFontManager_getFontPathNative
511
(JNIEnv *env, jobject thiz, jboolean noType1, jboolean isX11) {
512
jstring ret;
513
static char *ptr = NULL; /* retain result across calls */
514
515
if (ptr == NULL) {
516
ptr = getPlatformFontPathChars(env, noType1, isX11);
517
}
518
ret = (*env)->NewStringUTF(env, ptr);
519
return ret;
520
}
521
522
#include <dlfcn.h>
523
524
#include <fontconfig/fontconfig.h>
525
526
527
static void* openFontConfig() {
528
529
char *homeEnv;
530
static char *homeEnvStr = "HOME="; /* must be static */
531
void* libfontconfig = NULL;
532
533
/* Private workaround to not use fontconfig library.
534
* May be useful during testing/debugging
535
*/
536
char *useFC = getenv("USE_J2D_FONTCONFIG");
537
if (useFC != NULL && !strcmp(useFC, "no")) {
538
return NULL;
539
}
540
541
#if defined(_AIX)
542
/* On AIX, fontconfig is not a standard package supported by IBM.
543
* instead it has to be installed from the "AIX Toolbox for Linux Applications"
544
* site http://www-03.ibm.com/systems/power/software/aix/linux/toolbox/alpha.html
545
* and will be installed under /opt/freeware/lib/libfontconfig.a.
546
* Notice that the archive contains the real 32- and 64-bit shared libraries.
547
* We first try to load 'libfontconfig.so' from the default library path in the
548
* case the user has installed a private version of the library and if that
549
* doesn't succeed, we try the version from /opt/freeware/lib/libfontconfig.a
550
*/
551
libfontconfig = dlopen("libfontconfig.so", RTLD_LOCAL|RTLD_LAZY);
552
if (libfontconfig == NULL) {
553
libfontconfig = dlopen("/opt/freeware/lib/libfontconfig.a(libfontconfig.so.1)", RTLD_MEMBER|RTLD_LOCAL|RTLD_LAZY);
554
if (libfontconfig == NULL) {
555
return NULL;
556
}
557
}
558
#else
559
/* 64 bit sparc should pick up the right version from the lib path.
560
* New features may be added to libfontconfig, this is expected to
561
* be compatible with old features, but we may need to start
562
* distinguishing the library version, to know whether to expect
563
* certain symbols - and functionality - to be available.
564
* Also add explicit search for .so.1 in case .so symlink doesn't exist.
565
*/
566
libfontconfig = dlopen(FONTCONFIG_DLL_VERSIONED, RTLD_LOCAL|RTLD_LAZY);
567
if (libfontconfig == NULL) {
568
libfontconfig = dlopen(FONTCONFIG_DLL, RTLD_LOCAL|RTLD_LAZY);
569
if (libfontconfig == NULL) {
570
return NULL;
571
}
572
}
573
#endif
574
575
/* Version 1.0 of libfontconfig crashes if HOME isn't defined in
576
* the environment. This should generally never happen, but we can't
577
* control it, and can't control the version of fontconfig, so iff
578
* its not defined we set it to an empty value which is sufficient
579
* to prevent a crash. I considered unsetting it before exit, but
580
* it doesn't appear to work on Solaris, so I will leave it set.
581
*/
582
homeEnv = getenv("HOME");
583
if (homeEnv == NULL) {
584
putenv(homeEnvStr);
585
}
586
587
return libfontconfig;
588
}
589
590
typedef void* (FcFiniFuncType)();
591
592
static void closeFontConfig(void* libfontconfig, jboolean fcFini) {
593
594
/* NB FcFini is not in (eg) the Solaris 10 version of fontconfig. Its not
595
* clear if this means we are really leaking resources in those cases
596
* but it seems we should call this function when its available.
597
* But since the Swing GTK code may be still accessing the lib, its probably
598
* safest for now to just let this "leak" rather than potentially
599
* concurrently free global data still in use by other code.
600
*/
601
#if 0
602
if (fcFini) { /* release resources */
603
FcFiniFuncType FcFini = (FcFiniFuncType)dlsym(libfontconfig, "FcFini");
604
605
if (FcFini != NULL) {
606
(*FcFini)();
607
}
608
}
609
#endif
610
dlclose(libfontconfig);
611
}
612
613
typedef FcConfig* (*FcInitLoadConfigFuncType)();
614
typedef FcPattern* (*FcPatternBuildFuncType)(FcPattern *orig, ...);
615
typedef FcObjectSet* (*FcObjectSetFuncType)(const char *first, ...);
616
typedef FcFontSet* (*FcFontListFuncType)(FcConfig *config,
617
FcPattern *p,
618
FcObjectSet *os);
619
typedef FcResult (*FcPatternGetBoolFuncType)(const FcPattern *p,
620
const char *object,
621
int n,
622
FcBool *b);
623
typedef FcResult (*FcPatternGetIntegerFuncType)(const FcPattern *p,
624
const char *object,
625
int n,
626
int *i);
627
typedef FcResult (*FcPatternGetStringFuncType)(const FcPattern *p,
628
const char *object,
629
int n,
630
FcChar8 ** s);
631
typedef FcChar8* (*FcStrDirnameFuncType)(const FcChar8 *file);
632
typedef void (*FcPatternDestroyFuncType)(FcPattern *p);
633
typedef void (*FcFontSetDestroyFuncType)(FcFontSet *s);
634
typedef FcPattern* (*FcNameParseFuncType)(const FcChar8 *name);
635
typedef FcBool (*FcPatternAddStringFuncType)(FcPattern *p,
636
const char *object,
637
const FcChar8 *s);
638
typedef void (*FcDefaultSubstituteFuncType)(FcPattern *p);
639
typedef FcBool (*FcConfigSubstituteFuncType)(FcConfig *config,
640
FcPattern *p,
641
FcMatchKind kind);
642
typedef FcPattern* (*FcFontMatchFuncType)(FcConfig *config,
643
FcPattern *p,
644
FcResult *result);
645
typedef FcFontSet* (*FcFontSetCreateFuncType)();
646
typedef FcBool (*FcFontSetAddFuncType)(FcFontSet *s, FcPattern *font);
647
648
typedef FcResult (*FcPatternGetCharSetFuncType)(FcPattern *p,
649
const char *object,
650
int n,
651
FcCharSet **c);
652
typedef FcFontSet* (*FcFontSortFuncType)(FcConfig *config,
653
FcPattern *p,
654
FcBool trim,
655
FcCharSet **csp,
656
FcResult *result);
657
typedef FcCharSet* (*FcCharSetUnionFuncType)(const FcCharSet *a,
658
const FcCharSet *b);
659
typedef FcChar32 (*FcCharSetSubtractCountFuncType)(const FcCharSet *a,
660
const FcCharSet *b);
661
662
typedef int (*FcGetVersionFuncType)();
663
664
typedef FcStrList* (*FcConfigGetCacheDirsFuncType)(FcConfig *config);
665
typedef FcChar8* (*FcStrListNextFuncType)(FcStrList *list);
666
typedef FcChar8* (*FcStrListDoneFuncType)(FcStrList *list);
667
668
static char **getFontConfigLocations() {
669
670
char **fontdirs;
671
int numdirs = 0;
672
FcInitLoadConfigFuncType FcInitLoadConfig;
673
FcPatternBuildFuncType FcPatternBuild;
674
FcObjectSetFuncType FcObjectSetBuild;
675
FcFontListFuncType FcFontList;
676
FcPatternGetStringFuncType FcPatternGetString;
677
FcStrDirnameFuncType FcStrDirname;
678
FcPatternDestroyFuncType FcPatternDestroy;
679
FcFontSetDestroyFuncType FcFontSetDestroy;
680
681
FcConfig *fontconfig;
682
FcPattern *pattern;
683
FcObjectSet *objset;
684
FcFontSet *fontSet;
685
FcStrList *strList;
686
FcChar8 *str;
687
int i, f, found, len=0;
688
char **fontPath;
689
690
void* libfontconfig = openFontConfig();
691
692
if (libfontconfig == NULL) {
693
return NULL;
694
}
695
696
FcPatternBuild =
697
(FcPatternBuildFuncType)dlsym(libfontconfig, "FcPatternBuild");
698
FcObjectSetBuild =
699
(FcObjectSetFuncType)dlsym(libfontconfig, "FcObjectSetBuild");
700
FcFontList =
701
(FcFontListFuncType)dlsym(libfontconfig, "FcFontList");
702
FcPatternGetString =
703
(FcPatternGetStringFuncType)dlsym(libfontconfig, "FcPatternGetString");
704
FcStrDirname =
705
(FcStrDirnameFuncType)dlsym(libfontconfig, "FcStrDirname");
706
FcPatternDestroy =
707
(FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy");
708
FcFontSetDestroy =
709
(FcFontSetDestroyFuncType)dlsym(libfontconfig, "FcFontSetDestroy");
710
711
if (FcPatternBuild == NULL ||
712
FcObjectSetBuild == NULL ||
713
FcPatternGetString == NULL ||
714
FcFontList == NULL ||
715
FcStrDirname == NULL ||
716
FcPatternDestroy == NULL ||
717
FcFontSetDestroy == NULL) { /* problem with the library: return. */
718
closeFontConfig(libfontconfig, JNI_FALSE);
719
return NULL;
720
}
721
722
/* Make calls into the fontconfig library to build a search for
723
* outline fonts, and to get the set of full file paths from the matches.
724
* This set is returned from the call to FcFontList(..)
725
* We allocate an array of char* pointers sufficient to hold all
726
* the matches + 1 extra which ensures there will be a NULL after all
727
* valid entries.
728
* We call FcStrDirname strip the file name from the path, and
729
* check if we have yet seen this directory. If not we add a pointer to
730
* it into our array of char*. Note that FcStrDirname returns newly
731
* allocated storage so we can use this in the return char** value.
732
* Finally we clean up, freeing allocated resources, and return the
733
* array of unique directories.
734
*/
735
pattern = (*FcPatternBuild)(NULL, FC_OUTLINE, FcTypeBool, FcTrue, NULL);
736
objset = (*FcObjectSetBuild)(FC_FILE, NULL);
737
fontSet = (*FcFontList)(NULL, pattern, objset);
738
if (fontSet == NULL) {
739
/* FcFontList() may return NULL if fonts are not installed. */
740
fontdirs = NULL;
741
} else {
742
fontdirs = (char**)calloc(fontSet->nfont+1, sizeof(char*));
743
if (fontdirs == NULL) {
744
(*FcFontSetDestroy)(fontSet);
745
goto cleanup;
746
}
747
for (f=0; f < fontSet->nfont; f++) {
748
FcChar8 *file;
749
FcChar8 *dir;
750
if ((*FcPatternGetString)(fontSet->fonts[f], FC_FILE, 0, &file) ==
751
FcResultMatch) {
752
dir = (*FcStrDirname)(file);
753
found = 0;
754
for (i=0;i<numdirs; i++) {
755
if (strcmp(fontdirs[i], (char*)dir) == 0) {
756
found = 1;
757
break;
758
}
759
}
760
if (!found) {
761
fontdirs[numdirs++] = (char*)dir;
762
} else {
763
free((char*)dir);
764
}
765
}
766
}
767
/* Free fontset if one was returned */
768
(*FcFontSetDestroy)(fontSet);
769
}
770
771
cleanup:
772
/* Free memory and close the ".so" */
773
(*FcPatternDestroy)(pattern);
774
closeFontConfig(libfontconfig, JNI_TRUE);
775
return fontdirs;
776
}
777
778
/* These are copied from sun.awt.SunHints.
779
* Consider initialising them as ints using JNI for more robustness.
780
*/
781
#define TEXT_AA_OFF 1
782
#define TEXT_AA_ON 2
783
#define TEXT_AA_LCD_HRGB 4
784
#define TEXT_AA_LCD_HBGR 5
785
#define TEXT_AA_LCD_VRGB 6
786
#define TEXT_AA_LCD_VBGR 7
787
788
JNIEXPORT jint JNICALL
789
Java_sun_font_FontConfigManager_getFontConfigAASettings
790
(JNIEnv *env, jclass obj, jstring localeStr, jstring fcNameStr) {
791
792
FcNameParseFuncType FcNameParse;
793
FcPatternAddStringFuncType FcPatternAddString;
794
FcConfigSubstituteFuncType FcConfigSubstitute;
795
FcDefaultSubstituteFuncType FcDefaultSubstitute;
796
FcFontMatchFuncType FcFontMatch;
797
FcPatternGetBoolFuncType FcPatternGetBool;
798
FcPatternGetIntegerFuncType FcPatternGetInteger;
799
FcPatternDestroyFuncType FcPatternDestroy;
800
801
FcPattern *pattern, *matchPattern;
802
FcResult result;
803
FcBool antialias = FcFalse;
804
int rgba = 0;
805
const char *locale=NULL, *fcName=NULL;
806
void* libfontconfig;
807
808
if (fcNameStr == NULL || localeStr == NULL) {
809
return -1;
810
}
811
812
fcName = (*env)->GetStringUTFChars(env, fcNameStr, 0);
813
if (fcName == NULL) {
814
return -1;
815
}
816
locale = (*env)->GetStringUTFChars(env, localeStr, 0);
817
818
if ((libfontconfig = openFontConfig()) == NULL) {
819
(*env)->ReleaseStringUTFChars(env, fcNameStr, (const char*)fcName);
820
if (locale) {
821
(*env)->ReleaseStringUTFChars(env, localeStr,(const char*)locale);
822
}
823
return -1;
824
}
825
826
FcNameParse = (FcNameParseFuncType)dlsym(libfontconfig, "FcNameParse");
827
FcPatternAddString =
828
(FcPatternAddStringFuncType)dlsym(libfontconfig, "FcPatternAddString");
829
FcConfigSubstitute =
830
(FcConfigSubstituteFuncType)dlsym(libfontconfig, "FcConfigSubstitute");
831
FcDefaultSubstitute = (FcDefaultSubstituteFuncType)
832
dlsym(libfontconfig, "FcDefaultSubstitute");
833
FcFontMatch = (FcFontMatchFuncType)dlsym(libfontconfig, "FcFontMatch");
834
FcPatternGetBool = (FcPatternGetBoolFuncType)
835
dlsym(libfontconfig, "FcPatternGetBool");
836
FcPatternGetInteger = (FcPatternGetIntegerFuncType)
837
dlsym(libfontconfig, "FcPatternGetInteger");
838
FcPatternDestroy =
839
(FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy");
840
841
if (FcNameParse == NULL ||
842
FcPatternAddString == NULL ||
843
FcConfigSubstitute == NULL ||
844
FcDefaultSubstitute == NULL ||
845
FcFontMatch == NULL ||
846
FcPatternGetBool == NULL ||
847
FcPatternGetInteger == NULL ||
848
FcPatternDestroy == NULL) { /* problem with the library: return. */
849
850
(*env)->ReleaseStringUTFChars(env, fcNameStr, (const char*)fcName);
851
if (locale) {
852
(*env)->ReleaseStringUTFChars(env, localeStr,(const char*)locale);
853
}
854
closeFontConfig(libfontconfig, JNI_FALSE);
855
return -1;
856
}
857
858
859
pattern = (*FcNameParse)((FcChar8 *)fcName);
860
if (locale != NULL) {
861
(*FcPatternAddString)(pattern, FC_LANG, (unsigned char*)locale);
862
}
863
(*FcConfigSubstitute)(NULL, pattern, FcMatchPattern);
864
(*FcDefaultSubstitute)(pattern);
865
matchPattern = (*FcFontMatch)(NULL, pattern, &result);
866
/* Perhaps should call FcFontRenderPrepare() here as some pattern
867
* elements might change as a result of that call, but I'm not seeing
868
* any difference in testing.
869
*/
870
if (matchPattern) {
871
(*FcPatternGetBool)(matchPattern, FC_ANTIALIAS, 0, &antialias);
872
(*FcPatternGetInteger)(matchPattern, FC_RGBA, 0, &rgba);
873
(*FcPatternDestroy)(matchPattern);
874
}
875
(*FcPatternDestroy)(pattern);
876
877
(*env)->ReleaseStringUTFChars(env, fcNameStr, (const char*)fcName);
878
if (locale) {
879
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
880
}
881
closeFontConfig(libfontconfig, JNI_TRUE);
882
883
if (antialias == FcFalse) {
884
return TEXT_AA_OFF;
885
} else if (rgba <= FC_RGBA_UNKNOWN || rgba >= FC_RGBA_NONE) {
886
return TEXT_AA_ON;
887
} else {
888
switch (rgba) {
889
case FC_RGBA_RGB : return TEXT_AA_LCD_HRGB;
890
case FC_RGBA_BGR : return TEXT_AA_LCD_HBGR;
891
case FC_RGBA_VRGB : return TEXT_AA_LCD_VRGB;
892
case FC_RGBA_VBGR : return TEXT_AA_LCD_VBGR;
893
default : return TEXT_AA_LCD_HRGB; // should not get here.
894
}
895
}
896
}
897
898
JNIEXPORT jint JNICALL
899
Java_sun_font_FontConfigManager_getFontConfigVersion
900
(JNIEnv *env, jclass obj) {
901
902
void* libfontconfig;
903
FcGetVersionFuncType FcGetVersion;
904
int version = 0;
905
906
if ((libfontconfig = openFontConfig()) == NULL) {
907
return 0;
908
}
909
910
FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion");
911
912
if (FcGetVersion == NULL) {
913
closeFontConfig(libfontconfig, JNI_FALSE);
914
return 0;
915
}
916
version = (*FcGetVersion)();
917
closeFontConfig(libfontconfig, JNI_FALSE);
918
919
return version;
920
}
921
922
923
JNIEXPORT void JNICALL
924
Java_sun_font_FontConfigManager_getFontConfig
925
(JNIEnv *env, jclass obj, jstring localeStr, jobject fcInfoObj,
926
jobjectArray fcCompFontArray, jboolean includeFallbacks) {
927
928
FcNameParseFuncType FcNameParse;
929
FcPatternAddStringFuncType FcPatternAddString;
930
FcConfigSubstituteFuncType FcConfigSubstitute;
931
FcDefaultSubstituteFuncType FcDefaultSubstitute;
932
FcFontMatchFuncType FcFontMatch;
933
FcPatternGetStringFuncType FcPatternGetString;
934
FcPatternDestroyFuncType FcPatternDestroy;
935
FcPatternGetCharSetFuncType FcPatternGetCharSet;
936
FcFontSortFuncType FcFontSort;
937
FcFontSetDestroyFuncType FcFontSetDestroy;
938
FcCharSetUnionFuncType FcCharSetUnion;
939
FcCharSetSubtractCountFuncType FcCharSetSubtractCount;
940
FcGetVersionFuncType FcGetVersion;
941
FcConfigGetCacheDirsFuncType FcConfigGetCacheDirs;
942
FcStrListNextFuncType FcStrListNext;
943
FcStrListDoneFuncType FcStrListDone;
944
945
int i, arrlen;
946
jobject fcCompFontObj;
947
jstring fcNameStr, jstr;
948
const char *locale, *fcName;
949
FcPattern *pattern;
950
FcResult result;
951
void* libfontconfig;
952
jfieldID fcNameID, fcFirstFontID, fcAllFontsID, fcVersionID, fcCacheDirsID;
953
jfieldID familyNameID, styleNameID, fullNameID, fontFileID;
954
jmethodID fcFontCons;
955
char* debugMinGlyphsStr = getenv("J2D_DEBUG_MIN_GLYPHS");
956
jclass fcInfoClass;
957
jclass fcCompFontClass;
958
jclass fcFontClass;
959
960
CHECK_NULL(fcInfoObj);
961
CHECK_NULL(fcCompFontArray);
962
963
fcInfoClass =
964
(*env)->FindClass(env, "sun/font/FontConfigManager$FontConfigInfo");
965
CHECK_NULL(fcInfoClass);
966
fcCompFontClass =
967
(*env)->FindClass(env, "sun/font/FontConfigManager$FcCompFont");
968
CHECK_NULL(fcCompFontClass);
969
fcFontClass =
970
(*env)->FindClass(env, "sun/font/FontConfigManager$FontConfigFont");
971
CHECK_NULL(fcFontClass);
972
973
974
CHECK_NULL(fcVersionID = (*env)->GetFieldID(env, fcInfoClass, "fcVersion", "I"));
975
CHECK_NULL(fcCacheDirsID = (*env)->GetFieldID(env, fcInfoClass, "cacheDirs",
976
"[Ljava/lang/String;"));
977
CHECK_NULL(fcNameID = (*env)->GetFieldID(env, fcCompFontClass,
978
"fcName", "Ljava/lang/String;"));
979
CHECK_NULL(fcFirstFontID = (*env)->GetFieldID(env, fcCompFontClass, "firstFont",
980
"Lsun/font/FontConfigManager$FontConfigFont;"));
981
CHECK_NULL(fcAllFontsID = (*env)->GetFieldID(env, fcCompFontClass, "allFonts",
982
"[Lsun/font/FontConfigManager$FontConfigFont;"));
983
CHECK_NULL(fcFontCons = (*env)->GetMethodID(env, fcFontClass, "<init>", "()V"));
984
CHECK_NULL(familyNameID = (*env)->GetFieldID(env, fcFontClass,
985
"familyName", "Ljava/lang/String;"));
986
CHECK_NULL(styleNameID = (*env)->GetFieldID(env, fcFontClass,
987
"styleStr", "Ljava/lang/String;"));
988
CHECK_NULL(fullNameID = (*env)->GetFieldID(env, fcFontClass,
989
"fullName", "Ljava/lang/String;"));
990
CHECK_NULL(fontFileID = (*env)->GetFieldID(env, fcFontClass,
991
"fontFile", "Ljava/lang/String;"));
992
993
if ((libfontconfig = openFontConfig()) == NULL) {
994
return;
995
}
996
997
FcNameParse = (FcNameParseFuncType)dlsym(libfontconfig, "FcNameParse");
998
FcPatternAddString =
999
(FcPatternAddStringFuncType)dlsym(libfontconfig, "FcPatternAddString");
1000
FcConfigSubstitute =
1001
(FcConfigSubstituteFuncType)dlsym(libfontconfig, "FcConfigSubstitute");
1002
FcDefaultSubstitute = (FcDefaultSubstituteFuncType)
1003
dlsym(libfontconfig, "FcDefaultSubstitute");
1004
FcFontMatch = (FcFontMatchFuncType)dlsym(libfontconfig, "FcFontMatch");
1005
FcPatternGetString =
1006
(FcPatternGetStringFuncType)dlsym(libfontconfig, "FcPatternGetString");
1007
FcPatternDestroy =
1008
(FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy");
1009
FcPatternGetCharSet =
1010
(FcPatternGetCharSetFuncType)dlsym(libfontconfig,
1011
"FcPatternGetCharSet");
1012
FcFontSort =
1013
(FcFontSortFuncType)dlsym(libfontconfig, "FcFontSort");
1014
FcFontSetDestroy =
1015
(FcFontSetDestroyFuncType)dlsym(libfontconfig, "FcFontSetDestroy");
1016
FcCharSetUnion =
1017
(FcCharSetUnionFuncType)dlsym(libfontconfig, "FcCharSetUnion");
1018
FcCharSetSubtractCount =
1019
(FcCharSetSubtractCountFuncType)dlsym(libfontconfig,
1020
"FcCharSetSubtractCount");
1021
FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion");
1022
1023
if (FcNameParse == NULL ||
1024
FcPatternAddString == NULL ||
1025
FcConfigSubstitute == NULL ||
1026
FcDefaultSubstitute == NULL ||
1027
FcFontMatch == NULL ||
1028
FcPatternGetString == NULL ||
1029
FcPatternDestroy == NULL ||
1030
FcPatternGetCharSet == NULL ||
1031
FcFontSetDestroy == NULL ||
1032
FcCharSetUnion == NULL ||
1033
FcGetVersion == NULL ||
1034
FcCharSetSubtractCount == NULL) {/* problem with the library: return.*/
1035
closeFontConfig(libfontconfig, JNI_FALSE);
1036
return;
1037
}
1038
1039
(*env)->SetIntField(env, fcInfoObj, fcVersionID, (*FcGetVersion)());
1040
1041
/* Optionally get the cache dir locations. This isn't
1042
* available until v 2.4.x, but this is OK since on those later versions
1043
* we can check the time stamps on the cache dirs to see if we
1044
* are out of date. There are a couple of assumptions here. First
1045
* that the time stamp on the directory changes when the contents are
1046
* updated. Secondly that the locations don't change. The latter is
1047
* most likely if a new version of fontconfig is installed, but we also
1048
* invalidate the cache if we detect that. Arguably even that is "rare",
1049
* and most likely is tied to an OS upgrade which gets a new file anyway.
1050
*/
1051
FcConfigGetCacheDirs =
1052
(FcConfigGetCacheDirsFuncType)dlsym(libfontconfig,
1053
"FcConfigGetCacheDirs");
1054
FcStrListNext =
1055
(FcStrListNextFuncType)dlsym(libfontconfig, "FcStrListNext");
1056
FcStrListDone =
1057
(FcStrListDoneFuncType)dlsym(libfontconfig, "FcStrListDone");
1058
if (FcStrListNext != NULL && FcStrListDone != NULL &&
1059
FcConfigGetCacheDirs != NULL) {
1060
1061
FcStrList* cacheDirs;
1062
FcChar8* cacheDir;
1063
int cnt = 0;
1064
jobject cacheDirArray =
1065
(*env)->GetObjectField(env, fcInfoObj, fcCacheDirsID);
1066
int max = (*env)->GetArrayLength(env, cacheDirArray);
1067
1068
cacheDirs = (*FcConfigGetCacheDirs)(NULL);
1069
if (cacheDirs != NULL) {
1070
while ((cnt < max) && (cacheDir = (*FcStrListNext)(cacheDirs))) {
1071
jstr = (*env)->NewStringUTF(env, (const char*)cacheDir);
1072
JNU_CHECK_EXCEPTION(env);
1073
1074
(*env)->SetObjectArrayElement(env, cacheDirArray, cnt++, jstr);
1075
(*env)->DeleteLocalRef(env, jstr);
1076
}
1077
(*FcStrListDone)(cacheDirs);
1078
}
1079
}
1080
1081
locale = (*env)->GetStringUTFChars(env, localeStr, 0);
1082
if (locale == NULL) {
1083
(*env)->ExceptionClear(env);
1084
JNU_ThrowOutOfMemoryError(env, "Could not create locale");
1085
return;
1086
}
1087
1088
arrlen = (*env)->GetArrayLength(env, fcCompFontArray);
1089
for (i=0; i<arrlen; i++) {
1090
FcFontSet* fontset;
1091
int fn, j, fontCount, nfonts;
1092
unsigned int minGlyphs;
1093
FcChar8 **family, **styleStr, **fullname, **file;
1094
jarray fcFontArr = NULL;
1095
FcCharSet *unionCharset = NULL;
1096
1097
fcCompFontObj = (*env)->GetObjectArrayElement(env, fcCompFontArray, i);
1098
fcNameStr =
1099
(jstring)((*env)->GetObjectField(env, fcCompFontObj, fcNameID));
1100
fcName = (*env)->GetStringUTFChars(env, fcNameStr, 0);
1101
if (fcName == NULL) {
1102
(*env)->DeleteLocalRef(env, fcCompFontObj);
1103
(*env)->DeleteLocalRef(env, fcNameStr);
1104
continue;
1105
}
1106
pattern = (*FcNameParse)((FcChar8 *)fcName);
1107
(*env)->ReleaseStringUTFChars(env, fcNameStr, (const char*)fcName);
1108
(*env)->DeleteLocalRef(env, fcNameStr);
1109
if (pattern == NULL) {
1110
closeFontConfig(libfontconfig, JNI_FALSE);
1111
if (locale) {
1112
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1113
}
1114
return;
1115
}
1116
1117
/* locale may not usually be necessary as fontconfig appears to apply
1118
* this anyway based on the user's environment. However we want
1119
* to use the value of the JDK startup locale so this should take
1120
* care of it.
1121
*/
1122
if (locale != NULL) {
1123
(*FcPatternAddString)(pattern, FC_LANG, (unsigned char*)locale);
1124
}
1125
(*FcConfigSubstitute)(NULL, pattern, FcMatchPattern);
1126
(*FcDefaultSubstitute)(pattern);
1127
fontset = (*FcFontSort)(NULL, pattern, FcTrue, NULL, &result);
1128
if (fontset == NULL) {
1129
(*FcPatternDestroy)(pattern);
1130
closeFontConfig(libfontconfig, JNI_FALSE);
1131
if (locale) {
1132
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1133
}
1134
return;
1135
}
1136
1137
/* fontconfig returned us "nfonts". If we are just getting the
1138
* first font, we set nfont to zero. Otherwise we use "nfonts".
1139
* Next create separate C arrrays of length nfonts for family file etc.
1140
* Inspect the returned fonts and the ones we like (adds enough glyphs)
1141
* are added to the arrays and we increment 'fontCount'.
1142
*/
1143
nfonts = fontset->nfont;
1144
family = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
1145
styleStr = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
1146
fullname = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
1147
file = (FcChar8**)calloc(nfonts, sizeof(FcChar8*));
1148
if (family == NULL || styleStr == NULL ||
1149
fullname == NULL || file == NULL) {
1150
if (family != NULL) {
1151
free(family);
1152
}
1153
if (styleStr != NULL) {
1154
free(styleStr);
1155
}
1156
if (fullname != NULL) {
1157
free(fullname);
1158
}
1159
if (file != NULL) {
1160
free(file);
1161
}
1162
(*FcPatternDestroy)(pattern);
1163
(*FcFontSetDestroy)(fontset);
1164
closeFontConfig(libfontconfig, JNI_FALSE);
1165
if (locale) {
1166
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1167
}
1168
return;
1169
}
1170
fontCount = 0;
1171
minGlyphs = 20;
1172
if (debugMinGlyphsStr != NULL) {
1173
int val = minGlyphs;
1174
sscanf(debugMinGlyphsStr, "%5d", &val);
1175
if (val >= 0 && val <= 65536) {
1176
minGlyphs = val;
1177
}
1178
}
1179
1180
for (j=0; j<nfonts; j++) {
1181
FcPattern *fontPattern = fontset->fonts[j];
1182
FcChar8 *fontformat;
1183
FcCharSet *charset = NULL;
1184
1185
fontformat = NULL;
1186
(*FcPatternGetString)(fontPattern, FC_FONTFORMAT, 0, &fontformat);
1187
/* We only want TrueType fonts but some Linuxes still depend
1188
* on Type 1 fonts for some Locale support, so we'll allow
1189
* them there.
1190
*/
1191
if (fontformat != NULL
1192
&& (strcmp((char*)fontformat, "TrueType") != 0)
1193
#if defined(__linux__) || defined(_AIX)
1194
&& (strcmp((char*)fontformat, "Type 1") != 0)
1195
&& (strcmp((char*)fontformat, "CFF") != 0)
1196
#endif
1197
) {
1198
continue;
1199
}
1200
result = (*FcPatternGetCharSet)(fontPattern,
1201
FC_CHARSET, 0, &charset);
1202
if (result != FcResultMatch) {
1203
free(family);
1204
free(fullname);
1205
free(styleStr);
1206
free(file);
1207
(*FcPatternDestroy)(pattern);
1208
(*FcFontSetDestroy)(fontset);
1209
closeFontConfig(libfontconfig, JNI_FALSE);
1210
if (locale) {
1211
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1212
}
1213
return;
1214
}
1215
1216
/* We don't want 20 or 30 fonts, so once we hit 10 fonts,
1217
* then require that they really be adding value. Too many
1218
* adversely affects load time for minimal value-add.
1219
* This is still likely far more than we've had in the past.
1220
*/
1221
if (j==10) {
1222
minGlyphs = 50;
1223
}
1224
if (unionCharset == NULL) {
1225
unionCharset = charset;
1226
} else {
1227
if ((*FcCharSetSubtractCount)(charset, unionCharset)
1228
> minGlyphs) {
1229
unionCharset = (* FcCharSetUnion)(unionCharset, charset);
1230
} else {
1231
continue;
1232
}
1233
}
1234
1235
fontCount++; // found a font we will use.
1236
(*FcPatternGetString)(fontPattern, FC_FILE, 0, &file[j]);
1237
(*FcPatternGetString)(fontPattern, FC_FAMILY, 0, &family[j]);
1238
(*FcPatternGetString)(fontPattern, FC_STYLE, 0, &styleStr[j]);
1239
(*FcPatternGetString)(fontPattern, FC_FULLNAME, 0, &fullname[j]);
1240
if (!includeFallbacks) {
1241
break;
1242
}
1243
if (fontCount == 254) {
1244
break; // CompositeFont will only use up to 254 slots from here.
1245
}
1246
}
1247
1248
/* Once we get here 'fontCount' is the number of returned fonts
1249
* we actually want to use, so we create 'fcFontArr' of that length.
1250
* The non-null entries of "family[]" etc are those fonts.
1251
* Then loop again over all nfonts adding just those non-null ones
1252
* to 'fcFontArr'. If its null (we didn't want the font)
1253
* then we don't enter the main body.
1254
* So we should never get more than 'fontCount' entries.
1255
*/
1256
if (includeFallbacks) {
1257
fcFontArr =
1258
(*env)->NewObjectArray(env, fontCount, fcFontClass, NULL);
1259
if (IS_NULL(fcFontArr)) {
1260
free(family);
1261
free(fullname);
1262
free(styleStr);
1263
free(file);
1264
(*FcPatternDestroy)(pattern);
1265
(*FcFontSetDestroy)(fontset);
1266
closeFontConfig(libfontconfig, JNI_FALSE);
1267
if (locale) {
1268
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1269
}
1270
return;
1271
}
1272
(*env)->SetObjectField(env,fcCompFontObj, fcAllFontsID, fcFontArr);
1273
}
1274
fn=0;
1275
1276
for (j=0;j<nfonts;j++) {
1277
if (family[j] != NULL) {
1278
jobject fcFont =
1279
(*env)->NewObject(env, fcFontClass, fcFontCons);
1280
if (IS_NULL(fcFont)) break;
1281
jstr = (*env)->NewStringUTF(env, (const char*)family[j]);
1282
if (IS_NULL(jstr)) break;
1283
(*env)->SetObjectField(env, fcFont, familyNameID, jstr);
1284
(*env)->DeleteLocalRef(env, jstr);
1285
if (file[j] != NULL) {
1286
jstr = (*env)->NewStringUTF(env, (const char*)file[j]);
1287
if (IS_NULL(jstr)) break;
1288
(*env)->SetObjectField(env, fcFont, fontFileID, jstr);
1289
(*env)->DeleteLocalRef(env, jstr);
1290
}
1291
if (styleStr[j] != NULL) {
1292
jstr = (*env)->NewStringUTF(env, (const char*)styleStr[j]);
1293
if (IS_NULL(jstr)) break;
1294
(*env)->SetObjectField(env, fcFont, styleNameID, jstr);
1295
(*env)->DeleteLocalRef(env, jstr);
1296
}
1297
if (fullname[j] != NULL) {
1298
jstr = (*env)->NewStringUTF(env, (const char*)fullname[j]);
1299
if (IS_NULL(jstr)) break;
1300
(*env)->SetObjectField(env, fcFont, fullNameID, jstr);
1301
(*env)->DeleteLocalRef(env, jstr);
1302
}
1303
if (fn==0) {
1304
(*env)->SetObjectField(env, fcCompFontObj,
1305
fcFirstFontID, fcFont);
1306
}
1307
if (includeFallbacks) {
1308
(*env)->SetObjectArrayElement(env, fcFontArr, fn++,fcFont);
1309
} else {
1310
(*env)->DeleteLocalRef(env, fcFont);
1311
break;
1312
}
1313
(*env)->DeleteLocalRef(env, fcFont);
1314
}
1315
}
1316
if (includeFallbacks) {
1317
(*env)->DeleteLocalRef(env, fcFontArr);
1318
}
1319
(*env)->DeleteLocalRef(env, fcCompFontObj);
1320
(*FcFontSetDestroy)(fontset);
1321
(*FcPatternDestroy)(pattern);
1322
free(family);
1323
free(styleStr);
1324
free(fullname);
1325
free(file);
1326
}
1327
1328
/* release resources and close the ".so" */
1329
1330
if (locale) {
1331
(*env)->ReleaseStringUTFChars(env, localeStr, (const char*)locale);
1332
}
1333
closeFontConfig(libfontconfig, JNI_TRUE);
1334
}
1335
1336