123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566 |
- /*
- * Copyright (c) 2021, the SerenityOS developers.
- * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include <Kernel/Arch/Processor.h>
- #include <Kernel/Process.h>
- #include <Kernel/Sections.h>
- #include <Kernel/WaitQueue.h>
- #include <Kernel/WorkQueue.h>
- namespace Kernel {
- WorkQueue* g_io_work;
- WorkQueue* g_ata_work;
- UNMAP_AFTER_INIT void WorkQueue::initialize()
- {
- g_io_work = new WorkQueue("IO WorkQueue Task"sv);
- g_ata_work = new WorkQueue("ATA WorkQueue Task"sv);
- }
- UNMAP_AFTER_INIT WorkQueue::WorkQueue(StringView name)
- {
- auto name_kstring = KString::try_create(name);
- if (name_kstring.is_error())
- TODO();
- auto [_, thread] = Process::create_kernel_process(name_kstring.release_value(), [this] {
- #if ARCH(AARCH64)
- // FIXME: This function expects to be executed with interrupts disabled, however on
- // aarch64 we spawn (kernel) threads with interrupts enabled, so we need to disable them.
- // This code should be written in a way that it is able to be executed with interrupts enabled.
- Processor::disable_interrupts();
- #endif
- for (;;) {
- WorkItem* item;
- bool have_more;
- m_items.with([&](auto& items) {
- item = items.take_first();
- have_more = !items.is_empty();
- });
- if (item) {
- item->function();
- delete item;
- if (have_more)
- continue;
- }
- [[maybe_unused]] auto result = m_wait_queue.wait_on({});
- }
- }).release_value_but_fixme_should_propagate_errors();
- m_thread = move(thread);
- }
- void WorkQueue::do_queue(WorkItem& item)
- {
- m_items.with([&](auto& items) {
- items.append(item);
- });
- m_wait_queue.wake_one();
- }
- }
|