SharedFramebufferVMObject.cpp 5.1 KB

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