mirror of
https://github.com/encounter/dawn-cmake.git
synced 2025-06-24 07:23:35 +00:00
dawn_native: deduplicate pipeline layouts
This is the first step to do pipeline deduplication. It also introduces tests for deduplication. BUG=dawn:143 Change-Id: Ib22496f543f8d1f9cfde04f725612504132c7d72 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/6861 Commit-Queue: Corentin Wallez <cwallez@chromium.org> Reviewed-by: Kai Ninomiya <kainino@chromium.org>
This commit is contained in:
parent
cc141797df
commit
0ee9859c91
1
BUILD.gn
1
BUILD.gn
@ -643,6 +643,7 @@ test("dawn_end2end_tests") {
|
|||||||
"src/tests/end2end/IndexFormatTests.cpp",
|
"src/tests/end2end/IndexFormatTests.cpp",
|
||||||
"src/tests/end2end/InputStateTests.cpp",
|
"src/tests/end2end/InputStateTests.cpp",
|
||||||
"src/tests/end2end/MultisampledRenderingTests.cpp",
|
"src/tests/end2end/MultisampledRenderingTests.cpp",
|
||||||
|
"src/tests/end2end/ObjectCachingTests.cpp",
|
||||||
"src/tests/end2end/PrimitiveTopologyTests.cpp",
|
"src/tests/end2end/PrimitiveTopologyTests.cpp",
|
||||||
"src/tests/end2end/PushConstantTests.cpp",
|
"src/tests/end2end/PushConstantTests.cpp",
|
||||||
"src/tests/end2end/RenderPassLoadOpTests.cpp",
|
"src/tests/end2end/RenderPassLoadOpTests.cpp",
|
||||||
|
@ -114,15 +114,13 @@ namespace dawn_native {
|
|||||||
return mBindingInfo;
|
return mBindingInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
// BindGroupLayoutCacheFuncs
|
size_t BindGroupLayoutBase::HashFunc::operator()(const BindGroupLayoutBase* bgl) const {
|
||||||
|
return HashBindingInfo(bgl->mBindingInfo);
|
||||||
size_t BindGroupLayoutCacheFuncs::operator()(const BindGroupLayoutBase* bgl) const {
|
|
||||||
return HashBindingInfo(bgl->GetBindingInfo());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BindGroupLayoutCacheFuncs::operator()(const BindGroupLayoutBase* a,
|
bool BindGroupLayoutBase::EqualityFunc::operator()(const BindGroupLayoutBase* a,
|
||||||
const BindGroupLayoutBase* b) const {
|
const BindGroupLayoutBase* b) const {
|
||||||
return a->GetBindingInfo() == b->GetBindingInfo();
|
return a->mBindingInfo == b->mBindingInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace dawn_native
|
} // namespace dawn_native
|
||||||
|
@ -46,6 +46,14 @@ namespace dawn_native {
|
|||||||
};
|
};
|
||||||
const LayoutBindingInfo& GetBindingInfo() const;
|
const LayoutBindingInfo& GetBindingInfo() const;
|
||||||
|
|
||||||
|
// Functors necessary for the unordered_set<BGLBase*>-based cache.
|
||||||
|
struct HashFunc {
|
||||||
|
size_t operator()(const BindGroupLayoutBase* bgl) const;
|
||||||
|
};
|
||||||
|
struct EqualityFunc {
|
||||||
|
bool operator()(const BindGroupLayoutBase* a, const BindGroupLayoutBase* b) const;
|
||||||
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
BindGroupLayoutBase(DeviceBase* device, ObjectBase::ErrorTag tag);
|
BindGroupLayoutBase(DeviceBase* device, ObjectBase::ErrorTag tag);
|
||||||
|
|
||||||
@ -53,15 +61,6 @@ namespace dawn_native {
|
|||||||
bool mIsBlueprint = false;
|
bool mIsBlueprint = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Implements the functors necessary for the unordered_set<BGL*>-based cache.
|
|
||||||
struct BindGroupLayoutCacheFuncs {
|
|
||||||
// The hash function
|
|
||||||
size_t operator()(const BindGroupLayoutBase* bgl) const;
|
|
||||||
|
|
||||||
// The equality predicate
|
|
||||||
bool operator()(const BindGroupLayoutBase* a, const BindGroupLayoutBase* b) const;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace dawn_native
|
} // namespace dawn_native
|
||||||
|
|
||||||
#endif // DAWNNATIVE_BINDGROUPLAYOUT_H_
|
#endif // DAWNNATIVE_BINDGROUPLAYOUT_H_
|
||||||
|
@ -42,11 +42,13 @@ namespace dawn_native {
|
|||||||
|
|
||||||
// The caches are unordered_sets of pointers with special hash and compare functions
|
// The caches are unordered_sets of pointers with special hash and compare functions
|
||||||
// to compare the value of the objects, instead of the pointers.
|
// to compare the value of the objects, instead of the pointers.
|
||||||
using BindGroupLayoutCache = std::
|
template <typename Object>
|
||||||
unordered_set<BindGroupLayoutBase*, BindGroupLayoutCacheFuncs, BindGroupLayoutCacheFuncs>;
|
using ContentLessObjectCache =
|
||||||
|
std::unordered_set<Object*, typename Object::HashFunc, typename Object::EqualityFunc>;
|
||||||
|
|
||||||
struct DeviceBase::Caches {
|
struct DeviceBase::Caches {
|
||||||
BindGroupLayoutCache bindGroupLayouts;
|
ContentLessObjectCache<BindGroupLayoutBase> bindGroupLayouts;
|
||||||
|
ContentLessObjectCache<PipelineLayoutBase> pipelineLayouts;
|
||||||
};
|
};
|
||||||
|
|
||||||
// DeviceBase
|
// DeviceBase
|
||||||
@ -114,7 +116,29 @@ namespace dawn_native {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void DeviceBase::UncacheBindGroupLayout(BindGroupLayoutBase* obj) {
|
void DeviceBase::UncacheBindGroupLayout(BindGroupLayoutBase* obj) {
|
||||||
mCaches->bindGroupLayouts.erase(obj);
|
size_t removedCount = mCaches->bindGroupLayouts.erase(obj);
|
||||||
|
ASSERT(removedCount == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultOrError<PipelineLayoutBase*> DeviceBase::GetOrCreatePipelineLayout(
|
||||||
|
const PipelineLayoutDescriptor* descriptor) {
|
||||||
|
PipelineLayoutBase blueprint(this, descriptor, true);
|
||||||
|
|
||||||
|
auto iter = mCaches->pipelineLayouts.find(&blueprint);
|
||||||
|
if (iter != mCaches->pipelineLayouts.end()) {
|
||||||
|
(*iter)->Reference();
|
||||||
|
return *iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
PipelineLayoutBase* backendObj;
|
||||||
|
DAWN_TRY_ASSIGN(backendObj, CreatePipelineLayoutImpl(descriptor));
|
||||||
|
mCaches->pipelineLayouts.insert(backendObj);
|
||||||
|
return backendObj;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DeviceBase::UncachePipelineLayout(PipelineLayoutBase* obj) {
|
||||||
|
size_t removedCount = mCaches->pipelineLayouts.erase(obj);
|
||||||
|
ASSERT(removedCount == 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Object creation API methods
|
// Object creation API methods
|
||||||
@ -331,7 +355,7 @@ namespace dawn_native {
|
|||||||
PipelineLayoutBase** result,
|
PipelineLayoutBase** result,
|
||||||
const PipelineLayoutDescriptor* descriptor) {
|
const PipelineLayoutDescriptor* descriptor) {
|
||||||
DAWN_TRY(ValidatePipelineLayoutDescriptor(this, descriptor));
|
DAWN_TRY(ValidatePipelineLayoutDescriptor(this, descriptor));
|
||||||
DAWN_TRY_ASSIGN(*result, CreatePipelineLayoutImpl(descriptor));
|
DAWN_TRY_ASSIGN(*result, GetOrCreatePipelineLayout(descriptor));
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,6 +84,10 @@ namespace dawn_native {
|
|||||||
const BindGroupLayoutDescriptor* descriptor);
|
const BindGroupLayoutDescriptor* descriptor);
|
||||||
void UncacheBindGroupLayout(BindGroupLayoutBase* obj);
|
void UncacheBindGroupLayout(BindGroupLayoutBase* obj);
|
||||||
|
|
||||||
|
ResultOrError<PipelineLayoutBase*> GetOrCreatePipelineLayout(
|
||||||
|
const PipelineLayoutDescriptor* descriptor);
|
||||||
|
void UncachePipelineLayout(PipelineLayoutBase* obj);
|
||||||
|
|
||||||
// Dawn API
|
// Dawn API
|
||||||
BindGroupBase* CreateBindGroup(const BindGroupDescriptor* descriptor);
|
BindGroupBase* CreateBindGroup(const BindGroupDescriptor* descriptor);
|
||||||
BindGroupLayoutBase* CreateBindGroupLayout(const BindGroupLayoutDescriptor* descriptor);
|
BindGroupLayoutBase* CreateBindGroupLayout(const BindGroupLayoutDescriptor* descriptor);
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
#include "dawn_native/PipelineLayout.h"
|
#include "dawn_native/PipelineLayout.h"
|
||||||
|
|
||||||
#include "common/Assert.h"
|
#include "common/Assert.h"
|
||||||
|
#include "common/BitSetIterator.h"
|
||||||
|
#include "common/HashUtils.h"
|
||||||
#include "dawn_native/BindGroupLayout.h"
|
#include "dawn_native/BindGroupLayout.h"
|
||||||
#include "dawn_native/Device.h"
|
#include "dawn_native/Device.h"
|
||||||
|
|
||||||
@ -39,8 +41,9 @@ namespace dawn_native {
|
|||||||
// PipelineLayoutBase
|
// PipelineLayoutBase
|
||||||
|
|
||||||
PipelineLayoutBase::PipelineLayoutBase(DeviceBase* device,
|
PipelineLayoutBase::PipelineLayoutBase(DeviceBase* device,
|
||||||
const PipelineLayoutDescriptor* descriptor)
|
const PipelineLayoutDescriptor* descriptor,
|
||||||
: ObjectBase(device) {
|
bool blueprint)
|
||||||
|
: ObjectBase(device), mIsBlueprint(blueprint) {
|
||||||
ASSERT(descriptor->bindGroupLayoutCount <= kMaxBindGroups);
|
ASSERT(descriptor->bindGroupLayoutCount <= kMaxBindGroups);
|
||||||
for (uint32_t group = 0; group < descriptor->bindGroupLayoutCount; ++group) {
|
for (uint32_t group = 0; group < descriptor->bindGroupLayoutCount; ++group) {
|
||||||
mBindGroupLayouts[group] = descriptor->bindGroupLayouts[group];
|
mBindGroupLayouts[group] = descriptor->bindGroupLayouts[group];
|
||||||
@ -52,6 +55,14 @@ namespace dawn_native {
|
|||||||
: ObjectBase(device, tag) {
|
: ObjectBase(device, tag) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PipelineLayoutBase::~PipelineLayoutBase() {
|
||||||
|
// Do not uncache the actual cached object if we are a blueprint
|
||||||
|
if (!mIsBlueprint) {
|
||||||
|
ASSERT(!IsError());
|
||||||
|
GetDevice()->UncachePipelineLayout(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// static
|
// static
|
||||||
PipelineLayoutBase* PipelineLayoutBase::MakeError(DeviceBase* device) {
|
PipelineLayoutBase* PipelineLayoutBase::MakeError(DeviceBase* device) {
|
||||||
return new PipelineLayoutBase(device, ObjectBase::kError);
|
return new PipelineLayoutBase(device, ObjectBase::kError);
|
||||||
@ -86,4 +97,29 @@ namespace dawn_native {
|
|||||||
return kMaxBindGroups;
|
return kMaxBindGroups;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t PipelineLayoutBase::HashFunc::operator()(const PipelineLayoutBase* pl) const {
|
||||||
|
size_t hash = Hash(pl->mMask);
|
||||||
|
|
||||||
|
for (uint32_t group : IterateBitSet(pl->mMask)) {
|
||||||
|
HashCombine(&hash, pl->GetBindGroupLayout(group));
|
||||||
|
}
|
||||||
|
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PipelineLayoutBase::EqualityFunc::operator()(const PipelineLayoutBase* a,
|
||||||
|
const PipelineLayoutBase* b) const {
|
||||||
|
if (a->mMask != b->mMask) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (uint32_t group : IterateBitSet(a->mMask)) {
|
||||||
|
if (a->GetBindGroupLayout(group) != b->GetBindGroupLayout(group)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace dawn_native
|
} // namespace dawn_native
|
||||||
|
@ -34,7 +34,10 @@ namespace dawn_native {
|
|||||||
|
|
||||||
class PipelineLayoutBase : public ObjectBase {
|
class PipelineLayoutBase : public ObjectBase {
|
||||||
public:
|
public:
|
||||||
PipelineLayoutBase(DeviceBase* device, const PipelineLayoutDescriptor* descriptor);
|
PipelineLayoutBase(DeviceBase* device,
|
||||||
|
const PipelineLayoutDescriptor* descriptor,
|
||||||
|
bool blueprint = false);
|
||||||
|
~PipelineLayoutBase() override;
|
||||||
|
|
||||||
static PipelineLayoutBase* MakeError(DeviceBase* device);
|
static PipelineLayoutBase* MakeError(DeviceBase* device);
|
||||||
|
|
||||||
@ -49,11 +52,20 @@ namespace dawn_native {
|
|||||||
// [1, kMaxBindGroups + 1]
|
// [1, kMaxBindGroups + 1]
|
||||||
uint32_t GroupsInheritUpTo(const PipelineLayoutBase* other) const;
|
uint32_t GroupsInheritUpTo(const PipelineLayoutBase* other) const;
|
||||||
|
|
||||||
|
// Functors necessary for the unordered_set<PipelineLayoutBase*>-based cache.
|
||||||
|
struct HashFunc {
|
||||||
|
size_t operator()(const PipelineLayoutBase* pl) const;
|
||||||
|
};
|
||||||
|
struct EqualityFunc {
|
||||||
|
bool operator()(const PipelineLayoutBase* a, const PipelineLayoutBase* b) const;
|
||||||
|
};
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
PipelineLayoutBase(DeviceBase* device, ObjectBase::ErrorTag tag);
|
PipelineLayoutBase(DeviceBase* device, ObjectBase::ErrorTag tag);
|
||||||
|
|
||||||
BindGroupLayoutArray mBindGroupLayouts;
|
BindGroupLayoutArray mBindGroupLayouts;
|
||||||
std::bitset<kMaxBindGroups> mMask;
|
std::bitset<kMaxBindGroups> mMask;
|
||||||
|
bool mIsBlueprint = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace dawn_native
|
} // namespace dawn_native
|
||||||
|
@ -166,7 +166,7 @@ void DawnTestEnvironment::SetUp() {
|
|||||||
std::cout << std::endl;
|
std::cout << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DawnTestEnvironment::UseWire() const {
|
bool DawnTestEnvironment::UsesWire() const {
|
||||||
return mUseWire;
|
return mUseWire;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -266,6 +266,10 @@ bool DawnTest::IsMacOS() const {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool DawnTest::UsesWire() const {
|
||||||
|
return gTestEnv->UsesWire();
|
||||||
|
}
|
||||||
|
|
||||||
void DawnTest::SetUp() {
|
void DawnTest::SetUp() {
|
||||||
// Get an adapter for the backend to use, and create the device.
|
// Get an adapter for the backend to use, and create the device.
|
||||||
dawn_native::Adapter backendAdapter;
|
dawn_native::Adapter backendAdapter;
|
||||||
@ -314,7 +318,7 @@ void DawnTest::SetUp() {
|
|||||||
DawnDevice cDevice = nullptr;
|
DawnDevice cDevice = nullptr;
|
||||||
DawnProcTable procs;
|
DawnProcTable procs;
|
||||||
|
|
||||||
if (gTestEnv->UseWire()) {
|
if (gTestEnv->UsesWire()) {
|
||||||
mC2sBuf = std::make_unique<utils::TerribleCommandBuffer>();
|
mC2sBuf = std::make_unique<utils::TerribleCommandBuffer>();
|
||||||
mS2cBuf = std::make_unique<utils::TerribleCommandBuffer>();
|
mS2cBuf = std::make_unique<utils::TerribleCommandBuffer>();
|
||||||
|
|
||||||
@ -473,7 +477,7 @@ void DawnTest::SwapBuffersForCapture() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void DawnTest::FlushWire() {
|
void DawnTest::FlushWire() {
|
||||||
if (gTestEnv->UseWire()) {
|
if (gTestEnv->UsesWire()) {
|
||||||
bool C2SFlushed = mC2sBuf->Flush();
|
bool C2SFlushed = mC2sBuf->Flush();
|
||||||
bool S2CFlushed = mS2cBuf->Flush();
|
bool S2CFlushed = mS2cBuf->Flush();
|
||||||
ASSERT(C2SFlushed);
|
ASSERT(C2SFlushed);
|
||||||
|
@ -106,7 +106,7 @@ class DawnTestEnvironment : public testing::Environment {
|
|||||||
|
|
||||||
void SetUp() override;
|
void SetUp() override;
|
||||||
|
|
||||||
bool UseWire() const;
|
bool UsesWire() const;
|
||||||
dawn_native::Instance* GetInstance() const;
|
dawn_native::Instance* GetInstance() const;
|
||||||
GLFWwindow* GetWindowForBackend(dawn_native::BackendType type) const;
|
GLFWwindow* GetWindowForBackend(dawn_native::BackendType type) const;
|
||||||
|
|
||||||
@ -146,6 +146,8 @@ class DawnTest : public ::testing::TestWithParam<DawnTestParam> {
|
|||||||
bool IsLinux() const;
|
bool IsLinux() const;
|
||||||
bool IsMacOS() const;
|
bool IsMacOS() const;
|
||||||
|
|
||||||
|
bool UsesWire() const;
|
||||||
|
|
||||||
void StartExpectDeviceError();
|
void StartExpectDeviceError();
|
||||||
bool EndExpectDeviceError();
|
bool EndExpectDeviceError();
|
||||||
|
|
||||||
|
51
src/tests/end2end/ObjectCachingTests.cpp
Normal file
51
src/tests/end2end/ObjectCachingTests.cpp
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
// Copyright 2019 The Dawn 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 "tests/DawnTest.h"
|
||||||
|
|
||||||
|
#include "utils/DawnHelpers.h"
|
||||||
|
|
||||||
|
class ObjectCachingTest : public DawnTest {};
|
||||||
|
|
||||||
|
// Test that BindGroupLayouts are correctly deduplicated.
|
||||||
|
TEST_P(ObjectCachingTest, BindGroupLayoutDeduplication) {
|
||||||
|
dawn::BindGroupLayout bgl = utils::MakeBindGroupLayout(
|
||||||
|
device, {{1, dawn::ShaderStageBit::Fragment, dawn::BindingType::UniformBuffer}});
|
||||||
|
dawn::BindGroupLayout sameBgl = utils::MakeBindGroupLayout(
|
||||||
|
device, {{1, dawn::ShaderStageBit::Fragment, dawn::BindingType::UniformBuffer}});
|
||||||
|
dawn::BindGroupLayout otherBgl = utils::MakeBindGroupLayout(
|
||||||
|
device, {{1, dawn::ShaderStageBit::Vertex, dawn::BindingType::UniformBuffer}});
|
||||||
|
|
||||||
|
EXPECT_NE(bgl.Get(), otherBgl.Get());
|
||||||
|
EXPECT_EQ(bgl.Get() == sameBgl.Get(), !UsesWire());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test that PipelineLayouts are correctly deduplicated.
|
||||||
|
TEST_P(ObjectCachingTest, PipelineLayoutDeduplication) {
|
||||||
|
dawn::BindGroupLayout bgl = utils::MakeBindGroupLayout(
|
||||||
|
device, {{1, dawn::ShaderStageBit::Fragment, dawn::BindingType::UniformBuffer}});
|
||||||
|
dawn::BindGroupLayout otherBgl = utils::MakeBindGroupLayout(
|
||||||
|
device, {{1, dawn::ShaderStageBit::Vertex, dawn::BindingType::UniformBuffer}});
|
||||||
|
|
||||||
|
dawn::PipelineLayout pl = utils::MakeBasicPipelineLayout(device, &bgl);
|
||||||
|
dawn::PipelineLayout samePl = utils::MakeBasicPipelineLayout(device, &bgl);
|
||||||
|
dawn::PipelineLayout otherPl1 = utils::MakeBasicPipelineLayout(device, nullptr);
|
||||||
|
dawn::PipelineLayout otherPl2 = utils::MakeBasicPipelineLayout(device, &otherBgl);
|
||||||
|
|
||||||
|
EXPECT_NE(pl.Get(), otherPl1.Get());
|
||||||
|
EXPECT_NE(pl.Get(), otherPl2.Get());
|
||||||
|
EXPECT_EQ(pl.Get() == samePl.Get(), !UsesWire());
|
||||||
|
}
|
||||||
|
|
||||||
|
DAWN_INSTANTIATE_TEST(ObjectCachingTest, D3D12Backend, MetalBackend, OpenGLBackend, VulkanBackend);
|
@ -278,7 +278,7 @@ namespace utils {
|
|||||||
dawn::PipelineLayout MakeBasicPipelineLayout(const dawn::Device& device,
|
dawn::PipelineLayout MakeBasicPipelineLayout(const dawn::Device& device,
|
||||||
const dawn::BindGroupLayout* bindGroupLayout) {
|
const dawn::BindGroupLayout* bindGroupLayout) {
|
||||||
dawn::PipelineLayoutDescriptor descriptor;
|
dawn::PipelineLayoutDescriptor descriptor;
|
||||||
if (bindGroupLayout) {
|
if (bindGroupLayout != nullptr) {
|
||||||
descriptor.bindGroupLayoutCount = 1;
|
descriptor.bindGroupLayoutCount = 1;
|
||||||
descriptor.bindGroupLayouts = bindGroupLayout;
|
descriptor.bindGroupLayouts = bindGroupLayout;
|
||||||
} else {
|
} else {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user