上一节我们看到在启动AndroidDebugBridge的过程中会调用其start方法,而该方法会做2个主要的事情:
其中第一点我们上一小节已经做了详尽分析了,那么我们往下就去分析下第2点。
DeviceMonitor主要的功能就是监控是否有安卓设备拔除或者连接上来,然后对每个连接上来的设备的所有可调式进程进行监控。这个功能更多是给DDMS这个调试工具使用的,大家从AndroidDebugBridge所在的jar包ddmlib.jar应该就能猜到其实DDMS用到的也是这个jar包。
Android的每一个应用程序都是运行在独立的进程中的,且每个进程都是在自己的虚拟机(VM)中运行的,其实这个就基本上是Android操作系统的沙箱(SandBox)的概念了。目的就是以防你的进程到处乱闯搞破坏。每个进程的VM都会提供唯一的一个端口作连接调试器用。DDMS启动时会连接到AndroidDebugBridge(也就是ADB服务器了)。当一个设备被连接时,一个虚拟机在设备上启动或停止时它会通知DDMS,在DDMS和adb之间创建一个虚拟机的监控服务。一旦虚拟机运行,DDMS会通过AndroidDebugBridge的DeviceMonitor线程检索虚拟机的进程ID(pid),并通过设备上的adb守护进程打开一个连接到虚拟机的调试器。这样DDMS就能和虚拟机对话了。
图8-4-1 DDMS界面
DeviceMonitor这个类对于MonkeyRunner框架来说最重要的功能是它保存了一个代表所有连接上来的设备的Device列表,这个Device类非常的重要,基本可以说MonkeyRunner和ADB服务器的沟通最终都是通过它来做的,所以它相当于一个安卓设备在PC端的代理,我们下一章会对这个类做详细的分析。下面我们先看下DeviceMonitor这个类:
图8-4-2 DeviceMonitor类
这个类是ddmlib库的一个类,上面类图列出了DeviceMonitor关键的成员变量和方法,我们先对它们做初步的介绍,这样有利于大家更好的跟随我在下面对这个类的源代码进行分析:
有了以上的基本认识后,我们继续往下分析其代码实现。AndroidDebugBridge启动起来后,下一步就是把这个ADB实例传到DeviceMonitor来去监测所有连接到adb服务器也就是pc主机端的android设备的状态:
70 DeviceMonitor(AndroidDebugBridge server) 71 { 72 this.mServer = server; 73 74 this.mDebuggerPorts.add(Integer.valueOf(DdmPreferences.getDebugPortBase())); 75 }代码8-4-1 DeviceMonitor构造函数
保存好AndroidDebugBridge实例后,下一步就是继续AndroidDebugBridge启动函数start()启动DeviceMonitor设备监控线程:
79 void start() 80 { 81 new Thread("Device List Monitor") 82 { 83 public void run() { 84 DeviceMonitor.this.deviceMonitorLoop(); 85 } 86 }.start(); 87 }代码8-4-2 DeviceMonitor - start
第81-86行,整个方法的主体就是创建一个”Device List Monitor”的线程。线程运行方法run直接调用DeviceMonitor的deviceMonitorLoop方法来进行无限循环监控设备状态了。
155 private void deviceMonitorLoop() 156 { 157 do 158 { 159 try 160 { 161 if (this.mMainAdbConnection == null) { 162 Log.d("DeviceMonitor", "Opening adb connection"); 163 this.mMainAdbConnection = openAdbConnection(); 164 if (this.mMainAdbConnection == null) { 165 this.mConnectionAttempt += 1; 166 Log.e("DeviceMonitor", "Connection attempts: " + this.mConnectionAttempt); 167 if (this.mConnectionAttempt > 10) { 168 if (!this.mServer.startAdb()) { 169 this.mRestartAttemptCount += 1; 170 Log.e("DeviceMonitor", "adb restart attempts: " + this.mRestartAttemptCount); 171 } 172 else { 173 this.mRestartAttemptCount = 0; 174 } 175 } 176 waitABit(); 177 } else { 178 Log.d("DeviceMonitor", "Connected to adb for device monitoring"); 179 this.mConnectionAttempt = 0; 180 } 181 } 182 183 if ((this.mMainAdbConnection != null) && (!this.mMonitoring)) { 184 this.mMonitoring = sendDeviceListMonitoringRequest(); 185 } 186 187 if (this.mMonitoring) 188 { 189 int length = readLength(this.mMainAdbConnection, this.mLengthBuffer); 190 191 if (length >= 0) 192 { 193 processIncomingDeviceData(length); 194 195 196 this.mInitialDeviceListDone = true; 197 } 198 } 199 } 200 catch (AsynchronousCloseException ace) {}catch (TimeoutException ioe) 201 { 202 handleExpectionInMonitorLoop(ioe); 203 } catch (IOException ioe) { 204 handleExpectionInMonitorLoop(ioe); 205 } 206 } while (!this.mQuit); 207 }代码8-4-3 DeviceMonitor - deviceMonitorLoop
我们先看第一步,在上一节中我们已经看到ADB服务器的启动过程了,但是我们还没有看到ADB客户端是怎么连接上服务器的,一下的代码就是一个实例:
255 private SocketChannel openAdbConnection() 256 { 257 Log.d("DeviceMonitor", "Connecting to adb for Device List Monitoring..."); 258 259 SocketChannel adbChannel = null; 260 try { 261 adbChannel = SocketChannel.open( AndroidDebugBridge.getSocketAddress()); 262 adbChannel.socket().setTcpNoDelay(true); 263 } 264 catch (IOException e) {} 265 266 return adbChannel; 267 }代码8-4-4 DeviceMonitor - openAdbConnection
261行创建一个和ADB服务器监听的Socket端口的一个异步非阻塞SocketChannel连接,该连接就是专门用于往后往ADB服务器发送命令用的,返回给deviceMonitorLoop方法后会被保存到mMainAdbConnection中,请大家记住它,我们往下会用到它。
第二步关于如何调用startAdb来开启ADB服务器是上一节的重点,所以我们不会重新分析了。
第三步是向ADB服务器发送设备监控命令,我们跳进去:
272 private boolean sendDeviceListMonitoringRequest() 273 throws TimeoutException, IOException 274 { 275 byte[] request = AdbHelper.formAdbRequest("host:track-devices"); 276 try 277 { 278 AdbHelper.write(this.mMainAdbConnection, request); 279 280 AdbHelper.AdbResponse resp = AdbHelper.readAdbResponse(this.mMainAdbConnection, false); 281 282 283 if (!resp.okay) 284 { 285 Log.e("DeviceMonitor", "adb refused request: " + resp.message); 286 } 287 288 return resp.okay; 289 } catch (IOException e) { 290 Log.e("DeviceMonitor", "Sending Tracking request failed!"); 291 this.mMainAdbConnection.close(); 292 throw e; 293 } 294 }代码8-4-5 DeviceMonitor - sendDeviceListMonitoringRequest
整个方法的功能就是去构建一个发送到ADB服务器请求服务的命令,然后发送,读取结果,返回,错误处理:
对应的 AdbHelper相应方法的实现细节我们下一章会详尽描述。这里我们只需要清楚这个方法做的事情就是刚才提到的往ADB服务器发送”host:track-devices”命令去请求相应设备监控服务就够了。那么这个服务请求是怎么回事呢?其实这个在第一章中已经描述过,用来就是让ADB服务器周期性的往客户端,也就是往这里的DeviceMonitor线程发送设备更新列表:
host:track-devices
这个服务是以上的host:devices的一个变种,客户端和ADB服务器的连接会一直保持,当有增加/移除设备或者设备状态改变的时候会主动的往连接上的客户端发送新的设备列表信息(4字节16进制长度+内容)。这样做的话就可以允许DDMS这些工具来实时跟踪所有连接上来的设备的状态,而不需要客户端每次都去连接ADB服务器获取对应信息。
最终这个命令返回格式跟你在命令行调用ADB命令行客户端发送命令”adb devices”,返回来的就是“设备序列号 设备状态”的格式: 图8-4-2 adb devices命令返回结果
注意这里device的状态其实就是oneline, 在ddmlib的IDevice类中有相应的定义:
86 public static enum DeviceState { BOOTLOADER("bootloader"), 87 OFFLINE("offline"), 88 ONLINE("device"), 89 RECOVERY("recovery"), 90 UNAUTHORIZED("unauthorized"); 91 92 private String mState; 93 94 private DeviceState(String state) { 95 this.mState = state; 96 }代码8-4-6 IDevice - DeviceState
“host:track-devices”这个监控请求命令发送一次后就会不停的周期性获得ADB服务器发送过来的设备列表,所以前面的deviceMonitorLoop循环在第一次循环之后其实可以简化成以下几行代码:
155 private void deviceMonitorLoop() 156 { 157 do 158 { 159 try 160 { ... 187 if (this.mMonitoring) 188 { 189 int length = readLength(this.mMainAdbConnection, this.mLengthBuffer); 190 191 if (length >= 0) 192 { 193 processIncomingDeviceData(length); 194 195 196 this.mInitialDeviceListDone = true; 197 } 198 } 199 } ... 206 } while (!this.mQuit); 207 }代码8-4-3 DeviceMonitor - deviceMonitorLoop简化版
所以最终重点就是调用processIncomingDeviceData来处理更新后的设备列表。
296 private void processIncomingDeviceData(int length) throws IOException 297 { 298 ArrayList<Device> list = new ArrayList(); 299 300 if (length > 0) { 301 byte[] buffer = new byte[length]; 302 String result = read(this.mMainAdbConnection, buffer); 303 304 String[] devices = result.split("\n"); 305 306 for (String d : devices) { 307 String[] param = d.split("\t"); 308 if (param.length == 2) 309 { 310 Device device = new Device(this, param[0], IDevice.DeviceState.getState(param[1])); 311 312 313 314 list.add(device); 315 } 316 } 317 } 318 319 320 updateDevices(list); 321 }
代码8-4-4 DeviceMonitor - processIncomingDeviceData
方法体首先在302行获得ADB服务器发送过来的”设备序列号 状态”的设备列表,然后在后续几行循环将每个设备的序列号的和状态解析出来,然后就是根据序列号和状态创建一个代表该设备的Device对象并把其存储到一个列表list里面,最后就是调用updateDevices方法进行下一步动作了。这了我们先简单看下Device的构造函数,至于它的详细分析将会留到下一章。
677 Device(DeviceMonitor monitor, String serialNumber, IDevice.DeviceState deviceState) 678 { 679 this.mMonitor = monitor; 680 this.mSerialNumber = serialNumber; 681 this.mState = deviceState; 682 }代码8-4-5 Device构造函数
Device的构造函数非常简单,就是把上面传进来的DeviceMonitor实例,ADB服务器主动发送过来的设备的序列号字串,以及设备当前的状态给保存起来到对应的成员变量中而已。这里要注意的的士mSerialNumber和mState,在往下分析“启动Monkey“的时候我们需要用到。
我们往下继续分析updateDevices这个方法。这个方法的代码稍微长那么一点点,我们把它分开来分析:
323 /** 324 * Updates the device list with the new items received from the monitoring service. 325 */ 326 private void updateDevices(ArrayList<Device> newList) { 327 // because we are going to call mServer.deviceDisconnected which will acquire this lock 328 // we lock it first, so that the AndroidDebugBridge lock is always locked first. 329 synchronized (AndroidDebugBridge.getLock()) { 330 // array to store the devices that must be queried for information. 331 // it's important to not do it inside the synchronized loop as this could block 332 // the whole workspace (this lock is acquired during build too). 333 ArrayList<Device> devicesToQuery = new ArrayList<Device>(); 334 synchronized (mDevices) { 335 // For each device in the current list, we look for a matching the new list. 336 // * if we find it, we update the current object with whatever new information 337 // there is 338 // (mostly state change, if the device becomes ready, we query for build info). 339 // We also remove the device from the new list to mark it as "processed" 340 // * if we do not find it, we remove it from the current list. 341 // Once this is done, the new list contains device we aren't monitoring yet, so we 342 // add them to the list, and start monitoring them. 343 344 for (int d = 0 ; d < mDevices.size() ;) { 345 Device device = mDevices.get(d); 346 347 // look for a similar device in the new list. 348 int count = newList.size(); 349 boolean foundMatch = false; 350 for (int dd = 0 ; dd < count ; dd++) { 351 Device newDevice = newList.get(dd); 352 // see if it matches in id and serial number. 353 if (newDevice.getSerialNumber().equals(device.getSerialNumber())) { 354 foundMatch = true; 355 356 // update the state if needed. 357 if (device.getState() != newDevice.getState()) { 358 device.setState(newDevice.getState()); 359 device.update(Device.CHANGE_STATE); 360 361 // if the device just got ready/online, we need to start 362 // monitoring it. 363 if (device.isOnline()) { 364 if (AndroidDebugBridge.getClientSupport()) { 365 if (!startMonitoringDevice(device)) { 366 Log.e("DeviceMonitor", 367 "Failed to start monitoring " 368 + device.getSerialNumber()); 369 } 370 } 371 372 if (device.getPropertyCount() == 0) { 373 devicesToQuery.add(device); 374 } 375 } 376 } 377 378 // remove the new device from the list since it's been used 379 newList.remove(dd); 380 break; 381 } 382 } 383 384 if (!foundMatch) { 385 // the device is gone, we need to remove it, and keep current index 386 // to process the next one. 387 removeDevice(device); 388 mServer.deviceDisconnected(device); 389 } else { 390 // process the next one 391 d++; 392 } 393 } ... }代码8-4-5 DeviceMonitor - updateDevices处理移除和状态改变设备
这一部分的代码逻辑关系是这样的:
以上代码是设备被移除和设备状态有更新时候的处理,那么新设备该怎么处理呢?毕竟一开始设备都是新的,这个才是关键点。
326 private void updateDevices(ArrayList<Device> newList) { ... 395 // at this point we should still have some new devices in newList, so we 396 // process them. 397 for (Device newDevice : newList) { 398 // add them to the list 399 mDevices.add(newDevice); 400 mServer.deviceConnected(newDevice); 401 402 // start monitoring them. 403 if (AndroidDebugBridge.getClientSupport()) { 404 if (newDevice.isOnline()) { 405 startMonitoringDevice(newDevice); 406 } 407 } 408 409 // look for their build info. 410 if (newDevice.isOnline()) { 411 devicesToQuery.add(newDevice); 412 } 413 } 414 } 415 416 // query the new devices for info. 417 for (Device d : devicesToQuery) { 418 queryNewDeviceForInfo(d); 419 } 420 } 421 newList.clear(); 422 }代码8-4-6 DeviceMonitor - updateDevices处理新增加设备
这里我们先重点看405行startMonitoringDevice:
509 private boolean startMonitoringDevice(Device device) { 510 SocketChannel socketChannel = openAdbConnection(); 511 512 if (socketChannel != null) { 513 try { 514 boolean result = sendDeviceMonitoringRequest( socketChannel, device); 515 if (result) { 516 517 if (mSelector == null) { 518 startDeviceMonitorThread(); 519 } 520 521 device.setClientMonitoringSocket(socketChannel); 522 523 synchronized (mDevices) { 524 // always wakeup before doing the register. The synchronized block 525 // ensure that the selector won't select() before the end of this block. 526 // @see deviceClientMonitorLoop 527 mSelector.wakeup(); 528 529 socketChannel.configureBlocking(false); 530 socketChannel.register(mSelector, SelectionKey.OP_READ, device); 531 } 532 533 return true; 534 } 535 } ... //省略错误处理代码 }代码8-4-7 DeviceMonitor - startMonitoringDevice
514行首先给ADB服务器发送监听请求获得所有可调试的应用进程PID列表:
674 private boolean sendDeviceMonitoringRequest(SocketChannel socket, Device device) 675 throws TimeoutException, AdbCommandRejectedException, IOException { 676 677 try { 678 AdbHelper.setDevice(socket, device); 679 680 byte[] request = AdbHelper.formAdbRequest("track-jdwp"); //$NON-NLS-1$ 681 682 AdbHelper.write(socket, request); 683 684 AdbResponse resp = AdbHelper.readAdbResponse(socket, false /* readDiagString */); 685 686 if (!resp.okay) { 687 // request was refused by adb! 688 Log.e("DeviceMonitor", "adb refused request: " + resp.message); 689 } 690 691 return resp.okay; 692 } ...//省略错误处理代码 }代码8-4-8 DeviceMonitor - sendDeviceMonitoringRequest
其实这段代码和上面的“代码8-4-5 DeviceMonitor - sendDeviceListMonitoringRequest”是类似的,只是发送是要在678行先把连接切换到目标监控设备(AdbHelper.setDevice方法将在下一章进行想尽描述)以及最后发送的命令变成是”track-jdwp”命令而已。最终这个命令其实等同于你在命令调用ADB命令行客户端发送命令”adb jdwp”,返回来的就是所有可调式应用进程的PID,请看以下输出结果示例,其与上图8-4-1中DDMS的Devices模块打印的进程PID是一致的:
图 8-4-2 adb jdwp 命令输出
获取到设备里面运行的可调试进程PID列表后,大家应该也可以想到下一步动作就是为每一个PID,也就是为每一个进程的vm虚拟机创建一个客户端线程来通过JDWP协议监控调试了,这也就是为什么DDMS能够动态获得每个进程的动态信息的原因了。
进程VM虚拟机监控代码分析到这里在本书中应该就算差不多了,如果再往下分析的话就需要去分析DDMS更多的知识以及JDWP协议相关的东西了,毕竟这不是我们这本书的重点,所以分析到这里让大家对DDMS工作原理有个基本认知就好了,再往下分析一大堆不相关代码就有走题和凑字数的嫌疑了。
这里我们根据上面承诺的,还是要看看“代码8-4-6 DeviceMonitor - updateDevices处理新增加设备”中481行对新增加的设备是如何通过调用“queryNewDeviceForInfo”这个方法来获取基本信息的,获取的又是什么信息:
442 private void queryNewDeviceForInfo(Device device) { 443 // TODO: do this in a separate thread. 444 try { 445 // first get the list of properties. 446 device.executeShellCommand( “getprop”, 447 new GetPropReceiver(device)); 448 449 queryNewDeviceForMountingPoint(device, “EXTERNAL_STORAGE”); 450 queryNewDeviceForMountingPoint(device, “ANDROID_DATA”); 451 queryNewDeviceForMountingPoint(device, “ANDROID_ROOT”); 452 453 // now get the emulator Virtual Device name (if applicable). 454 if (device.isEmulator()) { 455 EmulatorConsole console = EmulatorConsole.getConsole(device); 456 if (console != null) { 457 device.setAvdName(console.getAvdName()); 458 console.close(); 459 } 460 } 461 } ...//省略错误处理部分代码 }代码8-4-9 DeviceMonitor - queryNewDeviceForInfo
这个方法所做的事情就是:
获取完系统属性后,我们就要看下新设备的文件系统的那几个挂载点是怎么获得的了,我们进入到对应方法:
483 private void queryNewDeviceForMountingPoint(final Device device, final String name) 484 throws TimeoutException, AdbCommandRejectedException, ShellCommandUnresponsiveException, IOException 485 { 486 device.executeShellCommand("echo $" + name, new MultiLineReceiver() 487 { 488 public boolean isCancelled() { 489 return false; 490 } 491 492 public void processNewLines(String[] lines) 493 { 494 for (String line : lines) { 495 if (!line.isEmpty()) 496 { 497 device.setMountingPoint(name, line); 498 } 499 } 500 } 501 }); 502 }
代码8-4-10 DeviceMonitor - queryNewDeviceForMountingPoint
这个跟上面的发送getprop命令有类似的地方,只是命令换了”adb shell $name”和返回值处理类是重新实现的而已,但原理都一样。这里$name换成上面调用方法形参对应的”EXTERNAL_STORAGE”,”ANDROID_DATA”和“ANDROID_ROOT”就行了,以下就是本人通过命令行执行的效果:
图8-4-4 挂载点
最后把这个几个挂载点保存起来到Device实例的mMountpoints这个映射表里面:
67 private final Map<String, String> mMountPoints = new HashMap(); ... 783 void setMountingPoint(String name, String value) { 784 this.mMountPoints.put(name, value); 785 }代码8-4-11 Device - setMountingPoint
注:更多文章请关注公众号:techgogogo或个人博客http://techgogogo.com。当然,也非常欢迎您直接微信(zhubaitian1)勾搭。本文由天地会珠海分舵原创。转载请自觉,是否投诉维权看心情。