mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-13 18:00:05 +00:00
7249506b73
Instead of using the source/table.h methods, this CL switches the stats tool to use the spvtools::Context class and assign the message consumer through the public API.
439 lines
15 KiB
C++
439 lines
15 KiB
C++
// Copyright (c) 2017 Google Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
// Tests for unique type declaration rules validator.
|
|
|
|
#include <string>
|
|
#include <unordered_map>
|
|
|
|
#include "test/test_fixture.h"
|
|
#include "test/unit_spirv.h"
|
|
#include "tools/stats/spirv_stats.h"
|
|
|
|
namespace spvtools {
|
|
namespace stats {
|
|
namespace {
|
|
|
|
using spvtest::ScopedContext;
|
|
|
|
void DiagnosticsMessageHandler(spv_message_level_t level, const char*,
|
|
const spv_position_t& position,
|
|
const char* message) {
|
|
switch (level) {
|
|
case SPV_MSG_FATAL:
|
|
case SPV_MSG_INTERNAL_ERROR:
|
|
case SPV_MSG_ERROR:
|
|
std::cerr << "error: " << position.index << ": " << message << std::endl;
|
|
break;
|
|
case SPV_MSG_WARNING:
|
|
std::cout << "warning: " << position.index << ": " << message
|
|
<< std::endl;
|
|
break;
|
|
case SPV_MSG_INFO:
|
|
std::cout << "info: " << position.index << ": " << message << std::endl;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Calls AggregateStats for binary compiled from |code|.
|
|
void CompileAndAggregateStats(const std::string& code, SpirvStats* stats,
|
|
spv_target_env env = SPV_ENV_UNIVERSAL_1_1) {
|
|
spvtools::Context ctx(env);
|
|
ctx.SetMessageConsumer(DiagnosticsMessageHandler);
|
|
spv_binary binary;
|
|
ASSERT_EQ(SPV_SUCCESS, spvTextToBinary(ctx.CContext(), code.c_str(),
|
|
code.size(), &binary, nullptr));
|
|
|
|
ASSERT_EQ(SPV_SUCCESS, AggregateStats(ctx.CContext(), binary->code,
|
|
binary->wordCount, nullptr, stats));
|
|
spvBinaryDestroy(binary);
|
|
}
|
|
|
|
TEST(AggregateStats, CapabilityHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability GenericPointer
|
|
OpCapability Linkage
|
|
OpMemoryModel Physical32 OpenCL
|
|
)";
|
|
|
|
const std::string code2 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(4u, stats.capability_hist.size());
|
|
EXPECT_EQ(0u, stats.capability_hist.count(SpvCapabilityShader));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityAddresses));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityKernel));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityGenericPointer));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityLinkage));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(5u, stats.capability_hist.size());
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityShader));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityAddresses));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityKernel));
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityGenericPointer));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityLinkage));
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(5u, stats.capability_hist.size());
|
|
EXPECT_EQ(1u, stats.capability_hist.at(SpvCapabilityShader));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityAddresses));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityKernel));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityGenericPointer));
|
|
EXPECT_EQ(3u, stats.capability_hist.at(SpvCapabilityLinkage));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(5u, stats.capability_hist.size());
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityShader));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityAddresses));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityKernel));
|
|
EXPECT_EQ(2u, stats.capability_hist.at(SpvCapabilityGenericPointer));
|
|
EXPECT_EQ(4u, stats.capability_hist.at(SpvCapabilityLinkage));
|
|
}
|
|
|
|
TEST(AggregateStats, ExtensionHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability GenericPointer
|
|
OpCapability Linkage
|
|
OpExtension "SPV_KHR_16bit_storage"
|
|
OpMemoryModel Physical32 OpenCL
|
|
)";
|
|
|
|
const std::string code2 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpExtension "SPV_NV_viewport_array2"
|
|
OpExtension "greatest_extension_ever"
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(1u, stats.extension_hist.size());
|
|
EXPECT_EQ(0u, stats.extension_hist.count("SPV_NV_viewport_array2"));
|
|
EXPECT_EQ(1u, stats.extension_hist.at("SPV_KHR_16bit_storage"));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(3u, stats.extension_hist.size());
|
|
EXPECT_EQ(1u, stats.extension_hist.at("SPV_NV_viewport_array2"));
|
|
EXPECT_EQ(1u, stats.extension_hist.at("SPV_KHR_16bit_storage"));
|
|
EXPECT_EQ(1u, stats.extension_hist.at("greatest_extension_ever"));
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(3u, stats.extension_hist.size());
|
|
EXPECT_EQ(1u, stats.extension_hist.at("SPV_NV_viewport_array2"));
|
|
EXPECT_EQ(2u, stats.extension_hist.at("SPV_KHR_16bit_storage"));
|
|
EXPECT_EQ(1u, stats.extension_hist.at("greatest_extension_ever"));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(3u, stats.extension_hist.size());
|
|
EXPECT_EQ(2u, stats.extension_hist.at("SPV_NV_viewport_array2"));
|
|
EXPECT_EQ(2u, stats.extension_hist.at("SPV_KHR_16bit_storage"));
|
|
EXPECT_EQ(2u, stats.extension_hist.at("greatest_extension_ever"));
|
|
}
|
|
|
|
TEST(AggregateStats, VersionHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(1u, stats.version_hist.size());
|
|
EXPECT_EQ(1u, stats.version_hist.at(0x00010100));
|
|
|
|
CompileAndAggregateStats(code1, &stats, SPV_ENV_UNIVERSAL_1_0);
|
|
EXPECT_EQ(2u, stats.version_hist.size());
|
|
EXPECT_EQ(1u, stats.version_hist.at(0x00010100));
|
|
EXPECT_EQ(1u, stats.version_hist.at(0x00010000));
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(2u, stats.version_hist.size());
|
|
EXPECT_EQ(2u, stats.version_hist.at(0x00010100));
|
|
EXPECT_EQ(1u, stats.version_hist.at(0x00010000));
|
|
|
|
CompileAndAggregateStats(code1, &stats, SPV_ENV_UNIVERSAL_1_0);
|
|
EXPECT_EQ(2u, stats.version_hist.size());
|
|
EXPECT_EQ(2u, stats.version_hist.at(0x00010100));
|
|
EXPECT_EQ(2u, stats.version_hist.at(0x00010000));
|
|
}
|
|
|
|
TEST(AggregateStats, GeneratorHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
const uint32_t kGeneratorKhronosAssembler = SPV_GENERATOR_KHRONOS_ASSEMBLER
|
|
<< 16;
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(1u, stats.generator_hist.size());
|
|
EXPECT_EQ(1u, stats.generator_hist.at(kGeneratorKhronosAssembler));
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(1u, stats.generator_hist.size());
|
|
EXPECT_EQ(2u, stats.generator_hist.at(kGeneratorKhronosAssembler));
|
|
}
|
|
|
|
TEST(AggregateStats, OpcodeHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability Int64
|
|
OpCapability Linkage
|
|
OpMemoryModel Physical32 OpenCL
|
|
%u64 = OpTypeInt 64 0
|
|
%u32 = OpTypeInt 32 0
|
|
%f32 = OpTypeFloat 32
|
|
)";
|
|
|
|
const std::string code2 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpExtension "SPV_NV_viewport_array2"
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(4u, stats.opcode_hist.size());
|
|
EXPECT_EQ(4u, stats.opcode_hist.at(SpvOpCapability));
|
|
EXPECT_EQ(1u, stats.opcode_hist.at(SpvOpMemoryModel));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u, stats.opcode_hist.at(SpvOpTypeFloat));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(5u, stats.opcode_hist.size());
|
|
EXPECT_EQ(6u, stats.opcode_hist.at(SpvOpCapability));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpMemoryModel));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u, stats.opcode_hist.at(SpvOpTypeFloat));
|
|
EXPECT_EQ(1u, stats.opcode_hist.at(SpvOpExtension));
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(5u, stats.opcode_hist.size());
|
|
EXPECT_EQ(10u, stats.opcode_hist.at(SpvOpCapability));
|
|
EXPECT_EQ(3u, stats.opcode_hist.at(SpvOpMemoryModel));
|
|
EXPECT_EQ(4u, stats.opcode_hist.at(SpvOpTypeInt));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpTypeFloat));
|
|
EXPECT_EQ(1u, stats.opcode_hist.at(SpvOpExtension));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(5u, stats.opcode_hist.size());
|
|
EXPECT_EQ(12u, stats.opcode_hist.at(SpvOpCapability));
|
|
EXPECT_EQ(4u, stats.opcode_hist.at(SpvOpMemoryModel));
|
|
EXPECT_EQ(4u, stats.opcode_hist.at(SpvOpTypeInt));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpTypeFloat));
|
|
EXPECT_EQ(2u, stats.opcode_hist.at(SpvOpExtension));
|
|
}
|
|
|
|
TEST(AggregateStats, OpcodeMarkovHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpExtension "SPV_NV_viewport_array2"
|
|
OpMemoryModel Logical GLSL450
|
|
)";
|
|
|
|
const std::string code2 = R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability Int64
|
|
OpCapability Linkage
|
|
OpMemoryModel Physical32 OpenCL
|
|
%u64 = OpTypeInt 64 0
|
|
%u32 = OpTypeInt 32 0
|
|
%f32 = OpTypeFloat 32
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
stats.opcode_markov_hist.resize(2);
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
ASSERT_EQ(2u, stats.opcode_markov_hist.size());
|
|
EXPECT_EQ(2u, stats.opcode_markov_hist[0].size());
|
|
EXPECT_EQ(2u, stats.opcode_markov_hist[0].at(SpvOpCapability).size());
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[0].at(SpvOpExtension).size());
|
|
EXPECT_EQ(
|
|
1u, stats.opcode_markov_hist[0].at(SpvOpCapability).at(SpvOpCapability));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[0].at(SpvOpCapability).at(SpvOpExtension));
|
|
EXPECT_EQ(
|
|
1u, stats.opcode_markov_hist[0].at(SpvOpExtension).at(SpvOpMemoryModel));
|
|
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[1].size());
|
|
EXPECT_EQ(2u, stats.opcode_markov_hist[1].at(SpvOpCapability).size());
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpExtension));
|
|
EXPECT_EQ(
|
|
1u, stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpMemoryModel));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
ASSERT_EQ(2u, stats.opcode_markov_hist.size());
|
|
EXPECT_EQ(4u, stats.opcode_markov_hist[0].size());
|
|
EXPECT_EQ(3u, stats.opcode_markov_hist[0].at(SpvOpCapability).size());
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[0].at(SpvOpExtension).size());
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[0].at(SpvOpMemoryModel).size());
|
|
EXPECT_EQ(2u, stats.opcode_markov_hist[0].at(SpvOpTypeInt).size());
|
|
EXPECT_EQ(
|
|
4u, stats.opcode_markov_hist[0].at(SpvOpCapability).at(SpvOpCapability));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[0].at(SpvOpCapability).at(SpvOpExtension));
|
|
EXPECT_EQ(
|
|
1u, stats.opcode_markov_hist[0].at(SpvOpCapability).at(SpvOpMemoryModel));
|
|
EXPECT_EQ(
|
|
1u, stats.opcode_markov_hist[0].at(SpvOpExtension).at(SpvOpMemoryModel));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[0].at(SpvOpMemoryModel).at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[0].at(SpvOpTypeInt).at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[0].at(SpvOpTypeInt).at(SpvOpTypeFloat));
|
|
|
|
EXPECT_EQ(3u, stats.opcode_markov_hist[1].size());
|
|
EXPECT_EQ(4u, stats.opcode_markov_hist[1].at(SpvOpCapability).size());
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[1].at(SpvOpMemoryModel).size());
|
|
EXPECT_EQ(1u, stats.opcode_markov_hist[1].at(SpvOpTypeInt).size());
|
|
EXPECT_EQ(
|
|
2u, stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpCapability));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpExtension));
|
|
EXPECT_EQ(
|
|
2u, stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpMemoryModel));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[1].at(SpvOpCapability).at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[1].at(SpvOpMemoryModel).at(SpvOpTypeInt));
|
|
EXPECT_EQ(1u,
|
|
stats.opcode_markov_hist[1].at(SpvOpTypeInt).at(SpvOpTypeFloat));
|
|
}
|
|
|
|
TEST(AggregateStats, ConstantLiteralsHistogram) {
|
|
const std::string code1 = R"(
|
|
OpCapability Addresses
|
|
OpCapability Kernel
|
|
OpCapability GenericPointer
|
|
OpCapability Linkage
|
|
OpCapability Float64
|
|
OpCapability Int16
|
|
OpCapability Int64
|
|
OpMemoryModel Physical32 OpenCL
|
|
%u16 = OpTypeInt 16 0
|
|
%u32 = OpTypeInt 32 0
|
|
%u64 = OpTypeInt 64 0
|
|
%f32 = OpTypeFloat 32
|
|
%f64 = OpTypeFloat 64
|
|
%1 = OpConstant %f32 0.1
|
|
%2 = OpConstant %f32 -2
|
|
%3 = OpConstant %f64 -2
|
|
%4 = OpConstant %u16 16
|
|
%5 = OpConstant %u16 2
|
|
%6 = OpConstant %u32 32
|
|
%7 = OpConstant %u64 64
|
|
)";
|
|
|
|
const std::string code2 = R"(
|
|
OpCapability Shader
|
|
OpCapability Linkage
|
|
OpCapability Int16
|
|
OpCapability Int64
|
|
OpMemoryModel Logical GLSL450
|
|
%f32 = OpTypeFloat 32
|
|
%u16 = OpTypeInt 16 0
|
|
%s16 = OpTypeInt 16 1
|
|
%u32 = OpTypeInt 32 0
|
|
%s32 = OpTypeInt 32 1
|
|
%u64 = OpTypeInt 64 0
|
|
%s64 = OpTypeInt 64 1
|
|
%1 = OpConstant %f32 0.1
|
|
%2 = OpConstant %f32 -2
|
|
%3 = OpConstant %u16 1
|
|
%4 = OpConstant %u16 16
|
|
%5 = OpConstant %u16 2
|
|
%6 = OpConstant %s16 -16
|
|
%7 = OpConstant %u32 32
|
|
%8 = OpConstant %s32 2
|
|
%9 = OpConstant %s32 -32
|
|
%10 = OpConstant %u64 64
|
|
%11 = OpConstant %s64 -64
|
|
)";
|
|
|
|
SpirvStats stats;
|
|
|
|
CompileAndAggregateStats(code1, &stats);
|
|
EXPECT_EQ(2u, stats.f32_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.f64_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.f32_constant_hist.at(0.1f));
|
|
EXPECT_EQ(1u, stats.f32_constant_hist.at(-2.f));
|
|
EXPECT_EQ(1u, stats.f64_constant_hist.at(-2));
|
|
|
|
EXPECT_EQ(2u, stats.u16_constant_hist.size());
|
|
EXPECT_EQ(0u, stats.s16_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.u32_constant_hist.size());
|
|
EXPECT_EQ(0u, stats.s32_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.u64_constant_hist.size());
|
|
EXPECT_EQ(0u, stats.s64_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.u16_constant_hist.at(16));
|
|
EXPECT_EQ(1u, stats.u16_constant_hist.at(2));
|
|
EXPECT_EQ(1u, stats.u32_constant_hist.at(32));
|
|
EXPECT_EQ(1u, stats.u64_constant_hist.at(64));
|
|
|
|
CompileAndAggregateStats(code2, &stats);
|
|
EXPECT_EQ(2u, stats.f32_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.f64_constant_hist.size());
|
|
EXPECT_EQ(2u, stats.f32_constant_hist.at(0.1f));
|
|
EXPECT_EQ(2u, stats.f32_constant_hist.at(-2.f));
|
|
EXPECT_EQ(1u, stats.f64_constant_hist.at(-2));
|
|
|
|
EXPECT_EQ(3u, stats.u16_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.s16_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.u32_constant_hist.size());
|
|
EXPECT_EQ(2u, stats.s32_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.u64_constant_hist.size());
|
|
EXPECT_EQ(1u, stats.s64_constant_hist.size());
|
|
EXPECT_EQ(2u, stats.u16_constant_hist.at(16));
|
|
EXPECT_EQ(2u, stats.u16_constant_hist.at(2));
|
|
EXPECT_EQ(1u, stats.u16_constant_hist.at(1));
|
|
EXPECT_EQ(1u, stats.s16_constant_hist.at(-16));
|
|
EXPECT_EQ(2u, stats.u32_constant_hist.at(32));
|
|
EXPECT_EQ(1u, stats.s32_constant_hist.at(2));
|
|
EXPECT_EQ(1u, stats.s32_constant_hist.at(-32));
|
|
EXPECT_EQ(2u, stats.u64_constant_hist.at(64));
|
|
EXPECT_EQ(1u, stats.s64_constant_hist.at(-64));
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace stats
|
|
} // namespace spvtools
|