mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2024-11-29 06:21:06 +00:00
Fix optimizer on when to write the binary
The spvtools::Optimizer::Run method should also write the output binary if optimization succeeds without changes but the output binary vector does not have exactly the same contents as the input binary. We have to check both the base pointer of the storage and the size of the vector Added a test for this too. Fixes https://github.com/KhronosGroup/SPIRV-Tools/issues/611
This commit is contained in:
parent
ad3b08280d
commit
afc60bbebf
2
CHANGES
2
CHANGES
@ -16,6 +16,8 @@ v2016.7-dev 2017-01-06
|
|||||||
header.
|
header.
|
||||||
#548: Validator: Error when the reserved OpImageSparseSampleProj* opcodes
|
#548: Validator: Error when the reserved OpImageSparseSampleProj* opcodes
|
||||||
are used.
|
are used.
|
||||||
|
#611: spvtools::Optimizer was failing to save the module to the output
|
||||||
|
binary vector when all passes succeded without changes.
|
||||||
|
|
||||||
v2016.6 2016-12-13
|
v2016.6 2016-12-13
|
||||||
- Published the C++ interface for assembling, disassembling, validation, and
|
- Published the C++ interface for assembling, disassembling, validation, and
|
||||||
|
@ -75,7 +75,10 @@ bool Optimizer::Run(const uint32_t* original_binary,
|
|||||||
if (module == nullptr) return false;
|
if (module == nullptr) return false;
|
||||||
|
|
||||||
auto status = impl_->pass_manager.Run(module.get());
|
auto status = impl_->pass_manager.Run(module.get());
|
||||||
if (status == opt::Pass::Status::SuccessWithChange) {
|
if (status == opt::Pass::Status::SuccessWithChange ||
|
||||||
|
(status == opt::Pass::Status::SuccessWithoutChange &&
|
||||||
|
(optimized_binary->data() != original_binary ||
|
||||||
|
optimized_binary->size() != original_binary_size))) {
|
||||||
optimized_binary->clear();
|
optimized_binary->clear();
|
||||||
module->ToBinary(optimized_binary, /* skip_nop = */ true);
|
module->ToBinary(optimized_binary, /* skip_nop = */ true);
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,11 @@ add_spvtools_unittest(TARGET pass_manager
|
|||||||
LIBS SPIRV-Tools-opt
|
LIBS SPIRV-Tools-opt
|
||||||
)
|
)
|
||||||
|
|
||||||
|
add_spvtools_unittest(TARGET optimizer
|
||||||
|
SRCS optimizer_test.cpp
|
||||||
|
LIBS SPIRV-Tools-opt
|
||||||
|
)
|
||||||
|
|
||||||
add_spvtools_unittest(TARGET pass_strip_debug_info
|
add_spvtools_unittest(TARGET pass_strip_debug_info
|
||||||
SRCS strip_debug_info_test.cpp pass_utils.cpp
|
SRCS strip_debug_info_test.cpp pass_utils.cpp
|
||||||
LIBS SPIRV-Tools-opt
|
LIBS SPIRV-Tools-opt
|
||||||
|
109
test/opt/optimizer_test.cpp
Normal file
109
test/opt/optimizer_test.cpp
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
#include <gmock/gmock.h>
|
||||||
|
|
||||||
|
#include "spirv-tools/libspirv.hpp"
|
||||||
|
#include "spirv-tools/optimizer.hpp"
|
||||||
|
|
||||||
|
#include "pass_fixture.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
using spvtools::CreateNullPass;
|
||||||
|
using spvtools::CreateStripDebugInfoPass;
|
||||||
|
using spvtools::Optimizer;
|
||||||
|
using spvtools::SpirvTools;
|
||||||
|
using ::testing::Eq;
|
||||||
|
|
||||||
|
TEST(Optimizer, CanRunNullPassWithDistinctInputOutputVectors) {
|
||||||
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
std::vector<uint32_t> binary_in;
|
||||||
|
tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary_in);
|
||||||
|
|
||||||
|
Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
opt.RegisterPass(CreateNullPass());
|
||||||
|
std::vector<uint32_t> binary_out;
|
||||||
|
opt.Run(binary_in.data(), binary_in.size(), &binary_out);
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
tools.Disassemble(binary_out.data(), binary_out.size(), &disassembly);
|
||||||
|
EXPECT_THAT(disassembly, Eq("OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Optimizer, CanRunTransformingPassWithDistinctInputOutputVectors) {
|
||||||
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
std::vector<uint32_t> binary_in;
|
||||||
|
tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary_in);
|
||||||
|
|
||||||
|
Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
opt.RegisterPass(CreateStripDebugInfoPass());
|
||||||
|
std::vector<uint32_t> binary_out;
|
||||||
|
opt.Run(binary_in.data(), binary_in.size(), &binary_out);
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
tools.Disassemble(binary_out.data(), binary_out.size(), &disassembly);
|
||||||
|
EXPECT_THAT(disassembly, Eq("%void = OpTypeVoid\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Optimizer, CanRunNullPassWithAliasedVectors) {
|
||||||
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
std::vector<uint32_t> binary;
|
||||||
|
tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
|
||||||
|
|
||||||
|
Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
opt.RegisterPass(CreateNullPass());
|
||||||
|
opt.Run(binary.data(), binary.size(), &binary); // This is the key.
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
tools.Disassemble(binary.data(), binary.size(), &disassembly);
|
||||||
|
EXPECT_THAT(disassembly, Eq("OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Optimizer, CanRunNullPassWithAliasedVectorDataButDifferentSize) {
|
||||||
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
std::vector<uint32_t> binary;
|
||||||
|
tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
|
||||||
|
|
||||||
|
Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
opt.RegisterPass(CreateNullPass());
|
||||||
|
auto orig_size = binary.size();
|
||||||
|
// Now change the size. Add a word that will be ignored
|
||||||
|
// by the optimizer.
|
||||||
|
binary.push_back(42);
|
||||||
|
EXPECT_THAT(orig_size + 1, Eq(binary.size()));
|
||||||
|
opt.Run(binary.data(), orig_size, &binary); // This is the key.
|
||||||
|
// The binary vector should have been rewritten.
|
||||||
|
EXPECT_THAT(binary.size(), Eq(orig_size));
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
tools.Disassemble(binary.data(), binary.size(), &disassembly);
|
||||||
|
EXPECT_THAT(disassembly, Eq("OpName %foo \"foo\"\n%foo = OpTypeVoid\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Optimizer, CanRunTransformingPassWithAliasedVectors) {
|
||||||
|
SpirvTools tools(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
std::vector<uint32_t> binary;
|
||||||
|
tools.Assemble("OpName %foo \"foo\"\n%foo = OpTypeVoid", &binary);
|
||||||
|
|
||||||
|
Optimizer opt(SPV_ENV_UNIVERSAL_1_0);
|
||||||
|
opt.RegisterPass(CreateStripDebugInfoPass());
|
||||||
|
opt.Run(binary.data(), binary.size(), &binary); // This is the key
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
tools.Disassemble(binary.data(), binary.size(), &disassembly);
|
||||||
|
EXPECT_THAT(disassembly, Eq("%void = OpTypeVoid\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
Loading…
Reference in New Issue
Block a user