sync0rw.c
上传用户:romrleung
上传日期:2022-05-23
资源大小:18897k
文件大小:22k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /******************************************************
  2. The read-write lock (for thread synchronization)
  3. (c) 1995 Innobase Oy
  4. Created 9/11/1995 Heikki Tuuri
  5. *******************************************************/
  6. #include "sync0rw.h"
  7. #ifdef UNIV_NONINL
  8. #include "sync0rw.ic"
  9. #endif
  10. #include "os0thread.h"
  11. #include "mem0mem.h"
  12. #include "srv0srv.h"
  13. ulint rw_s_system_call_count = 0;
  14. ulint rw_s_spin_wait_count = 0;
  15. ulint rw_s_os_wait_count = 0;
  16. ulint rw_s_exit_count = 0;
  17. ulint rw_x_system_call_count = 0;
  18. ulint rw_x_spin_wait_count = 0;
  19. ulint rw_x_os_wait_count = 0;
  20. ulint rw_x_exit_count = 0;
  21. /* The global list of rw-locks */
  22. rw_lock_list_t rw_lock_list;
  23. mutex_t rw_lock_list_mutex;
  24. #ifdef UNIV_SYNC_DEBUG
  25. /* The global mutex which protects debug info lists of all rw-locks.
  26. To modify the debug info list of an rw-lock, this mutex has to be
  27. acquired in addition to the mutex protecting the lock. */
  28. mutex_t rw_lock_debug_mutex;
  29. os_event_t rw_lock_debug_event; /* If deadlock detection does not
  30. get immediately the mutex, it may
  31. wait for this event */
  32. ibool rw_lock_debug_waiters; /* This is set to TRUE, if there may
  33. be waiters for the event */
  34. /**********************************************************************
  35. Creates a debug info struct. */
  36. static
  37. rw_lock_debug_t*
  38. rw_lock_debug_create(void);
  39. /*======================*/
  40. /**********************************************************************
  41. Frees a debug info struct. */
  42. static
  43. void
  44. rw_lock_debug_free(
  45. /*===============*/
  46. rw_lock_debug_t* info);
  47. /**********************************************************************
  48. Creates a debug info struct. */
  49. static
  50. rw_lock_debug_t*
  51. rw_lock_debug_create(void)
  52. /*======================*/
  53. {
  54. return((rw_lock_debug_t*) mem_alloc(sizeof(rw_lock_debug_t)));
  55. }
  56. /**********************************************************************
  57. Frees a debug info struct. */
  58. static
  59. void
  60. rw_lock_debug_free(
  61. /*===============*/
  62. rw_lock_debug_t* info)
  63. {
  64. mem_free(info);
  65. }
  66. #endif /* UNIV_SYNC_DEBUG */
  67. /**********************************************************************
  68. Creates, or rather, initializes an rw-lock object in a specified memory
  69. location (which must be appropriately aligned). The rw-lock is initialized
  70. to the non-locked state. Explicit freeing of the rw-lock with rw_lock_free
  71. is necessary only if the memory block containing it is freed. */
  72. void
  73. rw_lock_create_func(
  74. /*================*/
  75. rw_lock_t* lock, /* in: pointer to memory */
  76. const char* cfile_name, /* in: file name where created */
  77. ulint cline) /* in: file line where created */
  78. {
  79. /* If this is the very first time a synchronization
  80. object is created, then the following call initializes
  81. the sync system. */
  82. mutex_create(rw_lock_get_mutex(lock));
  83. mutex_set_level(rw_lock_get_mutex(lock), SYNC_NO_ORDER_CHECK);
  84. lock->mutex.cfile_name = cfile_name;
  85. lock->mutex.cline = cline;
  86. rw_lock_set_waiters(lock, 0);
  87. rw_lock_set_writer(lock, RW_LOCK_NOT_LOCKED);
  88. lock->writer_count = 0;
  89. rw_lock_set_reader_count(lock, 0);
  90. lock->writer_is_wait_ex = FALSE;
  91. #ifdef UNIV_SYNC_DEBUG
  92. UT_LIST_INIT(lock->debug_list);
  93. lock->level = SYNC_LEVEL_NONE;
  94. #endif /* UNIV_SYNC_DEBUG */
  95. lock->magic_n = RW_LOCK_MAGIC_N;
  96. lock->cfile_name = cfile_name;
  97. lock->cline = cline;
  98. lock->last_s_file_name = "not yet reserved";
  99. lock->last_x_file_name = "not yet reserved";
  100. lock->last_s_line = 0;
  101. lock->last_x_line = 0;
  102. mutex_enter(&rw_lock_list_mutex);
  103. if (UT_LIST_GET_LEN(rw_lock_list) > 0) {
  104. ut_a(UT_LIST_GET_FIRST(rw_lock_list)->magic_n
  105. == RW_LOCK_MAGIC_N);
  106. }
  107. UT_LIST_ADD_FIRST(list, rw_lock_list, lock);
  108. mutex_exit(&rw_lock_list_mutex);
  109. }
  110. /**********************************************************************
  111. Calling this function is obligatory only if the memory buffer containing
  112. the rw-lock is freed. Removes an rw-lock object from the global list. The
  113. rw-lock is checked to be in the non-locked state. */
  114. void
  115. rw_lock_free(
  116. /*=========*/
  117. rw_lock_t* lock) /* in: rw-lock */
  118. {
  119. #ifdef UNIV_DEBUG
  120. ut_a(rw_lock_validate(lock));
  121. #endif /* UNIV_DEBUG */
  122. ut_a(rw_lock_get_writer(lock) == RW_LOCK_NOT_LOCKED);
  123. ut_a(rw_lock_get_waiters(lock) == 0);
  124. ut_a(rw_lock_get_reader_count(lock) == 0);
  125. lock->magic_n = 0;
  126. mutex_free(rw_lock_get_mutex(lock));
  127. mutex_enter(&rw_lock_list_mutex);
  128. if (UT_LIST_GET_PREV(list, lock)) {
  129. ut_a(UT_LIST_GET_PREV(list, lock)->magic_n == RW_LOCK_MAGIC_N);
  130. }
  131. if (UT_LIST_GET_NEXT(list, lock)) {
  132. ut_a(UT_LIST_GET_NEXT(list, lock)->magic_n == RW_LOCK_MAGIC_N);
  133. }
  134. UT_LIST_REMOVE(list, rw_lock_list, lock);
  135. mutex_exit(&rw_lock_list_mutex);
  136. }
  137. /**********************************************************************
  138. Checks that the rw-lock has been initialized and that there are no
  139. simultaneous shared and exclusive locks. */
  140. ibool
  141. rw_lock_validate(
  142. /*=============*/
  143. rw_lock_t* lock)
  144. {
  145. ut_a(lock);
  146. mutex_enter(rw_lock_get_mutex(lock));
  147. ut_a(lock->magic_n == RW_LOCK_MAGIC_N);
  148. ut_a((rw_lock_get_reader_count(lock) == 0)
  149.      || (rw_lock_get_writer(lock) != RW_LOCK_EX));
  150. ut_a((rw_lock_get_writer(lock) == RW_LOCK_EX)
  151.      || (rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX)
  152.      || (rw_lock_get_writer(lock) == RW_LOCK_NOT_LOCKED));
  153. ut_a((rw_lock_get_waiters(lock) == 0)
  154.      || (rw_lock_get_waiters(lock) == 1));
  155. ut_a((lock->writer != RW_LOCK_EX) || (lock->writer_count > 0));
  156.      
  157. mutex_exit(rw_lock_get_mutex(lock));
  158. return(TRUE);
  159. }
  160. /**********************************************************************
  161. Lock an rw-lock in shared mode for the current thread. If the rw-lock is
  162. locked in exclusive mode, or there is an exclusive lock request waiting,
  163. the function spins a preset time (controlled by SYNC_SPIN_ROUNDS), waiting
  164. for the lock, before suspending the thread. */
  165. void
  166. rw_lock_s_lock_spin(
  167. /*================*/
  168.         rw_lock_t*    lock,   /* in: pointer to rw-lock */
  169. ulint pass, /* in: pass value; != 0, if the lock
  170. will be passed to another thread to unlock */
  171. const char* file_name, /* in: file name where lock requested */
  172. ulint line) /* in: line where requested */
  173. {
  174.         ulint    index; /* index of the reserved wait cell */
  175.         ulint    i;    /* spin round count */
  176.         
  177.         ut_ad(rw_lock_validate(lock));
  178. lock_loop:
  179. rw_s_spin_wait_count++;
  180. /* Spin waiting for the writer field to become free */
  181.         i = 0;
  182.         while (rw_lock_get_writer(lock) != RW_LOCK_NOT_LOCKED
  183. && i < SYNC_SPIN_ROUNDS) {
  184.          if (srv_spin_wait_delay) {
  185.          ut_delay(ut_rnd_interval(0, srv_spin_wait_delay));
  186.          }
  187.          i++;
  188.         }
  189. if (i == SYNC_SPIN_ROUNDS) {
  190. os_thread_yield();
  191. }
  192. if (srv_print_latch_waits) {
  193. fprintf(stderr,
  194. "Thread %lu spin wait rw-s-lock at %p cfile %s cline %lu rnds %lun",
  195. (ulong) os_thread_pf(os_thread_get_curr_id()), lock,
  196. lock->cfile_name, (ulong) lock->cline, (ulong) i);
  197. }
  198. mutex_enter(rw_lock_get_mutex(lock));
  199.         /* We try once again to obtain the lock */
  200. if (TRUE == rw_lock_s_lock_low(lock, pass, file_name, line)) {
  201. mutex_exit(rw_lock_get_mutex(lock));
  202. return; /* Success */
  203. } else {
  204. /* If we get here, locking did not succeed, we may
  205. suspend the thread to wait in the wait array */
  206. rw_s_system_call_count++;
  207.          sync_array_reserve_cell(sync_primary_wait_array,
  208. lock, RW_LOCK_SHARED,
  209. file_name, line,
  210. &index);
  211. rw_lock_set_waiters(lock, 1);
  212. mutex_exit(rw_lock_get_mutex(lock));
  213. if (srv_print_latch_waits) {
  214. fprintf(stderr,
  215. "Thread %lu OS wait rw-s-lock at %p cfile %s cline %lun",
  216. os_thread_pf(os_thread_get_curr_id()),
  217.         lock, lock->cfile_name, (ulong) lock->cline);
  218. }
  219. rw_s_system_call_count++;
  220. rw_s_os_wait_count++;
  221.           sync_array_wait_event(sync_primary_wait_array, index);
  222.          goto lock_loop;
  223. }        
  224. }
  225. /**********************************************************************
  226. This function is used in the insert buffer to move the ownership of an
  227. x-latch on a buffer frame to the current thread. The x-latch was set by
  228. the buffer read operation and it protected the buffer frame while the
  229. read was done. The ownership is moved because we want that the current
  230. thread is able to acquire a second x-latch which is stored in an mtr.
  231. This, in turn, is needed to pass the debug checks of index page
  232. operations. */
  233. void
  234. rw_lock_x_lock_move_ownership(
  235. /*==========================*/
  236. rw_lock_t* lock) /* in: lock which was x-locked in the
  237. buffer read */
  238. {
  239. ut_ad(rw_lock_is_locked(lock, RW_LOCK_EX));
  240. mutex_enter(&(lock->mutex));
  241. lock->writer_thread = os_thread_get_curr_id();
  242. lock->pass = 0;
  243. mutex_exit(&(lock->mutex));
  244. }
  245. /**********************************************************************
  246. Low-level function for acquiring an exclusive lock. */
  247. UNIV_INLINE
  248. ulint
  249. rw_lock_x_lock_low(
  250. /*===============*/
  251. /* out: RW_LOCK_NOT_LOCKED if did
  252. not succeed, RW_LOCK_EX if success,
  253. RW_LOCK_WAIT_EX, if got wait reservation */
  254.         rw_lock_t*    lock,   /* in: pointer to rw-lock */
  255. ulint pass, /* in: pass value; != 0, if the lock will
  256. be passed to another thread to unlock */
  257. const char* file_name,/* in: file name where lock requested */
  258. ulint line) /* in: line where requested */
  259. {
  260. #ifdef UNIV_SYNC_DEBUG
  261. ut_ad(mutex_own(rw_lock_get_mutex(lock)));
  262. #endif /* UNIV_SYNC_DEBUG */
  263. if (rw_lock_get_writer(lock) == RW_LOCK_NOT_LOCKED) {
  264. if (rw_lock_get_reader_count(lock) == 0) {
  265. rw_lock_set_writer(lock, RW_LOCK_EX);
  266. lock->writer_thread = os_thread_get_curr_id();
  267. lock->writer_count++;
  268. lock->pass = pass;
  269. #ifdef UNIV_SYNC_DEBUG
  270. rw_lock_add_debug_info(lock, pass, RW_LOCK_EX,
  271. file_name, line);
  272. #endif
  273. lock->last_x_file_name = file_name;
  274. lock->last_x_line = line;
  275. /* Locking succeeded, we may return */
  276. return(RW_LOCK_EX);
  277. } else {
  278. /* There are readers, we have to wait */
  279. rw_lock_set_writer(lock, RW_LOCK_WAIT_EX);
  280. lock->writer_thread = os_thread_get_curr_id();
  281. lock->pass = pass;
  282. lock->writer_is_wait_ex = TRUE;
  283. #ifdef UNIV_SYNC_DEBUG
  284. rw_lock_add_debug_info(lock, pass, RW_LOCK_WAIT_EX,
  285. file_name, line);
  286. #endif
  287. return(RW_LOCK_WAIT_EX);
  288. }
  289. } else if ((rw_lock_get_writer(lock) == RW_LOCK_WAIT_EX)
  290.    && os_thread_eq(lock->writer_thread,
  291. os_thread_get_curr_id())) {
  292. if (rw_lock_get_reader_count(lock) == 0) {
  293. rw_lock_set_writer(lock, RW_LOCK_EX);
  294. lock->writer_count++;
  295. lock->pass = pass;
  296. lock->writer_is_wait_ex = FALSE;
  297. #ifdef UNIV_SYNC_DEBUG
  298. rw_lock_remove_debug_info(lock, pass, RW_LOCK_WAIT_EX);
  299. rw_lock_add_debug_info(lock, pass, RW_LOCK_EX,
  300. file_name, line);
  301. #endif
  302. lock->last_x_file_name = file_name;
  303. lock->last_x_line = line;
  304. /* Locking succeeded, we may return */
  305. return(RW_LOCK_EX);
  306. }
  307. return(RW_LOCK_WAIT_EX);
  308. } else if ((rw_lock_get_writer(lock) == RW_LOCK_EX)
  309.    && os_thread_eq(lock->writer_thread,
  310. os_thread_get_curr_id())
  311.    && (lock->pass == 0)
  312.    && (pass == 0)) {
  313. lock->writer_count++;
  314. #ifdef UNIV_SYNC_DEBUG
  315. rw_lock_add_debug_info(lock, pass, RW_LOCK_EX, file_name,
  316. line);
  317. #endif
  318. lock->last_x_file_name = file_name;
  319. lock->last_x_line = line;
  320. /* Locking succeeded, we may return */
  321. return(RW_LOCK_EX);
  322. }
  323. /* Locking did not succeed */
  324. return(RW_LOCK_NOT_LOCKED);
  325. }
  326. /**********************************************************************
  327. NOTE! Use the corresponding macro, not directly this function! Lock an
  328. rw-lock in exclusive mode for the current thread. If the rw-lock is locked
  329. in shared or exclusive mode, or there is an exclusive lock request waiting,
  330. the function spins a preset time (controlled by SYNC_SPIN_ROUNDS), waiting
  331. for the lock before suspending the thread. If the same thread has an x-lock
  332. on the rw-lock, locking succeed, with the following exception: if pass != 0,
  333. only a single x-lock may be taken on the lock. NOTE: If the same thread has
  334. an s-lock, locking does not succeed! */
  335. void
  336. rw_lock_x_lock_func(
  337. /*================*/
  338.         rw_lock_t*    lock,   /* in: pointer to rw-lock */
  339. ulint pass, /* in: pass value; != 0, if the lock will
  340. be passed to another thread to unlock */
  341. const char* file_name,/* in: file name where lock requested */
  342. ulint line) /* in: line where requested */
  343. {
  344.         ulint index;  /* index of the reserved wait cell */
  345.         ulint state; /* lock state acquired */
  346.         ulint i; /* spin round count */
  347.         
  348.         ut_ad(rw_lock_validate(lock));
  349. lock_loop:
  350.         /* Acquire the mutex protecting the rw-lock fields */
  351. mutex_enter_fast(&(lock->mutex));
  352. state = rw_lock_x_lock_low(lock, pass, file_name, line);
  353. mutex_exit(&(lock->mutex));
  354.         
  355. if (state == RW_LOCK_EX) {
  356. return; /* Locking succeeded */
  357. } else if (state == RW_LOCK_NOT_LOCKED) {
  358.   /* Spin waiting for the writer field to become free */
  359. i = 0;
  360.          while (rw_lock_get_writer(lock) != RW_LOCK_NOT_LOCKED 
  361.                 && i < SYNC_SPIN_ROUNDS) {
  362.          if (srv_spin_wait_delay) {
  363. ut_delay(ut_rnd_interval(0,
  364. srv_spin_wait_delay));
  365.          }
  366.         
  367.          i++;
  368.          }
  369. if (i == SYNC_SPIN_ROUNDS) {
  370. os_thread_yield();
  371. }
  372.         } else if (state == RW_LOCK_WAIT_EX) {
  373.   /* Spin waiting for the reader count field to become zero */
  374. i = 0;
  375.          while (rw_lock_get_reader_count(lock) != 0 
  376.                 && i < SYNC_SPIN_ROUNDS) {
  377.          if (srv_spin_wait_delay) {
  378. ut_delay(ut_rnd_interval(0,
  379. srv_spin_wait_delay));
  380.          }
  381. i++;
  382.          }
  383. if (i == SYNC_SPIN_ROUNDS) {
  384. os_thread_yield();
  385. }
  386.         } else {
  387. i = 0; /* Eliminate a compiler warning */
  388. ut_error;
  389. }
  390. if (srv_print_latch_waits) {
  391. fprintf(stderr,
  392. "Thread %lu spin wait rw-x-lock at %p cfile %s cline %lu rnds %lun",
  393. os_thread_pf(os_thread_get_curr_id()), lock,
  394. lock->cfile_name, (ulong) lock->cline, (ulong) i);
  395. }
  396. rw_x_spin_wait_count++;
  397.         /* We try once again to obtain the lock. Acquire the mutex protecting
  398. the rw-lock fields */
  399. mutex_enter(rw_lock_get_mutex(lock));
  400. state = rw_lock_x_lock_low(lock, pass, file_name, line);
  401. if (state == RW_LOCK_EX) {
  402. mutex_exit(rw_lock_get_mutex(lock));
  403. return; /* Locking succeeded */
  404. }
  405. rw_x_system_call_count++;
  406.         sync_array_reserve_cell(sync_primary_wait_array,
  407. lock, RW_LOCK_EX,
  408. file_name, line,
  409. &index);
  410. rw_lock_set_waiters(lock, 1);
  411. mutex_exit(rw_lock_get_mutex(lock));
  412. if (srv_print_latch_waits) {
  413. fprintf(stderr,
  414. "Thread %lu OS wait for rw-x-lock at %p cfile %s cline %lun",
  415. os_thread_pf(os_thread_get_curr_id()), lock,
  416. lock->cfile_name, (ulong) lock->cline);
  417. }
  418. rw_x_system_call_count++;
  419. rw_x_os_wait_count++;
  420.         sync_array_wait_event(sync_primary_wait_array, index);
  421.         goto lock_loop;
  422. }
  423. #ifdef UNIV_SYNC_DEBUG
  424. /**********************************************************************
  425. Acquires the debug mutex. We cannot use the mutex defined in sync0sync,
  426. because the debug mutex is also acquired in sync0arr while holding the OS
  427. mutex protecting the sync array, and the ordinary mutex_enter might
  428. recursively call routines in sync0arr, leading to a deadlock on the OS
  429. mutex. */
  430. void
  431. rw_lock_debug_mutex_enter(void)
  432. /*==========================*/
  433. {
  434. loop:
  435. if (0 == mutex_enter_nowait(&rw_lock_debug_mutex,
  436. __FILE__, __LINE__)) {
  437. return;
  438. }
  439. os_event_reset(rw_lock_debug_event);
  440. rw_lock_debug_waiters = TRUE;
  441. if (0 == mutex_enter_nowait(&rw_lock_debug_mutex,
  442. __FILE__, __LINE__)) {
  443. return;
  444. }
  445. os_event_wait(rw_lock_debug_event);
  446. goto loop;
  447. }
  448. /**********************************************************************
  449. Releases the debug mutex. */
  450. void
  451. rw_lock_debug_mutex_exit(void)
  452. /*==========================*/
  453. {
  454. mutex_exit(&rw_lock_debug_mutex);
  455. if (rw_lock_debug_waiters) {
  456. rw_lock_debug_waiters = FALSE;
  457. os_event_set(rw_lock_debug_event);
  458. }
  459. }
  460. /**********************************************************************
  461. Inserts the debug information for an rw-lock. */
  462. void
  463. rw_lock_add_debug_info(
  464. /*===================*/
  465. rw_lock_t* lock, /* in: rw-lock */
  466. ulint pass, /* in: pass value */
  467. ulint lock_type, /* in: lock type */
  468. const char* file_name, /* in: file where requested */
  469. ulint line) /* in: line where requested */
  470. {
  471. rw_lock_debug_t* info;
  472. ut_ad(lock);
  473. ut_ad(file_name);
  474. info = rw_lock_debug_create();
  475. rw_lock_debug_mutex_enter();
  476. info->file_name = file_name;
  477. info->line  = line;
  478. info->lock_type = lock_type;
  479. info->thread_id = os_thread_get_curr_id();
  480. info->pass = pass;
  481. UT_LIST_ADD_FIRST(list, lock->debug_list, info);
  482. rw_lock_debug_mutex_exit();
  483. if ((pass == 0) && (lock_type != RW_LOCK_WAIT_EX)) {
  484. sync_thread_add_level(lock, lock->level);
  485. }
  486. }
  487. /**********************************************************************
  488. Removes a debug information struct for an rw-lock. */
  489. void
  490. rw_lock_remove_debug_info(
  491. /*======================*/
  492. rw_lock_t* lock, /* in: rw-lock */
  493. ulint pass, /* in: pass value */
  494. ulint lock_type) /* in: lock type */
  495. {
  496. rw_lock_debug_t* info;
  497. ut_ad(lock);
  498. if ((pass == 0) && (lock_type != RW_LOCK_WAIT_EX)) {
  499. sync_thread_reset_level(lock);
  500. }
  501. rw_lock_debug_mutex_enter();
  502. info = UT_LIST_GET_FIRST(lock->debug_list);
  503. while (info != NULL) {
  504. if ((pass == info->pass)
  505.     && ((pass != 0)
  506. || os_thread_eq(info->thread_id,
  507. os_thread_get_curr_id()))
  508.     && (info->lock_type == lock_type)) {
  509.      /* Found! */
  510.      UT_LIST_REMOVE(list, lock->debug_list, info);
  511. rw_lock_debug_mutex_exit();
  512.      rw_lock_debug_free(info);
  513.      return;
  514. }
  515. info = UT_LIST_GET_NEXT(list, info);
  516. }
  517. ut_error;
  518. }
  519. #endif /* UNIV_SYNC_DEBUG */
  520. /**********************************************************************
  521. Sets the rw-lock latching level field. */
  522. void
  523. rw_lock_set_level(
  524. /*==============*/
  525. rw_lock_t* lock, /* in: rw-lock */
  526. ulint level) /* in: level */
  527. {
  528. lock->level = level;
  529. }
  530. #ifdef UNIV_SYNC_DEBUG
  531. /**********************************************************************
  532. Checks if the thread has locked the rw-lock in the specified mode, with
  533. the pass value == 0. */
  534. ibool
  535. rw_lock_own(
  536. /*========*/
  537. /* out: TRUE if locked */
  538. rw_lock_t* lock, /* in: rw-lock */
  539. ulint lock_type) /* in: lock type: RW_LOCK_SHARED,
  540. RW_LOCK_EX */
  541. {
  542. rw_lock_debug_t* info;
  543. ut_ad(lock);
  544. ut_ad(rw_lock_validate(lock));
  545. mutex_enter(&(lock->mutex));
  546. info = UT_LIST_GET_FIRST(lock->debug_list);
  547. while (info != NULL) {
  548. if (os_thread_eq(info->thread_id, os_thread_get_curr_id())
  549.     && (info->pass == 0)
  550.     && (info->lock_type == lock_type)) {
  551. mutex_exit(&(lock->mutex));
  552.      /* Found! */
  553.      return(TRUE);
  554. }
  555. info = UT_LIST_GET_NEXT(list, info);
  556. }
  557. mutex_exit(&(lock->mutex));
  558. return(FALSE);
  559. }
  560. #endif /* UNIV_SYNC_DEBUG */
  561. /**********************************************************************
  562. Checks if somebody has locked the rw-lock in the specified mode. */
  563. ibool
  564. rw_lock_is_locked(
  565. /*==============*/
  566. /* out: TRUE if locked */
  567. rw_lock_t* lock, /* in: rw-lock */
  568. ulint lock_type) /* in: lock type: RW_LOCK_SHARED,
  569. RW_LOCK_EX */
  570. {
  571. ibool ret = FALSE;
  572. ut_ad(lock);
  573. ut_ad(rw_lock_validate(lock));
  574. mutex_enter(&(lock->mutex));
  575. if (lock_type == RW_LOCK_SHARED) {
  576. if (lock->reader_count > 0) {
  577. ret = TRUE;
  578. }
  579. } else if (lock_type == RW_LOCK_EX) {
  580. if (lock->writer == RW_LOCK_EX) {
  581. ret = TRUE;
  582. }
  583. } else {
  584. ut_error;
  585. }
  586. mutex_exit(&(lock->mutex));
  587. return(ret);
  588. }
  589. #ifdef UNIV_SYNC_DEBUG
  590. /*******************************************************************
  591. Prints debug info of currently locked rw-locks. */
  592. void
  593. rw_lock_list_print_info(void)
  594. /*=========================*/
  595. {
  596. rw_lock_t* lock;
  597. ulint count = 0;
  598. rw_lock_debug_t* info;
  599. mutex_enter(&rw_lock_list_mutex);
  600. fputs("-------------n"
  601. "RW-LATCH INFOn"
  602. "-------------n", stderr);
  603. lock = UT_LIST_GET_FIRST(rw_lock_list);
  604. while (lock != NULL) {
  605. count++;
  606. mutex_enter(&(lock->mutex));
  607. if ((rw_lock_get_writer(lock) != RW_LOCK_NOT_LOCKED)
  608.     || (rw_lock_get_reader_count(lock) != 0)
  609.     || (rw_lock_get_waiters(lock) != 0)) {
  610. fprintf(stderr, "RW-LOCK: %p ", lock);
  611. if (rw_lock_get_waiters(lock)) {
  612. fputs(" Waiters for the lock existn", stderr);
  613. } else {
  614. putc('n', stderr);
  615. }
  616.     
  617. info = UT_LIST_GET_FIRST(lock->debug_list);
  618. while (info != NULL) {
  619. rw_lock_debug_print(info);
  620. info = UT_LIST_GET_NEXT(list, info);
  621. }
  622. }
  623. mutex_exit(&(lock->mutex));
  624. lock = UT_LIST_GET_NEXT(list, lock);
  625. }
  626. fprintf(stderr, "Total number of rw-locks %ldn", count);
  627. mutex_exit(&rw_lock_list_mutex);
  628. }
  629. /*******************************************************************
  630. Prints debug info of an rw-lock. */
  631. void
  632. rw_lock_print(
  633. /*==========*/
  634. rw_lock_t* lock) /* in: rw-lock */
  635. {
  636. rw_lock_debug_t* info;
  637. fprintf(stderr,
  638. "-------------n"
  639. "RW-LATCH INFOn"
  640. "RW-LATCH: %p ", lock);
  641. if ((rw_lock_get_writer(lock) != RW_LOCK_NOT_LOCKED)
  642.     || (rw_lock_get_reader_count(lock) != 0)
  643.     || (rw_lock_get_waiters(lock) != 0)) {
  644. if (rw_lock_get_waiters(lock)) {
  645. fputs(" Waiters for the lock existn", stderr);
  646. } else {
  647. putc('n', stderr);
  648. }
  649.     
  650. info = UT_LIST_GET_FIRST(lock->debug_list);
  651. while (info != NULL) {
  652. rw_lock_debug_print(info);
  653. info = UT_LIST_GET_NEXT(list, info);
  654. }
  655. }
  656. }
  657. /*************************************************************************
  658. Prints info of a debug struct. */
  659. void
  660. rw_lock_debug_print(
  661. /*================*/
  662. rw_lock_debug_t* info) /* in: debug struct */
  663. {
  664. ulint rwt;
  665. rwt    = info->lock_type;
  666. fprintf(stderr, "Locked: thread %ld file %s line %ld  ",
  667. (ulong) os_thread_pf(info->thread_id), info->file_name,
  668.         (ulong) info->line);
  669. if (rwt == RW_LOCK_SHARED) {
  670. fputs("S-LOCK", stderr);
  671. } else if (rwt == RW_LOCK_EX) {
  672. fputs("X-LOCK", stderr);
  673. } else if (rwt == RW_LOCK_WAIT_EX) {
  674. fputs("WAIT X-LOCK", stderr);
  675. } else {
  676. ut_error;
  677. }
  678. if (info->pass != 0) {
  679. fprintf(stderr, " pass value %lu", (ulong) info->pass);
  680. }
  681. putc('n', stderr);
  682. }
  683. /*******************************************************************
  684. Returns the number of currently locked rw-locks. Works only in the debug
  685. version. */
  686. ulint
  687. rw_lock_n_locked(void)
  688. /*==================*/
  689. {
  690. rw_lock_t* lock;
  691. ulint count = 0;
  692. mutex_enter(&rw_lock_list_mutex);
  693. lock = UT_LIST_GET_FIRST(rw_lock_list);
  694. while (lock != NULL) {
  695. mutex_enter(rw_lock_get_mutex(lock));
  696. if ((rw_lock_get_writer(lock) != RW_LOCK_NOT_LOCKED)
  697. || (rw_lock_get_reader_count(lock) != 0)) {
  698. count++;
  699. }
  700. mutex_exit(rw_lock_get_mutex(lock));
  701. lock = UT_LIST_GET_NEXT(list, lock);
  702. }
  703. mutex_exit(&rw_lock_list_mutex);
  704. return(count);
  705. }
  706. #endif /* UNIV_SYNC_DEBUG */