1 /* 2 * Copyright (C) 2014 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.systemui.recents.events; 18 19 import android.content.BroadcastReceiver; 20 import android.content.Context; 21 import android.content.Intent; 22 import android.content.IntentFilter; 23 import android.os.Bundle; 24 import android.os.Handler; 25 import android.os.Looper; 26 import android.os.SystemClock; 27 import android.os.UserHandle; 28 import android.util.Log; 29 import android.util.MutableBoolean; 30 31 import com.android.systemui.recents.misc.ReferenceCountedTrigger; 32 33 import java.io.PrintWriter; 34 import java.lang.ref.WeakReference; 35 import java.lang.reflect.Constructor; 36 import java.lang.reflect.InvocationTargetException; 37 import java.lang.reflect.Method; 38 import java.lang.reflect.Modifier; 39 import java.util.ArrayList; 40 import java.util.Collections; 41 import java.util.Comparator; 42 import java.util.HashMap; 43 import java.util.List; 44 45 /** 46 * Represents a subscriber, which implements various event bus handler methods. 47 */ 48 class Subscriber { 49 private WeakReference<Object> mSubscriber; 50 51 long registrationTime; 52 53 Subscriber(Object subscriber, long registrationTime) { 54 mSubscriber = new WeakReference<>(subscriber); 55 this.registrationTime = registrationTime; 56 } 57 58 public String toString(int priority) { 59 Object sub = mSubscriber.get(); 60 String id = Integer.toHexString(System.identityHashCode(sub)); 61 return sub.getClass().getSimpleName() + " [0x" + id + ", P" + priority + "]"; 62 } 63 64 public Object getReference() { 65 return mSubscriber.get(); 66 } 67 } 68 69 /** 70 * Represents an event handler with a priority. 71 */ 72 class EventHandler { 73 int priority; 74 Subscriber subscriber; 75 EventHandlerMethod method; 76 77 EventHandler(Subscriber subscriber, EventHandlerMethod method, int priority) { 78 this.subscriber = subscriber; 79 this.method = method; 80 this.priority = priority; 81 } 82 83 @Override 84 public String toString() { 85 return subscriber.toString(priority) + " " + method.toString(); 86 } 87 } 88 89 /** 90 * Represents the low level method handling a particular event. 91 */ 92 class EventHandlerMethod { 93 private Method mMethod; 94 Class<? extends EventBus.Event> eventType; 95 96 EventHandlerMethod(Method method, Class<? extends EventBus.Event> eventType) { 97 mMethod = method; 98 mMethod.setAccessible(true); 99 this.eventType = eventType; 100 } 101 102 public void invoke(Object target, EventBus.Event event) 103 throws InvocationTargetException, IllegalAccessException { 104 mMethod.invoke(target, event); 105 } 106 107 @Override 108 public String toString() { 109 return mMethod.getName() + "(" + eventType.getSimpleName() + ")"; 110 } 111 } 112 113 /** 114 * A simple in-process event bus. It is simple because we can make assumptions about the state of 115 * SystemUI and Recent's lifecycle. 116 * 117 * <p> 118 * Currently, there is a single EventBus that handles {@link EventBus.Event}s for each subscriber 119 * on the main application thread. Publishers can send() events to synchronously call subscribers 120 * of that event, or post() events to be processed in the next run of the {@link Looper}. In 121 * addition, the EventBus supports sending and handling {@link EventBus.InterprocessEvent}s 122 * (within the same package) implemented using standard {@link BroadcastReceiver} mechanism. 123 * Interprocess events must be posted using postInterprocess() to ensure that it is dispatched 124 * correctly across processes. 125 * 126 * <p> 127 * Subscribers must be registered with a particular EventBus before they will receive events, and 128 * handler methods must match a specific signature. 129 * 130 * <p> 131 * Event method signature:<ul> 132 * <li>Methods must be public final 133 * <li>Methods must return void 134 * <li>Methods must be called "onBusEvent" 135 * <li>Methods must take one parameter, of class type deriving from {@link EventBus.Event} 136 * </ul> 137 * 138 * <p> 139 * Interprocess-Event method signature:<ul> 140 * <li>Methods must be public final 141 * <li>Methods must return void 142 * <li>Methods must be called "onInterprocessBusEvent" 143 * <li>Methods must take one parameter, of class type deriving from {@link EventBus.InterprocessEvent} 144 * </ul> 145 * </p> 146 * 147 * </p> 148 * Each subscriber can be registered with a given priority (default 1), and events will be dispatch 149 * in decreasing order of priority. For subscribers with the same priority, events will be 150 * dispatched by latest registration time to earliest. 151 * 152 * <p> 153 * Interprocess events must extend {@link EventBus.InterprocessEvent}, have a constructor which 154 * takes a {@link Bundle} and implement toBundle(). This allows us to serialize events to be sent 155 * across processes. 156 * 157 * <p> 158 * Caveats:<ul> 159 * <li>The EventBus keeps a {@link WeakReference} to the publisher to prevent memory leaks, so 160 * there must be another strong reference to the publisher for it to not get garbage-collected and 161 * continue receiving events. 162 * <li>Because the event handlers are called back using reflection, the EventBus is not intended 163 * for use in tight, performance criticial loops. For most user input/system callback events, this 164 * is generally of low enough frequency to use the EventBus. 165 * <li>Because the event handlers are called back using reflection, there will often be no 166 * references to them from actual code. The proguard configuration will be need to be updated to 167 * keep these extra methods: 168 * 169 * -keepclassmembers class ** { 170 * public void onBusEvent(**); 171 * public void onInterprocessBusEvent(**); 172 * } 173 * -keepclassmembers class ** extends **.EventBus$InterprocessEvent { 174 * public <init>(android.os.Bundle); 175 * } 176 * 177 * <li>Subscriber registration can be expensive depending on the subscriber's {@link Class}. This 178 * is only done once per class type, but if possible, it is best to pre-register an instance of 179 * that class beforehand or when idle. 180 * <li>Each event should be sent once. Events may hold internal information about the current 181 * dispatch, or may be queued to be dispatched on another thread (if posted from a non-main thread), 182 * so it may be unsafe to edit, change, or re-send the event again. 183 * <li>Events should follow a pattern of public-final POD (plain old data) objects, where they are 184 * initialized by the constructor and read by each subscriber of that event. Subscribers should 185 * never alter events as they are processed, and this enforces that pattern. 186 * </ul> 187 * 188 * <p> 189 * Future optimizations: 190 * <li>throw exception/log when a subscriber loses the reference 191 * <li>trace cost per registration & invocation 192 * <li>trace cross-process invocation 193 * <li>register(subscriber, Class<?>...) -- pass in exact class types you want registered 194 * <li>setSubscriberEventHandlerPriority(subscriber, Class<Event>, priority) 195 * <li>allow subscribers to implement interface, ie. EventBus.Subscriber, which lets then test a 196 * message before invocation (ie. check if task id == this task id) 197 * <li>add postOnce() which automatically debounces 198 * <li>add postDelayed() which delays / postDelayedOnce() which delays and bounces 199 * <li>consolidate register() and registerInterprocess() 200 * <li>sendForResult<ReturnType>(Event) to send and get a result, but who will send the 201 * result? 202 * </p> 203 */ 204 public class EventBus extends BroadcastReceiver { 205 206 private static final String TAG = "EventBus"; 207 private static final boolean DEBUG_TRACE_ALL = false; 208 209 /** 210 * An event super class that allows us to track internal event state across subscriber 211 * invocations. 212 * 213 * Events should not be edited by subscribers. 214 */ 215 public static class Event implements Cloneable { 216 // Indicates that this event's dispatch should be traced and logged to logcat 217 boolean trace; 218 // Indicates that this event must be posted on the EventBus's looper thread before invocation 219 boolean requiresPost; 220 // Not currently exposed, allows a subscriber to cancel further dispatch of this event 221 boolean cancelled; 222 223 // Only accessible from derived events 224 protected Event() {} 225 226 /** 227 * Called by the EventBus prior to dispatching this event to any subscriber of this event. 228 */ 229 void onPreDispatch() { 230 // Do nothing 231 } 232 233 /** 234 * Called by the EventBus after dispatching this event to every subscriber of this event. 235 */ 236 void onPostDispatch() { 237 // Do nothing 238 } 239 240 @Override 241 protected Object clone() throws CloneNotSupportedException { 242 Event evt = (Event) super.clone(); 243 // When cloning an event, reset the cancelled-dispatch state 244 evt.cancelled = false; 245 return evt; 246 } 247 } 248 249 /** 250 * An event that represents an animated state change, which allows subscribers to coordinate 251 * callbacks which happen after the animation has taken place. 252 * 253 * Internally, it is guaranteed that increment() and decrement() will be called before and the 254 * after the event is dispatched. 255 */ 256 public static class AnimatedEvent extends Event { 257 258 private final ReferenceCountedTrigger mTrigger = new ReferenceCountedTrigger(); 259 260 // Only accessible from derived events 261 protected AnimatedEvent() {} 262 263 /** 264 * Returns the reference counted trigger that coordinates the animations for this event. 265 */ 266 public ReferenceCountedTrigger getAnimationTrigger() { 267 return mTrigger; 268 } 269 270 /** 271 * Adds a callback that is guaranteed to be called after the state has changed regardless of 272 * whether an actual animation took place. 273 */ 274 public void addPostAnimationCallback(Runnable r) { 275 mTrigger.addLastDecrementRunnable(r); 276 } 277 278 @Override 279 void onPreDispatch() { 280 mTrigger.increment(); 281 } 282 283 @Override 284 void onPostDispatch() { 285 mTrigger.decrement(); 286 } 287 288 @Override 289 protected Object clone() throws CloneNotSupportedException { 290 throw new CloneNotSupportedException(); 291 } 292 } 293 294 /** 295 * An event that can be reusable, only used for situations where we want to reduce memory 296 * allocations when events are sent frequently (ie. on scroll). 297 */ 298 public static class ReusableEvent extends Event { 299 300 private int mDispatchCount; 301 302 protected ReusableEvent() {} 303 304 @Override 305 void onPostDispatch() { 306 super.onPostDispatch(); 307 mDispatchCount++; 308 } 309 310 @Override 311 protected Object clone() throws CloneNotSupportedException { 312 throw new CloneNotSupportedException(); 313 } 314 } 315 316 /** 317 * An inter-process event super class that allows us to track user state across subscriber 318 * invocations. 319 */ 320 public static class InterprocessEvent extends Event { 321 private static final String EXTRA_USER = "_user"; 322 323 // The user which this event originated from 324 public final int user; 325 326 // Only accessible from derived events 327 protected InterprocessEvent(int user) { 328 this.user = user; 329 } 330 331 /** 332 * Called from the event bus 333 */ 334 protected InterprocessEvent(Bundle b) { 335 user = b.getInt(EXTRA_USER); 336 } 337 338 protected Bundle toBundle() { 339 Bundle b = new Bundle(); 340 b.putInt(EXTRA_USER, user); 341 return b; 342 } 343 } 344 345 /** 346 * Proguard must also know, and keep, all methods matching this signature. 347 * 348 * -keepclassmembers class ** { 349 * public void onBusEvent(**); 350 * public void onInterprocessBusEvent(**); 351 * } 352 */ 353 private static final String METHOD_PREFIX = "onBusEvent"; 354 private static final String INTERPROCESS_METHOD_PREFIX = "onInterprocessBusEvent"; 355 356 // Ensures that interprocess events can only be sent from a process holding this permission. */ 357 private static final String PERMISSION_SELF = "com.android.systemui.permission.SELF"; 358 359 // Used for passing event data across process boundaries 360 private static final String EXTRA_INTERPROCESS_EVENT_BUNDLE = "interprocess_event_bundle"; 361 362 // The default priority of all subscribers 363 private static final int DEFAULT_SUBSCRIBER_PRIORITY = 1; 364 365 // Orders the handlers by priority and registration time 366 private static final Comparator<EventHandler> EVENT_HANDLER_COMPARATOR = new Comparator<EventHandler>() { 367 @Override 368 public int compare(EventHandler h1, EventHandler h2) { 369 // Rank the handlers by priority descending, followed by registration time descending. 370 // aka. the later registered 371 if (h1.priority != h2.priority) { 372 return h2.priority - h1.priority; 373 } else { 374 return Long.compare(h2.subscriber.registrationTime, h1.subscriber.registrationTime); 375 } 376 } 377 }; 378 379 // Used for initializing the default bus 380 private static final Object sLock = new Object(); 381 private static EventBus sDefaultBus; 382 383 // The handler to post all events 384 private Handler mHandler; 385 386 // Keep track of whether we have registered a broadcast receiver already, so that we can 387 // unregister ourselves before re-registering again with a new IntentFilter. 388 private boolean mHasRegisteredReceiver; 389 390 /** 391 * Map from event class -> event handler list. Keeps track of the actual mapping from event 392 * to subscriber method. 393 */ 394 private HashMap<Class<? extends Event>, ArrayList<EventHandler>> mEventTypeMap = new HashMap<>(); 395 396 /** 397 * Map from subscriber class -> event handler method lists. Used to determine upon registration 398 * of a new subscriber whether we need to read all the subscriber's methods again using 399 * reflection or whether we can just add the subscriber to the event type map. 400 */ 401 private HashMap<Class<? extends Object>, ArrayList<EventHandlerMethod>> mSubscriberTypeMap = new HashMap<>(); 402 403 /** 404 * Map from interprocess event name -> interprocess event class. Used for mapping the event 405 * name after receiving the broadcast, to the event type. After which a new instance is created 406 * and posted in the local process. 407 */ 408 private HashMap<String, Class<? extends InterprocessEvent>> mInterprocessEventNameMap = new HashMap<>(); 409 410 /** 411 * Set of all currently registered subscribers 412 */ 413 private ArrayList<Subscriber> mSubscribers = new ArrayList<>(); 414 415 // For tracing 416 private int mCallCount; 417 private long mCallDurationMicros; 418 419 /** 420 * Private constructor to create an event bus for a given looper. 421 */ 422 private EventBus(Looper looper) { 423 mHandler = new Handler(looper); 424 } 425 426 /** 427 * @return the default event bus for the application's main thread. 428 */ 429 public static EventBus getDefault() { 430 if (sDefaultBus == null) 431 synchronized (sLock) { 432 if (sDefaultBus == null) { 433 if (DEBUG_TRACE_ALL) { 434 logWithPid("New EventBus"); 435 } 436 sDefaultBus = new EventBus(Looper.getMainLooper()); 437 } 438 } 439 return sDefaultBus; 440 } 441 442 /** 443 * Registers a subscriber to receive events with the default priority. 444 * 445 * @param subscriber the subscriber to handle events. If this is the first instance of the 446 * subscriber's class type that has been registered, the class's methods will 447 * be scanned for appropriate event handler methods. 448 */ 449 public void register(Object subscriber) { 450 registerSubscriber(subscriber, DEFAULT_SUBSCRIBER_PRIORITY, null); 451 } 452 453 /** 454 * Registers a subscriber to receive events with the given priority. 455 * 456 * @param subscriber the subscriber to handle events. If this is the first instance of the 457 * subscriber's class type that has been registered, the class's methods will 458 * be scanned for appropriate event handler methods. 459 * @param priority the priority that this subscriber will receive events relative to other 460 * subscribers 461 */ 462 public void register(Object subscriber, int priority) { 463 registerSubscriber(subscriber, priority, null); 464 } 465 466 /** 467 * Explicitly registers a subscriber to receive interprocess events with the default priority. 468 * 469 * @param subscriber the subscriber to handle events. If this is the first instance of the 470 * subscriber's class type that has been registered, the class's methods will 471 * be scanned for appropriate event handler methods. 472 */ 473 public void registerInterprocessAsCurrentUser(Context context, Object subscriber) { 474 registerInterprocessAsCurrentUser(context, subscriber, DEFAULT_SUBSCRIBER_PRIORITY); 475 } 476 477 /** 478 * Registers a subscriber to receive interprocess events with the given priority. 479 * 480 * @param subscriber the subscriber to handle events. If this is the first instance of the 481 * subscriber's class type that has been registered, the class's methods will 482 * be scanned for appropriate event handler methods. 483 * @param priority the priority that this subscriber will receive events relative to other 484 * subscribers 485 */ 486 public void registerInterprocessAsCurrentUser(Context context, Object subscriber, int priority) { 487 if (DEBUG_TRACE_ALL) { 488 logWithPid("registerInterprocessAsCurrentUser(" + subscriber.getClass().getSimpleName() + ")"); 489 } 490 491 // Register the subscriber normally, and update the broadcast receiver filter if this is 492 // a new subscriber type with interprocess events 493 MutableBoolean hasInterprocessEventsChanged = new MutableBoolean(false); 494 registerSubscriber(subscriber, priority, hasInterprocessEventsChanged); 495 if (DEBUG_TRACE_ALL) { 496 logWithPid("hasInterprocessEventsChanged: " + hasInterprocessEventsChanged.value); 497 } 498 if (hasInterprocessEventsChanged.value) { 499 registerReceiverForInterprocessEvents(context); 500 } 501 } 502 503 /** 504 * Remove all EventHandlers pointing to the specified subscriber. This does not remove the 505 * mapping of subscriber type to event handler method, in case new instances of this subscriber 506 * are registered. 507 */ 508 public void unregister(Object subscriber) { 509 if (DEBUG_TRACE_ALL) { 510 logWithPid("unregister()"); 511 } 512 513 // Fail immediately if we are being called from the non-main thread 514 long callingThreadId = Thread.currentThread().getId(); 515 if (callingThreadId != mHandler.getLooper().getThread().getId()) { 516 throw new RuntimeException("Can not unregister() a subscriber from a non-main thread."); 517 } 518 519 // Return early if this is not a registered subscriber 520 if (!findRegisteredSubscriber(subscriber, true /* removeFoundSubscriber */)) { 521 return; 522 } 523 524 Class<?> subscriberType = subscriber.getClass(); 525 ArrayList<EventHandlerMethod> subscriberMethods = mSubscriberTypeMap.get(subscriberType); 526 if (subscriberMethods != null) { 527 // For each of the event handlers the subscriber handles, remove all references of that 528 // handler 529 for (EventHandlerMethod method : subscriberMethods) { 530 ArrayList<EventHandler> eventHandlers = mEventTypeMap.get(method.eventType); 531 for (int i = eventHandlers.size() - 1; i >= 0; i--) { 532 if (eventHandlers.get(i).subscriber.getReference() == subscriber) { 533 eventHandlers.remove(i); 534 } 535 } 536 } 537 } 538 } 539 540 /** 541 * Explicit unregistration for interprocess event subscribers. This actually behaves exactly 542 * the same as unregister() since we also do not want to stop listening for specific 543 * inter-process messages in case new instances of that subscriber is registered. 544 */ 545 public void unregisterInterprocess(Context context, Object subscriber) { 546 if (DEBUG_TRACE_ALL) { 547 logWithPid("unregisterInterprocess()"); 548 } 549 unregister(subscriber); 550 } 551 552 /** 553 * Sends an event to the subscribers of the given event type immediately. This can only be 554 * called from the same thread as the EventBus's looper thread (for the default EventBus, this 555 * is the main application thread). 556 */ 557 public void send(Event event) { 558 // Fail immediately if we are being called from the non-main thread 559 long callingThreadId = Thread.currentThread().getId(); 560 if (callingThreadId != mHandler.getLooper().getThread().getId()) { 561 throw new RuntimeException("Can not send() a message from a non-main thread."); 562 } 563 564 if (DEBUG_TRACE_ALL) { 565 logWithPid("send(" + event.getClass().getSimpleName() + ")"); 566 } 567 568 // Reset the event's cancelled state 569 event.requiresPost = false; 570 event.cancelled = false; 571 queueEvent(event); 572 } 573 574 /** 575 * Post a message to the subscribers of the given event type. The messages will be posted on 576 * the EventBus's looper thread (for the default EventBus, this is the main application thread). 577 */ 578 public void post(Event event) { 579 if (DEBUG_TRACE_ALL) { 580 logWithPid("post(" + event.getClass().getSimpleName() + ")"); 581 } 582 583 // Reset the event's cancelled state 584 event.requiresPost = true; 585 event.cancelled = false; 586 queueEvent(event); 587 } 588 589 /** 590 * If this method is called from the main thread, it will be handled directly. If this method 591 * is not called from the main thread, it will be posted onto the main thread. 592 */ 593 public void sendOntoMainThread(Event event) { 594 long callingThreadId = Thread.currentThread().getId(); 595 if (callingThreadId != mHandler.getLooper().getThread().getId()) { 596 post(event); 597 } else { 598 send(event); 599 } 600 } 601 602 /** Prevent post()ing an InterprocessEvent */ 603 @Deprecated 604 public void post(InterprocessEvent event) { 605 throw new RuntimeException("Not supported, use postInterprocess"); 606 } 607 608 /** Prevent send()ing an InterprocessEvent */ 609 @Deprecated 610 public void send(InterprocessEvent event) { 611 throw new RuntimeException("Not supported, use postInterprocess"); 612 } 613 614 /** 615 * Posts an interprocess event. 616 */ 617 public void postInterprocess(Context context, final InterprocessEvent event) { 618 if (DEBUG_TRACE_ALL) { 619 logWithPid("postInterprocess(" + event.getClass().getSimpleName() + ")"); 620 } 621 String eventType = event.getClass().getName(); 622 Bundle eventBundle = event.toBundle(); 623 Intent intent = new Intent(eventType); 624 intent.setPackage(context.getPackageName()); 625 intent.putExtra(EXTRA_INTERPROCESS_EVENT_BUNDLE, eventBundle); 626 intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT | 627 Intent.FLAG_RECEIVER_FOREGROUND); 628 context.sendBroadcastAsUser(intent, UserHandle.ALL); 629 } 630 631 /** 632 * Receiver for interprocess events. 633 */ 634 @Override 635 public void onReceive(Context context, Intent intent) { 636 if (DEBUG_TRACE_ALL) { 637 logWithPid("onReceive(" + intent.getAction() + ", user " + UserHandle.myUserId() + ")"); 638 } 639 640 Bundle eventBundle = intent.getBundleExtra(EXTRA_INTERPROCESS_EVENT_BUNDLE); 641 Class<? extends InterprocessEvent> eventType = mInterprocessEventNameMap.get(intent.getAction()); 642 try { 643 Constructor<? extends InterprocessEvent> ctor = eventType.getConstructor(Bundle.class); 644 send((Event) ctor.newInstance(eventBundle)); 645 } catch (NoSuchMethodException| 646 InvocationTargetException| 647 InstantiationException| 648 IllegalAccessException e) { 649 Log.e(TAG, "Failed to create InterprocessEvent", e.getCause()); 650 } 651 } 652 653 /** 654 * @return a dump of the current state of the EventBus 655 */ 656 public void dump(String prefix, PrintWriter writer) { 657 writer.println(dumpInternal(prefix)); 658 } 659 660 public String dumpInternal(String prefix) { 661 String innerPrefix = prefix + " "; 662 String innerInnerPrefix = innerPrefix + " "; 663 StringBuilder output = new StringBuilder(); 664 output.append(prefix); 665 output.append("Registered class types:"); 666 output.append("\n"); 667 ArrayList<Class<?>> subsciberTypes = new ArrayList<>(mSubscriberTypeMap.keySet()); 668 Collections.sort(subsciberTypes, new Comparator<Class<?>>() { 669 @Override 670 public int compare(Class<?> o1, Class<?> o2) { 671 return o1.getSimpleName().compareTo(o2.getSimpleName()); 672 } 673 }); 674 for (int i = 0; i < subsciberTypes.size(); i++) { 675 Class<?> clz = subsciberTypes.get(i); 676 output.append(innerPrefix); 677 output.append(clz.getSimpleName()); 678 output.append("\n"); 679 } 680 output.append(prefix); 681 output.append("Event map:"); 682 output.append("\n"); 683 ArrayList<Class<?>> classes = new ArrayList<>(mEventTypeMap.keySet()); 684 Collections.sort(classes, new Comparator<Class<?>>() { 685 @Override 686 public int compare(Class<?> o1, Class<?> o2) { 687 return o1.getSimpleName().compareTo(o2.getSimpleName()); 688 } 689 }); 690 for (int i = 0; i < classes.size(); i++) { 691 Class<?> clz = classes.get(i); 692 output.append(innerPrefix); 693 output.append(clz.getSimpleName()); 694 output.append(" -> "); 695 output.append("\n"); 696 ArrayList<EventHandler> handlers = mEventTypeMap.get(clz); 697 for (EventHandler handler : handlers) { 698 Object subscriber = handler.subscriber.getReference(); 699 if (subscriber != null) { 700 String id = Integer.toHexString(System.identityHashCode(subscriber)); 701 output.append(innerInnerPrefix); 702 output.append(subscriber.getClass().getSimpleName()); 703 output.append(" [0x" + id + ", #" + handler.priority + "]"); 704 output.append("\n"); 705 } 706 } 707 } 708 return output.toString(); 709 } 710 711 /** 712 * Registers a new subscriber. 713 */ 714 private void registerSubscriber(Object subscriber, int priority, 715 MutableBoolean hasInterprocessEventsChangedOut) { 716 // Fail immediately if we are being called from the non-main thread 717 long callingThreadId = Thread.currentThread().getId(); 718 if (callingThreadId != mHandler.getLooper().getThread().getId()) { 719 throw new RuntimeException("Can not register() a subscriber from a non-main thread."); 720 } 721 722 // Return immediately if this exact subscriber is already registered 723 if (findRegisteredSubscriber(subscriber, false /* removeFoundSubscriber */)) { 724 return; 725 } 726 727 long t1 = 0; 728 if (DEBUG_TRACE_ALL) { 729 t1 = SystemClock.currentTimeMicro(); 730 logWithPid("registerSubscriber(" + subscriber.getClass().getSimpleName() + ")"); 731 } 732 Subscriber sub = new Subscriber(subscriber, SystemClock.uptimeMillis()); 733 Class<?> subscriberType = subscriber.getClass(); 734 ArrayList<EventHandlerMethod> subscriberMethods = mSubscriberTypeMap.get(subscriberType); 735 if (subscriberMethods != null) { 736 if (DEBUG_TRACE_ALL) { 737 logWithPid("Subscriber class type already registered"); 738 } 739 740 // If we've parsed this subscriber type before, just add to the set for all the known 741 // events 742 for (EventHandlerMethod method : subscriberMethods) { 743 ArrayList<EventHandler> eventTypeHandlers = mEventTypeMap.get(method.eventType); 744 eventTypeHandlers.add(new EventHandler(sub, method, priority)); 745 sortEventHandlersByPriority(eventTypeHandlers); 746 } 747 mSubscribers.add(sub); 748 return; 749 } else { 750 if (DEBUG_TRACE_ALL) { 751 logWithPid("Subscriber class type requires registration"); 752 } 753 754 // If we are parsing this type from scratch, ensure we add it to the subscriber type 755 // map, and pull out he handler methods below 756 subscriberMethods = new ArrayList<>(); 757 mSubscriberTypeMap.put(subscriberType, subscriberMethods); 758 mSubscribers.add(sub); 759 } 760 761 // Find all the valid event bus handler methods of the subscriber 762 MutableBoolean isInterprocessEvent = new MutableBoolean(false); 763 Method[] methods = subscriberType.getDeclaredMethods(); 764 for (Method m : methods) { 765 Class<?>[] parameterTypes = m.getParameterTypes(); 766 isInterprocessEvent.value = false; 767 if (isValidEventBusHandlerMethod(m, parameterTypes, isInterprocessEvent)) { 768 Class<? extends Event> eventType = (Class<? extends Event>) parameterTypes[0]; 769 ArrayList<EventHandler> eventTypeHandlers = mEventTypeMap.get(eventType); 770 if (eventTypeHandlers == null) { 771 eventTypeHandlers = new ArrayList<>(); 772 mEventTypeMap.put(eventType, eventTypeHandlers); 773 } 774 if (isInterprocessEvent.value) { 775 try { 776 // Enforce that the event must have a Bundle constructor 777 eventType.getConstructor(Bundle.class); 778 779 mInterprocessEventNameMap.put(eventType.getName(), 780 (Class<? extends InterprocessEvent>) eventType); 781 if (hasInterprocessEventsChangedOut != null) { 782 hasInterprocessEventsChangedOut.value = true; 783 } 784 } catch (NoSuchMethodException e) { 785 throw new RuntimeException("Expected InterprocessEvent to have a Bundle constructor"); 786 } 787 } 788 EventHandlerMethod method = new EventHandlerMethod(m, eventType); 789 EventHandler handler = new EventHandler(sub, method, priority); 790 eventTypeHandlers.add(handler); 791 subscriberMethods.add(method); 792 sortEventHandlersByPriority(eventTypeHandlers); 793 794 if (DEBUG_TRACE_ALL) { 795 logWithPid(" * Method: " + m.getName() + 796 " event: " + parameterTypes[0].getSimpleName() + 797 " interprocess? " + isInterprocessEvent.value); 798 } 799 } 800 } 801 if (DEBUG_TRACE_ALL) { 802 logWithPid("Registered " + subscriber.getClass().getSimpleName() + " in " + 803 (SystemClock.currentTimeMicro() - t1) + " microseconds"); 804 } 805 } 806 807 /** 808 * Adds a new message. 809 */ 810 private void queueEvent(final Event event) { 811 ArrayList<EventHandler> eventHandlers = mEventTypeMap.get(event.getClass()); 812 if (eventHandlers == null) { 813 return; 814 } 815 816 // Prepare this event 817 boolean hasPostedEvent = false; 818 event.onPreDispatch(); 819 820 // We need to clone the list in case a subscriber unregisters itself during traversal 821 // TODO: Investigate whether we can skip the object creation here 822 eventHandlers = (ArrayList<EventHandler>) eventHandlers.clone(); 823 int eventHandlerCount = eventHandlers.size(); 824 for (int i = 0; i < eventHandlerCount; i++) { 825 final EventHandler eventHandler = eventHandlers.get(i); 826 if (eventHandler.subscriber.getReference() != null) { 827 if (event.requiresPost) { 828 mHandler.post(new Runnable() { 829 @Override 830 public void run() { 831 processEvent(eventHandler, event); 832 } 833 }); 834 hasPostedEvent = true; 835 } else { 836 processEvent(eventHandler, event); 837 } 838 } 839 } 840 841 // Clean up after this event, deferring until all subscribers have been called 842 if (hasPostedEvent) { 843 mHandler.post(new Runnable() { 844 @Override 845 public void run() { 846 event.onPostDispatch(); 847 } 848 }); 849 } else { 850 event.onPostDispatch(); 851 } 852 } 853 854 /** 855 * Processes and dispatches the given event to the given event handler, on the thread of whoever 856 * calls this method. 857 */ 858 private void processEvent(final EventHandler eventHandler, final Event event) { 859 // Skip if the event was already cancelled 860 if (event.cancelled) { 861 if (event.trace || DEBUG_TRACE_ALL) { 862 logWithPid("Event dispatch cancelled"); 863 } 864 return; 865 } 866 867 try { 868 if (event.trace || DEBUG_TRACE_ALL) { 869 logWithPid(" -> " + eventHandler.toString()); 870 } 871 Object sub = eventHandler.subscriber.getReference(); 872 if (sub != null) { 873 long t1 = 0; 874 if (DEBUG_TRACE_ALL) { 875 t1 = SystemClock.currentTimeMicro(); 876 } 877 eventHandler.method.invoke(sub, event); 878 if (DEBUG_TRACE_ALL) { 879 long duration = (SystemClock.currentTimeMicro() - t1); 880 mCallDurationMicros += duration; 881 mCallCount++; 882 logWithPid(eventHandler.method.toString() + " duration: " + duration + 883 " microseconds, avg: " + (mCallDurationMicros / mCallCount)); 884 } 885 } else { 886 Log.e(TAG, "Failed to deliver event to null subscriber"); 887 } 888 } catch (IllegalAccessException e) { 889 Log.e(TAG, "Failed to invoke method", e.getCause()); 890 } catch (InvocationTargetException e) { 891 throw new RuntimeException(e.getCause()); 892 } 893 } 894 895 /** 896 * Re-registers the broadcast receiver for any new messages that we want to listen for. 897 */ 898 private void registerReceiverForInterprocessEvents(Context context) { 899 if (DEBUG_TRACE_ALL) { 900 logWithPid("registerReceiverForInterprocessEvents()"); 901 } 902 // Rebuild the receiver filter with the new interprocess events 903 IntentFilter filter = new IntentFilter(); 904 for (String eventName : mInterprocessEventNameMap.keySet()) { 905 filter.addAction(eventName); 906 if (DEBUG_TRACE_ALL) { 907 logWithPid(" filter: " + eventName); 908 } 909 } 910 // Re-register the receiver with the new filter 911 if (mHasRegisteredReceiver) { 912 context.unregisterReceiver(this); 913 } 914 context.registerReceiverAsUser(this, UserHandle.ALL, filter, PERMISSION_SELF, mHandler); 915 mHasRegisteredReceiver = true; 916 } 917 918 /** 919 * Returns whether this subscriber is currently registered. If {@param removeFoundSubscriber} 920 * is true, then remove the subscriber before returning. 921 */ 922 private boolean findRegisteredSubscriber(Object subscriber, boolean removeFoundSubscriber) { 923 for (int i = mSubscribers.size() - 1; i >= 0; i--) { 924 Subscriber sub = mSubscribers.get(i); 925 if (sub.getReference() == subscriber) { 926 if (removeFoundSubscriber) { 927 mSubscribers.remove(i); 928 } 929 return true; 930 } 931 } 932 return false; 933 } 934 935 /** 936 * @return whether {@param method} is a valid (normal or interprocess) event bus handler method 937 */ 938 private boolean isValidEventBusHandlerMethod(Method method, Class<?>[] parameterTypes, 939 MutableBoolean isInterprocessEventOut) { 940 int modifiers = method.getModifiers(); 941 if (Modifier.isPublic(modifiers) && 942 Modifier.isFinal(modifiers) && 943 method.getReturnType().equals(Void.TYPE) && 944 parameterTypes.length == 1) { 945 if (EventBus.InterprocessEvent.class.isAssignableFrom(parameterTypes[0]) && 946 method.getName().startsWith(INTERPROCESS_METHOD_PREFIX)) { 947 isInterprocessEventOut.value = true; 948 return true; 949 } else if (EventBus.Event.class.isAssignableFrom(parameterTypes[0]) && 950 method.getName().startsWith(METHOD_PREFIX)) { 951 isInterprocessEventOut.value = false; 952 return true; 953 } else { 954 if (DEBUG_TRACE_ALL) { 955 if (!EventBus.Event.class.isAssignableFrom(parameterTypes[0])) { 956 logWithPid(" Expected method take an Event-based parameter: " + method.getName()); 957 } else if (!method.getName().startsWith(INTERPROCESS_METHOD_PREFIX) && 958 !method.getName().startsWith(METHOD_PREFIX)) { 959 logWithPid(" Expected method start with method prefix: " + method.getName()); 960 } 961 } 962 } 963 } else { 964 if (DEBUG_TRACE_ALL) { 965 if (!Modifier.isPublic(modifiers)) { 966 logWithPid(" Expected method to be public: " + method.getName()); 967 } else if (!Modifier.isFinal(modifiers)) { 968 logWithPid(" Expected method to be final: " + method.getName()); 969 } else if (!method.getReturnType().equals(Void.TYPE)) { 970 logWithPid(" Expected method to return null: " + method.getName()); 971 } 972 } 973 } 974 return false; 975 } 976 977 /** 978 * Sorts the event handlers by priority and registration time. 979 */ 980 private void sortEventHandlersByPriority(List<EventHandler> eventHandlers) { 981 Collections.sort(eventHandlers, EVENT_HANDLER_COMPARATOR); 982 } 983 984 /** 985 * Helper method to log the given {@param text} with the current process and user id. 986 */ 987 private static void logWithPid(String text) { 988 Log.d(TAG, "[" + android.os.Process.myPid() + ", u" + UserHandle.myUserId() + "] " + text); 989 } 990 } 991