背景

最近在做百度carlife手机互联项目,项目要求支持无线连接,USB连接和无感连接,USB连接要支持Android手机与Iphone手机连接。我主要负责的是iphone手机与车机连接的功能,现将主要技术要点做个简单的总结。

连接流程

根据百度和苹果给的文档iphone车机Carlife连接要经过经下几步才能够通信。

  • 手机插入车机后,车机会根据枚举的信息(设备VID、PID)来判断当前是什么类型

的手机,如果不是苹果手机仍然保持HOST 模式。

  • 检测到是苹果手机,车机会通过USB 的EP0 给手机发送role switch 命令,来切换模式

手机收到role switch 命令后,首先会判断是否支持,手机不支持role switch 命令,HU 不会做任何处理手机支持role switch 命令,车机会主动断开与手机的USB 连接,并拉到D+等待。

  • 手机来重新枚举。
  • 车机在切换到从模式时,需要想手机上报两个USB 端口:EA Native Transport、IAP2端口。分别用来做CarLife 的数据传输和苹果的IAP2 认证。
  • 通过IAP2端口进行认证。
  • 认证成功后通过EAP端口进行通信。

步骤说明

设备枚举

手机插入车机后,车机会根据枚举的信息(设备VID、PID)来判断当前是什么类型

的手机,如果不是苹果手机仍然保持HOST 模式。

手机枚举过程:由于项目使用的车机是安卓系统,所以我们使用了Android DeviceManager提

供的API和接收安卓接供的USB插入的时收到的广播来来获取当前设备的实例。

主要代码如下:

UsbManager mUsbManager = (UsbManager) mContext.getSystemService(Context.USB_SERVICE);
//获取USBManager.
HashMap<String, UsbDevice> deviceList = mUsbManager.getDeviceList();
   //遍历设备
    Iterator<UsbDevice> deviceIterator = deviceList.values().iterator();
    while (deviceIterator.hasNext()) {
      UsbDevice device = deviceIterator.next();
       If (device == null) {
          continue;
       } 
}
   //遍历设备      
     mUsbManager.requestPermission(device, mPermissionIntent);
    //请求权限
     int vid = device.getVendorId();
     int pid = device.getProductId();
   //获取设备VID和PID来识别当前设备是否是Iphone

模式切换

检测到是苹果手机,车机会通过USB的EP0 给手机发送role switch 命令,来切换模式

手机收到role switch 命令后,首先会判断是否支持,手机不支持role switch 命令,HU 不会做任何处理手机支持role switch 命令,车机会主动断开与手机的USB 连接,并拉到D+等待。

  1. 首先通示USB的EP0端口来将手机由从模式切换到主模式。

切换方法指令如下:

bmRequest:0x40
bRequest:0x51
wValue:0x00
wIndex:0x00
wLength:0x00

     Android发送指令示令程序:

 

mUsbDeviceConnection = mUsbManager.openDevice(mUsbDevice); //与USB设备建立连接
int ret = mUsbDeviceConnection.controlTransfer(0x40, 0x51,0x00, 0x00, null, 0x00, 0);//切换
  1. 再将车机由主模式切换到从模式

将车机由主模式切到从模式,这个每个车机都不同,基本方法为向USB的控制器的节点里写入特定的值。

重新枚举

手机来重新枚举车机,这一步是由手机完成的,但是需要注意如下问题。

车机为主手机为从设备时,车机在给手机供电,当进行模式切换后正常情况车机不会再给手机供电,需要查看一下是否能够正常枚举,如果不能需要给车机的vbus上电。以便车机继续给手机供电完成枚举。

端口上报

车机在切换到从模式时,需要想手机上报两个USB 端口:EA Native Transport、IAP2端口。分别用来做CarLife 的数据传输和苹果的IAP2 认证。

具体端口中的参数请参照Carlife文档。这一步是由驱动程序完成的。需要向外暴露出两个备节点。/dev/eap和/dev/iap(名字可以更改)。这部分主要用到的技术为USB gadget驱动开发相关的技术。

可以参照gadget驱动中的其它代码完成。由于本人非驱动开发者具体技术细节不再叙述。现在把这要更技术点说明如下:

  1. 修改kernel里面的代码,包括在drivers/usb/gadget/function里面添加对应的驱动文件(驱动需要自己实现)。
  2. 修改对应的KConfig和makefile文件。使得加入的文件和功能可被配置进内核。
  3. 修改相关的启动文件。如init.rc或是其它脚本,使得编译出来的ko文件可以被插入到内核中。insert iap.ko/insert eap.ko
  4. 在配置configfs的脚本添加相关的功能脚本.这个过程网上有人写过详细过程。

可以在网上找一下。

5、编译。

认证过程

经过上面的几个步骤,我们就可以开始进行iap认证了。IAP认证过程如下图所示:

    

三星手机怎么连接Android studio 三星手机怎么连接carlife_android

 

手机与HU通过IAP端口进行通信。认证程序收到对应的认证请求后,通过去挂载在i2c总线上的MFI芯片进行认证,MFI芯片把认证后的数据发送给认证程序。认证程序再发给手机。

LINK

  在开始认证之前车机要与手机进行Link操作。车机需要给手机发送如下指令:

