rwsem.c 3.3 KB

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