zl程序教程

您现在的位置是:首页 >  移动开发

当前栏目

Android培训班(33)

Android 33 培训班
2023-09-14 09:10:36 时间

<!-- @page { margin: 2cm } P { margin-bottom: 0.21cm } -->

init.rc文件里,可以看到加载下面的服务:

service vold /system/bin/vold

socket vold stream 0660 root mount

vold服务的代码在目录:

Android-2.0/system/core/vold

vold服务的作用主要是负责完成系统的动态卷管理,比如CDROM、U盘、MMC卡等外存储的管理。当有这外存储设备插入时,就需要监视这种变化,并加载相应的驱动程序,然后报告给系统和应用程序有新存储设备可以使用。

Vold处理过程大致分为三步:
1.
创建链接:
vold作为一个守护进程,一方面接受驱动的信息,并把信息传给应用层;另一方面接受上层的命令并完成相应。所以这里的链接一共有两条:
1vold socket: 负责vold与应用层的信息传递;
2)访问udevsocket: 负责vold与底层的信息传递;
这两个链接都是在进程的一开始完成创建的。

2.
引导:
这里主要是在vold启动时,对现有外设存储设备的处理。首先,要加载并解析vold.conf
并检查挂载点是否已经被挂载; 其次,执行MMC卡挂载; 最后,处理USB大容量存储。

3.
事件处理:
这里通过对两个链接的监听,完成对动态事件的处理,以及对上层应用操作的响应。

 

下面来分析一下main函数的代码如下:

int main(int argc, char **argv)

{

int door_sock = -1;

int uevent_sock = -1;

struct sockaddr_nl nladdr;

int uevent_sz = 64 * 1024;

 

LOGI("Android Volume Daemon version %d.%d", ver_major, ver_minor);

 

/*

* Create all the various sockets we'll need

*/

 

// Socket to listen on for incomming framework connections

if ((door_sock = android_get_control_socket(VOLD_SOCKET)) < 0) {

LOGE("Obtaining file descriptor socket '%s' failed: %s",

VOLD_SOCKET, strerror(errno));

exit(1);

}

这段代码是创建一个负责vold与应用层的信息传递的SOCKET。

 

if (listen(door_sock, 4) < 0) {

LOGE("Unable to listen on fd '%d' for socket '%s': %s",

door_sock, VOLD_SOCKET, strerror(errno));

exit(1);

}

 

这里开始监听。

 

mkdir("/dev/block/vold", 0755);

创建相应的设备目录。

 

 

// Socket to listen on for uevent changes

memset(&nladdr, 0, sizeof(nladdr));

nladdr.nl_family = AF_NETLINK;

nladdr.nl_pid = getpid();

nladdr.nl_groups = 0xffffffff;

 

if ((uevent_sock = socket(PF_NETLINK,

SOCK_DGRAM,NETLINK_KOBJECT_UEVENT)) < 0) {

LOGE("Unable to create uevent socket: %s", strerror(errno));

exit(1);

}

这段代码是 负责vold与底层的信息传递的SOCKET。

 

 

if (setsockopt(uevent_sock, SOL_SOCKET, SO_RCVBUFFORCE, &uevent_sz,

sizeof(uevent_sz)) < 0) {

LOGE("Unable to set uevent socket options: %s", strerror(errno));

exit(1);

}

 

if (bind(uevent_sock, (struct sockaddr *) &nladdr, sizeof(nladdr)) < 0) {

LOGE("Unable to bind uevent socket: %s", strerror(errno));

exit(1);

}

 

/*

* Bootstrap

*/

 

 

 

bootstrap = 1;

// Volume Manager

volmgr_bootstrap();

 

// SD Card system

mmc_bootstrap();

 

// USB Mass Storage

ums_bootstrap();

 

// Switch

switch_bootstrap();

 

上面这段是vold启动时,对现有外设存储设备的处理。

 

 

bootstrap = 0;

/*

* Main loop

*/

LOG_VOL("Bootstrapping complete");

while(1) {

fd_set read_fds;

struct timeval to;

int max = 0;

int rc = 0;

 

to.tv_sec = (60 * 60);

to.tv_usec = 0;

 

FD_ZERO(&read_fds);

FD_SET(door_sock, &read_fds);

if (door_sock > max)

max = door_sock;

FD_SET(uevent_sock, &read_fds);

if (uevent_sock > max)

max = uevent_sock;

 

if (fw_sock != -1) {

FD_SET(fw_sock, &read_fds);

if (fw_sock > max)

max = fw_sock;

}

 

if ((rc = select(max + 1, &read_fds, NULL, NULL, &to)) < 0) {

LOGE("select() failed (%s)", strerror(errno));

sleep(1);

continue;

}

 

if (!rc) {

continue;

}

 

if (FD_ISSET(door_sock, &read_fds)) {

struct sockaddr addr;

socklen_t alen;

 

alen = sizeof(addr);

 

if (fw_sock != -1) {

LOGE("Dropping duplicate framework connection");

int tmp = accept(door_sock, &addr, &alen);

close(tmp);

continue;

}

 

if ((fw_sock = accept(door_sock, &addr, &alen)) < 0) {

LOGE("Unable to accept framework connection (%s)",

strerror(errno));

}

LOG_VOL("Accepted connection from framework");

if ((rc = volmgr_send_states()) < 0) {

LOGE("Unable to send volmgr status to framework (%d)", rc);

}

}

 

if (FD_ISSET(fw_sock, &read_fds)) {

if ((rc = process_framework_command(fw_sock)) < 0) {

if (rc == -ECONNRESET) {

LOGE("Framework disconnected");

close(fw_sock);

fw_sock = -1;

} else {

LOGE("Error processing framework command (%s)",

strerror(errno));

}

}

}

 

if (FD_ISSET(uevent_sock, &read_fds)) {

if ((rc = process_uevent_message(uevent_sock)) < 0) {

LOGE("Error processing uevent msg (%s)", strerror(errno));

}

}

} // while

 

}

上面这段代码是进入事件处理。