/* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ package com.sun.star.lib.util; import java.io.File; import java.net.URL; import java.net.URLClassLoader; /** * Helper functions to locate and load native files. * *

The methods in this class are designed to find the requested resources in * as many cases as possible. They search various places, roughly from most * specific to most general. This works well if a component is known to bring * with it a certain resource, and that resource has to be found. However, it * might not work very well in cases where you want to check whether a * component brings with it a certain resource or not: a similarly named * resource from another component might be found by the eager search * algorithm.

*/ public final class NativeLibraryLoader { /** * Load a system library, using a given class loader to locate the library. * *

This is similar to System.loadLibrary.

* * @param loader a class loader; may be null. * @param libname the library name; how this name is mapped to a system * library name is system dependent. */ public static void loadLibrary(ClassLoader loader, String libname) { String sysname = System.mapLibraryName(libname); // At least Oracle's 1.7.0_51 now maps to .dylib rather than .jnilib: if (System.getProperty("os.name").startsWith("Mac") && sysname.endsWith(".dylib")) { sysname = sysname.substring(0, sysname.length() - "dylib".length()) + "jnilib"; } File path = getResource(loader, sysname); if (path == null) { // If the library cannot be found as a class loader resource, try // the global System.loadLibrary as a last resort: System.loadLibrary(libname); } else { System.load(path.getAbsolutePath()); } } /** * Locate a system resource, using a given class loader. * *

This is similar to ClassLoader.getResource, but only works * for local resources (local files), and adds additional functionality for * URLClassLoaders.

* * @param loader a class loader; may be null. * @param name a resource name (that is, the name of a file). * @return a File locating the resource, or null if the resource was not * found. */ public static File getResource(ClassLoader loader, String name) { if (loader != null) { File path = UrlToFileMapper.mapUrlToFile(loader.getResource(name)); if (path != null) { return path; } } // URLClassLoaders work on lists of URLs, which are typically URLs // locating JAR files (scheme://auth/dir1/dir2/some.jar). The following // code looks for resource name beside the JAR file // (scheme://auth/dir1/dir2/name) and one directory up // (scheme://auth/dir1/name). The second step is important in a typical // OOo installation, where the JAR files are in the program/classes // directory while the shared libraries are in the program directory. if (!(loader instanceof URLClassLoader)) { return null; } URL[] urls = ((URLClassLoader) loader).getURLs(); for (int i = 0; i < urls.length; ++i) { File path = UrlToFileMapper.mapUrlToFile(urls[i]); if (path != null) { File dir = path.isDirectory() ? path : path.getParentFile(); if (dir != null) { path = new File(dir, name); if (path.exists()) { return path; } dir = dir.getParentFile(); if (dir != null) { path = new File(dir, name); if (path.exists()) { return path; } // On macOS, dir is now the Resources dir, // we want to look in Frameworks if (System.getProperty("os.name").startsWith("Mac") && dir.getName().equals("Resources")) { dir = dir.getParentFile(); path = new File(dir, "Frameworks/" + name); if (path.exists()) { return path; } } } } } } return null; } private NativeLibraryLoader() {} // do not instantiate } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */