Android Automotive navigation(谷歌车载嵌入式操作系统之导航仪表)

前言

随着时代的发展,科技的进步,越来越多的车载系统添加了对液晶仪表系统的支持。不同的车载主机厂商与不同的仪表提供厂商之间对中控仪表互联的协议也不尽相同,由于大家对传输协议都有各自的理解,所以目前市场上出现了多种中控仪与导航应用的互联协议,比较常见的做法是中控系统与导航应用及仪表系统定制AIDL/Broadcast传输协议。


meter.png

但由于这一协议只是使用三方协定的,更换设备提供商之后还需要重新协定,无法复用。谷歌也正式看出了市场上目前存在问题,所以在这一次的Automotive Library中新增了navigation模块,专门解决导航与仪表之间的信息交互。


google_navigation.png

源代码分析

在navigation下有两个文件,一个是CarNavigationInstrumentCluster数据载体,一个是控制中心CarNavigationStatusManager。


目录结构.png

我们先通过一个表格来展现一下CarNavigationInstrumentCluster这个数据载体中所包含的内容。

序号 类型 定义 意义
1 int mMinIntervalMillis 仪表信息最小更新时间
2 @IntDef mType 转向标取值类型
3 int mImageWidth 图片的像素宽度
4 int mImageHeight 图片的像素高度
5 int mImageColorDepthBits 图片的素材度(8,16或者32)
6 Bundle mExtra 额外信息

转向标取值类型是通过Android中提供的注解代替枚举,它的取值一共有两个CLUSTER_TYPE_CUSTOM_IMAGES_SUPPORTED 和 CLUSTER_TYPE_IMAGE_CODES_ONLY分别代表导航转向标信息,可以是图片或枚举和只能是枚举类型的转向标信息。CarNavigationInstrumentCluster的初始化有两种方式:

    /**
     * 创建只支持枚举类型
     * */
    public static CarNavigationInstrumentCluster createCluster(int minIntervalMillis) {
        return new CarNavigationInstrumentCluster(minIntervalMillis, CLUSTER_TYPE_IMAGE_CODES_ONLY,
                0, 0, 0);
    }

    /**
     * 创建支持枚举与图片类型
     * */
    public static CarNavigationInstrumentCluster createCustomImageCluster(int minIntervalMs,
            int imageWidth, int imageHeight, int imageColorDepthBits) {
        return new CarNavigationInstrumentCluster(minIntervalMs,
                CLUSTER_TYPE_CUSTOM_IMAGES_SUPPORTED,
                imageWidth, imageHeight, imageColorDepthBits);
    }

接下来我们来看一下navigation模块的控制中心CarNavigationStatusManager,在CarNavigationStatusManager中有两个主要的方法sendEvent和getInstrumentClusterInfo即数据的发送和接收,我们首先来看一下数据发送:

    private final IInstrumentClusterNavigation mService;
    /**
     * Only for CarServiceLoader
     * @hide
     */
    public CarNavigationStatusManager(IBinder service) {
        mService = IInstrumentClusterNavigation.Stub.asInterface(service);
    }
    /**
     * Sends events from navigation app to instrument cluster.
     * 导航向仪表发送信息
     * <p>The event type and bundle can be populated by
     * {@link android.support.car.navigation.CarNavigationStatusEvent}.
     *
     * @param eventType event type 自定义的事件类型
     * @param bundle object that holds data about the event 自定义事件类型携带的数据
     * @throws CarNotConnectedException if the connection to the car service has been lost.抛出    
     *  车辆未连接异常
     */
    public void sendEvent(int eventType, Bundle bundle) throws CarNotConnectedException {
        try {
            mService.onEvent(eventType, bundle);
        } catch (IllegalStateException e) {
            CarApiUtil.checkCarNotConnectedExceptionFromCarService(e);
        } catch (RemoteException e) {
            handleCarServiceRemoteExceptionAndThrow(e);
        }
    }

在车辆连接状态情况下,调用sendEvent函数会触发IInstrumentClusterNavigation这个AIDL的onEvent,通知仪表导航数据已发生更新,此时仪表可以通过getInstrumentClusterInfo获取CarNavigationInstrumentCluster系统定义的转向类型信息,也可以通过eventType获取导航应用的自定义信息Bundle 。

