You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
288 lines
11 KiB
288 lines
11 KiB
4 months ago
|
/*
|
||
|
* Copyright (C) 2015 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.
|
||
|
*/
|
||
|
|
||
|
#ifndef _ACAMERA_MANAGER_H
|
||
|
#define _ACAMERA_MANAGER_H
|
||
|
|
||
|
#include <camera/NdkCameraManager.h>
|
||
|
|
||
|
#include <android-base/parseint.h>
|
||
|
#include <android/hardware/ICameraService.h>
|
||
|
#include <android/hardware/BnCameraServiceListener.h>
|
||
|
#include <camera/CameraMetadata.h>
|
||
|
#include <binder/IServiceManager.h>
|
||
|
#include <utils/StrongPointer.h>
|
||
|
#include <utils/Mutex.h>
|
||
|
|
||
|
#include <media/stagefright/foundation/ALooper.h>
|
||
|
#include <media/stagefright/foundation/AHandler.h>
|
||
|
#include <media/stagefright/foundation/AMessage.h>
|
||
|
|
||
|
#include <set>
|
||
|
#include <map>
|
||
|
|
||
|
namespace android {
|
||
|
namespace acam {
|
||
|
|
||
|
/**
|
||
|
* Per-process singleton instance of CameraManger. Shared by all ACameraManager
|
||
|
* instances. Created when first ACameraManager is created and destroyed when
|
||
|
* all ACameraManager instances are deleted.
|
||
|
*
|
||
|
* TODO: maybe CameraManagerGlobal is better suited in libcameraclient?
|
||
|
*/
|
||
|
class CameraManagerGlobal final : public RefBase {
|
||
|
public:
|
||
|
static CameraManagerGlobal& getInstance();
|
||
|
sp<hardware::ICameraService> getCameraService();
|
||
|
|
||
|
void registerAvailabilityCallback(
|
||
|
const ACameraManager_AvailabilityCallbacks *callback);
|
||
|
void unregisterAvailabilityCallback(
|
||
|
const ACameraManager_AvailabilityCallbacks *callback);
|
||
|
|
||
|
void registerExtendedAvailabilityCallback(
|
||
|
const ACameraManager_ExtendedAvailabilityCallbacks* callback);
|
||
|
void unregisterExtendedAvailabilityCallback(
|
||
|
const ACameraManager_ExtendedAvailabilityCallbacks* callback);
|
||
|
|
||
|
/**
|
||
|
* Return camera IDs that support camera2
|
||
|
*/
|
||
|
void getCameraIdList(std::vector<String8> *cameraIds);
|
||
|
|
||
|
private:
|
||
|
sp<hardware::ICameraService> mCameraService;
|
||
|
const int kCameraServicePollDelay = 500000; // 0.5s
|
||
|
const char* kCameraServiceName = "media.camera";
|
||
|
Mutex mLock;
|
||
|
|
||
|
template<class T>
|
||
|
void registerAvailCallback(const T *callback);
|
||
|
|
||
|
class DeathNotifier : public IBinder::DeathRecipient {
|
||
|
public:
|
||
|
explicit DeathNotifier(CameraManagerGlobal* cm) : mCameraManager(cm) {}
|
||
|
protected:
|
||
|
// IBinder::DeathRecipient implementation
|
||
|
virtual void binderDied(const wp<IBinder>& who);
|
||
|
private:
|
||
|
const wp<CameraManagerGlobal> mCameraManager;
|
||
|
};
|
||
|
sp<DeathNotifier> mDeathNotifier;
|
||
|
|
||
|
class CameraServiceListener final : public hardware::BnCameraServiceListener {
|
||
|
public:
|
||
|
explicit CameraServiceListener(CameraManagerGlobal* cm) : mCameraManager(cm) {}
|
||
|
virtual binder::Status onStatusChanged(int32_t status, const String16& cameraId);
|
||
|
virtual binder::Status onPhysicalCameraStatusChanged(int32_t status,
|
||
|
const String16& cameraId, const String16& physicalCameraId);
|
||
|
|
||
|
// Torch API not implemented yet
|
||
|
virtual binder::Status onTorchStatusChanged(int32_t, const String16&) {
|
||
|
return binder::Status::ok();
|
||
|
}
|
||
|
|
||
|
virtual binder::Status onCameraAccessPrioritiesChanged();
|
||
|
virtual binder::Status onCameraOpened(const String16&, const String16&) {
|
||
|
return binder::Status::ok();
|
||
|
}
|
||
|
virtual binder::Status onCameraClosed(const String16&) {
|
||
|
return binder::Status::ok();
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
const wp<CameraManagerGlobal> mCameraManager;
|
||
|
};
|
||
|
sp<CameraServiceListener> mCameraServiceListener;
|
||
|
|
||
|
// Wrapper of ACameraManager_AvailabilityCallbacks so we can store it in std::set
|
||
|
struct Callback {
|
||
|
explicit Callback(const ACameraManager_AvailabilityCallbacks *callback) :
|
||
|
mAvailable(callback->onCameraAvailable),
|
||
|
mUnavailable(callback->onCameraUnavailable),
|
||
|
mAccessPriorityChanged(nullptr),
|
||
|
mPhysicalCamAvailable(nullptr),
|
||
|
mPhysicalCamUnavailable(nullptr),
|
||
|
mContext(callback->context) {}
|
||
|
|
||
|
explicit Callback(const ACameraManager_ExtendedAvailabilityCallbacks *callback) :
|
||
|
mAvailable(callback->availabilityCallbacks.onCameraAvailable),
|
||
|
mUnavailable(callback->availabilityCallbacks.onCameraUnavailable),
|
||
|
mAccessPriorityChanged(callback->onCameraAccessPrioritiesChanged),
|
||
|
mPhysicalCamAvailable(callback->onPhysicalCameraAvailable),
|
||
|
mPhysicalCamUnavailable(callback->onPhysicalCameraUnavailable),
|
||
|
mContext(callback->availabilityCallbacks.context) {}
|
||
|
|
||
|
bool operator == (const Callback& other) const {
|
||
|
return (mAvailable == other.mAvailable &&
|
||
|
mUnavailable == other.mUnavailable &&
|
||
|
mAccessPriorityChanged == other.mAccessPriorityChanged &&
|
||
|
mPhysicalCamAvailable == other.mPhysicalCamAvailable &&
|
||
|
mPhysicalCamUnavailable == other.mPhysicalCamUnavailable &&
|
||
|
mContext == other.mContext);
|
||
|
}
|
||
|
bool operator != (const Callback& other) const {
|
||
|
return !(*this == other);
|
||
|
}
|
||
|
bool operator < (const Callback& other) const {
|
||
|
if (*this == other) return false;
|
||
|
if (mContext != other.mContext) return mContext < other.mContext;
|
||
|
if (mPhysicalCamAvailable != other.mPhysicalCamAvailable) {
|
||
|
return mPhysicalCamAvailable < other.mPhysicalCamAvailable;
|
||
|
}
|
||
|
if (mPhysicalCamUnavailable != other.mPhysicalCamUnavailable) {
|
||
|
return mPhysicalCamUnavailable < other.mPhysicalCamUnavailable;
|
||
|
}
|
||
|
if (mAccessPriorityChanged != other.mAccessPriorityChanged) {
|
||
|
return mAccessPriorityChanged < other.mAccessPriorityChanged;
|
||
|
}
|
||
|
if (mAvailable != other.mAvailable) return mAvailable < other.mAvailable;
|
||
|
return mUnavailable < other.mUnavailable;
|
||
|
}
|
||
|
bool operator > (const Callback& other) const {
|
||
|
return (*this != other && !(*this < other));
|
||
|
}
|
||
|
ACameraManager_AvailabilityCallback mAvailable;
|
||
|
ACameraManager_AvailabilityCallback mUnavailable;
|
||
|
ACameraManager_AccessPrioritiesChangedCallback mAccessPriorityChanged;
|
||
|
ACameraManager_PhysicalCameraAvailabilityCallback mPhysicalCamAvailable;
|
||
|
ACameraManager_PhysicalCameraAvailabilityCallback mPhysicalCamUnavailable;
|
||
|
void* mContext;
|
||
|
};
|
||
|
|
||
|
android::Condition mCallbacksCond;
|
||
|
size_t mPendingCallbackCnt = 0;
|
||
|
void onCallbackCalled();
|
||
|
void drainPendingCallbacksLocked();
|
||
|
|
||
|
std::set<Callback> mCallbacks;
|
||
|
|
||
|
// definition of handler and message
|
||
|
enum {
|
||
|
kWhatSendSingleCallback,
|
||
|
kWhatSendSingleAccessCallback,
|
||
|
kWhatSendSinglePhysicalCameraCallback,
|
||
|
};
|
||
|
static const char* kCameraIdKey;
|
||
|
static const char* kPhysicalCameraIdKey;
|
||
|
static const char* kCallbackFpKey;
|
||
|
static const char* kContextKey;
|
||
|
static const nsecs_t kCallbackDrainTimeout;
|
||
|
class CallbackHandler : public AHandler {
|
||
|
public:
|
||
|
CallbackHandler(wp<CameraManagerGlobal> parent) : mParent(parent) {}
|
||
|
void onMessageReceived(const sp<AMessage> &msg) override;
|
||
|
|
||
|
private:
|
||
|
wp<CameraManagerGlobal> mParent;
|
||
|
void notifyParent();
|
||
|
void onMessageReceivedInternal(const sp<AMessage> &msg);
|
||
|
};
|
||
|
sp<CallbackHandler> mHandler;
|
||
|
sp<ALooper> mCbLooper; // Looper thread where callbacks actually happen on
|
||
|
|
||
|
sp<hardware::ICameraService> getCameraServiceLocked();
|
||
|
void onCameraAccessPrioritiesChanged();
|
||
|
void onStatusChanged(int32_t status, const String8& cameraId);
|
||
|
void onStatusChangedLocked(int32_t status, const String8& cameraId);
|
||
|
void onStatusChanged(int32_t status, const String8& cameraId, const String8& physicalCameraId);
|
||
|
void onStatusChangedLocked(int32_t status, const String8& cameraId,
|
||
|
const String8& physicalCameraId);
|
||
|
// Utils for status
|
||
|
static bool validStatus(int32_t status);
|
||
|
static bool isStatusAvailable(int32_t status);
|
||
|
bool supportsCamera2ApiLocked(const String8 &cameraId);
|
||
|
|
||
|
// The sort logic must match the logic in
|
||
|
// libcameraservice/common/CameraProviderManager.cpp::getAPI1CompatibleCameraDeviceIds
|
||
|
struct CameraIdComparator {
|
||
|
bool operator()(const String8& a, const String8& b) const {
|
||
|
uint32_t aUint = 0, bUint = 0;
|
||
|
bool aIsUint = base::ParseUint(a.c_str(), &aUint);
|
||
|
bool bIsUint = base::ParseUint(b.c_str(), &bUint);
|
||
|
|
||
|
// Uint device IDs first
|
||
|
if (aIsUint && bIsUint) {
|
||
|
return aUint < bUint;
|
||
|
} else if (aIsUint) {
|
||
|
return true;
|
||
|
} else if (bIsUint) {
|
||
|
return false;
|
||
|
}
|
||
|
// Simple string compare if both id are not uint
|
||
|
return a < b;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
struct StatusAndHAL3Support {
|
||
|
private:
|
||
|
int32_t status = hardware::ICameraServiceListener::STATUS_NOT_PRESENT;
|
||
|
mutable std::mutex mLock;
|
||
|
std::set<String8> unavailablePhysicalIds;
|
||
|
public:
|
||
|
const bool supportsHAL3 = false;
|
||
|
StatusAndHAL3Support(int32_t st, bool HAL3support):
|
||
|
status(st), supportsHAL3(HAL3support) { };
|
||
|
StatusAndHAL3Support() = default;
|
||
|
|
||
|
bool addUnavailablePhysicalId(const String8& physicalCameraId);
|
||
|
bool removeUnavailablePhysicalId(const String8& physicalCameraId);
|
||
|
int32_t getStatus();
|
||
|
void updateStatus(int32_t newStatus);
|
||
|
std::set<String8> getUnavailablePhysicalIds();
|
||
|
};
|
||
|
|
||
|
// Map camera_id -> status
|
||
|
std::map<String8, StatusAndHAL3Support, CameraIdComparator> mDeviceStatusMap;
|
||
|
|
||
|
// For the singleton instance
|
||
|
static Mutex sLock;
|
||
|
static CameraManagerGlobal* sInstance;
|
||
|
CameraManagerGlobal() {};
|
||
|
~CameraManagerGlobal();
|
||
|
};
|
||
|
|
||
|
} // namespace acam;
|
||
|
} // namespace android;
|
||
|
|
||
|
/**
|
||
|
* ACameraManager opaque struct definition
|
||
|
* Leave outside of android namespace because it's NDK struct
|
||
|
*/
|
||
|
struct ACameraManager {
|
||
|
ACameraManager() :
|
||
|
mGlobalManager(&(android::acam::CameraManagerGlobal::getInstance())) {}
|
||
|
~ACameraManager();
|
||
|
camera_status_t getCameraIdList(ACameraIdList** cameraIdList);
|
||
|
static void deleteCameraIdList(ACameraIdList* cameraIdList);
|
||
|
|
||
|
camera_status_t getCameraCharacteristics(
|
||
|
const char* cameraId, android::sp<ACameraMetadata>* characteristics);
|
||
|
camera_status_t openCamera(const char* cameraId,
|
||
|
ACameraDevice_StateCallbacks* callback,
|
||
|
/*out*/ACameraDevice** device);
|
||
|
|
||
|
private:
|
||
|
enum {
|
||
|
kCameraIdListNotInit = -1
|
||
|
};
|
||
|
android::Mutex mLock;
|
||
|
android::sp<android::acam::CameraManagerGlobal> mGlobalManager;
|
||
|
};
|
||
|
|
||
|
#endif //_ACAMERA_MANAGER_H
|