Home | History | Annotate | Download | only in init
      1 /*
      2  * Copyright (C) 2010 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 <poll.h>
     18 #include <fcntl.h>
     19 #include <string.h>
     20 #include <stdlib.h>
     21 #include <stdio.h>
     22 #include <ctype.h>
     23 #include <signal.h>
     24 
     25 #include <private/android_filesystem_config.h>
     26 
     27 #include "ueventd.h"
     28 #include "log.h"
     29 #include "util.h"
     30 #include "devices.h"
     31 #include "ueventd_parser.h"
     32 
     33 static char hardware[32];
     34 static unsigned revision = 0;
     35 
     36 static void import_kernel_nv(char *name, int in_qemu)
     37 {
     38     if (*name != '\0') {
     39         char *value = strchr(name, '=');
     40         if (value != NULL) {
     41             *value++ = 0;
     42             if (!strcmp(name,"androidboot.hardware"))
     43             {
     44                 strlcpy(hardware, value, sizeof(hardware));
     45             }
     46         }
     47     }
     48 }
     49 
     50 int ueventd_main(int argc, char **argv)
     51 {
     52     struct pollfd ufd;
     53     int nr;
     54     char tmp[32];
     55 
     56     /*
     57      * init sets the umask to 077 for forked processes. We need to
     58      * create files with exact permissions, without modification by
     59      * the umask.
     60      */
     61     umask(000);
     62 
     63     /* Prevent fire-and-forget children from becoming zombies.
     64      * If we should need to wait() for some children in the future
     65      * (as opposed to none right now), double-forking here instead
     66      * of ignoring SIGCHLD may be the better solution.
     67      */
     68     signal(SIGCHLD, SIG_IGN);
     69 
     70     open_devnull_stdio();
     71     klog_init();
     72 
     73     INFO("starting ueventd\n");
     74 
     75     /* Respect hardware passed in through the kernel cmd line. Here we will look
     76      * for androidboot.hardware param in kernel cmdline, and save its value in
     77      * hardware[]. */
     78     import_kernel_cmdline(0, import_kernel_nv);
     79 
     80     get_hardware_name(hardware, &revision);
     81 
     82     ueventd_parse_config_file("/ueventd.rc");
     83 
     84     snprintf(tmp, sizeof(tmp), "/ueventd.%s.rc", hardware);
     85     ueventd_parse_config_file(tmp);
     86 
     87     device_init();
     88 
     89     ufd.events = POLLIN;
     90     ufd.fd = get_device_fd();
     91 
     92     while(1) {
     93         ufd.revents = 0;
     94         nr = poll(&ufd, 1, -1);
     95         if (nr <= 0)
     96             continue;
     97         if (ufd.revents == POLLIN)
     98                handle_device_fd();
     99     }
    100 }
    101 
    102 static int get_android_id(const char *id)
    103 {
    104     unsigned int i;
    105     for (i = 0; i < ARRAY_SIZE(android_ids); i++)
    106         if (!strcmp(id, android_ids[i].name))
    107             return android_ids[i].aid;
    108     return -1;
    109 }
    110 
    111 void set_device_permission(int nargs, char **args)
    112 {
    113     char *name;
    114     char *attr = 0;
    115     mode_t perm;
    116     uid_t uid;
    117     gid_t gid;
    118     int prefix = 0;
    119     char *endptr;
    120     int ret;
    121     char *tmp = 0;
    122 
    123     if (nargs == 0)
    124         return;
    125 
    126     if (args[0][0] == '#')
    127         return;
    128 
    129     name = args[0];
    130 
    131     if (!strncmp(name,"/sys/", 5) && (nargs == 5)) {
    132         INFO("/sys/ rule %s %s\n",args[0],args[1]);
    133         attr = args[1];
    134         args++;
    135         nargs--;
    136     }
    137 
    138     if (nargs != 4) {
    139         ERROR("invalid line ueventd.rc line for '%s'\n", args[0]);
    140         return;
    141     }
    142 
    143     /* If path starts with mtd@ lookup the mount number. */
    144     if (!strncmp(name, "mtd@", 4)) {
    145         int n = mtd_name_to_number(name + 4);
    146         if (n >= 0)
    147             asprintf(&tmp, "/dev/mtd/mtd%d", n);
    148         name = tmp;
    149     } else {
    150         int len = strlen(name);
    151         if (name[len - 1] == '*') {
    152             prefix = 1;
    153             name[len - 1] = '\0';
    154         }
    155     }
    156 
    157     perm = strtol(args[1], &endptr, 8);
    158     if (!endptr || *endptr != '\0') {
    159         ERROR("invalid mode '%s'\n", args[1]);
    160         free(tmp);
    161         return;
    162     }
    163 
    164     ret = get_android_id(args[2]);
    165     if (ret < 0) {
    166         ERROR("invalid uid '%s'\n", args[2]);
    167         free(tmp);
    168         return;
    169     }
    170     uid = ret;
    171 
    172     ret = get_android_id(args[3]);
    173     if (ret < 0) {
    174         ERROR("invalid gid '%s'\n", args[3]);
    175         free(tmp);
    176         return;
    177     }
    178     gid = ret;
    179 
    180     add_dev_perms(name, attr, perm, uid, gid, prefix);
    181     free(tmp);
    182 }
    183