mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-28 11:02:36 +00:00 
			
		
		
		
	 828441852f
			
		
	
	
		828441852f
		
	
	
	
	
		
			
			Now that we have OS macros for essentially every supported OS, let's try to use them everywhere.
		
			
				
	
	
		
			80 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			80 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/Platform.h>
 | |
| #include <AK/Random.h>
 | |
| #include <AK/Vector.h>
 | |
| #include <LibJS/Heap/BlockAllocator.h>
 | |
| #include <LibJS/Heap/HeapBlock.h>
 | |
| #include <sys/mman.h>
 | |
| 
 | |
| #ifdef HAS_ADDRESS_SANITIZER
 | |
| #    include <sanitizer/asan_interface.h>
 | |
| #endif
 | |
| 
 | |
| namespace JS {
 | |
| 
 | |
| BlockAllocator::~BlockAllocator()
 | |
| {
 | |
|     for (auto* block : m_blocks) {
 | |
|         ASAN_UNPOISON_MEMORY_REGION(block, HeapBlock::block_size);
 | |
| #ifdef AK_OS_SERENITY
 | |
|         if (munmap(block, HeapBlock::block_size) < 0) {
 | |
|             perror("munmap");
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| #else
 | |
|         free(block);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| void* BlockAllocator::allocate_block([[maybe_unused]] char const* name)
 | |
| {
 | |
|     if (!m_blocks.is_empty()) {
 | |
|         // To reduce predictability, take a random block from the cache.
 | |
|         size_t random_index = get_random_uniform(m_blocks.size());
 | |
|         auto* block = m_blocks.unstable_take(random_index);
 | |
|         ASAN_UNPOISON_MEMORY_REGION(block, HeapBlock::block_size);
 | |
| #ifdef AK_OS_SERENITY
 | |
|         if (set_mmap_name(block, HeapBlock::block_size, name) < 0) {
 | |
|             perror("set_mmap_name");
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| #endif
 | |
|         return block;
 | |
|     }
 | |
| 
 | |
| #ifdef AK_OS_SERENITY
 | |
|     auto* block = (HeapBlock*)serenity_mmap(nullptr, HeapBlock::block_size, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_RANDOMIZED | MAP_PRIVATE, 0, 0, HeapBlock::block_size, name);
 | |
|     VERIFY(block != MAP_FAILED);
 | |
| #else
 | |
|     auto* block = (HeapBlock*)aligned_alloc(HeapBlock::block_size, HeapBlock::block_size);
 | |
|     VERIFY(block);
 | |
| #endif
 | |
|     return block;
 | |
| }
 | |
| 
 | |
| void BlockAllocator::deallocate_block(void* block)
 | |
| {
 | |
|     VERIFY(block);
 | |
|     if (m_blocks.size() >= max_cached_blocks) {
 | |
| #ifdef AK_OS_SERENITY
 | |
|         if (munmap(block, HeapBlock::block_size) < 0) {
 | |
|             perror("munmap");
 | |
|             VERIFY_NOT_REACHED();
 | |
|         }
 | |
| #else
 | |
|         free(block);
 | |
| #endif
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     ASAN_POISON_MEMORY_REGION(block, HeapBlock::block_size);
 | |
|     m_blocks.append(block);
 | |
| }
 | |
| 
 | |
| }
 |