| // Copyright 2013 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "flutter/fml/mapping.h" |
| |
| #include <fcntl.h> |
| #include <sys/mman.h> |
| #include <sys/stat.h> |
| #include <unistd.h> |
| |
| #include <type_traits> |
| |
| #include "flutter/fml/build_config.h" |
| #include "flutter/fml/eintr_wrapper.h" |
| #include "flutter/fml/unique_fd.h" |
| |
| namespace fml { |
| |
| static int ToPosixProtectionFlags( |
| std::initializer_list<FileMapping::Protection> protection_flags) { |
| int flags = 0; |
| for (auto protection : protection_flags) { |
| switch (protection) { |
| case FileMapping::Protection::kRead: |
| flags |= PROT_READ; |
| break; |
| case FileMapping::Protection::kWrite: |
| flags |= PROT_WRITE; |
| break; |
| case FileMapping::Protection::kExecute: |
| flags |= PROT_READ | PROT_EXEC; |
| break; |
| } |
| } |
| return flags; |
| } |
| |
| static bool IsWritable( |
| std::initializer_list<FileMapping::Protection> protection_flags) { |
| for (auto protection : protection_flags) { |
| if (protection == FileMapping::Protection::kWrite) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| Mapping::Mapping() = default; |
| |
| Mapping::~Mapping() = default; |
| |
| FileMapping::FileMapping(const fml::UniqueFD& handle, |
| std::initializer_list<Protection> protection) |
| : size_(0), mapping_(nullptr) { |
| if (!handle.is_valid()) { |
| return; |
| } |
| |
| struct stat stat_buffer = {}; |
| |
| if (::fstat(handle.get(), &stat_buffer) != 0) { |
| return; |
| } |
| |
| if (stat_buffer.st_size == 0) { |
| valid_ = true; |
| return; |
| } |
| |
| const auto is_writable = IsWritable(protection); |
| |
| auto* mapping = |
| ::mmap(nullptr, stat_buffer.st_size, ToPosixProtectionFlags(protection), |
| is_writable ? MAP_SHARED : MAP_PRIVATE, handle.get(), 0); |
| |
| if (mapping == MAP_FAILED) { |
| return; |
| } |
| |
| mapping_ = static_cast<uint8_t*>(mapping); |
| size_ = stat_buffer.st_size; |
| valid_ = true; |
| if (is_writable) { |
| mutable_mapping_ = mapping_; |
| } |
| } |
| |
| FileMapping::~FileMapping() { |
| if (mapping_ != nullptr) { |
| ::munmap(mapping_, size_); |
| } |
| } |
| |
| size_t FileMapping::GetSize() const { |
| return size_; |
| } |
| |
| const uint8_t* FileMapping::GetMapping() const { |
| return mapping_; |
| } |
| |
| bool FileMapping::IsValid() const { |
| return valid_; |
| } |
| |
| } // namespace fml |