ReaderWriterLockSlimTest.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775
  1. //
  2. // ReaderWriterLockSlimTest.cs
  3. //
  4. // Authors:
  5. // Marek Safar ([email protected])
  6. //
  7. // Copyright (C) 2009 Novell, Inc (http://www.novell.com)
  8. //
  9. // Permission is hereby granted, free of charge, to any person obtaining
  10. // a copy of this software and associated documentation files (the
  11. // "Software"), to deal in the Software without restriction, including
  12. // without limitation the rights to use, copy, modify, merge, publish,
  13. // distribute, sublicense, and/or sell copies of the Software, and to
  14. // permit persons to whom the Software is furnished to do so, subject to
  15. // the following conditions:
  16. //
  17. // The above copyright notice and this permission notice shall be
  18. // included in all copies or substantial portions of the Software.
  19. //
  20. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  21. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  22. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  23. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  24. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  25. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  26. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  27. //
  28. using System;
  29. using NUnit.Framework;
  30. using System.Threading;
  31. using System.Linq;
  32. using System.Threading.Tasks;
  33. namespace MonoTests.System.Threading
  34. {
  35. [TestFixture]
  36. public class ReaderWriterLockSlimTests
  37. {
  38. [Test]
  39. public void DefaultValues ()
  40. {
  41. var v = new ReaderWriterLockSlim ();
  42. Assert.AreEqual (0, v.CurrentReadCount, "1");
  43. Assert.AreEqual (false, v.IsReadLockHeld, "2");
  44. Assert.AreEqual (false, v.IsUpgradeableReadLockHeld, "3");
  45. Assert.AreEqual (false, v.IsWriteLockHeld, "4");
  46. Assert.AreEqual (LockRecursionPolicy.NoRecursion, v.RecursionPolicy, "5");
  47. Assert.AreEqual (0, v.RecursiveReadCount, "6");
  48. Assert.AreEqual (0, v.RecursiveUpgradeCount, "7");
  49. Assert.AreEqual (0, v.RecursiveWriteCount, "8");
  50. Assert.AreEqual (0, v.WaitingReadCount, "9");
  51. Assert.AreEqual (0, v.WaitingUpgradeCount, "10");
  52. Assert.AreEqual (0, v.WaitingWriteCount, "11");
  53. }
  54. [Test]
  55. public void Dispose_Errors ()
  56. {
  57. var v = new ReaderWriterLockSlim ();
  58. v.Dispose ();
  59. try {
  60. v.EnterUpgradeableReadLock ();
  61. Assert.Fail ("1");
  62. } catch (ObjectDisposedException) {
  63. }
  64. try {
  65. v.EnterReadLock ();
  66. Assert.Fail ("2");
  67. } catch (ObjectDisposedException) {
  68. }
  69. try {
  70. v.EnterWriteLock ();
  71. Assert.Fail ("3");
  72. } catch (ObjectDisposedException) {
  73. }
  74. }
  75. [Test]
  76. public void Dispose_WithReadLock ()
  77. {
  78. var rwl = new ReaderWriterLockSlim ();
  79. rwl.EnterReadLock ();
  80. try {
  81. rwl.Dispose ();
  82. Assert.Fail ("1");
  83. } catch (SynchronizationLockException) {
  84. }
  85. }
  86. [Test]
  87. public void Dispose_WithWriteLock ()
  88. {
  89. var rwl = new ReaderWriterLockSlim ();
  90. rwl.EnterWriteLock ();
  91. try {
  92. rwl.Dispose ();
  93. Assert.Fail ("1");
  94. } catch (SynchronizationLockException) {
  95. }
  96. }
  97. [Test]
  98. public void Dispose_UpgradeableReadLock ()
  99. {
  100. var rwl = new ReaderWriterLockSlim ();
  101. rwl.EnterUpgradeableReadLock ();
  102. try {
  103. rwl.Dispose ();
  104. Assert.Fail ("1");
  105. } catch (SynchronizationLockException) {
  106. }
  107. }
  108. [Test]
  109. public void TryEnterReadLock_OutOfRange ()
  110. {
  111. var v = new ReaderWriterLockSlim ();
  112. try {
  113. v.TryEnterReadLock (-2);
  114. Assert.Fail ("1");
  115. } catch (ArgumentOutOfRangeException) {
  116. }
  117. try {
  118. v.TryEnterReadLock (TimeSpan.MaxValue);
  119. Assert.Fail ("2");
  120. } catch (ArgumentOutOfRangeException) {
  121. }
  122. try {
  123. v.TryEnterReadLock (TimeSpan.MinValue);
  124. Assert.Fail ("3");
  125. } catch (ArgumentOutOfRangeException) {
  126. }
  127. }
  128. [Test]
  129. public void TryEnterUpgradeableReadLock_OutOfRange ()
  130. {
  131. var v = new ReaderWriterLockSlim ();
  132. try {
  133. v.TryEnterUpgradeableReadLock (-2);
  134. Assert.Fail ("1");
  135. } catch (ArgumentOutOfRangeException) {
  136. }
  137. try {
  138. v.TryEnterUpgradeableReadLock (TimeSpan.MaxValue);
  139. Assert.Fail ("2");
  140. } catch (ArgumentOutOfRangeException) {
  141. }
  142. try {
  143. v.TryEnterUpgradeableReadLock (TimeSpan.MinValue);
  144. Assert.Fail ("3");
  145. } catch (ArgumentOutOfRangeException) {
  146. }
  147. }
  148. [Test]
  149. public void TryEnterWriteLock_OutOfRange ()
  150. {
  151. var v = new ReaderWriterLockSlim ();
  152. try {
  153. v.TryEnterWriteLock (-2);
  154. Assert.Fail ("1");
  155. } catch (ArgumentOutOfRangeException) {
  156. }
  157. try {
  158. v.TryEnterWriteLock (TimeSpan.MaxValue);
  159. Assert.Fail ("2");
  160. } catch (ArgumentOutOfRangeException) {
  161. }
  162. try {
  163. v.TryEnterWriteLock (TimeSpan.MinValue);
  164. Assert.Fail ("3");
  165. } catch (ArgumentOutOfRangeException) {
  166. }
  167. }
  168. [Test, ExpectedException (typeof (SynchronizationLockException))]
  169. public void ExitReadLock ()
  170. {
  171. var v = new ReaderWriterLockSlim ();
  172. v.ExitReadLock ();
  173. }
  174. [Test, ExpectedException (typeof (SynchronizationLockException))]
  175. public void ExitWriteLock ()
  176. {
  177. var v = new ReaderWriterLockSlim ();
  178. v.ExitWriteLock ();
  179. }
  180. [Test]
  181. public void EnterReadLock_NoRecursionError ()
  182. {
  183. var v = new ReaderWriterLockSlim ();
  184. v.EnterReadLock ();
  185. Assert.AreEqual (1, v.RecursiveReadCount);
  186. try {
  187. v.EnterReadLock ();
  188. Assert.Fail ("1");
  189. } catch (LockRecursionException) {
  190. }
  191. try {
  192. v.EnterWriteLock ();
  193. Assert.Fail ("2");
  194. } catch (LockRecursionException) {
  195. }
  196. }
  197. [Test]
  198. public void EnterReadLock ()
  199. {
  200. var v = new ReaderWriterLockSlim ();
  201. v.EnterReadLock ();
  202. Assert.IsTrue (v.IsReadLockHeld, "A");
  203. Assert.AreEqual (0, v.RecursiveWriteCount, "A1");
  204. Assert.AreEqual (1, v.RecursiveReadCount, "A2");
  205. Assert.AreEqual (0, v.RecursiveUpgradeCount, "A3");
  206. Assert.AreEqual (0, v.WaitingReadCount, "A4");
  207. Assert.AreEqual (0, v.WaitingUpgradeCount, "A5");
  208. Assert.AreEqual (0, v.WaitingWriteCount, "A6");
  209. v.ExitReadLock ();
  210. v.EnterReadLock ();
  211. Assert.IsTrue (v.IsReadLockHeld, "B");
  212. Assert.AreEqual (0, v.RecursiveWriteCount, "B1");
  213. Assert.AreEqual (1, v.RecursiveReadCount, "B2");
  214. Assert.AreEqual (0, v.RecursiveUpgradeCount, "B3");
  215. Assert.AreEqual (0, v.WaitingReadCount, "B4");
  216. Assert.AreEqual (0, v.WaitingUpgradeCount, "B5");
  217. Assert.AreEqual (0, v.WaitingWriteCount, "B6");
  218. v.ExitReadLock ();
  219. }
  220. [Test]
  221. public void EnterWriteLock_NoRecursionError ()
  222. {
  223. var v = new ReaderWriterLockSlim ();
  224. v.EnterWriteLock ();
  225. Assert.AreEqual (1, v.RecursiveWriteCount);
  226. try {
  227. v.EnterWriteLock ();
  228. Assert.Fail ("1");
  229. } catch (LockRecursionException) {
  230. }
  231. try {
  232. v.EnterReadLock ();
  233. Assert.Fail ("2");
  234. } catch (LockRecursionException) {
  235. }
  236. }
  237. [Test]
  238. public void EnterWriteLock ()
  239. {
  240. var v = new ReaderWriterLockSlim ();
  241. v.EnterWriteLock ();
  242. Assert.IsTrue (v.IsWriteLockHeld, "A");
  243. Assert.AreEqual (1, v.RecursiveWriteCount, "A1");
  244. Assert.AreEqual (0, v.RecursiveReadCount, "A2");
  245. Assert.AreEqual (0, v.RecursiveUpgradeCount, "A3");
  246. Assert.AreEqual (0, v.WaitingReadCount, "A4");
  247. Assert.AreEqual (0, v.WaitingUpgradeCount, "A5");
  248. Assert.AreEqual (0, v.WaitingWriteCount, "A6");
  249. v.ExitWriteLock ();
  250. v.EnterWriteLock ();
  251. Assert.IsTrue (v.IsWriteLockHeld, "B");
  252. Assert.AreEqual (1, v.RecursiveWriteCount, "B1");
  253. Assert.AreEqual (0, v.RecursiveReadCount, "B2");
  254. Assert.AreEqual (0, v.RecursiveUpgradeCount, "B3");
  255. Assert.AreEqual (0, v.WaitingReadCount, "B4");
  256. Assert.AreEqual (0, v.WaitingUpgradeCount, "B5");
  257. Assert.AreEqual (0, v.WaitingWriteCount, "B6");
  258. v.ExitWriteLock ();
  259. }
  260. [Test]
  261. public void EnterUpgradeableReadLock_NoRecursionError ()
  262. {
  263. var v = new ReaderWriterLockSlim ();
  264. v.EnterUpgradeableReadLock ();
  265. Assert.AreEqual (1, v.RecursiveUpgradeCount);
  266. try {
  267. v.EnterUpgradeableReadLock ();
  268. Assert.Fail ("2");
  269. } catch (LockRecursionException) {
  270. }
  271. }
  272. [Test]
  273. public void EnterUpgradeableReadLock ()
  274. {
  275. var v = new ReaderWriterLockSlim ();
  276. v.EnterUpgradeableReadLock ();
  277. Assert.IsTrue (v.IsUpgradeableReadLockHeld, "A");
  278. Assert.AreEqual (0, v.RecursiveWriteCount, "A1");
  279. Assert.AreEqual (0, v.RecursiveReadCount, "A2");
  280. Assert.AreEqual (1, v.RecursiveUpgradeCount, "A3");
  281. Assert.AreEqual (0, v.WaitingReadCount, "A4");
  282. Assert.AreEqual (0, v.WaitingUpgradeCount, "A5");
  283. Assert.AreEqual (0, v.WaitingWriteCount, "A6");
  284. v.ExitUpgradeableReadLock ();
  285. v.EnterUpgradeableReadLock ();
  286. Assert.IsTrue (v.IsUpgradeableReadLockHeld, "B");
  287. Assert.AreEqual (0, v.RecursiveWriteCount, "B1");
  288. Assert.AreEqual (0, v.RecursiveReadCount, "B2");
  289. Assert.AreEqual (1, v.RecursiveUpgradeCount, "B3");
  290. Assert.AreEqual (0, v.WaitingReadCount, "B4");
  291. Assert.AreEqual (0, v.WaitingUpgradeCount, "B5");
  292. Assert.AreEqual (0, v.WaitingWriteCount, "B6");
  293. v.EnterReadLock ();
  294. v.ExitUpgradeableReadLock ();
  295. Assert.IsTrue (v.IsReadLockHeld, "C");
  296. Assert.AreEqual (0, v.RecursiveWriteCount, "C1");
  297. Assert.AreEqual (1, v.RecursiveReadCount, "C2");
  298. Assert.AreEqual (0, v.RecursiveUpgradeCount, "C3");
  299. Assert.AreEqual (0, v.WaitingReadCount, "C4");
  300. Assert.AreEqual (0, v.WaitingUpgradeCount, "C5");
  301. Assert.AreEqual (0, v.WaitingWriteCount, "C6");
  302. v.ExitReadLock ();
  303. }
  304. [Test]
  305. [Category ("MultiThreaded")]
  306. public void EnterReadLock_MultiRead ()
  307. {
  308. var v = new ReaderWriterLockSlim ();
  309. int local = 10;
  310. var r = from i in Enumerable.Range (1, 30) select new Thread (() => {
  311. // Just to cause some contention
  312. Thread.Sleep (100);
  313. v.EnterReadLock ();
  314. Assert.AreEqual (10, local);
  315. });
  316. var threads = r.ToList ();
  317. foreach (var t in threads) {
  318. t.Start ();
  319. }
  320. foreach (var t in threads) {
  321. // Console.WriteLine (t.ThreadState);
  322. t.Join ();
  323. }
  324. }
  325. [Test]
  326. [Category ("MultiThreaded")]
  327. public void TryEnterWriteLock_WhileReading ()
  328. {
  329. var v = new ReaderWriterLockSlim ();
  330. AutoResetEvent ev = new AutoResetEvent (false);
  331. AutoResetEvent ev2 = new AutoResetEvent (false);
  332. Thread t1 = new Thread (() => {
  333. v.EnterReadLock ();
  334. ev2.Set ();
  335. ev.WaitOne ();
  336. v.ExitReadLock ();
  337. });
  338. t1.Start ();
  339. ev2.WaitOne ();
  340. Assert.IsFalse (v.TryEnterWriteLock (100));
  341. ev.Set ();
  342. t1.Join ();
  343. Assert.IsTrue (v.TryEnterWriteLock (100));
  344. }
  345. [Test]
  346. [Category ("MultiThreaded")]
  347. public void EnterWriteLock_MultiRead ()
  348. {
  349. var v = new ReaderWriterLockSlim ();
  350. int local = 10;
  351. int ready_count = 0;
  352. int entered_count = 0;
  353. const int thread_count = 10;
  354. var r = from i in Enumerable.Range (1, thread_count) select new Thread (() => {
  355. Interlocked.Increment (ref ready_count);
  356. v.EnterReadLock ();
  357. Interlocked.Increment (ref entered_count);
  358. Assert.AreEqual (11, local);
  359. });
  360. v.EnterWriteLock ();
  361. var threads = r.ToList ();
  362. foreach (var t in threads) {
  363. t.Start ();
  364. }
  365. while (ready_count != thread_count)
  366. Thread.Sleep (10);
  367. /* Extra up to 2s of sleep to ensure all threads got the chance to enter the lock */
  368. for (int i = 0; i < 200 && v.WaitingReadCount != thread_count; ++i)
  369. Thread.Sleep (10);
  370. local = 11;
  371. Assert.AreEqual (0, v.WaitingWriteCount, "in waiting write");
  372. Assert.AreEqual (thread_count, v.WaitingReadCount, "in waiting read");
  373. Assert.AreEqual (0, v.WaitingUpgradeCount, "in waiting upgrade");
  374. v.ExitWriteLock ();
  375. foreach (var t in threads) {
  376. // Console.WriteLine (t.ThreadState);
  377. t.Join ();
  378. }
  379. }
  380. [Test]
  381. public void EnterWriteLock_After_ExitUpgradeableReadLock ()
  382. {
  383. var v = new ReaderWriterLockSlim ();
  384. v.EnterUpgradeableReadLock ();
  385. Assert.IsTrue (v.TryEnterWriteLock (100));
  386. v.ExitWriteLock ();
  387. v.ExitUpgradeableReadLock ();
  388. Assert.IsTrue (v.TryEnterWriteLock (100));
  389. v.ExitWriteLock ();
  390. }
  391. [Test]
  392. [Category ("MultiThreaded")]
  393. public void EnterWriteLockWhileInUpgradeAndOtherWaiting ()
  394. {
  395. var v = new ReaderWriterLockSlim ();
  396. var task2 = new Task(() => {
  397. v.EnterWriteLock();
  398. v.ExitWriteLock();
  399. });
  400. var task1 = new Task(() =>
  401. {
  402. v.EnterUpgradeableReadLock ();
  403. task2.Start ();
  404. Thread.Sleep (100);
  405. v.EnterWriteLock ();
  406. v.ExitWriteLock ();
  407. v.ExitUpgradeableReadLock ();
  408. });
  409. task1.Start ();
  410. Assert.IsTrue (task1.Wait (500));
  411. }
  412. [Test]
  413. public void RecursiveReadLockTest ()
  414. {
  415. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  416. Assert.IsTrue (v.TryEnterReadLock (100), "#1");
  417. Assert.IsTrue (v.TryEnterReadLock (100), "#2");
  418. Assert.IsTrue (v.TryEnterReadLock (100), "#3");
  419. Assert.AreEqual (3, v.RecursiveReadCount);
  420. }
  421. [Test]
  422. public void RecursiveReadPlusWriteLockTest ()
  423. {
  424. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  425. try {
  426. v.EnterReadLock ();
  427. v.EnterWriteLock ();
  428. Assert.Fail ("1");
  429. } catch (LockRecursionException ex) {
  430. Assert.IsNotNull (ex, "#1");
  431. }
  432. }
  433. [Test]
  434. public void RecursiveReadPlusUpgradeableLockTest ()
  435. {
  436. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  437. try {
  438. v.EnterReadLock ();
  439. v.EnterUpgradeableReadLock ();
  440. Assert.Fail ("1");
  441. } catch (LockRecursionException ex) {
  442. Assert.IsNotNull (ex, "#1");
  443. }
  444. }
  445. [Test]
  446. public void RecursiveWriteLockTest ()
  447. {
  448. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  449. Assert.IsTrue (v.TryEnterWriteLock (100), "#1");
  450. Assert.IsTrue (v.TryEnterWriteLock (100), "#2");
  451. Assert.IsTrue (v.TryEnterWriteLock (100), "#3");
  452. Assert.AreEqual (3, v.RecursiveWriteCount);
  453. }
  454. [Test]
  455. public void RecursiveWritePlusReadLockTest ()
  456. {
  457. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  458. Assert.IsTrue (v.TryEnterWriteLock (100), "#1");
  459. Assert.AreEqual (1, v.RecursiveWriteCount, "1b");
  460. Assert.AreEqual (0, v.RecursiveReadCount, "1c");
  461. Assert.IsTrue (v.TryEnterReadLock (100), "#2");
  462. Assert.AreEqual (1, v.RecursiveWriteCount, "2b");
  463. Assert.AreEqual (1, v.RecursiveReadCount, "2c");
  464. Assert.IsTrue (v.TryEnterReadLock (100), "#3");
  465. Assert.AreEqual (1, v.RecursiveWriteCount, "3b");
  466. Assert.AreEqual (2, v.RecursiveReadCount, "3c");
  467. v.ExitReadLock ();
  468. Assert.AreEqual (1, v.RecursiveWriteCount, "4b");
  469. Assert.AreEqual (1, v.RecursiveReadCount, "4c");
  470. }
  471. [Test]
  472. public void RecursiveUpgradeableReadLockTest ()
  473. {
  474. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  475. Assert.IsTrue (v.TryEnterUpgradeableReadLock (100), "#1");
  476. Assert.IsTrue (v.TryEnterUpgradeableReadLock (100), "#2");
  477. Assert.IsTrue (v.TryEnterUpgradeableReadLock (100), "#3");
  478. Assert.AreEqual (3, v.RecursiveUpgradeCount);
  479. }
  480. [Test]
  481. [Category ("MultiThreaded")]
  482. public void RecursiveReadPropertiesTest ()
  483. {
  484. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  485. v.EnterReadLock ();
  486. v.EnterReadLock ();
  487. Assert.AreEqual (true, v.IsReadLockHeld, "#1a");
  488. Assert.AreEqual (1, v.CurrentReadCount, "#2a");
  489. Assert.AreEqual (2, v.RecursiveReadCount, "#3a");
  490. bool rLock = true;
  491. int cReadCount = -1, rReadCount = -1;
  492. Thread t = new Thread ((_) => {
  493. rLock = v.IsReadLockHeld;
  494. cReadCount = v.CurrentReadCount;
  495. rReadCount = v.RecursiveReadCount;
  496. });
  497. t.Start ();
  498. t.Join ();
  499. Assert.AreEqual (false, rLock, "#1b");
  500. Assert.AreEqual (1, cReadCount, "#2b");
  501. Assert.AreEqual (0, rReadCount, "#3b");
  502. }
  503. [Test]
  504. [Category ("MultiThreaded")]
  505. public void RecursiveUpgradePropertiesTest ()
  506. {
  507. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  508. v.EnterUpgradeableReadLock ();
  509. v.EnterUpgradeableReadLock ();
  510. Assert.AreEqual (true, v.IsUpgradeableReadLockHeld, "#1a");
  511. Assert.AreEqual (false, v.IsReadLockHeld, "#11a");
  512. Assert.AreEqual (0, v.CurrentReadCount, "#2a");
  513. Assert.AreEqual (2, v.RecursiveUpgradeCount, "#3a");
  514. bool upLock = false, rLock = false;
  515. int rCount = -1, rUCount = -1;
  516. Thread t = new Thread ((_) => {
  517. upLock = v.IsUpgradeableReadLockHeld;
  518. rLock = v.IsReadLockHeld;
  519. rCount = v.CurrentReadCount;
  520. rUCount = v.RecursiveUpgradeCount;
  521. });
  522. t.Start ();
  523. t.Join ();
  524. Assert.AreEqual (false, upLock, "#1b");
  525. Assert.AreEqual (false, rLock, "#11b");
  526. Assert.AreEqual (0, rCount, "#2b");
  527. Assert.AreEqual (0, rUCount, "#3b");
  528. }
  529. [Test]
  530. [Category ("MultiThreaded")]
  531. public void RecursiveWritePropertiesTest ()
  532. {
  533. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  534. v.EnterWriteLock ();
  535. v.EnterWriteLock ();
  536. Assert.AreEqual (true, v.IsWriteLockHeld, "#1a");
  537. Assert.AreEqual (2, v.RecursiveWriteCount, "#3a");
  538. bool wLock = false;
  539. int rWrite = -1;
  540. Thread t = new Thread ((_) => {
  541. wLock = v.IsWriteLockHeld;
  542. rWrite = v.RecursiveWriteCount;
  543. });
  544. t.Start ();
  545. t.Join ();
  546. Assert.AreEqual (false, wLock, "#1b");
  547. Assert.AreEqual (0, rWrite, "#3b");
  548. }
  549. [Test]
  550. public void RecursiveEnterExitReadTest ()
  551. {
  552. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  553. v.EnterReadLock ();
  554. v.EnterReadLock ();
  555. v.EnterReadLock ();
  556. Assert.IsTrue (v.IsReadLockHeld);
  557. Assert.AreEqual (3, v.RecursiveReadCount);
  558. v.ExitReadLock ();
  559. Assert.IsTrue (v.IsReadLockHeld);
  560. Assert.AreEqual (2, v.RecursiveReadCount);
  561. }
  562. [Test]
  563. public void RecursiveEnterExitWriteTest ()
  564. {
  565. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  566. v.EnterWriteLock ();
  567. v.EnterWriteLock ();
  568. v.EnterWriteLock ();
  569. Assert.IsTrue (v.IsWriteLockHeld);
  570. Assert.AreEqual (3, v.RecursiveWriteCount);
  571. v.ExitWriteLock ();
  572. v.ExitWriteLock ();
  573. Assert.IsTrue (v.IsWriteLockHeld);
  574. Assert.AreEqual (1, v.RecursiveWriteCount);
  575. }
  576. [Test]
  577. public void RecursiveEnterExitUpgradableTest ()
  578. {
  579. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  580. v.EnterUpgradeableReadLock ();
  581. v.EnterUpgradeableReadLock ();
  582. v.EnterUpgradeableReadLock ();
  583. Assert.IsTrue (v.IsUpgradeableReadLockHeld);
  584. Assert.AreEqual (3, v.RecursiveUpgradeCount);
  585. v.ExitUpgradeableReadLock ();
  586. Assert.IsTrue (v.IsUpgradeableReadLockHeld);
  587. Assert.AreEqual (2, v.RecursiveUpgradeCount);
  588. }
  589. [Test]
  590. public void RecursiveWriteUpgradeReadTest ()
  591. {
  592. var rwlock = new ReaderWriterLockSlim(LockRecursionPolicy.SupportsRecursion);
  593. rwlock.EnterWriteLock ();
  594. Assert.IsTrue (rwlock.IsWriteLockHeld);
  595. rwlock.EnterUpgradeableReadLock ();
  596. Assert.IsTrue (rwlock.IsUpgradeableReadLockHeld);
  597. rwlock.EnterReadLock ();
  598. Assert.IsTrue (rwlock.IsReadLockHeld);
  599. rwlock.ExitUpgradeableReadLock();
  600. Assert.IsFalse (rwlock.IsUpgradeableReadLockHeld);
  601. Assert.IsTrue (rwlock.IsReadLockHeld);
  602. Assert.IsTrue (rwlock.IsWriteLockHeld);
  603. rwlock.ExitReadLock ();
  604. Assert.IsTrue (rwlock.IsWriteLockHeld);
  605. }
  606. [Test]
  607. public void RecursiveWriteUpgradeTest ()
  608. {
  609. ReaderWriterLockSlim rwlock = new ReaderWriterLockSlim(LockRecursionPolicy.SupportsRecursion);
  610. rwlock.EnterWriteLock ();
  611. Assert.IsTrue (rwlock.IsWriteLockHeld);
  612. rwlock.EnterUpgradeableReadLock ();
  613. Assert.IsTrue (rwlock.IsUpgradeableReadLockHeld);
  614. rwlock.ExitUpgradeableReadLock ();
  615. Assert.IsFalse (rwlock.IsUpgradeableReadLockHeld);
  616. Assert.IsTrue (rwlock.IsWriteLockHeld);
  617. rwlock.ExitWriteLock ();
  618. Assert.IsFalse (rwlock.IsWriteLockHeld);
  619. rwlock.EnterWriteLock ();
  620. Assert.IsTrue (rwlock.IsWriteLockHeld);
  621. }
  622. [Test]
  623. [Category ("MultiThreaded")]
  624. public void RecursiveWriteReadAcquisitionInterleaving ()
  625. {
  626. var v = new ReaderWriterLockSlim (LockRecursionPolicy.SupportsRecursion);
  627. v.EnterWriteLock ();
  628. Assert.IsTrue (v.IsWriteLockHeld, "#1");
  629. bool result = true;
  630. var t = new Thread (delegate () {
  631. result = v.TryEnterReadLock (100);
  632. });
  633. t.Start ();
  634. t.Join ();
  635. Assert.IsFalse (result, "#2");
  636. v.ExitWriteLock ();
  637. t = new Thread (delegate () {
  638. result = v.TryEnterReadLock (100);
  639. });
  640. t.Start ();
  641. t.Join ();
  642. Assert.IsTrue (result, "#3");
  643. }
  644. }
  645. }