diff options
| -rw-r--r-- | src/shader_recompiler/backend/spirv/emit_spirv_image.cpp | 22 | ||||
| -rw-r--r-- | src/shader_recompiler/backend/spirv/spirv_emit_context.cpp | 22 | ||||
| -rw-r--r-- | src/shader_recompiler/backend/spirv/spirv_emit_context.h | 2 | ||||
| -rw-r--r-- | src/shader_recompiler/environment.h | 2 | ||||
| -rw-r--r-- | src/shader_recompiler/ir_opt/texture_pass.cpp | 25 | ||||
| -rw-r--r-- | src/shader_recompiler/shader_info.h | 105 | ||||
| -rw-r--r-- | src/video_core/renderer_opengl/gl_shader_cache.cpp | 2 | ||||
| -rw-r--r-- | src/video_core/renderer_vulkan/vk_buffer_cache.cpp | 32 | ||||
| -rw-r--r-- | src/video_core/renderer_vulkan/vk_buffer_cache.h | 2 | ||||
| -rw-r--r-- | src/video_core/renderer_vulkan/vk_pipeline_cache.cpp | 2 | ||||
| -rw-r--r-- | src/video_core/shader_environment.cpp | 35 | ||||
| -rw-r--r-- | src/video_core/shader_environment.h | 6 | 
12 files changed, 210 insertions, 47 deletions
| diff --git a/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp b/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp index 22ceca19c..800754554 100644 --- a/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp +++ b/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp @@ -214,16 +214,16 @@ Id TextureImage(EmitContext& ctx, IR::TextureInstInfo info, const IR::Value& ind      }  } -Id Image(EmitContext& ctx, const IR::Value& index, IR::TextureInstInfo info) { +std::pair<Id, bool> Image(EmitContext& ctx, const IR::Value& index, IR::TextureInstInfo info) {      if (!index.IsImmediate() || index.U32() != 0) {          throw NotImplementedException("Indirect image indexing");      }      if (info.type == TextureType::Buffer) {          const ImageBufferDefinition def{ctx.image_buffers.at(info.descriptor_index)}; -        return ctx.OpLoad(def.image_type, def.id); +        return {ctx.OpLoad(def.image_type, def.id), def.is_integer};      } else {          const ImageDefinition def{ctx.images.at(info.descriptor_index)}; -        return ctx.OpLoad(def.image_type, def.id); +        return {ctx.OpLoad(def.image_type, def.id), def.is_integer};      }  } @@ -566,13 +566,23 @@ Id EmitImageRead(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id co          LOG_WARNING(Shader_SPIRV, "Typeless image read not supported by host");          return ctx.ConstantNull(ctx.U32[4]);      } -    return Emit(&EmitContext::OpImageSparseRead, &EmitContext::OpImageRead, ctx, inst, ctx.U32[4], -                Image(ctx, index, info), coords, std::nullopt, std::span<const Id>{}); +    const auto [image, is_integer] = Image(ctx, index, info); +    const Id result_type{is_integer ? ctx.U32[4] : ctx.F32[4]}; +    Id color{Emit(&EmitContext::OpImageSparseRead, &EmitContext::OpImageRead, ctx, inst, +                  result_type, image, coords, std::nullopt, std::span<const Id>{})}; +    if (!is_integer) { +        color = ctx.OpBitcast(ctx.U32[4], color); +    } +    return color;  }  void EmitImageWrite(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, Id color) {      const auto info{inst->Flags<IR::TextureInstInfo>()}; -    ctx.OpImageWrite(Image(ctx, index, info), coords, color); +    const auto [image, is_integer] = Image(ctx, index, info); +    if (!is_integer) { +        color = ctx.OpBitcast(ctx.F32[4], color); +    } +    ctx.OpImageWrite(image, coords, color);  }  Id EmitIsTextureScaled(EmitContext& ctx, const IR::Value& index) { diff --git a/src/shader_recompiler/backend/spirv/spirv_emit_context.cpp b/src/shader_recompiler/backend/spirv/spirv_emit_context.cpp index 2abc21a17..ed023fcfe 100644 --- a/src/shader_recompiler/backend/spirv/spirv_emit_context.cpp +++ b/src/shader_recompiler/backend/spirv/spirv_emit_context.cpp @@ -74,20 +74,19 @@ spv::ImageFormat GetImageFormat(ImageFormat format) {      throw InvalidArgument("Invalid image format {}", format);  } -Id ImageType(EmitContext& ctx, const ImageDescriptor& desc) { +Id ImageType(EmitContext& ctx, const ImageDescriptor& desc, Id sampled_type) {      const spv::ImageFormat format{GetImageFormat(desc.format)}; -    const Id type{ctx.U32[1]};      switch (desc.type) {      case TextureType::Color1D: -        return ctx.TypeImage(type, spv::Dim::Dim1D, false, false, false, 2, format); +        return ctx.TypeImage(sampled_type, spv::Dim::Dim1D, false, false, false, 2, format);      case TextureType::ColorArray1D: -        return ctx.TypeImage(type, spv::Dim::Dim1D, false, true, false, 2, format); +        return ctx.TypeImage(sampled_type, spv::Dim::Dim1D, false, true, false, 2, format);      case TextureType::Color2D: -        return ctx.TypeImage(type, spv::Dim::Dim2D, false, false, false, 2, format); +        return ctx.TypeImage(sampled_type, spv::Dim::Dim2D, false, false, false, 2, format);      case TextureType::ColorArray2D: -        return ctx.TypeImage(type, spv::Dim::Dim2D, false, true, false, 2, format); +        return ctx.TypeImage(sampled_type, spv::Dim::Dim2D, false, true, false, 2, format);      case TextureType::Color3D: -        return ctx.TypeImage(type, spv::Dim::Dim3D, false, false, false, 2, format); +        return ctx.TypeImage(sampled_type, spv::Dim::Dim3D, false, false, false, 2, format);      case TextureType::Buffer:          throw NotImplementedException("Image buffer");      default: @@ -1273,7 +1272,9 @@ void EmitContext::DefineImageBuffers(const Info& info, u32& binding) {              throw NotImplementedException("Array of image buffers");          }          const spv::ImageFormat format{GetImageFormat(desc.format)}; -        const Id image_type{TypeImage(U32[1], spv::Dim::Buffer, false, false, false, 2, format)}; +        const Id sampled_type{desc.is_integer ? U32[1] : F32[1]}; +        const Id image_type{ +            TypeImage(sampled_type, spv::Dim::Buffer, false, false, false, 2, format)};          const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, image_type)};          const Id id{AddGlobalVariable(pointer_type, spv::StorageClass::UniformConstant)};          Decorate(id, spv::Decoration::Binding, binding); @@ -1283,6 +1284,7 @@ void EmitContext::DefineImageBuffers(const Info& info, u32& binding) {              .id = id,              .image_type = image_type,              .count = desc.count, +            .is_integer = desc.is_integer,          });          if (profile.supported_spirv >= 0x00010400) {              interfaces.push_back(id); @@ -1327,7 +1329,8 @@ void EmitContext::DefineImages(const Info& info, u32& binding, u32& scaling_inde          if (desc.count != 1) {              throw NotImplementedException("Array of images");          } -        const Id image_type{ImageType(*this, desc)}; +        const Id sampled_type{desc.is_integer ? U32[1] : F32[1]}; +        const Id image_type{ImageType(*this, desc, sampled_type)};          const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, image_type)};          const Id id{AddGlobalVariable(pointer_type, spv::StorageClass::UniformConstant)};          Decorate(id, spv::Decoration::Binding, binding); @@ -1337,6 +1340,7 @@ void EmitContext::DefineImages(const Info& info, u32& binding, u32& scaling_inde              .id = id,              .image_type = image_type,              .count = desc.count, +            .is_integer = desc.is_integer,          });          if (profile.supported_spirv >= 0x00010400) {              interfaces.push_back(id); diff --git a/src/shader_recompiler/backend/spirv/spirv_emit_context.h b/src/shader_recompiler/backend/spirv/spirv_emit_context.h index 1aa79863d..56019ad89 100644 --- a/src/shader_recompiler/backend/spirv/spirv_emit_context.h +++ b/src/shader_recompiler/backend/spirv/spirv_emit_context.h @@ -47,12 +47,14 @@ struct ImageBufferDefinition {      Id id;      Id image_type;      u32 count; +    bool is_integer;  };  struct ImageDefinition {      Id id;      Id image_type;      u32 count; +    bool is_integer;  };  struct UniformDefinitions { diff --git a/src/shader_recompiler/environment.h b/src/shader_recompiler/environment.h index 15285ab0a..e30bf094a 100644 --- a/src/shader_recompiler/environment.h +++ b/src/shader_recompiler/environment.h @@ -24,6 +24,8 @@ public:      [[nodiscard]] virtual TexturePixelFormat ReadTexturePixelFormat(u32 raw_handle) = 0; +    [[nodiscard]] virtual bool IsTexturePixelFormatInteger(u32 raw_handle) = 0; +      [[nodiscard]] virtual u32 ReadViewportTransformState() = 0;      [[nodiscard]] virtual u32 TextureBoundBuffer() const = 0; diff --git a/src/shader_recompiler/ir_opt/texture_pass.cpp b/src/shader_recompiler/ir_opt/texture_pass.cpp index d374c976a..100437f0e 100644 --- a/src/shader_recompiler/ir_opt/texture_pass.cpp +++ b/src/shader_recompiler/ir_opt/texture_pass.cpp @@ -372,6 +372,10 @@ TexturePixelFormat ReadTexturePixelFormat(Environment& env, const ConstBufferAdd      return env.ReadTexturePixelFormat(GetTextureHandle(env, cbuf));  } +bool IsTexturePixelFormatInteger(Environment& env, const ConstBufferAddr& cbuf) { +    return env.IsTexturePixelFormatInteger(GetTextureHandle(env, cbuf)); +} +  class Descriptors {  public:      explicit Descriptors(TextureBufferDescriptors& texture_buffer_descriptors_, @@ -403,6 +407,7 @@ public:          })};          image_buffer_descriptors[index].is_written |= desc.is_written;          image_buffer_descriptors[index].is_read |= desc.is_read; +        image_buffer_descriptors[index].is_integer |= desc.is_integer;          return index;      } @@ -432,6 +437,7 @@ public:          })};          image_descriptors[index].is_written |= desc.is_written;          image_descriptors[index].is_read |= desc.is_read; +        image_descriptors[index].is_integer |= desc.is_integer;          return index;      } @@ -469,6 +475,20 @@ void PatchImageSampleImplicitLod(IR::Block& block, IR::Inst& inst) {                          ir.FPRecip(ir.ConvertUToF(32, 32, ir.CompositeExtract(texture_size, 1))))));  } +bool IsPixelFormatSNorm(TexturePixelFormat pixel_format) { +    switch (pixel_format) { +    case TexturePixelFormat::A8B8G8R8_SNORM: +    case TexturePixelFormat::R8G8_SNORM: +    case TexturePixelFormat::R8_SNORM: +    case TexturePixelFormat::R16G16B16A16_SNORM: +    case TexturePixelFormat::R16G16_SNORM: +    case TexturePixelFormat::R16_SNORM: +        return true; +    default: +        return false; +    } +} +  void PatchTexelFetch(IR::Block& block, IR::Inst& inst, TexturePixelFormat pixel_format) {      const auto it{IR::Block::InstructionList::s_iterator_to(inst)};      IR::IREmitter ir{block, IR::Block::InstructionList::s_iterator_to(inst)}; @@ -587,11 +607,13 @@ void TexturePass(Environment& env, IR::Program& program, const HostTranslateInfo              }              const bool is_written{inst->GetOpcode() != IR::Opcode::ImageRead};              const bool is_read{inst->GetOpcode() != IR::Opcode::ImageWrite}; +            const bool is_integer{IsTexturePixelFormatInteger(env, cbuf)};              if (flags.type == TextureType::Buffer) {                  index = descriptors.Add(ImageBufferDescriptor{                      .format = flags.image_format,                      .is_written = is_written,                      .is_read = is_read, +                    .is_integer = is_integer,                      .cbuf_index = cbuf.index,                      .cbuf_offset = cbuf.offset,                      .count = cbuf.count, @@ -603,6 +625,7 @@ void TexturePass(Environment& env, IR::Program& program, const HostTranslateInfo                      .format = flags.image_format,                      .is_written = is_written,                      .is_read = is_read, +                    .is_integer = is_integer,                      .cbuf_index = cbuf.index,                      .cbuf_offset = cbuf.offset,                      .count = cbuf.count, @@ -658,7 +681,7 @@ void TexturePass(Environment& env, IR::Program& program, const HostTranslateInfo          if (!host_info.support_snorm_render_buffer && inst->GetOpcode() == IR::Opcode::ImageFetch &&              flags.type == TextureType::Buffer) {              const auto pixel_format = ReadTexturePixelFormat(env, cbuf); -            if (pixel_format != TexturePixelFormat::OTHER) { +            if (IsPixelFormatSNorm(pixel_format)) {                  PatchTexelFetch(*texture_inst.block, *texture_inst.inst, pixel_format);              }          } diff --git a/src/shader_recompiler/shader_info.h b/src/shader_recompiler/shader_info.h index 1419b8fe7..ed13e6820 100644 --- a/src/shader_recompiler/shader_info.h +++ b/src/shader_recompiler/shader_info.h @@ -35,14 +35,109 @@ enum class TextureType : u32 {  };  constexpr u32 NUM_TEXTURE_TYPES = 9; -enum class TexturePixelFormat : u32 { +enum class TexturePixelFormat { +    A8B8G8R8_UNORM,      A8B8G8R8_SNORM, +    A8B8G8R8_SINT, +    A8B8G8R8_UINT, +    R5G6B5_UNORM, +    B5G6R5_UNORM, +    A1R5G5B5_UNORM, +    A2B10G10R10_UNORM, +    A2B10G10R10_UINT, +    A2R10G10B10_UNORM, +    A1B5G5R5_UNORM, +    A5B5G5R1_UNORM, +    R8_UNORM,      R8_SNORM, -    R8G8_SNORM, +    R8_SINT, +    R8_UINT, +    R16G16B16A16_FLOAT, +    R16G16B16A16_UNORM,      R16G16B16A16_SNORM, -    R16G16_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, -    OTHER +    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_10X6_UNORM, +    ASTC_2D_10X6_SRGB, +    ASTC_2D_10X5_UNORM, +    ASTC_2D_10X5_SRGB, +    ASTC_2D_10X10_UNORM, +    ASTC_2D_10X10_SRGB, +    ASTC_2D_12X10_UNORM, +    ASTC_2D_12X10_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, +    D32_FLOAT, +    D16_UNORM, +    X8_D24_UNORM, +    S8_UINT, +    D24_UNORM_S8_UINT, +    S8_UINT_D24_UNORM, +    D32_FLOAT_S8_UINT,  };  enum class ImageFormat : u32 { @@ -97,6 +192,7 @@ struct ImageBufferDescriptor {      ImageFormat format;      bool is_written;      bool is_read; +    bool is_integer;      u32 cbuf_index;      u32 cbuf_offset;      u32 count; @@ -129,6 +225,7 @@ struct ImageDescriptor {      ImageFormat format;      bool is_written;      bool is_read; +    bool is_integer;      u32 cbuf_index;      u32 cbuf_offset;      u32 count; diff --git a/src/video_core/renderer_opengl/gl_shader_cache.cpp b/src/video_core/renderer_opengl/gl_shader_cache.cpp index b5999362a..30df41b7d 100644 --- a/src/video_core/renderer_opengl/gl_shader_cache.cpp +++ b/src/video_core/renderer_opengl/gl_shader_cache.cpp @@ -51,7 +51,7 @@ using VideoCommon::LoadPipelines;  using VideoCommon::SerializePipeline;  using Context = ShaderContext::Context; -constexpr u32 CACHE_VERSION = 9; +constexpr u32 CACHE_VERSION = 10;  template <typename Container>  auto MakeSpan(Container& container) { diff --git a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp index 2267069e7..3c61799fa 100644 --- a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp +++ b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp @@ -78,8 +78,15 @@ vk::Buffer CreateBuffer(const Device& device, const MemoryAllocator& memory_allo  }  } // Anonymous namespace -Buffer::Buffer(BufferCacheRuntime&, VideoCommon::NullBufferParams null_params) -    : VideoCommon::BufferBase<VideoCore::RasterizerInterface>(null_params), tracker{4096} {} +Buffer::Buffer(BufferCacheRuntime& runtime, VideoCommon::NullBufferParams null_params) +    : VideoCommon::BufferBase<VideoCore::RasterizerInterface>(null_params), tracker{4096} { +    if (runtime.device.HasNullDescriptor()) { +        return; +    } +    device = &runtime.device; +    buffer = runtime.CreateNullBuffer(); +    is_null = true; +}  Buffer::Buffer(BufferCacheRuntime& runtime, VideoCore::RasterizerInterface& rasterizer_,                 VAddr cpu_addr_, u64 size_bytes_) @@ -93,8 +100,12 @@ Buffer::Buffer(BufferCacheRuntime& runtime, VideoCore::RasterizerInterface& rast  VkBufferView Buffer::View(u32 offset, u32 size, VideoCore::Surface::PixelFormat format) {      if (!device) { -        // Null buffer, return a null descriptor +        // Null buffer supported, return a null descriptor          return VK_NULL_HANDLE; +    } else if (is_null) { +        // Null buffer not supported, adjust offset and size +        offset = 0; +        size = 0;      }      const auto it{std::ranges::find_if(views, [offset, size, format](const BufferView& view) {          return offset == view.offset && size == view.size && format == view.format; @@ -627,9 +638,12 @@ void BufferCacheRuntime::BindTransformFeedbackBuffers(VideoCommon::HostBindings<  }  void BufferCacheRuntime::ReserveNullBuffer() { -    if (null_buffer) { -        return; +    if (!null_buffer) { +        null_buffer = CreateNullBuffer();      } +} + +vk::Buffer BufferCacheRuntime::CreateNullBuffer() {      VkBufferCreateInfo create_info{          .sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,          .pNext = nullptr, @@ -644,15 +658,17 @@ void BufferCacheRuntime::ReserveNullBuffer() {      if (device.IsExtTransformFeedbackSupported()) {          create_info.usage |= VK_BUFFER_USAGE_TRANSFORM_FEEDBACK_BUFFER_BIT_EXT;      } -    null_buffer = memory_allocator.CreateBuffer(create_info, MemoryUsage::DeviceLocal); +    vk::Buffer ret = memory_allocator.CreateBuffer(create_info, MemoryUsage::DeviceLocal);      if (device.HasDebuggingToolAttached()) { -        null_buffer.SetObjectNameEXT("Null buffer"); +        ret.SetObjectNameEXT("Null buffer");      }      scheduler.RequestOutsideRenderPassOperationContext(); -    scheduler.Record([buffer = *null_buffer](vk::CommandBuffer cmdbuf) { +    scheduler.Record([buffer = *ret](vk::CommandBuffer cmdbuf) {          cmdbuf.FillBuffer(buffer, 0, VK_WHOLE_SIZE, 0);      }); + +    return ret;  }  } // namespace Vulkan diff --git a/src/video_core/renderer_vulkan/vk_buffer_cache.h b/src/video_core/renderer_vulkan/vk_buffer_cache.h index 0b3fbd6d0..dc300d7cb 100644 --- a/src/video_core/renderer_vulkan/vk_buffer_cache.h +++ b/src/video_core/renderer_vulkan/vk_buffer_cache.h @@ -63,6 +63,7 @@ private:      vk::Buffer buffer;      std::vector<BufferView> views;      VideoCommon::UsageTracker tracker; +    bool is_null{};  };  class QuadArrayIndexBuffer; @@ -151,6 +152,7 @@ private:      }      void ReserveNullBuffer(); +    vk::Buffer CreateNullBuffer();      const Device& device;      MemoryAllocator& memory_allocator; diff --git a/src/video_core/renderer_vulkan/vk_pipeline_cache.cpp b/src/video_core/renderer_vulkan/vk_pipeline_cache.cpp index fa63d6228..d1841198d 100644 --- a/src/video_core/renderer_vulkan/vk_pipeline_cache.cpp +++ b/src/video_core/renderer_vulkan/vk_pipeline_cache.cpp @@ -54,7 +54,7 @@ using VideoCommon::FileEnvironment;  using VideoCommon::GenericEnvironment;  using VideoCommon::GraphicsEnvironment; -constexpr u32 CACHE_VERSION = 10; +constexpr u32 CACHE_VERSION = 11;  constexpr std::array<char, 8> VULKAN_CACHE_MAGIC_NUMBER{'y', 'u', 'z', 'u', 'v', 'k', 'c', 'h'};  template <typename Container> diff --git a/src/video_core/shader_environment.cpp b/src/video_core/shader_environment.cpp index 4edbe5700..492440ac4 100644 --- a/src/video_core/shader_environment.cpp +++ b/src/video_core/shader_environment.cpp @@ -62,23 +62,9 @@ static Shader::TextureType ConvertTextureType(const Tegra::Texture::TICEntry& en  }  static Shader::TexturePixelFormat ConvertTexturePixelFormat(const Tegra::Texture::TICEntry& entry) { -    switch (PixelFormatFromTextureInfo(entry.format, entry.r_type, entry.g_type, entry.b_type, -                                       entry.a_type, entry.srgb_conversion)) { -    case VideoCore::Surface::PixelFormat::A8B8G8R8_SNORM: -        return Shader::TexturePixelFormat::A8B8G8R8_SNORM; -    case VideoCore::Surface::PixelFormat::R8_SNORM: -        return Shader::TexturePixelFormat::R8_SNORM; -    case VideoCore::Surface::PixelFormat::R8G8_SNORM: -        return Shader::TexturePixelFormat::R8G8_SNORM; -    case VideoCore::Surface::PixelFormat::R16G16B16A16_SNORM: -        return Shader::TexturePixelFormat::R16G16B16A16_SNORM; -    case VideoCore::Surface::PixelFormat::R16G16_SNORM: -        return Shader::TexturePixelFormat::R16G16_SNORM; -    case VideoCore::Surface::PixelFormat::R16_SNORM: -        return Shader::TexturePixelFormat::R16_SNORM; -    default: -        return Shader::TexturePixelFormat::OTHER; -    } +    return static_cast<Shader::TexturePixelFormat>( +        PixelFormatFromTextureInfo(entry.format, entry.r_type, entry.g_type, entry.b_type, +                                   entry.a_type, entry.srgb_conversion));  }  static std::string_view StageToPrefix(Shader::Stage stage) { @@ -398,6 +384,11 @@ Shader::TexturePixelFormat GraphicsEnvironment::ReadTexturePixelFormat(u32 handl      return result;  } +bool GraphicsEnvironment::IsTexturePixelFormatInteger(u32 handle) { +    return VideoCore::Surface::IsPixelFormatInteger( +        static_cast<VideoCore::Surface::PixelFormat>(ReadTexturePixelFormat(handle))); +} +  u32 GraphicsEnvironment::ReadViewportTransformState() {      const auto& regs{maxwell3d->regs};      viewport_transform_state = regs.viewport_scale_offset_enabled; @@ -448,6 +439,11 @@ Shader::TexturePixelFormat ComputeEnvironment::ReadTexturePixelFormat(u32 handle      return result;  } +bool ComputeEnvironment::IsTexturePixelFormatInteger(u32 handle) { +    return VideoCore::Surface::IsPixelFormatInteger( +        static_cast<VideoCore::Surface::PixelFormat>(ReadTexturePixelFormat(handle))); +} +  u32 ComputeEnvironment::ReadViewportTransformState() {      return viewport_transform_state;  } @@ -551,6 +547,11 @@ Shader::TexturePixelFormat FileEnvironment::ReadTexturePixelFormat(u32 handle) {      return it->second;  } +bool FileEnvironment::IsTexturePixelFormatInteger(u32 handle) { +    return VideoCore::Surface::IsPixelFormatInteger( +        static_cast<VideoCore::Surface::PixelFormat>(ReadTexturePixelFormat(handle))); +} +  u32 FileEnvironment::ReadViewportTransformState() {      return viewport_transform_state;  } diff --git a/src/video_core/shader_environment.h b/src/video_core/shader_environment.h index b90f3d44e..6b372e336 100644 --- a/src/video_core/shader_environment.h +++ b/src/video_core/shader_environment.h @@ -115,6 +115,8 @@ public:      Shader::TexturePixelFormat ReadTexturePixelFormat(u32 handle) override; +    bool IsTexturePixelFormatInteger(u32 handle) override; +      u32 ReadViewportTransformState() override;      std::optional<Shader::ReplaceConstant> GetReplaceConstBuffer(u32 bank, u32 offset) override; @@ -139,6 +141,8 @@ public:      Shader::TexturePixelFormat ReadTexturePixelFormat(u32 handle) override; +    bool IsTexturePixelFormatInteger(u32 handle) override; +      u32 ReadViewportTransformState() override;      std::optional<Shader::ReplaceConstant> GetReplaceConstBuffer( @@ -171,6 +175,8 @@ public:      [[nodiscard]] Shader::TexturePixelFormat ReadTexturePixelFormat(u32 handle) override; +    [[nodiscard]] bool IsTexturePixelFormatInteger(u32 handle) override; +      [[nodiscard]] u32 ReadViewportTransformState() override;      [[nodiscard]] u32 LocalMemorySize() const override; | 
