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

jnr.ffi.Platform Maven / Gradle / Ivy

There is a newer version: 2.2.16
Show 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;

import java.io.File;
import java.io.FilenameFilter;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public abstract class Platform {
    private static final java.util.Locale LOCALE = java.util.Locale.ENGLISH;
    private final OS os;
    private final CPU cpu;
    private final int addressSize;
    private final int longSize;
    protected final Pattern libPattern;

    private static final class SingletonHolder {
        static final Platform PLATFORM = determinePlatform();
    }

    /**
     * The common names of supported operating systems.
     */
    public enum OS {
        /*
         * Note The names of the enum values are used in other parts of the
         * code to determine where to find the native stub library.  Do not rename.
         */

        /** MacOSX */
        DARWIN,
        /** FreeBSD */
        FREEBSD,
        /** NetBSD */
        NETBSD,
        /** OpenBSD */
        OPENBSD,
        /** DragonFly */
        DRAGONFLY,
        /** Linux */
        LINUX,
        /** Solaris (and OpenSolaris) */
        SOLARIS,
        /** The evil borg operating system */
        WINDOWS,
        /** IBM AIX */
        AIX,
        /** IBM i */
        IBMI,
        /** IBM zOS **/
        ZLINUX,
        /** MidnightBSD **/
        MIDNIGHTBSD,
        /** No idea what the operating system is */
        UNKNOWN;

        @Override
        public String toString() {
            return name().toLowerCase(LOCALE);
        }
    }

    /**
     * The supported CPU architectures.
     */
    public enum CPU {
        /*
         * Note The names of the enum values are used in other parts of the
         * code to determine where to find the native stub library.  Do NOT rename.
         */

        /** 32 bit legacy Intel */
        I386,

        /** 64 bit AMD (aka EM64T/X64) */
        X86_64,

        /** 32 bit Power PC */
        PPC,

        /** 64 bit Power PC */
        PPC64,

        /** 64 bit Power PC little endian */
        PPC64LE,

        /** 32 bit Sun sparc */
        SPARC,

        /** 64 bit Sun sparc */
        SPARCV9,

        /** IBM zSeries S/390 */
        S390X,

        /** 32 bit MIPS (used by nestedvm) */
        MIPS32,

        /** 32 bit ARM */
        ARM,

        /** 64 bit ARM */
        AARCH64,

        /** 64 bit MIPS */
        MIPS64EL,

        /** 64 bit LOONGARCH */
        LOONGARCH64,

        /**
         * Unknown CPU architecture.  A best effort will be made to infer architecture
         * specific values such as address and long size.
         */
        UNKNOWN;

        /**
         * Returns a {@code String} object representing this {@code CPU} object.
         *
         * @return the name of the cpu architecture as a lower case {@code String}.
         */
        @Override
        public String toString() {
            return name().toLowerCase(LOCALE);
        }
    }

    /**
     * Determines the operating system jffi is running on
     *
     * @return An member of the OS enum.
     */
    private static OS determineOS() {
        String osName = System.getProperty("os.name").split(" ")[0];
        if (startsWithIgnoreCase(osName, "mac") || startsWithIgnoreCase(osName, "darwin")) {
            return OS.DARWIN;
        } else if (startsWithIgnoreCase(osName, "linux")) {
            return OS.LINUX;
        } else if (startsWithIgnoreCase(osName, "sunos") || startsWithIgnoreCase(osName, "solaris")) {
            return OS.SOLARIS;
        } else if (startsWithIgnoreCase(osName, "aix")) {
            return OS.AIX;
        } else if (startsWithIgnoreCase(osName, "os400") || startsWithIgnoreCase(osName, "os/400")) {
            return OS.IBMI;
        } else if (startsWithIgnoreCase(osName, "openbsd")) {
            return OS.OPENBSD;
        } else if (startsWithIgnoreCase(osName, "freebsd")) {
            return OS.FREEBSD;
        } else if (startsWithIgnoreCase(osName, "dragonfly")) {
            return OS.DRAGONFLY;
        } else if (startsWithIgnoreCase(osName, "windows")) {
            return OS.WINDOWS;
        } else if (startsWithIgnoreCase(osName, "midnightbsd")) {
            return OS.MIDNIGHTBSD;
        } else {
            return OS.UNKNOWN;
        }
    }

    /**
     * Determines the Platform that best describes the OS
     *
     * @param os The operating system.
     * @return An instance of Platform
     */
    private static Platform determinePlatform(OS os) {
        switch (os) {
            case DARWIN:
                return new Darwin();
            case LINUX:
                return new Linux();
            case WINDOWS:
                return new Windows();
            case IBMI:
                return new IbmI();
            case UNKNOWN:
                return new Unsupported(os);
            default:
                return new Default(os);
        }
    }

    private static Platform determinePlatform() {
        String providerName = System.getProperty("jnr.ffi.provider");
        try {
            Class c = Class.forName(providerName + "$Platform");
            return (Platform) c.newInstance();
        } catch (ClassNotFoundException ex) {
            return determinePlatform(determineOS());
        } catch (IllegalAccessException ex) {
            throw new ExceptionInInitializerError(ex);
        } catch (InstantiationException ex) {
            throw new ExceptionInInitializerError(ex);
        }
    }
    
    private static CPU determineCPU() {
        String archString = System.getProperty("os.arch");
        if (equalsIgnoreCase("x86", archString) ||
                equalsIgnoreCase("i386", archString) ||
                equalsIgnoreCase("i86pc", archString) ||
                equalsIgnoreCase("i686", archString)) {
            return CPU.I386;
        } else if (equalsIgnoreCase("x86_64", archString) || equalsIgnoreCase("amd64", archString)) {
            return CPU.X86_64;
        } else if (equalsIgnoreCase("ppc", archString) || equalsIgnoreCase("powerpc", archString)) {
            if (OS.IBMI.equals(determineOS()))
                return CPU.PPC64;
            return CPU.PPC;
        } else if (equalsIgnoreCase("ppc64", archString) || equalsIgnoreCase("powerpc64", archString)) {
            if ("little".equals(System.getProperty("sun.cpu.endian"))) {
                return CPU.PPC64LE;
            }
            return CPU.PPC64;
        } else if (equalsIgnoreCase("ppc64le", archString) || equalsIgnoreCase("powerpc64le", archString)) {
            return CPU.PPC64LE;
        } else if (equalsIgnoreCase("s390", archString) || equalsIgnoreCase("s390x", archString)) {
            return CPU.S390X;
        } else if (equalsIgnoreCase("aarch64", archString)) {
            return CPU.AARCH64;
        } else if (equalsIgnoreCase("arm", archString) || equalsIgnoreCase("armv7l", archString)) {
            return CPU.ARM;
        } else if (equalsIgnoreCase("mips64", archString) || equalsIgnoreCase("mips64el", archString)) {
            return CPU.MIPS64EL;
        } else if (equalsIgnoreCase("loongarch64", archString)) {
           return CPU.LOONGARCH64;
        }

        // Try to find by lookup up in the CPU list
        for (CPU cpu : CPU.values()) {
            if (equalsIgnoreCase(cpu.name(), archString)) {
                return cpu;
            }
        }

        return CPU.UNKNOWN;
    }

    public Platform(OS os, CPU cpu, int addressSize, int longSize, String libPattern) {
        this.os = os;
        this.cpu = cpu;
        this.addressSize = addressSize;
        this.longSize = longSize;
        this.libPattern = Pattern.compile(libPattern);
    }
    
    private Platform(OS os) {
        this.os = os;
        this.cpu = determineCPU();
        
        String libpattern;
        switch (os) {
            case WINDOWS:
                libpattern = ".*\\.dll$";
                break;
            case DARWIN:
                libpattern = "lib.*\\.(dylib|jnilib)$";
                break;
            case IBMI:
                libpattern = "lib.*\\.(so|a\\(shr.o\\)|a\\(shr_64.o\\)|a|so.[\\.0-9]+)$";
                break;
            default:
                libpattern = "lib.*\\.so.*$";
                break;
        }
        libPattern = Pattern.compile(libpattern);

        this.addressSize = calculateAddressSize(cpu);
        this.longSize = os == OS.WINDOWS ? 32 : addressSize;
    }

    private static int calculateAddressSize(CPU cpu) {
        Integer dataModel = Integer.getInteger("sun.arch.data.model");
        if (dataModel == null || dataModel != 32 && dataModel != 64) {
            switch (cpu) {
                case I386:
                case PPC:
                case SPARC:
                    dataModel = 32;
                    break;
                case X86_64:
                case PPC64:
                case PPC64LE:
                case SPARCV9:
                case S390X:
                case AARCH64:
		case MIPS64EL:
		case LOONGARCH64:
                    dataModel = 64;
                    break;
                default:
                    throw new ExceptionInInitializerError("Cannot determine cpu address size");
            }
        }

        return dataModel;
    }

    /**
     * Gets the native Platform
     *
     * @return The current platform.
     */
    public static Platform getNativePlatform() {
        return SingletonHolder.PLATFORM;
    }

    @Deprecated
    public static Platform getPlatform() {
        return SingletonHolder.PLATFORM;
    }

    /**
     * Gets the current Operating System.
     *
     * @return A OS value representing the current Operating System.
     */
    public final OS getOS() {
        return os;
    }

    /**
     * Gets the current processor architecture the JVM is running on.
     *
     * @return A CPU value representing the current processor architecture.
     */
    public final CPU getCPU() {
        return cpu;
    }
    
    public final boolean isBSD() {
        return os == OS.FREEBSD || os == OS.OPENBSD || os == OS.NETBSD || os == OS.DARWIN || os == OS.DRAGONFLY | os == OS.MIDNIGHTBSD;
    }
    public final boolean isUnix() {
        return os != OS.WINDOWS;
    }

    /**
     * Gets the size of a C 'long' on the native platform.
     *
     * @return the size of a long in bits
     * @deprecated Use {@link Runtime#longSize()} instead.
     */
    public final int longSize() {
        return longSize;
    }

    /**
     * Gets the size of a C address/pointer on the native platform.
     *
     * @return the size of a pointer in bits
     * @deprecated Use {@link Runtime#addressSize()} instead.
     */
    public final int addressSize() {
        return addressSize;
    }

    /**
     * @return true if this platform is 32 bit, else false
     * Note: do not use this to determine long size, instead use {@link Runtime#longSize()}
     */
    public final boolean is32Bit() {
        return addressSize == 32;
    }

    /**
     * @return true if this platform is 64 bit, else false
     * Note: do not use this to determine long size, instead use {@link Runtime#longSize()}
     */
    public final boolean is64Bit() {
        return addressSize == 64;
    }

    /**
     * Returns true if the current platform is little endian
     * @return true if little endian, false otherwise or if cannot determine
     */
    public final boolean isLittleEndian() {
        return "little".equals(System.getProperty("sun.cpu.endian"));
    }

    /**
     * Returns true if the current platform is big endian
     * @return true if big endian, false otherwise or if cannot determine
     */
    public final boolean isBigEndian() {
        return "big".equals(System.getProperty("sun.cpu.endian"));
    }

    /**
     * @return the String representing the OS name from the System property {@code os.name} or null if none was found
     * This is not the same as {@link #getOS()} which returns the {@link OS}.
     * For example: Mac OS X is the {@link OS#DARWIN} {@link OS} but returns  "Mac OS X" from this method
     */
    public final String getOSName() {
        return System.getProperty("os.name", null);
    }

    /**
     * Gets the name of this Platform.
     *
     * @return The name of this platform.
     */
    public String getName() {
        return cpu + "-" + os;
    }

    /**
     * Gets the version of this platform as specified by the system property "os.version"
     * @return the String representing the version of this platform, or null if none could be found
     */
    public String getVersion() {
        return System.getProperty("os.version", null);
    }

    /**
     * @return the list of version numbers found from {@link #getVersion()} or an empty list if none were found
     */
    private List getVersionNumbers() {
        String version = getVersion();
        if (version == null) return Collections.emptyList();
        Matcher matcher = Pattern.compile("[\\d]+").matcher(version); // get digits only
        ArrayList result = new ArrayList<>();
        while (matcher.find()) {
            result.add(matcher.group());
        }
        return result;
    }

    /**
     * Gets the number representing the major version of this platform
     * This uses the first number from {@link #getVersion()}
     * @return the number representing the major version of this platform or -1 if none was found
     */
    public int getVersionMajor() {
        List versionNumbers = getVersionNumbers();
        return versionNumbers.size() < 1 ? -1 : Integer.parseInt(versionNumbers.get(0));
    }

    /**
     * Gets the number representing the minor version of this platform
     * This uses the second number from {@link #getVersion()}
     * @return the number representing the minor version of this platform or -1 if none was found
     */
    public int getVersionMinor() {
        List versionNumbers = getVersionNumbers();
        return versionNumbers.size() < 2 ? -1 : Integer.parseInt(versionNumbers.get(1));
    }

    /**
     * Returns the platform specific standard C library name
     * 
     * @return The standard C library name
     */
    public String getStandardCLibraryName() {
        switch (os) {
        case LINUX:
            return "libc.so.6";
        case SOLARIS:
            return "c";
        case DRAGONFLY:
        case FREEBSD:
        case MIDNIGHTBSD:
        case NETBSD:
            return "c";
        case AIX:
        case IBMI:
            return addressSize == 32
                ? "libc.a(shr.o)"
                : "libc.a(shr_64.o)";
        case WINDOWS:
            return "msvcrt";
        default:
            return "c";
        }
    }
    
    /**
     * Maps from a generic library name (e.g. "c") to the platform specific library name.
     *
     * @param libName The library name to map
     * @return The mapped library name.
     */
    public String mapLibraryName(String libName) {
        //
        // A specific version was requested - use as is for search
        //
        if (libPattern.matcher(libName).find()) {
            return libName;
        }
        return System.mapLibraryName(libName);
    }

    /**
     * Searches through a list of directories for a native library.
     *
     * @param libName the base name (e.g. "c") of the library to locate
     * @param libraryPath the list of directories to search
     * @return the path of the library
     */
    public String locateLibrary(String libName, List libraryPath) {
        String mappedName = mapLibraryName(libName);
        for (String path : libraryPath) {
            File libFile = new File(path, mappedName);
            if (libFile.exists()) {
                return libFile.getAbsolutePath();
            }
        }
        // Default to letting the system search for it
        return mappedName;
    }

    /**
     * Searches through a list of directories for a native library.
     *
     * @param libName      the base name (e.g. "c") of the library to locate
     * @param libraryPaths the list of directories to search
     * @param options      map of {@link LibraryOption}s to customize search behavior
     *                     such as {@link LibraryOption#PreferCustomPaths}
     * @return the path of the library
     */
    public String locateLibrary(String libName, List libraryPaths, Map options) {
        return locateLibrary(libName, libraryPaths);
    }

    /**
     * Returns a list of absolute paths to the found locations of a library with the base name {@code libName},
     * if the returned list is empty then the library could not be found and will fail to be loaded as a result.
     *
     * Even if a library is found, this does not guarantee that it will successfully be loaded, it only guarantees
     * that the reason for the failure was not that it was not found.
     *
     * @param libName         the base name (e.g. "c") of the library to locate
     * @param additionalPaths additional paths to search, these take precedence over default paths,
     *                        (as is the behavior in {@link LibraryLoader})
     *                        pass null to only search in the default paths
     * @return the list of absolute paths where the library was found
     */
    public List libraryLocations(String libName, List additionalPaths) {
        ArrayList result = new ArrayList<>();
        ArrayList libDirs = new ArrayList<>();
        if (additionalPaths != null) libDirs.addAll(additionalPaths); // customPaths first!
        libDirs.addAll(LibraryLoader.DefaultLibPaths.PATHS);

        // locateLibrary can either give us an absolute path with the version at the end (for Linux)
        //  or just the name (forwards to mapLibraryName), either way we only want the name, we will
        //  add the parent later from libDirs
        String name = new File(locateLibrary(libName, libDirs)).getName();
        for (String libDir : libDirs) {
            File libFile = new File(libDir, name);
            if (libFile.exists()) {
                result.add(libFile.getAbsolutePath());
            }
        }
        return result;
    }

    private static class Supported extends Platform {
        public Supported(OS os) {
            super(os);
        }
    }

    private static class Unsupported extends Platform {
        public Unsupported(OS os) {
            super(os);
        }
    }

    private static final class Default extends Supported {
        public Default(OS os) {
            super(os);
        }
    }
    /**
     * A {@link Platform} subclass representing the MacOS system.
     */
    private static final class Darwin extends Supported {

        public Darwin() {
            super(OS.DARWIN);
        }

        @Override
        public String mapLibraryName(String libName) {
            //
            // A specific version was requested - use as is for search
            //
            if (libPattern.matcher(libName).find()) {
                return libName;
            }
            return "lib" + libName + ".dylib";
        }

    }

    static final class IbmI extends Supported {
        public IbmI() {
            super(OS.IBMI);
        }

        @Override
        public String mapLibraryName(String libName) {
            //
            // A specific version was requested - use as is for search
            //
            if (libPattern.matcher(libName).find()) {
                return libName;
            }
            return "lib" + libName + ".a(shr_64.o)";
        }
        @Override
        public String locateLibrary(final String libName, List libraryPaths) {
            final Pattern versionedLibPattern = Pattern.compile("lib" + libName + "\\.so((?:\\.[0-9]+)*)$");
            final Pattern dotAorSoPattern = Pattern.compile("lib" + libName + "\\.(a|so)$");
            List dotAorSoFiles = new java.util.LinkedList();

            List searchPaths = new java.util.LinkedList();
            searchPaths.addAll(libraryPaths);
            searchPaths.add("/QOpenSys/pkgs/lib");
            searchPaths.add("/QOpenSys/usr/lib");

            FilenameFilter filter = new FilenameFilter() {
                public boolean accept(File dir, String name) {
                    return dotAorSoPattern.matcher(name).matches() ||  versionedLibPattern.matcher(name).matches() ;
                }
            };
            Map matches = new LinkedHashMap();
            for (String path : searchPaths) {
                if(path.toLowerCase(LOCALE).startsWith("/qsys")) {
                    continue;
                }
                File libraryPath = new File(path);
                File[] files = libraryPath.listFiles(filter);
                if (files == null) {
                    continue;
                }

                for (File file : files) {
                    if (dotAorSoPattern.matcher(file.getName()).matches()) {
                        dotAorSoFiles.add(file);
                        continue;
                    }
                    Matcher matcher = versionedLibPattern.matcher(file.getName());
                    String versionString = matcher.matches() ? matcher.group(1) : "";
                    int[] version;
                    if (versionString == null || versionString.isEmpty()) {
                        version = new int[0];
                    } else {
                        String[] parts = versionString.split("\\.");
                        version = new int[parts.length - 1];
                        for (int i = 1; i < parts.length; i++) {
                            version[i - 1] = Integer.parseInt(parts[i]);
                        }
                    }
                    matches.put(file.getAbsolutePath(), version);
                }
            }

            //
            // Search through the results and return the highest numbered version
            // i.e. libc.so.6 is preferred over libc.so.5
            //
            int[] bestVersion = null;
            String bestMatch = null;
            for (Map.Entry entry : matches.entrySet()) {
                String file = entry.getKey();
                int[] fileVersion = entry.getValue();

                if (Linux.compareVersions(fileVersion, bestVersion) > 0) {
                    bestMatch = file;
                    bestVersion = fileVersion;
                }
            }
            if (null != bestMatch) {
                return bestMatch;
            }
            if (!dotAorSoFiles.isEmpty()) {
                String qualifiedAorSo = dotAorSoFiles.get(0).getAbsolutePath();
                if(qualifiedAorSo.endsWith(".a")) {
                    qualifiedAorSo +="(shr_64.o)";
                }
                return qualifiedAorSo;
            }

            return mapLibraryName(libName);
        }
    }

    /**
     * A {@link Platform} subclass representing the Linux operating system.
     */
    static final class Linux extends Supported {

        // represents a valid library file that matches the search
        private static class Match implements Comparable {
            String path; // absolute path of library file
            int[] version; // version of library, empty if no version specified
            boolean isCustom; // if path is from a custom searchPath specified by user and not default path

            @Override
            public int compareTo(Match o) { // for Collections.sort() to work
                return compareVersions(o.version, this.version);
            }
        }

        public Linux() {
            super(OS.LINUX);
        }

        @Override
        public String locateLibrary(String libName, List libraryPaths) {
            return locateLibrary(libName, libraryPaths, null);
        }

        @Override
        public String locateLibrary(final String libName, List libraryPaths,
                                    Map options) {
            List matches = getMatches(libName, libraryPaths);
            if (matches.isEmpty()) return mapLibraryName(libName); // no matches, default behavior returns mapped name

            boolean preferCustom = options != null && options.containsKey(LibraryOption.PreferCustomPaths);

            Collections.sort(matches); // sort by version, regardless of location

            Match best = null;
            if (preferCustom) {
                for (Match match : matches) {
                    if (match.isCustom) {
                        best = match;
                        break;
                    }
                }
            }
            return best != null ? best.path : matches.get(0).path;
        }

        private List getMatches(String libName, List libraryPaths) {
            List customPaths = new ArrayList<>();
            if (LibraryLoader.DefaultLibPaths.PATHS.size() > 0 &&
                    libraryPaths.size() >= LibraryLoader.DefaultLibPaths.PATHS.size()) {
                // we were probably called by JNR-FFI, customs will always be before system paths
                String firstSystemPath = LibraryLoader.DefaultLibPaths.PATHS.get(0);

                // everything before last occurrence of first system path is custom
                int firstSystemPathIndex = libraryPaths.lastIndexOf(firstSystemPath);
                for (int i = 0; i < firstSystemPathIndex; i++) {
                    customPaths.add(libraryPaths.get(i));
                }
            } else {
                // we were probably called by user and not by JNR-FFI, assume all paths are custom
                customPaths.addAll(libraryPaths);
            }

            Pattern exclude;
            // there are /libx32 directories in wild on ubuntu 14.04 and the
            // oracle-java8-installer package
            if (getCPU() == CPU.X86_64) {
                exclude = Pattern.compile(".*(lib[a-z]*32|i[0-9]86).*"); // ignore 32 bit libs on 64-bit
            } else {
                exclude = Pattern.compile(".*(lib[a-z]*64|amd64|x86_64).*"); // ignore 64 bit libs on 32-bit
            }

            final Pattern versionedLibPattern = Pattern.compile("lib" + libName + "\\.so((?:\\.[0-9]+)*)$");

            FilenameFilter filter = new FilenameFilter() {
                public boolean accept(File dir, String name) {
                    return versionedLibPattern.matcher(name).matches();
                }
            };

            List matches = new ArrayList<>();
            for (String path : libraryPaths) {
                if (exclude.matcher(path).matches()) {
                    continue;
                }

                File libraryPath = new File(path);
                File[] files = libraryPath.listFiles(filter);
                if (files == null) {
                    continue;
                }

                for (File file : files) {
                    Matcher matcher = versionedLibPattern.matcher(file.getName());
                    String versionString = matcher.matches() ? matcher.group(1) : "";
                    int[] version;
                    if (versionString == null || versionString.isEmpty()) {
                        version = new int[0];
                    } else {
                        String[] parts = versionString.split("\\.");
                        version = new int[parts.length - 1];
                        for (int i = 1; i < parts.length; i++) {
                            version[i - 1] = Integer.parseInt(parts[i]);
                        }
                    }
                    Match match = new Match();
                    match.path = file.getAbsolutePath();
                    match.version = version;
                    match.isCustom = customPaths.contains(path);
                    matches.add(match);
                }
            }
            return matches;
        }

        private static int compareVersions(int[] version1, int[] version2) {
            // Null is always smallest
            if (version1 == null) {
                return version2 == null ? 0 : -1;
            }
            if (version2 == null) {
                return 1;
            }

            // Compare component by component
            int commonLength = Math.min(version1.length, version2.length);
            for (int i = 0; i < commonLength; i++) {
                if (version1[i] < version2[i]) {
                    return -1;
                } else if (version1[i] > version2[i]) {
                    return 1;
                }
            }

            // If all components are equal, version with fewest components is smallest
            return Integer.compare(version1.length, version2.length);
        }

        @Override
        public String mapLibraryName(String libName) {
            // Older JDK on linux map 'c' to 'libc.so' which doesn't work
            return "c".equals(libName) || "libc.so".equals(libName)
                    ? "libc.so.6" : super.mapLibraryName(libName);
        }
    }

    /**
     * A {@link Platform} subclass representing the Windows system.
     */
    private static class Windows extends Supported {
        public Windows() {
            super(OS.WINDOWS);
        }

        // This list only includes the Windows versions supported by Java 8+ (our minimum JDK)
        private static final String WINDOWS_SERVER = "server";
        private static final String WINDOWS_VISTA = "windows vista";
        private static final String WINDOWS_7 = "windows 7";
        private static final String WINDOWS_8 = "windows 8";
        private static final String WINDOWS_10 = "windows 10";
        private static final String WINDOWS_11 = "windows 11";

        private String osName() {
            return System.getProperty("os.name").toLowerCase();
        }

        /**
         * @return true if this Windows version is a Windows server version
         */
        public boolean isServer() {
            return osName().contains(WINDOWS_SERVER);
        }

        /**
         * @return true if this Windows version is Windows Vista
         */
        public boolean isVista() {
            return osName().contains(WINDOWS_VISTA);
        }

        /**
         * @return true if this Windows version is Windows 7
         */
        public boolean is7() {
            return osName().contains(WINDOWS_7);
        }

        /**
         * @return true if this Windows version is Windows 8 (or 8.1)
         */
        public boolean is8() {
            return osName().contains(WINDOWS_8);
        }

        /**
         * @return true if this Windows version is Windows 10
         */
        public boolean is10() {
            return osName().contains(WINDOWS_10);
        }

        /**
         * @return true if this Windows versions is Windows 11
         */
        public boolean is11() {
            return osName().contains(WINDOWS_11);
        }
    }

    private static boolean startsWithIgnoreCase(String s1, String s2) {
        return s1.startsWith(s2)
            || s1.toUpperCase(LOCALE).startsWith(s2.toUpperCase(LOCALE))
            || s1.toLowerCase(LOCALE).startsWith(s2.toLowerCase(LOCALE));
    }

    private static boolean equalsIgnoreCase(String s1, String s2) {
        return s1.equalsIgnoreCase(s2)
            || s1.toUpperCase(LOCALE).equals(s2.toUpperCase(LOCALE))
            || s1.toLowerCase(LOCALE).equals(s2.toLowerCase(LOCALE));
    }
}





© 2015 - 2024 Weber Informatics LLC | Privacy Policy