mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-31 02:52:43 +00:00 
			
		
		
		
	 9201a06027
			
		
	
	
		9201a06027
		
	
	
	
	
		
			
			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;
 | |
| }
 | |
| 
 | |
| }
 |