blob: 840f6bd85935f0e201f42f7f435e2cdb3bd5824b [file] [log] [blame]
Jamie Madill1f46bc12018-02-20 16:09:43 -05001//
2// Copyright 2017 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6// CommandGraph:
7// Deferred work constructed by GL calls, that will later be flushed to Vulkan.
8//
9
10#ifndef LIBANGLE_RENDERER_VULKAN_COMMAND_GRAPH_H_
11#define LIBANGLE_RENDERER_VULKAN_COMMAND_GRAPH_H_
12
13#include "libANGLE/renderer/vulkan/vk_cache_utils.h"
14
15namespace rx
16{
17
18namespace vk
19{
Jamie Madill3d61ac22018-08-28 16:58:55 -040020enum class VisitedState
21{
22 Unvisited,
23 Ready,
24 Visited,
25};
26
Jamie Madill0da73fe2018-10-02 09:31:39 -040027enum class CommandGraphResourceType
28{
29 Buffer,
30 Framebuffer,
31 Image,
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -040032 Query,
33};
34
35// Certain functionality cannot be put in secondary command buffers, so they are special-cased in
36// the node.
37enum class CommandGraphNodeFunction
38{
39 Generic,
40 BeginQuery,
41 EndQuery,
Shahbaz Youssefic2b576d2018-10-12 14:45:34 -040042 WriteTimestamp,
Jamie Madill0da73fe2018-10-02 09:31:39 -040043};
44
Jamie Madill85ca1892019-01-16 13:27:15 -050045// Receives notifications when a command buffer is no longer able to record. Can be used with
46// inheritance. Faster than using an interface class since it has inlined methods. Could be used
47// with composition by adding a getCommandBuffer method.
48class CommandBufferOwner
49{
50 public:
51 CommandBufferOwner() = default;
52 virtual ~CommandBufferOwner() {}
53
54 ANGLE_INLINE void onCommandBufferFinished() { mCommandBuffer = nullptr; }
55
56 protected:
57 vk::CommandBuffer *mCommandBuffer = nullptr;
58};
59
Jamie Madill3d61ac22018-08-28 16:58:55 -040060// Only used internally in the command graph. Kept in the header for better inlining performance.
61class CommandGraphNode final : angle::NonCopyable
62{
63 public:
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -040064 CommandGraphNode(CommandGraphNodeFunction function);
Jamie Madill3d61ac22018-08-28 16:58:55 -040065 ~CommandGraphNode();
66
67 // Immutable queries for when we're walking the commands tree.
68 CommandBuffer *getOutsideRenderPassCommands();
69
70 CommandBuffer *getInsideRenderPassCommands()
71 {
72 ASSERT(!mHasChildren);
73 return &mInsideRenderPassCommands;
74 }
75
76 // For outside the render pass (copies, transitions, etc).
77 angle::Result beginOutsideRenderPassRecording(Context *context,
78 const CommandPool &commandPool,
79 CommandBuffer **commandsOut);
80
81 // For rendering commands (draws).
82 angle::Result beginInsideRenderPassRecording(Context *context, CommandBuffer **commandsOut);
83
84 // storeRenderPassInfo and append*RenderTarget store info relevant to the RenderPass.
85 void storeRenderPassInfo(const Framebuffer &framebuffer,
86 const gl::Rectangle renderArea,
87 const vk::RenderPassDesc &renderPassDesc,
88 const std::vector<VkClearValue> &clearValues);
89
90 // Dependency commands order node execution in the command graph.
91 // Once a node has commands that must happen after it, recording is stopped and the node is
92 // frozen forever.
93 static void SetHappensBeforeDependency(CommandGraphNode *beforeNode,
Jamie Madillc759b8b2019-01-03 15:16:50 -050094 CommandGraphNode *afterNode)
95 {
96 ASSERT(beforeNode != afterNode && !beforeNode->isChildOf(afterNode));
97 afterNode->mParents.emplace_back(beforeNode);
98 beforeNode->setHasChildren();
99 }
100
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400101 static void SetHappensBeforeDependencies(CommandGraphNode **beforeNodes,
102 size_t beforeNodesCount,
Jamie Madill3d61ac22018-08-28 16:58:55 -0400103 CommandGraphNode *afterNode);
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400104 static void SetHappensBeforeDependencies(CommandGraphNode *beforeNode,
105 CommandGraphNode **afterNodes,
106 size_t afterNodesCount);
Jamie Madill3d61ac22018-08-28 16:58:55 -0400107 bool hasParents() const;
108 bool hasChildren() const { return mHasChildren; }
109
110 // Commands for traversing the node on a flush operation.
111 VisitedState visitedState() const;
112 void visitParents(std::vector<CommandGraphNode *> *stack);
113 angle::Result visitAndExecute(Context *context,
114 Serial serial,
115 RenderPassCache *renderPassCache,
116 CommandBuffer *primaryCommandBuffer);
117
Jamie Madill0da73fe2018-10-02 09:31:39 -0400118 // Only used in the command graph diagnostics.
119 const std::vector<CommandGraphNode *> &getParentsForDiagnostics() const;
120 void setDiagnosticInfo(CommandGraphResourceType resourceType, uintptr_t resourceID);
121
122 CommandGraphResourceType getResourceTypeForDiagnostics() const { return mResourceType; }
123 uintptr_t getResourceIDForDiagnostics() const { return mResourceID; }
124
Jamie Madill3d61ac22018-08-28 16:58:55 -0400125 const gl::Rectangle &getRenderPassRenderArea() const;
126
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400127 CommandGraphNodeFunction getFunction() const { return mFunction; }
128
129 void setQueryPool(const QueryPool *queryPool, uint32_t queryIndex);
130
Jamie Madillc759b8b2019-01-03 15:16:50 -0500131 ANGLE_INLINE void addGlobalMemoryBarrier(VkFlags srcAccess, VkFlags dstAccess)
132 {
133 mGlobalMemoryBarrierSrcAccess |= srcAccess;
134 mGlobalMemoryBarrierDstAccess |= dstAccess;
135 }
Jamie Madill03d1a5e2018-11-12 11:34:24 -0500136
Jamie Madill85ca1892019-01-16 13:27:15 -0500137 // This can only be set for RenderPass nodes. Each RenderPass node can have at most one owner.
138 void setCommandBufferOwner(CommandBufferOwner *owner)
139 {
140 ASSERT(mCommandBufferOwner == nullptr);
141 mCommandBufferOwner = owner;
142 }
143
Jamie Madill3d61ac22018-08-28 16:58:55 -0400144 private:
Jamie Madill85ca1892019-01-16 13:27:15 -0500145 ANGLE_INLINE void setHasChildren()
146 {
147 mHasChildren = true;
148 if (mCommandBufferOwner)
149 {
150 mCommandBufferOwner->onCommandBufferFinished();
151 }
152 }
Jamie Madill3d61ac22018-08-28 16:58:55 -0400153
154 // Used for testing only.
155 bool isChildOf(CommandGraphNode *parent);
156
157 // Only used if we need a RenderPass for these commands.
158 RenderPassDesc mRenderPassDesc;
159 Framebuffer mRenderPassFramebuffer;
160 gl::Rectangle mRenderPassRenderArea;
161 gl::AttachmentArray<VkClearValue> mRenderPassClearValues;
162
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400163 CommandGraphNodeFunction mFunction;
164
165 // Keep separate buffers for commands inside and outside a RenderPass.
Jamie Madill3d61ac22018-08-28 16:58:55 -0400166 // TODO(jmadill): We might not need inside and outside RenderPass commands separate.
167 CommandBuffer mOutsideRenderPassCommands;
168 CommandBuffer mInsideRenderPassCommands;
169
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400170 // Special-function additional data:
171 VkQueryPool mQueryPool;
172 uint32_t mQueryIndex;
173
Jamie Madill3d61ac22018-08-28 16:58:55 -0400174 // Parents are commands that must be submitted before 'this' CommandNode can be submitted.
175 std::vector<CommandGraphNode *> mParents;
176
177 // If this is true, other commands exist that must be submitted after 'this' command.
178 bool mHasChildren;
179
180 // Used when traversing the dependency graph.
181 VisitedState mVisitedState;
Jamie Madill0da73fe2018-10-02 09:31:39 -0400182
183 // Additional diagnostic information.
184 CommandGraphResourceType mResourceType;
185 uintptr_t mResourceID;
Jamie Madill03d1a5e2018-11-12 11:34:24 -0500186
187 // For global memory barriers.
188 VkFlags mGlobalMemoryBarrierSrcAccess;
189 VkFlags mGlobalMemoryBarrierDstAccess;
Jamie Madill85ca1892019-01-16 13:27:15 -0500190
191 // Command buffer notifications.
192 CommandBufferOwner *mCommandBufferOwner;
Jamie Madill3d61ac22018-08-28 16:58:55 -0400193};
Jamie Madill1f46bc12018-02-20 16:09:43 -0500194
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400195// This is a helper class for back-end objects used in Vk command buffers. It records a serial
196// at command recording times indicating an order in the queue. We use Fences to detect when
197// commands finish, and then release any unreferenced and deleted resources based on the stored
198// queue serial in a special 'garbage' queue. Resources also track current read and write
199// dependencies. Only one command buffer node can be writing to the Resource at a time, but many
200// can be reading from it. Together the dependencies will form a command graph at submission time.
Jamie Madill0da73fe2018-10-02 09:31:39 -0400201class CommandGraphResource : angle::NonCopyable
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400202{
203 public:
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400204 virtual ~CommandGraphResource();
205
Jamie Madillc57ee252018-05-30 19:53:48 -0400206 // Returns true if the resource is in use by the renderer.
207 bool isResourceInUse(RendererVk *renderer) const;
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400208
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400209 // Returns true if the resource has unsubmitted work pending.
210 bool hasPendingWork(RendererVk *renderer) const;
211
Jamie Madill193a2842018-10-30 17:28:41 -0400212 // Get the current queue serial for this resource. Used to release resources, and for
213 // queries, to know if the queue they are submitted on has finished execution.
Jamie Madillc759b8b2019-01-03 15:16:50 -0500214 Serial getStoredQueueSerial() const { return mStoredQueueSerial; }
Jamie Madill193a2842018-10-30 17:28:41 -0400215
216 protected:
217 explicit CommandGraphResource(CommandGraphResourceType resourceType);
218
219 Serial mStoredQueueSerial;
220
221 // Additional diagnostic information.
222 CommandGraphResourceType mResourceType;
223
224 // Current command graph writing node.
225 CommandGraphNode *mCurrentWritingNode;
226};
227
228// Subclass of graph resources that can record command buffers. Images/Buffers/Framebuffers.
229// Does not include Query graph resources.
230class RecordableGraphResource : public CommandGraphResource
231{
232 public:
233 ~RecordableGraphResource() override;
234
Jamie Madilld014c9e2018-05-18 15:15:59 -0400235 // Sets up dependency relations. 'this' resource is the resource being written to.
Jamie Madill193a2842018-10-30 17:28:41 -0400236 void addWriteDependency(RecordableGraphResource *writingResource);
Jamie Madilld014c9e2018-05-18 15:15:59 -0400237
238 // Sets up dependency relations. 'this' resource is the resource being read.
Jamie Madill193a2842018-10-30 17:28:41 -0400239 void addReadDependency(RecordableGraphResource *readingResource);
Jamie Madilld014c9e2018-05-18 15:15:59 -0400240
Shahbaz Youssefi254b32c2018-11-26 11:58:03 -0500241 // Updates the in-use serial tracked for this resource. Will clear dependencies if the resource
242 // was not used in this set of command nodes.
Jamie Madillc759b8b2019-01-03 15:16:50 -0500243 ANGLE_INLINE void updateQueueSerial(Serial queueSerial)
244 {
245 ASSERT(queueSerial >= mStoredQueueSerial);
246
247 if (queueSerial > mStoredQueueSerial)
248 {
249 mCurrentWritingNode = nullptr;
250 mCurrentReadingNodes.clear();
251 mStoredQueueSerial = queueSerial;
252 }
253 }
Shahbaz Youssefi254b32c2018-11-26 11:58:03 -0500254
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400255 // Allocates a write node via getNewWriteNode and returns a started command buffer.
256 // The started command buffer will render outside of a RenderPass.
Jamie Madille2d22702018-09-19 08:11:48 -0400257 // Will append to an existing command buffer/graph node if possible.
258 angle::Result recordCommands(Context *context, CommandBuffer **commandBufferOut);
Jamie Madill316c6062018-05-29 10:49:45 -0400259
260 // Begins a command buffer on the current graph node for in-RenderPass rendering.
Shahbaz Youssefif83a28a2018-12-09 03:48:34 +0100261 // Called from FramebufferVk::startNewRenderPass and UtilsVk functions.
Jamie Madill85ca1892019-01-16 13:27:15 -0500262 angle::Result beginRenderPass(ContextVk *contextVk,
Jamie Madill21061022018-07-12 23:56:30 -0400263 const Framebuffer &framebuffer,
264 const gl::Rectangle &renderArea,
265 const RenderPassDesc &renderPassDesc,
266 const std::vector<VkClearValue> &clearValues,
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400267 CommandBuffer **commandBufferOut);
268
Jamie Madill5dca6512018-05-30 10:53:51 -0400269 // Checks if we're in a RenderPass, returning true if so. Updates serial internally.
270 // Returns the started command buffer in commandBufferOut.
Jamie Madillc759b8b2019-01-03 15:16:50 -0500271 ANGLE_INLINE bool appendToStartedRenderPass(Serial currentQueueSerial,
272 CommandBuffer **commandBufferOut)
273 {
274 updateQueueSerial(currentQueueSerial);
275 if (hasStartedRenderPass())
276 {
277 *commandBufferOut = mCurrentWritingNode->getInsideRenderPassCommands();
278 return true;
279 }
280 else
281 {
282 return false;
283 }
284 }
Jamie Madill316c6062018-05-29 10:49:45 -0400285
286 // Accessor for RenderPass RenderArea.
287 const gl::Rectangle &getRenderPassRenderArea() const;
288
289 // Called when 'this' object changes, but we'd like to start a new command buffer later.
Jamie Madille2d22702018-09-19 08:11:48 -0400290 void finishCurrentCommands(RendererVk *renderer);
Jamie Madill316c6062018-05-29 10:49:45 -0400291
Jamie Madill03d1a5e2018-11-12 11:34:24 -0500292 // Store a deferred memory barrier. Will be recorded into a primary command buffer at submit.
293 void addGlobalMemoryBarrier(VkFlags srcAccess, VkFlags dstAccess)
294 {
295 ASSERT(mCurrentWritingNode);
296 mCurrentWritingNode->addGlobalMemoryBarrier(srcAccess, dstAccess);
297 }
298
Jamie Madill2d03ff42018-09-27 15:04:26 -0400299 protected:
Jamie Madill193a2842018-10-30 17:28:41 -0400300 explicit RecordableGraphResource(CommandGraphResourceType resourceType);
Jamie Madill0da73fe2018-10-02 09:31:39 -0400301
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400302 private:
Jamie Madill316c6062018-05-29 10:49:45 -0400303 // Returns true if this node has a current writing node with no children.
Jamie Madillc759b8b2019-01-03 15:16:50 -0500304 ANGLE_INLINE bool hasChildlessWritingNode() const
Jamie Madill3d61ac22018-08-28 16:58:55 -0400305 {
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400306 // Note: currently, we don't have a resource that can issue both generic and special
307 // commands. We don't create read/write dependencies between mixed generic/special
308 // resources either. As such, we expect the function to always be generic here. If such a
309 // resource is added in the future, this can add a check for function == generic and fail if
310 // false.
311 ASSERT(mCurrentWritingNode == nullptr ||
312 mCurrentWritingNode->getFunction() == CommandGraphNodeFunction::Generic);
Jamie Madill3d61ac22018-08-28 16:58:55 -0400313 return (mCurrentWritingNode != nullptr && !mCurrentWritingNode->hasChildren());
314 }
Jamie Madill316c6062018-05-29 10:49:45 -0400315
Jamie Madill5dca6512018-05-30 10:53:51 -0400316 // Checks if we're in a RenderPass without children.
Jamie Madill3d61ac22018-08-28 16:58:55 -0400317 bool hasStartedRenderPass() const
318 {
319 return hasChildlessWritingNode() &&
320 mCurrentWritingNode->getInsideRenderPassCommands()->valid();
321 }
Jamie Madill5dca6512018-05-30 10:53:51 -0400322
Jamie Madill193a2842018-10-30 17:28:41 -0400323 void startNewCommands(RendererVk *renderer);
Jamie Madill0da73fe2018-10-02 09:31:39 -0400324
Jamie Madill193a2842018-10-30 17:28:41 -0400325 void onWriteImpl(CommandGraphNode *writingNode, Serial currentSerial);
326
327 std::vector<CommandGraphNode *> mCurrentReadingNodes;
328};
329
330// Specialized command graph node for queries. Not for use with any exposed command buffers.
331class QueryGraphResource : public CommandGraphResource
332{
333 public:
334 ~QueryGraphResource() override;
335
336 void beginQuery(Context *context, const QueryPool *queryPool, uint32_t queryIndex);
337 void endQuery(Context *context, const QueryPool *queryPool, uint32_t queryIndex);
338 void writeTimestamp(Context *context, const QueryPool *queryPool, uint32_t queryIndex);
339
340 protected:
341 QueryGraphResource();
342
343 private:
344 void startNewCommands(RendererVk *renderer, CommandGraphNodeFunction function);
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400345};
346
Jamie Madill1f46bc12018-02-20 16:09:43 -0500347// Translating OpenGL commands into Vulkan and submitting them immediately loses out on some
348// of the powerful flexiblity Vulkan offers in RenderPasses. Load/Store ops can automatically
349// clear RenderPass attachments, or preserve the contents. RenderPass automatic layout transitions
350// can improve certain performance cases. Also, we can remove redundant RenderPass Begin and Ends
351// when processing interleaved draw operations on independent Framebuffers.
352//
353// ANGLE's CommandGraph (and CommandGraphNode) attempt to solve these problems using deferred
354// command submission. We also sometimes call this command re-ordering. A brief summary:
355//
356// During GL command processing, we record Vulkan commands into secondary command buffers, which
357// are stored in CommandGraphNodes, and these nodes are chained together via dependencies to
358// for a directed acyclic CommandGraph. When we need to submit the CommandGraph, say during a
359// SwapBuffers or ReadPixels call, we begin a primary Vulkan CommandBuffer, and walk the
360// CommandGraph, starting at the most senior nodes, recording secondary CommandBuffers inside
Jamie Madill6c7ab7f2018-03-31 14:19:15 -0400361// and outside RenderPasses as necessary, filled with the right load/store operations. Once
Jamie Madill1f46bc12018-02-20 16:09:43 -0500362// the primary CommandBuffer has recorded all of the secondary CommandBuffers from all the open
363// CommandGraphNodes, we submit the primary CommandBuffer to the VkQueue on the device.
Jamie Madilla5e06072018-05-18 14:36:05 -0400364//
Jamie Madill1f46bc12018-02-20 16:09:43 -0500365// The Command Graph consists of an array of open Command Graph Nodes. It supports allocating new
366// nodes for the graph, which are linked via dependency relation calls in CommandGraphNode, and
367// also submitting the whole command graph via submitCommands.
368class CommandGraph final : angle::NonCopyable
369{
370 public:
Jamie Madill0da73fe2018-10-02 09:31:39 -0400371 explicit CommandGraph(bool enableGraphDiagnostics);
Jamie Madill1f46bc12018-02-20 16:09:43 -0500372 ~CommandGraph();
373
374 // Allocates a new CommandGraphNode and adds it to the list of current open nodes. No ordering
375 // relations exist in the node by default. Call CommandGraphNode::SetHappensBeforeDependency
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400376 // to set up dependency relations. If the node is a barrier, it will automatically add
377 // dependencies between the previous barrier, the new barrier and all nodes in between.
Jamie Madill193a2842018-10-30 17:28:41 -0400378 CommandGraphNode *allocateNode(CommandGraphNodeFunction function);
Jamie Madill1f46bc12018-02-20 16:09:43 -0500379
Jamie Madill21061022018-07-12 23:56:30 -0400380 angle::Result submitCommands(Context *context,
381 Serial serial,
382 RenderPassCache *renderPassCache,
383 CommandPool *commandPool,
384 CommandBuffer *primaryCommandBufferOut);
Jamie Madill1f46bc12018-02-20 16:09:43 -0500385 bool empty() const;
Yuly Novikovb56ddbb2018-11-02 16:53:18 -0400386 void clear();
Jamie Madill1f46bc12018-02-20 16:09:43 -0500387
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400388 CommandGraphNode *getLastBarrierNode(size_t *indexOut);
389
Jamie Madill193a2842018-10-30 17:28:41 -0400390 void setNewBarrier(CommandGraphNode *newBarrier);
391
Jamie Madill1f46bc12018-02-20 16:09:43 -0500392 private:
Jamie Madill0da73fe2018-10-02 09:31:39 -0400393 void dumpGraphDotFile(std::ostream &out) const;
Jamie Madill1f46bc12018-02-20 16:09:43 -0500394
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400395 void addDependenciesToNextBarrier(size_t begin, size_t end, CommandGraphNode *nextBarrier);
396
Jamie Madill0da73fe2018-10-02 09:31:39 -0400397 std::vector<CommandGraphNode *> mNodes;
398 bool mEnableGraphDiagnostics;
Shahbaz Youssefi563fbaa2018-10-02 11:22:01 -0400399
400 // A set of nodes (eventually) exist that act as barriers to guarantee submission order. For
401 // example, a glMemoryBarrier() calls would lead to such a barrier or beginning and ending a
402 // query. This is because the graph can reorder operations if it sees fit. Let's call a barrier
403 // node Bi, and the other nodes Ni. The edges between Ni don't interest us. Before a barrier is
404 // inserted, we have:
405 //
406 // N0 N1 ... Na
407 // \___\__/_/ (dependency egdes, which we don't care about so I'll stop drawing them.
408 // \/
409 //
410 // When the first barrier is inserted, we will have:
411 //
412 // ______
413 // / ____\
414 // / / \
415 // / / /\
416 // N0 N1 ... Na B0
417 //
418 // This makes sure all N0..Na are called before B0. From then on, B0 will be the current
419 // "barrier point" which extends an edge to every next node:
420 //
421 // ______
422 // / ____\
423 // / / \
424 // / / /\
425 // N0 N1 ... Na B0 Na+1 ... Nb
426 // \/ /
427 // \______/
428 //
429 //
430 // When the next barrier B1 is met, all nodes between B0 and B1 will add a depenency on B1 as
431 // well, and the "barrier point" is updated.
432 //
433 // ______
434 // / ____\ ______ ______
435 // / / \ / \ / \
436 // / / /\ / /\ / /\
437 // N0 N1 ... Na B0 Na+1 ... Nb B1 Nb+1 ... Nc B2 ...
438 // \/ / / \/ / /
439 // \______/ / \______/ /
440 // \_______/ \_______/
441 //
442 //
443 // When barrier Bi is introduced, all nodes added since Bi-1 need to add a dependency to Bi
444 // (including Bi-1). We therefore keep track of the node index of the last barrier that was
445 // issued.
446 static constexpr size_t kInvalidNodeIndex = std::numeric_limits<std::size_t>::max();
447 size_t mLastBarrierIndex;
Jamie Madill0da73fe2018-10-02 09:31:39 -0400448};
Jamie Madill1f46bc12018-02-20 16:09:43 -0500449} // namespace vk
450} // namespace rx
451
452#endif // LIBANGLE_RENDERER_VULKAN_COMMAND_GRAPH_H_