#include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <pthread.h> #include <signal.h> #include <string.h> #include <sys/wait.h> #include <sys/types.h> #include <sys/mount.h> #include <daemon.hpp> #include <utils.hpp> #include "magiskhide.hpp" using namespace std::literals; [[noreturn]] static void usage(char *arg0) { fprintf(stderr, "MagiskHide - Hide Config CLI\n\n" "Usage: %s [action [arguments...] ]\n\n" "Actions:\n" " status Return the status of magiskhide\n" " enable Start magiskhide\n" " disable Stop magiskhide\n" " add PKG [PROC] Add a new target to the hide list\n" " rm PKG [PROC] Remove target(s) from the hide list\n" " ls Print the current hide list\n" " exec CMDs... Execute commands in isolated mount\n" " namespace and do all hide unmounts\n" #ifdef MAGISK_DEBUG " test Run process monitor test\n" #endif , arg0); exit(1); } void magiskhide_handler(int client) { int req = read_int(client); int res = DAEMON_ERROR; switch (req) { case STOP_MAGISKHIDE: case ADD_HIDELIST: case RM_HIDELIST: case LS_HIDELIST: if (!hide_enabled()) { write_int(client, HIDE_NOT_ENABLED); close(client); return; } } switch (req) { case LAUNCH_MAGISKHIDE: res = launch_magiskhide(); break; case STOP_MAGISKHIDE: res = stop_magiskhide(); break; case ADD_HIDELIST: res = add_list(client); break; case RM_HIDELIST: res = rm_list(client); break; case LS_HIDELIST: ls_list(client); return; case HIDE_STATUS: res = hide_enabled() ? HIDE_IS_ENABLED : HIDE_NOT_ENABLED; break; } write_int(client, res); close(client); } int magiskhide_main(int argc, char *argv[]) { if (argc < 2) usage(argv[0]); // CLI backwards compatibility const char *opt = argv[1]; if (opt[0] == '-' && opt[1] == '-') opt += 2; int req; if (opt == "enable"sv) req = LAUNCH_MAGISKHIDE; else if (opt == "disable"sv) req = STOP_MAGISKHIDE; else if (opt == "add"sv) req = ADD_HIDELIST; else if (opt == "rm"sv) req = RM_HIDELIST; else if (opt == "ls"sv) req = LS_HIDELIST; else if (opt == "status"sv) req = HIDE_STATUS; else if (opt == "exec"sv && argc > 2) { xunshare(CLONE_NEWNS); xmount(nullptr, "/", nullptr, MS_PRIVATE | MS_REC, nullptr); hide_unmount(); execvp(argv[2], argv + 2); exit(1); } #if 0 else if (opt == "test"sv) test_proc_monitor(); #endif else usage(argv[0]); // Send request int fd = connect_daemon(); write_int(fd, MAGISKHIDE); write_int(fd, req); if (req == ADD_HIDELIST || req == RM_HIDELIST) { write_string(fd, argv[2]); write_string(fd, argv[3] ? argv[3] : ""); } if (req == LS_HIDELIST) send_fd(fd, STDOUT_FILENO); // Get response int code = read_int(fd); switch (code) { case DAEMON_SUCCESS: break; case HIDE_NOT_ENABLED: fprintf(stderr, "MagiskHide is not enabled\n"); break; case HIDE_IS_ENABLED: fprintf(stderr, "MagiskHide is enabled\n"); break; case HIDE_ITEM_EXIST: fprintf(stderr, "Target already exists in hide list\n"); break; case HIDE_ITEM_NOT_EXIST: fprintf(stderr, "Target does not exist in hide list\n"); break; case HIDE_NO_NS: fprintf(stderr, "Your kernel doesn't support mount namespace\n"); break; case HIDE_INVALID_PKG: fprintf(stderr, "Invalid package / process name\n"); break; case ROOT_REQUIRED: fprintf(stderr, "Root is required for this operation\n"); break; case DAEMON_ERROR: default: fprintf(stderr, "Daemon error\n"); return DAEMON_ERROR; } return req == HIDE_STATUS ? (code == HIDE_IS_ENABLED ? 0 : 1) : code != DAEMON_SUCCESS; }