package com.yanghaoyi.aosp.car.cluster.renderer;
import com.yanghaoyi.aosp.car.navigation.CarNavigationInstrumentCluster;
import android.graphics.Bitmap;
import android.os.Bundle;

interface IInstrumentClusterNavigation {
    void onEvent(int eventType, in Bundle bundle);
    CarNavigationInstrumentCluster getInstrumentClusterInfo();
}

那么导航应用是怎样实现信息流的发送呢?通过对AOSP(Android Open Source Project)的查阅,我们找到了CarNavigationStatusManager的初始化方法,在Car.Java文件中有一个getCarManager方法,这个方法的作用就是通过服务的名称获取对应的Manager,例如我们之前提到的CarNavigationStatusManager,我们就可以通过定义的"car_navigation_service"在Car中获取到CarNavigationStatusManager对象。

    /** Service name for {@link CarNavigationStatusManager} */
    public static final String CAR_NAVIGATION_SERVICE = "car_navigation_service";

    /**
     * Get car specific service as in {@link Context#getSystemService(String)}. Returned
     * {@link Object} should be type-casted to the desired service.
     * For example, to get sensor service,
     * SensorManagerService sensorManagerService = car.getCarManager(Car.SENSOR_SERVICE);
     * @param serviceName Name of service that should be created like {@link #SENSOR_SERVICE}.
     * @return Matching service manager or null if there is no such service.
     * @throws CarNotConnectedException if the connection to the car service has been lost.
     */
    public Object getCarManager(String serviceName) throws CarNotConnectedException {
        CarManagerBase manager;
        ICar service = getICarOrThrow();
        synchronized (mCarManagerLock) {
            manager = mServiceMap.get(serviceName);
            if (manager == null) {
                try {
                    IBinder binder = service.getCarService(serviceName);
                    if (binder == null) {
                        Log.w(CarLibLog.TAG_CAR, "getCarManager could not get binder for service:" +
                                serviceName);
                        return null;
                    }
                    manager = createCarManager(serviceName, binder);
                    if (manager == null) {
                        Log.w(CarLibLog.TAG_CAR,
                                "getCarManager could not create manager for service:" +
                                        serviceName);
                        return null;
                    }
                    mServiceMap.put(serviceName, manager);
                } catch (RemoteException e) {
                    handleRemoteException(e);
                }
            }
        }
        return manager;
    }
  
  //通过服务名创建manager实例
    private CarManagerBase createCarManager(String serviceName, IBinder binder)
            throws CarNotConnectedException {
        CarManagerBase manager = null;
        switch (serviceName) {
            case CAR_NAVIGATION_SERVICE:
                manager = new CarNavigationStatusManager(binder);
                break;
          //...省略其他与示例无关的case
            default:
                break;
        }
        return manager;
    }

信息流发送实例(出自谷歌/car/kitchensink/cluster目录下的InstrumentClusterFragment):

    private CarNavigationStatusManager mCarNavigationStatusManager;
    
        private final CarConnectionCallback mCarConnectionCallback = new CarConnectionCallback() {
            @Override
            public void onConnected(Car car) {
                Log.d(TAG, "Connected to Car Service");
                try {
              //车辆连接情况下,获取CarNavigationStatusManager对象
                    mCarNavigationStatusManager =
                            (CarNavigationStatusManager) mCarApi.getCarManager(CAR_NAVIGATION_SERVICE);
                    mCarAppFocusManager = (CarAppFocusManager) mCarApi.getCarManager(APP_FOCUS_SERVICE);
                } catch (CarNotConnectedException e) {
                    Log.e(TAG, "Car is not connected!", e);
                }
            }

            @Override
            public void onDisconnected(Car car) {
            Log.d(TAG, "Disconnect from Car Service");
        }
    };
    
  //发送导航信息流
    private void sendTurn() {
        // TODO(deanh): Make this actually meaningful.
        Bundle bundle = new Bundle();
        bundle.putString("someName", "someValue time=" + System.currentTimeMillis());
        try {
            mCarNavigationStatusManager.sendEvent(1, bundle);
        } catch(CarNotConnectedException e) {
            Log.e(TAG, "Failed to send turn information.", e);
        }
    }

