| /* |
| * Copyright (C) 2010 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #define LOG_TAG "GLConsumerUtils" |
| //#define LOG_NDEBUG 0 |
| |
| #include <gui/GLConsumer.h> |
| #include <math/mat4.h> |
| #include <system/window.h> |
| #include <utils/Log.h> |
| |
| namespace android { |
| |
| void GLConsumer::computeTransformMatrix(float outTransform[16], |
| const sp<GraphicBuffer>& buf, const Rect& cropRect, uint32_t transform, |
| bool filtering) { |
| // Transform matrices |
| static const mat4 mtxFlipH( |
| -1, 0, 0, 0, |
| 0, 1, 0, 0, |
| 0, 0, 1, 0, |
| 1, 0, 0, 1 |
| ); |
| static const mat4 mtxFlipV( |
| 1, 0, 0, 0, |
| 0, -1, 0, 0, |
| 0, 0, 1, 0, |
| 0, 1, 0, 1 |
| ); |
| static const mat4 mtxRot90( |
| 0, 1, 0, 0, |
| -1, 0, 0, 0, |
| 0, 0, 1, 0, |
| 1, 0, 0, 1 |
| ); |
| |
| mat4 xform; |
| if (transform & NATIVE_WINDOW_TRANSFORM_FLIP_H) { |
| xform *= mtxFlipH; |
| } |
| if (transform & NATIVE_WINDOW_TRANSFORM_FLIP_V) { |
| xform *= mtxFlipV; |
| } |
| if (transform & NATIVE_WINDOW_TRANSFORM_ROT_90) { |
| xform *= mtxRot90; |
| } |
| |
| if (!cropRect.isEmpty()) { |
| float tx = 0.0f, ty = 0.0f, sx = 1.0f, sy = 1.0f; |
| float bufferWidth = buf->getWidth(); |
| float bufferHeight = buf->getHeight(); |
| float shrinkAmount = 0.0f; |
| if (filtering) { |
| // In order to prevent bilinear sampling beyond the edge of the |
| // crop rectangle we may need to shrink it by 2 texels in each |
| // dimension. Normally this would just need to take 1/2 a texel |
| // off each end, but because the chroma channels of YUV420 images |
| // are subsampled we may need to shrink the crop region by a whole |
| // texel on each side. |
| switch (buf->getPixelFormat()) { |
| case PIXEL_FORMAT_RGBA_8888: |
| case PIXEL_FORMAT_RGBX_8888: |
| case PIXEL_FORMAT_RGBA_FP16: |
| case PIXEL_FORMAT_RGBA_1010102: |
| case PIXEL_FORMAT_RGB_888: |
| case PIXEL_FORMAT_RGB_565: |
| case PIXEL_FORMAT_BGRA_8888: |
| // We know there's no subsampling of any channels, so we |
| // only need to shrink by a half a pixel. |
| shrinkAmount = 0.5; |
| break; |
| |
| default: |
| // If we don't recognize the format, we must assume the |
| // worst case (that we care about), which is YUV420. |
| shrinkAmount = 1.0; |
| break; |
| } |
| } |
| |
| // Only shrink the dimensions that are not the size of the buffer. |
| if (cropRect.width() < bufferWidth) { |
| tx = (float(cropRect.left) + shrinkAmount) / bufferWidth; |
| sx = (float(cropRect.width()) - (2.0f * shrinkAmount)) / |
| bufferWidth; |
| } |
| if (cropRect.height() < bufferHeight) { |
| ty = (float(bufferHeight - cropRect.bottom) + shrinkAmount) / |
| bufferHeight; |
| sy = (float(cropRect.height()) - (2.0f * shrinkAmount)) / |
| bufferHeight; |
| } |
| |
| mat4 crop( |
| sx, 0, 0, 0, |
| 0, sy, 0, 0, |
| 0, 0, 1, 0, |
| tx, ty, 0, 1 |
| ); |
| xform = crop * xform; |
| } |
| |
| // GLConsumer uses the GL convention where (0, 0) is the bottom-left |
| // corner and (1, 1) is the top-right corner. Add an additional vertical |
| // flip after all other transforms to map from GL convention to buffer |
| // queue memory layout, where (0, 0) is the top-left corner. |
| xform = mtxFlipV * xform; |
| |
| memcpy(outTransform, xform.asArray(), sizeof(xform)); |
| } |
| |
| }; // namespace android |