|
@@ -350,3 +350,31 @@ bool refcount_dec_and_lock(refcount_t *r, spinlock_t *lock)
|
|
|
}
|
|
|
EXPORT_SYMBOL(refcount_dec_and_lock);
|
|
|
|
|
|
+/**
|
|
|
+ * refcount_dec_and_lock_irqsave - return holding spinlock with disabled
|
|
|
+ * interrupts if able to decrement refcount to 0
|
|
|
+ * @r: the refcount
|
|
|
+ * @lock: the spinlock to be locked
|
|
|
+ * @flags: saved IRQ-flags if the is acquired
|
|
|
+ *
|
|
|
+ * Same as refcount_dec_and_lock() above except that the spinlock is acquired
|
|
|
+ * with disabled interupts.
|
|
|
+ *
|
|
|
+ * Return: true and hold spinlock if able to decrement refcount to 0, false
|
|
|
+ * otherwise
|
|
|
+ */
|
|
|
+bool refcount_dec_and_lock_irqsave(refcount_t *r, spinlock_t *lock,
|
|
|
+ unsigned long *flags)
|
|
|
+{
|
|
|
+ if (refcount_dec_not_one(r))
|
|
|
+ return false;
|
|
|
+
|
|
|
+ spin_lock_irqsave(lock, *flags);
|
|
|
+ if (!refcount_dec_and_test(r)) {
|
|
|
+ spin_unlock_irqrestore(lock, *flags);
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ return true;
|
|
|
+}
|
|
|
+EXPORT_SYMBOL(refcount_dec_and_lock_irqsave);
|