• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 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 #define LOG_TAG "SoftwareRenderer"
18 #include <utils/Log.h>
19 
20 #include "../include/SoftwareRenderer.h"
21 #include <cutils/properties.h> // for property_get
22 #include <media/stagefright/foundation/ADebug.h>
23 #include <media/stagefright/foundation/AMessage.h>
24 #include <media/stagefright/foundation/ColorUtils.h>
25 #include <media/stagefright/SurfaceUtils.h>
26 #include <system/window.h>
27 #include <ui/Fence.h>
28 #include <ui/GraphicBufferMapper.h>
29 #include <ui/GraphicBuffer.h>
30 #include <ui/Rect.h>
31 
32 namespace android {
33 
initDstYUV(const android_ycbcr & ycbcr,int32_t cropTop,int32_t cropLeft,uint8_t ** dst_y,uint8_t ** dst_u,uint8_t ** dst_v)34 inline void initDstYUV(
35         const android_ycbcr &ycbcr, int32_t cropTop, int32_t cropLeft,
36         uint8_t **dst_y, uint8_t **dst_u, uint8_t **dst_v) {
37     *dst_y = (uint8_t *)ycbcr.y + cropTop * ycbcr.ystride + cropLeft;
38 
39     int32_t c_offset = (cropTop / 2) * ycbcr.cstride + cropLeft / 2;
40     *dst_v = (uint8_t *)ycbcr.cr + c_offset;
41     *dst_u = (uint8_t *)ycbcr.cb + c_offset;
42 }
43 
SoftwareRenderer(const sp<ANativeWindow> & nativeWindow,int32_t rotation)44 SoftwareRenderer::SoftwareRenderer(
45         const sp<ANativeWindow> &nativeWindow, int32_t rotation)
46     : mColorFormat(OMX_COLOR_FormatUnused),
47       mConverter(NULL),
48       mYUVMode(None),
49       mNativeWindow(nativeWindow),
50       mWidth(0),
51       mHeight(0),
52       mStride(0),
53       mCropLeft(0),
54       mCropTop(0),
55       mCropRight(0),
56       mCropBottom(0),
57       mCropWidth(0),
58       mCropHeight(0),
59       mRotationDegrees(rotation),
60       mDataSpace(HAL_DATASPACE_UNKNOWN) {
61     memset(&mHDRStaticInfo, 0, sizeof(mHDRStaticInfo));
62 }
63 
~SoftwareRenderer()64 SoftwareRenderer::~SoftwareRenderer() {
65     delete mConverter;
66     mConverter = NULL;
67 }
68 
resetFormatIfChanged(const sp<AMessage> & format,size_t numOutputBuffers)69 void SoftwareRenderer::resetFormatIfChanged(
70         const sp<AMessage> &format, size_t numOutputBuffers) {
71     CHECK(format != NULL);
72 
73     int32_t colorFormatNew;
74     CHECK(format->findInt32("color-format", &colorFormatNew));
75 
76     int32_t widthNew, heightNew, strideNew;
77     CHECK(format->findInt32("width", &widthNew));
78     CHECK(format->findInt32("slice-height", &heightNew));
79     CHECK(format->findInt32("stride", &strideNew));
80 
81     int32_t cropLeftNew, cropTopNew, cropRightNew, cropBottomNew;
82     if (!format->findRect(
83             "crop", &cropLeftNew, &cropTopNew, &cropRightNew, &cropBottomNew)) {
84         cropLeftNew = cropTopNew = 0;
85         cropRightNew = widthNew - 1;
86         cropBottomNew = heightNew - 1;
87     }
88 
89     // The native window buffer format for high-bitdepth content could
90     // depend on the dataspace also.
91     android_dataspace dataSpace;
92     bool dataSpaceChangedForPlanar16 = false;
93     if (colorFormatNew == OMX_COLOR_FormatYUV420Planar16
94             && format->findInt32("android._dataspace", (int32_t *)&dataSpace)
95             && dataSpace != mDataSpace) {
96         // Do not modify mDataSpace here, it's only modified at last
97         // when we do native_window_set_buffers_data_space().
98         dataSpaceChangedForPlanar16 = true;
99     }
100 
101     if (static_cast<int32_t>(mColorFormat) == colorFormatNew &&
102         mWidth == widthNew &&
103         mHeight == heightNew &&
104         mCropLeft == cropLeftNew &&
105         mCropTop == cropTopNew &&
106         mCropRight == cropRightNew &&
107         mCropBottom == cropBottomNew &&
108         !dataSpaceChangedForPlanar16) {
109         // Nothing changed, no need to reset renderer.
110         return;
111     }
112 
113     mColorFormat = static_cast<OMX_COLOR_FORMATTYPE>(colorFormatNew);
114     mWidth = widthNew;
115     mHeight = heightNew;
116     mStride = strideNew;
117     mCropLeft = cropLeftNew;
118     mCropTop = cropTopNew;
119     mCropRight = cropRightNew;
120     mCropBottom = cropBottomNew;
121 
122     mCropWidth = mCropRight - mCropLeft + 1;
123     mCropHeight = mCropBottom - mCropTop + 1;
124 
125     // by default convert everything to RGB565
126     int halFormat = HAL_PIXEL_FORMAT_RGB_565;
127     size_t bufWidth = mCropWidth;
128     size_t bufHeight = mCropHeight;
129 
130     // hardware has YUV12 and RGBA8888 support, so convert known formats
131     {
132         switch (mColorFormat) {
133             case OMX_COLOR_FormatYUV420Planar:
134             case OMX_COLOR_FormatYUV420SemiPlanar:
135             case OMX_TI_COLOR_FormatYUV420PackedSemiPlanar:
136             {
137                 halFormat = HAL_PIXEL_FORMAT_YV12;
138                 bufWidth = (mCropWidth + 1) & ~1;
139                 bufHeight = (mCropHeight + 1) & ~1;
140                 break;
141             }
142             case OMX_COLOR_Format24bitRGB888:
143             {
144                 halFormat = HAL_PIXEL_FORMAT_RGB_888;
145                 bufWidth = (mCropWidth + 1) & ~1;
146                 bufHeight = (mCropHeight + 1) & ~1;
147                 break;
148             }
149             case OMX_COLOR_Format32bitARGB8888:
150             case OMX_COLOR_Format32BitRGBA8888:
151             {
152                 halFormat = HAL_PIXEL_FORMAT_RGBA_8888;
153                 bufWidth = (mCropWidth + 1) & ~1;
154                 bufHeight = (mCropHeight + 1) & ~1;
155                 break;
156             }
157             case OMX_COLOR_FormatYUV420Planar16:
158             {
159                 if (((dataSpace & HAL_DATASPACE_STANDARD_MASK) == HAL_DATASPACE_STANDARD_BT2020)
160                  && ((dataSpace & HAL_DATASPACE_TRANSFER_MASK) == HAL_DATASPACE_TRANSFER_ST2084)) {
161                     // Here we would convert OMX_COLOR_FormatYUV420Planar16 into
162                     // OMX_COLOR_FormatYUV444Y410, and put it inside a buffer with
163                     // format HAL_PIXEL_FORMAT_RGBA_1010102. Surfaceflinger will
164                     // use render engine to convert it to RGB if needed.
165                     halFormat = HAL_PIXEL_FORMAT_RGBA_1010102;
166                 } else {
167                     halFormat = HAL_PIXEL_FORMAT_YV12;
168                 }
169                 bufWidth = (mCropWidth + 1) & ~1;
170                 bufHeight = (mCropHeight + 1) & ~1;
171                 break;
172             }
173             default:
174             {
175                 break;
176             }
177         }
178     }
179 
180     if (halFormat == HAL_PIXEL_FORMAT_RGB_565) {
181         mConverter = new ColorConverter(
182                 mColorFormat, OMX_COLOR_Format16bitRGB565);
183         CHECK(mConverter->isValid());
184     } else if (halFormat == HAL_PIXEL_FORMAT_RGBA_1010102) {
185         mConverter = new ColorConverter(
186                 mColorFormat, OMX_COLOR_FormatYUV444Y410);
187         CHECK(mConverter->isValid());
188     }
189 
190     CHECK(mNativeWindow != NULL);
191     CHECK(mCropWidth > 0);
192     CHECK(mCropHeight > 0);
193     CHECK(mConverter == NULL || mConverter->isValid());
194 
195     CHECK_EQ(0,
196             native_window_set_usage(
197             mNativeWindow.get(),
198             GRALLOC_USAGE_SW_READ_NEVER | GRALLOC_USAGE_SW_WRITE_RARELY
199             | GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_EXTERNAL_DISP));
200 
201     CHECK_EQ(0,
202             native_window_set_scaling_mode(
203             mNativeWindow.get(),
204             NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW));
205 
206     // Width must be multiple of 32???
207     CHECK_EQ(0, native_window_set_buffers_dimensions(
208                 mNativeWindow.get(),
209                 bufWidth,
210                 bufHeight));
211     CHECK_EQ(0, native_window_set_buffers_format(
212                 mNativeWindow.get(),
213                 halFormat));
214     if (OK != native_window_set_buffer_count(
215                 mNativeWindow.get(), numOutputBuffers + 4)) {
216         ALOGE("Failed to set native window buffer count to (%zu + 4)",
217                 numOutputBuffers);
218     }
219 
220     // NOTE: native window uses extended right-bottom coordinate
221     android_native_rect_t crop;
222     crop.left = mCropLeft;
223     crop.top = mCropTop;
224     crop.right = mCropRight + 1;
225     crop.bottom = mCropBottom + 1;
226     ALOGV("setting crop: [%d, %d, %d, %d] for size [%zu, %zu]",
227           crop.left, crop.top, crop.right, crop.bottom, bufWidth, bufHeight);
228 
229     CHECK_EQ(0, native_window_set_crop(mNativeWindow.get(), &crop));
230 
231     int32_t rotationDegrees;
232     if (!format->findInt32("rotation-degrees", &rotationDegrees)) {
233         rotationDegrees = mRotationDegrees;
234     }
235     uint32_t transform;
236     switch (rotationDegrees) {
237         case 0: transform = 0; break;
238         case 90: transform = HAL_TRANSFORM_ROT_90; break;
239         case 180: transform = HAL_TRANSFORM_ROT_180; break;
240         case 270: transform = HAL_TRANSFORM_ROT_270; break;
241         default: transform = 0; break;
242     }
243 
244     CHECK_EQ(0, native_window_set_buffers_transform(
245                 mNativeWindow.get(), transform));
246 }
247 
clearTracker()248 void SoftwareRenderer::clearTracker() {
249     mRenderTracker.clear(-1 /* lastRenderTimeNs */);
250 }
251 
render(const void * data,size_t,int64_t mediaTimeUs,nsecs_t renderTimeNs,size_t numOutputBuffers,const sp<AMessage> & format)252 std::list<FrameRenderTracker::Info> SoftwareRenderer::render(
253         const void *data, size_t , int64_t mediaTimeUs, nsecs_t renderTimeNs,
254         size_t numOutputBuffers, const sp<AMessage>& format) {
255     resetFormatIfChanged(format, numOutputBuffers);
256     FrameRenderTracker::Info *info = NULL;
257 
258     ANativeWindowBuffer *buf;
259     int fenceFd = -1;
260     int err = mNativeWindow->dequeueBuffer(mNativeWindow.get(), &buf, &fenceFd);
261     if (err == 0 && fenceFd >= 0) {
262         info = mRenderTracker.updateInfoForDequeuedBuffer(buf, fenceFd, 0);
263         sp<Fence> fence = new Fence(fenceFd);
264         err = fence->waitForever("SoftwareRenderer::render");
265     }
266     if (err != 0) {
267         ALOGW("Surface::dequeueBuffer returned error %d", err);
268         // complete (drop) dequeued frame if fence wait failed; otherwise,
269         // this returns an empty list as no frames should have rendered and not yet returned.
270         return mRenderTracker.checkFencesAndGetRenderedFrames(info, false /* dropIncomplete */);
271     }
272 
273     GraphicBufferMapper &mapper = GraphicBufferMapper::get();
274 
275     Rect bounds(mCropWidth, mCropHeight);
276 
277     void *dst = NULL;
278     struct android_ycbcr ycbcr;
279     if ( !mConverter &&
280          (mColorFormat == OMX_COLOR_FormatYUV420Planar ||
281          mColorFormat == OMX_COLOR_FormatYUV420SemiPlanar ||
282          mColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar ||
283          mColorFormat == OMX_COLOR_FormatYUV420Planar16)) {
284         CHECK_EQ(0, mapper.lockYCbCr(buf->handle,
285                 GRALLOC_USAGE_SW_READ_NEVER | GRALLOC_USAGE_SW_WRITE_RARELY,
286                 bounds, &ycbcr));
287     } else {
288         CHECK_EQ(0, mapper.lock(buf->handle,
289                 GRALLOC_USAGE_SW_READ_NEVER | GRALLOC_USAGE_SW_WRITE_RARELY,
290                 bounds, &dst));
291     }
292 
293     // TODO move the other conversions also into ColorConverter, and
294     // fix cropping issues (when mCropLeft/Top != 0 or mWidth != mCropWidth)
295     if (mConverter) {
296         mConverter->convert(
297                 data,
298                 mWidth, mHeight, mStride,
299                 mCropLeft, mCropTop, mCropRight, mCropBottom,
300                 dst,
301                 buf->stride, buf->height, 0,
302                 0, 0, mCropWidth - 1, mCropHeight - 1);
303     } else if (mColorFormat == OMX_COLOR_FormatYUV420Planar) {
304         const uint8_t *src_y = (const uint8_t *)data + mCropTop * mStride + mCropLeft;
305         const uint8_t *src_u = (const uint8_t *)data + mStride * mHeight + mCropTop * mStride / 4;
306         const uint8_t *src_v = (const uint8_t *)src_u + mStride * mHeight / 4;
307 
308         uint8_t *dst_y, *dst_u, *dst_v;
309         initDstYUV(ycbcr, mCropTop, mCropLeft, &dst_y, &dst_u, &dst_v);
310 
311         for (int y = 0; y < mCropHeight; ++y) {
312             memcpy(dst_y, src_y, mCropWidth);
313 
314             src_y += mStride;
315             dst_y += ycbcr.ystride;
316         }
317 
318         for (int y = 0; y < (mCropHeight + 1) / 2; ++y) {
319             memcpy(dst_u, src_u, (mCropWidth + 1) / 2);
320             memcpy(dst_v, src_v, (mCropWidth + 1) / 2);
321 
322             src_u += mStride / 2;
323             src_v += mStride / 2;
324             dst_u += ycbcr.cstride;
325             dst_v += ycbcr.cstride;
326         }
327     } else if (mColorFormat == OMX_COLOR_FormatYUV420Planar16) {
328         const uint8_t *src_y = (const uint8_t *)data + mCropTop * mStride + mCropLeft * 2;
329         const uint8_t *src_u = (const uint8_t *)data + mStride * mHeight + mCropTop * mStride / 4;
330         const uint8_t *src_v = (const uint8_t *)src_u + mStride * mHeight / 4;
331 
332         uint8_t *dst_y, *dst_u, *dst_v;
333         initDstYUV(ycbcr, mCropTop, mCropLeft, &dst_y, &dst_u, &dst_v);
334 
335         for (int y = 0; y < mCropHeight; ++y) {
336             for (int x = 0; x < mCropWidth; ++x) {
337                 dst_y[x] = (uint8_t)(((uint16_t *)src_y)[x] >> 2);
338             }
339 
340             src_y += mStride;
341             dst_y += ycbcr.ystride;
342         }
343 
344         for (int y = 0; y < (mCropHeight + 1) / 2; ++y) {
345             for (int x = 0; x < (mCropWidth + 1) / 2; ++x) {
346                 dst_u[x] = (uint8_t)(((uint16_t *)src_u)[x] >> 2);
347                 dst_v[x] = (uint8_t)(((uint16_t *)src_v)[x] >> 2);
348             }
349 
350             src_u += mStride / 2;
351             src_v += mStride / 2;
352             dst_u += ycbcr.cstride;
353             dst_v += ycbcr.cstride;
354         }
355     } else if (mColorFormat == OMX_TI_COLOR_FormatYUV420PackedSemiPlanar
356             || mColorFormat == OMX_COLOR_FormatYUV420SemiPlanar) {
357         const uint8_t *src_y = (const uint8_t *)data;
358         const uint8_t *src_uv = (const uint8_t *)data
359                 + mWidth * mHeight;
360 
361         src_y += mCropLeft + mCropTop * mWidth;
362         src_uv += (mCropLeft + mCropTop * mWidth) / 2;
363 
364         uint8_t *dst_y, *dst_u, *dst_v;
365         initDstYUV(ycbcr, mCropTop, mCropLeft, &dst_y, &dst_u, &dst_v);
366 
367         for (int y = 0; y < mCropHeight; ++y) {
368             memcpy(dst_y, src_y, mCropWidth);
369 
370             src_y += mWidth;
371             dst_y += ycbcr.ystride;
372         }
373 
374         for (int y = 0; y < (mCropHeight + 1) / 2; ++y) {
375             size_t tmp = (mCropWidth + 1) / 2;
376             for (size_t x = 0; x < tmp; ++x) {
377                 dst_u[x] = src_uv[2 * x];
378                 dst_v[x] = src_uv[2 * x + 1];
379             }
380 
381             src_uv += mWidth;
382             dst_u += ycbcr.cstride;
383             dst_v += ycbcr.cstride;
384         }
385     } else if (mColorFormat == OMX_COLOR_Format24bitRGB888) {
386         uint8_t* srcPtr = (uint8_t*)data + mWidth * mCropTop * 3 + mCropLeft * 3;
387         uint8_t* dstPtr = (uint8_t*)dst + buf->stride * mCropTop * 3 + mCropLeft * 3;
388 
389         for (size_t y = 0; y < (size_t)mCropHeight; ++y) {
390             memcpy(dstPtr, srcPtr, mCropWidth * 3);
391             srcPtr += mWidth * 3;
392             dstPtr += buf->stride * 3;
393         }
394     } else if (mColorFormat == OMX_COLOR_Format32bitARGB8888) {
395         uint8_t *srcPtr, *dstPtr;
396 
397         for (size_t y = 0; y < (size_t)mCropHeight; ++y) {
398             srcPtr = (uint8_t*)data + mWidth * 4 * (y + mCropTop) + mCropLeft * 4;
399             dstPtr = (uint8_t*)dst + buf->stride * 4 * (y + mCropTop) + mCropLeft * 4;
400             for (size_t x = 0; x < (size_t)mCropWidth; ++x) {
401                 uint8_t a = *srcPtr++;
402                 for (size_t i = 0; i < 3; ++i) {   // copy RGB
403                     *dstPtr++ = *srcPtr++;
404                 }
405                 *dstPtr++ = a;  // alpha last (ARGB to RGBA)
406             }
407         }
408     } else if (mColorFormat == OMX_COLOR_Format32BitRGBA8888) {
409         uint8_t* srcPtr = (uint8_t*)data + mWidth * mCropTop * 4 + mCropLeft * 4;
410         uint8_t* dstPtr = (uint8_t*)dst + buf->stride * mCropTop * 4 + mCropLeft * 4;
411 
412         for (size_t y = 0; y < (size_t)mCropHeight; ++y) {
413             memcpy(dstPtr, srcPtr, mCropWidth * 4);
414             srcPtr += mWidth * 4;
415             dstPtr += buf->stride * 4;
416         }
417     } else {
418         LOG_ALWAYS_FATAL("bad color format %#x", mColorFormat);
419     }
420 
421 skip_copying:
422     CHECK_EQ(0, mapper.unlock(buf->handle));
423 
424     if (renderTimeNs >= 0) {
425         if ((err = native_window_set_buffers_timestamp(mNativeWindow.get(),
426                 renderTimeNs)) != 0) {
427             ALOGW("Surface::set_buffers_timestamp returned error %d", err);
428         }
429     }
430 
431     // TODO: propagate color aspects to software renderer to allow better
432     // color conversion to RGB. For now, just mark dataspace for YUV rendering.
433     android_dataspace dataSpace;
434     if (format->findInt32("android._dataspace", (int32_t *)&dataSpace) && dataSpace != mDataSpace) {
435         mDataSpace = dataSpace;
436 
437         if (mConverter != NULL && mConverter->isDstRGB()) {
438             // graphics only supports full range RGB. ColorConverter should have
439             // converted any YUV to full range.
440             dataSpace = (android_dataspace)
441                     ((dataSpace & ~HAL_DATASPACE_RANGE_MASK) | HAL_DATASPACE_RANGE_FULL);
442         }
443 
444         ALOGD("setting dataspace on output surface to %#x", dataSpace);
445         if ((err = native_window_set_buffers_data_space(mNativeWindow.get(), dataSpace))) {
446             ALOGW("failed to set dataspace on surface (%d)", err);
447         }
448     }
449     if (format->contains("hdr-static-info")) {
450         HDRStaticInfo info;
451         if (ColorUtils::getHDRStaticInfoFromFormat(format, &info)
452             && memcmp(&mHDRStaticInfo, &info, sizeof(info))) {
453             setNativeWindowHdrMetadata(mNativeWindow.get(), &info);
454             mHDRStaticInfo = info;
455         }
456     }
457 
458     if ((err = mNativeWindow->queueBuffer(mNativeWindow.get(), buf, -1)) != 0) {
459         ALOGW("Surface::queueBuffer returned error %d", err);
460     } else {
461         mRenderTracker.onFrameQueued(mediaTimeUs, (GraphicBuffer *)buf, Fence::NO_FENCE);
462     }
463 
464     buf = NULL;
465     return mRenderTracker.checkFencesAndGetRenderedFrames(info, info != NULL /* dropIncomplete */);
466 }
467 
468 }  // namespace android
469