You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
152 lines
4.9 KiB
152 lines
4.9 KiB
4 months ago
|
/*
|
||
|
* Copyright 2018 Google Inc.
|
||
|
*
|
||
|
* Use of this source code is governed by a BSD-style license that can be
|
||
|
* found in the LICENSE file.
|
||
|
*/
|
||
|
|
||
|
#ifndef GrProxyRef_DEFINED
|
||
|
#define GrProxyRef_DEFINED
|
||
|
|
||
|
#include "GrSurfaceProxy.h"
|
||
|
#include "GrTextureProxy.h"
|
||
|
#include "GrTypesPriv.h"
|
||
|
|
||
|
/**
|
||
|
* Helper for owning a ref and/or pending IO on a GrSurfaceProxy. This is useful when ownership
|
||
|
* must transform from ref'ed to pending IO when the owner is recorded into a GrOpList.
|
||
|
*/
|
||
|
template <typename T> class GrProxyRef {
|
||
|
public:
|
||
|
GrProxyRef() = default;
|
||
|
GrProxyRef(const GrProxyRef&) = delete;
|
||
|
GrProxyRef& operator=(const GrProxyRef&) = delete;
|
||
|
|
||
|
/** ioType expresses what type of IO operations will be marked as pending on the proxy when
|
||
|
markPendingIO is called. */
|
||
|
GrProxyRef(sk_sp<T> proxy, GrIOType ioType) { this->setProxy(std::move(proxy), ioType); }
|
||
|
|
||
|
~GrProxyRef() { this->reset(); }
|
||
|
|
||
|
/** ioType expresses what type of IO operations will be marked as
|
||
|
pending on the proxy when markPendingIO is called. */
|
||
|
void setProxy(sk_sp<T> proxy, GrIOType ioType) {
|
||
|
SkASSERT(!fPendingIO);
|
||
|
SkASSERT(SkToBool(fProxy) == fOwnRef);
|
||
|
SkSafeUnref(fProxy);
|
||
|
if (!proxy) {
|
||
|
fProxy = nullptr;
|
||
|
fOwnRef = false;
|
||
|
} else {
|
||
|
fProxy = proxy.release(); // due to the semantics of this class we unpack from sk_sp
|
||
|
fOwnRef = true;
|
||
|
fIOType = ioType;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
T* get() const { return fProxy; }
|
||
|
|
||
|
/** Does this object own a pending read or write on the resource it is wrapping. */
|
||
|
bool ownsPendingIO() const { return fPendingIO; }
|
||
|
|
||
|
/** What type of IO does this represent? This is independent of whether a normal ref or a
|
||
|
pending IO is currently held. */
|
||
|
GrIOType ioType() const { return fIOType; }
|
||
|
|
||
|
/** Shortcut for calling setProxy() with NULL. It cannot be called after markingPendingIO
|
||
|
is called. */
|
||
|
void reset() {
|
||
|
if (fPendingIO) {
|
||
|
SkASSERT(fProxy);
|
||
|
switch (fIOType) {
|
||
|
case kRead_GrIOType:
|
||
|
fProxy->completedRead();
|
||
|
break;
|
||
|
case kWrite_GrIOType:
|
||
|
fProxy->completedWrite();
|
||
|
break;
|
||
|
case kRW_GrIOType:
|
||
|
fProxy->completedRead();
|
||
|
fProxy->completedWrite();
|
||
|
break;
|
||
|
}
|
||
|
fPendingIO = false;
|
||
|
}
|
||
|
if (fOwnRef) {
|
||
|
SkASSERT(fProxy);
|
||
|
fProxy->unref();
|
||
|
fOwnRef = false;
|
||
|
}
|
||
|
fProxy = nullptr;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Called when transferring into an op list and therefore scheduled for an IO operation. It can
|
||
|
* only be called once.
|
||
|
*/
|
||
|
void markPendingIO() const {
|
||
|
// This should only be called when the owning GrProgramElement gets its first
|
||
|
// pendingExecution ref.
|
||
|
SkASSERT(!fPendingIO);
|
||
|
SkASSERT(fProxy);
|
||
|
fPendingIO = true;
|
||
|
switch (fIOType) {
|
||
|
case kRead_GrIOType:
|
||
|
fProxy->addPendingRead();
|
||
|
break;
|
||
|
case kWrite_GrIOType:
|
||
|
fProxy->addPendingWrite();
|
||
|
break;
|
||
|
case kRW_GrIOType:
|
||
|
fProxy->addPendingRead();
|
||
|
fProxy->addPendingWrite();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/** Called when the program element/draw state is no longer owned by GrOpList-client code.
|
||
|
This lets the cache know that the drawing code will no longer schedule additional reads or
|
||
|
writes to the resource using the program element or draw state. It can only be called once.
|
||
|
*/
|
||
|
void removeRef() const {
|
||
|
SkASSERT(fOwnRef);
|
||
|
SkASSERT(fPendingIO);
|
||
|
SkASSERT(fProxy);
|
||
|
fProxy->unref();
|
||
|
fOwnRef = false;
|
||
|
}
|
||
|
|
||
|
/** Called to indicate that the previous pending IO is complete. Useful when the owning object
|
||
|
still has refs, so it is not about to destroy this GrGpuResourceRef, but its previously
|
||
|
pending executions have been complete. Can only be called if removeRef() was not previously
|
||
|
called. */
|
||
|
void pendingIOComplete() const {
|
||
|
SkASSERT(fOwnRef);
|
||
|
SkASSERT(fPendingIO);
|
||
|
switch (fIOType) {
|
||
|
case kRead_GrIOType:
|
||
|
fProxy->completedRead();
|
||
|
break;
|
||
|
case kWrite_GrIOType:
|
||
|
fProxy->completedWrite();
|
||
|
break;
|
||
|
case kRW_GrIOType:
|
||
|
fProxy->completedRead();
|
||
|
fProxy->completedWrite();
|
||
|
break;
|
||
|
}
|
||
|
fPendingIO = false;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
T* fProxy = nullptr;
|
||
|
mutable bool fOwnRef = false;
|
||
|
mutable bool fPendingIO = false;
|
||
|
GrIOType fIOType = kRead_GrIOType;
|
||
|
};
|
||
|
|
||
|
using GrSurfaceProxyRef = GrProxyRef<GrSurfaceProxy>;
|
||
|
using GrTextureProxyRef = GrProxyRef<GrTextureProxy>;
|
||
|
|
||
|
#endif
|