mirror of
https://github.com/topjohnwu/Magisk.git
synced 2025-01-12 17:43:36 +00:00
Cleanup implementation
This commit is contained in:
parent
17569005a4
commit
cd7a335d0f
@ -127,7 +127,7 @@ string find_preinit_device() {
|
|||||||
part_t ext4_type = UNKNOWN;
|
part_t ext4_type = UNKNOWN;
|
||||||
part_t f2fs_type = UNKNOWN;
|
part_t f2fs_type = UNKNOWN;
|
||||||
|
|
||||||
bool encrypted = getprop("ro.crypto.state") == "encrypted";
|
bool encrypted = get_prop("ro.crypto.state") == "encrypted";
|
||||||
bool mount = getuid() == 0 && getenv("MAGISKTMP");
|
bool mount = getuid() == 0 && getenv("MAGISKTMP");
|
||||||
bool make_dev = mount && getenv("MAKEDEV");
|
bool make_dev = mount && getenv("MAKEDEV");
|
||||||
|
|
||||||
@ -283,14 +283,14 @@ static bool check_data() {
|
|||||||
});
|
});
|
||||||
if (!mnt)
|
if (!mnt)
|
||||||
return false;
|
return false;
|
||||||
auto crypto = getprop("ro.crypto.state");
|
auto crypto = get_prop("ro.crypto.state");
|
||||||
if (!crypto.empty()) {
|
if (!crypto.empty()) {
|
||||||
if (crypto != "encrypted") {
|
if (crypto != "encrypted") {
|
||||||
// Unencrypted, we can directly access data
|
// Unencrypted, we can directly access data
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
// Encrypted, check whether vold is started
|
// Encrypted, check whether vold is started
|
||||||
return !getprop("init.svc.vold").empty();
|
return !get_prop("init.svc.vold").empty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ro.crypto.state is not set, assume it's unencrypted
|
// ro.crypto.state is not set, assume it's unencrypted
|
||||||
@ -391,8 +391,8 @@ static void post_fs_data() {
|
|||||||
goto early_abort;
|
goto early_abort;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getprop("persist.sys.safemode", true) == "1" ||
|
if (get_prop("persist.sys.safemode", true) == "1" ||
|
||||||
getprop("ro.sys.safemode") == "1" || check_key_combo()) {
|
get_prop("ro.sys.safemode") == "1" || check_key_combo()) {
|
||||||
boot_state |= FLAG_SAFE_MODE;
|
boot_state |= FLAG_SAFE_MODE;
|
||||||
// Disable all modules and denylist so next boot will be clean
|
// Disable all modules and denylist so next boot will be clean
|
||||||
disable_modules();
|
disable_modules();
|
||||||
|
@ -330,7 +330,7 @@ static void daemon_entry() {
|
|||||||
switch_cgroup("/acct", pid);
|
switch_cgroup("/acct", pid);
|
||||||
switch_cgroup("/dev/cg2_bpf", pid);
|
switch_cgroup("/dev/cg2_bpf", pid);
|
||||||
switch_cgroup("/sys/fs/cgroup", pid);
|
switch_cgroup("/sys/fs/cgroup", pid);
|
||||||
if (getprop("ro.config.per_app_memcg") != "false") {
|
if (get_prop("ro.config.per_app_memcg") != "false") {
|
||||||
switch_cgroup("/dev/memcg/apps", pid);
|
switch_cgroup("/dev/memcg/apps", pid);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -350,7 +350,7 @@ static void daemon_entry() {
|
|||||||
});
|
});
|
||||||
if (SDK_INT < 0) {
|
if (SDK_INT < 0) {
|
||||||
// In case some devices do not store this info in build.prop, fallback to getprop
|
// In case some devices do not store this info in build.prop, fallback to getprop
|
||||||
auto sdk = getprop("ro.build.version.sdk");
|
auto sdk = get_prop("ro.build.version.sdk");
|
||||||
if (!sdk.empty()) {
|
if (!sdk.empty()) {
|
||||||
SDK_INT = parse_int(sdk);
|
SDK_INT = parse_int(sdk);
|
||||||
}
|
}
|
||||||
|
@ -264,7 +264,8 @@ void load_modules() {
|
|||||||
strcpy(b, "system.prop");
|
strcpy(b, "system.prop");
|
||||||
if (access(buf, F_OK) == 0) {
|
if (access(buf, F_OK) == 0) {
|
||||||
LOGI("%s: loading [system.prop]\n", module);
|
LOGI("%s: loading [system.prop]\n", module);
|
||||||
load_prop_file(buf, false);
|
// Do NOT go through property service as it could cause boot lock
|
||||||
|
load_prop_file(buf, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check whether skip mounting
|
// Check whether skip mounting
|
||||||
|
@ -3,10 +3,7 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
|
||||||
int setprop(const char *name, const char *value, bool prop_svc = true);
|
std::string get_prop(const char *name, bool persist = false);
|
||||||
std::string getprop(const char *name, bool persist = false);
|
int delete_prop(const char *name, bool persist = false);
|
||||||
std::string getpropcontext(const char *name);
|
int set_prop(const char *name, const char *value, bool skip_svc = false);
|
||||||
void getprops(void (*callback)(const char *, const char *, void *),
|
void load_prop_file(const char *filename, bool skip_svc = false);
|
||||||
void *cookie = nullptr, bool persist = false);
|
|
||||||
int delprop(const char *name, bool persist = false);
|
|
||||||
void load_prop_file(const char *filename, bool prop_svc = true);
|
|
||||||
|
@ -25,6 +25,21 @@ static void (*system_property_read_callback)(
|
|||||||
static int (*system_property_foreach)(void (*)(const prop_info*, void*), void*);
|
static int (*system_property_foreach)(void (*)(const prop_info*, void*), void*);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
struct PropFlags {
|
||||||
|
void setSkipSvc() { flags |= 1; }
|
||||||
|
void setPersist() { flags |= (1 << 1); }
|
||||||
|
void setContext() { flags |= (1 << 2); }
|
||||||
|
void setDelete() { flags |= (1 << 3); }
|
||||||
|
void setLoadFile() { flags |= (1 << 4); }
|
||||||
|
bool isSkipSvc() const { return flags & 1; }
|
||||||
|
bool isPersist() const { return flags & (1 << 1); }
|
||||||
|
bool isContext() const { return flags & (1 << 2); }
|
||||||
|
bool isDelete() const { return flags & (1 << 3); }
|
||||||
|
bool isLoadFile() const { return flags & (1 << 4); }
|
||||||
|
private:
|
||||||
|
uint32_t flags = 0;
|
||||||
|
};
|
||||||
|
|
||||||
[[noreturn]] static void usage(char* arg0) {
|
[[noreturn]] static void usage(char* arg0) {
|
||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
R"EOF(resetprop - System Property Manipulation Tool
|
R"EOF(resetprop - System Property Manipulation Tool
|
||||||
@ -127,34 +142,35 @@ struct resetprop {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int setprop(const char *name, const char *value, bool prop_svc) {
|
int set_prop(const char *name, const char *value, PropFlags flags) {
|
||||||
if (!check_legal_property_name(name))
|
if (!check_legal_property_name(name))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
const char *msg = prop_svc ? "property_service" : "modifying prop data structure";
|
const char *msg = flags.isSkipSvc() ? "modifying prop data structure" : "property_service";
|
||||||
|
|
||||||
int ret;
|
|
||||||
auto pi = const_cast<prop_info *>(__system_property_find(name));
|
auto pi = const_cast<prop_info *>(__system_property_find(name));
|
||||||
|
|
||||||
// Always delete existing read-only properties, because they could be
|
// Always delete existing read-only properties, because they could be
|
||||||
// long properties and cannot directly go through __system_property_update
|
// long properties and cannot directly go through __system_property_update
|
||||||
if (pi != nullptr && str_starts(name, "ro.")) {
|
if (pi != nullptr && str_starts(name, "ro.")) {
|
||||||
__system_property_delete(name, false /* Do NOT trim node */);
|
// Skip pruning nodes as we will add it back ASAP
|
||||||
|
__system_property_delete(name, false);
|
||||||
pi = nullptr;
|
pi = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int ret;
|
||||||
if (pi != nullptr) {
|
if (pi != nullptr) {
|
||||||
if (prop_svc) {
|
if (flags.isSkipSvc()) {
|
||||||
ret = system_property_set(name, value);
|
|
||||||
} else {
|
|
||||||
ret = __system_property_update(pi, value, strlen(value));
|
ret = __system_property_update(pi, value, strlen(value));
|
||||||
|
} else {
|
||||||
|
ret = system_property_set(name, value);
|
||||||
}
|
}
|
||||||
LOGD("resetprop: update prop [%s]: [%s] by %s\n", name, value, msg);
|
LOGD("resetprop: update prop [%s]: [%s] by %s\n", name, value, msg);
|
||||||
} else {
|
} else {
|
||||||
if (prop_svc) {
|
if (flags.isSkipSvc()) {
|
||||||
ret = system_property_set(name, value);
|
|
||||||
} else {
|
|
||||||
ret = __system_property_add(name, strlen(name), value, strlen(value));
|
ret = __system_property_add(name, strlen(name), value, strlen(value));
|
||||||
|
} else {
|
||||||
|
ret = system_property_set(name, value);
|
||||||
}
|
}
|
||||||
LOGD("resetprop: create prop [%s]: [%s] by %s\n", name, value, msg);
|
LOGD("resetprop: create prop [%s]: [%s] by %s\n", name, value, msg);
|
||||||
}
|
}
|
||||||
@ -165,10 +181,17 @@ struct resetprop {
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
string getprop(const char *name, bool persist) {
|
string get_prop(const char *name, PropFlags flags) {
|
||||||
string val;
|
|
||||||
if (!check_legal_property_name(name))
|
if (!check_legal_property_name(name))
|
||||||
|
return "";
|
||||||
|
|
||||||
|
if (flags.isContext()) {
|
||||||
|
auto val = __system_property_get_context(name) ?: "";
|
||||||
|
LOGD("resetprop: getcontext [%s]: [%s]\n", name, val);
|
||||||
return val;
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
string val;
|
||||||
auto pi = system_property_find(name);
|
auto pi = system_property_find(name);
|
||||||
if (pi == nullptr)
|
if (pi == nullptr)
|
||||||
return val;
|
return val;
|
||||||
@ -176,102 +199,55 @@ struct resetprop {
|
|||||||
read_prop_with_cb(pi, &cb);
|
read_prop_with_cb(pi, &cb);
|
||||||
LOGD("resetprop: getprop [%s]: [%s]\n", name, val.data());
|
LOGD("resetprop: getprop [%s]: [%s]\n", name, val.data());
|
||||||
|
|
||||||
if (val.empty() && persist && strncmp(name, "persist.", 8) == 0)
|
if (val.empty() && flags.isPersist() && strncmp(name, "persist.", 8) == 0)
|
||||||
val = persist_getprop(name);
|
val = persist_getprop(name);
|
||||||
if (val.empty())
|
if (val.empty())
|
||||||
LOGD("resetprop: prop [%s] does not exist\n", name);
|
LOGD("resetprop: prop [%s] does not exist\n", name);
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
string getcontext(const char *name) {
|
void print_props(PropFlags flags) {
|
||||||
if (!check_legal_property_name(name))
|
|
||||||
return "";
|
|
||||||
auto val = __system_property_get_context(name) ?: "";
|
|
||||||
LOGD("resetprop: getcontext [%s]: [%s]\n", name, val);
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
void getprops(void (*callback)(const char *, const char *, void *),
|
|
||||||
void *cookie, bool persist) {
|
|
||||||
prop_list list;
|
prop_list list;
|
||||||
prop_collector collector(list);
|
prop_collector collector(list);
|
||||||
system_property_foreach(read_prop_with_cb, &collector);
|
system_property_foreach(read_prop_with_cb, &collector);
|
||||||
if (persist)
|
if (flags.isPersist())
|
||||||
persist_getprops(&collector);
|
persist_getprops(&collector);
|
||||||
for (auto &[key, val] : list)
|
for (auto &[key, val] : list) {
|
||||||
callback(key.data(), val.data(), cookie);
|
const char *v = flags.isContext() ?
|
||||||
|
(__system_property_get_context(key.data()) ?: "") :
|
||||||
|
val.data();
|
||||||
|
printf("[%s]: [%s]\n", key.data(), v);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not an error when something is deleted
|
// Not an error when something is deleted
|
||||||
int delprop(const char *name, bool persist) {
|
int delete_prop(const char *name, PropFlags flags) {
|
||||||
if (!check_legal_property_name(name))
|
|
||||||
return 1;
|
|
||||||
LOGD("resetprop: delete prop [%s]\n", name);
|
LOGD("resetprop: delete prop [%s]\n", name);
|
||||||
|
|
||||||
int ret = __system_property_delete(name, true);
|
int ret = __system_property_delete(name, true);
|
||||||
if (persist && str_starts(name, "persist.")) {
|
if (flags.isPersist() && str_starts(name, "persist.")) {
|
||||||
if (persist_deleteprop(name))
|
if (persist_deleteprop(name))
|
||||||
ret = 0;
|
ret = 0;
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void load_file(const char *filename, PropFlags flags) {
|
||||||
|
LOGD("resetprop: Parse prop file [%s]\n", filename);
|
||||||
|
parse_prop_file(filename, [&](auto key, auto val) -> bool {
|
||||||
|
set_prop(key.data(), val.data(), flags);
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
static resetprop *get_impl() {
|
static resetprop *get_impl() {
|
||||||
static resetprop *impl = new resetprop();
|
static resetprop impl;
|
||||||
return impl;
|
return &impl;
|
||||||
}
|
|
||||||
|
|
||||||
/***********************************
|
|
||||||
* Implementation of top-level APIs
|
|
||||||
***********************************/
|
|
||||||
|
|
||||||
static void print_props(bool persist) {
|
|
||||||
getprops([](const char *name, const char *value, auto) {
|
|
||||||
printf("[%s]: [%s]\n", name, value);
|
|
||||||
}, nullptr, persist);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void print_props_context() {
|
|
||||||
getprops([](const char *name, const char *, auto) {
|
|
||||||
printf("[%s]: [%s]\n", name, getpropcontext(name).data());
|
|
||||||
}, nullptr, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
string getprop(const char *name, bool persist) {
|
|
||||||
return get_impl()->getprop(name, persist);
|
|
||||||
}
|
|
||||||
|
|
||||||
string getpropcontext(const char *name) {
|
|
||||||
return get_impl()->getcontext(name);
|
|
||||||
}
|
|
||||||
|
|
||||||
void getprops(void (*callback)(const char *, const char *, void *), void *cookie, bool persist) {
|
|
||||||
get_impl()->getprops(callback, cookie, persist);
|
|
||||||
}
|
|
||||||
|
|
||||||
int setprop(const char *name, const char *value, bool prop_svc) {
|
|
||||||
return get_impl()->setprop(name, value, prop_svc);
|
|
||||||
}
|
|
||||||
|
|
||||||
int delprop(const char *name, bool persist) {
|
|
||||||
return get_impl()->delprop(name, persist);
|
|
||||||
}
|
|
||||||
|
|
||||||
void load_prop_file(const char *filename, bool prop_svc) {
|
|
||||||
auto impl = get_impl();
|
|
||||||
LOGD("resetprop: Parse prop file [%s]\n", filename);
|
|
||||||
parse_prop_file(filename, [=](auto key, auto val) -> bool {
|
|
||||||
impl->setprop(key.data(), val.data(), prop_svc);
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int resetprop_main(int argc, char *argv[]) {
|
int resetprop_main(int argc, char *argv[]) {
|
||||||
bool prop_svc = true;
|
PropFlags flags;
|
||||||
bool persist = false;
|
|
||||||
bool verbose = false;
|
|
||||||
bool context = false;
|
|
||||||
char *argv0 = argv[0];
|
char *argv0 = argv[0];
|
||||||
|
|
||||||
--argc;
|
--argc;
|
||||||
@ -282,25 +258,24 @@ int resetprop_main(int argc, char *argv[]) {
|
|||||||
for (int idx = 1; true; ++idx) {
|
for (int idx = 1; true; ++idx) {
|
||||||
switch (argv[0][idx]) {
|
switch (argv[0][idx]) {
|
||||||
case '-':
|
case '-':
|
||||||
if (strcmp(argv[0], "--file") == 0 && argc == 2) {
|
if (argv[0] == "--file"sv) {
|
||||||
load_prop_file(argv[1], prop_svc);
|
flags.setLoadFile();
|
||||||
return 0;
|
} else if (argv[0] == "--delete"sv) {
|
||||||
} else if (strcmp(argv[0], "--delete") == 0 && argc == 2) {
|
flags.setDelete();
|
||||||
return delprop(argv[1], persist);
|
} else if (argv[0] == "--help"sv) {
|
||||||
} else if (strcmp(argv[0], "--help") == 0) {
|
|
||||||
usage(argv0);
|
usage(argv0);
|
||||||
}
|
}
|
||||||
case 'v':
|
case 'v':
|
||||||
verbose = true;
|
set_log_level_state(LogLevel::Debug, true);
|
||||||
continue;
|
continue;
|
||||||
case 'p':
|
case 'p':
|
||||||
persist = true;
|
flags.setPersist();
|
||||||
continue;
|
continue;
|
||||||
case 'n':
|
case 'n':
|
||||||
prop_svc = false;
|
flags.setSkipSvc();
|
||||||
continue;
|
continue;
|
||||||
case 'Z':
|
case 'Z':
|
||||||
context = true;
|
flags.setContext();
|
||||||
continue;
|
continue;
|
||||||
case '\0':
|
case '\0':
|
||||||
break;
|
break;
|
||||||
@ -314,25 +289,63 @@ int resetprop_main(int argc, char *argv[]) {
|
|||||||
++argv;
|
++argv;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_log_level_state(LogLevel::Debug, verbose);
|
resetprop *impl = get_impl();
|
||||||
|
|
||||||
|
if (flags.isDelete()) {
|
||||||
|
if (argc != 1)
|
||||||
|
usage(argv0);
|
||||||
|
return impl->delete_prop(argv[0], flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flags.isLoadFile()) {
|
||||||
|
if (argc != 1)
|
||||||
|
usage(argv0);
|
||||||
|
impl->load_file(argv[0], flags);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
switch (argc) {
|
switch (argc) {
|
||||||
case 0:
|
case 0:
|
||||||
if (context)
|
impl->print_props(flags);
|
||||||
print_props_context();
|
|
||||||
else
|
|
||||||
print_props(persist);
|
|
||||||
return 0;
|
return 0;
|
||||||
case 1: {
|
case 1: {
|
||||||
string prop = context ? getpropcontext(argv[0]) : getprop(argv[0], persist);
|
string val = impl->get_prop(argv[0], flags);
|
||||||
if (prop.empty())
|
if (val.empty())
|
||||||
return 1;
|
return 1;
|
||||||
printf("%s\n", prop.data());
|
printf("%s\n", val.data());
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
case 2:
|
case 2:
|
||||||
return setprop(argv[0], argv[1], prop_svc);
|
return impl->set_prop(argv[0], argv[1], flags);
|
||||||
default:
|
default:
|
||||||
usage(argv0);
|
usage(argv0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/***********************************
|
||||||
|
* Implementation of top-level APIs
|
||||||
|
***********************************/
|
||||||
|
|
||||||
|
string get_prop(const char *name, bool persist) {
|
||||||
|
PropFlags flags;
|
||||||
|
if (persist) flags.setPersist();
|
||||||
|
return get_impl()->get_prop(name, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
int delete_prop(const char *name, bool persist) {
|
||||||
|
PropFlags flags;
|
||||||
|
if (persist) flags.setPersist();
|
||||||
|
return get_impl()->delete_prop(name, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
int set_prop(const char *name, const char *value, bool skip_svc) {
|
||||||
|
PropFlags flags;
|
||||||
|
if (skip_svc) flags.setSkipSvc();
|
||||||
|
return get_impl()->set_prop(name, value, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_prop_file(const char *filename, bool skip_svc) {
|
||||||
|
PropFlags flags;
|
||||||
|
if (skip_svc) flags.setSkipSvc();
|
||||||
|
get_impl()->load_file(filename, flags);
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user