blob: b26de442161f2ca69c732992761c2b10c418ab31 [file] [log] [blame]
// Copyright (c) 2013 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 "content/child/child_shared_bitmap_manager.h"
#include <stddef.h>
#include <utility>
#include "base/debug/alias.h"
#include "base/memory/ptr_util.h"
#include "base/process/memory.h"
#include "base/process/process_metrics.h"
#include "build/build_config.h"
#include "content/child/child_thread_impl.h"
#include "content/common/child_process_messages.h"
#include "mojo/public/cpp/system/platform_handle.h"
#include "ui/gfx/geometry/size.h"
namespace content {
namespace {
class ChildSharedBitmap : public SharedMemoryBitmap {
public:
ChildSharedBitmap(
const scoped_refptr<mojom::ThreadSafeRenderMessageFilterAssociatedPtr>&
render_message_filter_ptr,
base::SharedMemory* shared_memory,
const cc::SharedBitmapId& id)
: SharedMemoryBitmap(static_cast<uint8_t*>(shared_memory->memory()),
id,
shared_memory),
render_message_filter_ptr_(render_message_filter_ptr) {}
ChildSharedBitmap(
const scoped_refptr<mojom::ThreadSafeRenderMessageFilterAssociatedPtr>&
render_message_filter_ptr,
std::unique_ptr<base::SharedMemory> shared_memory_holder,
const cc::SharedBitmapId& id)
: ChildSharedBitmap(render_message_filter_ptr,
shared_memory_holder.get(),
id) {
shared_memory_holder_ = std::move(shared_memory_holder);
}
~ChildSharedBitmap() override {
(*render_message_filter_ptr_)->DeletedSharedBitmap(id());
}
private:
scoped_refptr<mojom::ThreadSafeRenderMessageFilterAssociatedPtr>
render_message_filter_ptr_;
std::unique_ptr<base::SharedMemory> shared_memory_holder_;
};
// Collect extra information for debugging bitmap creation failures.
void CollectMemoryUsageAndDie(const gfx::Size& size, size_t alloc_size) {
#if defined(OS_WIN)
int width = size.width();
int height = size.height();
DWORD last_error = GetLastError();
std::unique_ptr<base::ProcessMetrics> metrics(
base::ProcessMetrics::CreateProcessMetrics(
base::GetCurrentProcessHandle()));
size_t private_bytes = 0;
size_t shared_bytes = 0;
metrics->GetMemoryBytes(&private_bytes, &shared_bytes);
base::debug::Alias(&width);
base::debug::Alias(&height);
base::debug::Alias(&last_error);
base::debug::Alias(&private_bytes);
base::debug::Alias(&shared_bytes);
#endif
base::TerminateBecauseOutOfMemory(alloc_size);
}
} // namespace
SharedMemoryBitmap::SharedMemoryBitmap(uint8_t* pixels,
const cc::SharedBitmapId& id,
base::SharedMemory* shared_memory)
: SharedBitmap(pixels, id), shared_memory_(shared_memory) {}
ChildSharedBitmapManager::ChildSharedBitmapManager(
const scoped_refptr<mojom::ThreadSafeRenderMessageFilterAssociatedPtr>&
render_message_filter_ptr)
: render_message_filter_ptr_(render_message_filter_ptr) {}
ChildSharedBitmapManager::~ChildSharedBitmapManager() {}
std::unique_ptr<cc::SharedBitmap>
ChildSharedBitmapManager::AllocateSharedBitmap(const gfx::Size& size) {
TRACE_EVENT2("renderer",
"ChildSharedBitmapManager::AllocateSharedMemoryBitmap", "width",
size.width(), "height", size.height());
size_t memory_size;
if (!cc::SharedBitmap::SizeInBytes(size, &memory_size))
return std::unique_ptr<SharedMemoryBitmap>();
cc::SharedBitmapId id = cc::SharedBitmap::GenerateId();
std::unique_ptr<base::SharedMemory> memory =
ChildThreadImpl::AllocateSharedMemory(memory_size);
if (!memory || !memory->Map(memory_size))
CollectMemoryUsageAndDie(size, memory_size);
NotifyAllocatedSharedBitmap(memory.get(), id);
// Close the associated FD to save resources, the previously mapped memory
// remains available.
memory->Close();
return base::MakeUnique<ChildSharedBitmap>(render_message_filter_ptr_,
std::move(memory), id);
}
std::unique_ptr<cc::SharedBitmap>
ChildSharedBitmapManager::GetSharedBitmapFromId(const gfx::Size&,
const cc::SharedBitmapId&) {
NOTREACHED();
return std::unique_ptr<cc::SharedBitmap>();
}
std::unique_ptr<cc::SharedBitmap>
ChildSharedBitmapManager::GetBitmapForSharedMemory(base::SharedMemory* mem) {
cc::SharedBitmapId id = cc::SharedBitmap::GenerateId();
NotifyAllocatedSharedBitmap(mem, cc::SharedBitmap::GenerateId());
return base::MakeUnique<ChildSharedBitmap>(render_message_filter_ptr_,
std::move(mem), id);
}
// Notifies the browser process that a shared bitmap with the given ID was
// allocated. Caller keeps ownership of |memory|.
void ChildSharedBitmapManager::NotifyAllocatedSharedBitmap(
base::SharedMemory* memory,
const cc::SharedBitmapId& id) {
base::SharedMemoryHandle handle_to_send =
base::SharedMemory::DuplicateHandle(memory->handle());
if (!base::SharedMemory::IsHandleValid(handle_to_send)) {
LOG(ERROR) << "Failed to duplicate shared memory handle for bitmap.";
return;
}
mojo::ScopedSharedBufferHandle buffer_handle = mojo::WrapSharedMemoryHandle(
handle_to_send, memory->mapped_size(), true /* read_only */);
(*render_message_filter_ptr_)
->AllocatedSharedBitmap(std::move(buffer_handle), id);
}
} // namespace content