/*! * Copyright (c) 2018 by Contributors * \file codegen_spirv.cc * \brief Generate SPIRV block */ #if TVM_VULKAN_RUNTIME #include <tvm/ir.h> #include <tvm/ir_pass.h> #include "../codegen_common.h" #include "./codegen_spirv.h" namespace tvm { namespace codegen { std::vector<uint32_t> CodeGenSPIRV::BuildFunction(const LoweredFunc& f) { this->InitFuncState(); CHECK(f->is_restricted) << "SPIRV only takes restricted memory model"; std::vector<Var> pod_args; uint32_t num_buffer = 0; for (Var arg : f->args) { Type t = arg.type(); if (t.is_handle()) { auto it = f->handle_data_type.find(arg); if (it != f->handle_data_type.end()) { Type value_type = (*it).second.type(); spirv::Value arg_value = builder_->BufferArgument( builder_->GetSType(value_type), 0, num_buffer); storage_info_[arg.get()].UpdateContentType(value_type); var_map_[arg.get()] = arg_value; } else { LOG(FATAL) << "require all handles to be typed"; } ++num_buffer; } else { pod_args.push_back(arg); } } spirv::Value func_ptr = builder_->DeclareKenrelFunction(f->name); builder_->StartFunction(func_ptr); // All the POD arguments are passed in through PushConstant if (pod_args.size() != 0) { std::vector<spirv::SType> value_types; for (size_t i = 0; i < pod_args.size(); ++i) { value_types.push_back(builder_->GetSType(pod_args[i].type())); } spirv::Value ptr = builder_->DeclarePushConstant(value_types); for (size_t i = 0; i < pod_args.size(); ++i) { spirv::Value value = builder_->GetPushConstant( ptr, value_types[i], static_cast<uint32_t>(i)); var_map_[pod_args[i].get()] = value; } } this->VisitStmt(f->body); builder_->SetLocalSize(func_ptr, workgroup_size_); builder_->MakeInst(spv::OpReturn); builder_->MakeInst(spv::OpFunctionEnd); return builder_->Finalize(); } void CodeGenSPIRV::InitFuncState() { std::fill(workgroup_size_, workgroup_size_ + 3, 1); var_map_.clear(); storage_info_.clear(); align_map_.clear(); builder_.reset(new spirv::IRBuilder()); builder_->InitHeader(); } spirv::Value CodeGenSPIRV::GetThreadIndex( const IterVar& iv, const Expr& extent) { runtime::ThreadScope ts = runtime::ThreadScope::make(iv->thread_tag); spirv::Value v; if (ts.rank == 1) { v = builder_->GetLocalID(ts.dim_index); int size = 0; CHECK(arith::GetConstInt(extent, &size)) << "SPIRV only allows constant thread group size " << " get " << extent; CHECK_LT(ts.dim_index, 3); workgroup_size_[ts.dim_index] = static_cast<uint32_t>(size); } else { v = builder_->GetWorkgroupID(ts.dim_index); } return builder_->Cast(builder_->GetSType(iv->var.type()), v); } spirv::Value CodeGenSPIRV::CreateStorageSync(const Call* op) { const std::string& sync = op->args[0].as<StringImm>()->value; spirv::Value value; if (sync == "warp") { return value; } else if (sync == "shared") { builder_->MakeInst( spv::OpControlBarrier, spv::ScopeWorkgroup, spv::ScopeWorkgroup, spv::MemorySemanticsSequentiallyConsistentMask | spv::MemorySemanticsWorkgroupMemoryMask); } else { LOG(FATAL) << "Do not support sync " << sync; } return value; } spirv::Value CodeGenSPIRV::VisitExpr_(const Variable* op) { auto it = var_map_.find(op); CHECK(it != var_map_.end()) << "cannot find variable " << op->name_hint; return it->second; } spirv::Value CodeGenSPIRV::VisitExpr_(const IntImm* op) { return builder_->IntImm(builder_->GetSType(op->type), op->value); } spirv::Value CodeGenSPIRV::VisitExpr_(const UIntImm* op) { return builder_->UIntImm(builder_->GetSType(op->type), op->value); } spirv::Value CodeGenSPIRV::VisitExpr_(const FloatImm* op) { return builder_->FloatImm(builder_->GetSType(op->type), op->value); } spirv::Value CodeGenSPIRV::VisitExpr_(const StringImm* op) { LOG(FATAL) << "StringImm is not supported in Device code"; return spirv::Value(); } spirv::Value CodeGenSPIRV::VisitExpr_(const Cast* op) { return builder_->Cast(builder_->GetSType(op->type), MakeValue(op->value)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Add* op) { return builder_->Add(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Sub* op) { return builder_->Sub(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Mul* op) { return builder_->Mul(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Div* op) { return builder_->Div(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Mod* op) { return builder_->Mod(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Min* op) { spirv::Value a = MakeValue(op->a); spirv::Value b = MakeValue(op->b); return builder_->Select(builder_->LT(a, b), a, b); } spirv::Value CodeGenSPIRV::VisitExpr_(const Max* op) { spirv::Value a = MakeValue(op->a); spirv::Value b = MakeValue(op->b); return builder_->Select(builder_->GT(a, b), a, b); } spirv::Value CodeGenSPIRV::VisitExpr_(const LT* op) { return builder_->LT(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const LE* op) { return builder_->LE(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const GT* op) { return builder_->GT(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const GE* op) { return builder_->GE(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const EQ* op) { return builder_->EQ(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const NE* op) { return builder_->NE(MakeValue(op->a), MakeValue(op->b)); } spirv::Value CodeGenSPIRV::VisitExpr_(const And* op) { spirv::Value a = MakeValue(op->a); spirv::Value b = MakeValue(op->b); return builder_->MakeValue(spv::OpLogicalAnd, a.stype, a, b); } spirv::Value CodeGenSPIRV::VisitExpr_(const Or* op) { spirv::Value a = MakeValue(op->a); spirv::Value b = MakeValue(op->b); return builder_->MakeValue(spv::OpLogicalOr, a.stype, a, b); } spirv::Value CodeGenSPIRV::VisitExpr_(const Not* op) { spirv::Value a = MakeValue(op->a); return builder_->MakeValue(spv::OpLogicalNot, a.stype, a); } spirv::Value CodeGenSPIRV::VisitExpr_(const Select* op) { return builder_->Select(MakeValue(op->condition), MakeValue(op->true_value), MakeValue(op->false_value)); } spirv::Value CodeGenSPIRV::VisitExpr_(const Let* op) { CHECK(!var_map_.count(op->var.get())); var_map_[op->var.get()] = MakeValue(op->value); align_map_[op->var.get()] = EvalModular(op->value, align_map_); return MakeValue(op->body); } spirv::Value CodeGenSPIRV::VisitExpr_(const Call* op) { if (op->is_intrinsic("spirv_glsl450")) { CHECK_GE(op->args.size(), 2U); uint32_t inst_id = op->args[0].as<UIntImm>()->value; std::vector<spirv::Value> values; for (size_t i = 1; i < op->args.size(); ++i) { values.push_back(MakeValue(op->args[i])); } return builder_->CallGLSL450( builder_->GetSType(op->type), inst_id, values); } else if (op->is_intrinsic(Call::bitwise_and)) { CHECK_EQ(op->args.size(), 2U); spirv::Value a = MakeValue(op->args[0]); spirv::Value b = MakeValue(op->args[1]); return builder_->MakeValue(spv::OpBitwiseAnd, a.stype, a, b); } else if (op->is_intrinsic(Call::bitwise_xor)) { CHECK_EQ(op->args.size(), 2U); spirv::Value a = MakeValue(op->args[0]); spirv::Value b = MakeValue(op->args[1]); return builder_->MakeValue(spv::OpBitwiseXor, a.stype, a, b); } else if (op->is_intrinsic(Call::bitwise_or)) { CHECK_EQ(op->args.size(), 2U); spirv::Value a = MakeValue(op->args[0]); spirv::Value b = MakeValue(op->args[1]); return builder_->MakeValue(spv::OpBitwiseOr, a.stype, a, b); } else if (op->is_intrinsic(Call::bitwise_not)) { CHECK_EQ(op->args.size(), 1U); spirv::Value a = MakeValue(op->args[0]); return builder_->MakeValue(spv::OpNot, a.stype, a); } else if (op->is_intrinsic(Call::shift_left)) { CHECK_EQ(op->args.size(), 2U); spirv::Value a = MakeValue(op->args[0]); spirv::Value b = MakeValue(op->args[1]); return builder_->MakeValue(spv::OpShiftLeftLogical, a.stype, a, b); } else if (op->is_intrinsic(Call::shift_right)) { CHECK_EQ(op->args.size(), 2U); spirv::Value a = MakeValue(op->args[0]); spirv::Value b = MakeValue(op->args[1]); if (op->args[0].type().is_int()) { return builder_->MakeValue(spv::OpShiftRightArithmetic, a.stype, a, b); } else { return builder_->MakeValue(spv::OpShiftRightLogical, a.stype, a, b); } } else if (op->is_intrinsic(intrinsic::tvm_storage_sync)) { return this->CreateStorageSync(op); } else if (op->is_intrinsic(intrinsic::tvm_if_then_else)) { CHECK_EQ(op->args.size(), 3U); spirv::Value cond = MakeValue(op->args[0]); spirv::Label then_label = builder_->NewLabel(); spirv::Label else_label = builder_->NewLabel(); spirv::Label merge_label = builder_->NewLabel(); builder_->MakeInst( spv::OpSelectionMerge, merge_label, spv::SelectionControlMaskNone); builder_->MakeInst( spv::OpBranchConditional, cond, then_label, else_label); // then block, must get label after we see the value builder_->StartLabel(then_label); spirv::Value then_value = MakeValue(op->args[1]); spirv::Label then_value_label = builder_->CurrentLabel(); builder_->MakeInst(spv::OpBranch, merge_label); // else block builder_->StartLabel(else_label); spirv::Value else_value = MakeValue(op->args[2]); spirv::Label else_value_label = builder_->CurrentLabel(); builder_->MakeInst(spv::OpBranch, merge_label); // merge block builder_->StartLabel(merge_label); spirv::PhiValue phi = builder_->MakePhi(then_value.stype, 2); phi.SetIncoming(0, then_value, then_value_label); phi.SetIncoming(1, else_value, else_value_label); return phi; } else if (op->is_intrinsic("popcount")) { return builder_->MakeValue( spv::OpBitCount, builder_->GetSType(op->type), MakeValue(op->args[0])); } else { if (op->call_type == Call::Intrinsic || op->call_type == Call::PureIntrinsic) { LOG(FATAL) << "Unresolved intrinsic " << op->name << " with return type " << op->type; } else if (op->call_type == Call::Extern || op->call_type == Call::PureExtern) { LOG(FATAL) << "Unresolved extern " << op->name << " with return type " << op->type; } else { LOG(FATAL) << "Unresolved call type " << op->call_type; } return spirv::Value(); } } spirv::Value CodeGenSPIRV::VisitExpr_(const Ramp* op) { std::vector<spirv::Value> values; spirv::Value base = MakeValue(op->base); for (int i = 0; i < op->lanes; ++i) { spirv::Value v = base; if (i != 0) { spirv::Value offset = MakeValue( arith::ComputeExpr<Mul>(make_const(op->stride.type(), i), op->stride)); v = builder_->Add(v, offset); } values.push_back(v); } return builder_->Concat(values); } spirv::Value CodeGenSPIRV::VisitExpr_(const Broadcast* op) { std::vector<spirv::Value> values; spirv::Value v = MakeValue(op->value); for (int i = 0; i < op->lanes; i++) { values.push_back(v); } return builder_->Concat(values); } spirv::Value CodeGenSPIRV::VisitExpr_(const Load* op) { CHECK(is_one(op->predicate)); auto it = storage_info_.find(op->buffer_var.get()); CHECK(it != storage_info_.end()); StorageInfo& info = it->second; if (!info.content_fixed) { info.UpdateContentType(op->type); } spirv::SType content_type = builder_->GetSType(info.content_type); spirv::Value buffer = MakeValue(op->buffer_var); spirv::SType ptr_type = builder_->GetPointerType( content_type, buffer.stype.storage_class); uint32_t mask = spv::MemoryAccessMaskNone; if (info.is_volatile) { mask |= spv::MemoryAccessVolatileMask; } if (op->type.lanes() == 1) { CHECK_EQ(info.content_type, op->type) << "Vulkan only allow one type access to the same buffer"; spirv::Value index = MakeValue(op->index); spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, index); return builder_->MakeValue(spv::OpLoad, content_type, ptr, mask); } else { if (op->type.element_of() == info.content_type) { // because content type is element type, we can only do scalarize load. std::vector<spirv::Value> values; auto f = [&](int i, spirv::Value index) { spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, index); values.emplace_back( builder_->MakeValue(spv::OpLoad, content_type, ptr, mask)); }; this->Scalarize(op->index, f); return builder_->Concat(values); } else { if (const Ramp* ramp = op->index.as<Ramp>()) { if (is_one(ramp->stride)) { CHECK_EQ(ramp->lanes, op->type.lanes()); arith::ModularEntry me = arith::EvalModular(ramp->base, align_map_); CHECK((me.coeff % ramp->lanes) == 0 && (me.base % ramp->lanes) == 0) << "Only aligned vector access is allowed in SPIRV"; Expr vec_index = ir::Simplify( ramp->base / make_const(ramp->base.type(), ramp->lanes)); spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, MakeValue(vec_index)); return builder_->MakeValue(spv::OpLoad, content_type, ptr, mask); } } } LOG(FATAL) << "Only aligned continuous vector access is allowed in SPIRV"; } LOG(FATAL) << "Only aligned continuous vector access is allowed in SPIRV"; return spirv::Value(); } void CodeGenSPIRV::Scalarize(const Expr& e, std::function<void(int i, spirv::Value v)> f) { if (const Ramp* ramp = e.as<Ramp>()) { for (int i = 0; i < ramp->type.lanes(); ++i) { Expr offset = arith::ComputeExpr<Add>( ramp->base, arith::ComputeExpr<Mul>(ramp->stride, i)); f(i, MakeValue(offset)); } } else { spirv::SType etype = builder_->GetSType(e.type().element_of()); spirv::Value value = MakeValue(e); for (int i = 0; i < e.type().lanes(); ++i) { f(i, builder_->MakeValue( spv::OpCompositeExtract, etype, value, i)); } } } void CodeGenSPIRV::VisitStmt_(const Store* op) { CHECK(is_one(op->predicate)); auto it = storage_info_.find(op->buffer_var.get()); CHECK(it != storage_info_.end()); StorageInfo& info = it->second; if (!info.content_fixed) { info.UpdateContentType(op->value.type()); } spirv::SType content_type = builder_->GetSType(info.content_type); spirv::Value buffer = MakeValue(op->buffer_var); spirv::Value value = MakeValue(op->value); spirv::SType ptr_type = builder_->GetPointerType( content_type, buffer.stype.storage_class); uint32_t mask = spv::MemoryAccessMaskNone; if (info.is_volatile) { mask |= spv::MemoryAccessVolatileMask; } if (op->value.type().lanes() == 1) { CHECK_EQ(info.content_type, op->value.type()) << "Vulkan only allow one type access to the same buffer"; spirv::Value index = MakeValue(op->index); spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, index); builder_->MakeInst(spv::OpStore, ptr, value, mask); } else { if (op->value.type().element_of() == info.content_type) { // because content type is element type, we can only do scalarize load. auto f = [&](int i, spirv::Value index) { spirv::Value elem = builder_->MakeValue( spv::OpCompositeExtract, content_type, value, i); spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, index); builder_->MakeInst(spv::OpStore, ptr, elem, mask); }; this->Scalarize(op->index, f); } else { if (const Ramp* ramp = op->index.as<Ramp>()) { if (is_one(ramp->stride)) { CHECK_EQ(ramp->lanes, op->value.type().lanes()); arith::ModularEntry me = arith::EvalModular(ramp->base, align_map_); CHECK((me.coeff % ramp->lanes) == 0 && (me.base % ramp->lanes) == 0) << "Only aligned vector access is allowed in SPIRV"; Expr vec_index = ir::Simplify( ramp->base / make_const(ramp->base.type(), ramp->lanes)); spirv::Value ptr = builder_->StructArrayAccess( ptr_type, buffer, MakeValue(vec_index)); builder_->MakeInst(spv::OpStore, ptr, value, mask); return; } } LOG(FATAL) << "Only aligned continuous vector access is allowed in SPIRV"; } } } void CodeGenSPIRV::VisitStmt_(const For* op) { CHECK(is_zero(op->min)); spirv::Value init_value = MakeValue(op->min); spirv::Value extent_value = MakeValue(op->extent); // Must get init label after making value(to make sure they are correct) spirv::Label init_label = builder_->CurrentLabel(); spirv::Label head_label = builder_->NewLabel(); spirv::Label body_label = builder_->NewLabel(); spirv::Label continue_label = builder_->NewLabel(); spirv::Label merge_label = builder_->NewLabel(); builder_->MakeInst(spv::OpBranch, head_label); // Loop head builder_->StartLabel(head_label); spirv::PhiValue loop_var = builder_->MakePhi(init_value.stype, 2); loop_var.SetIncoming(0, init_value, init_label); spirv::Value loop_cond = builder_->LT(loop_var, extent_value); uint32_t control = ( op->for_type == ForType::Unrolled ? spv::LoopControlUnrollMask : spv::LoopControlMaskNone); builder_->MakeInst( spv::OpLoopMerge, merge_label, continue_label, control); builder_->MakeInst( spv::OpBranchConditional, loop_cond, body_label, merge_label, weight_likely_branch_, 1); // loop body builder_->StartLabel(body_label); var_map_[op->loop_var.get()] = spirv::Value(loop_var); this->VisitStmt(op->body); builder_->MakeInst(spv::OpBranch, continue_label); // loop continue builder_->StartLabel(continue_label); spirv::Value one = op->loop_var.type().is_int() ? builder_->IntImm(loop_var.stype, 1) : builder_->UIntImm(loop_var.stype, 1); spirv::Value next_value = builder_->Add(loop_var, one); loop_var.SetIncoming(1, next_value, builder_->CurrentLabel()); builder_->MakeInst(spv::OpBranch, head_label); // loop merge builder_->StartLabel(merge_label); } void CodeGenSPIRV::VisitStmt_(const IfThenElse* op) { spirv::Value cond = MakeValue(op->condition); spirv::Label then_label = builder_->NewLabel(); spirv::Label merge_label = builder_->NewLabel(); if (op->else_case.defined()) { spirv::Label else_label = builder_->NewLabel(); builder_->MakeInst( spv::OpSelectionMerge, merge_label, spv::SelectionControlMaskNone); builder_->MakeInst( spv::OpBranchConditional, cond, then_label, else_label); // then block builder_->StartLabel(then_label); this->VisitStmt(op->then_case); builder_->MakeInst(spv::OpBranch, merge_label); // else block builder_->StartLabel(else_label); this->VisitStmt(op->else_case); builder_->MakeInst(spv::OpBranch, merge_label); } else { builder_->MakeInst( spv::OpSelectionMerge, merge_label, spv::SelectionControlMaskNone); builder_->MakeInst( spv::OpBranchConditional, cond, then_label, merge_label, weight_likely_branch_, 1); // then block builder_->StartLabel(then_label); this->VisitStmt(op->then_case); builder_->MakeInst(spv::OpBranch, merge_label); } // start merge label; builder_->StartLabel(merge_label); } void CodeGenSPIRV::VisitStmt_(const Allocate* op) { CHECK(!is_zero(op->condition)); CHECK(!op->new_expr.defined()); CHECK(!op->type.is_handle()); int32_t constant_size = op->constant_allocation_size(); CHECK_GT(constant_size, 0) << "Can only handle constant size stack allocation in GPU"; spirv::Value buf; StorageInfo& info = storage_info_[op->buffer_var.get()]; spirv::SType etype = builder_->GetSType(op->type); if (info.scope.rank == runtime::StorageRank::kLocal) { buf = builder_->Allocate( etype, static_cast<uint32_t>(constant_size), spv::StorageClassFunction); } else { // shared memory CHECK(info.scope.rank == runtime::StorageRank::kShared) << "Can only allocate shared or local memory inside kernel"; // Shared memory buf = builder_->Allocate( etype, static_cast<uint32_t>(constant_size), spv::StorageClassWorkgroup); } CHECK(!info.content_fixed); info.UpdateContentType(op->type); CHECK(!var_map_.count(op->buffer_var.get())); var_map_[op->buffer_var.get()] = buf; this->VisitStmt(op->body); } void CodeGenSPIRV::VisitStmt_(const AttrStmt* op) { if (op->attr_key == attr::thread_extent) { IterVar iv(op->node.node_); if (iv->thread_tag.length() != 0) { if (!var_map_.count(iv->var.get())) { var_map_[iv->var.get()] = GetThreadIndex(iv, op->value); } } } else if (op->attr_key == ir::attr::storage_scope) { const Variable* v = op->node.as<Variable>(); CHECK(v); storage_info_[v].scope = runtime::StorageScope::make(op->value.as<StringImm>()->value); } else if (op->attr_key == ir::attr::volatile_scope) { const Variable* v = op->node.as<Variable>(); CHECK(v); storage_info_[v].is_volatile = true; } this->VisitStmt(op->body); } void CodeGenSPIRV::VisitStmt_(const AssertStmt* op) { VisitAssert(op, &align_map_, [this](const Stmt& body) { this->VisitStmt(body); }); } void CodeGenSPIRV::VisitStmt_(const LetStmt* op) { CHECK(!var_map_.count(op->var.get())); CHECK(!align_map_.count(op->var.get())); CHECK(!op->var.type().is_handle()); var_map_[op->var.get()] = MakeValue(op->value); align_map_[op->var.get()] = EvalModular(op->value, align_map_); this->VisitStmt(op->body); } void CodeGenSPIRV::VisitStmt_(const Block* op) { VisitStmt(op->first); if (op->rest.defined()) { this->VisitStmt(op->rest); } } void CodeGenSPIRV::VisitStmt_(const Evaluate* op) { MakeValue(op->value); } void CodeGenSPIRV::VisitStmt_(const ProducerConsumer* op) { this->VisitStmt(op->body); } } // namespace codegen } // namespace tvm #endif // TVM_VULKAN_RUNTIME