blob: a721d287f48b431b0a761f95132ed1e92a70af2c [file] [log] [blame]
/*
* Copyright (C) 2022 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.
*/
#include "AidlProviderInfo.h"
#include "common/HalConversionsTemplated.h"
#include "common/CameraProviderInfoTemplated.h"
#include <com_android_internal_camera_flags.h>
#include <cutils/properties.h>
#include <aidlcommonsupport/NativeHandle.h>
#include <android_companion_virtualdevice_flags.h>
#include <android/binder_manager.h>
#include <android/hardware/ICameraService.h>
#include <camera_metadata_hidden.h>
#include "device3/ZoomRatioMapper.h"
#include <utils/SessionConfigurationUtils.h>
#include <utils/Trace.h>
namespace {
const bool kEnableLazyHal(property_get_bool("ro.camera.enableLazyHal", false));
} // anonymous namespace
namespace android {
namespace SessionConfigurationUtils = ::android::camera3::SessionConfigurationUtils;
namespace flags = com::android::internal::camera::flags;
namespace vd_flags = android::companion::virtualdevice::flags;
using namespace aidl::android::hardware;
using namespace hardware::camera;
using hardware::camera2::utils::CameraIdAndSessionConfiguration;
using hardware::ICameraService;
using HalDeviceStatusType = aidl::android::hardware::camera::common::CameraDeviceStatus;
using ICameraProvider = aidl::android::hardware::camera::provider::ICameraProvider;
using StatusListener = CameraProviderManager::StatusListener;
static status_t mapExceptionCodeToStatusT(binder_exception_t binderException) {
switch (binderException) {
case EX_NONE:
return OK;
case EX_ILLEGAL_ARGUMENT:
case EX_NULL_POINTER:
case EX_BAD_PARCELABLE:
case EX_ILLEGAL_STATE:
return BAD_VALUE;
case EX_UNSUPPORTED_OPERATION:
return INVALID_OPERATION;
case EX_TRANSACTION_FAILED:
return DEAD_OBJECT;
default:
return UNKNOWN_ERROR;
}
}
status_t AidlProviderInfo::mapToStatusT(const ndk::ScopedAStatus& s) {
using Status = aidl::android::hardware::camera::common::Status;
auto exceptionCode = s.getExceptionCode();
if (exceptionCode != EX_SERVICE_SPECIFIC) {
return mapExceptionCodeToStatusT(exceptionCode);
}
Status st = static_cast<Status>(s.getServiceSpecificError());
switch (st) {
case Status::OK:
return OK;
case Status::ILLEGAL_ARGUMENT:
return BAD_VALUE;
case Status::CAMERA_IN_USE:
return -EBUSY;
case Status::MAX_CAMERAS_IN_USE:
return -EUSERS;
case Status::OPERATION_NOT_SUPPORTED:
return INVALID_OPERATION;
case Status::CAMERA_DISCONNECTED:
return DEAD_OBJECT;
case Status::INTERNAL_ERROR:
return INVALID_OPERATION;
}
ALOGW("Unexpected HAL status code %d", static_cast<int>(st));
return INVALID_OPERATION;
}
AidlProviderInfo::AidlProviderInfo(
const std::string &providerName,
const std::string &providerInstance,
CameraProviderManager *manager) :
CameraProviderManager::ProviderInfo(providerName, providerInstance, manager) {}
status_t AidlProviderInfo::initializeAidlProvider(
std::shared_ptr<ICameraProvider>& interface, int64_t currentDeviceState) {
using aidl::android::hardware::camera::provider::ICameraProvider;
std::string parsedProviderName = mProviderName;
if (flags::lazy_aidl_wait_for_service()) {
parsedProviderName =
mProviderName.substr(std::string(ICameraProvider::descriptor).size() + 1);
}
status_t res = parseProviderName(parsedProviderName, &mType, &mId);
if (res != OK) {
ALOGE("%s: Invalid provider name, ignoring", __FUNCTION__);
return BAD_VALUE;
}
ALOGI("Connecting to new camera provider: %s, isRemote? %d",
mProviderName.c_str(), interface->isRemote());
// cameraDeviceStatusChange callbacks may be called (and causing new devices added)
// before setCallback returns
mCallbacks =
ndk::SharedRefBase::make<AidlProviderCallbacks>(this);
ndk::ScopedAStatus status =
interface->setCallback(mCallbacks);
if (!status.isOk()) {
ALOGE("%s: Transaction error setting up callbacks with camera provider '%s': %s",
__FUNCTION__, mProviderName.c_str(), status.getMessage());
return mapToStatusT(status);
}
mDeathRecipient = ndk::ScopedAIBinder_DeathRecipient(AIBinder_DeathRecipient_new(binderDied));
if (!vd_flags::virtual_camera_service_discovery() || interface->isRemote()) {
binder_status_t link =
AIBinder_linkToDeath(interface->asBinder().get(), mDeathRecipient.get(), this);
if (link != STATUS_OK) {
ALOGW("%s: Unable to link to provider '%s' death notifications (%d)", __FUNCTION__,
mProviderName.c_str(), link);
return DEAD_OBJECT;
}
}
if (!kEnableLazyHal) {
// Save HAL reference indefinitely
mSavedInterface = interface;
} else {
mActiveInterface = interface;
}
ALOGV("%s: Setting device state for %s: 0x%" PRIx64,
__FUNCTION__, mProviderName.c_str(), mDeviceState);
notifyDeviceStateChange(currentDeviceState);
res = setUpVendorTags();
if (res != OK) {
ALOGE("%s: Unable to set up vendor tags from provider '%s'",
__FUNCTION__, mProviderName.c_str());
return res;
}
// Get initial list of camera devices, if any
std::vector<std::string> devices;
std::vector<std::string> retDevices;
status = interface->getCameraIdList(&retDevices);
if (!status.isOk()) {
ALOGE("%s: Transaction error in getting camera ID list from provider '%s': %s",
__FUNCTION__, mProviderName.c_str(), status.getMessage());
return mapToStatusT(status);
}
for (auto& name : retDevices) {
uint16_t major, minor;
std::string type, id;
status_t res = parseDeviceName(name, &major, &minor, &type, &id);
if (res != OK) {
ALOGE("%s: Error parsing deviceName: %s: %d", __FUNCTION__, name.c_str(), res);
return res;
} else {
devices.push_back(name);
mProviderPublicCameraIds.push_back(id);
}
}
// Get list of concurrent streaming camera device combinations
res = getConcurrentCameraIdsInternalLocked(interface);
if (res != OK) {
return res;
}
mSetTorchModeSupported = true;
mIsRemote = interface->isRemote();
initializeProviderInfoCommon(devices);
return OK;
}
void AidlProviderInfo::binderDied(void *cookie) {
AidlProviderInfo *provider = reinterpret_cast<AidlProviderInfo *>(cookie);
ALOGI("Camera provider '%s' has died; removing it", provider->mProviderInstance.c_str());
provider->mManager->removeProvider(provider->mProviderInstance);
}
status_t AidlProviderInfo::setUpVendorTags() {
if (mVendorTagDescriptor != nullptr)
return OK;
std::vector<camera::common::VendorTagSection> vts;
::ndk::ScopedAStatus status;
const std::shared_ptr<ICameraProvider> interface = startProviderInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
status = interface->getVendorTags(&vts);
if (!status.isOk()) {
ALOGE("%s: Transaction error getting vendor tags from provider '%s': %s",
__FUNCTION__, mProviderName.c_str(), status.getMessage());
return mapToStatusT(status);
}
// Read all vendor tag definitions into a descriptor
status_t res;
if ((res =
IdlVendorTagDescriptor::
createDescriptorFromIdl<std::vector<camera::common::VendorTagSection>,
camera::common::VendorTagSection>(vts, /*out*/mVendorTagDescriptor))
!= OK) {
ALOGE("%s: Could not generate descriptor from vendor tag operations,"
"received error %s (%d). Camera clients will not be able to use"
"vendor tags", __FUNCTION__, strerror(res), res);
return res;
}
return OK;
}
status_t AidlProviderInfo::notifyDeviceStateChange(int64_t newDeviceState) {
mDeviceState = newDeviceState;
// Check if the provider is currently active - not going to start it up for this notification
auto interface = mSavedInterface != nullptr ? mSavedInterface : mActiveInterface.lock();
if (interface != nullptr) {
// Send current device state
interface->notifyDeviceStateChange(mDeviceState);
}
return OK;
}
bool AidlProviderInfo::successfullyStartedProviderInterface() {
return startProviderInterface() != nullptr;
}
std::shared_ptr<camera::device::ICameraDevice>
AidlProviderInfo::startDeviceInterface(const std::string &name) {
::ndk::ScopedAStatus status;
std::shared_ptr<camera::device::ICameraDevice> cameraInterface;
const std::shared_ptr<ICameraProvider> interface = startProviderInterface();
if (interface == nullptr) {
return nullptr;
}
status = interface->getCameraDeviceInterface(name, &cameraInterface);
if (!status.isOk()) {
ALOGE("%s: Transaction error trying to obtain interface for camera device %s: %s",
__FUNCTION__, name.c_str(), status.getMessage());
return nullptr;
}
return cameraInterface;
}
const std::shared_ptr<ICameraProvider> AidlProviderInfo::startProviderInterface() {
ATRACE_CALL();
ALOGV("Request to start camera provider: %s", mProviderName.c_str());
if (mSavedInterface != nullptr) {
return mSavedInterface;
}
if (!kEnableLazyHal) {
ALOGE("Bad provider state! Should not be here on a non-lazy HAL!");
return nullptr;
}
auto interface = mActiveInterface.lock();
if (interface != nullptr) {
ALOGV("Camera provider (%s) already in use. Re-using instance.", mProviderName.c_str());
return interface;
}
// Try to get service without starting
interface = ICameraProvider::fromBinder(
ndk::SpAIBinder(AServiceManager_checkService(mProviderName.c_str())));
if (interface != nullptr) {
// Service is already running. Cache and return.
mActiveInterface = interface;
return interface;
}
ALOGV("Camera provider actually needs restart, calling getService(%s)", mProviderName.c_str());
interface = mManager->mAidlServiceProxy->getService(mProviderName);
if (interface == nullptr) {
ALOGE("%s: %s service not started", __FUNCTION__, mProviderName.c_str());
return nullptr;
}
// Set all devices as ENUMERATING, provider should update status
// to PRESENT after initializing.
// This avoids failing getCameraDeviceInterface_V3_x before devices
// are ready.
for (auto& device : mDevices) {
device->mIsDeviceAvailable = false;
}
interface->setCallback(mCallbacks);
auto link = AIBinder_linkToDeath(interface->asBinder().get(), mDeathRecipient.get(),
this);
if (link != STATUS_OK) {
ALOGW("%s: Unable to link to provider '%s' death notifications",
__FUNCTION__, mProviderName.c_str());
mManager->removeProvider(mProviderInstance);
return nullptr;
}
// Send current device state
interface->notifyDeviceStateChange(mDeviceState);
// Cache interface to return early for future calls.
mActiveInterface = interface;
return interface;
}
::ndk::ScopedAStatus AidlProviderInfo::AidlProviderCallbacks::cameraDeviceStatusChange(
const std::string& cameraDeviceName,
HalDeviceStatusType newStatus) {
sp<AidlProviderInfo> parent = mParent.promote();
if (parent == nullptr) {
ALOGE("%s: Parent provider not alive", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return parent->cameraDeviceStatusChange(cameraDeviceName, newStatus);
}
::ndk::ScopedAStatus AidlProviderInfo::AidlProviderCallbacks::torchModeStatusChange(
const std::string& cameraDeviceName,
aidl::android::hardware::camera::common::TorchModeStatus newStatus) {
sp<AidlProviderInfo> parent = mParent.promote();
if (parent == nullptr) {
ALOGE("%s: Parent provider not alive", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return parent->torchModeStatusChange(cameraDeviceName, newStatus);
};
::ndk::ScopedAStatus AidlProviderInfo::AidlProviderCallbacks::physicalCameraDeviceStatusChange(
const std::string& cameraDeviceName,
const std::string& physicalCameraDeviceName,
HalDeviceStatusType newStatus) {
sp<AidlProviderInfo> parent = mParent.promote();
if (parent == nullptr) {
ALOGE("%s: Parent provider not alive", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return parent->physicalCameraDeviceStatusChange(cameraDeviceName, physicalCameraDeviceName,
newStatus);
};
::ndk::ScopedAStatus AidlProviderInfo::cameraDeviceStatusChange(const std::string& cameraDeviceName,
HalDeviceStatusType newStatus) {
cameraDeviceStatusChangeInternal(cameraDeviceName, HalToFrameworkCameraDeviceStatus(newStatus));
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus AidlProviderInfo::torchModeStatusChange(const std::string& cameraDeviceName,
aidl::android::hardware::camera::common::TorchModeStatus newStatus) {
torchModeStatusChangeInternal(cameraDeviceName, HalToFrameworkTorchModeStatus(newStatus));
return ::ndk::ScopedAStatus::ok();
};
::ndk::ScopedAStatus AidlProviderInfo::physicalCameraDeviceStatusChange(
const std::string& cameraDeviceName,
const std::string& physicalCameraDeviceName,
HalDeviceStatusType newStatus) {
physicalCameraDeviceStatusChangeInternal(cameraDeviceName, physicalCameraDeviceName,
HalToFrameworkCameraDeviceStatus(newStatus));
return ::ndk::ScopedAStatus::ok();
};
std::unique_ptr<CameraProviderManager::ProviderInfo::DeviceInfo>
AidlProviderInfo::initializeDeviceInfo(
const std::string &name, const metadata_vendor_id_t tagId,
const std::string &id, uint16_t /*minorVersion*/) {
::ndk::ScopedAStatus status;
auto cameraInterface = startDeviceInterface(name);
if (cameraInterface == nullptr) return nullptr;
camera::common::CameraResourceCost resourceCost;
status = cameraInterface->getResourceCost(&resourceCost);
if (!status.isOk()) {
ALOGE("%s: Unable to obtain resource costs for camera device %s: %s", __FUNCTION__,
name.c_str(), status.getMessage());
return nullptr;
}
for (auto& conflictName : resourceCost.conflictingDevices) {
uint16_t major, minor;
std::string type, id;
status_t res = parseDeviceName(conflictName, &major, &minor, &type, &id);
if (res != OK) {
ALOGE("%s: Failed to parse conflicting device %s", __FUNCTION__, conflictName.c_str());
return nullptr;
}
conflictName = id;
}
int32_t interfaceVersion = 0;
status = cameraInterface->getInterfaceVersion(&interfaceVersion);
if (!status.isOk()) {
ALOGE("%s: Unable to obtain interface version for camera device %s: %s", __FUNCTION__,
id.c_str(), status.getMessage());
return nullptr;
}
return std::unique_ptr<DeviceInfo3>(
new AidlDeviceInfo3(name, tagId, id, static_cast<uint16_t>(interfaceVersion),
HalToFrameworkResourceCost(resourceCost), this,
mProviderPublicCameraIds, cameraInterface));
}
status_t AidlProviderInfo::reCacheConcurrentStreamingCameraIdsLocked() {
// Check if the provider is currently active - not going to start it up for this notification
auto interface = mSavedInterface != nullptr ? mSavedInterface : mActiveInterface.lock();
if (interface == nullptr) {
ALOGE("%s: camera provider interface for %s is not valid", __FUNCTION__,
mProviderName.c_str());
return INVALID_OPERATION;
}
return getConcurrentCameraIdsInternalLocked(interface);
}
status_t AidlProviderInfo::getConcurrentCameraIdsInternalLocked(
std::shared_ptr<ICameraProvider> &interface) {
if (interface == nullptr) {
ALOGE("%s: null interface provided", __FUNCTION__);
return BAD_VALUE;
}
std::vector<aidl::android::hardware::camera::provider::ConcurrentCameraIdCombination> combs;
::ndk::ScopedAStatus status = interface->getConcurrentCameraIds(&combs);
if (!status.isOk()) {
ALOGE("%s: Transaction error in getting concurrent camera ID list from provider '%s'",
__FUNCTION__, mProviderName.c_str());
return mapToStatusT(status);
}
mConcurrentCameraIdCombinations.clear();
for (const auto& combination : combs) {
std::unordered_set<std::string> deviceIds;
for (const auto &cameraDeviceId : combination.combination) {
deviceIds.insert(cameraDeviceId);
}
mConcurrentCameraIdCombinations.push_back(std::move(deviceIds));
}
return OK;
}
AidlProviderInfo::AidlDeviceInfo3::AidlDeviceInfo3(
const std::string& name,
const metadata_vendor_id_t tagId,
const std::string &id, uint16_t minorVersion,
const CameraResourceCost& resourceCost,
sp<CameraProviderManager::ProviderInfo> parentProvider,
const std::vector<std::string>& publicCameraIds,
std::shared_ptr<aidl::android::hardware::camera::device::ICameraDevice> interface) :
DeviceInfo3(name, tagId, id, minorVersion, resourceCost, parentProvider, publicCameraIds) {
// Get camera characteristics and initialize flash unit availability
aidl::android::hardware::camera::device::CameraMetadata chars;
::ndk::ScopedAStatus status = interface->getCameraCharacteristics(&chars);
std::vector<uint8_t> &metadata = chars.metadata;
camera_metadata_t *buffer = reinterpret_cast<camera_metadata_t*>(metadata.data());
size_t expectedSize = metadata.size();
int resV = validate_camera_metadata_structure(buffer, &expectedSize);
if (resV == OK || resV == CAMERA_METADATA_VALIDATION_SHIFTED) {
set_camera_metadata_vendor_id(buffer, mProviderTagid);
mCameraCharacteristics = buffer;
} else {
ALOGE("%s: Malformed camera metadata received from HAL", __FUNCTION__);
return;
}
if (!status.isOk()) {
ALOGE("%s: Transaction error getting camera characteristics for device %s"
" to check for a flash unit: %s", __FUNCTION__, id.c_str(),
status.getMessage());
return;
}
if (mCameraCharacteristics.exists(ANDROID_INFO_DEVICE_STATE_ORIENTATIONS)) {
const auto &stateMap = mCameraCharacteristics.find(ANDROID_INFO_DEVICE_STATE_ORIENTATIONS);
if ((stateMap.count > 0) && ((stateMap.count % 2) == 0)) {
for (size_t i = 0; i < stateMap.count; i += 2) {
mDeviceStateOrientationMap.emplace(stateMap.data.i64[i], stateMap.data.i64[i+1]);
}
} else {
ALOGW("%s: Invalid ANDROID_INFO_DEVICE_STATE_ORIENTATIONS map size: %zu", __FUNCTION__,
stateMap.count);
}
}
mCompositeJpegRDisabled = mCameraCharacteristics.exists(
ANDROID_JPEGR_AVAILABLE_JPEG_R_STREAM_CONFIGURATIONS);
mSystemCameraKind = getSystemCameraKind();
status_t res = fixupMonochromeTags();
if (OK != res) {
ALOGE("%s: Unable to fix up monochrome tags based for older HAL version: %s (%d)",
__FUNCTION__, strerror(-res), res);
return;
}
if (flags::camera_manual_flash_strength_control()) {
res = fixupManualFlashStrengthControlTags(mCameraCharacteristics);
if (OK != res) {
ALOGE("%s: Unable to fix up manual flash strength control tags: %s (%d)",
__FUNCTION__, strerror(-res), res);
return;
}
}
auto stat = addDynamicDepthTags();
if (OK != stat) {
ALOGE("%s: Failed appending dynamic depth tags: %s (%d)", __FUNCTION__, strerror(-stat),
stat);
}
res = deriveHeicTags();
if (OK != res) {
ALOGE("%s: Unable to derive HEIC tags based on camera and media capabilities: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
res = deriveJpegRTags();
if (OK != res) {
ALOGE("%s: Unable to derive Jpeg/R tags based on camera and media capabilities: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
using camera3::SessionConfigurationUtils::supportsUltraHighResolutionCapture;
if (supportsUltraHighResolutionCapture(mCameraCharacteristics)) {
status_t status = addDynamicDepthTags(/*maxResolution*/true);
if (OK != status) {
ALOGE("%s: Failed appending dynamic depth tags for maximum resolution mode: %s (%d)",
__FUNCTION__, strerror(-status), status);
}
status = deriveHeicTags(/*maxResolution*/true);
if (OK != status) {
ALOGE("%s: Unable to derive HEIC tags based on camera and media capabilities for"
"maximum resolution mode: %s (%d)", __FUNCTION__, strerror(-status), status);
}
status = deriveJpegRTags(/*maxResolution*/true);
if (OK != status) {
ALOGE("%s: Unable to derive Jpeg/R tags based on camera and media capabilities for"
"maximum resolution mode: %s (%d)", __FUNCTION__, strerror(-status), status);
}
}
res = addRotateCropTags();
if (OK != res) {
ALOGE("%s: Unable to add default SCALER_ROTATE_AND_CROP tags: %s (%d)", __FUNCTION__,
strerror(-res), res);
}
res = addAutoframingTags();
if (OK != res) {
ALOGE("%s: Unable to add default AUTOFRAMING tags: %s (%d)", __FUNCTION__,
strerror(-res), res);
}
res = addPreCorrectionActiveArraySize();
if (OK != res) {
ALOGE("%s: Unable to add PRE_CORRECTION_ACTIVE_ARRAY_SIZE: %s (%d)", __FUNCTION__,
strerror(-res), res);
}
res = camera3::ZoomRatioMapper::overrideZoomRatioTags(
&mCameraCharacteristics, &mSupportNativeZoomRatio);
if (OK != res) {
ALOGE("%s: Unable to override zoomRatio related tags: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
res = addReadoutTimestampTag();
if (OK != res) {
ALOGE("%s: Unable to add sensorReadoutTimestamp tag: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
camera_metadata_entry flashAvailable =
mCameraCharacteristics.find(ANDROID_FLASH_INFO_AVAILABLE);
if (flashAvailable.count == 1 &&
flashAvailable.data.u8[0] == ANDROID_FLASH_INFO_AVAILABLE_TRUE) {
mHasFlashUnit = true;
// Fix up flash strength tags for devices without these keys.
res = fixupTorchStrengthTags();
if (OK != res) {
ALOGE("%s: Unable to add default ANDROID_FLASH_INFO_STRENGTH_DEFAULT_LEVEL and"
"ANDROID_FLASH_INFO_STRENGTH_MAXIMUM_LEVEL tags: %s (%d)", __FUNCTION__,
strerror(-res), res);
}
// b/247038031: In case of system_server crash, camera_server is
// restarted as well. If flashlight is turned on before the crash, it
// may be stuck to be on. As a workaround, set torch mode to be OFF.
interface->setTorchMode(false);
} else {
mHasFlashUnit = false;
}
if (flags::feature_combination_query()) {
res = addSessionConfigQueryVersionTag();
if (OK != res) {
ALOGE("%s: Unable to add sessionConfigurationQueryVersion tag: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
}
camera_metadata_entry entry =
mCameraCharacteristics.find(ANDROID_FLASH_INFO_STRENGTH_DEFAULT_LEVEL);
if (entry.count == 1) {
mTorchDefaultStrengthLevel = entry.data.i32[0];
} else {
mTorchDefaultStrengthLevel = 0;
}
entry = mCameraCharacteristics.find(ANDROID_FLASH_INFO_STRENGTH_MAXIMUM_LEVEL);
if (entry.count == 1) {
mTorchMaximumStrengthLevel = entry.data.i32[0];
} else {
mTorchMaximumStrengthLevel = 0;
}
mTorchStrengthLevel = 0;
queryPhysicalCameraIds();
// Get physical camera characteristics if applicable
if (mIsLogicalCamera) {
for (auto& id : mPhysicalIds) {
if (std::find(mPublicCameraIds.begin(), mPublicCameraIds.end(), id) !=
mPublicCameraIds.end()) {
continue;
}
aidl::android::hardware::camera::device::CameraMetadata pChars;
status = interface->getPhysicalCameraCharacteristics(id, &pChars);
if (!status.isOk()) {
ALOGE("%s: Transaction error getting physical camera %s characteristics for "
"logical id %s: %s", __FUNCTION__, id.c_str(), mId.c_str(),
status.getMessage());
return;
}
std::vector<uint8_t> &pMetadata = pChars.metadata;
camera_metadata_t *pBuffer =
reinterpret_cast<camera_metadata_t*>(pMetadata.data());
size_t expectedSize = pMetadata.size();
int res = validate_camera_metadata_structure(pBuffer, &expectedSize);
if (res == OK || res == CAMERA_METADATA_VALIDATION_SHIFTED) {
set_camera_metadata_vendor_id(pBuffer, mProviderTagid);
mPhysicalCameraCharacteristics[id] = pBuffer;
} else {
ALOGE("%s: Malformed camera metadata received from HAL", __FUNCTION__);
return;
}
res = camera3::ZoomRatioMapper::overrideZoomRatioTags(
&mPhysicalCameraCharacteristics[id], &mSupportNativeZoomRatio);
if (OK != res) {
ALOGE("%s: Unable to override zoomRatio related tags: %s (%d)",
__FUNCTION__, strerror(-res), res);
}
if (flags::camera_manual_flash_strength_control()) {
res = fixupManualFlashStrengthControlTags(mPhysicalCameraCharacteristics[id]);
if (OK != res) {
ALOGE("%s: Unable to fix up manual flash strength control tags: %s (%d)",
__FUNCTION__, strerror(-res), res);
return;
}
}
}
}
if (!kEnableLazyHal) {
// Save HAL reference indefinitely
mSavedInterface = interface;
}
}
status_t AidlProviderInfo::AidlDeviceInfo3::setTorchMode(bool enabled) {
const std::shared_ptr<camera::device::ICameraDevice> interface = startDeviceInterface();
::ndk::ScopedAStatus s = interface->setTorchMode(enabled);
if (!s.isOk()) {
ALOGE("%s Unable to set torch mode: %s", __FUNCTION__, s.getMessage());
return mapToStatusT(s);
}
return OK;
}
status_t AidlProviderInfo::AidlDeviceInfo3::turnOnTorchWithStrengthLevel(
int32_t torchStrength) {
const std::shared_ptr<camera::device::ICameraDevice> interface = startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
::ndk::ScopedAStatus s = interface->turnOnTorchWithStrengthLevel(torchStrength);
if (!s.isOk()) {
ALOGE("%s Unable to set torch mode strength %d : %s", __FUNCTION__, torchStrength,
s.getMessage());
return mapToStatusT(s);
}
mTorchStrengthLevel = torchStrength;
return OK;
}
status_t AidlProviderInfo::AidlDeviceInfo3::getTorchStrengthLevel(int32_t *torchStrength) {
if (torchStrength == nullptr) {
return BAD_VALUE;
}
const std::shared_ptr<camera::device::ICameraDevice> interface = startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
::ndk::ScopedAStatus status = interface->getTorchStrengthLevel(torchStrength);
if (!status.isOk()) {
ALOGE("%s: Couldn't get torch strength level: %s", __FUNCTION__, status.getMessage());
return mapToStatusT(status);
}
return OK;
}
std::shared_ptr<aidl::android::hardware::camera::device::ICameraDevice>
AidlProviderInfo::AidlDeviceInfo3::startDeviceInterface() {
Mutex::Autolock l(mDeviceAvailableLock);
std::shared_ptr<camera::device::ICameraDevice> device;
ATRACE_CALL();
if (mSavedInterface == nullptr) {
sp<AidlProviderInfo> parentProvider =
static_cast<AidlProviderInfo *>(mParentProvider.promote().get());
if (parentProvider != nullptr) {
// Wait for lazy HALs to confirm device availability
if (parentProvider->isExternalLazyHAL() && !mIsDeviceAvailable) {
ALOGV("%s: Wait for external device to become available %s",
__FUNCTION__,
mId.c_str());
auto res = mDeviceAvailableSignal.waitRelative(mDeviceAvailableLock,
kDeviceAvailableTimeout);
if (res != OK) {
ALOGE("%s: Failed waiting for device to become available",
__FUNCTION__);
return nullptr;
}
}
device = parentProvider->startDeviceInterface(mName);
}
} else {
device = mSavedInterface;
}
return device;
}
status_t AidlProviderInfo::AidlDeviceInfo3::dumpState(int fd) {
const std::shared_ptr<camera::device::ICameraDevice> interface = startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
const char *args = nullptr;
auto ret = interface->dump(fd, &args, /*numArgs*/0);
if (ret != OK) {
return ret;
}
return OK;
}
status_t AidlProviderInfo::AidlDeviceInfo3::isSessionConfigurationSupported(
const SessionConfiguration &configuration, bool overrideForPerfClass,
bool checkSessionParams, bool *status) {
auto operatingMode = configuration.getOperatingMode();
auto res = SessionConfigurationUtils::checkOperatingMode(operatingMode,
mCameraCharacteristics, mId);
if (!res.isOk()) {
return UNKNOWN_ERROR;
}
camera::device::StreamConfiguration streamConfiguration;
bool earlyExit = false;
camera3::metadataGetter getMetadata = [this](const std::string &id,
bool /*overrideForPerfClass*/) {return this->deviceInfo(id);};
auto bRes = SessionConfigurationUtils::convertToHALStreamCombination(configuration,
mId, mCameraCharacteristics, mCompositeJpegRDisabled, getMetadata,
mPhysicalIds, streamConfiguration, overrideForPerfClass, mProviderTagid,
checkSessionParams, &earlyExit);
if (!bRes.isOk()) {
return UNKNOWN_ERROR;
}
if (earlyExit) {
*status = false;
return OK;
}
const std::shared_ptr<camera::device::ICameraDevice> interface =
startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
::ndk::ScopedAStatus ret;
if (checkSessionParams) {
// Only interface version 1_3 or greater supports
// isStreamCombinationWIthSettingsSupported.
int deviceVersion = HARDWARE_DEVICE_API_VERSION(mVersion.get_major(), mVersion.get_minor());
if (deviceVersion < CAMERA_DEVICE_API_VERSION_1_3) {
ALOGI("%s: Camera device version (major %d, minor %d) doesn't support querying of "
"session configuration!", __FUNCTION__, mVersion.get_major(),
mVersion.get_minor());
return INVALID_OPERATION;
}
if (flags::feature_combination_query()) {
ret = interface->isStreamCombinationWithSettingsSupported(streamConfiguration, status);
} else {
return INVALID_OPERATION;
}
} else {
ret = interface->isStreamCombinationSupported(streamConfiguration, status);
}
if (!ret.isOk()) {
*status = false;
ALOGE("%s: Unexpected binder error: %s", __FUNCTION__, ret.getMessage());
return mapToStatusT(ret);
}
return OK;
}
status_t AidlProviderInfo::AidlDeviceInfo3::createDefaultRequest(
camera3::camera_request_template_t templateId, camera_metadata_t** metadata) {
const std::shared_ptr<camera::device::ICameraDevice> interface =
startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
int deviceVersion = HARDWARE_DEVICE_API_VERSION(mVersion.get_major(), mVersion.get_minor());
if (deviceVersion < CAMERA_DEVICE_API_VERSION_1_3) {
ALOGI("%s: Camera device minor version 0x%x doesn't support creating "
" default request!", __FUNCTION__, mVersion.get_minor());
return INVALID_OPERATION;
}
aidl::android::hardware::camera::device::CameraMetadata request;
using aidl::android::hardware::camera::device::RequestTemplate;
RequestTemplate id;
status_t res = SessionConfigurationUtils::mapRequestTemplateToAidl(
templateId, &id);
if (res != OK) {
return res;
}
if (!flags::feature_combination_query()) {
return INVALID_OPERATION;
}
auto err = interface->constructDefaultRequestSettings(id, &request);
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
const camera_metadata *r =
reinterpret_cast<const camera_metadata_t*>(request.metadata.data());
size_t expectedSize = request.metadata.size();
int ret = validate_camera_metadata_structure(r, &expectedSize);
if (ret == OK || ret == CAMERA_METADATA_VALIDATION_SHIFTED) {
*metadata = clone_camera_metadata(r);
if (*metadata == nullptr) {
ALOGE("%s: Unable to clone camera metadata received from HAL",
__FUNCTION__);
res = UNKNOWN_ERROR;
}
} else {
ALOGE("%s: Malformed camera metadata received from HAL", __FUNCTION__);
res = UNKNOWN_ERROR;
}
return res;
}
status_t AidlProviderInfo::AidlDeviceInfo3::getSessionCharacteristics(
const SessionConfiguration &configuration, bool overrideForPerfClass,
camera3::metadataGetter getMetadata, CameraMetadata *sessionCharacteristics) {
camera::device::StreamConfiguration streamConfiguration;
bool earlyExit = false;
auto res = SessionConfigurationUtils::convertToHALStreamCombination(configuration,
mId, mCameraCharacteristics, mCompositeJpegRDisabled, getMetadata,
mPhysicalIds, streamConfiguration, overrideForPerfClass, mProviderTagid,
/*checkSessionParams*/true, &earlyExit);
if (!res.isOk()) {
return UNKNOWN_ERROR;
}
if (earlyExit) {
return BAD_VALUE;
}
const std::shared_ptr<camera::device::ICameraDevice> interface =
startDeviceInterface();
if (interface == nullptr) {
return DEAD_OBJECT;
}
aidl::android::hardware::camera::device::CameraMetadata chars;
::ndk::ScopedAStatus ret =
interface->getSessionCharacteristics(streamConfiguration, &chars);
std::vector<uint8_t> &metadata = chars.metadata;
camera_metadata_t *buffer = reinterpret_cast<camera_metadata_t*>(metadata.data());
size_t expectedSize = metadata.size();
int resV = validate_camera_metadata_structure(buffer, &expectedSize);
if (resV == OK || resV == CAMERA_METADATA_VALIDATION_SHIFTED) {
set_camera_metadata_vendor_id(buffer, mProviderTagid);
*sessionCharacteristics = buffer;
} else {
ALOGE("%s: Malformed camera metadata received from HAL", __FUNCTION__);
return BAD_VALUE;
}
if (!ret.isOk()) {
ALOGE("%s: Unexpected binder error: %s", __FUNCTION__, ret.getMessage());
return mapToStatusT(ret);
}
return OK;
}
status_t AidlProviderInfo::convertToAidlHALStreamCombinationAndCameraIdsLocked(
const std::vector<CameraIdAndSessionConfiguration> &cameraIdsAndSessionConfigs,
const std::set<std::string>& perfClassPrimaryCameraIds,
int targetSdkVersion,
std::vector<camera::provider::CameraIdAndStreamCombination>
*halCameraIdsAndStreamCombinations,
bool *earlyExit) {
binder::Status bStatus = binder::Status::ok();
std::vector<camera::provider::CameraIdAndStreamCombination> halCameraIdsAndStreamsV;
bool shouldExit = false;
status_t res = OK;
for (auto &cameraIdAndSessionConfig : cameraIdsAndSessionConfigs) {
const std::string& cameraId = cameraIdAndSessionConfig.mCameraId;
camera::device::StreamConfiguration streamConfiguration;
CameraMetadata deviceInfo;
bool overrideForPerfClass =
SessionConfigurationUtils::targetPerfClassPrimaryCamera(
perfClassPrimaryCameraIds, cameraId, targetSdkVersion);
res = mManager->getCameraCharacteristicsLocked(cameraId, overrideForPerfClass, &deviceInfo,
/*overrideToPortrait*/false);
if (res != OK) {
return res;
}
camera3::metadataGetter getMetadata =
[this](const std::string &id, bool overrideForPerfClass) {
CameraMetadata physicalDeviceInfo;
mManager->getCameraCharacteristicsLocked(id, overrideForPerfClass,
&physicalDeviceInfo,
/*overrideToPortrait*/false);
return physicalDeviceInfo;
};
std::vector<std::string> physicalCameraIds;
mManager->isLogicalCameraLocked(cameraId, &physicalCameraIds);
bStatus =
SessionConfigurationUtils::convertToHALStreamCombination(
cameraIdAndSessionConfig.mSessionConfiguration,
cameraId, deviceInfo,
mManager->isCompositeJpegRDisabledLocked(cameraId), getMetadata,
physicalCameraIds, streamConfiguration,
overrideForPerfClass, mProviderTagid,
/*checkSessionParams*/false, &shouldExit);
if (!bStatus.isOk()) {
ALOGE("%s: convertToHALStreamCombination failed", __FUNCTION__);
return INVALID_OPERATION;
}
if (shouldExit) {
*earlyExit = true;
return OK;
}
camera::provider::CameraIdAndStreamCombination halCameraIdAndStream;
halCameraIdAndStream.cameraId = cameraId;
halCameraIdAndStream.streamConfiguration = streamConfiguration;
halCameraIdsAndStreamsV.push_back(halCameraIdAndStream);
}
*halCameraIdsAndStreamCombinations = halCameraIdsAndStreamsV;
return OK;
}
status_t AidlProviderInfo::isConcurrentSessionConfigurationSupported(
const std::vector<CameraIdAndSessionConfiguration> &cameraIdsAndSessionConfigs,
const std::set<std::string>& perfClassPrimaryCameraIds,
int targetSdkVersion, bool *isSupported) {
std::vector<camera::provider::CameraIdAndStreamCombination> halCameraIdsAndStreamCombinations;
bool knowUnsupported = false;
status_t res = convertToAidlHALStreamCombinationAndCameraIdsLocked(
cameraIdsAndSessionConfigs, perfClassPrimaryCameraIds,
targetSdkVersion, &halCameraIdsAndStreamCombinations, &knowUnsupported);
if (res != OK) {
ALOGE("%s unable to convert session configurations provided to HAL stream"
"combinations", __FUNCTION__);
return res;
}
if (knowUnsupported) {
// We got to know the streams aren't valid before doing the HAL
// call itself.
*isSupported = false;
return OK;
}
// Check if the provider is currently active - not going to start it up for this notification
auto interface = mSavedInterface != nullptr ? mSavedInterface : mActiveInterface.lock();
if (interface == nullptr) {
// TODO: This might be some other problem
return INVALID_OPERATION;
}
::ndk::ScopedAStatus status = interface->isConcurrentStreamCombinationSupported(
halCameraIdsAndStreamCombinations, isSupported);
if (!status.isOk()) {
*isSupported = false;
ALOGE("%s: hal interface session configuration query failed", __FUNCTION__);
return mapToStatusT(status);
}
return OK;
}
} //namespace android