2016-03-25 19:15:03 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2016 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef GrBuffer_DEFINED
|
|
|
|
#define GrBuffer_DEFINED
|
|
|
|
|
|
|
|
#include "GrGpuResource.h"
|
|
|
|
|
|
|
|
class GrGpu;
|
|
|
|
|
|
|
|
class GrBuffer : public GrGpuResource {
|
|
|
|
public:
|
|
|
|
/**
|
2016-07-13 17:16:32 +00:00
|
|
|
* Creates a client-side buffer.
|
2016-03-25 19:15:03 +00:00
|
|
|
*/
|
2016-07-13 17:16:32 +00:00
|
|
|
static SK_WARN_UNUSED_RESULT GrBuffer* CreateCPUBacked(GrGpu*, size_t sizeInBytes, GrBufferType,
|
|
|
|
const void* data = nullptr);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Computes a scratch key for a GPU-side buffer with a "dynamic" access pattern. (Buffers with
|
|
|
|
* "static" and "stream" patterns are disqualified by nature from being cached and reused.)
|
|
|
|
*/
|
|
|
|
static void ComputeScratchKeyForDynamicVBO(size_t size, GrBufferType, GrScratchKey*);
|
2016-03-25 19:15:03 +00:00
|
|
|
|
|
|
|
GrAccessPattern accessPattern() const { return fAccessPattern; }
|
2016-07-13 17:16:32 +00:00
|
|
|
size_t sizeInBytes() const { return fSizeInBytes; }
|
2016-03-25 19:15:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if the buffer is a wrapper around a CPU array. If true it
|
|
|
|
* indicates that map will always succeed and will be free.
|
|
|
|
*/
|
2016-07-13 17:16:32 +00:00
|
|
|
bool isCPUBacked() const { return SkToBool(fCPUData); }
|
|
|
|
size_t baseOffset() const { return reinterpret_cast<size_t>(fCPUData); }
|
2016-03-25 19:15:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps the buffer to be written by the CPU.
|
|
|
|
*
|
|
|
|
* The previous content of the buffer is invalidated. It is an error
|
|
|
|
* to draw from the buffer while it is mapped. It may fail if the backend
|
|
|
|
* doesn't support mapping the buffer. If the buffer is CPU backed then
|
|
|
|
* it will always succeed and is a free operation. Once a buffer is mapped,
|
|
|
|
* subsequent calls to map() are ignored.
|
|
|
|
*
|
|
|
|
* Note that buffer mapping does not go through GrContext and therefore is
|
|
|
|
* not serialized with other operations.
|
|
|
|
*
|
|
|
|
* @return a pointer to the data or nullptr if the map fails.
|
|
|
|
*/
|
|
|
|
void* map() {
|
|
|
|
if (!fMapPtr) {
|
|
|
|
this->onMap();
|
|
|
|
}
|
|
|
|
return fMapPtr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unmaps the buffer.
|
|
|
|
*
|
|
|
|
* The pointer returned by the previous map call will no longer be valid.
|
|
|
|
*/
|
|
|
|
void unmap() {
|
|
|
|
SkASSERT(fMapPtr);
|
|
|
|
this->onUnmap();
|
|
|
|
fMapPtr = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the same ptr that map() returned at time of map or nullptr if the
|
|
|
|
* is not mapped.
|
|
|
|
*
|
|
|
|
* @return ptr to mapped buffer data or nullptr if buffer is not mapped.
|
|
|
|
*/
|
|
|
|
void* mapPtr() const { return fMapPtr; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
Queries whether the buffer has been mapped.
|
|
|
|
|
|
|
|
@return true if the buffer is mapped, false otherwise.
|
|
|
|
*/
|
|
|
|
bool isMapped() const { return SkToBool(fMapPtr); }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the buffer data.
|
|
|
|
*
|
|
|
|
* The size of the buffer will be preserved. The src data will be
|
|
|
|
* placed at the beginning of the buffer and any remaining contents will
|
|
|
|
* be undefined. srcSizeInBytes must be <= to the buffer size.
|
|
|
|
*
|
|
|
|
* The buffer must not be mapped.
|
|
|
|
*
|
|
|
|
* Note that buffer updates do not go through GrContext and therefore are
|
|
|
|
* not serialized with other operations.
|
|
|
|
*
|
|
|
|
* @return returns true if the update succeeds, false otherwise.
|
|
|
|
*/
|
|
|
|
bool updateData(const void* src, size_t srcSizeInBytes) {
|
|
|
|
SkASSERT(!this->isMapped());
|
2016-07-13 17:16:32 +00:00
|
|
|
SkASSERT(srcSizeInBytes <= fSizeInBytes);
|
2016-03-25 19:15:03 +00:00
|
|
|
return this->onUpdateData(src, srcSizeInBytes);
|
|
|
|
}
|
|
|
|
|
2016-07-13 17:16:32 +00:00
|
|
|
~GrBuffer() override {
|
|
|
|
sk_free(fCPUData);
|
Refactor to separate backend object lifecycle and GpuResource budget decision
Refactor GrGpuResource to contain two different pieces of state:
a) instance is budgeted or not budgeted
b) instance references wrapped backend objects or not
The "object lifecycle" was also attached to backend object
handles (ids), which made the code a bit unclear. Backend objects
would be associated with GrGpuResource::LifeCycle, even though
GrGpuResource::LifeCycle refers to the GpuResource, and individual
backend objects in one GpuResource might be governed with different
"lifecycle".
Mark the budgeted/not budgeted with SkBudgeted::kYes, SkBudgeted::kNo.
This was previously GrGpuResource::kCached_LifeCycle,
GrGpuResource::kUncached_LifeCycle.
Mark the "references wrapped object" with boolean. This was previously
GrGpuResource::kBorrowed_LifeCycle,
GrGpuResource::kAdopted_LifeCycle for GrGpuResource.
Associate the backend object ownership status with
GrBackendObjectOwnership for the backend object handles.
The resource type leaf constuctors, such has GrGLTexture or
GrGLTextureRenderTarget take "budgeted" parameter. This parameter
is passed to GrGpuResource::registerWithCache().
The resource type intermediary constructors, such as GrGLTexture
constructors for class GrGLTextureRenderTarget do not take "budgeted"
parameters, intermediary construtors do not call registerWithCache.
Removes the need for tagging GrGpuResource -derived subclass
constructors with "Derived" parameter.
Makes instances that wrap backend objects be registered with
a new function GrGpuResource::registerWithCacheWrapped().
Removes "budgeted" parameter from classes such as StencilAttahment, as
they are always cached and never wrap any external backend objects.
Removes the use of concept "external" from the member function names.
The API refers to the objects as "wrapped", so make all related
functions use the term consistently.
No change in functionality. Resources referencing wrapped objects are
always inserted to the cache with budget decision kNo.
BUG=594928
GOLD_TRYBOT_URL= https://gold.skia.org/search2?unt=true&query=source_type%3Dgm&master=false&issue=1862043002
Review URL: https://codereview.chromium.org/1862043002
2016-04-22 08:48:29 +00:00
|
|
|
}
|
|
|
|
|
2016-07-13 17:16:32 +00:00
|
|
|
protected:
|
|
|
|
GrBuffer(GrGpu*, size_t sizeInBytes, GrBufferType, GrAccessPattern);
|
2016-03-25 19:15:03 +00:00
|
|
|
|
|
|
|
void* fMapPtr;
|
|
|
|
|
|
|
|
private:
|
2016-07-13 17:16:32 +00:00
|
|
|
/**
|
|
|
|
* Internal constructor to make a CPU-backed buffer.
|
|
|
|
*/
|
|
|
|
GrBuffer(GrGpu*, size_t sizeInBytes, GrBufferType, void* cpuData);
|
2016-03-25 19:15:03 +00:00
|
|
|
|
2016-07-13 17:16:32 +00:00
|
|
|
virtual void onMap() { SkASSERT(this->isCPUBacked()); fMapPtr = fCPUData; }
|
|
|
|
virtual void onUnmap() { SkASSERT(this->isCPUBacked()); }
|
|
|
|
virtual bool onUpdateData(const void* src, size_t srcSizeInBytes);
|
2016-03-25 19:15:03 +00:00
|
|
|
|
2016-07-13 17:16:32 +00:00
|
|
|
size_t onGpuMemorySize() const override { return fSizeInBytes; } // TODO: zero for cpu backed?
|
|
|
|
void computeScratchKey(GrScratchKey* key) const override;
|
|
|
|
|
|
|
|
size_t fSizeInBytes;
|
2016-03-25 19:15:03 +00:00
|
|
|
GrAccessPattern fAccessPattern;
|
2016-07-13 17:16:32 +00:00
|
|
|
void* fCPUData;
|
Refactor to separate backend object lifecycle and GpuResource budget decision
Refactor GrGpuResource to contain two different pieces of state:
a) instance is budgeted or not budgeted
b) instance references wrapped backend objects or not
The "object lifecycle" was also attached to backend object
handles (ids), which made the code a bit unclear. Backend objects
would be associated with GrGpuResource::LifeCycle, even though
GrGpuResource::LifeCycle refers to the GpuResource, and individual
backend objects in one GpuResource might be governed with different
"lifecycle".
Mark the budgeted/not budgeted with SkBudgeted::kYes, SkBudgeted::kNo.
This was previously GrGpuResource::kCached_LifeCycle,
GrGpuResource::kUncached_LifeCycle.
Mark the "references wrapped object" with boolean. This was previously
GrGpuResource::kBorrowed_LifeCycle,
GrGpuResource::kAdopted_LifeCycle for GrGpuResource.
Associate the backend object ownership status with
GrBackendObjectOwnership for the backend object handles.
The resource type leaf constuctors, such has GrGLTexture or
GrGLTextureRenderTarget take "budgeted" parameter. This parameter
is passed to GrGpuResource::registerWithCache().
The resource type intermediary constructors, such as GrGLTexture
constructors for class GrGLTextureRenderTarget do not take "budgeted"
parameters, intermediary construtors do not call registerWithCache.
Removes the need for tagging GrGpuResource -derived subclass
constructors with "Derived" parameter.
Makes instances that wrap backend objects be registered with
a new function GrGpuResource::registerWithCacheWrapped().
Removes "budgeted" parameter from classes such as StencilAttahment, as
they are always cached and never wrap any external backend objects.
Removes the use of concept "external" from the member function names.
The API refers to the objects as "wrapped", so make all related
functions use the term consistently.
No change in functionality. Resources referencing wrapped objects are
always inserted to the cache with budget decision kNo.
BUG=594928
GOLD_TRYBOT_URL= https://gold.skia.org/search2?unt=true&query=source_type%3Dgm&master=false&issue=1862043002
Review URL: https://codereview.chromium.org/1862043002
2016-04-22 08:48:29 +00:00
|
|
|
GrBufferType fIntendedType;
|
2016-07-13 17:16:32 +00:00
|
|
|
|
2016-03-25 19:15:03 +00:00
|
|
|
typedef GrGpuResource INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|