All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.qt.internal.NativeLibraryManager Maven / Gradle / Ivy

/****************************************************************************
**
** Copyright (C) 1992-2009 Nokia. All rights reserved.
** Copyright (C) 2009-2022 Dr. Peter Droste, Omix Visualization GmbH & Co. KG. All rights reserved.
**
** This file is part of Qt Jambi.
**
** ** $BEGIN_LICENSE$
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file.  Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
** 
** In addition, as a special exception, Nokia gives you certain
** additional rights. These rights are described in the Nokia Qt LGPL
** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
** package.
** 
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3.0 as published by the Free Software
** Foundation and appearing in the file LICENSE.GPL included in the
** packaging of this file.  Please review the following information to
** ensure the GNU General Public License version 3.0 requirements will be
** met: http://www.gnu.org/copyleft/gpl.html.
** $END_LICENSE$

**
** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
**
****************************************************************************/

package io.qt.internal;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.MalformedURLException;
import java.net.URISyntaxException;
import java.net.URL;
import java.nio.file.Files;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.SortedMap;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Executor;
import java.util.function.BiConsumer;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.prefs.Preferences;

import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;

import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;

import io.qt.NativeAccess;
import io.qt.QNoSuchSlotException;
import io.qt.QLibraryNotFoundError;
import io.qt.QLibraryNotLoadedError;
import io.qt.QtUninvokable;
import io.qt.core.QDeclarableSignals;

