mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 15:48:12 +00:00

Before we start disabling acquisition of the big process lock for specific syscalls, make sure to document and assert that all the lock is held during all syscalls.
74 lines
2.5 KiB
C++
74 lines
2.5 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/NonnullRefPtrVector.h>
|
|
#include <Kernel/Arch/x86/InterruptDisabler.h>
|
|
#include <Kernel/Process.h>
|
|
#include <Kernel/VM/AnonymousVMObject.h>
|
|
#include <Kernel/VM/InodeVMObject.h>
|
|
#include <Kernel/VM/MemoryManager.h>
|
|
|
|
namespace Kernel {
|
|
|
|
KResultOr<FlatPtr> Process::sys$purge(int mode)
|
|
{
|
|
VERIFY_PROCESS_BIG_LOCK_ACQUIRED(this)
|
|
REQUIRE_NO_PROMISES;
|
|
if (!is_superuser())
|
|
return EPERM;
|
|
int purged_page_count = 0;
|
|
if (mode & PURGE_ALL_VOLATILE) {
|
|
NonnullRefPtrVector<AnonymousVMObject> vmobjects;
|
|
{
|
|
KResult result(KSuccess);
|
|
InterruptDisabler disabler;
|
|
MM.for_each_vmobject([&](auto& vmobject) {
|
|
if (vmobject.is_anonymous()) {
|
|
// In the event that the append fails, only attempt to continue
|
|
// the purge if we have already appended something successfully.
|
|
if (!vmobjects.try_append(vmobject) && vmobjects.is_empty()) {
|
|
result = ENOMEM;
|
|
return IterationDecision::Break;
|
|
}
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
if (result.is_error())
|
|
return result.error();
|
|
}
|
|
for (auto& vmobject : vmobjects) {
|
|
purged_page_count += vmobject.purge();
|
|
}
|
|
}
|
|
if (mode & PURGE_ALL_CLEAN_INODE) {
|
|
NonnullRefPtrVector<InodeVMObject> vmobjects;
|
|
{
|
|
KResult result(KSuccess);
|
|
InterruptDisabler disabler;
|
|
MM.for_each_vmobject([&](auto& vmobject) {
|
|
if (vmobject.is_inode()) {
|
|
// In the event that the append fails, only attempt to continue
|
|
// the purge if we have already appended something successfully.
|
|
if (!vmobjects.try_append(static_cast<InodeVMObject&>(vmobject)) && vmobjects.is_empty()) {
|
|
result = ENOMEM;
|
|
return IterationDecision::Break;
|
|
}
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
if (result.is_error())
|
|
return result.error();
|
|
}
|
|
for (auto& vmobject : vmobjects) {
|
|
purged_page_count += vmobject.release_all_clean_pages();
|
|
}
|
|
}
|
|
return purged_page_count;
|
|
}
|
|
|
|
}
|