aboutsummaryrefslogtreecommitdiff
path: root/src/solaris/classes
diff options
context:
space:
mode:
authorprr <none@none>2008-06-12 13:17:33 -0700
committerprr <none@none>2008-06-12 13:17:33 -0700
commit8b084e76f5f0bd07a8336c96b9e06fe23064d788 (patch)
tree95d1c79aa7f0346c12e06d67a5ba85d7c5c1324b /src/solaris/classes
parent1bc01bd6ec65719bc4285eddf4d5424b1c37c6ee (diff)
downloadjdk8u_jdk-8b084e76f5f0bd07a8336c96b9e06fe23064d788.tar.gz
6378099: RFE: Use libfontconfig to create/synthesise a fontconfig.properties
Reviewed-by: tdv, igor
Diffstat (limited to 'src/solaris/classes')
-rw-r--r--src/solaris/classes/sun/awt/X11GraphicsEnvironment.java50
-rw-r--r--src/solaris/classes/sun/font/FcFontConfiguration.java517
2 files changed, 566 insertions, 1 deletions
diff --git a/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java b/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
index b531fa739b..4c52788f82 100644
--- a/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
+++ b/src/solaris/classes/sun/awt/X11GraphicsEnvironment.java
@@ -44,6 +44,7 @@ import java.util.*;
import java.util.logging.*;
import sun.awt.motif.MFontConfiguration;
+import sun.font.FcFontConfiguration;
import sun.font.Font2D;
import sun.font.FontManager;
import sun.font.NativeFont;
@@ -350,6 +351,14 @@ public class X11GraphicsEnvironment
* only to get called for these fonts.
*/
public String getFileNameFromPlatformName(String platName) {
+
+ /* If the FontConfig file doesn't use xlfds, or its
+ * FcFontConfiguration, this may be already a file name.
+ */
+ if (platName.startsWith("/")) {
+ return platName;
+ }
+
String fileName = null;
String fontID = specificFontIDForName(platName);
@@ -905,12 +914,50 @@ public class X11GraphicsEnvironment
// Implements SunGraphicsEnvironment.createFontConfiguration.
protected FontConfiguration createFontConfiguration() {
- return new MFontConfiguration(this);
+
+ /* The logic here decides whether to use a preconfigured
+ * fontconfig.properties file, or synthesise one using platform APIs.
+ * On Solaris (as opposed to OpenSolaris) we try to use the
+ * pre-configured ones, but if the files it specifies are missing
+ * we fail-safe to synthesising one. This might happen if Solaris
+ * changes its fonts.
+ * For OpenSolaris I don't expect us to ever create fontconfig files,
+ * so it will always synthesise. Note that if we misidentify
+ * OpenSolaris as Solaris, then the test for the presence of
+ * Solaris-only font files will correct this.
+ * For Linux we require an exact match of distro and version to
+ * use the preconfigured file, and also that it points to
+ * existent fonts.
+ * If synthesising fails, we fall back to any preconfigured file
+ * and do the best we can. For the commercial JDK this will be
+ * fine as it includes the Lucida fonts. OpenJDK should not hit
+ * this as the synthesis should always work on its platforms.
+ */
+ FontConfiguration mFontConfig = new MFontConfiguration(this);
+ if (isOpenSolaris ||
+ (isLinux &&
+ (!mFontConfig.foundOsSpecificFile() ||
+ !mFontConfig.fontFilesArePresent()) ||
+ (isSolaris && !mFontConfig.fontFilesArePresent()))) {
+ FcFontConfiguration fcFontConfig =
+ new FcFontConfiguration(this);
+ if (fcFontConfig.init()) {
+ return fcFontConfig;
+ }
+ }
+ mFontConfig.init();
+ return mFontConfig;
}
public FontConfiguration
createFontConfiguration(boolean preferLocaleFonts,
boolean preferPropFonts) {
+ FontConfiguration config = getFontConfiguration();
+ if (config instanceof FcFontConfiguration) {
+ // Doesn't need to implement the alternate support.
+ return config;
+ }
+
return new MFontConfiguration(this,
preferLocaleFonts, preferPropFonts);
}
@@ -921,6 +968,7 @@ public class X11GraphicsEnvironment
* for this platform.
*/
public String getDefaultFontFaceName() {
+
return null;
}
diff --git a/src/solaris/classes/sun/font/FcFontConfiguration.java b/src/solaris/classes/sun/font/FcFontConfiguration.java
new file mode 100644
index 0000000000..fe2e5dbf83
--- /dev/null
+++ b/src/solaris/classes/sun/font/FcFontConfiguration.java
@@ -0,0 +1,517 @@
+/*
+ * Copyright 2008 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
+ * 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.
+ */
+
+package sun.font;
+
+import java.awt.Font;
+import java.io.File;
+import java.io.FileInputStream;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+import java.nio.charset.Charset;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.logging.Logger;
+import java.util.Properties;
+import java.util.Scanner;
+import sun.awt.FontConfiguration;
+import sun.awt.FontDescriptor;
+import sun.awt.SunToolkit;
+import sun.font.CompositeFontDescriptor;
+import sun.font.FontManager;
+import sun.font.FontManager.FontConfigInfo;
+import sun.font.FontManager.FcCompFont;
+import sun.font.FontManager.FontConfigFont;
+import sun.java2d.SunGraphicsEnvironment;
+
+public class FcFontConfiguration extends FontConfiguration {
+
+ /** Version of the cache file format understood by this code.
+ * Its part of the file name so that we can rev this at
+ * any time, even in a minor JDK update.
+ * It is stored as the value of the "version" property.
+ * This is distinct from the version of "libfontconfig" that generated
+ * the cached results, and which is the "fcversion" property in the file.
+ * {@code FontConfiguration.getVersion()} also returns a version string,
+ * and has meant the version of the fontconfiguration.properties file
+ * that was read. Since this class doesn't use such files, then what
+ * that really means is whether the methods on this class return
+ * values that are compatible with the classes that do directly read
+ * from such files. It is a compatible subset of version "1".
+ */
+ private static final String fileVersion = "1";
+ private String fcInfoFileName = null;
+
+ private FcCompFont[] fcCompFonts = null;
+
+ public FcFontConfiguration(SunGraphicsEnvironment environment) {
+ super(environment);
+ init();
+ }
+
+ /* This isn't called but is needed to satisfy super-class contract. */
+ public FcFontConfiguration(SunGraphicsEnvironment environment,
+ boolean preferLocaleFonts,
+ boolean preferPropFonts) {
+ super(environment, preferLocaleFonts, preferPropFonts);
+ init();
+ }
+
+ @Override
+ public synchronized boolean init() {
+ if (fcCompFonts != null) {
+ return true;
+ }
+
+ readFcInfo();
+ if (fcCompFonts == null) {
+ fcCompFonts = FontManager.loadFontConfig();
+ if (fcCompFonts != null) {
+ try {
+ writeFcInfo();
+ } catch (Exception e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger =
+ Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Exception writing fcInfo " + e);
+ }
+ }
+ } else if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Failed to get info from libfontconfig");
+ }
+ } else {
+ FontManager.populateFontConfig(fcCompFonts);
+ }
+
+ if (fcCompFonts == null) {
+ return false; // couldn't load fontconfig.
+ }
+
+ // NB already in a privileged block from SGE
+ String javaHome = System.getProperty("java.home");
+ if (javaHome == null) {
+ throw new Error("java.home property not set");
+ }
+ String javaLib = javaHome + File.separator + "lib";
+ getInstalledFallbackFonts(javaLib);
+
+ return true;
+ }
+
+ @Override
+ public String getFallbackFamilyName(String fontName,
+ String defaultFallback) {
+ // maintain compatibility with old font.properties files, which either
+ // had aliases for TimesRoman & Co. or defined mappings for them.
+ String compatibilityName = getCompatibilityFamilyName(fontName);
+ if (compatibilityName != null) {
+ return compatibilityName;
+ }
+ return defaultFallback;
+ }
+
+ @Override
+ protected String
+ getFaceNameFromComponentFontName(String componentFontName) {
+ return null;
+ }
+
+ @Override
+ protected String
+ getFileNameFromComponentFontName(String componentFontName) {
+ return null;
+ }
+
+ @Override
+ public String getFileNameFromPlatformName(String platformName) {
+ /* Platform name is the file name, but rather than returning
+ * the arg, return null*/
+ return null;
+ }
+
+ @Override
+ protected Charset getDefaultFontCharset(String fontName) {
+ return Charset.forName("ISO8859_1");
+ }
+
+ @Override
+ protected String getEncoding(String awtFontName,
+ String characterSubsetName) {
+ return "default";
+ }
+
+ @Override
+ protected void initReorderMap() {
+ reorderMap = new HashMap();
+ }
+
+ @Override
+ public FontDescriptor[] getFontDescriptors(String fontName, int style) {
+ throw new InternalError("Not implemented");
+ }
+
+ @Override
+ public int getNumberCoreFonts() {
+ return 1;
+ }
+
+ @Override
+ public String[] getPlatformFontNames() {
+ HashSet<String> nameSet = new HashSet<String>();
+ FcCompFont[] fcCompFonts = FontManager.loadFontConfig();
+ for (int i=0; i<fcCompFonts.length; i++) {
+ for (int j=0; j<fcCompFonts[i].allFonts.length; j++) {
+ nameSet.add(fcCompFonts[i].allFonts[j].fontFile);
+ }
+ }
+ return nameSet.toArray(new String[0]);
+ }
+
+ @Override
+ public String getExtraFontPath() {
+ return null;
+ }
+
+ @Override
+ public boolean needToSearchForFile(String fileName) {
+ return false;
+ }
+
+ private FontConfigFont[] getFcFontList(FcCompFont[] fcFonts,
+ String fontname, int style) {
+
+ if (fontname.equals("dialog")) {
+ fontname = "sansserif";
+ } else if (fontname.equals("dialoginput")) {
+ fontname = "monospaced";
+ }
+ for (int i=0; i<fcFonts.length; i++) {
+ if (fontname.equals(fcFonts[i].jdkName) &&
+ style == fcFonts[i].style) {
+ return fcFonts[i].allFonts;
+ }
+ }
+ return fcFonts[0].allFonts;
+ }
+
+ @Override
+ public CompositeFontDescriptor[] get2DCompositeFontInfo() {
+
+ FcCompFont[] fcCompFonts = FontManager.loadFontConfig();
+
+ CompositeFontDescriptor[] result =
+ new CompositeFontDescriptor[NUM_FONTS * NUM_STYLES];
+
+ for (int fontIndex = 0; fontIndex < NUM_FONTS; fontIndex++) {
+ String fontName = publicFontNames[fontIndex];
+
+ for (int styleIndex = 0; styleIndex < NUM_STYLES; styleIndex++) {
+
+ String faceName = fontName + "." + styleNames[styleIndex];
+ FontConfigFont[] fcFonts =
+ getFcFontList(fcCompFonts,
+ fontNames[fontIndex], styleIndex);
+
+ int numFonts = fcFonts.length;
+ // fall back fonts listed in the lib/fonts/fallback directory
+ if (installedFallbackFontFiles != null) {
+ numFonts += installedFallbackFontFiles.length;
+ }
+
+ String[] fileNames = new String[numFonts];
+
+ int index;
+ for (index = 0; index < fcFonts.length; index++) {
+ fileNames[index] = fcFonts[index].fontFile;
+ }
+
+ if (installedFallbackFontFiles != null) {
+ System.arraycopy(fileNames, index,
+ installedFallbackFontFiles,
+ 0, installedFallbackFontFiles.length);
+ }
+
+ result[fontIndex * NUM_STYLES + styleIndex]
+ = new CompositeFontDescriptor(
+ faceName,
+ 1,
+ null,
+ fileNames,
+ null, null);
+ }
+ }
+ return result;
+ }
+
+ /**
+ * Gets the OS version string from a Linux release-specific file.
+ */
+ private String getVersionString(File f){
+ try {
+ Scanner sc = new Scanner(f);
+ return sc.findInLine("(\\d)+((\\.)(\\d)+)*");
+ }
+ catch (Exception e){
+ }
+ return null;
+ }
+
+ /**
+ * Sets the OS name and version from environment information.
+ */
+ @Override
+ protected void setOsNameAndVersion() {
+
+ super.setOsNameAndVersion();
+
+ if (!osName.equals("Linux")) {
+ return;
+ }
+ try {
+ File f;
+ if ((f = new File("/etc/lsb-release")).canRead()) {
+ /* Ubuntu and (perhaps others) use only lsb-release.
+ * Syntax and encoding is compatible with java properties.
+ * For Ubuntu the ID is "Ubuntu".
+ */
+ Properties props = new Properties();
+ props.load(new FileInputStream(f));
+ osName = props.getProperty("DISTRIB_ID");
+ osVersion = props.getProperty("DISTRIB_RELEASE");
+ } else if ((f = new File("/etc/redhat-release")).canRead()) {
+ osName = "RedHat";
+ osVersion = getVersionString(f);
+ } else if ((f = new File("/etc/SuSE-release")).canRead()) {
+ osName = "SuSE";
+ osVersion = getVersionString(f);
+ } else if ((f = new File("/etc/turbolinux-release")).canRead()) {
+ osName = "Turbo";
+ osVersion = getVersionString(f);
+ } else if ((f = new File("/etc/fedora-release")).canRead()) {
+ osName = "Fedora";
+ osVersion = getVersionString(f);
+ } else if ((f = new File("/etc/sun-release")).canRead()) {
+ osName = "Sun";
+ osVersion = getVersionString(f);
+ }
+ } catch (Exception e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Exception identifying Linux distro.");
+ }
+ }
+ }
+
+ private File getFcInfoFile() {
+ if (fcInfoFileName == null) {
+ // NB need security permissions to get true IP address, and
+ // we should have those as the whole initialisation is in a
+ // doPrivileged block. But in this case no exception is thrown,
+ // and it returns the loop back address, and so we end up with
+ // "localhost"
+ String hostname;
+ try {
+ hostname = InetAddress.getLocalHost().getHostName();
+ } catch (UnknownHostException e) {
+ hostname = "localhost";
+ }
+ String userDir = System.getProperty("user.home");
+ String version = System.getProperty("java.version");
+ String fs = File.separator;
+ String dir = userDir+fs+".java"+fs+"fonts"+fs+version;
+ String lang = SunToolkit.getStartupLocale().getLanguage();
+ String name = "fcinfo-"+fileVersion+"-"+hostname+"-"+
+ osName+"-"+osVersion+"-"+lang+".properties";
+ fcInfoFileName = dir+fs+name;
+ }
+ return new File(fcInfoFileName);
+ }
+
+ private void writeFcInfo() {
+ Properties props = new Properties();
+ props.setProperty("version", fileVersion);
+ FontConfigInfo fcInfo = FontManager.getFontConfigInfo();
+ props.setProperty("fcversion", Integer.toString(fcInfo.fcVersion));
+ if (fcInfo.cacheDirs != null) {
+ for (int i=0;i<fcInfo.cacheDirs.length;i++) {
+ if (fcInfo.cacheDirs[i] != null) {
+ props.setProperty("cachedir."+i, fcInfo.cacheDirs[i]);
+ }
+ }
+ }
+ for (int i=0; i<fcCompFonts.length; i++) {
+ FcCompFont fci = fcCompFonts[i];
+ String styleKey = fci.jdkName+"."+fci.style;
+ props.setProperty(styleKey+".length",
+ Integer.toString(fci.allFonts.length));
+ for (int j=0; j<fci.allFonts.length; j++) {
+ props.setProperty(styleKey+"."+j+".family",
+ fci.allFonts[j].familyName);
+ props.setProperty(styleKey+"."+j+".file",
+ fci.allFonts[j].fontFile);
+ }
+ }
+ try {
+ /* This writes into a temp file then renames when done.
+ * Since the rename is an atomic action within the same
+ * directory no client will ever see a partially written file.
+ */
+ File fcInfoFile = getFcInfoFile();
+ File dir = fcInfoFile.getParentFile();
+ dir.mkdirs();
+ File tempFile = File.createTempFile("fcinfo", null, dir);
+ FileOutputStream fos = new FileOutputStream(tempFile);
+ props.store(fos,
+ "JDK Font Configuration Generated File: *Do Not Edit*");
+ fos.close();
+ boolean renamed = tempFile.renameTo(fcInfoFile);
+ if (!renamed && SunGraphicsEnvironment.debugFonts) {
+ System.out.println("rename failed");
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Failed renaming file to "+ getFcInfoFile());
+ }
+ } catch (Exception e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("IOException writing to "+ getFcInfoFile());
+ }
+ }
+ }
+
+ /* We want to be able to use this cache instead of invoking
+ * fontconfig except when we can detect the system cache has changed.
+ * But there doesn't seem to be a way to find the location of
+ * the system cache.
+ */
+ private void readFcInfo() {
+ File fcFile = getFcInfoFile();
+ if (!fcFile.exists()) {
+ return;
+ }
+ Properties props = new Properties();
+ try {
+ FileInputStream fis = new FileInputStream(fcFile);
+ props.load(fis);
+ fis.close();
+ } catch (IOException e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("IOException reading from "+fcFile.toString());
+ }
+ return;
+ }
+ String version = (String)props.get("version");
+ if (version == null || !version.equals(fileVersion)) {
+ return;
+ }
+
+ // If there's a new, different fontconfig installed on the
+ // system, we invalidate our fontconfig file.
+ String fcVersionStr = (String)props.get("fcversion");
+ if (fcVersionStr != null) {
+ int fcVersion;
+ try {
+ fcVersion = Integer.parseInt(fcVersionStr);
+ if (fcVersion != 0 &&
+ fcVersion != FontManager.getFontConfigVersion()) {
+ return;
+ }
+ } catch (Exception e) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger =
+ Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning("Exception parsing version " +
+ fcVersionStr);
+ }
+ return;
+ }
+ }
+
+ // If we can locate the fontconfig cache dirs, then compare the
+ // time stamp of those with our properties file. If we are out
+ // of date then re-generate.
+ long lastModified = fcFile.lastModified();
+ int cacheDirIndex = 0;
+ while (cacheDirIndex<4) { // should never be more than 2 anyway.
+ String dir = (String)props.get("cachedir."+cacheDirIndex);
+ if (dir == null) {
+ break;
+ }
+ File dirFile = new File(dir);
+ if (dirFile.exists() && dirFile.lastModified() > lastModified) {
+ return;
+ }
+ cacheDirIndex++;
+ }
+
+ String[] names = { "sansserif", "serif", "monospaced" };
+ String[] fcnames = { "sans", "serif", "monospace" };
+ int namesLen = names.length;
+ int numStyles = 4;
+ FcCompFont[] fci = new FcCompFont[namesLen*numStyles];
+
+ try {
+ for (int i=0; i<namesLen; i++) {
+ for (int s=0; s<numStyles; s++) {
+ int index = i*numStyles+s;
+ fci[index] = new FcCompFont();
+ String key = names[i]+"."+s;
+ fci[index].jdkName = names[i];
+ fci[index].fcFamily = fcnames[i];
+ fci[index].style = s;
+ String lenStr = (String)props.get(key+".length");
+ int nfonts = Integer.parseInt(lenStr);
+ if (nfonts <= 0) {
+ return; // bad file
+ }
+ fci[index].allFonts = new FontConfigFont[nfonts];
+ for (int f=0; f<nfonts; f++) {
+ fci[index].allFonts[f] = new FontConfigFont();
+ String fkey = key+"."+f+".family";
+ String family = (String)props.get(fkey);
+ fci[index].allFonts[f].familyName = family;
+ fkey = key+"."+f+".file";
+ String file = (String)props.get(fkey);
+ if (file == null) {
+ return; // bad file
+ }
+ fci[index].allFonts[f].fontFile = file;
+ }
+ fci[index].firstFont = fci[index].allFonts[0];
+
+ }
+ }
+ fcCompFonts = fci;
+ } catch (Throwable t) {
+ if (SunGraphicsEnvironment.debugFonts) {
+ Logger logger = Logger.getLogger("sun.awt.FontConfiguration");
+ logger.warning(t.toString());
+ }
+ }
+ }
+}