blob: fddeee82c14bcb87b12989ef489c82765e2e3d1a [file] [log] [blame]
/*
* Copyright 2017 The gRPC Authors
*
* 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 io.grpc.internal;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* A {@link LongCounter} that is implemented with a JDK8 {@link LongAdder}. Instantiates the object
* and invokes methods reflectively to avoid a compile time dependency on LongAdder.
*/
public final class ReflectionLongAdderCounter implements LongCounter {
private static final Logger logger = Logger.getLogger(ReflectionLongAdderCounter.class.getName());
private static final Constructor<?> defaultConstructor;
private static final Method addMethod;
private static final Method sumMethod;
private static final RuntimeException initializationException;
private final Object instance;
static {
Class<?> klass = null;
Constructor<?> defaultConstructorLookup = null;
Method addMethodLookup = null;
Method sumMethodLookup = null;
Throwable caught = null;
try {
klass = Class.forName("java.util.concurrent.atomic.LongAdder");
addMethodLookup = klass.getMethod("add", Long.TYPE);
sumMethodLookup = klass.getMethod("sum");
Constructor<?>[] constructors = klass.getConstructors();
for (Constructor<?> ctor : constructors) {
if (ctor.getParameterTypes().length == 0) {
defaultConstructorLookup = ctor;
break;
}
}
} catch (Throwable e) {
logger.log(
Level.FINE,
"LongAdder can not be found via reflection, this is normal for JDK7 and below",
e);
caught = e;
}
if (caught == null && defaultConstructorLookup != null) {
defaultConstructor = defaultConstructorLookup;
addMethod = addMethodLookup;
sumMethod = sumMethodLookup;
initializationException = null;
} else {
defaultConstructor = null;
addMethod = null;
sumMethod = null;
initializationException = new RuntimeException(caught);
}
}
ReflectionLongAdderCounter() {
if (initializationException != null) {
throw initializationException;
}
try {
instance = defaultConstructor.newInstance();
} catch (InstantiationException e) {
throw new RuntimeException(e);
} catch (IllegalAccessException e) {
throw new RuntimeException(e);
} catch (InvocationTargetException e) {
throw new RuntimeException(e);
}
}
/**
* Returns true if the environment supports LongAdder. In other words, we are running in >= JDK8.
*/
static boolean isAvailable() {
return initializationException == null;
}
private static final Object[] one = new Object[] { 1L };
@Override
public void add(long delta) {
try {
addMethod.invoke(instance, delta == 1L ? one : new Object[] { delta });
} catch (IllegalAccessException e) {
throw new RuntimeException(e);
} catch (InvocationTargetException e) {
throw new RuntimeException(e);
}
}
@Override
public long value() {
try {
return (Long) sumMethod.invoke(instance);
} catch (IllegalAccessException e) {
throw new RuntimeException();
} catch (InvocationTargetException e) {
throw new RuntimeException();
}
}
}