From 94ac8391d64134e3a1ac6388748c387b576b4a0e Mon Sep 17 00:00:00 2001 From: larsrc Date: Wed, 10 Mar 2021 02:47:05 -0800 Subject: [PATCH] Make WorkRequestHandler do a GC after some amount of CPU time has been used on requests. For Bazel and Blaze, defaults to 10s based on benchmarking. Linux benchmarks show no significant regressions in wall or CPU time, and possibly a reduction in system time. RELNOTES: None. PiperOrigin-RevId: 362006101 --- .../build/buildjar/BazelJavaBuilder.java | 4 +- .../build/lib/worker/WorkRequestHandler.java | 105 ++++++++++++++++-- 2 files changed, 99 insertions(+), 10 deletions(-) diff --git a/src/java_tools/buildjar/java/com/google/devtools/build/buildjar/BazelJavaBuilder.java b/src/java_tools/buildjar/java/com/google/devtools/build/buildjar/BazelJavaBuilder.java index ae78c446e35319..f1b12e62d19e7f 100644 --- a/src/java_tools/buildjar/java/com/google/devtools/build/buildjar/BazelJavaBuilder.java +++ b/src/java_tools/buildjar/java/com/google/devtools/build/buildjar/BazelJavaBuilder.java @@ -30,6 +30,7 @@ import java.io.PrintWriter; import java.io.Writer; import java.nio.charset.Charset; +import java.time.Duration; import java.util.Arrays; import java.util.List; @@ -46,7 +47,8 @@ public static void main(String[] args) { new WorkRequestHandler( builder::parseAndBuild, System.err, - new ProtoWorkerMessageProcessor(System.in, System.out)); + new ProtoWorkerMessageProcessor(System.in, System.out), + Duration.ofSeconds(10)); try { workerHandler.processRequests(); } catch (IOException e) { diff --git a/src/main/java/com/google/devtools/build/lib/worker/WorkRequestHandler.java b/src/main/java/com/google/devtools/build/lib/worker/WorkRequestHandler.java index 6a32a39aa95047..7da37742d48a15 100644 --- a/src/main/java/com/google/devtools/build/lib/worker/WorkRequestHandler.java +++ b/src/main/java/com/google/devtools/build/lib/worker/WorkRequestHandler.java @@ -13,14 +13,19 @@ // limitations under the License. package com.google.devtools.build.lib.worker; + import com.google.common.annotations.VisibleForTesting; import com.google.devtools.build.lib.worker.WorkerProtocol.WorkRequest; import com.google.devtools.build.lib.worker.WorkerProtocol.WorkResponse; +import com.sun.management.OperatingSystemMXBean; import java.io.IOException; import java.io.PrintStream; import java.io.PrintWriter; import java.io.StringWriter; +import java.lang.management.ManagementFactory; +import java.time.Duration; import java.util.List; +import java.util.concurrent.atomic.AtomicReference; import java.util.function.BiFunction; /** @@ -53,19 +58,50 @@ public interface WorkerMessageProcessor { final WorkerMessageProcessor messageProcessor; + private final CpuTimeBasedGcScheduler gcScheduler; + /** * Creates a {@code WorkRequestHandler} that will call {@code callback} for each WorkRequest - * received. The first argument to {@code callback} is the set of command-line arguments, the - * second is where all error messages and similar should be written to. The callback should return - * an exit code indicating success (0) or failure (nonzero). + * received. + * + * @param callback Callback method for executing a single WorkRequest in a thread. The first + * argument to {@code callback} is the set of command-line arguments, the second is where all + * error messages and other user-oriented messages should be written to. The callback must + * return an exit code indicating success (zero) or failure (nonzero). + * @param stderr Stream that log messages should be written to, typically the process' stderr. + * @param messageProcessor Object responsible for parsing {@code WorkRequest}s from the server and + * writing {@code WorkResponses} to the server. */ public WorkRequestHandler( BiFunction, PrintWriter, Integer> callback, PrintStream stderr, WorkerMessageProcessor messageProcessor) { + this(callback, stderr, messageProcessor, Duration.ZERO); + } + + /** + * Creates a {@code WorkRequestHandler} that will call {@code callback} for each WorkRequest + * received. + * + * @param callback Callback method for executing a single WorkRequest in a thread. The first + * argument to {@code callback} is the set of command-line arguments, the second is where all + * error messages and other user-oriented messages should be written to. The callback must + * return an exit code indicating success (zero) or failure (nonzero). + * @param stderr Stream that log messages should be written to, typically the process' stderr. + * @param messageProcessor Object responsible for parsing {@code WorkRequest}s from the server and + * writing {@code WorkResponses} to the server. + * @param cpuUsageBeforeGc The minimum amount of CPU time between explicit garbage collection + * calls. + */ + public WorkRequestHandler( + BiFunction, PrintWriter, Integer> callback, + PrintStream stderr, + WorkerMessageProcessor messageProcessor, + Duration cpuUsageBeforeGc) { this.callback = callback; this.stderr = stderr; this.messageProcessor = messageProcessor; + this.gcScheduler = new CpuTimeBasedGcScheduler(cpuUsageBeforeGc); } /** @@ -77,13 +113,13 @@ public WorkRequestHandler( public void processRequests() throws IOException { while (true) { WorkRequest request = messageProcessor.readWorkRequest(); - if (request == null) { - break; - } - if (request.getRequestId() != 0) { + if (request == null) { + break; + } + if (request.getRequestId() != 0) { Thread t = createResponseThread(request); - t.start(); - } else { + t.start(); + } else { respondToRequest(request); } } @@ -127,6 +163,7 @@ void respondToRequest(WorkRequest request) throws IOException { synchronized (this) { messageProcessor.writeWorkResponse(workResponse); } + gcScheduler.maybePerformGc(); } } @@ -134,4 +171,54 @@ void respondToRequest(WorkRequest request) throws IOException { public void close() throws IOException { messageProcessor.close(); } + + /** + * Class that performs GC occasionally, based on how much CPU time has passed. This strikes a + * compromise between blindly doing GC after e.g. every request, which takes too much CPU, and not + * doing explicit GC at all, which causes poor garbage collection in some cases. + */ + private static class CpuTimeBasedGcScheduler { + /** + * After this much CPU time has elapsed, we may force a GC run. Set to {@link Duration#ZERO} to + * disable. + */ + private final Duration cpuUsageBeforeGc; + + /** The total process CPU time at the last GC run (or from the start of the worker). */ + private final AtomicReference cpuTimeAtLastGc; + + /** Used to get the CPU time used by this process. */ + private static final OperatingSystemMXBean bean = + (OperatingSystemMXBean) ManagementFactory.getOperatingSystemMXBean(); + + /** + * Creates a new {@link CpuTimeBasedGcScheduler} that may perform GC after {@code + * cpuUsageBeforeGc} amount of CPU time has been used. + */ + public CpuTimeBasedGcScheduler(Duration cpuUsageBeforeGc) { + this.cpuUsageBeforeGc = cpuUsageBeforeGc; + this.cpuTimeAtLastGc = new AtomicReference<>(getCpuTime()); + } + + private Duration getCpuTime() { + return !cpuUsageBeforeGc.isZero() + ? Duration.ofNanos(bean.getProcessCpuTime()) + : Duration.ZERO; + } + + /** Call occasionally to perform a GC if enough CPU time has been used. */ + private void maybePerformGc() { + if (!cpuUsageBeforeGc.isZero()) { + Duration currentCpuTime = getCpuTime(); + Duration lastCpuTime = cpuTimeAtLastGc.get(); + // Do GC when enough CPU time has been used, but only if nobody else beat us to it. + if (currentCpuTime.minus(lastCpuTime).compareTo(cpuUsageBeforeGc) > 0 + && cpuTimeAtLastGc.compareAndSet(lastCpuTime, currentCpuTime)) { + System.gc(); + // Avoid counting GC CPU time against CPU time before next GC. + cpuTimeAtLastGc.compareAndSet(currentCpuTime, getCpuTime()); + } + } + } + } }