#include #include #include #include #include #include #include struct cmdline { bool skip_initramfs; bool force_normal_boot; char slot[3]; char dt_dir[128]; }; struct raw_data { uint8_t *buf = nullptr; size_t sz = 0; raw_data() = default; raw_data(const raw_data&) = delete; raw_data(raw_data &&d) { buf = d.buf; sz = d.sz; d.buf = nullptr; d.sz = 0; } ~raw_data() { free(buf); } }; /* ************* * Base classes * *************/ class BaseInit { protected: cmdline *cmd; char **argv; std::vector mount_list; void exec_init(const char *init = "/init") { cleanup(); execv(init, argv); exit(1); } virtual void cleanup(); public: BaseInit(char *argv[], cmdline *cmd) : cmd(cmd), argv(argv), mount_list{"/sys", "/proc"} {} virtual ~BaseInit() = default; virtual void start() = 0; }; class MagiskInit : public BaseInit { protected: raw_data self; std::string persist_dir; virtual void early_mount() = 0; bool patch_sepolicy(const char *file = "/sepolicy"); public: MagiskInit(char *argv[], cmdline *cmd) : BaseInit(argv, cmd) {} }; class SARBase : public MagiskInit { protected: raw_data config; std::vector overlays; std::string tmp_dir; void backup_files(); void patch_rootdir(); void mount_system_root(); public: SARBase(char *argv[], cmdline *cmd) : MagiskInit(argv, cmd) {} void start() override { early_mount(); patch_rootdir(); exec_init(); } }; /* ************* * 2 Stage Init * *************/ class ForcedFirstStageInit : public BaseInit { private: void prepare(); public: ForcedFirstStageInit(char *argv[], cmdline *cmd) : BaseInit(argv, cmd) { LOGD("%s\n", __FUNCTION__); }; void start() override { prepare(); exec_init("/system/bin/init"); } }; class FirstStageInit : public BaseInit { private: void prepare(); public: FirstStageInit(char *argv[], cmdline *cmd) : BaseInit(argv, cmd) { LOGD("%s\n", __FUNCTION__); }; void start() override { prepare(); exec_init(); } }; class SARFirstStageInit : public SARBase { private: void traced_exec_init(); protected: void early_mount() override; public: SARFirstStageInit(char *argv[], cmdline *cmd) : SARBase(argv, cmd) { LOGD("%s\n", __FUNCTION__); }; void start() override { early_mount(); traced_exec_init(); } }; class SecondStageInit : public SARBase { protected: void early_mount() override; void cleanup() override { /* Do not do any cleanup */ } public: SecondStageInit(char *argv[]) : SARBase(argv, nullptr) { LOGD("%s\n", __FUNCTION__); }; }; /* *********** * Legacy SAR * ***********/ class SARInit : public SARBase { protected: void early_mount() override; public: SARInit(char *argv[], cmdline *cmd) : SARBase(argv, cmd) { LOGD("%s\n", __FUNCTION__); }; }; /* ********** * Initramfs * **********/ class RootFSInit : public MagiskInit { private: void setup_rootfs(); protected: void early_mount() override; public: RootFSInit(char *argv[], cmdline *cmd) : MagiskInit(argv, cmd) { LOGD("%s\n", __FUNCTION__); } void start() override { early_mount(); setup_rootfs(); exec_init(); } }; #define INIT_SOCKET "MAGISKINIT" void load_kernel_info(cmdline *cmd); int dump_magisk(const char *path, mode_t mode); int magisk_proxy_main(int argc, char *argv[]); void setup_klog(); void setup_tmp(const char *path, const raw_data &self, const raw_data &config);