1 /* 2 * Copyright (C) 2019 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_SERVERS_CAMERA3_BUFFER_UTILS_H 18 #define ANDROID_SERVERS_CAMERA3_BUFFER_UTILS_H 19 20 #include <unordered_map> 21 #include <mutex> 22 #include <set> 23 24 #include <cutils/native_handle.h> 25 26 #include <android/hardware/camera/device/3.2/ICameraDevice.h> 27 28 #include <device3/Camera3OutputInterface.h> 29 30 namespace android { 31 32 namespace camera3 { 33 34 struct BufferHasher { operatorBufferHasher35 size_t operator()(const buffer_handle_t& buf) const { 36 if (buf == nullptr) 37 return 0; 38 39 size_t result = 1; 40 result = 31 * result + buf->numFds; 41 for (int i = 0; i < buf->numFds; i++) { 42 result = 31 * result + buf->data[i]; 43 } 44 return result; 45 } 46 }; 47 48 struct BufferComparator { operatorBufferComparator49 bool operator()(const buffer_handle_t& buf1, const buffer_handle_t& buf2) const { 50 if (buf1->numFds == buf2->numFds) { 51 for (int i = 0; i < buf1->numFds; i++) { 52 if (buf1->data[i] != buf2->data[i]) { 53 return false; 54 } 55 } 56 return true; 57 } 58 return false; 59 } 60 }; 61 62 // Per stream buffer native handle -> bufId map 63 typedef std::unordered_map<const buffer_handle_t, uint64_t, 64 BufferHasher, BufferComparator> BufferIdMap; 65 66 // streamId -> BufferIdMap 67 typedef std::unordered_map<int, BufferIdMap> BufferIdMaps; 68 69 // Map of inflight buffers sent along in capture requests. 70 // Key is composed by (frameNumber << 32 | streamId) 71 typedef std::unordered_map<uint64_t, buffer_handle_t*> InflightBufferMap; 72 73 // Map of inflight buffers dealt by requestStreamBuffers API 74 typedef std::unordered_map<uint64_t, std::pair<int32_t, buffer_handle_t*>> RequestedBufferMap; 75 76 // A struct containing all buffer tracking information like inflight buffers 77 // and buffer ID caches 78 class BufferRecords : public BufferRecordsInterface { 79 80 public: BufferRecords()81 BufferRecords() {} 82 BufferRecords(BufferRecords && other)83 BufferRecords(BufferRecords&& other) : 84 mBufferIdMaps(other.mBufferIdMaps), 85 mNextBufferId(other.mNextBufferId), 86 mInflightBufferMap(other.mInflightBufferMap), 87 mRequestedBufferMap(other.mRequestedBufferMap) {} 88 ~BufferRecords()89 virtual ~BufferRecords() {} 90 91 // Helper methods to help moving buffer records 92 void takeInflightBufferMap(BufferRecords& other); 93 void takeRequestedBufferMap(BufferRecords& other); 94 void takeBufferCaches(BufferRecords& other, const std::vector<int32_t>& streams); 95 96 // method to extract buffer's unique ID 97 // return pair of (newlySeenBuffer?, bufferId) 98 virtual std::pair<bool, uint64_t> getBufferId( 99 const buffer_handle_t& buf, int streamId) override; 100 101 void tryCreateBufferCache(int streamId); 102 103 void removeInactiveBufferCaches(const std::set<int32_t>& activeStreams); 104 105 // Return the removed buffer ID if input cache is found. 106 // Otherwise return BUFFER_ID_NO_BUFFER 107 uint64_t removeOneBufferCache(int streamId, const native_handle_t* handle) override; 108 109 // Clear all caches for input stream, but do not remove the stream 110 // Removed buffers' ID are returned 111 std::vector<uint64_t> clearBufferCaches(int streamId); 112 113 bool isStreamCached(int streamId); 114 115 // Return true if the input caches match what we have; otherwise false 116 bool verifyBufferIds(int32_t streamId, std::vector<uint64_t>& inBufIds); 117 118 // Get a vector of (frameNumber, streamId) pair of currently inflight 119 // buffers 120 void getInflightBufferKeys(std::vector<std::pair<int32_t, int32_t>>* out); 121 122 status_t pushInflightBuffer(int32_t frameNumber, int32_t streamId, 123 buffer_handle_t *buffer); 124 125 // Find a buffer_handle_t based on frame number and stream ID 126 virtual status_t popInflightBuffer(int32_t frameNumber, int32_t streamId, 127 /*out*/ buffer_handle_t **buffer) override; 128 129 // Pop inflight buffers based on pairs of (frameNumber,streamId) 130 void popInflightBuffers(const std::vector<std::pair<int32_t, int32_t>>& buffers); 131 132 // Get a vector of bufferId of currently inflight buffers 133 void getInflightRequestBufferKeys(std::vector<uint64_t>* out); 134 135 // Register a bufId (streamId, buffer_handle_t) to inflight request buffer 136 virtual status_t pushInflightRequestBuffer( 137 uint64_t bufferId, buffer_handle_t* buf, int32_t streamId) override; 138 139 // Find a buffer_handle_t based on bufferId 140 virtual status_t popInflightRequestBuffer(uint64_t bufferId, 141 /*out*/ buffer_handle_t** buffer, 142 /*optional out*/ int32_t* streamId = nullptr) override; 143 144 private: 145 std::mutex mBufferIdMapLock; 146 BufferIdMaps mBufferIdMaps; 147 uint64_t mNextBufferId = 1; // 0 means no buffer 148 149 std::mutex mInflightLock; 150 InflightBufferMap mInflightBufferMap; 151 152 std::mutex mRequestedBuffersLock; 153 RequestedBufferMap mRequestedBufferMap; 154 }; // class BufferRecords 155 156 static const uint64_t BUFFER_ID_NO_BUFFER = 0; 157 } // namespace camera3 158 159 } // namespace android 160 161 #endif 162