/* * 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 GrMtlCaps_DEFINED #define GrMtlCaps_DEFINED #include "include/private/base/SkTDArray.h" #include "src/gpu/ganesh/GrCaps.h" #include "src/gpu/ganesh/mtl/GrMtlAttachment.h" #import class GrMtlRenderTarget; /** * Stores some capabilities of a Mtl backend. */ class GrMtlCaps : public GrCaps { public: GrMtlCaps(const GrContextOptions& contextOptions, id device); bool isFormatSRGB(const GrBackendFormat&) const override; bool isFormatTexturable(const GrBackendFormat&, GrTextureType) const override; bool isFormatTexturable(MTLPixelFormat) const; bool isFormatCopyable(const GrBackendFormat&) const override { return true; } bool isFormatAsColorTypeRenderable(GrColorType ct, const GrBackendFormat& format, int sampleCount = 1) const override; bool isFormatRenderable(const GrBackendFormat& format, int sampleCount) const override; bool isFormatRenderable(MTLPixelFormat, int sampleCount) const; int getRenderTargetSampleCount(int requestedCount, const GrBackendFormat&) const override; int getRenderTargetSampleCount(int requestedCount, MTLPixelFormat) const; int maxRenderTargetSampleCount(const GrBackendFormat&) const override; int maxRenderTargetSampleCount(MTLPixelFormat) const; SupportedWrite supportedWritePixelsColorType(GrColorType surfaceColorType, const GrBackendFormat& surfaceFormat, GrColorType srcColorType) const override; SurfaceReadPixelsSupport surfaceSupportsReadPixels(const GrSurface*) const override; DstCopyRestrictions getDstCopyRestrictions(const GrRenderTargetProxy* src, GrColorType ct) const override; /** * Returns both a supported and most prefered stencil format to use in draws. */ MTLPixelFormat preferredStencilFormat() const { return fPreferredStencilFormat; } bool canCopyAsBlit(MTLPixelFormat dstFormat, int dstSampleCount, MTLPixelFormat srcFormat, int srcSampleCount, const SkIRect& srcRect, const SkIPoint& dstPoint, bool areDstSrcSameObj) const; bool canCopyAsResolve(MTLPixelFormat dstFormat, int dstSampleCount, MTLPixelFormat srcFormat, int srcSampleCount, bool srcIsRenderTarget, const SkISize srcDimensions, const SkIRect& srcRect, const SkIPoint& dstPoint, bool areDstSrcSameObj) const; GrBackendFormat getBackendFormatFromCompressionType(SkImage::CompressionType) const override; MTLPixelFormat getFormatFromColorType(GrColorType colorType) const { int idx = static_cast(colorType); return fColorTypeToFormatTable[idx]; } skgpu::Swizzle getWriteSwizzle(const GrBackendFormat&, GrColorType) const override; uint64_t computeFormatKey(const GrBackendFormat&) const override; GrProgramDesc makeDesc(GrRenderTarget*, const GrProgramInfo&, ProgramDescOverrideFlags) const override; MTLPixelFormat getStencilPixelFormat(const GrProgramDesc& desc); bool isMac() const { return fGPUFamily == GPUFamily::kMac; } bool isApple() const { return fGPUFamily == GPUFamily::kApple; } size_t getMinBufferAlignment() const { return this->isMac() ? 4 : 1; } // if true, MTLStoreActionStoreAndMultiplesampleResolve is available bool storeAndMultisampleResolveSupport() const { return fStoreAndMultisampleResolveSupport; } bool renderTargetSupportsDiscardableMSAA(const GrMtlRenderTarget*) const; #if GR_TEST_UTILS std::vector getTestingCombinations() const override; #endif void onDumpJSON(SkJSONWriter*) const override; private: void initGPUFamily(id device); void initStencilFormat(id device); void initGrCaps(id device); void initShaderCaps(); void applyDriverCorrectnessWorkarounds(const GrContextOptions&, const id); void initFormatTable(); bool onSurfaceSupportsWritePixels(const GrSurface*) const override; bool onCanCopySurface(const GrSurfaceProxy* dst, const SkIRect& dstRect, const GrSurfaceProxy* src, const SkIRect& srcRect) const override; GrBackendFormat onGetDefaultBackendFormat(GrColorType) const override; bool onAreColorTypeAndFormatCompatible(GrColorType, const GrBackendFormat&) const override; SupportedRead onSupportedReadPixelsColorType(GrColorType, const GrBackendFormat&, GrColorType) const override; skgpu::Swizzle onGetReadSwizzle(const GrBackendFormat&, GrColorType) const override; // ColorTypeInfo for a specific format struct ColorTypeInfo { GrColorType fColorType = GrColorType::kUnknown; enum { kUploadData_Flag = 0x1, // Does Ganesh itself support rendering to this colorType & format pair. Renderability // still additionally depends on if the format itself is renderable. kRenderable_Flag = 0x2, }; uint32_t fFlags = 0; skgpu::Swizzle fReadSwizzle; skgpu::Swizzle fWriteSwizzle; }; struct FormatInfo { uint32_t colorTypeFlags(GrColorType colorType) const { for (int i = 0; i < fColorTypeInfoCount; ++i) { if (fColorTypeInfos[i].fColorType == colorType) { return fColorTypeInfos[i].fFlags; } } return 0; } enum { kTexturable_Flag = 0x1, kRenderable_Flag = 0x2, // Color attachment and blendable kMSAA_Flag = 0x4, kResolve_Flag = 0x8, }; static const uint16_t kAllFlags = kTexturable_Flag | kRenderable_Flag | kMSAA_Flag | kResolve_Flag; uint16_t fFlags = 0; std::unique_ptr fColorTypeInfos; int fColorTypeInfoCount = 0; }; #ifdef SK_BUILD_FOR_IOS inline static constexpr size_t kNumMtlFormats = 18; #else inline static constexpr size_t kNumMtlFormats = 19; #endif static size_t GetFormatIndex(MTLPixelFormat); FormatInfo fFormatTable[kNumMtlFormats]; const FormatInfo& getFormatInfo(const MTLPixelFormat pixelFormat) const { size_t index = GetFormatIndex(pixelFormat); return fFormatTable[index]; } MTLPixelFormat fColorTypeToFormatTable[kGrColorTypeCnt]; void setColorType(GrColorType, std::initializer_list formats); enum class GPUFamily { kMac, kApple, }; bool getGPUFamily(id device, GPUFamily* gpuFamily, int* group); bool getGPUFamilyFromFeatureSet(id device, GrMtlCaps::GPUFamily* gpuFamily, int* group); GPUFamily fGPUFamily; int fFamilyGroup; SkTDArray fSampleCounts; MTLPixelFormat fPreferredStencilFormat; bool fStoreAndMultisampleResolveSupport : 1; using INHERITED = GrCaps; }; #endif