xref: /aosp_15_r20/external/angle/src/libANGLE/renderer/vulkan/CLEventVk.cpp (revision 8975f5c5ed3d1c378011245431ada316dfb6f244)
1*8975f5c5SAndroid Build Coastguard Worker //
2*8975f5c5SAndroid Build Coastguard Worker // Copyright 2021 The ANGLE Project Authors. All rights reserved.
3*8975f5c5SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
4*8975f5c5SAndroid Build Coastguard Worker // found in the LICENSE file.
5*8975f5c5SAndroid Build Coastguard Worker //
6*8975f5c5SAndroid Build Coastguard Worker // CLEventVk.cpp: Implements the class methods for CLEventVk.
7*8975f5c5SAndroid Build Coastguard Worker 
8*8975f5c5SAndroid Build Coastguard Worker #include "libANGLE/renderer/vulkan/CLEventVk.h"
9*8975f5c5SAndroid Build Coastguard Worker #include "libANGLE/renderer/vulkan/CLCommandQueueVk.h"
10*8975f5c5SAndroid Build Coastguard Worker 
11*8975f5c5SAndroid Build Coastguard Worker #include "libANGLE/cl_utils.h"
12*8975f5c5SAndroid Build Coastguard Worker 
13*8975f5c5SAndroid Build Coastguard Worker namespace rx
14*8975f5c5SAndroid Build Coastguard Worker {
15*8975f5c5SAndroid Build Coastguard Worker 
CLEventVk(const cl::Event & event)16*8975f5c5SAndroid Build Coastguard Worker CLEventVk::CLEventVk(const cl::Event &event)
17*8975f5c5SAndroid Build Coastguard Worker     : CLEventImpl(event),
18*8975f5c5SAndroid Build Coastguard Worker       mStatus(isUserEvent() ? CL_SUBMITTED : CL_QUEUED),
19*8975f5c5SAndroid Build Coastguard Worker       mProfilingTimestamps(ProfilingTimestamps{})
20*8975f5c5SAndroid Build Coastguard Worker {
21*8975f5c5SAndroid Build Coastguard Worker     ANGLE_CL_IMPL_TRY(setTimestamp(*mStatus));
22*8975f5c5SAndroid Build Coastguard Worker }
23*8975f5c5SAndroid Build Coastguard Worker 
~CLEventVk()24*8975f5c5SAndroid Build Coastguard Worker CLEventVk::~CLEventVk() {}
25*8975f5c5SAndroid Build Coastguard Worker 
getCommandExecutionStatus(cl_int & executionStatus)26*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::getCommandExecutionStatus(cl_int &executionStatus)
27*8975f5c5SAndroid Build Coastguard Worker {
28*8975f5c5SAndroid Build Coastguard Worker     executionStatus = *mStatus;
29*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
30*8975f5c5SAndroid Build Coastguard Worker }
31*8975f5c5SAndroid Build Coastguard Worker 
setUserEventStatus(cl_int executionStatus)32*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::setUserEventStatus(cl_int executionStatus)
33*8975f5c5SAndroid Build Coastguard Worker {
34*8975f5c5SAndroid Build Coastguard Worker     ASSERT(isUserEvent());
35*8975f5c5SAndroid Build Coastguard Worker 
36*8975f5c5SAndroid Build Coastguard Worker     // Not much to do here other than storing the user supplied state.
37*8975f5c5SAndroid Build Coastguard Worker     // Error checking and single call enforcement is responsibility of the front end.
38*8975f5c5SAndroid Build Coastguard Worker     ANGLE_TRY(setStatusAndExecuteCallback(executionStatus));
39*8975f5c5SAndroid Build Coastguard Worker 
40*8975f5c5SAndroid Build Coastguard Worker     // User event set and callback(s) finished - notify those waiting
41*8975f5c5SAndroid Build Coastguard Worker     mUserEventCondition.notify_all();
42*8975f5c5SAndroid Build Coastguard Worker 
43*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
44*8975f5c5SAndroid Build Coastguard Worker }
45*8975f5c5SAndroid Build Coastguard Worker 
setCallback(cl::Event & event,cl_int commandExecCallbackType)46*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::setCallback(cl::Event &event, cl_int commandExecCallbackType)
47*8975f5c5SAndroid Build Coastguard Worker {
48*8975f5c5SAndroid Build Coastguard Worker     ASSERT(commandExecCallbackType >= CL_COMPLETE);
49*8975f5c5SAndroid Build Coastguard Worker     ASSERT(commandExecCallbackType < CL_QUEUED);
50*8975f5c5SAndroid Build Coastguard Worker 
51*8975f5c5SAndroid Build Coastguard Worker     // Not much to do, acknowledge the presence of callback and returns
52*8975f5c5SAndroid Build Coastguard Worker     mHaveCallbacks->at(commandExecCallbackType) = true;
53*8975f5c5SAndroid Build Coastguard Worker 
54*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
55*8975f5c5SAndroid Build Coastguard Worker }
56*8975f5c5SAndroid Build Coastguard Worker 
getProfilingInfo(cl::ProfilingInfo name,size_t valueSize,void * value,size_t * valueSizeRet)57*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::getProfilingInfo(cl::ProfilingInfo name,
58*8975f5c5SAndroid Build Coastguard Worker                                           size_t valueSize,
59*8975f5c5SAndroid Build Coastguard Worker                                           void *value,
60*8975f5c5SAndroid Build Coastguard Worker                                           size_t *valueSizeRet)
61*8975f5c5SAndroid Build Coastguard Worker {
62*8975f5c5SAndroid Build Coastguard Worker     cl_ulong valueUlong   = 0;
63*8975f5c5SAndroid Build Coastguard Worker     size_t copySize       = 0;
64*8975f5c5SAndroid Build Coastguard Worker     const void *copyValue = nullptr;
65*8975f5c5SAndroid Build Coastguard Worker 
66*8975f5c5SAndroid Build Coastguard Worker     auto profilingTimestamps = mProfilingTimestamps.synchronize();
67*8975f5c5SAndroid Build Coastguard Worker 
68*8975f5c5SAndroid Build Coastguard Worker     switch (name)
69*8975f5c5SAndroid Build Coastguard Worker     {
70*8975f5c5SAndroid Build Coastguard Worker         case cl::ProfilingInfo::CommandQueued:
71*8975f5c5SAndroid Build Coastguard Worker             valueUlong = profilingTimestamps->commandQueuedTS;
72*8975f5c5SAndroid Build Coastguard Worker             break;
73*8975f5c5SAndroid Build Coastguard Worker         case cl::ProfilingInfo::CommandSubmit:
74*8975f5c5SAndroid Build Coastguard Worker             valueUlong = profilingTimestamps->commandSubmitTS;
75*8975f5c5SAndroid Build Coastguard Worker             break;
76*8975f5c5SAndroid Build Coastguard Worker         case cl::ProfilingInfo::CommandStart:
77*8975f5c5SAndroid Build Coastguard Worker             valueUlong = profilingTimestamps->commandStartTS;
78*8975f5c5SAndroid Build Coastguard Worker             break;
79*8975f5c5SAndroid Build Coastguard Worker         case cl::ProfilingInfo::CommandEnd:
80*8975f5c5SAndroid Build Coastguard Worker             valueUlong = profilingTimestamps->commandEndTS;
81*8975f5c5SAndroid Build Coastguard Worker             break;
82*8975f5c5SAndroid Build Coastguard Worker         case cl::ProfilingInfo::CommandComplete:
83*8975f5c5SAndroid Build Coastguard Worker             valueUlong = profilingTimestamps->commandCompleteTS;
84*8975f5c5SAndroid Build Coastguard Worker             break;
85*8975f5c5SAndroid Build Coastguard Worker         default:
86*8975f5c5SAndroid Build Coastguard Worker             UNREACHABLE();
87*8975f5c5SAndroid Build Coastguard Worker     }
88*8975f5c5SAndroid Build Coastguard Worker     copyValue = &valueUlong;
89*8975f5c5SAndroid Build Coastguard Worker     copySize  = sizeof(valueUlong);
90*8975f5c5SAndroid Build Coastguard Worker 
91*8975f5c5SAndroid Build Coastguard Worker     if ((value != nullptr) && (copyValue != nullptr))
92*8975f5c5SAndroid Build Coastguard Worker     {
93*8975f5c5SAndroid Build Coastguard Worker         memcpy(value, copyValue, std::min(valueSize, copySize));
94*8975f5c5SAndroid Build Coastguard Worker     }
95*8975f5c5SAndroid Build Coastguard Worker 
96*8975f5c5SAndroid Build Coastguard Worker     if (valueSizeRet != nullptr)
97*8975f5c5SAndroid Build Coastguard Worker     {
98*8975f5c5SAndroid Build Coastguard Worker         *valueSizeRet = copySize;
99*8975f5c5SAndroid Build Coastguard Worker     }
100*8975f5c5SAndroid Build Coastguard Worker 
101*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
102*8975f5c5SAndroid Build Coastguard Worker }
103*8975f5c5SAndroid Build Coastguard Worker 
waitForUserEventStatus()104*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::waitForUserEventStatus()
105*8975f5c5SAndroid Build Coastguard Worker {
106*8975f5c5SAndroid Build Coastguard Worker     ASSERT(isUserEvent());
107*8975f5c5SAndroid Build Coastguard Worker 
108*8975f5c5SAndroid Build Coastguard Worker     cl_int status = CL_QUEUED;
109*8975f5c5SAndroid Build Coastguard Worker     std::unique_lock<std::mutex> ul(mUserEventMutex);
110*8975f5c5SAndroid Build Coastguard Worker     ANGLE_TRY(getCommandExecutionStatus(status));
111*8975f5c5SAndroid Build Coastguard Worker     if (status > CL_COMPLETE)
112*8975f5c5SAndroid Build Coastguard Worker     {
113*8975f5c5SAndroid Build Coastguard Worker         // User is responsible for setting the user-event object, we need to wait for that event
114*8975f5c5SAndroid Build Coastguard Worker         // (We dont care what the outcome is, just need to wait until that event triggers)
115*8975f5c5SAndroid Build Coastguard Worker         INFO() << "Waiting for user-event (" << &mEvent
116*8975f5c5SAndroid Build Coastguard Worker                << ") to be set! (aka clSetUserEventStatus)";
117*8975f5c5SAndroid Build Coastguard Worker         mUserEventCondition.wait(ul);
118*8975f5c5SAndroid Build Coastguard Worker     }
119*8975f5c5SAndroid Build Coastguard Worker 
120*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
121*8975f5c5SAndroid Build Coastguard Worker }
122*8975f5c5SAndroid Build Coastguard Worker 
setStatusAndExecuteCallback(cl_int status)123*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::setStatusAndExecuteCallback(cl_int status)
124*8975f5c5SAndroid Build Coastguard Worker {
125*8975f5c5SAndroid Build Coastguard Worker     *mStatus = status;
126*8975f5c5SAndroid Build Coastguard Worker 
127*8975f5c5SAndroid Build Coastguard Worker     ANGLE_TRY(setTimestamp(status));
128*8975f5c5SAndroid Build Coastguard Worker     if (status >= CL_COMPLETE && status < CL_QUEUED && mHaveCallbacks->at(status))
129*8975f5c5SAndroid Build Coastguard Worker     {
130*8975f5c5SAndroid Build Coastguard Worker         auto haveCallbacks = mHaveCallbacks.synchronize();
131*8975f5c5SAndroid Build Coastguard Worker 
132*8975f5c5SAndroid Build Coastguard Worker         // Sanity check, callback(s) only from this exec status should be outstanding
133*8975f5c5SAndroid Build Coastguard Worker         ASSERT(std::count(haveCallbacks->begin() + status, haveCallbacks->end(), true) == 1);
134*8975f5c5SAndroid Build Coastguard Worker 
135*8975f5c5SAndroid Build Coastguard Worker         getFrontendObject().callback(status);
136*8975f5c5SAndroid Build Coastguard Worker         haveCallbacks->at(status) = false;
137*8975f5c5SAndroid Build Coastguard Worker     }
138*8975f5c5SAndroid Build Coastguard Worker 
139*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
140*8975f5c5SAndroid Build Coastguard Worker }
141*8975f5c5SAndroid Build Coastguard Worker 
setTimestamp(cl_int status)142*8975f5c5SAndroid Build Coastguard Worker angle::Result CLEventVk::setTimestamp(cl_int status)
143*8975f5c5SAndroid Build Coastguard Worker {
144*8975f5c5SAndroid Build Coastguard Worker     if (!isUserEvent() &&
145*8975f5c5SAndroid Build Coastguard Worker         mEvent.getCommandQueue()->getProperties().intersects(CL_QUEUE_PROFILING_ENABLE))
146*8975f5c5SAndroid Build Coastguard Worker     {
147*8975f5c5SAndroid Build Coastguard Worker         // TODO(aannestrand) Just get current CPU timestamp for now, look into Vulkan GPU device
148*8975f5c5SAndroid Build Coastguard Worker         // timestamp query instead and later make CPU timestamp a fallback if GPU timestamp cannot
149*8975f5c5SAndroid Build Coastguard Worker         // be queried http://anglebug.com/357902514
150*8975f5c5SAndroid Build Coastguard Worker         cl_ulong cpuTS =
151*8975f5c5SAndroid Build Coastguard Worker             std::chrono::time_point_cast<std::chrono::nanoseconds>(std::chrono::system_clock::now())
152*8975f5c5SAndroid Build Coastguard Worker                 .time_since_epoch()
153*8975f5c5SAndroid Build Coastguard Worker                 .count();
154*8975f5c5SAndroid Build Coastguard Worker 
155*8975f5c5SAndroid Build Coastguard Worker         auto profilingTimestamps = mProfilingTimestamps.synchronize();
156*8975f5c5SAndroid Build Coastguard Worker 
157*8975f5c5SAndroid Build Coastguard Worker         switch (status)
158*8975f5c5SAndroid Build Coastguard Worker         {
159*8975f5c5SAndroid Build Coastguard Worker             case CL_QUEUED:
160*8975f5c5SAndroid Build Coastguard Worker                 profilingTimestamps->commandQueuedTS = cpuTS;
161*8975f5c5SAndroid Build Coastguard Worker                 break;
162*8975f5c5SAndroid Build Coastguard Worker             case CL_SUBMITTED:
163*8975f5c5SAndroid Build Coastguard Worker                 profilingTimestamps->commandSubmitTS = cpuTS;
164*8975f5c5SAndroid Build Coastguard Worker                 break;
165*8975f5c5SAndroid Build Coastguard Worker             case CL_RUNNING:
166*8975f5c5SAndroid Build Coastguard Worker                 profilingTimestamps->commandStartTS = cpuTS;
167*8975f5c5SAndroid Build Coastguard Worker                 break;
168*8975f5c5SAndroid Build Coastguard Worker             case CL_COMPLETE:
169*8975f5c5SAndroid Build Coastguard Worker                 profilingTimestamps->commandEndTS = cpuTS;
170*8975f5c5SAndroid Build Coastguard Worker 
171*8975f5c5SAndroid Build Coastguard Worker                 // Returns a value equivalent to passing CL_PROFILING_COMMAND_END if the device
172*8975f5c5SAndroid Build Coastguard Worker                 // associated with event does not support device-side enqueue.
173*8975f5c5SAndroid Build Coastguard Worker                 // https://registry.khronos.org/OpenCL/specs/3.0-unified/html/OpenCL_API.html#_device_side_enqueue
174*8975f5c5SAndroid Build Coastguard Worker                 profilingTimestamps->commandCompleteTS = cpuTS;
175*8975f5c5SAndroid Build Coastguard Worker                 break;
176*8975f5c5SAndroid Build Coastguard Worker             default:
177*8975f5c5SAndroid Build Coastguard Worker                 UNREACHABLE();
178*8975f5c5SAndroid Build Coastguard Worker         }
179*8975f5c5SAndroid Build Coastguard Worker     }
180*8975f5c5SAndroid Build Coastguard Worker 
181*8975f5c5SAndroid Build Coastguard Worker     return angle::Result::Continue;
182*8975f5c5SAndroid Build Coastguard Worker }
183*8975f5c5SAndroid Build Coastguard Worker 
184*8975f5c5SAndroid Build Coastguard Worker }  // namespace rx
185