task_work: Introduce task_work_cancel() again
authorFrederic Weisbecker <frederic@kernel.org>
Fri, 21 Jun 2024 09:15:59 +0000 (11:15 +0200)
committerPeter Zijlstra <peterz@infradead.org>
Tue, 9 Jul 2024 11:26:32 +0000 (13:26 +0200)
Re-introduce task_work_cancel(), this time to cancel an actual callback
and not *any* callback pointing to a given function. This is going to be
needed for perf events event freeing.

Signed-off-by: Frederic Weisbecker <frederic@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Cc: stable@vger.kernel.org
Link: https://lore.kernel.org/r/20240621091601.18227-3-frederic@kernel.org
include/linux/task_work.h
kernel/task_work.c

index 23ab01ae185e504cfc0a70a71d980624855bd0cf..26b8a47f41fcacf98f5cf5d427e22c6f7c6f1715 100644 (file)
@@ -31,6 +31,7 @@ int task_work_add(struct task_struct *task, struct callback_head *twork,
 struct callback_head *task_work_cancel_match(struct task_struct *task,
        bool (*match)(struct callback_head *, void *data), void *data);
 struct callback_head *task_work_cancel_func(struct task_struct *, task_work_func_t);
+bool task_work_cancel(struct task_struct *task, struct callback_head *cb);
 void task_work_run(void);
 
 static inline void exit_task_work(struct task_struct *task)
index 54ac24059daaff2a5d05c3230b2544d777e948f9..2134ac8057a94e64b42d75607f9753e6e82526aa 100644 (file)
@@ -136,6 +136,30 @@ task_work_cancel_func(struct task_struct *task, task_work_func_t func)
        return task_work_cancel_match(task, task_work_func_match, func);
 }
 
+static bool task_work_match(struct callback_head *cb, void *data)
+{
+       return cb == data;
+}
+
+/**
+ * task_work_cancel - cancel a pending work added by task_work_add()
+ * @task: the task which should execute the work
+ * @cb: the callback to remove if queued
+ *
+ * Remove a callback from a task's queue if queued.
+ *
+ * RETURNS:
+ * True if the callback was queued and got cancelled, false otherwise.
+ */
+bool task_work_cancel(struct task_struct *task, struct callback_head *cb)
+{
+       struct callback_head *ret;
+
+       ret = task_work_cancel_match(task, task_work_match, cb);
+
+       return ret == cb;
+}
+
 /**
  * task_work_run - execute the works added by task_work_add()
  *