2016-05-22 18:11:24 +00:00
|
|
|
// Copyright (c) 2016 Google Inc.
|
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// 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
|
2016-05-22 18:11:24 +00:00
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2016-05-22 18:11:24 +00:00
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// 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.
|
2016-05-22 18:11:24 +00:00
|
|
|
|
|
|
|
// This file defines the language constructs for representing a SPIR-V
|
|
|
|
// module in memory.
|
|
|
|
|
2018-08-03 12:05:33 +00:00
|
|
|
#ifndef SOURCE_OPT_BASIC_BLOCK_H_
|
|
|
|
#define SOURCE_OPT_BASIC_BLOCK_H_
|
2016-05-22 18:11:24 +00:00
|
|
|
|
|
|
|
#include <functional>
|
2018-04-17 14:31:09 +00:00
|
|
|
#include <iterator>
|
2016-08-09 23:20:35 +00:00
|
|
|
#include <memory>
|
SSA rewrite pass.
This pass replaces the load/store elimination passes. It implements the
SSA re-writing algorithm proposed in
Simple and Efficient Construction of Static Single Assignment Form.
Braun M., Buchwald S., Hack S., Leißa R., Mallon C., Zwinkau A. (2013)
In: Jhala R., De Bosschere K. (eds)
Compiler Construction. CC 2013.
Lecture Notes in Computer Science, vol 7791.
Springer, Berlin, Heidelberg
https://link.springer.com/chapter/10.1007/978-3-642-37051-9_6
In contrast to common eager algorithms based on dominance and dominance
frontier information, this algorithm works backwards from load operations.
When a target variable is loaded, it queries the variable's reaching
definition. If the reaching definition is unknown at the current location,
it searches backwards in the CFG, inserting Phi instructions at join points
in the CFG along the way until it finds the desired store instruction.
The algorithm avoids repeated lookups using memoization.
For reducible CFGs, which are a superset of the structured CFGs in SPIRV,
this algorithm is proven to produce minimal SSA. That is, it inserts the
minimal number of Phi instructions required to ensure the SSA property, but
some Phi instructions may be dead
(https://en.wikipedia.org/wiki/Static_single_assignment_form).
2018-02-22 21:18:29 +00:00
|
|
|
#include <ostream>
|
2018-08-03 19:06:09 +00:00
|
|
|
#include <string>
|
2016-05-22 18:11:24 +00:00
|
|
|
#include <utility>
|
2016-08-09 23:20:35 +00:00
|
|
|
#include <vector>
|
2016-05-22 18:11:24 +00:00
|
|
|
|
2018-08-03 19:06:09 +00:00
|
|
|
#include "source/opt/instruction.h"
|
|
|
|
#include "source/opt/instruction_list.h"
|
|
|
|
#include "source/opt/iterator.h"
|
2016-05-22 18:11:24 +00:00
|
|
|
|
|
|
|
namespace spvtools {
|
2018-07-09 15:32:29 +00:00
|
|
|
namespace opt {
|
2016-05-22 18:11:24 +00:00
|
|
|
|
|
|
|
class Function;
|
2017-11-14 19:11:50 +00:00
|
|
|
class IRContext;
|
2016-05-22 18:11:24 +00:00
|
|
|
|
|
|
|
// A SPIR-V basic block.
|
|
|
|
class BasicBlock {
|
|
|
|
public:
|
2017-10-13 18:25:21 +00:00
|
|
|
using iterator = InstructionList::iterator;
|
|
|
|
using const_iterator = InstructionList::const_iterator;
|
2018-04-17 14:31:09 +00:00
|
|
|
using reverse_iterator = std::reverse_iterator<InstructionList::iterator>;
|
|
|
|
using const_reverse_iterator =
|
|
|
|
std::reverse_iterator<InstructionList::const_iterator>;
|
2016-08-12 13:13:04 +00:00
|
|
|
|
|
|
|
// Creates a basic block with the given starting |label|.
|
|
|
|
inline explicit BasicBlock(std::unique_ptr<Instruction> label);
|
2016-05-22 18:11:24 +00:00
|
|
|
|
2017-11-14 19:11:50 +00:00
|
|
|
explicit BasicBlock(const BasicBlock& bb) = delete;
|
|
|
|
|
|
|
|
// Creates a clone of the basic block in the given |context|
|
2017-07-13 00:16:51 +00:00
|
|
|
//
|
|
|
|
// The parent function will default to null and needs to be explicitly set by
|
|
|
|
// the user.
|
2018-09-26 21:36:27 +00:00
|
|
|
//
|
|
|
|
// If the inst-to-block map in |context| is valid, then the new instructions
|
|
|
|
// will be inserted into the map.
|
2017-11-14 19:11:50 +00:00
|
|
|
BasicBlock* Clone(IRContext*) const;
|
2017-07-13 00:16:51 +00:00
|
|
|
|
2016-05-22 18:11:24 +00:00
|
|
|
// Sets the enclosing function for this basic block.
|
|
|
|
void SetParent(Function* function) { function_ = function; }
|
2017-06-07 21:28:53 +00:00
|
|
|
|
2017-11-14 19:11:50 +00:00
|
|
|
// Return the enclosing function
|
|
|
|
inline Function* GetParent() const { return function_; }
|
|
|
|
|
2016-05-22 18:11:24 +00:00
|
|
|
// Appends an instruction to this basic block.
|
2016-08-09 23:20:35 +00:00
|
|
|
inline void AddInstruction(std::unique_ptr<Instruction> i);
|
2017-06-07 21:28:53 +00:00
|
|
|
|
|
|
|
// Appends all of block's instructions (except label) to this block
|
|
|
|
inline void AddInstructions(BasicBlock* bp);
|
|
|
|
|
2018-11-08 18:54:54 +00:00
|
|
|
// The pointer to the label starting this basic block.
|
|
|
|
std::unique_ptr<Instruction>& GetLabel() { return label_; }
|
|
|
|
|
2017-03-26 19:08:41 +00:00
|
|
|
// The label starting this basic block.
|
2017-06-07 21:28:53 +00:00
|
|
|
Instruction* GetLabelInst() { return label_.get(); }
|
2018-01-03 00:54:55 +00:00
|
|
|
const Instruction* GetLabelInst() const { return label_.get(); }
|
2016-05-22 18:11:24 +00:00
|
|
|
|
2017-08-31 15:37:17 +00:00
|
|
|
// Returns the merge instruction in this basic block, if it exists.
|
|
|
|
// Otherwise return null. May be used whenever tail() can be used.
|
|
|
|
const Instruction* GetMergeInst() const;
|
|
|
|
Instruction* GetMergeInst();
|
2017-10-30 21:42:26 +00:00
|
|
|
|
2017-08-31 15:37:17 +00:00
|
|
|
// Returns the OpLoopMerge instruciton in this basic block, if it exists.
|
|
|
|
// Otherwise return null. May be used whenever tail() can be used.
|
|
|
|
const Instruction* GetLoopMergeInst() const;
|
|
|
|
Instruction* GetLoopMergeInst();
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
// Returns the id of the label at the top of this block
|
2017-05-05 02:55:53 +00:00
|
|
|
inline uint32_t id() const { return label_->result_id(); }
|
2016-11-10 17:11:50 +00:00
|
|
|
|
2017-10-13 18:25:21 +00:00
|
|
|
iterator begin() { return insts_.begin(); }
|
|
|
|
iterator end() { return insts_.end(); }
|
2017-11-27 21:21:26 +00:00
|
|
|
const_iterator begin() const { return insts_.cbegin(); }
|
|
|
|
const_iterator end() const { return insts_.cend(); }
|
2017-10-13 18:25:21 +00:00
|
|
|
const_iterator cbegin() const { return insts_.cbegin(); }
|
|
|
|
const_iterator cend() const { return insts_.cend(); }
|
2016-08-12 13:13:04 +00:00
|
|
|
|
2018-04-17 14:31:09 +00:00
|
|
|
reverse_iterator rbegin() { return reverse_iterator(end()); }
|
|
|
|
reverse_iterator rend() { return reverse_iterator(begin()); }
|
|
|
|
const_reverse_iterator rbegin() const {
|
|
|
|
return const_reverse_iterator(cend());
|
|
|
|
}
|
|
|
|
const_reverse_iterator rend() const {
|
|
|
|
return const_reverse_iterator(cbegin());
|
|
|
|
}
|
|
|
|
const_reverse_iterator crbegin() const {
|
|
|
|
return const_reverse_iterator(cend());
|
|
|
|
}
|
|
|
|
const_reverse_iterator crend() const {
|
|
|
|
return const_reverse_iterator(cbegin());
|
|
|
|
}
|
|
|
|
|
2017-08-31 15:37:17 +00:00
|
|
|
// Returns an iterator pointing to the last instruction. This may only
|
|
|
|
// be used if this block has an instruction other than the OpLabel
|
|
|
|
// that defines it.
|
2017-06-07 21:28:53 +00:00
|
|
|
iterator tail() {
|
|
|
|
assert(!insts_.empty());
|
2017-10-13 18:25:21 +00:00
|
|
|
return --end();
|
2017-06-07 21:28:53 +00:00
|
|
|
}
|
2017-10-30 21:42:26 +00:00
|
|
|
|
2017-08-31 15:37:17 +00:00
|
|
|
// Returns a const iterator, but othewrise similar to tail().
|
|
|
|
const_iterator ctail() const {
|
|
|
|
assert(!insts_.empty());
|
2017-10-13 18:25:21 +00:00
|
|
|
return --insts_.cend();
|
2017-08-31 15:37:17 +00:00
|
|
|
}
|
2017-06-07 21:28:53 +00:00
|
|
|
|
2017-11-27 21:21:26 +00:00
|
|
|
// Returns true if the basic block has at least one successor.
|
|
|
|
inline bool hasSuccessor() const { return ctail()->IsBranch(); }
|
|
|
|
|
2016-08-20 13:47:00 +00:00
|
|
|
// Runs the given function |f| on each instruction in this basic block, and
|
|
|
|
// optionally on the debug line instructions that might precede them.
|
|
|
|
inline void ForEachInst(const std::function<void(Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false);
|
|
|
|
inline void ForEachInst(const std::function<void(const Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false) const;
|
2016-05-22 18:11:24 +00:00
|
|
|
|
2018-01-12 20:05:53 +00:00
|
|
|
// Runs the given function |f| on each instruction in this basic block, and
|
|
|
|
// optionally on the debug line instructions that might precede them. If |f|
|
|
|
|
// returns false, iteration is terminated and this function returns false.
|
|
|
|
inline bool WhileEachInst(const std::function<bool(Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false);
|
|
|
|
inline bool WhileEachInst(const std::function<bool(const Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false) const;
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
// Runs the given function |f| on each Phi instruction in this basic block,
|
|
|
|
// and optionally on the debug line instructions that might precede them.
|
|
|
|
inline void ForEachPhiInst(const std::function<void(Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false);
|
|
|
|
|
2018-01-16 16:15:06 +00:00
|
|
|
// Runs the given function |f| on each Phi instruction in this basic block,
|
|
|
|
// and optionally on the debug line instructions that might precede them. If
|
|
|
|
// |f| returns false, iteration is terminated and this function return false.
|
|
|
|
inline bool WhileEachPhiInst(const std::function<bool(Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts = false);
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
// Runs the given function |f| on each label id of each successor block
|
2018-01-04 22:04:03 +00:00
|
|
|
void ForEachSuccessorLabel(
|
|
|
|
const std::function<void(const uint32_t)>& f) const;
|
|
|
|
|
2018-01-26 12:07:10 +00:00
|
|
|
// Runs the given function |f| on each label id of each successor block.
|
|
|
|
// Modifying the pointed value will change the branch taken by the basic
|
|
|
|
// block. It is the caller responsibility to update or invalidate the CFG.
|
|
|
|
void ForEachSuccessorLabel(const std::function<void(uint32_t*)>& f);
|
|
|
|
|
2018-01-04 22:04:03 +00:00
|
|
|
// Returns true if |block| is a direct successor of |this|.
|
2018-07-12 19:14:43 +00:00
|
|
|
bool IsSuccessor(const BasicBlock* block) const;
|
2016-11-10 17:11:50 +00:00
|
|
|
|
2017-08-23 23:05:38 +00:00
|
|
|
// Runs the given function |f| on the merge and continue label, if any
|
2017-10-13 18:25:21 +00:00
|
|
|
void ForMergeAndContinueLabel(const std::function<void(const uint32_t)>& f);
|
2017-08-23 23:05:38 +00:00
|
|
|
|
2017-10-19 19:22:02 +00:00
|
|
|
// Returns true if this basic block has any Phi instructions.
|
|
|
|
bool HasPhiInstructions() {
|
2018-07-12 19:14:43 +00:00
|
|
|
return !WhileEachPhiInst([](Instruction*) { return false; });
|
2017-10-19 19:22:02 +00:00
|
|
|
}
|
|
|
|
|
2017-10-30 21:42:26 +00:00
|
|
|
// Return true if this block is a loop header block.
|
|
|
|
bool IsLoopHeader() const { return GetLoopMergeInst() != nullptr; }
|
|
|
|
|
|
|
|
// Returns the ID of the merge block declared by a merge instruction in this
|
2018-01-08 15:42:02 +00:00
|
|
|
// block, if any. If none, returns zero.
|
2017-10-30 21:42:26 +00:00
|
|
|
uint32_t MergeBlockIdIfAny() const;
|
|
|
|
|
|
|
|
// Returns the ID of the continue block declared by a merge instruction in
|
|
|
|
// this block, if any. If none, returns zero.
|
|
|
|
uint32_t ContinueBlockIdIfAny() const;
|
|
|
|
|
2018-01-04 23:52:38 +00:00
|
|
|
// Returns the terminator instruction. Assumes the terminator exists.
|
|
|
|
Instruction* terminator() { return &*tail(); }
|
2018-01-10 19:23:47 +00:00
|
|
|
const Instruction* terminator() const { return &*ctail(); }
|
2018-01-04 23:52:38 +00:00
|
|
|
|
2017-11-17 13:59:25 +00:00
|
|
|
// Returns true if this basic block exits this function and returns to its
|
|
|
|
// caller.
|
|
|
|
bool IsReturn() const { return ctail()->IsReturn(); }
|
|
|
|
|
2018-01-03 20:25:03 +00:00
|
|
|
// Returns true if this basic block exits this function or aborts execution.
|
|
|
|
bool IsReturnOrAbort() const { return ctail()->IsReturnOrAbort(); }
|
|
|
|
|
2018-02-12 21:42:15 +00:00
|
|
|
// Kill all instructions in this block. Whether or not to kill the label is
|
|
|
|
// indicated by |killLabel|.
|
|
|
|
void KillAllInsts(bool killLabel);
|
|
|
|
|
2018-03-06 16:20:28 +00:00
|
|
|
// Splits this basic block into two. Returns a new basic block with label
|
|
|
|
// |labelId| containing the instructions from |iter| onwards. Instructions
|
2018-09-18 12:52:47 +00:00
|
|
|
// prior to |iter| remain in this basic block. The new block will be added
|
|
|
|
// to the function immediately after the original block.
|
2018-03-06 16:20:28 +00:00
|
|
|
BasicBlock* SplitBasicBlock(IRContext* context, uint32_t label_id,
|
|
|
|
iterator iter);
|
|
|
|
|
SSA rewrite pass.
This pass replaces the load/store elimination passes. It implements the
SSA re-writing algorithm proposed in
Simple and Efficient Construction of Static Single Assignment Form.
Braun M., Buchwald S., Hack S., Leißa R., Mallon C., Zwinkau A. (2013)
In: Jhala R., De Bosschere K. (eds)
Compiler Construction. CC 2013.
Lecture Notes in Computer Science, vol 7791.
Springer, Berlin, Heidelberg
https://link.springer.com/chapter/10.1007/978-3-642-37051-9_6
In contrast to common eager algorithms based on dominance and dominance
frontier information, this algorithm works backwards from load operations.
When a target variable is loaded, it queries the variable's reaching
definition. If the reaching definition is unknown at the current location,
it searches backwards in the CFG, inserting Phi instructions at join points
in the CFG along the way until it finds the desired store instruction.
The algorithm avoids repeated lookups using memoization.
For reducible CFGs, which are a superset of the structured CFGs in SPIRV,
this algorithm is proven to produce minimal SSA. That is, it inserts the
minimal number of Phi instructions required to ensure the SSA property, but
some Phi instructions may be dead
(https://en.wikipedia.org/wiki/Static_single_assignment_form).
2018-02-22 21:18:29 +00:00
|
|
|
// Pretty-prints this basic block into a std::string by printing every
|
|
|
|
// instruction in it.
|
|
|
|
//
|
|
|
|
// |options| are the disassembly options. SPV_BINARY_TO_TEXT_OPTION_NO_HEADER
|
|
|
|
// is always added to |options|.
|
|
|
|
std::string PrettyPrint(uint32_t options = 0u) const;
|
|
|
|
|
2018-09-14 17:57:12 +00:00
|
|
|
// Dump this basic block on stderr. Useful when running interactive
|
|
|
|
// debuggers.
|
|
|
|
void Dump() const;
|
|
|
|
|
2016-05-22 18:11:24 +00:00
|
|
|
private:
|
2016-08-09 23:20:35 +00:00
|
|
|
// The enclosing function.
|
|
|
|
Function* function_;
|
|
|
|
// The label starting this basic block.
|
|
|
|
std::unique_ptr<Instruction> label_;
|
2016-08-20 13:47:00 +00:00
|
|
|
// Instructions inside this basic block, but not the OpLabel.
|
2017-10-13 18:25:21 +00:00
|
|
|
InstructionList insts_;
|
2016-05-22 18:11:24 +00:00
|
|
|
};
|
|
|
|
|
2018-01-10 19:23:47 +00:00
|
|
|
// Pretty-prints |block| to |str|. Returns |str|.
|
|
|
|
std::ostream& operator<<(std::ostream& str, const BasicBlock& block);
|
|
|
|
|
2016-08-12 13:13:04 +00:00
|
|
|
inline BasicBlock::BasicBlock(std::unique_ptr<Instruction> label)
|
|
|
|
: function_(nullptr), label_(std::move(label)) {}
|
|
|
|
|
2016-08-09 23:20:35 +00:00
|
|
|
inline void BasicBlock::AddInstruction(std::unique_ptr<Instruction> i) {
|
2017-10-13 18:25:21 +00:00
|
|
|
insts_.push_back(std::move(i));
|
2016-08-09 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
2017-06-07 21:28:53 +00:00
|
|
|
inline void BasicBlock::AddInstructions(BasicBlock* bp) {
|
|
|
|
auto bEnd = end();
|
2017-10-30 21:42:26 +00:00
|
|
|
(void)bEnd.MoveBefore(&bp->insts_);
|
2017-06-07 21:28:53 +00:00
|
|
|
}
|
|
|
|
|
2018-01-12 20:05:53 +00:00
|
|
|
inline bool BasicBlock::WhileEachInst(
|
|
|
|
const std::function<bool(Instruction*)>& f, bool run_on_debug_line_insts) {
|
|
|
|
if (label_) {
|
|
|
|
if (!label_->WhileEachInst(f, run_on_debug_line_insts)) return false;
|
|
|
|
}
|
2017-11-21 19:47:46 +00:00
|
|
|
if (insts_.empty()) {
|
2018-01-12 20:05:53 +00:00
|
|
|
return true;
|
2017-11-21 19:47:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Instruction* inst = &insts_.front();
|
|
|
|
while (inst != nullptr) {
|
|
|
|
Instruction* next_instruction = inst->NextNode();
|
2018-01-12 20:05:53 +00:00
|
|
|
if (!inst->WhileEachInst(f, run_on_debug_line_insts)) return false;
|
2017-11-21 19:47:46 +00:00
|
|
|
inst = next_instruction;
|
|
|
|
}
|
2018-01-12 20:05:53 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool BasicBlock::WhileEachInst(
|
|
|
|
const std::function<bool(const Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts) const {
|
|
|
|
if (label_) {
|
|
|
|
if (!static_cast<const Instruction*>(label_.get())
|
|
|
|
->WhileEachInst(f, run_on_debug_line_insts))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (const auto& inst : insts_) {
|
|
|
|
if (!static_cast<const Instruction*>(&inst)->WhileEachInst(
|
|
|
|
f, run_on_debug_line_insts))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void BasicBlock::ForEachInst(const std::function<void(Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts) {
|
|
|
|
WhileEachInst(
|
|
|
|
[&f](Instruction* inst) {
|
|
|
|
f(inst);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
run_on_debug_line_insts);
|
2016-05-22 18:11:24 +00:00
|
|
|
}
|
|
|
|
|
2016-08-20 13:47:00 +00:00
|
|
|
inline void BasicBlock::ForEachInst(
|
|
|
|
const std::function<void(const Instruction*)>& f,
|
|
|
|
bool run_on_debug_line_insts) const {
|
2018-01-12 20:05:53 +00:00
|
|
|
WhileEachInst(
|
|
|
|
[&f](const Instruction* inst) {
|
|
|
|
f(inst);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
run_on_debug_line_insts);
|
2016-05-22 18:11:24 +00:00
|
|
|
}
|
|
|
|
|
2018-01-16 16:15:06 +00:00
|
|
|
inline bool BasicBlock::WhileEachPhiInst(
|
|
|
|
const std::function<bool(Instruction*)>& f, bool run_on_debug_line_insts) {
|
2018-03-06 16:20:28 +00:00
|
|
|
if (insts_.empty()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Instruction* inst = &insts_.front();
|
|
|
|
while (inst != nullptr) {
|
|
|
|
Instruction* next_instruction = inst->NextNode();
|
|
|
|
if (inst->opcode() != SpvOpPhi) break;
|
|
|
|
if (!inst->WhileEachInst(f, run_on_debug_line_insts)) return false;
|
|
|
|
inst = next_instruction;
|
2016-11-10 17:11:50 +00:00
|
|
|
}
|
2018-01-16 16:15:06 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void BasicBlock::ForEachPhiInst(
|
|
|
|
const std::function<void(Instruction*)>& f, bool run_on_debug_line_insts) {
|
|
|
|
WhileEachPhiInst(
|
|
|
|
[&f](Instruction* inst) {
|
|
|
|
f(inst);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
run_on_debug_line_insts);
|
2016-11-10 17:11:50 +00:00
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
} // namespace opt
|
2016-05-22 18:11:24 +00:00
|
|
|
} // namespace spvtools
|
|
|
|
|
2018-08-03 12:05:33 +00:00
|
|
|
#endif // SOURCE_OPT_BASIC_BLOCK_H_
|