]> www.infradead.org Git - users/hch/dma-mapping.git/commitdiff
closures: closure_sync_timeout()
authorKent Overstreet <kent.overstreet@linux.dev>
Fri, 3 May 2024 18:43:54 +0000 (14:43 -0400)
committerKent Overstreet <kent.overstreet@linux.dev>
Wed, 8 May 2024 21:29:22 +0000 (17:29 -0400)
Add a new variant of closure_sync_timeout() that takes a timeout.

Note that when this returns -ETIME the closure will still be waiting on
something, i.e. it's not safe to return if you've got a stack allocated
closure.

Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
include/linux/closure.h
lib/closure.c

index c554c6a08768ad60cdf529a65cf962095363a4a9..99155df162d03ca4369fe4ada58c371cbc1b7813 100644 (file)
@@ -194,6 +194,18 @@ static inline void closure_sync(struct closure *cl)
                __closure_sync(cl);
 }
 
+int __closure_sync_timeout(struct closure *cl, unsigned long timeout);
+
+static inline int closure_sync_timeout(struct closure *cl, unsigned long timeout)
+{
+#ifdef CONFIG_DEBUG_CLOSURES
+       BUG_ON(closure_nr_remaining(cl) != 1 && !cl->closure_get_happened);
+#endif
+       return cl->closure_get_happened
+               ? __closure_sync_timeout(cl, timeout)
+               : 0;
+}
+
 #ifdef CONFIG_DEBUG_CLOSURES
 
 void closure_debug_create(struct closure *cl);
index c16540552d61bc14121b034a9d6e302045ff0dc5..07409e9e35a53ef96d0957c396c4685ce69a7f05 100644 (file)
@@ -139,6 +139,43 @@ void __sched __closure_sync(struct closure *cl)
 }
 EXPORT_SYMBOL(__closure_sync);
 
+int __sched __closure_sync_timeout(struct closure *cl, unsigned long timeout)
+{
+       struct closure_syncer s = { .task = current };
+       int ret = 0;
+
+       cl->s = &s;
+       continue_at(cl, closure_sync_fn, NULL);
+
+       while (1) {
+               set_current_state(TASK_UNINTERRUPTIBLE);
+               if (s.done)
+                       break;
+               if (!timeout) {
+                       /*
+                        * Carefully undo the continue_at() - but only if it
+                        * hasn't completed, i.e. the final closure_put() hasn't
+                        * happened yet:
+                        */
+                       unsigned old, new, v = atomic_read(&cl->remaining);
+                       do {
+                               old = v;
+                               if (!old || (old & CLOSURE_RUNNING))
+                                       goto success;
+
+                               new = old + CLOSURE_REMAINING_INITIALIZER;
+                       } while ((v = atomic_cmpxchg(&cl->remaining, old, new)) != old);
+                       ret = -ETIME;
+               }
+
+               timeout = schedule_timeout(timeout);
+       }
+success:
+       __set_current_state(TASK_RUNNING);
+       return ret;
+}
+EXPORT_SYMBOL(__closure_sync_timeout);
+
 #ifdef CONFIG_DEBUG_CLOSURES
 
 static LIST_HEAD(closure_list);