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

jnr.netdb.NativeProtocolsDB Maven / Gradle / Ivy

There is a newer version: 1.2.0
Show newest version
/*
 * Copyright (C) 2010 Wayne Meissner
 *
 * This file is part of jnr.
 *
 * 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.netdb;

import jnr.ffi.*;
import jnr.ffi.Runtime;
import jnr.ffi.annotations.Direct;

import java.util.ArrayList;

import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;

import static jnr.ffi.Platform.OS.*;

/**
 *
 */
abstract class NativeProtocolsDB implements ProtocolsDB {

    public static final NativeProtocolsDB getInstance() {
        return SingletonHolder.INSTANCE;
    }

    private static final class SingletonHolder {
        public static final NativeProtocolsDB INSTANCE = load();
    }

    private static final NativeProtocolsDB load() {
        try {
            Platform.OS os = Platform.getNativePlatform().getOS();

            // The protoent struct is only known to match on Windows, MacOSX, Linux, Solaris.
            // We assume FreeBSD and NetBSD also match.
            if (!(os.equals(DARWIN) || (os.equals(WINDOWS) && Platform.getNativePlatform().getCPU() == Platform.CPU.I386)
                    || os.equals(LINUX) || os.equals(SOLARIS)
                    || os.equals(FREEBSD) || os.equals(NETBSD))) {
                return null;
            }

            LibProto lib;
            if (os.equals(WINDOWS)) {
                Map options = new HashMap();
                options.put(LibraryOption.CallingConvention, CallingConvention.STDCALL);
                lib = Library.loadLibrary(LibProto.class, options, "Ws2_32");
            } else {
                String[] libnames = os.equals(SOLARIS)
                        ? new String[]{"socket", "nsl", "c"}
                        : new String[]{"c"};
                lib = os.equals(LINUX)
                    ? Library.loadLibrary(LinuxLibProto.class, libnames)
                    : Library.loadLibrary(LibProto.class, libnames);
            }

            NativeProtocolsDB protocolsDB = os.equals(LINUX)
                    ? new LinuxNativeProtocolsDB((LinuxLibProto) lib)
                    : new DefaultNativeProtocolsDB(lib);
            // Try to lookup a protocol to make sure the library loaded and found the functions
            protocolsDB.getProtocolByName("ip");
            protocolsDB.getProtocolByNumber(0);
            
            return protocolsDB;
        } catch (Throwable t) {
            Logger.getLogger(NativeProtocolsDB.class.getName()).log(Level.WARNING, "Failed to load native protocols db", t);
            return null;
        }
    }

    public static class UnixProtoent extends jnr.ffi.Struct {
        public final String name = new UTF8StringRef();
        public final Pointer aliases = new Pointer();
        public final Signed32 proto = new Signed32();

        public UnixProtoent(jnr.ffi.Runtime runtime) {
            super(runtime);
        }
    }

    public static interface LibProto {
        UnixProtoent getprotobyname(String name);
        UnixProtoent getprotobynumber(int proto);
        UnixProtoent getprotoent();
        void setprotoent(int stayopen);
        void endprotoent();
    }

    public static interface LinuxLibProto extends LibProto{
        int getprotobyname_r(String proto, @Direct UnixProtoent protoent, Pointer buf, NativeLong buflen, Pointer result);
        int getprotobynumber_r(int proto, @Direct UnixProtoent protoent, Pointer buf, NativeLong buflen, Pointer result);
        int getprotoent_r(@Direct UnixProtoent protoent, Pointer buf, NativeLong buflen, Pointer result);
    }

    static Protocol protocolFromNative(UnixProtoent p) {
        if (p == null) {
            return null;
        }

        List emptyAliases = Collections.emptyList();

        Pointer ptr;
        final Collection aliases = ((ptr = p.aliases.get()) != null)
                ? StringUtil.getNullTerminatedStringArray(ptr) : emptyAliases;

        return new Protocol(p.name.get(), (short) p.proto.get(), aliases);
    }

    static final class DefaultNativeProtocolsDB extends NativeProtocolsDB {
        private final LibProto lib;

        DefaultNativeProtocolsDB(LibProto lib) {
            this.lib = lib;
        }

        public synchronized Protocol getProtocolByName(String name) {
            return protocolFromNative(lib.getprotobyname(name));
        }

        public synchronized Protocol getProtocolByNumber(Integer proto) {
            return protocolFromNative(lib.getprotobynumber(proto));
        }

        public synchronized Collection getAllProtocols() {
            UnixProtoent p;
            List allProtocols = new ArrayList();

            lib.setprotoent(0);
            try {
                while ((p = lib.getprotoent()) != null) {
                    allProtocols.add(protocolFromNative(p));
                }
            } finally {
                lib.endprotoent();
            }

            return allProtocols;
        }
    }

    static final class LinuxNativeProtocolsDB extends NativeProtocolsDB {
        private static final int BUFLEN = 4096;
        private final Runtime runtime;
        private final Pointer buf;
        private final LinuxLibProto lib;


        LinuxNativeProtocolsDB(LinuxLibProto lib) {
            this.lib = lib;
            this.runtime = Library.getRuntime(lib);
            this.buf = Memory.allocateDirect(runtime, BUFLEN);
        }

        public synchronized Protocol getProtocolByName(String name) {
            UnixProtoent protoent = new UnixProtoent(runtime);
            Pointer result = Memory.allocateDirect(runtime, runtime.addressSize());
            if (lib.getprotobyname_r(name, protoent, buf, new NativeLong(BUFLEN), result) == 0) {
                return result.getPointer(0) != null ? protocolFromNative(protoent) : null;
            }

            throw new RuntimeException("getprotobyname_r failed");
        }

        public synchronized Protocol getProtocolByNumber(Integer number) {
            UnixProtoent protoent = new UnixProtoent(runtime);
            Pointer result = Memory.allocateDirect(runtime, runtime.addressSize());
            if (lib.getprotobynumber_r(number, protoent, buf, new NativeLong(BUFLEN), result) == 0) {
                return result.getPointer(0) != null ? protocolFromNative(protoent) : null;
            }

            throw new RuntimeException("getprotobynumber_r failed");
        }

        public synchronized Collection getAllProtocols() {
            UnixProtoent p = new UnixProtoent(runtime);
            List allProtocols = new ArrayList();
            Pointer result = Memory.allocateDirect(runtime, runtime.addressSize());
            NativeLong buflen = new NativeLong(BUFLEN);

            lib.setprotoent(0);
            try {
                while (lib.getprotoent_r(p, buf, buflen, result) == 0 && result.getPointer(0) != null) {
                    allProtocols.add(protocolFromNative(p));
                }
            } finally {
                lib.endprotoent();
            }

            return allProtocols;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy