/* * Copyright (C) 2023 The Android Open Source Project * * 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 android.view; import static android.Manifest.permission.READ_FRAME_BUFFER; import static android.content.pm.PackageManager.PERMISSION_GRANTED; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.RequiresPermission; import android.content.Context; import android.os.SystemClock; import android.util.Log; import com.android.internal.annotations.GuardedBy; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.util.GcUtils; import java.io.PrintWriter; import java.util.ArrayList; import java.util.Map; import java.util.WeakHashMap; /** * A thread-safe registry used to track surface controls that are active (not yet released) within a * process, to help debug and identify leaks. * @hide */ public class SurfaceControlRegistry { private static final String TAG = "SurfaceControlRegistry"; /** * An interface for processing the registered SurfaceControls when the threshold is exceeded. */ public interface Reporter { /** * Called when the set of layers exceeds the max threshold. This can be called on any * thread, and must be handled synchronously. */ void onMaxLayersExceeded(WeakHashMap surfaceControls, int limit, PrintWriter pw); } /** * The default implementation of the reporter which logs the existing registered surfaces to * logcat. */ private static class DefaultReporter implements Reporter { public void onMaxLayersExceeded(WeakHashMap surfaceControls, int limit, PrintWriter pw) { final long now = SystemClock.elapsedRealtime(); final ArrayList> entries = new ArrayList<>(); for (Map.Entry entry : surfaceControls.entrySet()) { entries.add(entry); } // Sort entries by time registered when dumping // TODO: Or should it sort by name? entries.sort((o1, o2) -> (int) (o1.getValue() - o2.getValue())); final int size = Math.min(entries.size(), limit); pw.println("SurfaceControlRegistry"); pw.println("----------------------"); pw.println("Listing oldest " + size + " of " + surfaceControls.size()); for (int i = 0; i < size; i++) { final Map.Entry entry = entries.get(i); final SurfaceControl sc = entry.getKey(); final long timeRegistered = entry.getValue(); pw.print(" "); pw.print(sc.getName()); pw.print(" (" + sc.getCallsite() + ")"); pw.println(" [" + ((now - timeRegistered) / 1000) + "s ago]"); } } } // The threshold at which to dump information about all the known active SurfaceControls in the // process when the number of layers exceeds a certain count. This should be significantly // smaller than the MAX_LAYERS (currently 4096) defined in SurfaceFlinger.h private static final int MAX_LAYERS_REPORTING_THRESHOLD = 1024; // The threshold at which to reset the dump state. Needs to be smaller than // MAX_LAYERS_REPORTING_THRESHOLD private static final int RESET_REPORTING_THRESHOLD = 256; // Number of surface controls to dump when the max threshold is exceeded private static final int DUMP_LIMIT = 256; // Static lock, must be held for all registry operations private static final Object sLock = new Object(); // The default reporter for printing out the registered surfaces private static final DefaultReporter sDefaultReporter = new DefaultReporter(); // The registry for a given process private static volatile SurfaceControlRegistry sProcessRegistry; // Mapping of the active SurfaceControls to the elapsed time when they were registered @GuardedBy("sLock") private final WeakHashMap mSurfaceControls; // The threshold at which we dump information about the current set of registered surfaces. // Once this threshold is reached, we no longer report until the number of layers drops below // mResetReportingThreshold to ensure that we don't spam logcat. private int mMaxLayersReportingThreshold = MAX_LAYERS_REPORTING_THRESHOLD; private int mResetReportingThreshold = RESET_REPORTING_THRESHOLD; // Whether the current set of layers has exceeded mMaxLayersReportingThreshold, and we have // already reported the set of registered surfaces. private boolean mHasReportedExceedingMaxThreshold = false; // The handler for when the registry exceeds the max threshold private Reporter mReporter = sDefaultReporter; private SurfaceControlRegistry() { mSurfaceControls = new WeakHashMap<>(256); } /** * Sets the thresholds at which the registry reports errors. * @param maxLayersReportingThreshold The max threshold (inclusive) * @param resetReportingThreshold The reset threshold (inclusive) * @hide */ @VisibleForTesting public void setReportingThresholds(int maxLayersReportingThreshold, int resetReportingThreshold, Reporter reporter) { synchronized (sLock) { if (maxLayersReportingThreshold <= 0 || resetReportingThreshold >= maxLayersReportingThreshold) { throw new IllegalArgumentException("Expected maxLayersReportingThreshold (" + maxLayersReportingThreshold + ") to be > 0 and resetReportingThreshold (" + resetReportingThreshold + ") to be < maxLayersReportingThreshold"); } if (reporter == null) { throw new IllegalArgumentException("Expected non-null reporter"); } mMaxLayersReportingThreshold = maxLayersReportingThreshold; mResetReportingThreshold = resetReportingThreshold; mHasReportedExceedingMaxThreshold = false; mReporter = reporter; } } /** * Creates and initializes the registry for all SurfaceControls in this process. The caller must * hold the READ_FRAME_BUFFER permission. * @hide */ @RequiresPermission(READ_FRAME_BUFFER) @NonNull public static void createProcessInstance(Context context) { if (context.checkSelfPermission(READ_FRAME_BUFFER) != PERMISSION_GRANTED) { throw new SecurityException("Expected caller to hold READ_FRAME_BUFFER"); } synchronized (sLock) { if (sProcessRegistry == null) { sProcessRegistry = new SurfaceControlRegistry(); } } } /** * Destroys the previously created registry this process. * @hide */ public static void destroyProcessInstance() { synchronized (sLock) { if (sProcessRegistry == null) { return; } sProcessRegistry = null; } } /** * Returns the instance of the registry for this process, only non-null if * createProcessInstance(Context) was previously called from a valid caller. * @hide */ @Nullable @VisibleForTesting public static SurfaceControlRegistry getProcessInstance() { synchronized (sLock) { return sProcessRegistry; } } /** * Adds a SurfaceControl to the registry. */ void add(SurfaceControl sc) { synchronized (sLock) { mSurfaceControls.put(sc, SystemClock.elapsedRealtime()); if (!mHasReportedExceedingMaxThreshold && mSurfaceControls.size() >= mMaxLayersReportingThreshold) { // Dump existing info to logcat for debugging purposes (but don't close the // System.out output stream otherwise we can't print to it after this call) PrintWriter pw = new PrintWriter(System.out, true /* autoFlush */); mReporter.onMaxLayersExceeded(mSurfaceControls, DUMP_LIMIT, pw); mHasReportedExceedingMaxThreshold = true; } } } /** * Removes a SurfaceControl from the registry. */ void remove(SurfaceControl sc) { synchronized (sLock) { mSurfaceControls.remove(sc); if (mHasReportedExceedingMaxThreshold && mSurfaceControls.size() <= mResetReportingThreshold) { mHasReportedExceedingMaxThreshold = false; } } } /** * Returns a hash of this registry and is a function of all the active surface controls. This * is useful for testing to determine whether the registry has changed between creating and * destroying new SurfaceControls. */ @Override public int hashCode() { synchronized (sLock) { // Return a hash of the surface controls return mSurfaceControls.keySet().hashCode(); } } /** * Forces the gc and finalizers to run, used prior to dumping to ensure we only dump strongly * referenced surface controls. */ private static void runGcAndFinalizers() { long t = SystemClock.elapsedRealtime(); GcUtils.runGcAndFinalizersSync(); Log.i(TAG, "Ran gc and finalizers (" + (SystemClock.elapsedRealtime() - t) + "ms)"); } /** * Dumps information about the set of SurfaceControls in the registry. * * @param limit the number of layers to report * @param runGc whether to run the GC and finalizers before dumping * @hide */ public static void dump(int limit, boolean runGc, PrintWriter pw) { if (runGc) { // This needs to run outside the lock since finalization isn't synchronous runGcAndFinalizers(); } synchronized (sLock) { if (sProcessRegistry != null) { sDefaultReporter.onMaxLayersExceeded(sProcessRegistry.mSurfaceControls, limit, pw); } } } }