2020-01-18 08:38:21 +00:00
|
|
|
/*
|
2021-01-10 14:55:54 +00:00
|
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
2023-09-12 18:21:23 +00:00
|
|
|
* Copyright (c) 2023, kleines Filmröllchen <filmroellchen@serenityos.org>
|
2020-01-18 08:38:21 +00:00
|
|
|
*
|
2021-04-22 08:24:48 +00:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 08:38:21 +00:00
|
|
|
*/
|
|
|
|
|
2021-01-10 14:55:54 +00:00
|
|
|
#include <AK/ScopeGuard.h>
|
2023-01-15 00:57:29 +00:00
|
|
|
#include <LibCore/File.h>
|
2021-11-23 10:32:25 +00:00
|
|
|
#include <LibCore/MappedFile.h>
|
2021-11-23 10:36:00 +00:00
|
|
|
#include <LibCore/System.h>
|
2019-05-28 09:53:16 +00:00
|
|
|
#include <fcntl.h>
|
2018-10-10 09:53:07 +00:00
|
|
|
#include <sys/mman.h>
|
2019-04-03 11:51:49 +00:00
|
|
|
#include <unistd.h>
|
2018-10-10 09:53:07 +00:00
|
|
|
|
2021-11-23 10:32:25 +00:00
|
|
|
namespace Core {
|
2018-10-10 09:53:07 +00:00
|
|
|
|
2023-11-05 20:31:17 +00:00
|
|
|
ErrorOr<NonnullOwnPtr<MappedFile>> MappedFile::map(StringView path, Mode mode)
|
2018-10-10 09:53:07 +00:00
|
|
|
{
|
2023-11-05 20:31:17 +00:00
|
|
|
auto const file_mode = mode == Mode::ReadOnly ? O_RDONLY : O_RDWR;
|
2023-09-12 18:21:23 +00:00
|
|
|
auto fd = TRY(Core::System::open(path, file_mode | O_CLOEXEC, 0));
|
|
|
|
return map_from_fd_and_close(fd, path, mode);
|
2021-08-06 01:47:55 +00:00
|
|
|
}
|
|
|
|
|
2023-09-25 22:54:34 +00:00
|
|
|
ErrorOr<NonnullOwnPtr<MappedFile>> MappedFile::map_from_file(NonnullOwnPtr<Core::File> stream, StringView path)
|
2023-01-15 00:57:29 +00:00
|
|
|
{
|
|
|
|
return map_from_fd_and_close(stream->leak_fd(Badge<MappedFile> {}), path);
|
|
|
|
}
|
|
|
|
|
2023-11-05 20:31:17 +00:00
|
|
|
ErrorOr<NonnullOwnPtr<MappedFile>> MappedFile::map_from_fd_and_close(int fd, [[maybe_unused]] StringView path, Mode mode)
|
2021-08-06 01:47:55 +00:00
|
|
|
{
|
2021-11-23 10:36:00 +00:00
|
|
|
TRY(Core::System::fcntl(fd, F_SETFD, FD_CLOEXEC));
|
2021-08-06 01:47:55 +00:00
|
|
|
|
2021-01-10 14:55:54 +00:00
|
|
|
ScopeGuard fd_close_guard = [fd] {
|
2023-09-12 18:21:23 +00:00
|
|
|
::close(fd);
|
2021-01-10 14:55:54 +00:00
|
|
|
};
|
2018-10-10 09:53:07 +00:00
|
|
|
|
2021-11-23 10:36:00 +00:00
|
|
|
auto stat = TRY(Core::System::fstat(fd));
|
|
|
|
auto size = stat.st_size;
|
2019-08-05 12:26:56 +00:00
|
|
|
|
2023-09-12 18:21:23 +00:00
|
|
|
int protection;
|
|
|
|
int flags;
|
|
|
|
switch (mode) {
|
2023-11-05 20:31:17 +00:00
|
|
|
case Mode::ReadOnly:
|
2023-09-12 18:21:23 +00:00
|
|
|
protection = PROT_READ;
|
|
|
|
flags = MAP_SHARED;
|
|
|
|
break;
|
2023-11-05 20:31:17 +00:00
|
|
|
case Mode::ReadWrite:
|
2023-09-12 18:21:23 +00:00
|
|
|
protection = PROT_READ | PROT_WRITE;
|
|
|
|
// Don't map a read-write mapping shared as a precaution.
|
|
|
|
flags = MAP_PRIVATE;
|
|
|
|
break;
|
|
|
|
}
|
2021-07-29 13:00:26 +00:00
|
|
|
|
2023-09-12 18:21:23 +00:00
|
|
|
auto* ptr = TRY(Core::System::mmap(nullptr, size, protection, flags, fd, 0, 0, path));
|
|
|
|
|
|
|
|
return adopt_own(*new MappedFile(ptr, size, mode));
|
2018-10-10 09:53:07 +00:00
|
|
|
}
|
|
|
|
|
2023-11-05 20:31:17 +00:00
|
|
|
MappedFile::MappedFile(void* ptr, size_t size, Mode mode)
|
|
|
|
: FixedMemoryStream(Bytes { ptr, size }, mode)
|
2023-09-12 18:21:23 +00:00
|
|
|
, m_data(ptr)
|
2021-01-10 14:55:54 +00:00
|
|
|
, m_size(size)
|
2018-10-10 09:53:07 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-01-10 14:55:54 +00:00
|
|
|
MappedFile::~MappedFile()
|
2019-04-03 11:51:49 +00:00
|
|
|
{
|
2022-11-20 03:23:14 +00:00
|
|
|
auto res = Core::System::munmap(m_data, m_size);
|
|
|
|
if (res.is_error())
|
|
|
|
dbgln("Failed to unmap MappedFile (@ {:p}): {}", m_data, res.error());
|
2019-04-03 11:51:49 +00:00
|
|
|
}
|
|
|
|
|
2018-10-10 09:53:07 +00:00
|
|
|
}
|