Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.onthegomap.planetiler.stats.ProcessInfo Maven / Gradle / Ivy
Go to download
Planetiler is tool to build planet-scale vector tilesets from OpenStreetMap data fast.
package com.onthegomap.planetiler.stats;
import com.onthegomap.planetiler.util.Parse;
import com.sun.management.GarbageCollectionNotificationInfo;
import com.sun.management.GcInfo;
import java.lang.management.BufferPoolMXBean;
import java.lang.management.GarbageCollectorMXBean;
import java.lang.management.ManagementFactory;
import java.lang.management.OperatingSystemMXBean;
import java.lang.management.ThreadInfo;
import java.lang.management.ThreadMXBean;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.time.Duration;
import java.util.Map;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.TreeMap;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.LongSupplier;
import java.util.stream.Collectors;
import javax.management.NotificationEmitter;
import javax.management.openmbean.CompositeData;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* A collection of utilities to gather runtime information about the JVM.
*/
public class ProcessInfo {
private static final Logger LOGGER = LoggerFactory.getLogger(ProcessInfo.class);
// listen on GC events to track memory pool sizes after each GC
private static final AtomicReference> postGcMemoryUsage = new AtomicReference<>(Map.of());
static {
for (GarbageCollectorMXBean garbageCollectorMXBean : ManagementFactory.getGarbageCollectorMXBeans()) {
if (garbageCollectorMXBean instanceof NotificationEmitter emitter) {
emitter.addNotificationListener((notification, handback) -> {
if (notification.getUserData() instanceof CompositeData compositeData) {
var info = GarbageCollectionNotificationInfo.from(compositeData);
GcInfo gcInfo = info.getGcInfo();
postGcMemoryUsage.set(gcInfo.getMemoryUsageAfterGc().entrySet().stream()
.map(e -> Map.entry(e.getKey(), e.getValue().getUsed()))
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue))
);
}
}, null, null);
}
}
ThreadMXBean threadBean = ManagementFactory.getThreadMXBean();
if (threadBean.isThreadContentionMonitoringSupported()) {
ManagementFactory.getThreadMXBean().setThreadContentionMonitoringEnabled(true);
} else {
LOGGER.debug("Thread contention monitoring not supported, will not have access to waiting/blocking time stats.");
}
}
/**
* Returns the amount of CPU time this processed hase used according to {@link OperatingSystemMXBean}, or empty if the
* JVM does not support this.
*/
public static Optional getProcessCpuTime() {
Long result;
Object obj = ManagementFactory.getOperatingSystemMXBean();
try {
result = callGetter(obj.getClass().getMethod("getProcessCpuTime"), obj, Long.class);
} catch (NoSuchMethodException | InvocationTargetException e) {
result = null;
}
return Optional
.ofNullable(result)
.map(Duration::ofNanos);
}
// reflection helper
private static T callGetter(Method method, Object obj, Class resultClazz) throws InvocationTargetException {
try {
return resultClazz.cast(method.invoke(obj));
} catch (IllegalAccessException e) {
// Expected, the declaring class or interface might not be public.
} catch (ClassCastException e) {
return null;
}
// Iterate over all implemented/extended interfaces and attempt invoking the method with the
// same name and parameters on each.
for (Class> clazz : method.getDeclaringClass().getInterfaces()) {
try {
Method interfaceMethod = clazz.getMethod(method.getName(), method.getParameterTypes());
T result = callGetter(interfaceMethod, obj, resultClazz);
if (result != null) {
return result;
}
} catch (NoSuchMethodException e) {
// Expected, class might implement multiple, unrelated interfaces.
}
}
return null;
}
/** Returns the {@code -Xmx} JVM property in bytes according to {@link Runtime#maxMemory()}. */
public static long getMaxMemoryBytes() {
return Runtime.getRuntime().maxMemory();
}
/** Returns the JVM used memory. */
public static long getOnHeapUsedMemoryBytes() {
var runtime = Runtime.getRuntime();
return runtime.totalMemory() - runtime.freeMemory();
}
/**
* Returns the amount of direct memory (allocated through {@link java.nio.ByteBuffer#allocateDirect(int)}) used by the
* JVM.
*/
public static long getDirectUsedMemoryBytes() {
return ManagementFactory.getPlatformMXBeans(BufferPoolMXBean.class).stream()
.filter(pool -> "direct".equals(pool.getName()))
.mapToLong(BufferPoolMXBean::getTotalCapacity)
.sum();
}
/**
* Returns an estimate the amount of direct memory limit for this JVM by parsing {@code -XX:MaxDirectMemorySize}
* argument.
*/
public static long getDirectUsedMemoryLimit() {
return ManagementFactory.getRuntimeMXBean().getInputArguments().stream()
.filter(arg -> arg.startsWith("-XX:MaxDirectMemorySize="))
.mapToLong(arg -> Parse.jvmMemoryStringToBytes(arg.replace("-XX:MaxDirectMemorySize=", "")))
.findFirst()
// if -XX:MaxDirectMemorySize not explicitly specified, then direct limit is equal to -Xmx so total memory
// used can be 2x that.
.orElseGet(ProcessInfo::getMaxMemoryBytes);
}
/**
* Returns the total amount of memory available on the system if available.
*/
public static OptionalLong getSystemMemoryBytes() {
if (ManagementFactory.getOperatingSystemMXBean() instanceof com.sun.management.OperatingSystemMXBean osBean) {
return OptionalLong.of(osBean.getTotalMemorySize());
} else {
return OptionalLong.empty();
}
}
/**
* Returns the amount of free memory on this system outside the JVM heap, if available.
*/
public static OptionalLong getSystemFreeMemoryBytes() {
return getSystemMemoryBytes().stream()
.map(value -> value - getMaxMemoryBytes())
.filter(value -> value > 0)
.findFirst();
}
/** Processor usage statistics for a thread. */
public record ThreadState(
String name, Duration cpuTime, Duration userTime, Duration waiting, Duration blocking, long id
) {
private static long zeroIfUnsupported(LongSupplier supplier) {
try {
return supplier.getAsLong();
} catch (UnsupportedOperationException e) {
return 0;
}
}
public ThreadState(ThreadMXBean threadMXBean, ThreadInfo thread) {
this(
thread.getThreadName(),
Duration.ofNanos(zeroIfUnsupported(() -> threadMXBean.getThreadCpuTime(thread.getThreadId()))),
Duration.ofNanos(zeroIfUnsupported(() -> threadMXBean.getThreadUserTime(thread.getThreadId()))),
Duration.ofMillis(zeroIfUnsupported(thread::getWaitedTime)),
Duration.ofMillis(zeroIfUnsupported(thread::getBlockedTime)),
thread.getThreadId());
}
public static final ThreadState DEFAULT = new ThreadState("", Duration.ZERO, Duration.ZERO, Duration.ZERO,
Duration.ZERO, -1);
/** Adds up the timers in two {@code ThreadState} instances */
public ThreadState plus(ThreadState other) {
return new ThreadState("",
cpuTime.plus(other.cpuTime),
userTime.plus(other.userTime),
waiting.plus(other.waiting),
blocking.plus(other.blocking),
-1
);
}
}
/** Returns the amount of time this JVM has spent in any kind of garbage collection since startup. */
public static Duration getGcTime() {
long total = 0;
for (final GarbageCollectorMXBean gc : ManagementFactory.getGarbageCollectorMXBeans()) {
total += gc.getCollectionTime();
}
return Duration.ofMillis(total);
}
/** Returns a map from memory pool name to the size of that pool in bytes after the last garbage-collection. */
public static Map getPostGcPoolSizes() {
return postGcMemoryUsage.get();
}
/** Returns the total memory usage in bytes after last GC, or empty if no GC has occurred yet. */
public static OptionalLong getMemoryUsageAfterLastGC() {
var lastGcPoolSizes = postGcMemoryUsage.get();
if (lastGcPoolSizes.isEmpty()) {
return OptionalLong.empty();
} else {
return OptionalLong.of(lastGcPoolSizes.values().stream().mapToLong(l -> l).sum());
}
}
/** Returns a map from thread ID to stats about that thread for every thread that has run, even completed ones. */
public static Map getThreadStats() {
Map threadState = new TreeMap<>();
ThreadMXBean threadMXBean = ManagementFactory.getThreadMXBean();
for (ThreadInfo thread : threadMXBean.dumpAllThreads(false, false)) {
threadState.put(thread.getThreadId(), new ThreadState(threadMXBean, thread));
}
return threadState;
}
public static ThreadState getCurrentThreadState() {
ThreadMXBean threadMXBean = ManagementFactory.getThreadMXBean();
ThreadInfo thread = threadMXBean.getThreadInfo(Thread.currentThread().getId());
return new ThreadState(threadMXBean, thread);
}
}