mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-12 09:20:15 +00:00
0530a532fc
Fixes #5283 * Validate group non-uniform instructions
959 lines
35 KiB
C++
959 lines
35 KiB
C++
// Copyright (c) 2019 Google LLC.
|
|
//
|
|
// 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.
|
|
|
|
#include <vector>
|
|
|
|
#include "gmock/gmock.h"
|
|
#include "test/opt/pass_fixture.h"
|
|
#include "test/opt/pass_utils.h"
|
|
|
|
namespace spvtools {
|
|
namespace opt {
|
|
namespace {
|
|
|
|
using AmdExtToKhrTest = PassTest<::testing::Test>;
|
|
|
|
using ::testing::HasSubstr;
|
|
|
|
std::string GetTest(std::string op_code, std::string new_op_code,
|
|
bool is_float = false) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_ballot"
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[undef:%\w+]] = OpUndef %
|
|
; CHECK-NEXT: )" + new_op_code +
|
|
" %" + (is_float ? "float" : "uint") +
|
|
R"( %uint_3 Reduce [[undef]]
|
|
OpCapability Shader
|
|
OpCapability Groups
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %)" +
|
|
(is_float ? "float" : "uint") + R"(
|
|
%8 = )" + op_code +
|
|
" %" + (is_float ? "float" : "uint") +
|
|
R"( %uint_3 Reduce %7
|
|
OpReturn
|
|
OpFunctionEnd
|
|
|
|
)";
|
|
return text;
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupIAddNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupIAddNonUniformAMD", "OpGroupNonUniformIAdd");
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupFAddNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupFAddNonUniformAMD", "OpGroupNonUniformFAdd", true);
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupUMinNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupUMinNonUniformAMD", "OpGroupNonUniformUMin");
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupSMinNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupSMinNonUniformAMD", "OpGroupNonUniformSMin");
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupFMinNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupFMinNonUniformAMD", "OpGroupNonUniformFMin", true);
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupUMaxNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupUMaxNonUniformAMD", "OpGroupNonUniformUMax");
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupSMaxNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupSMaxNonUniformAMD", "OpGroupNonUniformSMax");
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceGroupFMaxNonUniformAMD) {
|
|
std::string text =
|
|
GetTest("OpGroupFMaxNonUniformAMD", "OpGroupNonUniformFMax", true);
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceMbcntAMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_ballot"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_ballot"
|
|
; CHECK: OpDecorate [[var:%\w+]] BuiltIn SubgroupLtMask
|
|
; CHECK: [[var]] = OpVariable %_ptr_Input_v4uint Input
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[ld:%\w+]] = OpLoad %v4uint [[var]]
|
|
; CHECK-NEXT: [[shuffle:%\w+]] = OpVectorShuffle %v2uint [[ld]] [[ld]] 0 1
|
|
; CHECK-NEXT: [[bitcast:%\w+]] = OpBitcast %ulong [[shuffle]]
|
|
; CHECK-NEXT: [[and:%\w+]] = OpBitwiseAnd %ulong [[bitcast]] %ulong_0
|
|
; CHECK-NEXT: [[result:%\w+]] = OpBitCount %uint [[and]]
|
|
OpCapability Shader
|
|
OpCapability Int64
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%1 = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %2 "func"
|
|
OpExecutionMode %2 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%ulong = OpTypeInt 64 0
|
|
%ulong_0 = OpConstant %ulong 0
|
|
%2 = OpFunction %void None %4
|
|
%8 = OpLabel
|
|
%9 = OpExtInst %uint %1 MbcntAMD %ulong_0
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceSwizzleInvocationsAMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_ballot"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_ballot"
|
|
; CHECK: OpDecorate [[var:%\w+]] BuiltIn SubgroupLocalInvocationId
|
|
; CHECK: [[subgroup:%\w+]] = OpConstant %uint 3
|
|
; CHECK: [[offset:%\w+]] = OpConstantComposite %v4uint
|
|
; CHECK: [[var]] = OpVariable %_ptr_Input_uint Input
|
|
; CHECK: [[uint_max:%\w+]] = OpConstant %uint 4294967295
|
|
; CHECK: [[ballot_value:%\w+]] = OpConstantComposite %v4uint [[uint_max]] [[uint_max]] [[uint_max]] [[uint_max]]
|
|
; CHECK: [[null:%\w+]] = OpConstantNull [[type:%\w+]]
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[data:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[id:%\w+]] = OpLoad %uint [[var]]
|
|
; CHECK-NEXT: [[quad_idx:%\w+]] = OpBitwiseAnd %uint [[id]] %uint_3
|
|
; CHECK-NEXT: [[quad_ldr:%\w+]] = OpBitwiseXor %uint [[id]] [[quad_idx]]
|
|
; CHECK-NEXT: [[my_offset:%\w+]] = OpVectorExtractDynamic %uint [[offset]] [[quad_idx]]
|
|
; CHECK-NEXT: [[target_inv:%\w+]] = OpIAdd %uint [[quad_ldr]] [[my_offset]]
|
|
; CHECK-NEXT: [[is_active:%\w+]] = OpGroupNonUniformBallotBitExtract %bool [[subgroup]] [[ballot_value]] [[target_inv]]
|
|
; CHECK-NEXT: [[shuffle:%\w+]] = OpGroupNonUniformShuffle [[type]] [[subgroup]] [[data]] [[target_inv]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpSelect [[type]] [[is_active]] [[shuffle]] [[null]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%uint_x = OpConstant %uint 1
|
|
%uint_y = OpConstant %uint 2
|
|
%uint_z = OpConstant %uint 3
|
|
%uint_w = OpConstant %uint 0
|
|
%v4uint = OpTypeVector %uint 4
|
|
%offset = OpConstantComposite %v4uint %uint_x %uint_y %uint_z %uint_x
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%data = OpUndef %uint
|
|
%9 = OpExtInst %uint %ext SwizzleInvocationsAMD %data %offset
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
TEST_F(AmdExtToKhrTest, ReplaceSwizzleInvocationsMaskedAMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_ballot"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_ballot"
|
|
; CHECK: OpDecorate [[var:%\w+]] BuiltIn SubgroupLocalInvocationId
|
|
; CHECK: [[x:%\w+]] = OpConstant %uint 19
|
|
; CHECK: [[y:%\w+]] = OpConstant %uint 12
|
|
; CHECK: [[z:%\w+]] = OpConstant %uint 16
|
|
; CHECK: [[var]] = OpVariable %_ptr_Input_uint Input
|
|
; CHECK: [[mask_extend:%\w+]] = OpConstant %uint 4294967264
|
|
; CHECK: [[uint_max:%\w+]] = OpConstant %uint 4294967295
|
|
; CHECK: [[subgroup:%\w+]] = OpConstant %uint 3
|
|
; CHECK: [[ballot_value:%\w+]] = OpConstantComposite %v4uint [[uint_max]] [[uint_max]] [[uint_max]] [[uint_max]]
|
|
; CHECK: [[null:%\w+]] = OpConstantNull [[type:%\w+]]
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[data:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[id:%\w+]] = OpLoad %uint [[var]]
|
|
; CHECK-NEXT: [[and_mask:%\w+]] = OpBitwiseOr %uint [[x]] [[mask_extend]]
|
|
; CHECK-NEXT: [[and:%\w+]] = OpBitwiseAnd %uint [[id]] [[and_mask]]
|
|
; CHECK-NEXT: [[or:%\w+]] = OpBitwiseOr %uint [[and]] [[y]]
|
|
; CHECK-NEXT: [[target_inv:%\w+]] = OpBitwiseXor %uint [[or]] [[z]]
|
|
; CHECK-NEXT: [[is_active:%\w+]] = OpGroupNonUniformBallotBitExtract %bool [[subgroup]] [[ballot_value]] [[target_inv]]
|
|
; CHECK-NEXT: [[shuffle:%\w+]] = OpGroupNonUniformShuffle [[type]] [[subgroup]] [[data]] [[target_inv]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpSelect [[type]] [[is_active]] [[shuffle]] [[null]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%uint_x = OpConstant %uint 19
|
|
%uint_y = OpConstant %uint 12
|
|
%uint_z = OpConstant %uint 16
|
|
%v3uint = OpTypeVector %uint 3
|
|
%mask = OpConstantComposite %v3uint %uint_x %uint_y %uint_z
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%data = OpUndef %uint
|
|
%9 = OpExtInst %uint %ext SwizzleInvocationsMaskedAMD %data %mask
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceWriteInvocationAMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_ballot"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_ballot"
|
|
; CHECK: OpDecorate [[var:%\w+]] BuiltIn SubgroupLocalInvocationId
|
|
; CHECK: [[var]] = OpVariable %_ptr_Input_uint Input
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[input_val:%\w+]] = OpUndef %uint
|
|
; CHECK-NEXT: [[write_val:%\w+]] = OpUndef %uint
|
|
; CHECK-NEXT: [[ld:%\w+]] = OpLoad %uint [[var]]
|
|
; CHECK-NEXT: [[cmp:%\w+]] = OpIEqual %bool [[ld]] %uint_3
|
|
; CHECK-NEXT: [[result:%\w+]] = OpSelect %uint [[cmp]] [[write_val]] [[input_val]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %uint
|
|
%8 = OpUndef %uint
|
|
%9 = OpExtInst %uint %ext WriteInvocationAMD %7 %8 %uint_3
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceFMin3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeFloat 32
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] FMin [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] FMin [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %float
|
|
%8 = OpUndef %float
|
|
%9 = OpUndef %float
|
|
%10 = OpExtInst %float %ext FMin3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceSMin3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 1
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] SMin [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] SMin [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %int
|
|
%8 = OpUndef %int
|
|
%9 = OpUndef %int
|
|
%10 = OpExtInst %int %ext SMin3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceUMin3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 0
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] UMin [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] UMin [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %uint
|
|
%8 = OpUndef %uint
|
|
%9 = OpUndef %uint
|
|
%10 = OpExtInst %uint %ext UMin3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceFMax3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeFloat 32
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] FMax [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] FMax [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %float
|
|
%8 = OpUndef %float
|
|
%9 = OpUndef %float
|
|
%10 = OpExtInst %float %ext FMax3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceSMax3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 1
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] SMax [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] SMax [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %int
|
|
%8 = OpUndef %int
|
|
%9 = OpUndef %int
|
|
%10 = OpExtInst %int %ext SMax3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceUMax3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 0
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %uint
|
|
%8 = OpUndef %uint
|
|
%9 = OpUndef %uint
|
|
%10 = OpExtInst %uint %ext UMax3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceVecUMax3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeVector
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[temp:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[x]] [[y]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[temp]] [[z]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%vec = OpTypeVector %uint 4
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %vec
|
|
%8 = OpUndef %vec
|
|
%9 = OpUndef %vec
|
|
%10 = OpExtInst %vec %ext UMax3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceFMid3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeFloat 32
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[min:%\w+]] = OpExtInst [[type]] [[ext]] FMin [[y]] [[z]]
|
|
; CHECK-NEXT: [[max:%\w+]] = OpExtInst [[type]] [[ext]] FMax [[y]] [[z]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] FClamp [[x]] [[min]] [[max]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %float
|
|
%8 = OpUndef %float
|
|
%9 = OpUndef %float
|
|
%10 = OpExtInst %float %ext FMid3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceSMid3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 1
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[min:%\w+]] = OpExtInst [[type]] [[ext]] SMin [[y]] [[z]]
|
|
; CHECK-NEXT: [[max:%\w+]] = OpExtInst [[type]] [[ext]] SMax [[y]] [[z]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] SClamp [[x]] [[min]] [[max]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %int
|
|
%8 = OpUndef %int
|
|
%9 = OpUndef %int
|
|
%10 = OpExtInst %int %ext SMid3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceUMid3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeInt 32 0
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[min:%\w+]] = OpExtInst [[type]] [[ext]] UMin [[y]] [[z]]
|
|
; CHECK-NEXT: [[max:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[y]] [[z]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] UClamp [[x]] [[min]] [[max]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %uint
|
|
%8 = OpUndef %uint
|
|
%9 = OpUndef %uint
|
|
%10 = OpExtInst %uint %ext UMid3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceVecUMid3AMD) {
|
|
const std::string text = R"(
|
|
; CHECK: OpCapability Shader
|
|
; CHECK-NOT: OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
; CHECK: [[ext:%\w+]] = OpExtInstImport "GLSL.std.450"
|
|
; CHECK: [[type:%\w+]] = OpTypeVector
|
|
; CHECK: OpFunction
|
|
; CHECK-NEXT: OpLabel
|
|
; CHECK-NEXT: [[x:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[y:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[z:%\w+]] = OpUndef [[type]]
|
|
; CHECK-NEXT: [[min:%\w+]] = OpExtInst [[type]] [[ext]] UMin [[y]] [[z]]
|
|
; CHECK-NEXT: [[max:%\w+]] = OpExtInst [[type]] [[ext]] UMax [[y]] [[z]]
|
|
; CHECK-NEXT: [[result:%\w+]] = OpExtInst [[type]] [[ext]] UClamp [[x]] [[min]] [[max]]
|
|
OpCapability Shader
|
|
OpExtension "SPV_AMD_shader_trinary_minmax"
|
|
%ext = OpExtInstImport "SPV_AMD_shader_trinary_minmax"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %1 "func"
|
|
OpExecutionMode %1 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%3 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%vec = OpTypeVector %uint 3
|
|
%int = OpTypeInt 32 1
|
|
%float = OpTypeFloat 32
|
|
%uint_3 = OpConstant %uint 3
|
|
%1 = OpFunction %void None %3
|
|
%6 = OpLabel
|
|
%7 = OpUndef %vec
|
|
%8 = OpUndef %vec
|
|
%9 = OpUndef %vec
|
|
%10 = OpExtInst %vec %ext UMid3AMD %7 %8 %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceCubeFaceCoordAMD) {
|
|
// Sorry for the Check test. The code sequence is so long, I do not think
|
|
// that a match test would be anymore legible. This tests the replacement of
|
|
// the CubeFaceCoordAMD instruction.
|
|
const std::string before = R"(
|
|
OpCapability Shader
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
OpExtension "SPV_AMD_gcn_shader"
|
|
%1 = OpExtInstImport "SPV_AMD_gcn_shader"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint GLCompute %2 "main"
|
|
OpExecutionMode %2 LocalSize 1 1 1
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%v2float = OpTypeVector %float 2
|
|
%v3float = OpTypeVector %float 3
|
|
%2 = OpFunction %void None %4
|
|
%8 = OpLabel
|
|
%9 = OpUndef %v3float
|
|
%10 = OpExtInst %v2float %1 CubeFaceCoordAMD %9
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string after = R"(OpCapability Shader
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%12 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint GLCompute %2 "main"
|
|
OpExecutionMode %2 LocalSize 1 1 1
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%v2float = OpTypeVector %float 2
|
|
%v3float = OpTypeVector %float 3
|
|
%bool = OpTypeBool
|
|
%float_0 = OpConstant %float 0
|
|
%float_2 = OpConstant %float 2
|
|
%float_0_5 = OpConstant %float 0.5
|
|
%16 = OpConstantComposite %v2float %float_0_5 %float_0_5
|
|
%2 = OpFunction %void None %4
|
|
%8 = OpLabel
|
|
%9 = OpUndef %v3float
|
|
%17 = OpCompositeExtract %float %9 0
|
|
%18 = OpCompositeExtract %float %9 1
|
|
%19 = OpCompositeExtract %float %9 2
|
|
%20 = OpFNegate %float %17
|
|
%21 = OpFNegate %float %18
|
|
%22 = OpFNegate %float %19
|
|
%23 = OpExtInst %float %12 FAbs %17
|
|
%24 = OpExtInst %float %12 FAbs %18
|
|
%25 = OpExtInst %float %12 FAbs %19
|
|
%26 = OpFOrdLessThan %bool %19 %float_0
|
|
%27 = OpFOrdLessThan %bool %18 %float_0
|
|
%28 = OpFOrdLessThan %bool %17 %float_0
|
|
%29 = OpExtInst %float %12 FMax %23 %24
|
|
%30 = OpExtInst %float %12 FMax %25 %29
|
|
%31 = OpFMul %float %float_2 %30
|
|
%32 = OpFOrdGreaterThanEqual %bool %25 %29
|
|
%33 = OpLogicalNot %bool %32
|
|
%34 = OpFOrdGreaterThanEqual %bool %24 %23
|
|
%35 = OpLogicalAnd %bool %33 %34
|
|
%36 = OpSelect %float %26 %20 %17
|
|
%37 = OpSelect %float %28 %19 %22
|
|
%38 = OpSelect %float %35 %17 %37
|
|
%39 = OpSelect %float %32 %36 %38
|
|
%40 = OpSelect %float %27 %22 %19
|
|
%41 = OpSelect %float %35 %40 %21
|
|
%42 = OpCompositeConstruct %v2float %39 %41
|
|
%43 = OpCompositeConstruct %v2float %31 %31
|
|
%44 = OpFDiv %v2float %42 %43
|
|
%10 = OpFAdd %v2float %44 %16
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndCheck<AmdExtensionToKhrPass>(before, after, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, ReplaceCubeFaceIndexAMD) {
|
|
// Sorry for the Check test. The code sequence is so long, I do not think
|
|
// that a match test would be anymore legible. This tests the replacement of
|
|
// the CubeFaceIndexAMD instruction.
|
|
const std::string before = R"(OpCapability Shader
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
OpExtension "SPV_AMD_gcn_shader"
|
|
%1 = OpExtInstImport "SPV_AMD_gcn_shader"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint GLCompute %2 "main"
|
|
OpExecutionMode %2 LocalSize 1 1 1
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%v3float = OpTypeVector %float 3
|
|
%2 = OpFunction %void None %4
|
|
%7 = OpLabel
|
|
%8 = OpUndef %v3float
|
|
%9 = OpExtInst %float %1 CubeFaceIndexAMD %8
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
const std::string after = R"(OpCapability Shader
|
|
OpExtension "SPV_KHR_storage_buffer_storage_class"
|
|
%11 = OpExtInstImport "GLSL.std.450"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint GLCompute %2 "main"
|
|
OpExecutionMode %2 LocalSize 1 1 1
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%float = OpTypeFloat 32
|
|
%v3float = OpTypeVector %float 3
|
|
%bool = OpTypeBool
|
|
%float_0 = OpConstant %float 0
|
|
%float_1 = OpConstant %float 1
|
|
%float_2 = OpConstant %float 2
|
|
%float_3 = OpConstant %float 3
|
|
%float_4 = OpConstant %float 4
|
|
%float_5 = OpConstant %float 5
|
|
%2 = OpFunction %void None %4
|
|
%7 = OpLabel
|
|
%8 = OpUndef %v3float
|
|
%18 = OpCompositeExtract %float %8 0
|
|
%19 = OpCompositeExtract %float %8 1
|
|
%20 = OpCompositeExtract %float %8 2
|
|
%21 = OpExtInst %float %11 FAbs %18
|
|
%22 = OpExtInst %float %11 FAbs %19
|
|
%23 = OpExtInst %float %11 FAbs %20
|
|
%24 = OpFOrdLessThan %bool %20 %float_0
|
|
%25 = OpFOrdLessThan %bool %19 %float_0
|
|
%26 = OpFOrdLessThan %bool %18 %float_0
|
|
%27 = OpExtInst %float %11 FMax %21 %22
|
|
%28 = OpFOrdGreaterThanEqual %bool %23 %27
|
|
%29 = OpFOrdGreaterThanEqual %bool %22 %21
|
|
%30 = OpSelect %float %24 %float_5 %float_4
|
|
%31 = OpSelect %float %25 %float_3 %float_2
|
|
%32 = OpSelect %float %26 %float_1 %float_0
|
|
%33 = OpSelect %float %29 %31 %32
|
|
%9 = OpSelect %float %28 %30 %33
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndCheck<AmdExtensionToKhrPass>(before, after, true);
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, SetVersion) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Int64
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%1 = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %2 "func"
|
|
OpExecutionMode %2 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%ulong = OpTypeInt 64 0
|
|
%ulong_0 = OpConstant %ulong 0
|
|
%2 = OpFunction %void None %4
|
|
%8 = OpLabel
|
|
%9 = OpExtInst %uint %1 MbcntAMD %ulong_0
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
// Set the version to 1.1 and make sure it is upgraded to 1.3.
|
|
SetTargetEnv(SPV_ENV_UNIVERSAL_1_1);
|
|
SetDisassembleOptions(0);
|
|
auto result = SinglePassRunAndDisassemble<AmdExtensionToKhrPass>(
|
|
text, /* skip_nop = */ true, /* skip_validation = */ false);
|
|
|
|
EXPECT_EQ(Pass::Status::SuccessWithChange, std::get<1>(result));
|
|
const std::string& output = std::get<0>(result);
|
|
EXPECT_THAT(output, HasSubstr("Version: 1.3"));
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, SetVersion1) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Int64
|
|
OpExtension "SPV_AMD_shader_ballot"
|
|
%1 = OpExtInstImport "SPV_AMD_shader_ballot"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %2 "func"
|
|
OpExecutionMode %2 OriginUpperLeft
|
|
%void = OpTypeVoid
|
|
%4 = OpTypeFunction %void
|
|
%uint = OpTypeInt 32 0
|
|
%ulong = OpTypeInt 64 0
|
|
%ulong_0 = OpConstant %ulong 0
|
|
%2 = OpFunction %void None %4
|
|
%8 = OpLabel
|
|
%9 = OpExtInst %uint %1 MbcntAMD %ulong_0
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
// Set the version to 1.4 and make sure it is stays the same.
|
|
SetTargetEnv(SPV_ENV_UNIVERSAL_1_4);
|
|
SetDisassembleOptions(0);
|
|
auto result = SinglePassRunAndDisassemble<AmdExtensionToKhrPass>(
|
|
text, /* skip_nop = */ true, /* skip_validation = */ false);
|
|
|
|
EXPECT_EQ(Pass::Status::SuccessWithChange, std::get<1>(result));
|
|
const std::string& output = std::get<0>(result);
|
|
EXPECT_THAT(output, HasSubstr("Version: 1.4"));
|
|
}
|
|
|
|
TEST_F(AmdExtToKhrTest, TimeAMD) {
|
|
const std::string text = R"(
|
|
OpCapability Shader
|
|
OpCapability Int64
|
|
OpExtension "SPV_AMD_gcn_shader"
|
|
; CHECK-NOT: OpExtension "SPV_AMD_gcn_shader"
|
|
; CHECK: OpExtension "SPV_KHR_shader_clock"
|
|
%1 = OpExtInstImport "GLSL.std.450"
|
|
%2 = OpExtInstImport "SPV_AMD_gcn_shader"
|
|
; CHECK-NOT: OpExtInstImport "SPV_AMD_gcn_shader"
|
|
OpMemoryModel Logical GLSL450
|
|
OpEntryPoint Fragment %main "main"
|
|
OpExecutionMode %main OriginUpperLeft
|
|
OpSource GLSL 450
|
|
OpSourceExtension "GL_AMD_gcn_shader"
|
|
OpSourceExtension "GL_ARB_gpu_shader_int64"
|
|
OpName %main "main"
|
|
OpName %time "time"
|
|
%void = OpTypeVoid
|
|
%6 = OpTypeFunction %void
|
|
%ulong = OpTypeInt 64 0
|
|
%_ptr_Function_ulong = OpTypePointer Function %ulong
|
|
%main = OpFunction %void None %6
|
|
%9 = OpLabel
|
|
%time = OpVariable %_ptr_Function_ulong Function
|
|
; CHECK: [[uint:%\w+]] = OpTypeInt 32 0
|
|
; CHECK: [[uint_3:%\w+]] = OpConstant [[uint]] 3
|
|
%10 = OpExtInst %ulong %2 TimeAMD
|
|
; CHECK: %10 = OpReadClockKHR %ulong [[uint_3]]
|
|
OpStore %time %10
|
|
OpReturn
|
|
OpFunctionEnd
|
|
)";
|
|
|
|
SinglePassRunAndMatch<AmdExtensionToKhrPass>(text, true);
|
|
}
|
|
} // namespace
|
|
} // namespace opt
|
|
} // namespace spvtools
|