// SPDX-License-Identifier: GPL-2.0 #include #include #include static struct callback_head work_exited; /* all we need is ->next == NULL */ /** * task_work_add - ask the @task to execute @work->func() * @task: the task which should run the callback * @work: the callback to run * @notify: how to notify the targeted task * * Queue @work for task_work_run() below and notify the @task if @notify * is @TWA_RESUME, @TWA_SIGNAL, or @TWA_SIGNAL_NO_IPI. * * @TWA_SIGNAL works like signals, in that the it will interrupt the targeted * task and run the task_work, regardless of whether the task is currently * running in the kernel or userspace. * @TWA_SIGNAL_NO_IPI works like @TWA_SIGNAL, except it doesn't send a * reschedule IPI to force the targeted task to reschedule and run task_work. * This can be advantageous if there's no strict requirement that the * task_work be run as soon as possible, just whenever the task enters the * kernel anyway. * @TWA_RESUME work is run only when the task exits the kernel and returns to * user mode, or before entering guest mode. * * Fails if the @task is exiting/exited and thus it can't process this @work. * Otherwise @work->func() will be called when the @task goes through one of * the aforementioned transitions, or exits. * * If the targeted task is exiting, then an error is returned and the work item * is not queued. It's up to the caller to arrange for an alternative mechanism * in that case. * * Note: there is no ordering guarantee on works queued here. The task_work * list is LIFO. * * RETURNS: * 0 if succeeds or -ESRCH. */ int task_work_add(struct task_struct *task, struct callback_head *work, enum task_work_notify_mode notify) { struct callback_head *head; /* record the work call stack in order to print it in KASAN reports */ kasan_record_aux_stack(work); do { head = READ_ONCE(task->task_works); if (unlikely(head == &work_exited)) return -ESRCH; work->next = head; } while (cmpxchg(&task->task_works, head, work) != head); switch (notify) { case TWA_NONE: break; case TWA_RESUME: set_notify_resume(task); break; case TWA_SIGNAL: set_notify_signal(task); break; case TWA_SIGNAL_NO_IPI: __set_notify_signal(task); break; default: WARN_ON_ONCE(1); break; } return 0; } /** * task_work_cancel_match - cancel a pending work added by task_work_add() * @task: the task which should execute the work * @match: match function to call * * RETURNS: * The found work or NULL if not found. */ struct callback_head * task_work_cancel_match(struct task_struct *task, bool (*match)(struct callback_head *, void *data), void *data) { struct callback_head **pprev = &task->task_works; struct callback_head *work; unsigned long flags; if (likely(!task_work_pending(task))) return NULL; /* * If cmpxchg() fails we continue without updating pprev. * Either we raced with task_work_add() which added the * new entry before this work, we will find it again. Or * we raced with task_work_run(), *pprev == NULL/exited. */ raw_spin_lock_irqsave(&task->pi_lock, flags); while ((work = READ_ONCE(*pprev))) { if (!match(work, data)) pprev = &work->next; else if (cmpxchg(pprev, work, work->next) == work) break; } raw_spin_unlock_irqrestore(&task->pi_lock, flags); return work; } static bool task_work_func_match(struct callback_head *cb, void *data) { return cb->func == data; } /** * task_work_cancel - cancel a pending work added by task_work_add() * @task: the task which should execute the work * @func: identifies the work to remove * * Find the last queued pending work with ->func == @func and remove * it from queue. * * RETURNS: * The found work or NULL if not found. */ struct callback_head * task_work_cancel(struct task_struct *task, task_work_func_t func) { return task_work_cancel_match(task, task_work_func_match, func); } /** * task_work_run - execute the works added by task_work_add() * * Flush the pending works. Should be used by the core kernel code. * Called before the task returns to the user-mode or stops, or when * it exits. In the latter case task_work_add() can no longer add the * new work after task_work_run() returns. */ void task_work_run(void) { struct task_struct *task = current; struct callback_head *work, *head, *next; for (;;) { /* * work->func() can do task_work_add(), do not set * work_exited unless the list is empty. */ do { head = NULL; work = READ_ONCE(task->task_works); if (!work) { if (task->flags & PF_EXITING) head = &work_exited; else break; } } while (cmpxchg(&task->task_works, work, head) != work); if (!work) break; /* * Synchronize with task_work_cancel(). It can not remove * the first entry == work, cmpxchg(task_works) must fail. * But it can remove another entry from the ->next list. */ raw_spin_lock_irq(&task->pi_lock); raw_spin_unlock_irq(&task->pi_lock); do { next = work->next; work->func(work); work = next; cond_resched(); } while (work); } } .0-29 main, development code repositoryroot
summaryrefslogtreecommitdiff
ModeNameSize
-rw-r--r--.gitattributes26logplain
-rw-r--r--.gitignore1841logplain
-rw-r--r--.src_root0logplain
-rw-r--r--COPYING35147logplain
-rw-r--r--COPYING.LGPL7639logplain
-rw-r--r--COPYING.MPL25755logplain
-rw-r--r--GNUmakefile.mk1466logplain
-rw-r--r--Library_merged.mk1977logplain
-rw-r--r--Makefile.in5339logplain
d---------MathMLDTD247logplain
d---------Mesa180logplain
-rw-r--r--Module_tail_build.mk1808logplain
-rw-r--r--README.Android2265logplain
-rw-r--r--README.cross16265logplain
-rw-r--r--Repository.mk6268logplain
-rw-r--r--RepositoryExternal.mk15519logplain
-rw-r--r--RepositoryFixes.mk7270logplain
-rw-r--r--RepositoryModule_ooo.mk2460logplain
-rw-r--r--TEMPLATE.SOURCECODE.HEADER1279logplain
d---------UnoControls124logplain
d---------accessibility191logplain
d---------afms103logplain
d---------animations308logplain
d---------apache-commons95logplain
d---------apple_remote493logplain
d---------autodoc93logplain
-rwxr-xr-xautogen.sh4356logplain
d---------automation710logplain
d---------avmedia501logplain
d---------basctl410logplain
d---------basebmp410logplain
d---------basegfx493logplain
d---------basic635logplain
d---------bean378logplain
d---------beanshell116logplain
d---------berkeleydb201logplain
d---------bin1014logplain
d---------binaryurp353logplain
d---------boost481logplain
-rwxr-xr-xbootstrap2325logplain
d---------bridges197logplain
-rwxr-xr-xbuild_env.in217logplain
d---------cairo290logplain
d---------canvas747logplain
d---------chart2607logplain
d---------cli_ure296logplain
d---------codemaker165logplain
d---------comphelper474logplain
-rwxr-xr-xconfig.guess44461logplain
-rwxr-xr-xconfig.sub35356logplain
d---------configmgr262logplain
-rw-r--r--configure.in341180logplain
d---------connectivity372logplain
d---------cosv124logplain
d---------cppcanvas316logplain
d---------cppu153logplain
d---------cppuhelper188logplain
d---------cppunit513logplain
d---------cpputools63logplain
d---------crashrep63logplain
d---------cross_toolset / prj30logplain
d---------ct2n254logplain
d---------cui264logplain
d---------curl302logplain
d---------dbaccess1217logplain
d---------default_images1042logplain
d---------desktop2959logplain
d---------distro-configs548logplain
d---------dmake2458logplain
-rwxr-xr-xdownload10620logplain
d---------drawinglayer303logplain
d---------dtrans525logplain
d---------editeng438logplain
d---------embeddedobj184logplain
d---------embedserv94logplain
d---------epm110logplain
d---------eventattacher195logplain
d---------expat160logplain
d---------extensions401logplain
d---------external559logplain
d---------extras124logplain
d---------fileaccess193logplain
d---------filter2452logplain
d---------forms483logplain
d---------formula437logplain
d---------fpicker155logplain
d---------framework903logplain
-rwxr-xr-xg13003logplain
d---------gdk-pixbuf177logplain
d---------gettext177logplain
d---------git-hooks114logplain
d---------glib269logplain
d---------graphite269logplain
d---------guw70logplain
d---------hsqldb141logplain
d---------hunspell329logplain
d---------hwpfilter284logplain
d---------hyphen226logplain
d---------i18npool1490logplain
d---------i18nutil264logplain
d---------icc295logplain
d---------icu705logplain
d---------idl248logplain
d---------idlc124logplain
-rw-r--r--install-sh6315logplain
d---------instsetoo_native323logplain
d---------io124logplain
d---------javaunohelper185logplain
d---------