zoukankan      html  css  js  c++  java
  • 深入学习Android系统上mount命令的使用

    博客链接:http://blog.csdn.net/qq1084283172/article/details/52493227


    在Android系统的预装apk病毒和elf病毒的清除时,经常需要先获取root权限,再执行 “mount -o remount,rw /system” 命令修改系统分区属性为可写,然后才能将system/xbin、system/bin以及system/app下的病毒清除干净。在清除Android系统病毒的这个过程中,必须涉及到 mount修改Android系统的分区属性为可写的行为,这里就学习和研究一下Android系统的mount命令。mount命令在Android安全学习的过程中经常会遇到,这里就学习这个命令。



    一、mount 命令代码实现


    在Android4.4.2的源码路径android4.4.2/system/core/init/builtins.c路径下有Android系统mount命令的代码实现,do_mount()函数的具体实现的功能就是Android系统的mount命令对应的实现。do_mount()函数最终调用Linux系统mount()函数来实现修改系统分区属性的功能。

    /*
     * 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 <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include <unistd.h>
    #include <string.h>
    #include <stdio.h>
    #include <linux/kd.h>
    #include <errno.h>
    #include <sys/socket.h>
    #include <netinet/in.h>
    #include <linux/if.h>
    #include <arpa/inet.h>
    #include <stdlib.h>
    #include <sys/mount.h>
    #include <sys/resource.h>
    #include <sys/wait.h>
    #include <linux/loop.h>
    #include <cutils/partition_utils.h>
    #include <cutils/android_reboot.h>
    #include <sys/system_properties.h>
    #include <fs_mgr.h>
    
    #include <selinux/selinux.h>
    #include <selinux/label.h>
    
    #include "init.h"
    #include "keywords.h"
    #include "property_service.h"
    #include "devices.h"
    #include "init_parser.h"
    #include "util.h"
    #include "log.h"
    
    #include <private/android_filesystem_config.h>
    
    void add_environment(const char *name, const char *value);
    
    extern int init_module(void *, unsigned long, const char *);
    
    static int write_file(const char *path, const char *value)
    {
        int fd, ret, len;
    
        fd = open(path, O_WRONLY|O_CREAT|O_NOFOLLOW, 0600);
    
        if (fd < 0)
            return -errno;
    
        len = strlen(value);
    
        do {
            ret = write(fd, value, len);
        } while (ret < 0 && errno == EINTR);
    
        close(fd);
        if (ret < 0) {
            return -errno;
        } else {
            return 0;
        }
    }
    
    static int _open(const char *path)
    {
        int fd;
    
        fd = open(path, O_RDONLY | O_NOFOLLOW);
        if (fd < 0)
            fd = open(path, O_WRONLY | O_NOFOLLOW);
    
        return fd;
    }
    
    static int _chown(const char *path, unsigned int uid, unsigned int gid)
    {
        int fd;
        int ret;
    
        fd = _open(path);
        if (fd < 0) {
            return -1;
        }
    
        ret = fchown(fd, uid, gid);
        if (ret < 0) {
            int errno_copy = errno;
            close(fd);
            errno = errno_copy;
            return -1;
        }
    
        close(fd);
    
        return 0;
    }
    
    static int _chmod(const char *path, mode_t mode)
    {
        int fd;
        int ret;
    
        fd = _open(path);
        if (fd < 0) {
            return -1;
        }
    
        ret = fchmod(fd, mode);
        if (ret < 0) {
            int errno_copy = errno;
            close(fd);
            errno = errno_copy;
            return -1;
        }
    
        close(fd);
    
        return 0;
    }
    
    static int insmod(const char *filename, char *options)
    {
        void *module;
        unsigned size;
        int ret;
    
        module = read_file(filename, &size);
        if (!module)
            return -1;
    
        ret = init_module(module, size, options);
    
        free(module);
    
        return ret;
    }
    
    static int setkey(struct kbentry *kbe)
    {
        int fd, ret;
    
        fd = open("/dev/tty0", O_RDWR | O_SYNC);
        if (fd < 0)
            return -1;
    
        ret = ioctl(fd, KDSKBENT, kbe);
    
        close(fd);
        return ret;
    }
    
    static int __ifupdown(const char *interface, int up)
    {
        struct ifreq ifr;
        int s, ret;
    
        strlcpy(ifr.ifr_name, interface, IFNAMSIZ);
    
        s = socket(AF_INET, SOCK_DGRAM, 0);
        if (s < 0)
            return -1;
    
        ret = ioctl(s, SIOCGIFFLAGS, &ifr);
        if (ret < 0) {
            goto done;
        }
    
        if (up)
            ifr.ifr_flags |= IFF_UP;
        else
            ifr.ifr_flags &= ~IFF_UP;
    
        ret = ioctl(s, SIOCSIFFLAGS, &ifr);
        
    done:
        close(s);
        return ret;
    }
    
    static void service_start_if_not_disabled(struct service *svc)
    {
        if (!(svc->flags & SVC_DISABLED)) {
            service_start(svc, NULL);
        }
    }
    
    int do_chdir(int nargs, char **args)
    {
        chdir(args[1]);
        return 0;
    }
    
    int do_chroot(int nargs, char **args)
    {
        chroot(args[1]);
        return 0;
    }
    
    int do_class_start(int nargs, char **args)
    {
            /* Starting a class does not start services
             * which are explicitly disabled.  They must
             * be started individually.
             */
        service_for_each_class(args[1], service_start_if_not_disabled);
        return 0;
    }
    
    int do_class_stop(int nargs, char **args)
    {
        service_for_each_class(args[1], service_stop);
        return 0;
    }
    
    int do_class_reset(int nargs, char **args)
    {
        service_for_each_class(args[1], service_reset);
        return 0;
    }
    
    int do_domainname(int nargs, char **args)
    {
        return write_file("/proc/sys/kernel/domainname", args[1]);
    }
    
    int do_exec(int nargs, char **args)
    {
        return -1;
    }
    
    int do_export(int nargs, char **args)
    {
        add_environment(args[1], args[2]);
        return 0;
    }
    
    int do_hostname(int nargs, char **args)
    {
        return write_file("/proc/sys/kernel/hostname", args[1]);
    }
    
    int do_ifup(int nargs, char **args)
    {
        return __ifupdown(args[1], 1);
    }
    
    
    static int do_insmod_inner(int nargs, char **args, int opt_len)
    {
        char options[opt_len + 1];
        int i;
    
        options[0] = '';
        if (nargs > 2) {
            strcpy(options, args[2]);
            for (i = 3; i < nargs; ++i) {
                strcat(options, " ");
                strcat(options, args[i]);
            }
        }
    
        return insmod(args[1], options);
    }
    
    int do_insmod(int nargs, char **args)
    {
        int i;
        int size = 0;
    
        if (nargs > 2) {
            for (i = 2; i < nargs; ++i)
                size += strlen(args[i]) + 1;
        }
    
        return do_insmod_inner(nargs, args, size);
    }
    
    int do_mkdir(int nargs, char **args)
    {
        mode_t mode = 0755;
        int ret;
    
        /* mkdir <path> [mode] [owner] [group] */
    
        if (nargs >= 3) {
            mode = strtoul(args[2], 0, 8);
        }
    
        ret = make_dir(args[1], mode);
        /* chmod in case the directory already exists */
        if (ret == -1 && errno == EEXIST) {
            ret = _chmod(args[1], mode);
        }
        if (ret == -1) {
            return -errno;
        }
    
        if (nargs >= 4) {
            uid_t uid = decode_uid(args[3]);
            gid_t gid = -1;
    
            if (nargs == 5) {
                gid = decode_uid(args[4]);
            }
    
            if (_chown(args[1], uid, gid) < 0) {
                return -errno;
            }
    
            /* chown may have cleared S_ISUID and S_ISGID, chmod again */
            if (mode & (S_ISUID | S_ISGID)) {
                ret = _chmod(args[1], mode);
                if (ret == -1) {
                    return -errno;
                }
            }
        }
    
        return 0;
    }
    
    static struct {
        const char *name;
        unsigned flag;
    } mount_flags[] = {
        { "noatime",    MS_NOATIME },
        { "noexec",     MS_NOEXEC },
        { "nosuid",     MS_NOSUID },
        { "nodev",      MS_NODEV },
        { "nodiratime", MS_NODIRATIME },
        { "ro",         MS_RDONLY },
        { "rw",         0 },
        { "remount",    MS_REMOUNT },
        { "bind",       MS_BIND },
        { "rec",        MS_REC },
        { "unbindable", MS_UNBINDABLE },
        { "private",    MS_PRIVATE },
        { "slave",      MS_SLAVE },
        { "shared",     MS_SHARED },
        { "defaults",   0 },
        { 0,            0 },
    };
    
    #define DATA_MNT_POINT "/data"
    
    /* mount <type> <device> <path> <flags ...> <options> */
    int do_mount(int nargs, char **args)
    {
        char tmp[64];
        char *source, *target, *system;
        char *options = NULL;
        unsigned flags = 0;
        int n, i;
        int wait = 0;
    
        for (n = 4; n < nargs; n++) {
            for (i = 0; mount_flags[i].name; i++) {
                if (!strcmp(args[n], mount_flags[i].name)) {
                    flags |= mount_flags[i].flag;
                    break;
                }
            }
    
            if (!mount_flags[i].name) {
                if (!strcmp(args[n], "wait"))
                    wait = 1;
                /* if our last argument isn't a flag, wolf it up as an option string */
                else if (n + 1 == nargs)
                    options = args[n];
            }
        }
    
        system = args[1];
        source = args[2];
        target = args[3];
    
        if (!strncmp(source, "mtd@", 4)) {
            n = mtd_name_to_number(source + 4);
            if (n < 0) {
                return -1;
            }
    
            sprintf(tmp, "/dev/block/mtdblock%d", n);
    
            if (wait)
                wait_for_file(tmp, COMMAND_RETRY_TIMEOUT);
            if (mount(tmp, target, system, flags, options) < 0) {
                return -1;
            }
    
            goto exit_success;
        } else if (!strncmp(source, "loop@", 5)) {
            int mode, loop, fd;
            struct loop_info info;
    
            mode = (flags & MS_RDONLY) ? O_RDONLY : O_RDWR;
            fd = open(source + 5, mode);
            if (fd < 0) {
                return -1;
            }
    
            for (n = 0; ; n++) {
                sprintf(tmp, "/dev/block/loop%d", n);
                loop = open(tmp, mode);
                if (loop < 0) {
                    return -1;
                }
    
                /* if it is a blank loop device */
                if (ioctl(loop, LOOP_GET_STATUS, &info) < 0 && errno == ENXIO) {
                    /* if it becomes our loop device */
                    if (ioctl(loop, LOOP_SET_FD, fd) >= 0) {
                        close(fd);
    
                        if (mount(tmp, target, system, flags, options) < 0) {
                            ioctl(loop, LOOP_CLR_FD, 0);
                            close(loop);
                            return -1;
                        }
    
                        close(loop);
                        goto exit_success;
                    }
                }
    
                close(loop);
            }
    
            close(fd);
            ERROR("out of loopback devices");
            return -1;
        } else {
            if (wait)
                wait_for_file(source, COMMAND_RETRY_TIMEOUT);
            if (mount(source, target, system, flags, options) < 0) {
                return -1;
            }
    
        }
    
    exit_success:
        return 0;
    
    }
    
    int do_mount_all(int nargs, char **args)
    {
        pid_t pid;
        int ret = -1;
        int child_ret = -1;
        int status;
        const char *prop;
        struct fstab *fstab;
    
        if (nargs != 2) {
            return -1;
        }
    
        /*
         * Call fs_mgr_mount_all() to mount all filesystems.  We fork(2) and
         * do the call in the child to provide protection to the main init
         * process if anything goes wrong (crash or memory leak), and wait for
         * the child to finish in the parent.
         */
        pid = fork();
        if (pid > 0) {
            /* Parent.  Wait for the child to return */
            waitpid(pid, &status, 0);
            if (WIFEXITED(status)) {
                ret = WEXITSTATUS(status);
            } else {
                ret = -1;
            }
        } else if (pid == 0) {
            /* child, call fs_mgr_mount_all() */
            klog_set_level(6);  /* So we can see what fs_mgr_mount_all() does */
            fstab = fs_mgr_read_fstab(args[1]);
            child_ret = fs_mgr_mount_all(fstab);
            fs_mgr_free_fstab(fstab);
            if (child_ret == -1) {
                ERROR("fs_mgr_mount_all returned an error
    ");
            }
            exit(child_ret);
        } else {
            /* fork failed, return an error */
            return -1;
        }
    
        /* ret is 1 if the device is encrypted, 0 if not, and -1 on error */
        if (ret == 1) {
            property_set("ro.crypto.state", "encrypted");
            property_set("vold.decrypt", "1");
        } else if (ret == 0) {
            property_set("ro.crypto.state", "unencrypted");
            /* If fs_mgr determined this is an unencrypted device, then trigger
             * that action.
             */
            action_for_each_trigger("nonencrypted", action_add_queue_tail);
        }
    
        return ret;
    }
    
    int do_swapon_all(int nargs, char **args)
    {
        struct fstab *fstab;
        int ret;
    
        fstab = fs_mgr_read_fstab(args[1]);
        ret = fs_mgr_swapon_all(fstab);
        fs_mgr_free_fstab(fstab);
    
        return ret;
    }
    
    int do_setcon(int nargs, char **args) {
        if (is_selinux_enabled() <= 0)
            return 0;
        if (setcon(args[1]) < 0) {
            return -errno;
        }
        return 0;
    }
    
    int do_setenforce(int nargs, char **args) {
        if (is_selinux_enabled() <= 0)
            return 0;
        if (security_setenforce(atoi(args[1])) < 0) {
            return -errno;
        }
        return 0;
    }
    
    int do_setkey(int nargs, char **args)
    {
        struct kbentry kbe;
        kbe.kb_table = strtoul(args[1], 0, 0);
        kbe.kb_index = strtoul(args[2], 0, 0);
        kbe.kb_value = strtoul(args[3], 0, 0);
        return setkey(&kbe);
    }
    
    int do_setprop(int nargs, char **args)
    {
        const char *name = args[1];
        const char *value = args[2];
        char prop_val[PROP_VALUE_MAX];
        int ret;
    
        ret = expand_props(prop_val, value, sizeof(prop_val));
        if (ret) {
            ERROR("cannot expand '%s' while assigning to '%s'
    ", value, name);
            return -EINVAL;
        }
        property_set(name, prop_val);
        return 0;
    }
    
    int do_setrlimit(int nargs, char **args)
    {
        struct rlimit limit;
        int resource;
        resource = atoi(args[1]);
        limit.rlim_cur = atoi(args[2]);
        limit.rlim_max = atoi(args[3]);
        return setrlimit(resource, &limit);
    }
    
    int do_start(int nargs, char **args)
    {
        struct service *svc;
        svc = service_find_by_name(args[1]);
        if (svc) {
            service_start(svc, NULL);
        }
        return 0;
    }
    
    int do_stop(int nargs, char **args)
    {
        struct service *svc;
        svc = service_find_by_name(args[1]);
        if (svc) {
            service_stop(svc);
        }
        return 0;
    }
    
    int do_restart(int nargs, char **args)
    {
        struct service *svc;
        svc = service_find_by_name(args[1]);
        if (svc) {
            service_restart(svc);
        }
        return 0;
    }
    
    int do_powerctl(int nargs, char **args)
    {
        char command[PROP_VALUE_MAX];
        int res;
        int len = 0;
        int cmd = 0;
        char *reboot_target;
    
        res = expand_props(command, args[1], sizeof(command));
        if (res) {
            ERROR("powerctl: cannot expand '%s'
    ", args[1]);
            return -EINVAL;
        }
    
        if (strncmp(command, "shutdown", 8) == 0) {
            cmd = ANDROID_RB_POWEROFF;
            len = 8;
        } else if (strncmp(command, "reboot", 6) == 0) {
            cmd = ANDROID_RB_RESTART2;
            len = 6;
        } else {
            ERROR("powerctl: unrecognized command '%s'
    ", command);
            return -EINVAL;
        }
    
        if (command[len] == ',') {
            reboot_target = &command[len + 1];
        } else if (command[len] == '') {
            reboot_target = "";
        } else {
            ERROR("powerctl: unrecognized reboot target '%s'
    ", &command[len]);
            return -EINVAL;
        }
    
        return android_reboot(cmd, 0, reboot_target);
    }
    
    int do_trigger(int nargs, char **args)
    {
        action_for_each_trigger(args[1], action_add_queue_tail);
        return 0;
    }
    
    int do_symlink(int nargs, char **args)
    {
        return symlink(args[1], args[2]);
    }
    
    int do_rm(int nargs, char **args)
    {
        return unlink(args[1]);
    }
    
    int do_rmdir(int nargs, char **args)
    {
        return rmdir(args[1]);
    }
    
    int do_sysclktz(int nargs, char **args)
    {
        struct timezone tz;
    
        if (nargs != 2)
            return -1;
    
        memset(&tz, 0, sizeof(tz));
        tz.tz_minuteswest = atoi(args[1]);   
        if (settimeofday(NULL, &tz))
            return -1;
        return 0;
    }
    
    int do_write(int nargs, char **args)
    {
        const char *path = args[1];
        const char *value = args[2];
        char prop_val[PROP_VALUE_MAX];
        int ret;
    
        ret = expand_props(prop_val, value, sizeof(prop_val));
        if (ret) {
            ERROR("cannot expand '%s' while writing to '%s'
    ", value, path);
            return -EINVAL;
        }
        return write_file(path, prop_val);
    }
    
    int do_copy(int nargs, char **args)
    {
        char *buffer = NULL;
        int rc = 0;
        int fd1 = -1, fd2 = -1;
        struct stat info;
        int brtw, brtr;
        char *p;
    
        if (nargs != 3)
            return -1;
    
        if (stat(args[1], &info) < 0) 
            return -1;
    
        if ((fd1 = open(args[1], O_RDONLY)) < 0) 
            goto out_err;
    
        if ((fd2 = open(args[2], O_WRONLY|O_CREAT|O_TRUNC, 0660)) < 0)
            goto out_err;
    
        if (!(buffer = malloc(info.st_size)))
            goto out_err;
    
        p = buffer;
        brtr = info.st_size;
        while(brtr) {
            rc = read(fd1, p, brtr);
            if (rc < 0)
                goto out_err;
            if (rc == 0)
                break;
            p += rc;
            brtr -= rc;
        }
    
        p = buffer;
        brtw = info.st_size;
        while(brtw) {
            rc = write(fd2, p, brtw);
            if (rc < 0)
                goto out_err;
            if (rc == 0)
                break;
            p += rc;
            brtw -= rc;
        }
    
        rc = 0;
        goto out;
    out_err:
        rc = -1;
    out:
        if (buffer)
            free(buffer);
        if (fd1 >= 0)
            close(fd1);
        if (fd2 >= 0)
            close(fd2);
        return rc;
    }
    
    int do_chown(int nargs, char **args) {
        /* GID is optional. */
        if (nargs == 3) {
            if (_chown(args[2], decode_uid(args[1]), -1) < 0)
                return -errno;
        } else if (nargs == 4) {
            if (_chown(args[3], decode_uid(args[1]), decode_uid(args[2])) < 0)
                return -errno;
        } else {
            return -1;
        }
        return 0;
    }
    
    static mode_t get_mode(const char *s) {
        mode_t mode = 0;
        while (*s) {
            if (*s >= '0' && *s <= '7') {
                mode = (mode<<3) | (*s-'0');
            } else {
                return -1;
            }
            s++;
        }
        return mode;
    }
    
    int do_chmod(int nargs, char **args) {
        mode_t mode = get_mode(args[1]);
        if (_chmod(args[2], mode) < 0) {
            return -errno;
        }
        return 0;
    }
    
    int do_restorecon(int nargs, char **args) {
        int i;
    
        for (i = 1; i < nargs; i++) {
            if (restorecon(args[i]) < 0)
                return -errno;
        }
        return 0;
    }
    
    int do_setsebool(int nargs, char **args) {
        const char *name = args[1];
        const char *value = args[2];
        SELboolean b;
        int ret;
    
        if (is_selinux_enabled() <= 0)
            return 0;
    
        b.name = name;
        if (!strcmp(value, "1") || !strcasecmp(value, "true") || !strcasecmp(value, "on"))
            b.value = 1;
        else if (!strcmp(value, "0") || !strcasecmp(value, "false") || !strcasecmp(value, "off"))
            b.value = 0;
        else {
            ERROR("setsebool: invalid value %s
    ", value);
            return -EINVAL;
        }
    
        if (security_set_boolean_list(1, &b, 0) < 0) {
            ret = -errno;
            ERROR("setsebool: could not set %s to %s
    ", name, value);
            return ret;
        }
    
        return 0;
    }
    
    int do_loglevel(int nargs, char **args) {
        if (nargs == 2) {
            klog_set_level(atoi(args[1]));
            return 0;
        }
        return -1;
    }
    
    int do_load_persist_props(int nargs, char **args) {
        if (nargs == 1) {
            load_persist_props();
            return 0;
        }
        return -1;
    }
    
    int do_wait(int nargs, char **args)
    {
        if (nargs == 2) {
            return wait_for_file(args[1], COMMAND_RETRY_TIMEOUT);
        } else if (nargs == 3) {
            return wait_for_file(args[1], atoi(args[2]));
        } else
            return -1;
    }


    Linux系统调用--mount/umount函数的使用说明

    【 mount/umount系统调用】   
        
    功能描述:
    mount挂上文件系统,umount执行相反的操作。
      
    用法:  
    #include <sys/mount.h>
    int mount(const char *source, const char *target,
       const char *filesystemtype, unsigned long mountflags, const void *data);
    int umount(const char *target);
    int umount2(const char *target, int flags);
    
    参数:   
    source:将要挂上的文件系统,通常是一个设备名。
    target:文件系统所要挂在的目标目录。
    filesystemtype:文件系统的类型,可以是"ext2","ext3","msdos","proc","nfs","iso9660" 。。。
    mountflags:指定文件系统的读写访问标志,可能值有以下
    MS_BIND:执行bind挂载,使文件或者子目录树在文件系统内的另一个点上可视。
    MS_DIRSYNC:同步目录的更新。
    MS_MANDLOCK:允许在文件上执行强制锁。
    MS_MOVE:移动子目录树。
    MS_NOATIME:不要更新文件上的访问时间。
    MS_NODEV:不允许访问设备文件。
    MS_NODIRATIME:不允许更新目录上的访问时间。
    MS_NOEXEC:不允许在挂上的文件系统上执行程序。
    MS_NOSUID:执行程序时,不遵照set-user-ID 和 set-group-ID位。
    MS_RDONLY:指定文件系统为只读。
    MS_REMOUNT:重新加载文件系统。这允许你改变现存文件系统的mountflag和数据,而无需使用先卸载,再挂上文件系统的方式。
    MS_SYNCHRONOUS:同步文件的更新。
    MNT_FORCE:强制卸载,即使文件系统处于忙状态。
    MNT_EXPIRE:将挂载点标志为过时。
    data:文件系统特有的参数。
        
    返回说明:   
    成功执行时,返回0。失败返回-1,errno被设为以下的某个值   
    EACCES:权能不足,可能原因是,路径的一部分不可搜索,或者挂载只读的文件系统时,没有指定 MS_RDONLY 标志。
    EAGAIN:成功地将不处于忙状态的文件系统标志为过时。
    EBUSY:一. 源文件系统已被挂上。或者不可以以只读的方式重新挂载,因为它还拥有以写方式打开的文件。二. 目标处于忙状态。
    EFAULT: 内存空间访问出错。
    EINVAL:操作无效,可能是源文件系统超级块无效。
    ELOOP :路径解析的过程中存在太多的符号连接。
    EMFILE:无需块设备要求的情况下,无用设备表已满。
    ENAMETOOLONG:路径名超出可允许的长度。
    ENODEV:内核不支持某中文件系统。
    ENOENT:路径名部分内容表示的目录不存在。
    ENOMEM: 核心内存不足。
    ENOTBLK:source不是块设备。
    ENOTDIR:路径名的部分内容不是目录。
    EPERM : 调用者权能不足。
    ENXIO:块主设备号超出所允许的范围。


    二、Android手机上mount命令的使用


    已经获取到手机root权限的情况下,在电脑上手动执行错误的 mount 命令的时候,adb shell终端会提示 “mount [-r] [-w] [-o options] [-t type] device directory”,可能就是说我们在adb shell终端环境下执行的mount命令不正确。下面就网上一些博客分享的Android手机上执行mount命令的方法,总结整理一下。


    A、方法一


    如果是愣头青,搞不清清楚Android手机系统的修改分区属性的命令,那就老老实实的使用简化的命令:


    <pre name="code" class="cpp">
    
    #修改系统分区属性为可写
    mount -o remount,rw /system
    
    #修改系统分区属性为只读
    mount -o remount,ro /system
    
    

    因为linux系统自己会去维护一个已经mount的表,因此只需要输入现有的挂载点,系统会根据现有的挂载点去寻找对应的需要挂载的设备文件。

    B、方法二

    自己手动为mount命令寻找需要挂载的设备文件,并写到mount命令中去(使用Nexus 5设备Android4.4.4的系统为测试环境),说明下-不同的手机设备/system文件对应的系统设备文件是不同的,不能一概而论。

    1.使用 df  命令查看档案系统的状况或是看所有档案系统的状况(预设值),发现/system分区有1009.3MB的大小。

    接着执行命令cat  /proc/partitions 查看/proc下的partitions分区的大小。

    看的出来,分区mmcblk0p25的大小最接近1009.3MB,判断挂载点/system就对应该设备文件了。但是这种方法明显操作起来不是很方便而且比较繁琐,需要计算和比较。

    #修改系统分区属性为可写
    mount -o remount,rw /dev/block/mmcblk0p25 /system
    
    #修改系统分区属性为只读
    mount -o remount,ro /dev/block/mmcblk0p25 /system




    其实在 adb shell 的环境情况下,执行 cat  /proc/mounts 或者mount命令也能找到/system文件对应的设备文件。




    #修改系统分区属性为可写
    mount -o remount,rw /dev/block/platform/msm_sdcc.1/by-name/system /system
    
    #修改系统分区属性为只读
    mount -o remount,ro /dev/block/platform/msm_sdcc.1/by-name/system /system


    在这里需要注意下,可能由于手机的CPU的类型不同,执行cat /proc/mounts 或者 mount 命令显示的结果还是有所区别的,上面的结果是高通的CPU显示的结果。其中也有博客的作者提到在adb shell的环境下执行cat /proc/mtd 命令来看手机设备的分区的信息,但是呢,经过在Nexus 5的手机上测试发现 cat /proc/mtd 是执行不成功的,会提示“/system/bin/sh: cat: /proc/mtd: No such file or directory”错误。


    特地mark一下,还有博主提到 尝试用其他信息代替”/dev/block/mmcblk0p25“,试了一下居然也是可行的;甚至用任何分块号mtdblock3、mtdblock4、mtdblock11等等都替换 /dev/block/mmcblk0p25 也能正常运行!那么这种错误命令为什么能成功呢?其实我们的命令 参数“-o remount”其实自动 忽略了/dev/block/mtdblock 这一段参数,只是简单的把/system重新挂载了一下而已。


    C、方法三

    </pre><pre code_snippet_id="1875306" snippet_file_name="blog_20160910_8_46145" name="code" class="cpp">#修改系统分区属性为可写
    mount -o remount,rw mtd@system /system
    
    #修改系统分区属性为只读
    mount -o remount,ro mtd@system /system

    这种方法虽然也是可以的。之所以这么写是参考init.rc里面的mount写法。不清楚这个是怎么回事,要是原生的linux mount命令应该不会这样的。具体的原因,我猜应该看看,联想一下mount命令的实现源码,也许执行下面的命令也能修改系统分区的属性:

    #修改系统分区属性为可写
    mount -o remount,rw loop@system /system
    
    #修改系统分区属性为只读
    mount -o remount,ro loop@system /system



    不墨迹了,该整理的也整理了。很多的内容是参阅大神的博客,本人Linux菜鸟一枚,再此不一一致谢了。本文的有些地方(知识点)还是需要验证和再思考的。如果有问题,希望大神们能指出来,拿砖拍我。关于Android系统上mount命令的使用就此做个比较,方便自己也方便他人。


    三、某安全应用修改Android系统分区属性所采用的方法


    很多Rom喜欢内置恶意的app软件,清除掉这些预转的恶意apk应用就需要在root授权的情况下,修改Android系统分区的属性为可写,将/system/app下的预装apk删除掉。某安全应用就提供了卸载预装系统apk的功能,比较好奇就稍微逆向分析了一下,该应用修改Android系统分区的方法如下:


    第1次修改Android系统分区,执行的命令操作:



    当前面第1次修改Android系统分区属性的方法失败,采用类似 执行cat /proc/mounts 或者 mount 找挂载设备文件的方法,再次执行mount命令。





    补充一点:

    如果想看某些分区的别名信息,对于高通平台上来说,可以通过下面的命令:

    [plain] view plain copy
     在CODE上查看代码片派生到我的代码片
    1. ls -al /dev/block/platform/msm_sdcc.1/by-name  

    在Google Nexus 6设备上,命令的显示结果如下:

    这些信息可以帮助发现每个分区到底是用来干什么的,比如/dev/block/mmcblk0p41这个分区就是用来存放/system的。

    有了这些信息,就可以使用dd命令,将感兴趣的分区全部倒出来进行分析。比如,如果想将TrustZone相关的tz分区倒出到sdcard上,可以使用下面的命令:

    [plain] view plain copy
     在CODE上查看代码片派生到我的代码片
    1. dd if=/dev/block/platform/msm_sdcc.1/by-name of=/sdcard/tz.img


    感谢链接:

    http://blog.chinaunix.net/uid-22731254-id-3222708.html

    http://blog.csdn.net/cainiao413/article/details/6156812

    http://bbs.csdn.net/topics/330050292

    http://blog.sina.com.cn/s/blog_4a4aca6501008ath.html

    http://blog.csdn.net/candyguy242/article/details/8054973

    http://blog.chinaunix.net/uid-20564848-id-73964.html

    http://www.miui.com/thread-611911-1-1.html

    http://blog.csdn.net/roland_sun?viewmode=contents









  • 相关阅读:
    5.2.9.字符设备驱动代码实践2
    5.2.8.字符设备驱动代码实践1
    5.2.7.字符设备驱动工作原理2
    5.2.6.字符设备驱动工作原理1
    带参宏定义的思考
    重读gets()与is函数的用法
    地址/指针和字符串
    总体来说,require_once 肯定要比 require 性能好
    auto_prepend_file与auto_append_file使用方法
    经验分享:CSS浮动(float,clear)通俗讲解
  • 原文地址:https://www.cnblogs.com/csnd/p/11800673.html
Copyright © 2011-2022 走看看