2015-05-24 06:32:32 +02:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2015-05-24 06:32:32 +02:00
|
|
|
|
2022-08-05 21:10:17 -07:00
|
|
|
#include <bit>
|
2015-03-12 11:27:22 +01:00
|
|
|
#include <limits>
|
|
|
|
#include <memory>
|
|
|
|
#include <tuple>
|
|
|
|
#include <type_traits>
|
2014-12-13 00:23:54 +01:00
|
|
|
#include <unordered_set>
|
2014-08-01 23:42:36 -07:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
#include <gtest/gtest.h> // NOLINT
|
|
|
|
|
2014-08-01 23:42:36 -07:00
|
|
|
#include "Common/Common.h"
|
2021-01-28 12:49:28 +01:00
|
|
|
#include "Common/MathUtil.h"
|
2015-03-12 11:27:22 +01:00
|
|
|
#include "VideoCommon/CPMemory.h"
|
2014-08-01 23:42:36 -07:00
|
|
|
#include "VideoCommon/DataReader.h"
|
2015-03-12 11:27:22 +01:00
|
|
|
#include "VideoCommon/OpcodeDecoding.h"
|
2014-12-13 01:51:14 +01:00
|
|
|
#include "VideoCommon/VertexLoaderBase.h"
|
2015-05-30 00:42:45 +12:00
|
|
|
#include "VideoCommon/VertexLoaderManager.h"
|
2014-08-01 23:42:36 -07:00
|
|
|
|
|
|
|
TEST(VertexLoaderUID, UniqueEnough)
|
|
|
|
{
|
2014-12-13 00:23:54 +01:00
|
|
|
std::unordered_set<VertexLoaderUID> uids;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-01 23:42:36 -07:00
|
|
|
TVtxDesc vtx_desc;
|
|
|
|
VAT vat;
|
|
|
|
uids.insert(VertexLoaderUID(vtx_desc, vat));
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-06-08 21:35:43 -07:00
|
|
|
vtx_desc.low.Hex = 0x76543210;
|
|
|
|
vtx_desc.high.Hex = 0xFEDCBA98;
|
2024-08-15 14:20:16 -07:00
|
|
|
EXPECT_FALSE(uids.contains(VertexLoaderUID(vtx_desc, vat)));
|
2014-08-01 23:42:36 -07:00
|
|
|
uids.insert(VertexLoaderUID(vtx_desc, vat));
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-01 23:42:36 -07:00
|
|
|
vat.g0.Hex = 0xFFFFFFFF;
|
|
|
|
vat.g1.Hex = 0xFFFFFFFF;
|
|
|
|
vat.g2.Hex = 0xFFFFFFFF;
|
2024-08-15 14:20:16 -07:00
|
|
|
EXPECT_FALSE(uids.contains(VertexLoaderUID(vtx_desc, vat)));
|
2014-08-01 23:42:36 -07:00
|
|
|
uids.insert(VertexLoaderUID(vtx_desc, vat));
|
|
|
|
}
|
|
|
|
|
2014-08-02 14:02:38 -07:00
|
|
|
static u8 input_memory[16 * 1024 * 1024];
|
|
|
|
static u8 output_memory[16 * 1024 * 1024];
|
|
|
|
|
2014-08-01 23:42:36 -07:00
|
|
|
class VertexLoaderTest : public testing::Test
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
void SetUp() override
|
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
memset(input_memory, 0, sizeof(input_memory));
|
|
|
|
memset(output_memory, 0xFF, sizeof(input_memory));
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-03-18 19:24:35 -07:00
|
|
|
m_vtx_desc.low.Hex = 0;
|
|
|
|
m_vtx_desc.high.Hex = 0;
|
|
|
|
|
|
|
|
m_vtx_attr.g0.Hex = 0;
|
|
|
|
m_vtx_attr.g1.Hex = 0;
|
|
|
|
m_vtx_attr.g2.Hex = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
m_loader = nullptr;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-02 14:02:38 -07:00
|
|
|
ResetPointers();
|
2014-08-01 23:42:36 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-17 07:05:19 +01:00
|
|
|
void CreateAndCheckSizes(size_t input_size, size_t output_size)
|
2015-03-12 11:27:22 +01:00
|
|
|
{
|
2015-12-22 19:59:32 -05:00
|
|
|
m_loader = VertexLoaderBase::CreateVertexLoader(m_vtx_desc, m_vtx_attr);
|
2021-03-11 12:55:25 -08:00
|
|
|
ASSERT_EQ(input_size, m_loader->m_vertex_size);
|
2015-03-12 11:27:22 +01:00
|
|
|
ASSERT_EQ((int)output_size, m_loader->m_native_vtx_decl.stride);
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-01 23:42:36 -07:00
|
|
|
template <typename T>
|
|
|
|
void Input(T val)
|
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
// Write swapped.
|
|
|
|
m_src.Write<T, true>(val);
|
2014-08-01 23:42:36 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2018-05-10 10:35:00 -04:00
|
|
|
void ExpectOut(float expected)
|
2014-08-01 23:42:36 -07:00
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
// Read unswapped.
|
2018-05-10 10:35:00 -04:00
|
|
|
const float actual = m_dst.Read<float, false>();
|
|
|
|
|
2024-08-14 08:22:57 +02:00
|
|
|
if (!actual || std::isnan(actual))
|
2022-08-05 21:10:17 -07:00
|
|
|
EXPECT_EQ(std::bit_cast<u32>(expected), std::bit_cast<u32>(actual));
|
2015-03-12 11:27:22 +01:00
|
|
|
else
|
2018-05-10 10:35:00 -04:00
|
|
|
EXPECT_EQ(expected, actual);
|
2014-08-01 23:42:36 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
void RunVertices(int count, int expected_count = -1)
|
2014-08-02 14:02:38 -07:00
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
if (expected_count == -1)
|
|
|
|
expected_count = count;
|
|
|
|
ResetPointers();
|
2022-11-22 16:54:05 -08:00
|
|
|
int actual_count = m_loader->RunVertices(m_src.GetPointer(), m_dst.GetPointer(), count);
|
2015-03-12 11:27:22 +01:00
|
|
|
EXPECT_EQ(actual_count, expected_count);
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-02 14:02:38 -07:00
|
|
|
void ResetPointers()
|
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
m_src = DataReader(input_memory, input_memory + sizeof(input_memory));
|
|
|
|
m_dst = DataReader(output_memory, output_memory + sizeof(output_memory));
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
DataReader m_src;
|
|
|
|
DataReader m_dst;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-02 14:02:38 -07:00
|
|
|
TVtxDesc m_vtx_desc;
|
|
|
|
VAT m_vtx_attr;
|
2015-03-12 11:27:22 +01:00
|
|
|
std::unique_ptr<VertexLoaderBase> m_loader;
|
2014-08-01 23:42:36 -07:00
|
|
|
};
|
|
|
|
|
2021-02-08 15:22:10 -08:00
|
|
|
class VertexLoaderParamTest
|
|
|
|
: public VertexLoaderTest,
|
|
|
|
public ::testing::WithParamInterface<
|
|
|
|
std::tuple<VertexComponentFormat, ComponentFormat, CoordComponentCount, int>>
|
2015-03-12 11:27:22 +01:00
|
|
|
{
|
|
|
|
};
|
2023-01-13 16:21:39 -08:00
|
|
|
INSTANTIATE_TEST_SUITE_P(
|
2021-02-08 15:22:10 -08:00
|
|
|
AllCombinations, VertexLoaderParamTest,
|
|
|
|
::testing::Combine(
|
|
|
|
::testing::Values(VertexComponentFormat::Direct, VertexComponentFormat::Index8,
|
|
|
|
VertexComponentFormat::Index16),
|
|
|
|
::testing::Values(ComponentFormat::UByte, ComponentFormat::Byte, ComponentFormat::UShort,
|
2024-04-01 21:37:11 -07:00
|
|
|
ComponentFormat::Short, ComponentFormat::Float,
|
|
|
|
ComponentFormat::InvalidFloat5, ComponentFormat::InvalidFloat6,
|
|
|
|
ComponentFormat::InvalidFloat7),
|
2021-02-08 15:22:10 -08:00
|
|
|
::testing::Values(CoordComponentCount::XY, CoordComponentCount::XYZ),
|
|
|
|
::testing::Values(0, 1, 31) // frac
|
|
|
|
));
|
2015-03-12 11:27:22 +01:00
|
|
|
|
|
|
|
TEST_P(VertexLoaderParamTest, PositionAll)
|
2014-08-01 23:42:36 -07:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
VertexComponentFormat addr;
|
|
|
|
ComponentFormat format;
|
|
|
|
CoordComponentCount elements;
|
|
|
|
int frac;
|
2015-03-12 11:27:22 +01:00
|
|
|
std::tie(addr, format, elements, frac) = GetParam();
|
2021-02-08 15:22:10 -08:00
|
|
|
this->m_vtx_desc.low.Position = addr;
|
2015-03-12 11:27:22 +01:00
|
|
|
this->m_vtx_attr.g0.PosFormat = format;
|
|
|
|
this->m_vtx_attr.g0.PosElements = elements;
|
|
|
|
this->m_vtx_attr.g0.PosFrac = frac;
|
|
|
|
this->m_vtx_attr.g0.ByteDequant = true;
|
2021-02-08 15:22:10 -08:00
|
|
|
const u32 elem_size = GetElementSize(format);
|
|
|
|
const u32 elem_count = elements == CoordComponentCount::XY ? 2 : 3;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
std::vector<float> values = {
|
|
|
|
std::numeric_limits<float>::lowest(),
|
|
|
|
std::numeric_limits<float>::denorm_min(),
|
|
|
|
std::numeric_limits<float>::min(),
|
|
|
|
std::numeric_limits<float>::max(),
|
|
|
|
std::numeric_limits<float>::quiet_NaN(),
|
|
|
|
std::numeric_limits<float>::infinity(),
|
|
|
|
-0x8000,
|
|
|
|
-0x80,
|
|
|
|
-1,
|
2021-01-28 12:49:28 +01:00
|
|
|
-0.0,
|
2015-03-12 11:27:22 +01:00
|
|
|
0,
|
|
|
|
1,
|
|
|
|
123,
|
|
|
|
0x7F,
|
|
|
|
0xFF,
|
|
|
|
0x7FFF,
|
|
|
|
0xFFFF,
|
|
|
|
12345678,
|
|
|
|
};
|
|
|
|
ASSERT_EQ(0u, values.size() % 2);
|
|
|
|
ASSERT_EQ(0u, values.size() % 3);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-02-08 15:22:10 -08:00
|
|
|
int count = (int)values.size() / elem_count;
|
|
|
|
size_t input_size = elem_count * elem_size;
|
|
|
|
if (IsIndexed(addr))
|
2015-03-12 11:27:22 +01:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
input_size = addr == VertexComponentFormat::Index8 ? 1 : 2;
|
2015-03-12 11:27:22 +01:00
|
|
|
for (int i = 0; i < count; i++)
|
2024-04-01 21:37:11 -07:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
if (addr == VertexComponentFormat::Index8)
|
2015-03-12 11:27:22 +01:00
|
|
|
Input<u8>(i);
|
|
|
|
else
|
|
|
|
Input<u16>(i);
|
2024-04-01 21:37:11 -07:00
|
|
|
}
|
2021-06-20 13:47:57 -07:00
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Position] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Position] = elem_count * elem_size;
|
2015-03-12 11:27:22 +01:00
|
|
|
}
|
2021-02-08 15:22:10 -08:00
|
|
|
CreateAndCheckSizes(input_size, elem_count * sizeof(float));
|
2015-03-12 11:27:22 +01:00
|
|
|
for (float value : values)
|
|
|
|
{
|
|
|
|
switch (format)
|
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::UByte:
|
2021-01-28 12:49:28 +01:00
|
|
|
Input(MathUtil::SaturatingCast<u8>(value));
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Byte:
|
2021-01-28 12:49:28 +01:00
|
|
|
Input(MathUtil::SaturatingCast<s8>(value));
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::UShort:
|
2021-01-28 12:49:28 +01:00
|
|
|
Input(MathUtil::SaturatingCast<u16>(value));
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Short:
|
2021-01-28 12:49:28 +01:00
|
|
|
Input(MathUtil::SaturatingCast<s16>(value));
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Float:
|
2024-04-01 21:37:11 -07:00
|
|
|
case ComponentFormat::InvalidFloat5:
|
|
|
|
case ComponentFormat::InvalidFloat6:
|
|
|
|
case ComponentFormat::InvalidFloat7:
|
2015-03-12 11:27:22 +01:00
|
|
|
Input(value);
|
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
}
|
2015-03-12 11:27:22 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
RunVertices(count);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2024-04-01 21:37:11 -07:00
|
|
|
float scale = 1.f / (1u << (format >= ComponentFormat::Float ? 0 : frac));
|
2015-03-12 11:27:22 +01:00
|
|
|
for (auto iter = values.begin(); iter != values.end();)
|
|
|
|
{
|
|
|
|
float f, g;
|
|
|
|
switch (format)
|
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::UByte:
|
2021-01-28 12:49:28 +01:00
|
|
|
f = MathUtil::SaturatingCast<u8>(*iter++);
|
|
|
|
g = MathUtil::SaturatingCast<u8>(*iter++);
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Byte:
|
2021-01-28 12:49:28 +01:00
|
|
|
f = MathUtil::SaturatingCast<s8>(*iter++);
|
|
|
|
g = MathUtil::SaturatingCast<s8>(*iter++);
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::UShort:
|
2021-01-28 12:49:28 +01:00
|
|
|
f = MathUtil::SaturatingCast<u16>(*iter++);
|
|
|
|
g = MathUtil::SaturatingCast<u16>(*iter++);
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Short:
|
2021-01-28 12:49:28 +01:00
|
|
|
f = MathUtil::SaturatingCast<s16>(*iter++);
|
|
|
|
g = MathUtil::SaturatingCast<s16>(*iter++);
|
2015-03-12 11:27:22 +01:00
|
|
|
break;
|
2021-02-08 15:22:10 -08:00
|
|
|
case ComponentFormat::Float:
|
2024-04-01 21:37:11 -07:00
|
|
|
case ComponentFormat::InvalidFloat5:
|
|
|
|
case ComponentFormat::InvalidFloat6:
|
|
|
|
case ComponentFormat::InvalidFloat7:
|
2015-03-12 11:27:22 +01:00
|
|
|
f = *iter++;
|
|
|
|
g = *iter++;
|
|
|
|
break;
|
2017-03-25 15:21:35 -07:00
|
|
|
default:
|
|
|
|
FAIL() << "Unknown format";
|
2015-03-12 11:27:22 +01:00
|
|
|
}
|
|
|
|
ExpectOut(f * scale);
|
|
|
|
ExpectOut(g * scale);
|
|
|
|
}
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
TEST_F(VertexLoaderTest, PositionIndex16FloatXY)
|
2014-08-02 14:02:38 -07:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
2015-03-17 07:05:19 +01:00
|
|
|
CreateAndCheckSizes(sizeof(u16), 2 * sizeof(float));
|
2015-03-12 11:27:22 +01:00
|
|
|
Input<u16>(1);
|
|
|
|
Input<u16>(0);
|
2021-06-20 13:47:57 -07:00
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Position] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Position] = sizeof(float); // ;)
|
2015-03-12 11:27:22 +01:00
|
|
|
Input(1.f);
|
|
|
|
Input(2.f);
|
|
|
|
Input(3.f);
|
|
|
|
RunVertices(2);
|
2015-03-17 07:05:19 +01:00
|
|
|
ExpectOut(2);
|
|
|
|
ExpectOut(3);
|
|
|
|
ExpectOut(1);
|
|
|
|
ExpectOut(2);
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
class VertexLoaderSpeedTest : public VertexLoaderTest,
|
2021-02-08 15:22:10 -08:00
|
|
|
public ::testing::WithParamInterface<std::tuple<ComponentFormat, int>>
|
2015-03-12 11:27:22 +01:00
|
|
|
{
|
|
|
|
};
|
2023-01-13 16:21:39 -08:00
|
|
|
INSTANTIATE_TEST_SUITE_P(
|
2021-02-08 15:22:10 -08:00
|
|
|
FormatsAndElements, VertexLoaderSpeedTest,
|
|
|
|
::testing::Combine(::testing::Values(ComponentFormat::UByte, ComponentFormat::Byte,
|
|
|
|
ComponentFormat::UShort, ComponentFormat::Short,
|
|
|
|
ComponentFormat::Float),
|
|
|
|
::testing::Values(0, 1)));
|
2015-03-12 11:27:22 +01:00
|
|
|
|
|
|
|
TEST_P(VertexLoaderSpeedTest, PositionDirectAll)
|
2014-08-02 14:02:38 -07:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
ComponentFormat format;
|
|
|
|
int elements_i;
|
|
|
|
std::tie(format, elements_i) = GetParam();
|
|
|
|
CoordComponentCount elements = static_cast<CoordComponentCount>(elements_i);
|
|
|
|
fmt::print("format: {}, elements: {}\n", format, elements);
|
|
|
|
const u32 elem_count = elements == CoordComponentCount::XY ? 2 : 3;
|
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Direct;
|
2015-03-12 11:27:22 +01:00
|
|
|
m_vtx_attr.g0.PosFormat = format;
|
|
|
|
m_vtx_attr.g0.PosElements = elements;
|
2021-02-08 15:22:10 -08:00
|
|
|
const size_t elem_size = GetElementSize(format);
|
|
|
|
CreateAndCheckSizes(elem_count * elem_size, elem_count * sizeof(float));
|
2014-08-02 14:02:38 -07:00
|
|
|
for (int i = 0; i < 1000; ++i)
|
2015-03-12 11:27:22 +01:00
|
|
|
RunVertices(100000);
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
2014-08-01 23:42:36 -07:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
TEST_P(VertexLoaderSpeedTest, TexCoordSingleElement)
|
2014-08-02 14:02:38 -07:00
|
|
|
{
|
2021-02-08 15:22:10 -08:00
|
|
|
ComponentFormat format;
|
|
|
|
int elements_i;
|
|
|
|
std::tie(format, elements_i) = GetParam();
|
|
|
|
TexComponentCount elements = static_cast<TexComponentCount>(elements_i);
|
|
|
|
fmt::print("format: {}, elements: {}\n", format, elements);
|
|
|
|
const u32 elem_count = elements == TexComponentCount::S ? 1 : 2;
|
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Byte;
|
|
|
|
m_vtx_desc.high.Tex0Coord = VertexComponentFormat::Direct;
|
2015-03-12 11:27:22 +01:00
|
|
|
m_vtx_attr.g0.Tex0CoordFormat = format;
|
|
|
|
m_vtx_attr.g0.Tex0CoordElements = elements;
|
2021-02-08 15:22:10 -08:00
|
|
|
const size_t elem_size = GetElementSize(format);
|
|
|
|
CreateAndCheckSizes(2 * sizeof(s8) + elem_count * elem_size,
|
|
|
|
2 * sizeof(float) + elem_count * sizeof(float));
|
2014-08-02 14:02:38 -07:00
|
|
|
for (int i = 0; i < 1000; ++i)
|
2015-03-12 11:27:22 +01:00
|
|
|
RunVertices(100000);
|
2014-08-02 14:02:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(VertexLoaderTest, LargeFloatVertexSpeed)
|
|
|
|
{
|
2015-03-12 11:27:22 +01:00
|
|
|
// Enables most attributes in floating point indexed mode to test speed.
|
2024-07-28 22:28:58 +02:00
|
|
|
m_vtx_desc.low.PosMatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex0MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex1MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex2MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex3MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex4MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex5MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex6MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex7MatIdx = true;
|
2021-02-08 15:22:10 -08:00
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.low.Normal = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.low.Color0 = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.low.Color1 = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex0Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex1Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex2Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex3Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex4Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex5Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex6Coord = VertexComponentFormat::Index16;
|
|
|
|
m_vtx_desc.high.Tex7Coord = VertexComponentFormat::Index16;
|
|
|
|
|
|
|
|
m_vtx_attr.g0.PosElements = CoordComponentCount::XYZ;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
2022-05-17 11:54:24 -07:00
|
|
|
m_vtx_attr.g0.NormalElements = NormalComponentCount::NTB;
|
2021-02-08 15:22:10 -08:00
|
|
|
m_vtx_attr.g0.NormalFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g0.Color0Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color0Comp = ColorFormat::RGBA8888;
|
|
|
|
m_vtx_attr.g0.Color1Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color1Comp = ColorFormat::RGBA8888;
|
|
|
|
m_vtx_attr.g0.Tex0CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g0.Tex0CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex1CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex1CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex2CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex2CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex3CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex3CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex4CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex4CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex5CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex5CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex6CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex6CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex7CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex7CoordFormat = ComponentFormat::Float;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-03-12 11:27:22 +01:00
|
|
|
CreateAndCheckSizes(33, 156);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2021-03-26 15:46:37 -07:00
|
|
|
for (int i = 0; i < NUM_VERTEX_COMPONENT_ARRAYS; i++)
|
2015-03-12 11:27:22 +01:00
|
|
|
{
|
2021-06-20 13:47:57 -07:00
|
|
|
VertexLoaderManager::cached_arraybases[static_cast<CPArray>(i)] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[static_cast<CPArray>(i)] = 129;
|
2015-03-12 11:27:22 +01:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-08-02 14:02:38 -07:00
|
|
|
// This test is only done 100x in a row since it's ~20x slower using the
|
|
|
|
// current vertex loader implementation.
|
|
|
|
for (int i = 0; i < 100; ++i)
|
2015-03-12 11:27:22 +01:00
|
|
|
RunVertices(100000);
|
2014-08-01 23:42:36 -07:00
|
|
|
}
|
2022-07-14 15:54:32 -07:00
|
|
|
|
|
|
|
TEST_F(VertexLoaderTest, DirectAllComponents)
|
|
|
|
{
|
2024-07-28 22:28:58 +02:00
|
|
|
m_vtx_desc.low.PosMatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex0MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex1MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex2MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex3MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex4MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex5MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex6MatIdx = true;
|
|
|
|
m_vtx_desc.low.Tex7MatIdx = true;
|
2022-07-14 15:54:32 -07:00
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.low.Normal = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.low.Color0 = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.low.Color1 = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex0Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex1Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex2Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex3Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex4Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex5Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex6Coord = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_desc.high.Tex7Coord = VertexComponentFormat::Direct;
|
|
|
|
|
|
|
|
m_vtx_attr.g0.PosElements = CoordComponentCount::XYZ;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g0.NormalElements = NormalComponentCount::NTB;
|
|
|
|
m_vtx_attr.g0.NormalFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g0.Color0Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color0Comp = ColorFormat::RGBA8888;
|
|
|
|
m_vtx_attr.g0.Color1Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color1Comp = ColorFormat::RGBA8888;
|
|
|
|
m_vtx_attr.g0.Tex0CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g0.Tex0CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex1CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex1CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex2CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex2CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex3CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex3CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g1.Tex4CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g1.Tex4CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex5CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex5CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex6CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex6CoordFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g2.Tex7CoordElements = TexComponentCount::ST;
|
|
|
|
m_vtx_attr.g2.Tex7CoordFormat = ComponentFormat::Float;
|
|
|
|
|
|
|
|
CreateAndCheckSizes(129, 39 * sizeof(float));
|
|
|
|
|
|
|
|
// Pos matrix idx
|
|
|
|
Input<u8>(20);
|
|
|
|
// Tex matrix idx
|
|
|
|
Input<u8>(0);
|
|
|
|
Input<u8>(1);
|
|
|
|
Input<u8>(2);
|
|
|
|
Input<u8>(3);
|
|
|
|
Input<u8>(4);
|
|
|
|
Input<u8>(5);
|
|
|
|
Input<u8>(6);
|
|
|
|
Input<u8>(7);
|
|
|
|
// Position
|
|
|
|
Input(-1.0f);
|
|
|
|
Input(-2.0f);
|
|
|
|
Input(-3.0f);
|
|
|
|
// Normal
|
|
|
|
Input(-4.0f);
|
|
|
|
Input(-5.0f);
|
|
|
|
Input(-6.0f);
|
|
|
|
// Tangent
|
|
|
|
Input(-7.0f);
|
|
|
|
Input(-8.0f);
|
|
|
|
Input(-9.0f);
|
|
|
|
// Binormal
|
|
|
|
Input(-10.0f);
|
|
|
|
Input(-11.0f);
|
|
|
|
Input(-12.0f);
|
|
|
|
// Colors
|
|
|
|
Input<u32>(0x01234567);
|
|
|
|
Input<u32>(0x89abcdef);
|
|
|
|
// Texture coordinates
|
|
|
|
Input(0.1f);
|
|
|
|
Input(-0.9f);
|
|
|
|
Input(1.1f);
|
|
|
|
Input(-1.9f);
|
|
|
|
Input(2.1f);
|
|
|
|
Input(-2.9f);
|
|
|
|
Input(3.1f);
|
|
|
|
Input(-3.9f);
|
|
|
|
Input(4.1f);
|
|
|
|
Input(-4.9f);
|
|
|
|
Input(5.1f);
|
|
|
|
Input(-5.9f);
|
|
|
|
Input(6.1f);
|
|
|
|
Input(-6.9f);
|
|
|
|
Input(7.1f);
|
|
|
|
Input(-7.9f);
|
|
|
|
|
|
|
|
RunVertices(1);
|
|
|
|
|
|
|
|
// Position matrix
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.posmtx.offset, 0 * sizeof(float));
|
|
|
|
EXPECT_EQ((m_dst.Read<u32, false>()), 20u);
|
|
|
|
// Position
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.position.offset, 1 * sizeof(float));
|
|
|
|
ExpectOut(-1.0f);
|
|
|
|
ExpectOut(-2.0f);
|
|
|
|
ExpectOut(-3.0f);
|
|
|
|
// Normal
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.normals[0].offset, 4 * sizeof(float));
|
|
|
|
ExpectOut(-4.0f);
|
|
|
|
ExpectOut(-5.0f);
|
|
|
|
ExpectOut(-6.0f);
|
|
|
|
// Tangent
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.normals[1].offset, 7 * sizeof(float));
|
|
|
|
ExpectOut(-7.0f);
|
|
|
|
ExpectOut(-8.0f);
|
|
|
|
ExpectOut(-9.0f);
|
|
|
|
// Binormal
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.normals[2].offset, 10 * sizeof(float));
|
|
|
|
ExpectOut(-10.0f);
|
|
|
|
ExpectOut(-11.0f);
|
|
|
|
ExpectOut(-12.0f);
|
|
|
|
// Colors
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[0].offset, 13 * sizeof(float));
|
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x01234567u);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[1].offset, 14 * sizeof(float));
|
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x89abcdefu);
|
|
|
|
// Texture coordinates and matrices (interleaved)
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[0].offset, 15 * sizeof(float));
|
|
|
|
ExpectOut(0.1f); // S
|
|
|
|
ExpectOut(-0.9f); // T
|
|
|
|
ExpectOut(0.0f); // matrix (yes, a float)
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[1].offset, 18 * sizeof(float));
|
|
|
|
ExpectOut(1.1f);
|
|
|
|
ExpectOut(-1.9f);
|
|
|
|
ExpectOut(1.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[2].offset, 21 * sizeof(float));
|
|
|
|
ExpectOut(2.1f);
|
|
|
|
ExpectOut(-2.9f);
|
|
|
|
ExpectOut(2.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[3].offset, 24 * sizeof(float));
|
|
|
|
ExpectOut(3.1f);
|
|
|
|
ExpectOut(-3.9f);
|
|
|
|
ExpectOut(3.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[4].offset, 27 * sizeof(float));
|
|
|
|
ExpectOut(4.1f);
|
|
|
|
ExpectOut(-4.9f);
|
|
|
|
ExpectOut(4.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[5].offset, 30 * sizeof(float));
|
|
|
|
ExpectOut(5.1f);
|
|
|
|
ExpectOut(-5.9f);
|
|
|
|
ExpectOut(5.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[6].offset, 33 * sizeof(float));
|
|
|
|
ExpectOut(6.1f);
|
|
|
|
ExpectOut(-6.9f);
|
|
|
|
ExpectOut(6.0f);
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[7].offset, 36 * sizeof(float));
|
|
|
|
ExpectOut(7.1f);
|
|
|
|
ExpectOut(-7.9f);
|
|
|
|
ExpectOut(7.0f);
|
|
|
|
}
|
2022-07-14 17:34:08 -07:00
|
|
|
|
|
|
|
class VertexLoaderNormalTest
|
|
|
|
: public VertexLoaderTest,
|
|
|
|
public ::testing::WithParamInterface<
|
|
|
|
std::tuple<VertexComponentFormat, ComponentFormat, NormalComponentCount, bool>>
|
|
|
|
{
|
|
|
|
};
|
2023-01-13 16:21:39 -08:00
|
|
|
INSTANTIATE_TEST_SUITE_P(
|
2022-07-14 17:34:08 -07:00
|
|
|
AllCombinations, VertexLoaderNormalTest,
|
|
|
|
::testing::Combine(
|
|
|
|
::testing::Values(VertexComponentFormat::NotPresent, VertexComponentFormat::Direct,
|
|
|
|
VertexComponentFormat::Index8, VertexComponentFormat::Index16),
|
|
|
|
::testing::Values(ComponentFormat::UByte, ComponentFormat::Byte, ComponentFormat::UShort,
|
2024-04-01 21:37:11 -07:00
|
|
|
ComponentFormat::Short, ComponentFormat::Float,
|
|
|
|
ComponentFormat::InvalidFloat5, ComponentFormat::InvalidFloat6,
|
|
|
|
ComponentFormat::InvalidFloat7),
|
2022-07-14 17:34:08 -07:00
|
|
|
::testing::Values(NormalComponentCount::N, NormalComponentCount::NTB),
|
|
|
|
::testing::Values(false, true)));
|
|
|
|
|
|
|
|
TEST_P(VertexLoaderNormalTest, NormalAll)
|
|
|
|
{
|
|
|
|
VertexComponentFormat addr;
|
|
|
|
ComponentFormat format;
|
|
|
|
NormalComponentCount elements;
|
|
|
|
bool index3;
|
|
|
|
std::tie(addr, format, elements, index3) = GetParam();
|
|
|
|
|
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Direct;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g0.PosElements = CoordComponentCount::XY;
|
|
|
|
m_vtx_attr.g0.PosFrac = 0;
|
|
|
|
m_vtx_desc.low.Normal = addr;
|
|
|
|
m_vtx_attr.g0.NormalFormat = format;
|
|
|
|
m_vtx_attr.g0.NormalElements = elements;
|
|
|
|
m_vtx_attr.g0.NormalIndex3 = index3;
|
|
|
|
|
|
|
|
const u32 in_size = [&]() -> u32 {
|
|
|
|
if (addr == VertexComponentFormat::NotPresent)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (IsIndexed(addr))
|
|
|
|
{
|
|
|
|
const u32 base_size = (addr == VertexComponentFormat::Index8) ? 1 : 2;
|
|
|
|
if (elements == NormalComponentCount::NTB)
|
|
|
|
return (index3 ? 3 : 1) * base_size;
|
|
|
|
else
|
|
|
|
return 1 * base_size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const u32 base_count = (elements == NormalComponentCount::NTB) ? 9 : 3;
|
|
|
|
const u32 base_size = GetElementSize(format);
|
|
|
|
return base_count * base_size;
|
|
|
|
}
|
|
|
|
}();
|
|
|
|
const u32 out_size = [&]() -> u32 {
|
|
|
|
if (addr == VertexComponentFormat::NotPresent)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
const u32 base_count = (elements == NormalComponentCount::NTB) ? 9 : 3;
|
|
|
|
return base_count * sizeof(float);
|
|
|
|
}();
|
|
|
|
|
|
|
|
CreateAndCheckSizes(2 * sizeof(float) + in_size, 2 * sizeof(float) + out_size);
|
|
|
|
|
|
|
|
auto input_with_expected_type = [&](float value) {
|
|
|
|
switch (format)
|
|
|
|
{
|
|
|
|
case ComponentFormat::UByte:
|
|
|
|
Input<u8>(value * (1 << 7));
|
|
|
|
break;
|
|
|
|
case ComponentFormat::Byte:
|
|
|
|
Input<s8>(value * (1 << 6));
|
|
|
|
break;
|
|
|
|
case ComponentFormat::UShort:
|
|
|
|
Input<u16>(value * (1 << 15));
|
|
|
|
break;
|
|
|
|
case ComponentFormat::Short:
|
|
|
|
Input<s16>(value * (1 << 14));
|
|
|
|
break;
|
|
|
|
case ComponentFormat::Float:
|
2024-04-01 21:37:11 -07:00
|
|
|
case ComponentFormat::InvalidFloat5:
|
|
|
|
case ComponentFormat::InvalidFloat6:
|
|
|
|
case ComponentFormat::InvalidFloat7:
|
2022-07-14 17:34:08 -07:00
|
|
|
Input<float>(value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
auto create_normal = [&](int counter_base) {
|
|
|
|
if (addr == VertexComponentFormat::Direct)
|
|
|
|
{
|
|
|
|
input_with_expected_type(counter_base / 32.f);
|
|
|
|
input_with_expected_type((counter_base + 1) / 32.f);
|
|
|
|
input_with_expected_type((counter_base + 2) / 32.f);
|
|
|
|
}
|
|
|
|
else if (addr == VertexComponentFormat::Index8)
|
|
|
|
{
|
|
|
|
// We set up arrays so that this works
|
|
|
|
Input<u8>(counter_base);
|
|
|
|
}
|
|
|
|
else if (addr == VertexComponentFormat::Index16)
|
|
|
|
{
|
|
|
|
Input<u16>(counter_base);
|
|
|
|
}
|
|
|
|
// Do nothing for NotPresent
|
|
|
|
};
|
|
|
|
auto create_tangent_and_binormal = [&](int counter_base) {
|
|
|
|
if (IsIndexed(addr))
|
|
|
|
{
|
|
|
|
// With NormalIndex3, specifying the same index 3 times should give the same result
|
|
|
|
// as specifying one index in non-index3 mode (as the index is biased by bytes).
|
|
|
|
// If index3 is disabled, we don't want to write any more indices.
|
|
|
|
if (index3)
|
|
|
|
{
|
|
|
|
// Tangent
|
|
|
|
create_normal(counter_base);
|
|
|
|
// Binormal
|
|
|
|
create_normal(counter_base);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Tangent
|
|
|
|
create_normal(counter_base + 3);
|
|
|
|
// Binormal
|
|
|
|
create_normal(counter_base + 6);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Create our two vertices
|
|
|
|
// Position 1
|
|
|
|
Input(4.0f);
|
|
|
|
Input(8.0f);
|
|
|
|
// Normal 1
|
|
|
|
create_normal(1);
|
|
|
|
if (elements == NormalComponentCount::NTB)
|
|
|
|
{
|
|
|
|
create_tangent_and_binormal(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Position 2
|
|
|
|
Input(6.0f);
|
|
|
|
Input(12.0f);
|
|
|
|
// Normal 1
|
|
|
|
create_normal(10);
|
|
|
|
if (elements == NormalComponentCount::NTB)
|
|
|
|
{
|
|
|
|
create_tangent_and_binormal(10);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create an array for indexed representations
|
|
|
|
for (int i = 0; i < NUM_VERTEX_COMPONENT_ARRAYS; i++)
|
|
|
|
{
|
|
|
|
VertexLoaderManager::cached_arraybases[static_cast<CPArray>(i)] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[static_cast<CPArray>(i)] = GetElementSize(format);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < 32; i++)
|
|
|
|
input_with_expected_type(i / 32.f);
|
|
|
|
|
|
|
|
// Pre-fill these values to detect if they're modified
|
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
|
|
|
VertexLoaderManager::normal_cache = {-42.f, -43.f, -44.f, -45.f};
|
2022-07-14 17:34:08 -07:00
|
|
|
VertexLoaderManager::binormal_cache = {42.f, 43.f, 44.f, 45.f};
|
|
|
|
VertexLoaderManager::tangent_cache = {46.f, 47.f, 48.f, 49.f};
|
|
|
|
|
|
|
|
RunVertices(2);
|
|
|
|
|
|
|
|
// First vertex, position
|
|
|
|
ExpectOut(4.0f);
|
|
|
|
ExpectOut(8.0f);
|
|
|
|
if (addr != VertexComponentFormat::NotPresent)
|
|
|
|
{
|
|
|
|
// Normal
|
|
|
|
ExpectOut(1 / 32.f);
|
|
|
|
ExpectOut(2 / 32.f);
|
|
|
|
ExpectOut(3 / 32.f);
|
|
|
|
if (elements == NormalComponentCount::NTB)
|
|
|
|
{
|
|
|
|
// Tangent
|
|
|
|
ExpectOut(4 / 32.f);
|
|
|
|
ExpectOut(5 / 32.f);
|
|
|
|
ExpectOut(6 / 32.f);
|
|
|
|
// Binormal
|
|
|
|
ExpectOut(7 / 32.f);
|
|
|
|
ExpectOut(8 / 32.f);
|
|
|
|
ExpectOut(9 / 32.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Second vertex, position
|
|
|
|
ExpectOut(6.0f);
|
|
|
|
ExpectOut(12.0f);
|
|
|
|
if (addr != VertexComponentFormat::NotPresent)
|
|
|
|
{
|
|
|
|
// Normal
|
|
|
|
ExpectOut(10 / 32.f);
|
|
|
|
ExpectOut(11 / 32.f);
|
|
|
|
ExpectOut(12 / 32.f);
|
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
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[0], 10 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[1], 11 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[2], 12 / 32.f);
|
2022-07-14 17:34:08 -07:00
|
|
|
if (elements == NormalComponentCount::NTB)
|
|
|
|
{
|
|
|
|
// Tangent
|
|
|
|
ExpectOut(13 / 32.f);
|
|
|
|
ExpectOut(14 / 32.f);
|
|
|
|
ExpectOut(15 / 32.f);
|
|
|
|
// Binormal
|
|
|
|
ExpectOut(16 / 32.f);
|
|
|
|
ExpectOut(17 / 32.f);
|
|
|
|
ExpectOut(18 / 32.f);
|
|
|
|
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[0], 13 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[1], 14 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[2], 15 / 32.f);
|
|
|
|
// Last index is padding/junk
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[0], 16 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[1], 17 / 32.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[2], 18 / 32.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
if (addr == VertexComponentFormat::NotPresent)
|
|
|
|
{
|
|
|
|
// Expect these to not be written
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[0], -42.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[1], -43.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[2], -44.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::normal_cache[3], -45.f);
|
|
|
|
}
|
2022-07-14 17:34:08 -07:00
|
|
|
if (addr == VertexComponentFormat::NotPresent || elements == NormalComponentCount::N)
|
|
|
|
{
|
|
|
|
// Expect these to not be written
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[0], 42.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[1], 43.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[2], 44.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::binormal_cache[3], 45.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[0], 46.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[1], 47.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[2], 48.f);
|
|
|
|
EXPECT_EQ(VertexLoaderManager::tangent_cache[3], 49.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-02 14:13:39 -08:00
|
|
|
class VertexLoaderSkippedColorsTest : public VertexLoaderTest,
|
|
|
|
public ::testing::WithParamInterface<std::tuple<bool, bool>>
|
|
|
|
{
|
|
|
|
};
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AllCombinations, VertexLoaderSkippedColorsTest,
|
|
|
|
::testing::Combine(::testing::Values(false, true),
|
|
|
|
::testing::Values(false, true)));
|
|
|
|
|
|
|
|
TEST_P(VertexLoaderSkippedColorsTest, SkippedColors)
|
|
|
|
{
|
|
|
|
bool enable_color_0, enable_color_1;
|
|
|
|
std::tie(enable_color_0, enable_color_1) = GetParam();
|
|
|
|
|
|
|
|
size_t input_size = 1;
|
|
|
|
size_t output_size = 3 * sizeof(float);
|
|
|
|
size_t color_0_offset = 0;
|
|
|
|
size_t color_1_offset = 0;
|
|
|
|
|
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Index8;
|
|
|
|
if (enable_color_0)
|
|
|
|
{
|
|
|
|
m_vtx_desc.low.Color0 = VertexComponentFormat::Index8;
|
|
|
|
input_size++;
|
|
|
|
color_0_offset = output_size;
|
|
|
|
output_size += sizeof(u32);
|
|
|
|
}
|
|
|
|
if (enable_color_1)
|
|
|
|
{
|
|
|
|
m_vtx_desc.low.Color1 = VertexComponentFormat::Index8;
|
|
|
|
input_size++;
|
|
|
|
color_1_offset = output_size;
|
|
|
|
output_size += sizeof(u32);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_vtx_attr.g0.PosElements = CoordComponentCount::XYZ;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
|
|
|
m_vtx_attr.g0.Color0Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color0Comp = ColorFormat::RGBA8888;
|
|
|
|
m_vtx_attr.g0.Color1Elements = ColorComponentCount::RGBA;
|
|
|
|
m_vtx_attr.g0.Color1Comp = ColorFormat::RGBA8888;
|
|
|
|
|
|
|
|
CreateAndCheckSizes(input_size, output_size);
|
|
|
|
|
|
|
|
// Vertex 0
|
|
|
|
Input<u8>(1);
|
|
|
|
if (enable_color_0)
|
|
|
|
Input<u8>(1);
|
|
|
|
if (enable_color_1)
|
|
|
|
Input<u8>(1);
|
|
|
|
// Vertex 1
|
|
|
|
Input<u8>(0);
|
|
|
|
if (enable_color_0)
|
|
|
|
Input<u8>(0);
|
|
|
|
if (enable_color_1)
|
|
|
|
Input<u8>(0);
|
|
|
|
// Position array
|
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Position] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Position] =
|
|
|
|
sizeof(float); // so 1, 2, 3 for index 0; 2, 3, 4 for index 1
|
|
|
|
Input(1.f);
|
|
|
|
Input(2.f);
|
|
|
|
Input(3.f);
|
|
|
|
Input(4.f);
|
|
|
|
// Color array 0
|
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Color0] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Color0] = sizeof(u32);
|
|
|
|
Input<u32>(0x00010203u);
|
|
|
|
Input<u32>(0x04050607u);
|
|
|
|
// Color array 1
|
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Color1] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Color1] = sizeof(u32);
|
|
|
|
Input<u32>(0x08090a0bu);
|
|
|
|
Input<u32>(0x0c0d0e0fu);
|
|
|
|
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[0].enable, enable_color_0);
|
|
|
|
if (enable_color_0)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[0].offset, color_0_offset);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[1].enable, enable_color_1);
|
|
|
|
if (enable_color_1)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.colors[1].offset, color_1_offset);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
|
|
|
|
RunVertices(2);
|
|
|
|
// Vertex 0
|
|
|
|
ExpectOut(2);
|
|
|
|
ExpectOut(3);
|
|
|
|
ExpectOut(4);
|
|
|
|
if (enable_color_0)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x04050607u);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
if (enable_color_1)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x0c0d0e0fu);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
// Vertex 1
|
|
|
|
ExpectOut(1);
|
|
|
|
ExpectOut(2);
|
|
|
|
ExpectOut(3);
|
|
|
|
if (enable_color_0)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x00010203u);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
if (enable_color_1)
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 14:13:39 -08:00
|
|
|
EXPECT_EQ((m_dst.Read<u32, true>()), 0x08090a0bu);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 14:13:39 -08:00
|
|
|
}
|
|
|
|
|
2023-12-02 15:39:43 -08:00
|
|
|
class VertexLoaderSkippedTexCoordsTest : public VertexLoaderTest,
|
|
|
|
public ::testing::WithParamInterface<u32>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static constexpr u32 NUM_COMPONENTS_TO_TEST = 3;
|
|
|
|
static constexpr u32 NUM_PARAMETERS_PER_COMPONENT = 3;
|
|
|
|
static constexpr u32 NUM_COMBINATIONS =
|
|
|
|
1 << (NUM_COMPONENTS_TO_TEST * NUM_PARAMETERS_PER_COMPONENT);
|
|
|
|
};
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AllCombinations, VertexLoaderSkippedTexCoordsTest,
|
|
|
|
::testing::Range(0u, VertexLoaderSkippedTexCoordsTest::NUM_COMBINATIONS));
|
|
|
|
|
|
|
|
TEST_P(VertexLoaderSkippedTexCoordsTest, SkippedTextures)
|
|
|
|
{
|
|
|
|
std::array<bool, NUM_COMPONENTS_TO_TEST> enable_tex, enable_matrix, use_st;
|
|
|
|
const u32 param = GetParam();
|
|
|
|
for (u32 component = 0; component < NUM_COMPONENTS_TO_TEST; component++)
|
|
|
|
{
|
|
|
|
const u32 bits = param >> (component * NUM_PARAMETERS_PER_COMPONENT);
|
|
|
|
enable_tex[component] = (bits & 1);
|
|
|
|
enable_matrix[component] = (bits & 2);
|
|
|
|
use_st[component] = (bits & 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t input_size = 1;
|
|
|
|
size_t output_size = 3 * sizeof(float);
|
|
|
|
|
|
|
|
std::array<bool, NUM_COMPONENTS_TO_TEST> component_enabled{};
|
|
|
|
std::array<size_t, NUM_COMPONENTS_TO_TEST> component_offset{};
|
|
|
|
|
|
|
|
m_vtx_desc.low.Position = VertexComponentFormat::Index8;
|
|
|
|
m_vtx_attr.g0.PosElements = CoordComponentCount::XYZ;
|
|
|
|
m_vtx_attr.g0.PosFormat = ComponentFormat::Float;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
if (enable_matrix[i] || enable_tex[i])
|
|
|
|
{
|
|
|
|
component_enabled[i] = true;
|
|
|
|
component_offset[i] = output_size;
|
|
|
|
if (enable_matrix[i])
|
|
|
|
{
|
|
|
|
output_size += 3 * sizeof(float);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (use_st[i])
|
|
|
|
{
|
|
|
|
output_size += 2 * sizeof(float);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
output_size += sizeof(float);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enable_matrix[i])
|
|
|
|
{
|
|
|
|
m_vtx_desc.low.TexMatIdx[i] = enable_matrix[i];
|
|
|
|
input_size++;
|
|
|
|
}
|
|
|
|
if (enable_tex[i])
|
|
|
|
{
|
|
|
|
m_vtx_desc.high.TexCoord[i] = VertexComponentFormat::Index8;
|
|
|
|
input_size++;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_vtx_attr.SetTexElements(i, use_st[i] ? TexComponentCount::ST : TexComponentCount::S);
|
|
|
|
m_vtx_attr.SetTexFormat(i, ComponentFormat::Float);
|
|
|
|
m_vtx_attr.SetTexFrac(i, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
CreateAndCheckSizes(input_size, output_size);
|
|
|
|
|
|
|
|
// Vertex 0
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
if (enable_matrix[i])
|
|
|
|
Input<u8>(u8(20 + i));
|
|
|
|
}
|
|
|
|
Input<u8>(1); // Position
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
if (enable_tex[i])
|
|
|
|
Input<u8>(1);
|
|
|
|
}
|
|
|
|
// Vertex 1
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
if (enable_matrix[i])
|
|
|
|
Input<u8>(u8(10 + i));
|
|
|
|
}
|
|
|
|
Input<u8>(0); // Position
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
if (enable_tex[i])
|
|
|
|
Input<u8>(0);
|
|
|
|
}
|
|
|
|
// Position array
|
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::Position] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::Position] =
|
|
|
|
sizeof(float); // so 1, 2, 3 for index 0; 2, 3, 4 for index 1
|
|
|
|
Input(1.f);
|
|
|
|
Input(2.f);
|
|
|
|
Input(3.f);
|
|
|
|
Input(4.f);
|
|
|
|
// Texture coord arrays
|
|
|
|
for (u8 i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
VertexLoaderManager::cached_arraybases[CPArray::TexCoord0 + i] = m_src.GetPointer();
|
|
|
|
g_main_cp_state.array_strides[CPArray::TexCoord0 + i] = 2 * sizeof(float);
|
|
|
|
Input<float>(i * 100 + 11);
|
|
|
|
Input<float>(i * 100 + 12);
|
|
|
|
Input<float>(i * 100 + 21);
|
|
|
|
Input<float>(i * 100 + 22);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[i].enable, component_enabled[i]);
|
|
|
|
if (component_enabled[i])
|
2023-12-11 18:45:23 -05:00
|
|
|
{
|
2023-12-02 15:39:43 -08:00
|
|
|
ASSERT_EQ(m_loader->m_native_vtx_decl.texcoords[i].offset, component_offset[i]);
|
2023-12-11 18:45:23 -05:00
|
|
|
}
|
2023-12-02 15:39:43 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
RunVertices(2);
|
|
|
|
|
|
|
|
// Vertex 0
|
|
|
|
ExpectOut(2);
|
|
|
|
ExpectOut(3);
|
|
|
|
ExpectOut(4);
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
size_t num_read = 0;
|
|
|
|
if (enable_tex[i])
|
|
|
|
{
|
|
|
|
ExpectOut(i * 100 + 21);
|
|
|
|
num_read++;
|
|
|
|
if (use_st[i])
|
|
|
|
{
|
|
|
|
ExpectOut(i * 100 + 22);
|
|
|
|
num_read++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enable_matrix[i])
|
|
|
|
{
|
|
|
|
// With a matrix there are always 3 components; otherwise-unused components should be 0
|
|
|
|
while (num_read++ < 2)
|
|
|
|
ExpectOut(0);
|
|
|
|
ExpectOut(20 + i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Vertex 1
|
|
|
|
ExpectOut(1);
|
|
|
|
ExpectOut(2);
|
|
|
|
ExpectOut(3);
|
|
|
|
for (size_t i = 0; i < NUM_COMPONENTS_TO_TEST; i++)
|
|
|
|
{
|
|
|
|
size_t num_read = 0;
|
|
|
|
if (enable_tex[i])
|
|
|
|
{
|
|
|
|
ExpectOut(i * 100 + 11);
|
|
|
|
num_read++;
|
|
|
|
if (use_st[i])
|
|
|
|
{
|
|
|
|
ExpectOut(i * 100 + 12);
|
|
|
|
num_read++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enable_matrix[i])
|
|
|
|
{
|
|
|
|
// With a matrix there are always 3 components; otherwise-unused components should be 0
|
|
|
|
while (num_read++ < 2)
|
|
|
|
ExpectOut(0);
|
|
|
|
ExpectOut(10 + i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-14 17:34:08 -07:00
|
|
|
// For gtest, which doesn't know about our fmt::formatters by default
|
|
|
|
static void PrintTo(const VertexComponentFormat& t, std::ostream* os)
|
|
|
|
{
|
|
|
|
*os << fmt::to_string(t);
|
|
|
|
}
|
|
|
|
static void PrintTo(const ComponentFormat& t, std::ostream* os)
|
|
|
|
{
|
|
|
|
*os << fmt::to_string(t);
|
|
|
|
}
|
|
|
|
static void PrintTo(const CoordComponentCount& t, std::ostream* os)
|
|
|
|
{
|
|
|
|
*os << fmt::to_string(t);
|
|
|
|
}
|
|
|
|
static void PrintTo(const NormalComponentCount& t, std::ostream* os)
|
|
|
|
{
|
|
|
|
*os << fmt::to_string(t);
|
|
|
|
}
|