AnonymousBuffer.cpp 2.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. /*
  2. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Try.h>
  7. #include <LibCore/AnonymousBuffer.h>
  8. #include <LibIPC/File.h>
  9. #include <fcntl.h>
  10. #include <sys/mman.h>
  11. #if defined(__serenity__)
  12. # include <serenity.h>
  13. #endif
  14. #if defined(__linux__) && !defined(MFD_CLOEXEC)
  15. # include <linux/memfd.h>
  16. # include <sys/syscall.h>
  17. static int memfd_create(const char* name, unsigned int flags)
  18. {
  19. return syscall(SYS_memfd_create, name, flags);
  20. }
  21. #endif
  22. namespace Core {
  23. ErrorOr<AnonymousBuffer> AnonymousBuffer::create_with_size(size_t size)
  24. {
  25. int fd = -1;
  26. #if defined(__serenity__)
  27. fd = anon_create(round_up_to_power_of_two(size, PAGE_SIZE), O_CLOEXEC);
  28. if (fd < 0)
  29. return Error::from_errno(errno);
  30. #elif defined(__linux__)
  31. fd = memfd_create("", MFD_CLOEXEC);
  32. if (fd < 0)
  33. return Error::from_errno(errno);
  34. if (ftruncate(fd, size) < 0) {
  35. close(fd);
  36. return Error::from_errno(errno);
  37. }
  38. #endif
  39. if (fd < 0)
  40. return Error::from_errno(errno);
  41. return create_from_anon_fd(fd, size);
  42. }
  43. ErrorOr<NonnullRefPtr<AnonymousBufferImpl>> AnonymousBufferImpl::create(int fd, size_t size)
  44. {
  45. auto* data = mmap(nullptr, round_up_to_power_of_two(size, PAGE_SIZE), PROT_READ | PROT_WRITE, MAP_FILE | MAP_SHARED, fd, 0);
  46. if (data == MAP_FAILED)
  47. return Error::from_errno(errno);
  48. return AK::adopt_nonnull_ref_or_enomem(new (nothrow) AnonymousBufferImpl(fd, size, data));
  49. }
  50. AnonymousBufferImpl::~AnonymousBufferImpl()
  51. {
  52. if (m_fd != -1) {
  53. auto rc = close(m_fd);
  54. VERIFY(rc == 0);
  55. }
  56. auto rc = munmap(m_data, round_up_to_power_of_two(m_size, PAGE_SIZE));
  57. VERIFY(rc == 0);
  58. }
  59. ErrorOr<AnonymousBuffer> AnonymousBuffer::create_from_anon_fd(int fd, size_t size)
  60. {
  61. auto impl = TRY(AnonymousBufferImpl::create(fd, size));
  62. return AnonymousBuffer(move(impl));
  63. }
  64. AnonymousBuffer::AnonymousBuffer(NonnullRefPtr<AnonymousBufferImpl> impl)
  65. : m_impl(move(impl))
  66. {
  67. }
  68. AnonymousBufferImpl::AnonymousBufferImpl(int fd, size_t size, void* data)
  69. : m_fd(fd)
  70. , m_size(size)
  71. , m_data(data)
  72. {
  73. }
  74. }