2017-05-31 00:03:44 +00:00
|
|
|
// Copyright 2017 The NXT 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.
|
|
|
|
|
2017-07-06 18:41:13 +00:00
|
|
|
#include "backend/Pipeline.h"
|
2017-05-31 00:03:44 +00:00
|
|
|
|
2017-07-06 18:41:13 +00:00
|
|
|
#include "backend/DepthStencilState.h"
|
2017-11-24 18:59:42 +00:00
|
|
|
#include "backend/Device.h"
|
2017-07-06 18:41:13 +00:00
|
|
|
#include "backend/InputState.h"
|
|
|
|
#include "backend/PipelineLayout.h"
|
|
|
|
#include "backend/ShaderModule.h"
|
2017-05-31 00:03:44 +00:00
|
|
|
|
|
|
|
namespace backend {
|
|
|
|
|
|
|
|
// PipelineBase
|
|
|
|
|
|
|
|
PipelineBase::PipelineBase(PipelineBuilder* builder)
|
2017-11-23 18:32:51 +00:00
|
|
|
: mStageMask(builder->mStageMask), mLayout(std::move(builder->mLayout)) {
|
|
|
|
if (!mLayout) {
|
2018-06-27 23:21:39 +00:00
|
|
|
nxt::PipelineLayoutDescriptor descriptor;
|
|
|
|
descriptor.numBindGroupLayouts = 0;
|
|
|
|
descriptor.bindGroupLayouts = nullptr;
|
|
|
|
mLayout = builder->GetParentBuilder()->GetDevice()->CreatePipelineLayout(&descriptor);
|
2018-01-12 19:18:45 +00:00
|
|
|
// Remove the external ref objects are created with
|
|
|
|
mLayout->Release();
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
auto FillPushConstants = [](const ShaderModuleBase* module, PushConstantInfo* info) {
|
|
|
|
const auto& moduleInfo = module->GetPushConstants();
|
|
|
|
info->mask = moduleInfo.mask;
|
|
|
|
|
|
|
|
for (uint32_t i = 0; i < moduleInfo.names.size(); i++) {
|
2017-07-22 00:00:22 +00:00
|
|
|
uint32_t size = moduleInfo.sizes[i];
|
2017-05-31 00:03:44 +00:00
|
|
|
if (size == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t offset = 0; offset < size; offset++) {
|
|
|
|
info->types[i + offset] = moduleInfo.types[i];
|
|
|
|
}
|
|
|
|
i += size - 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-11-23 18:32:51 +00:00
|
|
|
for (auto stageBit : IterateStages(builder->mStageMask)) {
|
|
|
|
if (!builder->mStages[stageBit].module->IsCompatibleWithPipelineLayout(mLayout.Get())) {
|
2017-07-14 14:58:07 +00:00
|
|
|
builder->GetParentBuilder()->HandleError("Stage not compatible with layout");
|
2017-05-31 00:03:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-11-23 18:32:51 +00:00
|
|
|
FillPushConstants(builder->mStages[stageBit].module.Get(), &mPushConstants[stageBit]);
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-24 18:59:42 +00:00
|
|
|
const PipelineBase::PushConstantInfo& PipelineBase::GetPushConstants(
|
|
|
|
nxt::ShaderStage stage) const {
|
2017-11-23 18:32:51 +00:00
|
|
|
return mPushConstants[stage];
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nxt::ShaderStageBit PipelineBase::GetStageMask() const {
|
2017-11-23 18:32:51 +00:00
|
|
|
return mStageMask;
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
PipelineLayoutBase* PipelineBase::GetLayout() {
|
2017-11-23 18:32:51 +00:00
|
|
|
return mLayout.Get();
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// PipelineBuilder
|
|
|
|
|
2017-07-14 14:58:07 +00:00
|
|
|
PipelineBuilder::PipelineBuilder(BuilderBase* parentBuilder)
|
2017-11-23 18:32:51 +00:00
|
|
|
: mParentBuilder(parentBuilder), mStageMask(static_cast<nxt::ShaderStageBit>(0)) {
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const PipelineBuilder::StageInfo& PipelineBuilder::GetStageInfo(nxt::ShaderStage stage) const {
|
2017-11-23 18:32:51 +00:00
|
|
|
ASSERT(mStageMask & StageBit(stage));
|
|
|
|
return mStages[stage];
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
2017-07-14 14:58:07 +00:00
|
|
|
BuilderBase* PipelineBuilder::GetParentBuilder() const {
|
2017-11-23 18:32:51 +00:00
|
|
|
return mParentBuilder;
|
2017-06-01 15:30:03 +00:00
|
|
|
}
|
|
|
|
|
2017-05-31 00:03:44 +00:00
|
|
|
void PipelineBuilder::SetLayout(PipelineLayoutBase* layout) {
|
2017-11-23 18:32:51 +00:00
|
|
|
mLayout = layout;
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
2017-11-24 18:59:42 +00:00
|
|
|
void PipelineBuilder::SetStage(nxt::ShaderStage stage,
|
|
|
|
ShaderModuleBase* module,
|
|
|
|
const char* entryPoint) {
|
2017-05-31 00:03:44 +00:00
|
|
|
if (entryPoint != std::string("main")) {
|
2017-11-23 18:32:51 +00:00
|
|
|
mParentBuilder->HandleError("Currently the entry point has to be main()");
|
2017-05-31 00:03:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stage != module->GetExecutionModel()) {
|
2017-11-23 18:32:51 +00:00
|
|
|
mParentBuilder->HandleError("Setting module with wrong execution model");
|
2017-05-31 00:03:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nxt::ShaderStageBit bit = StageBit(stage);
|
2017-11-23 18:32:51 +00:00
|
|
|
if (mStageMask & bit) {
|
|
|
|
mParentBuilder->HandleError("Setting already set stage");
|
2017-05-31 00:03:44 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-11-23 18:32:51 +00:00
|
|
|
mStageMask |= bit;
|
2017-05-31 00:03:44 +00:00
|
|
|
|
2017-11-23 18:32:51 +00:00
|
|
|
mStages[stage].module = module;
|
|
|
|
mStages[stage].entryPoint = entryPoint;
|
2017-05-31 00:03:44 +00:00
|
|
|
}
|
|
|
|
|
2017-11-24 18:59:42 +00:00
|
|
|
} // namespace backend
|