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/Device.h"
|
|
|
|
#include "backend/DepthStencilState.h"
|
|
|
|
#include "backend/InputState.h"
|
|
|
|
#include "backend/PipelineLayout.h"
|
|
|
|
#include "backend/RenderPass.h"
|
|
|
|
#include "backend/ShaderModule.h"
|
2017-05-31 00:03:44 +00:00
|
|
|
|
|
|
|
namespace backend {
|
|
|
|
|
|
|
|
// PipelineBase
|
|
|
|
|
|
|
|
PipelineBase::PipelineBase(PipelineBuilder* builder)
|
2017-06-27 15:10:29 +00:00
|
|
|
: stageMask(builder->stageMask), layout(std::move(builder->layout)),
|
2017-05-31 00:03:44 +00:00
|
|
|
renderPass(std::move(builder->renderPass)), subpass(builder->subpass),
|
|
|
|
inputState(std::move(builder->inputState)), depthStencilState(std::move(builder->depthStencilState)) {
|
|
|
|
|
|
|
|
if (stageMask != (nxt::ShaderStageBit::Vertex | nxt::ShaderStageBit::Fragment) &&
|
|
|
|
stageMask != nxt::ShaderStageBit::Compute) {
|
|
|
|
builder->HandleError("Wrong combination of stage for pipeline");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-06-01 15:30:03 +00:00
|
|
|
if (IsCompute() && depthStencilState) {
|
|
|
|
builder->HandleError("Compute pipeline cannot have depth stencil state");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:03:44 +00:00
|
|
|
if (!IsCompute() && !renderPass) {
|
|
|
|
builder->HandleError("Pipeline render pass not set");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO(kainino@chromium.org): Need to verify the pipeline against its render subpass.
|
|
|
|
|
|
|
|
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++) {
|
|
|
|
unsigned int size = moduleInfo.sizes[i];
|
|
|
|
if (size == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint32_t offset = 0; offset < size; offset++) {
|
|
|
|
info->types[i + offset] = moduleInfo.types[i];
|
|
|
|
}
|
|
|
|
i += size - 1;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto stageBit : IterateStages(builder->stageMask)) {
|
|
|
|
if (!builder->stages[stageBit].module->IsCompatibleWithPipelineLayout(layout.Get())) {
|
|
|
|
builder->HandleError("Stage not compatible with layout");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
FillPushConstants(builder->stages[stageBit].module.Get(), &pushConstants[stageBit]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!IsCompute()) {
|
|
|
|
if ((builder->stages[nxt::ShaderStage::Vertex].module->GetUsedVertexAttributes() & ~inputState->GetAttributesSetMask()).any()) {
|
|
|
|
builder->HandleError("Pipeline vertex stage uses inputs not in the input state");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const PipelineBase::PushConstantInfo& PipelineBase::GetPushConstants(nxt::ShaderStage stage) const {
|
|
|
|
return pushConstants[stage];
|
|
|
|
}
|
|
|
|
|
|
|
|
nxt::ShaderStageBit PipelineBase::GetStageMask() const {
|
|
|
|
return stageMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
PipelineLayoutBase* PipelineBase::GetLayout() {
|
|
|
|
return layout.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
RenderPassBase* PipelineBase::GetRenderPass() {
|
|
|
|
return renderPass.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
InputStateBase* PipelineBase::GetInputState() {
|
|
|
|
return inputState.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
DepthStencilStateBase* PipelineBase::GetDepthStencilState() {
|
|
|
|
return depthStencilState.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PipelineBase::IsCompute() const {
|
|
|
|
return stageMask == nxt::ShaderStageBit::Compute;
|
|
|
|
}
|
|
|
|
|
|
|
|
// PipelineBuilder
|
|
|
|
|
|
|
|
PipelineBuilder::PipelineBuilder(DeviceBase* device)
|
|
|
|
: Builder(device), stageMask(static_cast<nxt::ShaderStageBit>(0)) {
|
|
|
|
}
|
|
|
|
|
|
|
|
const PipelineBuilder::StageInfo& PipelineBuilder::GetStageInfo(nxt::ShaderStage stage) const {
|
|
|
|
ASSERT(stageMask & StageBit(stage));
|
|
|
|
return stages[stage];
|
|
|
|
}
|
|
|
|
|
|
|
|
PipelineBase* PipelineBuilder::GetResultImpl() {
|
|
|
|
// TODO(cwallez@chromium.org): the layout should be required, and put the default objects in the device
|
|
|
|
if (!layout) {
|
|
|
|
layout = device->CreatePipelineLayoutBuilder()->GetResult();
|
|
|
|
}
|
2017-06-02 18:31:53 +00:00
|
|
|
if (!inputState && !IsCompute()) {
|
2017-05-31 00:03:44 +00:00
|
|
|
inputState = device->CreateInputStateBuilder()->GetResult();
|
|
|
|
}
|
2017-06-01 15:30:03 +00:00
|
|
|
if (!depthStencilState && !IsCompute()) {
|
2017-05-31 00:03:44 +00:00
|
|
|
depthStencilState = device->CreateDepthStencilStateBuilder()->GetResult();
|
|
|
|
}
|
|
|
|
|
|
|
|
return device->CreatePipeline(this);
|
|
|
|
}
|
|
|
|
|
2017-06-01 15:30:03 +00:00
|
|
|
bool PipelineBuilder::IsCompute() const {
|
|
|
|
return stageMask == nxt::ShaderStageBit::Compute;
|
|
|
|
}
|
|
|
|
|
2017-05-31 00:03:44 +00:00
|
|
|
void PipelineBuilder::SetLayout(PipelineLayoutBase* layout) {
|
|
|
|
this->layout = layout;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PipelineBuilder::SetSubpass(RenderPassBase* renderPass, uint32_t subpass) {
|
|
|
|
this->renderPass = renderPass;
|
|
|
|
this->subpass = subpass;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PipelineBuilder::SetStage(nxt::ShaderStage stage, ShaderModuleBase* module, const char* entryPoint) {
|
|
|
|
if (entryPoint != std::string("main")) {
|
|
|
|
HandleError("Currently the entry point has to be main()");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stage != module->GetExecutionModel()) {
|
|
|
|
HandleError("Setting module with wrong execution model");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nxt::ShaderStageBit bit = StageBit(stage);
|
|
|
|
if (stageMask & bit) {
|
|
|
|
HandleError("Setting already set stage");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
stageMask |= bit;
|
|
|
|
|
|
|
|
stages[stage].module = module;
|
|
|
|
stages[stage].entryPoint = entryPoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PipelineBuilder::SetInputState(InputStateBase* inputState) {
|
|
|
|
this->inputState = inputState;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PipelineBuilder::SetDepthStencilState(DepthStencilStateBase* depthStencilState) {
|
|
|
|
this->depthStencilState = depthStencilState;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|