// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #pragma once #include #include #include "common/assert.h" #include "common/common_types.h" #include "common/logging/log.h" #include "video_core/gpu.h" #include "video_core/textures/texture.h" namespace VideoCore::Surface { enum class PixelFormat { A8B8G8R8_UNORM, A8B8G8R8_SNORM, A8B8G8R8_SINT, A8B8G8R8_UINT, R5G6B5_UNORM, B5G6R5_UNORM, A1R5G5B5_UNORM, A2B10G10R10_UNORM, A2B10G10R10_UINT, A1B5G5R5_UNORM, A5B5G5R1_UNORM, R8_UNORM, R8_SNORM, R8_SINT, R8_UINT, R16G16B16A16_FLOAT, R16G16B16A16_UNORM, R16G16B16A16_SNORM, R16G16B16A16_SINT, R16G16B16A16_UINT, B10G11R11_FLOAT, R32G32B32A32_UINT, BC1_RGBA_UNORM, BC2_UNORM, BC3_UNORM, BC4_UNORM, BC4_SNORM, BC5_UNORM, BC5_SNORM, BC7_UNORM, BC6H_UFLOAT, BC6H_SFLOAT, ASTC_2D_4X4_UNORM, B8G8R8A8_UNORM, R32G32B32A32_FLOAT, R32G32B32A32_SINT, R32G32_FLOAT, R32G32_SINT, R32_FLOAT, R16_FLOAT, R16_UNORM, R16_SNORM, R16_UINT, R16_SINT, R16G16_UNORM, R16G16_FLOAT, R16G16_UINT, R16G16_SINT, R16G16_SNORM, R32G32B32_FLOAT, A8B8G8R8_SRGB, R8G8_UNORM, R8G8_SNORM, R8G8_SINT, R8G8_UINT, R32G32_UINT, R16G16B16X16_FLOAT, R32_UINT, R32_SINT, ASTC_2D_8X8_UNORM, ASTC_2D_8X5_UNORM, ASTC_2D_5X4_UNORM, B8G8R8A8_SRGB, BC1_RGBA_SRGB, BC2_SRGB, BC3_SRGB, BC7_SRGB, A4B4G4R4_UNORM, G4R4_UNORM, ASTC_2D_4X4_SRGB, ASTC_2D_8X8_SRGB, ASTC_2D_8X5_SRGB, ASTC_2D_5X4_SRGB, ASTC_2D_5X5_UNORM, ASTC_2D_5X5_SRGB, ASTC_2D_10X8_UNORM, ASTC_2D_10X8_SRGB, ASTC_2D_6X6_UNORM, ASTC_2D_6X6_SRGB, ASTC_2D_10X10_UNORM, ASTC_2D_10X10_SRGB, ASTC_2D_12X12_UNORM, ASTC_2D_12X12_SRGB, ASTC_2D_8X6_UNORM, ASTC_2D_8X6_SRGB, ASTC_2D_6X5_UNORM, ASTC_2D_6X5_SRGB, E5B9G9R9_FLOAT, MaxColorFormat, // Depth formats D32_FLOAT = MaxColorFormat, D16_UNORM, MaxDepthFormat, // Stencil formats S8_UINT = MaxDepthFormat, MaxStencilFormat, // DepthStencil formats D24_UNORM_S8_UINT = MaxStencilFormat, S8_UINT_D24_UNORM, D32_FLOAT_S8_UINT, MaxDepthStencilFormat, Max = MaxDepthStencilFormat, Invalid = 255, }; constexpr std::size_t MaxPixelFormat = static_cast(PixelFormat::Max); enum class SurfaceType { ColorTexture = 0, Depth = 1, Stencil = 2, DepthStencil = 3, Invalid = 4, }; enum class SurfaceTarget { Texture1D, TextureBuffer, Texture2D, Texture3D, Texture1DArray, Texture2DArray, TextureCubemap, TextureCubeArray, }; constexpr std::array BLOCK_WIDTH_TABLE = {{ 1, // A8B8G8R8_UNORM 1, // A8B8G8R8_SNORM 1, // A8B8G8R8_SINT 1, // A8B8G8R8_UINT 1, // R5G6B5_UNORM 1, // B5G6R5_UNORM 1, // A1R5G5B5_UNORM 1, // A2B10G10R10_UNORM 1, // A2B10G10R10_UINT 1, // A1B5G5R5_UNORM 1, // A5B5G5R1_UNORM 1, // R8_UNORM 1, // R8_SNORM 1, // R8_SINT 1, // R8_UINT 1, // R16G16B16A16_FLOAT 1, // R16G16B16A16_UNORM 1, // R16G16B16A16_SNORM 1, // R16G16B16A16_SINT 1, // R16G16B16A16_UINT 1, // B10G11R11_FLOAT 1, // R32G32B32A32_UINT 4, // BC1_RGBA_UNORM 4, // BC2_UNORM 4, // BC3_UNORM 4, // BC4_UNORM 4, // BC4_SNORM 4, // BC5_UNORM 4, // BC5_SNORM 4, // BC7_UNORM 4, // BC6H_UFLOAT 4, // BC6H_SFLOAT 4, // ASTC_2D_4X4_UNORM 1, // B8G8R8A8_UNORM 1, // R32G32B32A32_FLOAT 1, // R32G32B32A32_SINT 1, // R32G32_FLOAT 1, // R32G32_SINT 1, // R32_FLOAT 1, // R16_FLOAT 1, // R16_UNORM 1, // R16_SNORM 1, // R16_UINT 1, // R16_SINT 1, // R16G16_UNORM 1, // R16G16_FLOAT 1, // R16G16_UINT 1, // R16G16_SINT 1, // R16G16_SNORM 1, // R32G32B32_FLOAT 1, // A8B8G8R8_SRGB 1, // R8G8_UNORM 1, // R8G8_SNORM 1, // R8G8_SINT 1, // R8G8_UINT 1, // R32G32_UINT 1, // R16G16B16X16_FLOAT 1, // R32_UINT 1, // R32_SINT 8, // ASTC_2D_8X8_UNORM 8, // ASTC_2D_8X5_UNORM 5, // ASTC_2D_5X4_UNORM 1, // B8G8R8A8_SRGB 4, // BC1_RGBA_SRGB 4, // BC2_SRGB 4, // BC3_SRGB 4, // BC7_SRGB 1, // A4B4G4R4_UNORM 1, // G4R4_UNORM 4, // ASTC_2D_4X4_SRGB 8, // ASTC_2D_8X8_SRGB 8, // ASTC_2D_8X5_SRGB 5, // ASTC_2D_5X4_SRGB 5, // ASTC_2D_5X5_UNORM 5, // ASTC_2D_5X5_SRGB 10, // ASTC_2D_10X8_UNORM 10, // ASTC_2D_10X8_SRGB 6, // ASTC_2D_6X6_UNORM 6, // ASTC_2D_6X6_SRGB 10, // ASTC_2D_10X10_UNORM 10, // ASTC_2D_10X10_SRGB 12, // ASTC_2D_12X12_UNORM 12, // ASTC_2D_12X12_SRGB 8, // ASTC_2D_8X6_UNORM 8, // ASTC_2D_8X6_SRGB 6, // ASTC_2D_6X5_UNORM 6, // ASTC_2D_6X5_SRGB 1, // E5B9G9R9_FLOAT 1, // D32_FLOAT 1, // D16_UNORM 1, // S8_UINT 1, // D24_UNORM_S8_UINT 1, // S8_UINT_D24_UNORM 1, // D32_FLOAT_S8_UINT }}; constexpr u32 DefaultBlockWidth(PixelFormat format) { ASSERT(static_cast(format) < BLOCK_WIDTH_TABLE.size()); return BLOCK_WIDTH_TABLE[static_cast(format)]; } constexpr std::array BLOCK_HEIGHT_TABLE = {{ 1, // A8B8G8R8_UNORM 1, // A8B8G8R8_SNORM 1, // A8B8G8R8_SINT 1, // A8B8G8R8_UINT 1, // R5G6B5_UNORM 1, // B5G6R5_UNORM 1, // A1R5G5B5_UNORM 1, // A2B10G10R10_UNORM 1, // A2B10G10R10_UINT 1, // A1B5G5R5_UNORM 1, // A5B5G5R1_UNORM 1, // R8_UNORM 1, // R8_SNORM 1, // R8_SINT 1, // R8_UINT 1, // R16G16B16A16_FLOAT 1, // R16G16B16A16_UNORM 1, // R16G16B16A16_SNORM 1, // R16G16B16A16_SINT 1, // R16G16B16A16_UINT 1, // B10G11R11_FLOAT 1, // R32G32B32A32_UINT 4, // BC1_RGBA_UNORM 4, // BC2_UNORM 4, // BC3_UNORM 4, // BC4_UNORM 4, // BC4_SNORM 4, // BC5_UNORM 4, // BC5_SNORM 4, // BC7_UNORM 4, // BC6H_UFLOAT 4, // BC6H_SFLOAT 4, // ASTC_2D_4X4_UNORM 1, // B8G8R8A8_UNORM 1, // R32G32B32A32_FLOAT 1, // R32G32B32A32_SINT 1, // R32G32_FLOAT 1, // R32G32_SINT 1, // R32_FLOAT 1, // R16_FLOAT 1, // R16_UNORM 1, // R16_SNORM 1, // R16_UINT 1, // R16_SINT 1, // R16G16_UNORM 1, // R16G16_FLOAT 1, // R16G16_UINT 1, // R16G16_SINT 1, // R16G16_SNORM 1, // R32G32B32_FLOAT 1, // A8B8G8R8_SRGB 1, // R8G8_UNORM 1, // R8G8_SNORM 1, // R8G8_SINT 1, // R8G8_UINT 1, // R32G32_UINT 1, // R16G16B16X16_FLOAT 1, // R32_UINT 1, // R32_SINT 8, // ASTC_2D_8X8_UNORM 5, // ASTC_2D_8X5_UNORM 4, // ASTC_2D_5X4_UNORM 1, // B8G8R8A8_SRGB 4, // BC1_RGBA_SRGB 4, // BC2_SRGB 4, // BC3_SRGB 4, // BC7_SRGB 1, // A4B4G4R4_UNORM 1, // G4R4_UNORM 4, // ASTC_2D_4X4_SRGB 8, // ASTC_2D_8X8_SRGB 5, // ASTC_2D_8X5_SRGB 4, // ASTC_2D_5X4_SRGB 5, // ASTC_2D_5X5_UNORM 5, // ASTC_2D_5X5_SRGB 8, // ASTC_2D_10X8_UNORM 8, // ASTC_2D_10X8_SRGB 6, // ASTC_2D_6X6_UNORM 6, // ASTC_2D_6X6_SRGB 10, // ASTC_2D_10X10_UNORM 10, // ASTC_2D_10X10_SRGB 12, // ASTC_2D_12X12_UNORM 12, // ASTC_2D_12X12_SRGB 6, // ASTC_2D_8X6_UNORM 6, // ASTC_2D_8X6_SRGB 5, // ASTC_2D_6X5_UNORM 5, // ASTC_2D_6X5_SRGB 1, // E5B9G9R9_FLOAT 1, // D32_FLOAT 1, // D16_UNORM 1, // S8_UINT 1, // D24_UNORM_S8_UINT 1, // S8_UINT_D24_UNORM 1, // D32_FLOAT_S8_UINT }}; constexpr u32 DefaultBlockHeight(PixelFormat format) { ASSERT(static_cast(format) < BLOCK_HEIGHT_TABLE.size()); return BLOCK_HEIGHT_TABLE[static_cast(format)]; } constexpr std::array BITS_PER_BLOCK_TABLE = {{ 32, // A8B8G8R8_UNORM 32, // A8B8G8R8_SNORM 32, // A8B8G8R8_SINT 32, // A8B8G8R8_UINT 16, // R5G6B5_UNORM 16, // B5G6R5_UNORM 16, // A1R5G5B5_UNORM 32, // A2B10G10R10_UNORM 32, // A2B10G10R10_UINT 16, // A1B5G5R5_UNORM 16, // A5B5G5R1_UNORM 8, // R8_UNORM 8, // R8_SNORM 8, // R8_SINT 8, // R8_UINT 64, // R16G16B16A16_FLOAT 64, // R16G16B16A16_UNORM 64, // R16G16B16A16_SNORM 64, // R16G16B16A16_SINT 64, // R16G16B16A16_UINT 32, // B10G11R11_FLOAT 128, // R32G32B32A32_UINT 64, // BC1_RGBA_UNORM 128, // BC2_UNORM 128, // BC3_UNORM 64, // BC4_UNORM 64, // BC4_SNORM 128, // BC5_UNORM 128, // BC5_SNORM 128, // BC7_UNORM 128, // BC6H_UFLOAT 128, // BC6H_SFLOAT 128, // ASTC_2D_4X4_UNORM 32, // B8G8R8A8_UNORM 128, // R32G32B32A32_FLOAT 128, // R32G32B32A32_SINT 64, // R32G32_FLOAT 64, // R32G32_SINT 32, // R32_FLOAT 16, // R16_FLOAT 16, // R16_UNORM 16, // R16_SNORM 16, // R16_UINT 16, // R16_SINT 32, // R16G16_UNORM 32, // R16G16_FLOAT 32, // R16G16_UINT 32, // R16G16_SINT 32, // R16G16_SNORM 96, // R32G32B32_FLOAT 32, // A8B8G8R8_SRGB 16, // R8G8_UNORM 16, // R8G8_SNORM 16, // R8G8_SINT 16, // R8G8_UINT 64, // R32G32_UINT 64, // R16G16B16X16_FLOAT 32, // R32_UINT 32, // R32_SINT 128, // ASTC_2D_8X8_UNORM 128, // ASTC_2D_8X5_UNORM 128, // ASTC_2D_5X4_UNORM 32, // B8G8R8A8_SRGB 64, // BC1_RGBA_SRGB 128, // BC2_SRGB 128, // BC3_SRGB 128, // BC7_UNORM 16, // A4B4G4R4_UNORM 8, // G4R4_UNORM 128, // ASTC_2D_4X4_SRGB 128, // ASTC_2D_8X8_SRGB 128, // ASTC_2D_8X5_SRGB 128, // ASTC_2D_5X4_SRGB 128, // ASTC_2D_5X5_UNORM 128, // ASTC_2D_5X5_SRGB 128, // ASTC_2D_10X8_UNORM 128, // ASTC_2D_10X8_SRGB 128, // ASTC_2D_6X6_UNORM 128, // ASTC_2D_6X6_SRGB 128, // ASTC_2D_10X10_UNORM 128, // ASTC_2D_10X10_SRGB 128, // ASTC_2D_12X12_UNORM 128, // ASTC_2D_12X12_SRGB 128, // ASTC_2D_8X6_UNORM 128, // ASTC_2D_8X6_SRGB 128, // ASTC_2D_6X5_UNORM 128, // ASTC_2D_6X5_SRGB 32, // E5B9G9R9_FLOAT 32, // D32_FLOAT 16, // D16_UNORM 8, // S8_UINT 32, // D24_UNORM_S8_UINT 32, // S8_UINT_D24_UNORM 64, // D32_FLOAT_S8_UINT }}; constexpr u32 BitsPerBlock(PixelFormat format) { ASSERT(static_cast(format) < BITS_PER_BLOCK_TABLE.size()); return BITS_PER_BLOCK_TABLE[static_cast(format)]; } /// Returns the sizer in bytes of the specified pixel format constexpr u32 BytesPerBlock(PixelFormat pixel_format) { return BitsPerBlock(pixel_format) / CHAR_BIT; } SurfaceTarget SurfaceTargetFromTextureType(Tegra::Texture::TextureType texture_type); bool SurfaceTargetIsLayered(SurfaceTarget target); bool SurfaceTargetIsArray(SurfaceTarget target); PixelFormat PixelFormatFromDepthFormat(Tegra::DepthFormat format); PixelFormat PixelFormatFromRenderTargetFormat(Tegra::RenderTargetFormat format); PixelFormat PixelFormatFromGPUPixelFormat(Service::android::PixelFormat format); SurfaceType GetFormatType(PixelFormat pixel_format); bool IsPixelFormatASTC(PixelFormat format); bool IsPixelFormatSRGB(PixelFormat format); bool IsPixelFormatInteger(PixelFormat format); bool IsPixelFormatSignedInteger(PixelFormat format); size_t PixelComponentSizeBitsInteger(PixelFormat format); std::pair GetASTCBlockSize(PixelFormat format); u64 EstimatedDecompressedSize(u64 base_size, PixelFormat format); } // namespace VideoCore::Surface