aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/main/java
diff options
context:
space:
mode:
authorGravatar Janak Ramakrishnan <janakr@google.com>2016-03-23 17:26:12 +0000
committerGravatar Damien Martin-Guillerez <dmarting@google.com>2016-03-24 10:31:43 +0000
commit8cc772ef98604678d99b6a685e412a11a6508ba5 (patch)
tree59e3b96e6813a04ca9d8692529850e7c90f171d8 /src/main/java
parentb92c097ffac2e2e50632fc0a6e4f6909386aee45 (diff)
Add startup option --experimental_oom_more_eagerly_threshold, with default value 90. When --experimental_oom_more_eagerly is enabled, if after two full GCs the old gen is still >=--experimental_oom_more_eagerly_threshold% full, exit the JVM.
-- MOS_MIGRATED_REVID=117943361
Diffstat (limited to 'src/main/java')
-rw-r--r--src/main/java/com/google/devtools/build/lib/runtime/BlazeRuntime.java1
-rw-r--r--src/main/java/com/google/devtools/build/lib/runtime/BlazeServerStartupOptions.java10
-rw-r--r--src/main/java/com/google/devtools/build/lib/runtime/RetainedHeapLimiter.java130
3 files changed, 141 insertions, 0 deletions
diff --git a/src/main/java/com/google/devtools/build/lib/runtime/BlazeRuntime.java b/src/main/java/com/google/devtools/build/lib/runtime/BlazeRuntime.java
index 5a0e51262f..5b291c039a 100644
--- a/src/main/java/com/google/devtools/build/lib/runtime/BlazeRuntime.java
+++ b/src/main/java/com/google/devtools/build/lib/runtime/BlazeRuntime.java
@@ -1133,6 +1133,7 @@ public final class BlazeRuntime {
BlazeServerStartupOptions startupOptions = options.getOptions(BlazeServerStartupOptions.class);
if (startupOptions.batch && startupOptions.oomMoreEagerly) {
new OomSignalHandler();
+ new RetainedHeapLimiter(startupOptions.oomMoreEagerlyThreshold).install();
}
PathFragment workspaceDirectory = startupOptions.workspaceDirectory;
PathFragment installBase = startupOptions.installBase;
diff --git a/src/main/java/com/google/devtools/build/lib/runtime/BlazeServerStartupOptions.java b/src/main/java/com/google/devtools/build/lib/runtime/BlazeServerStartupOptions.java
index bd07e166e1..47e3cacaa7 100644
--- a/src/main/java/com/google/devtools/build/lib/runtime/BlazeServerStartupOptions.java
+++ b/src/main/java/com/google/devtools/build/lib/runtime/BlazeServerStartupOptions.java
@@ -177,6 +177,16 @@ public class BlazeServerStartupOptions extends OptionsBase {
)
public boolean oomMoreEagerly;
+ @Option(
+ name = "experimental_oom_more_eagerly_threshold",
+ defaultValue = "90", // NOTE: purely decorative! See class docstring.
+ category = "server startup",
+ help =
+ "If --experimental_oom_more_eagerly is set, Blaze will OOM if, after two full GC's, more "
+ + "than this percentage of the (old gen) heap is still occupied."
+ )
+ public int oomMoreEagerlyThreshold;
+
@Option(name = "block_for_lock",
defaultValue = "true", // NOTE: purely decorative! See class docstring.
category = "server startup",
diff --git a/src/main/java/com/google/devtools/build/lib/runtime/RetainedHeapLimiter.java b/src/main/java/com/google/devtools/build/lib/runtime/RetainedHeapLimiter.java
new file mode 100644
index 0000000000..a98e540737
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/runtime/RetainedHeapLimiter.java
@@ -0,0 +1,130 @@
+// Copyright 2016 The Bazel Authors. All rights reserved.
+//
+// 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 com.google.devtools.build.lib.runtime;
+
+import com.google.devtools.build.lib.util.Preconditions;
+import com.sun.management.GarbageCollectionNotificationInfo;
+import java.lang.management.GarbageCollectorMXBean;
+import java.lang.management.ManagementFactory;
+import java.lang.management.MemoryUsage;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.atomic.AtomicBoolean;
+import java.util.logging.Logger;
+import javax.management.Notification;
+import javax.management.NotificationEmitter;
+import javax.management.NotificationListener;
+import javax.management.openmbean.CompositeData;
+
+/**
+ * Monitor the size of the retained heap and exit promptly if it grows too large. Specifically,
+ * check the size of the tenured space after each major GC; if it exceeds 90%, call
+ * {@code System.gc()} to trigger a stop-the-world collection; if it's still more than 90% full,
+ * exit with an {@link OutOfMemoryError}.
+ */
+class RetainedHeapLimiter implements NotificationListener {
+ private static final Logger LOG = Logger.getLogger(RetainedHeapLimiter.class.getName());
+ private static final long MIN_TIME_BETWEEN_TRIGGERED_GC_MILLISECONDS = 60000;
+
+ private boolean installed = false;
+ private final AtomicBoolean throwingOom = new AtomicBoolean(false);
+ private long lastTriggeredGcInMilliseconds = 0;
+ private final int occupiedHeapPercentageThreshold;
+
+ RetainedHeapLimiter(int occupiedHeapPercentageThreshold) {
+ this.occupiedHeapPercentageThreshold = occupiedHeapPercentageThreshold;
+ }
+
+ void install() {
+ Preconditions.checkState(!installed, "RetainedHeapLimiter installed twice");
+ installed = true;
+ List<GarbageCollectorMXBean> gcbeans = ManagementFactory.getGarbageCollectorMXBeans();
+ boolean foundTenured = false;
+ // Examine all collectors and register for notifications from those which collect the tenured
+ // space. Normally there is one such collector.
+ for (GarbageCollectorMXBean gcbean : gcbeans) {
+ boolean collectsTenured = false;
+ for (String name : gcbean.getMemoryPoolNames()) {
+ collectsTenured |= isTenuredSpace(name);
+ }
+ if (collectsTenured) {
+ foundTenured = true;
+ NotificationEmitter emitter = (NotificationEmitter) gcbean;
+ emitter.addNotificationListener(this, null, null);
+ }
+ }
+ if (!foundTenured) {
+ throw new IllegalStateException(
+ "Can't find tenured space; update this class for a new collector");
+ }
+ }
+
+ @Override
+ public void handleNotification(Notification notification, Object handback) {
+ if (!notification
+ .getType()
+ .equals(GarbageCollectionNotificationInfo.GARBAGE_COLLECTION_NOTIFICATION)) {
+ return;
+ }
+ GarbageCollectionNotificationInfo info =
+ GarbageCollectionNotificationInfo.from((CompositeData) notification.getUserData());
+ Map<String, MemoryUsage> spaces = info.getGcInfo().getMemoryUsageAfterGc();
+ for (Map.Entry<String, MemoryUsage> entry : spaces.entrySet()) {
+ if (isTenuredSpace(entry.getKey())) {
+ MemoryUsage space = entry.getValue();
+ if (space.getMax() == 0) {
+ // The CMS collector sometimes passes us nonsense stats.
+ continue;
+ }
+
+ long percentUsed = 100 * space.getUsed() / space.getMax();
+ if (percentUsed > occupiedHeapPercentageThreshold) {
+ if (info.getGcCause().equals("System.gc()") && !throwingOom.getAndSet(true)) {
+ // Assume we got here from a GC initiated by the other branch.
+ String exitMsg =
+ String.format(
+ "RetainedHeapLimiter forcing exit due to GC thrashing: tenured space "
+ + "%s out of %s (>%s%%) occupied after back-to-back full GCs",
+ space.getUsed(),
+ space.getMax(),
+ occupiedHeapPercentageThreshold);
+ System.err.println(exitMsg);
+ LOG.info(exitMsg);
+ // Exits the runtime.
+ BugReport.handleCrash(new OutOfMemoryError(exitMsg));
+ } else if (System.currentTimeMillis() - lastTriggeredGcInMilliseconds
+ > MIN_TIME_BETWEEN_TRIGGERED_GC_MILLISECONDS) {
+ LOG.info(
+ "Triggering a full GC with "
+ + space.getUsed()
+ + " out of "
+ + space.getMax()
+ + " used");
+ // Force a full stop-the-world GC and see if it can get us below the threshold.
+ System.gc();
+ lastTriggeredGcInMilliseconds = System.currentTimeMillis();
+ }
+ }
+ }
+ }
+ }
+
+ private static boolean isTenuredSpace(String name) {
+ return "CMS Old Gen".equals(name)
+ || "G1 Old Gen".equals(name)
+ || "PS Old Gen".equals(name)
+ || "Tenured Gen".equals(name);
+ }
+}