/*
* Copyright (C) 2014 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.systemui.recents.events;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.SystemClock;
import android.os.UserHandle;
import android.util.Log;
import android.util.MutableBoolean;
import com.android.systemui.recents.misc.ReferenceCountedTrigger;
import java.io.PrintWriter;
import java.lang.ref.WeakReference;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
/**
* Represents a subscriber, which implements various event bus handler methods.
*/
class Subscriber {
private WeakReference
*
*
* Each subscriber can be registered with a given priority (default 1), and events will be dispatch
* in decreasing order of priority. For subscribers with the same priority, events will be
* dispatched by latest registration time to earliest.
*
*
* Interprocess events must extend {@link EventBus.InterprocessEvent}, have a constructor which
* takes a {@link Bundle} and implement toBundle(). This allows us to serialize events to be sent
* across processes.
*
*
* Caveats:
*
The EventBus keeps a {@link WeakReference} to the publisher to prevent memory leaks, so
* there must be another strong reference to the publisher for it to not get garbage-collected and
* continue receiving events.
*
Because the event handlers are called back using reflection, the EventBus is not intended
* for use in tight, performance criticial loops. For most user input/system callback events, this
* is generally of low enough frequency to use the EventBus.
*
Because the event handlers are called back using reflection, there will often be no
* references to them from actual code. The proguard configuration will be need to be updated to
* keep these extra methods:
*
* -keepclassmembers class ** {
* public void onBusEvent(**);
* public void onInterprocessBusEvent(**);
* }
* -keepclassmembers class ** extends **.EventBus$InterprocessEvent {
* public (android.os.Bundle);
* }
*
*
Subscriber registration can be expensive depending on the subscriber's {@link Class}. This
* is only done once per class type, but if possible, it is best to pre-register an instance of
* that class beforehand or when idle.
*
Each event should be sent once. Events may hold internal information about the current
* dispatch, or may be queued to be dispatched on another thread (if posted from a non-main thread),
* so it may be unsafe to edit, change, or re-send the event again.
*
Events should follow a pattern of public-final POD (plain old data) objects, where they are
* initialized by the constructor and read by each subscriber of that event. Subscribers should
* never alter events as they are processed, and this enforces that pattern.
*
*
*
* Future optimizations:
*
throw exception/log when a subscriber loses the reference
*
trace cost per registration & invocation
*
trace cross-process invocation
*
register(subscriber, Class<?>...) -- pass in exact class types you want registered
*
allow subscribers to implement interface, ie. EventBus.Subscriber, which lets then test a
* message before invocation (ie. check if task id == this task id)
*
add postOnce() which automatically debounces
*
add postDelayed() which delays / postDelayedOnce() which delays and bounces
*
consolidate register() and registerInterprocess()
*
sendForResult<ReturnType>(Event) to send and get a result, but who will send the
* result?
*
*/
public class EventBus extends BroadcastReceiver {
private static final String TAG = "EventBus";
private static final boolean DEBUG_TRACE_ALL = false;
/**
* An event super class that allows us to track internal event state across subscriber
* invocations.
*
* Events should not be edited by subscribers.
*/
public static class Event implements Cloneable {
// Indicates that this event's dispatch should be traced and logged to logcat
boolean trace;
// Indicates that this event must be posted on the EventBus's looper thread before invocation
boolean requiresPost;
// Not currently exposed, allows a subscriber to cancel further dispatch of this event
boolean cancelled;
// Only accessible from derived events
protected Event() {}
/**
* Called by the EventBus prior to dispatching this event to any subscriber of this event.
*/
void onPreDispatch() {
// Do nothing
}
/**
* Called by the EventBus after dispatching this event to every subscriber of this event.
*/
void onPostDispatch() {
// Do nothing
}
@Override
protected Object clone() throws CloneNotSupportedException {
Event evt = (Event) super.clone();
// When cloning an event, reset the cancelled-dispatch state
evt.cancelled = false;
return evt;
}
}
/**
* An event that represents an animated state change, which allows subscribers to coordinate
* callbacks which happen after the animation has taken place.
*
* Internally, it is guaranteed that increment() and decrement() will be called before and the
* after the event is dispatched.
*/
public static class AnimatedEvent extends Event {
private final ReferenceCountedTrigger mTrigger = new ReferenceCountedTrigger();
// Only accessible from derived events
protected AnimatedEvent() {}
/**
* Returns the reference counted trigger that coordinates the animations for this event.
*/
public ReferenceCountedTrigger getAnimationTrigger() {
return mTrigger;
}
/**
* Adds a callback that is guaranteed to be called after the state has changed regardless of
* whether an actual animation took place.
*/
public void addPostAnimationCallback(Runnable r) {
mTrigger.addLastDecrementRunnable(r);
}
@Override
void onPreDispatch() {
mTrigger.increment();
}
@Override
void onPostDispatch() {
mTrigger.decrement();
}
@Override
protected Object clone() throws CloneNotSupportedException {
throw new CloneNotSupportedException();
}
}
/**
* An event that can be reusable, only used for situations where we want to reduce memory
* allocations when events are sent frequently (ie. on scroll).
*/
public static class ReusableEvent extends Event {
private int mDispatchCount;
protected ReusableEvent() {}
@Override
void onPostDispatch() {
super.onPostDispatch();
mDispatchCount++;
}
@Override
protected Object clone() throws CloneNotSupportedException {
throw new CloneNotSupportedException();
}
}
/**
* An inter-process event super class that allows us to track user state across subscriber
* invocations.
*/
public static class InterprocessEvent extends Event {
private static final String EXTRA_USER = "_user";
// The user which this event originated from
public final int user;
// Only accessible from derived events
protected InterprocessEvent(int user) {
this.user = user;
}
/**
* Called from the event bus
*/
protected InterprocessEvent(Bundle b) {
user = b.getInt(EXTRA_USER);
}
protected Bundle toBundle() {
Bundle b = new Bundle();
b.putInt(EXTRA_USER, user);
return b;
}
}
/**
* Proguard must also know, and keep, all methods matching this signature.
*
* -keepclassmembers class ** {
* public void onBusEvent(**);
* public void onInterprocessBusEvent(**);
* }
*/
private static final String METHOD_PREFIX = "onBusEvent";
private static final String INTERPROCESS_METHOD_PREFIX = "onInterprocessBusEvent";
// Ensures that interprocess events can only be sent from a process holding this permission. */
private static final String PERMISSION_SELF = "com.android.systemui.permission.SELF";
// Used for passing event data across process boundaries
private static final String EXTRA_INTERPROCESS_EVENT_BUNDLE = "interprocess_event_bundle";
// The default priority of all subscribers
private static final int DEFAULT_SUBSCRIBER_PRIORITY = 1;
// Orders the handlers by priority and registration time
private static final Comparator EVENT_HANDLER_COMPARATOR = new Comparator() {
@Override
public int compare(EventHandler h1, EventHandler h2) {
// Rank the handlers by priority descending, followed by registration time descending.
// aka. the later registered
if (h1.priority != h2.priority) {
return h2.priority - h1.priority;
} else {
return Long.compare(h2.subscriber.registrationTime, h1.subscriber.registrationTime);
}
}
};
// Used for initializing the default bus
private static final Object sLock = new Object();
private static EventBus sDefaultBus;
// The handler to post all events
private Handler mHandler;
// Keep track of whether we have registered a broadcast receiver already, so that we can
// unregister ourselves before re-registering again with a new IntentFilter.
private boolean mHasRegisteredReceiver;
/**
* Map from event class -> event handler list. Keeps track of the actual mapping from event
* to subscriber method.
*/
private HashMap, ArrayList> mEventTypeMap = new HashMap<>();
/**
* Map from subscriber class -> event handler method lists. Used to determine upon registration
* of a new subscriber whether we need to read all the subscriber's methods again using
* reflection or whether we can just add the subscriber to the event type map.
*/
private HashMap, ArrayList> mSubscriberTypeMap = new HashMap<>();
/**
* Map from interprocess event name -> interprocess event class. Used for mapping the event
* name after receiving the broadcast, to the event type. After which a new instance is created
* and posted in the local process.
*/
private HashMap> mInterprocessEventNameMap = new HashMap<>();
/**
* Set of all currently registered subscribers
*/
private ArrayList mSubscribers = new ArrayList<>();
// For tracing
private int mCallCount;
private long mCallDurationMicros;
/**
* Private constructor to create an event bus for a given looper.
*/
private EventBus(Looper looper) {
mHandler = new Handler(looper);
}
/**
* @return the default event bus for the application's main thread.
*/
public static EventBus getDefault() {
if (sDefaultBus == null)
synchronized (sLock) {
if (sDefaultBus == null) {
if (DEBUG_TRACE_ALL) {
logWithPid("New EventBus");
}
sDefaultBus = new EventBus(Looper.getMainLooper());
}
}
return sDefaultBus;
}
/**
* Registers a subscriber to receive events with the default priority.
*
* @param subscriber the subscriber to handle events. If this is the first instance of the
* subscriber's class type that has been registered, the class's methods will
* be scanned for appropriate event handler methods.
*/
public void register(Object subscriber) {
registerSubscriber(subscriber, DEFAULT_SUBSCRIBER_PRIORITY, null);
}
/**
* Registers a subscriber to receive events with the given priority.
*
* @param subscriber the subscriber to handle events. If this is the first instance of the
* subscriber's class type that has been registered, the class's methods will
* be scanned for appropriate event handler methods.
* @param priority the priority that this subscriber will receive events relative to other
* subscribers
*/
public void register(Object subscriber, int priority) {
registerSubscriber(subscriber, priority, null);
}
/**
* Explicitly registers a subscriber to receive interprocess events with the default priority.
*
* @param subscriber the subscriber to handle events. If this is the first instance of the
* subscriber's class type that has been registered, the class's methods will
* be scanned for appropriate event handler methods.
*/
public void registerInterprocessAsCurrentUser(Context context, Object subscriber) {
registerInterprocessAsCurrentUser(context, subscriber, DEFAULT_SUBSCRIBER_PRIORITY);
}
/**
* Registers a subscriber to receive interprocess events with the given priority.
*
* @param subscriber the subscriber to handle events. If this is the first instance of the
* subscriber's class type that has been registered, the class's methods will
* be scanned for appropriate event handler methods.
* @param priority the priority that this subscriber will receive events relative to other
* subscribers
*/
public void registerInterprocessAsCurrentUser(Context context, Object subscriber, int priority) {
if (DEBUG_TRACE_ALL) {
logWithPid("registerInterprocessAsCurrentUser(" + subscriber.getClass().getSimpleName() + ")");
}
// Register the subscriber normally, and update the broadcast receiver filter if this is
// a new subscriber type with interprocess events
MutableBoolean hasInterprocessEventsChanged = new MutableBoolean(false);
registerSubscriber(subscriber, priority, hasInterprocessEventsChanged);
if (DEBUG_TRACE_ALL) {
logWithPid("hasInterprocessEventsChanged: " + hasInterprocessEventsChanged.value);
}
if (hasInterprocessEventsChanged.value) {
registerReceiverForInterprocessEvents(context);
}
}
/**
* Remove all EventHandlers pointing to the specified subscriber. This does not remove the
* mapping of subscriber type to event handler method, in case new instances of this subscriber
* are registered.
*/
public void unregister(Object subscriber) {
if (DEBUG_TRACE_ALL) {
logWithPid("unregister()");
}
// Fail immediately if we are being called from the non-main thread
long callingThreadId = Thread.currentThread().getId();
if (callingThreadId != mHandler.getLooper().getThread().getId()) {
throw new RuntimeException("Can not unregister() a subscriber from a non-main thread.");
}
// Return early if this is not a registered subscriber
if (!findRegisteredSubscriber(subscriber, true /* removeFoundSubscriber */)) {
return;
}
Class> subscriberType = subscriber.getClass();
ArrayList subscriberMethods = mSubscriberTypeMap.get(subscriberType);
if (subscriberMethods != null) {
// For each of the event handlers the subscriber handles, remove all references of that
// handler
for (EventHandlerMethod method : subscriberMethods) {
ArrayList eventHandlers = mEventTypeMap.get(method.eventType);
for (int i = eventHandlers.size() - 1; i >= 0; i--) {
if (eventHandlers.get(i).subscriber.getReference() == subscriber) {
eventHandlers.remove(i);
}
}
}
}
}
/**
* Explicit unregistration for interprocess event subscribers. This actually behaves exactly
* the same as unregister() since we also do not want to stop listening for specific
* inter-process messages in case new instances of that subscriber is registered.
*/
public void unregisterInterprocess(Context context, Object subscriber) {
if (DEBUG_TRACE_ALL) {
logWithPid("unregisterInterprocess()");
}
unregister(subscriber);
}
/**
* Sends an event to the subscribers of the given event type immediately. This can only be
* called from the same thread as the EventBus's looper thread (for the default EventBus, this
* is the main application thread).
*/
public void send(Event event) {
// Fail immediately if we are being called from the non-main thread
long callingThreadId = Thread.currentThread().getId();
if (callingThreadId != mHandler.getLooper().getThread().getId()) {
throw new RuntimeException("Can not send() a message from a non-main thread.");
}
if (DEBUG_TRACE_ALL) {
logWithPid("send(" + event.getClass().getSimpleName() + ")");
}
// Reset the event's cancelled state
event.requiresPost = false;
event.cancelled = false;
queueEvent(event);
}
/**
* Post a message to the subscribers of the given event type. The messages will be posted on
* the EventBus's looper thread (for the default EventBus, this is the main application thread).
*/
public void post(Event event) {
if (DEBUG_TRACE_ALL) {
logWithPid("post(" + event.getClass().getSimpleName() + ")");
}
// Reset the event's cancelled state
event.requiresPost = true;
event.cancelled = false;
queueEvent(event);
}
/**
* If this method is called from the main thread, it will be handled directly. If this method
* is not called from the main thread, it will be posted onto the main thread.
*/
public void sendOntoMainThread(Event event) {
long callingThreadId = Thread.currentThread().getId();
if (callingThreadId != mHandler.getLooper().getThread().getId()) {
post(event);
} else {
send(event);
}
}
/** Prevent post()ing an InterprocessEvent */
@Deprecated
public void post(InterprocessEvent event) {
throw new RuntimeException("Not supported, use postInterprocess");
}
/** Prevent send()ing an InterprocessEvent */
@Deprecated
public void send(InterprocessEvent event) {
throw new RuntimeException("Not supported, use postInterprocess");
}
/**
* Posts an interprocess event.
*/
public void postInterprocess(Context context, final InterprocessEvent event) {
if (DEBUG_TRACE_ALL) {
logWithPid("postInterprocess(" + event.getClass().getSimpleName() + ")");
}
String eventType = event.getClass().getName();
Bundle eventBundle = event.toBundle();
Intent intent = new Intent(eventType);
intent.setPackage(context.getPackageName());
intent.putExtra(EXTRA_INTERPROCESS_EVENT_BUNDLE, eventBundle);
intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT |
Intent.FLAG_RECEIVER_FOREGROUND);
context.sendBroadcastAsUser(intent, UserHandle.ALL);
}
/**
* Receiver for interprocess events.
*/
@Override
public void onReceive(Context context, Intent intent) {
if (DEBUG_TRACE_ALL) {
logWithPid("onReceive(" + intent.getAction() + ", user " + UserHandle.myUserId() + ")");
}
Bundle eventBundle = intent.getBundleExtra(EXTRA_INTERPROCESS_EVENT_BUNDLE);
Class extends InterprocessEvent> eventType = mInterprocessEventNameMap.get(intent.getAction());
try {
Constructor extends InterprocessEvent> ctor = eventType.getConstructor(Bundle.class);
send((Event) ctor.newInstance(eventBundle));
} catch (NoSuchMethodException|
InvocationTargetException|
InstantiationException|
IllegalAccessException e) {
Log.e(TAG, "Failed to create InterprocessEvent", e.getCause());
}
}
/**
* @return a dump of the current state of the EventBus
*/
public void dump(String prefix, PrintWriter writer) {
writer.println(dumpInternal(prefix));
}
public String dumpInternal(String prefix) {
String innerPrefix = prefix + " ";
String innerInnerPrefix = innerPrefix + " ";
StringBuilder output = new StringBuilder();
output.append(prefix);
output.append("Registered class types:");
output.append("\n");
ArrayList> subsciberTypes = new ArrayList<>(mSubscriberTypeMap.keySet());
Collections.sort(subsciberTypes, new Comparator>() {
@Override
public int compare(Class> o1, Class> o2) {
return o1.getSimpleName().compareTo(o2.getSimpleName());
}
});
for (int i = 0; i < subsciberTypes.size(); i++) {
Class> clz = subsciberTypes.get(i);
output.append(innerPrefix);
output.append(clz.getSimpleName());
output.append("\n");
}
output.append(prefix);
output.append("Event map:");
output.append("\n");
ArrayList> classes = new ArrayList<>(mEventTypeMap.keySet());
Collections.sort(classes, new Comparator>() {
@Override
public int compare(Class> o1, Class> o2) {
return o1.getSimpleName().compareTo(o2.getSimpleName());
}
});
for (int i = 0; i < classes.size(); i++) {
Class> clz = classes.get(i);
output.append(innerPrefix);
output.append(clz.getSimpleName());
output.append(" -> ");
output.append("\n");
ArrayList handlers = mEventTypeMap.get(clz);
for (EventHandler handler : handlers) {
Object subscriber = handler.subscriber.getReference();
if (subscriber != null) {
String id = Integer.toHexString(System.identityHashCode(subscriber));
output.append(innerInnerPrefix);
output.append(subscriber.getClass().getSimpleName());
output.append(" [0x" + id + ", #" + handler.priority + "]");
output.append("\n");
}
}
}
return output.toString();
}
/**
* Registers a new subscriber.
*/
private void registerSubscriber(Object subscriber, int priority,
MutableBoolean hasInterprocessEventsChangedOut) {
// Fail immediately if we are being called from the non-main thread
long callingThreadId = Thread.currentThread().getId();
if (callingThreadId != mHandler.getLooper().getThread().getId()) {
throw new RuntimeException("Can not register() a subscriber from a non-main thread.");
}
// Return immediately if this exact subscriber is already registered
if (findRegisteredSubscriber(subscriber, false /* removeFoundSubscriber */)) {
return;
}
long t1 = 0;
if (DEBUG_TRACE_ALL) {
t1 = SystemClock.currentTimeMicro();
logWithPid("registerSubscriber(" + subscriber.getClass().getSimpleName() + ")");
}
Subscriber sub = new Subscriber(subscriber, SystemClock.uptimeMillis());
Class> subscriberType = subscriber.getClass();
ArrayList subscriberMethods = mSubscriberTypeMap.get(subscriberType);
if (subscriberMethods != null) {
if (DEBUG_TRACE_ALL) {
logWithPid("Subscriber class type already registered");
}
// If we've parsed this subscriber type before, just add to the set for all the known
// events
for (EventHandlerMethod method : subscriberMethods) {
ArrayList eventTypeHandlers = mEventTypeMap.get(method.eventType);
eventTypeHandlers.add(new EventHandler(sub, method, priority));
sortEventHandlersByPriority(eventTypeHandlers);
}
mSubscribers.add(sub);
return;
} else {
if (DEBUG_TRACE_ALL) {
logWithPid("Subscriber class type requires registration");
}
// If we are parsing this type from scratch, ensure we add it to the subscriber type
// map, and pull out he handler methods below
subscriberMethods = new ArrayList<>();
mSubscriberTypeMap.put(subscriberType, subscriberMethods);
mSubscribers.add(sub);
}
// Find all the valid event bus handler methods of the subscriber
MutableBoolean isInterprocessEvent = new MutableBoolean(false);
Method[] methods = subscriberType.getDeclaredMethods();
for (Method m : methods) {
Class>[] parameterTypes = m.getParameterTypes();
isInterprocessEvent.value = false;
if (isValidEventBusHandlerMethod(m, parameterTypes, isInterprocessEvent)) {
Class extends Event> eventType = (Class extends Event>) parameterTypes[0];
ArrayList eventTypeHandlers = mEventTypeMap.get(eventType);
if (eventTypeHandlers == null) {
eventTypeHandlers = new ArrayList<>();
mEventTypeMap.put(eventType, eventTypeHandlers);
}
if (isInterprocessEvent.value) {
try {
// Enforce that the event must have a Bundle constructor
eventType.getConstructor(Bundle.class);
mInterprocessEventNameMap.put(eventType.getName(),
(Class extends InterprocessEvent>) eventType);
if (hasInterprocessEventsChangedOut != null) {
hasInterprocessEventsChangedOut.value = true;
}
} catch (NoSuchMethodException e) {
throw new RuntimeException("Expected InterprocessEvent to have a Bundle constructor");
}
}
EventHandlerMethod method = new EventHandlerMethod(m, eventType);
EventHandler handler = new EventHandler(sub, method, priority);
eventTypeHandlers.add(handler);
subscriberMethods.add(method);
sortEventHandlersByPriority(eventTypeHandlers);
if (DEBUG_TRACE_ALL) {
logWithPid(" * Method: " + m.getName() +
" event: " + parameterTypes[0].getSimpleName() +
" interprocess? " + isInterprocessEvent.value);
}
}
}
if (DEBUG_TRACE_ALL) {
logWithPid("Registered " + subscriber.getClass().getSimpleName() + " in " +
(SystemClock.currentTimeMicro() - t1) + " microseconds");
}
}
/**
* Adds a new message.
*/
private void queueEvent(final Event event) {
ArrayList eventHandlers = mEventTypeMap.get(event.getClass());
if (eventHandlers == null) {
return;
}
// Prepare this event
boolean hasPostedEvent = false;
event.onPreDispatch();
// We need to clone the list in case a subscriber unregisters itself during traversal
// TODO: Investigate whether we can skip the object creation here
eventHandlers = (ArrayList) eventHandlers.clone();
int eventHandlerCount = eventHandlers.size();
for (int i = 0; i < eventHandlerCount; i++) {
final EventHandler eventHandler = eventHandlers.get(i);
if (eventHandler.subscriber.getReference() != null) {
if (event.requiresPost) {
mHandler.post(new Runnable() {
@Override
public void run() {
processEvent(eventHandler, event);
}
});
hasPostedEvent = true;
} else {
processEvent(eventHandler, event);
}
}
}
// Clean up after this event, deferring until all subscribers have been called
if (hasPostedEvent) {
mHandler.post(new Runnable() {
@Override
public void run() {
event.onPostDispatch();
}
});
} else {
event.onPostDispatch();
}
}
/**
* Processes and dispatches the given event to the given event handler, on the thread of whoever
* calls this method.
*/
private void processEvent(final EventHandler eventHandler, final Event event) {
// Skip if the event was already cancelled
if (event.cancelled) {
if (event.trace || DEBUG_TRACE_ALL) {
logWithPid("Event dispatch cancelled");
}
return;
}
try {
if (event.trace || DEBUG_TRACE_ALL) {
logWithPid(" -> " + eventHandler.toString());
}
Object sub = eventHandler.subscriber.getReference();
if (sub != null) {
long t1 = 0;
if (DEBUG_TRACE_ALL) {
t1 = SystemClock.currentTimeMicro();
}
eventHandler.method.invoke(sub, event);
if (DEBUG_TRACE_ALL) {
long duration = (SystemClock.currentTimeMicro() - t1);
mCallDurationMicros += duration;
mCallCount++;
logWithPid(eventHandler.method.toString() + " duration: " + duration +
" microseconds, avg: " + (mCallDurationMicros / mCallCount));
}
} else {
Log.e(TAG, "Failed to deliver event to null subscriber");
}
} catch (IllegalAccessException e) {
Log.e(TAG, "Failed to invoke method", e.getCause());
} catch (InvocationTargetException e) {
throw new RuntimeException(e.getCause());
}
}
/**
* Re-registers the broadcast receiver for any new messages that we want to listen for.
*/
private void registerReceiverForInterprocessEvents(Context context) {
if (DEBUG_TRACE_ALL) {
logWithPid("registerReceiverForInterprocessEvents()");
}
// Rebuild the receiver filter with the new interprocess events
IntentFilter filter = new IntentFilter();
for (String eventName : mInterprocessEventNameMap.keySet()) {
filter.addAction(eventName);
if (DEBUG_TRACE_ALL) {
logWithPid(" filter: " + eventName);
}
}
// Re-register the receiver with the new filter
if (mHasRegisteredReceiver) {
context.unregisterReceiver(this);
}
context.registerReceiverAsUser(this, UserHandle.ALL, filter, PERMISSION_SELF, mHandler);
mHasRegisteredReceiver = true;
}
/**
* Returns whether this subscriber is currently registered. If {@param removeFoundSubscriber}
* is true, then remove the subscriber before returning.
*/
private boolean findRegisteredSubscriber(Object subscriber, boolean removeFoundSubscriber) {
for (int i = mSubscribers.size() - 1; i >= 0; i--) {
Subscriber sub = mSubscribers.get(i);
if (sub.getReference() == subscriber) {
if (removeFoundSubscriber) {
mSubscribers.remove(i);
}
return true;
}
}
return false;
}
/**
* @return whether {@param method} is a valid (normal or interprocess) event bus handler method
*/
private boolean isValidEventBusHandlerMethod(Method method, Class>[] parameterTypes,
MutableBoolean isInterprocessEventOut) {
int modifiers = method.getModifiers();
if (Modifier.isPublic(modifiers) &&
Modifier.isFinal(modifiers) &&
method.getReturnType().equals(Void.TYPE) &&
parameterTypes.length == 1) {
if (EventBus.InterprocessEvent.class.isAssignableFrom(parameterTypes[0]) &&
method.getName().startsWith(INTERPROCESS_METHOD_PREFIX)) {
isInterprocessEventOut.value = true;
return true;
} else if (EventBus.Event.class.isAssignableFrom(parameterTypes[0]) &&
method.getName().startsWith(METHOD_PREFIX)) {
isInterprocessEventOut.value = false;
return true;
} else {
if (DEBUG_TRACE_ALL) {
if (!EventBus.Event.class.isAssignableFrom(parameterTypes[0])) {
logWithPid(" Expected method take an Event-based parameter: " + method.getName());
} else if (!method.getName().startsWith(INTERPROCESS_METHOD_PREFIX) &&
!method.getName().startsWith(METHOD_PREFIX)) {
logWithPid(" Expected method start with method prefix: " + method.getName());
}
}
}
} else {
if (DEBUG_TRACE_ALL) {
if (!Modifier.isPublic(modifiers)) {
logWithPid(" Expected method to be public: " + method.getName());
} else if (!Modifier.isFinal(modifiers)) {
logWithPid(" Expected method to be final: " + method.getName());
} else if (!method.getReturnType().equals(Void.TYPE)) {
logWithPid(" Expected method to return null: " + method.getName());
}
}
}
return false;
}
/**
* Sorts the event handlers by priority and registration time.
*/
private void sortEventHandlersByPriority(List eventHandlers) {
Collections.sort(eventHandlers, EVENT_HANDLER_COMPARATOR);
}
/**
* Helper method to log the given {@param text} with the current process and user id.
*/
private static void logWithPid(String text) {
Log.d(TAG, "[" + android.os.Process.myPid() + ", u" + UserHandle.myUserId() + "] " + text);
}
}