1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
// Copyright 2014 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 "ui/gl/gl_image_shared_memory.h"
#include "base/logging.h"
#include "base/numerics/safe_math.h"
#include "base/process/process_handle.h"
namespace gfx {
namespace {
// Returns true if the size is valid and false otherwise.
bool SizeInBytes(const gfx::Size& size,
gfx::BufferFormat format,
size_t* size_in_bytes) {
if (size.IsEmpty())
return false;
size_t stride_in_bytes = 0;
if (!GLImageMemory::StrideInBytes(size.width(), format, &stride_in_bytes))
return false;
base::CheckedNumeric<size_t> s = stride_in_bytes;
s *= size.height();
if (!s.IsValid())
return false;
*size_in_bytes = s.ValueOrDie();
return true;
}
} // namespace
GLImageSharedMemory::GLImageSharedMemory(const gfx::Size& size,
unsigned internalformat)
: GLImageMemory(size, internalformat) {
}
GLImageSharedMemory::~GLImageSharedMemory() {
DCHECK(!shared_memory_);
}
bool GLImageSharedMemory::Initialize(
const base::SharedMemoryHandle& handle,
gfx::GenericSharedMemoryId shared_memory_id,
gfx::BufferFormat format) {
size_t size_in_bytes;
if (!SizeInBytes(GetSize(), format, &size_in_bytes))
return false;
if (!base::SharedMemory::IsHandleValid(handle))
return false;
base::SharedMemory shared_memory(handle, true);
// Duplicate the handle.
base::SharedMemoryHandle duped_shared_memory_handle;
if (!shared_memory.ShareToProcess(base::GetCurrentProcessHandle(),
&duped_shared_memory_handle)) {
DVLOG(0) << "Failed to duplicate shared memory handle.";
return false;
}
scoped_ptr<base::SharedMemory> duped_shared_memory(
new base::SharedMemory(duped_shared_memory_handle, true));
if (!duped_shared_memory->Map(size_in_bytes)) {
DVLOG(0) << "Failed to map shared memory.";
return false;
}
if (!GLImageMemory::Initialize(
static_cast<unsigned char*>(duped_shared_memory->memory()), format)) {
return false;
}
DCHECK(!shared_memory_);
shared_memory_ = duped_shared_memory.Pass();
shared_memory_id_ = shared_memory_id;
return true;
}
void GLImageSharedMemory::Destroy(bool have_context) {
GLImageMemory::Destroy(have_context);
shared_memory_.reset();
}
void GLImageSharedMemory::OnMemoryDump(
base::trace_event::ProcessMemoryDump* pmd,
uint64_t process_tracing_id,
const std::string& dump_name) {
size_t size_in_bytes = 0;
if (shared_memory_) {
bool result = SizeInBytes(GetSize(), format(), &size_in_bytes);
DCHECK(result);
}
// Dump under "/shared_memory", as the base class may also dump to
// "/texture_memory".
base::trace_event::MemoryAllocatorDump* dump =
pmd->CreateAllocatorDump(dump_name + "/private_memory");
dump->AddScalar(base::trace_event::MemoryAllocatorDump::kNameSize,
base::trace_event::MemoryAllocatorDump::kUnitsBytes,
static_cast<uint64_t>(size_in_bytes));
auto guid = gfx::GetGenericSharedMemoryGUIDForTracing(process_tracing_id,
shared_memory_id_);
pmd->CreateSharedGlobalAllocatorDump(guid);
pmd->AddOwnershipEdge(dump->guid(), guid);
// Also dump the base class's texture memory.
GLImageMemory::OnMemoryDump(pmd, process_tracing_id, dump_name);
}
} // namespace gfx
|