• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 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 _ACAMERA_MANAGER_H
18 #define _ACAMERA_MANAGER_H
19 
20 #include <CameraMetadata.h>
21 #include <aidl/android/frameworks/cameraservice/common/Status.h>
22 #include <aidl/android/frameworks/cameraservice/common/VendorTag.h>
23 #include <aidl/android/frameworks/cameraservice/common/VendorTagSection.h>
24 #include <aidl/android/frameworks/cameraservice/service/BnCameraServiceListener.h>
25 #include <aidl/android/frameworks/cameraservice/service/CameraDeviceStatus.h>
26 #include <aidl/android/frameworks/cameraservice/service/CameraStatusAndId.h>
27 #include <aidl/android/frameworks/cameraservice/service/ICameraService.h>
28 #include <android-base/parseint.h>
29 #include <camera/NdkCameraManager.h>
30 #include <map>
31 #include <media/stagefright/foundation/AHandler.h>
32 #include <media/stagefright/foundation/ALooper.h>
33 #include <media/stagefright/foundation/AMessage.h>
34 #include <set>
35 #include <utility>
36 #include <utils/Mutex.h>
37 #include <utils/StrongPointer.h>
38 
39 namespace android {
40 namespace acam {
41 
42 using ::aidl::android::frameworks::cameraservice::common::Status;
43 using ::aidl::android::frameworks::cameraservice::common::VendorTag;
44 using ::aidl::android::frameworks::cameraservice::common::VendorTagSection;
45 using ::aidl::android::frameworks::cameraservice::service::BnCameraServiceListener;
46 using ::aidl::android::frameworks::cameraservice::service::CameraDeviceStatus;
47 using ::aidl::android::frameworks::cameraservice::service::CameraStatusAndId;
48 using ::aidl::android::frameworks::cameraservice::service::ICameraService;
49 
50 /**
51  * Per-process singleton instance of CameraManger. Shared by all ACameraManager
52  * instances. Created when first ACameraManager is created and destroyed when
53  * all ACameraManager instances are deleted.
54  *
55  * TODO: maybe CameraManagerGlobal is better suited in libcameraclient?
56  */
57 class CameraManagerGlobal final: public std::enable_shared_from_this<CameraManagerGlobal> {
58   public:
59     static std::shared_ptr<CameraManagerGlobal> getInstance();
60     static void binderDeathCallback(void* cookie);
61 
CameraManagerGlobal()62     CameraManagerGlobal() {};
63     ~CameraManagerGlobal();
64 
65     std::shared_ptr<ICameraService> getCameraService();
66 
67     void registerAvailabilityCallback(const ACameraManager_AvailabilityCallbacks *callback);
68     void unregisterAvailabilityCallback(const ACameraManager_AvailabilityCallbacks *callback);
69 
70     void registerExtendedAvailabilityCallback(
71             const ACameraManager_ExtendedAvailabilityCallbacks* callback);
72     void unregisterExtendedAvailabilityCallback(
73             const ACameraManager_ExtendedAvailabilityCallbacks* callback);
74 
75     /**
76      * Return camera IDs that support camera2
77      */
78     void getCameraIdList(std::vector<std::string> *cameraIds);
79 
80   private:
81     std::shared_ptr<ICameraService> mCameraService;
82     const int          kCameraServicePollDelay = 500000; // 0.5s
83     Mutex              mLock;
84     ::ndk::ScopedAIBinder_DeathRecipient mDeathRecipient;
85 
86     class CameraServiceListener final : public BnCameraServiceListener {
87       public:
CameraServiceListener(std::weak_ptr<CameraManagerGlobal> cm)88         explicit CameraServiceListener(std::weak_ptr<CameraManagerGlobal> cm) :
89               mCameraManager(std::move(cm)) {}
90         ndk::ScopedAStatus onPhysicalCameraStatusChanged(
91                 CameraDeviceStatus in_status, const std::string& in_cameraId,
92                 const std::string& in_physicalCameraId) override;
93         ndk::ScopedAStatus onStatusChanged(CameraDeviceStatus in_status,
94                                            const std::string& in_cameraId) override;
95 
96       private:
97         const std::weak_ptr<CameraManagerGlobal> mCameraManager;
98     };
99     std::shared_ptr<CameraServiceListener> mCameraServiceListener;
100 
101     // Wrapper of ACameraManager_AvailabilityCallbacks so we can store it in std::set
102     struct Callback {
CallbackCallback103         explicit Callback(const ACameraManager_AvailabilityCallbacks *callback) :
104             mAvailable(callback->onCameraAvailable),
105             mUnavailable(callback->onCameraUnavailable),
106             mAccessPriorityChanged(nullptr),
107             mPhysicalCamAvailable(nullptr),
108             mPhysicalCamUnavailable(nullptr),
109             mContext(callback->context) {}
110 
CallbackCallback111         explicit Callback(const ACameraManager_ExtendedAvailabilityCallbacks *callback) :
112             mAvailable(callback->availabilityCallbacks.onCameraAvailable),
113             mUnavailable(callback->availabilityCallbacks.onCameraUnavailable),
114             mAccessPriorityChanged(callback->onCameraAccessPrioritiesChanged),
115             mPhysicalCamAvailable(callback->onPhysicalCameraAvailable),
116             mPhysicalCamUnavailable(callback->onPhysicalCameraUnavailable),
117             mContext(callback->availabilityCallbacks.context) {}
118 
119         bool operator == (const Callback& other) const {
120             return (mAvailable == other.mAvailable &&
121                     mUnavailable == other.mUnavailable &&
122                     mAccessPriorityChanged == other.mAccessPriorityChanged &&
123                     mPhysicalCamAvailable == other.mPhysicalCamAvailable &&
124                     mPhysicalCamUnavailable == other.mPhysicalCamUnavailable &&
125                     mContext == other.mContext);
126         }
127         bool operator != (const Callback& other) const {
128             return !(*this == other);
129         }
130         bool operator < (const Callback& other) const {
131 #pragma GCC diagnostic push
132 #pragma GCC diagnostic ignored "-Wordered-compare-function-pointers"
133             if (*this == other) return false;
134             if (mContext != other.mContext) return mContext < other.mContext;
135             if (mAvailable != other.mAvailable) return mAvailable < other.mAvailable;
136             if (mAccessPriorityChanged != other.mAccessPriorityChanged)
137                     return mAccessPriorityChanged < other.mAccessPriorityChanged;
138             if (mPhysicalCamAvailable != other.mPhysicalCamAvailable)
139                     return mPhysicalCamAvailable < other.mPhysicalCamAvailable;
140             if (mPhysicalCamUnavailable != other.mPhysicalCamUnavailable)
141                     return mPhysicalCamUnavailable < other.mPhysicalCamUnavailable;
142             return mUnavailable < other.mUnavailable;
143 #pragma GCC diagnostic pop
144         }
145         bool operator > (const Callback& other) const {
146             return (*this != other && !(*this < other));
147         }
148         ACameraManager_AvailabilityCallback mAvailable;
149         ACameraManager_AvailabilityCallback mUnavailable;
150         ACameraManager_AccessPrioritiesChangedCallback mAccessPriorityChanged;
151         ACameraManager_PhysicalCameraAvailabilityCallback mPhysicalCamAvailable;
152         ACameraManager_PhysicalCameraAvailabilityCallback mPhysicalCamUnavailable;
153         void*                               mContext;
154     };
155 
156     android::Condition mCallbacksCond;
157     size_t mPendingCallbackCnt = 0;
158     void onCallbackCalled();
159     void drainPendingCallbacksLocked();
160 
161     std::set<Callback> mCallbacks;
162 
163     // definition of handler and message
164     enum {
165         kWhatSendSingleCallback,
166         kWhatSendSinglePhysicalCameraCallback,
167     };
168     static const char* kCameraIdKey;
169     static const char* kPhysicalCameraIdKey;
170     static const char* kCallbackFpKey;
171     static const char* kContextKey;
172     static const nsecs_t kCallbackDrainTimeout;
173     class CallbackHandler : public AHandler {
174       public:
CallbackHandler(std::weak_ptr<CameraManagerGlobal> parent)175         CallbackHandler(std::weak_ptr<CameraManagerGlobal> parent) : mParent(std::move(parent)) {}
176         void onMessageReceived(const sp<AMessage> &msg) override;
177       private:
178         std::weak_ptr<CameraManagerGlobal> mParent;
179         void notifyParent();
180         void onMessageReceivedInternal(const sp<AMessage> &msg);
181     };
182     sp<CallbackHandler> mHandler;
183     sp<ALooper>         mCbLooper; // Looper thread where callbacks actually happen on
184 
185     void onStatusChanged(const CameraDeviceStatus &status, const std::string &cameraId);
186     void onStatusChangedLocked(const CameraDeviceStatus &status, const std::string &cameraId);
187     void onStatusChanged(const CameraDeviceStatus &status, const std::string &cameraId,
188                          const std::string &physicalCameraId);
189     void onStatusChangedLocked(const CameraDeviceStatus &status, const std::string &cameraId,
190                                const std::string &physicalCameraId);
191     bool setupVendorTags();
192 
193     // Utils for status
194     static bool validStatus(CameraDeviceStatus status);
195     static bool isStatusAvailable(CameraDeviceStatus status);
196 
197     // The sort logic must match the logic in
198     // libcameraservice/common/CameraProviderManager.cpp::getAPI1CompatibleCameraDeviceIds
199     struct CameraIdComparator {
operatorCameraIdComparator200         bool operator()(const std::string& a, const std::string& b) const {
201             uint32_t aUint = 0, bUint = 0;
202             bool aIsUint = base::ParseUint(a.c_str(), &aUint);
203             bool bIsUint = base::ParseUint(b.c_str(), &bUint);
204 
205             // Uint device IDs first
206             if (aIsUint && bIsUint) {
207                 return aUint < bUint;
208             } else if (aIsUint) {
209                 return true;
210             } else if (bIsUint) {
211                 return false;
212             }
213             // Simple string compare if both id are not uint
214             return a < b;
215         }
216     };
217 
218     struct CameraStatus {
219       private:
220         CameraDeviceStatus status = CameraDeviceStatus::STATUS_NOT_PRESENT;
221         mutable std::mutex mLock;
222         std::set<std::string> unavailablePhysicalIds;
223       public:
CameraStatusCameraStatus224         CameraStatus(CameraDeviceStatus st): status(st) { };
225         CameraStatus() = default;
226 
227         bool addUnavailablePhysicalId(const std::string& physicalCameraId);
228         bool removeUnavailablePhysicalId(const std::string& physicalCameraId);
229         CameraDeviceStatus getStatus();
230         void updateStatus(CameraDeviceStatus newStatus);
231         std::set<std::string> getUnavailablePhysicalIds();
232     };
233 
234     template <class T>
235     void registerAvailCallback(const T *callback);
236 
237     // Map camera_id -> status
238     std::map<std::string, CameraStatus, CameraIdComparator> mDeviceStatusMap;
239 
240     // For the singleton instance
241     static Mutex sLock;
242     // Static instance is stored in a weak pointer, so will only exist if there is at least one
243     // active consumer of CameraManagerGlobal
244     static std::weak_ptr<CameraManagerGlobal> sInstance;
245 };
246 
247 } // namespace acam;
248 } // namespace android;
249 
250 /**
251  * ACameraManager opaque struct definition
252  * Leave outside of android namespace because it's NDK struct
253  */
254 struct ACameraManager {
ACameraManagerACameraManager255     ACameraManager() :
256             mGlobalManager(android::acam::CameraManagerGlobal::getInstance()) {}
257     ~ACameraManager();
258     camera_status_t getCameraIdList(ACameraIdList** cameraIdList);
259     static void     deleteCameraIdList(ACameraIdList* cameraIdList);
260 
261     camera_status_t getCameraCharacteristics(
262             const char* cameraId, android::sp<ACameraMetadata>* characteristics);
263 
264     camera_status_t openCamera(const char* cameraId,
265                                ACameraDevice_StateCallbacks* callback,
266                                /*out*/ACameraDevice** device);
267     camera_status_t getTagFromName(const char *cameraId, const char *name, uint32_t *tag);
268 
269   private:
270     enum {
271         kCameraIdListNotInit = -1
272     };
273     android::Mutex         mLock;
274     std::shared_ptr<android::acam::CameraManagerGlobal> mGlobalManager;
275 };
276 
277 #endif //_ACAMERA_MANAGER_H
278