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

jnr.ffi.provider.jffi.NativeLibrary Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2008-2010 Wayne Meissner
 *
 * This file is part of the JNR project.
 *
 * Licensed 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
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package jnr.ffi.provider.jffi;

import com.kenai.jffi.Library;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import jnr.ffi.LibraryLoader;
import jnr.ffi.LibraryOption;
import jnr.ffi.Platform;
import jnr.ffi.Runtime;

public class NativeLibrary {
    private final List libraryNames;
    private final List searchPaths;
    private final List successfulPaths = new ArrayList<>();
    private final Map options;

    private volatile List nativeLibraries = Collections.emptyList();

    NativeLibrary(Collection libraryNames, Collection searchPaths,
                  Map options) {
        this.libraryNames = Collections.unmodifiableList(new ArrayList<>(libraryNames));
        this.searchPaths = Collections.unmodifiableList(new ArrayList<>(searchPaths));
        this.options = options;
        if (options.containsKey(LibraryOption.LoadNow)) getNativeLibraries();
    }

    private String locateLibrary(String libraryName) {
        return Platform.getNativePlatform().locateLibrary(libraryName, searchPaths, options);
    }

    long getSymbolAddress(String name) {
        for (com.kenai.jffi.Library l : getNativeLibraries()) {
            long address = l.getSymbolAddress(name);
            if (address != 0) {
                return address;
            }
        }
        return 0;
    }

    long findSymbolAddress(String name) {
        long address = getSymbolAddress(name);
        if (address == 0) {
            throw new SymbolNotFoundError(com.kenai.jffi.Library.getLastError());
        }
        return address;
    }

    private synchronized List getNativeLibraries() {
        if (!this.nativeLibraries.isEmpty()) {
            return nativeLibraries;
        }
        return nativeLibraries = loadNativeLibraries();
    }

    private synchronized List loadNativeLibraries() {
        List libs = new ArrayList<>();

        for (String libraryName : libraryNames) {
            if (libraryName == null) continue;
            if (libraryName.equals(LibraryLoader.DEFAULT_LIBRARY)) {
                libs.add(Library.getDefault());
                continue;
            }

            com.kenai.jffi.Library lib;

            // try opening ignoring searchPaths AND any name mapping, so just literal given name
            lib = openLibrary(libraryName, successfulPaths);
            if (lib == null) {
                String path = locateLibrary(libraryName); // try opening with mapping and searchPaths
                if (!libraryName.equals(path)) {
                    lib = openLibrary(path, successfulPaths);
                }
            }
            if (lib == null) {
                throw new UnsatisfiedLinkError(com.kenai.jffi.Library.getLastError() +
                        "\nLibrary names\n" + libraryNames.toString() +
                        "\nSearch paths:\n" + searchPaths.toString());
            }
            libs.add(lib);
        }
        putLibraryIntoRuntime(); // successfulPaths have been set and library has been loaded

        return Collections.unmodifiableList(libs);
    }

    private static final Pattern BAD_ELF = Pattern.compile("(.*): (invalid ELF header|file too short|invalid file format)");
    private static final Pattern ELF_GROUP = Pattern.compile("GROUP\\s*\\(\\s*(\\S*).*\\)");

    private static com.kenai.jffi.Library openLibrary(String path, List successfulPaths) {
        com.kenai.jffi.Library lib;

        lib = com.kenai.jffi.Library.getCachedInstance(path, com.kenai.jffi.Library.LAZY | com.kenai.jffi.Library.GLOBAL);
        if (lib != null) {
            successfulPaths.add(path);
            return lib;
        }

        // If dlopen() fails with 'invalid ELF header', then it is likely to be a ld script - parse it for the real library path
        Matcher badElf = BAD_ELF.matcher(com.kenai.jffi.Library.getLastError());
        if (badElf.lookingAt()) {
            File f = new File(badElf.group(1));
            if (f.isFile() && f.length() < (4 * 1024)) {
                Matcher sharedObject = ELF_GROUP.matcher(readAll(f));
                if (sharedObject.find()) {
                    lib = com.kenai.jffi.Library.getCachedInstance(sharedObject.group(1), com.kenai.jffi.Library.LAZY | com.kenai.jffi.Library.GLOBAL);
                    if (lib != null) successfulPaths.add(path);
                    return lib;
                }
            }
        }

        return null;
    }

    private static String readAll(File f) {
        BufferedReader br = null;
        try {
            br = new BufferedReader(new InputStreamReader(new FileInputStream(f)));
            StringBuilder sb = new StringBuilder();
            String line;
            while ((line = br.readLine()) != null) {
                sb.append(line);
            }
            return sb.toString();
        } catch (FileNotFoundException e) {
            throw new RuntimeException(e);

        } catch (IOException ioe) {
            throw new RuntimeException(ioe);

        } finally {
            if (br != null) try { br.close(); } catch (IOException e) { throw new RuntimeException(e); }
        }
    }

    private void putLibraryIntoRuntime() {
        if (Runtime.getSystemRuntime() instanceof NativeRuntime) {
            ((NativeRuntime) Runtime.getSystemRuntime())
                    .loadedLibraries.put(this, new LoadedLibraryData(libraryNames, searchPaths, successfulPaths));
        }
    }

    /**
     * Data class containing information about a loaded native library.
     *
     * A list of all currently loaded libraries can be queried using {@link Runtime#getLoadedLibraries()} which will
     * return a list of {@link LoadedLibraryData}s.
     */
    public static class LoadedLibraryData {

        private final List libraryNames;
        private final List searchPaths;
        private final List successfulPaths;

        LoadedLibraryData(List libraryNames, List searchPaths, List successfulPaths) {
            this.libraryNames = Collections.unmodifiableList(libraryNames);
            this.searchPaths = Collections.unmodifiableList(searchPaths);
            this.successfulPaths = Collections.unmodifiableList(successfulPaths);
        }

        /**
         * @return the list of library names that were provided when this library was loaded
         */
        public List getLibraryNames() {
            return libraryNames;
        }

        /**
         * @return the list of paths that were used to search for the library, custom paths will always appear before
         *         any system default paths
         */
        public List getSearchPaths() {
            return searchPaths;
        }

        /**
         * @return the list of absolute paths of the loaded library files (.so, .dylib, .dll etc) that were actually
         *         loaded, these are the native library files that are being used for this library
         */
        public List getSuccessfulPaths() {
            return successfulPaths;
        }

        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (!(o instanceof LoadedLibraryData)) return false;
            LoadedLibraryData that = (LoadedLibraryData) o;
            return Objects.equals(libraryNames, that.libraryNames) &&
                    Objects.equals(searchPaths, that.searchPaths) &&
                    Objects.equals(successfulPaths, that.successfulPaths);
        }

        @Override
        public int hashCode() {
            return Objects.hash(libraryNames, searchPaths, successfulPaths);
        }

        @Override
        public String toString() {
            return "LoadedLibraryData {" +
                    "libraryNames=" + libraryNames +
                    ", searchPaths=" + searchPaths +
                    ", successfulPaths=" + successfulPaths +
                    '}';
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy