2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2008 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2009-03-07 08:35:01 +00:00
|
|
|
|
2019-06-01 07:55:09 -04:00
|
|
|
#include "VideoCommon/VertexLoaderManager.h"
|
|
|
|
|
2010-03-06 02:07:48 +00:00
|
|
|
#include <algorithm>
|
2019-06-01 07:55:09 -04:00
|
|
|
#include <iterator>
|
2014-06-05 17:55:21 +02:00
|
|
|
#include <memory>
|
2014-08-24 23:53:28 -04:00
|
|
|
#include <mutex>
|
2016-01-17 16:54:31 -05:00
|
|
|
#include <string>
|
2023-03-23 12:49:09 -05:00
|
|
|
#include <type_traits>
|
2012-12-10 00:40:28 -06:00
|
|
|
#include <unordered_map>
|
2014-08-15 16:17:06 +02:00
|
|
|
#include <utility>
|
2009-03-07 08:35:01 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2021-06-20 13:47:57 -07:00
|
|
|
#include "Common/EnumMap.h"
|
2021-02-28 13:53:32 -08:00
|
|
|
#include "Common/Logging/Log.h"
|
2021-03-07 15:42:10 -08:00
|
|
|
|
2022-05-17 13:42:31 -07:00
|
|
|
#include "Core/DolphinAnalytics.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Core/HW/Memmap.h"
|
2022-12-02 20:07:30 +01:00
|
|
|
#include "Core/System.h"
|
2009-03-07 08:35:01 +00:00
|
|
|
|
2023-01-27 11:34:59 +13:00
|
|
|
#include "VideoCommon/AbstractGfx.h"
|
2014-08-04 19:51:42 -07:00
|
|
|
#include "VideoCommon/BPMemory.h"
|
2021-02-28 13:53:32 -08:00
|
|
|
#include "VideoCommon/CPMemory.h"
|
2016-01-31 14:51:55 -05:00
|
|
|
#include "VideoCommon/DataReader.h"
|
2014-08-01 23:42:15 -07:00
|
|
|
#include "VideoCommon/IndexGenerator.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "VideoCommon/NativeVertexFormat.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/Statistics.h"
|
2014-12-13 01:51:14 +01:00
|
|
|
#include "VideoCommon/VertexLoaderBase.h"
|
2014-07-26 01:10:44 +02:00
|
|
|
#include "VideoCommon/VertexManagerBase.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "VideoCommon/VertexShaderManager.h"
|
2022-07-26 03:57:30 -05:00
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2022-05-17 13:42:31 -07:00
|
|
|
#include "VideoCommon/XFMemory.h"
|
2010-03-06 02:07:48 +00:00
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
namespace VertexLoaderManager
|
|
|
|
{
|
2022-04-13 16:12:53 -07:00
|
|
|
// Used by zfreeze
|
2022-04-14 12:01:57 -07:00
|
|
|
std::array<u32, 3> position_matrix_index_cache;
|
|
|
|
// 3 vertices, 4 floats each to allow SIMD overwrite
|
|
|
|
alignas(sizeof(std::array<float, 4>)) std::array<std::array<float, 4>, 3> position_cache;
|
Cache normals in addition to binormals and tangents
Fixes LIT (https://bugs.dolphin-emu.org/issues/13635). The text does not include normals, but has lighting enabled. With the previous default of (0, 0, 0), lighting was always black (as dot(X, (0, 0, 0)) is always 0). It seems like the normal from the map in the background (0, 0, 1) is re-used.
LIT also has the vertex color enabled while vertex color is not specified, the same as SMS's debug cubes; the default MissingColorValue GameINI value of solid white seems to work correctly in this case.
2024-09-24 23:46:45 -07:00
|
|
|
alignas(sizeof(std::array<float, 4>)) std::array<float, 4> normal_cache;
|
2022-04-13 22:03:34 -07:00
|
|
|
alignas(sizeof(std::array<float, 4>)) std::array<float, 4> tangent_cache;
|
|
|
|
alignas(sizeof(std::array<float, 4>)) std::array<float, 4> binormal_cache;
|
2015-06-01 19:58:27 +02:00
|
|
|
|
2014-12-12 08:53:48 +01:00
|
|
|
static NativeVertexFormatMap s_native_vertex_map;
|
|
|
|
static NativeVertexFormat* s_current_vtx_fmt;
|
2015-11-01 22:39:31 +01:00
|
|
|
u32 g_current_components;
|
2014-12-12 08:53:48 +01:00
|
|
|
|
2014-12-13 01:51:14 +01:00
|
|
|
typedef std::unordered_map<VertexLoaderUID, std::unique_ptr<VertexLoaderBase>> VertexLoaderMap;
|
2014-08-24 23:53:28 -04:00
|
|
|
static std::mutex s_vertex_loader_map_lock;
|
|
|
|
static VertexLoaderMap s_vertex_loader_map;
|
2009-03-07 08:35:01 +00:00
|
|
|
// TODO - change into array of pointers. Keep a map of all seen so far.
|
|
|
|
|
2021-06-20 13:47:57 -07:00
|
|
|
Common::EnumMap<u8*, CPArray::TexCoord7> cached_arraybases;
|
|
|
|
|
|
|
|
BitSet8 g_main_vat_dirty;
|
|
|
|
BitSet8 g_preprocess_vat_dirty;
|
|
|
|
bool g_bases_dirty; // Main only
|
|
|
|
std::array<VertexLoaderBase*, CP_NUM_VAT_REG> g_main_vertex_loaders;
|
|
|
|
std::array<VertexLoaderBase*, CP_NUM_VAT_REG> g_preprocess_vertex_loaders;
|
2022-11-16 14:39:29 +01:00
|
|
|
bool g_needs_cp_xf_consistency_check;
|
2015-05-30 00:42:45 +12:00
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
MarkAllDirty();
|
2024-01-31 19:46:24 -05:00
|
|
|
g_main_vertex_loaders.fill(nullptr);
|
|
|
|
g_preprocess_vertex_loaders.fill(nullptr);
|
2019-07-10 23:34:50 -04:00
|
|
|
SETSTAT(g_stats.num_vertex_loaders, 0);
|
2009-03-07 08:35:01 +00:00
|
|
|
}
|
|
|
|
|
2016-01-13 21:14:20 +01:00
|
|
|
void Clear()
|
2009-03-07 08:35:01 +00:00
|
|
|
{
|
2014-08-24 23:53:28 -04:00
|
|
|
std::lock_guard<std::mutex> lk(s_vertex_loader_map_lock);
|
|
|
|
s_vertex_loader_map.clear();
|
2014-12-12 08:53:48 +01:00
|
|
|
s_native_vertex_map.clear();
|
2009-03-07 08:35:01 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 00:42:45 +12:00
|
|
|
void UpdateVertexArrayPointers()
|
|
|
|
{
|
2015-05-30 03:58:27 +12:00
|
|
|
// Anything to update?
|
2022-09-15 23:38:57 +02:00
|
|
|
if (!g_bases_dirty) [[likely]]
|
2015-05-30 03:58:27 +12:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2022-12-02 20:07:30 +01:00
|
|
|
auto& system = Core::System::GetInstance();
|
|
|
|
auto& memory = system.GetMemory();
|
|
|
|
|
2015-05-30 00:42:45 +12:00
|
|
|
// Some games such as Burnout 2 can put invalid addresses into
|
|
|
|
// the array base registers. (see issue 8591)
|
|
|
|
// But the vertex arrays with invalid addresses aren't actually enabled.
|
|
|
|
// Note: Only array bases 0 through 11 are used by the Vertex loaders.
|
|
|
|
// 12 through 15 are used for loading data into xfmem.
|
2021-02-08 15:22:10 -08:00
|
|
|
// We also only update the array base if the vertex description states we are going to use it.
|
2024-04-13 12:08:43 +02:00
|
|
|
// TODO: For memory safety, we need to check the sizes returned by GetSpanForAddress
|
2021-02-08 15:22:10 -08:00
|
|
|
if (IsIndexed(g_main_cp_state.vtx_desc.low.Position))
|
2024-04-13 12:08:43 +02:00
|
|
|
{
|
2021-06-20 13:47:57 -07:00
|
|
|
cached_arraybases[CPArray::Position] =
|
2024-04-13 12:08:43 +02:00
|
|
|
memory.GetSpanForAddress(g_main_cp_state.array_bases[CPArray::Position]).data();
|
|
|
|
}
|
2021-02-08 15:22:10 -08:00
|
|
|
|
|
|
|
if (IsIndexed(g_main_cp_state.vtx_desc.low.Normal))
|
2024-04-13 12:08:43 +02:00
|
|
|
{
|
2021-06-20 13:47:57 -07:00
|
|
|
cached_arraybases[CPArray::Normal] =
|
2024-04-13 12:08:43 +02:00
|
|
|
memory.GetSpanForAddress(g_main_cp_state.array_bases[CPArray::Normal]).data();
|
|
|
|
}
|
2021-02-08 15:22:10 -08:00
|
|
|
|
2021-06-20 13:47:57 -07:00
|
|
|
for (u8 i = 0; i < g_main_cp_state.vtx_desc.low.Color.Size(); i++)
|
2021-02-08 15:22:10 -08:00
|
|
|
{
|
|
|
|
if (IsIndexed(g_main_cp_state.vtx_desc.low.Color[i]))
|
2024-04-13 12:08:43 +02:00
|
|
|
{
|
2021-06-20 13:47:57 -07:00
|
|
|
cached_arraybases[CPArray::Color0 + i] =
|
2024-04-13 12:08:43 +02:00
|
|
|
memory.GetSpanForAddress(g_main_cp_state.array_bases[CPArray::Color0 + i]).data();
|
|
|
|
}
|
2021-02-08 15:22:10 -08:00
|
|
|
}
|
|
|
|
|
2021-06-20 13:47:57 -07:00
|
|
|
for (u8 i = 0; i < g_main_cp_state.vtx_desc.high.TexCoord.Size(); i++)
|
2015-05-30 00:42:45 +12:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
if (IsIndexed(g_main_cp_state.vtx_desc.high.TexCoord[i]))
|
2024-04-13 12:08:43 +02:00
|
|
|
{
|
2021-06-20 13:47:57 -07:00
|
|
|
cached_arraybases[CPArray::TexCoord0 + i] =
|
2024-04-13 12:08:43 +02:00
|
|
|
memory.GetSpanForAddress(g_main_cp_state.array_bases[CPArray::TexCoord0 + i]).data();
|
|
|
|
}
|
2015-05-30 00:42:45 +12:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-05-13 16:05:31 -07:00
|
|
|
g_bases_dirty = false;
|
2015-05-30 00:42:45 +12:00
|
|
|
}
|
|
|
|
|
2009-03-07 08:35:01 +00:00
|
|
|
void MarkAllDirty()
|
|
|
|
{
|
2022-10-07 12:25:04 -07:00
|
|
|
g_bases_dirty = true;
|
2021-05-13 16:05:31 -07:00
|
|
|
g_main_vat_dirty = BitSet8::AllTrue(8);
|
|
|
|
g_preprocess_vat_dirty = BitSet8::AllTrue(8);
|
2022-11-16 14:39:29 +01:00
|
|
|
g_needs_cp_xf_consistency_check = true;
|
2009-03-07 08:35:01 +00:00
|
|
|
}
|
|
|
|
|
2017-07-03 19:43:47 +10:00
|
|
|
NativeVertexFormat* GetOrCreateMatchingFormat(const PortableVertexDeclaration& decl)
|
|
|
|
{
|
|
|
|
auto iter = s_native_vertex_map.find(decl);
|
|
|
|
if (iter == s_native_vertex_map.end())
|
|
|
|
{
|
2023-01-27 11:34:59 +13:00
|
|
|
std::unique_ptr<NativeVertexFormat> fmt = g_gfx->CreateNativeVertexFormat(decl);
|
2017-07-03 19:43:47 +10:00
|
|
|
auto ipair = s_native_vertex_map.emplace(decl, std::move(fmt));
|
|
|
|
iter = ipair.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
return iter->second.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
NativeVertexFormat* GetUberVertexFormat(const PortableVertexDeclaration& decl)
|
|
|
|
{
|
|
|
|
// The padding in the structs can cause the memcmp() in the map to create duplicates.
|
|
|
|
// Avoid this by initializing the padding to zero.
|
|
|
|
PortableVertexDeclaration new_decl;
|
2023-03-23 12:49:09 -05:00
|
|
|
static_assert(std::is_trivially_copyable_v<PortableVertexDeclaration>);
|
|
|
|
std::memset(static_cast<void*>(&new_decl), 0, sizeof(new_decl));
|
2017-07-03 19:43:47 +10:00
|
|
|
new_decl.stride = decl.stride;
|
|
|
|
|
2021-06-26 12:48:28 -07:00
|
|
|
auto MakeDummyAttribute = [](AttributeFormat& attr, ComponentFormat type, int components,
|
|
|
|
bool integer) {
|
2017-07-03 19:43:47 +10:00
|
|
|
attr.type = type;
|
|
|
|
attr.components = components;
|
|
|
|
attr.offset = 0;
|
|
|
|
attr.enable = true;
|
|
|
|
attr.integer = integer;
|
|
|
|
};
|
|
|
|
auto CopyAttribute = [](AttributeFormat& attr, const AttributeFormat& src) {
|
|
|
|
attr.type = src.type;
|
|
|
|
attr.components = src.components;
|
|
|
|
attr.offset = src.offset;
|
|
|
|
attr.enable = src.enable;
|
|
|
|
attr.integer = src.integer;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (decl.position.enable)
|
|
|
|
CopyAttribute(new_decl.position, decl.position);
|
|
|
|
else
|
2021-06-26 12:48:28 -07:00
|
|
|
MakeDummyAttribute(new_decl.position, ComponentFormat::Float, 1, false);
|
2019-06-01 07:55:09 -04:00
|
|
|
for (size_t i = 0; i < std::size(new_decl.normals); i++)
|
2017-07-03 19:43:47 +10:00
|
|
|
{
|
|
|
|
if (decl.normals[i].enable)
|
|
|
|
CopyAttribute(new_decl.normals[i], decl.normals[i]);
|
|
|
|
else
|
2021-06-26 12:48:28 -07:00
|
|
|
MakeDummyAttribute(new_decl.normals[i], ComponentFormat::Float, 1, false);
|
2017-07-03 19:43:47 +10:00
|
|
|
}
|
2019-06-01 07:55:09 -04:00
|
|
|
for (size_t i = 0; i < std::size(new_decl.colors); i++)
|
2017-07-03 19:43:47 +10:00
|
|
|
{
|
|
|
|
if (decl.colors[i].enable)
|
|
|
|
CopyAttribute(new_decl.colors[i], decl.colors[i]);
|
|
|
|
else
|
2021-06-26 12:48:28 -07:00
|
|
|
MakeDummyAttribute(new_decl.colors[i], ComponentFormat::UByte, 4, false);
|
2017-07-03 19:43:47 +10:00
|
|
|
}
|
2019-06-01 07:55:09 -04:00
|
|
|
for (size_t i = 0; i < std::size(new_decl.texcoords); i++)
|
2017-07-03 19:43:47 +10:00
|
|
|
{
|
|
|
|
if (decl.texcoords[i].enable)
|
|
|
|
CopyAttribute(new_decl.texcoords[i], decl.texcoords[i]);
|
|
|
|
else
|
2021-06-26 12:48:28 -07:00
|
|
|
MakeDummyAttribute(new_decl.texcoords[i], ComponentFormat::Float, 1, false);
|
2017-07-03 19:43:47 +10:00
|
|
|
}
|
|
|
|
if (decl.posmtx.enable)
|
|
|
|
CopyAttribute(new_decl.posmtx, decl.posmtx);
|
|
|
|
else
|
2021-06-26 12:48:28 -07:00
|
|
|
MakeDummyAttribute(new_decl.posmtx, ComponentFormat::UByte, 1, true);
|
2017-07-03 19:43:47 +10:00
|
|
|
|
|
|
|
return GetOrCreateMatchingFormat(new_decl);
|
|
|
|
}
|
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
namespace detail
|
2014-08-15 16:17:06 +02:00
|
|
|
{
|
2022-09-15 23:38:57 +02:00
|
|
|
template <bool IsPreprocess>
|
|
|
|
VertexLoaderBase* GetOrCreateLoader(int vtx_attr_group)
|
|
|
|
{
|
|
|
|
constexpr CPState* state = IsPreprocess ? &g_preprocess_cp_state : &g_main_cp_state;
|
|
|
|
constexpr BitSet8& attr_dirty = IsPreprocess ? g_preprocess_vat_dirty : g_main_vat_dirty;
|
|
|
|
constexpr auto& vertex_loaders =
|
|
|
|
IsPreprocess ? g_preprocess_vertex_loaders : g_main_vertex_loaders;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-12-13 01:51:14 +01:00
|
|
|
VertexLoaderBase* loader;
|
2022-09-15 23:38:57 +02:00
|
|
|
|
|
|
|
// We are not allowed to create a native vertex format on preprocessing as this is on the wrong
|
|
|
|
// thread
|
|
|
|
bool check_for_native_format = !IsPreprocess;
|
|
|
|
|
|
|
|
VertexLoaderUID uid(state->vtx_desc, state->vtx_attr[vtx_attr_group]);
|
|
|
|
std::lock_guard<std::mutex> lk(s_vertex_loader_map_lock);
|
|
|
|
VertexLoaderMap::iterator iter = s_vertex_loader_map.find(uid);
|
|
|
|
if (iter != s_vertex_loader_map.end())
|
2014-08-15 16:17:06 +02:00
|
|
|
{
|
2022-09-15 23:38:57 +02:00
|
|
|
loader = iter->second.get();
|
|
|
|
check_for_native_format &= !loader->m_native_vertex_format;
|
2014-08-24 23:53:28 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-09-15 23:38:57 +02:00
|
|
|
auto [it, added] = s_vertex_loader_map.try_emplace(
|
|
|
|
uid,
|
|
|
|
VertexLoaderBase::CreateVertexLoader(state->vtx_desc, state->vtx_attr[vtx_attr_group]));
|
|
|
|
loader = it->second.get();
|
|
|
|
INCSTAT(g_stats.num_vertex_loaders);
|
2014-08-15 16:17:06 +02:00
|
|
|
}
|
2022-09-15 23:38:57 +02:00
|
|
|
if (check_for_native_format)
|
|
|
|
{
|
|
|
|
// search for a cached native vertex format
|
|
|
|
loader->m_native_vertex_format = GetOrCreateMatchingFormat(loader->m_native_vtx_decl);
|
|
|
|
}
|
|
|
|
vertex_loaders[vtx_attr_group] = loader;
|
|
|
|
attr_dirty[vtx_attr_group] = false;
|
2014-08-24 23:53:28 -04:00
|
|
|
return loader;
|
2014-08-15 16:17:06 +02:00
|
|
|
}
|
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
} // namespace detail
|
|
|
|
|
2022-05-17 13:42:31 -07:00
|
|
|
static void CheckCPConfiguration(int vtx_attr_group)
|
|
|
|
{
|
|
|
|
// Validate that the XF input configuration matches the CP configuration
|
|
|
|
u32 num_cp_colors = std::count_if(
|
|
|
|
g_main_cp_state.vtx_desc.low.Color.begin(), g_main_cp_state.vtx_desc.low.Color.end(),
|
|
|
|
[](auto format) { return format != VertexComponentFormat::NotPresent; });
|
|
|
|
u32 num_cp_tex_coords = std::count_if(
|
|
|
|
g_main_cp_state.vtx_desc.high.TexCoord.begin(), g_main_cp_state.vtx_desc.high.TexCoord.end(),
|
|
|
|
[](auto format) { return format != VertexComponentFormat::NotPresent; });
|
|
|
|
|
|
|
|
u32 num_cp_normals;
|
|
|
|
if (g_main_cp_state.vtx_desc.low.Normal == VertexComponentFormat::NotPresent)
|
|
|
|
num_cp_normals = 0;
|
|
|
|
else if (g_main_cp_state.vtx_attr[vtx_attr_group].g0.NormalElements == NormalComponentCount::NTB)
|
|
|
|
num_cp_normals = 3;
|
|
|
|
else
|
|
|
|
num_cp_normals = 1;
|
|
|
|
|
|
|
|
std::optional<u32> num_xf_normals;
|
|
|
|
switch (xfmem.invtxspec.numnormals)
|
|
|
|
{
|
|
|
|
case NormalCount::None:
|
|
|
|
num_xf_normals = 0;
|
|
|
|
break;
|
|
|
|
case NormalCount::Normal:
|
|
|
|
num_xf_normals = 1;
|
|
|
|
break;
|
|
|
|
case NormalCount::NormalTangentBinormal:
|
2022-10-24 22:36:43 -07:00
|
|
|
case NormalCount::Invalid: // see https://bugs.dolphin-emu.org/issues/13070
|
2022-05-17 13:42:31 -07:00
|
|
|
num_xf_normals = 3;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num_cp_colors != xfmem.invtxspec.numcolors || num_cp_normals != num_xf_normals ||
|
2022-11-16 14:39:29 +01:00
|
|
|
num_cp_tex_coords != xfmem.invtxspec.numtextures) [[unlikely]]
|
2022-05-17 13:42:31 -07:00
|
|
|
{
|
|
|
|
PanicAlertFmt("Mismatched configuration between CP and XF stages - {}/{} colors, {}/{} "
|
|
|
|
"normals, {}/{} texture coordinates. Please report on the issue tracker.\n\n"
|
|
|
|
"VCD: {:08x} {:08x}\nVAT {}: {:08x} {:08x} {:08x}\nXF vertex spec: {:08x}",
|
|
|
|
num_cp_colors, xfmem.invtxspec.numcolors, num_cp_normals,
|
|
|
|
num_xf_normals.has_value() ? fmt::to_string(num_xf_normals.value()) : "invalid",
|
|
|
|
num_cp_tex_coords, xfmem.invtxspec.numtextures, g_main_cp_state.vtx_desc.low.Hex,
|
|
|
|
g_main_cp_state.vtx_desc.high.Hex, vtx_attr_group,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g1.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g2.Hex, xfmem.invtxspec.hex);
|
|
|
|
|
|
|
|
// Analytics reporting so we can discover which games have this problem, that way when we
|
|
|
|
// eventually simulate the behavior we have test cases for it.
|
2022-11-16 14:39:29 +01:00
|
|
|
if (num_cp_colors != xfmem.invtxspec.numcolors) [[unlikely]]
|
2022-05-17 13:42:31 -07:00
|
|
|
{
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(
|
|
|
|
GameQuirk::MISMATCHED_GPU_COLORS_BETWEEN_CP_AND_XF);
|
|
|
|
}
|
2022-11-16 14:39:29 +01:00
|
|
|
if (num_cp_normals != num_xf_normals) [[unlikely]]
|
2022-05-17 13:42:31 -07:00
|
|
|
{
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(
|
|
|
|
GameQuirk::MISMATCHED_GPU_NORMALS_BETWEEN_CP_AND_XF);
|
|
|
|
}
|
2022-11-16 14:39:29 +01:00
|
|
|
if (num_cp_tex_coords != xfmem.invtxspec.numtextures) [[unlikely]]
|
2022-05-17 13:42:31 -07:00
|
|
|
{
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(
|
|
|
|
GameQuirk::MISMATCHED_GPU_TEX_COORDS_BETWEEN_CP_AND_XF);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't bail out, though; we can still render something successfully
|
|
|
|
// (real hardware seems to hang in this case, though)
|
|
|
|
}
|
2022-05-17 13:58:54 -07:00
|
|
|
|
|
|
|
if (g_main_cp_state.matrix_index_a.Hex != xfmem.MatrixIndexA.Hex ||
|
2022-11-16 14:39:29 +01:00
|
|
|
g_main_cp_state.matrix_index_b.Hex != xfmem.MatrixIndexB.Hex) [[unlikely]]
|
2022-05-17 13:58:54 -07:00
|
|
|
{
|
2022-07-17 18:12:08 -07:00
|
|
|
WARN_LOG_FMT(VIDEO,
|
|
|
|
"Mismatched matrix index configuration between CP and XF stages - "
|
|
|
|
"index A: {:08x}/{:08x}, index B {:08x}/{:08x}.",
|
|
|
|
g_main_cp_state.matrix_index_a.Hex, xfmem.MatrixIndexA.Hex,
|
|
|
|
g_main_cp_state.matrix_index_b.Hex, xfmem.MatrixIndexB.Hex);
|
2022-05-17 13:58:54 -07:00
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(
|
|
|
|
GameQuirk::MISMATCHED_GPU_MATRIX_INDICES_BETWEEN_CP_AND_XF);
|
|
|
|
}
|
2024-04-01 21:37:11 -07:00
|
|
|
|
|
|
|
if (g_main_cp_state.vtx_attr[vtx_attr_group].g0.PosFormat >= ComponentFormat::InvalidFloat5)
|
|
|
|
{
|
|
|
|
WARN_LOG_FMT(VIDEO, "Invalid position format {} for VAT {} - {:08x} {:08x} {:08x}",
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.PosFormat, vtx_attr_group,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g1.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g2.Hex);
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(GameQuirk::INVALID_POSITION_COMPONENT_FORMAT);
|
|
|
|
}
|
|
|
|
if (g_main_cp_state.vtx_attr[vtx_attr_group].g0.NormalFormat >= ComponentFormat::InvalidFloat5)
|
|
|
|
{
|
|
|
|
WARN_LOG_FMT(VIDEO, "Invalid normal format {} for VAT {} - {:08x} {:08x} {:08x}",
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.NormalFormat, vtx_attr_group,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g1.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g2.Hex);
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(GameQuirk::INVALID_NORMAL_COMPONENT_FORMAT);
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < 8; i++)
|
|
|
|
{
|
|
|
|
if (g_main_cp_state.vtx_attr[vtx_attr_group].GetTexFormat(i) >= ComponentFormat::InvalidFloat5)
|
|
|
|
{
|
|
|
|
WARN_LOG_FMT(VIDEO,
|
|
|
|
"Invalid texture coordinate {} format {} for VAT {} - {:08x} {:08x} {:08x}", i,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].GetTexFormat(i), vtx_attr_group,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g1.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g2.Hex);
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(
|
|
|
|
GameQuirk::INVALID_TEXTURE_COORDINATE_COMPONENT_FORMAT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < 2; i++)
|
|
|
|
{
|
|
|
|
if (g_main_cp_state.vtx_attr[vtx_attr_group].GetColorFormat(i) > ColorFormat::RGBA8888)
|
|
|
|
{
|
|
|
|
WARN_LOG_FMT(VIDEO, "Invalid color {} format {} for VAT {} - {:08x} {:08x} {:08x}", i,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].GetColorFormat(i), vtx_attr_group,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g0.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g1.Hex,
|
|
|
|
g_main_cp_state.vtx_attr[vtx_attr_group].g2.Hex);
|
|
|
|
DolphinAnalytics::Instance().ReportGameQuirk(GameQuirk::INVALID_COLOR_COMPONENT_FORMAT);
|
|
|
|
}
|
|
|
|
}
|
2022-05-17 13:42:31 -07:00
|
|
|
}
|
|
|
|
|
2024-10-20 15:16:59 -05:00
|
|
|
static bool CanSplit(OpcodeDecoder::Primitive primitive)
|
|
|
|
{
|
|
|
|
// Splitting is currently only implemented for the easy cases (individual lines/points/triangles)
|
|
|
|
switch (primitive)
|
|
|
|
{
|
|
|
|
case OpcodeDecoder::Primitive::GX_DRAW_QUADS:
|
|
|
|
case OpcodeDecoder::Primitive::GX_DRAW_QUADS_2:
|
|
|
|
case OpcodeDecoder::Primitive::GX_DRAW_TRIANGLES:
|
|
|
|
case OpcodeDecoder::Primitive::GX_DRAW_LINES:
|
|
|
|
case OpcodeDecoder::Primitive::GX_DRAW_POINTS:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
template <bool IsPreprocess>
|
2022-11-22 16:54:05 -08:00
|
|
|
int RunVertices(int vtx_attr_group, OpcodeDecoder::Primitive primitive, int count, const u8* src)
|
2009-03-07 08:35:01 +00:00
|
|
|
{
|
2022-11-16 14:39:29 +01:00
|
|
|
if (count == 0) [[unlikely]]
|
2014-12-09 08:35:04 +01:00
|
|
|
return 0;
|
2022-04-14 12:01:57 -07:00
|
|
|
ASSERT(count > 0);
|
2014-08-27 13:38:00 -04:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
VertexLoaderBase* loader = RefreshLoader<IsPreprocess>(vtx_attr_group);
|
2014-07-26 01:10:44 +02:00
|
|
|
|
2021-03-11 12:55:25 -08:00
|
|
|
int size = count * loader->m_vertex_size;
|
Refactor opcode decoding a bit to kill FifoCommandRunnable.
Separated out from my gpu-determinism branch by request. It's not a big
commit; I just like to write long commit messages.
The main reason to kill it is hopefully a slight performance improvement
from avoiding the double switch (especially in single core mode);
however, this also improves cycle calculation, as described below.
- FifoCommandRunnable is removed; in its stead, Decode returns the
number of cycles (which only matters for "sync" GPU mode), or 0 if there
was not enough data, and is also responsible for unknown opcode alerts.
Decode and DecodeSemiNop are almost identical, so the latter is replaced
with a skipped_frame parameter to Decode. Doesn't mean we can't improve
skipped_frame mode to do less work; if, at such a point, branching on it
has too much overhead (it certainly won't now), it can always be changed
to a template parameter.
- FifoCommandRunnable used a fixed, large cycle count for display lists,
regardless of the contents. Presumably the actual hardware's processing
time is mostly the processing time of whatever commands are in the list,
and with this change InterpretDisplayList can just return the list's
cycle count to be added to the total. (Since the calculation for this
is part of Decode, it didn't seem easy to split this change up.)
To facilitate this, Decode also gains an explicit 'end' parameter in
lieu of FifoCommandRunnable's call to GetVideoBufferEndPtr, which can
point to there or to the end of a display list (or elsewhere in
gpu-determinism, but that's another story). Also, as a small
optimization, InterpretDisplayList now calls OpcodeDecoder_Run rather
than having its own Decode loop, to allow Decode to be inlined (haven't
checked whether this actually happens though).
skipped_frame mode still does not traverse display lists and uses the
old fake value of 45 cycles. degasus has suggested that this hack is
not essential for performance and can be removed, but I want to separate
any potential performance impact of that from this commit.
2014-09-01 01:11:32 -04:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
if constexpr (!IsPreprocess)
|
|
|
|
{
|
|
|
|
// Doing early return for the opposite case would be cleaner
|
|
|
|
// but triggers a false unreachable code warning in MSVC debug builds.
|
2014-08-04 19:51:42 -07:00
|
|
|
|
2023-03-19 23:25:00 +01:00
|
|
|
if (g_needs_cp_xf_consistency_check) [[unlikely]]
|
|
|
|
{
|
|
|
|
CheckCPConfiguration(vtx_attr_group);
|
|
|
|
g_needs_cp_xf_consistency_check = false;
|
|
|
|
}
|
2022-05-17 13:42:31 -07:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
// If the native vertex format changed, force a flush.
|
|
|
|
if (loader->m_native_vertex_format != s_current_vtx_fmt ||
|
2022-11-16 14:39:29 +01:00
|
|
|
loader->m_native_components != g_current_components) [[unlikely]]
|
2022-09-15 23:38:57 +02:00
|
|
|
{
|
|
|
|
g_vertex_manager->Flush();
|
2023-03-19 23:25:00 +01:00
|
|
|
|
|
|
|
s_current_vtx_fmt = loader->m_native_vertex_format;
|
|
|
|
g_current_components = loader->m_native_components;
|
|
|
|
auto& system = Core::System::GetInstance();
|
|
|
|
auto& vertex_shader_manager = system.GetVertexShaderManager();
|
|
|
|
vertex_shader_manager.SetVertexFormat(loader->m_native_components,
|
|
|
|
loader->m_native_vertex_format->GetVertexDeclaration());
|
2022-09-15 23:38:57 +02:00
|
|
|
}
|
2014-07-26 01:10:44 +02:00
|
|
|
|
2022-07-26 03:57:30 -05:00
|
|
|
// CPUCull's performance increase comes from encoding fewer GPU commands, not sending less data
|
|
|
|
// Therefore it's only useful to check if culling could remove a flush
|
2024-10-20 15:16:59 -05:00
|
|
|
bool can_cpu_cull = g_ActiveConfig.bCPUCull &&
|
|
|
|
primitive < OpcodeDecoder::Primitive::GX_DRAW_LINES &&
|
|
|
|
!g_vertex_manager->HasSendableVertices();
|
2022-07-26 03:57:30 -05:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
// if cull mode is CULL_ALL, tell VertexManager to skip triangles and quads.
|
2022-07-26 03:57:30 -05:00
|
|
|
// They still need to go through vertex loading, because we need to calculate a zfreeze
|
|
|
|
// reference slope.
|
|
|
|
const bool cullall = (bpmem.genMode.cullmode == CullMode::All &&
|
|
|
|
primitive < OpcodeDecoder::Primitive::GX_DRAW_LINES);
|
2015-01-24 14:37:20 +13:00
|
|
|
|
2022-07-26 03:57:30 -05:00
|
|
|
const int stride = loader->m_native_vtx_decl.stride;
|
2024-10-20 15:16:59 -05:00
|
|
|
do
|
|
|
|
{
|
|
|
|
const int max_vertices = 16380; // Max is 16383, but 16380 is divisible by both 4 and 3
|
|
|
|
const int run = CanSplit(primitive) && count > max_vertices ? max_vertices : count;
|
|
|
|
count -= run;
|
|
|
|
DataReader dst = g_vertex_manager->PrepareForAdditionalData(primitive, run, stride,
|
|
|
|
cullall || can_cpu_cull);
|
2014-08-01 23:42:15 -07:00
|
|
|
|
2024-10-20 15:16:59 -05:00
|
|
|
const int num_loaded = loader->RunVertices(src, dst.GetPointer(), run);
|
|
|
|
src += loader->m_vertex_size * max_vertices;
|
2014-08-01 23:42:15 -07:00
|
|
|
|
2024-10-20 15:16:59 -05:00
|
|
|
if (can_cpu_cull && !cullall)
|
2022-07-26 03:57:30 -05:00
|
|
|
{
|
2024-10-20 15:16:59 -05:00
|
|
|
const bool all_culled =
|
|
|
|
g_vertex_manager->AreAllVerticesCulled(loader, primitive, dst.GetPointer(), num_loaded);
|
|
|
|
if (!all_culled)
|
|
|
|
{
|
|
|
|
DataReader new_dst = g_vertex_manager->DisableCullAll(stride);
|
|
|
|
memmove(new_dst.GetPointer(), dst.GetPointer(), num_loaded * stride);
|
|
|
|
can_cpu_cull = false;
|
|
|
|
}
|
2022-07-26 03:57:30 -05:00
|
|
|
}
|
|
|
|
|
2024-10-20 15:16:59 -05:00
|
|
|
g_vertex_manager->AddIndices(primitive, num_loaded);
|
|
|
|
g_vertex_manager->FlushData(num_loaded, stride);
|
|
|
|
|
|
|
|
ADDSTAT(g_stats.this_frame.num_prims, num_loaded);
|
|
|
|
} while (count);
|
2014-12-09 08:35:04 +01:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
INCSTAT(g_stats.this_frame.num_primitive_joins);
|
|
|
|
}
|
2014-12-09 08:35:04 +01:00
|
|
|
return size;
|
2014-01-30 14:48:23 +01:00
|
|
|
}
|
2009-08-08 01:39:56 +00:00
|
|
|
|
2022-09-15 23:38:57 +02:00
|
|
|
template int RunVertices<false>(int vtx_attr_group, OpcodeDecoder::Primitive primitive, int count,
|
2022-11-22 16:54:05 -08:00
|
|
|
const u8* src);
|
2022-09-15 23:38:57 +02:00
|
|
|
template int RunVertices<true>(int vtx_attr_group, OpcodeDecoder::Primitive primitive, int count,
|
2022-11-22 16:54:05 -08:00
|
|
|
const u8* src);
|
2022-09-15 23:38:57 +02:00
|
|
|
|
2014-07-26 01:10:44 +02:00
|
|
|
NativeVertexFormat* GetCurrentVertexFormat()
|
2014-06-05 17:55:21 +02:00
|
|
|
{
|
2014-07-26 01:10:44 +02:00
|
|
|
return s_current_vtx_fmt;
|
2014-06-05 17:55:21 +02:00
|
|
|
}
|
|
|
|
|
2019-05-05 23:48:12 +00:00
|
|
|
} // namespace VertexLoaderManager
|