WaitQueue.cpp 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/Debug.h>
  7. #include <Kernel/Thread.h>
  8. #include <Kernel/WaitQueue.h>
  9. namespace Kernel {
  10. bool WaitQueue::should_add_blocker(Thread::Blocker& b, void* data)
  11. {
  12. VERIFY(data != nullptr); // Thread that is requesting to be blocked
  13. VERIFY(m_lock.is_locked());
  14. VERIFY(b.blocker_type() == Thread::Blocker::Type::Queue);
  15. if (m_wake_requested || !m_should_block) {
  16. m_wake_requested = false;
  17. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: do not block thread {}, {}", this, data, m_should_block ? "wake was pending" : "not blocking");
  18. return false;
  19. }
  20. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: should block thread {}", this, data);
  21. return true;
  22. }
  23. u32 WaitQueue::wake_one()
  24. {
  25. u32 did_wake = 0;
  26. SpinlockLocker lock(m_lock);
  27. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_one", this);
  28. bool did_unblock_one = do_unblock([&](Thread::Blocker& b, void* data, bool& stop_iterating) {
  29. VERIFY(data);
  30. VERIFY(b.blocker_type() == Thread::Blocker::Type::Queue);
  31. auto& blocker = static_cast<Thread::QueueBlocker&>(b);
  32. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_one unblocking {}", this, data);
  33. if (blocker.unblock()) {
  34. stop_iterating = true;
  35. did_wake = 1;
  36. return true;
  37. }
  38. return false;
  39. });
  40. m_wake_requested = !did_unblock_one;
  41. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_one woke {} threads", this, did_wake);
  42. return did_wake;
  43. }
  44. u32 WaitQueue::wake_n(u32 wake_count)
  45. {
  46. if (wake_count == 0)
  47. return 0; // should we assert instead?
  48. SpinlockLocker lock(m_lock);
  49. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_n({})", this, wake_count);
  50. u32 did_wake = 0;
  51. bool did_unblock_some = do_unblock([&](Thread::Blocker& b, void* data, bool& stop_iterating) {
  52. VERIFY(data);
  53. VERIFY(b.blocker_type() == Thread::Blocker::Type::Queue);
  54. auto& blocker = static_cast<Thread::QueueBlocker&>(b);
  55. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_n unblocking {}", this, data);
  56. VERIFY(did_wake < wake_count);
  57. if (blocker.unblock()) {
  58. if (++did_wake >= wake_count)
  59. stop_iterating = true;
  60. return true;
  61. }
  62. return false;
  63. });
  64. m_wake_requested = !did_unblock_some;
  65. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_n({}) woke {} threads", this, wake_count, did_wake);
  66. return did_wake;
  67. }
  68. u32 WaitQueue::wake_all()
  69. {
  70. SpinlockLocker lock(m_lock);
  71. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_all", this);
  72. u32 did_wake = 0;
  73. bool did_unblock_any = do_unblock([&](Thread::Blocker& b, void* data, bool&) {
  74. VERIFY(data);
  75. VERIFY(b.blocker_type() == Thread::Blocker::Type::Queue);
  76. auto& blocker = static_cast<Thread::QueueBlocker&>(b);
  77. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_all unblocking {}", this, data);
  78. if (blocker.unblock()) {
  79. did_wake++;
  80. return true;
  81. }
  82. return false;
  83. });
  84. m_wake_requested = !did_unblock_any;
  85. dbgln_if(WAITQUEUE_DEBUG, "WaitQueue @ {}: wake_all woke {} threads", this, did_wake);
  86. return did_wake;
  87. }
  88. }