// 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.
#include "source/val/validate.h"
#include "source/val/function.h"
#include "source/val/validation_state.h"
namespace spvtools {
namespace val {
spv_result_t ValidateExecutionLimitations(ValidationState_t& _,
const Instruction* inst) {
if (inst->opcode() != SpvOpFunction) {
return SPV_SUCCESS;
}
const auto func = _.function(inst->id());
if (!func) {
return _.diag(SPV_ERROR_INTERNAL, inst)
<< "Internal error: missing function id " << inst->id() << ".";
}
for (uint32_t entry_id : _.FunctionEntryPoints(inst->id())) {
const auto* models = _.GetExecutionModels(entry_id);
if (models) {
if (models->empty()) {
return _.diag(SPV_ERROR_INTERNAL, inst)
<< "Internal error: empty execution models for function id "
<< entry_id << ".";
}
for (const auto model : *models) {
std::string reason;
if (!func->IsCompatibleWithExecutionModel(model, &reason)) {
return _.diag(SPV_ERROR_INVALID_ID, inst)
<< "OpEntryPoint Entry Point <id> '" << _.getIdName(entry_id)
<< "'s callgraph contains function <id> "
<< _.getIdName(inst->id())
<< ", which cannot be used with the current execution "
"model:\n"
<< reason;
}
}
}
}
return SPV_SUCCESS;
}
} // namespace val
} // namespace spvtools