FF 55 02 00 EE 10

  成功会给车机返回

FF 55 02 00 EE 10

协商

   连接成功后需要和手机进行通信前的协商。

通信报文

  

三星手机怎么连接Android studio 三星手机怎么连接carlife_序列号_02

 

在与IAP通信的过程中这个报文格式是一直在使用的,唯一不同的是协商的payload数据的格式与认证的不同。

协商数据报文格式

 

三星手机怎么连接Android studio 三星手机怎么连接carlife_android_03

具体的传输协议需要阅读苹果的相关文档。这部分苹果给源码,大家可以参照源码来实现。

认证流程

协商完成后就可以进行认证了。具体的认证过程如下:

(需要注意的是协商的session version版本号是1还是2)

三星手机怎么连接Android studio 三星手机怎么连接carlife_Android_04

 

认证Payload数据格式

 

三星手机怎么连接Android studio 三星手机怎么连接carlife_数据_05

后面所有的数据格式都是control session data.

 

EAP功能认证

  功能认证流程

三星手机怎么连接Android studio 三星手机怎么连接carlife_android_06

 

IdentificationInformation 命令的具体含义请参照苹果文档中的IdentificationInformation一节,IdentificationInformation中SupportedExternalAccessoryProtocol及USBHostTransportComponent是非常重要的两个参数。SupportedExternalAccessoryProtocol 命令中的ExternalAccessoryProtocolName 都是com.baidu.CarLifeVehicleProtocol。

关于IdentificationInformation参数需要注意以下几点:

utf8字符串需要带上’\0’。也就是说如果使用strlen来求取长度,求得的长度需要加1.

如果使用memcpy(str1,str2,strlen(str2))来拷贝字符串,需要加上0x00.

拉起手机端APP

  拉起手起端APP车机向手机发送一条EA02的指令。需要选择无提示的拉起,否则不好处理EAP的读取。

数据通信

打开EAP节点读取数据。

 

传输协议介绍

三星手机怎么连接Android studio 三星手机怎么连接carlife_序列号_07

 

Start of Packet MSB(0xFF): 固定值FF

Start of Packet LSB (0x5A): 固定值0x5A

Package Length MSB和Package Length LSB:两字节报文长度,为报文的全部长度从Start of Packet MSB开始到Payload CheckSum的长度。

Control Byte:控制标识:SYN:0x80,ACK:0x40,AYN&ACK:0xc0,RST:0x10

Packaget Sequence Number:序列号0~255,初始值为随机数,每发一条报文需要加1.

Package Acknowledgement number: ACK序列号,为收到报文的序列号。如果收到连序的报文,序列号为1,2,3,5.这里候回ACK的序列号为3,因为5不是连续的。

Session identifier :session 标识符。

Checksum:从Start of Package MSB到session identifier的checksum值。

Payload checksum: Payload的checksum值。

 

Checksum的计算方法:

苹果文档上面有给出具体的函数。

 

 

 

Payload报文

协商报文

 

Link Version:固定值0x01.

三星手机怎么连接Android studio 三星手机怎么连接carlife_序列号_08

下面的数据不同的传输方式有不同的值,苹果文档中有给出建议值,可以参照。

 

 

 

 

 

Control Session Payload

三星手机怎么连接Android studio 三星手机怎么连接carlife_数据_09

 

Start of Message MSB :固定值 0x40.

Start of Message LSB :固定值0x40.

 

Message Length : Message的长度包括参数,包括Start of Message MSB和Start of Message LSB

和长度。

Message ID:需要发送的Message ID.

Parameter :参数。

参数格式:

Parameter Length MSB

Parameter Length LSB

Parameter ID MSB

Parameter ID LSB

Param Data

 

例子:比如要发送0xAA00这条消息:

 0xFF,0x5A,0x00,0x10,0x40,0xA7,0xE5,0x01,0xCA,0x40,0x40,0x00,0x06,0xAA,0x00,0x0d

红色部分为控制报文,黑色部分为Control session payload.

对上面的报文回复ACK为:0xFF,0x5A,0x00,0x09,0x40,0xE5,0xA7,0x00,0xd2

主要关注蓝色部分的序列号和ACK序列号。回复ACK时序列号不需要加1.

协商完成后所有的数据的control bytes都是0x40 ACK。但是回复ACK时不需要带数据只需要发送9个字节的报文。发送Message时有数据序列号需要加1.

例子:对上面的报文做带数据的回复

      0xAA00这条消息是手机发送给车机请求进行IAP证,车机端需要回复0xAA01证书数据给手机,回复报文如下:

0xFF,0x5A,0x00,0x10,0x40,0xE6,0xA7,0x01,0x62,0x40,0x40,0x02,0x6b,0xAA,0x01, 0xxx,0x0xx,0x00,0x00,00……..

多个参数传输:

0xFF,0x5A,0x00,0x10,0x40,0xE6,0xA7,0x01,0x62,0x40,0x40,0x02,0x6b,0xEA,0x02,0x00,0x05,0x00,0x00,00
0x00,0x06,0x01,0x00,0x01.

这条报文有两个参数每别是0和1,从紫色部分开始。