From 22e719b16e7687337ead65a3160af24c4b0a6e87 Mon Sep 17 00:00:00 2001 From: Tsuyoshi Ozawa Date: Wed, 13 Jul 2016 21:28:04 +0900 Subject: [PATCH] HADOOP-11361. Fix a race condition in MetricsSourceAdapter.updateJmxCache. Contributed by Vinayakumar B, Yongjun Zhang, and Brahma Reddy Battula. (ozawa) (cherry picked from commit 77ffe7621212be9f462ca37a542a13d167eca4e0) --- .../metrics2/impl/MetricsSourceAdapter.java | 33 ++++++++----------- 1 file changed, 14 insertions(+), 19 deletions(-) diff --git a/hadoop-common-project/hadoop-common/src/main/java/org/apache/hadoop/metrics2/impl/MetricsSourceAdapter.java b/hadoop-common-project/hadoop-common/src/main/java/org/apache/hadoop/metrics2/impl/MetricsSourceAdapter.java index d56ee532201..04cd88cef4c 100644 --- a/hadoop-common-project/hadoop-common/src/main/java/org/apache/hadoop/metrics2/impl/MetricsSourceAdapter.java +++ b/hadoop-common-project/hadoop-common/src/main/java/org/apache/hadoop/metrics2/impl/MetricsSourceAdapter.java @@ -31,6 +31,7 @@ import javax.management.ReflectionException; import static com.google.common.base.Preconditions.*; import com.google.common.annotations.VisibleForTesting; +import com.google.common.base.Preconditions; import com.google.common.collect.Maps; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -59,7 +60,6 @@ class MetricsSourceAdapter implements DynamicMBean { private final MBeanInfoBuilder infoBuilder; private final Iterable injectedTags; - private Iterable lastRecs; private boolean lastRecsCleared; private long jmxCacheTS = 0; private long jmxCacheTTL; @@ -175,18 +175,19 @@ class MetricsSourceAdapter implements DynamicMBean { } } + // HADOOP-11361: Release lock here for avoid deadlock between + // MetricsSystemImpl's lock and MetricsSourceAdapter's lock. + Iterable lastRecs = null; if (getAllMetrics) { - MetricsCollectorImpl builder = new MetricsCollectorImpl(); - getMetrics(builder, true); + lastRecs = getMetrics(new MetricsCollectorImpl(), true); } - synchronized(this) { - updateAttrCache(); - if (getAllMetrics) { - updateInfoCache(); + synchronized (this) { + if (lastRecs != null) { + updateAttrCache(lastRecs); + updateInfoCache(lastRecs); } jmxCacheTS = Time.now(); - lastRecs = null; // in case regular interval update is not running lastRecsCleared = true; } } @@ -194,11 +195,6 @@ class MetricsSourceAdapter implements DynamicMBean { Iterable getMetrics(MetricsCollectorImpl builder, boolean all) { builder.setRecordFilter(recordFilter).setMetricFilter(metricFilter); - synchronized(this) { - if (lastRecs == null && jmxCacheTS == 0) { - all = true; // Get all the metrics to populate the sink caches - } - } try { source.getMetrics(builder, all); } @@ -210,10 +206,7 @@ class MetricsSourceAdapter implements DynamicMBean { rb.add(t); } } - synchronized(this) { - lastRecs = builder.getRecords(); - return lastRecs; - } + return builder.getRecords(); } synchronized void stop() { @@ -247,13 +240,15 @@ class MetricsSourceAdapter implements DynamicMBean { return jmxCacheTTL; } - private void updateInfoCache() { + private void updateInfoCache(Iterable lastRecs) { + Preconditions.checkNotNull(lastRecs, "LastRecs should not be null"); LOG.debug("Updating info cache..."); infoCache = infoBuilder.reset(lastRecs).get(); LOG.debug("Done"); } - private int updateAttrCache() { + private int updateAttrCache(Iterable lastRecs) { + Preconditions.checkNotNull(lastRecs, "LastRecs should not be null"); LOG.debug("Updating attr cache..."); int recNo = 0; int numMetrics = 0;