]> granicus.if.org Git - zfs/commitdiff
Disable rw_tryupgrade() for newer kernels
authorBrian Behlendorf <behlendorf1@llnl.gov>
Tue, 20 Apr 2010 22:16:27 +0000 (15:16 -0700)
committerBrian Behlendorf <behlendorf1@llnl.gov>
Thu, 22 Apr 2010 19:28:19 +0000 (12:28 -0700)
For kernels using the CONFIG_RWSEM_GENERIC_SPINLOCK implementation
nothing has changed.  But if your kernel is building with arch
specific rwsems rw_tryupgrade() has been disabled until it can
be implemented correctly.  In particular, the x86 implementation
now leverages atomic primatives for serialization rather than
spinlocks.  So to get this working again it will need to be
implemented as a cmpxchg for x86 and likely something similiar
for other arches we are interested in.  For now it's safest
to simply disable it.

include/sys/rwlock.h

index 89fdfa537e42e9e8abd44f8541a9326016880edf..0fc8d24f7c4c7b30041768057375ff726fdfe632 100644 (file)
@@ -70,7 +70,7 @@ extern int __down_write_trylock_locked(struct rw_semaphore *);
  */
 # if defined(_I386_RWSEM_H) || defined(_ASM_X86_RWSEM_H)
 #  define RW_COUNT(rwp)                 ((SEM(rwp)->count < 0) ? (-1) : \
-                                        (SEM(rwp)->count & RWSEM_ACTIVE_MASK))
+                                        (SEM(rwp)->count & RWSEM_ACTIVE_MASK))
 # else
 #  define RW_COUNT(rwp)                 (SEM(rwp)->count & RWSEM_ACTIVE_MASK)
 # endif
@@ -225,6 +225,7 @@ RW_LOCK_HELD(krwlock_t *rwp)
         downgrade_write(SEM(rwp));                                      \
 })
 
+#if defined(CONFIG_RWSEM_GENERIC_SPINLOCK)
 #define rw_tryupgrade(rwp)                                              \
 ({                                                                      \
         unsigned long _flags_;                                          \
@@ -239,6 +240,14 @@ RW_LOCK_HELD(krwlock_t *rwp)
         spin_unlock_irqrestore(&SEM(rwp)->wait_lock, _flags_);          \
         _rc_;                                                           \
 })
+#else
+/*
+ * This can be done correctly but for each supported arch we will need
+ * a custom cmpxchg() to atomically check and promote the rwsem.  That's
+ * not worth the trouble for now so rw_tryupgrade() will always fail.
+ */
+#define rw_tryupgrade(rwp)      ({ 0; })
+#endif
 
 int spl_rw_init(void);
 void spl_rw_fini(void);