Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 1 | /* |
Shuzhen Wang | c28189a | 2017-11-27 23:05:10 -0800 | [diff] [blame] | 2 | * Copyright (C) 2013-2018 The Android Open Source Project |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 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 | #ifndef ANDROID_SERVERS_CAMERA3_OUTPUT_STREAM_H |
| 18 | #define ANDROID_SERVERS_CAMERA3_OUTPUT_STREAM_H |
| 19 | |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 20 | #include <mutex> |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 21 | #include <utils/RefBase.h> |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 22 | #include <gui/IProducerListener.h> |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 23 | #include <gui/Surface.h> |
Shuzhen Wang | 00abbeb | 2022-02-25 17:14:42 -0800 | [diff] [blame] | 24 | #include <gui/DisplayEventReceiver.h> |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 25 | |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 26 | #include "utils/IPCTransport.h" |
Shuzhen Wang | 686f644 | 2017-06-20 16:16:04 -0700 | [diff] [blame] | 27 | #include "utils/LatencyHistogram.h" |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 28 | #include "Camera3Stream.h" |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 29 | #include "Camera3IOStreamBase.h" |
Igor Murashkin | 2fba584 | 2013-04-22 14:03:54 -0700 | [diff] [blame] | 30 | #include "Camera3OutputStreamInterface.h" |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 31 | #include "Camera3BufferManager.h" |
Shuzhen Wang | ba92d77 | 2022-04-11 11:47:24 -0700 | [diff] [blame] | 32 | #include "PreviewFrameSpacer.h" |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 33 | |
| 34 | namespace android { |
| 35 | |
| 36 | namespace camera3 { |
| 37 | |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 38 | class Camera3BufferManager; |
| 39 | |
| 40 | /** |
| 41 | * Stream info structure that holds the necessary stream info for buffer manager to use for |
| 42 | * buffer allocation and management. |
| 43 | */ |
| 44 | struct StreamInfo { |
| 45 | int streamId; |
| 46 | int streamSetId; |
| 47 | uint32_t width; |
| 48 | uint32_t height; |
| 49 | uint32_t format; |
| 50 | android_dataspace dataSpace; |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 51 | uint64_t combinedUsage; |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 52 | size_t totalBufferCount; |
| 53 | bool isConfigured; |
Shuzhen Wang | 83bff12 | 2020-11-20 15:51:39 -0800 | [diff] [blame] | 54 | bool isMultiRes; |
Chih-Hung Hsieh | d19d994 | 2016-08-29 14:21:14 -0700 | [diff] [blame] | 55 | explicit StreamInfo(int id = CAMERA3_STREAM_ID_INVALID, |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 56 | int setId = CAMERA3_STREAM_SET_ID_INVALID, |
| 57 | uint32_t w = 0, |
| 58 | uint32_t h = 0, |
| 59 | uint32_t fmt = 0, |
| 60 | android_dataspace ds = HAL_DATASPACE_UNKNOWN, |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 61 | uint64_t usage = 0, |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 62 | size_t bufferCount = 0, |
Shuzhen Wang | 83bff12 | 2020-11-20 15:51:39 -0800 | [diff] [blame] | 63 | bool configured = false, |
| 64 | bool multiRes = false) : |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 65 | streamId(id), |
| 66 | streamSetId(setId), |
| 67 | width(w), |
| 68 | height(h), |
| 69 | format(fmt), |
| 70 | dataSpace(ds), |
| 71 | combinedUsage(usage), |
| 72 | totalBufferCount(bufferCount), |
Shuzhen Wang | 83bff12 | 2020-11-20 15:51:39 -0800 | [diff] [blame] | 73 | isConfigured(configured), |
| 74 | isMultiRes(multiRes) {} |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 75 | }; |
| 76 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 77 | /** |
| 78 | * A class for managing a single stream of output data from the camera device. |
| 79 | */ |
Igor Murashkin | 2fba584 | 2013-04-22 14:03:54 -0700 | [diff] [blame] | 80 | class Camera3OutputStream : |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 81 | public Camera3IOStreamBase, |
Igor Murashkin | 2fba584 | 2013-04-22 14:03:54 -0700 | [diff] [blame] | 82 | public Camera3OutputStreamInterface { |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 83 | public: |
| 84 | /** |
| 85 | * Set up a stream for formats that have 2 dimensions, such as RAW and YUV. |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 86 | * A valid stream set id needs to be set to support buffer sharing between multiple |
| 87 | * streams. |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 88 | */ |
Eino-Ville Talvala | 727d172 | 2015-06-09 13:44:19 -0700 | [diff] [blame] | 89 | Camera3OutputStream(int id, sp<Surface> consumer, |
Eino-Ville Talvala | 3d82c0d | 2015-02-23 15:19:19 -0800 | [diff] [blame] | 90 | uint32_t width, uint32_t height, int format, |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 91 | android_dataspace dataSpace, camera_stream_rotation_t rotation, |
Shuzhen Wang | c28189a | 2017-11-27 23:05:10 -0800 | [diff] [blame] | 92 | nsecs_t timestampOffset, const String8& physicalCameraId, |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 93 | const std::unordered_set<int32_t> &sensorPixelModesUsed, IPCTransport transport, |
Emilian Peev | 2295df7 | 2021-11-12 18:14:10 -0800 | [diff] [blame] | 94 | int setId = CAMERA3_STREAM_SET_ID_INVALID, bool isMultiResolution = false, |
Emilian Peev | c81a759 | 2022-02-14 17:38:18 -0800 | [diff] [blame] | 95 | int64_t dynamicProfile = ANDROID_REQUEST_AVAILABLE_DYNAMIC_RANGE_PROFILES_MAP_STANDARD, |
Shuzhen Wang | 8ed1e87 | 2022-03-08 16:34:33 -0800 | [diff] [blame] | 96 | int64_t streamUseCase = ANDROID_SCALER_AVAILABLE_STREAM_USE_CASES_DEFAULT, |
Shuzhen Wang | e420892 | 2022-02-01 16:52:48 -0800 | [diff] [blame] | 97 | bool deviceTimeBaseIsRealtime = false, |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 98 | int timestampBase = OutputConfiguration::TIMESTAMP_BASE_DEFAULT, |
| 99 | int mirrorMode = OutputConfiguration::MIRROR_MODE_AUTO); |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 100 | /** |
| 101 | * Set up a stream for formats that have a variable buffer size for the same |
| 102 | * dimensions, such as compressed JPEG. |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 103 | * A valid stream set id needs to be set to support buffer sharing between multiple |
| 104 | * streams. |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 105 | */ |
Eino-Ville Talvala | 727d172 | 2015-06-09 13:44:19 -0700 | [diff] [blame] | 106 | Camera3OutputStream(int id, sp<Surface> consumer, |
Eino-Ville Talvala | 3d82c0d | 2015-02-23 15:19:19 -0800 | [diff] [blame] | 107 | uint32_t width, uint32_t height, size_t maxSize, int format, |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 108 | android_dataspace dataSpace, camera_stream_rotation_t rotation, |
Shuzhen Wang | c28189a | 2017-11-27 23:05:10 -0800 | [diff] [blame] | 109 | nsecs_t timestampOffset, const String8& physicalCameraId, |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 110 | const std::unordered_set<int32_t> &sensorPixelModesUsed, IPCTransport transport, |
Emilian Peev | 2295df7 | 2021-11-12 18:14:10 -0800 | [diff] [blame] | 111 | int setId = CAMERA3_STREAM_SET_ID_INVALID, bool isMultiResolution = false, |
Emilian Peev | c81a759 | 2022-02-14 17:38:18 -0800 | [diff] [blame] | 112 | int64_t dynamicProfile = ANDROID_REQUEST_AVAILABLE_DYNAMIC_RANGE_PROFILES_MAP_STANDARD, |
Shuzhen Wang | 8ed1e87 | 2022-03-08 16:34:33 -0800 | [diff] [blame] | 113 | int64_t streamUseCase = ANDROID_SCALER_AVAILABLE_STREAM_USE_CASES_DEFAULT, |
Shuzhen Wang | e420892 | 2022-02-01 16:52:48 -0800 | [diff] [blame] | 114 | bool deviceTimeBaseIsRealtime = false, |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 115 | int timestampBase = OutputConfiguration::TIMESTAMP_BASE_DEFAULT, |
| 116 | int mirrorMode = OutputConfiguration::MIRROR_MODE_AUTO); |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 117 | /** |
| 118 | * Set up a stream with deferred consumer for formats that have 2 dimensions, such as |
| 119 | * RAW and YUV. The consumer must be set before using this stream for output. A valid |
| 120 | * stream set id needs to be set to support buffer sharing between multiple streams. |
| 121 | */ |
| 122 | Camera3OutputStream(int id, uint32_t width, uint32_t height, int format, |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 123 | uint64_t consumerUsage, android_dataspace dataSpace, |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 124 | camera_stream_rotation_t rotation, nsecs_t timestampOffset, |
Shuzhen Wang | c28189a | 2017-11-27 23:05:10 -0800 | [diff] [blame] | 125 | const String8& physicalCameraId, |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 126 | const std::unordered_set<int32_t> &sensorPixelModesUsed, IPCTransport transport, |
Emilian Peev | 2295df7 | 2021-11-12 18:14:10 -0800 | [diff] [blame] | 127 | int setId = CAMERA3_STREAM_SET_ID_INVALID, bool isMultiResolution = false, |
Emilian Peev | c81a759 | 2022-02-14 17:38:18 -0800 | [diff] [blame] | 128 | int64_t dynamicProfile = ANDROID_REQUEST_AVAILABLE_DYNAMIC_RANGE_PROFILES_MAP_STANDARD, |
Shuzhen Wang | 8ed1e87 | 2022-03-08 16:34:33 -0800 | [diff] [blame] | 129 | int64_t streamUseCase = ANDROID_SCALER_AVAILABLE_STREAM_USE_CASES_DEFAULT, |
Shuzhen Wang | e420892 | 2022-02-01 16:52:48 -0800 | [diff] [blame] | 130 | bool deviceTimeBaseIsRealtime = false, |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 131 | int timestampBase = OutputConfiguration::TIMESTAMP_BASE_DEFAULT, |
| 132 | int mirrorMode = OutputConfiguration::MIRROR_MODE_AUTO); |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 133 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 134 | virtual ~Camera3OutputStream(); |
| 135 | |
| 136 | /** |
| 137 | * Camera3Stream interface |
| 138 | */ |
| 139 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 140 | virtual void dump(int fd, const Vector<String16> &args) const; |
| 141 | |
| 142 | /** |
| 143 | * Set the transform on the output stream; one of the |
| 144 | * HAL_TRANSFORM_* / NATIVE_WINDOW_TRANSFORM_* constants. |
| 145 | */ |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 146 | status_t setTransform(int transform, bool mayChangeMirror); |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 147 | |
Chien-Yu Chen | 85a6455 | 2015-08-28 15:46:12 -0700 | [diff] [blame] | 148 | /** |
| 149 | * Return if this output stream is for video encoding. |
| 150 | */ |
| 151 | bool isVideoStream() const; |
Shuzhen Wang | 13a6963 | 2016-01-26 09:51:07 -0800 | [diff] [blame] | 152 | /** |
| 153 | * Return if this output stream is consumed by hardware composer. |
| 154 | */ |
| 155 | bool isConsumedByHWComposer() const; |
Chien-Yu Chen | 85a6455 | 2015-08-28 15:46:12 -0700 | [diff] [blame] | 156 | |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 157 | /** |
Zhijun He | f0645c1 | 2016-08-02 00:58:11 -0700 | [diff] [blame] | 158 | * Return if this output stream is consumed by hardware texture. |
| 159 | */ |
| 160 | bool isConsumedByHWTexture() const; |
| 161 | |
| 162 | /** |
Shuzhen Wang | fe8a2a3 | 2022-05-10 18:18:54 -0700 | [diff] [blame] | 163 | * Return if this output stream is consumed by CPU. |
| 164 | */ |
| 165 | bool isConsumedByCPU() const; |
| 166 | |
| 167 | /** |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 168 | * Return if the consumer configuration of this stream is deferred. |
| 169 | */ |
Shuzhen Wang | 0129d52 | 2016-10-30 22:43:41 -0700 | [diff] [blame] | 170 | virtual bool isConsumerConfigurationDeferred(size_t surface_id) const; |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 171 | |
| 172 | /** |
Shuzhen Wang | 758c215 | 2017-01-10 18:26:18 -0800 | [diff] [blame] | 173 | * Set the consumer surfaces to the output stream. |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 174 | */ |
Shuzhen Wang | 758c215 | 2017-01-10 18:26:18 -0800 | [diff] [blame] | 175 | virtual status_t setConsumers(const std::vector<sp<Surface>>& consumers); |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 176 | |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 177 | class BufferProducerListener : public SurfaceListener { |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 178 | public: |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 179 | BufferProducerListener(wp<Camera3OutputStream> parent, bool needsReleaseNotify) |
| 180 | : mParent(parent), mNeedsReleaseNotify(needsReleaseNotify) {} |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 181 | |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 182 | /** |
| 183 | * Implementation of IProducerListener, used to notify this stream that the consumer |
| 184 | * has returned a buffer and it is ready to return to Camera3BufferManager for reuse. |
| 185 | */ |
| 186 | virtual void onBufferReleased(); |
| 187 | virtual bool needsReleaseNotify() { return mNeedsReleaseNotify; } |
| 188 | virtual void onBuffersDiscarded(const std::vector<sp<GraphicBuffer>>& buffers); |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 189 | |
| 190 | private: |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 191 | wp<Camera3OutputStream> mParent; |
| 192 | bool mNeedsReleaseNotify; |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 193 | }; |
| 194 | |
Eino-Ville Talvala | 77c1a35 | 2016-06-13 12:32:43 -0700 | [diff] [blame] | 195 | virtual status_t detachBuffer(sp<GraphicBuffer>* buffer, int* fenceFd); |
| 196 | |
Shuzhen Wang | bee0f0a | 2017-01-24 14:51:37 -0800 | [diff] [blame] | 197 | /** |
| 198 | * Notify that the buffer is being released to the buffer queue instead of |
| 199 | * being queued to the consumer. |
| 200 | */ |
| 201 | virtual status_t notifyBufferReleased(ANativeWindowBuffer *anwBuffer); |
Shuzhen Wang | 0129d52 | 2016-10-30 22:43:41 -0700 | [diff] [blame] | 202 | |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 203 | /** |
Chien-Yu Chen | a936ac2 | 2017-10-23 15:59:49 -0700 | [diff] [blame] | 204 | * Drop buffers if dropping is true. If dropping is false, do not drop buffers. |
| 205 | */ |
| 206 | virtual status_t dropBuffers(bool dropping) override; |
| 207 | |
| 208 | /** |
Shuzhen Wang | 5c22c15 | 2017-12-31 17:12:25 -0800 | [diff] [blame] | 209 | * Query the physical camera id for the output stream. |
| 210 | */ |
| 211 | virtual const String8& getPhysicalCameraId() const override; |
| 212 | |
| 213 | /** |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 214 | * Set the graphic buffer manager to get/return the stream buffers. |
| 215 | * |
| 216 | * It is only legal to call this method when stream is in STATE_CONSTRUCTED state. |
| 217 | */ |
| 218 | status_t setBufferManager(sp<Camera3BufferManager> bufferManager); |
| 219 | |
Emilian Peev | 40ead60 | 2017-09-26 15:46:36 +0100 | [diff] [blame] | 220 | /** |
| 221 | * Query the ouput surface id. |
| 222 | */ |
| 223 | virtual ssize_t getSurfaceId(const sp<Surface> &/*surface*/) { return 0; } |
| 224 | |
Yin-Chia Yeh | 58b1b4e | 2018-10-15 12:18:36 -0700 | [diff] [blame] | 225 | virtual status_t getUniqueSurfaceIds(const std::vector<size_t>&, |
| 226 | /*out*/std::vector<size_t>*) { return INVALID_OPERATION; }; |
| 227 | |
Emilian Peev | 40ead60 | 2017-09-26 15:46:36 +0100 | [diff] [blame] | 228 | /** |
| 229 | * Update the stream output surfaces. |
| 230 | */ |
| 231 | virtual status_t updateStream(const std::vector<sp<Surface>> &outputSurfaces, |
| 232 | const std::vector<OutputStreamInfo> &outputInfo, |
| 233 | const std::vector<size_t> &removedSurfaceIds, |
| 234 | KeyedVector<sp<Surface>, size_t> *outputMap/*out*/); |
| 235 | |
Emilian Peev | 35ae826 | 2018-11-08 13:11:32 +0000 | [diff] [blame] | 236 | /** |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 237 | * Set the batch size for buffer operations. The output stream will request |
| 238 | * buffers from buffer queue on a batch basis. Currently only video streams |
| 239 | * are allowed to set the batch size. Also if the stream is managed by |
| 240 | * buffer manager (Surface group in Java API) then batching is also not |
| 241 | * supported. Changing batch size on the fly while there is already batched |
| 242 | * buffers in the stream is also not supported. |
| 243 | * If the batch size is larger than the max dequeue count set |
| 244 | * by the camera HAL, the batch size will be set to the max dequeue count |
| 245 | * instead. |
| 246 | */ |
| 247 | virtual status_t setBatchSize(size_t batchSize = 1) override; |
| 248 | |
| 249 | /** |
Shuzhen Wang | 00abbeb | 2022-02-25 17:14:42 -0800 | [diff] [blame] | 250 | * Notify the stream on change of min frame durations. |
| 251 | */ |
| 252 | virtual void onMinDurationChanged(nsecs_t duration) override; |
| 253 | |
| 254 | /** |
Emilian Peev | 35ae826 | 2018-11-08 13:11:32 +0000 | [diff] [blame] | 255 | * Apply ZSL related consumer usage quirk. |
| 256 | */ |
| 257 | static void applyZSLUsageQuirk(int format, uint64_t *consumerUsage /*inout*/); |
| 258 | |
Shuzhen Wang | abbcb6b | 2020-12-09 22:32:44 -0800 | [diff] [blame] | 259 | void setImageDumpMask(int mask) { mImageDumpMask = mask; } |
Shuzhen Wang | ba92d77 | 2022-04-11 11:47:24 -0700 | [diff] [blame] | 260 | bool shouldLogError(status_t res); |
Shuzhen Wang | abbcb6b | 2020-12-09 22:32:44 -0800 | [diff] [blame] | 261 | |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 262 | protected: |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 263 | Camera3OutputStream(int id, camera_stream_type_t type, |
Eino-Ville Talvala | 3d82c0d | 2015-02-23 15:19:19 -0800 | [diff] [blame] | 264 | uint32_t width, uint32_t height, int format, |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 265 | android_dataspace dataSpace, camera_stream_rotation_t rotation, |
Shuzhen Wang | c28189a | 2017-11-27 23:05:10 -0800 | [diff] [blame] | 266 | const String8& physicalCameraId, |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 267 | const std::unordered_set<int32_t> &sensorPixelModesUsed, IPCTransport transport, |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 268 | uint64_t consumerUsage = 0, nsecs_t timestampOffset = 0, |
Emilian Peev | 2295df7 | 2021-11-12 18:14:10 -0800 | [diff] [blame] | 269 | int setId = CAMERA3_STREAM_SET_ID_INVALID, bool isMultiResolution = false, |
Emilian Peev | c81a759 | 2022-02-14 17:38:18 -0800 | [diff] [blame] | 270 | int64_t dynamicProfile = ANDROID_REQUEST_AVAILABLE_DYNAMIC_RANGE_PROFILES_MAP_STANDARD, |
Shuzhen Wang | 8ed1e87 | 2022-03-08 16:34:33 -0800 | [diff] [blame] | 271 | int64_t streamUseCase = ANDROID_SCALER_AVAILABLE_STREAM_USE_CASES_DEFAULT, |
Shuzhen Wang | e420892 | 2022-02-01 16:52:48 -0800 | [diff] [blame] | 272 | bool deviceTimeBaseIsRealtime = false, |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 273 | int timestampBase = OutputConfiguration::TIMESTAMP_BASE_DEFAULT, |
| 274 | int mirrorMode = OutputConfiguration::MIRROR_MODE_AUTO); |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 275 | |
Zhijun He | 124ccf4 | 2013-05-22 14:01:30 -0700 | [diff] [blame] | 276 | /** |
| 277 | * Note that we release the lock briefly in this function |
| 278 | */ |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 279 | virtual status_t returnBufferCheckedLocked( |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 280 | const camera_stream_buffer &buffer, |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 281 | nsecs_t timestamp, |
Shuzhen Wang | 90708ea | 2021-11-04 11:40:49 -0700 | [diff] [blame] | 282 | nsecs_t readoutTimestamp, |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 283 | bool output, |
Emilian Peev | 5104fe9 | 2021-10-21 14:27:09 -0700 | [diff] [blame] | 284 | int32_t transform, |
Yin-Chia Yeh | 58b1b4e | 2018-10-15 12:18:36 -0700 | [diff] [blame] | 285 | const std::vector<size_t>& surface_ids, |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 286 | /*out*/ |
| 287 | sp<Fence> *releaseFenceOut); |
| 288 | |
Zhijun He | 0a21051 | 2014-07-24 13:45:15 -0700 | [diff] [blame] | 289 | virtual status_t disconnectLocked(); |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 290 | status_t fixUpHidlJpegBlobHeader(ANativeWindowBuffer* anwBuffer, int fence); |
Zhijun He | 0a21051 | 2014-07-24 13:45:15 -0700 | [diff] [blame] | 291 | |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 292 | status_t getEndpointUsageForSurface(uint64_t *usage, |
Shuzhen Wang | 0129d52 | 2016-10-30 22:43:41 -0700 | [diff] [blame] | 293 | const sp<Surface>& surface) const; |
Shuzhen Wang | ba92d77 | 2022-04-11 11:47:24 -0700 | [diff] [blame] | 294 | status_t configureConsumerQueueLocked(bool allowPreviewRespace); |
Shuzhen Wang | 0129d52 | 2016-10-30 22:43:41 -0700 | [diff] [blame] | 295 | |
| 296 | // Consumer as the output of camera HAL |
Eino-Ville Talvala | 727d172 | 2015-06-09 13:44:19 -0700 | [diff] [blame] | 297 | sp<Surface> mConsumer; |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 298 | |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 299 | uint64_t getPresetConsumerUsage() const { return mConsumerUsage; } |
Zhijun He | f0645c1 | 2016-08-02 00:58:11 -0700 | [diff] [blame] | 300 | |
| 301 | static const nsecs_t kDequeueBufferTimeout = 1000000000; // 1 sec |
| 302 | |
Shuzhen Wang | bee0f0a | 2017-01-24 14:51:37 -0800 | [diff] [blame] | 303 | status_t getBufferLockedCommon(ANativeWindowBuffer** anb, int* fenceFd); |
| 304 | |
| 305 | |
Shuzhen Wang | 0129d52 | 2016-10-30 22:43:41 -0700 | [diff] [blame] | 306 | private: |
| 307 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 308 | int mTransform; |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 309 | |
Igor Murashkin | e3a9f96 | 2013-05-08 18:03:15 -0700 | [diff] [blame] | 310 | virtual status_t setTransformLocked(int transform); |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 311 | |
Ruchit Sharma | e0711f2 | 2014-08-18 13:48:24 -0400 | [diff] [blame] | 312 | bool mTraceFirstBuffer; |
| 313 | |
Eino-Ville Talvala | 727d172 | 2015-06-09 13:44:19 -0700 | [diff] [blame] | 314 | // Name of Surface consumer |
| 315 | String8 mConsumerName; |
| 316 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 317 | /** |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 318 | * GraphicBuffer manager this stream is registered to. Used to replace the buffer |
| 319 | * allocation/deallocation role of BufferQueue. |
| 320 | */ |
| 321 | sp<Camera3BufferManager> mBufferManager; |
| 322 | |
| 323 | /** |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 324 | * Buffer producer listener, used to handle notification when a buffer is released |
| 325 | * from consumer side, or a set of buffers are discarded by the consumer. |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 326 | */ |
Shuzhen Wang | 0160ddd | 2019-08-15 09:11:56 -0700 | [diff] [blame] | 327 | sp<BufferProducerListener> mBufferProducerListener; |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 328 | |
| 329 | /** |
| 330 | * Flag indicating if the buffer manager is used to allocate the stream buffers |
| 331 | */ |
| 332 | bool mUseBufferManager; |
Shuzhen Wang | c28dccc | 2016-02-11 23:48:46 -0800 | [diff] [blame] | 333 | |
| 334 | /** |
Shuzhen Wang | e420892 | 2022-02-01 16:52:48 -0800 | [diff] [blame] | 335 | * Offset used to override camera HAL produced timestamps |
| 336 | * |
| 337 | * The offset is first initialized to bootTime - monotonicTime in |
| 338 | * constructor, and may later be updated based on the client's timestampBase |
| 339 | * setting. |
Shuzhen Wang | c28dccc | 2016-02-11 23:48:46 -0800 | [diff] [blame] | 340 | */ |
| 341 | nsecs_t mTimestampOffset; |
| 342 | |
Zhijun He | 125684a | 2015-12-26 15:07:30 -0800 | [diff] [blame] | 343 | /** |
Shuzhen Wang | ffc4c01 | 2022-04-20 15:55:46 -0700 | [diff] [blame] | 344 | * If camera readout time is used rather than the start-of-exposure time. |
| 345 | */ |
| 346 | bool mUseReadoutTime; |
| 347 | |
| 348 | /** |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 349 | * Consumer end point usage flag set by the constructor for the deferred |
| 350 | * consumer case. |
| 351 | */ |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 352 | uint64_t mConsumerUsage; |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 353 | |
Chien-Yu Chen | a936ac2 | 2017-10-23 15:59:49 -0700 | [diff] [blame] | 354 | // Whether to drop valid buffers. |
| 355 | bool mDropBuffers; |
| 356 | |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 357 | |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 358 | |
| 359 | // The batch size for buffer operation |
Shuzhen Wang | c762946 | 2021-07-12 15:02:58 -0700 | [diff] [blame] | 360 | std::atomic_size_t mBatchSize = 1; |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 361 | |
Shuzhen Wang | c762946 | 2021-07-12 15:02:58 -0700 | [diff] [blame] | 362 | // Protecting batch states below, must be acquired after mLock |
| 363 | std::mutex mBatchLock; |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 364 | // Prefetched buffers (ready to be handed to client) |
| 365 | std::vector<Surface::BatchBuffer> mBatchedBuffers; |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 366 | // ---- End of mBatchLock protected scope ---- |
| 367 | |
Shuzhen Wang | 610d7b8 | 2022-02-08 14:37:22 -0800 | [diff] [blame] | 368 | const int mMirrorMode; |
| 369 | |
Zhijun He | 5d677d1 | 2016-05-29 16:52:39 -0700 | [diff] [blame] | 370 | /** |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 371 | * Internal Camera3Stream interface |
| 372 | */ |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 373 | virtual status_t getBufferLocked(camera_stream_buffer *buffer, |
Shuzhen Wang | bee0f0a | 2017-01-24 14:51:37 -0800 | [diff] [blame] | 374 | const std::vector<size_t>& surface_ids); |
| 375 | |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 376 | virtual status_t getBuffersLocked(/*out*/std::vector<OutstandingBuffer>* buffers) override; |
| 377 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 378 | virtual status_t returnBufferLocked( |
Emilian Peev | f481670 | 2020-04-03 15:44:51 -0700 | [diff] [blame] | 379 | const camera_stream_buffer &buffer, |
Shuzhen Wang | 90708ea | 2021-11-04 11:40:49 -0700 | [diff] [blame] | 380 | nsecs_t timestamp, nsecs_t readoutTimestamp, |
| 381 | int32_t transform, const std::vector<size_t>& surface_ids); |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 382 | |
Shuzhen Wang | bee0f0a | 2017-01-24 14:51:37 -0800 | [diff] [blame] | 383 | virtual status_t queueBufferToConsumer(sp<ANativeWindow>& consumer, |
Yin-Chia Yeh | 58b1b4e | 2018-10-15 12:18:36 -0700 | [diff] [blame] | 384 | ANativeWindowBuffer* buffer, int anwReleaseFence, |
| 385 | const std::vector<size_t>& surface_ids); |
Shuzhen Wang | bee0f0a | 2017-01-24 14:51:37 -0800 | [diff] [blame] | 386 | |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 387 | virtual status_t configureQueueLocked(); |
Eino-Ville Talvala | b2f5b19 | 2013-07-30 14:36:03 -0700 | [diff] [blame] | 388 | |
Emilian Peev | 050f5dc | 2017-05-18 14:43:56 +0100 | [diff] [blame] | 389 | virtual status_t getEndpointUsage(uint64_t *usage) const; |
Eino-Ville Talvala | b2f5b19 | 2013-07-30 14:36:03 -0700 | [diff] [blame] | 390 | |
Yin-Chia Yeh | 89954d9 | 2017-05-21 17:28:53 -0700 | [diff] [blame] | 391 | /** |
| 392 | * Private methods |
| 393 | */ |
Yin-Chia Yeh | 017d49c | 2017-03-31 19:11:00 -0700 | [diff] [blame] | 394 | void onBuffersRemovedLocked(const std::vector<sp<GraphicBuffer>>&); |
Yin-Chia Yeh | 89954d9 | 2017-05-21 17:28:53 -0700 | [diff] [blame] | 395 | status_t detachBufferLocked(sp<GraphicBuffer>* buffer, int* fenceFd); |
Yin-Chia Yeh | bf1b8b9 | 2019-03-06 14:56:08 -0800 | [diff] [blame] | 396 | // Call this after each dequeueBuffer/attachBuffer/detachNextBuffer call to get update on |
| 397 | // removed buffers. Set notifyBufferManager to false when the call is initiated by buffer |
| 398 | // manager so buffer manager doesn't need to be notified. |
| 399 | void checkRemovedBuffersLocked(bool notifyBufferManager = true); |
| 400 | |
| 401 | // Check return status of IGBP calls and set abandoned state accordingly |
| 402 | void checkRetAndSetAbandonedLocked(status_t res); |
Yin-Chia Yeh | 017d49c | 2017-03-31 19:11:00 -0700 | [diff] [blame] | 403 | |
Yin-Chia Yeh | a1b56c8 | 2019-03-27 15:50:39 -0700 | [diff] [blame] | 404 | // If the status indicates abandonded stream, only log when state hasn't been updated to |
| 405 | // STATE_ABANDONED |
| 406 | static bool shouldLogError(status_t res, StreamState state); |
| 407 | |
Shuzhen Wang | abbcb6b | 2020-12-09 22:32:44 -0800 | [diff] [blame] | 408 | // Dump images to disk before returning to consumer |
| 409 | void dumpImageToDisk(nsecs_t timestamp, ANativeWindowBuffer* anwBuffer, int fence); |
| 410 | |
Yin-Chia Yeh | 14ef48d | 2020-02-10 15:06:37 -0800 | [diff] [blame] | 411 | void returnPrefetchedBuffersLocked(); |
| 412 | |
Shuzhen Wang | 00abbeb | 2022-02-25 17:14:42 -0800 | [diff] [blame] | 413 | |
Shuzhen Wang | 686f644 | 2017-06-20 16:16:04 -0700 | [diff] [blame] | 414 | static const int32_t kDequeueLatencyBinSize = 5; // in ms |
| 415 | CameraLatencyHistogram mDequeueBufferLatency; |
Jayant Chowdhary | c67af1b | 2022-04-07 18:05:04 +0000 | [diff] [blame] | 416 | IPCTransport mIPCTransport = IPCTransport::INVALID; |
Shuzhen Wang | 686f644 | 2017-06-20 16:16:04 -0700 | [diff] [blame] | 417 | |
Shuzhen Wang | abbcb6b | 2020-12-09 22:32:44 -0800 | [diff] [blame] | 418 | int mImageDumpMask = 0; |
| 419 | |
Shuzhen Wang | ba92d77 | 2022-04-11 11:47:24 -0700 | [diff] [blame] | 420 | // Re-space frames by overriding timestamp to align with display Vsync. |
| 421 | // Default is on for SurfaceView bound streams. |
Shuzhen Wang | 00abbeb | 2022-02-25 17:14:42 -0800 | [diff] [blame] | 422 | nsecs_t mMinExpectedDuration = 0; |
| 423 | bool mSyncToDisplay = false; |
| 424 | DisplayEventReceiver mDisplayEventReceiver; |
| 425 | nsecs_t mLastCaptureTime = 0; |
| 426 | nsecs_t mLastPresentTime = 0; |
| 427 | nsecs_t mCaptureToPresentOffset = 0; |
| 428 | static constexpr size_t kDisplaySyncExtraBuffer = 2; |
Shuzhen Wang | ed08fbe | 2022-06-21 01:00:50 -0700 | [diff] [blame] | 429 | static constexpr nsecs_t kSpacingResetIntervalNs = 50000000LL; // 50 millisecond |
Shuzhen Wang | 00abbeb | 2022-02-25 17:14:42 -0800 | [diff] [blame] | 430 | static constexpr nsecs_t kTimelineThresholdNs = 1000000LL; // 1 millisecond |
Shuzhen Wang | 34a5e28 | 2022-06-17 14:48:35 -0700 | [diff] [blame] | 431 | static constexpr float kMaxIntervalRatioDeviation = 0.05f; |
Shuzhen Wang | ed08fbe | 2022-06-21 01:00:50 -0700 | [diff] [blame] | 432 | static constexpr int kMaxTimelines = 3; |
Shuzhen Wang | ba92d77 | 2022-04-11 11:47:24 -0700 | [diff] [blame] | 433 | nsecs_t syncTimestampToDisplayLocked(nsecs_t t); |
| 434 | |
| 435 | // Re-space frames by delaying queueBuffer so that frame delivery has |
| 436 | // the same cadence as capture. Default is on for SurfaceTexture bound |
| 437 | // streams. |
| 438 | sp<PreviewFrameSpacer> mPreviewFrameSpacer; |
Eino-Ville Talvala | fd58f1a | 2013-03-06 16:20:06 -0800 | [diff] [blame] | 439 | }; // class Camera3OutputStream |
| 440 | |
| 441 | } // namespace camera3 |
| 442 | |
| 443 | } // namespace android |
| 444 | |
| 445 | #endif |