Serenity Operating System
at master 120 lines 5.3 kB view raw
1/* 2 * Copyright (c) 2022, Liav A. <liavalb@hotmail.co.il> 3 * 4 * SPDX-License-Identifier: BSD-2-Clause 5 */ 6 7#include <Kernel/FileSystem/Inode.h> 8#include <Kernel/Locking/Spinlock.h> 9#include <Kernel/Memory/SharedFramebufferVMObject.h> 10 11namespace Kernel::Memory { 12 13ErrorOr<NonnullLockRefPtr<SharedFramebufferVMObject>> SharedFramebufferVMObject::try_create_for_physical_range(PhysicalAddress paddr, size_t size) 14{ 15 auto real_framebuffer_vmobject = TRY(AnonymousVMObject::try_create_for_physical_range(paddr, size)); 16 auto new_physical_pages = TRY(VMObject::try_create_physical_pages(size)); 17 auto committed_pages = TRY(MM.commit_physical_pages(ceil_div(size, static_cast<size_t>(PAGE_SIZE)))); 18 auto vm_object = TRY(adopt_nonnull_lock_ref_or_enomem(new (nothrow) SharedFramebufferVMObject(move(new_physical_pages), move(committed_pages), real_framebuffer_vmobject))); 19 TRY(vm_object->create_fake_writes_framebuffer_vm_object()); 20 TRY(vm_object->create_real_writes_framebuffer_vm_object()); 21 return vm_object; 22} 23 24ErrorOr<NonnullLockRefPtr<SharedFramebufferVMObject>> SharedFramebufferVMObject::try_create_at_arbitrary_physical_range(size_t size) 25{ 26 auto real_framebuffer_vmobject = TRY(AnonymousVMObject::try_create_with_size(size, AllocationStrategy::AllocateNow)); 27 auto new_physical_pages = TRY(VMObject::try_create_physical_pages(size)); 28 auto committed_pages = TRY(MM.commit_physical_pages(ceil_div(size, static_cast<size_t>(PAGE_SIZE)))); 29 auto vm_object = TRY(adopt_nonnull_lock_ref_or_enomem(new (nothrow) SharedFramebufferVMObject(move(new_physical_pages), move(committed_pages), real_framebuffer_vmobject))); 30 TRY(vm_object->create_fake_writes_framebuffer_vm_object()); 31 TRY(vm_object->create_real_writes_framebuffer_vm_object()); 32 return vm_object; 33} 34 35ErrorOr<NonnullLockRefPtr<SharedFramebufferVMObject::FakeWritesFramebufferVMObject>> SharedFramebufferVMObject::FakeWritesFramebufferVMObject::try_create(Badge<SharedFramebufferVMObject>, SharedFramebufferVMObject const& parent_object) 36{ 37 auto new_physical_pages = TRY(VMObject::try_create_physical_pages(0)); 38 return adopt_nonnull_lock_ref_or_enomem(new (nothrow) FakeWritesFramebufferVMObject(parent_object, move(new_physical_pages))); 39} 40 41ErrorOr<NonnullLockRefPtr<SharedFramebufferVMObject::RealWritesFramebufferVMObject>> SharedFramebufferVMObject::RealWritesFramebufferVMObject::try_create(Badge<SharedFramebufferVMObject>, SharedFramebufferVMObject const& parent_object) 42{ 43 auto new_physical_pages = TRY(VMObject::try_create_physical_pages(0)); 44 return adopt_nonnull_lock_ref_or_enomem(new (nothrow) RealWritesFramebufferVMObject(parent_object, move(new_physical_pages))); 45} 46 47ErrorOr<void> SharedFramebufferVMObject::create_fake_writes_framebuffer_vm_object() 48{ 49 m_fake_writes_framebuffer_vmobject = TRY(FakeWritesFramebufferVMObject::try_create({}, *this)); 50 return {}; 51} 52 53ErrorOr<void> SharedFramebufferVMObject::create_real_writes_framebuffer_vm_object() 54{ 55 m_real_writes_framebuffer_vmobject = TRY(RealWritesFramebufferVMObject::try_create({}, *this)); 56 return {}; 57} 58 59Span<RefPtr<PhysicalPage>> SharedFramebufferVMObject::real_framebuffer_physical_pages() 60{ 61 return m_real_framebuffer_vmobject->physical_pages(); 62} 63ReadonlySpan<RefPtr<PhysicalPage>> SharedFramebufferVMObject::real_framebuffer_physical_pages() const 64{ 65 return m_real_framebuffer_vmobject->physical_pages(); 66} 67 68Span<RefPtr<PhysicalPage>> SharedFramebufferVMObject::fake_sink_framebuffer_physical_pages() 69{ 70 return m_physical_pages.span(); 71} 72 73ReadonlySpan<RefPtr<PhysicalPage>> SharedFramebufferVMObject::fake_sink_framebuffer_physical_pages() const 74{ 75 return m_physical_pages.span(); 76} 77 78void SharedFramebufferVMObject::switch_to_fake_sink_framebuffer_writes(Badge<Kernel::DisplayConnector>) 79{ 80 SpinlockLocker locker(m_writes_state_lock); 81 m_writes_are_faked = true; 82 for_each_region([](Region& region) { 83 region.remap(); 84 }); 85} 86void SharedFramebufferVMObject::switch_to_real_framebuffer_writes(Badge<Kernel::DisplayConnector>) 87{ 88 SpinlockLocker locker(m_writes_state_lock); 89 m_writes_are_faked = false; 90 for_each_region([](Region& region) { 91 region.remap(); 92 }); 93} 94 95ReadonlySpan<RefPtr<PhysicalPage>> SharedFramebufferVMObject::physical_pages() const 96{ 97 SpinlockLocker locker(m_writes_state_lock); 98 if (m_writes_are_faked) 99 return VMObject::physical_pages(); 100 return m_real_framebuffer_vmobject->physical_pages(); 101} 102Span<RefPtr<PhysicalPage>> SharedFramebufferVMObject::physical_pages() 103{ 104 SpinlockLocker locker(m_writes_state_lock); 105 if (m_writes_are_faked) 106 return VMObject::physical_pages(); 107 return m_real_framebuffer_vmobject->physical_pages(); 108} 109 110SharedFramebufferVMObject::SharedFramebufferVMObject(FixedArray<RefPtr<PhysicalPage>>&& new_physical_pages, CommittedPhysicalPageSet committed_pages, AnonymousVMObject& real_framebuffer_vmobject) 111 : VMObject(move(new_physical_pages)) 112 , m_real_framebuffer_vmobject(real_framebuffer_vmobject) 113 , m_committed_pages(move(committed_pages)) 114{ 115 // Allocate all pages right now. We know we can get all because we committed the amount needed 116 for (size_t i = 0; i < page_count(); ++i) 117 m_physical_pages[i] = m_committed_pages.take_one(); 118} 119 120}