1 /*
2 * Copyright (C) 2011-2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "FrameBuffer.h"
18
19 #include <stdio.h>
20 #include <string.h>
21 #include <time.h>
22
23 #include <iomanip>
24
25 #if defined(__linux__)
26 #include <sys/resource.h>
27 #endif
28
29 #include "ContextHelper.h"
30 #include "Hwc2.h"
31 #include "NativeSubWindow.h"
32 #include "RenderThreadInfo.h"
33 #include "SyncThread.h"
34 #include "aemu/base/LayoutResolver.h"
35 #include "aemu/base/Metrics.h"
36 #include "aemu/base/SharedLibrary.h"
37 #include "aemu/base/Tracing.h"
38 #include "aemu/base/containers/Lookup.h"
39 #include "aemu/base/files/StreamSerializing.h"
40 #include "aemu/base/memory/MemoryTracker.h"
41 #include "aemu/base/synchronization/Lock.h"
42 #include "aemu/base/system/System.h"
43
44 #if GFXSTREAM_ENABLE_HOST_GLES
45 #include "GLESVersionDetector.h"
46 #include "OpenGLESDispatch/DispatchTables.h"
47 #include "OpenGLESDispatch/EGLDispatch.h"
48 #include "PostWorkerGl.h"
49 #include "RenderControl.h"
50 #include "RenderThreadInfoGl.h"
51 #include "gl/YUVConverter.h"
52 #include "gl/gles2_dec/gles2_dec.h"
53 #include "gl/glestranslator/EGL/EglGlobalInfo.h"
54 #endif
55
56 #include "host-common/GfxstreamFatalError.h"
57 #include "host-common/crash_reporter.h"
58 #include "host-common/feature_control.h"
59 #include "host-common/logging.h"
60 #include "host-common/misc.h"
61 #include "host-common/opengl/misc.h"
62 #include "host-common/vm_operations.h"
63 #include "render-utils/MediaNative.h"
64 #include "vulkan/DisplayVk.h"
65 #include "vulkan/PostWorkerVk.h"
66 #include "vulkan/VkCommonOperations.h"
67 #include "vulkan/VkDecoderGlobalState.h"
68
69 namespace gfxstream {
70
71 using android::base::AutoLock;
72 using android::base::ManagedDescriptor;
73 using android::base::MetricEventVulkanOutOfMemory;
74 using android::base::Stream;
75 using android::base::WorkerProcessingResult;
76 using emugl::ABORT_REASON_OTHER;
77 using emugl::CreateHealthMonitor;
78 using emugl::FatalError;
79 using emugl::GfxApiLogger;
80 using gfxstream::host::FeatureSet;
81
82 #if GFXSTREAM_ENABLE_HOST_GLES
83 using gl::DisplaySurfaceGl;
84 using gl::EmulatedEglConfig;
85 using gl::EmulatedEglConfigList;
86 using gl::EmulatedEglContext;
87 using gl::EmulatedEglContextMap;
88 using gl::EmulatedEglContextPtr;
89 using gl::EmulatedEglFenceSync;
90 using gl::EmulatedEglWindowSurface;
91 using gl::EmulatedEglWindowSurfaceMap;
92 using gl::EmulatedEglWindowSurfacePtr;
93 using gl::EmulationGl;
94 using gl::GLES_DISPATCH_MAX_VERSION_2;
95 using gl::GLESApi;
96 using gl::GLESApi_2;
97 using gl::GLESApi_CM;
98 using gl::GLESDispatchMaxVersion;
99 using gl::RenderThreadInfoGl;
100 using gl::s_egl;
101 using gl::s_gles2;
102 using gl::TextureDraw;
103 using gl::YUVConverter;
104 using gl::YUVPlane;
105 #endif
106
107 using gfxstream::vk::AstcEmulationMode;
108 using gfxstream::vk::VkEmulationFeatures;
109
110 // static std::string getTimeStampString() {
111 // const time_t timestamp = android::base::getUnixTimeUs();
112 // const struct tm *timeinfo = localtime(×tamp);
113 // // Target format: 07-31 4:44:33
114 // char b[64];
115 // snprintf(
116 // b,
117 // sizeof(b) - 1,
118 // "%02u-%02u %02u:%02u:%02u",
119 // timeinfo->tm_mon + 1,
120 // timeinfo->tm_mday,
121 // timeinfo->tm_hour,
122 // timeinfo->tm_min,
123 // timeinfo->tm_sec);
124 // return std::string(b);
125 // }
126
127 // static unsigned int getUptimeMs() {
128 // return android::base::getUptimeMs();
129 // }
130
dumpPerfStats()131 static void dumpPerfStats() {
132 // auto usage = System::get()->getMemUsage();
133 // std::string memoryStats =
134 // emugl::getMemoryTracker()
135 // ? emugl::getMemoryTracker()->printUsage()
136 // : "";
137 // auto cpuUsage = emugl::getCpuUsage();
138 // std::string lastStats =
139 // cpuUsage ? cpuUsage->printUsage() : "";
140 // printf("%s Uptime: %u ms Resident memory: %f mb %s \n%s\n",
141 // getTimeStampString().c_str(), getUptimeMs(),
142 // (float)usage.resident / 1048576.0f, lastStats.c_str(),
143 // memoryStats.c_str());
144 }
145
146 class PerfStatThread : public android::base::Thread {
147 public:
PerfStatThread(bool * perfStatActive)148 PerfStatThread(bool* perfStatActive) :
149 Thread(), m_perfStatActive(perfStatActive) {}
150
main()151 virtual intptr_t main() {
152 while (*m_perfStatActive) {
153 sleepMs(1000);
154 dumpPerfStats();
155 }
156 return 0;
157 }
158
159 private:
160 bool* m_perfStatActive;
161 };
162
163 FrameBuffer* FrameBuffer::s_theFrameBuffer = NULL;
164 HandleType FrameBuffer::s_nextHandle = 0;
165
166 // A condition variable needed to wait for framebuffer initialization.
167 namespace {
168 struct InitializedGlobals {
169 android::base::Lock lock;
170 android::base::ConditionVariable condVar;
171 };
172
postOnlyOnMainThread()173 bool postOnlyOnMainThread() {
174 #ifdef __APPLE__
175 return true;
176 #else
177 return false;
178 #endif
179 }
180
getAstcEmulationMode()181 AstcEmulationMode getAstcEmulationMode() {
182 return AstcEmulationMode::Gpu;
183 // return AstcEmulationMode::Cpu;
184 }
185
186 } // namespace
187
188 // |sInitialized| caches the initialized framebuffer state - this way
189 // happy path doesn't need to lock the mutex.
190 static std::atomic<bool> sInitialized{false};
sGlobals()191 static InitializedGlobals* sGlobals() {
192 static InitializedGlobals* g = new InitializedGlobals;
193 return g;
194 }
195
waitUntilInitialized()196 void FrameBuffer::waitUntilInitialized() {
197 if (sInitialized.load(std::memory_order_relaxed)) {
198 return;
199 }
200
201 #if SNAPSHOT_PROFILE > 1
202 const auto startTime = android::base::getHighResTimeUs();
203 #endif
204 {
205 AutoLock l(sGlobals()->lock);
206 sGlobals()->condVar.wait(
207 &l, [] { return sInitialized.load(std::memory_order_acquire); });
208 }
209 #if SNAPSHOT_PROFILE > 1
210 printf("Waited for FrameBuffer initialization for %.03f ms\n",
211 (android::base::getHighResTimeUs() - startTime) / 1000.0);
212 #endif
213 }
214
MaybeIncreaseFileDescriptorSoftLimit()215 void MaybeIncreaseFileDescriptorSoftLimit() {
216 #if defined(__linux__)
217 // Cuttlefish with Gfxstream on Nvidia and SwiftShader often hits the default nofile
218 // soft limit (1024) when running large test suites.
219 struct rlimit nofileLimits = {
220 .rlim_cur = 0,
221 .rlim_max = 0,
222 };
223
224 int ret = getrlimit(RLIMIT_NOFILE, &nofileLimits);
225 if (ret) {
226 ERR("Warning: failed to query nofile limits.");
227 return;
228 }
229
230 const auto softLimit = nofileLimits.rlim_cur;
231 const auto hardLimit = nofileLimits.rlim_max;
232
233 constexpr const rlim_t kDesiredNofileSoftLimit = 4096;
234
235 if (softLimit < kDesiredNofileSoftLimit) {
236 if (softLimit == hardLimit) {
237 ERR("Warning: unable to raise nofile soft limit - already at hard limit.");
238 return;
239 }
240
241 if (kDesiredNofileSoftLimit > hardLimit) {
242 ERR("Warning: unable to raise nofile soft limit to desired %d - hard limit is %d.",
243 static_cast<int>(kDesiredNofileSoftLimit), static_cast<int>(hardLimit));
244 }
245
246 const rlim_t requestedSoftLimit = std::min(kDesiredNofileSoftLimit, hardLimit);
247
248 struct rlimit requestedNofileLimits = {
249 .rlim_cur = requestedSoftLimit,
250 .rlim_max = hardLimit,
251 };
252
253 ret = setrlimit(RLIMIT_NOFILE, &requestedNofileLimits);
254 if (ret) {
255 ERR("Warning: failed to raise nofile soft limit to %d: %s (%d)",
256 static_cast<int>(requestedSoftLimit), strerror(errno), errno);
257 return;
258 }
259
260 INFO("Raised nofile soft limit to %d.", static_cast<int>(requestedSoftLimit));
261 } else {
262 INFO("Not raising nofile soft limit from %d.", static_cast<int>(softLimit));
263 }
264 #endif
265 }
266
initialize(int width,int height,gfxstream::host::FeatureSet features,bool useSubWindow,bool egl2egl)267 bool FrameBuffer::initialize(int width, int height, gfxstream::host::FeatureSet features,
268 bool useSubWindow, bool egl2egl) {
269 GL_LOG("FrameBuffer::initialize");
270
271 if (s_theFrameBuffer != NULL) {
272 return true;
273 }
274
275 MaybeIncreaseFileDescriptorSoftLimit();
276
277 android::base::initializeTracing();
278
279 //
280 // allocate space for the FrameBuffer object
281 //
282 std::unique_ptr<FrameBuffer> fb(new FrameBuffer(width, height, features, useSubWindow));
283 if (!fb) {
284 GL_LOG("Failed to create fb");
285 ERR("Failed to create fb\n");
286 return false;
287 }
288
289 std::unique_ptr<emugl::RenderDocWithMultipleVkInstances> renderDocMultipleVkInstances = nullptr;
290 if (!android::base::getEnvironmentVariable("ANDROID_EMU_RENDERDOC").empty()) {
291 SharedLibrary* renderdocLib = nullptr;
292 #ifdef _WIN32
293 renderdocLib = SharedLibrary::open(R"(C:\Program Files\RenderDoc\renderdoc.dll)");
294 #elif defined(__linux__)
295 renderdocLib = SharedLibrary::open("librenderdoc.so");
296 #endif
297 fb->m_renderDoc = emugl::RenderDoc::create(renderdocLib);
298 if (fb->m_renderDoc) {
299 INFO("RenderDoc integration enabled.");
300 renderDocMultipleVkInstances =
301 std::make_unique<emugl::RenderDocWithMultipleVkInstances>(*fb->m_renderDoc);
302 if (!renderDocMultipleVkInstances) {
303 ERR("Failed to initialize RenderDoc with multiple VkInstances. Can't capture any "
304 "information from guest VkInstances with RenderDoc.");
305 }
306 }
307 }
308 // Initialize Vulkan emulation state
309 //
310 // Note: This must happen before any use of s_egl,
311 // or it's possible that the existing EGL display and contexts
312 // used by underlying EGL driver might become invalid,
313 // preventing new contexts from being created that share
314 // against those contexts.
315 vk::VkEmulation* vkEmu = nullptr;
316 vk::VulkanDispatch* vkDispatch = nullptr;
317 if (fb->m_features.Vulkan.enabled) {
318 vkDispatch = vk::vkDispatch(false /* not for testing */);
319 vkEmu = vk::createGlobalVkEmulation(vkDispatch, fb->m_features);
320 if (!vkEmu) {
321 ERR("Failed to initialize global Vulkan emulation. Disable the Vulkan support.");
322 }
323 fb->m_emulationVk = vkEmu;
324 }
325 if (vkEmu) {
326 fb->m_vulkanEnabled = true;
327 if (fb->m_features.VulkanNativeSwapchain.enabled) {
328 fb->m_vkInstance = vkEmu->instance;
329 }
330 if (vkEmu->deviceInfo.supportsIdProperties) {
331 GL_LOG("Supports id properties, got a vulkan device UUID");
332 fprintf(stderr, "%s: Supports id properties, got a vulkan device UUID\n", __func__);
333 memcpy(fb->m_vulkanUUID.data(), vkEmu->deviceInfo.idProps.deviceUUID, VK_UUID_SIZE);
334 } else {
335 GL_LOG("Doesn't support id properties, no vulkan device UUID");
336 fprintf(stderr, "%s: Doesn't support id properties, no vulkan device UUID\n", __func__);
337 }
338 }
339
340 #if GFXSTREAM_ENABLE_HOST_GLES
341 // Do not initialize GL emulation if the guest is using ANGLE.
342 if (!fb->m_features.GuestUsesAngle.enabled) {
343 fb->m_emulationGl = EmulationGl::create(width, height, fb->m_features, useSubWindow, egl2egl);
344 if (!fb->m_emulationGl) {
345 ERR("Failed to initialize GL emulation.");
346 return false;
347 }
348 }
349 #endif
350
351 fb->m_guestUsesAngle = fb->m_features.GuestUsesAngle.enabled;
352
353 fb->m_useVulkanComposition = fb->m_features.GuestUsesAngle.enabled ||
354 fb->m_features.VulkanNativeSwapchain.enabled;
355
356 std::unique_ptr<VkEmulationFeatures> vkEmulationFeatures =
357 std::make_unique<VkEmulationFeatures>(VkEmulationFeatures{
358 .glInteropSupported = false, // Set later.
359 .deferredCommands =
360 android::base::getEnvironmentVariable("ANDROID_EMU_VK_DISABLE_DEFERRED_COMMANDS")
361 .empty(),
362 .createResourceWithRequirements =
363 android::base::getEnvironmentVariable(
364 "ANDROID_EMU_VK_DISABLE_USE_CREATE_RESOURCES_WITH_REQUIREMENTS")
365 .empty(),
366 .useVulkanComposition = fb->m_useVulkanComposition,
367 .useVulkanNativeSwapchain = fb->m_features.VulkanNativeSwapchain.enabled,
368 .guestRenderDoc = std::move(renderDocMultipleVkInstances),
369 .astcLdrEmulationMode = AstcEmulationMode::Gpu,
370 .enableEtc2Emulation = true,
371 .enableYcbcrEmulation = false,
372 .guestUsesAngle = fb->m_guestUsesAngle,
373 .useDedicatedAllocations = false, // Set later.
374 });
375
376 //
377 // Cache the GL strings so we don't have to think about threading or
378 // current-context when asked for them.
379 //
380 bool useVulkanGraphicsDiagInfo =
381 vkEmu && fb->m_features.VulkanNativeSwapchain.enabled && fb->m_guestUsesAngle;
382
383 if (useVulkanGraphicsDiagInfo) {
384 fb->m_graphicsAdapterVendor = vkEmu->deviceInfo.driverVendor;
385 fb->m_graphicsAdapterName = vkEmu->deviceInfo.physdevProps.deviceName;
386
387 uint32_t vkVersion = vkEmu->vulkanInstanceVersion;
388
389 std::stringstream versionStringBuilder;
390 versionStringBuilder << "Vulkan " << VK_API_VERSION_MAJOR(vkVersion) << "."
391 << VK_API_VERSION_MINOR(vkVersion) << "."
392 << VK_API_VERSION_PATCH(vkVersion) << " "
393 << vkEmu->deviceInfo.driverVendor << " "
394 << vkEmu->deviceInfo.driverVersion;
395 fb->m_graphicsApiVersion = versionStringBuilder.str();
396
397 std::stringstream instanceExtensionsStringBuilder;
398 for (auto& ext : vkEmu->instanceExtensions) {
399 if (instanceExtensionsStringBuilder.tellp() != 0) {
400 instanceExtensionsStringBuilder << " ";
401 }
402 instanceExtensionsStringBuilder << ext.extensionName;
403 }
404
405 fb->m_graphicsApiExtensions = instanceExtensionsStringBuilder.str();
406
407 std::stringstream deviceExtensionsStringBuilder;
408 for (auto& ext : vkEmu->deviceInfo.extensions) {
409 if (deviceExtensionsStringBuilder.tellp() != 0) {
410 deviceExtensionsStringBuilder << " ";
411 }
412 deviceExtensionsStringBuilder << ext.extensionName;
413 }
414
415 fb->m_graphicsDeviceExtensions = deviceExtensionsStringBuilder.str();
416 } else if (fb->m_emulationGl) {
417 #if GFXSTREAM_ENABLE_HOST_GLES
418 fb->m_graphicsAdapterVendor = fb->m_emulationGl->getGlesVendor();
419 fb->m_graphicsAdapterName = fb->m_emulationGl->getGlesRenderer();
420 fb->m_graphicsApiVersion = fb->m_emulationGl->getGlesVersionString();
421 fb->m_graphicsApiExtensions = fb->m_emulationGl->getGlesExtensionsString();
422 fb->m_graphicsDeviceExtensions = "N/A";
423 #endif
424 } else {
425 fb->m_graphicsAdapterVendor = "N/A";
426 fb->m_graphicsAdapterName = "N/A";
427 fb->m_graphicsApiVersion = "N/A";
428 fb->m_graphicsApiExtensions = "N/A";
429 fb->m_graphicsDeviceExtensions = "N/A";
430 }
431
432 // Attempt to get the device UUID of the gles and match with Vulkan. If
433 // they match, interop is possible. If they don't, then don't trust the
434 // result of interop query to egl and fall back to CPU copy, as we might
435 // have initialized Vulkan devices and GLES contexts from different
436 // physical devices.
437
438 bool vulkanInteropSupported = true;
439 // First, if the VkEmulation instance doesn't support ext memory capabilities,
440 // it won't support uuids.
441 if (!vkEmu || !vkEmu->deviceInfo.supportsIdProperties) {
442 vulkanInteropSupported = false;
443 }
444 if (!fb->m_emulationGl) {
445 vulkanInteropSupported = false;
446 } else {
447 #if GFXSTREAM_ENABLE_HOST_GLES
448 if (!fb->m_emulationGl->isGlesVulkanInteropSupported()) {
449 vulkanInteropSupported = false;
450 }
451 const auto& glesDeviceUuid = fb->m_emulationGl->getGlesDeviceUuid();
452 if (!glesDeviceUuid || glesDeviceUuid != fb->m_vulkanUUID) {
453 vulkanInteropSupported = false;
454 }
455 #endif
456 }
457 // TODO: 0-copy gl interop on swiftshader vk
458 if (android::base::getEnvironmentVariable("ANDROID_EMU_VK_ICD") == "swiftshader") {
459 vulkanInteropSupported = false;
460 GL_LOG("vk icd swiftshader, disable interop");
461 }
462
463 fb->m_vulkanInteropSupported = vulkanInteropSupported;
464 GL_LOG("interop? %d", fb->m_vulkanInteropSupported);
465
466 #if GFXSTREAM_ENABLE_HOST_GLES
467 if (vulkanInteropSupported && fb->m_emulationGl && fb->m_emulationGl->isMesa()) {
468 // Mesa currently expects dedicated allocations for external memory sharing
469 // between GL and VK. See b/265186355.
470 vkEmulationFeatures->useDedicatedAllocations = true;
471 }
472 #endif
473
474 GL_LOG("glvk interop final: %d", fb->m_vulkanInteropSupported);
475 vkEmulationFeatures->glInteropSupported = fb->m_vulkanInteropSupported;
476 if (fb->m_features.Vulkan.enabled) {
477 vk::initVkEmulationFeatures(std::move(vkEmulationFeatures));
478 if (vkEmu && vkEmu->displayVk) {
479 fb->m_displayVk = vkEmu->displayVk.get();
480 fb->m_displaySurfaceUsers.push_back(fb->m_displayVk);
481 }
482 }
483
484 if (fb->m_useVulkanComposition) {
485 if (!vkEmu->compositorVk) {
486 ERR("Failed to get CompositorVk from VkEmulation.");
487 return false;
488 }
489 GL_LOG("Performing composition using CompositorVk.");
490 fb->m_compositor = vkEmu->compositorVk.get();
491 } else {
492 GL_LOG("Performing composition using CompositorGl.");
493 #if GFXSTREAM_ENABLE_HOST_GLES
494 auto compositorGl = fb->m_emulationGl->getCompositor();
495 fb->m_compositor = compositorGl;
496 #endif
497 }
498
499 #if GFXSTREAM_ENABLE_HOST_GLES
500 if (fb->m_emulationGl) {
501 auto displayGl = fb->m_emulationGl->getDisplay();
502 fb->m_displayGl = displayGl;
503 fb->m_displaySurfaceUsers.push_back(displayGl);
504 }
505 #endif
506
507 INFO("Graphics Adapter Vendor %s", fb->m_graphicsAdapterVendor.c_str());
508 INFO("Graphics Adapter %s", fb->m_graphicsAdapterName.c_str());
509 INFO("Graphics API Version %s", fb->m_graphicsApiVersion.c_str());
510 INFO("Graphics API Extensions %s", fb->m_graphicsApiExtensions.c_str());
511 INFO("Graphics Device Extensions %s", fb->m_graphicsDeviceExtensions.c_str());
512
513 if (fb->m_useVulkanComposition) {
514 fb->m_postWorker.reset(new PostWorkerVk(fb.get(), fb->m_compositor, fb->m_displayVk));
515 } else {
516 const bool shouldPostOnlyOnMainThread = postOnlyOnMainThread();
517
518 #if GFXSTREAM_ENABLE_HOST_GLES
519 PostWorkerGl* postWorkerGl =
520 new PostWorkerGl(shouldPostOnlyOnMainThread, fb.get(), fb->m_compositor,
521 fb->m_displayGl, fb->m_emulationGl.get());
522 fb->m_postWorker.reset(postWorkerGl);
523 fb->m_displaySurfaceUsers.push_back(postWorkerGl);
524 #endif
525 }
526
527 // Start up the single sync thread. If we are using Vulkan native
528 // swapchain, then don't initialize SyncThread worker threads with EGL
529 // contexts.
530 SyncThread::initialize(
531 /* hasGL */ fb->m_emulationGl != nullptr, fb->getHealthMonitor());
532
533 // Start the vsync thread
534 const uint64_t kOneSecondNs = 1000000000ULL;
535 fb->m_vsyncThread.reset(new VsyncThread((uint64_t)kOneSecondNs / (uint64_t)fb->m_vsyncHz));
536
537 //
538 // Keep the singleton framebuffer pointer
539 //
540 s_theFrameBuffer = fb.release();
541 {
542 AutoLock lock(sGlobals()->lock);
543 sInitialized.store(true, std::memory_order_release);
544 sGlobals()->condVar.broadcastAndUnlock(&lock);
545 }
546
547 // Nothing else to do - we're ready to rock!
548 return true;
549 }
550
finalize()551 void FrameBuffer::finalize() {
552 FrameBuffer* fb = s_theFrameBuffer;
553 s_theFrameBuffer = nullptr;
554 if (fb) {
555 delete fb;
556 }
557 }
558
FrameBuffer(int p_width,int p_height,gfxstream::host::FeatureSet features,bool useSubWindow)559 FrameBuffer::FrameBuffer(int p_width, int p_height, gfxstream::host::FeatureSet features, bool useSubWindow)
560 : m_features(features),
561 m_framebufferWidth(p_width),
562 m_framebufferHeight(p_height),
563 m_windowWidth(p_width),
564 m_windowHeight(p_height),
565 m_useSubWindow(useSubWindow),
566 m_fpsStats(getenv("SHOW_FPS_STATS") != nullptr),
567 m_perfStats(!android::base::getEnvironmentVariable("SHOW_PERF_STATS").empty()),
568 m_perfThread(new PerfStatThread(&m_perfStats)),
569 m_readbackThread(
570 [this](FrameBuffer::Readback&& readback) { return sendReadbackWorkerCmd(readback); }),
571 m_refCountPipeEnabled(features.RefCountPipe.enabled),
572 m_noDelayCloseColorBufferEnabled(features.NoDelayCloseColorBuffer.enabled ||
573 features.Minigbm.enabled),
__anon1be705cb0402(Post&& post) 574 m_postThread([this](Post&& post) { return postWorkerFunc(post); }),
575 m_logger(CreateMetricsLogger()),
576 m_healthMonitor(CreateHealthMonitor(*m_logger)) {
577 mDisplayActiveConfigId = 0;
578 mDisplayConfigs[0] = {p_width, p_height, 160, 160};
579 uint32_t displayId = 0;
580 if (createDisplay(&displayId) < 0) {
581 fprintf(stderr, "Failed to create default display\n");
582 }
583
584 setDisplayPose(displayId, 0, 0, getWidth(), getHeight(), 0);
585 m_perfThread->start();
586 }
587
~FrameBuffer()588 FrameBuffer::~FrameBuffer() {
589 AutoLock fbLock(m_lock);
590
591 m_perfStats = false;
592 m_perfThread->wait(NULL);
593
594 m_postThread.enqueue({PostCmd::Exit});
595 m_postThread.join();
596 m_postWorker.reset();
597
598 // Run other cleanup callbacks
599 // Avoid deadlock by first storing a separate list of callbacks
600 std::vector<std::function<void()>> callbacks;
601 for (auto procIte : m_procOwnedCleanupCallbacks)
602 {
603 for (auto it : procIte.second) {
604 callbacks.push_back(it.second);
605 }
606 }
607 m_procOwnedCleanupCallbacks.clear();
608
609 fbLock.unlock();
610
611 for (auto cb : callbacks) {
612 cb();
613 }
614
615 fbLock.lock();
616
617 if (m_useSubWindow) {
618 removeSubWindow_locked();
619 }
620
621 m_readbackThread.enqueue({ReadbackCmd::Exit});
622 m_readbackThread.join();
623
624 m_vsyncThread.reset();
625
626 delete m_perfThread;
627
628 SyncThread::destroy();
629
630 sweepColorBuffersLocked();
631
632 m_buffers.clear();
633 {
634 AutoLock lock(m_colorBufferMapLock);
635 m_colorbuffers.clear();
636 }
637 m_colorBufferDelayedCloseList.clear();
638
639 #if GFXSTREAM_ENABLE_HOST_GLES
640 m_windows.clear();
641 m_contexts.clear();
642
643 for (auto it : m_platformEglContexts) {
644 destroySharedTrivialContext(it.second.context, it.second.surface);
645 }
646 #endif
647
648 vk::teardownGlobalVkEmulation();
649
650 sInitialized.store(false, std::memory_order_relaxed);
651 }
652
653 WorkerProcessingResult
sendReadbackWorkerCmd(const Readback & readback)654 FrameBuffer::sendReadbackWorkerCmd(const Readback& readback) {
655 ensureReadbackWorker();
656 switch (readback.cmd) {
657 case ReadbackCmd::Init:
658 m_readbackWorker->init();
659 return WorkerProcessingResult::Continue;
660 case ReadbackCmd::GetPixels:
661 m_readbackWorker->getPixels(readback.displayId, readback.pixelsOut, readback.bytes);
662 return WorkerProcessingResult::Continue;
663 case ReadbackCmd::AddRecordDisplay:
664 m_readbackWorker->initReadbackForDisplay(readback.displayId, readback.width, readback.height);
665 return WorkerProcessingResult::Continue;
666 case ReadbackCmd::DelRecordDisplay:
667 m_readbackWorker->deinitReadbackForDisplay(readback.displayId);
668 return WorkerProcessingResult::Continue;
669 case ReadbackCmd::Exit:
670 return WorkerProcessingResult::Stop;
671 }
672 return WorkerProcessingResult::Stop;
673 }
674
postWorkerFunc(Post & post)675 WorkerProcessingResult FrameBuffer::postWorkerFunc(Post& post) {
676 auto annotations = std::make_unique<EventHangMetadata::HangAnnotations>();
677 if (m_healthMonitor)
678 annotations->insert(
679 {"Post command opcode", std::to_string(static_cast<uint64_t>(post.cmd))});
680 auto watchdog = WATCHDOG_BUILDER(m_healthMonitor.get(), "PostWorker main function")
681 .setAnnotations(std::move(annotations))
682 .build();
683 switch (post.cmd) {
684 case PostCmd::Post: {
685 // We wrap the callback like this to workaround a bug in the MS STL implementation.
686 auto packagePostCmdCallback =
687 std::shared_ptr<Post::CompletionCallback>(std::move(post.completionCallback));
688 std::unique_ptr<Post::CompletionCallback> postCallback =
689 std::make_unique<Post::CompletionCallback>(
690 [packagePostCmdCallback](std::shared_future<void> waitForGpu) {
691 SyncThread::get()->triggerGeneral(
692 [composeCallback = std::move(packagePostCmdCallback), waitForGpu] {
693 (*composeCallback)(waitForGpu);
694 },
695 "Wait for post");
696 });
697 m_postWorker->post(post.cb, std::move(postCallback));
698 decColorBufferRefCountNoDestroy(post.cbHandle);
699 break;
700 }
701 case PostCmd::Viewport:
702 m_postWorker->viewport(post.viewport.width,
703 post.viewport.height);
704 break;
705 case PostCmd::Compose: {
706 std::unique_ptr<FlatComposeRequest> composeRequest;
707 std::unique_ptr<Post::CompletionCallback> composeCallback;
708 if (post.composeVersion <= 1) {
709 composeCallback = std::move(post.completionCallback);
710 composeRequest = ToFlatComposeRequest((ComposeDevice*)post.composeBuffer.data());
711 } else {
712 // std::shared_ptr(std::move(...)) is WA for MSFT STL implementation bug:
713 // https://developercommunity.visualstudio.com/t/unable-to-move-stdpackaged-task-into-any-stl-conta/108672
714 auto packageComposeCallback =
715 std::shared_ptr<Post::CompletionCallback>(std::move(post.completionCallback));
716 composeCallback = std::make_unique<Post::CompletionCallback>(
717 [packageComposeCallback](
718 std::shared_future<void> waitForGpu) {
719 SyncThread::get()->triggerGeneral(
720 [composeCallback = std::move(packageComposeCallback), waitForGpu] {
721 (*composeCallback)(waitForGpu);
722 },
723 "Wait for host composition");
724 });
725 composeRequest = ToFlatComposeRequest((ComposeDevice_v2*)post.composeBuffer.data());
726 }
727 m_postWorker->compose(std::move(composeRequest), std::move(composeCallback));
728 break;
729 }
730 case PostCmd::Clear:
731 m_postWorker->clear();
732 break;
733 case PostCmd::Screenshot:
734 m_postWorker->screenshot(
735 post.screenshot.cb, post.screenshot.screenwidth,
736 post.screenshot.screenheight, post.screenshot.format,
737 post.screenshot.type, post.screenshot.rotation,
738 post.screenshot.pixels, post.screenshot.rect);
739 decColorBufferRefCountNoDestroy(post.cbHandle);
740 break;
741 case PostCmd::Block:
742 m_postWorker->block(std::move(post.block->scheduledSignal),
743 std::move(post.block->continueSignal));
744 break;
745 case PostCmd::Exit:
746 m_postWorker->exit();
747 return WorkerProcessingResult::Stop;
748 default:
749 break;
750 }
751 return WorkerProcessingResult::Continue;
752 }
753
sendPostWorkerCmd(Post post)754 std::future<void> FrameBuffer::sendPostWorkerCmd(Post post) {
755 bool expectedPostThreadStarted = false;
756 if (m_postThreadStarted.compare_exchange_strong(expectedPostThreadStarted, true)) {
757 m_postThread.start();
758 }
759
760 bool shouldPostOnlyOnMainThread = postOnlyOnMainThread();
761 // If we want to run only in the main thread and we are actually running
762 // in the main thread already, don't use the PostWorker thread. Ideally,
763 // PostWorker should handle this and dispatch directly, but we'll need to
764 // transfer ownership of the thread to PostWorker.
765 // TODO(lfy): do that refactor
766 // For now, this fixes a screenshot issue on macOS.
767 std::future<void> res = std::async(std::launch::deferred, [] {});
768 res.wait();
769 if (shouldPostOnlyOnMainThread && (PostCmd::Screenshot == post.cmd) &&
770 emugl::get_emugl_window_operations().isRunningInUiThread()) {
771 post.cb->readToBytesScaled(post.screenshot.screenwidth, post.screenshot.screenheight,
772 post.screenshot.format, post.screenshot.type,
773 post.screenshot.rotation, post.screenshot.rect,
774 post.screenshot.pixels);
775 } else {
776 std::future<void> completeFuture =
777 m_postThread.enqueue(Post(std::move(post)));
778 if (!shouldPostOnlyOnMainThread ||
779 (PostCmd::Screenshot == post.cmd &&
780 !emugl::get_emugl_window_operations().isRunningInUiThread())) {
781 res = std::move(completeFuture);
782 }
783 }
784 return res;
785 }
786
setPostCallback(Renderer::OnPostCallback onPost,void * onPostContext,uint32_t displayId,bool useBgraReadback)787 void FrameBuffer::setPostCallback(Renderer::OnPostCallback onPost, void* onPostContext,
788 uint32_t displayId, bool useBgraReadback) {
789 AutoLock lock(m_lock);
790 if (onPost) {
791 uint32_t w, h;
792 if (!emugl::get_emugl_multi_display_operations().getMultiDisplay(displayId,
793 nullptr,
794 nullptr,
795 &w, &h,
796 nullptr,
797 nullptr,
798 nullptr)) {
799 ERR("display %d not exist, cancelling OnPost callback", displayId);
800 return;
801 }
802 if (m_onPost.find(displayId) != m_onPost.end()) {
803 ERR("display %d already configured for recording", displayId);
804 return;
805 }
806 m_onPost[displayId].cb = onPost;
807 m_onPost[displayId].context = onPostContext;
808 m_onPost[displayId].displayId = displayId;
809 m_onPost[displayId].width = w;
810 m_onPost[displayId].height = h;
811 m_onPost[displayId].img = new unsigned char[4 * w * h];
812 m_onPost[displayId].readBgra = useBgraReadback;
813 bool expectedReadbackThreadStarted = false;
814 if (m_readbackThreadStarted.compare_exchange_strong(expectedReadbackThreadStarted, true)) {
815 m_readbackThread.start();
816 m_readbackThread.enqueue({ ReadbackCmd::Init });
817 }
818 std::future<void> completeFuture = m_readbackThread.enqueue(
819 {ReadbackCmd::AddRecordDisplay, displayId, nullptr, 0, w, h});
820 completeFuture.wait();
821 } else {
822 std::future<void> completeFuture = m_readbackThread.enqueue(
823 {ReadbackCmd::DelRecordDisplay, displayId});
824 completeFuture.wait();
825 m_onPost.erase(displayId);
826 }
827 }
828
subWindowRepaint(void * param)829 static void subWindowRepaint(void* param) {
830 GL_LOG("call repost from subWindowRepaint callback");
831 auto fb = static_cast<FrameBuffer*>(param);
832 fb->repost();
833 }
834
setupSubWindow(FBNativeWindowType p_window,int wx,int wy,int ww,int wh,int fbw,int fbh,float dpr,float zRot,bool deleteExisting,bool hideWindow)835 bool FrameBuffer::setupSubWindow(FBNativeWindowType p_window,
836 int wx,
837 int wy,
838 int ww,
839 int wh,
840 int fbw,
841 int fbh,
842 float dpr,
843 float zRot,
844 bool deleteExisting,
845 bool hideWindow) {
846 GL_LOG("Begin setupSubWindow");
847 if (!m_useSubWindow) {
848 ERR("%s: Cannot create native sub-window in this configuration\n",
849 __FUNCTION__);
850 return false;
851 }
852
853 // Do a quick check before even taking the lock - maybe we don't need to
854 // do anything here.
855
856 const bool shouldCreateSubWindow = !m_subWin || deleteExisting;
857
858 // On Mac, since window coordinates are Y-up and not Y-down, the
859 // subwindow may not change dimensions, but because the main window
860 // did, the subwindow technically needs to be re-positioned. This
861 // can happen on rotation, so a change in Z-rotation can be checked
862 // for this case. However, this *should not* be done on Windows/Linux,
863 // because the functions used to resize a native window on those hosts
864 // will block if the shape doesn't actually change, freezing the
865 // emulator.
866 const bool shouldMoveSubWindow =
867 !shouldCreateSubWindow &&
868 !(m_x == wx && m_y == wy && m_windowWidth == ww && m_windowHeight == wh
869 #if defined(__APPLE__)
870 && m_zRot == zRot
871 #endif
872 );
873
874 const bool redrawSubwindow =
875 shouldCreateSubWindow || shouldMoveSubWindow || m_zRot != zRot || m_dpr != dpr ||
876 m_windowContentFullWidth != fbw || m_windowContentFullHeight != fbh;
877 if (!shouldCreateSubWindow && !shouldMoveSubWindow && !redrawSubwindow) {
878 assert(sInitialized.load(std::memory_order_relaxed));
879 GL_LOG("Exit setupSubWindow (nothing to do)");
880 #if SNAPSHOT_PROFILE > 1
881 // printf("FrameBuffer::%s(): nothing to do at %lld ms\n", __func__,
882 // (long long)System::get()->getProcessTimes().wallClockMs);
883 #endif
884 return true;
885 }
886
887 #if SNAPSHOT_PROFILE > 1
888 // printf("FrameBuffer::%s(%s): start at %lld ms\n", __func__,
889 // deleteExisting ? "deleteExisting" : "keepExisting",
890 // (long long)System::get()->getProcessTimes().wallClockMs);
891 #endif
892 class ScopedPromise {
893 public:
894 ~ScopedPromise() { mPromise.set_value(); }
895 std::future<void> getFuture() { return mPromise.get_future(); }
896 DISALLOW_COPY_ASSIGN_AND_MOVE(ScopedPromise);
897 static std::tuple<std::unique_ptr<ScopedPromise>, std::future<void>> create() {
898 auto scopedPromise = std::unique_ptr<ScopedPromise>(new ScopedPromise());
899 auto future = scopedPromise->mPromise.get_future();
900 return std::make_tuple(std::move(scopedPromise), std::move(future));
901 }
902
903 private:
904 ScopedPromise() = default;
905 std::promise<void> mPromise;
906 };
907 std::unique_ptr<ScopedPromise> postWorkerContinueSignal;
908 std::future<void> postWorkerContinueSignalFuture;
909 std::tie(postWorkerContinueSignal, postWorkerContinueSignalFuture) = ScopedPromise::create();
910 {
911 auto watchdog =
912 WATCHDOG_BUILDER(m_healthMonitor.get(), "Wait for other tasks on PostWorker")
913 .setTimeoutMs(6000)
914 .build();
915 blockPostWorker(std::move(postWorkerContinueSignalFuture)).wait();
916 }
917 if (m_displayVk) {
918 auto watchdog = WATCHDOG_BUILDER(m_healthMonitor.get(), "Draining the VkQueue")
919 .setTimeoutMs(6000)
920 .build();
921 m_displayVk->drainQueues();
922 }
923 auto lockWatchdog =
924 WATCHDOG_BUILDER(m_healthMonitor.get(), "Wait for the FrameBuffer global lock").build();
925 auto lockWatchdogId = lockWatchdog->release();
926 AutoLock mutex(m_lock);
927 if (lockWatchdogId.has_value()) {
928 m_healthMonitor->stopMonitoringTask(lockWatchdogId.value());
929 }
930
931 #if SNAPSHOT_PROFILE > 1
932 // printf("FrameBuffer::%s(): got lock at %lld ms\n", __func__,
933 // (long long)System::get()->getProcessTimes().wallClockMs);
934 #endif
935
936 if (deleteExisting) {
937 removeSubWindow_locked();
938 }
939
940 bool success = false;
941
942 // If the subwindow doesn't exist, create it with the appropriate dimensions
943 if (!m_subWin) {
944 // Create native subwindow for FB display output
945 m_x = wx;
946 m_y = wy;
947 m_windowWidth = ww;
948 m_windowHeight = wh;
949
950 if (!hideWindow) {
951 m_subWin = createSubWindow(p_window, m_x, m_y, m_windowWidth, m_windowHeight, dpr,
952 subWindowRepaint, this, hideWindow);
953 }
954 if (m_subWin) {
955 m_nativeWindow = p_window;
956
957
958
959 if (m_displayVk) {
960 m_displaySurface =
961 vk::createDisplaySurface(m_subWin, m_windowWidth * dpr, m_windowHeight * dpr);
962 } else if (m_emulationGl) {
963 #if GFXSTREAM_ENABLE_HOST_GLES
964 m_displaySurface = m_emulationGl->createWindowSurface(m_windowWidth * dpr,
965 m_windowHeight * dpr,
966 m_subWin);
967 #endif
968 } else {
969 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
970 << "Unhandled window surface creation.";
971 }
972
973 if (m_displaySurface) {
974 // Some backends use a default display surface. Unbind from that before
975 // binding the new display surface. which potentially needs to be unbound.
976 for (auto* displaySurfaceUser : m_displaySurfaceUsers) {
977 displaySurfaceUser->unbindFromSurface();
978 }
979
980 // TODO: Make RenderDoc a DisplaySurfaceUser.
981 if (m_displayVk) {
982 if (m_renderDoc) {
983 m_renderDoc->call(emugl::RenderDoc::kSetActiveWindow,
984 RENDERDOC_DEVICEPOINTER_FROM_VKINSTANCE(m_vkInstance),
985 reinterpret_cast<RENDERDOC_WindowHandle>(m_subWin));
986 }
987 }
988
989 m_px = 0;
990 m_py = 0;
991 for (auto* displaySurfaceUser : m_displaySurfaceUsers) {
992 displaySurfaceUser->bindToSurface(m_displaySurface.get());
993 }
994 success = true;
995 } else {
996 // Display surface creation failed.
997 if (m_emulationGl) {
998 // NOTE: This can typically happen with software-only renderers like OSMesa.
999 destroySubWindow(m_subWin);
1000 m_subWin = (EGLNativeWindowType)0;
1001 } else {
1002 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
1003 << "Failed to create DisplaySurface.";
1004 }
1005 }
1006 }
1007 }
1008
1009 auto watchdog = WATCHDOG_BUILDER(m_healthMonitor.get(), "Updating subwindow state").build();
1010 // At this point, if the subwindow doesn't exist, it is because it either
1011 // couldn't be created
1012 // in the first place or the EGLSurface couldn't be created.
1013 if (m_subWin) {
1014 if (!shouldMoveSubWindow) {
1015 // Ensure that at least viewport parameters are properly updated.
1016 success = true;
1017 } else {
1018 // Only attempt to update window geometry if anything has actually
1019 // changed.
1020 m_x = wx;
1021 m_y = wy;
1022 m_windowWidth = ww;
1023 m_windowHeight = wh;
1024
1025 {
1026 auto watchdog = WATCHDOG_BUILDER(m_healthMonitor.get(), "Moving subwindow").build();
1027 success = moveSubWindow(m_nativeWindow, m_subWin, m_x, m_y, m_windowWidth,
1028 m_windowHeight, dpr);
1029 }
1030 m_displaySurface->updateSize(m_windowWidth * dpr, m_windowHeight * dpr);
1031 }
1032 // We are safe to unblock the PostWorker thread now, because we have completed all the
1033 // operations that could modify the state of the m_subWin. We need to unblock the PostWorker
1034 // here because we may need to send and wait for other tasks dispatched to the PostWorker
1035 // later, e.g. the viewport command or the post command issued later.
1036 postWorkerContinueSignal.reset();
1037
1038 if (success && redrawSubwindow) {
1039 // Subwin creation or movement was successful,
1040 // update viewport and z rotation and draw
1041 // the last posted color buffer.
1042 m_dpr = dpr;
1043 m_zRot = zRot;
1044 if (m_displayVk == nullptr) {
1045 Post postCmd;
1046 postCmd.cmd = PostCmd::Viewport;
1047 postCmd.viewport.width = fbw;
1048 postCmd.viewport.height = fbh;
1049 sendPostWorkerCmd(std::move(postCmd));
1050
1051 if (m_lastPostedColorBuffer) {
1052 GL_LOG("setupSubwindow: draw last posted cb");
1053 postImpl(m_lastPostedColorBuffer,
1054 [](std::shared_future<void> waitForGpu) {}, false);
1055 } else {
1056 Post postCmd;
1057 postCmd.cmd = PostCmd::Clear;
1058 sendPostWorkerCmd(std::move(postCmd));
1059 }
1060 }
1061 m_windowContentFullWidth = fbw;
1062 m_windowContentFullHeight = fbh;
1063 }
1064 }
1065
1066 mutex.unlock();
1067
1068 // Nobody ever checks for the return code, so there will be no retries or
1069 // even aborted run; if we don't mark the framebuffer as initialized here
1070 // its users will hang forever; if we do mark it, they will crash - which
1071 // is a better outcome (crash report == bug fixed).
1072 AutoLock lock(sGlobals()->lock);
1073 sInitialized.store(true, std::memory_order_relaxed);
1074 sGlobals()->condVar.broadcastAndUnlock(&lock);
1075
1076 #if SNAPSHOT_PROFILE > 1
1077 // printf("FrameBuffer::%s(): end at %lld ms\n", __func__,
1078 // (long long)System::get()->getProcessTimes().wallClockMs);
1079 #endif
1080
1081 GL_LOG("Exit setupSubWindow (successful setup)");
1082 return success;
1083 }
1084
removeSubWindow()1085 bool FrameBuffer::removeSubWindow() {
1086 if (!m_useSubWindow) {
1087 ERR("Cannot remove native sub-window in this configuration");
1088 return false;
1089 }
1090 AutoLock lock(sGlobals()->lock);
1091 sInitialized.store(false, std::memory_order_relaxed);
1092 sGlobals()->condVar.broadcastAndUnlock(&lock);
1093
1094 AutoLock mutex(m_lock);
1095 return removeSubWindow_locked();
1096 }
1097
removeSubWindow_locked()1098 bool FrameBuffer::removeSubWindow_locked() {
1099 if (!m_useSubWindow) {
1100 ERR("Cannot remove native sub-window in this configuration");
1101 return false;
1102 }
1103 bool removed = false;
1104 if (m_subWin) {
1105 for (auto* displaySurfaceUser : m_displaySurfaceUsers) {
1106 displaySurfaceUser->unbindFromSurface();
1107 }
1108 m_displaySurface.reset();
1109
1110 destroySubWindow(m_subWin);
1111
1112 m_subWin = (EGLNativeWindowType)0;
1113 removed = true;
1114 }
1115 return removed;
1116 }
1117
genHandle_locked()1118 HandleType FrameBuffer::genHandle_locked() {
1119 HandleType id;
1120 do {
1121 id = ++s_nextHandle;
1122 } while (id == 0 ||
1123 #if GFXSTREAM_ENABLE_HOST_GLES
1124 m_contexts.find(id) != m_contexts.end() || m_windows.find(id) != m_windows.end() ||
1125 #endif
1126 m_colorbuffers.find(id) != m_colorbuffers.end() ||
1127 m_buffers.find(id) != m_buffers.end());
1128
1129 return id;
1130 }
1131
createColorBuffer(int p_width,int p_height,GLenum p_internalFormat,FrameworkFormat p_frameworkFormat)1132 HandleType FrameBuffer::createColorBuffer(int p_width,
1133 int p_height,
1134 GLenum p_internalFormat,
1135 FrameworkFormat p_frameworkFormat) {
1136
1137 AutoLock mutex(m_lock);
1138 sweepColorBuffersLocked();
1139 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1140
1141 return createColorBufferWithHandleLocked(p_width, p_height, p_internalFormat, p_frameworkFormat,
1142 genHandle_locked());
1143 }
1144
createColorBufferWithHandle(int p_width,int p_height,GLenum p_internalFormat,FrameworkFormat p_frameworkFormat,HandleType handle,bool p_linear)1145 void FrameBuffer::createColorBufferWithHandle(int p_width, int p_height, GLenum p_internalFormat,
1146 FrameworkFormat p_frameworkFormat, HandleType handle,
1147 bool p_linear) {
1148 {
1149 AutoLock mutex(m_lock);
1150 sweepColorBuffersLocked();
1151
1152 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1153
1154 // Check for handle collision
1155 if (m_colorbuffers.count(handle) != 0) {
1156 // emugl::emugl_crash_reporter(
1157 // "FATAL: color buffer with handle %u already exists",
1158 // handle);
1159 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER));
1160 }
1161
1162 createColorBufferWithHandleLocked(p_width, p_height, p_internalFormat, p_frameworkFormat,
1163 handle, p_linear);
1164 }
1165 }
1166
createColorBufferWithHandleLocked(int p_width,int p_height,GLenum p_internalFormat,FrameworkFormat p_frameworkFormat,HandleType handle,bool p_linear)1167 HandleType FrameBuffer::createColorBufferWithHandleLocked(int p_width, int p_height,
1168 GLenum p_internalFormat,
1169 FrameworkFormat p_frameworkFormat,
1170 HandleType handle, bool p_linear) {
1171 ColorBufferPtr cb =
1172 ColorBuffer::create(m_emulationGl.get(), m_emulationVk, p_width, p_height, p_internalFormat,
1173 p_frameworkFormat, handle, nullptr /*stream*/, p_linear);
1174 if (cb.get() == nullptr) {
1175 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
1176 << "Failed to create ColorBuffer:" << handle << " format:" << p_internalFormat
1177 << " framework-format:" << p_frameworkFormat << " width:" << p_width
1178 << " height:" << p_height;
1179 }
1180
1181 assert(m_colorbuffers.count(handle) == 0);
1182 // When guest feature flag RefCountPipe is on, no reference counting is
1183 // needed. We only memoize the mapping from handle to ColorBuffer.
1184 // Explicitly set refcount to 1 to avoid the colorbuffer being added to
1185 // m_colorBufferDelayedCloseList in FrameBuffer::onLoad().
1186 if (m_refCountPipeEnabled) {
1187 m_colorbuffers.try_emplace(handle, ColorBufferRef{std::move(cb), 1, false, 0});
1188 } else {
1189 // Android master default api level is 1000
1190 int apiLevel = 1000;
1191 emugl::getAvdInfo(nullptr, &apiLevel);
1192 // pre-O and post-O use different color buffer memory management
1193 // logic
1194 if (apiLevel > 0 && apiLevel < 26) {
1195 m_colorbuffers.try_emplace(handle, ColorBufferRef{std::move(cb), 1, false, 0});
1196
1197 RenderThreadInfo* tInfo = RenderThreadInfo::get();
1198 uint64_t puid = tInfo->m_puid;
1199 if (puid) {
1200 m_procOwnedColorBuffers[puid].insert(handle);
1201 }
1202
1203 } else {
1204 m_colorbuffers.try_emplace(handle, ColorBufferRef{std::move(cb), 0, false, 0});
1205 }
1206 }
1207
1208 return handle;
1209 }
1210
createBuffer(uint64_t p_size,uint32_t memoryProperty)1211 HandleType FrameBuffer::createBuffer(uint64_t p_size, uint32_t memoryProperty) {
1212 AutoLock mutex(m_lock);
1213 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1214 return createBufferWithHandleLocked(p_size, genHandle_locked(), memoryProperty);
1215 }
1216
createBufferWithHandle(uint64_t size,HandleType handle)1217 void FrameBuffer::createBufferWithHandle(uint64_t size, HandleType handle) {
1218 AutoLock mutex(m_lock);
1219 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1220
1221 if (m_buffers.count(handle) != 0) {
1222 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
1223 << "Buffer already exists with handle " << handle;
1224 }
1225
1226 createBufferWithHandleLocked(size, handle, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT);
1227 }
1228
createBufferWithHandleLocked(int p_size,HandleType handle,uint32_t memoryProperty)1229 HandleType FrameBuffer::createBufferWithHandleLocked(int p_size, HandleType handle,
1230 uint32_t memoryProperty) {
1231 if (m_buffers.count(handle) != 0) {
1232 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
1233 << "Buffer already exists with handle " << handle;
1234 }
1235
1236 BufferPtr buffer(Buffer::create(m_emulationGl.get(), m_emulationVk, p_size, handle));
1237 if (!buffer) {
1238 ERR("Create buffer failed.\n");
1239 return 0;
1240 }
1241
1242 m_buffers[handle] = {std::move(buffer)};
1243
1244 return handle;
1245 }
1246
openColorBuffer(HandleType p_colorbuffer)1247 int FrameBuffer::openColorBuffer(HandleType p_colorbuffer) {
1248 // When guest feature flag RefCountPipe is on, no reference counting is
1249 // needed.
1250 if (m_refCountPipeEnabled) return 0;
1251
1252 RenderThreadInfo* tInfo = RenderThreadInfo::get();
1253
1254 AutoLock mutex(m_lock);
1255
1256 ColorBufferMap::iterator c;
1257 {
1258 AutoLock colorBuffermapLock(m_colorBufferMapLock);
1259 c = m_colorbuffers.find(p_colorbuffer);
1260 if (c == m_colorbuffers.end()) {
1261 // bad colorbuffer handle
1262 ERR("FB: openColorBuffer cb handle %#x not found", p_colorbuffer);
1263 return -1;
1264 }
1265 c->second.refcount++;
1266 markOpened(&c->second);
1267 }
1268
1269 uint64_t puid = tInfo ? tInfo->m_puid : 0;
1270 if (puid) {
1271 m_procOwnedColorBuffers[puid].insert(p_colorbuffer);
1272 }
1273 return 0;
1274 }
1275
closeColorBuffer(HandleType p_colorbuffer)1276 void FrameBuffer::closeColorBuffer(HandleType p_colorbuffer) {
1277 // When guest feature flag RefCountPipe is on, no reference counting is
1278 // needed.
1279 if (m_refCountPipeEnabled) {
1280 return;
1281 }
1282
1283 RenderThreadInfo* tInfo = RenderThreadInfo::get();
1284
1285 std::vector<HandleType> toCleanup;
1286
1287 AutoLock mutex(m_lock);
1288 uint64_t puid = tInfo ? tInfo->m_puid : 0;
1289 if (puid) {
1290 auto ite = m_procOwnedColorBuffers.find(puid);
1291 if (ite != m_procOwnedColorBuffers.end()) {
1292 const auto& cb = ite->second.find(p_colorbuffer);
1293 if (cb != ite->second.end()) {
1294 ite->second.erase(cb);
1295 if (closeColorBufferLocked(p_colorbuffer)) {
1296 toCleanup.push_back(p_colorbuffer);
1297 }
1298 }
1299 }
1300 } else {
1301 if (closeColorBufferLocked(p_colorbuffer)) {
1302 toCleanup.push_back(p_colorbuffer);
1303 }
1304 }
1305 }
1306
closeBuffer(HandleType p_buffer)1307 void FrameBuffer::closeBuffer(HandleType p_buffer) {
1308 AutoLock mutex(m_lock);
1309
1310 auto it = m_buffers.find(p_buffer);
1311 if (it == m_buffers.end()) {
1312 ERR("Failed to find Buffer:%d", p_buffer);
1313 return;
1314 }
1315
1316 m_buffers.erase(it);
1317 }
1318
closeColorBufferLocked(HandleType p_colorbuffer,bool forced)1319 bool FrameBuffer::closeColorBufferLocked(HandleType p_colorbuffer, bool forced) {
1320 // When guest feature flag RefCountPipe is on, no reference counting is
1321 // needed.
1322 if (m_refCountPipeEnabled) {
1323 return false;
1324 }
1325 bool deleted = false;
1326 {
1327 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1328
1329 if (m_noDelayCloseColorBufferEnabled) forced = true;
1330
1331 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
1332 if (c == m_colorbuffers.end()) {
1333 // This is harmless: it is normal for guest system to issue
1334 // closeColorBuffer command when the color buffer is already
1335 // garbage collected on the host. (we don't have a mechanism
1336 // to give guest a notice yet)
1337 return false;
1338 }
1339
1340 // The guest can and will gralloc_alloc/gralloc_free and then
1341 // gralloc_register a buffer, due to API level (O+) or
1342 // timing issues.
1343 // So, we don't actually close the color buffer when refcount
1344 // reached zero, unless it has been opened at least once already.
1345 // Instead, put it on a 'delayed close' list to return to it later.
1346 if (--c->second.refcount == 0) {
1347 if (forced) {
1348 eraseDelayedCloseColorBufferLocked(c->first, c->second.closedTs);
1349 m_colorbuffers.erase(c);
1350 deleted = true;
1351 } else {
1352 c->second.closedTs = android::base::getUnixTimeUs();
1353 m_colorBufferDelayedCloseList.push_back({c->second.closedTs, p_colorbuffer});
1354 }
1355 }
1356 }
1357
1358 performDelayedColorBufferCloseLocked(false);
1359
1360 return deleted;
1361 }
1362
decColorBufferRefCountNoDestroy(HandleType p_colorbuffer)1363 void FrameBuffer::decColorBufferRefCountNoDestroy(HandleType p_colorbuffer) {
1364 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1365
1366 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
1367 if (c == m_colorbuffers.end()) {
1368 return;
1369 }
1370
1371 if (--c->second.refcount == 0) {
1372 c->second.closedTs = android::base::getUnixTimeUs();
1373 m_colorBufferDelayedCloseList.push_back({c->second.closedTs, p_colorbuffer});
1374 }
1375 }
1376
performDelayedColorBufferCloseLocked(bool forced)1377 void FrameBuffer::performDelayedColorBufferCloseLocked(bool forced) {
1378 // Let's wait just long enough to make sure it's not because of instant
1379 // timestamp change (end of previous second -> beginning of a next one),
1380 // but not for long - this is a workaround for race conditions, and they
1381 // are quick.
1382 static constexpr uint64_t kColorBufferClosingDelayUs = 1000000LL;
1383
1384 const auto now = android::base::getUnixTimeUs();
1385 auto it = m_colorBufferDelayedCloseList.begin();
1386 while (it != m_colorBufferDelayedCloseList.end() &&
1387 (forced ||
1388 it->ts + kColorBufferClosingDelayUs <= now)) {
1389 if (it->cbHandle != 0) {
1390 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1391 const auto& cb = m_colorbuffers.find(it->cbHandle);
1392 if (cb != m_colorbuffers.end()) {
1393 m_colorbuffers.erase(cb);
1394 }
1395 }
1396 ++it;
1397 }
1398 m_colorBufferDelayedCloseList.erase(
1399 m_colorBufferDelayedCloseList.begin(), it);
1400 }
1401
eraseDelayedCloseColorBufferLocked(HandleType cb,uint64_t ts)1402 void FrameBuffer::eraseDelayedCloseColorBufferLocked(
1403 HandleType cb, uint64_t ts)
1404 {
1405 // Find the first delayed buffer with a timestamp <= |ts|
1406 auto it = std::lower_bound(
1407 m_colorBufferDelayedCloseList.begin(),
1408 m_colorBufferDelayedCloseList.end(), ts,
1409 [](const ColorBufferCloseInfo& ci, uint64_t ts) {
1410 return ci.ts < ts;
1411 });
1412 while (it != m_colorBufferDelayedCloseList.end() &&
1413 it->ts == ts) {
1414 // if this is the one we need - clear it out.
1415 if (it->cbHandle == cb) {
1416 it->cbHandle = 0;
1417 break;
1418 }
1419 ++it;
1420 }
1421 }
1422
createGraphicsProcessResources(uint64_t puid)1423 void FrameBuffer::createGraphicsProcessResources(uint64_t puid) {
1424 AutoLock mutex(m_lock);
1425 bool inserted = m_procOwnedResources.try_emplace(puid, ProcessResources::create()).second;
1426 if (!inserted) {
1427 WARN("Failed to create process resource for puid %" PRIu64 ".", puid);
1428 }
1429 }
1430
removeGraphicsProcessResources(uint64_t puid)1431 std::unique_ptr<ProcessResources> FrameBuffer::removeGraphicsProcessResources(uint64_t puid) {
1432 std::unordered_map<uint64_t, std::unique_ptr<ProcessResources>>::node_type node;
1433 {
1434 AutoLock mutex(m_lock);
1435 node = m_procOwnedResources.extract(puid);
1436 }
1437 if (node.empty()) {
1438 WARN("Failed to find process resource for puid %" PRIu64 ".", puid);
1439 return nullptr;
1440 }
1441 std::unique_ptr<ProcessResources> res = std::move(node.mapped());
1442 return res;
1443 }
1444
cleanupProcGLObjects(uint64_t puid)1445 void FrameBuffer::cleanupProcGLObjects(uint64_t puid) {
1446 bool renderThreadWithThisPuidExists = false;
1447
1448 do {
1449 renderThreadWithThisPuidExists = false;
1450 RenderThreadInfo::forAllRenderThreadInfos(
1451 [puid, &renderThreadWithThisPuidExists](RenderThreadInfo* i) {
1452 if (i->m_puid == puid) {
1453 renderThreadWithThisPuidExists = true;
1454 }
1455 });
1456 android::base::sleepUs(10000);
1457 } while (renderThreadWithThisPuidExists);
1458
1459
1460 AutoLock mutex(m_lock);
1461
1462 cleanupProcGLObjects_locked(puid);
1463
1464 // Run other cleanup callbacks
1465 // Avoid deadlock by first storing a separate list of callbacks
1466 std::vector<std::function<void()>> callbacks;
1467
1468 {
1469 auto procIte = m_procOwnedCleanupCallbacks.find(puid);
1470 if (procIte != m_procOwnedCleanupCallbacks.end()) {
1471 for (auto it : procIte->second) {
1472 callbacks.push_back(it.second);
1473 }
1474 m_procOwnedCleanupCallbacks.erase(procIte);
1475 }
1476 }
1477
1478 mutex.unlock();
1479
1480 for (auto cb : callbacks) {
1481 cb();
1482 }
1483 }
1484
cleanupProcGLObjects_locked(uint64_t puid,bool forced)1485 std::vector<HandleType> FrameBuffer::cleanupProcGLObjects_locked(uint64_t puid, bool forced) {
1486 std::vector<HandleType> colorBuffersToCleanup;
1487 {
1488 std::unique_ptr<RecursiveScopedContextBind> bind = nullptr;
1489 #if GFXSTREAM_ENABLE_HOST_GLES
1490 if (m_emulationGl) {
1491 bind = std::make_unique<RecursiveScopedContextBind>(getPbufferSurfaceContextHelper());
1492 }
1493 // Clean up window surfaces
1494 if (m_emulationGl) {
1495 auto procIte = m_procOwnedEmulatedEglWindowSurfaces.find(puid);
1496 if (procIte != m_procOwnedEmulatedEglWindowSurfaces.end()) {
1497 for (auto whndl : procIte->second) {
1498 auto w = m_windows.find(whndl);
1499 // TODO(b/265186226): figure out if we are leaking?
1500 if (w == m_windows.end()) {
1501 continue;
1502 }
1503 if (!m_guestManagedColorBufferLifetime) {
1504 if (m_refCountPipeEnabled) {
1505 if (decColorBufferRefCountLocked(w->second.second)) {
1506 colorBuffersToCleanup.push_back(w->second.second);
1507 }
1508 } else {
1509 if (closeColorBufferLocked(w->second.second, forced)) {
1510 colorBuffersToCleanup.push_back(w->second.second);
1511 }
1512 }
1513 }
1514 m_windows.erase(w);
1515 }
1516 m_procOwnedEmulatedEglWindowSurfaces.erase(procIte);
1517 }
1518 }
1519 #endif
1520
1521 // Clean up color buffers.
1522 // A color buffer needs to be closed as many times as it is opened by
1523 // the guest process, to give the correct reference count.
1524 // (Note that a color buffer can be shared across guest processes.)
1525 {
1526 if (!m_guestManagedColorBufferLifetime) {
1527 auto procIte = m_procOwnedColorBuffers.find(puid);
1528 if (procIte != m_procOwnedColorBuffers.end()) {
1529 for (auto cb : procIte->second) {
1530 if (closeColorBufferLocked(cb, forced)) {
1531 colorBuffersToCleanup.push_back(cb);
1532 }
1533 }
1534 m_procOwnedColorBuffers.erase(procIte);
1535 }
1536 }
1537 }
1538
1539 #if GFXSTREAM_ENABLE_HOST_GLES
1540 // Clean up EGLImage handles
1541 if (m_emulationGl) {
1542 auto procImagesIt = m_procOwnedEmulatedEglImages.find(puid);
1543 if (procImagesIt != m_procOwnedEmulatedEglImages.end()) {
1544 for (auto image : procImagesIt->second) {
1545 m_images.erase(image);
1546 }
1547 m_procOwnedEmulatedEglImages.erase(procImagesIt);
1548 }
1549 }
1550 #endif
1551 }
1552
1553 #if GFXSTREAM_ENABLE_HOST_GLES
1554 // Unbind before cleaning up contexts
1555 // Cleanup render contexts
1556 if (m_emulationGl) {
1557 auto procIte = m_procOwnedEmulatedEglContexts.find(puid);
1558 if (procIte != m_procOwnedEmulatedEglContexts.end()) {
1559 for (auto ctx : procIte->second) {
1560 m_contexts.erase(ctx);
1561 }
1562 m_procOwnedEmulatedEglContexts.erase(procIte);
1563 }
1564 }
1565 #endif
1566
1567 return colorBuffersToCleanup;
1568 }
1569
markOpened(ColorBufferRef * cbRef)1570 void FrameBuffer::markOpened(ColorBufferRef* cbRef) {
1571 cbRef->opened = true;
1572 eraseDelayedCloseColorBufferLocked(cbRef->cb->getHndl(), cbRef->closedTs);
1573 cbRef->closedTs = 0;
1574 }
1575
readBuffer(HandleType handle,uint64_t offset,uint64_t size,void * bytes)1576 void FrameBuffer::readBuffer(HandleType handle, uint64_t offset, uint64_t size, void* bytes) {
1577 AutoLock mutex(m_lock);
1578
1579 BufferPtr buffer = findBuffer(handle);
1580 if (!buffer) {
1581 ERR("Failed to read buffer: buffer %d not found.", handle);
1582 return;
1583 }
1584
1585 buffer->readToBytes(offset, size, bytes);
1586 }
1587
readColorBuffer(HandleType p_colorbuffer,int x,int y,int width,int height,GLenum format,GLenum type,void * pixels)1588 void FrameBuffer::readColorBuffer(HandleType p_colorbuffer, int x, int y, int width, int height,
1589 GLenum format, GLenum type, void* pixels) {
1590 AutoLock mutex(m_lock);
1591
1592 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
1593 if (!colorBuffer) {
1594 // bad colorbuffer handle
1595 return;
1596 }
1597
1598 colorBuffer->readToBytes(x, y, width, height, format, type, pixels);
1599 }
1600
readColorBufferYUV(HandleType p_colorbuffer,int x,int y,int width,int height,void * pixels,uint32_t pixels_size)1601 void FrameBuffer::readColorBufferYUV(HandleType p_colorbuffer, int x, int y, int width, int height,
1602 void* pixels, uint32_t pixels_size) {
1603 AutoLock mutex(m_lock);
1604
1605 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
1606 if (!colorBuffer) {
1607 // bad colorbuffer handle
1608 return;
1609 }
1610
1611 colorBuffer->readYuvToBytes(x, y, width, height, pixels, pixels_size);
1612 }
1613
updateBuffer(HandleType p_buffer,uint64_t offset,uint64_t size,void * bytes)1614 bool FrameBuffer::updateBuffer(HandleType p_buffer, uint64_t offset, uint64_t size, void* bytes) {
1615 AutoLock mutex(m_lock);
1616
1617 BufferPtr buffer = findBuffer(p_buffer);
1618 if (!buffer) {
1619 ERR("Failed to update buffer: buffer %d not found.", p_buffer);
1620 return false;
1621 }
1622
1623 return buffer->updateFromBytes(offset, size, bytes);
1624 }
1625
updateColorBuffer(HandleType p_colorbuffer,int x,int y,int width,int height,GLenum format,GLenum type,void * pixels)1626 bool FrameBuffer::updateColorBuffer(HandleType p_colorbuffer,
1627 int x,
1628 int y,
1629 int width,
1630 int height,
1631 GLenum format,
1632 GLenum type,
1633 void* pixels) {
1634 if (width == 0 || height == 0) {
1635 return false;
1636 }
1637
1638 AutoLock mutex(m_lock);
1639
1640 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
1641 if (!colorBuffer) {
1642 // bad colorbuffer handle
1643 return false;
1644 }
1645
1646 colorBuffer->updateFromBytes(x, y, width, height, format, type, pixels);
1647
1648 return true;
1649 }
1650
updateColorBufferFromFrameworkFormat(HandleType p_colorbuffer,int x,int y,int width,int height,FrameworkFormat fwkFormat,GLenum format,GLenum type,void * pixels,void * metadata)1651 bool FrameBuffer::updateColorBufferFromFrameworkFormat(HandleType p_colorbuffer, int x, int y,
1652 int width, int height,
1653 FrameworkFormat fwkFormat, GLenum format,
1654 GLenum type, void* pixels, void* metadata) {
1655 if (width == 0 || height == 0) {
1656 return false;
1657 }
1658
1659 AutoLock mutex(m_lock);
1660
1661 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
1662 if (c == m_colorbuffers.end()) {
1663 // bad colorbuffer handle
1664 return false;
1665 }
1666
1667 (*c).second.cb->updateFromBytes(x, y, width, height, fwkFormat, format, type, pixels, metadata);
1668 return true;
1669 }
1670
getColorBufferInfo(HandleType p_colorbuffer,int * width,int * height,GLint * internalformat,FrameworkFormat * frameworkFormat)1671 bool FrameBuffer::getColorBufferInfo(
1672 HandleType p_colorbuffer, int* width, int* height, GLint* internalformat,
1673 FrameworkFormat* frameworkFormat) {
1674
1675 AutoLock mutex(m_lock);
1676
1677 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
1678 if (!colorBuffer) {
1679 // bad colorbuffer handle
1680 return false;
1681 }
1682
1683 *width = colorBuffer->getWidth();
1684 *height = colorBuffer->getHeight();
1685 *internalformat = colorBuffer->getFormat();
1686 if (frameworkFormat) {
1687 *frameworkFormat = colorBuffer->getFrameworkFormat();
1688 }
1689
1690 return true;
1691 }
1692
getBufferInfo(HandleType p_buffer,int * size)1693 bool FrameBuffer::getBufferInfo(HandleType p_buffer, int* size) {
1694 AutoLock mutex(m_lock);
1695
1696 BufferMap::iterator c(m_buffers.find(p_buffer));
1697 if (c == m_buffers.end()) {
1698 // Bad buffer handle.
1699 return false;
1700 }
1701
1702 auto buf = (*c).second.buffer;
1703 *size = buf->getSize();
1704 return true;
1705 }
1706
post(HandleType p_colorbuffer,bool needLockAndBind)1707 bool FrameBuffer::post(HandleType p_colorbuffer, bool needLockAndBind) {
1708 #if GFXSTREAM_ENABLE_HOST_GLES
1709 if (m_guestUsesAngle) {
1710 flushColorBufferFromGl(p_colorbuffer);
1711 }
1712 #endif
1713
1714 auto res = postImplSync(p_colorbuffer, needLockAndBind);
1715 if (res) setGuestPostedAFrame();
1716 return res;
1717 }
1718
postWithCallback(HandleType p_colorbuffer,Post::CompletionCallback callback,bool needLockAndBind)1719 void FrameBuffer::postWithCallback(HandleType p_colorbuffer, Post::CompletionCallback callback,
1720 bool needLockAndBind) {
1721 #if GFXSTREAM_ENABLE_HOST_GLES
1722 if (m_guestUsesAngle) {
1723 flushColorBufferFromGl(p_colorbuffer);
1724 }
1725 #endif
1726
1727 AsyncResult res = postImpl(p_colorbuffer, callback, needLockAndBind);
1728 if (res.Succeeded()) {
1729 setGuestPostedAFrame();
1730 }
1731
1732 if (!res.CallbackScheduledOrFired()) {
1733 // If postImpl fails, we have not fired the callback. postWithCallback
1734 // should always ensure the callback fires.
1735 std::shared_future<void> callbackRes = std::async(std::launch::deferred, [] {});
1736 callback(callbackRes);
1737 }
1738 }
1739
postImplSync(HandleType p_colorbuffer,bool needLockAndBind,bool repaint)1740 bool FrameBuffer::postImplSync(HandleType p_colorbuffer, bool needLockAndBind, bool repaint) {
1741 std::promise<void> promise;
1742 std::future<void> completeFuture = promise.get_future();
1743 auto posted = postImpl(
1744 p_colorbuffer,
1745 [&](std::shared_future<void> waitForGpu) {
1746 waitForGpu.wait();
1747 promise.set_value();
1748 },
1749 needLockAndBind, repaint);
1750 if (posted.CallbackScheduledOrFired()) {
1751 completeFuture.wait();
1752 }
1753
1754 return posted.Succeeded();
1755 }
1756
postImpl(HandleType p_colorbuffer,Post::CompletionCallback callback,bool needLockAndBind,bool repaint)1757 AsyncResult FrameBuffer::postImpl(HandleType p_colorbuffer, Post::CompletionCallback callback,
1758 bool needLockAndBind, bool repaint) {
1759 std::unique_ptr<RecursiveScopedContextBind> bind;
1760 if (needLockAndBind) {
1761 m_lock.lock();
1762 #if GFXSTREAM_ENABLE_HOST_GLES
1763 if (m_emulationGl) {
1764 bind = std::make_unique<RecursiveScopedContextBind>(getPbufferSurfaceContextHelper());
1765 }
1766 #endif
1767 }
1768 AsyncResult ret = AsyncResult::FAIL_AND_CALLBACK_NOT_SCHEDULED;
1769
1770 ColorBufferPtr colorBuffer = nullptr;
1771 {
1772 AutoLock colorBufferMapLock(m_colorBufferMapLock);
1773 ColorBufferMap::iterator c = m_colorbuffers.find(p_colorbuffer);
1774 if (c != m_colorbuffers.end()) {
1775 colorBuffer = c->second.cb;
1776 c->second.refcount++;
1777 markOpened(&c->second);
1778 }
1779 }
1780 if (!colorBuffer) {
1781 goto EXIT;
1782 }
1783
1784 m_lastPostedColorBuffer = p_colorbuffer;
1785
1786 colorBuffer->touch();
1787 if (m_subWin) {
1788 Post postCmd;
1789 postCmd.cmd = PostCmd::Post;
1790 postCmd.cb = colorBuffer.get();
1791 postCmd.cbHandle = p_colorbuffer;
1792 postCmd.completionCallback = std::make_unique<Post::CompletionCallback>(callback);
1793 sendPostWorkerCmd(std::move(postCmd));
1794 ret = AsyncResult::OK_AND_CALLBACK_SCHEDULED;
1795 } else {
1796 // If there is no sub-window, don't display anything, the client will
1797 // rely on m_onPost to get the pixels instead.
1798 ret = AsyncResult::OK_AND_CALLBACK_NOT_SCHEDULED;
1799 }
1800
1801 //
1802 // output FPS and performance usage statistics
1803 //
1804 if (m_fpsStats) {
1805 long long currTime = android::base::getHighResTimeUs() / 1000;
1806 m_statsNumFrames++;
1807 if (currTime - m_statsStartTime >= 1000) {
1808 if (m_fpsStats) {
1809 float dt = (float)(currTime - m_statsStartTime) / 1000.0f;
1810 printf("FPS: %5.3f \n", (float)m_statsNumFrames / dt);
1811 m_statsNumFrames = 0;
1812 }
1813 m_statsStartTime = currTime;
1814 }
1815 }
1816
1817 //
1818 // Send framebuffer (without FPS overlay) to callback
1819 //
1820 if (m_onPost.size() == 0) {
1821 goto DEC_REFCOUNT_AND_EXIT;
1822 }
1823 for (auto& iter : m_onPost) {
1824 ColorBufferPtr cb;
1825 if (iter.first == 0) {
1826 cb = colorBuffer;
1827 } else {
1828 uint32_t colorBuffer;
1829 if (getDisplayColorBuffer(iter.first, &colorBuffer) < 0) {
1830 ERR("Failed to get color buffer for display %d, skip onPost", iter.first);
1831 continue;
1832 }
1833
1834 cb = findColorBuffer(colorBuffer);
1835 if (!cb) {
1836 ERR("Failed to find colorbuffer %d, skip onPost", colorBuffer);
1837 continue;
1838 }
1839 }
1840
1841 if (asyncReadbackSupported()) {
1842 ensureReadbackWorker();
1843 const auto status = m_readbackWorker->doNextReadback(
1844 iter.first, cb.get(), iter.second.img, repaint, iter.second.readBgra);
1845 if (status == ReadbackWorker::DoNextReadbackResult::OK_READY_FOR_READ) {
1846 doPostCallback(iter.second.img, iter.first);
1847 }
1848 } else {
1849 #if GFXSTREAM_ENABLE_HOST_GLES
1850 cb->glOpReadback(iter.second.img, iter.second.readBgra);
1851 #endif
1852 doPostCallback(iter.second.img, iter.first);
1853 }
1854 }
1855 DEC_REFCOUNT_AND_EXIT:
1856 if (!m_subWin) { // m_subWin is supposed to be false
1857 decColorBufferRefCountLocked(p_colorbuffer);
1858 }
1859
1860 EXIT:
1861 if (needLockAndBind) {
1862 bind.reset();
1863 m_lock.unlock();
1864 }
1865 return ret;
1866 }
1867
doPostCallback(void * pixels,uint32_t displayId)1868 void FrameBuffer::doPostCallback(void* pixels, uint32_t displayId) {
1869 const auto& iter = m_onPost.find(displayId);
1870 if (iter == m_onPost.end()) {
1871 ERR("Cannot find post callback function for display %d", displayId);
1872 return;
1873 }
1874 iter->second.cb(iter->second.context, displayId, iter->second.width, iter->second.height, -1,
1875 GL_RGBA, GL_UNSIGNED_BYTE, (unsigned char*)pixels);
1876 }
1877
getPixels(void * pixels,uint32_t bytes,uint32_t displayId)1878 void FrameBuffer::getPixels(void* pixels, uint32_t bytes, uint32_t displayId) {
1879 const auto& iter = m_onPost.find(displayId);
1880 if (iter == m_onPost.end()) {
1881 ERR("Display %d not configured for recording yet", displayId);
1882 return;
1883 }
1884 std::future<void> completeFuture =
1885 m_readbackThread.enqueue({ReadbackCmd::GetPixels, displayId, pixels, bytes});
1886 completeFuture.wait();
1887 }
1888
flushReadPipeline(int displayId)1889 void FrameBuffer::flushReadPipeline(int displayId) {
1890 const auto& iter = m_onPost.find(displayId);
1891 if (iter == m_onPost.end()) {
1892 ERR("Cannot find onPost pixels for display %d", displayId);
1893 return;
1894 }
1895
1896 ensureReadbackWorker();
1897
1898 const auto status = m_readbackWorker->flushPipeline(displayId);
1899 if (status == ReadbackWorker::FlushResult::OK_READY_FOR_READ) {
1900 doPostCallback(nullptr, displayId);
1901 }
1902 }
1903
ensureReadbackWorker()1904 void FrameBuffer::ensureReadbackWorker() {
1905 #if GFXSTREAM_ENABLE_HOST_GLES
1906 if (!m_readbackWorker) {
1907 if (!m_emulationGl) {
1908 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not enabled.";
1909 }
1910 m_readbackWorker = m_emulationGl->getReadbackWorker();
1911 }
1912 #endif
1913 }
1914
sFrameBuffer_ReadPixelsCallback(void * pixels,uint32_t bytes,uint32_t displayId)1915 static void sFrameBuffer_ReadPixelsCallback(void* pixels, uint32_t bytes, uint32_t displayId) {
1916 FrameBuffer::getFB()->getPixels(pixels, bytes, displayId);
1917 }
1918
sFrameBuffer_FlushReadPixelPipeline(int displayId)1919 static void sFrameBuffer_FlushReadPixelPipeline(int displayId) {
1920 FrameBuffer::getFB()->flushReadPipeline(displayId);
1921 }
1922
asyncReadbackSupported()1923 bool FrameBuffer::asyncReadbackSupported() {
1924 #if GFXSTREAM_ENABLE_HOST_GLES
1925 return m_emulationGl && m_emulationGl->isAsyncReadbackSupported();
1926 #else
1927 return false;
1928 #endif
1929 }
1930
getReadPixelsCallback()1931 Renderer::ReadPixelsCallback FrameBuffer::getReadPixelsCallback() {
1932 return sFrameBuffer_ReadPixelsCallback;
1933 }
1934
getFlushReadPixelPipeline()1935 Renderer::FlushReadPixelPipeline FrameBuffer::getFlushReadPixelPipeline() {
1936 return sFrameBuffer_FlushReadPixelPipeline;
1937 }
1938
repost(bool needLockAndBind)1939 bool FrameBuffer::repost(bool needLockAndBind) {
1940 GL_LOG("Reposting framebuffer.");
1941 if (m_displayVk) {
1942 setGuestPostedAFrame();
1943 return true;
1944 }
1945 if (m_lastPostedColorBuffer && sInitialized.load(std::memory_order_relaxed)) {
1946 GL_LOG("Has last posted colorbuffer and is initialized; post.");
1947 auto res = postImplSync(m_lastPostedColorBuffer, needLockAndBind, true);
1948 if (res) setGuestPostedAFrame();
1949 return res;
1950 } else {
1951 GL_LOG("No repost: no last posted color buffer");
1952 if (!sInitialized.load(std::memory_order_relaxed)) {
1953 GL_LOG("No repost: initialization is not finished.");
1954 }
1955 }
1956 return false;
1957 }
1958
1959 template <class Collection>
saveProcOwnedCollection(Stream * stream,const Collection & c)1960 static void saveProcOwnedCollection(Stream* stream, const Collection& c) {
1961 // Exclude empty handle lists from saving as they add no value but only
1962 // increase the snapshot size; keep the format compatible with
1963 // android::base::saveCollection() though.
1964 const int count = std::count_if(
1965 c.begin(), c.end(),
1966 [](const typename Collection::value_type& pair) { return !pair.second.empty(); });
1967 stream->putBe32(count);
1968 for (const auto& pair : c) {
1969 if (pair.second.empty()) {
1970 continue;
1971 }
1972 stream->putBe64(pair.first);
1973 saveCollection(stream, pair.second, [](Stream* s, HandleType h) { s->putBe32(h); });
1974 }
1975 }
1976
1977 template <class Collection>
loadProcOwnedCollection(Stream * stream,Collection * c)1978 static void loadProcOwnedCollection(Stream* stream, Collection* c) {
1979 loadCollection(stream, c, [](Stream* stream) -> typename Collection::value_type {
1980 const int processId = stream->getBe64();
1981 typename Collection::mapped_type handles;
1982 loadCollection(stream, &handles, [](Stream* s) { return s->getBe32(); });
1983 return {processId, std::move(handles)};
1984 });
1985 }
1986
getScreenshot(unsigned int nChannels,unsigned int * width,unsigned int * height,uint8_t * pixels,size_t * cPixels,int displayId,int desiredWidth,int desiredHeight,int desiredRotation,Rect rect)1987 int FrameBuffer::getScreenshot(unsigned int nChannels, unsigned int* width, unsigned int* height,
1988 uint8_t* pixels, size_t* cPixels, int displayId, int desiredWidth,
1989 int desiredHeight, int desiredRotation, Rect rect) {
1990 #ifdef CONFIG_AEMU
1991 if (emugl::shouldSkipDraw()) {
1992 *width = 0;
1993 *height = 0;
1994 *cPixels = 0;
1995 return -1;
1996 }
1997 #endif
1998
1999 AutoLock mutex(m_lock);
2000 uint32_t w, h, cb, screenWidth, screenHeight;
2001 if (!emugl::get_emugl_multi_display_operations().getMultiDisplay(
2002 displayId, nullptr, nullptr, &w, &h, nullptr, nullptr, nullptr)) {
2003 ERR("Screenshot of invalid display %d", displayId);
2004 *width = 0;
2005 *height = 0;
2006 *cPixels = 0;
2007 return -1;
2008 }
2009 if (nChannels != 3 && nChannels != 4) {
2010 ERR("Screenshot only support 3(RGB) or 4(RGBA) channels");
2011 *width = 0;
2012 *height = 0;
2013 *cPixels = 0;
2014 return -1;
2015 }
2016 emugl::get_emugl_multi_display_operations().getDisplayColorBuffer(displayId, &cb);
2017 if (displayId == 0) {
2018 cb = m_lastPostedColorBuffer;
2019 }
2020 ColorBufferPtr colorBuffer = findColorBuffer(cb);
2021 if (!colorBuffer) {
2022 *width = 0;
2023 *height = 0;
2024 *cPixels = 0;
2025 return -1;
2026 }
2027
2028 screenWidth = (desiredWidth == 0) ? w : desiredWidth;
2029 screenHeight = (desiredHeight == 0) ? h : desiredHeight;
2030
2031 bool useSnipping = (rect.size.w != 0 && rect.size.h != 0);
2032 if (useSnipping) {
2033 if (desiredWidth == 0 || desiredHeight == 0) {
2034 ERR("Must provide non-zero desiredWidth and desireRectanlge "
2035 "when using rectangle snipping");
2036 *width = 0;
2037 *height = 0;
2038 *cPixels = 0;
2039 return -1;
2040 }
2041 if ((rect.pos.x < 0 || rect.pos.y < 0) ||
2042 (desiredWidth < rect.pos.x + rect.size.w || desiredHeight < rect.pos.y + rect.size.h)) {
2043 return -1;
2044 }
2045 }
2046
2047 if (useSnipping) {
2048 *width = rect.size.w;
2049 *height = rect.size.h;
2050 } else {
2051 *width = screenWidth;
2052 *height = screenHeight;
2053 }
2054
2055 int needed =
2056 useSnipping ? (nChannels * rect.size.w * rect.size.h) : (nChannels * (*width) * (*height));
2057
2058 if (*cPixels < needed) {
2059 *cPixels = needed;
2060 return -2;
2061 }
2062 *cPixels = needed;
2063 if (desiredRotation == SKIN_ROTATION_90 || desiredRotation == SKIN_ROTATION_270) {
2064 std::swap(*width, *height);
2065 std::swap(screenWidth, screenHeight);
2066 std::swap(rect.size.w, rect.size.h);
2067 }
2068 // Transform the x, y coordinates given the rotation.
2069 // Assume (0, 0) represents the top left corner of the screen.
2070 if (useSnipping) {
2071 int x, y;
2072 switch (desiredRotation) {
2073 case SKIN_ROTATION_0:
2074 x = rect.pos.x;
2075 y = rect.pos.y;
2076 break;
2077 case SKIN_ROTATION_90:
2078 x = rect.pos.y;
2079 y = rect.pos.x;
2080 break;
2081 case SKIN_ROTATION_180:
2082 x = screenWidth - rect.pos.x - rect.size.w;
2083 y = rect.pos.y;
2084 break;
2085 case SKIN_ROTATION_270:
2086 x = rect.pos.y;
2087 y = screenHeight - rect.pos.x - rect.size.h;
2088 break;
2089 }
2090 rect.pos.x = x;
2091 rect.pos.y = y;
2092 }
2093
2094 GLenum format = nChannels == 3 ? GL_RGB : GL_RGBA;
2095 Post scrCmd;
2096 scrCmd.cmd = PostCmd::Screenshot;
2097 scrCmd.screenshot.cb = colorBuffer.get();
2098 scrCmd.screenshot.screenwidth = screenWidth;
2099 scrCmd.screenshot.screenheight = screenHeight;
2100 scrCmd.screenshot.format = format;
2101 scrCmd.screenshot.type = GL_UNSIGNED_BYTE;
2102 scrCmd.screenshot.rotation = desiredRotation;
2103 scrCmd.screenshot.pixels = pixels;
2104 scrCmd.screenshot.rect = rect;
2105
2106 std::future<void> completeFuture = sendPostWorkerCmd(std::move(scrCmd));
2107
2108 mutex.unlock();
2109 completeFuture.wait();
2110 return 0;
2111 }
2112
onLastColorBufferRef(uint32_t handle)2113 void FrameBuffer::onLastColorBufferRef(uint32_t handle) {
2114 if (!mOutstandingColorBufferDestroys.trySend((HandleType)handle)) {
2115 ERR("warning: too many outstanding "
2116 "color buffer destroys. leaking handle 0x%x",
2117 handle);
2118 }
2119 }
2120
decColorBufferRefCountLocked(HandleType p_colorbuffer)2121 bool FrameBuffer::decColorBufferRefCountLocked(HandleType p_colorbuffer) {
2122 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2123 const auto& it = m_colorbuffers.find(p_colorbuffer);
2124 if (it != m_colorbuffers.end()) {
2125 it->second.refcount -= 1;
2126 if (it->second.refcount == 0) {
2127 m_colorbuffers.erase(p_colorbuffer);
2128 return true;
2129 }
2130 }
2131 return false;
2132 }
2133
compose(uint32_t bufferSize,void * buffer,bool needPost)2134 bool FrameBuffer::compose(uint32_t bufferSize, void* buffer, bool needPost) {
2135 std::promise<void> promise;
2136 std::future<void> completeFuture = promise.get_future();
2137 auto composeRes =
2138 composeWithCallback(bufferSize, buffer, [&](std::shared_future<void> waitForGpu) {
2139 waitForGpu.wait();
2140 promise.set_value();
2141 });
2142 if (!composeRes.Succeeded()) {
2143 return false;
2144 }
2145
2146 if (composeRes.CallbackScheduledOrFired()) {
2147 completeFuture.wait();
2148 }
2149
2150 const auto& multiDisplay = emugl::get_emugl_multi_display_operations();
2151 const bool is_pixel_fold = multiDisplay.isPixelFold();
2152 if (needPost) {
2153 // AEMU with -no-window mode uses this code path.
2154 ComposeDevice* composeDevice = (ComposeDevice*)buffer;
2155
2156 switch (composeDevice->version) {
2157 case 1: {
2158 post(composeDevice->targetHandle, true);
2159 break;
2160 }
2161 case 2: {
2162 ComposeDevice_v2* composeDeviceV2 = (ComposeDevice_v2*)buffer;
2163 if (is_pixel_fold || composeDeviceV2->displayId == 0) {
2164 post(composeDeviceV2->targetHandle, true);
2165 }
2166 break;
2167 }
2168 default: {
2169 return false;
2170 }
2171 }
2172 }
2173 return true;
2174 }
2175
composeWithCallback(uint32_t bufferSize,void * buffer,Post::CompletionCallback callback)2176 AsyncResult FrameBuffer::composeWithCallback(uint32_t bufferSize, void* buffer,
2177 Post::CompletionCallback callback) {
2178 ComposeDevice* p = (ComposeDevice*)buffer;
2179 AutoLock mutex(m_lock);
2180
2181 switch (p->version) {
2182 case 1: {
2183 Post composeCmd;
2184 composeCmd.composeVersion = 1;
2185 composeCmd.composeBuffer.resize(bufferSize);
2186 memcpy(composeCmd.composeBuffer.data(), buffer, bufferSize);
2187 composeCmd.completionCallback = std::make_unique<Post::CompletionCallback>(callback);
2188 composeCmd.cmd = PostCmd::Compose;
2189 sendPostWorkerCmd(std::move(composeCmd));
2190 return AsyncResult::OK_AND_CALLBACK_SCHEDULED;
2191 }
2192
2193 case 2: {
2194 // support for multi-display
2195 ComposeDevice_v2* p2 = (ComposeDevice_v2*)buffer;
2196 if (p2->displayId != 0) {
2197 mutex.unlock();
2198 setDisplayColorBuffer(p2->displayId, p2->targetHandle);
2199 mutex.lock();
2200 }
2201 Post composeCmd;
2202 composeCmd.composeVersion = 2;
2203 composeCmd.composeBuffer.resize(bufferSize);
2204 memcpy(composeCmd.composeBuffer.data(), buffer, bufferSize);
2205 composeCmd.completionCallback = std::make_unique<Post::CompletionCallback>(callback);
2206 composeCmd.cmd = PostCmd::Compose;
2207 sendPostWorkerCmd(std::move(composeCmd));
2208 return AsyncResult::OK_AND_CALLBACK_SCHEDULED;
2209 }
2210
2211 default:
2212 ERR("yet to handle composition device version: %d", p->version);
2213 return AsyncResult::FAIL_AND_CALLBACK_NOT_SCHEDULED;
2214 }
2215 }
2216
onSave(Stream * stream,const android::snapshot::ITextureSaverPtr & textureSaver)2217 void FrameBuffer::onSave(Stream* stream, const android::snapshot::ITextureSaverPtr& textureSaver) {
2218 // Things we do not need to snapshot:
2219 // m_eglSurface
2220 // m_eglContext
2221 // m_pbufSurface
2222 // m_pbufContext
2223 // m_prevContext
2224 // m_prevReadSurf
2225 // m_prevDrawSurf
2226 AutoLock mutex(m_lock);
2227
2228 std::unique_ptr<RecursiveScopedContextBind> bind;
2229 #if GFXSTREAM_ENABLE_HOST_GLES
2230 if (m_emulationGl) {
2231 // Some snapshot commands try using GL.
2232 bind = std::make_unique<RecursiveScopedContextBind>(getPbufferSurfaceContextHelper());
2233 if (!bind->isOk()) {
2234 ERR("Failed to make context current for saving snapshot.");
2235 }
2236
2237 // eglPreSaveContext labels all guest context textures to be saved
2238 // (textures created by the host are not saved!)
2239 // eglSaveAllImages labels all EGLImages (both host and guest) to be saved
2240 // and save all labeled textures and EGLImages.
2241 if (s_egl.eglPreSaveContext && s_egl.eglSaveAllImages) {
2242 for (const auto& ctx : m_contexts) {
2243 s_egl.eglPreSaveContext(getDisplay(), ctx.second->getEGLContext(), stream);
2244 }
2245 s_egl.eglSaveAllImages(getDisplay(), stream, &textureSaver);
2246 }
2247 }
2248 #endif
2249
2250 // Don't save subWindow's x/y/w/h here - those are related to the current
2251 // emulator UI state, not guest state that we're saving.
2252 stream->putBe32(m_framebufferWidth);
2253 stream->putBe32(m_framebufferHeight);
2254 stream->putFloat(m_dpr);
2255 stream->putBe32(mDisplayActiveConfigId);
2256 saveCollection(stream, mDisplayConfigs,
2257 [](Stream* s, const std::map<int, DisplayConfig>::value_type& pair) {
2258 s->putBe32(pair.first);
2259 s->putBe32(pair.second.w);
2260 s->putBe32(pair.second.h);
2261 s->putBe32(pair.second.dpiX);
2262 s->putBe32(pair.second.dpiY);
2263 });
2264
2265 stream->putBe32(m_useSubWindow);
2266 stream->putBe32(/*Obsolete m_eglContextInitialized =*/1);
2267
2268 stream->putBe32(m_fpsStats);
2269 stream->putBe32(m_statsNumFrames);
2270 stream->putBe64(m_statsStartTime);
2271
2272 // Save all contexts.
2273 // Note: some of the contexts might not be restored yet. In such situation
2274 // we skip reading from GPU (for non-texture objects) or force a restore in
2275 // previous eglPreSaveContext and eglSaveAllImages calls (for texture
2276 // objects).
2277 // TODO: skip reading from GPU even for texture objects.
2278 #if GFXSTREAM_ENABLE_HOST_GLES
2279 saveCollection(
2280 stream, m_contexts,
2281 [](Stream* s, const EmulatedEglContextMap::value_type& pair) { pair.second->onSave(s); });
2282 #endif
2283
2284 // We don't need to save |m_colorBufferCloseTsMap| here - there's enough
2285 // information to reconstruct it when loading.
2286 uint64_t now = android::base::getUnixTimeUs();
2287
2288 {
2289 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2290 stream->putByte(m_guestManagedColorBufferLifetime);
2291 saveCollection(stream, m_colorbuffers,
2292 [now](Stream* s, const ColorBufferMap::value_type& pair) {
2293 pair.second.cb->onSave(s);
2294 s->putBe32(pair.second.refcount);
2295 s->putByte(pair.second.opened);
2296 s->putBe32(std::max<uint64_t>(0, now - pair.second.closedTs));
2297 });
2298 }
2299 stream->putBe32(m_lastPostedColorBuffer);
2300 #if GFXSTREAM_ENABLE_HOST_GLES
2301 saveCollection(stream, m_windows,
2302 [](Stream* s, const EmulatedEglWindowSurfaceMap::value_type& pair) {
2303 pair.second.first->onSave(s);
2304 s->putBe32(pair.second.second); // Color buffer handle.
2305 });
2306 #endif
2307
2308 #if GFXSTREAM_ENABLE_HOST_GLES
2309 saveProcOwnedCollection(stream, m_procOwnedEmulatedEglWindowSurfaces);
2310 #endif
2311 saveProcOwnedCollection(stream, m_procOwnedColorBuffers);
2312 #if GFXSTREAM_ENABLE_HOST_GLES
2313 saveProcOwnedCollection(stream, m_procOwnedEmulatedEglImages);
2314 saveProcOwnedCollection(stream, m_procOwnedEmulatedEglContexts);
2315 #endif
2316
2317 // TODO(b/309858017): remove if when ready to bump snapshot version
2318 if (m_features.VulkanSnapshots.enabled) {
2319 stream->putBe64(m_procOwnedResources.size());
2320 for (const auto& element : m_procOwnedResources) {
2321 stream->putBe64(element.first);
2322 stream->putBe32(element.second->getSequenceNumberPtr()->load());
2323 }
2324 }
2325
2326 // Save Vulkan state
2327 if (m_features.VulkanSnapshots.enabled && vk::VkDecoderGlobalState::get()) {
2328 vk::VkDecoderGlobalState::get()->save(stream);
2329 }
2330
2331 #if GFXSTREAM_ENABLE_HOST_GLES
2332 if (m_emulationGl) {
2333 if (s_egl.eglPostSaveContext) {
2334 for (const auto& ctx : m_contexts) {
2335 s_egl.eglPostSaveContext(getDisplay(), ctx.second->getEGLContext(), stream);
2336 }
2337 // We need to run the post save step for m_eglContext
2338 // to mark their texture handles dirty
2339 if (getContext() != EGL_NO_CONTEXT) {
2340 s_egl.eglPostSaveContext(getDisplay(), getContext(), stream);
2341 }
2342 }
2343
2344 EmulatedEglFenceSync::onSave(stream);
2345 }
2346 #endif
2347 }
2348
onLoad(Stream * stream,const android::snapshot::ITextureLoaderPtr & textureLoader)2349 bool FrameBuffer::onLoad(Stream* stream,
2350 const android::snapshot::ITextureLoaderPtr& textureLoader) {
2351 AutoLock lock(m_lock);
2352 // cleanups
2353 {
2354 sweepColorBuffersLocked();
2355
2356 std::unique_ptr<RecursiveScopedContextBind> bind;
2357 #if GFXSTREAM_ENABLE_HOST_GLES
2358 if (m_emulationGl) {
2359 // Some snapshot commands try using GL.
2360 bind = std::make_unique<RecursiveScopedContextBind>(getPbufferSurfaceContextHelper());
2361 if (!bind->isOk()) {
2362 ERR("Failed to make context current for loading snapshot.");
2363 }
2364 }
2365 #endif
2366
2367 bool cleanupComplete = false;
2368 {
2369 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2370 if (m_procOwnedCleanupCallbacks.empty() && m_procOwnedColorBuffers.empty() &&
2371 #if GFXSTREAM_ENABLE_HOST_GLES
2372 m_procOwnedEmulatedEglContexts.empty() && m_procOwnedEmulatedEglImages.empty() &&
2373 m_procOwnedEmulatedEglWindowSurfaces.empty() &&
2374 #endif
2375 (
2376 #if GFXSTREAM_ENABLE_HOST_GLES
2377 !m_contexts.empty() || !m_windows.empty() ||
2378 #endif
2379 m_colorbuffers.size() > m_colorBufferDelayedCloseList.size())) {
2380 // we are likely on a legacy system image, which does not have
2381 // process owned objects. We need to force cleanup everything
2382 #if GFXSTREAM_ENABLE_HOST_GLES
2383 m_contexts.clear();
2384 m_windows.clear();
2385 #endif
2386 m_colorbuffers.clear();
2387 cleanupComplete = true;
2388 }
2389 }
2390 if (!cleanupComplete) {
2391 std::vector<HandleType> colorBuffersToCleanup;
2392
2393 #if GFXSTREAM_ENABLE_HOST_GLES
2394 while (m_procOwnedEmulatedEglWindowSurfaces.size()) {
2395 auto cleanupHandles = cleanupProcGLObjects_locked(
2396 m_procOwnedEmulatedEglWindowSurfaces.begin()->first, true);
2397 colorBuffersToCleanup.insert(colorBuffersToCleanup.end(), cleanupHandles.begin(),
2398 cleanupHandles.end());
2399 }
2400 #endif
2401 while (m_procOwnedColorBuffers.size()) {
2402 auto cleanupHandles =
2403 cleanupProcGLObjects_locked(m_procOwnedColorBuffers.begin()->first, true);
2404 colorBuffersToCleanup.insert(colorBuffersToCleanup.end(), cleanupHandles.begin(),
2405 cleanupHandles.end());
2406 }
2407 #if GFXSTREAM_ENABLE_HOST_GLES
2408 while (m_procOwnedEmulatedEglImages.size()) {
2409 auto cleanupHandles =
2410 cleanupProcGLObjects_locked(m_procOwnedEmulatedEglImages.begin()->first, true);
2411 colorBuffersToCleanup.insert(colorBuffersToCleanup.end(), cleanupHandles.begin(),
2412 cleanupHandles.end());
2413 }
2414 while (m_procOwnedEmulatedEglContexts.size()) {
2415 auto cleanupHandles = cleanupProcGLObjects_locked(
2416 m_procOwnedEmulatedEglContexts.begin()->first, true);
2417 colorBuffersToCleanup.insert(colorBuffersToCleanup.end(), cleanupHandles.begin(),
2418 cleanupHandles.end());
2419 }
2420 #endif
2421
2422 std::vector<std::function<void()>> cleanupCallbacks;
2423
2424 while (m_procOwnedCleanupCallbacks.size()) {
2425 auto it = m_procOwnedCleanupCallbacks.begin();
2426 while (it != m_procOwnedCleanupCallbacks.end()) {
2427 for (auto it2 : it->second) {
2428 cleanupCallbacks.push_back(it2.second);
2429 }
2430 it = m_procOwnedCleanupCallbacks.erase(it);
2431 }
2432 }
2433
2434 m_procOwnedResources.clear();
2435
2436 performDelayedColorBufferCloseLocked(true);
2437
2438 lock.unlock();
2439
2440 for (auto cb : cleanupCallbacks) {
2441 cb();
2442 }
2443
2444 lock.lock();
2445 cleanupComplete = true;
2446 }
2447 m_colorBufferDelayedCloseList.clear();
2448 #if GFXSTREAM_ENABLE_HOST_GLES
2449 assert(m_contexts.empty());
2450 assert(m_windows.empty());
2451 #endif
2452 {
2453 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2454 if (!m_colorbuffers.empty()) {
2455 ERR("warning: on load, stale colorbuffers: %zu", m_colorbuffers.size());
2456 m_colorbuffers.clear();
2457 }
2458 assert(m_colorbuffers.empty());
2459 }
2460 #ifdef SNAPSHOT_PROFILE
2461 uint64_t texTime = android::base::getUnixTimeUs();
2462 #endif
2463 #if GFXSTREAM_ENABLE_HOST_GLES
2464 if (m_emulationGl) {
2465 if (s_egl.eglLoadAllImages) {
2466 s_egl.eglLoadAllImages(getDisplay(), stream, &textureLoader);
2467 }
2468 }
2469 #endif
2470 #ifdef SNAPSHOT_PROFILE
2471 printf("Texture load time: %lld ms\n",
2472 (long long)(android::base::getUnixTimeUs() - texTime) / 1000);
2473 #endif
2474 }
2475 // See comment about subwindow position in onSave().
2476 m_framebufferWidth = stream->getBe32();
2477 m_framebufferHeight = stream->getBe32();
2478 m_dpr = stream->getFloat();
2479 mDisplayActiveConfigId = stream->getBe32();
2480 loadCollection(stream, &mDisplayConfigs,
2481 [](Stream* s) -> std::map<int, DisplayConfig>::value_type {
2482 int idx = static_cast<int>(s->getBe32());
2483 int w = static_cast<int>(s->getBe32());
2484 int h = static_cast<int>(s->getBe32());
2485 int dpiX = static_cast<int>(s->getBe32());
2486 int dpiY = static_cast<int>(s->getBe32());
2487 return {idx, {w, h, dpiX, dpiY}};
2488 });
2489
2490 // TODO: resize the window
2491 //
2492 m_useSubWindow = stream->getBe32();
2493 /*Obsolete m_eglContextInitialized =*/stream->getBe32();
2494
2495 m_fpsStats = stream->getBe32();
2496 m_statsNumFrames = stream->getBe32();
2497 m_statsStartTime = stream->getBe64();
2498
2499 #if GFXSTREAM_ENABLE_HOST_GLES
2500 loadCollection(
2501 stream, &m_contexts, [this](Stream* stream) -> EmulatedEglContextMap::value_type {
2502 if (!m_emulationGl) {
2503 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not enabled.";
2504 }
2505
2506 auto context = m_emulationGl->loadEmulatedEglContext(stream);
2507 auto contextHandle = context ? context->getHndl() : 0;
2508 return {contextHandle, std::move(context)};
2509 });
2510 assert(!android::base::find(m_contexts, 0));
2511 #endif
2512
2513 auto now = android::base::getUnixTimeUs();
2514 {
2515 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2516 m_guestManagedColorBufferLifetime = stream->getByte();
2517 loadCollection(
2518 stream, &m_colorbuffers, [this, now](Stream* stream) -> ColorBufferMap::value_type {
2519 ColorBufferPtr cb = ColorBuffer::onLoad(m_emulationGl.get(), m_emulationVk, stream);
2520 const HandleType handle = cb->getHndl();
2521 const unsigned refCount = stream->getBe32();
2522 const bool opened = stream->getByte();
2523 const uint64_t closedTs = now - stream->getBe32();
2524 if (refCount == 0) {
2525 m_colorBufferDelayedCloseList.push_back({closedTs, handle});
2526 }
2527 return {handle, ColorBufferRef{std::move(cb), refCount, opened, closedTs}};
2528 });
2529 }
2530 m_lastPostedColorBuffer = static_cast<HandleType>(stream->getBe32());
2531 GL_LOG("Got lasted posted color buffer from snapshot");
2532
2533 {
2534 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2535 #if GFXSTREAM_ENABLE_HOST_GLES
2536 loadCollection(
2537 stream, &m_windows, [this](Stream* stream) -> EmulatedEglWindowSurfaceMap::value_type {
2538 if (!m_emulationGl) {
2539 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
2540 << "GL/EGL emulation not enabled.";
2541 }
2542
2543 auto window =
2544 m_emulationGl->loadEmulatedEglWindowSurface(stream, m_colorbuffers, m_contexts);
2545
2546 HandleType handle = window->getHndl();
2547 HandleType colorBufferHandle = stream->getBe32();
2548 return {handle, {std::move(window), colorBufferHandle}};
2549 });
2550 #endif
2551 }
2552
2553 #if GFXSTREAM_ENABLE_HOST_GLES
2554 loadProcOwnedCollection(stream, &m_procOwnedEmulatedEglWindowSurfaces);
2555 #endif
2556 loadProcOwnedCollection(stream, &m_procOwnedColorBuffers);
2557 #if GFXSTREAM_ENABLE_HOST_GLES
2558 loadProcOwnedCollection(stream, &m_procOwnedEmulatedEglImages);
2559 loadProcOwnedCollection(stream, &m_procOwnedEmulatedEglContexts);
2560 #endif
2561 // TODO(b/309858017): remove if when ready to bump snapshot version
2562 if (m_features.VulkanSnapshots.enabled) {
2563 size_t resourceCount = stream->getBe64();
2564 for (size_t i = 0; i < resourceCount; i++) {
2565 uint64_t puid = stream->getBe64();
2566 uint32_t sequenceNumber = stream->getBe32();
2567 std::unique_ptr<ProcessResources> processResources = ProcessResources::create();
2568 processResources->getSequenceNumberPtr()->store(sequenceNumber);
2569 m_procOwnedResources.emplace(puid, std::move(processResources));
2570 }
2571 }
2572
2573 #if GFXSTREAM_ENABLE_HOST_GLES
2574 if (m_emulationGl) {
2575 if (s_egl.eglPostLoadAllImages) {
2576 s_egl.eglPostLoadAllImages(getDisplay(), stream);
2577 }
2578 }
2579
2580 registerTriggerWait();
2581 #endif
2582
2583 {
2584 std::unique_ptr<RecursiveScopedContextBind> bind;
2585 #if GFXSTREAM_ENABLE_HOST_GLES
2586 if (m_emulationGl) {
2587 // Some snapshot commands try using GL.
2588 bind = std::make_unique<RecursiveScopedContextBind>(getPbufferSurfaceContextHelper());
2589 if (!bind->isOk()) {
2590 ERR("Failed to make context current for loading snapshot.");
2591 }
2592 }
2593 #endif
2594
2595 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2596 for (auto& it : m_colorbuffers) {
2597 if (it.second.cb) {
2598 it.second.cb->touch();
2599 }
2600 }
2601 }
2602
2603 // Restore Vulkan state
2604 if (m_features.VulkanSnapshots.enabled && vk::VkDecoderGlobalState::get()) {
2605 lock.unlock();
2606 GfxApiLogger gfxLogger;
2607 vk::VkDecoderGlobalState::get()->load(stream, gfxLogger, m_healthMonitor.get());
2608 lock.lock();
2609 }
2610
2611 repost(false);
2612
2613 #if GFXSTREAM_ENABLE_HOST_GLES
2614 if (m_emulationGl) {
2615 EmulatedEglFenceSync::onLoad(stream);
2616 }
2617 #endif
2618
2619 return true;
2620 // TODO: restore memory management
2621 }
2622
lock()2623 void FrameBuffer::lock() { m_lock.lock(); }
2624
unlock()2625 void FrameBuffer::unlock() { m_lock.unlock(); }
2626
findColorBuffer(HandleType p_colorbuffer)2627 ColorBufferPtr FrameBuffer::findColorBuffer(HandleType p_colorbuffer) {
2628 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2629 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
2630 if (c == m_colorbuffers.end()) {
2631 return nullptr;
2632 } else {
2633 return c->second.cb;
2634 }
2635 }
2636
findBuffer(HandleType p_buffer)2637 BufferPtr FrameBuffer::findBuffer(HandleType p_buffer) {
2638 AutoLock colorBufferMapLock(m_colorBufferMapLock);
2639 BufferMap::iterator b(m_buffers.find(p_buffer));
2640 if (b == m_buffers.end()) {
2641 return nullptr;
2642 } else {
2643 return b->second.buffer;
2644 }
2645 }
2646
registerProcessCleanupCallback(void * key,std::function<void ()> cb)2647 void FrameBuffer::registerProcessCleanupCallback(void* key, std::function<void()> cb) {
2648 AutoLock mutex(m_lock);
2649 RenderThreadInfo* tInfo = RenderThreadInfo::get();
2650 if (!tInfo) return;
2651
2652 auto& callbackMap = m_procOwnedCleanupCallbacks[tInfo->m_puid];
2653 callbackMap[key] = cb;
2654 }
2655
unregisterProcessCleanupCallback(void * key)2656 void FrameBuffer::unregisterProcessCleanupCallback(void* key) {
2657 AutoLock mutex(m_lock);
2658 RenderThreadInfo* tInfo = RenderThreadInfo::get();
2659 if (!tInfo) return;
2660
2661 auto& callbackMap = m_procOwnedCleanupCallbacks[tInfo->m_puid];
2662 if (callbackMap.find(key) == callbackMap.end()) {
2663 ERR("warning: tried to erase nonexistent key %p "
2664 "associated with process %llu",
2665 key, (unsigned long long)(tInfo->m_puid));
2666 }
2667 callbackMap.erase(key);
2668 }
2669
getProcessResources(uint64_t puid)2670 const ProcessResources* FrameBuffer::getProcessResources(uint64_t puid) {
2671 AutoLock mutex(m_lock);
2672 auto i = m_procOwnedResources.find(puid);
2673 if (i == m_procOwnedResources.end()) {
2674 ERR("Failed to find process owned resources for puid %" PRIu64 ".", puid);
2675 return nullptr;
2676 }
2677 return i->second.get();
2678 }
2679
createDisplay(uint32_t * displayId)2680 int FrameBuffer::createDisplay(uint32_t* displayId) {
2681 return emugl::get_emugl_multi_display_operations().createDisplay(displayId);
2682 }
2683
createDisplay(uint32_t displayId)2684 int FrameBuffer::createDisplay(uint32_t displayId) {
2685 return emugl::get_emugl_multi_display_operations().createDisplay(&displayId);
2686 }
2687
destroyDisplay(uint32_t displayId)2688 int FrameBuffer::destroyDisplay(uint32_t displayId) {
2689 return emugl::get_emugl_multi_display_operations().destroyDisplay(displayId);
2690 }
2691
setDisplayColorBuffer(uint32_t displayId,uint32_t colorBuffer)2692 int FrameBuffer::setDisplayColorBuffer(uint32_t displayId, uint32_t colorBuffer) {
2693 return emugl::get_emugl_multi_display_operations().setDisplayColorBuffer(displayId,
2694 colorBuffer);
2695 }
2696
getDisplayColorBuffer(uint32_t displayId,uint32_t * colorBuffer)2697 int FrameBuffer::getDisplayColorBuffer(uint32_t displayId, uint32_t* colorBuffer) {
2698 return emugl::get_emugl_multi_display_operations().getDisplayColorBuffer(displayId,
2699 colorBuffer);
2700 }
2701
getColorBufferDisplay(uint32_t colorBuffer,uint32_t * displayId)2702 int FrameBuffer::getColorBufferDisplay(uint32_t colorBuffer, uint32_t* displayId) {
2703 return emugl::get_emugl_multi_display_operations().getColorBufferDisplay(colorBuffer,
2704 displayId);
2705 }
2706
getDisplayPose(uint32_t displayId,int32_t * x,int32_t * y,uint32_t * w,uint32_t * h)2707 int FrameBuffer::getDisplayPose(uint32_t displayId, int32_t* x, int32_t* y, uint32_t* w,
2708 uint32_t* h) {
2709 return emugl::get_emugl_multi_display_operations().getDisplayPose(displayId, x, y, w, h);
2710 }
2711
setDisplayPose(uint32_t displayId,int32_t x,int32_t y,uint32_t w,uint32_t h,uint32_t dpi)2712 int FrameBuffer::setDisplayPose(uint32_t displayId, int32_t x, int32_t y, uint32_t w, uint32_t h,
2713 uint32_t dpi) {
2714 return emugl::get_emugl_multi_display_operations().setDisplayPose(displayId, x, y, w, h, dpi);
2715 }
2716
sweepColorBuffersLocked()2717 void FrameBuffer::sweepColorBuffersLocked() {
2718 HandleType handleToDestroy;
2719 while (mOutstandingColorBufferDestroys.tryReceive(&handleToDestroy)) {
2720 decColorBufferRefCountLocked(handleToDestroy);
2721 }
2722 }
2723
blockPostWorker(std::future<void> continueSignal)2724 std::future<void> FrameBuffer::blockPostWorker(std::future<void> continueSignal) {
2725 std::promise<void> scheduled;
2726 std::future<void> scheduledFuture = scheduled.get_future();
2727 Post postCmd = {
2728 .cmd = PostCmd::Block,
2729 .block = std::make_unique<Post::Block>(Post::Block{
2730 .scheduledSignal = std::move(scheduled),
2731 .continueSignal = std::move(continueSignal),
2732 }),
2733 };
2734 sendPostWorkerCmd(std::move(postCmd));
2735 return scheduledFuture;
2736 }
2737
waitForGpuVulkan(uint64_t deviceHandle,uint64_t fenceHandle)2738 void FrameBuffer::waitForGpuVulkan(uint64_t deviceHandle, uint64_t fenceHandle) {
2739 (void)deviceHandle;
2740 if (!m_emulationGl) {
2741 // Guest ANGLE should always use the asyncWaitForGpuVulkanWithCb call. EmulatedEglFenceSync
2742 // is a wrapper over EGLSyncKHR and should not be used for pure Vulkan environment.
2743 return;
2744 }
2745
2746 #if GFXSTREAM_ENABLE_HOST_GLES
2747 // Note: this will always be nullptr.
2748 EmulatedEglFenceSync* fenceSync = EmulatedEglFenceSync::getFromHandle(fenceHandle);
2749
2750 // Note: This will always signal right away.
2751 SyncThread::get()->triggerBlockedWaitNoTimeline(fenceSync);
2752 #endif
2753 }
2754
asyncWaitForGpuVulkanWithCb(uint64_t deviceHandle,uint64_t fenceHandle,FenceCompletionCallback cb)2755 void FrameBuffer::asyncWaitForGpuVulkanWithCb(uint64_t deviceHandle, uint64_t fenceHandle,
2756 FenceCompletionCallback cb) {
2757 (void)deviceHandle;
2758 SyncThread::get()->triggerWaitVkWithCompletionCallback((VkFence)fenceHandle, std::move(cb));
2759 }
2760
asyncWaitForGpuVulkanQsriWithCb(uint64_t image,FenceCompletionCallback cb)2761 void FrameBuffer::asyncWaitForGpuVulkanQsriWithCb(uint64_t image, FenceCompletionCallback cb) {
2762 SyncThread::get()->triggerWaitVkQsriWithCompletionCallback((VkImage)image, std::move(cb));
2763 }
2764
waitForGpuVulkanQsri(uint64_t image)2765 void FrameBuffer::waitForGpuVulkanQsri(uint64_t image) {
2766 (void)image;
2767 // Signal immediately, because this was a sync wait and it's vulkan.
2768 #if GFXSTREAM_ENABLE_HOST_GLES
2769 SyncThread::get()->triggerBlockedWaitNoTimeline(nullptr);
2770 #endif
2771 }
2772
setGuestManagedColorBufferLifetime(bool guestManaged)2773 void FrameBuffer::setGuestManagedColorBufferLifetime(bool guestManaged) {
2774 m_guestManagedColorBufferLifetime = guestManaged;
2775 }
2776
platformImportResource(uint32_t handle,uint32_t info,void * resource)2777 bool FrameBuffer::platformImportResource(uint32_t handle, uint32_t info, void* resource) {
2778 if (!resource) {
2779 ERR("Error: resource was null");
2780 }
2781
2782 AutoLock mutex(m_lock);
2783
2784 ColorBufferPtr colorBuffer = findColorBuffer(handle);
2785 if (!colorBuffer) {
2786 ERR("Error: resource %u not found as a ColorBuffer", handle);
2787 return false;
2788 }
2789
2790 uint32_t type = (info & RESOURCE_TYPE_MASK);
2791 bool preserveContent = (info & RESOURCE_USE_PRESERVE);
2792
2793 switch (type) {
2794 #if GFXSTREAM_ENABLE_HOST_GLES
2795 case RESOURCE_TYPE_EGL_NATIVE_PIXMAP:
2796 return colorBuffer->glOpImportEglNativePixmap(resource, preserveContent);
2797 case RESOURCE_TYPE_EGL_IMAGE:
2798 return colorBuffer->glOpImportEglImage(resource, preserveContent);
2799 #endif
2800 // Note: Additional non-EGL resource-types can be added here, and will
2801 // be propagated through color-buffer import functionality
2802 case RESOURCE_TYPE_VK_EXT_MEMORY_HANDLE:
2803 return colorBuffer->importNativeResource(resource, type, preserveContent);
2804 default:
2805 ERR("Error: unsupported resource type: %u", type);
2806 return false;
2807 }
2808
2809 return true;
2810 }
2811
borrowColorBufferForComposition(uint32_t colorBufferHandle,bool colorBufferIsTarget)2812 std::unique_ptr<BorrowedImageInfo> FrameBuffer::borrowColorBufferForComposition(
2813 uint32_t colorBufferHandle, bool colorBufferIsTarget) {
2814 ColorBufferPtr colorBufferPtr = findColorBuffer(colorBufferHandle);
2815 if (!colorBufferPtr) {
2816 ERR("Failed to get borrowed image info for ColorBuffer:%d", colorBufferHandle);
2817 return nullptr;
2818 }
2819
2820 if (m_useVulkanComposition) {
2821 invalidateColorBufferForVk(colorBufferHandle);
2822 } else {
2823 #if GFXSTREAM_ENABLE_HOST_GLES
2824 invalidateColorBufferForGl(colorBufferHandle);
2825 #endif
2826 }
2827
2828 const auto api = m_useVulkanComposition ? ColorBuffer::UsedApi::kVk : ColorBuffer::UsedApi::kGl;
2829 return colorBufferPtr->borrowForComposition(api, colorBufferIsTarget);
2830 }
2831
borrowColorBufferForDisplay(uint32_t colorBufferHandle)2832 std::unique_ptr<BorrowedImageInfo> FrameBuffer::borrowColorBufferForDisplay(
2833 uint32_t colorBufferHandle) {
2834 ColorBufferPtr colorBufferPtr = findColorBuffer(colorBufferHandle);
2835 if (!colorBufferPtr) {
2836 ERR("Failed to get borrowed image info for ColorBuffer:%d", colorBufferHandle);
2837 return nullptr;
2838 }
2839
2840 if (m_useVulkanComposition) {
2841 invalidateColorBufferForVk(colorBufferHandle);
2842 } else {
2843 #if GFXSTREAM_ENABLE_HOST_GLES
2844 invalidateColorBufferForGl(colorBufferHandle);
2845 #else
2846 ERR("Failed to invalidate ColorBuffer:%d", colorBufferHandle);
2847 #endif
2848 }
2849
2850 const auto api = m_useVulkanComposition ? ColorBuffer::UsedApi::kVk : ColorBuffer::UsedApi::kGl;
2851 return colorBufferPtr->borrowForDisplay(api);
2852 }
2853
logVulkanOutOfMemory(VkResult result,const char * function,int line,std::optional<uint64_t> allocationSize)2854 void FrameBuffer::logVulkanOutOfMemory(VkResult result, const char* function, int line,
2855 std::optional<uint64_t> allocationSize) {
2856 m_logger->logMetricEvent(MetricEventVulkanOutOfMemory{
2857 .vkResultCode = result,
2858 .function = function,
2859 .line = std::make_optional(line),
2860 .allocationSize = allocationSize,
2861 });
2862 }
2863
setVsyncHz(int vsyncHz)2864 void FrameBuffer::setVsyncHz(int vsyncHz) {
2865 const uint64_t kOneSecondNs = 1000000000ULL;
2866 m_vsyncHz = vsyncHz;
2867 if (m_vsyncThread) {
2868 m_vsyncThread->setPeriod(kOneSecondNs / (uint64_t)m_vsyncHz);
2869 }
2870 }
2871
scheduleVsyncTask(VsyncThread::VsyncTask task)2872 void FrameBuffer::scheduleVsyncTask(VsyncThread::VsyncTask task) {
2873 if (!m_vsyncThread) {
2874 fprintf(stderr, "%s: warning: no vsync thread exists\n", __func__);
2875 task(0);
2876 return;
2877 }
2878
2879 m_vsyncThread->schedule(task);
2880 }
2881
setDisplayConfigs(int configId,int w,int h,int dpiX,int dpiY)2882 void FrameBuffer::setDisplayConfigs(int configId, int w, int h, int dpiX, int dpiY) {
2883 AutoLock mutex(m_lock);
2884 mDisplayConfigs[configId] = {w, h, dpiX, dpiY};
2885 INFO("Setting display: %d configuration to: %dx%d, dpi: %dx%d ", configId,
2886 w, h, dpiX, dpiY);
2887 }
2888
setDisplayActiveConfig(int configId)2889 void FrameBuffer::setDisplayActiveConfig(int configId) {
2890 AutoLock mutex(m_lock);
2891 if (mDisplayConfigs.find(configId) == mDisplayConfigs.end()) {
2892 ERR("config %d not set", configId);
2893 return;
2894 }
2895 mDisplayActiveConfigId = configId;
2896 m_framebufferWidth = mDisplayConfigs[configId].w;
2897 m_framebufferHeight = mDisplayConfigs[configId].h;
2898 setDisplayPose(0, 0, 0, getWidth(), getHeight(), 0);
2899 INFO("setDisplayActiveConfig %d", configId);
2900 }
2901
getDisplayConfigsCount()2902 const int FrameBuffer::getDisplayConfigsCount() {
2903 AutoLock mutex(m_lock);
2904 return mDisplayConfigs.size();
2905 }
2906
getDisplayConfigsParam(int configId,EGLint param)2907 const int FrameBuffer::getDisplayConfigsParam(int configId, EGLint param) {
2908 AutoLock mutex(m_lock);
2909 if (mDisplayConfigs.find(configId) == mDisplayConfigs.end()) {
2910 return -1;
2911 }
2912 switch (param) {
2913 case FB_WIDTH:
2914 return mDisplayConfigs[configId].w;
2915 case FB_HEIGHT:
2916 return mDisplayConfigs[configId].h;
2917 case FB_XDPI:
2918 return mDisplayConfigs[configId].dpiX;
2919 case FB_YDPI:
2920 return mDisplayConfigs[configId].dpiY;
2921 case FB_FPS:
2922 return 60;
2923 case FB_MIN_SWAP_INTERVAL:
2924 return -1;
2925 case FB_MAX_SWAP_INTERVAL:
2926 return -1;
2927 default:
2928 return -1;
2929 }
2930 }
2931
getDisplayActiveConfig()2932 const int FrameBuffer::getDisplayActiveConfig() {
2933 AutoLock mutex(m_lock);
2934 return mDisplayActiveConfigId >= 0 ? mDisplayActiveConfigId : -1;
2935 }
2936
flushColorBufferFromVk(HandleType colorBufferHandle)2937 bool FrameBuffer::flushColorBufferFromVk(HandleType colorBufferHandle) {
2938 auto colorBuffer = findColorBuffer(colorBufferHandle);
2939 if (!colorBuffer) {
2940 ERR("Failed to find ColorBuffer:%d", colorBufferHandle);
2941 return false;
2942 }
2943 return colorBuffer->flushFromVk();
2944 }
2945
flushColorBufferFromVkBytes(HandleType colorBufferHandle,const void * bytes,size_t bytesSize)2946 bool FrameBuffer::flushColorBufferFromVkBytes(HandleType colorBufferHandle, const void* bytes,
2947 size_t bytesSize) {
2948 AutoLock mutex(m_lock);
2949
2950 auto colorBuffer = findColorBuffer(colorBufferHandle);
2951 if (!colorBuffer) {
2952 ERR("Failed to find ColorBuffer:%d", colorBufferHandle);
2953 return false;
2954 }
2955 return colorBuffer->flushFromVkBytes(bytes, bytesSize);
2956 }
2957
invalidateColorBufferForVk(HandleType colorBufferHandle)2958 bool FrameBuffer::invalidateColorBufferForVk(HandleType colorBufferHandle) {
2959 // It reads contents from GL, which requires a context lock.
2960 // Also we should not do this in PostWorkerGl, otherwise it will deadlock.
2961 //
2962 // b/283524158
2963 // b/273986739
2964 AutoLock mutex(m_lock);
2965 auto colorBuffer = findColorBuffer(colorBufferHandle);
2966 if (!colorBuffer) {
2967 ERR("Failed to find ColorBuffer:%d", colorBufferHandle);
2968 return false;
2969 }
2970 return colorBuffer->invalidateForVk();
2971 }
2972
2973 #if GFXSTREAM_ENABLE_HOST_GLES
getEmulatedEglWindowSurfaceColorBufferHandle(HandleType p_surface)2974 HandleType FrameBuffer::getEmulatedEglWindowSurfaceColorBufferHandle(HandleType p_surface) {
2975 AutoLock mutex(m_lock);
2976
2977 auto it = m_EmulatedEglWindowSurfaceToColorBuffer.find(p_surface);
2978 if (it == m_EmulatedEglWindowSurfaceToColorBuffer.end()) {
2979 return 0;
2980 }
2981
2982 return it->second;
2983 }
2984
createTrivialContext(HandleType shared,HandleType * contextOut,HandleType * surfOut)2985 void FrameBuffer::createTrivialContext(HandleType shared, HandleType* contextOut,
2986 HandleType* surfOut) {
2987 assert(contextOut);
2988 assert(surfOut);
2989
2990 *contextOut = createEmulatedEglContext(0, shared, GLESApi_2);
2991 // Zero size is formally allowed here, but SwiftShader doesn't like it and
2992 // fails.
2993 *surfOut = createEmulatedEglWindowSurface(0, 1, 1);
2994 }
2995
createSharedTrivialContext(EGLContext * contextOut,EGLSurface * surfOut)2996 void FrameBuffer::createSharedTrivialContext(EGLContext* contextOut, EGLSurface* surfOut) {
2997 assert(contextOut);
2998 assert(surfOut);
2999
3000 const EmulatedEglConfig* config = getConfigs()->get(0 /* p_config */);
3001 if (!config) return;
3002
3003 int maj, min;
3004 emugl::getGlesVersion(&maj, &min);
3005
3006 const EGLint contextAttribs[] = {EGL_CONTEXT_MAJOR_VERSION_KHR, maj,
3007 EGL_CONTEXT_MINOR_VERSION_KHR, min, EGL_NONE};
3008
3009 *contextOut = s_egl.eglCreateContext(getDisplay(), config->getHostEglConfig(),
3010 getGlobalEGLContext(), contextAttribs);
3011
3012 const EGLint pbufAttribs[] = {EGL_WIDTH, 1, EGL_HEIGHT, 1, EGL_NONE};
3013
3014 *surfOut = s_egl.eglCreatePbufferSurface(getDisplay(), config->getHostEglConfig(), pbufAttribs);
3015 }
3016
destroySharedTrivialContext(EGLContext context,EGLSurface surface)3017 void FrameBuffer::destroySharedTrivialContext(EGLContext context, EGLSurface surface) {
3018 if (getDisplay() != EGL_NO_DISPLAY) {
3019 s_egl.eglDestroyContext(getDisplay(), context);
3020 s_egl.eglDestroySurface(getDisplay(), surface);
3021 }
3022 }
3023
getConfigs() const3024 const EmulatedEglConfigList* FrameBuffer::getConfigs() const {
3025 if (!m_emulationGl) {
3026 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3027 }
3028
3029 return &m_emulationGl->getEmulationEglConfigs();
3030 }
3031
setEmulatedEglWindowSurfaceColorBuffer(HandleType p_surface,HandleType p_colorbuffer)3032 bool FrameBuffer::setEmulatedEglWindowSurfaceColorBuffer(HandleType p_surface,
3033 HandleType p_colorbuffer) {
3034 AutoLock mutex(m_lock);
3035
3036 EmulatedEglWindowSurfaceMap::iterator w(m_windows.find(p_surface));
3037 if (w == m_windows.end()) {
3038 // bad surface handle
3039 ERR("bad window surface handle %#x", p_surface);
3040 return false;
3041 }
3042
3043 {
3044 AutoLock colorBufferMapLock(m_colorBufferMapLock);
3045 ColorBufferMap::iterator c(m_colorbuffers.find(p_colorbuffer));
3046 if (c == m_colorbuffers.end()) {
3047 ERR("bad color buffer handle %#x", p_colorbuffer);
3048 // bad colorbuffer handle
3049 return false;
3050 }
3051
3052 (*w).second.first->setColorBuffer((*c).second.cb);
3053 markOpened(&c->second);
3054 if (!m_guestManagedColorBufferLifetime) {
3055 c->second.refcount++;
3056 }
3057 }
3058 if (w->second.second) {
3059 if (!m_guestManagedColorBufferLifetime) {
3060 if (m_refCountPipeEnabled) {
3061 decColorBufferRefCountLocked(w->second.second);
3062 } else {
3063 closeColorBufferLocked(w->second.second);
3064 }
3065 }
3066 }
3067
3068 (*w).second.second = p_colorbuffer;
3069
3070 m_EmulatedEglWindowSurfaceToColorBuffer[p_surface] = p_colorbuffer;
3071
3072 return true;
3073 }
3074
createEmulatedEglContext(int config,HandleType shareContextHandle,GLESApi version)3075 HandleType FrameBuffer::createEmulatedEglContext(int config, HandleType shareContextHandle,
3076 GLESApi version) {
3077 if (!m_emulationGl) {
3078 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation unavailable.";
3079 }
3080
3081 AutoLock mutex(m_lock);
3082 android::base::AutoWriteLock contextLock(m_contextStructureLock);
3083 // Hold the ColorBuffer map lock so that the new handle won't collide with a ColorBuffer handle.
3084 AutoLock colorBufferMapLock(m_colorBufferMapLock);
3085
3086 EmulatedEglContextPtr shareContext = nullptr;
3087 if (shareContextHandle != 0) {
3088 auto shareContextIt = m_contexts.find(shareContextHandle);
3089 if (shareContextIt == m_contexts.end()) {
3090 ERR("Failed to find share EmulatedEglContext:%d", shareContextHandle);
3091 return 0;
3092 }
3093 shareContext = shareContextIt->second;
3094 }
3095
3096 HandleType contextHandle = genHandle_locked();
3097 auto context =
3098 m_emulationGl->createEmulatedEglContext(config, shareContext.get(), version, contextHandle);
3099 if (!context) {
3100 ERR("Failed to create EmulatedEglContext.");
3101 return 0;
3102 }
3103
3104 m_contexts[contextHandle] = std::move(context);
3105
3106 RenderThreadInfo* tinfo = RenderThreadInfo::get();
3107 uint64_t puid = tinfo->m_puid;
3108 // The new emulator manages render contexts per guest process.
3109 // Fall back to per-thread management if the system image does not
3110 // support it.
3111 if (puid) {
3112 m_procOwnedEmulatedEglContexts[puid].insert(contextHandle);
3113 } else { // legacy path to manage context lifetime by threads
3114 if (!tinfo->m_glInfo) {
3115 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "Render thread GL not available.";
3116 }
3117 tinfo->m_glInfo->m_contextSet.insert(contextHandle);
3118 }
3119
3120 return contextHandle;
3121 }
3122
destroyEmulatedEglContext(HandleType contextHandle)3123 void FrameBuffer::destroyEmulatedEglContext(HandleType contextHandle) {
3124 AutoLock mutex(m_lock);
3125 sweepColorBuffersLocked();
3126
3127 android::base::AutoWriteLock contextLock(m_contextStructureLock);
3128 m_contexts.erase(contextHandle);
3129 RenderThreadInfo* tinfo = RenderThreadInfo::get();
3130 uint64_t puid = tinfo->m_puid;
3131 // The new emulator manages render contexts per guest process.
3132 // Fall back to per-thread management if the system image does not
3133 // support it.
3134 if (puid) {
3135 auto it = m_procOwnedEmulatedEglContexts.find(puid);
3136 if (it != m_procOwnedEmulatedEglContexts.end()) {
3137 it->second.erase(contextHandle);
3138 }
3139 } else {
3140 if (!tinfo->m_glInfo) {
3141 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "Render thread GL not available.";
3142 }
3143 tinfo->m_glInfo->m_contextSet.erase(contextHandle);
3144 }
3145 }
3146
createEmulatedEglWindowSurface(int p_config,int p_width,int p_height)3147 HandleType FrameBuffer::createEmulatedEglWindowSurface(int p_config, int p_width, int p_height) {
3148 if (!m_emulationGl) {
3149 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation unavailable.";
3150 }
3151
3152 AutoLock mutex(m_lock);
3153 // Hold the ColorBuffer map lock so that the new handle won't collide with a ColorBuffer handle.
3154 AutoLock colorBufferMapLock(m_colorBufferMapLock);
3155
3156 HandleType handle = genHandle_locked();
3157
3158 auto window =
3159 m_emulationGl->createEmulatedEglWindowSurface(p_config, p_width, p_height, handle);
3160 if (!window) {
3161 ERR("Failed to create EmulatedEglWindowSurface.");
3162 return 0;
3163 }
3164
3165 m_windows[handle] = {std::move(window), 0};
3166
3167 RenderThreadInfo* info = RenderThreadInfo::get();
3168 if (!info->m_glInfo) {
3169 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "RRenderThreadInfoGl not available.";
3170 }
3171
3172 uint64_t puid = info->m_puid;
3173 if (puid) {
3174 m_procOwnedEmulatedEglWindowSurfaces[puid].insert(handle);
3175 } else { // legacy path to manage window surface lifetime by threads
3176 info->m_glInfo->m_windowSet.insert(handle);
3177 }
3178
3179 return handle;
3180 }
3181
destroyEmulatedEglWindowSurface(HandleType p_surface)3182 void FrameBuffer::destroyEmulatedEglWindowSurface(HandleType p_surface) {
3183 if (m_shuttingDown) {
3184 return;
3185 }
3186 AutoLock mutex(m_lock);
3187 destroyEmulatedEglWindowSurfaceLocked(p_surface);
3188 }
3189
destroyEmulatedEglWindowSurfaceLocked(HandleType p_surface)3190 std::vector<HandleType> FrameBuffer::destroyEmulatedEglWindowSurfaceLocked(HandleType p_surface) {
3191 std::vector<HandleType> colorBuffersToCleanUp;
3192 const auto w = m_windows.find(p_surface);
3193 if (w != m_windows.end()) {
3194 RecursiveScopedContextBind bind(getPbufferSurfaceContextHelper());
3195 if (!m_guestManagedColorBufferLifetime) {
3196 if (m_refCountPipeEnabled) {
3197 if (decColorBufferRefCountLocked(w->second.second)) {
3198 colorBuffersToCleanUp.push_back(w->second.second);
3199 }
3200 } else {
3201 if (closeColorBufferLocked(w->second.second)) {
3202 colorBuffersToCleanUp.push_back(w->second.second);
3203 }
3204 }
3205 }
3206 m_windows.erase(w);
3207 RenderThreadInfo* tinfo = RenderThreadInfo::get();
3208 uint64_t puid = tinfo->m_puid;
3209 if (puid) {
3210 auto ite = m_procOwnedEmulatedEglWindowSurfaces.find(puid);
3211 if (ite != m_procOwnedEmulatedEglWindowSurfaces.end()) {
3212 ite->second.erase(p_surface);
3213 }
3214 } else {
3215 if (!tinfo->m_glInfo) {
3216 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
3217 << "Render thread GL not available.";
3218 }
3219 tinfo->m_glInfo->m_windowSet.erase(p_surface);
3220 }
3221 }
3222 return colorBuffersToCleanUp;
3223 }
3224
createEmulatedEglFenceSync(EGLenum type,int destroyWhenSignaled,uint64_t * outSync,uint64_t * outSyncThread)3225 void FrameBuffer::createEmulatedEglFenceSync(EGLenum type, int destroyWhenSignaled,
3226 uint64_t* outSync, uint64_t* outSyncThread) {
3227 if (!m_emulationGl) {
3228 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not available.";
3229 }
3230
3231 // TODO(b/233939967): move RenderThreadInfoGl usage to EmulationGl.
3232 RenderThreadInfoGl* const info = RenderThreadInfoGl::get();
3233 if (!info) {
3234 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "RenderThreadInfoGl not available.";
3235 }
3236 if (!info->currContext) {
3237 auto fb = FrameBuffer::getFB();
3238 uint32_t syncContext;
3239 uint32_t syncSurface;
3240 createTrivialContext(0, // There is no context to share.
3241 &syncContext, &syncSurface);
3242 bindContext(syncContext, syncSurface, syncSurface);
3243 // This context is then cleaned up when the render thread exits.
3244 }
3245
3246 auto sync = m_emulationGl->createEmulatedEglFenceSync(type, destroyWhenSignaled);
3247 if (!sync) {
3248 return;
3249 }
3250
3251 if (outSync) {
3252 *outSync = (uint64_t)(uintptr_t)sync.release();
3253 }
3254 if (outSyncThread) {
3255 *outSyncThread = reinterpret_cast<uint64_t>(SyncThread::get());
3256 }
3257 }
3258
drainGlRenderThreadResources()3259 void FrameBuffer::drainGlRenderThreadResources() {
3260 // If we're already exiting then snapshot should not contain
3261 // this thread information at all.
3262 if (isShuttingDown()) {
3263 return;
3264 }
3265
3266 // Release references to the current thread's context/surfaces if any
3267 bindContext(0, 0, 0);
3268
3269 drainGlRenderThreadSurfaces();
3270 drainGlRenderThreadContexts();
3271
3272 if (!s_egl.eglReleaseThread()) {
3273 ERR("Error: RenderThread @%p failed to eglReleaseThread()", this);
3274 }
3275 }
3276
drainGlRenderThreadContexts()3277 void FrameBuffer::drainGlRenderThreadContexts() {
3278 if (isShuttingDown()) {
3279 return;
3280 }
3281
3282 RenderThreadInfoGl* const tinfo = RenderThreadInfoGl::get();
3283 if (!tinfo) {
3284 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "Render thread GL not available.";
3285 }
3286
3287 if (tinfo->m_contextSet.empty()) {
3288 return;
3289 }
3290
3291 AutoLock mutex(m_lock);
3292 android::base::AutoWriteLock contextLock(m_contextStructureLock);
3293 for (const HandleType contextHandle : tinfo->m_contextSet) {
3294 m_contexts.erase(contextHandle);
3295 }
3296 tinfo->m_contextSet.clear();
3297 }
3298
drainGlRenderThreadSurfaces()3299 void FrameBuffer::drainGlRenderThreadSurfaces() {
3300 if (isShuttingDown()) {
3301 return;
3302 }
3303
3304 RenderThreadInfoGl* const tinfo = RenderThreadInfoGl::get();
3305 if (!tinfo) {
3306 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "Render thread GL not available.";
3307 }
3308
3309 if (tinfo->m_windowSet.empty()) {
3310 return;
3311 }
3312
3313 std::vector<HandleType> colorBuffersToCleanup;
3314
3315 AutoLock mutex(m_lock);
3316 RecursiveScopedContextBind bind(getPbufferSurfaceContextHelper());
3317 for (const HandleType winHandle : tinfo->m_windowSet) {
3318 const auto winIt = m_windows.find(winHandle);
3319 if (winIt != m_windows.end()) {
3320 if (const HandleType oldColorBufferHandle = winIt->second.second) {
3321 if (!m_guestManagedColorBufferLifetime) {
3322 if (m_refCountPipeEnabled) {
3323 if (decColorBufferRefCountLocked(oldColorBufferHandle)) {
3324 colorBuffersToCleanup.push_back(oldColorBufferHandle);
3325 }
3326 } else {
3327 if (closeColorBufferLocked(oldColorBufferHandle)) {
3328 colorBuffersToCleanup.push_back(oldColorBufferHandle);
3329 }
3330 }
3331 }
3332 m_windows.erase(winIt);
3333 }
3334 }
3335 }
3336 tinfo->m_windowSet.clear();
3337 }
3338
getEmulationGl()3339 EmulationGl& FrameBuffer::getEmulationGl() {
3340 if (!m_emulationGl) {
3341 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not enabled.";
3342 }
3343 return *m_emulationGl;
3344 }
3345
getDisplay() const3346 EGLDisplay FrameBuffer::getDisplay() const {
3347 if (!m_emulationGl) {
3348 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3349 }
3350 return m_emulationGl->mEglDisplay;
3351 }
3352
getWindowSurface() const3353 EGLSurface FrameBuffer::getWindowSurface() const {
3354 if (!m_emulationGl) {
3355 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3356 }
3357
3358 if (!m_emulationGl->mWindowSurface) {
3359 return EGL_NO_SURFACE;
3360 }
3361
3362 const auto* displaySurfaceGl =
3363 reinterpret_cast<const DisplaySurfaceGl*>(m_emulationGl->mWindowSurface->getImpl());
3364
3365 return displaySurfaceGl->getSurface();
3366 }
3367
getContext() const3368 EGLContext FrameBuffer::getContext() const {
3369 if (!m_emulationGl) {
3370 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3371 }
3372 return m_emulationGl->mEglContext;
3373 }
3374
getConfig() const3375 EGLContext FrameBuffer::getConfig() const {
3376 if (!m_emulationGl) {
3377 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3378 }
3379 return m_emulationGl->mEglConfig;
3380 }
3381
getGlobalEGLContext() const3382 EGLContext FrameBuffer::getGlobalEGLContext() const {
3383 if (!m_emulationGl) {
3384 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3385 }
3386
3387 if (!m_emulationGl->mPbufferSurface) {
3388 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
3389 << "FrameBuffer pbuffer surface not available.";
3390 }
3391
3392 const auto* displaySurfaceGl =
3393 reinterpret_cast<const DisplaySurfaceGl*>(m_emulationGl->mPbufferSurface->getImpl());
3394
3395 return displaySurfaceGl->getContextForShareContext();
3396 }
3397
getContext_locked(HandleType p_context)3398 EmulatedEglContextPtr FrameBuffer::getContext_locked(HandleType p_context) {
3399 return android::base::findOrDefault(m_contexts, p_context);
3400 }
3401
getWindowSurface_locked(HandleType p_windowsurface)3402 EmulatedEglWindowSurfacePtr FrameBuffer::getWindowSurface_locked(HandleType p_windowsurface) {
3403 return android::base::findOrDefault(m_windows, p_windowsurface).first;
3404 }
3405
getTextureDraw() const3406 TextureDraw* FrameBuffer::getTextureDraw() const {
3407 if (!m_emulationGl) {
3408 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3409 }
3410
3411 return m_emulationGl->mTextureDraw.get();
3412 }
3413
isFastBlitSupported() const3414 bool FrameBuffer::isFastBlitSupported() const {
3415 if (!m_emulationGl) {
3416 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3417 }
3418
3419 return m_emulationGl->isFastBlitSupported();
3420 }
3421
disableFastBlitForTesting()3422 void FrameBuffer::disableFastBlitForTesting() {
3423 if (!m_emulationGl) {
3424 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3425 }
3426
3427 m_emulationGl->disableFastBlitForTesting();
3428 }
3429
createEmulatedEglImage(HandleType contextHandle,EGLenum target,GLuint buffer)3430 HandleType FrameBuffer::createEmulatedEglImage(HandleType contextHandle, EGLenum target,
3431 GLuint buffer) {
3432 if (!m_emulationGl) {
3433 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not enabled.";
3434 }
3435
3436 AutoLock mutex(m_lock);
3437
3438 EmulatedEglContext* context = nullptr;
3439 if (contextHandle) {
3440 android::base::AutoWriteLock contextLock(m_contextStructureLock);
3441
3442 auto it = m_contexts.find(contextHandle);
3443 if (it == m_contexts.end()) {
3444 ERR("Failed to find EmulatedEglContext:%d", contextHandle);
3445 return false;
3446 }
3447
3448 context = it->second.get();
3449 }
3450
3451 auto image = m_emulationGl->createEmulatedEglImage(context, target,
3452 reinterpret_cast<EGLClientBuffer>(buffer));
3453 if (!image) {
3454 ERR("Failed to create EmulatedEglImage");
3455 return false;
3456 }
3457
3458 HandleType imageHandle = image->getHandle();
3459
3460 m_images[imageHandle] = std::move(image);
3461
3462 RenderThreadInfo* tInfo = RenderThreadInfo::get();
3463 uint64_t puid = tInfo->m_puid;
3464 if (puid) {
3465 m_procOwnedEmulatedEglImages[puid].insert(imageHandle);
3466 }
3467 return imageHandle;
3468 }
3469
destroyEmulatedEglImage(HandleType imageHandle)3470 EGLBoolean FrameBuffer::destroyEmulatedEglImage(HandleType imageHandle) {
3471 if (!m_emulationGl) {
3472 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "GL/EGL emulation not enabled.";
3473 }
3474
3475 AutoLock mutex(m_lock);
3476
3477 auto imageIt = m_images.find(imageHandle);
3478 if (imageIt == m_images.end()) {
3479 ERR("Failed to find EmulatedEglImage:%d", imageHandle);
3480 return false;
3481 }
3482 auto& image = imageIt->second;
3483
3484 EGLBoolean success = image->destroy();
3485 m_images.erase(imageIt);
3486
3487 RenderThreadInfo* tInfo = RenderThreadInfo::get();
3488 uint64_t puid = tInfo->m_puid;
3489 if (puid) {
3490 m_procOwnedEmulatedEglImages[puid].erase(imageHandle);
3491 // We don't explicitly call m_procOwnedEmulatedEglImages.erase(puid) when the
3492 // size reaches 0, since it could go between zero and one many times in
3493 // the lifetime of a process. It will be cleaned up by
3494 // cleanupProcGLObjects(puid) when the process is dead.
3495 }
3496 return success;
3497 }
3498
flushEmulatedEglWindowSurfaceColorBuffer(HandleType p_surface)3499 bool FrameBuffer::flushEmulatedEglWindowSurfaceColorBuffer(HandleType p_surface) {
3500 AutoLock mutex(m_lock);
3501
3502 auto it = m_windows.find(p_surface);
3503 if (it == m_windows.end()) {
3504 ERR("FB::flushEmulatedEglWindowSurfaceColorBuffer: window handle %#x not found", p_surface);
3505 // bad surface handle
3506 return false;
3507 }
3508
3509 EmulatedEglWindowSurface* surface = it->second.first.get();
3510 surface->flushColorBuffer();
3511
3512 return true;
3513 }
3514
getMaxGLESVersion()3515 GLESDispatchMaxVersion FrameBuffer::getMaxGLESVersion() {
3516 if (!m_emulationGl) {
3517 return GLES_DISPATCH_MAX_VERSION_2;
3518 }
3519 return m_emulationGl->getGlesMaxDispatchVersion();
3520 }
3521
fillGLESUsages(android_studio::EmulatorGLESUsages * usages)3522 void FrameBuffer::fillGLESUsages(android_studio::EmulatorGLESUsages* usages) {
3523 if (s_egl.eglFillUsages) {
3524 s_egl.eglFillUsages(usages);
3525 }
3526 }
3527
platformCreateSharedEglContext(void)3528 void* FrameBuffer::platformCreateSharedEglContext(void) {
3529 AutoLock lock(m_lock);
3530
3531 EGLContext context = 0;
3532 EGLSurface surface = 0;
3533 createSharedTrivialContext(&context, &surface);
3534
3535 void* underlyingContext = s_egl.eglGetNativeContextANDROID(getDisplay(), context);
3536 if (!underlyingContext) {
3537 ERR("Error: Underlying egl backend could not produce a native EGL context.");
3538 return nullptr;
3539 }
3540
3541 m_platformEglContexts[underlyingContext] = {context, surface};
3542
3543 #if defined(__QNX__)
3544 EGLDisplay currDisplay = eglGetCurrentDisplay();
3545 EGLSurface currRead = eglGetCurrentSurface(EGL_READ);
3546 EGLSurface currDraw = eglGetCurrentSurface(EGL_DRAW);
3547 EGLSurface currContext = eglGetCurrentContext();
3548 // Make this context current to ensure thread-state is initialized
3549 s_egl.eglMakeCurrent(getDisplay(), surface, surface, context);
3550 // Revert back to original state
3551 s_egl.eglMakeCurrent(currDisplay, currRead, currDraw, currContext);
3552 #endif
3553
3554 return underlyingContext;
3555 }
3556
platformDestroySharedEglContext(void * underlyingContext)3557 bool FrameBuffer::platformDestroySharedEglContext(void* underlyingContext) {
3558 AutoLock lock(m_lock);
3559
3560 auto it = m_platformEglContexts.find(underlyingContext);
3561 if (it == m_platformEglContexts.end()) {
3562 ERR("Error: Could not find underlying egl context %p (perhaps already destroyed?)",
3563 underlyingContext);
3564 return false;
3565 }
3566
3567 destroySharedTrivialContext(it->second.context, it->second.surface);
3568
3569 m_platformEglContexts.erase(it);
3570
3571 return true;
3572 }
3573
flushColorBufferFromGl(HandleType colorBufferHandle)3574 bool FrameBuffer::flushColorBufferFromGl(HandleType colorBufferHandle) {
3575 auto colorBuffer = findColorBuffer(colorBufferHandle);
3576 if (!colorBuffer) {
3577 ERR("Failed to find ColorBuffer:%d", colorBufferHandle);
3578 return false;
3579 }
3580 return colorBuffer->flushFromGl();
3581 }
3582
invalidateColorBufferForGl(HandleType colorBufferHandle)3583 bool FrameBuffer::invalidateColorBufferForGl(HandleType colorBufferHandle) {
3584 auto colorBuffer = findColorBuffer(colorBufferHandle);
3585 if (!colorBuffer) {
3586 ERR("Failed to find ColorBuffer:%d", colorBufferHandle);
3587 return false;
3588 }
3589 return colorBuffer->invalidateForGl();
3590 }
3591
getPbufferSurfaceContextHelper() const3592 ContextHelper* FrameBuffer::getPbufferSurfaceContextHelper() const {
3593 if (!m_emulationGl) {
3594 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3595 }
3596 if (!m_emulationGl->mPbufferSurface) {
3597 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER))
3598 << "EGL emulation pbuffer surface not available.";
3599 }
3600 const auto* displaySurfaceGl =
3601 reinterpret_cast<const DisplaySurfaceGl*>(m_emulationGl->mPbufferSurface->getImpl());
3602
3603 return displaySurfaceGl->getContextHelper();
3604 }
3605
bindColorBufferToTexture(HandleType p_colorbuffer)3606 bool FrameBuffer::bindColorBufferToTexture(HandleType p_colorbuffer) {
3607 AutoLock mutex(m_lock);
3608
3609 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
3610 if (!colorBuffer) {
3611 // bad colorbuffer handle
3612 return false;
3613 }
3614
3615 return colorBuffer->glOpBindToTexture();
3616 }
3617
bindColorBufferToTexture2(HandleType p_colorbuffer)3618 bool FrameBuffer::bindColorBufferToTexture2(HandleType p_colorbuffer) {
3619 // This is only called when using multi window display
3620 // It will deadlock when posting from main thread.
3621 std::unique_ptr<AutoLock> mutex;
3622 if (!postOnlyOnMainThread()) {
3623 mutex = std::make_unique<AutoLock>(m_lock);
3624 }
3625
3626 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
3627 if (!colorBuffer) {
3628 // bad colorbuffer handle
3629 return false;
3630 }
3631
3632 return colorBuffer->glOpBindToTexture2();
3633 }
3634
bindColorBufferToRenderbuffer(HandleType p_colorbuffer)3635 bool FrameBuffer::bindColorBufferToRenderbuffer(HandleType p_colorbuffer) {
3636 AutoLock mutex(m_lock);
3637
3638 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
3639 if (!colorBuffer) {
3640 // bad colorbuffer handle
3641 return false;
3642 }
3643
3644 return colorBuffer->glOpBindToRenderbuffer();
3645 }
3646
bindContext(HandleType p_context,HandleType p_drawSurface,HandleType p_readSurface)3647 bool FrameBuffer::bindContext(HandleType p_context, HandleType p_drawSurface,
3648 HandleType p_readSurface) {
3649 if (m_shuttingDown) {
3650 return false;
3651 }
3652
3653 AutoLock mutex(m_lock);
3654
3655 EmulatedEglWindowSurfacePtr draw, read;
3656 EmulatedEglContextPtr ctx;
3657
3658 //
3659 // if this is not an unbind operation - make sure all handles are good
3660 //
3661 if (p_context || p_drawSurface || p_readSurface) {
3662 ctx = getContext_locked(p_context);
3663 if (!ctx) return false;
3664 EmulatedEglWindowSurfaceMap::iterator w(m_windows.find(p_drawSurface));
3665 if (w == m_windows.end()) {
3666 // bad surface handle
3667 return false;
3668 }
3669 draw = (*w).second.first;
3670
3671 if (p_readSurface != p_drawSurface) {
3672 EmulatedEglWindowSurfaceMap::iterator w(m_windows.find(p_readSurface));
3673 if (w == m_windows.end()) {
3674 // bad surface handle
3675 return false;
3676 }
3677 read = (*w).second.first;
3678 } else {
3679 read = draw;
3680 }
3681 } else {
3682 // if unbind operation, sweep color buffers
3683 sweepColorBuffersLocked();
3684 }
3685
3686 if (!s_egl.eglMakeCurrent(getDisplay(), draw ? draw->getEGLSurface() : EGL_NO_SURFACE,
3687 read ? read->getEGLSurface() : EGL_NO_SURFACE,
3688 ctx ? ctx->getEGLContext() : EGL_NO_CONTEXT)) {
3689 ERR("eglMakeCurrent failed");
3690 return false;
3691 }
3692
3693 //
3694 // Bind the surface(s) to the context
3695 //
3696 RenderThreadInfoGl* const tinfo = RenderThreadInfoGl::get();
3697 if (!tinfo) {
3698 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "Render thread GL not available.";
3699 }
3700
3701 EmulatedEglWindowSurfacePtr bindDraw, bindRead;
3702 if (draw.get() == NULL && read.get() == NULL) {
3703 // Unbind the current read and draw surfaces from the context
3704 bindDraw = tinfo->currDrawSurf;
3705 bindRead = tinfo->currReadSurf;
3706 } else {
3707 bindDraw = draw;
3708 bindRead = read;
3709 }
3710
3711 if (bindDraw.get() != NULL && bindRead.get() != NULL) {
3712 if (bindDraw.get() != bindRead.get()) {
3713 bindDraw->bind(ctx, EmulatedEglWindowSurface::BIND_DRAW);
3714 bindRead->bind(ctx, EmulatedEglWindowSurface::BIND_READ);
3715 } else {
3716 bindDraw->bind(ctx, EmulatedEglWindowSurface::BIND_READDRAW);
3717 }
3718 }
3719
3720 //
3721 // update thread info with current bound context
3722 //
3723 tinfo->currContext = ctx;
3724 tinfo->currDrawSurf = draw;
3725 tinfo->currReadSurf = read;
3726 if (ctx) {
3727 if (ctx->clientVersion() > GLESApi_CM)
3728 tinfo->m_gl2Dec.setContextData(&ctx->decoderContextData());
3729 else
3730 tinfo->m_glDec.setContextData(&ctx->decoderContextData());
3731 } else {
3732 tinfo->m_glDec.setContextData(NULL);
3733 tinfo->m_gl2Dec.setContextData(NULL);
3734 }
3735 return true;
3736 }
3737
createYUVTextures(uint32_t type,uint32_t count,int width,int height,uint32_t * output)3738 void FrameBuffer::createYUVTextures(uint32_t type, uint32_t count, int width, int height,
3739 uint32_t* output) {
3740 FrameworkFormat format = static_cast<FrameworkFormat>(type);
3741 AutoLock mutex(m_lock);
3742 RecursiveScopedContextBind bind(getPbufferSurfaceContextHelper());
3743 for (uint32_t i = 0; i < count; ++i) {
3744 if (format == FRAMEWORK_FORMAT_NV12) {
3745 YUVConverter::createYUVGLTex(GL_TEXTURE0, width, height, format, m_features.Yuv420888ToNv21.enabled,
3746 YUVPlane::Y, &output[2 * i]);
3747 YUVConverter::createYUVGLTex(GL_TEXTURE1, width / 2, height / 2, format, m_features.Yuv420888ToNv21.enabled, YUVPlane::UV,
3748 &output[2 * i + 1]);
3749 } else if (format == FRAMEWORK_FORMAT_YUV_420_888) {
3750 YUVConverter::createYUVGLTex(GL_TEXTURE0, width, height, format, m_features.Yuv420888ToNv21.enabled, YUVPlane::Y,
3751 &output[3 * i]);
3752 YUVConverter::createYUVGLTex(GL_TEXTURE1, width / 2, height / 2, format, m_features.Yuv420888ToNv21.enabled, YUVPlane::U,
3753 &output[3 * i + 1]);
3754 YUVConverter::createYUVGLTex(GL_TEXTURE2, width / 2, height / 2, format, m_features.Yuv420888ToNv21.enabled, YUVPlane::V,
3755 &output[3 * i + 2]);
3756 }
3757 }
3758 }
3759
destroyYUVTextures(uint32_t type,uint32_t count,uint32_t * textures)3760 void FrameBuffer::destroyYUVTextures(uint32_t type, uint32_t count, uint32_t* textures) {
3761 AutoLock mutex(m_lock);
3762 RecursiveScopedContextBind bind(getPbufferSurfaceContextHelper());
3763 if (type == FRAMEWORK_FORMAT_NV12) {
3764 s_gles2.glDeleteTextures(2 * count, textures);
3765 } else if (type == FRAMEWORK_FORMAT_YUV_420_888) {
3766 s_gles2.glDeleteTextures(3 * count, textures);
3767 }
3768 }
3769
updateYUVTextures(uint32_t type,uint32_t * textures,void * privData,void * func)3770 void FrameBuffer::updateYUVTextures(uint32_t type, uint32_t* textures, void* privData, void* func) {
3771 AutoLock mutex(m_lock);
3772 RecursiveScopedContextBind bind(getPbufferSurfaceContextHelper());
3773
3774 yuv_updater_t updater = (yuv_updater_t)func;
3775 uint32_t gtextures[3] = {0, 0, 0};
3776
3777 if (type == FRAMEWORK_FORMAT_NV12) {
3778 gtextures[0] = s_gles2.glGetGlobalTexName(textures[0]);
3779 gtextures[1] = s_gles2.glGetGlobalTexName(textures[1]);
3780 } else if (type == FRAMEWORK_FORMAT_YUV_420_888) {
3781 gtextures[0] = s_gles2.glGetGlobalTexName(textures[0]);
3782 gtextures[1] = s_gles2.glGetGlobalTexName(textures[1]);
3783 gtextures[2] = s_gles2.glGetGlobalTexName(textures[2]);
3784 }
3785
3786 #ifdef __APPLE__
3787 EGLContext prevContext = s_egl.eglGetCurrentContext();
3788 auto mydisp = EglGlobalInfo::getInstance()->getDisplayFromDisplayType(EGL_DEFAULT_DISPLAY);
3789 void* nativecontext = mydisp->getLowLevelContext(prevContext);
3790 struct MediaNativeCallerData callerdata;
3791 callerdata.ctx = nativecontext;
3792 callerdata.converter = nsConvertVideoFrameToNV12Textures;
3793 void* pcallerdata = &callerdata;
3794 #else
3795 void* pcallerdata = nullptr;
3796 #endif
3797
3798 updater(privData, type, gtextures, pcallerdata);
3799 }
3800
swapTexturesAndUpdateColorBuffer(uint32_t p_colorbuffer,int x,int y,int width,int height,uint32_t format,uint32_t type,uint32_t texture_type,uint32_t * textures)3801 void FrameBuffer::swapTexturesAndUpdateColorBuffer(uint32_t p_colorbuffer, int x, int y, int width,
3802 int height, uint32_t format, uint32_t type,
3803 uint32_t texture_type, uint32_t* textures) {
3804 {
3805 AutoLock mutex(m_lock);
3806 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
3807 if (!colorBuffer) {
3808 // bad colorbuffer handle
3809 return;
3810 }
3811 colorBuffer->glOpSwapYuvTexturesAndUpdate(
3812 format, type, static_cast<FrameworkFormat>(texture_type), textures);
3813 }
3814 }
3815
readColorBufferContents(HandleType p_colorbuffer,size_t * numBytes,void * pixels)3816 bool FrameBuffer::readColorBufferContents(HandleType p_colorbuffer, size_t* numBytes,
3817 void* pixels) {
3818 AutoLock mutex(m_lock);
3819
3820 ColorBufferPtr colorBuffer = findColorBuffer(p_colorbuffer);
3821 if (!colorBuffer) {
3822 // bad colorbuffer handle
3823 return false;
3824 }
3825
3826 return colorBuffer->glOpReadContents(numBytes, pixels);
3827 }
3828
waitForGpu(uint64_t eglsync)3829 void FrameBuffer::waitForGpu(uint64_t eglsync) {
3830 EmulatedEglFenceSync* fenceSync = EmulatedEglFenceSync::getFromHandle(eglsync);
3831
3832 if (!fenceSync) {
3833 ERR("err: fence sync 0x%llx not found", (unsigned long long)eglsync);
3834 return;
3835 }
3836
3837 SyncThread::get()->triggerBlockedWaitNoTimeline(fenceSync);
3838 }
3839
asyncWaitForGpuWithCb(uint64_t eglsync,FenceCompletionCallback cb)3840 void FrameBuffer::asyncWaitForGpuWithCb(uint64_t eglsync, FenceCompletionCallback cb) {
3841 EmulatedEglFenceSync* fenceSync = EmulatedEglFenceSync::getFromHandle(eglsync);
3842
3843 if (!fenceSync) {
3844 ERR("err: fence sync 0x%llx not found", (unsigned long long)eglsync);
3845 return;
3846 }
3847
3848 SyncThread::get()->triggerWaitWithCompletionCallback(fenceSync, std::move(cb));
3849 }
3850
getGles2Dispatch()3851 const gl::GLESv2Dispatch* FrameBuffer::getGles2Dispatch() {
3852 if (!m_emulationGl) {
3853 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3854 }
3855
3856 return m_emulationGl->getGles2Dispatch();
3857 }
3858
getEglDispatch()3859 const gl::EGLDispatch* FrameBuffer::getEglDispatch() {
3860 if (!m_emulationGl) {
3861 GFXSTREAM_ABORT(FatalError(ABORT_REASON_OTHER)) << "EGL emulation not enabled.";
3862 }
3863
3864 return m_emulationGl->getEglDispatch();
3865 }
3866
3867 #endif
3868
3869 } // namespace gfxstream
3870