// 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. #include "BindGroup.h" #include "BindGroupLayout.h" #include "Buffer.h" #include "Device.h" #include "Texture.h" namespace backend { // BindGroup BindGroupBase::BindGroupBase(BindGroupBuilder* builder) : layout(std::move(builder->layout)), usage(builder->usage), bindings(std::move(builder->bindings)) { } const BindGroupLayoutBase* BindGroupBase::GetLayout() const { return layout.Get(); } nxt::BindGroupUsage BindGroupBase::GetUsage() const { return usage; } BufferViewBase* BindGroupBase::GetBindingAsBufferView(size_t binding) { ASSERT(binding < kMaxBindingsPerGroup); ASSERT(layout->GetBindingInfo().mask[binding]); ASSERT(layout->GetBindingInfo().types[binding] == nxt::BindingType::UniformBuffer || layout->GetBindingInfo().types[binding] == nxt::BindingType::StorageBuffer); return reinterpret_cast(bindings[binding].Get()); } SamplerBase* BindGroupBase::GetBindingAsSampler(size_t binding) { ASSERT(binding < kMaxBindingsPerGroup); ASSERT(layout->GetBindingInfo().mask[binding]); ASSERT(layout->GetBindingInfo().types[binding] == nxt::BindingType::Sampler); return reinterpret_cast(bindings[binding].Get()); } TextureViewBase* BindGroupBase::GetBindingAsTextureView(size_t binding) { ASSERT(binding < kMaxBindingsPerGroup); ASSERT(layout->GetBindingInfo().mask[binding]); ASSERT(layout->GetBindingInfo().types[binding] == nxt::BindingType::SampledTexture); return reinterpret_cast(bindings[binding].Get()); } // BindGroupBuilder enum BindGroupSetProperties { BINDGROUP_PROPERTY_USAGE = 0x1, BINDGROUP_PROPERTY_LAYOUT = 0x2, }; BindGroupBuilder::BindGroupBuilder(DeviceBase* device) : device(device) { } bool BindGroupBuilder::WasConsumed() const { return consumed; } BindGroupBase* BindGroupBuilder::GetResult() { constexpr int allProperties = BINDGROUP_PROPERTY_USAGE | BINDGROUP_PROPERTY_LAYOUT; if ((propertiesSet & allProperties) != allProperties) { device->HandleError("Bindgroup missing properties"); return nullptr; } if (setMask != layout->GetBindingInfo().mask) { device->HandleError("Bindgroup missing bindings"); return nullptr; } consumed = true; return device->CreateBindGroup(this); } void BindGroupBuilder::SetLayout(BindGroupLayoutBase* layout) { if ((propertiesSet & BINDGROUP_PROPERTY_LAYOUT) != 0) { device->HandleError("Bindgroup layout property set multiple times"); return; } this->layout = layout; propertiesSet |= BINDGROUP_PROPERTY_LAYOUT; } void BindGroupBuilder::SetUsage(nxt::BindGroupUsage usage) { if ((propertiesSet & BINDGROUP_PROPERTY_USAGE) != 0) { device->HandleError("Bindgroup usage property set multiple times"); return; } this->usage = usage; propertiesSet |= BINDGROUP_PROPERTY_USAGE; } void BindGroupBuilder::SetBufferViews(uint32_t start, uint32_t count, BufferViewBase* const * bufferViews) { if (!SetBindingsValidationBase(start, count)) { return; } const auto& layoutInfo = layout->GetBindingInfo(); for (size_t i = start, j = 0; i < start + count; ++i, ++j) { nxt::BufferUsageBit requiredBit; switch (layoutInfo.types[i]) { case nxt::BindingType::UniformBuffer: requiredBit = nxt::BufferUsageBit::Uniform; break; case nxt::BindingType::StorageBuffer: requiredBit = nxt::BufferUsageBit::Storage; break; case nxt::BindingType::Sampler: case nxt::BindingType::SampledTexture: device->HandleError("Setting buffer for a wrong binding type"); return; } if (!(bufferViews[j]->GetBuffer()->GetAllowedUsage() & requiredBit)) { device->HandleError("Buffer needs to allow the correct usage bit"); return; } } SetBindingsBase(start, count, reinterpret_cast(bufferViews)); } void BindGroupBuilder::SetSamplers(uint32_t start, uint32_t count, SamplerBase* const * samplers) { if (!SetBindingsValidationBase(start, count)) { return; } const auto& layoutInfo = layout->GetBindingInfo(); for (size_t i = start, j = 0; i < start + count; ++i, ++j) { if (layoutInfo.types[i] != nxt::BindingType::Sampler) { device->HandleError("Setting binding for a wrong layout binding type"); return; } } SetBindingsBase(start, count, reinterpret_cast(samplers)); } void BindGroupBuilder::SetTextureViews(uint32_t start, uint32_t count, TextureViewBase* const * textureViews) { if (!SetBindingsValidationBase(start, count)) { return; } const auto& layoutInfo = layout->GetBindingInfo(); for (size_t i = start, j = 0; i < start + count; ++i, ++j) { if (layoutInfo.types[i] != nxt::BindingType::SampledTexture) { device->HandleError("Setting binding for a wrong layout binding type"); return; } if (!(textureViews[j]->GetTexture()->GetAllowedUsage() & nxt::TextureUsageBit::Sampled)) { device->HandleError("Texture needs to allow the sampled usage bit"); return; } } SetBindingsBase(start, count, reinterpret_cast(textureViews)); } void BindGroupBuilder::SetBindingsBase(uint32_t start, uint32_t count, RefCounted* const * objects) { for (size_t i = start, j = 0; i < start + count; ++i, ++j) { setMask.set(i); bindings[i] = objects[j]; } } bool BindGroupBuilder::SetBindingsValidationBase(uint32_t start, uint32_t count) { if (start + count > kMaxBindingsPerGroup) { device->HandleError("Setting bindings type over maximum number of bindings"); return false; } if ((propertiesSet & BINDGROUP_PROPERTY_LAYOUT) == 0) { device->HandleError("Bindgroup layout must be set before views"); return false; } const auto& layoutInfo = layout->GetBindingInfo(); for (size_t i = start, j = 0; i < start + count; ++i, ++j) { if (setMask[i]) { device->HandleError("Setting already set binding"); return false; } if (!layoutInfo.mask[i]) { device->HandleError("Setting binding that isn't present in the layout"); return false; } } return true; } }