From 0e264515c9e43640ed3c43b8166a8c33dd0eaf20 Mon Sep 17 00:00:00 2001 From: Roberto Scolaro Date: Thu, 18 Jan 2024 16:34:27 +0000 Subject: [PATCH] fix(driver): fix potential deadlock Signed-off-by: Roberto Scolaro --- driver/ppm_fillers.c | 16 ++++++++++++---- 1 file changed, 12 insertions(+), 4 deletions(-) diff --git a/driver/ppm_fillers.c b/driver/ppm_fillers.c index 231059c53a..a5acb10eca 100644 --- a/driver/ppm_fillers.c +++ b/driver/ppm_fillers.c @@ -7416,15 +7416,23 @@ int f_sched_prog_exec(struct event_filler_arguments *args) { /* Support exe_writable */ #if LINUX_VERSION_CODE >= KERNEL_VERSION(6, 3, 0) - exe_writable |= (file_permission(exe_file, MAY_WRITE) == 0); + exe_writable |= (file_permission(exe_file, MAY_WRITE | MAY_NOT_BLOCK) == 0); exe_writable |= inode_owner_or_capable(file_mnt_idmap(exe_file), file_inode(exe_file)); #elif LINUX_VERSION_CODE >= KERNEL_VERSION(5, 12, 0) - exe_writable |= (inode_permission(current_user_ns(), file_inode(exe_file), MAY_WRITE) == 0); + exe_writable |= (inode_permission(current_user_ns(), file_inode(exe_file), MAY_WRITE | MAY_NOT_BLOCK) == 0); exe_writable |= inode_owner_or_capable(current_user_ns(), file_inode(exe_file)); -#else - exe_writable |= (inode_permission(file_inode(exe_file), MAY_WRITE) == 0); +#elif LINUX_VERSION_CODE >= KERNEL_VERSION(3, 1, 0) + exe_writable |= (inode_permission(file_inode(exe_file), MAY_WRITE | MAY_NOT_BLOCK) == 0); exe_writable |= inode_owner_or_capable(file_inode(exe_file)); #endif + /* + * Kernels < 3.1.0 doesn't support the exe_writable flags due to the MAY_NOT_BLOCK not being + * available. This limitation is related to the fact that this function (f_sched_prog_exec) + * is in a RCU critical section: this means that this function (and its callee) MUST NOT + * call functions that can yield the processor (e.g. inode_permission that deep down in its + * call stack calls a down_read()). This is addressed after the Kernel 3.1.0 where the + * MAY_OT_BLOCK flag is introduced and avoids the processor to being yield. + */ /* Support exe_upper_layer */ exe_upper_layer = ppm_is_upper_layer(exe_file);