Magisk/native/jni/core/bootstages.c

822 lines
21 KiB
C
Raw Normal View History

2017-07-10 16:48:14 +02:00
/* bootstages.c - Core bootstage operations
*
* All bootstage operations, including simple mount in post-fs,
* magisk mount in post-fs-data, various image handling, script
* execution, load modules, install Magisk Manager etc.
2017-04-30 19:58:52 +02:00
*/
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <dirent.h>
#include <sys/mount.h>
#include <sys/wait.h>
2017-05-03 20:39:53 +02:00
#include <selinux/selinux.h>
2017-04-30 19:58:52 +02:00
#include "magisk.h"
#include "utils.h"
#include "daemon.h"
#include "resetprop.h"
2018-04-14 21:18:18 +02:00
#include "magiskpolicy.h"
2017-04-30 19:58:52 +02:00
static char *buf, *buf2;
static struct vector module_list;
2017-08-30 20:20:06 +02:00
extern char **environ;
2017-04-30 19:58:52 +02:00
/******************
* Node structure *
******************/
2017-06-11 21:29:01 +02:00
// Precedence: MODULE > SKEL > INTER > DUMMY
#define IS_DUMMY 0x01 /* mount from mirror */
#define IS_INTER 0x02 /* intermediate node */
#define IS_SKEL 0x04 /* mount from skeleton */
#define IS_MODULE 0x08 /* mount from module */
2017-08-11 05:04:02 +02:00
#define IS_DIR(n) (n->type == DT_DIR)
#define IS_LNK(n) (n->type == DT_LNK)
#define IS_REG(n) (n->type == DT_REG)
2017-04-30 19:58:52 +02:00
struct node_entry {
const char *module; /* Only used when status & IS_MODULE */
2017-04-30 19:58:52 +02:00
char *name;
uint8_t type;
uint8_t status;
struct node_entry *parent;
struct vector *children;
};
2017-08-11 05:04:02 +02:00
static void concat_path(struct node_entry *node) {
if (node->parent)
concat_path(node->parent);
size_t len = strlen(buf);
2017-08-11 05:04:02 +02:00
buf[len] = '/';
strcpy(buf + len + 1, node->name);
}
static char *get_full_path(struct node_entry *node) {
buf[0] = '\0';
concat_path(node);
return strdup(buf);
}
// Free the node
static void destroy_node(struct node_entry *node) {
free(node->name);
vec_destroy(node->children);
free(node->children);
free(node);
}
// Free the node and all children recursively
static void destroy_subtree(struct node_entry *node) {
// Never free parent, since it shall be freed by themselves
struct node_entry *e;
vec_for_each(node->children, e) {
destroy_subtree(e);
}
destroy_node(node);
}
// Return the child
static struct node_entry *insert_child(struct node_entry *p, struct node_entry *c) {
c->parent = p;
if (p->children == NULL) {
p->children = xmalloc(sizeof(struct vector));
vec_init(p->children);
}
struct node_entry *e;
vec_for_each(p->children, e) {
if (strcmp(e->name, c->name) == 0) {
// Exist duplicate
if (c->status > e->status) {
// Precedence is higher, replace with new node
destroy_subtree(e);
2017-09-07 13:22:30 +02:00
vec_cur(p->children) = c;
2017-08-11 05:04:02 +02:00
return c;
} else {
// Free the new entry, return old
destroy_node(c);
return e;
}
}
}
// New entry, push back
vec_push_back(p->children, c);
return c;
}
2017-04-30 19:58:52 +02:00
/***********
2017-08-30 20:20:06 +02:00
* setenvs *
2017-04-30 19:58:52 +02:00
***********/
static void set_path(struct vector *v) {
2017-08-30 20:20:06 +02:00
for (int i = 0; environ[i]; ++i) {
if (strncmp(environ[i], "PATH=", 5) == 0) {
vec_push_back(v, strdup("PATH=" BBPATH ":/sbin:" MIRRDIR "/system/bin:"
MIRRDIR "/system/xbin:" MIRRDIR "/vendor/bin"));
2017-08-30 20:20:06 +02:00
} else {
vec_push_back(v, strdup(environ[i]));
}
}
vec_push_back(v, NULL);
}
static void uninstall_env(struct vector *v) {
vec_push_back(v, strdup("BOOTMODE=true"));
set_path(v);
}
2017-08-30 20:20:06 +02:00
/***********
* Scripts *
***********/
2017-08-01 09:34:16 +02:00
static void exec_common_script(const char* stage) {
2017-04-30 19:58:52 +02:00
DIR *dir;
struct dirent *entry;
snprintf(buf2, PATH_MAX, "%s/%s.d", COREDIR, stage);
2017-04-30 19:58:52 +02:00
if (!(dir = xopendir(buf2)))
2017-04-30 19:58:52 +02:00
return;
while ((entry = xreaddir(dir))) {
if (entry->d_type == DT_REG) {
snprintf(buf2, PATH_MAX, "%s/%s.d/%s", COREDIR, stage, entry->d_name);
2017-05-04 22:16:00 +02:00
if (access(buf2, X_OK) == -1)
2017-04-30 19:58:52 +02:00
continue;
LOGI("%s.d: exec [%s]\n", stage, entry->d_name);
int pid = exec_command(0, NULL, set_path, "sh", buf2, NULL);
2017-04-30 19:58:52 +02:00
if (pid != -1)
waitpid(pid, NULL, 0);
}
}
closedir(dir);
}
2017-08-01 09:34:16 +02:00
static void exec_module_script(const char* stage) {
2017-04-30 19:58:52 +02:00
char *module;
vec_for_each(&module_list, module) {
2017-08-03 12:07:34 +02:00
snprintf(buf2, PATH_MAX, "%s/%s/%s.sh", MOUNTPOINT, module, stage);
snprintf(buf, PATH_MAX, "%s/%s/disable", MOUNTPOINT, module);
if (access(buf2, F_OK) == -1 || access(buf, F_OK) == 0)
2017-04-30 19:58:52 +02:00
continue;
LOGI("%s: exec [%s.sh]\n", module, stage);
int pid = exec_command(0, NULL, set_path, "sh", buf2, NULL);
2017-04-30 19:58:52 +02:00
if (pid != -1)
waitpid(pid, NULL, 0);
}
}
/***************
* Magic Mount *
***************/
static void construct_tree(const char *module, struct node_entry *parent) {
2017-04-30 19:58:52 +02:00
DIR *dir;
struct dirent *entry;
struct node_entry *node;
char *parent_path = get_full_path(parent);
snprintf(buf, PATH_MAX, "%s/%s%s", MOUNTPOINT, module, parent_path);
2017-04-30 19:58:52 +02:00
2017-10-13 18:08:12 +02:00
if (!(dir = xopendir(buf)))
goto cleanup;
2017-04-30 19:58:52 +02:00
while ((entry = xreaddir(dir))) {
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
continue;
// Create new node
node = xcalloc(sizeof(*node), 1);
node->module = module;
node->name = strdup(entry->d_name);
node->type = entry->d_type;
snprintf(buf, PATH_MAX, "%s/%s", parent_path, node->name);
2017-06-11 21:29:01 +02:00
/*
* Clone the parent in the following condition:
* 1. File in module is a symlink
* 2. Target file do not exist
* 3. Target file is a symlink, but not /system/vendor
2017-07-30 14:14:12 +02:00
*/
2017-06-11 21:29:01 +02:00
int clone = 0;
if (IS_LNK(node) || access(buf, F_OK) == -1) {
2017-06-11 21:29:01 +02:00
clone = 1;
2017-08-11 05:04:02 +02:00
} else if (parent->parent != NULL || strcmp(node->name, "vendor") != 0) {
2017-06-11 21:29:01 +02:00
struct stat s;
xstat(buf, &s);
if (S_ISLNK(s.st_mode))
clone = 1;
}
if (clone) {
2017-04-30 19:58:52 +02:00
// Mark the parent folder as a skeleton
2017-06-11 21:29:01 +02:00
parent->status |= IS_SKEL; /* This will not overwrite if parent is module */
node->status = IS_MODULE;
} else if (IS_DIR(node)) {
// Check if marked as replace
snprintf(buf2, PATH_MAX, "%s/%s%s/.replace", MOUNTPOINT, module, buf);
if (access(buf2, F_OK) == 0) {
// Replace everything, mark as leaf
2017-06-11 21:29:01 +02:00
node->status = IS_MODULE;
} else {
// This will be an intermediate node
node->status = IS_INTER;
2017-04-30 19:58:52 +02:00
}
} else if (IS_REG(node)) {
// This is a leaf, mark as target
2017-06-11 21:29:01 +02:00
node->status = IS_MODULE;
}
node = insert_child(parent, node);
2017-06-13 18:55:41 +02:00
if (node->status & (IS_SKEL | IS_INTER)) {
// Intermediate folder, travel deeper
construct_tree(module, node);
2017-04-30 19:58:52 +02:00
}
}
2017-07-30 14:14:12 +02:00
2017-04-30 19:58:52 +02:00
closedir(dir);
cleanup:
free(parent_path);
2017-04-30 19:58:52 +02:00
}
static void clone_skeleton(struct node_entry *node) {
2017-04-30 19:58:52 +02:00
DIR *dir;
struct dirent *entry;
struct node_entry *dummy, *child;
// Clone the structure
char *full_path = get_full_path(node);
2017-06-11 21:29:01 +02:00
snprintf(buf, PATH_MAX, "%s%s", MIRRDIR, full_path);
2017-10-13 18:08:12 +02:00
if (!(dir = xopendir(buf)))
goto cleanup;
2017-04-30 19:58:52 +02:00
while ((entry = xreaddir(dir))) {
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
continue;
// Create dummy node
dummy = xcalloc(sizeof(*dummy), 1);
dummy->name = strdup(entry->d_name);
dummy->type = entry->d_type;
2017-06-11 21:29:01 +02:00
dummy->status = IS_DUMMY;
insert_child(node, dummy);
2017-04-30 19:58:52 +02:00
}
2017-05-01 22:55:55 +02:00
closedir(dir);
2017-04-30 19:58:52 +02:00
if (node->status & IS_SKEL) {
struct stat s;
char *con;
xstat(full_path, &s);
getfilecon(full_path, &con);
2017-12-31 14:54:39 +01:00
LOGI("mnt_tmpfs : %s\n", full_path);
2017-10-13 18:08:12 +02:00
xmount("tmpfs", full_path, "tmpfs", 0, NULL);
chmod(full_path, s.st_mode & 0777);
chown(full_path, s.st_uid, s.st_gid);
setfilecon(full_path, con);
free(con);
}
2017-04-30 19:58:52 +02:00
vec_for_each(node->children, child) {
snprintf(buf, PATH_MAX, "%s/%s", full_path, child->name);
2017-06-11 21:29:01 +02:00
// Create the dummy file/directory
if (IS_DIR(child))
2017-06-30 17:22:51 +02:00
xmkdir(buf, 0755);
else if (IS_REG(child))
2017-11-09 18:51:41 +01:00
close(creat(buf, 0644));
2017-06-11 21:29:01 +02:00
// Links will be handled later
2017-08-11 05:29:35 +02:00
if (child->parent->parent == NULL && strcmp(child->name, "vendor") == 0) {
2017-06-11 21:29:01 +02:00
if (IS_LNK(child)) {
cp_afc(MIRRDIR "/system/vendor", "/system/vendor");
2017-12-31 14:54:39 +01:00
LOGI("creat_link: %s <- %s\n", "/system/vendor", MIRRDIR "/system/vendor");
2017-06-11 21:29:01 +02:00
}
// Skip
continue;
} else if (child->status & IS_MODULE) {
2017-04-30 19:58:52 +02:00
// Mount from module file to dummy file
snprintf(buf2, PATH_MAX, "%s/%s%s/%s", MOUNTPOINT, child->module, full_path, child->name);
2017-06-11 21:29:01 +02:00
} else if (child->status & (IS_SKEL | IS_INTER)) {
// It's an intermediate folder, recursive clone
clone_skeleton(child);
2017-04-30 19:58:52 +02:00
continue;
} else if (child->status & IS_DUMMY) {
// Mount from mirror to dummy file
snprintf(buf2, PATH_MAX, "%s%s/%s", MIRRDIR, full_path, child->name);
2017-04-30 19:58:52 +02:00
}
if (IS_LNK(child)) {
2017-06-11 21:29:01 +02:00
// Copy symlinks directly
cp_afc(buf2, buf);
2018-06-03 08:43:03 +02:00
#ifdef MAGISK_DEBUG
LOGI("creat_link: %s <- %s\n",buf, buf2);
#else
LOGI("creat_link: %s\n", buf);
#endif
2017-04-30 19:58:52 +02:00
} else {
snprintf(buf, PATH_MAX, "%s/%s", full_path, child->name);
2017-04-30 19:58:52 +02:00
bind_mount(buf2, buf);
}
}
cleanup:
free(full_path);
2017-04-30 19:58:52 +02:00
}
static void magic_mount(struct node_entry *node) {
char *real_path;
struct node_entry *child;
2017-06-11 21:29:01 +02:00
if (node->status & IS_MODULE) {
// The real deal, mount module item
real_path = get_full_path(node);
snprintf(buf, PATH_MAX, "%s/%s%s", MOUNTPOINT, node->module, real_path);
bind_mount(buf, real_path);
free(real_path);
} else if (node->status & IS_SKEL) {
// The node is labeled to be cloned with skeleton, lets do it
clone_skeleton(node);
} else if (node->status & IS_INTER) {
// It's an intermediate node, travel deeper
vec_for_each(node->children, child)
magic_mount(child);
2017-04-30 19:58:52 +02:00
}
2017-06-11 21:29:01 +02:00
// The only thing goes here should be vendor placeholder
// There should be no dummies, so don't need to handle it here
2017-04-30 19:58:52 +02:00
}
2017-05-03 20:39:53 +02:00
/****************
* Simple Mount *
****************/
static void simple_mount(const char *path) {
DIR *dir;
struct dirent *entry;
snprintf(buf, PATH_MAX, "%s%s", SIMPLEMOUNT, path);
2017-05-03 20:39:53 +02:00
if (!(dir = opendir(buf)))
return;
while ((entry = xreaddir(dir))) {
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
continue;
// Target file path
snprintf(buf2, PATH_MAX, "%s/%s", path, entry->d_name);
// Only mount existing file
if (access(buf2, F_OK) == -1)
continue;
if (entry->d_type == DT_DIR) {
char *new_path = strdup(buf2);
simple_mount(new_path);
free(new_path);
} else if (entry->d_type == DT_REG) {
// Actual file path
snprintf(buf, PATH_MAX, "%s%s", SIMPLEMOUNT, buf2);
2017-05-03 20:39:53 +02:00
// Clone all attributes
2017-05-04 22:16:00 +02:00
clone_attr(buf2, buf);
2017-05-03 20:39:53 +02:00
// Finally, mount the file
bind_mount(buf, buf2);
}
}
closedir(dir);
}
2017-08-11 05:43:45 +02:00
/*****************
* Miscellaneous *
*****************/
2017-12-25 20:23:58 +01:00
#define alt_img ((char *[]) \
{ "/cache/magisk.img", "/data/magisk_merge.img", "/data/adb/magisk_merge.img", NULL })
2017-08-12 19:15:18 +02:00
static int prepare_img() {
2017-12-25 20:23:58 +01:00
// Merge images
for (int i = 0; alt_img[i]; ++i) {
if (merge_img(alt_img[i], MAINIMG)) {
LOGE("Image merge %s -> " MAINIMG " failed!\n", alt_img[i]);
return 1;
}
2017-10-11 21:39:39 +02:00
}
2017-08-12 19:15:18 +02:00
if (access(MAINIMG, F_OK) == -1) {
if (create_img(MAINIMG, 64))
return 1;
}
LOGI("* Mounting " MAINIMG "\n");
// Mounting magisk image
char *magiskloop = mount_image(MAINIMG, MOUNTPOINT);
if (magiskloop == NULL)
return 1;
2017-09-27 23:50:06 +02:00
xmkdir(COREDIR, 0755);
xmkdir(COREDIR "/post-fs-data.d", 0755);
xmkdir(COREDIR "/service.d", 0755);
xmkdir(COREDIR "/props", 0755);
DIR *dir = xopendir(MOUNTPOINT);
struct dirent *entry;
while ((entry = xreaddir(dir))) {
if (entry->d_type == DT_DIR) {
if (strcmp(entry->d_name, ".") == 0 ||
strcmp(entry->d_name, "..") == 0 ||
strcmp(entry->d_name, ".core") == 0 ||
strcmp(entry->d_name, "lost+found") == 0)
continue;
snprintf(buf, PATH_MAX, "%s/%s/remove", MOUNTPOINT, entry->d_name);
if (access(buf, F_OK) == 0) {
snprintf(buf, PATH_MAX, "%s/%s", MOUNTPOINT, entry->d_name);
2017-10-11 20:57:18 +02:00
rm_rf(buf);
2017-09-27 23:50:06 +02:00
continue;
2017-08-12 19:15:18 +02:00
}
snprintf(buf, PATH_MAX, "%s/%s/update", MOUNTPOINT, entry->d_name);
unlink(buf);
2017-09-27 23:50:06 +02:00
snprintf(buf, PATH_MAX, "%s/%s/disable", MOUNTPOINT, entry->d_name);
if (access(buf, F_OK) == 0)
continue;
vec_push_back(&module_list, strdup(entry->d_name));
2017-08-12 19:15:18 +02:00
}
2017-09-27 23:50:06 +02:00
}
2017-08-12 19:15:18 +02:00
2017-09-27 23:50:06 +02:00
closedir(dir);
2017-08-12 19:15:18 +02:00
2017-09-27 23:50:06 +02:00
// Trim image
umount_image(MOUNTPOINT, magiskloop);
free(magiskloop);
trim_img(MAINIMG);
// Remount them back :)
magiskloop = mount_image(MAINIMG, MOUNTPOINT);
free(magiskloop);
2017-08-12 19:15:18 +02:00
return 0;
}
2017-04-30 19:58:52 +02:00
/****************
* Entry points *
****************/
2017-05-03 19:13:04 +02:00
static void unblock_boot_process() {
close(xopen(UNBLOCKFILE, O_RDONLY | O_CREAT, 0));
2017-05-03 19:13:04 +02:00
pthread_exit(NULL);
}
static const char wrapper[] =
"#!/system/bin/sh\n"
"unset LD_LIBRARY_PATH\n"
"unset LD_PRELOAD\n"
"exec /sbin/magisk.bin \"${0##*/}\" \"$@\"\n";
void startup() {
if (!check_data())
return;
2017-04-30 19:58:52 +02:00
// uninstaller or core-only mode
2017-05-03 19:13:04 +02:00
if (access(UNINSTALLER, F_OK) == 0 || access(DISABLEFILE, F_OK) == 0)
goto initialize;
2017-05-03 19:13:04 +02:00
2017-04-30 19:58:52 +02:00
// Allocate buffer
buf = xmalloc(PATH_MAX);
buf2 = xmalloc(PATH_MAX);
2017-05-03 20:39:53 +02:00
simple_mount("/system");
simple_mount("/vendor");
initialize:
LOGI("** Initializing Magisk\n");
// Unlock all blocks for rw
unlock_blocks();
// Magisk binaries
char *bin_path = NULL;
if (access("/cache/data_bin", F_OK) == 0)
bin_path = "/cache/data_bin";
else if (access("/data/data/com.topjohnwu.magisk/install", F_OK) == 0)
bin_path = "/data/data/com.topjohnwu.magisk/install";
else if (access("/data/user_de/0/com.topjohnwu.magisk/install", F_OK) == 0)
bin_path = "/data/user_de/0/com.topjohnwu.magisk/install";
if (bin_path) {
rm_rf(DATABIN);
cp_afc(bin_path, DATABIN);
rm_rf(bin_path);
}
// Migration
rm_rf("/data/magisk");
unlink("/data/magisk.img");
unlink("/data/magisk_debug.log");
xmkdir("/data/adb", 0700);
chmod("/data/adb", 0700);
LOGI("* Creating /sbin overlay");
DIR *dir;
struct dirent *entry;
int root, sbin, fd;
char buf[PATH_MAX];
void *magisk, *init;
size_t magisk_size, init_size;
xmount(NULL, "/", NULL, MS_REMOUNT, NULL);
2018-04-29 06:34:36 +02:00
// Remove some traits of Magisk
unlink("/init.magisk.rc");
// Create hardlink mirror of /sbin to /root
mkdir("/root", 0750);
full_read("/sbin/magisk", &magisk, &magisk_size);
unlink("/sbin/magisk");
full_read("/sbin/magiskinit", &init, &init_size);
unlink("/sbin/magiskinit");
root = xopen("/root", O_RDONLY | O_CLOEXEC);
sbin = xopen("/sbin", O_RDONLY | O_CLOEXEC);
link_dir(sbin, root);
close(sbin);
// Mount the /sbin tmpfs overlay
xmount("tmpfs", "/sbin", "tmpfs", 0, NULL);
chmod("/sbin", 0755);
setfilecon("/sbin", "u:object_r:rootfs:s0");
sbin = xopen("/sbin", O_RDONLY | O_CLOEXEC);
// Create wrapper
fd = creat("/sbin/magisk", 0755);
xwrite(fd, wrapper, sizeof(wrapper) - 1);
close(fd);
setfilecon("/sbin/magisk", "u:object_r:"SEPOL_FILE_DOMAIN":s0");
// Setup magisk symlinks
fd = creat("/sbin/magisk.bin", 0755);
xwrite(fd, magisk, magisk_size);
close(fd);
free(magisk);
setfilecon("/sbin/magisk.bin", "u:object_r:"SEPOL_FILE_DOMAIN":s0");
for (int i = 0; applet[i]; ++i) {
snprintf(buf, PATH_MAX, "/sbin/%s", applet[i]);
xsymlink("/sbin/magisk", buf);
}
// Setup magiskinit symlinks
fd = creat("/sbin/magiskinit", 0755);
xwrite(fd, init, init_size);
close(fd);
free(init);
setfilecon("/sbin/magiskinit", "u:object_r:"SEPOL_FILE_DOMAIN":s0");
for (int i = 0; init_applet[i]; ++i) {
snprintf(buf, PATH_MAX, "/sbin/%s", init_applet[i]);
xsymlink("/sbin/magiskinit", buf);
}
// Create symlinks pointing back to /root
dir = xfdopendir(root);
while((entry = xreaddir(dir))) {
if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0) continue;
snprintf(buf, PATH_MAX, "/root/%s", entry->d_name);
symlinkat(buf, sbin, entry->d_name);
}
close(sbin);
close(root);
2018-05-26 15:25:59 +02:00
// Create directories in tmpfs overlay
xmkdirs(MIRRDIR "/system", 0755);
xmkdir(MIRRDIR "/bin", 0755);
xmkdir(BBPATH, 0755);
xmkdir(MOUNTPOINT, 0755);
LOGI("* Mounting mirrors");
struct vector mounts;
vec_init(&mounts);
file_to_vector("/proc/mounts", &mounts);
char *line;
int skip_initramfs = 0;
// Check whether skip_initramfs device
vec_for_each(&mounts, line) {
if (strstr(line, " /system_root ")) {
bind_mount("/system_root/system", MIRRDIR "/system");
skip_initramfs = 1;
2018-05-26 15:25:59 +02:00
} else if (!skip_initramfs && strstr(line, " /system ")) {
sscanf(line, "%s", buf);
xmount(buf, MIRRDIR "/system", "ext4", MS_RDONLY, NULL);
#ifdef MAGISK_DEBUG
LOGI("mount: %s <- %s\n", MIRRDIR "/system", buf);
#else
LOGI("mount: %s\n", MIRRDIR "/system");
#endif
} else if (strstr(line, " /vendor ")) {
seperate_vendor = 1;
sscanf(line, "%s", buf);
2018-05-26 15:25:59 +02:00
xmkdir(MIRRDIR "/vendor", 0755);
xmount(buf, MIRRDIR "/vendor", "ext4", MS_RDONLY, NULL);
#ifdef MAGISK_DEBUG
LOGI("mount: %s <- %s\n", MIRRDIR "/vendor", buf);
#else
LOGI("mount: %s\n", MIRRDIR "/vendor");
#endif
}
free(line);
}
vec_destroy(&mounts);
if (!seperate_vendor) {
xsymlink(MIRRDIR "/system/vendor", MIRRDIR "/vendor");
#ifdef MAGISK_DEBUG
LOGI("link: %s <- %s\n", MIRRDIR "/vendor", MIRRDIR "/system/vendor");
#else
LOGI("link: %s\n", MIRRDIR "/vendor");
#endif
}
2018-05-13 12:14:44 +02:00
xmkdirs(DATABIN, 0755);
bind_mount(DATABIN, MIRRDIR "/bin");
2018-05-13 12:14:44 +02:00
if (access(MIRRDIR "/bin/busybox", X_OK) == 0) {
LOGI("* Setting up internal busybox");
exec_command_sync(MIRRDIR "/bin/busybox", "--install", "-s", BBPATH, NULL);
xsymlink(MIRRDIR "/bin/busybox", BBPATH "/busybox");
}
// uninstall
if (access(UNINSTALLER, F_OK) == 0) {
close(open(UNBLOCKFILE, O_RDONLY | O_CREAT));
exec_command(0, NULL, uninstall_env, "sh", UNINSTALLER, NULL);
return;
}
// Start post-fs-data mode
execl("/sbin/magisk", "magisk", "--post-fs-data", NULL);
2017-04-30 19:58:52 +02:00
}
void post_fs_data(int client) {
// ack
write_int(client, 0);
close(client);
// Start the debug log
start_debug_full_log();
2017-06-08 13:15:51 +02:00
2017-05-03 19:13:04 +02:00
LOGI("** post-fs-data mode running\n");
xmount(NULL, "/", NULL, MS_REMOUNT | MS_RDONLY, NULL);
full_patch_pid = exec_command(0, NULL, NULL, "/sbin/magiskpolicy", "--live", "allow "SEPOL_PROC_DOMAIN" * * *", NULL);
2017-04-30 19:58:52 +02:00
// Allocate buffer
buf = xmalloc(PATH_MAX);
buf2 = xmalloc(PATH_MAX);
2017-12-15 21:40:54 +01:00
vec_init(&module_list);
2017-04-30 19:58:52 +02:00
2017-10-11 21:39:39 +02:00
// Merge, trim, mount magisk.img, which will also travel through the modules
2017-08-12 19:15:18 +02:00
// After this, it will create the module list
if (prepare_img())
goto core_only; // Mounting fails, we can only do core only stuffs
2017-04-30 19:58:52 +02:00
2018-06-03 08:43:03 +02:00
restorecon();
2017-08-12 19:15:18 +02:00
// Run common scripts
LOGI("* Running post-fs-data.d scripts\n");
exec_common_script("post-fs-data");
2017-06-02 23:52:49 +02:00
2017-05-03 19:13:04 +02:00
// Core only mode
if (access(DISABLEFILE, F_OK) == 0)
goto core_only;
2017-05-03 19:13:04 +02:00
2017-08-12 19:15:18 +02:00
// Execute module scripts
LOGI("* Running module post-fs-data scripts\n");
exec_module_script("post-fs-data");
2017-04-30 19:58:52 +02:00
char *module;
struct node_entry *sys_root, *ven_root = NULL, *child;
// Create the system root entry
sys_root = xcalloc(sizeof(*sys_root), 1);
2017-08-11 05:04:02 +02:00
sys_root->name = strdup("system");
2017-06-11 21:29:01 +02:00
sys_root->status = IS_INTER;
2017-04-30 19:58:52 +02:00
int has_modules = 0;
LOGI("* Loading modules\n");
2017-08-12 19:15:18 +02:00
vec_for_each(&module_list, module) {
// Read props
snprintf(buf, PATH_MAX, "%s/%s/system.prop", MOUNTPOINT, module);
if (access(buf, F_OK) == 0) {
LOGI("%s: loading [system.prop]\n", module);
read_prop_file(buf, 0);
}
// Check whether enable auto_mount
snprintf(buf, PATH_MAX, "%s/%s/auto_mount", MOUNTPOINT, module);
if (access(buf, F_OK) == -1)
continue;
// Double check whether the system folder exists
snprintf(buf, PATH_MAX, "%s/%s/system", MOUNTPOINT, module);
if (access(buf, F_OK) == -1)
continue;
2017-05-03 19:13:04 +02:00
2017-08-12 19:15:18 +02:00
// Construct structure
has_modules = 1;
LOGI("%s: constructing magic mount structure\n", module);
// If /system/vendor exists in module, create a link outside
snprintf(buf, PATH_MAX, "%s/%s/system/vendor", MOUNTPOINT, module);
if (access(buf, F_OK) == 0) {
snprintf(buf2, PATH_MAX, "%s/%s/vendor", MOUNTPOINT, module);
unlink(buf2);
2017-10-13 18:08:12 +02:00
xsymlink(buf, buf2);
2017-04-30 19:58:52 +02:00
}
2017-08-12 19:15:18 +02:00
construct_tree(module, sys_root);
2017-04-30 19:58:52 +02:00
}
if (has_modules) {
2017-06-11 21:29:01 +02:00
// Extract the vendor node out of system tree and swap with placeholder
2017-04-30 19:58:52 +02:00
vec_for_each(sys_root->children, child) {
if (strcmp(child->name, "vendor") == 0) {
ven_root = child;
2017-06-11 21:29:01 +02:00
child = xcalloc(sizeof(*child), 1);
child->type = seperate_vendor ? DT_LNK : DT_DIR;
child->parent = ven_root->parent;
2017-08-11 05:29:35 +02:00
child->name = strdup("vendor");
child->status = 0;
// Swap!
2017-09-07 13:22:30 +02:00
vec_cur(sys_root->children) = child;
2017-04-30 19:58:52 +02:00
ven_root->parent = NULL;
break;
}
}
2017-06-11 21:29:01 +02:00
// Magic!!
magic_mount(sys_root);
if (ven_root) magic_mount(ven_root);
2017-04-30 19:58:52 +02:00
}
// Cleanup memory
destroy_subtree(sys_root);
2017-06-11 21:29:01 +02:00
if (ven_root) destroy_subtree(ven_root);
2017-04-30 19:58:52 +02:00
core_only:
2017-05-03 21:05:37 +02:00
// Systemless hosts
if (access(HOSTSFILE, F_OK) == 0) {
LOGI("* Enabling systemless hosts file support");
bind_mount(HOSTSFILE, "/system/etc/hosts");
}
2017-10-10 13:49:15 +02:00
auto_start_magiskhide();
2017-04-30 19:58:52 +02:00
unblock_boot_process();
}
void late_start(int client) {
LOGI("** late_start service mode running\n");
// ack
write_int(client, 0);
close(client);
2017-05-04 22:16:00 +02:00
// Allocate buffer
if (buf == NULL) buf = xmalloc(PATH_MAX);
if (buf2 == NULL) buf2 = xmalloc(PATH_MAX);
2017-04-30 19:58:52 +02:00
// Wait till the full patch is done
2018-04-14 21:18:18 +02:00
waitpid(full_patch_pid, NULL, 0);
2017-04-30 19:58:52 +02:00
2017-08-12 19:15:18 +02:00
// Run scripts after full patch, most reliable way to run scripts
LOGI("* Running service.d scripts\n");
exec_common_script("service");
2017-06-11 10:51:44 +02:00
// Core only mode
2017-07-31 17:31:40 +02:00
if (access(DISABLEFILE, F_OK) == 0)
goto core_only;
2017-06-11 10:51:44 +02:00
2017-04-30 19:58:52 +02:00
LOGI("* Running module service scripts\n");
exec_module_script("service");
2017-07-31 17:31:40 +02:00
core_only:
// Install Magisk Manager if exists
if (access(MANAGERAPK, F_OK) == 0) {
rename(MANAGERAPK, "/data/magisk.apk");
2018-04-14 21:18:18 +02:00
setfilecon("/data/magisk.apk", "u:object_r:"SEPOL_FILE_DOMAIN":s0");
2017-06-11 10:51:44 +02:00
while (1) {
sleep(5);
2018-04-07 21:22:22 +02:00
LOGD("apk_install: attempting to install APK");
2017-07-10 16:29:53 +02:00
int apk_res = -1, pid;
2018-04-07 21:22:22 +02:00
pid = exec_command(1, &apk_res, NULL,
"/system/bin/pm", "install", "-r", "/data/magisk.apk", NULL);
2017-08-01 09:34:16 +02:00
if (pid != -1) {
int err = 0;
while (fdgets(buf, PATH_MAX, apk_res) > 0) {
LOGD("apk_install: %s", buf);
err |= strstr(buf, "Error:") != NULL;
}
2017-08-01 09:34:16 +02:00
waitpid(pid, NULL, 0);
close(apk_res);
// Keep trying until pm is started
if (err)
continue;
break;
2017-08-01 09:34:16 +02:00
}
}
unlink("/data/magisk.apk");
}
2017-04-30 19:58:52 +02:00
// All boot stage done, cleanup everything
free(buf);
free(buf2);
2017-07-10 19:53:45 +02:00
buf = buf2 = NULL;
2017-04-30 19:58:52 +02:00
vec_deep_destroy(&module_list);
2017-06-07 21:20:49 +02:00
stop_debug_full_log();
2017-04-30 19:58:52 +02:00
}