mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-12-01 01:36:09 +03:00
116 lines
2.9 KiB
C++
116 lines
2.9 KiB
C++
|
// Copyright 2017 The Chromium 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 "base/memory/shared_memory_handle.h"
|
||
|
|
||
|
#include <sys/mman.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "base/logging.h"
|
||
|
#include "base/posix/eintr_wrapper.h"
|
||
|
#include "base/posix/unix_domain_socket.h"
|
||
|
#include "base/unguessable_token.h"
|
||
|
#include "third_party/ashmem/ashmem.h"
|
||
|
|
||
|
namespace base {
|
||
|
|
||
|
static int GetAshmemRegionProtectionMask(int fd) {
|
||
|
int prot = ashmem_get_prot_region(fd);
|
||
|
if (prot < 0) {
|
||
|
DPLOG(ERROR) << "ashmem_get_prot_region";
|
||
|
return -1;
|
||
|
}
|
||
|
return prot;
|
||
|
}
|
||
|
|
||
|
SharedMemoryHandle::SharedMemoryHandle() = default;
|
||
|
|
||
|
SharedMemoryHandle::SharedMemoryHandle(
|
||
|
const base::FileDescriptor& file_descriptor,
|
||
|
size_t size,
|
||
|
const base::UnguessableToken& guid)
|
||
|
: guid_(guid), size_(size) {
|
||
|
DCHECK_GE(file_descriptor.fd, 0);
|
||
|
file_descriptor_ = file_descriptor;
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
SharedMemoryHandle SharedMemoryHandle::ImportHandle(int fd, size_t size) {
|
||
|
SharedMemoryHandle handle;
|
||
|
handle.file_descriptor_.fd = fd;
|
||
|
handle.file_descriptor_.auto_close = false;
|
||
|
handle.guid_ = UnguessableToken::Create();
|
||
|
handle.size_ = size;
|
||
|
return handle;
|
||
|
}
|
||
|
|
||
|
int SharedMemoryHandle::GetHandle() const {
|
||
|
DCHECK(IsValid());
|
||
|
return file_descriptor_.fd;
|
||
|
}
|
||
|
|
||
|
bool SharedMemoryHandle::IsValid() const {
|
||
|
return file_descriptor_.fd >= 0;
|
||
|
}
|
||
|
|
||
|
void SharedMemoryHandle::Close() const {
|
||
|
DCHECK(IsValid());
|
||
|
if (IGNORE_EINTR(close(file_descriptor_.fd)) < 0)
|
||
|
PLOG(ERROR) << "close";
|
||
|
}
|
||
|
|
||
|
int SharedMemoryHandle::Release() {
|
||
|
int old_fd = file_descriptor_.fd;
|
||
|
file_descriptor_.fd = -1;
|
||
|
return old_fd;
|
||
|
}
|
||
|
|
||
|
SharedMemoryHandle SharedMemoryHandle::Duplicate() const {
|
||
|
DCHECK(IsValid());
|
||
|
SharedMemoryHandle result;
|
||
|
int duped_handle = HANDLE_EINTR(dup(file_descriptor_.fd));
|
||
|
if (duped_handle >= 0) {
|
||
|
result = SharedMemoryHandle(FileDescriptor(duped_handle, true), GetSize(),
|
||
|
GetGUID());
|
||
|
if (IsReadOnly())
|
||
|
result.SetReadOnly();
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
void SharedMemoryHandle::SetOwnershipPassesToIPC(bool ownership_passes) {
|
||
|
file_descriptor_.auto_close = ownership_passes;
|
||
|
}
|
||
|
|
||
|
bool SharedMemoryHandle::OwnershipPassesToIPC() const {
|
||
|
return file_descriptor_.auto_close;
|
||
|
}
|
||
|
|
||
|
bool SharedMemoryHandle::IsRegionReadOnly() const {
|
||
|
int prot = GetAshmemRegionProtectionMask(file_descriptor_.fd);
|
||
|
return (prot >= 0 && (prot & PROT_WRITE) == 0);
|
||
|
}
|
||
|
|
||
|
bool SharedMemoryHandle::SetRegionReadOnly() const {
|
||
|
int fd = file_descriptor_.fd;
|
||
|
int prot = GetAshmemRegionProtectionMask(fd);
|
||
|
if (prot < 0)
|
||
|
return false;
|
||
|
|
||
|
if ((prot & PROT_WRITE) == 0) {
|
||
|
// Region is already read-only.
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
prot &= ~PROT_WRITE;
|
||
|
int ret = ashmem_set_prot_region(fd, prot);
|
||
|
if (ret != 0) {
|
||
|
DPLOG(ERROR) << "ashmem_set_prot_region";
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
} // namespace base
|