Update resetprop to partially use system impl

This commit is contained in:
topjohnwu 2020-05-07 23:54:00 -07:00
parent aa8b23105f
commit 232ae2a189
5 changed files with 172 additions and 130 deletions

View File

@ -3,7 +3,6 @@
#include <string> #include <string>
#include <functional> #include <functional>
bool prop_exist(const char *name);
int setprop(const char *name, const char *value, bool trigger = true); int setprop(const char *name, const char *value, bool trigger = true);
std::string getprop(const char *name, bool persist = false); std::string getprop(const char *name, bool persist = false);
void getprops(void (*callback)(const char *, const char *, void *), void getprops(void (*callback)(const char *, const char *, void *),

View File

@ -88,6 +88,8 @@ const pb_field_t PersistentProperties_fields[2] = {
* End of auto generated code * End of auto generated code
* ***************************/ * ***************************/
bool use_pb = false;
#ifdef APPLET_STUB_MAIN #ifdef APPLET_STUB_MAIN
struct { struct {
void push_back(...){}; void push_back(...){};

View File

@ -1,10 +1,4 @@
#include <stdio.h> #include <dlfcn.h>
#include <stdint.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <dirent.h>
#include <sys/types.h> #include <sys/types.h>
#include <vector> #include <vector>
#include <map> #include <map>
@ -21,9 +15,36 @@
using namespace std; using namespace std;
bool use_pb = false;
static bool verbose = false; static bool verbose = false;
static int (*system_property_get)(const char*, char*);
static int (*system_property_set)(const char*, const char*);
[[noreturn]] static void usage(char* arg0) {
fprintf(stderr,
NAME_WITH_VER(resetprop) R"EOF(" - System Props Modification Tool
Usage: %s [flags] [options...]
Options:
-h, --help show this message
(no arguments) print all properties
NAME get property
NAME VALUE set property entry NAME with VALUE
--file FILE load props from FILE
--delete NAME delete property
Flags:
-v print verbose output to stderr
-n set properties without going through init
affects setprop and prop file loading
-p also access props directly from persist storage
affects getprop and delprop
)EOF", arg0);
exit(1);
}
static bool check_legal_property_name(const char *name) { static bool check_legal_property_name(const char *name) {
int namelen = strlen(name); int namelen = strlen(name);
@ -53,70 +74,76 @@ illegal:
return false; return false;
} }
[[noreturn]] static void usage(char* arg0) { static void read_prop(const prop_info *pi, void *cb) {
fprintf(stderr,
NAME_WITH_VER(resetprop) R"EOF(" - System Props Modification Tool
Usage: %s [flags] [options...]
Options:
-h, --help show this message
(no arguments) print all properties
NAME get property
NAME VALUE set property entry NAME with VALUE
--file FILE load props from FILE
--delete NAME delete property
Flags:
-v print verbose output to stderr
-n set properties without going through init
affects setprop and prop file loading
-p also access props directly from persist storage
affects getprop and delprop
)EOF", arg0);
exit(1);
}
static void read_props(const prop_info *pi, void *cb) {
__system_property_read_callback( __system_property_read_callback(
pi, [](auto cb, auto name, auto value, auto) { pi, [](auto cb, auto name, auto value, auto) {
reinterpret_cast<prop_cb*>(cb)->exec(name, value); reinterpret_cast<prop_cb*>(cb)->exec(name, value);
}, cb); }, cb);
} }
static int init_resetprop() {
use_pb = use_pb ? true : access(PERSISTENT_PROPERTY_DIR "/persistent_properties", R_OK) == 0;
if (__system_properties_init()) {
LOGE("resetprop: Initialize error\n");
return -1;
}
return 0;
}
static void print_props(bool persist) { static void print_props(bool persist) {
getprops([](auto name, auto value, auto) { getprops([](auto name, auto value, auto) {
printf("[%s]: [%s]\n", name, value); printf("[%s]: [%s]\n", name, value);
}, nullptr, persist); }, nullptr, persist);
} }
/* ************************************************** struct sysprop {
* Implementations of functions in resetprop.h (APIs) virtual int setprop(const char *name, const char *value, bool trigger) { return 1; }
* **************************************************/ virtual string getprop(const char *name, bool persist) { return ""; }
virtual void getprops(void (*callback)(const char *, const char *, void *),
void *cookie, bool persist) {}
virtual int delprop(const char *name, bool persist) { return 1; }
};
#define ENSURE_INIT(ret) if (init_resetprop()) return ret struct stock_sysprop : public sysprop {
int setprop(const char *name, const char *value, bool) override {
return system_property_set(name, value);
}
string getprop(const char *name, bool) override {
char buf[PROP_VALUE_MAX];
buf[0] = '\0';
system_property_get(name, buf);
return buf;
}
};
bool prop_exist(const char *name) { struct resetprop : public sysprop {
ENSURE_INIT(false); int setprop(const char *name, const char *value, bool trigger) override {
return __system_property_find(name) != nullptr; if (!check_legal_property_name(name))
return 1;
int ret;
auto pi = const_cast<prop_info *>(__system_property_find(name));
if (pi != nullptr) {
if (trigger) {
if (strncmp(name, "ro.", 3) == 0)
delprop(name, false);
ret = system_property_set(name, value);
} else {
ret = __system_property_update(pi, value, strlen(value));
}
LOGD("resetprop: update prop [%s]: [%s] by %s\n", name, value,
trigger ? "property_service" : "modifying prop data structure");
} else {
if (trigger) {
ret = system_property_set(name, value);
} else {
ret = __system_property_add(name, strlen(name), value, strlen(value));
}
LOGD("resetprop: create prop [%s]: [%s] by %s\n", name, value,
trigger ? "property_service" : "modifying prop data structure");
} }
// Get prop by name, return string if (ret)
string getprop(const char *name, bool persist) { LOGE("resetprop: setprop error\n");
return ret;
}
string getprop(const char *name, bool persist) override {
if (!check_legal_property_name(name)) if (!check_legal_property_name(name))
return string(); return string();
ENSURE_INIT(string()); auto pi = __system_property_find(name);
const prop_info *pi = __system_property_find(name);
if (pi == nullptr) { if (pi == nullptr) {
if (persist && strncmp(name, "persist.", 8) == 0) { if (persist && strncmp(name, "persist.", 8) == 0) {
auto value = persist_getprop(name); auto value = persist_getprop(name);
@ -128,76 +155,90 @@ not_found:
LOGD("resetprop: prop [%s] does not exist\n", name); LOGD("resetprop: prop [%s] does not exist\n", name);
return string(); return string();
} else { } else {
char buf[PROP_VALUE_MAX]; string val;
auto reader = make_prop_cb(buf, [](auto, auto value, auto buf) { strcpy(buf, value); }); auto reader = make_prop_cb(val, [](auto, auto value, auto str){ str = value; });
read_props(pi, &reader); read_prop(pi, &reader);
LOGD("resetprop: getprop [%s]: [%s]\n", name, buf); LOGD("resetprop: getprop [%s]: [%s]\n", name, val.data());
return buf; return val;
} }
} }
void getprops(void (*callback)(const char *, const char *, void *), void *cookie, bool persist) { void getprops(void (*callback)(const char *, const char *, void *),
ENSURE_INIT(); void *cookie, bool persist) override {
prop_list list; prop_list list;
prop_collector collector(list); prop_collector collector(list);
__system_property_foreach(read_props, &collector); __system_property_foreach(read_prop, &collector);
if (persist) if (persist)
persist_getprops(&collector); persist_getprops(&collector);
for (auto &[key, val] : list) for (auto &[key, val] : list)
callback(key.data(), val.data(), cookie); callback(key.data(), val.data(), cookie);
if (persist)
persist_cleanup(); persist_cleanup();
} }
int setprop(const char *name, const char *value, bool trigger) { int delprop(const char *name, bool persist) override {
if (!check_legal_property_name(name)) if (!check_legal_property_name(name))
return 1; return 1;
ENSURE_INIT(-1); LOGD("resetprop: delete prop [%s]\n", name);
if (persist && strncmp(name, "persist.", 8) == 0)
int ret; persist = persist_deleteprop(name);
return __system_property_delete(name) && !(persist && strncmp(name, "persist.", 8) == 0);
auto pi = (prop_info*) __system_property_find(name);
if (pi != nullptr) {
if (trigger) {
if (strncmp(name, "ro.", 3) == 0) delprop(name);
ret = __system_property_set(name, value);
} else {
ret = __system_property_update(pi, value, strlen(value));
} }
} else { };
LOGD("resetprop: New prop [%s]\n", name);
if (trigger) { #define DLOAD(name) \
ret = __system_property_set(name, value); *(void **) &name = dlsym(RTLD_DEFAULT, "__" #name)
} else {
ret = __system_property_add(name, strlen(name), value, strlen(value)); static sysprop *get_impl() {
static sysprop *impl = nullptr;
if (impl == nullptr) {
use_pb = access(PERSISTENT_PROPERTY_DIR "/persistent_properties", R_OK) == 0;
#ifdef APPLET_STUB_MAIN
if (__system_properties_init()) {
LOGE("resetprop: __system_properties_init error\n");
exit(1);
} }
impl = new resetprop();
system_property_set = &__system_property_set;
#else
DLOAD(system_property_set); /* Always prefer to use setprop from platform */
if (__system_properties_init()) {
LOGW("resetprop: __system_properties_init error\n");
DLOAD(system_property_get);
impl = system_property_get ? new stock_sysprop() : new sysprop();
} else {
impl = new resetprop();
}
#endif
}
return impl;
} }
LOGD("resetprop: setprop [%s]: [%s] by %s\n", name, value, /*********************************
trigger ? "property_service" : "modifing prop data structure"); * Implementation of public APIs
*********************************/
if (ret) string getprop(const char *name, bool persist) {
LOGE("resetprop: setprop error\n"); return get_impl()->getprop(name, persist);
}
return ret; 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 trigger) {
return get_impl()->setprop(name, value, trigger);
} }
int delprop(const char *name, bool persist) { int delprop(const char *name, bool persist) {
if (!check_legal_property_name(name)) return get_impl()->delprop(name, persist);
return 1;
ENSURE_INIT(-1);
char path[PATH_MAX];
path[0] = '\0';
LOGD("resetprop: delprop [%s]\n", name);
if (persist && strncmp(name, "persist.", 8) == 0)
persist = persist_deleteprop(name);
return __system_property_del(name) && !(persist && strncmp(name, "persist.", 8) == 0);
} }
void load_prop_file(const char *filename, bool trigger) { void load_prop_file(const char *filename, bool trigger) {
ENSURE_INIT(); auto impl = get_impl();
LOGD("resetprop: Parse prop file [%s]\n", filename); LOGD("resetprop: Parse prop file [%s]\n", filename);
parse_prop_file(filename, [=](auto key, auto val) -> bool { parse_prop_file(filename, [=](auto key, auto val) -> bool {
setprop(key.data(), val.data(), trigger); impl->setprop(key.data(), val.data(), trigger);
return true; return true;
}); });
} }
@ -253,7 +294,7 @@ int resetprop_main(int argc, char *argv[]) {
case 1: case 1:
prop = getprop(argv[0], persist); prop = getprop(argv[0], persist);
if (prop.empty()) return 1; if (prop.empty()) return 1;
printf("%s\n", prop.c_str()); printf("%s\n", prop.data());
return 0; return 0;
case 2: case 2:
return setprop(argv[0], argv[1], trigger); return setprop(argv[0], argv[1], trigger);

View File

@ -109,7 +109,7 @@ int __system_property_add(const char* __name, unsigned int __name_length, const
** **
** Returns 0 on success, -1 if the property area is full. ** Returns 0 on success, -1 if the property area is full.
*/ */
int __system_property_del(const char *__name); int __system_property_delete(const char *__name);
/* Update the value of a system property returned by /* Update the value of a system property returned by
** __system_property_find. Can only be done by a single process ** __system_property_find. Can only be done by a single process

View File

@ -100,7 +100,7 @@ int __system_property_add(const char* name, unsigned int namelen, const char* va
} }
__BIONIC_WEAK_FOR_NATIVE_BRIDGE __BIONIC_WEAK_FOR_NATIVE_BRIDGE
int __system_property_del(const char* name) { int __system_property_delete(const char* name) {
return system_properties.Delete(name); return system_properties.Delete(name);
} }