/* ** Copyright 2018, John Wu (@topjohnwu) ** Copyright 2010, Adam Shanks (@ChainsDD) ** Copyright 2008, Zinx Verituse (@zinxv) ** */ #include #include #include #include #include #include #include #include #include #include "su.h" #define BROADCAST_REBOOT \ "/system/bin/app_process", "/system/bin", "com.android.commands.am.Am", \ "broadcast", "-n", nullptr, "-a", "android.intent.action.REBOOT", \ "-f", "0x10000020" static inline const char *get_command(const struct su_request *to) { if (to->command[0]) return to->command; if (to->shell[0]) return to->shell; return DEFAULT_SHELL; } static void silent_run(const char **args, struct su_info *info) { char component[128]; sprintf(component, "%s/a.h", info->str[SU_MANAGER].data()); args[5] = component; exec_t exec { .pre_exec = []() -> void { int null = xopen("/dev/null", O_WRONLY | O_CLOEXEC); dup2(null, STDOUT_FILENO); dup2(null, STDERR_FILENO); setenv("CLASSPATH", "/system/framework/am.jar", 1); }, .fork = fork_dont_care, .argv = args }; exec_command(exec); } static void setup_user(char *user, struct su_info *info) { switch (info->cfg[SU_MULTIUSER_MODE]) { case MULTIUSER_MODE_OWNER_ONLY: case MULTIUSER_MODE_OWNER_MANAGED: sprintf(user, "%d", 0); break; case MULTIUSER_MODE_USER: sprintf(user, "%d", info->uid / 100000); break; } } void app_log(struct su_context *ctx) { char user[8]; setup_user(user, ctx->info); char fromUid[8]; sprintf(fromUid, "%d", ctx->info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_OWNER_MANAGED ? ctx->info->uid % 100000 : ctx->info->uid); char toUid[8]; sprintf(toUid, "%d", ctx->req.uid); char pid[8]; sprintf(pid, "%d", ctx->pid); char policy[2]; sprintf(policy, "%d", ctx->info->access.policy); const char *cmd[] = { BROADCAST_REBOOT, "--user", user, "--es", "action", "log", "--ei", "from.uid", fromUid, "--ei", "to.uid", toUid, "--ei", "pid", pid, "--ei", "policy", policy, "--es", "command", get_command(&ctx->req), "--ez", "notify", ctx->info->access.notify ? "true" : "false", nullptr }; silent_run(cmd, ctx->info); } void app_notify(struct su_context *ctx) { char user[8]; setup_user(user, ctx->info); char fromUid[8]; sprintf(fromUid, "%d", ctx->info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_OWNER_MANAGED ? ctx->info->uid % 100000 : ctx->info->uid); char policy[2]; sprintf(policy, "%d", ctx->info->access.policy); const char *cmd[] = { BROADCAST_REBOOT, "--user", user, "--es", "action", "notify", "--ei", "from.uid", fromUid, "--ei", "policy", policy, nullptr }; silent_run(cmd, ctx->info); } void app_connect(const char *socket, struct su_info *info) { char user[8]; setup_user(user, info); const char *cmd[] = { BROADCAST_REBOOT, "--user", user, "--es", "action", "request", "--es", "socket", socket, nullptr }; silent_run(cmd, info); } void socket_send_request(int fd, struct su_info *info) { write_key_token(fd, "uid", info->uid); write_string_be(fd, "eof"); }