blob: 0a063067e24b84107c1bf6096e283d4c7235ef84 [file] [log] [blame]
/*
* Copyright 2017 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef GrMockCaps_DEFINED
#define GrMockCaps_DEFINED
#include "include/gpu/mock/GrMockTypes.h"
#include "src/gpu/GrCaps.h"
#include "src/gpu/SkGr.h"
class GrMockCaps : public GrCaps {
public:
GrMockCaps(const GrContextOptions& contextOptions, const GrMockOptions& options)
: INHERITED(contextOptions), fOptions(options) {
fMipmapSupport = options.fMipmapSupport;
fDrawInstancedSupport = options.fDrawInstancedSupport;
fHalfFloatVertexAttributeSupport = options.fHalfFloatVertexAttributeSupport;
fMapBufferFlags = options.fMapBufferFlags;
fBufferMapThreshold = SK_MaxS32; // Overridable in GrContextOptions.
fMaxTextureSize = options.fMaxTextureSize;
fMaxWindowRectangles = options.fMaxWindowRectangles;
fMaxRenderTargetSize = std::min(options.fMaxRenderTargetSize, fMaxTextureSize);
fMaxPreferredRenderTargetSize = fMaxRenderTargetSize;
fMaxVertexAttributes = options.fMaxVertexAttributes;
fSampleLocationsSupport = true;
fShaderCaps.reset(new GrShaderCaps(contextOptions));
fShaderCaps->fGeometryShaderSupport = options.fGeometryShaderSupport;
fShaderCaps->fIntegerSupport = options.fIntegerSupport;
fShaderCaps->fFlatInterpolationSupport = options.fFlatInterpolationSupport;
fShaderCaps->fMaxFragmentSamplers = options.fMaxFragmentSamplers;
fShaderCaps->fShaderDerivativeSupport = options.fShaderDerivativeSupport;
fShaderCaps->fDualSourceBlendingSupport = options.fDualSourceBlendingSupport;
fShaderCaps->fSampleMaskSupport = true;
fShaderCaps->fMaxTessellationSegments = options.fMaxTessellationSegments;
this->finishInitialization(contextOptions);
}
bool isFormatSRGB(const GrBackendFormat& format) const override {
SkImage::CompressionType compression = format.asMockCompressionType();
if (compression != SkImage::CompressionType::kNone) {
return false;
}
auto ct = format.asMockColorType();
return GrGetColorTypeDesc(ct).encoding() == GrColorTypeEncoding::kSRGBUnorm;
}
bool isFormatTexturable(const GrBackendFormat& format) const override {
SkImage::CompressionType compression = format.asMockCompressionType();
if (compression != SkImage::CompressionType::kNone) {
return fOptions.fCompressedOptions[(int)compression].fTexturable;
}
auto index = static_cast<int>(format.asMockColorType());
return fOptions.fConfigOptions[index].fTexturable;
}
bool isFormatCopyable(const GrBackendFormat& format) const override {
return false;
}
bool isFormatAsColorTypeRenderable(GrColorType ct, const GrBackendFormat& format,
int sampleCount = 1) const override {
// Currently we don't allow RGB_888X to be renderable because we don't have a way to
// handle blends that reference dst alpha when the values in the dst alpha channel are
// uninitialized.
if (ct == GrColorType::kRGB_888x) {
return false;
}
return this->isFormatRenderable(format, sampleCount);
}
bool isFormatRenderable(const GrBackendFormat& format, int sampleCount) const override {
if (format.asMockCompressionType() != SkImage::CompressionType::kNone) {
return false; // compressed formats are never renderable
}
return sampleCount <= this->maxRenderTargetSampleCount(format.asMockColorType());
}
int getRenderTargetSampleCount(int requestCount, GrColorType ct) const {
requestCount = std::max(requestCount, 1);
switch (fOptions.fConfigOptions[(int)ct].fRenderability) {
case GrMockOptions::ConfigOptions::Renderability::kNo:
return 0;
case GrMockOptions::ConfigOptions::Renderability::kNonMSAA:
return requestCount > 1 ? 0 : 1;
case GrMockOptions::ConfigOptions::Renderability::kMSAA:
return requestCount > kMaxSampleCnt ? 0 : GrNextPow2(requestCount);
}
return 0;
}
int getRenderTargetSampleCount(int requestCount,
const GrBackendFormat& format) const override {
SkImage::CompressionType compression = format.asMockCompressionType();
if (compression != SkImage::CompressionType::kNone) {
return 0; // no compressed format is renderable
}
return this->getRenderTargetSampleCount(requestCount, format.asMockColorType());
}
int maxRenderTargetSampleCount(GrColorType ct) const {
switch (fOptions.fConfigOptions[(int)ct].fRenderability) {
case GrMockOptions::ConfigOptions::Renderability::kNo:
return 0;
case GrMockOptions::ConfigOptions::Renderability::kNonMSAA:
return 1;
case GrMockOptions::ConfigOptions::Renderability::kMSAA:
return kMaxSampleCnt;
}
return 0;
}
int maxRenderTargetSampleCount(const GrBackendFormat& format) const override {
SkImage::CompressionType compression = format.asMockCompressionType();
if (compression != SkImage::CompressionType::kNone) {
return 0; // no compressed format is renderable
}
return this->maxRenderTargetSampleCount(format.asMockColorType());
}
SupportedWrite supportedWritePixelsColorType(GrColorType surfaceColorType,
const GrBackendFormat& surfaceFormat,
GrColorType srcColorType) const override {
return {surfaceColorType, 1};
}
SurfaceReadPixelsSupport surfaceSupportsReadPixels(const GrSurface*) const override {
return SurfaceReadPixelsSupport::kSupported;
}
GrBackendFormat getBackendFormatFromCompressionType(SkImage::CompressionType) const override {
return {};
}
GrSwizzle getWriteSwizzle(const GrBackendFormat& format, GrColorType ct) const override {
SkASSERT(this->areColorTypeAndFormatCompatible(ct, format));
return GrSwizzle("rgba");
}
uint64_t computeFormatKey(const GrBackendFormat&) const override;
GrProgramDesc makeDesc(GrRenderTarget*,
const GrProgramInfo&,
ProgramDescOverrideFlags) const override;
#if GR_TEST_UTILS
std::vector<GrCaps::TestFormatColorTypeCombination> getTestingCombinations() const override;
#endif
private:
bool onSurfaceSupportsWritePixels(const GrSurface*) const override { return true; }
bool onCanCopySurface(const GrSurfaceProxy* dst, const GrSurfaceProxy* src,
const SkIRect& srcRect, const SkIPoint& dstPoint) const override {
return true;
}
GrBackendFormat onGetDefaultBackendFormat(GrColorType ct) const override {
return GrBackendFormat::MakeMock(ct, SkImage::CompressionType::kNone);
}
bool onAreColorTypeAndFormatCompatible(GrColorType ct,
const GrBackendFormat& format) const override {
if (ct == GrColorType::kUnknown) {
return false;
}
SkImage::CompressionType compression = format.asMockCompressionType();
if (compression == SkImage::CompressionType::kETC2_RGB8_UNORM ||
compression == SkImage::CompressionType::kBC1_RGB8_UNORM) {
return ct == GrColorType::kRGB_888x; // TODO: this may be too restrictive
}
if (compression == SkImage::CompressionType::kBC1_RGBA8_UNORM) {
return ct == GrColorType::kRGBA_8888;
}
return ct == format.asMockColorType();
}
SupportedRead onSupportedReadPixelsColorType(GrColorType srcColorType, const GrBackendFormat&,
GrColorType) const override {
return SupportedRead{srcColorType, 1};
}
GrSwizzle onGetReadSwizzle(const GrBackendFormat& format, GrColorType ct) const override {
SkASSERT(this->areColorTypeAndFormatCompatible(ct, format));
return GrSwizzle("rgba");
}
static const int kMaxSampleCnt = 16;
GrMockOptions fOptions;
using INHERITED = GrCaps;
};
#endif