1、"SD卡不受支持"通知的来源
由上述代码段我们知道出现问题的原因是,系统上报了一个存储容量大于0的Disk设备,但是该Disk的volume数量为0。
这样,问题就被转移了。要解释Disk的volume count为何为0,需要分析vold系统进程。
2、vold进程
vold进程用来管理和控制系统存储设备,比如SD/U盘热插拨、挂载、卸载、格式化。
vold进程由init进程启动,在init.rc里面注册:
对上,vold进程监听到kernel上报的事件(比如设备add,chang,remove等),处理后告知Java层。
这部分功能vold使用socket和Java层通信,负责通信实现的Java类是NativeDaemonConnector:
对下,void监听kernel事件,并对事件做相应的处理。这部分的原理是:
kernel检测到某些设备发生变化(典型的例子就是热插拔),会通过uevent事件告知用户,事件传递方向:内核空间->用户空间。
用来实现上述事件传递的,就是Netlink机制,它是一个socket,用户空间使用标准socket API就可以实现和kernel的通信:
StorageNotification.java (frameworks\base\packages\systemui\src\com\android\systemui\usb)
private void onDiskScannedInternal(DiskInfo disk, int volumeCount) {
if (volumeCount == 0 && disk.size > 0) {
// 构建"SD卡不受支持"通知
}
}
由上述代码段我们知道出现问题的原因是,系统上报了一个存储容量大于0的Disk设备,但是该Disk的volume数量为0。
这样,问题就被转移了。要解释Disk的volume count为何为0,需要分析vold系统进程。
2、vold进程
vold进程用来管理和控制系统存储设备,比如SD/U盘热插拨、挂载、卸载、格式化。
vold进程由init进程启动,在init.rc里面注册:
service vold /system/bin/vold
class core
socket vold stream 0660 root mount
socket cryptd stream 0660 root mount
ioprio be 2
对上,vold进程监听到kernel上报的事件(比如设备add,chang,remove等),处理后告知Java层。
这部分功能vold使用socket和Java层通信,负责通信实现的Java类是NativeDaemonConnector:
NativeDaemonConnector.java (frameworks\base\services\core\java\com\android\server)
@Override
public void run() {
while (true) {
try {
listenToSocket();
} catch (Exception e) {
}
}
}
对下,void监听kernel事件,并对事件做相应的处理。这部分的原理是:
kernel检测到某些设备发生变化(典型的例子就是热插拔),会通过uevent事件告知用户,事件传递方向:内核空间->用户空间。
用来实现上述事件传递的,就是Netlink机制,它是一个socket,用户空间使用标准socket API就可以实现和kernel的通信: