mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-07-02 19:25:56 +00:00
This CL adds Assignment and CompoundAssignment statements to the IR builder. Bug: tint:1718 Change-Id: I3037da0115c7f4fe68941565b7e48866d421bbbf Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/129201 Reviewed-by: Ben Clayton <bclayton@google.com> Kokoro: Kokoro <noreply+kokoro@google.com> Commit-Queue: Dan Sinclair <dsinclair@chromium.org>
235 lines
7.6 KiB
C++
235 lines
7.6 KiB
C++
// Copyright 2022 The Tint Authors.
|
|
//
|
|
// 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 "src/tint/ir/builder.h"
|
|
|
|
#include <utility>
|
|
|
|
#include "src/tint/ir/builder_impl.h"
|
|
|
|
namespace tint::ir {
|
|
|
|
Builder::Builder() {}
|
|
|
|
Builder::Builder(Module&& mod) : ir(std::move(mod)) {}
|
|
|
|
Builder::~Builder() = default;
|
|
|
|
Block* Builder::CreateBlock() {
|
|
return ir.flow_nodes.Create<Block>();
|
|
}
|
|
|
|
Terminator* Builder::CreateTerminator() {
|
|
return ir.flow_nodes.Create<Terminator>();
|
|
}
|
|
|
|
Function* Builder::CreateFunction() {
|
|
auto* ir_func = ir.flow_nodes.Create<Function>();
|
|
ir_func->start_target = CreateBlock();
|
|
ir_func->end_target = CreateTerminator();
|
|
|
|
// Function is always branching into the start target
|
|
ir_func->start_target->inbound_branches.Push(ir_func);
|
|
|
|
return ir_func;
|
|
}
|
|
|
|
If* Builder::CreateIf() {
|
|
auto* ir_if = ir.flow_nodes.Create<If>();
|
|
ir_if->true_.target = CreateBlock();
|
|
ir_if->false_.target = CreateBlock();
|
|
ir_if->merge.target = CreateBlock();
|
|
|
|
// An if always branches to both the true and false block.
|
|
ir_if->true_.target->inbound_branches.Push(ir_if);
|
|
ir_if->false_.target->inbound_branches.Push(ir_if);
|
|
|
|
return ir_if;
|
|
}
|
|
|
|
Loop* Builder::CreateLoop() {
|
|
auto* ir_loop = ir.flow_nodes.Create<Loop>();
|
|
ir_loop->start.target = CreateBlock();
|
|
ir_loop->continuing.target = CreateBlock();
|
|
ir_loop->merge.target = CreateBlock();
|
|
|
|
// A loop always branches to the start block.
|
|
ir_loop->start.target->inbound_branches.Push(ir_loop);
|
|
|
|
return ir_loop;
|
|
}
|
|
|
|
Switch* Builder::CreateSwitch() {
|
|
auto* ir_switch = ir.flow_nodes.Create<Switch>();
|
|
ir_switch->merge.target = CreateBlock();
|
|
return ir_switch;
|
|
}
|
|
|
|
Block* Builder::CreateCase(Switch* s, utils::VectorRef<Switch::CaseSelector> selectors) {
|
|
s->cases.Push(Switch::Case{selectors, {CreateBlock(), utils::Empty}});
|
|
|
|
Block* b = s->cases.Back().start.target->As<Block>();
|
|
// Switch branches into the case block
|
|
b->inbound_branches.Push(s);
|
|
return b;
|
|
}
|
|
|
|
void Builder::Branch(Block* from, FlowNode* to, utils::VectorRef<Value*> args) {
|
|
TINT_ASSERT(IR, from);
|
|
TINT_ASSERT(IR, to);
|
|
from->branch.target = to;
|
|
from->branch.args = args;
|
|
to->inbound_branches.Push(from);
|
|
}
|
|
|
|
Runtime::Id Builder::AllocateRuntimeId() {
|
|
return next_runtime_id++;
|
|
}
|
|
|
|
Binary* Builder::CreateBinary(Binary::Kind kind, const type::Type* type, Value* lhs, Value* rhs) {
|
|
return ir.instructions.Create<ir::Binary>(kind, Runtime(type), lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::And(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kAnd, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Or(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kOr, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Xor(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kXor, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::LogicalAnd(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kLogicalAnd, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::LogicalOr(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kLogicalOr, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Equal(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kEqual, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::NotEqual(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kNotEqual, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::LessThan(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kLessThan, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::GreaterThan(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kGreaterThan, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::LessThanEqual(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kLessThanEqual, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::GreaterThanEqual(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kGreaterThanEqual, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::ShiftLeft(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kShiftLeft, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::ShiftRight(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kShiftRight, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Add(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kAdd, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Subtract(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kSubtract, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Multiply(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kMultiply, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Divide(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kDivide, type, lhs, rhs);
|
|
}
|
|
|
|
Binary* Builder::Modulo(const type::Type* type, Value* lhs, Value* rhs) {
|
|
return CreateBinary(Binary::Kind::kModulo, type, lhs, rhs);
|
|
}
|
|
|
|
Unary* Builder::CreateUnary(Unary::Kind kind, const type::Type* type, Value* val) {
|
|
return ir.instructions.Create<ir::Unary>(kind, Runtime(type), val);
|
|
}
|
|
|
|
Unary* Builder::AddressOf(const type::Type* type, Value* val) {
|
|
return CreateUnary(Unary::Kind::kAddressOf, type, val);
|
|
}
|
|
|
|
Unary* Builder::Complement(const type::Type* type, Value* val) {
|
|
return CreateUnary(Unary::Kind::kComplement, type, val);
|
|
}
|
|
|
|
Unary* Builder::Indirection(const type::Type* type, Value* val) {
|
|
return CreateUnary(Unary::Kind::kIndirection, type, val);
|
|
}
|
|
|
|
Unary* Builder::Negation(const type::Type* type, Value* val) {
|
|
return CreateUnary(Unary::Kind::kNegation, type, val);
|
|
}
|
|
|
|
Unary* Builder::Not(const type::Type* type, Value* val) {
|
|
return CreateUnary(Unary::Kind::kNot, type, val);
|
|
}
|
|
|
|
ir::Bitcast* Builder::Bitcast(const type::Type* type, Value* val) {
|
|
return ir.instructions.Create<ir::Bitcast>(Runtime(type), val);
|
|
}
|
|
|
|
ir::Discard* Builder::Discard() {
|
|
return ir.instructions.Create<ir::Discard>(Runtime(ir.types.Get<type::Void>()));
|
|
}
|
|
|
|
ir::UserCall* Builder::UserCall(const type::Type* type,
|
|
Symbol name,
|
|
utils::VectorRef<Value*> args) {
|
|
return ir.instructions.Create<ir::UserCall>(Runtime(type), name, std::move(args));
|
|
}
|
|
|
|
ir::Convert* Builder::Convert(const type::Type* to,
|
|
const type::Type* from,
|
|
utils::VectorRef<Value*> args) {
|
|
return ir.instructions.Create<ir::Convert>(Runtime(to), from, std::move(args));
|
|
}
|
|
|
|
ir::Construct* Builder::Construct(const type::Type* to, utils::VectorRef<Value*> args) {
|
|
return ir.instructions.Create<ir::Construct>(Runtime(to), std::move(args));
|
|
}
|
|
|
|
ir::Builtin* Builder::Builtin(const type::Type* type,
|
|
builtin::Function func,
|
|
utils::VectorRef<Value*> args) {
|
|
return ir.instructions.Create<ir::Builtin>(Runtime(type), func, args);
|
|
}
|
|
|
|
ir::Store* Builder::Store(Value* to, Value* from) {
|
|
return ir.instructions.Create<ir::Store>(to, from);
|
|
}
|
|
|
|
} // namespace tint::ir
|