1 /* 2 * Copyright (C) 2008 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include <stdio.h> 18 #include <stdlib.h> 19 #include <signal.h> 20 #include <errno.h> 21 #include <string.h> 22 #include <sys/stat.h> 23 #include <sys/types.h> 24 #include <sys/wait.h> 25 26 #include <fcntl.h> 27 #include <dirent.h> 28 29 #define LOG_TAG "Netd" 30 31 #include "cutils/log.h" 32 33 #include "CommandListener.h" 34 #include "NetlinkManager.h" 35 #include "DnsProxyListener.h" 36 37 static void coldboot(const char *path); 38 static void sigchld_handler(int sig); 39 40 int main() { 41 42 CommandListener *cl; 43 NetlinkManager *nm; 44 DnsProxyListener *dpl; 45 46 LOGI("Netd 1.0 starting"); 47 48 // signal(SIGCHLD, sigchld_handler); 49 50 if (!(nm = NetlinkManager::Instance())) { 51 LOGE("Unable to create NetlinkManager"); 52 exit(1); 53 }; 54 55 56 cl = new CommandListener(); 57 nm->setBroadcaster((SocketListener *) cl); 58 59 if (nm->start()) { 60 LOGE("Unable to start NetlinkManager (%s)", strerror(errno)); 61 exit(1); 62 } 63 64 // Set local DNS mode, to prevent bionic from proxying 65 // back to this service, recursively. 66 setenv("ANDROID_DNS_MODE", "local", 1); 67 dpl = new DnsProxyListener(); 68 if (dpl->startListener()) { 69 LOGE("Unable to start DnsProxyListener (%s)", strerror(errno)); 70 exit(1); 71 } 72 73 /* 74 * Now that we're up, we can respond to commands 75 */ 76 if (cl->startListener()) { 77 LOGE("Unable to start CommandListener (%s)", strerror(errno)); 78 exit(1); 79 } 80 81 // Eventually we'll become the monitoring thread 82 while(1) { 83 sleep(1000); 84 } 85 86 LOGI("Netd exiting"); 87 exit(0); 88 } 89 90 static void do_coldboot(DIR *d, int lvl) 91 { 92 struct dirent *de; 93 int dfd, fd; 94 95 dfd = dirfd(d); 96 97 fd = openat(dfd, "uevent", O_WRONLY); 98 if(fd >= 0) { 99 write(fd, "add\n", 4); 100 close(fd); 101 } 102 103 while((de = readdir(d))) { 104 DIR *d2; 105 106 if (de->d_name[0] == '.') 107 continue; 108 109 if (de->d_type != DT_DIR && lvl > 0) 110 continue; 111 112 fd = openat(dfd, de->d_name, O_RDONLY | O_DIRECTORY); 113 if(fd < 0) 114 continue; 115 116 d2 = fdopendir(fd); 117 if(d2 == 0) 118 close(fd); 119 else { 120 do_coldboot(d2, lvl + 1); 121 closedir(d2); 122 } 123 } 124 } 125 126 static void coldboot(const char *path) 127 { 128 DIR *d = opendir(path); 129 if(d) { 130 do_coldboot(d, 0); 131 closedir(d); 132 } 133 } 134 135 static void sigchld_handler(int sig) { 136 pid_t pid = wait(NULL); 137 LOGD("Child process %d exited", pid); 138 } 139