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 #ifndef ANDROID_GUI_BUFFERHUBPRODUCER_H_ 18 #define ANDROID_GUI_BUFFERHUBPRODUCER_H_ 19 20 #include <gui/BufferSlot.h> 21 #include <gui/IGraphicBufferProducer.h> 22 #include <private/dvr/buffer_hub_queue_client.h> 23 #include <private/dvr/buffer_hub_queue_parcelable.h> 24 25 namespace android { 26 27 class BufferHubProducer : public IGraphicBufferProducer { 28 public: 29 static constexpr int kNoConnectedApi = -1; 30 31 // TODO(b/36187402) The actual implementation of BufferHubQueue's consumer 32 // side logic doesn't limit the number of buffer it can acquire 33 // simultaneously. We need a way for consumer logic to configure and enforce 34 // that. 35 static constexpr int kDefaultUndequeuedBuffers = 1; 36 37 // Creates a BufferHubProducer instance by importing an existing prodcuer 38 // queue. 39 static sp<BufferHubProducer> Create(const std::shared_ptr<dvr::ProducerQueue>& producer); 40 41 // Creates a BufferHubProducer instance by importing an existing prodcuer 42 // parcelable. Note that this call takes the ownership of the parcelable 43 // object and is guaranteed to succeed if parcelable object is valid. 44 static sp<BufferHubProducer> Create(dvr::ProducerQueueParcelable parcelable); 45 46 // See |IGraphicBufferProducer::requestBuffer| 47 status_t requestBuffer(int slot, sp<GraphicBuffer>* buf) override; 48 49 // For the BufferHub based implementation. All buffers in the queue are 50 // allowed to be dequeued from the consumer side. It call always returns 51 // 0 for |NATIVE_WINDOW_MIN_UNDEQUEUED_BUFFERS| query. Thus setting 52 // |max_dequeued_buffers| here can be considered the same as setting queue 53 // capacity. 54 // 55 // See |IGraphicBufferProducer::setMaxDequeuedBufferCount| for more info 56 status_t setMaxDequeuedBufferCount(int max_dequeued_buffers) override; 57 58 // See |IGraphicBufferProducer::setAsyncMode| 59 status_t setAsyncMode(bool async) override; 60 61 // See |IGraphicBufferProducer::dequeueBuffer| 62 status_t dequeueBuffer(int* out_slot, sp<Fence>* out_fence, uint32_t width, uint32_t height, 63 PixelFormat format, uint64_t usage, uint64_t* outBufferAge, 64 FrameEventHistoryDelta* outTimestamps) override; 65 66 // See |IGraphicBufferProducer::detachBuffer| 67 status_t detachBuffer(int slot) override; 68 69 // See |IGraphicBufferProducer::detachNextBuffer| 70 status_t detachNextBuffer(sp<GraphicBuffer>* out_buffer, sp<Fence>* out_fence) override; 71 72 // See |IGraphicBufferProducer::attachBuffer| 73 status_t attachBuffer(int* out_slot, const sp<GraphicBuffer>& buffer) override; 74 75 // See |IGraphicBufferProducer::queueBuffer| 76 status_t queueBuffer(int slot, const QueueBufferInput& input, 77 QueueBufferOutput* output) override; 78 79 // See |IGraphicBufferProducer::cancelBuffer| 80 status_t cancelBuffer(int slot, const sp<Fence>& fence) override; 81 82 // See |IGraphicBufferProducer::query| 83 status_t query(int what, int* out_value) override; 84 85 // See |IGraphicBufferProducer::connect| 86 status_t connect(const sp<IProducerListener>& listener, int api, 87 bool producer_controlled_by_app, QueueBufferOutput* output) override; 88 89 // See |IGraphicBufferProducer::disconnect| 90 status_t disconnect(int api, DisconnectMode mode = DisconnectMode::Api) override; 91 92 // See |IGraphicBufferProducer::setSidebandStream| 93 status_t setSidebandStream(const sp<NativeHandle>& stream) override; 94 95 // See |IGraphicBufferProducer::allocateBuffers| 96 void allocateBuffers(uint32_t width, uint32_t height, PixelFormat format, 97 uint64_t usage) override; 98 99 // See |IGraphicBufferProducer::allowAllocation| 100 status_t allowAllocation(bool allow) override; 101 102 // See |IGraphicBufferProducer::setGenerationNumber| 103 status_t setGenerationNumber(uint32_t generation_number) override; 104 105 // See |IGraphicBufferProducer::getConsumerName| 106 String8 getConsumerName() const override; 107 108 // See |IGraphicBufferProducer::setSharedBufferMode| 109 status_t setSharedBufferMode(bool shared_buffer_mode) override; 110 111 // See |IGraphicBufferProducer::setAutoRefresh| 112 status_t setAutoRefresh(bool auto_refresh) override; 113 114 // See |IGraphicBufferProducer::setDequeueTimeout| 115 status_t setDequeueTimeout(nsecs_t timeout) override; 116 117 // See |IGraphicBufferProducer::getLastQueuedBuffer| 118 status_t getLastQueuedBuffer(sp<GraphicBuffer>* out_buffer, sp<Fence>* out_fence, 119 float out_transform_matrix[16]) override; 120 121 // See |IGraphicBufferProducer::getFrameTimestamps| 122 void getFrameTimestamps(FrameEventHistoryDelta* /*outDelta*/) override; 123 124 // See |IGraphicBufferProducer::getUniqueId| 125 status_t getUniqueId(uint64_t* out_id) const override; 126 127 // See |IGraphicBufferProducer::getConsumerUsage| 128 status_t getConsumerUsage(uint64_t* out_usage) const override; 129 130 // Takes out the current producer as a binder parcelable object. Note that the 131 // producer must be disconnected to be exportable. After successful export, 132 // the producer queue can no longer be connected again. Returns NO_ERROR when 133 // takeout is successful and out_parcelable will hold the new parcelable 134 // object. Also note that out_parcelable cannot be NULL and must points to an 135 // invalid parcelable. 136 status_t TakeAsParcelable(dvr::ProducerQueueParcelable* out_parcelable); 137 138 IBinder* onAsBinder() override; 139 140 protected: 141 // See |IGraphicBufferProducer::exportToParcel| 142 status_t exportToParcel(Parcel* parcel) override; 143 144 private: 145 using LocalHandle = pdx::LocalHandle; 146 147 // Private constructor to force use of |Create|. BufferHubProducer()148 BufferHubProducer() {} 149 genUniqueId()150 static uint64_t genUniqueId() { 151 static std::atomic<uint32_t> counter{0}; 152 static uint64_t id = static_cast<uint64_t>(getpid()) << 32; 153 return id | counter++; 154 } 155 156 // Allocate new buffer through BufferHub and add it into |queue_| for 157 // bookkeeping. 158 status_t AllocateBuffer(uint32_t width, uint32_t height, uint32_t layer_count, 159 PixelFormat format, uint64_t usage); 160 161 // Remove a buffer via BufferHubRPC. 162 status_t RemoveBuffer(size_t slot); 163 164 // Free all buffers which are owned by the prodcuer. Note that if graphic 165 // buffers are acquired by the consumer, we can't . 166 status_t FreeAllBuffers(); 167 168 // Concreate implementation backed by BufferHubBuffer. 169 std::shared_ptr<dvr::ProducerQueue> queue_; 170 171 // Mutex for thread safety. 172 std::mutex mutex_; 173 174 // Connect client API, should be one of the NATIVE_WINDOW_API_* flags. 175 int connected_api_{kNoConnectedApi}; 176 177 // |max_buffer_count_| sets the capacity of the underlying buffer queue. 178 int32_t max_buffer_count_{dvr::BufferHubQueue::kMaxQueueCapacity}; 179 180 // |max_dequeued_buffer_count_| set the maximum number of buffers that can 181 // be dequeued at the same momment. 182 int32_t max_dequeued_buffer_count_{1}; 183 184 // Sets how long dequeueBuffer or attachBuffer will block if a buffer or 185 // slot is not yet available. The timeout is stored in milliseconds. 186 int dequeue_timeout_ms_{dvr::BufferHubQueue::kNoTimeOut}; 187 188 // |generation_number_| stores the current generation number of the attached 189 // producer. Any attempt to attach a buffer with a different generation 190 // number will fail. 191 // TOOD(b/38137191) Currently not used as we don't support 192 // IGraphicBufferProducer::detachBuffer. 193 uint32_t generation_number_{0}; 194 195 // |buffers_| stores the buffers that have been dequeued from 196 // |dvr::BufferHubQueue|, It is initialized to invalid buffers, and gets 197 // filled in with the result of |Dequeue|. 198 // TODO(jwcai) The buffer allocated to a slot will also be replaced if the 199 // requested buffer usage or geometry differs from that of the buffer 200 // allocated to a slot. 201 struct BufferHubSlot : public BufferSlot { BufferHubSlotBufferHubSlot202 BufferHubSlot() : mBufferProducer(nullptr), mIsReallocating(false) {} 203 // BufferSlot comes from android framework, using m prefix to comply with 204 // the name convention with the reset of data fields from BufferSlot. 205 std::shared_ptr<dvr::BufferProducer> mBufferProducer; 206 bool mIsReallocating; 207 }; 208 BufferHubSlot buffers_[dvr::BufferHubQueue::kMaxQueueCapacity]; 209 210 // A uniqueId used by IGraphicBufferProducer interface. 211 const uint64_t unique_id_{genUniqueId()}; 212 213 // A pending parcelable object which keeps the bufferhub channel alive. 214 dvr::ProducerQueueParcelable pending_producer_parcelable_; 215 }; 216 217 } // namespace android 218 219 #endif // ANDROID_GUI_BUFFERHUBPRODUCER_H_ 220