看到这里你可能会产生一个疑问,导航信息流所发送的数据并没有包含我们一开始提到的CarNavigationInstrumentCluster,那么CarNavigationInstrumentCluster又是在哪里赋值并传递的呢?让我们继续深入源码。通过对CarNavigationInstrumentCluster createCluster的追溯,我们发现在android/car/cluster/sample目录下的SampleClusterServiceImpl类中有对createCluster的调用:

    //初始化CarNavigationInstrumentCluster仪表转向标信息
    @Override
    protected NavigationRenderer getNavigationRenderer() {
        NavigationRenderer navigationRenderer = new NavigationRenderer() {
            @Override
            public CarNavigationInstrumentCluster getNavigationProperties() {
                Log.i(TAG, "getNavigationProperties");
                //创建只支持枚举类型的仪表信息 更新时间1000毫秒
                CarNavigationInstrumentCluster config =
                        CarNavigationInstrumentCluster.createCluster(1000);
                Log.i(TAG, "getNavigationProperties, returns: " + config);
                return config;
            }

            @Override
            public void onEvent(int eventType, Bundle bundle) {
                StringBuilder bundleSummary = new StringBuilder();
                for (String key : bundle.keySet()) {
                    bundleSummary.append(key);
                    bundleSummary.append("=");
                    bundleSummary.append(bundle.get(key));
                    bundleSummary.append(" ");
                }
                Log.i(TAG, "onEvent(" + eventType + ", " + bundleSummary + ")");
            }
        };

        Log.i(TAG, "createNavigationRenderer, returns: " + navigationRenderer);
        return navigationRenderer;
    }

SampleClusterServiceImpl是客户端的实现,它继承自car/cluster/renderer目录下的InstrumentClusterRenderingService,谷歌对这个类的解释是用于导航与仪表之间的通信服务,CarService可以为导航应用提供内部绑定接口。导航应用实现与中控系统的通信还需要一个绑定操作,是通过AIDL文件IInstrumentCluster的setNavigationContextOwner来实现的,通过设置应用唯一标识UID与进程PID建立供应关系。

interface IInstrumentCluster {
    /** Returns {@link IInstrumentClusterNavigation} that will be passed to the Nav app
     * 返回IInstrumentClusterNavigation 将会被传递给导航应用
      * */
    IInstrumentClusterNavigation getNavigationService();

    /**
    * Supplies Instrument Cluster Renderer with current owner of Navigation app context
    * 为仪表渲染提供支持的导航应用
    * @param uid 应用唯一标识
    * @param pid 进程ID
    * */
    oneway void setNavigationContextOwner(int uid, int pid);

    /** Called when key event that was addressed to instrument cluster display has been received. */
    oneway void onKeyEvent(in KeyEvent keyEvent);
}

总结

最后,我们来梳理一下android automotive navigation帮助我们实现的传输协议,首先导航应用需要通过继承InstrumentClusterRenderingService实现一个自己的仪表渲染服务,在这个服务中,导航应用会通过UID与PID向IInstrumentCluster 绑定与仪表的通信关系。在服务的onBind方法中,设置与仪表通信数据的刷新时间,与导航信息流事件。导航也可以使用自定义事件sendEvent向仪表发送自定义信息流数据。仪表通过IInstrumentClusterNavigation的getInstrumentClusterInfo,onEvent获取方向信息数据与导航自定义数据,当然navigation中的实现远不止这么简单,automotive通过CarConnectionCallback帮助我们实现了对车辆连接状态的监听,以及通过Car.Java实现对多个服务控制中心的初始化操作,以及对渲染操作的线程保护等等。


传输示意图 .png

代码

由于谷歌服务是外网环境,为了方便阅读,笔者将文中所提及相关AOSP源代码制作成了模块,已经上传至了GitHub,有兴趣的朋友可以从Github上下载以便对文章的理解。
GitHub代码连接

source_github.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270