blob: bafa3ed8cb0bc5f0c6a2077d68aea29e778d54a2 [file] [log] [blame]
// Copyright 2023 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#ifndef SRC_TINT_LANG_CORE_IR_OPERAND_INSTRUCTION_H_
#define SRC_TINT_LANG_CORE_IR_OPERAND_INSTRUCTION_H_
#include <utility>
#include "src/tint/lang/core/ir/instruction.h"
#include "src/tint/lang/core/ir/instruction_result.h"
#include "src/tint/utils/ice/ice.h"
namespace tint::core::ir {
/// An instruction in the IR that expects one or more operands.
/// @tparam N the number of operands before spilling to the heap
/// @tparam R the number of result values before spilling to the heap
template <unsigned N, unsigned R>
class OperandInstruction : public Castable<OperandInstruction<N, R>, Instruction> {
public:
/// Constructor
/// @param id the instruction id
explicit OperandInstruction(Instruction::Id id) : OperandInstruction::Base(id) {}
/// Destructor
~OperandInstruction() override = default;
/// @copydoc tint::core::ir::Value::Destroy
void Destroy() override {
ClearOperands();
Instruction::Destroy();
}
/// Set an operand at a given index.
/// @param index the operand index
/// @param value the value to use
void SetOperand(size_t index, ir::Value* value) override {
TINT_ASSERT(index < operands_.Length());
if (operands_[index]) {
operands_[index]->RemoveUsage({this, index});
}
operands_[index] = value;
if (value) {
value->AddUsage({this, index});
}
}
/// Replaces the operands of the instruction
/// @param operands the new operands of the instruction
void SetOperands(VectorRef<ir::Value*> operands) override {
ClearOperands();
operands_ = std::move(operands);
for (size_t i = 0; i < operands_.Length(); i++) {
if (operands_[i]) {
operands_[i]->AddUsage({this, static_cast<uint32_t>(i)});
}
}
}
/// Removes all operands from the instruction
void ClearOperands() {
for (uint32_t i = 0; i < operands_.Length(); i++) {
if (!operands_[i]) {
continue;
}
operands_[i]->RemoveUsage({this, i});
}
operands_.Clear();
}
/// Replaces the results of the instruction
/// @param results the new results of the instruction
void SetResults(VectorRef<ir::InstructionResult*> results) override {
ClearResults();
results_ = std::move(results);
for (auto* result : results_) {
if (result) {
result->SetInstruction(this);
}
}
}
/// Sets the results of the instruction
/// @param values the new result values
template <typename... ARGS,
typename = std::enable_if_t<!tint::IsVectorLike<
tint::traits::Decay<tint::traits::NthTypeOf<0, ARGS..., void>>>>>
void SetResults(ARGS&&... values) {
SetResults(Vector{std::forward<ARGS>(values)...});
}
/// Removes all results from the instruction.
void ClearResults() {
for (auto* result : results_) {
if (result && result->Instruction() == this) {
result->SetInstruction(nullptr);
}
}
results_.Clear();
}
/// @returns the operands of the instruction
VectorRef<ir::Value*> Operands() override { return operands_; }
/// @returns the operands of the instruction
VectorRef<const ir::Value*> Operands() const override { return operands_; }
/// @returns the result values for this instruction
VectorRef<InstructionResult*> Results() override { return results_; }
/// @returns the result values for this instruction
VectorRef<const InstructionResult*> Results() const override { return results_; }
/// @param idx the index of the result
/// @returns the result with index @p idx, or `nullptr` if there are no results or the index is
/// out of bounds.
InstructionResult* Result(size_t idx) {
return idx < results_.Length() ? results_[idx] : nullptr;
}
/// @param idx the index of the result
/// @returns the result with index @p idx, or `nullptr` if there are no results or the index is
/// out of bounds.
const InstructionResult* Result(size_t idx) const {
return idx < results_.Length() ? results_[idx] : nullptr;
}
protected:
/// Append a new operand to the operand list for this instruction.
/// @param idx the index the operand should be at
/// @param value the operand value to append
void AddOperand(size_t idx, ir::Value* value) {
TINT_ASSERT(idx == operands_.Length());
if (value) {
value->AddUsage({this, static_cast<uint32_t>(operands_.Length())});
}
operands_.Push(value);
}
/// Append a list of operands to the operand list for this instruction.
/// @param start_idx the index from which the values should start
/// @param values the operand values to append
void AddOperands(size_t start_idx, VectorRef<ir::Value*> values) {
size_t idx = start_idx;
for (auto* val : values) {
AddOperand(idx, val);
idx += 1;
}
}
/// Appends a result value to the instruction
/// @param value the value to append
void AddResult(InstructionResult* value) {
if (value) {
value->SetInstruction(this);
}
results_.Push(value);
}
/// The operands to this instruction.
Vector<ir::Value*, N> operands_;
/// The results of this instruction.
Vector<ir::InstructionResult*, R> results_;
/// The default number of operands
static constexpr size_t kDefaultNumOperands = N;
};
} // namespace tint::core::ir
#endif // SRC_TINT_LANG_CORE_IR_OPERAND_INSTRUCTION_H_