2018-01-31 21:29:54 +00:00
|
|
|
// Copyright (c) 2018 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.
|
|
|
|
|
|
|
|
// Validates correctness of barrier SPIR-V instructions.
|
|
|
|
|
2018-08-03 19:06:09 +00:00
|
|
|
#include <string>
|
|
|
|
|
2018-07-11 14:27:34 +00:00
|
|
|
#include "source/opcode.h"
|
|
|
|
#include "source/spirv_constant.h"
|
|
|
|
#include "source/val/instruction.h"
|
2019-05-07 16:27:18 +00:00
|
|
|
#include "source/val/validate.h"
|
2018-12-06 19:38:15 +00:00
|
|
|
#include "source/val/validate_memory_semantics.h"
|
2018-11-29 18:48:42 +00:00
|
|
|
#include "source/val/validate_scopes.h"
|
2018-07-11 14:27:34 +00:00
|
|
|
#include "source/val/validation_state.h"
|
2018-01-31 21:29:54 +00:00
|
|
|
|
2018-07-07 13:38:00 +00:00
|
|
|
namespace spvtools {
|
2018-07-10 03:18:44 +00:00
|
|
|
namespace val {
|
2018-01-31 21:29:54 +00:00
|
|
|
|
|
|
|
// Validates correctness of barrier instructions.
|
2018-07-10 14:57:52 +00:00
|
|
|
spv_result_t BarriersPass(ValidationState_t& _, const Instruction* inst) {
|
2022-11-04 21:27:10 +00:00
|
|
|
const spv::Op opcode = inst->opcode();
|
2018-07-10 14:57:52 +00:00
|
|
|
const uint32_t result_type = inst->type_id();
|
2018-01-31 21:29:54 +00:00
|
|
|
|
|
|
|
switch (opcode) {
|
2022-11-04 21:27:10 +00:00
|
|
|
case spv::Op::OpControlBarrier: {
|
2019-05-07 16:27:18 +00:00
|
|
|
if (_.version() < SPV_SPIRV_VERSION_WORD(1, 3)) {
|
2018-08-03 16:57:11 +00:00
|
|
|
_.function(inst->function()->id())
|
|
|
|
->RegisterExecutionModelLimitation(
|
2022-11-04 21:27:10 +00:00
|
|
|
[](spv::ExecutionModel model, std::string* message) {
|
|
|
|
if (model != spv::ExecutionModel::TessellationControl &&
|
|
|
|
model != spv::ExecutionModel::GLCompute &&
|
|
|
|
model != spv::ExecutionModel::Kernel &&
|
|
|
|
model != spv::ExecutionModel::TaskNV &&
|
|
|
|
model != spv::ExecutionModel::MeshNV) {
|
2018-08-03 16:57:11 +00:00
|
|
|
if (message) {
|
|
|
|
*message =
|
|
|
|
"OpControlBarrier requires one of the following "
|
|
|
|
"Execution "
|
2022-09-02 00:36:15 +00:00
|
|
|
"Models: TessellationControl, GLCompute, Kernel, "
|
|
|
|
"MeshNV or TaskNV";
|
2018-08-03 16:57:11 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
2018-03-27 16:01:21 +00:00
|
|
|
}
|
2018-01-31 21:29:54 +00:00
|
|
|
|
2018-07-10 14:57:52 +00:00
|
|
|
const uint32_t execution_scope = inst->word(1);
|
|
|
|
const uint32_t memory_scope = inst->word(2);
|
2018-01-31 21:29:54 +00:00
|
|
|
|
|
|
|
if (auto error = ValidateExecutionScope(_, inst, execution_scope)) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto error = ValidateMemoryScope(_, inst, memory_scope)) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2021-03-16 14:53:37 +00:00
|
|
|
if (auto error = ValidateMemorySemantics(_, inst, 2, memory_scope)) {
|
2018-01-31 21:29:54 +00:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-11-04 21:27:10 +00:00
|
|
|
case spv::Op::OpMemoryBarrier: {
|
2018-07-10 14:57:52 +00:00
|
|
|
const uint32_t memory_scope = inst->word(1);
|
2018-01-31 21:29:54 +00:00
|
|
|
|
|
|
|
if (auto error = ValidateMemoryScope(_, inst, memory_scope)) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2021-03-16 14:53:37 +00:00
|
|
|
if (auto error = ValidateMemorySemantics(_, inst, 1, memory_scope)) {
|
2018-01-31 21:29:54 +00:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-11-04 21:27:10 +00:00
|
|
|
case spv::Op::OpNamedBarrierInitialize: {
|
|
|
|
if (_.GetIdOpcode(result_type) != spv::Op::OpTypeNamedBarrier) {
|
2018-07-31 22:44:35 +00:00
|
|
|
return _.diag(SPV_ERROR_INVALID_DATA, inst)
|
2018-01-31 21:29:54 +00:00
|
|
|
<< spvOpcodeString(opcode)
|
|
|
|
<< ": expected Result Type to be OpTypeNamedBarrier";
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint32_t subgroup_count_type = _.GetOperandTypeId(inst, 2);
|
|
|
|
if (!_.IsIntScalarType(subgroup_count_type) ||
|
|
|
|
_.GetBitWidth(subgroup_count_type) != 32) {
|
2018-07-31 22:44:35 +00:00
|
|
|
return _.diag(SPV_ERROR_INVALID_DATA, inst)
|
2018-01-31 21:29:54 +00:00
|
|
|
<< spvOpcodeString(opcode)
|
|
|
|
<< ": expected Subgroup Count to be a 32-bit int";
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2022-11-04 21:27:10 +00:00
|
|
|
case spv::Op::OpMemoryNamedBarrier: {
|
2018-01-31 21:29:54 +00:00
|
|
|
const uint32_t named_barrier_type = _.GetOperandTypeId(inst, 0);
|
2022-11-04 21:27:10 +00:00
|
|
|
if (_.GetIdOpcode(named_barrier_type) != spv::Op::OpTypeNamedBarrier) {
|
2018-07-31 22:44:35 +00:00
|
|
|
return _.diag(SPV_ERROR_INVALID_DATA, inst)
|
2018-01-31 21:29:54 +00:00
|
|
|
<< spvOpcodeString(opcode)
|
|
|
|
<< ": expected Named Barrier to be of type OpTypeNamedBarrier";
|
|
|
|
}
|
|
|
|
|
2018-07-10 14:57:52 +00:00
|
|
|
const uint32_t memory_scope = inst->word(2);
|
2018-01-31 21:29:54 +00:00
|
|
|
|
|
|
|
if (auto error = ValidateMemoryScope(_, inst, memory_scope)) {
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
2021-03-16 14:53:37 +00:00
|
|
|
if (auto error = ValidateMemorySemantics(_, inst, 2, memory_scope)) {
|
2018-01-31 21:29:54 +00:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SPV_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2018-07-10 03:18:44 +00:00
|
|
|
} // namespace val
|
2018-07-07 13:38:00 +00:00
|
|
|
} // namespace spvtools
|