| |
| /* |
| * Copyright (C) 2008 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include <errno.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <unistd.h> |
| #include <string.h> |
| #include <fcntl.h> |
| #include <pthread.h> |
| |
| #include <sys/socket.h> |
| #include <sys/select.h> |
| #include <sys/time.h> |
| #include <sys/types.h> |
| #include <sys/un.h> |
| |
| #include <cutils/config_utils.h> |
| #include <cutils/cpu_info.h> |
| #include <cutils/properties.h> |
| #include <cutils/sockets.h> |
| |
| #include <linux/netlink.h> |
| |
| #include <private/android_filesystem_config.h> |
| |
| #include "vold.h" |
| #include "volmgr.h" |
| |
| |
| #define VOLD_SOCKET "vold" |
| |
| /* |
| * Globals |
| */ |
| |
| static int ver_major = 2; |
| static int ver_minor = 0; |
| static pthread_mutex_t write_mutex = PTHREAD_MUTEX_INITIALIZER; |
| static int fw_sock = -1; |
| |
| int bootstrap = 0; |
| |
| 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); |
| } |
| |
| 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); |
| } |
| |
| 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(); |
| |
| 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 |
| |
| } |
| |
| int send_msg(char* message) |
| { |
| int result = -1; |
| |
| pthread_mutex_lock(&write_mutex); |
| |
| // LOG_VOL("send_msg(%s):", message); |
| |
| if (fw_sock >= 0) |
| result = write(fw_sock, message, strlen(message) + 1); |
| |
| pthread_mutex_unlock(&write_mutex); |
| |
| return result; |
| } |
| |
| int send_msg_with_data(char *message, char *data) |
| { |
| int result = -1; |
| |
| char* buffer = (char *)alloca(strlen(message) + strlen(data) + 1); |
| if (!buffer) { |
| LOGE("alloca failed in send_msg_with_data"); |
| return -1; |
| } |
| |
| strcpy(buffer, message); |
| strcat(buffer, data); |
| return send_msg(buffer); |
| } |