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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
|
// Copyright 2012 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.
#ifndef CCResourceProvider_h
#define CCResourceProvider_h
#include "IntSize.h"
#include "base/basictypes.h"
#include "base/hash_tables.h"
#include "base/memory/scoped_ptr.h"
#include "cc/graphics_context.h"
#include "cc/texture_copier.h"
#include "third_party/khronos/GLES2/gl2.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkCanvas.h"
#include <deque>
#include <vector>
namespace WebKit {
class WebGraphicsContext3D;
}
namespace cc {
class IntRect;
class LayerTextureSubImage;
class TextureCopier;
class TextureUploader;
// Thread-safety notes: this class is not thread-safe and can only be called
// from the thread it was created on (in practice, the compositor thread).
class ResourceProvider {
public:
typedef unsigned ResourceId;
typedef std::vector<ResourceId> ResourceIdArray;
typedef base::hash_map<ResourceId, ResourceId> ResourceIdMap;
enum TextureUsageHint { TextureUsageAny, TextureUsageFramebuffer };
enum ResourceType {
GLTexture = 1,
Bitmap,
};
struct Mailbox {
GLbyte name[64];
};
struct TransferableResource {
unsigned id;
GLenum format;
IntSize size;
Mailbox mailbox;
};
typedef std::vector<TransferableResource> TransferableResourceArray;
struct TransferableResourceList {
TransferableResourceList();
~TransferableResourceList();
TransferableResourceArray resources;
unsigned syncPoint;
};
static scoped_ptr<ResourceProvider> create(GraphicsContext*);
virtual ~ResourceProvider();
WebKit::WebGraphicsContext3D* graphicsContext3D();
TextureCopier* textureCopier() const { return m_textureCopier.get(); }
int maxTextureSize() const { return m_maxTextureSize; }
unsigned numResources() const { return m_resources.size(); }
// Checks whether a resource is in use by a consumer.
bool inUseByConsumer(ResourceId);
// Producer interface.
void setDefaultResourceType(ResourceType type) { m_defaultResourceType = type; }
ResourceType defaultResourceType() const { return m_defaultResourceType; }
ResourceType resourceType(ResourceId);
// Creates a resource of the default resource type.
ResourceId createResource(int pool, const IntSize&, GLenum format, TextureUsageHint);
// You can also explicitly create a specific resource type.
ResourceId createGLTexture(int pool, const IntSize&, GLenum format, TextureUsageHint);
ResourceId createBitmap(int pool, const IntSize&);
// Wraps an external texture into a GL resource.
ResourceId createResourceFromExternalTexture(unsigned textureId);
void deleteResource(ResourceId);
// Deletes all resources owned by a given pool.
void deleteOwnedResources(int pool);
// Upload data from image, copying sourceRect (in image) into destRect (in the resource).
void upload(ResourceId, const uint8_t* image, const IntRect& imageRect, const IntRect& sourceRect, const IntSize& destOffset);
// Check upload status.
size_t numBlockingUploads();
void markPendingUploadsAsNonBlocking();
double estimatedUploadsPerSecond();
// Flush all context operations, kicking uploads and ensuring ordering with
// respect to other contexts.
void flush();
// Only flush the command buffer if supported.
// Returns true if the shallow flush occurred, false otherwise.
bool shallowFlushIfSupported();
// Creates accounting for a child, and associate it with a pool. Resources
// transfered from that child will go to that pool. Returns a child ID.
int createChild(int pool);
// Destroys accounting for the child, deleting all resources from that pool.
void destroyChild(int child);
// Gets the child->parent resource ID map.
const ResourceIdMap& getChildToParentMap(int child) const;
// Prepares resources to be transfered to the parent, moving them to
// mailboxes and serializing meta-data into TransferableResources.
// Resources are not removed from the ResourceProvider, but are markes as
// "in use".
TransferableResourceList prepareSendToParent(const ResourceIdArray&);
// Prepares resources to be transfered back to the child, moving them to
// mailboxes and serializing meta-data into TransferableResources.
// Resources are removed from the ResourceProvider. Note: the resource IDs
// passed are in the parent namespace and will be translated to the child
// namespace when returned.
TransferableResourceList prepareSendToChild(int child, const ResourceIdArray&);
// Receives resources from a child, moving them from mailboxes. Resource IDs
// passed are in the child namespace, and will be translated to the parent
// namespace, added to the child->parent map.
// NOTE: if the syncPoint filed in TransferableResourceList is set, this
// will wait on it.
void receiveFromChild(int child, const TransferableResourceList&);
// Receives resources from the parent, moving them from mailboxes. Resource IDs
// passed are in the child namespace.
// NOTE: if the syncPoint filed in TransferableResourceList is set, this
// will wait on it.
void receiveFromParent(const TransferableResourceList&);
// Only for testing
size_t mailboxCount() const { return m_mailboxes.size(); }
// The following lock classes are part of the ResourceProvider API and are
// needed to read and write the resource contents. The user must ensure
// that they only use GL locks on GL resources, etc, and this is enforced
// by assertions.
class ScopedReadLockGL {
public:
ScopedReadLockGL(ResourceProvider*, ResourceProvider::ResourceId);
~ScopedReadLockGL();
unsigned textureId() const { return m_textureId; }
private:
ResourceProvider* m_resourceProvider;
ResourceProvider::ResourceId m_resourceId;
unsigned m_textureId;
DISALLOW_COPY_AND_ASSIGN(ScopedReadLockGL);
};
class ScopedWriteLockGL {
public:
ScopedWriteLockGL(ResourceProvider*, ResourceProvider::ResourceId);
~ScopedWriteLockGL();
unsigned textureId() const { return m_textureId; }
private:
ResourceProvider* m_resourceProvider;
ResourceProvider::ResourceId m_resourceId;
unsigned m_textureId;
DISALLOW_COPY_AND_ASSIGN(ScopedWriteLockGL);
};
class ScopedReadLockSoftware {
public:
ScopedReadLockSoftware(ResourceProvider*, ResourceProvider::ResourceId);
~ScopedReadLockSoftware();
const SkBitmap* skBitmap() const { return &m_skBitmap; }
private:
ResourceProvider* m_resourceProvider;
ResourceProvider::ResourceId m_resourceId;
SkBitmap m_skBitmap;
DISALLOW_COPY_AND_ASSIGN(ScopedReadLockSoftware);
};
class ScopedWriteLockSoftware {
public:
ScopedWriteLockSoftware(ResourceProvider*, ResourceProvider::ResourceId);
~ScopedWriteLockSoftware();
SkCanvas* skCanvas() { return m_skCanvas.get(); }
private:
ResourceProvider* m_resourceProvider;
ResourceProvider::ResourceId m_resourceId;
SkBitmap m_skBitmap;
scoped_ptr<SkCanvas> m_skCanvas;
DISALLOW_COPY_AND_ASSIGN(ScopedWriteLockSoftware);
};
// Temporary functions for debugging crashes in issue 151428 in canary.
// Do not use these!
static void debugNotifyEnterZone(unsigned int index);
static void debugNotifyLeaveZone();
private:
struct Resource {
Resource();
Resource(unsigned textureId, int pool, const IntSize& size, GLenum format);
Resource(uint8_t* pixels, int pool, const IntSize& size, GLenum format);
unsigned glId;
uint8_t* pixels;
int pool;
int lockForReadCount;
bool lockedForWrite;
bool external;
bool exported;
bool markedForDeletion;
IntSize size;
GLenum format;
ResourceType type;
};
typedef base::hash_map<ResourceId, Resource> ResourceMap;
struct Child {
Child();
~Child();
int pool;
ResourceIdMap childToParentMap;
ResourceIdMap parentToChildMap;
};
typedef base::hash_map<int, Child> ChildMap;
explicit ResourceProvider(GraphicsContext*);
bool initialize();
const Resource* lockForRead(ResourceId);
void unlockForRead(ResourceId);
const Resource* lockForWrite(ResourceId);
void unlockForWrite(ResourceId);
static void populateSkBitmapWithResource(SkBitmap*, const Resource*);
bool transferResource(WebKit::WebGraphicsContext3D*, ResourceId, TransferableResource*);
void trimMailboxDeque();
void deleteResourceInternal(ResourceMap::iterator it);
GraphicsContext* m_context;
ResourceId m_nextId;
ResourceMap m_resources;
int m_nextChild;
ChildMap m_children;
std::deque<Mailbox> m_mailboxes;
ResourceType m_defaultResourceType;
bool m_useTextureStorageExt;
bool m_useTextureUsageHint;
bool m_useShallowFlush;
scoped_ptr<TextureUploader> m_textureUploader;
scoped_ptr<AcceleratedTextureCopier> m_textureCopier;
int m_maxTextureSize;
DISALLOW_COPY_AND_ASSIGN(ResourceProvider);
};
}
#endif
|