| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <AK/Types.h>
 | 
					
						
							|  |  |  | #include <LibJS/Forward.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/Cell.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class HeapBlock { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     static constexpr size_t block_size = 16 * KB; | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     static NonnullOwnPtr<HeapBlock> create_with_cell_size(Heap&, size_t); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     void operator delete(void*); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     size_t cell_size() const { return m_cell_size; } | 
					
						
							|  |  |  |     size_t cell_count() const { return (block_size - sizeof(HeapBlock)) / m_cell_size; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Cell* allocate(); | 
					
						
							|  |  |  |     void deallocate(Cell*); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template<typename Callback> | 
					
						
							|  |  |  |     void for_each_cell(Callback callback) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (size_t i = 0; i < cell_count(); ++i) | 
					
						
							|  |  |  |             callback(cell(i)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     Heap& heap() { return m_heap; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 00:19:41 +02:00
										 |  |  |     static HeapBlock* from_cell(const Cell* cell) | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         return reinterpret_cast<HeapBlock*>((FlatPtr)cell & ~(block_size - 1)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  |     Cell* cell_from_possible_pointer(FlatPtr pointer) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (pointer < reinterpret_cast<FlatPtr>(m_storage)) | 
					
						
							|  |  |  |             return nullptr; | 
					
						
							|  |  |  |         size_t cell_index = (pointer - reinterpret_cast<FlatPtr>(m_storage)) / m_cell_size; | 
					
						
							|  |  |  |         return cell(cell_index); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     HeapBlock(Heap&, size_t cell_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |     struct FreelistEntry final : public Cell { | 
					
						
							|  |  |  |         FreelistEntry* next { nullptr }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         virtual const char* class_name() const override { return "FreelistEntry"; } | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-01 17:01:04 +03:00
										 |  |  |     Cell* cell(size_t index) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return reinterpret_cast<Cell*>(&m_storage[index * cell_size()]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FreelistEntry* init_freelist_entry(size_t index) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return new (&m_storage[index * cell_size()]) FreelistEntry(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     Heap& m_heap; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     size_t m_cell_size { 0 }; | 
					
						
							|  |  |  |     FreelistEntry* m_freelist { nullptr }; | 
					
						
							|  |  |  |     u8 m_storage[]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |