android -- 蓝牙 bluetooth (二) 打开蓝牙

分类: Android的原生应用分析 2013-05-23 23:57 4773人阅读 评论(20) 收藏 举报

4.2的蓝牙打开流程这一部分还是有些变化的,从界面上看蓝牙开关就是设置settings里那个switch开关,widget开关当然也可以,起点不 同而已,后续的流程是一样的。先来看systemServer.java的代码,蓝牙服务开启的地方,最后一个else分支是我们关心的,前两个是模拟器 的一个测试模式的。

  1. if (SystemProperties.get("ro.kernel.qemu").equals("1")) {
  2. Slog.i(TAG, "No Bluetooh Service (emulator)");
  3. } else if (factoryTest == SystemServer.FACTORY_TEST_LOW_LEVEL) {
  4. Slog.i(TAG, "No Bluetooth Service (factory test)");
  5. } else {
  6. Slog.i(TAG, "Bluetooth Manager Service");
  7. bluetooth = new BluetoothManagerService(context);
  8. ServiceManager.addService(BluetoothAdapter.BLUETOOTH_MANAGER_SERVICE, bluetooth);
  9. }

暂且看下bluetoothManagerService的构造方法,代码有点多,我们只看两个地 方, loadStoredNameAndAddress()是读取蓝牙打开默认名称的地 方,isBluetoothPersistedStateOn()是判断是否已打开蓝牙的,如果已打开,后续操作要执行开启蓝牙的动作,前面那几行注册广 播其中就有这个作用。

  1. BluetoothManagerService(Context context) {
  2. ...一些变量声明初始化...
  3. IntentFilter filter = new IntentFilter(Intent.ACTION_BOOT_COMPLETED);
  4. filter.addAction(BluetoothAdapter.ACTION_LOCAL_NAME_CHANGED);
  5. filter.addAction(Intent.ACTION_USER_SWITCHED);
  6. registerForAirplaneMode(filter);
  7. mContext.registerReceiver(mReceiver, filter);
  8. loadStoredNameAndAddress();
  9. if (isBluetoothPersistedStateOn()) {
  10. mEnableExternal = true;
  11. }
  12. }

回到界面开关那个看得着的地方,界面上开关就是BluetoothEnabler.java这个类了,而setBluetoothEnabled()则是具体开关动作。看下代码

  1. public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
  2. // Show toast message if Bluetooth is not allowed in airplane mode
  3. if (isChecked &&
  4. !WirelessSettings.isRadioAllowed(mContext, Settings.Global.RADIO_BLUETOOTH)) {
  5. Toast.makeText(mContext, R.string.wifi_in_airplane_mode, Toast.LENGTH_SHORT).show();
  6. // Reset switch to off
  7. buttonView.setChecked(false);
  8. }
  9. if (mLocalAdapter != null) {
  10. mLocalAdapter.setBluetoothEnabled(isChecked);
  11. }
  12. mSwitch.setEnabled(false);
  13. }

这里在判断是飞行模式不知道为什么没有return,如果是飞行模式会有提示toast弹出,既然这样源码为什么还要执行下面打开流程呢,也许是个
bug?不细究这个了,继续看setBluetoothEnabled()方法做什么了,很明显
mLocalAdapter(LocalBluetoothAdapter
)只是个过渡,里面的 mAdapter(BluetoothAdapter)才是真正的主角,代码如下:

  1. public void setBluetoothEnabled(boolean enabled) {
  2. boolean success = enabled   ? mAdapter.enable() : mAdapter.disable();
  3. if (success) {
  4. setBluetoothStateInt(enabled
  5. ? BluetoothAdapter.STATE_TURNING_ON
  6. : BluetoothAdapter.STATE_TURNING_OFF);
  7. } else {
  8. .........
  9. }
  10. }

在BluetoothAdapter.java里可以看到一个单例模式的应用,主要提供给其它程序调用蓝牙的一些方法用的,外部程序想调用蓝牙的方法就要先用这个

拿到BluetoothAdapter对象,代码也简单看下吧,里面是典型的binder应用。

  1. public static synchronized BluetoothAdapter getDefaultAdapter() {
  2. if (sAdapter == null) {
  3. IBinder b = ServiceManager.getService(BLUETOOTH_MANAGER_SERVICE);
  4. if (b != null) {
  5. IBluetoothManager managerService = IBluetoothManager.Stub.asInterface(b);
  6. sAdapter = new BluetoothAdapter(managerService);
  7. } else {
  8. Log.e(TAG, "Bluetooth binder is null");
  9. }
  10. }
  11. return sAdapter;

此时我们更关心mAdapter.enable()的后续操作,外部其它应用到getDefaultAdapter()也是调用enable(),注意,
到了BluetoothAdapter我们已经在framework层了,顺着BluetoothAdapter.java的enable()调用先回到
BluetoothManagerService.java的enable(),再进一步来到BluetoothManagerService.java
中的handleEnable()

方法,后面要跳转到新类了,贴出来一起看下,这部分好像不同版本还有些出入,不过核心的启动service是一样的,不影响理解。

  1. private void handleEnable(boolean persist, boolean quietMode) {
  2. synchronized(mConnection) {
  3. if ((mBluetooth == null) && (!mBinding)) {
  4. //Start bind timeout and bind
  5. Message timeoutMsg=mHandler.obtainMessage(MESSAGE_TIMEOUT_BIND);
  6. mHandler.sendMessageDelayed(timeoutMsg,TIMEOUT_BIND_MS);
  7. mConnection.setGetNameAddressOnly(false);
  8. Intent i = new Intent(IBluetooth.class.getName());
  9. if (!mContext.bindService(i, mConnection,Context.BIND_AUTO_CREATE,
  10. UserHandle.USER_CURRENT)) {
  11. mHandler.removeMessages(MESSAGE_TIMEOUT_BIND);
  12. Log.e(TAG, "Fail to bind to: " + IBluetooth.class.getName());
  13. } else {
  14. mBinding = true;
  15. }
  16. }

下面跑到哪个service里去了呢,在log信息里可以看到"ActivityManager: Start proc
com.android.bluetooth for service
com.android.bluetooth/.btservice.AdapterService:"

这样的信息,那就是去AdapterService里看看,里面一共有三个enable(),跳转关系不复杂,我们直接看最后一个关键的。

  1. public synchronized boolean enable(boolean quietMode) {
  2. enforceCallingOrSelfPermission(BLUETOOTH_ADMIN_PERM,
  3. "Need BLUETOOTH ADMIN permission");
  4. if (DBG)debugLog("Enable called with quiet mode status =  " + mQuietmode);
  5. mQuietmode  = quietMode;
  6. Message m =
  7. mAdapterStateMachine.obtainMessage(AdapterState.USER_TURN_ON);
  8. mAdapterStateMachine.sendMessage(m);
  9. return true;
  10. }

状态机来了,状态转换图,从一个状态接受命令跳到另一个状态,因为我们是在开启蓝牙,所以先去的AdapterState.java内部类offstate.java里面找,在这个分支USER_TURN_ON看到mAdapterService.processStart();在这里面可以看到蓝牙遍历下所支持的profile,最后又发出个带AdapterState.STARTED标识的消息

处理在同文件下面的代码里

  1. case STARTED:   {
  2. if (DBG) Log.d(TAG,"CURRENT_STATE=PENDING, MESSAGE = STARTED, isTurningOn=" + isTurningOn + ", isTurningOff=" + isTurningOff);
  3. //Remove start timeout
  4. removeMessages(START_TIMEOUT);
  5. //Enable
  6. boolean ret = mAdapterService.enableNative();
  7. if (!ret) {
  8. Log.e(TAG, "Error while turning Bluetooth On");
  9. notifyAdapterStateChange(BluetoothAdapter.STATE_OFF);
  10. transitionTo(mOffState);
  11. } else {
  12. sendMessageDelayed(ENABLE_TIMEOUT, ENABLE_TIMEOUT_DELAY);
  13. }

看到那个enableNative()函数调用了吧,又要用到JNI了,稍微回头看下前面的代码,我们先从应用界面开关BluetoothEnabler
走到framework的BluetoothAdapter,又回到package的adapterService,现在又要去JNI的C++代码了,往
常一般是packages -->framework-->下面一层,这次顺序有些颠倒了,不过这不能影响我们跟踪代码,最后

还是要到下面去的。一起往下看吧。

根据android
JNI的函数命名惯例很容易找到enableNative对应的C++函数在packages/apps/Bluetooth/jni
/com_android_bluetooth_btservice_AdapterService.cpp里面

  1. static jboolean enableNative(JNIEnv* env, jobject obj) {
  2. ALOGV("%s:",__FUNCTION__);
  3. jboolean result = JNI_FALSE;
  4. if (!sBluetoothInterface) return result;
  5. int ret = sBluetoothInterface->enable();
  6. result = (ret == BT_STATUS_SUCCESS) ? JNI_TRUE : JNI_FALSE;
  7. return result;
  8. }

代码瞬间简洁了不少,看来更多的故事还在下面,sBluetoothInterface这是什么,直接关系到下一步去哪的问题,看下变量声明,原来是

Const bt_interface_t *sBluetoothInterface = NULL; 再去找在哪初始化,搜索external目录可以找到/external/bluetooth/bluedroid/btif/src/bluetooth.c

  1. static const bt_interface_t bluetoothInterface = {
  2. sizeof(bt_interface_t),
  3. init,
  4. enable,
  5. disable,
  6. .............
  7. start_discovery,
  8. cancel_discovery,
  9. create_bond,
  10. remove_bond,
  11. cancel_bond,
  12. ...............
  13. };


来在这里,说下怎么找到,直接跳转是不成了,看这个文件夹下的mk文件,那里面有libhardware目录是编译的时候要用到,这个多半在
hardware目录里,在这里面很快可以看到bluetooth.h,那里面有最我们要找的结构体定义,头文件找到了,再找同名C文件就快了,好了继续
吧看下enable()里是怎么实现的

  1. static int enable( void )
  2. {
  3. ALOGI("enable");
  4. /* sanity check */
  5. if (interface_ready() == FALSE)
  6. return BT_STATUS_NOT_READY;
  7. return btif_enable_bluetooth();
  8. }

又是一个新函数,直接跳转,比起刚才的寻觅这太幸福了

  1. bt_status_t btif_enable_bluetooth(void)
  2. {
  3. BTIF_TRACE_DEBUG0("BTIF ENABLE BLUETOOTH");
  4. if (btif_core_state != BTIF_CORE_STATE_DISABLED)
  5. {
  6. ALOGD("not disabled\n");
  7. return BT_STATUS_DONE;
  8. }
  9. btif_core_state = BTIF_CORE_STATE_ENABLING;
  10. /* Create the GKI tasks and run them */
  11. bte_main_enable(btif_local_bd_addr.address);
  12. return BT_STATUS_SUCCESS;
  13. }

忘了写路径了 好在可以直接跳转,下面是/external/bluetooth/bluedroid/main/bte_main.c,有点长,暂时只关心set_power那部分就好了,

  1. void bte_main_enable(uint8_t *local_addr)
  2. {
  3. APPL_TRACE_DEBUG1("%s", __FUNCTION__);
  4. ........................
  5. #if (defined (BT_CLEAN_TURN_ON_DISABLED) && BT_CLEAN_TURN_ON_DISABLED == TRUE)
  6. APPL_TRACE_DEBUG1("%s  Not Turninig Off the BT before Turninig ON", __FUNCTION__);
  7. #else
  8. /* toggle chip power to ensure we will reset chip in case
  9. a previous stack shutdown wasn't completed gracefully */
  10. bt_hc_if->set_power(BT_HC_CHIP_PWR_OFF);
  11. #endif
  12. bt_hc_if->set_power(BT_HC_CHIP_PWR_ON);
  13. bt_hc_if->preload(NULL);
  14. }
  15. .............................
  16. }

路径在这里/external/bluetooth/bluedroid/hci/src/bt_hci_bdroid.c,看看set_power里面有什么,快到头了

  1. static void set_power(bt_hc_chip_power_state_t state)
  2. {
  3. int pwr_state;
  4. BTHCDBG("set_power %d", state);
  5. /* Calling vendor-specific part */
  6. pwr_state = (state == BT_HC_CHIP_PWR_ON) ? BT_VND_PWR_ON : BT_VND_PWR_OFF;
  7. if (bt_vnd_if)
  8. bt_vnd_if->op(BT_VND_OP_POWER_CTRL, &pwr_state);
  9. else
  10. ALOGE("vendor lib is missing!");
  11. }

这下又有新东西了bt_vnd_if,这个是什么,bt_vendor_interface_t
*bt_vnd_if=NULL;和刚才的bt_interface_t
一样,我们希望可以找到它的初始化,那样就可以继续跟踪了,不过看到下面的代码和注释,在源码中我们要绝望了。路径:/external
/bluetooth/bluedroid/hci/include/bt_vendor_lib.h

  1. /* Entry point of DLib --
  2. *      Vendor library needs to implement the body of bt_vendor_interface_t
  3. *      structure and uses the below name as the variable name. HCI library
  4. *      will use this symbol name to get address of the object through the
  5. *      dlsym call.
  6. */
  7. extern const bt_vendor_interface_t BLUETOOTH_VENDOR_LIB_INTERFACE;
  8. bt_vendor_interface_t *bt_vnd_if=NULL;

google定义好了接口,具体实现要看vendor厂商来做了,这后面怎么实现就看各家芯片商怎么写了,肯定各有不同,而且这一部分代码一般是不会公
开,当然授权购买后除外了。所以在4.2的源码中我们只跟到这里了,那后面会做什么呢,加载驱动和上电这两项肯定要有了,打开蓝牙没这两步怎么行,类似下
面的字符串

  1. static const char* BT_DRIVER_MODULE_PATH =    "/system/lib/modules/mbt8xxx.ko";
  2. static const char* BT_DRIVER_MODULE_NAME =     "bt8xxx";
  3. static const char* BT_DRIVER_MODULE_INIT_ARG = " init_cfg=";
  4. static const char* BT_DRIVER_MODULE_INIT_CFG_PATH = "bt_init_cfg.conf";

在有类似下面的动作,insmod加载驱动,rfkill控制上下电,具体厂商具体做法也不同。

  1. ret = insmod(BT_DRIVER_MODULE_PATH, arg_buf);
  2. ret = system("/system/bin/rfkill block all");

写到这,关于4.2源码的蓝牙打开流程就算结束了,比起4.1之前的代码感觉没有以前的直观,对于vendor那部分的代码大家只能看各自厂商的代码
了,一般蓝牙开启后才会上电,这样也比较符合逻辑和节省电量,是否上电可以连上手机用adb
shell看sys/class/rfkill目录下的state状态值,有些厂商会把蓝牙和wifi的上电算在一起,这个也是要注意的,小心误判。最后
呢,这次文章写得有点仓促,写错的或遗漏地方希望朋友指出来,非常感谢。

更多
0

 
查看评论
12楼 reality_jie 前天 22:31发表 [回复]
顶一下表示支持!
11楼 默默dede 3天前 17:16发表 [回复]
遇到连接蓝牙设备的时候出现p-data->con.status=12 authentication failed的问题,请问高手们怎么解决?
10楼 vicalwu 2014-01-08 16:52发表 [回复]
文章写的很好,支持楼主继续分享
9楼 syngalon螳螳 2013-10-26 15:35发表 [回复]
今天发现了一篇,看看是不是很像哈哈
http://blog.csdn.net/yutao52shi/article/details/12690353
8楼 躺着享受 2013-10-25 16:13发表 [回复]
就要到我最想看的部分之时嘎然而止,不过博主写的很好!!!!
Re: balmy 2013-10-25 21:57发表 [回复]
回复fankyo:哎,水平不够写不下去了。再深入的还不知道从哪下手呢。
7楼 xyp5299 2013-08-19 18:14发表 [回复]
帮助很大,请楼主继续 分享
6楼 bm7638 2013-08-19 17:57发表 [回复]
写的不错哦,整体思路很清晰,但是跟踪流程过程太随意了,建议能有个流程图保护起来,很多地方直接就跳过了。
Re: balmy 2013-08-25 09:13发表 [回复]
回复bm7638:呵呵 写这个的时候是懒了没有画图
5楼 kuna 2013-06-28 10:36发表 [回复]
请教一下,我一直在看初始化和使能部分log和代码,有一个问题一直没有搞懂:BluetoothAdapterApp是怎么起来的?

谢谢!

4楼 windjkl 2013-06-20 10:38发表 [回复]
设置-打开蓝牙:出现以下错误是什么原因:
D/BluetoothAdapter( 3463): 657778856: getState() : mService = null. Returning STATE_OFF
D/LocalBluetoothManager( 3463): setting foreground activity to non-null context
D/BluetoothAdapter( 3463): 657778856: getState() : mService = null. Returning STATE_OFF
Re: balmy 2013-06-20 21:37发表 [回复]

复windjkl:这个没遇到过呢,不了解你当时的环境,没啥好想法。另外答复你添加USB外置蓝牙模块的问题,也是我咨询来的,仅供参考吧。usb设备
应该都会有注册流程吧,这一部分好像android部分的源码没有,在linux的源码是有的,也就是android源码里kernel。目录下的类似这
个路径:kernel/drivers/bluetooth/btusb.c,按你说的已经识别出来是不是可以认为注册成功了?后续代码没见过,可能没有
也可能是没发现的,目前接触到都是走串口的,目前只知道这些,希望不会误导你。
Re: windjkl 2013-06-21 08:29发表 [回复]
回复baimy1985:谢谢楼主,注册已经成功,现在疑问是service的启动?jb4.2是否还需要在初始化时启动d-bus,hciattach这些服务呢?我看有些方案商的init.rc中并没有这些服务?是不是换成别的方式呢?
3楼 kuna 2013-06-14 16:19发表 [回复]
看了你的文章,受益不浅,不知道4.2时候还有4.0上hcidump那样的分析工具?
Re: balmy 2013-06-16 22:43发表 [回复]
回复kuna:这个我也想知道,目前也在求证中
Re: xuexingyang 2013-06-20 19:27发表 [回复]
回复baimy1985:4.2中hcidump没有了,只有靠confi文件来开关了。
Re: balmy 2013-06-20 21:13发表 [回复]
回复xuexingyang:谢谢分享
2楼 kuna 2013-06-01 15:16发表 [回复]
请教一下:

handleEnable(boolean persist, boolean quietMode)是怎么跳转到启动service的呢?


是“下面跑到哪个service里去了呢,在log信息里可以看到"ActivityManager: Start proc
com.android.bluetooth for service
com.android.bluetooth/.btservice.AdapterService:" ”这段。

Re: balmy 2013-06-02 20:23发表 [回复]
回复kuna:IBluetooth.class.getName()这个当时的取值是AdapterService,可以看下面的4.2.2源码,也许可以帮助你
/packages/apps/Bluetooth/src/com/android/bluetooth/btservice/AdapterService.java 480行的
private static class AdapterServiceBinder extends IBluetooth.Stub

ZT android -- 蓝牙 bluetooth (二) 打开蓝牙的更多相关文章

  1. 【转】Android:Bluetooth 的打开和关闭--不错

    原文网址:http://www.ifeegoo.com/android-turn-on-and-turn-off-bluetooth.html 摘要:Android 中打开和关闭 Bluetooth ...

  2. Bluetooth LE(低功耗蓝牙) - 第三部分

    回顾 在本系列的前两篇文章中,我们已经了解了一些关于Bluetooth LE的背景并建立一个简单的Activity / Service框架.   在这篇文章中,我们将探讨Bluetooth LE的细节 ...

  3. Android Developer -- Bluetooth篇 开发实例之四 API详解

    http://www.open-open.com/lib/view/open1390879771695.html 这篇文章将会详细解析BluetoothAdapter的详细api, 包括隐藏方法, 每 ...

  4. android -- 蓝牙 bluetooth (二) 打开蓝牙

    4.2的蓝牙打开流程这一部分还是有些变化的,从界面上看蓝牙开关就是设置settings里那个switch开关,widget开关当然也可以,起点不同而已,后续的流程是一样的.先来看systemServe ...

  5. ZT android -- 蓝牙 bluetooth (三)搜索蓝牙

    android -- 蓝牙 bluetooth (三)搜索蓝牙 分类: Android的原生应用分析 2013-05-31 22:03 2192人阅读 评论(8) 收藏 举报 bluetooth蓝牙s ...

  6. ZT android -- 蓝牙 bluetooth (五)接电话与听音乐

    android -- 蓝牙 bluetooth (五)接电话与听音乐 分类: Android的原生应用分析 2013-07-13 20:53 2165人阅读 评论(9) 收藏 举报 蓝牙android ...

  7. ZT android -- 蓝牙 bluetooth (四)OPP文件传输

    android -- 蓝牙 bluetooth (四)OPP文件传输 分类: Android的原生应用分析 2013-06-22 21:51 2599人阅读 评论(19) 收藏 举报 4.2源码AND ...

  8. ZT android -- 蓝牙 bluetooth (一) 入门

    android -- 蓝牙 bluetooth (一) 入门 分类: Android的原生应用分析 2013-05-19 21:44 4543人阅读 评论(37) 收藏 举报 bluetooth4.2 ...

  9. Android开发之蓝牙(Bluetooth)操作(二)--修改本机蓝牙设备的可见性,并扫描周围可用的蓝牙设备

    版权声明:本文为博主原创文章,未经博主允许不得转载. 一. 修改本机蓝牙设备的可见性 二. 扫描周围可用的蓝牙设备 Eg: 一.  清单文件AdroidManifest.xml: <?xml v ...

随机推荐

  1. [PY3]——合并多个字典或映射(collections模块中的ChainMap 类)

    问题 现在有多个字典或者映射,你想将它们从逻辑上合并为一个单一的映射后执行某些操作, 比如查找值或者检查某些键是否存在. 解决方案 使用 collections 模块中的 ChainMap 类 Cha ...

  2. PHP命名规则

    参考: http://nowhisky.diandian.com/post/2012-08-12/40033898638 就一般约定而言,类.函数和变量的名字应该是能够让代码阅读者能够容易地知道这些代 ...

  3. [转]Repeat Page Header on each Page for reports SSRS

    本文转自:https://stackoverflow.com/questions/3475144/i-want-to-repeat-page-header-on-each-page-for-repor ...

  4. 如鹏网学习笔记(五)MySql基础

    MySQL基础 一.数据库概念 1,网友装备信息.论坛帖子信息.QQ好友关系信息.学籍管理系统中的学生信息等都要“持久化”的保存到一个地方, 如果通过IO写到文件中,那么会非常麻烦,而且不利于多人共享 ...

  5. 记一次tomcat自动退出问题

    问题 环境: centos/tomcat8/jdk1.8 最近遇到部署在服务器的tomcat总是过一段时间就自动结束进程 ; 通过监控tomcat 日志文件(tail -f ./logs/catali ...

  6. HDU X问题 中国剩余定理--求满足条件的个数

    X问题 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...

  7. 常用SEO优化

  8. VUE配置项结构

    VUE配置项结构 config:项目的配置文件 index.js: 基础的配置信息 dev.env.js:开发环境配置信息 prod.env.js:线上环境配置信息 build: 项目打包所需要的内容 ...

  9. 测试canvas绘制旋转文字的性能

    canvas 绘制各种动画效果时,我们经常会使用画布旋转,使绘制上去的元素有旋转的效果. 最近在项目中碰到了很严重的性能问题,经常排查发现是因为绘制批量文字时使用了画布旋转,且每行文字的旋转角度是不一 ...

  10. 001Spring Boot中使用MongoDB

    01.下载MongoDB 点击标题链接,下载windows可用的MongoDB. 02.解压 将下载的压缩包放入C盘根目录(根据自己需要调整目录)---->解压到当前文件夹---->重命名 ...