Home | History | Annotate | Download | only in renderer
      1 /*
      2  * Copyright (C) 2016 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 package android.car.cluster.renderer;
     17 
     18 import android.annotation.Nullable;
     19 import android.car.navigation.CarNavigationInstrumentCluster;
     20 import android.graphics.Bitmap;
     21 import android.os.Bundle;
     22 import android.os.Handler;
     23 import android.os.Looper;
     24 import android.os.Message;
     25 
     26 import java.lang.ref.WeakReference;
     27 import java.util.concurrent.CountDownLatch;
     28 
     29 /**
     30  * A wrapper over {@link NavigationRenderer} that runs all its methods in the context of provided
     31  * looper. It is guaranteed that all calls will be invoked in order they were called.
     32  */
     33 // TODO(deanh): Does this class even need to exist?
     34 /* package */ class ThreadSafeNavigationRenderer extends NavigationRenderer {
     35 
     36     private final Handler mHandler;
     37     private final NavigationRenderer mRenderer;
     38 
     39     private final static int MSG_EVENT = 1;
     40 
     41     /** Creates thread-safe {@link NavigationRenderer}. Returns null if renderer == null */
     42     @Nullable
     43     static NavigationRenderer createFor(Looper looper, NavigationRenderer renderer) {
     44         return renderer == null ? null : new ThreadSafeNavigationRenderer(looper, renderer);
     45     }
     46 
     47     private ThreadSafeNavigationRenderer(Looper looper, NavigationRenderer renderer) {
     48         mRenderer = renderer;
     49         mHandler = new NavigationRendererHandler(looper, renderer);
     50     }
     51 
     52     @Override
     53     public CarNavigationInstrumentCluster getNavigationProperties() {
     54         if (mHandler.getLooper() == Looper.myLooper()) {
     55             return mRenderer.getNavigationProperties();
     56         } else {
     57             return runAndWaitResult(mHandler,
     58                     new RunnableWithResult<CarNavigationInstrumentCluster>() {
     59                         @Override
     60                         protected CarNavigationInstrumentCluster createResult() {
     61                             return mRenderer.getNavigationProperties();
     62                         }
     63                     });
     64         }
     65     }
     66 
     67     @Override
     68     public void onEvent(int eventType, Bundle bundle) {
     69         mHandler.sendMessage(mHandler.obtainMessage(MSG_EVENT, eventType, 0, bundle));
     70     }
     71 
     72     private static class NavigationRendererHandler extends RendererHandler<NavigationRenderer> {
     73 
     74         NavigationRendererHandler(Looper looper, NavigationRenderer renderer) {
     75             super(looper, renderer);
     76         }
     77 
     78         @Override
     79         public void handleMessage(Message msg, NavigationRenderer renderer) {
     80             switch (msg.what) {
     81                 case MSG_EVENT:
     82                     Bundle bundle = (Bundle) msg.obj;
     83                     renderer.onEvent(msg.arg1, bundle);
     84                     break;
     85                 default:
     86                     throw new IllegalArgumentException("Msg: " + msg.what);
     87             }
     88         }
     89     }
     90 
     91     private static <E> E runAndWaitResult(Handler handler, final RunnableWithResult<E> runnable) {
     92         final CountDownLatch latch = new CountDownLatch(1);
     93 
     94         Runnable wrappedRunnable = new Runnable() {
     95             @Override
     96             public void run() {
     97                 runnable.run();
     98                 latch.countDown();
     99             }
    100         };
    101 
    102         handler.post(wrappedRunnable);
    103 
    104         try {
    105             latch.await();
    106         } catch (InterruptedException e) {
    107             throw new RuntimeException(e);
    108         }
    109         return runnable.getResult();
    110     }
    111 
    112     private static abstract class RunnableWithResult<T> implements Runnable {
    113         private volatile T result;
    114 
    115         protected abstract T createResult();
    116 
    117         @Override
    118         public void run() {
    119             result = createResult();
    120         }
    121 
    122         public T getResult() {
    123             return result;
    124         }
    125     }
    126 
    127     private static abstract class RendererHandler<T> extends Handler {
    128 
    129         private final WeakReference<T> mRendererRef;
    130 
    131         RendererHandler(Looper looper, T renderer) {
    132             super(looper);
    133             mRendererRef = new WeakReference<>(renderer);
    134         }
    135 
    136         @Override
    137         public void handleMessage(Message msg) {
    138             T renderer = mRendererRef.get();
    139             if (renderer != null) {
    140                 handleMessage(msg, renderer);
    141             }
    142         }
    143 
    144         public abstract void handleMessage(Message msg, T renderer);
    145     }
    146 }
    147