rwsem.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. /* kernel/rwsem.c: R/W semaphores, public implementation
  2. *
  3. * Written by David Howells (dhowells@redhat.com).
  4. * Derived from asm-i386/semaphore.h
  5. */
  6. #include <linux/types.h>
  7. #include <linux/kernel.h>
  8. #include <linux/sched.h>
  9. #include <linux/export.h>
  10. #include <linux/rwsem.h>
  11. #include <linux/atomic.h>
  12. #ifdef CONFIG_RWSEM_SPIN_ON_OWNER
  13. static inline void rwsem_set_owner(struct rw_semaphore *sem)
  14. {
  15. sem->owner = current;
  16. }
  17. static inline void rwsem_clear_owner(struct rw_semaphore *sem)
  18. {
  19. sem->owner = NULL;
  20. }
  21. #else
  22. static inline void rwsem_set_owner(struct rw_semaphore *sem)
  23. {
  24. }
  25. static inline void rwsem_clear_owner(struct rw_semaphore *sem)
  26. {
  27. }
  28. #endif
  29. /*
  30. * lock for reading
  31. */
  32. void __sched down_read(struct rw_semaphore *sem)
  33. {
  34. might_sleep();
  35. rwsem_acquire_read(&sem->dep_map, 0, 0, _RET_IP_);
  36. LOCK_CONTENDED(sem, __down_read_trylock, __down_read);
  37. }
  38. EXPORT_SYMBOL(down_read);
  39. /*
  40. * trylock for reading -- returns 1 if successful, 0 if contention
  41. */
  42. int down_read_trylock(struct rw_semaphore *sem)
  43. {
  44. int ret = __down_read_trylock(sem);
  45. if (ret == 1)
  46. rwsem_acquire_read(&sem->dep_map, 0, 1, _RET_IP_);
  47. return ret;
  48. }
  49. EXPORT_SYMBOL(down_read_trylock);
  50. /*
  51. * lock for writing
  52. */
  53. void __sched down_write(struct rw_semaphore *sem)
  54. {
  55. might_sleep();
  56. rwsem_acquire(&sem->dep_map, 0, 0, _RET_IP_);
  57. LOCK_CONTENDED(sem, __down_write_trylock, __down_write);
  58. rwsem_set_owner(sem);
  59. }
  60. EXPORT_SYMBOL(down_write);
  61. /*
  62. * trylock for writing -- returns 1 if successful, 0 if contention
  63. */
  64. int down_write_trylock(struct rw_semaphore *sem)
  65. {
  66. int ret = __down_write_trylock(sem);
  67. if (ret == 1) {
  68. rwsem_acquire(&sem->dep_map, 0, 1, _RET_IP_);
  69. rwsem_set_owner(sem);
  70. }
  71. return ret;
  72. }
  73. EXPORT_SYMBOL(down_write_trylock);
  74. /*
  75. * release a read lock
  76. */
  77. void up_read(struct rw_semaphore *sem)
  78. {
  79. rwsem_release(&sem->dep_map, 1, _RET_IP_);
  80. __up_read(sem);
  81. }
  82. EXPORT_SYMBOL(up_read);
  83. /*
  84. * release a write lock
  85. */
  86. void up_write(struct rw_semaphore *sem)
  87. {
  88. rwsem_release(&sem->dep_map, 1, _RET_IP_);
  89. rwsem_clear_owner(sem);
  90. __up_write(sem);
  91. }
  92. EXPORT_SYMBOL(up_write);
  93. /*
  94. * downgrade write lock to read lock
  95. */
  96. void downgrade_write(struct rw_semaphore *sem)
  97. {
  98. /*
  99. * lockdep: a downgraded write will live on as a write
  100. * dependency.
  101. */
  102. rwsem_clear_owner(sem);
  103. __downgrade_write(sem);
  104. }
  105. EXPORT_SYMBOL(downgrade_write);
  106. #ifdef CONFIG_DEBUG_LOCK_ALLOC
  107. void down_read_nested(struct rw_semaphore *sem, int subclass)
  108. {
  109. might_sleep();
  110. rwsem_acquire_read(&sem->dep_map, subclass, 0, _RET_IP_);
  111. LOCK_CONTENDED(sem, __down_read_trylock, __down_read);
  112. }
  113. EXPORT_SYMBOL(down_read_nested);
  114. void _down_write_nest_lock(struct rw_semaphore *sem, struct lockdep_map *nest)
  115. {
  116. might_sleep();
  117. rwsem_acquire_nest(&sem->dep_map, 0, 0, nest, _RET_IP_);
  118. LOCK_CONTENDED(sem, __down_write_trylock, __down_write);
  119. rwsem_set_owner(sem);
  120. }
  121. EXPORT_SYMBOL(_down_write_nest_lock);
  122. void down_read_non_owner(struct rw_semaphore *sem)
  123. {
  124. might_sleep();
  125. __down_read(sem);
  126. }
  127. EXPORT_SYMBOL(down_read_non_owner);
  128. void down_write_nested(struct rw_semaphore *sem, int subclass)
  129. {
  130. might_sleep();
  131. rwsem_acquire(&sem->dep_map, subclass, 0, _RET_IP_);
  132. LOCK_CONTENDED(sem, __down_write_trylock, __down_write);
  133. rwsem_set_owner(sem);
  134. }
  135. EXPORT_SYMBOL(down_write_nested);
  136. void up_read_non_owner(struct rw_semaphore *sem)
  137. {
  138. __up_read(sem);
  139. }
  140. EXPORT_SYMBOL(up_read_non_owner);
  141. #endif