]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
x86/vdso: Prepare for robust futex unlock support
authorThomas Gleixner <tglx@kernel.org>
Tue, 2 Jun 2026 09:10:08 +0000 (11:10 +0200)
committerPeter Zijlstra <peterz@infradead.org>
Wed, 3 Jun 2026 09:38:52 +0000 (11:38 +0200)
There will be a VDSO function to unlock non-contended robust futexes in
user space. The unlock sequence is racy vs. clearing the list_pending_op
pointer in the task's robust list head. To plug this race the kernel needs
to know the critical section window so it can clear the pointer when the
task is interrupted within that race window. The window is determined by
labels in the inline assembly.

Add these symbols to the vdso2c generator and use them in the VDSO VMA code
to update the critical section addresses in mm_struct::futex on (re)map().

The symbols are not exported to user space, but available in the debug
version of the vDSO.

Signed-off-by: Thomas Gleixner <tglx@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Reviewed-by: André Almeida <andrealmeid@igalia.com>
Link: https://patch.msgid.link/20260602090535.828312645@kernel.org
arch/x86/entry/vdso/vma.c
arch/x86/include/asm/vdso.h
arch/x86/tools/vdso2c.c

index a6bfcc8243cd91cc5b0547d74dd42c97835a90b8..9a953e7c76dbe021a6246f3851acf2de7a215b21 100644 (file)
@@ -6,6 +6,7 @@
  */
 #include <linux/mm.h>
 #include <linux/err.h>
+#include <linux/futex.h>
 #include <linux/sched.h>
 #include <linux/sched/task_stack.h>
 #include <linux/slab.h>
@@ -73,6 +74,31 @@ static void vdso_fix_landing(const struct vdso_image *image,
                regs->ip = new_vma->vm_start + ipoffset;
 }
 
+#ifdef CONFIG_FUTEX_ROBUST_UNLOCK
+static void vdso_futex_robust_unlock_update_ips(void)
+{
+       const struct vdso_image *image = current->mm->context.vdso_image;
+       unsigned long vdso = (unsigned long) current->mm->context.vdso;
+       struct futex_mm_data *fd = &current->mm->futex;
+       unsigned int idx = 0;
+
+       futex_reset_cs_ranges(fd);
+
+#ifdef CONFIG_X86_64
+       futex_set_vdso_cs_range(fd, idx, vdso + image->sym___futex_list64_try_unlock_cs_start,
+                               vdso + image->sym___futex_list64_try_unlock_cs_end, false);
+       idx++;
+#endif /* CONFIG_X86_64 */
+
+#if defined(CONFIG_X86_32) || defined(CONFIG_COMPAT)
+       futex_set_vdso_cs_range(fd, idx, vdso + image->sym___futex_list32_try_unlock_cs_start,
+                               vdso + image->sym___futex_list32_try_unlock_cs_end, true);
+#endif /* CONFIG_X86_32 || CONFIG_COMPAT */
+}
+#else
+static inline void vdso_futex_robust_unlock_update_ips(void) { }
+#endif
+
 static int vdso_mremap(const struct vm_special_mapping *sm,
                struct vm_area_struct *new_vma)
 {
@@ -80,6 +106,7 @@ static int vdso_mremap(const struct vm_special_mapping *sm,
 
        vdso_fix_landing(image, new_vma);
        current->mm->context.vdso = (void __user *)new_vma->vm_start;
+       vdso_futex_robust_unlock_update_ips();
 
        return 0;
 }
@@ -185,6 +212,8 @@ static int map_vdso(const struct vdso_image *image, unsigned long addr)
        current->mm->context.vdso = (void __user *)text_start;
        current->mm->context.vdso_image = image;
 
+       vdso_futex_robust_unlock_update_ips();
+
 up_fail:
        mmap_write_unlock(mm);
        return ret;
index f2d49212ae90272a57974d2648c0f25a0ab2b1e5..4e735158c1b28315baf6a3b189acd13d20ed805f 100644 (file)
@@ -24,6 +24,10 @@ struct vdso_image {
        long sym_int80_landing_pad;
        long sym_vdso32_sigreturn_landing_pad;
        long sym_vdso32_rt_sigreturn_landing_pad;
+       long sym___futex_list64_try_unlock_cs_start;
+       long sym___futex_list64_try_unlock_cs_end;
+       long sym___futex_list32_try_unlock_cs_start;
+       long sym___futex_list32_try_unlock_cs_end;
 };
 
 extern const struct vdso_image vdso64_image;
index b8a555763f4376de6ad4611e421b81d1d7b40e81..64a636b9fc3795e4ab64d0635242c6ad7365d294 100644 (file)
@@ -75,12 +75,16 @@ struct vdso_sym {
 };
 
 struct vdso_sym required_syms[] = {
-       {"__kernel_vsyscall", true},
-       {"__kernel_sigreturn", true},
-       {"__kernel_rt_sigreturn", true},
-       {"int80_landing_pad", true},
-       {"vdso32_rt_sigreturn_landing_pad", true},
-       {"vdso32_sigreturn_landing_pad", true},
+       {"__kernel_vsyscall",                           true},
+       {"__kernel_sigreturn",                          true},
+       {"__kernel_rt_sigreturn",                       true},
+       {"int80_landing_pad",                           true},
+       {"vdso32_rt_sigreturn_landing_pad",             true},
+       {"vdso32_sigreturn_landing_pad",                true},
+       {"__futex_list64_try_unlock_cs_start",          true},
+       {"__futex_list64_try_unlock_cs_end",            true},
+       {"__futex_list32_try_unlock_cs_start",          true},
+       {"__futex_list32_try_unlock_cs_end",            true},
 };
 
 __attribute__((format(printf, 1, 2))) __attribute__((noreturn))