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