#include #include #include #include #include #include #include #include #include #include #include #include "core.hpp" using namespace std; int SDK_INT = -1; string MAGISKTMP; bool RECOVERY_MODE = false; int DAEMON_STATE = STATE_NONE; static struct stat self_st; static bool verify_client(pid_t pid) { // Verify caller is the same as server char path[32]; sprintf(path, "/proc/%d/exe", pid); struct stat st; return !(stat(path, &st) || st.st_dev != self_st.st_dev || st.st_ino != self_st.st_ino); } static bool check_zygote(pid_t pid) { char buf[32]; sprintf(buf, "/proc/%d/attr/current", pid); if (auto fp = open_file(buf, "r")) { fscanf(fp.get(), "%s", buf); return buf == "u:r:zygote:s0"sv; } else { return false; } } static void handle_request_async(int client, int code, ucred cred) { switch (code) { case MAGISKHIDE: magiskhide_handler(client, &cred); break; case SUPERUSER: su_daemon_handler(client, &cred); break; case POST_FS_DATA: post_fs_data(client); break; case LATE_START: late_start(client); break; case BOOT_COMPLETE: boot_complete(client); break; case SQLITE_CMD: exec_sql(client); break; case REMOVE_MODULES: remove_modules(); write_int(client, 0); close(client); reboot(); break; case ZYGISK_REQUEST: zygisk_handler(client, &cred); break; default: close(client); break; } } static void handle_request_sync(int client, int code) { switch (code) { case CHECK_VERSION: write_string(client, MAGISK_VERSION ":MAGISK"); break; case CHECK_VERSION_CODE: write_int(client, MAGISK_VER_CODE); break; case GET_PATH: write_string(client, MAGISKTMP.data()); break; case START_DAEMON: setup_logfile(true); break; case STOP_DAEMON: magiskhide_handler(-1, nullptr); write_int(client, 0); // Terminate the daemon! exit(0); } } static void handle_request(int client) { int code; // Verify client credentials ucred cred; get_client_cred(client, &cred); bool is_root = cred.uid == UID_ROOT; bool is_client = verify_client(cred.pid); bool is_zygote = !is_client && check_zygote(cred.pid); if (!is_root && !is_zygote && !is_client) goto done; code = read_int(client); if (code < 0 || (code & DAEMON_CODE_MASK) >= DAEMON_CODE_END) goto done; // Check client permissions switch (code) { case POST_FS_DATA: case LATE_START: case BOOT_COMPLETE: case SQLITE_CMD: case GET_PATH: case MAGISKHIDE: case STOP_DAEMON: if (!is_root) { write_int(client, ROOT_REQUIRED); goto done; } break; case REMOVE_MODULES: if (!is_root && cred.uid != UID_SHELL) { write_int(client, 1); goto done; } break; case ZYGISK_REQUEST: if (!is_zygote) { write_int(client, DAEMON_ERROR); goto done; } break; } if (code & SYNC_FLAG) { handle_request_sync(client, code); goto done; } // Handle complex requests in another thread exec_task([=] { handle_request_async(client, code, cred); }); return; done: close(client); } static int switch_cgroup(const char *cgroup, int pid) { char buf[32]; snprintf(buf, sizeof(buf), "%s/cgroup.procs", cgroup); int fd = open(buf, O_WRONLY | O_APPEND | O_CLOEXEC); if (fd == -1) return -1; snprintf(buf, sizeof(buf), "%d\n", pid); if (xwrite(fd, buf, strlen(buf)) == -1) { close(fd); return -1; } close(fd); return 0; } [[noreturn]] static void daemon_entry() { magisk_logging(); // Block all signals sigset_t block_set; sigfillset(&block_set); pthread_sigmask(SIG_SETMASK, &block_set, nullptr); // Change process name set_nice_name("magiskd"); int fd = xopen("/dev/null", O_WRONLY); xdup2(fd, STDOUT_FILENO); xdup2(fd, STDERR_FILENO); if (fd > STDERR_FILENO) close(fd); fd = xopen("/dev/zero", O_RDONLY); xdup2(fd, STDIN_FILENO); if (fd > STDERR_FILENO) close(fd); setsid(); setcon("u:r:" SEPOL_PROC_DOMAIN ":s0"); start_log_daemon(); LOGI(NAME_WITH_VER(Magisk) " daemon started\n"); // Escape from cgroup int pid = getpid(); if (switch_cgroup("/acct", pid) && switch_cgroup("/sys/fs/cgroup", pid)) LOGW("Can't switch cgroup\n"); // Get self stat char buf[64]; xreadlink("/proc/self/exe", buf, sizeof(buf)); MAGISKTMP = dirname(buf); xstat("/proc/self/exe", &self_st); // Get API level parse_prop_file("/system/build.prop", [](auto key, auto val) -> bool { if (key == "ro.build.version.sdk") { SDK_INT = parse_int(val); return false; } return true; }); if (SDK_INT < 0) { // In case some devices do not store this info in build.prop, fallback to getprop auto sdk = getprop("ro.build.version.sdk"); if (!sdk.empty()) { SDK_INT = parse_int(sdk); } } LOGI("* Device API level: %d\n", SDK_INT); restore_tmpcon(); // SAR cleanups auto mount_list = MAGISKTMP + "/" ROOTMNT; if (access(mount_list.data(), F_OK) == 0) { file_readline(true, mount_list.data(), [](string_view line) -> bool { umount2(line.data(), MNT_DETACH); return true; }); } unlink("/dev/.se"); // Load config status auto config = MAGISKTMP + "/" INTLROOT "/config"; parse_prop_file(config.data(), [](auto key, auto val) -> bool { if (key == "RECOVERYMODE" && val == "true") RECOVERY_MODE = true; return true; }); sockaddr_un sun; socklen_t len = setup_sockaddr(&sun, MAIN_SOCKET); fd = xsocket(AF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC, 0); if (xbind(fd, (sockaddr*) &sun, len)) exit(1); xlisten(fd, 10); // Loop forever to listen for requests for (;;) { int client = xaccept4(fd, nullptr, nullptr, SOCK_CLOEXEC); handle_request(client); } } int connect_daemon(bool create) { sockaddr_un sun; socklen_t len = setup_sockaddr(&sun, MAIN_SOCKET); int fd = xsocket(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0); if (connect(fd, (sockaddr*) &sun, len)) { if (!create || getuid() != UID_ROOT) { LOGE("No daemon is currently running!\n"); return -1; } if (fork_dont_care() == 0) { close(fd); daemon_entry(); } while (connect(fd, (struct sockaddr*) &sun, len)) usleep(10000); } return fd; }