您的位置:首页 > 移动开发 > Android开发

深入学习Android系统上mount命令的使用

2017-04-12 16:53 204 查看
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()函数来实现修改系统分区属性的功能。

[cpp] view
plain copy







/*

* 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] = '\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
, mount_flags[i].name)) {

flags |= mount_flags[i].flag;

break;

}

}

if (!mount_flags[i].name) {

if (!strcmp(args
, "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
;

}

}

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\n");

}

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'\n", 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'\n", 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'\n", command);

return -EINVAL;

}

if (command[len] == ',') {

reboot_target = &command[len + 1];

} else if (command[len] == '\0') {

reboot_target = "";

} else {

ERROR("powerctl: unrecognized reboot target '%s'\n", &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'\n", 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\n", value);

return -EINVAL;

}

if (security_set_boolean_list(1, &b, 0) < 0) {

ret = -errno;

ERROR("setsebool: could not set %s to %s\n", 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函数的使用说明

[cpp] view
plain copy







【 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手机系统的修改分区属性的命令,那就老老实实的使用简化的命令:

[cpp] view
plain copy







<pre name="code" class="cpp">

[cpp] view
plain copy







#修改系统分区属性为可写

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就对应该设备文件了。但是这种方法明显操作起来不是很方便而且比较繁琐,需要计算和比较。

[cpp] view
plain copy







#修改系统分区属性为可写

mount -o remount,rw /dev/block/mmcblk0p25 /system

#修改系统分区属性为只读

mount -o remount,ro /dev/block/mmcblk0p25 /system



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



[cpp] view
plain copy







#修改系统分区属性为可写

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、方法三

[cpp] view
plain copy







</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命令的实现源码,也许执行下面的命令也能修改系统分区的属性:

[cpp] view
plain copy







#修改系统分区属性为可写

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







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







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
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