182 lines
3.4 KiB
C++
182 lines
3.4 KiB
C++
#include <sys/mount.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <vector>
|
|
|
|
#include <logging.hpp>
|
|
|
|
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<std::string> 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<raw_file> 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);
|