2016-08-13 12:57:50 +00:00
|
|
|
// Copyright 2016 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2+
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2016-10-01 03:07:50 +00:00
|
|
|
#include <array>
|
2016-12-09 12:23:07 +00:00
|
|
|
#include <map>
|
2016-08-13 12:57:50 +00:00
|
|
|
#include <memory>
|
2016-12-09 12:23:07 +00:00
|
|
|
#include <utility>
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2016-10-01 03:07:50 +00:00
|
|
|
#include "Common/CommonTypes.h"
|
2016-11-19 13:25:23 +00:00
|
|
|
#include "VideoBackends/Vulkan/StreamBuffer.h"
|
2016-11-19 14:08:24 +00:00
|
|
|
#include "VideoBackends/Vulkan/TextureCache.h"
|
2016-12-09 12:23:07 +00:00
|
|
|
#include "VideoCommon/TextureConversionShader.h"
|
2016-11-19 13:25:23 +00:00
|
|
|
#include "VideoCommon/TextureDecoder.h"
|
2016-08-13 12:57:50 +00:00
|
|
|
#include "VideoCommon/VideoCommon.h"
|
|
|
|
|
2017-10-30 12:58:43 +00:00
|
|
|
class AbstractTexture;
|
|
|
|
class AbstractStagingTexture;
|
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
namespace Vulkan
|
|
|
|
{
|
|
|
|
class StagingTexture2D;
|
|
|
|
class Texture2D;
|
2017-04-23 04:44:34 +00:00
|
|
|
class VKTexture;
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2016-11-19 13:25:23 +00:00
|
|
|
class TextureConverter
|
2016-08-13 12:57:50 +00:00
|
|
|
{
|
|
|
|
public:
|
2016-11-19 13:25:23 +00:00
|
|
|
TextureConverter();
|
|
|
|
~TextureConverter();
|
2016-08-13 12:57:50 +00:00
|
|
|
|
|
|
|
bool Initialize();
|
|
|
|
|
2016-11-19 14:08:24 +00:00
|
|
|
// Applies palette to dst_entry, using indices from src_entry.
|
2017-04-23 04:44:34 +00:00
|
|
|
void ConvertTexture(TextureCacheBase::TCacheEntry* dst_entry,
|
|
|
|
TextureCache::TCacheEntry* src_entry, VkRenderPass render_pass,
|
2017-07-30 19:45:55 +00:00
|
|
|
const void* palette, TLUTFormat palette_format);
|
2016-11-19 13:25:23 +00:00
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
// Uses an encoding shader to copy src_texture to dest_ptr.
|
2016-11-19 13:25:23 +00:00
|
|
|
// NOTE: Executes the current command buffer.
|
2017-07-30 19:45:55 +00:00
|
|
|
void EncodeTextureToMemory(VkImageView src_texture, u8* dest_ptr, const EFBCopyParams& params,
|
2017-04-04 13:55:36 +00:00
|
|
|
u32 native_width, u32 bytes_per_row, u32 num_blocks_y,
|
2017-07-30 19:45:55 +00:00
|
|
|
u32 memory_stride, const EFBRectangle& src_rect, bool scale_by_half);
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2016-11-19 14:43:50 +00:00
|
|
|
// Encodes texture to guest memory in XFB (YUYV) format.
|
|
|
|
void EncodeTextureToMemoryYUYV(void* dst_ptr, u32 dst_width, u32 dst_stride, u32 dst_height,
|
|
|
|
Texture2D* src_texture, const MathUtil::Rectangle<int>& src_rect);
|
|
|
|
|
|
|
|
// Decodes data from guest memory in XFB (YUYV) format to a RGBA format texture on the GPU.
|
2017-04-23 04:44:34 +00:00
|
|
|
void DecodeYUYVTextureFromMemory(VKTexture* dst_texture, const void* src_ptr, u32 src_width,
|
|
|
|
u32 src_stride, u32 src_height);
|
2016-11-19 14:43:50 +00:00
|
|
|
|
2017-07-30 19:45:55 +00:00
|
|
|
bool SupportsTextureDecoding(TextureFormat format, TLUTFormat palette_format);
|
2017-06-10 13:41:10 +00:00
|
|
|
void DecodeTexture(VkCommandBuffer command_buffer, TextureCache::TCacheEntry* entry,
|
|
|
|
u32 dst_level, const u8* data, size_t data_size, TextureFormat format,
|
|
|
|
u32 width, u32 height, u32 aligned_width, u32 aligned_height, u32 row_stride,
|
2017-07-30 19:45:55 +00:00
|
|
|
const u8* palette, TLUTFormat palette_format);
|
2016-12-09 12:23:07 +00:00
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
private:
|
|
|
|
static const u32 ENCODING_TEXTURE_WIDTH = EFB_WIDTH * 4;
|
|
|
|
static const u32 ENCODING_TEXTURE_HEIGHT = 1024;
|
2017-10-30 12:58:43 +00:00
|
|
|
static const AbstractTextureFormat ENCODING_TEXTURE_FORMAT = AbstractTextureFormat::BGRA8;
|
2016-11-19 13:25:23 +00:00
|
|
|
static const size_t NUM_PALETTE_CONVERSION_SHADERS = 3;
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2016-12-09 12:23:07 +00:00
|
|
|
// Maximum size of a texture based on BP registers.
|
|
|
|
static const u32 DECODING_TEXTURE_WIDTH = 1024;
|
|
|
|
static const u32 DECODING_TEXTURE_HEIGHT = 1024;
|
|
|
|
|
2016-11-19 14:08:24 +00:00
|
|
|
bool CreateTexelBuffer();
|
|
|
|
VkBufferView CreateTexelBufferView(VkFormat format) const;
|
|
|
|
|
2016-11-19 13:25:23 +00:00
|
|
|
bool CompilePaletteConversionShaders();
|
2016-11-19 14:08:24 +00:00
|
|
|
|
2017-07-30 19:45:55 +00:00
|
|
|
VkShaderModule CompileEncodingShader(const EFBCopyParams& params);
|
|
|
|
VkShaderModule GetEncodingShader(const EFBCopyParams& params);
|
2017-04-04 13:55:36 +00:00
|
|
|
|
2016-08-13 12:57:50 +00:00
|
|
|
bool CreateEncodingRenderPass();
|
|
|
|
bool CreateEncodingTexture();
|
2016-12-09 12:23:07 +00:00
|
|
|
bool CreateDecodingTexture();
|
|
|
|
|
2016-11-19 14:43:50 +00:00
|
|
|
bool CompileYUYVConversionShaders();
|
|
|
|
|
2016-11-19 15:16:27 +00:00
|
|
|
// Allocates storage in the texel command buffer of the specified size.
|
|
|
|
// If the buffer does not have enough space, executes the current command buffer and tries again.
|
|
|
|
// If this is done, g_command_buffer_mgr->GetCurrentCommandBuffer() will return a different value,
|
|
|
|
// so it always should be re-obtained after calling this method.
|
|
|
|
// Once the data copy is done, call m_texel_buffer->CommitMemory(size).
|
|
|
|
bool ReserveTexelBufferStorage(size_t size, size_t alignment);
|
|
|
|
|
|
|
|
// Returns the command buffer that the texture conversion should occur in for the given texture.
|
|
|
|
// This can be the initialization/copy command buffer, or the drawing command buffer.
|
|
|
|
VkCommandBuffer GetCommandBufferForTextureConversion(const TextureCache::TCacheEntry* src_entry);
|
|
|
|
|
2016-11-19 14:08:24 +00:00
|
|
|
// Shared between conversion types
|
|
|
|
std::unique_ptr<StreamBuffer> m_texel_buffer;
|
2016-12-09 12:23:07 +00:00
|
|
|
VkBufferView m_texel_buffer_view_r8_uint = VK_NULL_HANDLE;
|
2016-11-19 14:08:24 +00:00
|
|
|
VkBufferView m_texel_buffer_view_r16_uint = VK_NULL_HANDLE;
|
2016-12-09 12:23:07 +00:00
|
|
|
VkBufferView m_texel_buffer_view_r32g32_uint = VK_NULL_HANDLE;
|
2017-08-08 05:09:25 +00:00
|
|
|
VkBufferView m_texel_buffer_view_rgba8_uint = VK_NULL_HANDLE;
|
2016-11-19 15:16:27 +00:00
|
|
|
VkBufferView m_texel_buffer_view_rgba8_unorm = VK_NULL_HANDLE;
|
2016-11-19 14:08:24 +00:00
|
|
|
size_t m_texel_buffer_size = 0;
|
2016-11-19 13:25:23 +00:00
|
|
|
|
2016-11-19 14:08:24 +00:00
|
|
|
// Palette conversion - taking an indexed texture and applying palette
|
|
|
|
std::array<VkShaderModule, NUM_PALETTE_CONVERSION_SHADERS> m_palette_conversion_shaders = {};
|
2016-08-13 12:57:50 +00:00
|
|
|
|
2016-11-19 14:08:24 +00:00
|
|
|
// Texture encoding - RGBA8->GX format in memory
|
2017-07-30 19:45:55 +00:00
|
|
|
std::map<EFBCopyParams, VkShaderModule> m_encoding_shaders;
|
2017-10-30 12:58:43 +00:00
|
|
|
std::unique_ptr<AbstractTexture> m_encoding_render_texture;
|
|
|
|
std::unique_ptr<AbstractStagingTexture> m_encoding_readback_texture;
|
2016-08-13 12:57:50 +00:00
|
|
|
VkRenderPass m_encoding_render_pass = VK_NULL_HANDLE;
|
2016-11-19 14:43:50 +00:00
|
|
|
|
2016-12-09 12:23:07 +00:00
|
|
|
// Texture decoding - GX format in memory->RGBA8
|
|
|
|
struct TextureDecodingPipeline
|
|
|
|
{
|
|
|
|
const TextureConversionShader::DecodingShaderInfo* base_info;
|
|
|
|
VkShaderModule compute_shader;
|
|
|
|
bool valid;
|
|
|
|
};
|
2017-07-30 19:45:55 +00:00
|
|
|
std::map<std::pair<TextureFormat, TLUTFormat>, TextureDecodingPipeline> m_decoding_pipelines;
|
2016-12-09 12:23:07 +00:00
|
|
|
std::unique_ptr<Texture2D> m_decoding_texture;
|
|
|
|
|
2016-11-19 14:43:50 +00:00
|
|
|
// XFB encoding/decoding shaders
|
|
|
|
VkShaderModule m_rgb_to_yuyv_shader = VK_NULL_HANDLE;
|
|
|
|
VkShaderModule m_yuyv_to_rgb_shader = VK_NULL_HANDLE;
|
2016-08-13 12:57:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Vulkan
|