/**
 * The NativeLibraryManager class is responsible for handling native
 * libraries in Qt Jambi. Native libraries can be loaded either
 * directly from the file system using
 * -Djava.library.path or indirectly JAR-file that
 * contain a deployment descriptor. For normal deployment, the JAR
 * approach is recommended.
 *
 * Loading libraries is done by calling the methods
 * loadQtLibrary and loadLibrary.
 *
 * When the indirect .jar file approach is taken, the .jar file will
 * be opened and the native library manager will search for a file
 * named "qtjambi-deployment.xml". This file contains a list of native
 * libraries that should unpacked to a temporary directory and loaded,
 * either right away or at a later time. There are three types of
 * libraries.
 *
 * 
    * *
  • System libraries; such as the system runtime * libraries. These libraries are usually loaded automatically by * the native library loader. * *
  • Normal libraries; such as the QtCore and * io_qt_core libraries. These are loaded at runtime on * demand. * *
  • Plugins; such as qjpeg. These are never loaded explicitly by * the native library manager, but are unpacked into the temporary * folder so that Qt can find and load them from the file system. * *
* * There are three possible deployment scenarios. The simplest and * most straightforward approach is when deploying a Pure Java * application based on Qt Jambi. In this case the prebuilt binaries * from the binary package can just be deployed as part of the * classpath and the rest will solve itself. * * When deploying a Qt Jambi application that is using native code * other than Qt Jambi, we recommend building a new .jar file with a * custom qtjambi-deployment.xml which contais the Qt Jambi libraries * and the custom native libraries. Deployment can then be done by * making sure that this new .jar file is available in the classpath. * * The final option for deployment is when users have a C++ * application which starts and makes use of Qt Jambi. In this case we * suggest that all dependent libraries are available in the file * system and via -Djava.library.path * * To get runtime information about how library loading works, specify * the -Dio.qt.verbose-loading system property * to the Virtual Machine. It possible to specify that the native * library manager should load debug versions of libraries as * well. This is done by specifying the system property * -Dio.qt.debug=debug * */ final class NativeLibraryManager { private NativeLibraryManager() { throw new RuntimeException();} private static final boolean VERBOSE_LOADING = Boolean.getBoolean("io.qt.verbose-loading") || Boolean.getBoolean("qtjambi.verbose-loading"); private static final int[] qtVersionArray = {QtJambiVersion.qtMajorVersion, QtJambiVersion.qtMinorVersion}; private static final int[] qtjambiVersionArray = {QtJambiVersion.qtMajorVersion, QtJambiVersion.qtMinorVersion, QtJambiVersion.qtJambiPatch}; private static final String qtjambiVersion = QtJambiVersion.qtMajorVersion + "." + QtJambiVersion.qtMinorVersion + "." + QtJambiVersion.qtJambiPatch; public static final String ICU_VERSION; public static final String LIBINFIX; // We use a List<> to make the collection read-only an array would not be suitable private static final List systemLibrariesList; private static final List jniLibdirBeforeList; private static final List jniLibdirList; private static boolean isMinGWBuilt = false; static final OperatingSystem operatingSystem = decideOperatingSystem(); private static final String osArchName; private static Configuration configuration = null; private static String qtJambiLibraryPath = null; private static String qtLibraryPath = null; private static Boolean dontUseFrameworks = operatingSystem==OperatingSystem.MacOSX ? null : Boolean.FALSE; private static boolean dontSearchDeploymentSpec = false; private static final Set loadedDeploymentSpecUrls = Collections.synchronizedSet(new HashSet<>()); private static final List deploymentSpec = Collections.synchronizedList(new ArrayList<>()); private static final String qtjambiVersionJarSuffix = String.format("-%1$s.%2$s.%3$s.jar", QtJambiVersion.qtMajorVersion, QtJambiVersion.qtMinorVersion, QtJambiVersion.qtJambiPatch); private static final Map libraryMap = Collections.synchronizedMap(new HashMap()); private static final Reporter reporter = new Reporter(); private static final File jambiDeploymentDir; private static final File jambiJarDir; private static final boolean isJava8Bundle; private static final List pluginLibraries = Collections.synchronizedList(new ArrayList<>()); public static final String K_SUNOS_X86 = "sunos-x86"; public static final String K_SUNOS_X64 = "sunos-x64"; public static final String K_WIN_X86 = "windows-x86"; public static final String K_WIN_X64 = "windows-x64"; public static final String K_WIN_ARM32 = "windows-arm32"; public static final String K_WIN_ARM64 = "windows-arm64"; public static final String K_LINUX_X86 = "linux-x86"; public static final String K_LINUX_X64 = "linux-x64"; public static final String K_LINUX_ARM32 = "linux-arm32"; public static final String K_LINUX_ARM64 = "linux-arm64"; public static final String K_FREEBSD_X86 = "freebsd-x86"; public static final String K_FREEBSD_X64 = "freebsd-x64"; private static final boolean deleteTmpDeployment; static { { File tmpDir = new File(System.getProperty("java.io.tmpdir")); String deploymentdir = System.getProperty("io.qt.deploymentdir", ""); boolean keepDeployment = Boolean.getBoolean("io.qt.keep-temp-deployment"); if(deploymentdir!=null && !deploymentdir.isEmpty() && !"tmp".equalsIgnoreCase(deploymentdir) && !"temp".equalsIgnoreCase(deploymentdir)) { keepDeployment = true; if("user".equalsIgnoreCase(deploymentdir)) { switch (operatingSystem) { case Windows: tmpDir = new File(System.getProperty("user.home"), "AppData\\Local\\QtJambi"); break; case Android: case Linux: tmpDir = new File(System.getProperty("user.home"), ".local/share/QtJambi"); break; case MacOSX: tmpDir = new File(System.getProperty("user.home"), "Library/Application Support/QtJambi"); break; default: break; } }else if("common".equalsIgnoreCase(deploymentdir)) { switch (operatingSystem) { case Windows: tmpDir = new File("C:\\ProgramData\\QtJambi"); break; case Android: case Linux: tmpDir = new File("/usr/local/share/QtJambi"); break; case MacOSX: tmpDir = new File("/Library/Application Support/QtJambi"); break; default: break; } }else { File deploymentDir = new File(deploymentdir); if(deploymentDir.isAbsolute()) { tmpDir = deploymentDir; }else { tmpDir = new File(System.getProperty("user.home"), deploymentdir); } } jambiDeploymentDir = new File(tmpDir, "v" + QtJambiVersion.qtMajorVersion + "." + QtJambiVersion.qtMinorVersion + "." + QtJambiVersion.qtJambiPatch); }else { if(keepDeployment) { jambiDeploymentDir = new File(tmpDir, "QtJambi" + QtJambiVersion.qtMajorVersion + "." + QtJambiVersion.qtMinorVersion + "." + QtJambiVersion.qtJambiPatch + "_" + System.getProperty("user.name")); }else { jambiDeploymentDir = new File(tmpDir, "QtJambi" + QtJambiVersion.qtMajorVersion + "." + QtJambiVersion.qtMinorVersion + "." + QtJambiVersion.qtJambiPatch + "_" + System.getProperty("user.name") + "_" + RetroHelper.processName()); } } deleteTmpDeployment = !keepDeployment; if(deleteTmpDeployment) Logger.getLogger("io.qt.internal").log(Level.FINEST, ()->"Requires deletion of tmp deployment directory "+jambiDeploymentDir.getAbsolutePath()+" during pogram termination."); } boolean _isJava8Bundle = false; { File _jambiJarDir = null; String classURL = ""+NativeLibraryManager.class.getResource("NativeLibraryManager.class"); int index; if(classURL.startsWith("jar:file:") && (index = classURL.indexOf("!/"))>0) { String jarFileURL = classURL.substring(4, index); File jarFile = null; try { jarFile = new File(new URL(jarFileURL).toURI()); } catch (URISyntaxException | MalformedURLException e) { } if(jarFile!=null && jarFile.exists()) { _jambiJarDir = jarFile.getParentFile(); _isJava8Bundle = jarFile.getName().startsWith("qtjambi-jre8-"); } } isJava8Bundle = _isJava8Bundle; jambiJarDir = _jambiJarDir; } String tmpICUVERSION_STRING = null; String tmpINFIX_STRING = null; List tmpSystemLibrariesList = null; List tmpJniLibdirBeforeList = null; List tmpJniLibdirList = null; try { tmpINFIX_STRING = QtJambiVersion.properties.getProperty("qtjambi.libinfix"); tmpICUVERSION_STRING = QtJambiVersion.properties.getProperty("qtjambi.icu.version", "56"); SortedMap tmpSystemLibrariesMap = new TreeMap(); SortedMap tmpJniLibdirBeforeMap = new TreeMap(); SortedMap tmpJniLibdirMap = new TreeMap(); Enumeration e = QtJambiVersion.properties.propertyNames(); while (e.hasMoreElements()) { String key = (String) e.nextElement(); String value = QtJambiVersion.properties.getProperty(key); if (key.equals("qtjambi.system.libraries") || key.startsWith("qtjambi.system.libraries.")) { tmpSystemLibrariesMap.put(key, value); } else if(key.equals("qtjambi.jni.libdir.before") || key.startsWith("qtjambi.jni.libdir.before.")) { tmpJniLibdirBeforeMap.put(key, value); } else if(key.equals("qtjambi.jni.libdir") || key.startsWith("qtjambi.jni.libdir.")) { tmpJniLibdirMap.put(key, value); } } // Map will automatically sort the lists { "", ".0", ".01", ".1", ".10", ".2", ".A", ".a" } tmpSystemLibrariesList = new ArrayList(); for (String v : tmpSystemLibrariesMap.values()) tmpSystemLibrariesList.add(v); if (tmpSystemLibrariesList.size() > 0) tmpSystemLibrariesList = resolveSystemLibraries(tmpSystemLibrariesList); else tmpSystemLibrariesList = null; tmpJniLibdirBeforeList = new ArrayList(); for (String v : tmpJniLibdirBeforeMap.values()) tmpJniLibdirBeforeList.add(v); if (tmpJniLibdirBeforeList.size() > 0) tmpJniLibdirBeforeList = Collections.unmodifiableList(tmpJniLibdirBeforeList); else tmpJniLibdirBeforeList = null; tmpJniLibdirList = new ArrayList(); for (String v : tmpJniLibdirMap.values()) tmpJniLibdirList.add(v); if (tmpJniLibdirList.size() > 0) tmpJniLibdirList = Collections.unmodifiableList(tmpJniLibdirList); else tmpJniLibdirList = null; } catch(Throwable e) { java.util.logging.Logger.getLogger("io.qt.internal").log(java.util.logging.Level.SEVERE, "", e); }finally { ICU_VERSION = tmpICUVERSION_STRING; systemLibrariesList = tmpSystemLibrariesList!=null ? Collections.unmodifiableList(tmpSystemLibrariesList) : Collections.emptyList(); jniLibdirBeforeList = tmpJniLibdirBeforeList!=null ? Collections.unmodifiableList(tmpJniLibdirBeforeList) : Collections.emptyList(); jniLibdirList = tmpJniLibdirList!=null ? Collections.unmodifiableList(tmpJniLibdirList) : Collections.emptyList(); LIBINFIX = tmpINFIX_STRING; } switch (operatingSystem) { case Windows: switch(System.getProperty("os.arch").toLowerCase()) { case "arm": case "arm32": osArchName = K_WIN_ARM32; break; case "arm64": case "aarch64": osArchName = K_WIN_ARM64; break; case "x86_64": case "x64": case "amd64": osArchName = K_WIN_X64; break; default: osArchName = K_WIN_X86; break; } break; case Linux: switch(System.getProperty("os.arch").toLowerCase()) { case "arm": case "arm32": osArchName = K_LINUX_ARM32; break; case "arm64": case "aarch64": osArchName = K_LINUX_ARM64; break; case "x86_64": case "x64": case "amd64": osArchName = K_LINUX_X64; break; default: osArchName = K_LINUX_X86; break; } break; case Android: osArchName = "android"; break; case MacOSX: osArchName = "macos"; break; default: osArchName = "unknown"; break; } reporter.setReportEnabled(VERBOSE_LOADING); try{ Preferences preferences = Preferences.userNodeForPackage(NativeLibraryManager.class); String dirs = preferences.get("qtjambi.previous.deployment.dir", null); if(dirs!=null && !dirs.isEmpty()) { preferences.remove("qtjambi.previous.deployment.dir"); for(String dir : dirs.split("\\"+File.pathSeparator)) { File jambiTempDir = new File(dir); if(jambiTempDir.exists() && jambiTempDir.isDirectory()) clearAndDelete(jambiTempDir); } } }catch(Throwable t) {} if(!(Boolean.getBoolean("io.qt.no-deployment-spec") || Boolean.getBoolean("qtjambi.no-deployment-spec"))) { ClassLoader loader = classLoader(); // Multiple descriptors maybe found Enumeration specsFound = Collections.emptyEnumeration(); try { specsFound = loader.getResources("qtjambi-deployment.xml"); } catch (IOException e) { Logger.getLogger("io.qt.internal").log(Level.WARNING, "", e); } // FIXME: Want searchForDescriptors/parse/resolve/unpack/load phases separated dontSearchDeploymentSpec = specsFound.hasMoreElements(); if(!dontSearchDeploymentSpec && jambiJarDir!=null) { List foundURLs = new ArrayList<>(); String postfix; if("debug".equals(System.getProperty("io.qt.debug"))) { postfix = "-native-" + osArchName + "-debug" + qtjambiVersionJarSuffix; File nativeFile = new File(jambiJarDir, "qtjambi"+postfix); if(nativeFile.exists()) { try { foundURLs.add(new URL("jar:"+nativeFile.toURI()+"!/qtjambi-deployment.xml")); } catch (IOException e) { } } }else { postfix = "-native-" + osArchName + qtjambiVersionJarSuffix; File nativeFile = new File(jambiJarDir, "qtjambi"+postfix); if(nativeFile.exists()) { try { foundURLs.add(new URL("jar:"+nativeFile.toURI()+"!/qtjambi-deployment.xml")); } catch (IOException e) { } }else { postfix = "-native-" + osArchName + "-debug" + qtjambiVersionJarSuffix; nativeFile = new File(jambiJarDir, "qtjambi"+postfix); if(nativeFile.exists()) { try { foundURLs.add(new URL("jar:"+nativeFile.toURI()+"!/qtjambi-deployment.xml")); } catch (IOException e) { } } } } String notPostfix = null; if(_isJava8Bundle) postfix = "-jre8" + postfix; else notPostfix = "-jre8" + postfix; for(String f : jambiJarDir.list()) { if(f.startsWith("qtjambi-plugin-") && f.endsWith(postfix) && (_isJava8Bundle || !f.endsWith(notPostfix))) { try { foundURLs.add(new URL("jar:"+new File(jambiJarDir, f).toURI()+"!/qtjambi-deployment.xml")); } catch (IOException e) { } } } if(!foundURLs.isEmpty()) specsFound = Collections.enumeration(foundURLs); } while (specsFound.hasMoreElements()) { URL url = specsFound.nextElement(); if(loadedDeploymentSpecUrls.contains(url)) continue; loadedDeploymentSpecUrls.add(url); reporter.report("Found ", url.toString()); DeploymentSpec spec = null; String protocol = url.getProtocol(); if (protocol.equals("jar")) { // Using toExternalForm() will convert a space character into %20 which breaks java.lang.File use of the resulting path. String eform = url.toExternalForm(); // Try to decide the name of the .jar file to have a // reference point for later.. int start = 4; //"jar:".length(); int end = eform.indexOf("!/", start); // eform has the "jar:url!/entry" format if (end != -1) { try { URL jarUrl = new URL(eform.substring(start, end)); String jarName = new File(jarUrl.getFile()).getName(); reporter.report("Loading ", jarName, " from ", eform); spec = unpackDeploymentSpec(url, jarName, null); } catch (ParserConfigurationException | SAXException | IOException e) { Logger.getLogger("io.qt.internal").log(Level.WARNING, "", e); } } } else if (protocol.equals("file")) { // No unpack since we presume we are already unpacked try { spec = unpackDeploymentSpec(url, null, Boolean.FALSE); } catch (ParserConfigurationException | SAXException | IOException e) { Logger.getLogger("io.qt.internal").log(Level.WARNING, "", e); } } if(spec != null) { if("qtjambi".equals(spec.getModule())) deploymentSpec.add(0, spec); else deploymentSpec.add(spec); } } if(!deploymentSpec.isEmpty()) { DeploymentSpec qtjambiSpec = deploymentSpec.get(0); if(Configuration.Release.toString().compareToIgnoreCase(qtjambiSpec.getConfiguration()) == 0) configuration = Configuration.Release; else if(Configuration.Debug.toString().compareToIgnoreCase(qtjambiSpec.getConfiguration()) == 0) configuration = Configuration.Debug; else if(Configuration.Test.toString().compareToIgnoreCase(qtjambiSpec.getConfiguration()) == 0) configuration = Configuration.Test; else configuration = Configuration.Release; switch (operatingSystem) { case Windows: qtJambiLibraryPath = new File(qtjambiSpec.getBaseDir(), "bin").getAbsolutePath(); break; default: qtJambiLibraryPath = new File(qtjambiSpec.getBaseDir(), "lib").getAbsolutePath(); break; } for(DeploymentSpec spec : deploymentSpec) { if(!spec.getCompiler().equals(qtjambiSpec.getCompiler())) { throw new DeploymentSpecException(String.format("Native deployments of different builts: %1$s and %2$s", qtjambiSpec.getCompiler(), spec.getCompiler())); }else if(!spec.getConfiguration().equals(qtjambiSpec.getConfiguration())) { throw new DeploymentSpecException(String.format("Native deployments of different configurations: %1$s and %2$s", qtjambiSpec.getConfiguration(), spec.getConfiguration())); } } }else { configuration = decideConfiguration(); } }else { configuration = decideConfiguration(); } if(VERBOSE_LOADING) System.out.println(reporter.recentReports()); } private static void clearAndDelete(File directory) { if(directory!=null) { if(directory.isDirectory()) { for(File file : directory.listFiles()) { if(file.getName().equals(".") || file.getName().equals("..")) continue; if(file.isDirectory() && !Files.isSymbolicLink(file.toPath())) { clearAndDelete(file); }else { file.delete(); } } } directory.delete(); } } // The purpose of this method is to resolve the names provided in the list into DSO paths relative to the project. private static List resolveSystemLibraries(List tmpSystemLibrariesList) { if(tmpSystemLibrariesList == null) return null; List resolvedList = new ArrayList(); for(String original : tmpSystemLibrariesList) { String s = original; if(operatingSystem==OperatingSystem.Windows) // convert "/" into "\" s = stringCharReplace(s, "/", File.separator); String resolvedPath = null; for(DeploymentSpec deploymentSpec : deploymentSpec) { File f = new File(deploymentSpec.getBaseDir(), s); if(f.isFile()) { resolvedPath = s; break; } File libDir = new File(deploymentSpec.getBaseDir(), "lib"); f = new File(libDir, s); if(f.isFile()) { resolvedPath = "lib" + File.separator + s; break; } File binDir = new File(deploymentSpec.getBaseDir(), "bin"); f = new File(binDir, s); if(f.isFile()) { resolvedPath = "bin" + File.separator + s; break; } } if(resolvedPath != null) resolvedList.add(resolvedPath); else System.err.println("IGNORED version.properties qtjambi.system.libraries entry \"" + original + "\": file could not be found"); } return Collections.unmodifiableList(resolvedList); } private static String stringCharReplace(String s, CharSequence fromCharSequence, CharSequence toCharSequence) { final int len = s.length(); StringBuilder sb = new StringBuilder(); int i = 0; while(i < len) { char c = s.charAt(i); if(c == fromCharSequence.charAt(0) && stringCompareAt(s, i, fromCharSequence, 0)) { // FIXME needle/haystack sb.append(toCharSequence); i += fromCharSequence.length(); // skip input sequence } else { sb.append(c); i++; } } return sb.toString(); } // a - haystack, b - needle private static boolean stringCompareAt(CharSequence a, int aOffset, CharSequence b, int bOffset) { final int aLen = a.length(); // used as end range final int bLen = b.length(); // used as end range while(aOffset < aLen && bOffset < bLen) { char ca = a.charAt(aOffset++); char ba = b.charAt(bOffset++); if(ca != ba) return false; } // we are only allowed to run out of chars on 'a' side if(bOffset < bLen) return false; // ran out of chars on 'b' side return true; } private static OperatingSystem decideOperatingSystem() { String osName = System.getProperty("os.name").toLowerCase(); if (osName.startsWith("windows")) return OperatingSystem.Windows; if (osName.startsWith("mac")) return OperatingSystem.MacOSX; if (osName.startsWith("android")) return OperatingSystem.Android; return OperatingSystem.Linux; } static Configuration configuration() { return configuration; } private static Configuration decideConfiguration() { Configuration configuration = null; String debugString = System.getProperty("io.qt.debug"); try { if(debugString != null) { // This was added to allow unit tests to specify Configuration (as no MANIFEST.MF is available) if(Configuration.Release.toString().compareToIgnoreCase(debugString) == 0) configuration = Configuration.Release; else if(Configuration.Debug.toString().compareToIgnoreCase(debugString) == 0) configuration = Configuration.Debug; else if(Configuration.Test.toString().compareToIgnoreCase(debugString) == 0) configuration = Configuration.Test; if(configuration == null) { Boolean booleanValue = Boolean.valueOf(debugString); if((booleanValue != null && booleanValue.booleanValue()) || debugString.length() == 0) { configuration = Configuration.Debug; // FIXME: When we can unambigiously auto-detect this from the MANIFEST we don't need to // emit this, we'd only emit it when both non-debug and debug were found and we selected // the debug kind. java.util.logging.Logger.getLogger("io.qt.internal").log(java.util.logging.Level.WARNING, "-Dio.qt.debug=" + Boolean.TRUE + "; is set instead of =debug."); } } } } catch(Exception e) { java.util.logging.Logger.getLogger("io.qt.internal").log(java.util.logging.Level.SEVERE, "-Dio.qt.debug=" + Boolean.FALSE + "; is assumed default", e); // only because Configuration.Release is assumed } if(configuration==null) configuration = Configuration.Release; return configuration; } static File jambiDeploymentDir() { return jambiDeploymentDir; } @NativeAccess @QtUninvokable static void resetDeploymentSpecs() { deploymentSpec.clear(); if(deleteTmpDeployment) { if(jambiDeploymentDir.exists() && jambiDeploymentDir.isDirectory()) { Logger.getLogger("io.qt.internal").log(Level.FINEST, ()->"Deleting tmp deployment directory "+jambiDeploymentDir.getAbsolutePath()+"."); clearAndDelete(jambiDeploymentDir); if(jambiDeploymentDir.exists() && jambiDeploymentDir.isDirectory()) { Logger.getLogger("io.qt.internal").log(Level.FINEST, "Preparing pending deletion..."); Preferences preferences = Preferences.userNodeForPackage(NativeLibraryManager.class); String dirs = preferences.get("qtjambi.previous.deployment.dir", null); if(dirs!=null && !dirs.isEmpty()) { preferences.put("qtjambi.previous.deployment.dir", dirs + File.pathSeparator + jambiDeploymentDir.getAbsolutePath()); }else { preferences.put("qtjambi.previous.deployment.dir", jambiDeploymentDir.getAbsolutePath()); } } }else { Logger.getLogger("io.qt.internal").log(Level.FINEST, ()->"Tmp deployment directory "+jambiDeploymentDir.getAbsolutePath()+" does not exist."); } } } private static class XMLHandler extends DefaultHandler { public DeploymentSpec spec; private final List direntCurrentPathList = new ArrayList(); public void startElement(String uri, String localName, String name, org.xml.sax.Attributes attributes) { if (name.equals("symlink")) { SymlinkEntry e = new SymlinkEntry(); e.setName(attributes.getValue("name")); e.setTarget(attributes.getValue("target")); if (e.getName() == null) { throw new DeploymentSpecException(" element missing required attribute \"name\""); } if (e.getTarget() == null) { throw new DeploymentSpecException(" element missing required attribute \"target\""); } e.setDeploymentSpec(spec); if(e.getName().startsWith("qml/")) { spec.setHasQmlPaths(true); }else if(e.getName().startsWith("plugins/")) { spec.setHasPluginPaths(true); }else if(e.getName().startsWith("translations/")) { spec.setHasTrPaths(true); }else{ String fileName = e.getName(); if(fileName.startsWith("lib/") || fileName.startsWith("bin/")) fileName = fileName.substring(4); // Add library name to the global map of libraries... LibraryEntry old = libraryMap.get(fileName); if (old == null) { reporter.report(" - adding '", fileName, "' to library map"); libraryMap.put(fileName, e); } } spec.addLibraryEntry(e); reporter.report(" - symlink: name='", e.getName(), "'"); } else if (name.equals("library") || name.equals("syslibrary")) { LibraryEntry e = new LibraryEntry(); e.setName(attributes.getValue("name")); if (e.getName() == null) { throw new DeploymentSpecException(" element missing required attribute \"name\""); } e.setDeploymentSpec(spec); if(e.getName().startsWith("qml/")) { spec.setHasQmlPaths(true); }else if(e.getName().startsWith("plugins/")) { spec.setHasPluginPaths(true); }else if(e.getName().startsWith("translations/")) { spec.setHasTrPaths(true); }else{ String fileName = e.getName(); if(fileName.startsWith("lib/") || fileName.startsWith("bin/")) fileName = fileName.substring(4); // Add library name to the global map of libraries... LibraryEntry old = libraryMap.get(fileName); if (old != null && !old.getDeploymentSpec().getSourceUrl().equals(spec.getSourceUrl())) { throw new DeploymentSpecException(" '" + e.getName() + "' is duplicated. Present in both '" + spec.getSourceUrl() + "' and '" + old.getDeploymentSpec().getSourceUrl() + "'."); } reporter.report(" - adding '", fileName, "' to library map"); libraryMap.put(fileName, e); } spec.addLibraryEntry(e); reporter.report(" - library: name='", e.getName(), "'"); } else if (name.equals("qtjambi-deploy")) { String system = attributes.getValue("system"); if (system == null || system.length() == 0) { throw new DeploymentSpecException(" element missing required attribute 'system'"); } else if (!system.equals(osArchName)) { throw new WrongSystemException(String.format("Expected version: %1$s, found: %2$s.", osArchName, system)); } spec.setSystem(system); String version = attributes.getValue("version"); if (version == null || version.length() == 0) { throw new DeploymentSpecException(" element missing required attribute 'version'"); } else if(!qtjambiVersion.equals(version)) throw new WrongVersionException(String.format("Expected version: %1$s, found: %2$s.", qtjambiVersion, version)); spec.setVersion(version); spec.setModule(attributes.getValue("module")); spec.setCompiler(attributes.getValue("compiler")); spec.setConfiguration(attributes.getValue("configuration")); spec.setDate(attributes.getValue("date")); spec.setTime(attributes.getValue("time")); } else if (name.equals("directory")) { String attrName = attributes.getValue("name"); if (attrName == null) { throw new DeploymentSpecException(" element missing required attribute \"name\""); } while(attrName.length() > 0 && attrName.charAt(0) == '/') attrName = attrName.substring(1); while(attrName.length() > 0 && attrName.charAt(attrName.length() - 1) == '/') attrName = attrName.substring(0, attrName.length() - 1 - 1); direntCurrentPathList.add(attrName); } else if (name.equals("file")) { String attrName = attributes.getValue("name"); if (attrName == null) { throw new DeploymentSpecException(" element missing required attribute \"name\""); } StringBuilder sb = new StringBuilder(); for(String s : direntCurrentPathList) { if(sb.length() > 0) sb.append('/'); sb.append(s); } if(sb.length() > 0) sb.append('/'); sb.append(attrName); spec.addDirentPath(sb.toString()); reporter.report(" - dirent path='", sb.toString(), "'"); } } public void endElement(String uri, String localName, String name) { if (name.equals("directory")) { if (direntCurrentPathList.isEmpty() == false) direntCurrentPathList.remove(direntCurrentPathList.size() - 1); // remove last } } } //no-one used this so I commented it out /*private static class ChecksumFileFilter implements FilenameFilter { public boolean accept(File dir, String name) { return name.endsWith(".chk"); } }*/ /** * Returns a file that is used for caching native libraries. The * path is a subdirectory of java.io.tmpdir, named * QtJambi_{user}_{architecture}_{version}_{key}. The * key is the same as the cache key used in the deployment * descriptor. The purpose of using different keys is to avoid * binary compatibility when various configurations of Qt and Qt * Jambi are used on the same system. * * When deployment descriptors are not used, this location will * not be used. * * @param key The cache key to. * @return A File representing the location directory for * temporary content. The directory is not explicitly created in * this here. */ private static File jambiTempDirBase(DeploymentSpec spec) { File tmpDir = new File(System.getProperty("java.io.tmpdir")); String user = System.getProperty("user.name"); return new File(tmpDir, "QtJambi_" + spec.getVersion() + "_" + spec.getSystem() + "_" + spec.getCompiler() + "_" + spec.getConfiguration() + "_" + spec.getDate() + "_" + spec.getTime() + "_" + user); } /** * Returns the list of all plugin paths that are specified in the * deployment descriptors. If deployment descriptors are not used, * this list will be an empty list. * * @return The list of plugin paths */ static List pluginPaths() { pluginLibraries.removeIf( l -> { try { l.extract(); }catch(Throwable t) { java.util.logging.Logger.getLogger("io.qt.internal").log(java.util.logging.Level.SEVERE, "Error while extracting library", t); } return true; }); List paths = new ArrayList(); for (DeploymentSpec spec : deploymentSpec) { File root = spec.getBaseDir(); if(spec.isHasPluginPaths()) paths.add(new File(root, "plugins").getAbsolutePath()); } return paths; } static void loadLibrary(String library) { switch (operatingSystem) { case Windows: loadNativeLibrary(Object.class, library + ".dll", Arrays.asList(""), false); break; case MacOSX: try { loadNativeLibrary(Object.class, "lib" + library + ".jnilib", Arrays.asList(""), false); } catch (Throwable e) { loadNativeLibrary(Object.class, "lib" + library + ".dylib", Arrays.asList(""), false); } break; case Android: case Linux: loadNativeLibrary(Object.class, "lib" + library + ".so", Arrays.asList(""), false); break; default: break; } } static void loadQtJambiLibrary() { loadQtJambiLibrary(NativeLibraryManager.class, null, "QtJambi"); } static void loadQtJambiLibrary(Class callerClass, String library) { loadQtJambiLibrary(callerClass, "QtJambi", library); } static void loadJambiLibrary(Class callerClass, String library) { loadQtJambiLibrary(callerClass, null, library); } private static void loadQtJambiLibrary(Class callerClass, String prefix, String library) { List replacements = new ArrayList<>(); String lib = jambiLibraryName(prefix, library, qtjambiVersionArray, replacements); loadNativeLibrary(callerClass, lib, replacements, true); } /** * Loads a library with name specified in library. * The library name will be expanded to the default shared library * name for a given platform, so the name "QtCore" and version "4" and "5" will be * expanded like this: * *
    *
  • Windows: QtCore4.dll *
  • Linux / Unix: libQtCore.so.4 *
  • Mac OS X: libQtCore.4.dylib *
*
    *
  • Windows: Qt5Core.dll *
  • Linux / Unix: libQt5Core.so.5 *
  • Mac OS X: libQt5Core.5.dylib *
* * When using loading libraries from the filesystem, this method * simply calls System.loadLibrary. * @param library The name of the library.. */ static void loadQtLibrary(String library) { findAndLoadLibrary(Object.class, "Qt", library, LIBINFIX, qtVersionArray); } static void loadUtilityLibrary(String library, String version) { findAndLoadLibrary(Object.class, library, version); } static File loadQtCore() { loadSystemLibraries(); if(isAvailableLibrary("icudata", NativeLibraryManager.ICU_VERSION)){ findAndLoadLibrary(Object.class, "icudata", NativeLibraryManager.ICU_VERSION); } if(isAvailableLibrary("icuuc", NativeLibraryManager.ICU_VERSION)){ findAndLoadLibrary(Object.class, "icuuc", NativeLibraryManager.ICU_VERSION); } if(isAvailableLibrary("icui18n", NativeLibraryManager.ICU_VERSION)){ findAndLoadLibrary(Object.class, "icui18n", NativeLibraryManager.ICU_VERSION); } try{ File lib = findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); qtLibraryPath = lib.getParentFile().getAbsolutePath(); return lib; } catch (RuntimeException e) { if(configuration==Configuration.Release) { // try to run qtjambi with debug libraries instead configuration=Configuration.Debug; try { return findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); } catch (Throwable e1) { e.addSuppressed(e1); configuration=Configuration.Release; if(dontUseFrameworks==null) { dontUseFrameworks = Boolean.TRUE; try { return findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); } catch (Throwable e2) { e.addSuppressed(e2); configuration=Configuration.Debug; try { return findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); } catch (Throwable e3) { e.addSuppressed(e3); configuration=Configuration.Release; throw e; } } }else{ throw e; } } }else { if(dontUseFrameworks==null) { dontUseFrameworks = Boolean.TRUE; try { return findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); } catch (Throwable e1) { e.addSuppressed(e1); throw e; } }else{ if(!isMinGWBuilt && operatingSystem==OperatingSystem.Windows) { Availability availability = getLibraryAvailability(null, "libstdc++-6", null, null, null, Configuration.Release); if(availability.entry!=null){ isMinGWBuilt = true; }else if(availability.file!=null){ isMinGWBuilt = new File(availability.file.getParentFile(), "Qt"+QtJambiVersion.qtMajorVersion+"Core.dll").isFile(); } if(isMinGWBuilt) { return findAndLoadLibrary(Object.class, "Qt", "Core", LIBINFIX, qtVersionArray); } } throw e; } } }finally { if(dontUseFrameworks==null) { dontUseFrameworks = Boolean.FALSE; } } } public static boolean isMinGWBuilt() { return isMinGWBuilt; } private static File findAndLoadLibrary(Class callerClass, String library, String version) { List replacements = new ArrayList<>(); String lib = qtLibraryName(null, library, null, null, version, configuration, replacements); return loadNativeLibrary(callerClass, lib, replacements, false); } private static File findAndLoadLibrary(Class callerClass, String qtprefix, String library, String libInfix, int[] version) { List replacements = new ArrayList<>(); String lib = qtLibraryName(qtprefix, library, libInfix, version, null, configuration, replacements); return loadNativeLibrary(callerClass, lib, replacements, false); } static boolean isAvailableQtLibrary(String library) { return getLibraryAvailability("Qt", library, LIBINFIX, qtVersionArray, null, configuration).isAvailable(); } static boolean isAvailableLibrary(String library, String version) { return getLibraryAvailability(null, library, null, null, version, configuration).isAvailable(); } private static class Availability{ public Availability(File file) { super(); this.file = file; this.entry = null; } public Availability() { super(); this.file = null; this.entry = null; } public Availability(LibraryEntry entry) { super(); this.file = null; this.entry = entry; } final File file; final LibraryEntry entry; boolean isAvailable() { return file!=null || entry!=null; } } private static Availability getLibraryAvailability(String qtprefix, String library, String libInfix, int[] version, String versionStrg, Configuration configuration) { List replacements = new ArrayList<>(); String libFormat = qtLibraryName(qtprefix, library, libInfix, version, versionStrg, configuration, replacements); // "QtDBus" => "libQtDBus.so.4" String libPaths = System.getProperty("io.qt.library-path-override"); if (libPaths == null || libPaths.length() == 0) { libPaths = System.getProperty("java.library.path"); } List paths = new ArrayList<>(); if (libPaths != null) paths.addAll(Arrays.asList(libPaths.split("\\"+File.pathSeparator))); switch (operatingSystem) { case Windows: libPaths = System.getenv("PATH"); break; case MacOSX: libPaths = System.getenv("DYLD_LIBRARY_PATH"); break; default: libPaths = System.getenv("LD_LIBRARY_PATH"); } if (libPaths != null) paths.addAll(Arrays.asList(libPaths.split("\\"+File.pathSeparator))); paths = mergeJniLibdir(paths); // search active deploymentSpec for existance of library for(String replacement : replacements) { String lib = String.format(libFormat, replacement); for(DeploymentSpec deploymentSpec : deploymentSpec) { List libraries = deploymentSpec.getLibraries(); for(LibraryEntry libraryEntry : libraries) { String name = libraryEntry.getName(); // name="lib/libQtFoo.so.4" if(name == null) continue; if(lib.equals(name)) // lib=="lib/libQtFoo.so.4" return new Availability(libraryEntry); String[] pathA = name.split("\\/"); if(pathA == null || pathA.length == 0) continue; String lastPart = pathA[pathA.length - 1]; if(lib.equals(lastPart)) // lib=="libQtFoo.so.4" return new Availability(libraryEntry); } } for (String path : paths) { File f = new File(path, lib); if (f.exists()) { return new Availability(f); } } } return new Availability(); } /** * Returns a classloader for current context... * @return The classloader */ private static ClassLoader classLoader() { ClassLoader loader = Thread.currentThread().getContextClassLoader(); if (loader == null) { loader = NativeLibraryManager.class.getClassLoader(); assert(loader != null); } return loader; } /** * Tries to load the specified library. It will first try to load * the library using the deploy spec, and if that fails, it will * try to load the library using a standard System.loadLibrary() * call. * @param lib The full name of the library to load, such as libQtCore.so.4 */ private static File loadNativeLibrary(Class callerClass, String lib, List replacements, boolean searchSpec) { File result = null; try { result = loadLibrary_helper(callerClass, lib, replacements, searchSpec); if (VERBOSE_LOADING) System.out.println(reporter.recentReports()); } catch (RuntimeException | Error e) { if(VERBOSE_LOADING) System.err.println(reporter.recentReports()); throw e; } catch (Throwable e) { if(reporter.toString().isEmpty()) { if(e.getMessage().isEmpty()) { throw new QLibraryNotLoadedError("Loading library failed.", e); }else { throw new QLibraryNotLoadedError(e.getMessage(), e); } }else { throw new QLibraryNotLoadedError("Loading library failed. Progress so far: " + reporter, e); } } return result; } private static BiConsumer, String> libraryLoader; static{ libraryLoader = (callerClass, lib) -> { synchronized(NativeLibraryManager.class) { @SuppressWarnings({ "rawtypes", "unchecked" }) QDeclarableSignals.Signal2, String> loadLibrary = new QDeclarableSignals.Signal2(Class.class, String.class); try { loadLibrary.connect(Runtime.getRuntime(), "load0(Class, String)"); libraryLoader = loadLibrary::emit; } catch (QNoSuchSlotException e) { libraryLoader = (_callerClass, _lib)->Runtime.getRuntime().load(_lib); } } libraryLoader.accept(callerClass, lib); }; } private static final Map> javaLibraryPaths = new HashMap<>(); private static final Map> ldLibraryPaths = new HashMap<>(); private static final Map> javaLibraryPathOverrides = new HashMap<>(); private static List splitPath(String path){ if(path!=null) { return Arrays.asList(path.split("\\"+File.pathSeparator)); }else { return Collections.emptyList(); } } private static final Set loadedLibraries = new TreeSet<>(); private static boolean autoExtractQtJambiLibs; private static File loadLibrary_helper(Class callerClass, final String libFormat, List replacements, boolean searchSpec) { if(loadedLibraries.contains(libFormat)) return null; File result = null; ClassLoader callerClassLoader = callerClass.getClassLoader(); boolean loaded = false; loop1: for(String replacement : replacements) { String library = String.format(libFormat, replacement); LibraryEntry e; // Try to load via deploy spec... e = libraryMap.get(library); if (e != null) { if (e.isLoaded()) { return result; } try { e.extract(); } catch (Exception e1) { throw new QLibraryNotLoadedError("Unable to extract library "+e.getName(), e1); } File libFile = e.getDeploymentSpec().buildPath(e.getName()); if(libFile!=null && libFile.isFile()) { reporter.report("Loading library: '", library, "' from deployment spec at ", libFile.getAbsolutePath()); if(callerClassLoader==NativeLibraryManager.class.getClassLoader() || callerClassLoader==Object.class.getClassLoader()) { Runtime.getRuntime().load(libFile.getAbsolutePath()); }else { libraryLoader.accept(callerClass, libFile.getAbsolutePath()); } loadedLibraries.add(libFormat); result = libFile; if(operatingSystem==OperatingSystem.MacOSX && (dontUseFrameworks==null || !dontUseFrameworks) && qtLibraryPath==null) { result = result.getParentFile().getParentFile().getParentFile(); } e.setLoaded(true); loaded = true; break loop1; } // Load via System.load() using default paths.. } } if(!loaded && (!autoExtractQtJambiLibs || !searchSpec)) { loop2: for(String replacement : replacements) { String library = String.format(libFormat, replacement); List libraryPaths = new ArrayList<>(); if(qtJambiLibraryPath!=null) libraryPaths.add(qtJambiLibraryPath); switch (operatingSystem) { case Windows: libraryPaths.add(new File(jambiDeploymentDir, "bin").getAbsolutePath()); if(jambiJarDir!=null) libraryPaths.add(new File(jambiJarDir, "bin").getAbsolutePath()); break; default: libraryPaths.add(new File(jambiDeploymentDir, "lib").getAbsolutePath()); if(jambiJarDir!=null) libraryPaths.add(new File(jambiJarDir, "lib").getAbsolutePath()); break; } if (System.getProperties().contains("io.qt.library-path-override")) { // reporter.report(" - using 'io.qt.library-path-override'"); libraryPaths.addAll(javaLibraryPathOverrides.computeIfAbsent(System.getProperty("io.qt.library-path-override"), NativeLibraryManager::splitPath)); } else { // reporter.report(" - using 'java.library.path'"); libraryPaths.addAll(javaLibraryPaths.computeIfAbsent(System.getProperty("java.library.path"), NativeLibraryManager::splitPath)); } libraryPaths = mergeJniLibdir(libraryPaths); for (String path : libraryPaths) { File f = new File(path, library); if (f.exists()) { if(callerClassLoader==NativeLibraryManager.class.getClassLoader() || callerClassLoader==Object.class.getClassLoader()) { Runtime.getRuntime().load(f.getAbsolutePath()); }else { libraryLoader.accept(callerClass, f.getAbsolutePath()); } loadedLibraries.add(libFormat); reporter.report("Loading library: '", library, "'. Path was: ", f.getAbsolutePath()); result = f; if(operatingSystem==OperatingSystem.MacOSX && (dontUseFrameworks==null || !dontUseFrameworks) && qtLibraryPath==null) { result = result.getParentFile().getParentFile().getParentFile(); } loaded = true; break loop2; } } } } if(!loaded && searchSpec && !dontSearchDeploymentSpec && !(Boolean.getBoolean("io.qt.no-deployment-spec") || Boolean.getBoolean("qtjambi.no-deployment-spec"))) { String className = callerClass.getName(); int idx = className.lastIndexOf('.'); className = className.substring(idx+1); String classURL = ""+callerClass.getResource(className+".class"); int index; if(classURL.startsWith("jar:file:") && (index = classURL.indexOf("!/"))>0) { String jarFileURL = classURL.substring(4, index); File jarFile = null; try { jarFile = new File(new URL(jarFileURL).toURI()); } catch (URISyntaxException | MalformedURLException e) { } if(jarFile!=null && jarFile.exists()) { File directory = jarFile.getParentFile(); String fileName = jarFile.getName(); if(isJava8Bundle) index = fileName.indexOf("-jre8"+qtjambiVersionJarSuffix); else index = fileName.indexOf(qtjambiVersionJarSuffix); if(index>0) { String moduleName = fileName.substring(0, index); String postfix = "-native-" + osArchName; if(configuration==Configuration.Debug) postfix += "-debug"; postfix += qtjambiVersionJarSuffix; File nativeFile = new File(directory, moduleName + postfix); boolean found = false; if(nativeFile.exists()) { try { URL nativeFileURL = new URL("jar:"+nativeFile.toURI()+"!/qtjambi-deployment.xml"); if(!loadedDeploymentSpecUrls.contains(nativeFileURL)) { loadedDeploymentSpecUrls.add(nativeFileURL); DeploymentSpec spec = unpackDeploymentSpec(nativeFileURL, nativeFile.getName(), null); if(spec != null && spec.getConfiguration().equalsIgnoreCase(configuration.name()) && (deploymentSpec.isEmpty() || spec.getCompiler().equals(deploymentSpec.get(0).getCompiler()))) { if("qtjambi".equals(spec.getModule())) deploymentSpec.add(0, spec); else deploymentSpec.add(spec); if(qtJambiLibraryPath==null) { switch (operatingSystem) { case Windows: qtJambiLibraryPath = new File(spec.getBaseDir(), "bin").getAbsolutePath(); break; default: qtJambiLibraryPath = new File(spec.getBaseDir(), "lib").getAbsolutePath(); break; } } found = true; } } } catch (ParserConfigurationException | SAXException | IOException e) { } } if(found) { loop3: for(String replacement : replacements) { String library = String.format(libFormat, replacement); LibraryEntry e; // Try to load via deploy spec... e = libraryMap.get(library); if (e != null) { if (e.isLoaded()) { return result; } try { e.extract(); } catch (Exception e1) { throw new QLibraryNotLoadedError("Unable to extract library "+e.getName(), e1); } File libFile = e.getDeploymentSpec().buildPath(e.getName()); if(libFile!=null && libFile.isFile()) { reporter.report("Loading library: '", library, "' from deployment spec at ", libFile.getAbsolutePath()); if(callerClassLoader==NativeLibraryManager.class.getClassLoader() || callerClassLoader==Object.class.getClassLoader()) { Runtime.getRuntime().load(libFile.getAbsolutePath()); }else { libraryLoader.accept(callerClass, libFile.getAbsolutePath()); } autoExtractQtJambiLibs = true; loadedLibraries.add(libFormat); result = libFile; if(operatingSystem==OperatingSystem.MacOSX && (dontUseFrameworks==null || !dontUseFrameworks) && qtLibraryPath==null) { result = result.getParentFile().getParentFile().getParentFile(); } e.setLoaded(true); loaded = true; break loop3; } // Load via System.load() using default paths.. } } } } } } } if(!loaded && (!autoExtractQtJambiLibs || !searchSpec)) { for(String replacement : replacements) { String library = String.format(libFormat, replacement); List libraryPaths = new ArrayList<>(); switch (operatingSystem) { case Windows: libraryPaths.addAll(ldLibraryPaths.computeIfAbsent(System.getenv("PATH"), NativeLibraryManager::splitPath)); break; case MacOSX: libraryPaths.addAll(ldLibraryPaths.computeIfAbsent(System.getenv("DYLD_LIBRARY_PATH"), NativeLibraryManager::splitPath)); break; default: libraryPaths.addAll(ldLibraryPaths.computeIfAbsent(System.getenv("LD_LIBRARY_PATH"), NativeLibraryManager::splitPath)); } libraryPaths = mergeJniLibdir(libraryPaths); for (String path : libraryPaths) { File f = new File(path, library); if (f.exists()) { if(callerClassLoader==NativeLibraryManager.class.getClassLoader() || callerClassLoader==Object.class.getClassLoader()) { Runtime.getRuntime().load(f.getAbsolutePath()); }else { libraryLoader.accept(callerClass, f.getAbsolutePath()); } loadedLibraries.add(libFormat); reporter.report("Loading library: '", library, "'. Path was: ", f.getAbsolutePath()); result = f; if(operatingSystem==OperatingSystem.MacOSX && (dontUseFrameworks==null || !dontUseFrameworks) && qtLibraryPath==null) { result = result.getParentFile().getParentFile().getParentFile(); } loaded = true; break; } } } } if (!loaded) { String lib = String.format(libFormat, replacements.get(0)); throw new QLibraryNotFoundError("Library '" + lib +"' was not found in 'java.library.path=" + System.getProperty("java.library.path") + "'"); } return result; } /** * * @param url This maybe "file" or "jar" protocol. "http" is untested. * @return * @throws SAXException * @throws ParserConfigurationException * @throws IOException * @throws Exception */ private static DeploymentSpec readDeploySpec(URL url) throws ParserConfigurationException, SAXException, IOException { reporter.report("Checking Archive '", url.toString(), "'"); DeploymentSpec spec = new DeploymentSpec(); spec.setSourceUrl(url); SAXParserFactory fact = SAXParserFactory.newInstance(); SAXParser parser = fact.newSAXParser(); XMLHandler handler = new XMLHandler(); handler.spec = spec; InputStream inStream = null; try { inStream = url.openStream(); parser.parse(inStream, handler); return spec; } finally { if(inStream != null) { inStream.close(); inStream = null; } } } private static class ExecutorThread extends Thread implements Executor{ private final List commands = new ArrayList<>(); private boolean noMoreExpected = false; private ExecutorThread() { super(); this.setDaemon(true); this.setName("QtJambi_LibraryExtractor"); } @Override public void execute(Runnable command) { synchronized(this.commands) { this.commands.add(command); this.commands.notifyAll(); } } @Override public void run() { try { while(true) { synchronized(this.commands) { if(this.commands.isEmpty()) { if(noMoreExpected) break; this.commands.wait(); } } List commands = new ArrayList<>(); synchronized(this.commands) { commands.addAll(this.commands); this.commands.clear(); } for (Runnable runnable : commands) { try { runnable.run(); } catch (Throwable e) { if(e instanceof InterruptedException) throw e; java.util.logging.Logger.getLogger("io.qt.internal").log(java.util.logging.Level.SEVERE, "", e); } } } } catch (InterruptedException e) { } } void setNoMoreExpected() { synchronized(this.commands) { noMoreExpected = true; this.commands.notifyAll(); } } } private static DeploymentSpec unpackDeploymentSpec(URL deploymentSpec, String jarName, Boolean shouldUnpack) throws ParserConfigurationException, SAXException, IOException{ if(jarName!=null) reporter.report("Unpacking .jar file: '", jarName, "'"); DeploymentSpec spec = null; try { spec = readDeploySpec(deploymentSpec); } catch (DeploymentSpecException e1) { Logger.getLogger("io.qt.internal").warning(String.format("Unable to load native libraries from %1$s: %2$s", (jarName==null ? deploymentSpec : jarName), e1.getMessage())); } if (spec==null) return null; File tmpDir = spec.getModule()==null ? jambiTempDirBase(spec) : jambiDeploymentDir; spec.setBaseDir(tmpDir); spec.setBaseUrl(new URL(convertAbsolutePathStringToFileUrlString(tmpDir))); reporter.report(" - using cache directory: '", tmpDir.getAbsolutePath(), "'"); File dummyFile = null; if(shouldUnpack == null) { if(spec.getModule()!=null) { shouldUnpack = Boolean.TRUE; }else { // If the dir exists and contains .dummy, sanity check the contents... dummyFile = new File(tmpDir, ".dummy"); if(spec.getModule()!=null) dummyFile = new File(tmpDir, "."+spec.getModule()+".dummy"); if (dummyFile.exists()) { reporter.report(" - cache directory exists"); shouldUnpack = Boolean.FALSE; } else { shouldUnpack = Boolean.TRUE; } } } // If the dir doesn't exists or it was only half completed, copy the files over... // FIXME: This should be a separate API call to cause loading, we want to be able to load up multiple DeploymentSpec // so we can choose which one we are going to activate (and allow enumeration of them). if (shouldUnpack.booleanValue()) { reporter.report(" - starting to copy content to cache directory..."); String urlBase = deploymentSpec.toString(); int idx = urlBase.indexOf("!/"); if(idx>0) { urlBase = urlBase.substring(0, idx+2); } for (String path : spec.getDirents()) { File outFile = new File(tmpDir, path.replace('/', File.separatorChar)); if(!outFile.exists()) { reporter.report(" - copying over: '", path, "'..."); InputStream in = null; OutputStream out = null; try { URL entryURL = new URL(urlBase+path); try { in = entryURL.openStream(); } catch (Exception e1) { continue; } reporter.report(" - matched url: ", entryURL.toExternalForm()); File outFileDir = outFile.getParentFile(); if (!outFileDir.exists()) { reporter.report(" - creating directory: ", outFileDir.getAbsolutePath()); outFileDir.mkdirs(); } out = new FileOutputStream(outFile); try { reporter.report(" - copying to ", outFile.getAbsolutePath()); copy(in, out); } finally { in = null; // copy() ALWAYS closes it for us out = null; // copy() ALWAYS closes it for us } } finally { if(in != null) { try { in.close(); } catch(IOException eat) { } } } } } Map entries = new TreeMap<>(); ExecutorThread executor = null; switch(spec.getModule()) { case "qtjambi": executor = new ExecutorThread(); executor.start(); break; } try { for (LibraryEntry e : spec.getLibraries()) { entries.put(e.getName(), e); if(!(e instanceof SymlinkEntry)) { String libName = e.getName(); File outFile = new File(tmpDir, libName.replace('/', File.separatorChar)); if(!outFile.exists()) { boolean isplugin = libName.startsWith("plugins/") || libName.startsWith("qml/") || libName.startsWith("translations/"); String _urlBase = urlBase; if(isplugin) { outFile.getParentFile().mkdirs(); pluginLibraries.add(e); } if(executor!=null && ((!libName.contains("QtJambiGui") && !libName.contains("QtJambiWidgets")) || isplugin)) { CompletableFuture future = CompletableFuture.supplyAsync(()->{ reporter.report(" - copying over: '", libName, "'..."); InputStream in = null; OutputStream out = null; try { URL entryURL = new URL(_urlBase+libName); try { in = entryURL.openStream(); } catch (Exception e1) { return null; } File outFileDir = outFile.getParentFile(); if (!outFileDir.exists()) { reporter.report(" - creating directory: ", outFileDir.getAbsolutePath()); outFileDir.mkdirs(); } out = new FileOutputStream(outFile); try { reporter.report(" - copying to ", outFile.getAbsolutePath()); copy(in, out); } finally { in = null; // copy() ALWAYS closes it for us out = null; // copy() ALWAYS closes it for us } } catch (IOException e1) { return e1; } finally { if(in != null) { try { in.close(); } catch(IOException eat) { } } } return null; }, executor); e.addExtractionFunction(()->{ Exception exn = future.get(); if(exn!=null) throw exn; }); }else { e.addExtractionFunction(()->{ reporter.report(" - copying over: '", libName, "'..."); InputStream in = null; OutputStream out = null; try { URL entryURL = new URL(_urlBase+libName); try { in = entryURL.openStream(); } catch (Exception e1) { return; } File outFileDir = outFile.getParentFile(); if (!outFileDir.exists()) { reporter.report(" - creating directory: ", outFileDir.getAbsolutePath()); outFileDir.mkdirs(); } out = new FileOutputStream(outFile); try { reporter.report(" - copying to ", outFile.getAbsolutePath()); copy(in, out); } finally { in = null; // copy() ALWAYS closes it for us out = null; // copy() ALWAYS closes it for us } } finally { if(in != null) { try { in.close(); } catch(IOException eat) { } } } }); } } } } List shiftedLinks = new ArrayList<>(); for (LibraryEntry e : spec.getLibraries()) { if(e instanceof SymlinkEntry) { File outFile = new File(tmpDir, e.getName().replace('/', File.separatorChar)); if(!outFile.exists()) { SymlinkEntry s = (SymlinkEntry)e; File outFileDir = outFile.getParentFile(); if (!outFileDir.exists()) { reporter.report(" - creating directory: ", outFileDir.getAbsolutePath()); outFileDir.mkdirs(); } File target = new File(tmpDir, s.getTarget().replace('/', File.separatorChar)); if(target.exists()) { reporter.report(" - creating symbolic link ", outFile.getAbsolutePath()); Files.createSymbolicLink(outFile.toPath(), outFile.getParentFile().toPath().relativize(target.toPath())); }else { LibraryEntry linkedEntry = entries.get(s.getTarget()); if(linkedEntry!=null) { linkedEntry.addExtractionFunction(()->{ reporter.report(" - creating symbolic link ", outFile.getAbsolutePath()); Files.createSymbolicLink(outFile.toPath(), outFile.getParentFile().toPath().relativize(target.toPath())); s.extract(); }); }else { shiftedLinks.add(s); } } } } } if(!shiftedLinks.isEmpty()) { List _shiftedLinks = new ArrayList<>(); while(!shiftedLinks.isEmpty()) { _shiftedLinks.clear(); for (SymlinkEntry s : shiftedLinks) { File outFile = new File(tmpDir, s.getName().replace('/', File.separatorChar)); File outFileDir = outFile.getParentFile(); if (!outFileDir.exists()) { reporter.report(" - creating directory: ", outFileDir.getAbsolutePath()); outFileDir.mkdirs(); } File target = new File(tmpDir, s.getTarget().replace('/', File.separatorChar)); if(target.exists()) { reporter.report(" - creating symbolic link ", outFile.getAbsolutePath()); Files.createSymbolicLink(outFile.toPath(), outFile.getParentFile().toPath().relativize(target.toPath())); }else { LibraryEntry linkedEntry = entries.get(s.getTarget()); if(linkedEntry!=null) { linkedEntry.addExtractionFunction(()->{ reporter.report(" - creating symbolic link ", outFile.getAbsolutePath()); Files.createSymbolicLink(outFile.toPath(), outFile.getParentFile().toPath().relativize(target.toPath())); s.extract(); }); }else { _shiftedLinks.add(s); } } } if(_shiftedLinks.size()==shiftedLinks.size()) { break; } shiftedLinks.clear(); shiftedLinks.addAll(_shiftedLinks); } } if (dummyFile != null && !dummyFile.createNewFile()) { throw new DeploymentSpecException("Can't create dummy file in cache directory"); } spec.setBaseDir(tmpDir); spec.setBaseUrl(new URL(convertAbsolutePathStringToFileUrlString(tmpDir))); }finally { if(executor!=null) executor.setNoMoreExpected(); } } else if(spec.getBaseUrl() == null) { String path = deploymentSpec.getPath(); int i = path.lastIndexOf('/'); // URL path if(i >= 0) path = path.substring(0, i); spec.setBaseDir(new File(path)); spec.setBaseUrl(new URL(deploymentSpec, path)); } return spec; } private static boolean loadSystemLibrary(String name) { reporter.report(" - trying to load: ", name); File foundFile = null; for(DeploymentSpec deploymentSpec : deploymentSpec) { File f = new File(deploymentSpec.getBaseDir(), name); if(f.isFile()) { foundFile = f; break; } } if(foundFile == null) return false; Runtime rt = Runtime.getRuntime(); rt.load(foundFile.getAbsolutePath()); reporter.report(" - ok!"); return true; } private static String jambiLibraryName(String prefix, String lib, int[] version, List replacements) { if(prefix!=null) lib = prefix + lib; switch (operatingSystem) { case Windows: if(version.length>0) replacements.add(""+version[0]); replacements.add(""); break; case MacOSX: switch(version.length) { default: replacements.add("."+version[0]); replacements.add("."+version[0]+"."+version[1]); replacements.add("."+version[0]+"."+version[1]+"."+version[2]); replacements.add(""); break; case 2: replacements.add("."+version[0]); replacements.add("."+version[0]+"."+version[1]); replacements.add(""); break; case 1: replacements.add("."+version[0]); case 0: replacements.add(""); break; } break; default: switch(version.length) { default: replacements.add("."+version[0]+"."+version[1]+"."+version[2]); case 2: replacements.add("."+version[0]+"."+version[1]); case 1: replacements.add("."+version[0]); case 0: replacements.add(""); break; } break; } switch (operatingSystem) { case Windows: { if (configuration == Configuration.Debug) lib += "d"; return lib + "%1$s.dll"; // "foobar1.dll" } case MacOSX: if (configuration == Configuration.Debug) lib += "_debug"; return "lib" + lib + "%1$s.jnilib"; case Linux: case Android: if (configuration == Configuration.Debug) lib += "_debug"; return "lib" + lib + ".so%1$s"; default: break; } throw new RuntimeException("Unreachable statement"); } private static String qtLibraryName(String qtprefix, String lib, String libInfix, int[] version, String versionStrg, Configuration configuration, List replacements) { if(libInfix==null){ libInfix = ""; } String _lib = lib; if(qtprefix!=null && version!=null && version.length>0){ _lib = qtprefix + version[0] + lib; } String prefix = _lib.startsWith("lib") ? "" : "lib"; switch (operatingSystem) { case Windows: replacements.add(""); break; default: if(version!=null) { switch(version.length) { default: replacements.add("."+version[0]+"."+version[1]+"."+version[2]); case 2: replacements.add("."+version[0]+"."+version[1]); case 1: replacements.add("."+version[0]); case 0: replacements.add(""); break; } }else if(versionStrg!=null) { replacements.add("."+versionStrg); replacements.add(""); }else { replacements.add(""); } break; } switch (operatingSystem) { case Windows: return configuration == Configuration.Debug && !isMinGWBuilt ? _lib + libInfix + "d.dll" // "QtFood4.dll" : _lib + libInfix + ".dll"; // "QtFoo4.dll" case MacOSX: if(dontUseFrameworks==null || !dontUseFrameworks) { if(version!=null && version.length>0 && version[0]<6) return String.format("%1$s%2$s.framework/Versions/%3$s/%1$s%2$s", qtprefix, lib, version[0]); else return String.format("%1$s%2$s.framework/Versions/A/%1$s%2$s", qtprefix, lib); }else { if(version!=null && version.length>1 && version[0]>=5 && version[1]>=14) { return prefix + _lib + libInfix + "%1$s.dylib"; }else { return configuration == Configuration.Debug ? prefix + _lib + libInfix + "_debug%1$s.dylib" // "libQtFoo_debug.4.dylib" : prefix + _lib + libInfix + "%1$s.dylib"; // "libQtFoo.4.dylib" } } case Linux: case Android: // Linux doesn't have a dedicated "debug" library since 4.2 return prefix + _lib + libInfix + ".so%1$s"; // "libQtFoo.so.4" default: break; } throw new RuntimeException("Unreachable statement"); } /** * Copies the data in the inputstream into the output stream. * @param in The source. * @param out The destination. * * @throws IOException when there is a problem... */ private static boolean copy(InputStream in, OutputStream out) throws IOException { boolean bf = false; try { byte buffer[] = new byte[1024 * 8]; int n; while((n = in.read(buffer)) > 0) { out.write(buffer, 0, n); } out.close(); out = null; in.close(); in = null; bf = true; } finally { if(out != null) { try { out.close(); } catch(IOException eat) { } out = null; } if(in != null) { try { in.close(); } catch(IOException eat) { } in = null; } } return bf; } static boolean isUsingDeploymentSpec() { return !deploymentSpec.isEmpty(); } // This is implemented so that String#replace(char, char) is not used as this is regexp based // and sounds heavy weight for us. Maybe I should performance test this point before just // implementing this but bleh! private static String stringCharReplace(String s, char fromChar, char toChar) { final int len = s.length(); StringBuilder sb = new StringBuilder(); for(int i = 0; i < len; i++) { char c = s.charAt(i); if(c == fromChar) sb.append(toChar); else sb.append(c); } return sb.toString(); } private static String convertAbsolutePathStringToFileUrlString(String pathString) { if(File.separatorChar == '\\') pathString = stringCharReplace(pathString, '\\', '/'); // windows String schemePrefix; if(pathString.length() > 0 && pathString.charAt(0) != '/') schemePrefix = "file:///"; // empty authority part, ensure leading / in path part else schemePrefix = "file://"; // empty authority part, pathString already has leading / return schemePrefix + pathString; } private static String convertAbsolutePathStringToFileUrlString(File file) { return convertAbsolutePathStringToFileUrlString(file.getAbsolutePath()); } private static List mergeJniLibdir(List middle) { if((jniLibdirBeforeList != null && !jniLibdirBeforeList.isEmpty()) || (jniLibdirList != null && !jniLibdirList.isEmpty())) { List newList = new ArrayList(); if(jniLibdirBeforeList != null) newList.addAll(jniLibdirBeforeList); newList.addAll(middle); if(jniLibdirList != null) newList.addAll(jniLibdirList); middle = newList; } Set set = new TreeSet<>(); for (int i = 0; i < middle.size(); ++i) { String a = middle.get(i); if(!set.contains(a)) set.add(a); else { middle.remove(i); --i; } } return middle; } static void loadSystemLibraries() { if (systemLibrariesList != null) { for (String s : systemLibrariesList) { // FIXME: We want only append the suffix (no prefix, no Qt version, no debug extra) // currently is does add a prefix (maybe also debug extra). NativeLibraryManager.loadSystemLibrary(s); } } } /** * Converts the string version number into an array of * @param versionString Version string such as "1.2.3" * @return Array of new int[] { 1, 2, 3 }; * @throws NumberFormatException */ static int[] getVersion(String versionString) { String[] sA = versionString.split("\\."); int[] vA = new int[sA.length]; int i = 0; for(String s : sA) vA[i++] = Integer.valueOf(s); return vA; } /** * Enum for defining the operation system. */ enum OperatingSystem { Windows, MacOSX, Linux, Android } /** * Enum for defining whether Qt is build in Release or Debug. */ enum Configuration { Release, Debug, Test; } static String qtJambiLibraryPath() { return qtJambiLibraryPath; } static String qtLibraryPath() { return qtLibraryPath; } } //!!NOTE!! This class can have no dependencies on Qt since //it is required for loading the libraries. class DeploymentSpecException extends RuntimeException { private static final long serialVersionUID = 1L; public DeploymentSpecException(String msg) { super(msg); } } class WrongSystemException extends DeploymentSpecException { private static final long serialVersionUID = 1L; public WrongSystemException(String msg) { super(msg); } } class WrongVersionException extends DeploymentSpecException { private static final long serialVersionUID = 1L; public WrongVersionException(String msg) { super(msg); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy