| 
									
										
										
										
											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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:29:22 +01:00
										 |  |  | #include <AK/Badge.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #include <AK/HashTable.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-18 20:03:17 +01:00
										 |  |  | #include <LibJS/Heap/Handle.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Heap/Heap.h>
 | 
					
						
							|  |  |  | #include <LibJS/Heap/HeapBlock.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/Object.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  | #include <setjmp.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #ifdef __serenity__
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #    include <serenity.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #elif __linux__
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #    include <pthread.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-25 09:49:14 +01:00
										 |  |  | #ifdef __serenity__
 | 
					
						
							| 
									
										
										
										
											2020-04-10 12:48:31 +02:00
										 |  |  | //#define HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-25 09:49:14 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Heap::Heap(Interpreter& interpreter) | 
					
						
							|  |  |  |     : m_interpreter(interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Heap::~Heap() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:11:19 +01:00
										 |  |  |     collect_garbage(CollectionType::CollectEverything); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Cell* Heap::allocate_cell(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-06 12:36:49 +02:00
										 |  |  |     if (should_collect_on_every_allocation()) { | 
					
						
							| 
									
										
										
										
											2020-03-16 19:18:46 +01:00
										 |  |  |         collect_garbage(); | 
					
						
							| 
									
										
										
										
											2020-04-06 12:36:49 +02:00
										 |  |  |     } else if (m_allocations_since_last_gc > m_max_allocations_between_gc) { | 
					
						
							|  |  |  |         m_allocations_since_last_gc = 0; | 
					
						
							|  |  |  |         collect_garbage(); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ++m_allocations_since_last_gc; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-16 19:18:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     for (auto& block : m_blocks) { | 
					
						
							|  |  |  |         if (size > block->cell_size()) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (auto* cell = block->allocate()) | 
					
						
							|  |  |  |             return cell; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-21 11:49:18 +01:00
										 |  |  |     size_t cell_size = round_up_to_power_of_two(size, 16); | 
					
						
							|  |  |  |     auto block = HeapBlock::create_with_cell_size(*this, cell_size); | 
					
						
							| 
									
										
										
										
											2020-03-13 11:01:44 +01:00
										 |  |  |     auto* cell = block->allocate(); | 
					
						
							|  |  |  |     m_blocks.append(move(block)); | 
					
						
							|  |  |  |     return cell; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 14:11:19 +01:00
										 |  |  | void Heap::collect_garbage(CollectionType collection_type) | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-23 14:11:19 +01:00
										 |  |  |     if (collection_type == CollectionType::CollectGarbage) { | 
					
						
							|  |  |  |         HashTable<Cell*> roots; | 
					
						
							|  |  |  |         gather_roots(roots); | 
					
						
							|  |  |  |         mark_live_cells(roots); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     sweep_dead_cells(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-15 15:12:34 +01:00
										 |  |  | void Heap::gather_roots(HashTable<Cell*>& roots) | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-15 15:12:34 +01:00
										 |  |  |     m_interpreter.gather_roots({}, roots); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  |     gather_conservative_roots(roots); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-18 20:03:17 +01:00
										 |  |  |     for (auto* handle : m_handles) | 
					
						
							|  |  |  |         roots.set(handle->cell()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  |     dbg() << "gather_roots:"; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     for (auto* root : roots) { | 
					
						
							|  |  |  |         dbg() << "  + " << root; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  | void Heap::gather_conservative_roots(HashTable<Cell*>& roots) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FlatPtr dummy; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |     dbg() << "gather_conservative_roots:"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     jmp_buf buf; | 
					
						
							|  |  |  |     setjmp(buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     HashTable<FlatPtr> possible_pointers; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  |     const FlatPtr* raw_jmp_buf = reinterpret_cast<const FlatPtr*>(buf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (size_t i = 0; i < sizeof(buf) / sizeof(FlatPtr); i += sizeof(FlatPtr)) | 
					
						
							|  |  |  |         possible_pointers.set(raw_jmp_buf[i]); | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     FlatPtr stack_base; | 
					
						
							|  |  |  |     size_t stack_size; | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef __serenity__
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  |     if (get_stack_bounds(&stack_base, &stack_size) < 0) { | 
					
						
							|  |  |  |         perror("get_stack_bounds"); | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-23 13:14:57 +01:00
										 |  |  | #elif __linux__
 | 
					
						
							|  |  |  |     pthread_attr_t attr = {}; | 
					
						
							|  |  |  |     if (int rc = pthread_getattr_np(pthread_self(), &attr) != 0) { | 
					
						
							|  |  |  |         fprintf(stderr, "pthread_getattr_np: %s\n", strerror(-rc)); | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (int rc = pthread_attr_getstack(&attr, (void**)&stack_base, &stack_size) != 0) { | 
					
						
							|  |  |  |         fprintf(stderr, "pthread_attr_getstack: %s\n", strerror(-rc)); | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     pthread_attr_destroy(&attr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-16 19:08:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     FlatPtr stack_reference = reinterpret_cast<FlatPtr>(&dummy); | 
					
						
							|  |  |  |     FlatPtr stack_top = stack_base + stack_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (FlatPtr stack_address = stack_reference; stack_address < stack_top; stack_address += sizeof(FlatPtr)) { | 
					
						
							|  |  |  |         auto data = *reinterpret_cast<FlatPtr*>(stack_address); | 
					
						
							|  |  |  |         possible_pointers.set(data); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto possible_pointer : possible_pointers) { | 
					
						
							|  |  |  |         if (!possible_pointer) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |         dbg() << "  ? " << (const void*)possible_pointer; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         if (auto* cell = cell_from_possible_pointer(possible_pointer)) { | 
					
						
							|  |  |  |             if (cell->is_live()) { | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |                 dbg() << "  ?-> " << (const void*)cell; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |                 roots.set(cell); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |                 dbg() << "  #-> " << (const void*)cell; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Cell* Heap::cell_from_possible_pointer(FlatPtr pointer) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto* possible_heap_block = HeapBlock::from_cell(reinterpret_cast<const Cell*>(pointer)); | 
					
						
							|  |  |  |     if (m_blocks.find([possible_heap_block](auto& block) { return block.ptr() == possible_heap_block; }) == m_blocks.end()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     return possible_heap_block->cell_from_possible_pointer(pointer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  | class MarkingVisitor final : public Cell::Visitor { | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |     MarkingVisitor() {} | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-16 16:07:50 +02:00
										 |  |  |     virtual void visit_impl(Cell* cell) | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |         if (cell->is_marked()) | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |         dbg() << "  ! " << cell; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |         cell->set_marked(true); | 
					
						
							|  |  |  |         cell->visit_children(*this); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  | void Heap::mark_live_cells(const HashTable<Cell*>& roots) | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | { | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |     dbg() << "mark_live_cells:"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |     MarkingVisitor visitor; | 
					
						
							| 
									
										
										
										
											2020-04-16 16:07:50 +02:00
										 |  |  |     for (auto* root : roots) | 
					
						
							| 
									
										
										
										
											2020-03-09 22:11:22 +01:00
										 |  |  |         visitor.visit(root); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Heap::sweep_dead_cells() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							|  |  |  |     dbg() << "sweep_dead_cells:"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |     Vector<HeapBlock*, 32> empty_blocks; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     for (auto& block : m_blocks) { | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |         bool block_has_live_cells = false; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |         block->for_each_cell([&](Cell* cell) { | 
					
						
							| 
									
										
										
										
											2020-03-08 23:17:34 +01:00
										 |  |  |             if (cell->is_live()) { | 
					
						
							|  |  |  |                 if (!cell->is_marked()) { | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-08 23:17:34 +01:00
										 |  |  |                     dbg() << "  ~ " << cell; | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-08 23:17:34 +01:00
										 |  |  |                     block->deallocate(cell); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     cell->set_marked(false); | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |                     block_has_live_cells = true; | 
					
						
							| 
									
										
										
										
											2020-03-08 23:17:34 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |         if (!block_has_live_cells) | 
					
						
							|  |  |  |             empty_blocks.append(block); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto* block : empty_blocks) { | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |         dbg() << " - Reclaim HeapBlock @ " << block << ": cell_size=" << block->cell_size(); | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |         m_blocks.remove_first_matching([block](auto& entry) { return entry == block; }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #ifdef HEAP_DEBUG
 | 
					
						
							| 
									
										
										
										
											2020-03-21 11:45:50 +01:00
										 |  |  |     for (auto& block : m_blocks) { | 
					
						
							|  |  |  |         dbg() << " > Live HeapBlock @ " << block << ": cell_size=" << block->cell_size(); | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-23 13:45:01 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-18 20:03:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void Heap::did_create_handle(Badge<HandleImpl>, HandleImpl& impl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(!m_handles.contains(&impl)); | 
					
						
							|  |  |  |     m_handles.set(&impl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Heap::did_destroy_handle(Badge<HandleImpl>, HandleImpl& impl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(m_handles.contains(&impl)); | 
					
						
							|  |  |  |     m_handles.remove(&impl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 19:23:58 +01:00
										 |  |  | } |