/* * Copyright 1998-2006 Sun Microsystems, Inc. All Rights Reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, * CA 95054 USA or visit www.sun.com if you need additional information or * have any questions. */ #ifdef __linux__ #include #endif /* __linux__ */ #include #include #include #include #include #include #include #include #ifdef __solaris__ #include #endif #include #include #include #ifndef HEADLESS #include #include #else /* locks ought to be included from awt.h */ #define AWT_LOCK() #define AWT_UNLOCK() #endif /* !HEADLESS */ #if defined(__linux__) && !defined(MAP_FAILED) #define MAP_FAILED ((caddr_t)-1) #endif #ifndef HEADLESS extern Display *awt_display; #endif /* !HEADLESS */ #define MAXFDIRS 512 /* Max number of directories that contain fonts */ #ifndef __linux__ /* * This can be set in the makefile to "/usr/X11" if so desired. */ #ifndef OPENWINHOMELIB #define OPENWINHOMELIB "/usr/openwin/lib/" #endif /* This is all known Solaris X11 directories on Solaris 8, 9 and 10. * It is ordered to give precedence to TrueType directories. * It is needed if fontconfig is not installed or configured properly. */ static char *fullSolarisFontPath[] = { OPENWINHOMELIB "X11/fonts/TrueType", OPENWINHOMELIB "locale/euro_fonts/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_2/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_5/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_7/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_8/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_9/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_13/X11/fonts/TrueType", OPENWINHOMELIB "locale/iso_8859_15/X11/fonts/TrueType", OPENWINHOMELIB "locale/ar/X11/fonts/TrueType", OPENWINHOMELIB "locale/hi_IN.UTF-8/X11/fonts/TrueType", OPENWINHOMELIB "locale/ja/X11/fonts/TT", OPENWINHOMELIB "locale/ko/X11/fonts/TrueType", OPENWINHOMELIB "locale/ko.UTF-8/X11/fonts/TrueType", OPENWINHOMELIB "locale/KOI8-R/X11/fonts/TrueType", OPENWINHOMELIB "locale/ru.ansi-1251/X11/fonts/TrueType", OPENWINHOMELIB "locale/th_TH/X11/fonts/TrueType", OPENWINHOMELIB "locale/zh_TW/X11/fonts/TrueType", OPENWINHOMELIB "locale/zh_TW.BIG5/X11/fonts/TT", OPENWINHOMELIB "locale/zh_HK.BIG5HK/X11/fonts/TT", OPENWINHOMELIB "locale/zh_CN.GB18030/X11/fonts/TrueType", OPENWINHOMELIB "locale/zh/X11/fonts/TrueType", OPENWINHOMELIB "locale/zh.GBK/X11/fonts/TrueType", OPENWINHOMELIB "X11/fonts/Type1", OPENWINHOMELIB "X11/fonts/Type1/sun", OPENWINHOMELIB "X11/fonts/Type1/sun/outline", OPENWINHOMELIB "locale/iso_8859_2/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_4/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_5/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_7/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_8/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_9/X11/fonts/Type1", OPENWINHOMELIB "locale/iso_8859_13/X11/fonts/Type1", OPENWINHOMELIB "locale/ar/X11/fonts/Type1", NULL, /* terminates the list */ }; #else /* __linux */ /* All the known interesting locations we have discovered on * various flavors of Linux */ static char *fullLinuxFontPath[] = { "/usr/X11R6/lib/X11/fonts/TrueType", /* RH 7.1+ */ "/usr/X11R6/lib/X11/fonts/truetype", /* SuSE */ "/usr/X11R6/lib/X11/fonts/tt", "/usr/X11R6/lib/X11/fonts/TTF", "/usr/X11R6/lib/X11/fonts/OTF", /* RH 9.0 (but empty!) */ "/usr/share/fonts/ja/TrueType", /* RH 7.2+ */ "/usr/share/fonts/truetype", "/usr/share/fonts/ko/TrueType", /* RH 9.0 */ "/usr/share/fonts/zh_CN/TrueType", /* RH 9.0 */ "/usr/share/fonts/zh_TW/TrueType", /* RH 9.0 */ "/var/lib/defoma/x-ttcidfont-conf.d/dirs/TrueType", /* Debian */ "/usr/X11R6/lib/X11/fonts/Type1", "/usr/share/fonts/default/Type1", /* RH 9.0 */ NULL, /* terminates the list */ }; #endif static char **getFontConfigLocations(); typedef struct { const char *name[MAXFDIRS]; int num; } fDirRecord, *fDirRecordPtr; #ifndef HEADLESS /* * Returns True if display is local, False of it's remote. */ jboolean isDisplayLocal(JNIEnv *env) { static jboolean isLocal = False; static jboolean isLocalSet = False; jboolean ret; if (isLocalSet) { return isLocal; } isLocal = JNU_CallStaticMethodByName(env, NULL, "sun/awt/X11GraphicsEnvironment", "_isDisplayLocal", "()Z").z; isLocalSet = True; return isLocal; } static void AddFontsToX11FontPath ( fDirRecord *fDirP ) { char *onePath; int index, nPaths; int origNumPaths, length; int origIndex; int totalDirCount; char **origFontPath; char **tempFontPath; int doNotAppend; int *appendDirList; char **newFontPath; int err, compareLength; char fontDirPath[512]; int dirFile; doNotAppend = 0; if ( fDirP->num == 0 ) return; appendDirList = malloc ( fDirP->num * sizeof ( int )); if ( appendDirList == NULL ) { return; /* if it fails we cannot do much */ } origFontPath = XGetFontPath ( awt_display, &nPaths ); totalDirCount = nPaths; origNumPaths = nPaths; tempFontPath = origFontPath; for (index = 0; index < fDirP->num; index++ ) { doNotAppend = 0; tempFontPath = origFontPath; for ( origIndex = 0; origIndex < nPaths; origIndex++ ) { onePath = *tempFontPath; compareLength = strlen ( onePath ); if ( onePath[compareLength -1] == '/' ) compareLength--; /* there is a slash at the end of every solaris X11 font path name */ if ( strncmp ( onePath, fDirP->name[index], compareLength ) == 0 ) { doNotAppend = 1; break; } tempFontPath++; } appendDirList[index] = 0; if ( doNotAppend == 0 ) { strcpy ( fontDirPath, fDirP->name[index] ); strcat ( fontDirPath, "/fonts.dir" ); dirFile = open ( fontDirPath, O_RDONLY, 0 ); if ( dirFile == -1 ) { doNotAppend = 1; } else { close ( dirFile ); totalDirCount++; appendDirList[index] = 1; } } } /* if no changes are required do not bother to do a setfontpath */ if ( totalDirCount == nPaths ) { free ( ( void *) appendDirList ); XFreeFontPath ( origFontPath ); return; } newFontPath = malloc ( totalDirCount * sizeof ( char **) ); /* if it fails free things and get out */ if ( newFontPath == NULL ) { free ( ( void *) appendDirList ); XFreeFontPath ( origFontPath ); return; } for ( origIndex = 0; origIndex < nPaths; origIndex++ ) { onePath = origFontPath[origIndex]; newFontPath[origIndex] = onePath; } /* now add the other font paths */ for (index = 0; index < fDirP->num; index++ ) { if ( appendDirList[index] == 1 ) { /* printf ( "Appending %s\n", fDirP->name[index] ); */ onePath = malloc ( ( strlen (fDirP->name[index]) + 2 )* sizeof( char ) ); strcpy ( onePath, fDirP->name[index] ); strcat ( onePath, "/" ); newFontPath[nPaths++] = onePath; /* printf ( "The path to be appended is %s\n", onePath ); */ } } /* printf ( "The dir count = %d\n", totalDirCount ); */ free ( ( void *) appendDirList ); XSetFontPath ( awt_display, newFontPath, totalDirCount ); for ( index = origNumPaths; index < totalDirCount; index++ ) { free( newFontPath[index] ); } free ( (void *) newFontPath ); XFreeFontPath ( origFontPath ); return; } #endif /* !HEADLESS */ #ifndef HEADLESS static char **getX11FontPath () { char **x11Path, **fontdirs; int i, pos, slen, nPaths, numDirs; x11Path = XGetFontPath (awt_display, &nPaths); /* This isn't ever going to be perfect: the font path may contain * much we aren't interested in, but the cost should be moderate * Exclude all directories that contain the strings "Speedo","/F3/", * "75dpi", "100dpi", "misc" or "bitmap", or don't begin with a "/", * the last of which should exclude font servers. * Also exclude the user specific ".gnome*" directories which * aren't going to contain the system fonts we need. * Hopefully we are left only with Type1 and TrueType directories. * It doesn't matter much if there are extraneous directories, it'll just * cost us a little wasted effort upstream. */ fontdirs = (char**)calloc(nPaths+1, sizeof(char*)); pos = 0; for (i=0; i < nPaths; i++) { if (x11Path[i][0] != '/') { continue; } if (strstr(x11Path[i], "/75dpi") != NULL) { continue; } if (strstr(x11Path[i], "/100dpi") != NULL) { continue; } if (strstr(x11Path[i], "/misc") != NULL) { continue; } if (strstr(x11Path[i], "/Speedo") != NULL) { continue; } if (strstr(x11Path[i], ".gnome") != NULL) { continue; } #ifdef __solaris__ if (strstr(x11Path[i], "/F3/") != NULL) { continue; } if (strstr(x11Path[i], "bitmap") != NULL) { continue; } #endif fontdirs[pos] = strdup(x11Path[i]); slen = strlen(fontdirs[pos]); if (slen > 0 && fontdirs[pos][slen-1] == '/') { fontdirs[pos][slen-1] = '\0'; /* null out trailing "/" */ } pos++; } XFreeFontPath(x11Path); if (pos == 0) { free(fontdirs); fontdirs = NULL; } return fontdirs; } #endif /* !HEADLESS */ #ifdef __linux__ /* from awt_LoadLibrary.c */ JNIEXPORT jboolean JNICALL AWTIsHeadless(); #endif /* This eliminates duplicates, at a non-linear but acceptable cost * since the lists are expected to be reasonably short, and then * deletes references to non-existent directories, and returns * a single path consisting of unique font directories. */ static char* mergePaths(char **p1, char **p2, char **p3, jboolean noType1) { int len1=0, len2=0, len3=0, totalLen=0, numDirs=0, currLen, i, j, found, pathLen=0; char **ptr, **fontdirs; char *fontPath = NULL; if (p1 != NULL) { ptr = p1; while (*ptr++ != NULL) len1++; } if (p2 != NULL) { ptr = p2; while (*ptr++ != NULL) len2++; } if (p3 != NULL) { ptr = p3; while (*ptr++ != NULL) len3++; } totalLen = len1+len2+len3; fontdirs = (char**)calloc(totalLen, sizeof(char*)); for (i=0; i < len1; i++) { if (noType1 && strstr(p1[i], "Type1") != NULL) { continue; } fontdirs[numDirs++] = p1[i]; } currLen = numDirs; /* only compare against previous path dirs */ for (i=0; i < len2; i++) { if (noType1 && strstr(p2[i], "Type1") != NULL) { continue; } found = 0; for (j=0; j < currLen; j++) { if (strcmp(fontdirs[j], p2[i]) == 0) { found = 1; break; } } if (!found) { fontdirs[numDirs++] = p2[i]; } } currLen = numDirs; /* only compare against previous path dirs */ for (i=0; i < len3; i++) { if (noType1 && strstr(p3[i], "Type1") != NULL) { continue; } found = 0; for (j=0; j < currLen; j++) { if (strcmp(fontdirs[j], p3[i]) == 0) { found = 1; break; } } if (!found) { fontdirs[numDirs++] = p3[i]; } } /* Now fontdirs contains unique dirs and numDirs records how many. * What we don't know is if they all exist. On reflection I think * this isn't an issue, so for now I will return all these locations, * converted to one string */ for (i=0; i 0 && (fontPath = malloc(pathLen))) { *fontPath = '\0'; for (i = 0; iNewStringUTF(env, ptr); return ret; } /* * In general setting the font path in a remote display situation is * problematic. But for Solaris->Solaris the paths needed by the JRE should * also be available to the server, although we have no way to check this * for sure. * So set the font path if we think its safe to do so: * All Solaris X servers at least back to 2.6 and up to Solaris 10 * define the exact same vendor string. * The version number for Solaris 2.6 is 3600, for 2.7 is 3610 and * for Solaris 8 6410 * we want to set the font path only for 2.8 and onwards. Earlier releases * are unlikely to have the right fonts and can't install "all locales" * as needed to be sure. Also Solaris 8 is the earliest release supported * by 1.5. */ #ifndef HEADLESS static int isSunXServer() { #ifdef __solaris__ return (strcmp("Sun Microsystems, Inc.", ServerVendor(awt_display)) == 0 && VendorRelease(awt_display) >= 6410); #else return 0; #endif /* __solaris__ */ } /* Avoid re-doing work for every call to setNativeFontPath */ static int doSetFontPath = -1; static int shouldSetXFontPath(JNIEnv *env) { if (doSetFontPath == -1) { doSetFontPath = awt_display != NULL && (isDisplayLocal(env) || isSunXServer()); } return doSetFontPath; } #endif /* !HEADLESS */ JNIEXPORT void JNICALL Java_sun_font_FontManager_setNativeFontPath (JNIEnv *env, jclass obj, jstring theString) { #ifdef HEADLESS return; #else fDirRecord fDir; const char *theChars; if (awt_display == NULL) { return; } AWT_LOCK(); if (shouldSetXFontPath(env)) { theChars = (*env)->GetStringUTFChars (env, theString, 0); fDir.num = 1; fDir.name[0] = theChars; /* printf ("Registering the font path here %s \n", theChars ); */ AddFontsToX11FontPath ( &fDir ); if (theChars) { (*env)->ReleaseStringUTFChars (env, theString, (const char*)theChars); } } AWT_UNLOCK(); #endif } /* This isn't yet used on unix, the implementation is added since shared * code calls this method in preparation for future use. */ /* Obtain all the fontname -> filename mappings. * This is called once and the results returned to Java code which can * use it for lookups to reduce or avoid the need to search font files. */ JNIEXPORT void JNICALL Java_sun_font_FontManager_populateFontFileNameMap (JNIEnv *env, jclass obj, jobject fontToFileMap, jobject fontToFamilyMap, jobject familyToFontListMap, jobject locale) { return; } #include #ifndef __linux__ /* i.e. is solaris */ #include #endif #include "fontconfig.h" static void* openFontConfig() { char *homeEnv; static char *homeEnvStr = "HOME="; /* must be static */ void* libfontconfig = NULL; #ifdef __solaris__ #define SYSINFOBUFSZ 8 char sysinfobuf[SYSINFOBUFSZ]; #endif /* Private workaround to not use fontconfig library. * May be useful during testing/debugging */ char *useFC = getenv("USE_J2D_FONTCONFIG"); if (useFC != NULL && !strcmp(useFC, "no")) { return NULL; } #ifdef __solaris__ /* fontconfig is likely not properly configured on S8/S9 - skip it, * although allow user to override this behaviour with an env. variable * ie if USE_J2D_FONTCONFIG=yes then we skip this test. * NB "4" is the length of a string which matches our patterns. */ if (useFC == NULL || strcmp(useFC, "yes")) { if (sysinfo(SI_RELEASE, sysinfobuf, SYSINFOBUFSZ) == 4) { if ((!strcmp(sysinfobuf, "5.8") || !strcmp(sysinfobuf, "5.9"))) { return NULL; } } } #endif /* 64 bit sparc should pick up the right version from the lib path. * New features may be added to libfontconfig, this is expected to * be compatible with old features, but we may need to start * distinguishing the library version, to know whether to expect * certain symbols - and functionality - to be available. * Also add explicit search for .so.1 in case .so symlink doesn't exist. */ libfontconfig = dlopen("libfontconfig.so.1", RTLD_LOCAL|RTLD_LAZY); if (libfontconfig == NULL) { libfontconfig = dlopen("libfontconfig.so", RTLD_LOCAL|RTLD_LAZY); if (libfontconfig == NULL) { return NULL; } } /* Version 1.0 of libfontconfig crashes if HOME isn't defined in * the environment. This should generally never happen, but we can't * control it, and can't control the version of fontconfig, so iff * its not defined we set it to an empty value which is sufficient * to prevent a crash. I considered unsetting it before exit, but * it doesn't appear to work on Solaris, so I will leave it set. */ homeEnv = getenv("HOME"); if (homeEnv == NULL) { putenv(homeEnvStr); } return libfontconfig; } typedef void* (FcFiniFuncType)(); static void closeFontConfig(void* libfontconfig, jboolean fcFini) { /* NB FcFini is not in (eg) the Solaris 10 version of fontconfig. Its not * clear if this means we are really leaking resources in those cases * but it seems we should call this function when its available. * But since the Swing GTK code may be still accessing the lib, its probably * safest for now to just let this "leak" rather than potentially * concurrently free global data still in use by other code. */ #if 0 if (fcFini) { /* release resources */ FcFiniFuncType FcFini = (FcFiniFuncType)dlsym(libfontconfig, "FcFini"); if (FcFini != NULL) { (*FcFini)(); } } #endif dlclose(libfontconfig); } typedef FcConfig* (*FcInitLoadConfigFuncType)(); typedef FcPattern* (*FcPatternBuildFuncType)(FcPattern *orig, ...); typedef FcObjectSet* (*FcObjectSetFuncType)(const char *first, ...); typedef FcFontSet* (*FcFontListFuncType)(FcConfig *config, FcPattern *p, FcObjectSet *os); typedef FcResult (*FcPatternGetBoolFuncType)(const FcPattern *p, const char *object, int n, FcBool *b); typedef FcResult (*FcPatternGetIntegerFuncType)(const FcPattern *p, const char *object, int n, int *i); typedef FcResult (*FcPatternGetStringFuncType)(const FcPattern *p, const char *object, int n, FcChar8 ** s); typedef FcChar8* (*FcStrDirnameFuncType)(const FcChar8 *file); typedef void (*FcPatternDestroyFuncType)(FcPattern *p); typedef void (*FcFontSetDestroyFuncType)(FcFontSet *s); typedef FcPattern* (*FcNameParseFuncType)(const FcChar8 *name); typedef FcBool (*FcPatternAddStringFuncType)(FcPattern *p, const char *object, const FcChar8 *s); typedef void (*FcDefaultSubstituteFuncType)(FcPattern *p); typedef FcBool (*FcConfigSubstituteFuncType)(FcConfig *config, FcPattern *p, FcMatchKind kind); typedef FcPattern* (*FcFontMatchFuncType)(FcConfig *config, FcPattern *p, FcResult *result); typedef FcFontSet* (*FcFontSetCreateFuncType)(); typedef FcBool (*FcFontSetAddFuncType)(FcFontSet *s, FcPattern *font); typedef FcResult (*FcPatternGetCharSetFuncType)(FcPattern *p, const char *object, int n, FcCharSet **c); typedef FcFontSet* (*FcFontSortFuncType)(FcConfig *config, FcPattern *p, FcBool trim, FcCharSet **csp, FcResult *result); typedef FcCharSet* (*FcCharSetUnionFuncType)(const FcCharSet *a, const FcCharSet *b); typedef FcChar32 (*FcCharSetSubtractCountFuncType)(const FcCharSet *a, const FcCharSet *b); typedef int (*FcGetVersionFuncType)(); typedef FcStrList* (*FcConfigGetCacheDirsFuncType)(FcConfig *config); typedef FcChar8* (*FcStrListNextFuncType)(FcStrList *list); typedef FcChar8* (*FcStrListDoneFuncType)(FcStrList *list); static char **getFontConfigLocations() { char **fontdirs; int numdirs = 0; FcInitLoadConfigFuncType FcInitLoadConfig; FcPatternBuildFuncType FcPatternBuild; FcObjectSetFuncType FcObjectSetBuild; FcFontListFuncType FcFontList; FcPatternGetStringFuncType FcPatternGetString; FcStrDirnameFuncType FcStrDirname; FcPatternDestroyFuncType FcPatternDestroy; FcFontSetDestroyFuncType FcFontSetDestroy; FcConfig *fontconfig; FcPattern *pattern; FcObjectSet *objset; FcFontSet *fontSet; FcStrList *strList; FcChar8 *str; int i, f, found, len=0; char **fontPath; void* libfontconfig = openFontConfig(); if (libfontconfig == NULL) { return NULL; } FcPatternBuild = (FcPatternBuildFuncType)dlsym(libfontconfig, "FcPatternBuild"); FcObjectSetBuild = (FcObjectSetFuncType)dlsym(libfontconfig, "FcObjectSetBuild"); FcFontList = (FcFontListFuncType)dlsym(libfontconfig, "FcFontList"); FcPatternGetString = (FcPatternGetStringFuncType)dlsym(libfontconfig, "FcPatternGetString"); FcStrDirname = (FcStrDirnameFuncType)dlsym(libfontconfig, "FcStrDirname"); FcPatternDestroy = (FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy"); FcFontSetDestroy = (FcFontSetDestroyFuncType)dlsym(libfontconfig, "FcFontSetDestroy"); if (FcPatternBuild == NULL || FcObjectSetBuild == NULL || FcPatternGetString == NULL || FcFontList == NULL || FcStrDirname == NULL || FcPatternDestroy == NULL || FcFontSetDestroy == NULL) { /* problem with the library: return. */ closeFontConfig(libfontconfig, JNI_FALSE); return NULL; } /* Make calls into the fontconfig library to build a search for * outline fonts, and to get the set of full file paths from the matches. * This set is returned from the call to FcFontList(..) * We allocate an array of char* pointers sufficient to hold all * the matches + 1 extra which ensures there will be a NULL after all * valid entries. * We call FcStrDirname strip the file name from the path, and * check if we have yet seen this directory. If not we add a pointer to * it into our array of char*. Note that FcStrDirname returns newly * allocated storage so we can use this in the return char** value. * Finally we clean up, freeing allocated resources, and return the * array of unique directories. */ pattern = (*FcPatternBuild)(NULL, FC_OUTLINE, FcTypeBool, FcTrue, NULL); objset = (*FcObjectSetBuild)(FC_FILE, NULL); fontSet = (*FcFontList)(NULL, pattern, objset); fontdirs = (char**)calloc(fontSet->nfont+1, sizeof(char*)); for (f=0; f < fontSet->nfont; f++) { FcChar8 *file; FcChar8 *dir; if ((*FcPatternGetString)(fontSet->fonts[f], FC_FILE, 0, &file) == FcResultMatch) { dir = (*FcStrDirname)(file); found = 0; for (i=0;iGetStringUTFChars(env, fcNameStr, 0); if (fcName == NULL) { return -1; } locale = (*env)->GetStringUTFChars(env, localeStr, 0); if ((libfontconfig = openFontConfig()) == NULL) { (*env)->ReleaseStringUTFChars (env, fcNameStr, (const char*)fcName); if (locale) { (*env)->ReleaseStringUTFChars (env, localeStr,(const char*)locale); } return -1; } FcNameParse = (FcNameParseFuncType)dlsym(libfontconfig, "FcNameParse"); FcPatternAddString = (FcPatternAddStringFuncType)dlsym(libfontconfig, "FcPatternAddString"); FcConfigSubstitute = (FcConfigSubstituteFuncType)dlsym(libfontconfig, "FcConfigSubstitute"); FcDefaultSubstitute = (FcDefaultSubstituteFuncType) dlsym(libfontconfig, "FcDefaultSubstitute"); FcFontMatch = (FcFontMatchFuncType)dlsym(libfontconfig, "FcFontMatch"); FcPatternGetBool = (FcPatternGetBoolFuncType) dlsym(libfontconfig, "FcPatternGetBool"); FcPatternGetInteger = (FcPatternGetIntegerFuncType) dlsym(libfontconfig, "FcPatternGetInteger"); FcPatternDestroy = (FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy"); if (FcNameParse == NULL || FcPatternAddString == NULL || FcConfigSubstitute == NULL || FcDefaultSubstitute == NULL || FcFontMatch == NULL || FcPatternGetBool == NULL || FcPatternGetInteger == NULL || FcPatternDestroy == NULL) { /* problem with the library: return. */ (*env)->ReleaseStringUTFChars (env, fcNameStr, (const char*)fcName); if (locale) { (*env)->ReleaseStringUTFChars (env, localeStr,(const char*)locale); } closeFontConfig(libfontconfig, JNI_FALSE); return -1; } pattern = (*FcNameParse)((FcChar8 *)fcName); if (locale != NULL) { (*FcPatternAddString)(pattern, FC_LANG, (unsigned char*)locale); } (*FcConfigSubstitute)(NULL, pattern, FcMatchPattern); (*FcDefaultSubstitute)(pattern); matchPattern = (*FcFontMatch)(NULL, pattern, &result); /* Perhaps should call FcFontRenderPrepare() here as some pattern * elements might change as a result of that call, but I'm not seeing * any difference in testing. */ if (matchPattern) { (*FcPatternGetBool)(matchPattern, FC_ANTIALIAS, 0, &antialias); (*FcPatternGetInteger)(matchPattern, FC_RGBA, 0, &rgba); (*FcPatternDestroy)(matchPattern); } (*FcPatternDestroy)(pattern); (*env)->ReleaseStringUTFChars (env, fcNameStr, (const char*)fcName); if (locale) { (*env)->ReleaseStringUTFChars (env, localeStr, (const char*)locale); } closeFontConfig(libfontconfig, JNI_TRUE); if (antialias == FcFalse) { return TEXT_AA_OFF; } else if (rgba <= FC_RGBA_UNKNOWN || rgba >= FC_RGBA_NONE) { return TEXT_AA_ON; } else { switch (rgba) { case FC_RGBA_RGB : return TEXT_AA_LCD_HRGB; case FC_RGBA_BGR : return TEXT_AA_LCD_HBGR; case FC_RGBA_VRGB : return TEXT_AA_LCD_VRGB; case FC_RGBA_VBGR : return TEXT_AA_LCD_VBGR; default : return TEXT_AA_LCD_HRGB; // should not get here. } } } JNIEXPORT jint JNICALL Java_sun_font_FontManager_getFontConfigVersion (JNIEnv *env, jclass obj) { void* libfontconfig; FcGetVersionFuncType FcGetVersion; int version = 0; if ((libfontconfig = openFontConfig()) == NULL) { return 0; } FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion"); if (FcGetVersion == NULL) { closeFontConfig(libfontconfig, JNI_FALSE); return 0; } version = (*FcGetVersion)(); closeFontConfig(libfontconfig, JNI_FALSE); return version; } JNIEXPORT void JNICALL Java_sun_font_FontManager_getFontConfig (JNIEnv *env, jclass obj, jstring localeStr, jobject fcInfoObj, jobjectArray fcCompFontArray, jboolean includeFallbacks) { FcNameParseFuncType FcNameParse; FcPatternAddStringFuncType FcPatternAddString; FcConfigSubstituteFuncType FcConfigSubstitute; FcDefaultSubstituteFuncType FcDefaultSubstitute; FcFontMatchFuncType FcFontMatch; FcPatternGetStringFuncType FcPatternGetString; FcPatternDestroyFuncType FcPatternDestroy; FcPatternGetCharSetFuncType FcPatternGetCharSet; FcFontSortFuncType FcFontSort; FcFontSetDestroyFuncType FcFontSetDestroy; FcCharSetUnionFuncType FcCharSetUnion; FcCharSetSubtractCountFuncType FcCharSetSubtractCount; FcGetVersionFuncType FcGetVersion; FcConfigGetCacheDirsFuncType FcConfigGetCacheDirs; FcStrListNextFuncType FcStrListNext; FcStrListDoneFuncType FcStrListDone; int i, arrlen; jobject fcCompFontObj; jstring fcNameStr, jstr; const char *locale, *fcName; FcPattern *pattern; FcResult result; void* libfontconfig; jfieldID fcNameID, fcFirstFontID, fcAllFontsID, fcVersionID, fcCacheDirsID; jfieldID familyNameID, styleNameID, fullNameID, fontFileID; jmethodID fcFontCons; char* debugMinGlyphsStr = getenv("J2D_DEBUG_MIN_GLYPHS"); jclass fcInfoClass = (*env)->FindClass(env, "sun/font/FontManager$FontConfigInfo"); jclass fcCompFontClass = (*env)->FindClass(env, "sun/font/FontManager$FcCompFont"); jclass fcFontClass = (*env)->FindClass(env, "sun/font/FontManager$FontConfigFont"); if (fcInfoObj == NULL || fcCompFontArray == NULL || fcInfoClass == NULL || fcCompFontClass == NULL || fcFontClass == NULL) { return; } fcVersionID = (*env)->GetFieldID(env, fcInfoClass, "fcVersion", "I"); fcCacheDirsID = (*env)->GetFieldID(env, fcInfoClass, "cacheDirs", "[Ljava/lang/String;"); fcNameID = (*env)->GetFieldID(env, fcCompFontClass, "fcName", "Ljava/lang/String;"); fcFirstFontID = (*env)->GetFieldID(env, fcCompFontClass, "firstFont", "Lsun/font/FontManager$FontConfigFont;"); fcAllFontsID = (*env)->GetFieldID(env, fcCompFontClass, "allFonts", "[Lsun/font/FontManager$FontConfigFont;"); fcFontCons = (*env)->GetMethodID(env, fcFontClass, "", "()V"); familyNameID = (*env)->GetFieldID(env, fcFontClass, "familyName", "Ljava/lang/String;"); styleNameID = (*env)->GetFieldID(env, fcFontClass, "styleStr", "Ljava/lang/String;"); fullNameID = (*env)->GetFieldID(env, fcFontClass, "fullName", "Ljava/lang/String;"); fontFileID = (*env)->GetFieldID(env, fcFontClass, "fontFile", "Ljava/lang/String;"); if (fcVersionID == NULL || fcCacheDirsID == NULL || fcNameID == NULL || fcFirstFontID == NULL || fcAllFontsID == NULL || fcFontCons == NULL || familyNameID == NULL || styleNameID == NULL || fullNameID == NULL || fontFileID == NULL) { return; } if ((libfontconfig = openFontConfig()) == NULL) { return; } FcNameParse = (FcNameParseFuncType)dlsym(libfontconfig, "FcNameParse"); FcPatternAddString = (FcPatternAddStringFuncType)dlsym(libfontconfig, "FcPatternAddString"); FcConfigSubstitute = (FcConfigSubstituteFuncType)dlsym(libfontconfig, "FcConfigSubstitute"); FcDefaultSubstitute = (FcDefaultSubstituteFuncType) dlsym(libfontconfig, "FcDefaultSubstitute"); FcFontMatch = (FcFontMatchFuncType)dlsym(libfontconfig, "FcFontMatch"); FcPatternGetString = (FcPatternGetStringFuncType)dlsym(libfontconfig, "FcPatternGetString"); FcPatternDestroy = (FcPatternDestroyFuncType)dlsym(libfontconfig, "FcPatternDestroy"); FcPatternGetCharSet = (FcPatternGetCharSetFuncType)dlsym(libfontconfig, "FcPatternGetCharSet"); FcFontSort = (FcFontSortFuncType)dlsym(libfontconfig, "FcFontSort"); FcFontSetDestroy = (FcFontSetDestroyFuncType)dlsym(libfontconfig, "FcFontSetDestroy"); FcCharSetUnion = (FcCharSetUnionFuncType)dlsym(libfontconfig, "FcCharSetUnion"); FcCharSetSubtractCount = (FcCharSetSubtractCountFuncType)dlsym(libfontconfig, "FcCharSetSubtractCount"); FcGetVersion = (FcGetVersionFuncType)dlsym(libfontconfig, "FcGetVersion"); if (FcNameParse == NULL || FcPatternAddString == NULL || FcConfigSubstitute == NULL || FcDefaultSubstitute == NULL || FcFontMatch == NULL || FcPatternGetString == NULL || FcPatternDestroy == NULL || FcPatternGetCharSet == NULL || FcFontSetDestroy == NULL || FcCharSetUnion == NULL || FcGetVersion == NULL || FcCharSetSubtractCount == NULL) {/* problem with the library: return.*/ closeFontConfig(libfontconfig, JNI_FALSE); return; } (*env)->SetIntField(env, fcInfoObj, fcVersionID, (*FcGetVersion)()); /* Optionally get the cache dir locations. This isn't * available until v 2.4.x, but this is OK since on those later versions * we can check the time stamps on the cache dirs to see if we * are out of date. There are a couple of assumptions here. First * that the time stamp on the directory changes when the contents are * updated. Secondly that the locations don't change. The latter is * most likely if a new version of fontconfig is installed, but we also * invalidate the cache if we detect that. Arguably even that is "rare", * and most likely is tied to an OS upgrade which gets a new file anyway. */ FcConfigGetCacheDirs = (FcConfigGetCacheDirsFuncType)dlsym(libfontconfig, "FcConfigGetCacheDirs"); FcStrListNext = (FcStrListNextFuncType)dlsym(libfontconfig, "FcStrListNext"); FcStrListDone = (FcStrListDoneFuncType)dlsym(libfontconfig, "FcStrListDone"); if (FcStrListNext != NULL && FcStrListDone != NULL && FcConfigGetCacheDirs != NULL) { FcStrList* cacheDirs; FcChar8* cacheDir; int cnt = 0; jobject cacheDirArray = (*env)->GetObjectField(env, fcInfoObj, fcCacheDirsID); int max = (*env)->GetArrayLength(env, cacheDirArray); cacheDirs = (*FcConfigGetCacheDirs)(NULL); if (cacheDirs != NULL) { while ((cnt < max) && (cacheDir = (*FcStrListNext)(cacheDirs))) { jstr = (*env)->NewStringUTF(env, (const char*)cacheDir); (*env)->SetObjectArrayElement(env, cacheDirArray, cnt++, jstr); } (*FcStrListDone)(cacheDirs); } } locale = (*env)->GetStringUTFChars(env, localeStr, 0); arrlen = (*env)->GetArrayLength(env, fcCompFontArray); for (i=0; iGetObjectArrayElement(env, fcCompFontArray, i); fcNameStr = (jstring)((*env)->GetObjectField(env, fcCompFontObj, fcNameID)); fcName = (*env)->GetStringUTFChars(env, fcNameStr, 0); if (fcName == NULL) { continue; } pattern = (*FcNameParse)((FcChar8 *)fcName); if (pattern == NULL) { closeFontConfig(libfontconfig, JNI_FALSE); return; } /* locale may not usually be necessary as fontconfig appears to apply * this anyway based on the user's environment. However we want * to use the value of the JDK startup locale so this should take * care of it. */ if (locale != NULL) { (*FcPatternAddString)(pattern, FC_LANG, (unsigned char*)locale); } (*FcConfigSubstitute)(NULL, pattern, FcMatchPattern); (*FcDefaultSubstitute)(pattern); fontset = (*FcFontSort)(NULL, pattern, FcTrue, NULL, &result); if (fontset == NULL) { closeFontConfig(libfontconfig, JNI_FALSE); return; } /* fontconfig returned us "nfonts". If we are just getting the * first font, we set nfont to zero. Otherwise we use "nfonts". * Next create separate C arrrays of length nfonts for family file etc. * Inspect the returned fonts and the ones we like (adds enough glyphs) * are added to the arrays and we increment 'fontCount'. */ if (includeFallbacks) { nfonts = fontset->nfont; } else { nfonts = 1; } family = (FcChar8**)calloc(nfonts, sizeof(FcChar8*)); styleStr = (FcChar8**)calloc(nfonts, sizeof(FcChar8*)); fullname = (FcChar8**)calloc(nfonts, sizeof(FcChar8*)); file = (FcChar8**)calloc(nfonts, sizeof(FcChar8*)); if (family == NULL || styleStr == NULL || fullname == NULL || file == NULL) { closeFontConfig(libfontconfig, JNI_FALSE); return; } fontCount = 0; minGlyphs = 20; if (debugMinGlyphsStr != NULL) { int val = minGlyphs; sscanf(debugMinGlyphsStr, "%5d", &val); if (val >= 0 && val <= 65536) { minGlyphs = val; } } for (j=0; jfonts[j]; FcChar8 *fontformat; FcCharSet *unionCharset = NULL, *charset; fontformat = NULL; (*FcPatternGetString)(fontPattern, FC_FONTFORMAT, 0, &fontformat); if (fontformat != NULL && strcmp((char*)fontformat, "TrueType") != 0) { continue; } result = (*FcPatternGetCharSet)(fontPattern, FC_CHARSET, 0, &charset); if (result != FcResultMatch) { closeFontConfig(libfontconfig, JNI_FALSE); return; } /* We don't want 20 or 30 fonts, so once we hit 10 fonts, * then require that they really be adding value. Too many * adversely affects load time for minimal value-add. * This is still likely far more than we've had in the past. */ if (nfonts==10) { minGlyphs = 50; } if (unionCharset == NULL) { unionCharset = charset; } else { if ((*FcCharSetSubtractCount)(charset, unionCharset) > minGlyphs) { unionCharset = (* FcCharSetUnion)(unionCharset, charset); } else { continue; } } fontCount++; // found a font we will use. (*FcPatternGetString)(fontPattern, FC_FILE, 0, &file[j]); (*FcPatternGetString)(fontPattern, FC_FAMILY, 0, &family[j]); (*FcPatternGetString)(fontPattern, FC_STYLE, 0, &styleStr[j]); (*FcPatternGetString)(fontPattern, FC_FULLNAME, 0, &fullname[j]); } /* Once we get here 'fontCount' is the number of returned fonts * we actually want to use, so we create 'fcFontArr' of that length. * The non-null entries of "family[]" etc are those fonts. * Then loop again over all nfonts adding just those non-null ones * to 'fcFontArr'. If its null (we didn't want the font) * then we don't enter the main body. * So we should never get more than 'fontCount' entries. */ if (includeFallbacks) { fcFontArr = (*env)->NewObjectArray(env, fontCount, fcFontClass, NULL); (*env)->SetObjectField(env,fcCompFontObj, fcAllFontsID, fcFontArr); } fn=0; for (j=0;jNewObject(env, fcFontClass, fcFontCons); jstr = (*env)->NewStringUTF(env, (const char*)family[j]); (*env)->SetObjectField(env, fcFont, familyNameID, jstr); if (file[j] != NULL) { jstr = (*env)->NewStringUTF(env, (const char*)file[j]); (*env)->SetObjectField(env, fcFont, fontFileID, jstr); } if (styleStr[j] != NULL) { jstr = (*env)->NewStringUTF(env, (const char*)styleStr[j]); (*env)->SetObjectField(env, fcFont, styleNameID, jstr); } if (fullname[j] != NULL) { jstr = (*env)->NewStringUTF(env, (const char*)fullname[j]); (*env)->SetObjectField(env, fcFont, fullNameID, jstr); } if (fn==0) { (*env)->SetObjectField(env, fcCompFontObj, fcFirstFontID, fcFont); } if (includeFallbacks) { (*env)->SetObjectArrayElement(env, fcFontArr, fn++,fcFont); } } } (*env)->ReleaseStringUTFChars (env, fcNameStr, (const char*)fcName); (*FcFontSetDestroy)(fontset); (*FcPatternDestroy)(pattern); free(family); free(styleStr); free(fullname); free(file); } /* release resources and close the ".so" */ if (locale) { (*env)->ReleaseStringUTFChars (env, localeStr, (const char*)locale); } closeFontConfig(libfontconfig, JNI_TRUE); }