Flutter Engine
 
Loading...
Searching...
No Matches
pipeline_library_vk.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <cstdint>
8
17
18namespace impeller {
19
20PipelineLibraryVK::PipelineLibraryVK(
21 const std::shared_ptr<DeviceHolderVK>& device_holder,
22 std::shared_ptr<const Capabilities> caps,
23 fml::UniqueFD cache_directory,
24 std::shared_ptr<fml::ConcurrentTaskRunner> worker_task_runner)
25 : device_holder_(device_holder),
26 pso_cache_(std::make_shared<PipelineCacheVK>(std::move(caps),
27 device_holder,
28 std::move(cache_directory))),
29 worker_task_runner_(std::move(worker_task_runner)) {
30 FML_DCHECK(worker_task_runner_);
31 if (!pso_cache_->IsValid() || !worker_task_runner_) {
32 return;
33 }
34
35 is_valid_ = true;
36}
37
38PipelineLibraryVK::~PipelineLibraryVK() = default;
39
40// |PipelineLibrary|
41bool PipelineLibraryVK::IsValid() const {
42 return is_valid_;
43}
44
45std::unique_ptr<ComputePipelineVK> PipelineLibraryVK::CreateComputePipeline(
46 const ComputePipelineDescriptor& desc,
47 PipelineKey pipeline_key) {
48 TRACE_EVENT0("flutter", __FUNCTION__);
49 vk::ComputePipelineCreateInfo pipeline_info;
50
51 //----------------------------------------------------------------------------
52 /// Shader Stage
53 ///
54 const auto entrypoint = desc.GetStageEntrypoint();
55 if (!entrypoint) {
56 VALIDATION_LOG << "Compute shader is missing an entrypoint.";
57 return nullptr;
58 }
59
60 std::shared_ptr<DeviceHolderVK> strong_device = device_holder_.lock();
61 if (!strong_device) {
62 return nullptr;
63 }
64 auto device_properties = strong_device->GetPhysicalDevice().getProperties();
65 auto max_wg_size = device_properties.limits.maxComputeWorkGroupSize;
66
67 // Give all compute shaders a specialization constant entry for the
68 // workgroup/threadgroup size.
69 vk::SpecializationMapEntry specialization_map_entry[1];
70
71 uint32_t workgroup_size_x = max_wg_size[0];
72 specialization_map_entry[0].constantID = 0;
73 specialization_map_entry[0].offset = 0;
74 specialization_map_entry[0].size = sizeof(uint32_t);
75
76 vk::SpecializationInfo specialization_info;
77 specialization_info.mapEntryCount = 1;
78 specialization_info.pMapEntries = &specialization_map_entry[0];
79 specialization_info.dataSize = sizeof(uint32_t);
80 specialization_info.pData = &workgroup_size_x;
81
82 vk::PipelineShaderStageCreateInfo info;
83 info.setStage(vk::ShaderStageFlagBits::eCompute);
84 info.setPName("main");
85 info.setModule(ShaderFunctionVK::Cast(entrypoint.get())->GetModule());
86 info.setPSpecializationInfo(&specialization_info);
87 pipeline_info.setStage(info);
88
89 //----------------------------------------------------------------------------
90 /// Pipeline Layout a.k.a the descriptor sets and uniforms.
91 ///
92 std::vector<vk::DescriptorSetLayoutBinding> desc_bindings;
93
94 for (auto layout : desc.GetDescriptorSetLayouts()) {
95 auto vk_desc_layout = ToVKDescriptorSetLayoutBinding(layout);
96 desc_bindings.push_back(vk_desc_layout);
97 }
98
99 vk::DescriptorSetLayoutCreateInfo descs_layout_info;
100 descs_layout_info.setBindings(desc_bindings);
101
102 auto [descs_result, descs_layout] =
103 strong_device->GetDevice().createDescriptorSetLayoutUnique(
104 descs_layout_info);
105 if (descs_result != vk::Result::eSuccess) {
106 VALIDATION_LOG << "unable to create uniform descriptors";
107 return nullptr;
108 }
109
110 ContextVK::SetDebugName(strong_device->GetDevice(), descs_layout.get(),
111 "Descriptor Set Layout " + desc.GetLabel());
112
113 //----------------------------------------------------------------------------
114 /// Create the pipeline layout.
115 ///
116 vk::PipelineLayoutCreateInfo pipeline_layout_info;
117 pipeline_layout_info.setSetLayouts(descs_layout.get());
118 auto pipeline_layout = strong_device->GetDevice().createPipelineLayoutUnique(
119 pipeline_layout_info);
120 if (pipeline_layout.result != vk::Result::eSuccess) {
121 VALIDATION_LOG << "Could not create pipeline layout for pipeline "
122 << desc.GetLabel() << ": "
123 << vk::to_string(pipeline_layout.result);
124 return nullptr;
125 }
126 pipeline_info.setLayout(pipeline_layout.value.get());
127
128 //----------------------------------------------------------------------------
129 /// Finally, all done with the setup info. Create the pipeline itself.
130 ///
131 auto pipeline = pso_cache_->CreatePipeline(pipeline_info);
132 if (!pipeline) {
133 VALIDATION_LOG << "Could not create graphics pipeline: " << desc.GetLabel();
134 return nullptr;
135 }
136
137 ContextVK::SetDebugName(strong_device->GetDevice(), *pipeline_layout.value,
138 "Pipeline Layout " + desc.GetLabel());
139 ContextVK::SetDebugName(strong_device->GetDevice(), *pipeline,
140 "Pipeline " + desc.GetLabel());
141
142 return std::make_unique<ComputePipelineVK>(
143 device_holder_,
144 weak_from_this(), //
145 desc, //
146 std::move(pipeline), //
147 std::move(pipeline_layout.value), //
148 std::move(descs_layout), //
149 pipeline_key);
150}
151
152// |PipelineLibrary|
153PipelineFuture<PipelineDescriptor> PipelineLibraryVK::GetPipeline(
154 PipelineDescriptor descriptor,
155 bool async,
156 bool threadsafe) {
157 Lock lock(pipelines_mutex_);
158 if (auto found = pipelines_.find(descriptor); found != pipelines_.end()) {
159 return found->second;
160 }
161
162 cache_dirty_ = true;
163 if (!IsValid()) {
164 return {
165 descriptor,
166 RealizedFuture<std::shared_ptr<Pipeline<PipelineDescriptor>>>(nullptr)};
167 }
168
169 auto promise = std::make_shared<
170 NoExceptionPromise<std::shared_ptr<Pipeline<PipelineDescriptor>>>>();
171 auto pipeline_future =
172 PipelineFuture<PipelineDescriptor>{descriptor, promise->get_future()};
173 pipelines_[descriptor] = pipeline_future;
174
175 auto weak_this = weak_from_this();
176
177 PipelineKey next_key = pipeline_key_++;
178 auto generation_task = [descriptor, weak_this, promise, next_key]() {
179 auto thiz = weak_this.lock();
180 if (!thiz) {
181 promise->set_value(nullptr);
182 VALIDATION_LOG << "Pipeline library was collected before the pipeline "
183 "could be created.";
184 return;
185 }
186
187 promise->set_value(PipelineVK::Create(
188 descriptor, //
189 PipelineLibraryVK::Cast(*thiz).device_holder_.lock(), //
190 weak_this, //
191 next_key //
192 ));
193 };
194
195 if (async) {
196 worker_task_runner_->PostTask(generation_task);
197 } else {
198 generation_task();
199 }
200
201 return pipeline_future;
202}
203
204// |PipelineLibrary|
205PipelineFuture<ComputePipelineDescriptor> PipelineLibraryVK::GetPipeline(
206 ComputePipelineDescriptor descriptor,
207 bool async) {
208 Lock lock(pipelines_mutex_);
209 if (auto found = compute_pipelines_.find(descriptor);
210 found != compute_pipelines_.end()) {
211 return found->second;
212 }
213
214 cache_dirty_ = true;
215 if (!IsValid()) {
216 return {
217 descriptor,
218 RealizedFuture<std::shared_ptr<Pipeline<ComputePipelineDescriptor>>>(
219 nullptr)};
220 }
221
222 auto promise = std::make_shared<
223 std::promise<std::shared_ptr<Pipeline<ComputePipelineDescriptor>>>>();
224 auto pipeline_future = PipelineFuture<ComputePipelineDescriptor>{
225 descriptor, promise->get_future()};
226 compute_pipelines_[descriptor] = pipeline_future;
227
228 auto weak_this = weak_from_this();
229
230 PipelineKey next_key = pipeline_key_++;
231 auto generation_task = [descriptor, weak_this, promise, next_key]() {
232 auto self = weak_this.lock();
233 if (!self) {
234 promise->set_value(nullptr);
235 VALIDATION_LOG << "Pipeline library was collected before the pipeline "
236 "could be created.";
237 return;
238 }
239
240 auto pipeline = PipelineLibraryVK::Cast(*self).CreateComputePipeline(
241 descriptor, next_key);
242 if (!pipeline) {
243 promise->set_value(nullptr);
244 VALIDATION_LOG << "Could not create pipeline: " << descriptor.GetLabel();
245 return;
246 }
247
248 promise->set_value(std::move(pipeline));
249 };
250
251 if (async) {
252 worker_task_runner_->PostTask(generation_task);
253 } else {
254 generation_task();
255 }
256
257 return pipeline_future;
258}
259
260// |PipelineLibrary|
261bool PipelineLibraryVK::HasPipeline(const PipelineDescriptor& descriptor) {
262 Lock lock(pipelines_mutex_);
263 return pipelines_.find(descriptor) != pipelines_.end();
264}
265
266// |PipelineLibrary|
267void PipelineLibraryVK::RemovePipelinesWithEntryPoint(
268 std::shared_ptr<const ShaderFunction> function) {
269 Lock lock(pipelines_mutex_);
270
271 fml::erase_if(pipelines_, [&](auto item) {
272 return item->first.GetEntrypointForStage(function->GetStage())
273 ->IsEqual(*function);
274 });
275}
276
277void PipelineLibraryVK::DidAcquireSurfaceFrame() {
278 if (++frames_acquired_ == 50u) {
279 if (cache_dirty_) {
280 cache_dirty_ = false;
281 PersistPipelineCacheToDisk();
282 }
283 frames_acquired_ = 0;
284 }
285}
286
287void PipelineLibraryVK::PersistPipelineCacheToDisk() {
288 worker_task_runner_->PostTask(
289 [weak_cache = decltype(pso_cache_)::weak_type(pso_cache_)]() {
290 auto cache = weak_cache.lock();
291 if (!cache) {
292 return;
293 }
294 cache->PersistCacheToDisk();
295 });
296}
297
298const std::shared_ptr<PipelineCacheVK>& PipelineLibraryVK::GetPSOCache() const {
299 return pso_cache_;
300}
301
302const std::shared_ptr<fml::ConcurrentTaskRunner>&
303PipelineLibraryVK::GetWorkerTaskRunner() const {
304 return worker_task_runner_;
305}
306
307} // namespace impeller
std::vector< std::pair< uint64_t, std::unique_ptr< GenericRenderPipelineHandle > > > pipelines_
#define FML_DCHECK(condition)
Definition logging.h:122
Dart_NativeFunction function
Definition fuchsia.cc:50
void erase_if(Collection &container, const std::function< bool(typename Collection::iterator)> &predicate)
Definition container.h:16
constexpr vk::DescriptorSetLayoutBinding ToVKDescriptorSetLayoutBinding(const DescriptorSetLayout &layout)
Definition formats_vk.h:296
int64_t PipelineKey
Definition pipeline.h:21
Definition ref_ptr.h:261
#define TRACE_EVENT0(category_group, name)
#define VALIDATION_LOG
Definition validation.h:91