/* * Copyright (C) 2008 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.os; import android.compat.annotation.UnsupportedAppUsage; import android.util.Log; import java.util.ArrayList; import java.util.HashMap; /** * UEventObserver is an abstract class that receives UEvents from the kernel.
* * Subclass UEventObserver, implementing onUEvent(UEvent event), then call * startObserving() with a match string. The UEvent thread will then call your * onUEvent() method when a UEvent occurs that contains your match string.
* * Call stopObserving() to stop receiving UEvents.
* * There is only one UEvent thread per process, even if that process has * multiple UEventObserver subclass instances. The UEvent thread starts when * the startObserving() is called for the first time in that process. Once * started the UEvent thread will not stop (although it can stop notifying * UEventObserver's via stopObserving()).
* * @hide */ public abstract class UEventObserver { private static final String TAG = "UEventObserver"; private static final boolean DEBUG = false; private static UEventThread sThread; private static native void nativeSetup(); private static native String nativeWaitForNextEvent(); private static native void nativeAddMatch(String match); private static native void nativeRemoveMatch(String match); @UnsupportedAppUsage public UEventObserver() { } @Override protected void finalize() throws Throwable { try { stopObserving(); } finally { super.finalize(); } } private static UEventThread getThread() { synchronized (UEventObserver.class) { if (sThread == null) { sThread = new UEventThread(); sThread.start(); } return sThread; } } private static UEventThread peekThread() { synchronized (UEventObserver.class) { return sThread; } } /** * Begin observation of UEvents.
* This method will cause the UEvent thread to start if this is the first * invocation of startObserving in this process.
* Once called, the UEvent thread will call onUEvent() when an incoming * UEvent matches the specified string.
* This method can be called multiple times to register multiple matches. * Only one call to stopObserving is required even with multiple registered * matches. * * @param match A substring of the UEvent to match. Try to be as specific * as possible to avoid incurring unintended additional cost from processing * irrelevant messages. Netlink messages can be moderately high bandwidth and * are expensive to parse. For example, some devices may send one netlink message * for each vsync period. */ @UnsupportedAppUsage public final void startObserving(String match) { if (match == null || match.isEmpty()) { throw new IllegalArgumentException("match substring must be non-empty"); } final UEventThread t = getThread(); t.addObserver(match, this); } /** * End observation of UEvents.
     * This process's UEvent thread will never call onUEvent() on this
     * UEventObserver after this call. Repeated calls have no effect.
     */
    @UnsupportedAppUsage
    public final void stopObserving() {
        final UEventThread t = peekThread();
        if (t != null) {
            t.removeObserver(this);
        }
    }
    /**
     * Subclasses of UEventObserver should override this method to handle
     * UEvents.
     */
    @UnsupportedAppUsage
    public abstract void onUEvent(UEvent event);
    /**
     * Representation of a UEvent.
     */
    public static final class UEvent {
        // collection of key=value pairs parsed from the uevent message
        private final HashMap