1 /*
2  * Copyright 2018, The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <media/stagefright/bqhelper/Conversion.h>
18 
19 namespace android {
20 namespace conversion {
21 
22 // native_handle_t helper functions.
23 
24 /**
25  * \brief Take an fd and create a native handle containing only the given fd.
26  * The created handle will need to be deleted manually with
27  * `native_handle_delete()`.
28  *
29  * \param[in] fd The source file descriptor (of type `int`).
30  * \return The create `native_handle_t*` that contains the given \p fd. If the
31  * supplied \p fd is negative, the created native handle will contain no file
32  * descriptors.
33  *
34  * If the native handle cannot be created, the return value will be
35  * `nullptr`.
36  *
37  * This function does not duplicate the file descriptor.
38  */
native_handle_create_from_fd(int fd)39 native_handle_t* native_handle_create_from_fd(int fd) {
40     if (fd < 2) {
41         return native_handle_create(0, 0);
42     }
43     native_handle_t* nh = native_handle_create(1, 0);
44     if (nh == nullptr) {
45         return nullptr;
46     }
47     nh->data[0] = fd;
48     return nh;
49 }
50 
51 /**
52  * \brief Extract a file descriptor from a native handle.
53  *
54  * \param[in] nh The source `native_handle_t*`.
55  * \param[in] index The index of the file descriptor in \p nh to read from. This
56  * input has the default value of `0`.
57  * \return The `index`-th file descriptor in \p nh. If \p nh does not have
58  * enough file descriptors, the returned value will be `-1`.
59  *
60  * This function does not duplicate the file descriptor.
61  */
native_handle_read_fd(native_handle_t const * nh,int index)62 int native_handle_read_fd(native_handle_t const* nh, int index) {
63     return ((nh == nullptr) || (nh->numFds == 0) ||
64             (nh->numFds <= index) || (index < 0)) ?
65             -1 : nh->data[index];
66 }
67 
68 /**
69  * Conversion functions
70  * ====================
71  *
72  * There are two main directions of conversion:
73  * - `inTargetType(...)`: Create a wrapper whose lifetime depends on the
74  *   input. The wrapper has type `TargetType`.
75  * - `toTargetType(...)`: Create a standalone object of type `TargetType` that
76  *   corresponds to the input. The lifetime of the output does not depend on the
77  *   lifetime of the input.
78  * - `wrapIn(TargetType*, ...)`: Same as `inTargetType()`, but for `TargetType`
79  *   that cannot be copied and/or moved efficiently, or when there are multiple
80  *   output arguments.
81  * - `convertTo(TargetType*, ...)`: Same as `toTargetType()`, but for
82  *   `TargetType` that cannot be copied and/or moved efficiently, or when there
83  *   are multiple output arguments.
84  *
85  * `wrapIn()` and `convertTo()` functions will take output arguments before
86  * input arguments. Some of these functions might return a value to indicate
87  * success or error.
88  *
89  * In converting or wrapping something as a Treble type that contains a
90  * `hidl_handle`, `native_handle_t*` will need to be created and returned as
91  * an additional output argument, hence only `wrapIn()` or `convertTo()` would
92  * be available. The caller must call `native_handle_delete()` to deallocate the
93  * returned native handle when it is no longer needed.
94  *
95  * For types that contain file descriptors, `inTargetType()` and `wrapAs()` do
96  * not perform duplication of file descriptors, while `toTargetType()` and
97  * `convertTo()` do.
98  */
99 
100 /**
101  * \brief Convert `Return<void>` to `binder::Status`.
102  *
103  * \param[in] t The source `Return<void>`.
104  * \return The corresponding `binder::Status`.
105  */
106 // convert: Return<void> -> ::android::binder::Status
toBinderStatus(Return<void> const & t)107 ::android::binder::Status toBinderStatus(
108         Return<void> const& t) {
109     return ::android::binder::Status::fromExceptionCode(
110             t.isOk() ? OK : UNKNOWN_ERROR,
111             t.description().c_str());
112 }
113 
114 /**
115  * \brief Convert `Return<void>` to `status_t`. This is for legacy binder calls.
116  *
117  * \param[in] t The source `Return<void>`.
118  * \return The corresponding `status_t`.
119  */
120 // convert: Return<void> -> status_t
toStatusT(Return<void> const & t)121 status_t toStatusT(Return<void> const& t) {
122     return t.isOk() ? OK : UNKNOWN_ERROR;
123 }
124 
125 /**
126  * \brief Wrap `native_handle_t*` in `hidl_handle`.
127  *
128  * \param[in] nh The source `native_handle_t*`.
129  * \return The `hidl_handle` that points to \p nh.
130  */
131 // wrap: native_handle_t* -> hidl_handle
inHidlHandle(native_handle_t const * nh)132 hidl_handle inHidlHandle(native_handle_t const* nh) {
133     return hidl_handle(nh);
134 }
135 
136 /**
137  * \brief Convert `int32_t` to `Dataspace`.
138  *
139  * \param[in] l The source `int32_t`.
140  * \result The corresponding `Dataspace`.
141  */
142 // convert: int32_t -> Dataspace
toHardwareDataspace(int32_t l)143 Dataspace toHardwareDataspace(int32_t l) {
144     return static_cast<Dataspace>(l);
145 }
146 
147 /**
148  * \brief Convert `Dataspace` to `int32_t`.
149  *
150  * \param[in] t The source `Dataspace`.
151  * \result The corresponding `int32_t`.
152  */
153 // convert: Dataspace -> int32_t
toRawDataspace(Dataspace const & t)154 int32_t toRawDataspace(Dataspace const& t) {
155     return static_cast<int32_t>(t);
156 }
157 
158 /**
159  * \brief Wrap an opaque buffer inside a `hidl_vec<uint8_t>`.
160  *
161  * \param[in] l The pointer to the beginning of the opaque buffer.
162  * \param[in] size The size of the buffer.
163  * \return A `hidl_vec<uint8_t>` that points to the buffer.
164  */
165 // wrap: void*, size_t -> hidl_vec<uint8_t>
inHidlBytes(void const * l,size_t size)166 hidl_vec<uint8_t> inHidlBytes(void const* l, size_t size) {
167     hidl_vec<uint8_t> t;
168     t.setToExternal(static_cast<uint8_t*>(const_cast<void*>(l)), size, false);
169     return t;
170 }
171 
172 /**
173  * \brief Create a `hidl_vec<uint8_t>` that is a copy of an opaque buffer.
174  *
175  * \param[in] l The pointer to the beginning of the opaque buffer.
176  * \param[in] size The size of the buffer.
177  * \return A `hidl_vec<uint8_t>` that is a copy of the input buffer.
178  */
179 // convert: void*, size_t -> hidl_vec<uint8_t>
toHidlBytes(void const * l,size_t size)180 hidl_vec<uint8_t> toHidlBytes(void const* l, size_t size) {
181     hidl_vec<uint8_t> t;
182     t.resize(size);
183     uint8_t const* src = static_cast<uint8_t const*>(l);
184     std::copy(src, src + size, t.data());
185     return t;
186 }
187 
188 /**
189  * \brief Wrap `GraphicBuffer` in `AnwBuffer`.
190  *
191  * \param[out] t The wrapper of type `AnwBuffer`.
192  * \param[in] l The source `GraphicBuffer`.
193  */
194 // wrap: GraphicBuffer -> AnwBuffer
wrapAs(AnwBuffer * t,GraphicBuffer const & l)195 void wrapAs(AnwBuffer* t, GraphicBuffer const& l) {
196     t->attr.width = l.getWidth();
197     t->attr.height = l.getHeight();
198     t->attr.stride = l.getStride();
199     t->attr.format = static_cast<PixelFormat>(l.getPixelFormat());
200     t->attr.layerCount = l.getLayerCount();
201     t->attr.usage = l.getUsage();
202     t->attr.id = l.getId();
203     t->attr.generationNumber = l.getGenerationNumber();
204     t->nativeHandle = hidl_handle(l.handle);
205 }
206 
207 /**
208  * \brief Convert `AnwBuffer` to `GraphicBuffer`.
209  *
210  * \param[out] l The destination `GraphicBuffer`.
211  * \param[in] t The source `AnwBuffer`.
212  *
213  * This function will duplicate all file descriptors in \p t.
214  */
215 // convert: AnwBuffer -> GraphicBuffer
216 // Ref: frameworks/native/libs/ui/GraphicBuffer.cpp: GraphicBuffer::flatten
convertTo(GraphicBuffer * l,AnwBuffer const & t)217 bool convertTo(GraphicBuffer* l, AnwBuffer const& t) {
218     native_handle_t* handle = t.nativeHandle == nullptr ?
219             nullptr : native_handle_clone(t.nativeHandle);
220 
221     size_t const numInts = 12 + (handle ? handle->numInts : 0);
222     int32_t* ints = new int32_t[numInts];
223 
224     size_t numFds = static_cast<size_t>(handle ? handle->numFds : 0);
225     int* fds = new int[numFds];
226 
227     ints[0] = 'GBFR';
228     ints[1] = static_cast<int32_t>(t.attr.width);
229     ints[2] = static_cast<int32_t>(t.attr.height);
230     ints[3] = static_cast<int32_t>(t.attr.stride);
231     ints[4] = static_cast<int32_t>(t.attr.format);
232     ints[5] = static_cast<int32_t>(t.attr.layerCount);
233     ints[6] = static_cast<int32_t>(t.attr.usage);
234     ints[7] = static_cast<int32_t>(t.attr.id >> 32);
235     ints[8] = static_cast<int32_t>(t.attr.id & 0xFFFFFFFF);
236     ints[9] = static_cast<int32_t>(t.attr.generationNumber);
237     ints[10] = 0;
238     ints[11] = 0;
239     if (handle) {
240         ints[10] = static_cast<int32_t>(handle->numFds);
241         ints[11] = static_cast<int32_t>(handle->numInts);
242         int* intsStart = handle->data + handle->numFds;
243         std::copy(handle->data, intsStart, fds);
244         std::copy(intsStart, intsStart + handle->numInts, &ints[12]);
245     }
246 
247     void const* constBuffer = static_cast<void const*>(ints);
248     size_t size = numInts * sizeof(int32_t);
249     int const* constFds = static_cast<int const*>(fds);
250     status_t status = l->unflatten(constBuffer, size, constFds, numFds);
251 
252     delete [] fds;
253     delete [] ints;
254     native_handle_delete(handle);
255     return status == NO_ERROR;
256 }
257 
258 /**
259  * Conversion functions for types outside media
260  * ============================================
261  *
262  * Some objects in libui and libgui that were made to go through binder calls do
263  * not expose ways to read or write their fields to the public. To pass an
264  * object of this kind through the HIDL boundary, translation functions need to
265  * work around the access restriction by using the publicly available
266  * `flatten()` and `unflatten()` functions.
267  *
268  * All `flatten()` and `unflatten()` overloads follow the same convention as
269  * follows:
270  *
271  *     status_t flatten(ObjectType const& object,
272  *                      [OtherType const& other, ...]
273  *                      void*& buffer, size_t& size,
274  *                      int*& fds, size_t& numFds)
275  *
276  *     status_t unflatten(ObjectType* object,
277  *                        [OtherType* other, ...,]
278  *                        void*& buffer, size_t& size,
279  *                        int*& fds, size_t& numFds)
280  *
281  * The number of `other` parameters varies depending on the `ObjectType`. For
282  * example, in the process of unflattening an object that contains
283  * `hidl_handle`, `other` is needed to hold `native_handle_t` objects that will
284  * be created.
285  *
286  * The last four parameters always work the same way in all overloads of
287  * `flatten()` and `unflatten()`:
288  * - For `flatten()`, `buffer` is the pointer to the non-fd buffer to be filled,
289  *   `size` is the size (in bytes) of the non-fd buffer pointed to by `buffer`,
290  *   `fds` is the pointer to the fd buffer to be filled, and `numFds` is the
291  *   size (in ints) of the fd buffer pointed to by `fds`.
292  * - For `unflatten()`, `buffer` is the pointer to the non-fd buffer to be read
293  *   from, `size` is the size (in bytes) of the non-fd buffer pointed to by
294  *   `buffer`, `fds` is the pointer to the fd buffer to be read from, and
295  *   `numFds` is the size (in ints) of the fd buffer pointed to by `fds`.
296  * - After a successful call to `flatten()` or `unflatten()`, `buffer` and `fds`
297  *   will be advanced, while `size` and `numFds` will be decreased to reflect
298  *   how much storage/data of the two buffers (fd and non-fd) have been used.
299  * - After an unsuccessful call, the values of `buffer`, `size`, `fds` and
300  *   `numFds` are invalid.
301  *
302  * The return value of a successful `flatten()` or `unflatten()` call will be
303  * `OK` (also aliased as `NO_ERROR`). Any other values indicate a failure.
304  *
305  * For each object type that supports flattening, there will be two accompanying
306  * functions: `getFlattenedSize()` and `getFdCount()`. `getFlattenedSize()` will
307  * return the size of the non-fd buffer that the object will need for
308  * flattening. `getFdCount()` will return the size of the fd buffer that the
309  * object will need for flattening.
310  *
311  * The set of these four functions, `getFlattenedSize()`, `getFdCount()`,
312  * `flatten()` and `unflatten()`, are similar to functions of the same name in
313  * the abstract class `Flattenable`. The only difference is that functions in
314  * this file are not member functions of the object type. For example, we write
315  *
316  *     flatten(x, buffer, size, fds, numFds)
317  *
318  * instead of
319  *
320  *     x.flatten(buffer, size, fds, numFds)
321  *
322  * because we cannot modify the type of `x`.
323  *
324  * There is one exception to the naming convention: `hidl_handle` that
325  * represents a fence. The four functions for this "Fence" type have the word
326  * "Fence" attched to their names because the object type, which is
327  * `hidl_handle`, does not carry the special meaning that the object itself can
328  * only contain zero or one file descriptor.
329  */
330 
331 // Ref: frameworks/native/libs/ui/Fence.cpp
332 
333 /**
334  * \brief Return the size of the non-fd buffer required to flatten a fence.
335  *
336  * \param[in] fence The input fence of type `hidl_handle`.
337  * \return The required size of the flat buffer.
338  *
339  * The current version of this function always returns 4, which is the number of
340  * bytes required to store the number of file descriptors contained in the fd
341  * part of the flat buffer.
342  */
getFenceFlattenedSize(hidl_handle const &)343 size_t getFenceFlattenedSize(hidl_handle const& /* fence */) {
344     return 4;
345 };
346 
347 /**
348  * \brief Return the number of file descriptors contained in a fence.
349  *
350  * \param[in] fence The input fence of type `hidl_handle`.
351  * \return `0` if \p fence does not contain a valid file descriptor, or `1`
352  * otherwise.
353  */
getFenceFdCount(hidl_handle const & fence)354 size_t getFenceFdCount(hidl_handle const& fence) {
355     return native_handle_read_fd(fence) == -1 ? 0 : 1;
356 }
357 
358 /**
359  * \brief Unflatten `Fence` to `hidl_handle`.
360  *
361  * \param[out] fence The destination `hidl_handle`.
362  * \param[out] nh The underlying native handle.
363  * \param[in,out] buffer The pointer to the flat non-fd buffer.
364  * \param[in,out] size The size of the flat non-fd buffer.
365  * \param[in,out] fds The pointer to the flat fd buffer.
366  * \param[in,out] numFds The size of the flat fd buffer.
367  * \return `NO_ERROR` on success; other value on failure.
368  *
369  * If the return value is `NO_ERROR`, \p nh will point to a newly created
370  * native handle, which needs to be deleted with `native_handle_delete()`
371  * afterwards.
372  */
unflattenFence(hidl_handle * fence,native_handle_t ** nh,void const * & buffer,size_t & size,int const * & fds,size_t & numFds)373 status_t unflattenFence(hidl_handle* fence, native_handle_t** nh,
374         void const*& buffer, size_t& size, int const*& fds, size_t& numFds) {
375     if (size < 4) {
376         return NO_MEMORY;
377     }
378 
379     uint32_t numFdsInHandle;
380     FlattenableUtils::read(buffer, size, numFdsInHandle);
381 
382     if (numFdsInHandle > 1) {
383         return BAD_VALUE;
384     }
385 
386     if (numFds < numFdsInHandle) {
387         return NO_MEMORY;
388     }
389 
390     if (numFdsInHandle) {
391         *nh = native_handle_create_from_fd(*fds);
392         if (*nh == nullptr) {
393             return NO_MEMORY;
394         }
395         *fence = *nh;
396         ++fds;
397         --numFds;
398     } else {
399         *nh = nullptr;
400         *fence = hidl_handle();
401     }
402 
403     return NO_ERROR;
404 }
405 
406 /**
407  * \brief Flatten `hidl_handle` as `Fence`.
408  *
409  * \param[in] t The source `hidl_handle`.
410  * \param[in,out] buffer The pointer to the flat non-fd buffer.
411  * \param[in,out] size The size of the flat non-fd buffer.
412  * \param[in,out] fds The pointer to the flat fd buffer.
413  * \param[in,out] numFds The size of the flat fd buffer.
414  * \return `NO_ERROR` on success; other value on failure.
415  */
flattenFence(hidl_handle const & fence,void * & buffer,size_t & size,int * & fds,size_t & numFds)416 status_t flattenFence(hidl_handle const& fence,
417         void*& buffer, size_t& size, int*& fds, size_t& numFds) {
418     if (size < getFenceFlattenedSize(fence) ||
419             numFds < getFenceFdCount(fence)) {
420         return NO_MEMORY;
421     }
422     // Cast to uint32_t since the size of a size_t can vary between 32- and
423     // 64-bit processes
424     FlattenableUtils::write(buffer, size,
425             static_cast<uint32_t>(getFenceFdCount(fence)));
426     int fd = native_handle_read_fd(fence);
427     if (fd != -1) {
428         *fds = fd;
429         ++fds;
430         --numFds;
431     }
432     return NO_ERROR;
433 }
434 
435 /**
436  * \brief Wrap `Fence` in `hidl_handle`.
437  *
438  * \param[out] t The wrapper of type `hidl_handle`.
439  * \param[out] nh The native handle pointed to by \p t.
440  * \param[in] l The source `Fence`.
441  *
442  * On success, \p nh will hold a newly created native handle, which must be
443  * deleted manually with `native_handle_delete()` afterwards.
444  */
445 // wrap: Fence -> hidl_handle
wrapAs(hidl_handle * t,native_handle_t ** nh,Fence const & l)446 bool wrapAs(hidl_handle* t, native_handle_t** nh, Fence const& l) {
447     size_t const baseSize = l.getFlattenedSize();
448     std::unique_ptr<uint8_t[]> baseBuffer(
449             new (std::nothrow) uint8_t[baseSize]);
450     if (!baseBuffer) {
451         return false;
452     }
453 
454     size_t const baseNumFds = l.getFdCount();
455     std::unique_ptr<int[]> baseFds(
456             new (std::nothrow) int[baseNumFds]);
457     if (!baseFds) {
458         return false;
459     }
460 
461     void* buffer = static_cast<void*>(baseBuffer.get());
462     size_t size = baseSize;
463     int* fds = static_cast<int*>(baseFds.get());
464     size_t numFds = baseNumFds;
465     if (l.flatten(buffer, size, fds, numFds) != NO_ERROR) {
466         return false;
467     }
468 
469     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
470     size = baseSize;
471     int const* constFds = static_cast<int const*>(baseFds.get());
472     numFds = baseNumFds;
473     if (unflattenFence(t, nh, constBuffer, size, constFds, numFds)
474             != NO_ERROR) {
475         return false;
476     }
477 
478     return true;
479 }
480 
481 /**
482  * \brief Convert `hidl_handle` to `Fence`.
483  *
484  * \param[out] l The destination `Fence`. `l` must not have been used
485  * (`l->isValid()` must return `false`) before this function is called.
486  * \param[in] t The source `hidl_handle`.
487  *
488  * If \p t contains a valid file descriptor, it will be duplicated.
489  */
490 // convert: hidl_handle -> Fence
convertTo(Fence * l,hidl_handle const & t)491 bool convertTo(Fence* l, hidl_handle const& t) {
492     int fd = native_handle_read_fd(t);
493     if (fd != -1) {
494         fd = dup(fd);
495         if (fd == -1) {
496             return false;
497         }
498     }
499     native_handle_t* nh = native_handle_create_from_fd(fd);
500     if (nh == nullptr) {
501         if (fd != -1) {
502             close(fd);
503         }
504         return false;
505     }
506 
507     size_t const baseSize = getFenceFlattenedSize(t);
508     std::unique_ptr<uint8_t[]> baseBuffer(
509             new (std::nothrow) uint8_t[baseSize]);
510     if (!baseBuffer) {
511         native_handle_delete(nh);
512         return false;
513     }
514 
515     size_t const baseNumFds = getFenceFdCount(t);
516     std::unique_ptr<int[]> baseFds(
517             new (std::nothrow) int[baseNumFds]);
518     if (!baseFds) {
519         native_handle_delete(nh);
520         return false;
521     }
522 
523     void* buffer = static_cast<void*>(baseBuffer.get());
524     size_t size = baseSize;
525     int* fds = static_cast<int*>(baseFds.get());
526     size_t numFds = baseNumFds;
527     if (flattenFence(hidl_handle(nh), buffer, size, fds, numFds) != NO_ERROR) {
528         native_handle_delete(nh);
529         return false;
530     }
531     native_handle_delete(nh);
532 
533     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
534     size = baseSize;
535     int const* constFds = static_cast<int const*>(baseFds.get());
536     numFds = baseNumFds;
537     if (l->unflatten(constBuffer, size, constFds, numFds) != NO_ERROR) {
538         return false;
539     }
540 
541     return true;
542 }
543 
544 // Ref: frameworks/native/libs/ui/FenceTime.cpp: FenceTime::Snapshot
545 
546 /**
547  * \brief Return the size of the non-fd buffer required to flatten
548  * `FenceTimeSnapshot`.
549  *
550  * \param[in] t The input `FenceTimeSnapshot`.
551  * \return The required size of the flat buffer.
552  */
getFlattenedSize(HGraphicBufferProducer::FenceTimeSnapshot const & t)553 size_t getFlattenedSize(
554         HGraphicBufferProducer::FenceTimeSnapshot const& t) {
555     constexpr size_t min = sizeof(t.state);
556     switch (t.state) {
557         case HGraphicBufferProducer::FenceTimeSnapshot::State::EMPTY:
558             return min;
559         case HGraphicBufferProducer::FenceTimeSnapshot::State::FENCE:
560             return min + getFenceFlattenedSize(t.fence);
561         case HGraphicBufferProducer::FenceTimeSnapshot::State::SIGNAL_TIME:
562             return min + sizeof(
563                     ::android::FenceTime::Snapshot::signalTime);
564     }
565     return 0;
566 }
567 
568 /**
569  * \brief Return the number of file descriptors contained in
570  * `FenceTimeSnapshot`.
571  *
572  * \param[in] t The input `FenceTimeSnapshot`.
573  * \return The number of file descriptors contained in \p snapshot.
574  */
getFdCount(HGraphicBufferProducer::FenceTimeSnapshot const & t)575 size_t getFdCount(
576         HGraphicBufferProducer::FenceTimeSnapshot const& t) {
577     return t.state ==
578             HGraphicBufferProducer::FenceTimeSnapshot::State::FENCE ?
579             getFenceFdCount(t.fence) : 0;
580 }
581 
582 /**
583  * \brief Flatten `FenceTimeSnapshot`.
584  *
585  * \param[in] t The source `FenceTimeSnapshot`.
586  * \param[in,out] buffer The pointer to the flat non-fd buffer.
587  * \param[in,out] size The size of the flat non-fd buffer.
588  * \param[in,out] fds The pointer to the flat fd buffer.
589  * \param[in,out] numFds The size of the flat fd buffer.
590  * \return `NO_ERROR` on success; other value on failure.
591  *
592  * This function will duplicate the file descriptor in `t.fence` if `t.state ==
593  * FENCE`.
594  */
flatten(HGraphicBufferProducer::FenceTimeSnapshot const & t,void * & buffer,size_t & size,int * & fds,size_t & numFds)595 status_t flatten(HGraphicBufferProducer::FenceTimeSnapshot const& t,
596         void*& buffer, size_t& size, int*& fds, size_t& numFds) {
597     if (size < getFlattenedSize(t)) {
598         return NO_MEMORY;
599     }
600 
601     switch (t.state) {
602         case HGraphicBufferProducer::FenceTimeSnapshot::State::EMPTY:
603             FlattenableUtils::write(buffer, size,
604                     ::android::FenceTime::Snapshot::State::EMPTY);
605             return NO_ERROR;
606         case HGraphicBufferProducer::FenceTimeSnapshot::State::FENCE:
607             FlattenableUtils::write(buffer, size,
608                     ::android::FenceTime::Snapshot::State::FENCE);
609             return flattenFence(t.fence, buffer, size, fds, numFds);
610         case HGraphicBufferProducer::FenceTimeSnapshot::State::SIGNAL_TIME:
611             FlattenableUtils::write(buffer, size,
612                     ::android::FenceTime::Snapshot::State::SIGNAL_TIME);
613             FlattenableUtils::write(buffer, size, t.signalTimeNs);
614             return NO_ERROR;
615     }
616     return NO_ERROR;
617 }
618 
619 /**
620  * \brief Unflatten `FenceTimeSnapshot`.
621  *
622  * \param[out] t The destination `FenceTimeSnapshot`.
623  * \param[out] nh The underlying native handle.
624  * \param[in,out] buffer The pointer to the flat non-fd buffer.
625  * \param[in,out] size The size of the flat non-fd buffer.
626  * \param[in,out] fds The pointer to the flat fd buffer.
627  * \param[in,out] numFds The size of the flat fd buffer.
628  * \return `NO_ERROR` on success; other value on failure.
629  *
630  * If the return value is `NO_ERROR` and the constructed snapshot contains a
631  * file descriptor, \p nh will be created to hold that file descriptor. In this
632  * case, \p nh needs to be deleted with `native_handle_delete()` afterwards.
633  */
unflatten(HGraphicBufferProducer::FenceTimeSnapshot * t,native_handle_t ** nh,void const * & buffer,size_t & size,int const * & fds,size_t & numFds)634 status_t unflatten(
635         HGraphicBufferProducer::FenceTimeSnapshot* t, native_handle_t** nh,
636         void const*& buffer, size_t& size, int const*& fds, size_t& numFds) {
637     if (size < sizeof(t->state)) {
638         return NO_MEMORY;
639     }
640 
641     *nh = nullptr;
642     ::android::FenceTime::Snapshot::State state;
643     FlattenableUtils::read(buffer, size, state);
644     switch (state) {
645         case ::android::FenceTime::Snapshot::State::EMPTY:
646             t->state = HGraphicBufferProducer::FenceTimeSnapshot::State::EMPTY;
647             return NO_ERROR;
648         case ::android::FenceTime::Snapshot::State::FENCE:
649             t->state = HGraphicBufferProducer::FenceTimeSnapshot::State::FENCE;
650             return unflattenFence(&t->fence, nh, buffer, size, fds, numFds);
651         case ::android::FenceTime::Snapshot::State::SIGNAL_TIME:
652             t->state = HGraphicBufferProducer::FenceTimeSnapshot::State::SIGNAL_TIME;
653             if (size < sizeof(t->signalTimeNs)) {
654                 return NO_MEMORY;
655             }
656             FlattenableUtils::read(buffer, size, t->signalTimeNs);
657             return NO_ERROR;
658     }
659     return NO_ERROR;
660 }
661 
662 // Ref: frameworks/native/libs/gui/FrameTimestamps.cpp: FrameEventsDelta
663 
664 /**
665  * \brief Return a lower bound on the size of the non-fd buffer required to
666  * flatten `FrameEventsDelta`.
667  *
668  * \param[in] t The input `FrameEventsDelta`.
669  * \return A lower bound on the size of the flat buffer.
670  */
minFlattenedSize(HGraphicBufferProducer::FrameEventsDelta const &)671 constexpr size_t minFlattenedSize(
672         HGraphicBufferProducer::FrameEventsDelta const& /* t */) {
673     return sizeof(uint64_t) + // mFrameNumber
674             sizeof(uint8_t) + // mIndex
675             sizeof(uint8_t) + // mAddPostCompositeCalled
676             sizeof(uint8_t) + // mAddRetireCalled
677             sizeof(uint8_t) + // mAddReleaseCalled
678             sizeof(nsecs_t) + // mPostedTime
679             sizeof(nsecs_t) + // mRequestedPresentTime
680             sizeof(nsecs_t) + // mLatchTime
681             sizeof(nsecs_t) + // mFirstRefreshStartTime
682             sizeof(nsecs_t); // mLastRefreshStartTime
683 }
684 
685 /**
686  * \brief Return the size of the non-fd buffer required to flatten
687  * `FrameEventsDelta`.
688  *
689  * \param[in] t The input `FrameEventsDelta`.
690  * \return The required size of the flat buffer.
691  */
getFlattenedSize(HGraphicBufferProducer::FrameEventsDelta const & t)692 size_t getFlattenedSize(
693         HGraphicBufferProducer::FrameEventsDelta const& t) {
694     return minFlattenedSize(t) +
695             getFlattenedSize(t.gpuCompositionDoneFence) +
696             getFlattenedSize(t.displayPresentFence) +
697             getFlattenedSize(t.displayRetireFence) +
698             getFlattenedSize(t.releaseFence);
699 };
700 
701 /**
702  * \brief Return the number of file descriptors contained in
703  * `FrameEventsDelta`.
704  *
705  * \param[in] t The input `FrameEventsDelta`.
706  * \return The number of file descriptors contained in \p t.
707  */
getFdCount(HGraphicBufferProducer::FrameEventsDelta const & t)708 size_t getFdCount(
709         HGraphicBufferProducer::FrameEventsDelta const& t) {
710     return getFdCount(t.gpuCompositionDoneFence) +
711             getFdCount(t.displayPresentFence) +
712             getFdCount(t.displayRetireFence) +
713             getFdCount(t.releaseFence);
714 };
715 
716 /**
717  * \brief Unflatten `FrameEventsDelta`.
718  *
719  * \param[out] t The destination `FrameEventsDelta`.
720  * \param[out] nh The underlying array of native handles.
721  * \param[in,out] buffer The pointer to the flat non-fd buffer.
722  * \param[in,out] size The size of the flat non-fd buffer.
723  * \param[in,out] fds The pointer to the flat fd buffer.
724  * \param[in,out] numFds The size of the flat fd buffer.
725  * \return `NO_ERROR` on success; other value on failure.
726  *
727  * If the return value is `NO_ERROR`, \p nh will have length 4, and it will be
728  * populated with `nullptr` or newly created handles. Each non-null slot in \p
729  * nh will need to be deleted manually with `native_handle_delete()`.
730  */
unflatten(HGraphicBufferProducer::FrameEventsDelta * t,std::vector<native_handle_t * > * nh,void const * & buffer,size_t & size,int const * & fds,size_t & numFds)731 status_t unflatten(HGraphicBufferProducer::FrameEventsDelta* t,
732         std::vector<native_handle_t*>* nh,
733         void const*& buffer, size_t& size, int const*& fds, size_t& numFds) {
734     if (size < minFlattenedSize(*t)) {
735         return NO_MEMORY;
736     }
737     FlattenableUtils::read(buffer, size, t->frameNumber);
738 
739     // These were written as uint8_t for alignment.
740     uint8_t temp = 0;
741     FlattenableUtils::read(buffer, size, temp);
742     size_t index = static_cast<size_t>(temp);
743     if (index >= ::android::FrameEventHistory::MAX_FRAME_HISTORY) {
744         return BAD_VALUE;
745     }
746     t->index = static_cast<uint32_t>(index);
747 
748     FlattenableUtils::read(buffer, size, temp);
749     t->addPostCompositeCalled = static_cast<bool>(temp);
750     FlattenableUtils::read(buffer, size, temp);
751     t->addRetireCalled = static_cast<bool>(temp);
752     FlattenableUtils::read(buffer, size, temp);
753     t->addReleaseCalled = static_cast<bool>(temp);
754 
755     FlattenableUtils::read(buffer, size, t->postedTimeNs);
756     FlattenableUtils::read(buffer, size, t->requestedPresentTimeNs);
757     FlattenableUtils::read(buffer, size, t->latchTimeNs);
758     FlattenableUtils::read(buffer, size, t->firstRefreshStartTimeNs);
759     FlattenableUtils::read(buffer, size, t->lastRefreshStartTimeNs);
760     FlattenableUtils::read(buffer, size, t->dequeueReadyTime);
761 
762     // Fences
763     HGraphicBufferProducer::FenceTimeSnapshot* tSnapshot[4];
764     tSnapshot[0] = &t->gpuCompositionDoneFence;
765     tSnapshot[1] = &t->displayPresentFence;
766     tSnapshot[2] = &t->displayRetireFence;
767     tSnapshot[3] = &t->releaseFence;
768     nh->resize(4);
769     for (size_t snapshotIndex = 0; snapshotIndex < 4; ++snapshotIndex) {
770         status_t status = unflatten(
771                 tSnapshot[snapshotIndex], &((*nh)[snapshotIndex]),
772                 buffer, size, fds, numFds);
773         if (status != NO_ERROR) {
774             while (snapshotIndex > 0) {
775                 --snapshotIndex;
776                 if ((*nh)[snapshotIndex] != nullptr) {
777                     native_handle_delete((*nh)[snapshotIndex]);
778                 }
779             }
780             return status;
781         }
782     }
783     return NO_ERROR;
784 }
785 
786 /**
787  * \brief Flatten `FrameEventsDelta`.
788  *
789  * \param[in] t The source `FrameEventsDelta`.
790  * \param[in,out] buffer The pointer to the flat non-fd buffer.
791  * \param[in,out] size The size of the flat non-fd buffer.
792  * \param[in,out] fds The pointer to the flat fd buffer.
793  * \param[in,out] numFds The size of the flat fd buffer.
794  * \return `NO_ERROR` on success; other value on failure.
795  *
796  * This function will duplicate file descriptors contained in \p t.
797  */
798 // Ref: frameworks/native/libs/gui/FrameTimestamp.cpp:
799 //      FrameEventsDelta::flatten
flatten(HGraphicBufferProducer::FrameEventsDelta const & t,void * & buffer,size_t & size,int * & fds,size_t numFds)800 status_t flatten(HGraphicBufferProducer::FrameEventsDelta const& t,
801         void*& buffer, size_t& size, int*& fds, size_t numFds) {
802     // Check that t.index is within a valid range.
803     if (t.index >= static_cast<uint32_t>(FrameEventHistory::MAX_FRAME_HISTORY)
804             || t.index > std::numeric_limits<uint8_t>::max()) {
805         return BAD_VALUE;
806     }
807 
808     FlattenableUtils::write(buffer, size, t.frameNumber);
809 
810     // These are static_cast to uint8_t for alignment.
811     FlattenableUtils::write(buffer, size, static_cast<uint8_t>(t.index));
812     FlattenableUtils::write(
813             buffer, size, static_cast<uint8_t>(t.addPostCompositeCalled));
814     FlattenableUtils::write(
815             buffer, size, static_cast<uint8_t>(t.addRetireCalled));
816     FlattenableUtils::write(
817             buffer, size, static_cast<uint8_t>(t.addReleaseCalled));
818 
819     FlattenableUtils::write(buffer, size, t.postedTimeNs);
820     FlattenableUtils::write(buffer, size, t.requestedPresentTimeNs);
821     FlattenableUtils::write(buffer, size, t.latchTimeNs);
822     FlattenableUtils::write(buffer, size, t.firstRefreshStartTimeNs);
823     FlattenableUtils::write(buffer, size, t.lastRefreshStartTimeNs);
824     FlattenableUtils::write(buffer, size, t.dequeueReadyTime);
825 
826     // Fences
827     HGraphicBufferProducer::FenceTimeSnapshot const* tSnapshot[4];
828     tSnapshot[0] = &t.gpuCompositionDoneFence;
829     tSnapshot[1] = &t.displayPresentFence;
830     tSnapshot[2] = &t.displayRetireFence;
831     tSnapshot[3] = &t.releaseFence;
832     for (size_t snapshotIndex = 0; snapshotIndex < 4; ++snapshotIndex) {
833         status_t status = flatten(
834                 *(tSnapshot[snapshotIndex]), buffer, size, fds, numFds);
835         if (status != NO_ERROR) {
836             return status;
837         }
838     }
839     return NO_ERROR;
840 }
841 
842 // Ref: frameworks/native/libs/gui/FrameTimestamps.cpp: FrameEventHistoryDelta
843 
844 /**
845  * \brief Return the size of the non-fd buffer required to flatten
846  * `HGraphicBufferProducer::FrameEventHistoryDelta`.
847  *
848  * \param[in] t The input `HGraphicBufferProducer::FrameEventHistoryDelta`.
849  * \return The required size of the flat buffer.
850  */
getFlattenedSize(HGraphicBufferProducer::FrameEventHistoryDelta const & t)851 size_t getFlattenedSize(
852         HGraphicBufferProducer::FrameEventHistoryDelta const& t) {
853     size_t size = 4 + // mDeltas.size()
854             sizeof(t.compositorTiming);
855     for (size_t i = 0; i < t.deltas.size(); ++i) {
856         size += getFlattenedSize(t.deltas[i]);
857     }
858     return size;
859 }
860 
861 /**
862  * \brief Return the number of file descriptors contained in
863  * `HGraphicBufferProducer::FrameEventHistoryDelta`.
864  *
865  * \param[in] t The input `HGraphicBufferProducer::FrameEventHistoryDelta`.
866  * \return The number of file descriptors contained in \p t.
867  */
getFdCount(HGraphicBufferProducer::FrameEventHistoryDelta const & t)868 size_t getFdCount(
869         HGraphicBufferProducer::FrameEventHistoryDelta const& t) {
870     size_t numFds = 0;
871     for (size_t i = 0; i < t.deltas.size(); ++i) {
872         numFds += getFdCount(t.deltas[i]);
873     }
874     return numFds;
875 }
876 
877 /**
878  * \brief Unflatten `FrameEventHistoryDelta`.
879  *
880  * \param[out] t The destination `FrameEventHistoryDelta`.
881  * \param[out] nh The underlying array of arrays of native handles.
882  * \param[in,out] buffer The pointer to the flat non-fd buffer.
883  * \param[in,out] size The size of the flat non-fd buffer.
884  * \param[in,out] fds The pointer to the flat fd buffer.
885  * \param[in,out] numFds The size of the flat fd buffer.
886  * \return `NO_ERROR` on success; other value on failure.
887  *
888  * If the return value is `NO_ERROR`, \p nh will be populated with `nullptr` or
889  * newly created handles. The second dimension of \p nh will be 4. Each non-null
890  * slot in \p nh will need to be deleted manually with `native_handle_delete()`.
891  */
unflatten(HGraphicBufferProducer::FrameEventHistoryDelta * t,std::vector<std::vector<native_handle_t * >> * nh,void const * & buffer,size_t & size,int const * & fds,size_t & numFds)892 status_t unflatten(
893         HGraphicBufferProducer::FrameEventHistoryDelta* t,
894         std::vector<std::vector<native_handle_t*> >* nh,
895         void const*& buffer, size_t& size, int const*& fds, size_t& numFds) {
896     if (size < 4) {
897         return NO_MEMORY;
898     }
899 
900     FlattenableUtils::read(buffer, size, t->compositorTiming);
901 
902     uint32_t deltaCount = 0;
903     FlattenableUtils::read(buffer, size, deltaCount);
904     if (static_cast<size_t>(deltaCount) >
905             ::android::FrameEventHistory::MAX_FRAME_HISTORY) {
906         return BAD_VALUE;
907     }
908     t->deltas.resize(deltaCount);
909     nh->resize(deltaCount);
910     for (size_t deltaIndex = 0; deltaIndex < deltaCount; ++deltaIndex) {
911         status_t status = unflatten(
912                 &(t->deltas[deltaIndex]), &((*nh)[deltaIndex]),
913                 buffer, size, fds, numFds);
914         if (status != NO_ERROR) {
915             return status;
916         }
917     }
918     return NO_ERROR;
919 }
920 
921 /**
922  * \brief Flatten `FrameEventHistoryDelta`.
923  *
924  * \param[in] t The source `FrameEventHistoryDelta`.
925  * \param[in,out] buffer The pointer to the flat non-fd buffer.
926  * \param[in,out] size The size of the flat non-fd buffer.
927  * \param[in,out] fds The pointer to the flat fd buffer.
928  * \param[in,out] numFds The size of the flat fd buffer.
929  * \return `NO_ERROR` on success; other value on failure.
930  *
931  * This function will duplicate file descriptors contained in \p t.
932  */
flatten(HGraphicBufferProducer::FrameEventHistoryDelta const & t,void * & buffer,size_t & size,int * & fds,size_t & numFds)933 status_t flatten(
934         HGraphicBufferProducer::FrameEventHistoryDelta const& t,
935         void*& buffer, size_t& size, int*& fds, size_t& numFds) {
936     if (t.deltas.size() > ::android::FrameEventHistory::MAX_FRAME_HISTORY) {
937         return BAD_VALUE;
938     }
939     if (size < getFlattenedSize(t)) {
940         return NO_MEMORY;
941     }
942 
943     FlattenableUtils::write(buffer, size, t.compositorTiming);
944 
945     FlattenableUtils::write(buffer, size, static_cast<uint32_t>(t.deltas.size()));
946     for (size_t deltaIndex = 0; deltaIndex < t.deltas.size(); ++deltaIndex) {
947         status_t status = flatten(t.deltas[deltaIndex], buffer, size, fds, numFds);
948         if (status != NO_ERROR) {
949             return status;
950         }
951     }
952     return NO_ERROR;
953 }
954 
955 /**
956  * \brief Wrap `::android::FrameEventHistoryData` in
957  * `HGraphicBufferProducer::FrameEventHistoryDelta`.
958  *
959  * \param[out] t The wrapper of type
960  * `HGraphicBufferProducer::FrameEventHistoryDelta`.
961  * \param[out] nh The array of array of native handles that are referred to by
962  * members of \p t.
963  * \param[in] l The source `::android::FrameEventHistoryDelta`.
964  *
965  * On success, each member of \p nh will be either `nullptr` or a newly created
966  * native handle. All the non-`nullptr` elements must be deleted individually
967  * with `native_handle_delete()`.
968  */
wrapAs(HGraphicBufferProducer::FrameEventHistoryDelta * t,std::vector<std::vector<native_handle_t * >> * nh,::android::FrameEventHistoryDelta const & l)969 bool wrapAs(HGraphicBufferProducer::FrameEventHistoryDelta* t,
970         std::vector<std::vector<native_handle_t*> >* nh,
971         ::android::FrameEventHistoryDelta const& l) {
972 
973     size_t const baseSize = l.getFlattenedSize();
974     std::unique_ptr<uint8_t[]> baseBuffer(
975             new (std::nothrow) uint8_t[baseSize]);
976     if (!baseBuffer) {
977         return false;
978     }
979 
980     size_t const baseNumFds = l.getFdCount();
981     std::unique_ptr<int[]> baseFds(
982             new (std::nothrow) int[baseNumFds]);
983     if (!baseFds) {
984         return false;
985     }
986 
987     void* buffer = static_cast<void*>(baseBuffer.get());
988     size_t size = baseSize;
989     int* fds = baseFds.get();
990     size_t numFds = baseNumFds;
991     if (l.flatten(buffer, size, fds, numFds) != NO_ERROR) {
992         return false;
993     }
994 
995     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
996     size = baseSize;
997     int const* constFds = static_cast<int const*>(baseFds.get());
998     numFds = baseNumFds;
999     if (unflatten(t, nh, constBuffer, size, constFds, numFds) != NO_ERROR) {
1000         return false;
1001     }
1002 
1003     return true;
1004 }
1005 
1006 /**
1007  * \brief Convert `HGraphicBufferProducer::FrameEventHistoryDelta` to
1008  * `::android::FrameEventHistoryDelta`.
1009  *
1010  * \param[out] l The destination `::android::FrameEventHistoryDelta`.
1011  * \param[in] t The source `HGraphicBufferProducer::FrameEventHistoryDelta`.
1012  *
1013  * This function will duplicate all file descriptors contained in \p t.
1014  */
convertTo(::android::FrameEventHistoryDelta * l,HGraphicBufferProducer::FrameEventHistoryDelta const & t)1015 bool convertTo(
1016         ::android::FrameEventHistoryDelta* l,
1017         HGraphicBufferProducer::FrameEventHistoryDelta const& t) {
1018 
1019     size_t const baseSize = getFlattenedSize(t);
1020     std::unique_ptr<uint8_t[]> baseBuffer(
1021             new (std::nothrow) uint8_t[baseSize]);
1022     if (!baseBuffer) {
1023         return false;
1024     }
1025 
1026     size_t const baseNumFds = getFdCount(t);
1027     std::unique_ptr<int[]> baseFds(
1028             new (std::nothrow) int[baseNumFds]);
1029     if (!baseFds) {
1030         return false;
1031     }
1032 
1033     void* buffer = static_cast<void*>(baseBuffer.get());
1034     size_t size = baseSize;
1035     int* fds = static_cast<int*>(baseFds.get());
1036     size_t numFds = baseNumFds;
1037     if (flatten(t, buffer, size, fds, numFds) != NO_ERROR) {
1038         return false;
1039     }
1040 
1041     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
1042     size = baseSize;
1043     int const* constFds = static_cast<int const*>(baseFds.get());
1044     numFds = baseNumFds;
1045     if (l->unflatten(constBuffer, size, constFds, numFds) != NO_ERROR) {
1046         return false;
1047     }
1048 
1049     return true;
1050 }
1051 
1052 // Ref: frameworks/native/libs/ui/Region.cpp
1053 
1054 /**
1055  * \brief Return the size of the buffer required to flatten `Region`.
1056  *
1057  * \param[in] t The input `Region`.
1058  * \return The required size of the flat buffer.
1059  */
getFlattenedSize(Region const & t)1060 size_t getFlattenedSize(Region const& t) {
1061     return sizeof(uint32_t) + t.size() * sizeof(::android::Rect);
1062 }
1063 
1064 /**
1065  * \brief Unflatten `Region`.
1066  *
1067  * \param[out] t The destination `Region`.
1068  * \param[in,out] buffer The pointer to the flat buffer.
1069  * \param[in,out] size The size of the flat buffer.
1070  * \return `NO_ERROR` on success; other value on failure.
1071  */
unflatten(Region * t,void const * & buffer,size_t & size)1072 status_t unflatten(Region* t, void const*& buffer, size_t& size) {
1073     if (size < sizeof(uint32_t)) {
1074         return NO_MEMORY;
1075     }
1076 
1077     uint32_t numRects = 0;
1078     FlattenableUtils::read(buffer, size, numRects);
1079     if (size < numRects * sizeof(Rect)) {
1080         return NO_MEMORY;
1081     }
1082     if (numRects > (UINT32_MAX / sizeof(Rect))) {
1083         return NO_MEMORY;
1084     }
1085 
1086     t->resize(numRects);
1087     for (size_t r = 0; r < numRects; ++r) {
1088         ::android::Rect rect(::android::Rect::EMPTY_RECT);
1089         status_t status = rect.unflatten(buffer, size);
1090         if (status != NO_ERROR) {
1091             return status;
1092         }
1093         FlattenableUtils::advance(buffer, size, sizeof(rect));
1094         (*t)[r] = Rect{
1095                 static_cast<int32_t>(rect.left),
1096                 static_cast<int32_t>(rect.top),
1097                 static_cast<int32_t>(rect.right),
1098                 static_cast<int32_t>(rect.bottom)};
1099     }
1100     return NO_ERROR;
1101 }
1102 
1103 /**
1104  * \brief Flatten `Region`.
1105  *
1106  * \param[in] t The source `Region`.
1107  * \param[in,out] buffer The pointer to the flat buffer.
1108  * \param[in,out] size The size of the flat buffer.
1109  * \return `NO_ERROR` on success; other value on failure.
1110  */
flatten(Region const & t,void * & buffer,size_t & size)1111 status_t flatten(Region const& t, void*& buffer, size_t& size) {
1112     if (size < getFlattenedSize(t)) {
1113         return NO_MEMORY;
1114     }
1115 
1116     FlattenableUtils::write(buffer, size, static_cast<uint32_t>(t.size()));
1117     for (size_t r = 0; r < t.size(); ++r) {
1118         ::android::Rect rect(
1119                 static_cast<int32_t>(t[r].left),
1120                 static_cast<int32_t>(t[r].top),
1121                 static_cast<int32_t>(t[r].right),
1122                 static_cast<int32_t>(t[r].bottom));
1123         status_t status = rect.flatten(buffer, size);
1124         if (status != NO_ERROR) {
1125             return status;
1126         }
1127         FlattenableUtils::advance(buffer, size, sizeof(rect));
1128     }
1129     return NO_ERROR;
1130 }
1131 
1132 /**
1133  * \brief Convert `::android::Region` to `Region`.
1134  *
1135  * \param[out] t The destination `Region`.
1136  * \param[in] l The source `::android::Region`.
1137  */
1138 // convert: ::android::Region -> Region
convertTo(Region * t,::android::Region const & l)1139 bool convertTo(Region* t, ::android::Region const& l) {
1140     size_t const baseSize = l.getFlattenedSize();
1141     std::unique_ptr<uint8_t[]> baseBuffer(
1142             new (std::nothrow) uint8_t[baseSize]);
1143     if (!baseBuffer) {
1144         return false;
1145     }
1146 
1147     void* buffer = static_cast<void*>(baseBuffer.get());
1148     size_t size = baseSize;
1149     if (l.flatten(buffer, size) != NO_ERROR) {
1150         return false;
1151     }
1152 
1153     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
1154     size = baseSize;
1155     if (unflatten(t, constBuffer, size) != NO_ERROR) {
1156         return false;
1157     }
1158 
1159     return true;
1160 }
1161 
1162 /**
1163  * \brief Convert `Region` to `::android::Region`.
1164  *
1165  * \param[out] l The destination `::android::Region`.
1166  * \param[in] t The source `Region`.
1167  */
1168 // convert: Region -> ::android::Region
convertTo(::android::Region * l,Region const & t)1169 bool convertTo(::android::Region* l, Region const& t) {
1170     size_t const baseSize = getFlattenedSize(t);
1171     std::unique_ptr<uint8_t[]> baseBuffer(
1172             new (std::nothrow) uint8_t[baseSize]);
1173     if (!baseBuffer) {
1174         return false;
1175     }
1176 
1177     void* buffer = static_cast<void*>(baseBuffer.get());
1178     size_t size = baseSize;
1179     if (flatten(t, buffer, size) != NO_ERROR) {
1180         return false;
1181     }
1182 
1183     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
1184     size = baseSize;
1185     if (l->unflatten(constBuffer, size) != NO_ERROR) {
1186         return false;
1187     }
1188 
1189     return true;
1190 }
1191 
1192 // Ref: frameworks/native/libs/gui/BGraphicBufferProducer.cpp:
1193 //      BGraphicBufferProducer::QueueBufferInput
1194 
1195 /**
1196  * \brief Return a lower bound on the size of the buffer required to flatten
1197  * `HGraphicBufferProducer::QueueBufferInput`.
1198  *
1199  * \param[in] t The input `HGraphicBufferProducer::QueueBufferInput`.
1200  * \return A lower bound on the size of the flat buffer.
1201  */
minFlattenedSize(HGraphicBufferProducer::QueueBufferInput const &)1202 constexpr size_t minFlattenedSize(
1203         HGraphicBufferProducer::QueueBufferInput const& /* t */) {
1204     return sizeof(int64_t) + // timestamp
1205             sizeof(int) + // isAutoTimestamp
1206             sizeof(android_dataspace) + // dataSpace
1207             sizeof(::android::Rect) + // crop
1208             sizeof(int) + // scalingMode
1209             sizeof(uint32_t) + // transform
1210             sizeof(uint32_t) + // stickyTransform
1211             sizeof(bool); // getFrameTimestamps
1212 }
1213 
1214 /**
1215  * \brief Return the size of the buffer required to flatten
1216  * `HGraphicBufferProducer::QueueBufferInput`.
1217  *
1218  * \param[in] t The input `HGraphicBufferProducer::QueueBufferInput`.
1219  * \return The required size of the flat buffer.
1220  */
getFlattenedSize(HGraphicBufferProducer::QueueBufferInput const & t)1221 size_t getFlattenedSize(HGraphicBufferProducer::QueueBufferInput const& t) {
1222     return minFlattenedSize(t) +
1223             getFenceFlattenedSize(t.fence) +
1224             getFlattenedSize(t.surfaceDamage) +
1225             sizeof(HdrMetadata::validTypes);
1226 }
1227 
1228 /**
1229  * \brief Return the number of file descriptors contained in
1230  * `HGraphicBufferProducer::QueueBufferInput`.
1231  *
1232  * \param[in] t The input `HGraphicBufferProducer::QueueBufferInput`.
1233  * \return The number of file descriptors contained in \p t.
1234  */
getFdCount(HGraphicBufferProducer::QueueBufferInput const & t)1235 size_t getFdCount(
1236         HGraphicBufferProducer::QueueBufferInput const& t) {
1237     return getFenceFdCount(t.fence);
1238 }
1239 
1240 /**
1241  * \brief Flatten `HGraphicBufferProducer::QueueBufferInput`.
1242  *
1243  * \param[in] t The source `HGraphicBufferProducer::QueueBufferInput`.
1244  * \param[out] nh The native handle cloned from `t.fence`.
1245  * \param[in,out] buffer The pointer to the flat non-fd buffer.
1246  * \param[in,out] size The size of the flat non-fd buffer.
1247  * \param[in,out] fds The pointer to the flat fd buffer.
1248  * \param[in,out] numFds The size of the flat fd buffer.
1249  * \return `NO_ERROR` on success; other value on failure.
1250  *
1251  * This function will duplicate the file descriptor in `t.fence`. */
flatten(HGraphicBufferProducer::QueueBufferInput const & t,native_handle_t ** nh,void * & buffer,size_t & size,int * & fds,size_t & numFds)1252 status_t flatten(HGraphicBufferProducer::QueueBufferInput const& t,
1253         native_handle_t** nh,
1254         void*& buffer, size_t& size, int*& fds, size_t& numFds) {
1255     if (size < getFlattenedSize(t)) {
1256         return NO_MEMORY;
1257     }
1258 
1259     FlattenableUtils::write(buffer, size, t.timestamp);
1260     FlattenableUtils::write(buffer, size, static_cast<int>(t.isAutoTimestamp));
1261     FlattenableUtils::write(buffer, size,
1262             static_cast<android_dataspace_t>(t.dataSpace));
1263     FlattenableUtils::write(buffer, size, ::android::Rect(
1264             static_cast<int32_t>(t.crop.left),
1265             static_cast<int32_t>(t.crop.top),
1266             static_cast<int32_t>(t.crop.right),
1267             static_cast<int32_t>(t.crop.bottom)));
1268     FlattenableUtils::write(buffer, size, static_cast<int>(t.scalingMode));
1269     FlattenableUtils::write(buffer, size, t.transform);
1270     FlattenableUtils::write(buffer, size, t.stickyTransform);
1271     FlattenableUtils::write(buffer, size, t.getFrameTimestamps);
1272 
1273     *nh = t.fence.getNativeHandle() == nullptr ?
1274             nullptr : native_handle_clone(t.fence);
1275     status_t status = flattenFence(hidl_handle(*nh), buffer, size, fds, numFds);
1276     if (status != NO_ERROR) {
1277         return status;
1278     }
1279     status = flatten(t.surfaceDamage, buffer, size);
1280     if (status != NO_ERROR) {
1281         return status;
1282     }
1283     FlattenableUtils::write(buffer, size, decltype(HdrMetadata::validTypes)(0));
1284     return NO_ERROR;
1285 }
1286 
1287 /**
1288  * \brief Unflatten `HGraphicBufferProducer::QueueBufferInput`.
1289  *
1290  * \param[out] t The destination `HGraphicBufferProducer::QueueBufferInput`.
1291  * \param[out] nh The underlying native handle for `t->fence`.
1292  * \param[in,out] buffer The pointer to the flat non-fd buffer.
1293  * \param[in,out] size The size of the flat non-fd buffer.
1294  * \param[in,out] fds The pointer to the flat fd buffer.
1295  * \param[in,out] numFds The size of the flat fd buffer.
1296  * \return `NO_ERROR` on success; other value on failure.
1297  *
1298  * If the return value is `NO_ERROR` and `t->fence` contains a valid file
1299  * descriptor, \p nh will be a newly created native handle holding that file
1300  * descriptor. \p nh needs to be deleted with `native_handle_delete()`
1301  * afterwards.
1302  */
unflatten(HGraphicBufferProducer::QueueBufferInput * t,native_handle_t ** nh,void const * & buffer,size_t & size,int const * & fds,size_t & numFds)1303 status_t unflatten(
1304         HGraphicBufferProducer::QueueBufferInput* t, native_handle_t** nh,
1305         void const*& buffer, size_t& size, int const*& fds, size_t& numFds) {
1306     if (size < minFlattenedSize(*t)) {
1307         return NO_MEMORY;
1308     }
1309 
1310     FlattenableUtils::read(buffer, size, t->timestamp);
1311     int lIsAutoTimestamp;
1312     FlattenableUtils::read(buffer, size, lIsAutoTimestamp);
1313     t->isAutoTimestamp = static_cast<int32_t>(lIsAutoTimestamp);
1314     android_dataspace_t lDataSpace;
1315     FlattenableUtils::read(buffer, size, lDataSpace);
1316     t->dataSpace = static_cast<Dataspace>(lDataSpace);
1317     Rect lCrop;
1318     FlattenableUtils::read(buffer, size, lCrop);
1319     t->crop = Rect{
1320             static_cast<int32_t>(lCrop.left),
1321             static_cast<int32_t>(lCrop.top),
1322             static_cast<int32_t>(lCrop.right),
1323             static_cast<int32_t>(lCrop.bottom)};
1324     int lScalingMode;
1325     FlattenableUtils::read(buffer, size, lScalingMode);
1326     t->scalingMode = static_cast<int32_t>(lScalingMode);
1327     FlattenableUtils::read(buffer, size, t->transform);
1328     FlattenableUtils::read(buffer, size, t->stickyTransform);
1329     FlattenableUtils::read(buffer, size, t->getFrameTimestamps);
1330 
1331     status_t status = unflattenFence(&(t->fence), nh,
1332             buffer, size, fds, numFds);
1333     if (status != NO_ERROR) {
1334         return status;
1335     }
1336     // HdrMetadata ignored
1337     return unflatten(&(t->surfaceDamage), buffer, size);
1338 }
1339 
1340 /**
1341  * \brief Wrap `BGraphicBufferProducer::QueueBufferInput` in
1342  * `HGraphicBufferProducer::QueueBufferInput`.
1343  *
1344  * \param[out] t The wrapper of type
1345  * `HGraphicBufferProducer::QueueBufferInput`.
1346  * \param[out] nh The underlying native handle for `t->fence`.
1347  * \param[in] l The source `BGraphicBufferProducer::QueueBufferInput`.
1348  *
1349  * If the return value is `true` and `t->fence` contains a valid file
1350  * descriptor, \p nh will be a newly created native handle holding that file
1351  * descriptor. \p nh needs to be deleted with `native_handle_delete()`
1352  * afterwards.
1353  */
wrapAs(HGraphicBufferProducer::QueueBufferInput * t,native_handle_t ** nh,BGraphicBufferProducer::QueueBufferInput const & l)1354 bool wrapAs(
1355         HGraphicBufferProducer::QueueBufferInput* t,
1356         native_handle_t** nh,
1357         BGraphicBufferProducer::QueueBufferInput const& l) {
1358 
1359     size_t const baseSize = l.getFlattenedSize();
1360     std::unique_ptr<uint8_t[]> baseBuffer(
1361             new (std::nothrow) uint8_t[baseSize]);
1362     if (!baseBuffer) {
1363         return false;
1364     }
1365 
1366     size_t const baseNumFds = l.getFdCount();
1367     std::unique_ptr<int[]> baseFds(
1368             new (std::nothrow) int[baseNumFds]);
1369     if (!baseFds) {
1370         return false;
1371     }
1372 
1373     void* buffer = static_cast<void*>(baseBuffer.get());
1374     size_t size = baseSize;
1375     int* fds = baseFds.get();
1376     size_t numFds = baseNumFds;
1377     if (l.flatten(buffer, size, fds, numFds) != NO_ERROR) {
1378         return false;
1379     }
1380 
1381     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
1382     size = baseSize;
1383     int const* constFds = static_cast<int const*>(baseFds.get());
1384     numFds = baseNumFds;
1385     if (unflatten(t, nh, constBuffer, size, constFds, numFds) != NO_ERROR) {
1386         return false;
1387     }
1388 
1389     return true;
1390 }
1391 
1392 /**
1393  * \brief Convert `HGraphicBufferProducer::QueueBufferInput` to
1394  * `BGraphicBufferProducer::QueueBufferInput`.
1395  *
1396  * \param[out] l The destination `BGraphicBufferProducer::QueueBufferInput`.
1397  * \param[in] t The source `HGraphicBufferProducer::QueueBufferInput`.
1398  *
1399  * If `t.fence` has a valid file descriptor, it will be duplicated.
1400  */
convertTo(BGraphicBufferProducer::QueueBufferInput * l,HGraphicBufferProducer::QueueBufferInput const & t)1401 bool convertTo(
1402         BGraphicBufferProducer::QueueBufferInput* l,
1403         HGraphicBufferProducer::QueueBufferInput const& t) {
1404 
1405     size_t const baseSize = getFlattenedSize(t);
1406     std::unique_ptr<uint8_t[]> baseBuffer(
1407             new (std::nothrow) uint8_t[baseSize]);
1408     if (!baseBuffer) {
1409         return false;
1410     }
1411 
1412     size_t const baseNumFds = getFdCount(t);
1413     std::unique_ptr<int[]> baseFds(
1414             new (std::nothrow) int[baseNumFds]);
1415     if (!baseFds) {
1416         return false;
1417     }
1418 
1419     void* buffer = static_cast<void*>(baseBuffer.get());
1420     size_t size = baseSize;
1421     int* fds = baseFds.get();
1422     size_t numFds = baseNumFds;
1423     native_handle_t* nh;
1424     if (flatten(t, &nh, buffer, size, fds, numFds) != NO_ERROR) {
1425         return false;
1426     }
1427 
1428     void const* constBuffer = static_cast<void const*>(baseBuffer.get());
1429     size = baseSize;
1430     int const* constFds = static_cast<int const*>(baseFds.get());
1431     numFds = baseNumFds;
1432     if (l->unflatten(constBuffer, size, constFds, numFds) != NO_ERROR) {
1433         if (nh != nullptr) {
1434             native_handle_close(nh);
1435             native_handle_delete(nh);
1436         }
1437         return false;
1438     }
1439 
1440     native_handle_delete(nh);
1441     return true;
1442 }
1443 
1444 // Ref: frameworks/native/libs/gui/BGraphicBufferProducer.cpp:
1445 //      BGraphicBufferProducer::QueueBufferOutput
1446 
1447 /**
1448  * \brief Wrap `BGraphicBufferProducer::QueueBufferOutput` in
1449  * `HGraphicBufferProducer::QueueBufferOutput`.
1450  *
1451  * \param[out] t The wrapper of type
1452  * `HGraphicBufferProducer::QueueBufferOutput`.
1453  * \param[out] nh The array of array of native handles that are referred to by
1454  * members of \p t.
1455  * \param[in] l The source `BGraphicBufferProducer::QueueBufferOutput`.
1456  *
1457  * On success, each member of \p nh will be either `nullptr` or a newly created
1458  * native handle. All the non-`nullptr` elements must be deleted individually
1459  * with `native_handle_delete()`.
1460  */
1461 // wrap: BGraphicBufferProducer::QueueBufferOutput ->
1462 // HGraphicBufferProducer::QueueBufferOutput
wrapAs(HGraphicBufferProducer::QueueBufferOutput * t,std::vector<std::vector<native_handle_t * >> * nh,BGraphicBufferProducer::QueueBufferOutput const & l)1463 bool wrapAs(HGraphicBufferProducer::QueueBufferOutput* t,
1464         std::vector<std::vector<native_handle_t*> >* nh,
1465         BGraphicBufferProducer::QueueBufferOutput const& l) {
1466     if (!wrapAs(&(t->frameTimestamps), nh, l.frameTimestamps)) {
1467         return false;
1468     }
1469     t->width = l.width;
1470     t->height = l.height;
1471     t->transformHint = l.transformHint;
1472     t->numPendingBuffers = l.numPendingBuffers;
1473     t->nextFrameNumber = l.nextFrameNumber;
1474     t->bufferReplaced = l.bufferReplaced;
1475     return true;
1476 }
1477 
1478 /**
1479  * \brief Convert `HGraphicBufferProducer::QueueBufferOutput` to
1480  * `BGraphicBufferProducer::QueueBufferOutput`.
1481  *
1482  * \param[out] l The destination `BGraphicBufferProducer::QueueBufferOutput`.
1483  * \param[in] t The source `HGraphicBufferProducer::QueueBufferOutput`.
1484  *
1485  * This function will duplicate all file descriptors contained in \p t.
1486  */
1487 // convert: HGraphicBufferProducer::QueueBufferOutput ->
1488 // BGraphicBufferProducer::QueueBufferOutput
convertTo(BGraphicBufferProducer::QueueBufferOutput * l,HGraphicBufferProducer::QueueBufferOutput const & t)1489 bool convertTo(
1490         BGraphicBufferProducer::QueueBufferOutput* l,
1491         HGraphicBufferProducer::QueueBufferOutput const& t) {
1492     if (!convertTo(&(l->frameTimestamps), t.frameTimestamps)) {
1493         return false;
1494     }
1495     l->width = t.width;
1496     l->height = t.height;
1497     l->transformHint = t.transformHint;
1498     l->numPendingBuffers = t.numPendingBuffers;
1499     l->nextFrameNumber = t.nextFrameNumber;
1500     l->bufferReplaced = t.bufferReplaced;
1501     return true;
1502 }
1503 
1504 /**
1505  * \brief Convert `BGraphicBufferProducer::DisconnectMode` to
1506  * `HGraphicBufferProducer::DisconnectMode`.
1507  *
1508  * \param[in] l The source `BGraphicBufferProducer::DisconnectMode`.
1509  * \return The corresponding `HGraphicBufferProducer::DisconnectMode`.
1510  */
toHidlDisconnectMode(BGraphicBufferProducer::DisconnectMode l)1511 HGraphicBufferProducer::DisconnectMode toHidlDisconnectMode(
1512         BGraphicBufferProducer::DisconnectMode l) {
1513     switch (l) {
1514         case BGraphicBufferProducer::DisconnectMode::Api:
1515             return HGraphicBufferProducer::DisconnectMode::API;
1516         case BGraphicBufferProducer::DisconnectMode::AllLocal:
1517             return HGraphicBufferProducer::DisconnectMode::ALL_LOCAL;
1518     }
1519     return HGraphicBufferProducer::DisconnectMode::API;
1520 }
1521 
1522 /**
1523  * \brief Convert `HGraphicBufferProducer::DisconnectMode` to
1524  * `BGraphicBufferProducer::DisconnectMode`.
1525  *
1526  * \param[in] l The source `HGraphicBufferProducer::DisconnectMode`.
1527  * \return The corresponding `BGraphicBufferProducer::DisconnectMode`.
1528  */
toGuiDisconnectMode(HGraphicBufferProducer::DisconnectMode t)1529 BGraphicBufferProducer::DisconnectMode toGuiDisconnectMode(
1530         HGraphicBufferProducer::DisconnectMode t) {
1531     switch (t) {
1532         case HGraphicBufferProducer::DisconnectMode::API:
1533             return BGraphicBufferProducer::DisconnectMode::Api;
1534         case HGraphicBufferProducer::DisconnectMode::ALL_LOCAL:
1535             return BGraphicBufferProducer::DisconnectMode::AllLocal;
1536     }
1537     return BGraphicBufferProducer::DisconnectMode::Api;
1538 }
1539 
1540 }  // namespace conversion
1541 }  // namespace android
1542 
1543