/* * 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 com.android.net.module.util; import android.annotation.NonNull; import android.os.Handler; import android.os.Looper; import java.util.concurrent.CountDownLatch; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicReference; /** * Helper class for Handler related utilities. * * @hide */ public class HandlerUtils { /** * Runs the specified task synchronously for dump method. *
* If the current thread is the same as the handler thread, then the runnable * runs immediately without being enqueued. Otherwise, posts the runnable * to the handler and waits for it to complete before returning. *
* This method is dangerous! Improper use can result in deadlocks. * Never call this method while any locks are held or use it in a * possibly re-entrant manner. *
* This method is made to let dump method access members on the handler thread to * avoid concurrent access problems or races. *
* If timeout occurs then this method returns false but the runnable
* will remain posted on the handler and may already be in progress or
* complete at a later time.
*
* When using this method, be sure to use {@link Looper#quitSafely} when * quitting the looper. Otherwise {@link #runWithScissorsForDump} may hang indefinitely. * (TODO: We should fix this by making MessageQueue aware of blocking runnables.) *
* * @param h The target handler. * @param r The Runnable that will be executed synchronously. * @param timeout The timeout in milliseconds, or 0 to not wait at all. * * @return Returns true if the Runnable was successfully executed. * Returns false on failure, usually because the * looper processing the message queue is exiting. * * @hide */ public static boolean runWithScissorsForDump(@NonNull Handler h, @NonNull Runnable r, long timeout) { if (r == null) { throw new IllegalArgumentException("runnable must not be null"); } if (timeout < 0) { throw new IllegalArgumentException("timeout must be non-negative"); } if (Looper.myLooper() == h.getLooper()) { r.run(); return true; } final CountDownLatch latch = new CountDownLatch(1); // Don't crash in the handler if something in the runnable throws an exception, // but try to propagate the exception to the caller. AtomicReference