Deadlock fixes (#1758)
Hi, This PR fixes deadlocks that I've noticed within my kernel, with some minor optimizations around it. Thanks. --------- Signed-off-by: Juhyung Park <qkrwngud825@gmail.com>
This commit is contained in:
@@ -5,7 +5,6 @@
|
|||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/types.h>
|
#include <linux/types.h>
|
||||||
#include <linux/version.h>
|
#include <linux/version.h>
|
||||||
#include <linux/workqueue.h>
|
|
||||||
|
|
||||||
#include "allowlist.h"
|
#include "allowlist.h"
|
||||||
#include "klog.h" // IWYU pragma: keep
|
#include "klog.h" // IWYU pragma: keep
|
||||||
@@ -16,8 +15,7 @@
|
|||||||
|
|
||||||
uid_t ksu_manager_uid = KSU_INVALID_UID;
|
uid_t ksu_manager_uid = KSU_INVALID_UID;
|
||||||
|
|
||||||
#define SYSTEM_PACKAGES_LIST_PATH "/data/system/packages.list"
|
#define SYSTEM_PACKAGES_LIST_PATH "/data/system/packages.list.tmp"
|
||||||
static struct work_struct ksu_update_uid_work;
|
|
||||||
|
|
||||||
struct uid_data {
|
struct uid_data {
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
@@ -94,8 +92,25 @@ static void crown_manager(const char *apk, struct list_head *uid_data)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define DATA_PATH_LEN 384 // 384 is enough for /data/app/<package>/base.apk
|
||||||
|
|
||||||
|
struct data_path {
|
||||||
|
char dirpath[DATA_PATH_LEN];
|
||||||
|
int depth;
|
||||||
|
struct list_head list;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct apk_path_hash {
|
||||||
|
unsigned int hash;
|
||||||
|
bool exists;
|
||||||
|
struct list_head list;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct list_head apk_path_hash_list = LIST_HEAD_INIT(apk_path_hash_list);
|
||||||
|
|
||||||
struct my_dir_context {
|
struct my_dir_context {
|
||||||
struct dir_context ctx;
|
struct dir_context ctx;
|
||||||
|
struct list_head *data_path_list;
|
||||||
char *parent_dir;
|
char *parent_dir;
|
||||||
void *private_data;
|
void *private_data;
|
||||||
int depth;
|
int depth;
|
||||||
@@ -119,8 +134,7 @@ FILLDIR_RETURN_TYPE my_actor(struct dir_context *ctx, const char *name,
|
|||||||
{
|
{
|
||||||
struct my_dir_context *my_ctx =
|
struct my_dir_context *my_ctx =
|
||||||
container_of(ctx, struct my_dir_context, ctx);
|
container_of(ctx, struct my_dir_context, ctx);
|
||||||
struct file *file;
|
char dirpath[DATA_PATH_LEN];
|
||||||
char dirpath[384]; // 384 is enough for /data/app/<package>/base.apk
|
|
||||||
|
|
||||||
if (!my_ctx) {
|
if (!my_ctx) {
|
||||||
pr_err("Invalid context\n");
|
pr_err("Invalid context\n");
|
||||||
@@ -134,8 +148,8 @@ FILLDIR_RETURN_TYPE my_actor(struct dir_context *ctx, const char *name,
|
|||||||
if (!strncmp(name, "..", namelen) || !strncmp(name, ".", namelen))
|
if (!strncmp(name, "..", namelen) || !strncmp(name, ".", namelen))
|
||||||
return FILLDIR_ACTOR_CONTINUE; // Skip "." and ".."
|
return FILLDIR_ACTOR_CONTINUE; // Skip "." and ".."
|
||||||
|
|
||||||
if (snprintf(dirpath, sizeof(dirpath), "%s/%.*s", my_ctx->parent_dir,
|
if (snprintf(dirpath, DATA_PATH_LEN, "%s/%.*s", my_ctx->parent_dir,
|
||||||
namelen, name) >= sizeof(dirpath)) {
|
namelen, name) >= DATA_PATH_LEN) {
|
||||||
pr_err("Path too long: %s/%.*s\n", my_ctx->parent_dir, namelen,
|
pr_err("Path too long: %s/%.*s\n", my_ctx->parent_dir, namelen,
|
||||||
name);
|
name);
|
||||||
return FILLDIR_ACTOR_CONTINUE;
|
return FILLDIR_ACTOR_CONTINUE;
|
||||||
@@ -143,29 +157,45 @@ FILLDIR_RETURN_TYPE my_actor(struct dir_context *ctx, const char *name,
|
|||||||
|
|
||||||
if (d_type == DT_DIR && my_ctx->depth > 0 &&
|
if (d_type == DT_DIR && my_ctx->depth > 0 &&
|
||||||
(my_ctx->stop && !*my_ctx->stop)) {
|
(my_ctx->stop && !*my_ctx->stop)) {
|
||||||
struct my_dir_context sub_ctx = { .ctx.actor = my_actor,
|
struct data_path *data = kmalloc(sizeof(struct data_path), GFP_ATOMIC);
|
||||||
.parent_dir = dirpath,
|
|
||||||
.private_data =
|
if (!data) {
|
||||||
my_ctx->private_data,
|
pr_err("Failed to allocate memory for %s\n", dirpath);
|
||||||
.depth = my_ctx->depth - 1,
|
|
||||||
.stop = my_ctx->stop };
|
|
||||||
file = ksu_filp_open_compat(dirpath, O_RDONLY | O_NOFOLLOW, 0);
|
|
||||||
if (IS_ERR(file)) {
|
|
||||||
pr_err("Failed to open directory: %s, err: %ld\n",
|
|
||||||
dirpath, PTR_ERR(file));
|
|
||||||
return FILLDIR_ACTOR_CONTINUE;
|
return FILLDIR_ACTOR_CONTINUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
iterate_dir(file, &sub_ctx.ctx);
|
strscpy(data->dirpath, dirpath, DATA_PATH_LEN);
|
||||||
filp_close(file, NULL);
|
data->depth = my_ctx->depth - 1;
|
||||||
|
list_add_tail(&data->list, my_ctx->data_path_list);
|
||||||
} else {
|
} else {
|
||||||
if ((namelen == 8) && (strncmp(name, "base.apk", namelen) == 0)) {
|
if ((namelen == 8) && (strncmp(name, "base.apk", namelen) == 0)) {
|
||||||
|
struct apk_path_hash *pos, *n;
|
||||||
|
unsigned int hash = full_name_hash(NULL, dirpath, strlen(dirpath));
|
||||||
|
|
||||||
|
list_for_each_entry(pos, &apk_path_hash_list, list) {
|
||||||
|
if (hash == pos->hash) {
|
||||||
|
pos->exists = true;
|
||||||
|
return FILLDIR_ACTOR_CONTINUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool is_manager = is_manager_apk(dirpath);
|
bool is_manager = is_manager_apk(dirpath);
|
||||||
pr_info("Found base.apk at path: %s, is_manager: %d\n",
|
pr_info("Found new base.apk at path: %s, is_manager: %d\n",
|
||||||
dirpath, is_manager);
|
dirpath, is_manager);
|
||||||
if (is_manager) {
|
if (is_manager) {
|
||||||
crown_manager(dirpath, my_ctx->private_data);
|
crown_manager(dirpath, my_ctx->private_data);
|
||||||
*my_ctx->stop = 1;
|
*my_ctx->stop = 1;
|
||||||
|
|
||||||
|
// Manager found, clear APK cache list
|
||||||
|
list_for_each_entry_safe(pos, n, &apk_path_hash_list, list) {
|
||||||
|
list_del(&pos->list);
|
||||||
|
kfree(pos);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
struct apk_path_hash *apk_data = kmalloc(sizeof(struct apk_path_hash), GFP_ATOMIC);
|
||||||
|
apk_data->hash = hash;
|
||||||
|
apk_data->exists = true;
|
||||||
|
list_add_tail(&apk_data->list, &apk_path_hash_list);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -175,17 +205,38 @@ FILLDIR_RETURN_TYPE my_actor(struct dir_context *ctx, const char *name,
|
|||||||
|
|
||||||
void search_manager(const char *path, int depth, struct list_head *uid_data)
|
void search_manager(const char *path, int depth, struct list_head *uid_data)
|
||||||
{
|
{
|
||||||
struct file *file;
|
int i, stop = 0;
|
||||||
int stop = 0;
|
struct list_head data_path_list;
|
||||||
struct my_dir_context ctx = { .ctx.actor = my_actor,
|
INIT_LIST_HEAD(&data_path_list);
|
||||||
.parent_dir = (char *)path,
|
|
||||||
.private_data = uid_data,
|
|
||||||
.depth = depth,
|
|
||||||
.stop = &stop };
|
|
||||||
|
|
||||||
file = ksu_filp_open_compat(path, O_RDONLY | O_NOFOLLOW, 0);
|
// Initialize APK cache list
|
||||||
|
struct apk_path_hash *pos, *n;
|
||||||
|
list_for_each_entry(pos, &apk_path_hash_list, list) {
|
||||||
|
pos->exists = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// First depth
|
||||||
|
struct data_path data;
|
||||||
|
strscpy(data.dirpath, path, DATA_PATH_LEN);
|
||||||
|
data.depth = depth;
|
||||||
|
list_add_tail(&data.list, &data_path_list);
|
||||||
|
|
||||||
|
for (i = depth; i > 0; i--) {
|
||||||
|
struct data_path *pos, *n;
|
||||||
|
|
||||||
|
list_for_each_entry_safe(pos, n, &data_path_list, list) {
|
||||||
|
struct my_dir_context ctx = { .ctx.actor = my_actor,
|
||||||
|
.data_path_list = &data_path_list,
|
||||||
|
.parent_dir = pos->dirpath,
|
||||||
|
.private_data = uid_data,
|
||||||
|
.depth = pos->depth,
|
||||||
|
.stop = &stop };
|
||||||
|
struct file *file;
|
||||||
|
|
||||||
|
if (!stop) {
|
||||||
|
file = ksu_filp_open_compat(pos->dirpath, O_RDONLY | O_NOFOLLOW, 0);
|
||||||
if (IS_ERR(file)) {
|
if (IS_ERR(file)) {
|
||||||
pr_err("Failed to open directory: %s\n", path);
|
pr_err("Failed to open directory: %s, err: %ld\n", pos->dirpath, PTR_ERR(file));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -193,6 +244,21 @@ void search_manager(const char *path, int depth, struct list_head *uid_data)
|
|||||||
filp_close(file, NULL);
|
filp_close(file, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
list_del(&pos->list);
|
||||||
|
if (pos != &data)
|
||||||
|
kfree(pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove stale cached APK entries
|
||||||
|
list_for_each_entry_safe(pos, n, &apk_path_hash_list, list) {
|
||||||
|
if (!pos->exists) {
|
||||||
|
list_del(&pos->list);
|
||||||
|
kfree(pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static bool is_uid_exist(uid_t uid, char *package, void *data)
|
static bool is_uid_exist(uid_t uid, char *package, void *data)
|
||||||
{
|
{
|
||||||
struct list_head *list = (struct list_head *)data;
|
struct list_head *list = (struct list_head *)data;
|
||||||
@@ -209,14 +275,13 @@ static bool is_uid_exist(uid_t uid, char *package, void *data)
|
|||||||
return exist;
|
return exist;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void do_update_uid(struct work_struct *work)
|
void track_throne()
|
||||||
{
|
{
|
||||||
struct file *fp =
|
struct file *fp =
|
||||||
ksu_filp_open_compat(SYSTEM_PACKAGES_LIST_PATH, O_RDONLY, 0);
|
ksu_filp_open_compat(SYSTEM_PACKAGES_LIST_PATH, O_RDONLY, 0);
|
||||||
if (IS_ERR(fp)) {
|
if (IS_ERR(fp)) {
|
||||||
pr_err("do_update_uid, open " SYSTEM_PACKAGES_LIST_PATH
|
pr_err("%s: open " SYSTEM_PACKAGES_LIST_PATH " failed: %ld\n",
|
||||||
" failed: %ld\n",
|
__func__, PTR_ERR(fp));
|
||||||
PTR_ERR(fp));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -303,14 +368,9 @@ out:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void track_throne()
|
|
||||||
{
|
|
||||||
ksu_queue_work(&ksu_update_uid_work);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ksu_throne_tracker_init()
|
void ksu_throne_tracker_init()
|
||||||
{
|
{
|
||||||
INIT_WORK(&ksu_update_uid_work, do_update_uid);
|
// nothing to do
|
||||||
}
|
}
|
||||||
|
|
||||||
void ksu_throne_tracker_exit()
|
void ksu_throne_tracker_exit()
|
||||||
|
|||||||
Reference in New Issue
Block a user