mirror of
https://github.com/tiann/KernelSU.git
synced 2025-02-20 11:43:32 +08:00
145 lines
3.1 KiB
C
145 lines
3.1 KiB
C
#include "linux/err.h"
|
|
#include "linux/fs.h"
|
|
#include "linux/list.h"
|
|
#include "linux/slab.h"
|
|
#include "linux/string.h"
|
|
#include "linux/types.h"
|
|
#include "linux/version.h"
|
|
#include "linux/workqueue.h"
|
|
|
|
#include "allowlist.h"
|
|
#include "klog.h" // IWYU pragma: keep
|
|
#include "ksu.h"
|
|
#include "manager.h"
|
|
#include "uid_observer.h"
|
|
#include "kernel_compat.h"
|
|
|
|
#define SYSTEM_PACKAGES_LIST_PATH "/data/system/packages.list"
|
|
static struct work_struct ksu_update_uid_work;
|
|
|
|
struct uid_data {
|
|
struct list_head list;
|
|
u32 uid;
|
|
char package[KSU_MAX_PACKAGE_NAME];
|
|
};
|
|
|
|
static bool is_uid_exist(uid_t uid, char *package, void *data)
|
|
{
|
|
struct list_head *list = (struct list_head *)data;
|
|
struct uid_data *np;
|
|
|
|
bool exist = false;
|
|
list_for_each_entry (np, list, list) {
|
|
if (np->uid == uid % 100000 &&
|
|
strncmp(np->package, package, KSU_MAX_PACKAGE_NAME) == 0) {
|
|
exist = true;
|
|
break;
|
|
}
|
|
}
|
|
return exist;
|
|
}
|
|
|
|
static void do_update_uid(struct work_struct *work)
|
|
{
|
|
struct file *fp =
|
|
ksu_filp_open_compat(SYSTEM_PACKAGES_LIST_PATH, O_RDONLY, 0);
|
|
if (IS_ERR(fp)) {
|
|
pr_err("do_update_uid, open " SYSTEM_PACKAGES_LIST_PATH
|
|
" failed: %ld\n",
|
|
PTR_ERR(fp));
|
|
return;
|
|
}
|
|
|
|
struct list_head uid_list;
|
|
INIT_LIST_HEAD(&uid_list);
|
|
|
|
char chr = 0;
|
|
loff_t pos = 0;
|
|
loff_t line_start = 0;
|
|
char buf[KSU_MAX_PACKAGE_NAME];
|
|
for (;;) {
|
|
ssize_t count =
|
|
ksu_kernel_read_compat(fp, &chr, sizeof(chr), &pos);
|
|
if (count != sizeof(chr))
|
|
break;
|
|
if (chr != '\n')
|
|
continue;
|
|
|
|
count = ksu_kernel_read_compat(fp, buf, sizeof(buf),
|
|
&line_start);
|
|
|
|
struct uid_data *data =
|
|
kzalloc(sizeof(struct uid_data), GFP_ATOMIC);
|
|
if (!data) {
|
|
goto out;
|
|
}
|
|
|
|
char *tmp = buf;
|
|
const char *delim = " ";
|
|
char *package = strsep(&tmp, delim);
|
|
char *uid = strsep(&tmp, delim);
|
|
if (!uid || !package) {
|
|
pr_err("update_uid: package or uid is NULL!\n");
|
|
break;
|
|
}
|
|
|
|
u32 res;
|
|
if (kstrtou32(uid, 10, &res)) {
|
|
pr_err("update_uid: uid parse err\n");
|
|
break;
|
|
}
|
|
data->uid = res;
|
|
strncpy(data->package, package, KSU_MAX_PACKAGE_NAME);
|
|
list_add_tail(&data->list, &uid_list);
|
|
// reset line start
|
|
line_start = pos;
|
|
}
|
|
|
|
// now update uid list
|
|
struct uid_data *np;
|
|
struct uid_data *n;
|
|
|
|
// first, check if manager_uid exist!
|
|
bool manager_exist = false;
|
|
list_for_each_entry (np, &uid_list, list) {
|
|
// if manager is installed in work profile, the uid in packages.list is still equals main profile
|
|
// don't delete it in this case!
|
|
int manager_uid = ksu_get_manager_uid() % 100000;
|
|
if (np->uid == manager_uid) {
|
|
manager_exist = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!manager_exist && ksu_is_manager_uid_valid()) {
|
|
pr_info("manager is uninstalled, invalidate it!\n");
|
|
ksu_invalidate_manager_uid();
|
|
}
|
|
|
|
// then prune the allowlist
|
|
ksu_prune_allowlist(is_uid_exist, &uid_list);
|
|
out:
|
|
// free uid_list
|
|
list_for_each_entry_safe (np, n, &uid_list, list) {
|
|
list_del(&np->list);
|
|
kfree(np);
|
|
}
|
|
filp_close(fp, 0);
|
|
}
|
|
|
|
void update_uid()
|
|
{
|
|
ksu_queue_work(&ksu_update_uid_work);
|
|
}
|
|
|
|
int ksu_uid_observer_init()
|
|
{
|
|
INIT_WORK(&ksu_update_uid_work, do_update_uid);
|
|
return 0;
|
|
}
|
|
|
|
int ksu_uid_observer_exit()
|
|
{
|
|
return 0;
|
|
}
|