TaskQueue.cpp 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. /*
  2. * Copyright (c) 2021-2024, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Heap/MarkedVector.h>
  7. #include <LibWeb/HTML/EventLoop/EventLoop.h>
  8. #include <LibWeb/HTML/EventLoop/TaskQueue.h>
  9. namespace Web::HTML {
  10. JS_DEFINE_ALLOCATOR(TaskQueue);
  11. TaskQueue::TaskQueue(HTML::EventLoop& event_loop)
  12. : m_event_loop(event_loop)
  13. {
  14. }
  15. TaskQueue::~TaskQueue() = default;
  16. void TaskQueue::visit_edges(Visitor& visitor)
  17. {
  18. Base::visit_edges(visitor);
  19. visitor.visit(m_event_loop);
  20. for (auto& task : m_tasks)
  21. visitor.visit(task);
  22. }
  23. void TaskQueue::add(JS::NonnullGCPtr<Task> task)
  24. {
  25. m_tasks.append(task);
  26. m_event_loop->schedule();
  27. }
  28. JS::GCPtr<Task> TaskQueue::take_first_runnable()
  29. {
  30. if (m_event_loop->execution_paused())
  31. return nullptr;
  32. for (size_t i = 0; i < m_tasks.size(); ++i) {
  33. auto const& task = m_tasks[i];
  34. if (m_event_loop->is_task_source_blocked(task->source()))
  35. continue;
  36. if (task->is_runnable())
  37. return m_tasks.take(i);
  38. }
  39. return nullptr;
  40. }
  41. bool TaskQueue::has_runnable_tasks() const
  42. {
  43. if (m_event_loop->execution_paused())
  44. return false;
  45. for (auto& task : m_tasks) {
  46. if (m_event_loop->is_task_source_blocked(task->source()))
  47. continue;
  48. if (task->is_runnable())
  49. return true;
  50. }
  51. return false;
  52. }
  53. void TaskQueue::remove_tasks_matching(Function<bool(HTML::Task const&)> filter)
  54. {
  55. m_tasks.remove_all_matching([&](auto& task) {
  56. return filter(*task);
  57. });
  58. }
  59. JS::MarkedVector<JS::NonnullGCPtr<Task>> TaskQueue::take_tasks_matching(Function<bool(HTML::Task const&)> filter)
  60. {
  61. JS::MarkedVector<JS::NonnullGCPtr<Task>> matching_tasks(heap());
  62. for (size_t i = 0; i < m_tasks.size();) {
  63. auto& task = m_tasks.at(i);
  64. if (filter(*task)) {
  65. matching_tasks.append(task);
  66. m_tasks.remove(i);
  67. } else {
  68. ++i;
  69. }
  70. }
  71. return matching_tasks;
  72. }
  73. Task const* TaskQueue::last_added_task() const
  74. {
  75. if (m_tasks.is_empty())
  76. return nullptr;
  77. return m_tasks.last();
  78. }
  79. }