| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |  *    list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |  *    this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |  *    and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
					
						
							|  |  |  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
					
						
							|  |  |  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
					
						
							|  |  |  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
					
						
							|  |  |  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
					
						
							|  |  |  |  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  |  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
					
						
							|  |  |  |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <AK/Assertions.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  | #include <AK/NonnullOwnPtr.h>
 | 
					
						
							|  |  |  | #include <AK/kmalloc.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Heap/HeapBlock.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-21 13:12:16 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  | #include <sys/mman.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  | NonnullOwnPtr<HeapBlock> HeapBlock::create_with_cell_size(Heap& heap, size_t cell_size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-21 13:12:16 +01:00
										 |  |  |     char name[64]; | 
					
						
							|  |  |  |     snprintf(name, sizeof(name), "LibJS: HeapBlock(%zu)", cell_size); | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #ifdef __serenity__
 | 
					
						
							| 
									
										
										
										
											2021-02-12 19:08:20 +01:00
										 |  |  |     auto* block = (HeapBlock*)serenity_mmap(nullptr, block_size, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_RANDOMIZED | MAP_PRIVATE, 0, 0, block_size, name); | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     auto* block = (HeapBlock*)aligned_alloc(block_size, block_size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     ASSERT(block != MAP_FAILED); | 
					
						
							|  |  |  |     new (block) HeapBlock(heap, cell_size); | 
					
						
							|  |  |  |     return NonnullOwnPtr<HeapBlock>(NonnullOwnPtr<HeapBlock>::Adopt, *block); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HeapBlock::operator delete(void* ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #ifdef __serenity__
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     int rc = munmap(ptr, block_size); | 
					
						
							|  |  |  |     ASSERT(rc == 0); | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     free(ptr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | HeapBlock::HeapBlock(Heap& heap, size_t cell_size) | 
					
						
							|  |  |  |     : m_heap(heap) | 
					
						
							|  |  |  |     , m_cell_size(cell_size) | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |     ASSERT(cell_size >= sizeof(FreelistEntry)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FreelistEntry* next = nullptr; | 
					
						
							|  |  |  |     for (ssize_t i = cell_count() - 1; i >= 0; i--) { | 
					
						
							|  |  |  |         auto* freelist_entry = init_freelist_entry(i); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |         freelist_entry->set_live(false); | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |         freelist_entry->next = next; | 
					
						
							|  |  |  |         next = freelist_entry; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |     m_freelist = next; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void HeapBlock::deallocate(Cell* cell) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(cell->is_live()); | 
					
						
							|  |  |  |     ASSERT(!cell->is_marked()); | 
					
						
							|  |  |  |     cell->~Cell(); | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |     auto* freelist_entry = new (cell) FreelistEntry(); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     freelist_entry->set_live(false); | 
					
						
							|  |  |  |     freelist_entry->next = m_freelist; | 
					
						
							|  |  |  |     m_freelist = freelist_entry; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |