sun.net.dns.ResolverConfigurationImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2002, 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package sun.net.dns;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
/*
* An implementation of sun.net.ResolverConfiguration for Windows.
*/
public class ResolverConfigurationImpl
extends ResolverConfiguration
{
// Lock held whilst loading configuration or checking
private static Object lock = new Object();
// Resolver options
private final Options opts;
// Addresses have changed. We default to true to make sure we
// resolve the first time it is requested.
private static boolean changed = true;
// Time of last refresh.
private static long lastRefresh;
// Cache timeout (120 seconds) - should be converted into property
// or configured as preference in the future.
private static final long TIMEOUT_NANOS = TimeUnit.SECONDS.toNanos(120);
// DNS suffix list and name servers populated by native method
private static String os_searchlist;
private static String os_nameservers;
// Cached lists
private static ArrayList searchlist;
private static ArrayList nameservers;
// Parse string that consists of token delimited by comma
// and return ArrayList. Refer to ResolverConfigurationImpl.c and
// strappend to see how the string is created.
private ArrayList stringToList(String str) {
// String is delimited by comma.
String[] tokens = str.split(",");
ArrayList l = new ArrayList<>(tokens.length);
for (String s : tokens) {
if (!s.isEmpty() && !l.contains(s)) {
l.add(s);
}
}
l.trimToSize();
return l;
}
// Parse string that consists of token delimited by comma
// and return ArrayList. Refer to ResolverConfigurationImpl.c and
// strappend to see how the string is created.
// In addition to splitting the string, converts IPv6 addresses to
// BSD-style.
private ArrayList addressesToList(String str) {
// String is delimited by comma
String[] tokens = str.split(",");
ArrayList l = new ArrayList<>(tokens.length);
for (String s : tokens) {
if (!s.isEmpty()) {
if (s.indexOf(':') >= 0 && s.charAt(0) != '[') {
// Not BSD style
s = '[' + s + ']';
}
if (!s.isEmpty() && !l.contains(s)) {
l.add(s);
}
}
}
l.trimToSize();
return l;
}
// Load DNS configuration from OS
private void loadConfig() {
assert Thread.holdsLock(lock);
// A change in the network address of the machine usually indicates
// a change in DNS configuration too so we always refresh the config
// after such a change.
if (changed) {
changed = false;
} else {
// Otherwise we refresh if TIMEOUT_NANOS has passed since last
// load.
long currTime = System.nanoTime();
// lastRefresh will always have been set once because we start with
// changed = true.
if ((currTime - lastRefresh) < TIMEOUT_NANOS) {
return;
}
}
// Native code that uses Windows API to find out the DNS server
// addresses and search suffixes. It builds a comma-delimited string
// of nameservers and domain suffixes and sets them to the static
// os_nameservers and os_searchlist. We then split these into Java
// Lists here.
loadDNSconfig0();
// Record the time of update and refresh the lists of addresses /
// domain suffixes.
lastRefresh = System.nanoTime();
searchlist = stringToList(os_searchlist);
nameservers = addressesToList(os_nameservers);
os_searchlist = null; // can be GC'ed
os_nameservers = null;
}
ResolverConfigurationImpl() {
opts = new OptionsImpl();
}
@SuppressWarnings("unchecked") // clone()
public List searchlist() {
synchronized (lock) {
loadConfig();
// List is mutable so return a shallow copy
return (List)searchlist.clone();
}
}
@SuppressWarnings("unchecked") // clone()
public List nameservers() {
synchronized (lock) {
loadConfig();
// List is mutable so return a shallow copy
return (List)nameservers.clone();
}
}
public Options options() {
return opts;
}
// --- Address Change Listener
static class AddressChangeListener extends Thread {
public void run() {
for (;;) {
// wait for configuration to change
if (notifyAddrChange0() != 0)
return;
synchronized (lock) {
changed = true;
}
}
}
}
// --- Native methods --
static native void init0();
static native void loadDNSconfig0();
static native int notifyAddrChange0();
static {
jdk.internal.loader.BootLoader.loadLibrary("net");
init0();
// start the address listener thread
AddressChangeListener thr = new AddressChangeListener();
thr.setDaemon(true);
thr.start();
}
}
/**
* Implementation of {@link ResolverConfiguration.Options}
*/
class OptionsImpl extends ResolverConfiguration.Options {
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy