mirror of
https://github.com/KhronosGroup/SPIRV-Tools
synced 2025-01-11 17:10:06 +00:00
Remove std::deque in favour of std::vector. (#1755)
This CL removes the two deque's from ValidationState and converts them into std::vectors. In order to maintain the stability of instructions we walk over the binary and counter the instructions and functions in the ValidationState constructor and reserve the required number of items in the module_functions_ and ordered_instructions_ vectors. Issue #1176.
This commit is contained in:
parent
fae987b470
commit
a504656dad
@ -49,6 +49,7 @@
|
||||
#include "markv_model.h"
|
||||
#include "opcode.h"
|
||||
#include "operand.h"
|
||||
#include "source/assembly_grammar.h"
|
||||
#include "spirv-tools/libspirv.h"
|
||||
#include "spirv_endian.h"
|
||||
#include "spirv_validator_options.h"
|
||||
@ -58,7 +59,6 @@
|
||||
#include "util/parse_number.h"
|
||||
#include "val/instruction.h"
|
||||
#include "val/validate.h"
|
||||
#include "val/validation_state.h"
|
||||
|
||||
namespace spvtools {
|
||||
namespace comp {
|
||||
|
@ -15,7 +15,6 @@
|
||||
#ifndef LIBSPIRV_OPERAND_H_
|
||||
#define LIBSPIRV_OPERAND_H_
|
||||
|
||||
#include <deque>
|
||||
#include <functional>
|
||||
|
||||
#include "spirv-tools/libspirv.h"
|
||||
|
@ -220,9 +220,11 @@ class StatsAggregator {
|
||||
void ProcessConstant() {
|
||||
const val::Instruction& inst = GetCurrentInstruction();
|
||||
if (inst.opcode() != SpvOpConstant) return;
|
||||
|
||||
const uint32_t type_id = inst.GetOperandAs<uint32_t>(0);
|
||||
const auto type_decl_it = vstate_->all_definitions().find(type_id);
|
||||
assert(type_decl_it != vstate_->all_definitions().end());
|
||||
|
||||
const val::Instruction& type_decl_inst = *type_decl_it->second;
|
||||
const SpvOp type_op = type_decl_inst.opcode();
|
||||
if (type_op == SpvOpTypeInt) {
|
||||
|
@ -238,16 +238,16 @@ spv_result_t ValidateBinaryUsingContextAndValidationState(
|
||||
}
|
||||
|
||||
// Look for OpExtension instructions and register extensions.
|
||||
// Diagnostics if any will be produced in the next pass (ProcessInstruction).
|
||||
spvBinaryParse(&context, vstate, words, num_words,
|
||||
/* parsed_header = */ nullptr, ProcessExtensions,
|
||||
/* diagnostic = */ nullptr);
|
||||
|
||||
// NOTE: Parse the module and perform inline validation checks. These
|
||||
// checks do not require the the knowledge of the whole module.
|
||||
// Parse the module and perform inline validation checks. These checks do
|
||||
// not require the the knowledge of the whole module.
|
||||
if (auto error = spvBinaryParse(&context, vstate, words, num_words, setHeader,
|
||||
ProcessInstruction, pDiagnostic))
|
||||
ProcessInstruction, pDiagnostic)) {
|
||||
return error;
|
||||
}
|
||||
|
||||
if (!vstate->has_memory_model_specified())
|
||||
return vstate->diag(SPV_ERROR_INVALID_LAYOUT, nullptr)
|
||||
|
@ -18,18 +18,12 @@
|
||||
#include <stack>
|
||||
|
||||
#include "opcode.h"
|
||||
#include "spirv-tools/libspirv.h"
|
||||
#include "spirv_target_env.h"
|
||||
#include "val/basic_block.h"
|
||||
#include "val/construct.h"
|
||||
#include "val/function.h"
|
||||
|
||||
using std::deque;
|
||||
using std::make_pair;
|
||||
using std::pair;
|
||||
using std::string;
|
||||
using std::unordered_map;
|
||||
using std::vector;
|
||||
|
||||
namespace spvtools {
|
||||
namespace val {
|
||||
namespace {
|
||||
@ -141,6 +135,16 @@ bool IsInstructionInLayoutSection(ModuleLayoutSection layout, SpvOp op) {
|
||||
return out;
|
||||
}
|
||||
|
||||
// Counts the number of instructions and functions in the file.
|
||||
spv_result_t CountInstructions(void* user_data,
|
||||
const spv_parsed_instruction_t* inst) {
|
||||
ValidationState_t& _ = *(reinterpret_cast<ValidationState_t*>(user_data));
|
||||
if (inst->opcode == SpvOpFunction) _.increment_total_functions();
|
||||
_.increment_total_instructions();
|
||||
|
||||
return SPV_SUCCESS;
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
ValidationState_t::ValidationState_t(const spv_const_context ctx,
|
||||
@ -187,6 +191,21 @@ ValidationState_t::ValidationState_t(const spv_const_context ctx,
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
// Only attempt to count if we have words, otherwise let the other validation
|
||||
// fail and generate an error.
|
||||
if (num_words > 0) {
|
||||
// Count the number of instructions in the binary.
|
||||
spvBinaryParse(ctx, this, words, num_words,
|
||||
/* parsed_header = */ nullptr, CountInstructions,
|
||||
/* diagnostic = */ nullptr);
|
||||
preallocateStorage();
|
||||
}
|
||||
}
|
||||
|
||||
void ValidationState_t::preallocateStorage() {
|
||||
ordered_instructions_.reserve(total_instructions_);
|
||||
module_functions_.reserve(total_functions_);
|
||||
}
|
||||
|
||||
spv_result_t ValidationState_t::ForwardDeclareId(uint32_t id) {
|
||||
@ -208,11 +227,11 @@ bool ValidationState_t::IsForwardPointer(uint32_t id) const {
|
||||
return (forward_pointer_ids_.find(id) != forward_pointer_ids_.end());
|
||||
}
|
||||
|
||||
void ValidationState_t::AssignNameToId(uint32_t id, string name) {
|
||||
void ValidationState_t::AssignNameToId(uint32_t id, std::string name) {
|
||||
operand_names_[id] = name;
|
||||
}
|
||||
|
||||
string ValidationState_t::getIdName(uint32_t id) const {
|
||||
std::string ValidationState_t::getIdName(uint32_t id) const {
|
||||
std::stringstream out;
|
||||
out << id;
|
||||
if (operand_names_.find(id) != end(operand_names_)) {
|
||||
@ -221,7 +240,7 @@ string ValidationState_t::getIdName(uint32_t id) const {
|
||||
return out.str();
|
||||
}
|
||||
|
||||
string ValidationState_t::getIdOrName(uint32_t id) const {
|
||||
std::string ValidationState_t::getIdOrName(uint32_t id) const {
|
||||
std::stringstream out;
|
||||
if (operand_names_.find(id) != end(operand_names_)) {
|
||||
out << operand_names_.at(id);
|
||||
@ -235,9 +254,9 @@ size_t ValidationState_t::unresolved_forward_id_count() const {
|
||||
return unresolved_forward_ids_.size();
|
||||
}
|
||||
|
||||
vector<uint32_t> ValidationState_t::UnresolvedForwardIds() const {
|
||||
vector<uint32_t> out(begin(unresolved_forward_ids_),
|
||||
end(unresolved_forward_ids_));
|
||||
std::vector<uint32_t> ValidationState_t::UnresolvedForwardIds() const {
|
||||
std::vector<uint32_t> out(std::begin(unresolved_forward_ids_),
|
||||
std::end(unresolved_forward_ids_));
|
||||
return out;
|
||||
}
|
||||
|
||||
@ -300,7 +319,9 @@ DiagnosticStream ValidationState_t::diag(spv_result_t error_code,
|
||||
error_code);
|
||||
}
|
||||
|
||||
deque<Function>& ValidationState_t::functions() { return module_functions_; }
|
||||
std::vector<Function>& ValidationState_t::functions() {
|
||||
return module_functions_;
|
||||
}
|
||||
|
||||
Function& ValidationState_t::current_function() {
|
||||
assert(in_function_body());
|
||||
@ -497,7 +518,7 @@ void ValidationState_t::RegisterDebugInstruction(const Instruction* inst) {
|
||||
}
|
||||
|
||||
void ValidationState_t::RegisterInstruction(Instruction* inst) {
|
||||
if (inst->id()) all_definitions_.insert(make_pair(inst->id(), inst));
|
||||
if (inst->id()) all_definitions_.insert(std::make_pair(inst->id(), inst));
|
||||
|
||||
// If the instruction is using an OpTypeSampledImage as an operand, it should
|
||||
// be recorded. The validator will ensure that all usages of an
|
||||
|
@ -15,7 +15,6 @@
|
||||
#ifndef LIBSPIRV_VAL_VALIDATIONSTATE_H_
|
||||
#define LIBSPIRV_VAL_VALIDATIONSTATE_H_
|
||||
|
||||
#include <deque>
|
||||
#include <set>
|
||||
#include <string>
|
||||
#include <tuple>
|
||||
@ -144,6 +143,17 @@ class ValidationState_t {
|
||||
/// Increments the instruction count. Used for diagnostic
|
||||
int increment_instruction_count();
|
||||
|
||||
/// Increments the total number of instructions in the file.
|
||||
void increment_total_instructions() { total_instructions_++; }
|
||||
|
||||
/// Increments the total number of functions in the file.
|
||||
void increment_total_functions() { total_functions_++; }
|
||||
|
||||
/// Allocates internal storage. Note, calling this will invalidate any
|
||||
/// pointers to |ordered_instructions_| or |module_functions_| and, hence,
|
||||
/// should only be called at the beginning of validation.
|
||||
void preallocateStorage();
|
||||
|
||||
/// Returns the current layout section which is being processed
|
||||
ModuleLayoutSection current_layout_section() const;
|
||||
|
||||
@ -157,7 +167,7 @@ class ValidationState_t {
|
||||
DiagnosticStream diag(spv_result_t error_code, const Instruction* inst) const;
|
||||
|
||||
/// Returns the function states
|
||||
std::deque<Function>& functions();
|
||||
std::vector<Function>& functions();
|
||||
|
||||
/// Returns the function states
|
||||
Function& current_function();
|
||||
@ -355,8 +365,8 @@ class ValidationState_t {
|
||||
/// nullptr
|
||||
Instruction* FindDef(uint32_t id);
|
||||
|
||||
/// Returns a deque of instructions in the order they appear in the binary
|
||||
const std::deque<Instruction>& ordered_instructions() const {
|
||||
/// Returns the instructions in the order they appear in the binary
|
||||
const std::vector<Instruction>& ordered_instructions() const {
|
||||
return ordered_instructions_;
|
||||
}
|
||||
|
||||
@ -520,6 +530,11 @@ class ValidationState_t {
|
||||
const uint32_t* words_;
|
||||
const size_t num_words_;
|
||||
|
||||
/// The total number of instructions in the binary.
|
||||
size_t total_instructions_ = 0;
|
||||
/// The total number of functions in the binary.
|
||||
size_t total_functions_ = 0;
|
||||
|
||||
/// Tracks the number of instructions evaluated by the validator
|
||||
int instruction_counter_;
|
||||
|
||||
@ -542,7 +557,7 @@ class ValidationState_t {
|
||||
/// A list of functions in the module.
|
||||
/// Pointers to objects in this container are guaranteed to be stable and
|
||||
/// valid until the end of lifetime of the validation state.
|
||||
std::deque<Function> module_functions_;
|
||||
std::vector<Function> module_functions_;
|
||||
|
||||
/// Capabilities declared in the module
|
||||
CapabilitySet module_capabilities_;
|
||||
@ -551,9 +566,7 @@ class ValidationState_t {
|
||||
ExtensionSet module_extensions_;
|
||||
|
||||
/// List of all instructions in the order they appear in the binary
|
||||
/// Pointers to objects in this container are guaranteed to be stable and
|
||||
/// valid until the end of lifetime of the validation state.
|
||||
std::deque<Instruction> ordered_instructions_;
|
||||
std::vector<Instruction> ordered_instructions_;
|
||||
|
||||
/// Instructions that can be referenced by Ids
|
||||
std::unordered_map<uint32_t, Instruction*> all_definitions_;
|
||||
|
Loading…
Reference in New Issue
Block a user