InodeVMObject.cpp 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/FileSystem/Inode.h>
  7. #include <Kernel/Memory/InodeVMObject.h>
  8. namespace Kernel::Memory {
  9. InodeVMObject::InodeVMObject(Inode& inode, FixedArray<RefPtr<PhysicalPage>>&& new_physical_pages, Bitmap dirty_pages)
  10. : VMObject(move(new_physical_pages))
  11. , m_inode(inode)
  12. , m_dirty_pages(move(dirty_pages))
  13. {
  14. }
  15. InodeVMObject::InodeVMObject(InodeVMObject const& other, FixedArray<RefPtr<PhysicalPage>>&& new_physical_pages, Bitmap dirty_pages)
  16. : VMObject(move(new_physical_pages))
  17. , m_inode(other.m_inode)
  18. , m_dirty_pages(move(dirty_pages))
  19. {
  20. for (size_t i = 0; i < page_count(); ++i)
  21. m_dirty_pages.set(i, other.m_dirty_pages.get(i));
  22. }
  23. InodeVMObject::~InodeVMObject() = default;
  24. size_t InodeVMObject::amount_clean() const
  25. {
  26. size_t count = 0;
  27. VERIFY(page_count() == m_dirty_pages.size());
  28. for (size_t i = 0; i < page_count(); ++i) {
  29. if (!m_dirty_pages.get(i) && m_physical_pages[i])
  30. ++count;
  31. }
  32. return count * PAGE_SIZE;
  33. }
  34. size_t InodeVMObject::amount_dirty() const
  35. {
  36. size_t count = 0;
  37. for (size_t i = 0; i < m_dirty_pages.size(); ++i) {
  38. if (m_dirty_pages.get(i))
  39. ++count;
  40. }
  41. return count * PAGE_SIZE;
  42. }
  43. int InodeVMObject::release_all_clean_pages()
  44. {
  45. SpinlockLocker locker(m_lock);
  46. int count = 0;
  47. for (size_t i = 0; i < page_count(); ++i) {
  48. if (!m_dirty_pages.get(i) && m_physical_pages[i]) {
  49. m_physical_pages[i] = nullptr;
  50. ++count;
  51. }
  52. }
  53. if (count) {
  54. for_each_region([](auto& region) {
  55. region.remap();
  56. });
  57. }
  58. return count;
  59. }
  60. u32 InodeVMObject::writable_mappings() const
  61. {
  62. u32 count = 0;
  63. const_cast<InodeVMObject&>(*this).for_each_region([&](auto& region) {
  64. if (region.is_writable())
  65. ++count;
  66. });
  67. return count;
  68. }
  69. u32 InodeVMObject::executable_mappings() const
  70. {
  71. u32 count = 0;
  72. const_cast<InodeVMObject&>(*this).for_each_region([&](auto& region) {
  73. if (region.is_executable())
  74. ++count;
  75. });
  76. return count;
  77. }
  78. }