1 // Copyright 2019 The Android Open Source Project
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 #include <vulkan/vulkan.h>
15
16 #include <cstdarg>
17 #include <cstdio>
18 #include <deque>
19 #include <type_traits>
20 #include <unordered_map>
21 #include <variant>
22
23 #include "BlobManager.h"
24 #include "FrameBuffer.h"
25 #include "GfxStreamAgents.h"
26 #include "VirtioGpuTimelines.h"
27 #include "VkCommonOperations.h"
28 #include "aemu/base/AlignedBuf.h"
29 #include "aemu/base/ManagedDescriptor.hpp"
30 #include "aemu/base/Metrics.h"
31 #include "aemu/base/Tracing.h"
32 #include "aemu/base/memory/SharedMemory.h"
33 #include "aemu/base/synchronization/Lock.h"
34 #include "aemu/base/threads/WorkerThread.h"
35 #include "gfxstream/Strings.h"
36 #include "gfxstream/host/Features.h"
37 #include "host-common/AddressSpaceService.h"
38 #include "host-common/GfxstreamFatalError.h"
39 #include "host-common/address_space_device.h"
40 #include "host-common/android_pipe_common.h"
41 #include "host-common/android_pipe_device.h"
42 #include "host-common/feature_control.h"
43 #include "host-common/globals.h"
44 #include "host-common/opengles-pipe.h"
45 #include "host-common/opengles.h"
46 #include "host-common/refcount-pipe.h"
47 #include "host-common/vm_operations.h"
48 #include "virgl_hw.h"
49 #include "virtgpu_gfxstream_protocol.h"
50 #include "vk_util.h"
51
52 #ifdef GFXSTREAM_ENABLE_HOST_VK_SNAPSHOT
53 #include "aemu/base/files/StdioStream.h"
54 #endif
55
56 extern "C" {
57 #include "drm_fourcc.h"
58 #include "gfxstream/virtio-gpu-gfxstream-renderer-unstable.h"
59 #include "gfxstream/virtio-gpu-gfxstream-renderer.h"
60 #include "host-common/goldfish_pipe.h"
61 #include "virgl_hw.h"
62 } // extern "C"
63
64 #if defined(_WIN32)
65 struct iovec {
66 void* iov_base; /* Starting address */
67 size_t iov_len; /* Length in bytes */
68 };
69 #else
70 #include <unistd.h>
71 #endif // _WIN32
72
73 #define MAX_DEBUG_BUFFER_SIZE 512
74
75 void* globalUserData = nullptr;
76 stream_renderer_debug_callback globalDebugCallback = nullptr;
77
stream_renderer_log(uint32_t type,const char * format,...)78 void stream_renderer_log(uint32_t type, const char* format, ...) {
79 char buf[MAX_DEBUG_BUFFER_SIZE];
80 va_list args;
81 va_start(args, format);
82 vsnprintf(buf, MAX_DEBUG_BUFFER_SIZE, format, args);
83 va_end(args);
84
85 if (globalUserData && globalDebugCallback) {
86 struct stream_renderer_debug debug = {0};
87 debug.debug_type = type;
88 debug.message = &buf[0];
89
90 globalDebugCallback(globalUserData, &debug);
91 } else {
92 fprintf(stderr, "%s\n", buf);
93 }
94 }
95
96 #if STREAM_RENDERER_LOG_LEVEL >= STREAM_RENDERER_DEBUG_ERROR
97 #define stream_renderer_error(format, ...) \
98 do { \
99 stream_renderer_log(STREAM_RENDERER_DEBUG_ERROR, "[%s(%d)] %s " format, __FILE__, \
100 __LINE__, __PRETTY_FUNCTION__, ##__VA_ARGS__); \
101 } while (0)
102 #else
103 #define stream_renderer_error(format, ...)
104 #endif
105
106 #if STREAM_RENDERER_LOG_LEVEL >= STREAM_RENDERER_DEBUG_WARN
107 #define stream_renderer_warn(format, ...) \
108 do { \
109 stream_renderer_log(STREAM_RENDERER_DEBUG_WARN, "[%s(%d)] %s " format, __FILE__, __LINE__, \
110 __PRETTY_FUNCTION__, ##__VA_ARGS__); \
111 } while (0)
112 #else
113 #define stream_renderer_warn(format, ...)
114 #endif
115
116 #if STREAM_RENDERER_LOG_LEVEL >= STREAM_RENDERER_DEBUG_INFO
117 #define stream_renderer_info(format, ...) \
118 do { \
119 stream_renderer_log(STREAM_RENDERER_DEBUG_INFO, "[%s(%d)] %s " format, __FILE__, __LINE__, \
120 __FUNCTION__, ##__VA_ARGS__); \
121 } while (0)
122 #else
123 #define stream_renderer_info(format, ...)
124 #endif
125
126 #if STREAM_RENDERER_LOG_LEVEL >= STREAM_RENDERER_DEBUG_DEBUG
127 #define stream_renderer_debug(format, ...) \
128 do { \
129 stream_renderer_log(STREAM_RENDERER_DEBUG_DEBUG, "[%s(%d)] %s " format, __FILE__, \
130 __LINE__, __PRETTY_FUNCTION__, ##__VA_ARGS__); \
131 } while (0)
132 #else
133 #define stream_renderer_debug(format, ...)
134 #endif
135
136 // Virtio Goldfish Pipe: Overview-----------------------------------------------
137 //
138 // Virtio Goldfish Pipe is meant for running goldfish pipe services with a
139 // stock Linux kernel that is already capable of virtio-gpu. It runs DRM
140 // VIRTGPU ioctls on top of a custom implementation of virglrenderer on the
141 // host side that doesn't (directly) do any rendering, but instead talks to
142 // host-side pipe services.
143 //
144 // This is mainly used for graphics at the moment, though it's possible to run
145 // other pipe services over virtio-gpu as well. virtio-gpu is selected over
146 // other devices primarily because of the existence of an API (virglrenderer)
147 // that is already somewhat separate from virtio-gpu, and not needing to create
148 // a new virtio device to handle goldfish pipe.
149 //
150 // How it works is, existing virglrenderer API are remapped to perform pipe
151 // operations. First of all, pipe operations consist of the following:
152 //
153 // - open() / close(): Starts or stops an instance of a pipe service.
154 //
155 // - write(const void* buf, size_t len) / read(const void* buf, size_t len):
156 // Sends or receives data over the pipe. The first write() is the name of the
157 // pipe service. After the pipe service is determined, the host calls
158 // resetPipe() to replace the host-side pipe instance with an instance of the
159 // pipe service.
160 //
161 // - reset(void* initialPipe, void* actualPipe): the operation that replaces an
162 // initial pipe with an instance of a pipe service.
163 //
164 // Next, here's how the pipe operations map to virglrenderer commands:
165 //
166 // - open() -> virgl_renderer_context_create(),
167 // virgl_renderer_resource_create(),
168 // virgl_renderer_resource_attach_iov()
169 //
170 // The open() corresponds to a guest-side open of a rendernode, which triggers
171 // context creation. Each pipe corresponds 1:1 with a drm virtgpu context id.
172 // We also associate an R8 resource with each pipe as the backing data for
173 // write/read.
174 //
175 // - close() -> virgl_rendrerer_resource_unref(),
176 // virgl_renderer_context_destroy()
177 //
178 // The close() corresponds to undoing the operations of open().
179 //
180 // - write() -> virgl_renderer_transfer_write_iov() OR
181 // virgl_renderer_submit_cmd()
182 //
183 // Pipe write() operation corresponds to performing a TRANSFER_TO_HOST ioctl on
184 // the resource created alongside open(), OR an EXECBUFFER ioctl.
185 //
186 // - read() -> virgl_renderer_transfer_read_iov()
187 //
188 // Pipe read() operation corresponds to performing a TRANSFER_FROM_HOST ioctl on
189 // the resource created alongside open().
190 //
191 // Details on transfer mechanism: mapping 2D transfer to 1D ones----------------
192 //
193 // Resource objects are typically 2D textures, while we're wanting to transmit
194 // 1D buffers to the pipe services on the host. DRM VIRTGPU uses the concept
195 // of a 'box' to represent transfers that do not involve an entire resource
196 // object. Each box has a x, y, width and height parameter to define the
197 // extent of the transfer for a 2D texture. In our use case, we only use the x
198 // and width parameters. We've also created the resource with R8 format
199 // (byte-by-byte) with width equal to the total size of the transfer buffer we
200 // want (around 1 MB).
201 //
202 // The resource object itself is currently backed via plain guest RAM, which
203 // can be physically not-contiguous from the guest POV, and therefore
204 // corresponds to a possibly-long list of pointers and sizes (iov) on the host
205 // side. The sync_iov helper function converts convert the list of pointers
206 // to one contiguous buffer on the host (or vice versa), at the cost of a copy.
207 // (TODO: see if we can use host coherent memory to do away with the copy).
208 //
209 // We can see this abstraction in use via the implementation of
210 // transferWriteIov and transferReadIov below, which sync the iovec to/from a
211 // linear buffer if necessary, and then perform a corresponding pip operation
212 // based on the box parameter's x and width values.
213
214 using android::AndroidPipe;
215 using android::base::AutoLock;
216 using android::base::DescriptorType;
217 using android::base::Lock;
218 using android::base::ManagedDescriptor;
219 using android::base::MetricsLogger;
220 using android::base::SharedMemory;
221
222 using emugl::FatalError;
223 using gfxstream::BlobManager;
224 using gfxstream::ManagedDescriptorInfo;
225
226 using VirtioGpuResId = uint32_t;
227
228 static constexpr int kPipeTryAgain = -2;
229
230 struct VirtioGpuCmd {
231 uint32_t op;
232 uint32_t cmdSize;
233 unsigned char buf[0];
234 } __attribute__((packed));
235
236 struct PipeCtxEntry {
237 std::string name;
238 uint32_t capsetId;
239 VirtioGpuCtxId ctxId;
240 GoldfishHostPipe* hostPipe;
241 int fence;
242 uint32_t addressSpaceHandle;
243 bool hasAddressSpaceHandle;
244 std::unordered_map<VirtioGpuResId, uint32_t> addressSpaceHandles;
245 };
246
247 enum class ResType {
248 // Used as a communication channel between the guest and the host
249 // which does not need an allocation on the host GPU.
250 PIPE,
251 // Used as a GPU data buffer.
252 BUFFER,
253 // Used as a GPU texture.
254 COLOR_BUFFER,
255 };
256
257 struct AlignedMemory {
258 void* addr = nullptr;
259
AlignedMemoryAlignedMemory260 AlignedMemory(size_t align, size_t size)
261 : addr(android::aligned_buf_alloc(align, size)) {}
262
~AlignedMemoryAlignedMemory263 ~AlignedMemory() {
264 if (addr != nullptr) {
265 android::aligned_buf_free(addr);
266 }
267 }
268
269 // AlignedMemory is neither copyable nor movable.
270 AlignedMemory(const AlignedMemory& other) = delete;
271 AlignedMemory& operator=(const AlignedMemory& other) = delete;
272 AlignedMemory(AlignedMemory&& other) = delete;
273 AlignedMemory& operator=(AlignedMemory&& other) = delete;
274 };
275
276 // Memory used as a ring buffer for communication between the guest and host.
277 class RingBlob : public std::variant<std::unique_ptr<AlignedMemory>,
278 std::unique_ptr<SharedMemory>> {
279 public:
280 using BaseType = std::variant<std::unique_ptr<AlignedMemory>,
281 std::unique_ptr<SharedMemory>>;
282 // Inherit constructors.
283 using BaseType::BaseType;
284
isExportable() const285 bool isExportable() const {
286 return std::holds_alternative<std::unique_ptr<SharedMemory>>(*this);
287 }
288
releaseHandle()289 SharedMemory::handle_type releaseHandle() {
290 if (!isExportable()) {
291 return SharedMemory::invalidHandle();
292 }
293 return std::get<std::unique_ptr<SharedMemory>>(*this)->releaseHandle();
294 }
295 };
296
297
298 struct PipeResEntry {
299 stream_renderer_resource_create_args args;
300 iovec* iov;
301 uint32_t numIovs;
302 void* linear;
303 size_t linearSize;
304 GoldfishHostPipe* hostPipe;
305 VirtioGpuCtxId ctxId;
306 void* hva;
307 uint64_t hvaSize;
308 uint64_t blobId;
309 uint32_t blobMem;
310 uint32_t blobFlags;
311 uint32_t caching;
312 ResType type;
313 std::shared_ptr<RingBlob> ringBlob;
314 bool externalAddr = false;
315 std::shared_ptr<ManagedDescriptorInfo> descriptorInfo = nullptr;
316 };
317
align_up(uint32_t n,uint32_t a)318 static inline uint32_t align_up(uint32_t n, uint32_t a) { return ((n + a - 1) / a) * a; }
319
align_up_power_of_2(uint32_t n,uint32_t a)320 static inline uint32_t align_up_power_of_2(uint32_t n, uint32_t a) {
321 return (n + (a - 1)) & ~(a - 1);
322 }
323
324 #define VIRGL_FORMAT_NV12 166
325 #define VIRGL_FORMAT_YV12 163
326 #define VIRGL_FORMAT_P010 314
327
328 const uint32_t kGlBgra = 0x80e1;
329 const uint32_t kGlRgba = 0x1908;
330 const uint32_t kGlRgba16f = 0x881A;
331 const uint32_t kGlRgb565 = 0x8d62;
332 const uint32_t kGlRgba1010102 = 0x8059;
333 const uint32_t kGlR8 = 0x8229;
334 const uint32_t kGlR16 = 0x822A;
335 const uint32_t kGlRg8 = 0x822b;
336 const uint32_t kGlLuminance = 0x1909;
337 const uint32_t kGlLuminanceAlpha = 0x190a;
338 const uint32_t kGlUnsignedByte = 0x1401;
339 const uint32_t kGlUnsignedShort565 = 0x8363;
340
341 constexpr uint32_t kFwkFormatGlCompat = 0;
342 constexpr uint32_t kFwkFormatYV12 = 1;
343 // constexpr uint32_t kFwkFormatYUV420888 = 2;
344 constexpr uint32_t kFwkFormatNV12 = 3;
345 constexpr uint32_t kFwkFormatP010 = 4;
346
virgl_format_is_yuv(uint32_t format)347 static inline bool virgl_format_is_yuv(uint32_t format) {
348 switch (format) {
349 case VIRGL_FORMAT_B8G8R8X8_UNORM:
350 case VIRGL_FORMAT_B8G8R8A8_UNORM:
351 case VIRGL_FORMAT_R8G8B8X8_UNORM:
352 case VIRGL_FORMAT_R8G8B8A8_UNORM:
353 case VIRGL_FORMAT_B5G6R5_UNORM:
354 case VIRGL_FORMAT_R8_UNORM:
355 case VIRGL_FORMAT_R16_UNORM:
356 case VIRGL_FORMAT_R16G16B16A16_FLOAT:
357 case VIRGL_FORMAT_R8G8_UNORM:
358 case VIRGL_FORMAT_R10G10B10A2_UNORM:
359 return false;
360 case VIRGL_FORMAT_NV12:
361 case VIRGL_FORMAT_P010:
362 case VIRGL_FORMAT_YV12:
363 return true;
364 default:
365 stream_renderer_error("Unknown virgl format 0x%x", format);
366 return false;
367 }
368 }
369
virgl_format_to_gl(uint32_t virgl_format)370 static inline uint32_t virgl_format_to_gl(uint32_t virgl_format) {
371 switch (virgl_format) {
372 case VIRGL_FORMAT_B8G8R8X8_UNORM:
373 case VIRGL_FORMAT_B8G8R8A8_UNORM:
374 return kGlBgra;
375 case VIRGL_FORMAT_R8G8B8X8_UNORM:
376 case VIRGL_FORMAT_R8G8B8A8_UNORM:
377 return kGlRgba;
378 case VIRGL_FORMAT_B5G6R5_UNORM:
379 return kGlRgb565;
380 case VIRGL_FORMAT_R16_UNORM:
381 return kGlR16;
382 case VIRGL_FORMAT_R16G16B16A16_FLOAT:
383 return kGlRgba16f;
384 case VIRGL_FORMAT_R8_UNORM:
385 return kGlR8;
386 case VIRGL_FORMAT_R8G8_UNORM:
387 return kGlRg8;
388 case VIRGL_FORMAT_NV12:
389 case VIRGL_FORMAT_P010:
390 case VIRGL_FORMAT_YV12:
391 // emulated as RGBA8888
392 return kGlRgba;
393 case VIRGL_FORMAT_R10G10B10A2_UNORM:
394 return kGlRgba1010102;
395 default:
396 return kGlRgba;
397 }
398 }
399
virgl_format_to_fwk_format(uint32_t virgl_format)400 static inline uint32_t virgl_format_to_fwk_format(uint32_t virgl_format) {
401 switch (virgl_format) {
402 case VIRGL_FORMAT_NV12:
403 return kFwkFormatNV12;
404 case VIRGL_FORMAT_P010:
405 return kFwkFormatP010;
406 case VIRGL_FORMAT_YV12:
407 return kFwkFormatYV12;
408 case VIRGL_FORMAT_R8_UNORM:
409 case VIRGL_FORMAT_R16_UNORM:
410 case VIRGL_FORMAT_R16G16B16A16_FLOAT:
411 case VIRGL_FORMAT_R8G8_UNORM:
412 case VIRGL_FORMAT_B8G8R8X8_UNORM:
413 case VIRGL_FORMAT_B8G8R8A8_UNORM:
414 case VIRGL_FORMAT_R8G8B8X8_UNORM:
415 case VIRGL_FORMAT_R8G8B8A8_UNORM:
416 case VIRGL_FORMAT_B5G6R5_UNORM:
417 case VIRGL_FORMAT_R10G10B10A2_UNORM:
418 default: // kFwkFormatGlCompat: No extra conversions needed
419 return kFwkFormatGlCompat;
420 }
421 }
422
gl_format_to_natural_type(uint32_t format)423 static inline uint32_t gl_format_to_natural_type(uint32_t format) {
424 switch (format) {
425 case kGlBgra:
426 case kGlRgba:
427 case kGlLuminance:
428 case kGlLuminanceAlpha:
429 return kGlUnsignedByte;
430 case kGlRgb565:
431 return kGlUnsignedShort565;
432 default:
433 return kGlUnsignedByte;
434 }
435 }
436
virgl_format_to_linear_base(uint32_t format,uint32_t totalWidth,uint32_t totalHeight,uint32_t x,uint32_t y,uint32_t w,uint32_t h)437 static inline size_t virgl_format_to_linear_base(uint32_t format, uint32_t totalWidth,
438 uint32_t totalHeight, uint32_t x, uint32_t y,
439 uint32_t w, uint32_t h) {
440 if (virgl_format_is_yuv(format)) {
441 return 0;
442 } else {
443 uint32_t bpp = 4;
444 switch (format) {
445 case VIRGL_FORMAT_R16G16B16A16_FLOAT:
446 bpp = 8;
447 break;
448 case VIRGL_FORMAT_B8G8R8X8_UNORM:
449 case VIRGL_FORMAT_B8G8R8A8_UNORM:
450 case VIRGL_FORMAT_R8G8B8X8_UNORM:
451 case VIRGL_FORMAT_R8G8B8A8_UNORM:
452 case VIRGL_FORMAT_R10G10B10A2_UNORM:
453 bpp = 4;
454 break;
455 case VIRGL_FORMAT_B5G6R5_UNORM:
456 case VIRGL_FORMAT_R8G8_UNORM:
457 case VIRGL_FORMAT_R16_UNORM:
458 bpp = 2;
459 break;
460 case VIRGL_FORMAT_R8_UNORM:
461 bpp = 1;
462 break;
463 default:
464 stream_renderer_error("Unknown virgl format: 0x%x", format);
465 return 0;
466 }
467
468 uint32_t stride = totalWidth * bpp;
469 return y * stride + x * bpp;
470 }
471 return 0;
472 }
473
virgl_format_to_total_xfer_len(uint32_t format,uint32_t totalWidth,uint32_t totalHeight,uint32_t x,uint32_t y,uint32_t w,uint32_t h)474 static inline size_t virgl_format_to_total_xfer_len(uint32_t format, uint32_t totalWidth,
475 uint32_t totalHeight, uint32_t x, uint32_t y,
476 uint32_t w, uint32_t h) {
477 if (virgl_format_is_yuv(format)) {
478 uint32_t bpp = format == VIRGL_FORMAT_P010 ? 2 : 1;
479
480 uint32_t yWidth = totalWidth;
481 uint32_t yHeight = totalHeight;
482 uint32_t yStridePixels;
483 if (format == VIRGL_FORMAT_NV12) {
484 yStridePixels = yWidth;
485 } else if (format == VIRGL_FORMAT_P010) {
486 yStridePixels = yWidth;
487 } else if (format == VIRGL_FORMAT_YV12) {
488 yStridePixels = align_up_power_of_2(yWidth, 32);
489 } else {
490 stream_renderer_error("Unknown virgl format: 0x%x", format);
491 return 0;
492 }
493 uint32_t yStrideBytes = yStridePixels * bpp;
494 uint32_t ySize = yStrideBytes * yHeight;
495
496 uint32_t uvStridePixels;
497 uint32_t uvPlaneCount;
498 if (format == VIRGL_FORMAT_NV12) {
499 uvStridePixels = yStridePixels;
500 uvPlaneCount = 1;
501 } else if (format == VIRGL_FORMAT_P010) {
502 uvStridePixels = yStridePixels;
503 uvPlaneCount = 1;
504 } else if (format == VIRGL_FORMAT_YV12) {
505 uvStridePixels = yStridePixels / 2;
506 uvPlaneCount = 2;
507 } else {
508 stream_renderer_error("Unknown virgl yuv format: 0x%x", format);
509 return 0;
510 }
511 uint32_t uvStrideBytes = uvStridePixels * bpp;
512 uint32_t uvHeight = totalHeight / 2;
513 uint32_t uvSize = uvStrideBytes * uvHeight * uvPlaneCount;
514
515 uint32_t dataSize = ySize + uvSize;
516 return dataSize;
517 } else {
518 uint32_t bpp = 4;
519 switch (format) {
520 case VIRGL_FORMAT_R16G16B16A16_FLOAT:
521 bpp = 8;
522 break;
523 case VIRGL_FORMAT_B8G8R8X8_UNORM:
524 case VIRGL_FORMAT_B8G8R8A8_UNORM:
525 case VIRGL_FORMAT_R8G8B8X8_UNORM:
526 case VIRGL_FORMAT_R8G8B8A8_UNORM:
527 case VIRGL_FORMAT_R10G10B10A2_UNORM:
528 bpp = 4;
529 break;
530 case VIRGL_FORMAT_B5G6R5_UNORM:
531 case VIRGL_FORMAT_R16_UNORM:
532 case VIRGL_FORMAT_R8G8_UNORM:
533 bpp = 2;
534 break;
535 case VIRGL_FORMAT_R8_UNORM:
536 bpp = 1;
537 break;
538 default:
539 stream_renderer_error("Unknown virgl format: 0x%x", format);
540 return 0;
541 }
542
543 uint32_t stride = totalWidth * bpp;
544 return (h - 1U) * stride + w * bpp;
545 }
546 return 0;
547 }
548
549 enum IovSyncDir {
550 IOV_TO_LINEAR = 0,
551 LINEAR_TO_IOV = 1,
552 };
553
sync_iov(PipeResEntry * res,uint64_t offset,const stream_renderer_box * box,IovSyncDir dir)554 static int sync_iov(PipeResEntry* res, uint64_t offset, const stream_renderer_box* box,
555 IovSyncDir dir) {
556 stream_renderer_debug("offset: 0x%llx box: %u %u %u %u size %u x %u iovs %u linearSize %zu",
557 (unsigned long long)offset, box->x, box->y, box->w, box->h,
558 res->args.width, res->args.height, res->numIovs, res->linearSize);
559
560 if (box->x > res->args.width || box->y > res->args.height) {
561 stream_renderer_error("Box out of range of resource");
562 return -EINVAL;
563 }
564 if (box->w == 0U || box->h == 0U) {
565 stream_renderer_error("Empty transfer");
566 return -EINVAL;
567 }
568 if (box->x + box->w > res->args.width) {
569 stream_renderer_error("Box overflows resource width");
570 return -EINVAL;
571 }
572
573 size_t linearBase = virgl_format_to_linear_base(
574 res->args.format, res->args.width, res->args.height, box->x, box->y, box->w, box->h);
575 size_t start = linearBase;
576 // height - 1 in order to treat the (w * bpp) row specially
577 // (i.e., the last row does not occupy the full stride)
578 size_t length = virgl_format_to_total_xfer_len(
579 res->args.format, res->args.width, res->args.height, box->x, box->y, box->w, box->h);
580 size_t end = start + length;
581
582 if (start == end) {
583 stream_renderer_error("nothing to transfer");
584 return -EINVAL;
585 }
586
587 if (end > res->linearSize) {
588 stream_renderer_error("start + length overflows!");
589 return -EINVAL;
590 }
591
592 uint32_t iovIndex = 0;
593 size_t iovOffset = 0;
594 size_t written = 0;
595 char* linear = static_cast<char*>(res->linear);
596
597 while (written < length) {
598 if (iovIndex >= res->numIovs) {
599 stream_renderer_error("write request overflowed numIovs");
600 return -EINVAL;
601 }
602
603 const char* iovBase_const = static_cast<const char*>(res->iov[iovIndex].iov_base);
604 char* iovBase = static_cast<char*>(res->iov[iovIndex].iov_base);
605 size_t iovLen = res->iov[iovIndex].iov_len;
606 size_t iovOffsetEnd = iovOffset + iovLen;
607
608 auto lower_intersect = std::max(iovOffset, start);
609 auto upper_intersect = std::min(iovOffsetEnd, end);
610 if (lower_intersect < upper_intersect) {
611 size_t toWrite = upper_intersect - lower_intersect;
612 switch (dir) {
613 case IOV_TO_LINEAR:
614 memcpy(linear + lower_intersect, iovBase_const + lower_intersect - iovOffset,
615 toWrite);
616 break;
617 case LINEAR_TO_IOV:
618 memcpy(iovBase + lower_intersect - iovOffset, linear + lower_intersect,
619 toWrite);
620 break;
621 default:
622 stream_renderer_error("Invalid synchronization dir");
623 return -EINVAL;
624 }
625 written += toWrite;
626 }
627 ++iovIndex;
628 iovOffset += iovLen;
629 }
630
631 return 0;
632 }
633
convert32to64(uint32_t lo,uint32_t hi)634 static uint64_t convert32to64(uint32_t lo, uint32_t hi) {
635 return ((uint64_t)lo) | (((uint64_t)hi) << 32);
636 }
637
638 class CleanupThread {
639 public:
640 using GenericCleanup = std::function<void()>;
641
__anonf6f38b3e0102(CleanupTask task) 642 CleanupThread() : mWorker([](CleanupTask task) {
643 return std::visit([](auto&& work) {
644 using T = std::decay_t<decltype(work)>;
645 if constexpr (std::is_same_v<T, GenericCleanup>) {
646 work();
647 return android::base::WorkerProcessingResult::Continue;
648 } else if constexpr (std::is_same_v<T, Exit>) {
649 return android::base::WorkerProcessingResult::Stop;
650 }
651 }, std::move(task));
652 }) {
653 mWorker.start();
654 }
655
~CleanupThread()656 ~CleanupThread() { stop(); }
657
658 // CleanupThread is neither copyable nor movable.
659 CleanupThread(const CleanupThread& other) = delete;
660 CleanupThread& operator=(const CleanupThread& other) = delete;
661 CleanupThread(CleanupThread&& other) = delete;
662 CleanupThread& operator=(CleanupThread&& other) = delete;
663
enqueueCleanup(GenericCleanup command)664 void enqueueCleanup(GenericCleanup command) {
665 mWorker.enqueue(std::move(command));
666 }
667
stop()668 void stop() {
669 mWorker.enqueue(Exit{});
670 mWorker.join();
671 }
672
673 private:
674 struct Exit {};
675 using CleanupTask = std::variant<GenericCleanup, Exit>;
676 android::base::WorkerThread<CleanupTask> mWorker;
677 };
678
679 class PipeVirglRenderer {
680 public:
681 PipeVirglRenderer() = default;
682
init(void * cookie,gfxstream::host::FeatureSet features,stream_renderer_fence_callback fence_callback)683 int init(void* cookie, gfxstream::host::FeatureSet features, stream_renderer_fence_callback fence_callback) {
684 stream_renderer_debug("cookie: %p", cookie);
685 mCookie = cookie;
686 mFeatures = features;
687 mFenceCallback = fence_callback;
688 mVirtioGpuOps = android_getVirtioGpuOps();
689 if (!mVirtioGpuOps) {
690 stream_renderer_error("Could not get virtio gpu ops!");
691 return -EINVAL;
692 }
693 mAddressSpaceDeviceControlOps = get_address_space_device_control_ops();
694 if (!mAddressSpaceDeviceControlOps) {
695 stream_renderer_error("Could not get address space device control ops!");
696 return -EINVAL;
697 }
698 mVirtioGpuTimelines = VirtioGpuTimelines::create(true);
699 mVirtioGpuTimelines = VirtioGpuTimelines::create(true);
700
701 #if !defined(_WIN32)
702 mPageSize = getpagesize();
703 #endif
704
705 mCleanupThread.reset(new CleanupThread());
706
707 return 0;
708 }
709
teardown()710 void teardown() {
711 mCleanupThread.reset();
712 }
713
resetPipe(GoldfishHwPipe * hwPipe,GoldfishHostPipe * hostPipe)714 int resetPipe(GoldfishHwPipe* hwPipe, GoldfishHostPipe* hostPipe) {
715 stream_renderer_debug("Want to reset hwpipe %p to hostpipe %p", hwPipe, hostPipe);
716 VirtioGpuCtxId asCtxId = (VirtioGpuCtxId)(uintptr_t)hwPipe;
717 auto it = mContexts.find(asCtxId);
718 if (it == mContexts.end()) {
719 stream_renderer_error("fatal: pipe id %u", asCtxId);
720 return -EINVAL;
721 }
722
723 auto& entry = it->second;
724 stream_renderer_debug("ctxid: %u prev hostpipe: %p", asCtxId, entry.hostPipe);
725 entry.hostPipe = hostPipe;
726 stream_renderer_debug("ctxid: %u next hostpipe: %p", asCtxId, entry.hostPipe);
727
728 // Also update any resources associated with it
729 auto resourcesIt = mContextResources.find(asCtxId);
730
731 if (resourcesIt == mContextResources.end()) {
732 return 0;
733 }
734
735 const auto& resIds = resourcesIt->second;
736
737 for (auto resId : resIds) {
738 auto resEntryIt = mResources.find(resId);
739 if (resEntryIt == mResources.end()) {
740 stream_renderer_error("entry with res id %u not found", resId);
741 return -EINVAL;
742 }
743
744 auto& resEntry = resEntryIt->second;
745 resEntry.hostPipe = hostPipe;
746 }
747
748 return 0;
749 }
750
createContext(VirtioGpuCtxId ctx_id,uint32_t nlen,const char * name,uint32_t context_init)751 int createContext(VirtioGpuCtxId ctx_id, uint32_t nlen, const char* name,
752 uint32_t context_init) {
753 std::string contextName(name, nlen);
754
755 stream_renderer_debug("ctxid: %u len: %u name: %s", ctx_id, nlen, contextName.c_str());
756 auto ops = ensureAndGetServiceOps();
757 auto hostPipe = ops->guest_open_with_flags(reinterpret_cast<GoldfishHwPipe*>(ctx_id),
758 0x1 /* is virtio */);
759
760 if (!hostPipe) {
761 stream_renderer_error("failed to create hw pipe!");
762 return -EINVAL;
763 }
764 std::unordered_map<uint32_t, uint32_t> map;
765
766 PipeCtxEntry res = {
767 std::move(contextName), // contextName
768 context_init, // capsetId
769 ctx_id, // ctxId
770 hostPipe, // hostPipe
771 0, // fence
772 0, // AS handle
773 false, // does not have an AS handle
774 map, // resourceId --> ASG handle map
775 };
776
777 stream_renderer_debug("initial host pipe for ctxid %u: %p", ctx_id, hostPipe);
778 mContexts[ctx_id] = res;
779 android_onGuestGraphicsProcessCreate(ctx_id);
780 return 0;
781 }
782
destroyContext(VirtioGpuCtxId handle)783 int destroyContext(VirtioGpuCtxId handle) {
784 stream_renderer_debug("ctxid: %u", handle);
785
786 auto it = mContexts.find(handle);
787 if (it == mContexts.end()) {
788 stream_renderer_error("could not find context handle %u", handle);
789 return -EINVAL;
790 }
791
792 if (it->second.hasAddressSpaceHandle) {
793 for (auto const& [resourceId, handle] : it->second.addressSpaceHandles) {
794 // Note: this can hang as is but this has only been observed to
795 // happen during shutdown. See b/329287602#comment8.
796 mAddressSpaceDeviceControlOps->destroy_handle(handle);
797 }
798 }
799
800 auto hostPipe = it->second.hostPipe;
801 if (!hostPipe) {
802 stream_renderer_error("0 is not a valid hostpipe");
803 return -EINVAL;
804 }
805
806 auto ops = ensureAndGetServiceOps();
807 ops->guest_close(hostPipe, GOLDFISH_PIPE_CLOSE_GRACEFUL);
808
809 android_cleanupProcGLObjects(handle);
810 mContexts.erase(it);
811 return 0;
812 }
813
setContextAddressSpaceHandleLocked(VirtioGpuCtxId ctxId,uint32_t handle,uint32_t resourceId)814 int setContextAddressSpaceHandleLocked(VirtioGpuCtxId ctxId, uint32_t handle,
815 uint32_t resourceId) {
816 auto ctxIt = mContexts.find(ctxId);
817 if (ctxIt == mContexts.end()) {
818 stream_renderer_error("ctx id %u is not found", ctxId);
819 return -EINVAL;
820 }
821
822 auto& ctxEntry = ctxIt->second;
823 ctxEntry.addressSpaceHandle = handle;
824 ctxEntry.hasAddressSpaceHandle = true;
825 ctxEntry.addressSpaceHandles[resourceId] = handle;
826 return 0;
827 }
828
getAddressSpaceHandleLocked(VirtioGpuCtxId ctxId,uint32_t resourceId)829 uint32_t getAddressSpaceHandleLocked(VirtioGpuCtxId ctxId, uint32_t resourceId) {
830 auto ctxIt = mContexts.find(ctxId);
831 if (ctxIt == mContexts.end()) {
832 stream_renderer_error("ctx id %u is not found", ctxId);
833 return -EINVAL;
834 }
835
836 auto& ctxEntry = ctxIt->second;
837
838 if (!ctxEntry.addressSpaceHandles.count(resourceId)) {
839 stream_renderer_error("ASG context with resource id %u", resourceId);
840 return -EINVAL;
841 }
842
843 return ctxEntry.addressSpaceHandles[resourceId];
844 }
845
846 #define DECODE(variable, type, input) \
847 type variable = {}; \
848 memcpy(&variable, input, sizeof(type));
849
addressSpaceProcessCmd(VirtioGpuCtxId ctxId,uint32_t * dwords)850 int addressSpaceProcessCmd(VirtioGpuCtxId ctxId, uint32_t* dwords) {
851 DECODE(header, gfxstream::gfxstreamHeader, dwords)
852
853 switch (header.opCode) {
854 case GFXSTREAM_CONTEXT_CREATE: {
855 DECODE(contextCreate, gfxstream::gfxstreamContextCreate, dwords)
856
857 auto resEntryIt = mResources.find(contextCreate.resourceId);
858 if (resEntryIt == mResources.end()) {
859 stream_renderer_error("ASG coherent resource %u not found",
860 contextCreate.resourceId);
861 return -EINVAL;
862 }
863
864 auto ctxIt = mContexts.find(ctxId);
865 if (ctxIt == mContexts.end()) {
866 stream_renderer_error("ctx id %u not found", ctxId);
867 return -EINVAL;
868 }
869
870 auto& ctxEntry = ctxIt->second;
871 auto& resEntry = resEntryIt->second;
872
873 std::string name = ctxEntry.name + "-" + std::to_string(contextCreate.resourceId);
874
875 // Note: resource ids can not be used as ASG handles because ASGs may outlive the
876 // containing resource due asynchronous ASG destruction.
877 uint32_t handle = mAddressSpaceDeviceControlOps->gen_handle();
878
879 struct AddressSpaceCreateInfo createInfo = {
880 .handle = handle,
881 .type = android::emulation::VirtioGpuGraphics,
882 .createRenderThread = true,
883 .externalAddr = resEntry.hva,
884 .externalAddrSize = resEntry.hvaSize,
885 .virtioGpuContextId = ctxId,
886 .virtioGpuCapsetId = ctxEntry.capsetId,
887 .contextName = name.c_str(),
888 .contextNameSize = static_cast<uint32_t>(ctxEntry.name.size()),
889 };
890
891 mAddressSpaceDeviceControlOps->create_instance(createInfo);
892 if (setContextAddressSpaceHandleLocked(ctxId, handle, contextCreate.resourceId)) {
893 return -EINVAL;
894 }
895 break;
896 }
897 case GFXSTREAM_CONTEXT_PING: {
898 DECODE(contextPing, gfxstream::gfxstreamContextPing, dwords)
899
900 struct android::emulation::AddressSpaceDevicePingInfo ping = {0};
901 ping.metadata = ASG_NOTIFY_AVAILABLE;
902
903 mAddressSpaceDeviceControlOps->ping_at_hva(
904 getAddressSpaceHandleLocked(ctxId, contextPing.resourceId), &ping);
905 break;
906 }
907 default:
908 break;
909 }
910
911 return 0;
912 }
913
submitCmd(struct stream_renderer_command * cmd)914 int submitCmd(struct stream_renderer_command* cmd) {
915 if (!cmd) return -EINVAL;
916
917 void* buffer = reinterpret_cast<void*>(cmd->cmd);
918
919 VirtioGpuRing ring = VirtioGpuRingGlobal{};
920 stream_renderer_debug("ctx: % u, ring: %s buffer: %p dwords: %d", cmd->ctx_id,
921 to_string(ring).c_str(), buffer, cmd->cmd_size);
922
923 if (!buffer) {
924 stream_renderer_error("error: buffer null");
925 return -EINVAL;
926 }
927
928 if (cmd->cmd_size < 4) {
929 stream_renderer_error("error: not enough bytes (got %d)", cmd->cmd_size);
930 return -EINVAL;
931 }
932
933 DECODE(header, gfxstream::gfxstreamHeader, buffer);
934 switch (header.opCode) {
935 case GFXSTREAM_CONTEXT_CREATE:
936 case GFXSTREAM_CONTEXT_PING:
937 case GFXSTREAM_CONTEXT_PING_WITH_RESPONSE:
938 if (addressSpaceProcessCmd(cmd->ctx_id, (uint32_t*)buffer)) {
939 return -EINVAL;
940 }
941 break;
942 case GFXSTREAM_CREATE_EXPORT_SYNC: {
943 DECODE(exportSync, gfxstream::gfxstreamCreateExportSync, buffer)
944
945 uint64_t sync_handle =
946 convert32to64(exportSync.syncHandleLo, exportSync.syncHandleHi);
947
948 stream_renderer_debug("wait for gpu ring %s", to_string(ring).c_str());
949 auto taskId = mVirtioGpuTimelines->enqueueTask(ring);
950 mVirtioGpuOps->async_wait_for_gpu_with_cb(sync_handle, [this, taskId] {
951 mVirtioGpuTimelines->notifyTaskCompletion(taskId);
952 });
953 break;
954 }
955 case GFXSTREAM_CREATE_EXPORT_SYNC_VK:
956 case GFXSTREAM_CREATE_IMPORT_SYNC_VK: {
957 // The guest sync export assumes fence context support and always uses
958 // VIRTGPU_EXECBUF_RING_IDX. With this, the task created here must use
959 // the same ring as the fence created for the virtio gpu command or the
960 // fence may be signaled without properly waiting for the task to complete.
961 ring = VirtioGpuRingContextSpecific{
962 .mCtxId = cmd->ctx_id,
963 .mRingIdx = 0,
964 };
965
966 DECODE(exportSyncVK, gfxstream::gfxstreamCreateExportSyncVK, buffer)
967
968 uint64_t device_handle =
969 convert32to64(exportSyncVK.deviceHandleLo, exportSyncVK.deviceHandleHi);
970
971 uint64_t fence_handle =
972 convert32to64(exportSyncVK.fenceHandleLo, exportSyncVK.fenceHandleHi);
973
974 stream_renderer_debug("wait for gpu ring %s", to_string(ring).c_str());
975 auto taskId = mVirtioGpuTimelines->enqueueTask(ring);
976 mVirtioGpuOps->async_wait_for_gpu_vulkan_with_cb(
977 device_handle, fence_handle,
978 [this, taskId] { mVirtioGpuTimelines->notifyTaskCompletion(taskId); });
979 break;
980 }
981 case GFXSTREAM_CREATE_QSRI_EXPORT_VK: {
982 // The guest QSRI export assumes fence context support and always uses
983 // VIRTGPU_EXECBUF_RING_IDX. With this, the task created here must use
984 // the same ring as the fence created for the virtio gpu command or the
985 // fence may be signaled without properly waiting for the task to complete.
986 ring = VirtioGpuRingContextSpecific{
987 .mCtxId = cmd->ctx_id,
988 .mRingIdx = 0,
989 };
990
991 DECODE(exportQSRI, gfxstream::gfxstreamCreateQSRIExportVK, buffer)
992
993 uint64_t image_handle =
994 convert32to64(exportQSRI.imageHandleLo, exportQSRI.imageHandleHi);
995
996 stream_renderer_debug("wait for gpu vk qsri ring %u image 0x%llx",
997 to_string(ring).c_str(), (unsigned long long)image_handle);
998 auto taskId = mVirtioGpuTimelines->enqueueTask(ring);
999 mVirtioGpuOps->async_wait_for_gpu_vulkan_qsri_with_cb(image_handle, [this, taskId] {
1000 mVirtioGpuTimelines->notifyTaskCompletion(taskId);
1001 });
1002 break;
1003 }
1004 case GFXSTREAM_PLACEHOLDER_COMMAND_VK: {
1005 // Do nothing, this is a placeholder command
1006 break;
1007 }
1008 default:
1009 return -EINVAL;
1010 }
1011
1012 return 0;
1013 }
1014
createFence(uint64_t fence_id,const VirtioGpuRing & ring)1015 int createFence(uint64_t fence_id, const VirtioGpuRing& ring) {
1016 stream_renderer_debug("fenceid: %llu ring: %s", (unsigned long long)fence_id,
1017 to_string(ring).c_str());
1018
1019 struct {
1020 FenceCompletionCallback operator()(const VirtioGpuRingGlobal&) {
1021 return [renderer = mRenderer, fenceId = mFenceId] {
1022 struct stream_renderer_fence fence = {0};
1023 fence.fence_id = fenceId;
1024 fence.flags = STREAM_RENDERER_FLAG_FENCE;
1025 renderer->mFenceCallback(renderer->mCookie, &fence);
1026 };
1027 }
1028 FenceCompletionCallback operator()(const VirtioGpuRingContextSpecific& ring) {
1029 return [renderer = mRenderer, fenceId = mFenceId, ring] {
1030 struct stream_renderer_fence fence = {0};
1031 fence.fence_id = fenceId;
1032 fence.flags = STREAM_RENDERER_FLAG_FENCE | STREAM_RENDERER_FLAG_FENCE_RING_IDX;
1033 fence.ctx_id = ring.mCtxId;
1034 fence.ring_idx = ring.mRingIdx;
1035 renderer->mFenceCallback(renderer->mCookie, &fence);
1036 };
1037 }
1038
1039 PipeVirglRenderer* mRenderer;
1040 VirtioGpuTimelines::FenceId mFenceId;
1041 } visitor{
1042 .mRenderer = this,
1043 .mFenceId = fence_id,
1044 };
1045 FenceCompletionCallback callback = std::visit(visitor, ring);
1046 if (!callback) {
1047 return -EINVAL;
1048 }
1049 mVirtioGpuTimelines->enqueueFence(ring, fence_id, std::move(callback));
1050
1051 return 0;
1052 }
1053
poll()1054 void poll() { mVirtioGpuTimelines->poll(); }
1055
1056 enum pipe_texture_target {
1057 PIPE_BUFFER,
1058 PIPE_TEXTURE_1D,
1059 PIPE_TEXTURE_2D,
1060 PIPE_TEXTURE_3D,
1061 PIPE_TEXTURE_CUBE,
1062 PIPE_TEXTURE_RECT,
1063 PIPE_TEXTURE_1D_ARRAY,
1064 PIPE_TEXTURE_2D_ARRAY,
1065 PIPE_TEXTURE_CUBE_ARRAY,
1066 PIPE_MAX_TEXTURE_TYPES,
1067 };
1068
1069 /**
1070 * * Resource binding flags -- state tracker must specify in advance all
1071 * * the ways a resource might be used.
1072 * */
1073 #define PIPE_BIND_DEPTH_STENCIL (1 << 0) /* create_surface */
1074 #define PIPE_BIND_RENDER_TARGET (1 << 1) /* create_surface */
1075 #define PIPE_BIND_BLENDABLE (1 << 2) /* create_surface */
1076 #define PIPE_BIND_SAMPLER_VIEW (1 << 3) /* create_sampler_view */
1077 #define PIPE_BIND_VERTEX_BUFFER (1 << 4) /* set_vertex_buffers */
1078 #define PIPE_BIND_INDEX_BUFFER (1 << 5) /* draw_elements */
1079 #define PIPE_BIND_CONSTANT_BUFFER (1 << 6) /* set_constant_buffer */
1080 #define PIPE_BIND_DISPLAY_TARGET (1 << 7) /* flush_front_buffer */
1081 /* gap */
1082 #define PIPE_BIND_STREAM_OUTPUT (1 << 10) /* set_stream_output_buffers */
1083 #define PIPE_BIND_CURSOR (1 << 11) /* mouse cursor */
1084 #define PIPE_BIND_CUSTOM (1 << 12) /* state-tracker/winsys usages */
1085 #define PIPE_BIND_GLOBAL (1 << 13) /* set_global_binding */
1086 #define PIPE_BIND_SHADER_BUFFER (1 << 14) /* set_shader_buffers */
1087 #define PIPE_BIND_SHADER_IMAGE (1 << 15) /* set_shader_images */
1088 #define PIPE_BIND_COMPUTE_RESOURCE (1 << 16) /* set_compute_resources */
1089 #define PIPE_BIND_COMMAND_ARGS_BUFFER (1 << 17) /* pipe_draw_info.indirect */
1090 #define PIPE_BIND_QUERY_BUFFER (1 << 18) /* get_query_result_resource */
1091
getResourceType(const struct stream_renderer_resource_create_args & args) const1092 ResType getResourceType(const struct stream_renderer_resource_create_args& args) const {
1093 if (args.target == PIPE_BUFFER) {
1094 return ResType::PIPE;
1095 }
1096
1097 if (args.format != VIRGL_FORMAT_R8_UNORM) {
1098 return ResType::COLOR_BUFFER;
1099 }
1100 if (args.bind & VIRGL_BIND_SAMPLER_VIEW) {
1101 return ResType::COLOR_BUFFER;
1102 }
1103 if (args.bind & VIRGL_BIND_RENDER_TARGET) {
1104 return ResType::COLOR_BUFFER;
1105 }
1106 if (args.bind & VIRGL_BIND_SCANOUT) {
1107 return ResType::COLOR_BUFFER;
1108 }
1109 if (args.bind & VIRGL_BIND_CURSOR) {
1110 return ResType::COLOR_BUFFER;
1111 }
1112 if (!(args.bind & VIRGL_BIND_LINEAR)) {
1113 return ResType::COLOR_BUFFER;
1114 }
1115
1116 return ResType::BUFFER;
1117 }
1118
handleCreateResourceBuffer(struct stream_renderer_resource_create_args * args)1119 void handleCreateResourceBuffer(struct stream_renderer_resource_create_args* args) {
1120 stream_renderer_debug("w:%u h:%u handle:%u", args->handle, args->width, args->height);
1121 mVirtioGpuOps->create_buffer_with_handle(args->width * args->height, args->handle);
1122 }
1123
handleCreateResourceColorBuffer(struct stream_renderer_resource_create_args * args)1124 void handleCreateResourceColorBuffer(struct stream_renderer_resource_create_args* args) {
1125 stream_renderer_debug("w h %u %u resid %u -> CreateColorBufferWithHandle", args->width,
1126 args->height, args->handle);
1127
1128 const uint32_t glformat = virgl_format_to_gl(args->format);
1129 const uint32_t fwkformat = virgl_format_to_fwk_format(args->format);
1130 const bool linear = !!(args->bind & VIRGL_BIND_LINEAR);
1131 mVirtioGpuOps->create_color_buffer_with_handle(args->width, args->height, glformat,
1132 fwkformat, args->handle, linear);
1133 mVirtioGpuOps->set_guest_managed_color_buffer_lifetime(true /* guest manages lifetime */);
1134 mVirtioGpuOps->open_color_buffer(args->handle);
1135 }
1136
createResource(struct stream_renderer_resource_create_args * args,struct iovec * iov,uint32_t num_iovs)1137 int createResource(struct stream_renderer_resource_create_args* args, struct iovec* iov,
1138 uint32_t num_iovs) {
1139 stream_renderer_debug("handle: %u. num iovs: %u", args->handle, num_iovs);
1140
1141 const auto resType = getResourceType(*args);
1142 switch (resType) {
1143 case ResType::PIPE:
1144 break;
1145 case ResType::BUFFER:
1146 handleCreateResourceBuffer(args);
1147 break;
1148 case ResType::COLOR_BUFFER:
1149 handleCreateResourceColorBuffer(args);
1150 break;
1151 }
1152
1153 PipeResEntry e;
1154 e.args = *args;
1155 e.linear = 0;
1156 e.hostPipe = 0;
1157 e.hva = nullptr;
1158 e.hvaSize = 0;
1159 e.blobId = 0;
1160 e.blobMem = 0;
1161 e.type = resType;
1162 allocResource(e, iov, num_iovs);
1163
1164 mResources[args->handle] = e;
1165 return 0;
1166 }
1167
unrefResource(uint32_t toUnrefId)1168 void unrefResource(uint32_t toUnrefId) {
1169 stream_renderer_debug("handle: %u", toUnrefId);
1170
1171 auto it = mResources.find(toUnrefId);
1172 if (it == mResources.end()) return;
1173
1174 auto contextsIt = mResourceContexts.find(toUnrefId);
1175 if (contextsIt != mResourceContexts.end()) {
1176 mResourceContexts.erase(contextsIt->first);
1177 }
1178
1179 for (auto& ctxIdResources : mContextResources) {
1180 detachResourceLocked(ctxIdResources.first, toUnrefId);
1181 }
1182
1183 auto& entry = it->second;
1184 switch (entry.type) {
1185 case ResType::PIPE:
1186 break;
1187 case ResType::BUFFER:
1188 mVirtioGpuOps->close_buffer(toUnrefId);
1189 break;
1190 case ResType::COLOR_BUFFER:
1191 mVirtioGpuOps->close_color_buffer(toUnrefId);
1192 break;
1193 }
1194
1195 if (entry.linear) {
1196 free(entry.linear);
1197 entry.linear = nullptr;
1198 }
1199
1200 if (entry.iov) {
1201 free(entry.iov);
1202 entry.iov = nullptr;
1203 entry.numIovs = 0;
1204 }
1205
1206 entry.hva = nullptr;
1207 entry.hvaSize = 0;
1208 entry.blobId = 0;
1209
1210 mResources.erase(it);
1211 }
1212
attachIov(int resId,iovec * iov,int num_iovs)1213 int attachIov(int resId, iovec* iov, int num_iovs) {
1214 stream_renderer_debug("resid: %d numiovs: %d", resId, num_iovs);
1215
1216 auto it = mResources.find(resId);
1217 if (it == mResources.end()) return ENOENT;
1218
1219 auto& entry = it->second;
1220 stream_renderer_debug("res linear: %p", entry.linear);
1221 if (!entry.linear) allocResource(entry, iov, num_iovs);
1222
1223 stream_renderer_debug("done");
1224 return 0;
1225 }
1226
detachIov(int resId,iovec ** iov,int * num_iovs)1227 void detachIov(int resId, iovec** iov, int* num_iovs) {
1228 auto it = mResources.find(resId);
1229 if (it == mResources.end()) return;
1230
1231 auto& entry = it->second;
1232
1233 if (num_iovs) {
1234 *num_iovs = entry.numIovs;
1235 stream_renderer_debug("resid: %d numIovs: %d", resId, *num_iovs);
1236 } else {
1237 stream_renderer_debug("resid: %d numIovs: 0", resId);
1238 }
1239
1240 entry.numIovs = 0;
1241
1242 if (entry.iov) free(entry.iov);
1243 entry.iov = nullptr;
1244
1245 if (iov) {
1246 *iov = entry.iov;
1247 }
1248
1249 allocResource(entry, entry.iov, entry.numIovs);
1250 stream_renderer_debug("done");
1251 }
1252
handleTransferReadPipe(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1253 int handleTransferReadPipe(PipeResEntry* res, uint64_t offset, stream_renderer_box* box) {
1254 if (res->type != ResType::PIPE) {
1255 stream_renderer_error("resid: %d not a PIPE resource", res->args.handle);
1256 return -EINVAL;
1257 }
1258
1259 // Do the pipe service op here, if there is an associated hostpipe.
1260 auto hostPipe = res->hostPipe;
1261 if (!hostPipe) return -EINVAL;
1262
1263 auto ops = ensureAndGetServiceOps();
1264
1265 size_t readBytes = 0;
1266 size_t wantedBytes = readBytes + (size_t)box->w;
1267
1268 while (readBytes < wantedBytes) {
1269 GoldfishPipeBuffer buf = {
1270 ((char*)res->linear) + box->x + readBytes,
1271 wantedBytes - readBytes,
1272 };
1273 auto status = ops->guest_recv(hostPipe, &buf, 1);
1274
1275 if (status > 0) {
1276 readBytes += status;
1277 } else if (status == kPipeTryAgain) {
1278 ops->wait_guest_recv(hostPipe);
1279 } else {
1280 return EIO;
1281 }
1282 }
1283
1284 return 0;
1285 }
1286
handleTransferWritePipe(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1287 int handleTransferWritePipe(PipeResEntry* res, uint64_t offset, stream_renderer_box* box) {
1288 if (res->type != ResType::PIPE) {
1289 stream_renderer_error("resid: %d not a PIPE resource", res->args.handle);
1290 return -EINVAL;
1291 }
1292
1293 // Do the pipe service op here, if there is an associated hostpipe.
1294 auto hostPipe = res->hostPipe;
1295 if (!hostPipe) {
1296 stream_renderer_error("No hostPipe");
1297 return -EINVAL;
1298 }
1299
1300 stream_renderer_debug("resid: %d offset: 0x%llx hostpipe: %p", res->args.handle,
1301 (unsigned long long)offset, hostPipe);
1302
1303 auto ops = ensureAndGetServiceOps();
1304
1305 size_t writtenBytes = 0;
1306 size_t wantedBytes = (size_t)box->w;
1307
1308 while (writtenBytes < wantedBytes) {
1309 GoldfishPipeBuffer buf = {
1310 ((char*)res->linear) + box->x + writtenBytes,
1311 wantedBytes - writtenBytes,
1312 };
1313
1314 // guest_send can now reallocate the pipe.
1315 void* hostPipeBefore = hostPipe;
1316 auto status = ops->guest_send(&hostPipe, &buf, 1);
1317 if (hostPipe != hostPipeBefore) {
1318 if (resetPipe((GoldfishHwPipe*)(uintptr_t)(res->ctxId), hostPipe)) {
1319 return -EINVAL;
1320 }
1321
1322 auto it = mResources.find(res->args.handle);
1323 res = &it->second;
1324 }
1325
1326 if (status > 0) {
1327 writtenBytes += status;
1328 } else if (status == kPipeTryAgain) {
1329 ops->wait_guest_send(hostPipe);
1330 } else {
1331 return EIO;
1332 }
1333 }
1334
1335 return 0;
1336 }
1337
handleTransferReadBuffer(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1338 int handleTransferReadBuffer(PipeResEntry* res, uint64_t offset, stream_renderer_box* box) {
1339 if (res->type != ResType::BUFFER) {
1340 stream_renderer_error("resid: %d not a BUFFER resource", res->args.handle);
1341 return -EINVAL;
1342 }
1343
1344 mVirtioGpuOps->read_buffer(res->args.handle, 0, res->args.width * res->args.height,
1345 res->linear);
1346 return 0;
1347 }
1348
handleTransferWriteBuffer(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1349 int handleTransferWriteBuffer(PipeResEntry* res, uint64_t offset, stream_renderer_box* box) {
1350 if (res->type != ResType::BUFFER) {
1351 stream_renderer_error("resid: %d not a BUFFER resource", res->args.handle);
1352 return -EINVAL;
1353 }
1354
1355 mVirtioGpuOps->update_buffer(res->args.handle, 0, res->args.width * res->args.height,
1356 res->linear);
1357 return 0;
1358 }
1359
handleTransferReadColorBuffer(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1360 int handleTransferReadColorBuffer(PipeResEntry* res, uint64_t offset,
1361 stream_renderer_box* box) {
1362 if (res->type != ResType::COLOR_BUFFER) {
1363 stream_renderer_error("resid: %d not a COLOR_BUFFER resource", res->args.handle);
1364 return -EINVAL;
1365 }
1366
1367 auto glformat = virgl_format_to_gl(res->args.format);
1368 auto gltype = gl_format_to_natural_type(glformat);
1369
1370 // We always xfer the whole thing again from GL
1371 // since it's fiddly to calc / copy-out subregions
1372 if (virgl_format_is_yuv(res->args.format)) {
1373 mVirtioGpuOps->read_color_buffer_yuv(res->args.handle, 0, 0, res->args.width,
1374 res->args.height, res->linear, res->linearSize);
1375 } else {
1376 mVirtioGpuOps->read_color_buffer(res->args.handle, 0, 0, res->args.width,
1377 res->args.height, glformat, gltype, res->linear);
1378 }
1379
1380 return 0;
1381 }
1382
handleTransferWriteColorBuffer(PipeResEntry * res,uint64_t offset,stream_renderer_box * box)1383 int handleTransferWriteColorBuffer(PipeResEntry* res, uint64_t offset,
1384 stream_renderer_box* box) {
1385 if (res->type != ResType::COLOR_BUFFER) {
1386 stream_renderer_error("resid: %d not a COLOR_BUFFER resource", res->args.handle);
1387 return -EINVAL;
1388 }
1389
1390 auto glformat = virgl_format_to_gl(res->args.format);
1391 auto gltype = gl_format_to_natural_type(glformat);
1392
1393 // We always xfer the whole thing again to GL
1394 // since it's fiddly to calc / copy-out subregions
1395 mVirtioGpuOps->update_color_buffer(res->args.handle, 0, 0, res->args.width,
1396 res->args.height, glformat, gltype, res->linear);
1397 return 0;
1398 }
1399
transferReadIov(int resId,uint64_t offset,stream_renderer_box * box,struct iovec * iov,int iovec_cnt)1400 int transferReadIov(int resId, uint64_t offset, stream_renderer_box* box, struct iovec* iov,
1401 int iovec_cnt) {
1402 auto it = mResources.find(resId);
1403 if (it == mResources.end()) return EINVAL;
1404
1405 int ret = 0;
1406
1407 auto& entry = it->second;
1408 switch (entry.type) {
1409 case ResType::PIPE:
1410 ret = handleTransferReadPipe(&entry, offset, box);
1411 break;
1412 case ResType::BUFFER:
1413 ret = handleTransferReadBuffer(&entry, offset, box);
1414 break;
1415 case ResType::COLOR_BUFFER:
1416 ret = handleTransferReadColorBuffer(&entry, offset, box);
1417 break;
1418 }
1419
1420 if (ret != 0) {
1421 return ret;
1422 }
1423
1424 if (iovec_cnt) {
1425 PipeResEntry e = {
1426 entry.args, iov, (uint32_t)iovec_cnt, entry.linear, entry.linearSize,
1427 };
1428 ret = sync_iov(&e, offset, box, LINEAR_TO_IOV);
1429 } else {
1430 ret = sync_iov(&entry, offset, box, LINEAR_TO_IOV);
1431 }
1432
1433 return ret;
1434 }
1435
transferWriteIov(int resId,uint64_t offset,stream_renderer_box * box,struct iovec * iov,int iovec_cnt)1436 int transferWriteIov(int resId, uint64_t offset, stream_renderer_box* box, struct iovec* iov,
1437 int iovec_cnt) {
1438 auto it = mResources.find(resId);
1439 if (it == mResources.end()) return EINVAL;
1440
1441 auto& entry = it->second;
1442
1443 int ret = 0;
1444 if (iovec_cnt) {
1445 PipeResEntry e = {
1446 entry.args, iov, (uint32_t)iovec_cnt, entry.linear, entry.linearSize,
1447 };
1448 ret = sync_iov(&e, offset, box, IOV_TO_LINEAR);
1449 } else {
1450 ret = sync_iov(&entry, offset, box, IOV_TO_LINEAR);
1451 }
1452
1453 if (ret != 0) {
1454 return ret;
1455 }
1456
1457 switch (entry.type) {
1458 case ResType::PIPE:
1459 ret = handleTransferWritePipe(&entry, offset, box);
1460 break;
1461 case ResType::BUFFER:
1462 ret = handleTransferWriteBuffer(&entry, offset, box);
1463 break;
1464 case ResType::COLOR_BUFFER:
1465 ret = handleTransferWriteColorBuffer(&entry, offset, box);
1466 break;
1467 }
1468
1469 return ret;
1470 }
1471
getCapset(uint32_t set,uint32_t * max_size)1472 void getCapset(uint32_t set, uint32_t* max_size) {
1473 switch (set) {
1474 case VIRTGPU_CAPSET_GFXSTREAM_VULKAN:
1475 *max_size = sizeof(struct gfxstream::vulkanCapset);
1476 break;
1477 case VIRTGPU_CAPSET_GFXSTREAM_MAGMA:
1478 *max_size = sizeof(struct gfxstream::magmaCapset);
1479 break;
1480 case VIRTGPU_CAPSET_GFXSTREAM_GLES:
1481 *max_size = sizeof(struct gfxstream::glesCapset);
1482 break;
1483 case VIRTGPU_CAPSET_GFXSTREAM_COMPOSER:
1484 *max_size = sizeof(struct gfxstream::composerCapset);
1485 break;
1486 default:
1487 stream_renderer_error("Incorrect capability set specified (%u)", set);
1488 }
1489 }
1490
fillCaps(uint32_t set,void * caps)1491 void fillCaps(uint32_t set, void* caps) {
1492 switch (set) {
1493 case VIRTGPU_CAPSET_GFXSTREAM_VULKAN: {
1494 struct gfxstream::vulkanCapset* capset =
1495 reinterpret_cast<struct gfxstream::vulkanCapset*>(caps);
1496
1497 memset(capset, 0, sizeof(*capset));
1498
1499 capset->protocolVersion = 1;
1500 capset->ringSize = 12288;
1501 capset->bufferSize = 1048576;
1502
1503 auto vk_emu = gfxstream::vk::getGlobalVkEmulation();
1504 if (vk_emu && vk_emu->live && vk_emu->representativeColorBufferMemoryTypeInfo) {
1505 capset->colorBufferMemoryIndex =
1506 vk_emu->representativeColorBufferMemoryTypeInfo->guestMemoryTypeIndex;
1507 }
1508
1509 capset->noRenderControlEnc = 1;
1510 capset->blobAlignment = mPageSize;
1511 if (vk_emu && vk_emu->live) {
1512 capset->deferredMapping = 1;
1513 }
1514 break;
1515 }
1516 case VIRTGPU_CAPSET_GFXSTREAM_MAGMA: {
1517 struct gfxstream::magmaCapset* capset =
1518 reinterpret_cast<struct gfxstream::magmaCapset*>(caps);
1519
1520 capset->protocolVersion = 1;
1521 capset->ringSize = 12288;
1522 capset->bufferSize = 1048576;
1523 capset->blobAlignment = mPageSize;
1524 break;
1525 }
1526 case VIRTGPU_CAPSET_GFXSTREAM_GLES: {
1527 struct gfxstream::glesCapset* capset =
1528 reinterpret_cast<struct gfxstream::glesCapset*>(caps);
1529
1530 capset->protocolVersion = 1;
1531 capset->ringSize = 12288;
1532 capset->bufferSize = 1048576;
1533 capset->blobAlignment = mPageSize;
1534 break;
1535 }
1536 case VIRTGPU_CAPSET_GFXSTREAM_COMPOSER: {
1537 struct gfxstream::composerCapset* capset =
1538 reinterpret_cast<struct gfxstream::composerCapset*>(caps);
1539
1540 capset->protocolVersion = 1;
1541 capset->ringSize = 12288;
1542 capset->bufferSize = 1048576;
1543 capset->blobAlignment = mPageSize;
1544 break;
1545 }
1546 default:
1547 stream_renderer_error("Incorrect capability set specified");
1548 }
1549 }
1550
attachResource(uint32_t ctxId,uint32_t resId)1551 void attachResource(uint32_t ctxId, uint32_t resId) {
1552 stream_renderer_debug("ctxid: %u resid: %u", ctxId, resId);
1553
1554 auto resourcesIt = mContextResources.find(ctxId);
1555
1556 if (resourcesIt == mContextResources.end()) {
1557 std::vector<VirtioGpuResId> ids;
1558 ids.push_back(resId);
1559 mContextResources[ctxId] = ids;
1560 } else {
1561 auto& ids = resourcesIt->second;
1562 auto idIt = std::find(ids.begin(), ids.end(), resId);
1563 if (idIt == ids.end()) ids.push_back(resId);
1564 }
1565
1566 auto contextsIt = mResourceContexts.find(resId);
1567
1568 if (contextsIt == mResourceContexts.end()) {
1569 std::vector<VirtioGpuCtxId> ids;
1570 ids.push_back(ctxId);
1571 mResourceContexts[resId] = ids;
1572 } else {
1573 auto& ids = contextsIt->second;
1574 auto idIt = std::find(ids.begin(), ids.end(), ctxId);
1575 if (idIt == ids.end()) ids.push_back(ctxId);
1576 }
1577
1578 // Associate the host pipe of the resource entry with the host pipe of
1579 // the context entry. That is, the last context to call attachResource
1580 // wins if there is any conflict.
1581 auto ctxEntryIt = mContexts.find(ctxId);
1582 auto resEntryIt = mResources.find(resId);
1583
1584 if (ctxEntryIt == mContexts.end() || resEntryIt == mResources.end()) return;
1585
1586 stream_renderer_debug("hostPipe: %p", ctxEntryIt->second.hostPipe);
1587 resEntryIt->second.hostPipe = ctxEntryIt->second.hostPipe;
1588 resEntryIt->second.ctxId = ctxId;
1589 }
1590
detachResource(uint32_t ctxId,uint32_t toUnrefId)1591 void detachResource(uint32_t ctxId, uint32_t toUnrefId) {
1592 stream_renderer_debug("ctxid: %u resid: %u", ctxId, toUnrefId);
1593 detachResourceLocked(ctxId, toUnrefId);
1594 }
1595
getResourceInfo(uint32_t resId,struct stream_renderer_resource_info * info)1596 int getResourceInfo(uint32_t resId, struct stream_renderer_resource_info* info) {
1597 stream_renderer_debug("resid: %u", resId);
1598 if (!info) return EINVAL;
1599
1600 auto it = mResources.find(resId);
1601 if (it == mResources.end()) return ENOENT;
1602
1603 auto& entry = it->second;
1604
1605 uint32_t bpp = 4U;
1606 switch (entry.args.format) {
1607 case VIRGL_FORMAT_B8G8R8A8_UNORM:
1608 info->drm_fourcc = DRM_FORMAT_ARGB8888;
1609 break;
1610 case VIRGL_FORMAT_B5G6R5_UNORM:
1611 info->drm_fourcc = DRM_FORMAT_RGB565;
1612 bpp = 2U;
1613 break;
1614 case VIRGL_FORMAT_R8G8B8A8_UNORM:
1615 info->drm_fourcc = DRM_FORMAT_ABGR8888;
1616 break;
1617 case VIRGL_FORMAT_R8G8B8X8_UNORM:
1618 info->drm_fourcc = DRM_FORMAT_XBGR8888;
1619 break;
1620 case VIRGL_FORMAT_R8_UNORM:
1621 info->drm_fourcc = DRM_FORMAT_R8;
1622 bpp = 1U;
1623 break;
1624 default:
1625 return EINVAL;
1626 }
1627
1628 info->stride = align_up(entry.args.width * bpp, 16U);
1629 info->virgl_format = entry.args.format;
1630 info->handle = entry.args.handle;
1631 info->height = entry.args.height;
1632 info->width = entry.args.width;
1633 info->depth = entry.args.depth;
1634 info->flags = entry.args.flags;
1635 info->tex_id = 0;
1636 return 0;
1637 }
1638
flushResource(uint32_t res_handle)1639 void flushResource(uint32_t res_handle) {
1640 auto taskId = mVirtioGpuTimelines->enqueueTask(VirtioGpuRingGlobal{});
1641 mVirtioGpuOps->async_post_color_buffer(
1642 res_handle, [this, taskId](std::shared_future<void> waitForGpu) {
1643 waitForGpu.wait();
1644 mVirtioGpuTimelines->notifyTaskCompletion(taskId);
1645 });
1646 }
1647
createRingBlob(PipeResEntry & entry,uint32_t res_handle,const struct stream_renderer_create_blob * create_blob,const struct stream_renderer_handle * handle)1648 int createRingBlob(PipeResEntry& entry, uint32_t res_handle,
1649 const struct stream_renderer_create_blob* create_blob,
1650 const struct stream_renderer_handle* handle) {
1651 if (mFeatures.ExternalBlob.enabled) {
1652 std::string name = "shared-memory-" + std::to_string(res_handle);
1653 auto shmem = std::make_unique<SharedMemory>(name, create_blob->size);
1654 int ret = shmem->create(0600);
1655 if (ret) {
1656 stream_renderer_error("Failed to create shared memory blob");
1657 return ret;
1658 }
1659
1660 entry.hva = shmem->get();
1661 entry.ringBlob = std::make_shared<RingBlob>(std::move(shmem));
1662
1663 } else {
1664 auto mem = std::make_unique<AlignedMemory>(mPageSize, create_blob->size);
1665 if (mem->addr == nullptr) {
1666 stream_renderer_error("Failed to allocate ring blob");
1667 return -ENOMEM;
1668 }
1669
1670 entry.hva = mem->addr;
1671 entry.ringBlob = std::make_shared<RingBlob>(std::move(mem));
1672 }
1673
1674 entry.hvaSize = create_blob->size;
1675 entry.externalAddr = true;
1676 entry.caching = STREAM_RENDERER_MAP_CACHE_CACHED;
1677
1678 return 0;
1679 }
1680
createBlob(uint32_t ctx_id,uint32_t res_handle,const struct stream_renderer_create_blob * create_blob,const struct stream_renderer_handle * handle)1681 int createBlob(uint32_t ctx_id, uint32_t res_handle,
1682 const struct stream_renderer_create_blob* create_blob,
1683 const struct stream_renderer_handle* handle) {
1684 stream_renderer_debug("ctx:%u res:%u blob-id:%u blob-size:%u", ctx_id, res_handle,
1685 create_blob->blob_id, create_blob->size);
1686
1687 PipeResEntry e;
1688 struct stream_renderer_resource_create_args args = {0};
1689 e.args = args;
1690 e.hostPipe = 0;
1691
1692 if (create_blob->blob_id == 0) {
1693 int ret = createRingBlob(e, res_handle, create_blob, handle);
1694 if (ret) {
1695 return ret;
1696 }
1697 } else if (mFeatures.ExternalBlob.enabled) {
1698 if (create_blob->blob_mem == STREAM_BLOB_MEM_GUEST &&
1699 (create_blob->blob_flags & STREAM_BLOB_FLAG_CREATE_GUEST_HANDLE)) {
1700 #if defined(__linux__) || defined(__QNX__)
1701 ManagedDescriptor managedHandle(handle->os_handle);
1702 BlobManager::get()->addDescriptorInfo(ctx_id, create_blob->blob_id,
1703 std::move(managedHandle), handle->handle_type,
1704 0, std::nullopt);
1705
1706 e.caching = STREAM_RENDERER_MAP_CACHE_CACHED;
1707 #else
1708 return -EINVAL;
1709 #endif
1710 } else {
1711 auto descriptorInfoOpt =
1712 BlobManager::get()->removeDescriptorInfo(ctx_id, create_blob->blob_id);
1713 if (descriptorInfoOpt) {
1714 e.descriptorInfo =
1715 std::make_shared<ManagedDescriptorInfo>(std::move(*descriptorInfoOpt));
1716 } else {
1717 return -EINVAL;
1718 }
1719
1720 e.caching = e.descriptorInfo->caching;
1721 }
1722 } else {
1723 auto entryOpt = BlobManager::get()->removeMapping(ctx_id, create_blob->blob_id);
1724 if (entryOpt) {
1725 e.hva = entryOpt->addr;
1726 e.caching = entryOpt->caching;
1727 e.hvaSize = create_blob->size;
1728 } else {
1729 return -EINVAL;
1730 }
1731 }
1732
1733 e.blobId = create_blob->blob_id;
1734 e.blobMem = create_blob->blob_mem;
1735 e.blobFlags = create_blob->blob_flags;
1736 e.iov = nullptr;
1737 e.numIovs = 0;
1738 e.linear = 0;
1739 e.linearSize = 0;
1740
1741 mResources[res_handle] = e;
1742 return 0;
1743 }
1744
resourceMap(uint32_t res_handle,void ** hvaOut,uint64_t * sizeOut)1745 int resourceMap(uint32_t res_handle, void** hvaOut, uint64_t* sizeOut) {
1746 if (mFeatures.ExternalBlob.enabled) return -EINVAL;
1747
1748 auto it = mResources.find(res_handle);
1749 if (it == mResources.end()) {
1750 if (hvaOut) *hvaOut = nullptr;
1751 if (sizeOut) *sizeOut = 0;
1752 return -EINVAL;
1753 }
1754
1755 const auto& entry = it->second;
1756
1757 if (hvaOut) *hvaOut = entry.hva;
1758 if (sizeOut) *sizeOut = entry.hvaSize;
1759 return 0;
1760 }
1761
resourceUnmap(uint32_t res_handle)1762 int resourceUnmap(uint32_t res_handle) {
1763 auto it = mResources.find(res_handle);
1764 if (it == mResources.end()) {
1765 return -EINVAL;
1766 }
1767
1768 // TODO(lfy): Good place to run any registered cleanup callbacks.
1769 // No-op for now.
1770 return 0;
1771 }
1772
platformImportResource(int res_handle,int res_info,void * resource)1773 int platformImportResource(int res_handle, int res_info, void* resource) {
1774 auto it = mResources.find(res_handle);
1775 if (it == mResources.end()) return -EINVAL;
1776 bool success = mVirtioGpuOps->platform_import_resource(res_handle, res_info, resource);
1777 return success ? 0 : -1;
1778 }
1779
platformResourceInfo(int res_handle,int * width,int * height,int * internal_format)1780 int platformResourceInfo(int res_handle, int* width, int* height, int* internal_format) {
1781 auto it = mResources.find(res_handle);
1782 if (it == mResources.end()) return -EINVAL;
1783 bool success =
1784 mVirtioGpuOps->platform_resource_info(res_handle, width, height, internal_format);
1785 return success ? 0 : -1;
1786 }
1787
platformCreateSharedEglContext()1788 void* platformCreateSharedEglContext() {
1789 return mVirtioGpuOps->platform_create_shared_egl_context();
1790 }
1791
platformDestroySharedEglContext(void * context)1792 int platformDestroySharedEglContext(void* context) {
1793 bool success = mVirtioGpuOps->platform_destroy_shared_egl_context(context);
1794 return success ? 0 : -1;
1795 }
1796
resourceMapInfo(uint32_t res_handle,uint32_t * map_info)1797 int resourceMapInfo(uint32_t res_handle, uint32_t* map_info) {
1798 auto it = mResources.find(res_handle);
1799 if (it == mResources.end()) return -EINVAL;
1800
1801 const auto& entry = it->second;
1802 *map_info = entry.caching;
1803 return 0;
1804 }
1805
exportBlob(uint32_t res_handle,struct stream_renderer_handle * handle)1806 int exportBlob(uint32_t res_handle, struct stream_renderer_handle* handle) {
1807 auto it = mResources.find(res_handle);
1808 if (it == mResources.end()) {
1809 return -EINVAL;
1810 }
1811
1812 auto& entry = it->second;
1813 if (entry.ringBlob && entry.ringBlob->isExportable()) {
1814 // Handle ownership transferred to VMM, gfxstream keeps the mapping.
1815 #ifdef _WIN32
1816 handle->os_handle =
1817 static_cast<int64_t>(reinterpret_cast<intptr_t>(entry.ringBlob->releaseHandle()));
1818 #else
1819 handle->os_handle = static_cast<int64_t>(entry.ringBlob->releaseHandle());
1820 #endif
1821 handle->handle_type = STREAM_MEM_HANDLE_TYPE_SHM;
1822 return 0;
1823 }
1824
1825 if (entry.descriptorInfo) {
1826 bool shareable = entry.blobFlags &
1827 (STREAM_BLOB_FLAG_USE_SHAREABLE | STREAM_BLOB_FLAG_USE_CROSS_DEVICE);
1828
1829 DescriptorType rawDescriptor;
1830 if (shareable) {
1831 // TODO: Add ManagedDescriptor::{clone, dup} method and use it;
1832 // This should have no affect since gfxstream allocates mappable-only buffers
1833 // currently
1834 return -EINVAL;
1835 } else {
1836 auto rawDescriptorOpt = entry.descriptorInfo->descriptor.release();
1837 if (rawDescriptorOpt)
1838 rawDescriptor = *rawDescriptorOpt;
1839 else
1840 return -EINVAL;
1841 }
1842
1843 handle->handle_type = entry.descriptorInfo->handleType;
1844
1845 #ifdef _WIN32
1846 handle->os_handle = static_cast<int64_t>(reinterpret_cast<intptr_t>(rawDescriptor));
1847 #else
1848 handle->os_handle = static_cast<int64_t>(rawDescriptor);
1849 #endif
1850
1851 return 0;
1852 }
1853
1854 return -EINVAL;
1855 }
1856
vulkanInfo(uint32_t res_handle,struct stream_renderer_vulkan_info * vulkan_info)1857 int vulkanInfo(uint32_t res_handle, struct stream_renderer_vulkan_info* vulkan_info) {
1858 auto it = mResources.find(res_handle);
1859 if (it == mResources.end()) return -EINVAL;
1860
1861 const auto& entry = it->second;
1862 if (entry.descriptorInfo && entry.descriptorInfo->vulkanInfoOpt) {
1863 vulkan_info->memory_index = (*entry.descriptorInfo->vulkanInfoOpt).memoryIndex;
1864 memcpy(vulkan_info->device_id.device_uuid,
1865 (*entry.descriptorInfo->vulkanInfoOpt).deviceUUID,
1866 sizeof(vulkan_info->device_id.device_uuid));
1867 memcpy(vulkan_info->device_id.driver_uuid,
1868 (*entry.descriptorInfo->vulkanInfoOpt).driverUUID,
1869 sizeof(vulkan_info->device_id.driver_uuid));
1870 return 0;
1871 }
1872
1873 return -EINVAL;
1874 }
1875
1876 #ifdef CONFIG_AEMU
setServiceOps(const GoldfishPipeServiceOps * ops)1877 void setServiceOps(const GoldfishPipeServiceOps* ops) { mServiceOps = ops; }
1878 #endif // CONFIG_AEMU
1879 private:
allocResource(PipeResEntry & entry,iovec * iov,int num_iovs)1880 void allocResource(PipeResEntry& entry, iovec* iov, int num_iovs) {
1881 stream_renderer_debug("entry linear: %p", entry.linear);
1882 if (entry.linear) free(entry.linear);
1883
1884 size_t linearSize = 0;
1885 for (uint32_t i = 0; i < num_iovs; ++i) {
1886 stream_renderer_debug("iov base: %p", iov[i].iov_base);
1887 linearSize += iov[i].iov_len;
1888 stream_renderer_debug("has iov of %zu. linearSize current: %zu", iov[i].iov_len,
1889 linearSize);
1890 }
1891 stream_renderer_debug("final linearSize: %zu", linearSize);
1892
1893 void* linear = nullptr;
1894
1895 if (linearSize) linear = malloc(linearSize);
1896
1897 entry.numIovs = num_iovs;
1898 entry.iov = (iovec*)malloc(sizeof(*iov) * num_iovs);
1899 if (entry.numIovs > 0) {
1900 memcpy(entry.iov, iov, num_iovs * sizeof(*iov));
1901 }
1902 entry.linear = linear;
1903 entry.linearSize = linearSize;
1904 }
1905
detachResourceLocked(uint32_t ctxId,uint32_t toUnrefId)1906 void detachResourceLocked(uint32_t ctxId, uint32_t toUnrefId) {
1907 stream_renderer_debug("ctxid: %u resid: %u", ctxId, toUnrefId);
1908
1909 auto it = mContextResources.find(ctxId);
1910 if (it == mContextResources.end()) return;
1911
1912 std::vector<VirtioGpuResId> withoutRes;
1913 for (auto resId : it->second) {
1914 if (resId != toUnrefId) {
1915 withoutRes.push_back(resId);
1916 }
1917 }
1918 mContextResources[ctxId] = withoutRes;
1919
1920 auto resourceIt = mResources.find(toUnrefId);
1921 if (resourceIt == mResources.end()) return;
1922 auto& resource = resourceIt->second;
1923
1924 resource.hostPipe = 0;
1925 resource.ctxId = 0;
1926
1927 auto ctxIt = mContexts.find(ctxId);
1928 if (ctxIt != mContexts.end()) {
1929 auto& ctxEntry = ctxIt->second;
1930 if (ctxEntry.addressSpaceHandles.count(toUnrefId)) {
1931 uint32_t asgHandle = ctxEntry.addressSpaceHandles[toUnrefId];
1932
1933 mCleanupThread->enqueueCleanup([this, asgBlob = resource.ringBlob, asgHandle](){
1934 mAddressSpaceDeviceControlOps->destroy_handle(asgHandle);
1935 });
1936
1937 ctxEntry.addressSpaceHandles.erase(toUnrefId);
1938 }
1939 }
1940 }
1941
ensureAndGetServiceOps()1942 inline const GoldfishPipeServiceOps* ensureAndGetServiceOps() {
1943 if (mServiceOps) return mServiceOps;
1944 mServiceOps = goldfish_pipe_get_service_ops();
1945 return mServiceOps;
1946 }
1947
1948 void* mCookie = nullptr;
1949 gfxstream::host::FeatureSet mFeatures;
1950 stream_renderer_fence_callback mFenceCallback;
1951 AndroidVirtioGpuOps* mVirtioGpuOps = nullptr;
1952 uint32_t mPageSize = 4096;
1953 struct address_space_device_control_ops* mAddressSpaceDeviceControlOps = nullptr;
1954
1955 const GoldfishPipeServiceOps* mServiceOps = nullptr;
1956
1957 std::unordered_map<VirtioGpuCtxId, PipeCtxEntry> mContexts;
1958 std::unordered_map<VirtioGpuResId, PipeResEntry> mResources;
1959 std::unordered_map<VirtioGpuCtxId, std::vector<VirtioGpuResId>> mContextResources;
1960 std::unordered_map<VirtioGpuResId, std::vector<VirtioGpuCtxId>> mResourceContexts;
1961
1962 // When we wait for gpu or wait for gpu vulkan, the next (and subsequent)
1963 // fences created for that context should not be signaled immediately.
1964 // Rather, they should get in line.
1965 std::unique_ptr<VirtioGpuTimelines> mVirtioGpuTimelines = nullptr;
1966
1967 std::unique_ptr<CleanupThread> mCleanupThread;
1968 };
1969
sRenderer()1970 static PipeVirglRenderer* sRenderer() {
1971 static PipeVirglRenderer* p = new PipeVirglRenderer;
1972 return p;
1973 }
1974
1975 extern "C" {
1976
stream_renderer_resource_create(struct stream_renderer_resource_create_args * args,struct iovec * iov,uint32_t num_iovs)1977 VG_EXPORT int stream_renderer_resource_create(struct stream_renderer_resource_create_args* args,
1978 struct iovec* iov, uint32_t num_iovs) {
1979 return sRenderer()->createResource(args, iov, num_iovs);
1980 }
1981
stream_renderer_resource_unref(uint32_t res_handle)1982 VG_EXPORT void stream_renderer_resource_unref(uint32_t res_handle) {
1983 sRenderer()->unrefResource(res_handle);
1984 }
1985
stream_renderer_context_destroy(uint32_t handle)1986 VG_EXPORT void stream_renderer_context_destroy(uint32_t handle) {
1987 sRenderer()->destroyContext(handle);
1988 }
1989
stream_renderer_submit_cmd(struct stream_renderer_command * cmd)1990 VG_EXPORT int stream_renderer_submit_cmd(struct stream_renderer_command* cmd) {
1991 return sRenderer()->submitCmd(cmd);
1992 }
1993
stream_renderer_transfer_read_iov(uint32_t handle,uint32_t ctx_id,uint32_t level,uint32_t stride,uint32_t layer_stride,struct stream_renderer_box * box,uint64_t offset,struct iovec * iov,int iovec_cnt)1994 VG_EXPORT int stream_renderer_transfer_read_iov(uint32_t handle, uint32_t ctx_id, uint32_t level,
1995 uint32_t stride, uint32_t layer_stride,
1996 struct stream_renderer_box* box, uint64_t offset,
1997 struct iovec* iov, int iovec_cnt) {
1998 return sRenderer()->transferReadIov(handle, offset, box, iov, iovec_cnt);
1999 }
2000
stream_renderer_transfer_write_iov(uint32_t handle,uint32_t ctx_id,int level,uint32_t stride,uint32_t layer_stride,struct stream_renderer_box * box,uint64_t offset,struct iovec * iovec,unsigned int iovec_cnt)2001 VG_EXPORT int stream_renderer_transfer_write_iov(uint32_t handle, uint32_t ctx_id, int level,
2002 uint32_t stride, uint32_t layer_stride,
2003 struct stream_renderer_box* box, uint64_t offset,
2004 struct iovec* iovec, unsigned int iovec_cnt) {
2005 return sRenderer()->transferWriteIov(handle, offset, box, iovec, iovec_cnt);
2006 }
2007
stream_renderer_get_cap_set(uint32_t set,uint32_t * max_ver,uint32_t * max_size)2008 VG_EXPORT void stream_renderer_get_cap_set(uint32_t set, uint32_t* max_ver, uint32_t* max_size) {
2009 // `max_ver` not useful
2010 return sRenderer()->getCapset(set, max_size);
2011 }
2012
stream_renderer_fill_caps(uint32_t set,uint32_t version,void * caps)2013 VG_EXPORT void stream_renderer_fill_caps(uint32_t set, uint32_t version, void* caps) {
2014 // `version` not useful
2015 return sRenderer()->fillCaps(set, caps);
2016 }
2017
stream_renderer_resource_attach_iov(int res_handle,struct iovec * iov,int num_iovs)2018 VG_EXPORT int stream_renderer_resource_attach_iov(int res_handle, struct iovec* iov, int num_iovs) {
2019 return sRenderer()->attachIov(res_handle, iov, num_iovs);
2020 }
2021
stream_renderer_resource_detach_iov(int res_handle,struct iovec ** iov,int * num_iovs)2022 VG_EXPORT void stream_renderer_resource_detach_iov(int res_handle, struct iovec** iov,
2023 int* num_iovs) {
2024 return sRenderer()->detachIov(res_handle, iov, num_iovs);
2025 }
2026
stream_renderer_ctx_attach_resource(int ctx_id,int res_handle)2027 VG_EXPORT void stream_renderer_ctx_attach_resource(int ctx_id, int res_handle) {
2028 sRenderer()->attachResource(ctx_id, res_handle);
2029 }
2030
stream_renderer_ctx_detach_resource(int ctx_id,int res_handle)2031 VG_EXPORT void stream_renderer_ctx_detach_resource(int ctx_id, int res_handle) {
2032 sRenderer()->detachResource(ctx_id, res_handle);
2033 }
2034
stream_renderer_resource_get_info(int res_handle,struct stream_renderer_resource_info * info)2035 VG_EXPORT int stream_renderer_resource_get_info(int res_handle,
2036 struct stream_renderer_resource_info* info) {
2037 return sRenderer()->getResourceInfo(res_handle, info);
2038 }
2039
stream_renderer_flush(uint32_t res_handle)2040 VG_EXPORT void stream_renderer_flush(uint32_t res_handle) {
2041 sRenderer()->flushResource(res_handle);
2042 }
2043
stream_renderer_create_blob(uint32_t ctx_id,uint32_t res_handle,const struct stream_renderer_create_blob * create_blob,const struct iovec * iovecs,uint32_t num_iovs,const struct stream_renderer_handle * handle)2044 VG_EXPORT int stream_renderer_create_blob(uint32_t ctx_id, uint32_t res_handle,
2045 const struct stream_renderer_create_blob* create_blob,
2046 const struct iovec* iovecs, uint32_t num_iovs,
2047 const struct stream_renderer_handle* handle) {
2048 sRenderer()->createBlob(ctx_id, res_handle, create_blob, handle);
2049 return 0;
2050 }
2051
stream_renderer_export_blob(uint32_t res_handle,struct stream_renderer_handle * handle)2052 VG_EXPORT int stream_renderer_export_blob(uint32_t res_handle,
2053 struct stream_renderer_handle* handle) {
2054 return sRenderer()->exportBlob(res_handle, handle);
2055 }
2056
stream_renderer_resource_map(uint32_t res_handle,void ** hvaOut,uint64_t * sizeOut)2057 VG_EXPORT int stream_renderer_resource_map(uint32_t res_handle, void** hvaOut, uint64_t* sizeOut) {
2058 return sRenderer()->resourceMap(res_handle, hvaOut, sizeOut);
2059 }
2060
stream_renderer_resource_unmap(uint32_t res_handle)2061 VG_EXPORT int stream_renderer_resource_unmap(uint32_t res_handle) {
2062 return sRenderer()->resourceUnmap(res_handle);
2063 }
2064
stream_renderer_context_create(uint32_t ctx_id,uint32_t nlen,const char * name,uint32_t context_init)2065 VG_EXPORT int stream_renderer_context_create(uint32_t ctx_id, uint32_t nlen, const char* name,
2066 uint32_t context_init) {
2067 return sRenderer()->createContext(ctx_id, nlen, name, context_init);
2068 }
2069
stream_renderer_create_fence(const struct stream_renderer_fence * fence)2070 VG_EXPORT int stream_renderer_create_fence(const struct stream_renderer_fence* fence) {
2071 if (fence->flags & STREAM_RENDERER_FLAG_FENCE_RING_IDX) {
2072 sRenderer()->createFence(fence->fence_id, VirtioGpuRingContextSpecific{
2073 .mCtxId = fence->ctx_id,
2074 .mRingIdx = fence->ring_idx,
2075 });
2076 } else {
2077 sRenderer()->createFence(fence->fence_id, VirtioGpuRingGlobal{});
2078 }
2079
2080 return 0;
2081 }
2082
stream_renderer_platform_import_resource(int res_handle,int res_info,void * resource)2083 VG_EXPORT int stream_renderer_platform_import_resource(int res_handle, int res_info,
2084 void* resource) {
2085 return sRenderer()->platformImportResource(res_handle, res_info, resource);
2086 }
2087
stream_renderer_platform_resource_info(int res_handle,int * width,int * height,int * internal_format)2088 VG_EXPORT int stream_renderer_platform_resource_info(int res_handle, int* width, int* height,
2089 int* internal_format) {
2090 return sRenderer()->platformResourceInfo(res_handle, width, height, internal_format);
2091 }
2092
stream_renderer_platform_create_shared_egl_context()2093 VG_EXPORT void* stream_renderer_platform_create_shared_egl_context() {
2094 return sRenderer()->platformCreateSharedEglContext();
2095 }
2096
stream_renderer_platform_destroy_shared_egl_context(void * context)2097 VG_EXPORT int stream_renderer_platform_destroy_shared_egl_context(void* context) {
2098 return sRenderer()->platformDestroySharedEglContext(context);
2099 }
2100
stream_renderer_resource_map_info(uint32_t res_handle,uint32_t * map_info)2101 VG_EXPORT int stream_renderer_resource_map_info(uint32_t res_handle, uint32_t* map_info) {
2102 return sRenderer()->resourceMapInfo(res_handle, map_info);
2103 }
2104
stream_renderer_vulkan_info(uint32_t res_handle,struct stream_renderer_vulkan_info * vulkan_info)2105 VG_EXPORT int stream_renderer_vulkan_info(uint32_t res_handle,
2106 struct stream_renderer_vulkan_info* vulkan_info) {
2107 return sRenderer()->vulkanInfo(res_handle, vulkan_info);
2108 }
2109
stream_renderer_snapshot(const char * dir)2110 VG_EXPORT int stream_renderer_snapshot(const char* dir) {
2111 #ifdef GFXSTREAM_ENABLE_HOST_VK_SNAPSHOT
2112 std::string dirString(dir);
2113
2114 std::string snapshotFileName = dirString + "snapshot.bin";
2115
2116 std::unique_ptr<android::base::StdioStream> stream(new android::base::StdioStream(
2117 fopen(snapshotFileName.c_str(), "wb"), android::base::StdioStream::kOwner));
2118
2119 android_getOpenglesRenderer()->pauseAllPreSave();
2120 android::snapshot::SnapshotSaveStream saveStream{
2121 .stream = stream.get(),
2122 };
2123
2124 android_getOpenglesRenderer()->save(saveStream.stream, saveStream.textureSaver);
2125 return 0;
2126 #else
2127 stream_renderer_error("Snapshot save requested without support.");
2128 return -EINVAL;
2129 #endif
2130 }
2131
stream_renderer_restore(const char * dir)2132 VG_EXPORT int stream_renderer_restore(const char* dir) {
2133 #ifdef GFXSTREAM_ENABLE_HOST_VK_SNAPSHOT
2134 std::string dirString(dir);
2135 std::string snapshotFileName = dirString + "snapshot.bin";
2136
2137 std::unique_ptr<android::base::StdioStream> stream(new android::base::StdioStream(
2138 fopen(snapshotFileName.c_str(), "rb"), android::base::StdioStream::kOwner));
2139
2140 android::snapshot::SnapshotLoadStream loadStream{
2141 .stream = stream.get(),
2142 };
2143
2144 android_getOpenglesRenderer()->load(loadStream.stream, loadStream.textureLoader);
2145
2146 // In end2end tests, we don't really do snapshot save for render threads.
2147 // We will need to resume all render threads without waiting for snapshot.
2148 android_getOpenglesRenderer()->resumeAll(false);
2149 return 0;
2150 #else
2151 stream_renderer_error("Snapshot save requested without support.");
2152 return -EINVAL;
2153 #endif
2154 }
2155
2156 static const GoldfishPipeServiceOps goldfish_pipe_service_ops = {
2157 // guest_open()
__anonf6f38b3e0b02() 2158 [](GoldfishHwPipe* hwPipe) -> GoldfishHostPipe* {
2159 return static_cast<GoldfishHostPipe*>(android_pipe_guest_open(hwPipe));
2160 },
2161 // guest_open_with_flags()
__anonf6f38b3e0c02() 2162 [](GoldfishHwPipe* hwPipe, uint32_t flags) -> GoldfishHostPipe* {
2163 return static_cast<GoldfishHostPipe*>(android_pipe_guest_open_with_flags(hwPipe, flags));
2164 },
2165 // guest_close()
__anonf6f38b3e0d02() 2166 [](GoldfishHostPipe* hostPipe, GoldfishPipeCloseReason reason) {
2167 static_assert((int)GOLDFISH_PIPE_CLOSE_GRACEFUL == (int)PIPE_CLOSE_GRACEFUL,
2168 "Invalid PIPE_CLOSE_GRACEFUL value");
2169 static_assert((int)GOLDFISH_PIPE_CLOSE_REBOOT == (int)PIPE_CLOSE_REBOOT,
2170 "Invalid PIPE_CLOSE_REBOOT value");
2171 static_assert((int)GOLDFISH_PIPE_CLOSE_LOAD_SNAPSHOT == (int)PIPE_CLOSE_LOAD_SNAPSHOT,
2172 "Invalid PIPE_CLOSE_LOAD_SNAPSHOT value");
2173 static_assert((int)GOLDFISH_PIPE_CLOSE_ERROR == (int)PIPE_CLOSE_ERROR,
2174 "Invalid PIPE_CLOSE_ERROR value");
2175
2176 android_pipe_guest_close(hostPipe, static_cast<PipeCloseReason>(reason));
2177 },
2178 // guest_pre_load()
__anonf6f38b3e0e02() 2179 [](QEMUFile* file) { (void)file; },
2180 // guest_post_load()
__anonf6f38b3e0f02() 2181 [](QEMUFile* file) { (void)file; },
2182 // guest_pre_save()
__anonf6f38b3e1002() 2183 [](QEMUFile* file) { (void)file; },
2184 // guest_post_save()
__anonf6f38b3e1102() 2185 [](QEMUFile* file) { (void)file; },
2186 // guest_load()
__anonf6f38b3e1202() 2187 [](QEMUFile* file, GoldfishHwPipe* hwPipe, char* force_close) -> GoldfishHostPipe* {
2188 (void)file;
2189 (void)hwPipe;
2190 (void)force_close;
2191 return nullptr;
2192 },
2193 // guest_save()
__anonf6f38b3e1302() 2194 [](GoldfishHostPipe* hostPipe, QEMUFile* file) {
2195 (void)hostPipe;
2196 (void)file;
2197 },
2198 // guest_poll()
__anonf6f38b3e1402() 2199 [](GoldfishHostPipe* hostPipe) {
2200 static_assert((int)GOLDFISH_PIPE_POLL_IN == (int)PIPE_POLL_IN, "invalid POLL_IN values");
2201 static_assert((int)GOLDFISH_PIPE_POLL_OUT == (int)PIPE_POLL_OUT, "invalid POLL_OUT values");
2202 static_assert((int)GOLDFISH_PIPE_POLL_HUP == (int)PIPE_POLL_HUP, "invalid POLL_HUP values");
2203
2204 return static_cast<GoldfishPipePollFlags>(android_pipe_guest_poll(hostPipe));
2205 },
2206 // guest_recv()
__anonf6f38b3e1502() 2207 [](GoldfishHostPipe* hostPipe, GoldfishPipeBuffer* buffers, int numBuffers) -> int {
2208 // NOTE: Assumes that AndroidPipeBuffer and GoldfishPipeBuffer
2209 // have exactly the same layout.
2210 static_assert(sizeof(AndroidPipeBuffer) == sizeof(GoldfishPipeBuffer),
2211 "Invalid PipeBuffer sizes");
2212 // We can't use a static_assert with offsetof() because in msvc, it uses
2213 // reinterpret_cast.
2214 // TODO: Add runtime assertion instead?
2215 // https://developercommunity.visualstudio.com/content/problem/22196/static-assert-cannot-compile-constexprs-method-tha.html
2216 #ifndef _MSC_VER
2217 static_assert(offsetof(AndroidPipeBuffer, data) == offsetof(GoldfishPipeBuffer, data),
2218 "Invalid PipeBuffer::data offsets");
2219 static_assert(offsetof(AndroidPipeBuffer, size) == offsetof(GoldfishPipeBuffer, size),
2220 "Invalid PipeBuffer::size offsets");
2221 #endif
2222 return android_pipe_guest_recv(hostPipe, reinterpret_cast<AndroidPipeBuffer*>(buffers),
2223 numBuffers);
2224 },
2225 // wait_guest_recv()
__anonf6f38b3e1602() 2226 [](GoldfishHostPipe* hostPipe) {
2227 android_pipe_wait_guest_recv(hostPipe);
2228 },
2229 // guest_send()
__anonf6f38b3e1702() 2230 [](GoldfishHostPipe** hostPipe, const GoldfishPipeBuffer* buffers, int numBuffers) -> int {
2231 return android_pipe_guest_send(reinterpret_cast<void**>(hostPipe),
2232 reinterpret_cast<const AndroidPipeBuffer*>(buffers),
2233 numBuffers);
2234 },
2235 // wait_guest_send()
__anonf6f38b3e1802() 2236 [](GoldfishHostPipe* hostPipe) {
2237 android_pipe_wait_guest_send(hostPipe);
2238 },
2239 // guest_wake_on()
__anonf6f38b3e1902() 2240 [](GoldfishHostPipe* hostPipe, GoldfishPipeWakeFlags wakeFlags) {
2241 android_pipe_guest_wake_on(hostPipe, static_cast<int>(wakeFlags));
2242 },
2243 // dma_add_buffer()
__anonf6f38b3e1a02() 2244 [](void* pipe, uint64_t paddr, uint64_t sz) {
2245 // not considered for virtio
2246 },
2247 // dma_remove_buffer()
__anonf6f38b3e1b02() 2248 [](uint64_t paddr) {
2249 // not considered for virtio
2250 },
2251 // dma_invalidate_host_mappings()
__anonf6f38b3e1c02() 2252 []() {
2253 // not considered for virtio
2254 },
2255 // dma_reset_host_mappings()
__anonf6f38b3e1d02() 2256 []() {
2257 // not considered for virtio
2258 },
2259 // dma_save_mappings()
__anonf6f38b3e1e02() 2260 [](QEMUFile* file) { (void)file; },
2261 // dma_load_mappings()
__anonf6f38b3e1f02() 2262 [](QEMUFile* file) { (void)file; },
2263 };
2264
stream_renderer_opengles_init(uint32_t display_width,uint32_t display_height,int renderer_flags,gfxstream::host::FeatureSet features)2265 static int stream_renderer_opengles_init(uint32_t display_width, uint32_t display_height,
2266 int renderer_flags, gfxstream::host::FeatureSet features) {
2267 stream_renderer_debug("start. display dimensions: width %u height %u, renderer flags: 0x%x",
2268 display_width, display_height, renderer_flags);
2269
2270 // Flags processing
2271
2272 // TODO: hook up "gfxstream egl" to the renderer flags
2273 // STREAM_RENDERER_FLAGS_USE_EGL_BIT in crosvm
2274 // as it's specified from launch_cvd.
2275 // At the moment, use ANDROID_GFXSTREAM_EGL=1
2276 // For test on GCE
2277 if (android::base::getEnvironmentVariable("ANDROID_GFXSTREAM_EGL") == "1") {
2278 android::base::setEnvironmentVariable("ANDROID_EGL_ON_EGL", "1");
2279 android::base::setEnvironmentVariable("ANDROID_EMUGL_LOG_PRINT", "1");
2280 android::base::setEnvironmentVariable("ANDROID_EMUGL_VERBOSE", "1");
2281 }
2282 // end for test on GCE
2283
2284 android::base::setEnvironmentVariable("ANDROID_EMU_HEADLESS", "1");
2285
2286 bool egl2eglByEnv = android::base::getEnvironmentVariable("ANDROID_EGL_ON_EGL") == "1";
2287 bool egl2eglByFlag = renderer_flags & STREAM_RENDERER_FLAGS_USE_EGL_BIT;
2288 bool enable_egl2egl = egl2eglByFlag || egl2eglByEnv;
2289 if (enable_egl2egl) {
2290 android::base::setEnvironmentVariable("ANDROID_GFXSTREAM_EGL", "1");
2291 android::base::setEnvironmentVariable("ANDROID_EGL_ON_EGL", "1");
2292 }
2293
2294 bool surfaceless = renderer_flags & STREAM_RENDERER_FLAGS_USE_SURFACELESS_BIT;
2295
2296 android::featurecontrol::productFeatureOverride();
2297
2298 gfxstream::vk::vkDispatch(false /* don't use test ICD */);
2299
2300 auto androidHw = aemu_get_android_hw();
2301
2302 androidHw->hw_gltransport_asg_writeBufferSize = 1048576;
2303 androidHw->hw_gltransport_asg_writeStepSize = 262144;
2304 androidHw->hw_gltransport_asg_dataRingSize = 524288;
2305 androidHw->hw_gltransport_drawFlushInterval = 10000;
2306
2307 EmuglConfig config;
2308
2309 // Make all the console agents available.
2310 android::emulation::injectGraphicsAgents(android::emulation::GfxStreamGraphicsAgentFactory());
2311
2312 emuglConfig_init(&config, true /* gpu enabled */, "auto",
2313 enable_egl2egl ? "swiftshader_indirect" : "host", 64, /* bitness */
2314 surfaceless, /* no window */
2315 false, /* blocklisted */
2316 false, /* has guest renderer */
2317 WINSYS_GLESBACKEND_PREFERENCE_AUTO, true /* force host gpu vulkan */);
2318
2319 emuglConfig_setupEnv(&config);
2320
2321 android_prepareOpenglesEmulation();
2322
2323 {
2324 static gfxstream::RenderLibPtr renderLibPtr = gfxstream::initLibrary();
2325 android_setOpenglesEmulation(renderLibPtr.get(), nullptr, nullptr);
2326 }
2327
2328 int maj;
2329 int min;
2330 android_startOpenglesRenderer(display_width, display_height, 1, 28, getGraphicsAgents()->vm,
2331 getGraphicsAgents()->emu, getGraphicsAgents()->multi_display,
2332 &features, &maj, &min);
2333
2334 char* vendor = nullptr;
2335 char* renderer = nullptr;
2336 char* version = nullptr;
2337
2338 android_getOpenglesHardwareStrings(&vendor, &renderer, &version);
2339
2340 stream_renderer_info("GL strings; [%s] [%s] [%s].", vendor, renderer, version);
2341
2342 auto openglesRenderer = android_getOpenglesRenderer();
2343
2344 if (!openglesRenderer) {
2345 stream_renderer_error("No renderer started, fatal");
2346 return -EINVAL;
2347 }
2348
2349 address_space_set_vm_operations(getGraphicsAgents()->vm);
2350 android_init_opengles_pipe();
2351 android_opengles_pipe_set_recv_mode(2 /* virtio-gpu */);
2352 android_init_refcount_pipe();
2353
2354 return 0;
2355 }
2356
2357 namespace {
2358
parseGfxstreamFeatures(const int renderer_flags,const std::string & renderer_features,gfxstream::host::FeatureSet & features)2359 int parseGfxstreamFeatures(const int renderer_flags,
2360 const std::string& renderer_features,
2361 gfxstream::host::FeatureSet& features) {
2362 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2363 &features, ExternalBlob,
2364 renderer_flags & STREAM_RENDERER_FLAGS_USE_EXTERNAL_BLOB);
2365 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2366 &features, GlAsyncSwap, false);
2367 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2368 &features, GlDirectMem, false);
2369 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2370 &features, GlDma, false);
2371 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2372 &features, GlesDynamicVersion, true);
2373 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2374 &features, GlPipeChecksum, false);
2375 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2376 &features, GuestUsesAngle,
2377 (renderer_flags & STREAM_RENDERER_FLAGS_USE_VK_BIT) &&
2378 !(renderer_flags & STREAM_RENDERER_FLAGS_USE_GLES_BIT));
2379 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2380 &features, HostComposition, true);
2381 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2382 &features, NativeTextureDecompression, false);
2383 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2384 &features, NoDelayCloseColorBuffer, true);
2385 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2386 &features, PlayStoreImage,
2387 !(renderer_flags & STREAM_RENDERER_FLAGS_USE_GLES_BIT));
2388 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2389 &features, RefCountPipe,
2390 /*Resources are ref counted via guest file objects.*/false);
2391 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2392 &features, SystemBlob,
2393 renderer_flags & STREAM_RENDERER_FLAGS_USE_SYSTEM_BLOB);
2394 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2395 &features, VirtioGpuFenceContexts, true);
2396 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2397 &features, VirtioGpuNativeSync, true);
2398 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2399 &features, VirtioGpuNext, true);
2400 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2401 &features, Vulkan,
2402 renderer_flags & STREAM_RENDERER_FLAGS_USE_VK_BIT);
2403 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2404 &features, VulkanBatchedDescriptorSetUpdate, true);
2405 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2406 &features, VulkanIgnoredHandles, true);
2407 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2408 &features, VulkanNativeSwapchain,
2409 renderer_flags & STREAM_RENDERER_FLAGS_VULKAN_NATIVE_SWAPCHAIN_BIT);
2410 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2411 &features, VulkanNullOptionalStrings, true);
2412 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2413 &features, VulkanQueueSubmitWithCommands, true);
2414 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2415 &features, VulkanShaderFloat16Int8, true);
2416 GFXSTREAM_SET_FEATURE_ON_CONDITION(
2417 &features, VulkanSnapshots,
2418 android::base::getEnvironmentVariable("ANDROID_GFXSTREAM_CAPTURE_VK_SNAPSHOT") == "1");
2419
2420 for (const std::string& renderer_feature : gfxstream::Split(renderer_features, ",")) {
2421 if (renderer_feature.empty()) continue;
2422
2423 const std::vector<std::string>& parts = gfxstream::Split(renderer_feature, ":");
2424 if (parts.size() != 2) {
2425 stream_renderer_error("Error: invalid renderer features: %s",
2426 renderer_features.c_str());
2427 return -EINVAL;
2428 }
2429
2430 const std::string& feature_name = parts[0];
2431
2432 auto feature_it = features.map.find(feature_name);
2433 if (feature_it == features.map.end()) {
2434 stream_renderer_error("Error: invalid renderer feature: '%s'", feature_name.c_str());
2435 return -EINVAL;
2436 }
2437
2438 const std::string& feature_status = parts[1];
2439 if (feature_status != "enabled" && feature_status != "disabled") {
2440 stream_renderer_error("Error: invalid option %s for renderer feature: %s",
2441 feature_status.c_str(), feature_name.c_str());
2442 return -EINVAL;
2443 }
2444
2445 auto& feature_info = feature_it->second;
2446 feature_info->enabled = feature_status == "enabled";
2447 feature_info->reason = "Overridden via STREAM_RENDERER_PARAM_RENDERER_FEATURES";
2448
2449 stream_renderer_error("Gfxstream feature %s %s", feature_name.c_str(),
2450 feature_status.c_str());
2451 }
2452
2453 if (features.SystemBlob.enabled) {
2454 if(!features.ExternalBlob.enabled) {
2455 stream_renderer_error("The SystemBlob features requires the ExternalBlob feature.");
2456 return -EINVAL;
2457 }
2458 #ifndef _WIN32
2459 stream_renderer_warn("Warning: USE_SYSTEM_BLOB has only been tested on Windows");
2460 #endif
2461 }
2462 if (features.VulkanNativeSwapchain.enabled && !features.Vulkan.enabled) {
2463 stream_renderer_error("can't enable vulkan native swapchain, Vulkan is disabled");
2464 return -EINVAL;
2465 }
2466
2467 return 0;
2468 }
2469
2470 } // namespace
2471
stream_renderer_init(struct stream_renderer_param * stream_renderer_params,uint64_t num_params)2472 VG_EXPORT int stream_renderer_init(struct stream_renderer_param* stream_renderer_params,
2473 uint64_t num_params) {
2474 // Required parameters.
2475 std::unordered_set<uint64_t> required_params{STREAM_RENDERER_PARAM_USER_DATA,
2476 STREAM_RENDERER_PARAM_RENDERER_FLAGS,
2477 STREAM_RENDERER_PARAM_FENCE_CALLBACK};
2478
2479 // String names of the parameters.
2480 std::unordered_map<uint64_t, std::string> param_strings{
2481 {STREAM_RENDERER_PARAM_USER_DATA, "USER_DATA"},
2482 {STREAM_RENDERER_PARAM_RENDERER_FLAGS, "RENDERER_FLAGS"},
2483 {STREAM_RENDERER_PARAM_FENCE_CALLBACK, "FENCE_CALLBACK"},
2484 {STREAM_RENDERER_PARAM_WIN0_WIDTH, "WIN0_WIDTH"},
2485 {STREAM_RENDERER_PARAM_WIN0_HEIGHT, "WIN0_HEIGHT"},
2486 {STREAM_RENDERER_PARAM_DEBUG_CALLBACK, "DEBUG_CALLBACK"},
2487 {STREAM_RENDERER_SKIP_OPENGLES_INIT, "SKIP_OPENGLES_INIT"},
2488 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT,
2489 "METRICS_CALLBACK_ADD_INSTANT_EVENT"},
2490 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_DESCRIPTOR,
2491 "METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_DESCRIPTOR"},
2492 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_METRIC,
2493 "METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_METRIC"},
2494 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_VULKAN_OUT_OF_MEMORY_EVENT,
2495 "METRICS_CALLBACK_ADD_VULKAN_OUT_OF_MEMORY_EVENT"},
2496 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_SET_ANNOTATION, "METRICS_CALLBACK_SET_ANNOTATION"},
2497 {STREAM_RENDERER_PARAM_METRICS_CALLBACK_ABORT, "METRICS_CALLBACK_ABORT"}};
2498
2499 // Print full values for these parameters:
2500 // Values here must not be pointers (e.g. callback functions), to avoid potentially identifying
2501 // someone via ASLR. Pointers in ASLR are randomized on boot, which means pointers may be
2502 // different between users but similar across a single user's sessions.
2503 // As a convenience, any value <= 4096 is also printed, to catch small or null pointer errors.
2504 std::unordered_set<uint64_t> printed_param_values{STREAM_RENDERER_PARAM_RENDERER_FLAGS,
2505 STREAM_RENDERER_PARAM_WIN0_WIDTH,
2506 STREAM_RENDERER_PARAM_WIN0_HEIGHT};
2507
2508 // We may have unknown parameters, so this function is lenient.
2509 auto get_param_string = [&](uint64_t key) -> std::string {
2510 auto param_string = param_strings.find(key);
2511 if (param_string != param_strings.end()) {
2512 return param_string->second;
2513 } else {
2514 return "Unknown param with key=" + std::to_string(key);
2515 }
2516 };
2517
2518 // Initialization data.
2519 uint32_t display_width = 0;
2520 uint32_t display_height = 0;
2521 void* renderer_cookie = nullptr;
2522 int renderer_flags = 0;
2523 std::string renderer_features_str;
2524 stream_renderer_fence_callback fence_callback = nullptr;
2525 bool skip_opengles = false;
2526
2527 // Iterate all parameters that we support.
2528 stream_renderer_debug("Reading stream renderer parameters:");
2529 for (uint64_t i = 0; i < num_params; ++i) {
2530 stream_renderer_param& param = stream_renderer_params[i];
2531
2532 // Print out parameter we are processing. See comment above `printed_param_values` before
2533 // adding new prints.
2534 if (printed_param_values.find(param.key) != printed_param_values.end() ||
2535 param.value <= 4096) {
2536 stream_renderer_debug("%s - %llu", get_param_string(param.key).c_str(),
2537 static_cast<unsigned long long>(param.value));
2538 } else {
2539 // If not full value, print that it was passed.
2540 stream_renderer_debug("%s", get_param_string(param.key).c_str());
2541 }
2542
2543 // Removing every param we process will leave required_params empty if all provided.
2544 required_params.erase(param.key);
2545
2546 switch (param.key) {
2547 case STREAM_RENDERER_PARAM_NULL:
2548 break;
2549 case STREAM_RENDERER_PARAM_USER_DATA: {
2550 renderer_cookie = reinterpret_cast<void*>(static_cast<uintptr_t>(param.value));
2551 globalUserData = renderer_cookie;
2552 break;
2553 }
2554 case STREAM_RENDERER_PARAM_RENDERER_FLAGS: {
2555 renderer_flags = static_cast<int>(param.value);
2556 break;
2557 }
2558 case STREAM_RENDERER_PARAM_FENCE_CALLBACK: {
2559 fence_callback = reinterpret_cast<stream_renderer_fence_callback>(
2560 static_cast<uintptr_t>(param.value));
2561 break;
2562 }
2563 case STREAM_RENDERER_PARAM_WIN0_WIDTH: {
2564 display_width = static_cast<uint32_t>(param.value);
2565 break;
2566 }
2567 case STREAM_RENDERER_PARAM_WIN0_HEIGHT: {
2568 display_height = static_cast<uint32_t>(param.value);
2569 break;
2570 }
2571 case STREAM_RENDERER_PARAM_DEBUG_CALLBACK: {
2572 globalDebugCallback = reinterpret_cast<stream_renderer_debug_callback>(
2573 static_cast<uintptr_t>(param.value));
2574 break;
2575 }
2576 case STREAM_RENDERER_SKIP_OPENGLES_INIT: {
2577 skip_opengles = static_cast<bool>(param.value);
2578 break;
2579 }
2580 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT: {
2581 MetricsLogger::add_instant_event_callback =
2582 reinterpret_cast<stream_renderer_param_metrics_callback_add_instant_event>(
2583 static_cast<uintptr_t>(param.value));
2584 break;
2585 }
2586 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_DESCRIPTOR: {
2587 MetricsLogger::add_instant_event_with_descriptor_callback = reinterpret_cast<
2588 stream_renderer_param_metrics_callback_add_instant_event_with_descriptor>(
2589 static_cast<uintptr_t>(param.value));
2590 break;
2591 }
2592 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_INSTANT_EVENT_WITH_METRIC: {
2593 MetricsLogger::add_instant_event_with_metric_callback = reinterpret_cast<
2594 stream_renderer_param_metrics_callback_add_instant_event_with_metric>(
2595 static_cast<uintptr_t>(param.value));
2596 break;
2597 }
2598 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_ADD_VULKAN_OUT_OF_MEMORY_EVENT: {
2599 MetricsLogger::add_vulkan_out_of_memory_event = reinterpret_cast<
2600 stream_renderer_param_metrics_callback_add_vulkan_out_of_memory_event>(
2601 static_cast<uintptr_t>(param.value));
2602 break;
2603 }
2604 case STREAM_RENDERER_PARAM_RENDERER_FEATURES: {
2605 renderer_features_str =
2606 std::string(reinterpret_cast<const char*>(static_cast<uintptr_t>(param.value)));
2607 break;
2608 }
2609 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_SET_ANNOTATION: {
2610 MetricsLogger::set_crash_annotation_callback =
2611 reinterpret_cast<stream_renderer_param_metrics_callback_set_annotation>(
2612 static_cast<uintptr_t>(param.value));
2613 break;
2614 }
2615 case STREAM_RENDERER_PARAM_METRICS_CALLBACK_ABORT: {
2616 emugl::setDieFunction(
2617 reinterpret_cast<stream_renderer_param_metrics_callback_abort>(
2618 static_cast<uintptr_t>(param.value)));
2619 break;
2620 }
2621 default: {
2622 // We skip any parameters we don't recognize.
2623 stream_renderer_error(
2624 "Skipping unknown parameter key: %llu. May need to upgrade gfxstream.",
2625 static_cast<unsigned long long>(param.key));
2626 break;
2627 }
2628 }
2629 }
2630 stream_renderer_debug("Finished reading parameters");
2631
2632 // Some required params not found.
2633 if (required_params.size() > 0) {
2634 stream_renderer_error("Missing required parameters:");
2635 for (uint64_t param : required_params) {
2636 stream_renderer_error("%s", get_param_string(param).c_str());
2637 }
2638 stream_renderer_error("Failing initialization intentionally");
2639 return -EINVAL;
2640 }
2641
2642 gfxstream::host::FeatureSet features;
2643 int ret = parseGfxstreamFeatures(renderer_flags, renderer_features_str, features);
2644 if (ret) {
2645 stream_renderer_error("Failed to initialize: failed to parse Gfxstream features.");
2646 return ret;
2647 }
2648
2649 stream_renderer_info("Gfxstream features:");
2650 for (const auto& [_, featureInfo] : features.map) {
2651 stream_renderer_info(" %s: %s (%s)", featureInfo->name.c_str(),
2652 (featureInfo->enabled ? "enabled" : "disabled"), featureInfo->reason.c_str());
2653 }
2654
2655 // Set non product-specific callbacks
2656 gfxstream::vk::vk_util::setVkCheckCallbacks(
2657 std::make_unique<gfxstream::vk::vk_util::VkCheckCallbacks>(
2658 gfxstream::vk::vk_util::VkCheckCallbacks{
2659 .onVkErrorOutOfMemory =
2660 [](VkResult result, const char* function, int line) {
2661 auto fb = gfxstream::FrameBuffer::getFB();
2662 if (!fb) {
2663 stream_renderer_error(
2664 "FrameBuffer not yet initialized. Dropping out of memory event");
2665 return;
2666 }
2667 fb->logVulkanOutOfMemory(result, function, line);
2668 },
2669 .onVkErrorOutOfMemoryOnAllocation =
2670 [](VkResult result, const char* function, int line,
2671 std::optional<uint64_t> allocationSize) {
2672 auto fb = gfxstream::FrameBuffer::getFB();
2673 if (!fb) {
2674 stream_renderer_error(
2675 "FrameBuffer not yet initialized. Dropping out of memory event");
2676 return;
2677 }
2678 fb->logVulkanOutOfMemory(result, function, line, allocationSize);
2679 }}));
2680
2681 if (!skip_opengles) {
2682 // aemu currently does its own opengles initialization in
2683 // qemu/android/android-emu/android/opengles.cpp.
2684 int ret = stream_renderer_opengles_init(display_width, display_height, renderer_flags, features);
2685 if (ret) {
2686 return ret;
2687 }
2688 }
2689
2690 sRenderer()->init(renderer_cookie, features, fence_callback);
2691 gfxstream::FrameBuffer::waitUntilInitialized();
2692
2693 stream_renderer_info("Gfxstream initialized successfully!");
2694 return 0;
2695 }
2696
gfxstream_backend_setup_window(void * native_window_handle,int32_t window_x,int32_t window_y,int32_t window_width,int32_t window_height,int32_t fb_width,int32_t fb_height)2697 VG_EXPORT void gfxstream_backend_setup_window(void* native_window_handle, int32_t window_x,
2698 int32_t window_y, int32_t window_width,
2699 int32_t window_height, int32_t fb_width,
2700 int32_t fb_height) {
2701 android_showOpenglesWindow(native_window_handle, window_x, window_y, window_width,
2702 window_height, fb_width, fb_height, 1.0f, 0, false, false);
2703 }
2704
stream_renderer_teardown()2705 VG_EXPORT void stream_renderer_teardown() {
2706 android_finishOpenglesRenderer();
2707 android_hideOpenglesWindow();
2708 android_stopOpenglesRenderer(true);
2709
2710 sRenderer()->teardown();
2711 stream_renderer_info("Gfxstream shut down completed!");
2712 }
2713
gfxstream_backend_set_screen_mask(int width,int height,const unsigned char * rgbaData)2714 VG_EXPORT void gfxstream_backend_set_screen_mask(int width, int height,
2715 const unsigned char* rgbaData) {
2716 android_setOpenglesScreenMask(width, height, rgbaData);
2717 }
2718
goldfish_pipe_get_service_ops()2719 const GoldfishPipeServiceOps* goldfish_pipe_get_service_ops() { return &goldfish_pipe_service_ops; }
2720
2721 static_assert(sizeof(struct stream_renderer_device_id) == 32,
2722 "stream_renderer_device_id must be 32 bytes");
2723 static_assert(offsetof(struct stream_renderer_device_id, device_uuid) == 0,
2724 "stream_renderer_device_id.device_uuid must be at offset 0");
2725 static_assert(offsetof(struct stream_renderer_device_id, driver_uuid) == 16,
2726 "stream_renderer_device_id.driver_uuid must be at offset 16");
2727
2728 static_assert(sizeof(struct stream_renderer_vulkan_info) == 36,
2729 "stream_renderer_vulkan_info must be 36 bytes");
2730 static_assert(offsetof(struct stream_renderer_vulkan_info, memory_index) == 0,
2731 "stream_renderer_vulkan_info.memory_index must be at offset 0");
2732 static_assert(offsetof(struct stream_renderer_vulkan_info, device_id) == 4,
2733 "stream_renderer_vulkan_info.device_id must be at offset 4");
2734
2735 static_assert(sizeof(struct stream_renderer_param_host_visible_memory_mask_entry) == 36,
2736 "stream_renderer_param_host_visible_memory_mask_entry must be 36 bytes");
2737 static_assert(offsetof(struct stream_renderer_param_host_visible_memory_mask_entry, device_id) == 0,
2738 "stream_renderer_param_host_visible_memory_mask_entry.device_id must be at offset 0");
2739 static_assert(
2740 offsetof(struct stream_renderer_param_host_visible_memory_mask_entry, memory_type_mask) == 32,
2741 "stream_renderer_param_host_visible_memory_mask_entry.memory_type_mask must be at offset 32");
2742
2743 static_assert(sizeof(struct stream_renderer_param_host_visible_memory_mask) == 16,
2744 "stream_renderer_param_host_visible_memory_mask must be 16 bytes");
2745 static_assert(offsetof(struct stream_renderer_param_host_visible_memory_mask, entries) == 0,
2746 "stream_renderer_param_host_visible_memory_mask.entries must be at offset 0");
2747 static_assert(offsetof(struct stream_renderer_param_host_visible_memory_mask, num_entries) == 8,
2748 "stream_renderer_param_host_visible_memory_mask.num_entries must be at offset 8");
2749
2750 static_assert(sizeof(struct stream_renderer_param) == 16, "stream_renderer_param must be 16 bytes");
2751 static_assert(offsetof(struct stream_renderer_param, key) == 0,
2752 "stream_renderer_param.key must be at offset 0");
2753 static_assert(offsetof(struct stream_renderer_param, value) == 8,
2754 "stream_renderer_param.value must be at offset 8");
2755
2756 #ifdef CONFIG_AEMU
2757
stream_renderer_set_service_ops(const GoldfishPipeServiceOps * ops)2758 VG_EXPORT void stream_renderer_set_service_ops(const GoldfishPipeServiceOps* ops) {
2759 sRenderer()->setServiceOps(ops);
2760 }
2761
2762 #endif // CONFIG_AEMU
2763
2764 } // extern "C"
2765