blob: e8ef692f0184c3781dcf927db5c316a3aef4a101 [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.
*/
#define LOG_TAG "AidlCamera3-Device"
#define ATRACE_TAG ATRACE_TAG_CAMERA
//#define LOG_NDEBUG 0
//#define LOG_NNDEBUG 0 // Per-frame verbose logging
#ifdef LOG_NNDEBUG
#define ALOGVV(...) ALOGV(__VA_ARGS__)
#else
#define ALOGVV(...) ((void)0)
#endif
// Convenience macro for transient errors
#define CLOGE(fmt, ...) ALOGE("Camera %s: %s: " fmt, mId.c_str(), __FUNCTION__, \
##__VA_ARGS__)
#define CLOGW(fmt, ...) ALOGW("Camera %s: %s: " fmt, mId.c_str(), __FUNCTION__, \
##__VA_ARGS__)
// Convenience macros for transitioning to the error state
#define SET_ERR(fmt, ...) setErrorState( \
"%s: " fmt, __FUNCTION__, \
##__VA_ARGS__)
#define SET_ERR_L(fmt, ...) setErrorStateLocked( \
"%s: " fmt, __FUNCTION__, \
##__VA_ARGS__)
#include <inttypes.h>
#include <utility>
#include <utils/Log.h>
#include <utils/Trace.h>
#include <utils/Timers.h>
#include <cutils/properties.h>
#include <aidl/android/hardware/camera/device/ICameraDeviceSession.h>
#include <aidl/android/hardware/camera/device/ICameraInjectionSession.h>
#include <aidlcommonsupport/NativeHandle.h>
#include <android-base/properties.h>
#include <android/binder_ibinder_platform.h>
#include <android/hardware/camera2/ICameraDeviceUser.h>
#include <camera/StringUtils.h>
#include <com_android_internal_camera_flags.h>
#include "utils/CameraTraces.h"
#include "utils/SessionConfigurationUtils.h"
#include "mediautils/SchedulingPolicyService.h"
#include "device3/Camera3OutputStream.h"
#include "device3/Camera3InputStream.h"
#include "device3/Camera3FakeStream.h"
#include "device3/Camera3SharedOutputStream.h"
#include "device3/aidl/AidlCamera3OutputUtils.h"
#include "device3/aidl/AidlCamera3OfflineSession.h"
#include "CameraService.h"
#include "utils/CameraThreadState.h"
#include "utils/SessionConfigurationUtils.h"
#include "utils/TraceHFR.h"
#include "utils/CameraServiceProxyWrapper.h"
#include "../../common/aidl/AidlProviderInfo.h"
#include <algorithm>
#include "AidlCamera3Device.h"
using namespace android::camera3;
using namespace android::camera3::SessionConfigurationUtils;
using namespace aidl::android::hardware;
using aidl::android::hardware::camera::metadata::SensorPixelMode;
using aidl::android::hardware::camera::metadata::RequestAvailableDynamicRangeProfilesMap;
using aidl::android::hardware::camera::metadata::ScalerAvailableStreamUseCases;
namespace flags = com::android::internal::camera::flags;
const int32_t AIDL_DEVICE_SESSION_V3 = 3;
namespace android {
RequestAvailableDynamicRangeProfilesMap
mapToAidlDynamicProfile(int64_t dynamicRangeProfile) {
return static_cast<RequestAvailableDynamicRangeProfilesMap>(dynamicRangeProfile);
}
aidl::android::hardware::graphics::common::PixelFormat AidlCamera3Device::mapToAidlPixelFormat(
int frameworkFormat) {
return (aidl::android::hardware::graphics::common::PixelFormat) frameworkFormat;
}
aidl::android::hardware::graphics::common::Dataspace AidlCamera3Device::mapToAidlDataspace(
android_dataspace dataSpace) {
return (aidl::android::hardware::graphics::common::Dataspace)dataSpace;
}
aidl::android::hardware::graphics::common::BufferUsage AidlCamera3Device::mapToAidlConsumerUsage(
uint64_t usage) {
return (aidl::android::hardware::graphics::common::BufferUsage)usage;
}
aidl::android::hardware::camera::device::StreamRotation
AidlCamera3Device::mapToAidlStreamRotation(camera_stream_rotation_t rotation) {
switch (rotation) {
case CAMERA_STREAM_ROTATION_0:
return aidl::android::hardware::camera::device::StreamRotation::ROTATION_0;
case CAMERA_STREAM_ROTATION_90:
return aidl::android::hardware::camera::device::StreamRotation::ROTATION_90;
case CAMERA_STREAM_ROTATION_180:
return aidl::android::hardware::camera::device::StreamRotation::ROTATION_180;
case CAMERA_STREAM_ROTATION_270:
return aidl::android::hardware::camera::device::StreamRotation::ROTATION_270;
}
ALOGE("%s: Unknown stream rotation %d", __FUNCTION__, rotation);
return aidl::android::hardware::camera::device::StreamRotation::ROTATION_0;
}
status_t AidlCamera3Device::mapToAidlStreamConfigurationMode(
camera_stream_configuration_mode_t operationMode,
aidl::android::hardware::camera::device::StreamConfigurationMode *mode) {
using StreamConfigurationMode =
aidl::android::hardware::camera::device::StreamConfigurationMode;
if (mode == nullptr) return BAD_VALUE;
if (operationMode < CAMERA_VENDOR_STREAM_CONFIGURATION_MODE_START) {
switch(operationMode) {
case CAMERA_STREAM_CONFIGURATION_NORMAL_MODE:
*mode = StreamConfigurationMode::NORMAL_MODE;
break;
case CAMERA_STREAM_CONFIGURATION_CONSTRAINED_HIGH_SPEED_MODE:
*mode = StreamConfigurationMode::CONSTRAINED_HIGH_SPEED_MODE;
break;
default:
ALOGE("%s: Unknown stream configuration mode %d", __FUNCTION__, operationMode);
return BAD_VALUE;
}
} else {
*mode = static_cast<StreamConfigurationMode>(operationMode);
}
return OK;
}
int AidlCamera3Device::mapToFrameworkFormat(
aidl::android::hardware::graphics::common::PixelFormat pixelFormat) {
return static_cast<uint32_t>(pixelFormat);
}
android_dataspace AidlCamera3Device::mapToFrameworkDataspace(
aidl::android::hardware::graphics::common::Dataspace dataSpace) {
return static_cast<android_dataspace>(dataSpace);
}
uint64_t AidlCamera3Device::mapConsumerToFrameworkUsage(
aidl::android::hardware::graphics::common::BufferUsage usage) {
return (uint64_t)usage;
}
uint64_t AidlCamera3Device::mapProducerToFrameworkUsage(
aidl::android::hardware::graphics::common::BufferUsage usage) {
return (uint64_t)usage;
}
AidlCamera3Device::AidlCamera3Device(
std::shared_ptr<CameraServiceProxyWrapper>& cameraServiceProxyWrapper,
const std::string& id, bool overrideForPerfClass, bool overrideToPortrait,
bool legacyClient) :
Camera3Device(cameraServiceProxyWrapper, id, overrideForPerfClass, overrideToPortrait,
legacyClient) {
mCallbacks = ndk::SharedRefBase::make<AidlCameraDeviceCallbacks>(this);
}
status_t AidlCamera3Device::initialize(sp<CameraProviderManager> manager,
const std::string& monitorTags) {
ATRACE_CALL();
Mutex::Autolock il(mInterfaceLock);
Mutex::Autolock l(mLock);
ALOGV("%s: Initializing AIDL device for camera %s", __FUNCTION__, mId.c_str());
if (mStatus != STATUS_UNINITIALIZED) {
CLOGE("Already initialized!");
return INVALID_OPERATION;
}
if (manager == nullptr) return INVALID_OPERATION;
std::shared_ptr<camera::device::ICameraDeviceSession> session;
ATRACE_BEGIN("CameraHal::openSession");
status_t res = manager->openAidlSession(mId, mCallbacks,
/*out*/ &session);
ATRACE_END();
if (res != OK) {
SET_ERR_L("Could not open camera session: %s (%d)", strerror(-res), res);
return res;
}
if (session == nullptr) {
SET_ERR("Session iface returned is null");
return INVALID_OPERATION;
}
res = manager->getCameraCharacteristics(mId, mOverrideForPerfClass, &mDeviceInfo,
mOverrideToPortrait);
if (res != OK) {
SET_ERR_L("Could not retrieve camera characteristics: %s (%d)", strerror(-res), res);
session->close();
return res;
}
mSupportNativeZoomRatio = manager->supportNativeZoomRatio(mId);
mIsCompositeJpegRDisabled = manager->isCompositeJpegRDisabled(mId);
std::vector<std::string> physicalCameraIds;
bool isLogical = manager->isLogicalCamera(mId, &physicalCameraIds);
if (isLogical) {
for (auto& physicalId : physicalCameraIds) {
// Do not override characteristics for physical cameras
res = manager->getCameraCharacteristics(
physicalId, /*overrideForPerfClass*/false, &mPhysicalDeviceInfoMap[physicalId],
mOverrideToPortrait);
if (res != OK) {
SET_ERR_L("Could not retrieve camera %s characteristics: %s (%d)",
physicalId.c_str(), strerror(-res), res);
session->close();
return res;
}
bool usePrecorrectArray =
DistortionMapper::isDistortionSupported(mPhysicalDeviceInfoMap[physicalId]);
if (usePrecorrectArray) {
res = mDistortionMappers[physicalId].setupStaticInfo(
mPhysicalDeviceInfoMap[physicalId]);
if (res != OK) {
SET_ERR_L("Unable to read camera %s's calibration fields for distortion "
"correction", physicalId.c_str());
session->close();
return res;
}
}
mZoomRatioMappers[physicalId] = ZoomRatioMapper(
&mPhysicalDeviceInfoMap[physicalId],
mSupportNativeZoomRatio, usePrecorrectArray);
if (SessionConfigurationUtils::supportsUltraHighResolutionCapture(
mPhysicalDeviceInfoMap[physicalId])) {
mUHRCropAndMeteringRegionMappers[physicalId] =
UHRCropAndMeteringRegionMapper(mPhysicalDeviceInfoMap[physicalId],
usePrecorrectArray);
}
}
}
std::shared_ptr<AidlRequestMetadataQueue> queue;
::aidl::android::hardware::common::fmq::MQDescriptor<
int8_t, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite> desc;
::ndk::ScopedAStatus requestQueueRet = session->getCaptureRequestMetadataQueue(&desc);
if (!requestQueueRet.isOk()) {
ALOGE("Transaction error when getting result metadata queue from camera session: %s",
requestQueueRet.getMessage());
return AidlProviderInfo::mapToStatusT(requestQueueRet);
}
queue = std::make_unique<AidlRequestMetadataQueue>(desc);
if (!queue->isValid() || queue->availableToWrite() <= 0) {
ALOGE("HAL returns empty result metadata fmq, not use it");
queue = nullptr;
// Don't use resQueue onwards.
}
std::unique_ptr<AidlResultMetadataQueue>& resQueue = mResultMetadataQueue;
::aidl::android::hardware::common::fmq::MQDescriptor<
int8_t, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite> resDesc;
::ndk::ScopedAStatus resultQueueRet = session->getCaptureResultMetadataQueue(&resDesc);
if (!resultQueueRet.isOk()) {
ALOGE("Transaction error when getting result metadata queue from camera session: %s",
resultQueueRet.getMessage());
return AidlProviderInfo::mapToStatusT(resultQueueRet);
}
resQueue = std::make_unique<AidlResultMetadataQueue>(resDesc);
if (!resQueue->isValid() || resQueue->availableToWrite() <= 0) {
ALOGE("HAL returns empty result metadata fmq, not use it");
resQueue = nullptr;
// Don't use resQueue onwards.
}
camera_metadata_entry bufMgrMode =
mDeviceInfo.find(ANDROID_INFO_SUPPORTED_BUFFER_MANAGEMENT_VERSION);
if (bufMgrMode.count > 0) {
mUseHalBufManager = (bufMgrMode.data.u8[0] ==
ANDROID_INFO_SUPPORTED_BUFFER_MANAGEMENT_VERSION_HIDL_DEVICE_3_5);
mSessionHalBufManager = (bufMgrMode.data.u8[0] ==
ANDROID_INFO_SUPPORTED_BUFFER_MANAGEMENT_VERSION_SESSION_CONFIGURABLE);
}
camera_metadata_entry_t capabilities = mDeviceInfo.find(ANDROID_REQUEST_AVAILABLE_CAPABILITIES);
for (size_t i = 0; i < capabilities.count; i++) {
uint8_t capability = capabilities.data.u8[i];
if (capability == ANDROID_REQUEST_AVAILABLE_CAPABILITIES_OFFLINE_PROCESSING) {
mSupportOfflineProcessing = true;
}
}
mInterface =
new AidlHalInterface(session, queue, mUseHalBufManager, mSupportOfflineProcessing,
mSessionHalBufManager);
std::string providerType;
mVendorTagId = manager->getProviderTagIdLocked(mId);
mTagMonitor.initialize(mVendorTagId);
if (!monitorTags.empty()) {
mTagMonitor.parseTagsToMonitor(monitorTags);
}
for (size_t i = 0; i < capabilities.count; i++) {
uint8_t capability = capabilities.data.u8[i];
if (capability == ANDROID_REQUEST_AVAILABLE_CAPABILITIES_MONOCHROME) {
mNeedFixupMonochromeTags = true;
}
}
// batch size limit is applied to the device with camera device version larger than 3.2 which is
// AIDL v2
hardware::hidl_version maxVersion{0, 0};
IPCTransport transport = IPCTransport::AIDL;
res = manager->getHighestSupportedVersion(mId, &maxVersion, &transport);
if (res != OK) {
ALOGE("%s: Error in getting camera device version id: %s (%d)", __FUNCTION__,
strerror(-res), res);
return res;
}
int deviceVersion = HARDWARE_DEVICE_API_VERSION(maxVersion.get_major(), maxVersion.get_minor());
mBatchSizeLimitEnabled = (deviceVersion >= CAMERA_DEVICE_API_VERSION_1_2);
camera_metadata_entry readoutSupported = mDeviceInfo.find(ANDROID_SENSOR_READOUT_TIMESTAMP);
if (readoutSupported.count == 0) {
ALOGW("%s: Could not find value corresponding to ANDROID_SENSOR_READOUT_TIMESTAMP. "
"Assuming true.", __FUNCTION__);
mSensorReadoutTimestampSupported = true;
} else {
mSensorReadoutTimestampSupported =
readoutSupported.data.u8[0] == ANDROID_SENSOR_READOUT_TIMESTAMP_HARDWARE;
}
return initializeCommonLocked();
}
::ndk::ScopedAStatus AidlCamera3Device::AidlCameraDeviceCallbacks::processCaptureResult(
const std::vector<camera::device::CaptureResult>& results) {
sp<AidlCamera3Device> p = mParent.promote();
if (p == nullptr) {
ALOGE("%s Parent AidlCameraDevice not alive, can't process callbacks", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return p->processCaptureResult(results);
}
::ndk::ScopedAStatus AidlCamera3Device::AidlCameraDeviceCallbacks::notify(
const std::vector<camera::device::NotifyMsg>& msgs) {
sp<AidlCamera3Device> p = mParent.promote();
if (p == nullptr) {
ALOGE("%s Parent AidlCameraDevice not alive, can't process callbacks", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return p->notify(msgs);
}
::ndk::ScopedAStatus AidlCamera3Device::processCaptureResult(
const std::vector<camera::device::CaptureResult>& results) {
// Ideally we should grab mLock, but that can lead to deadlock, and
// it's not super important to get up to date value of mStatus for this
// warning print, hence skipping the lock here
if (mStatus == STATUS_ERROR) {
// Per API contract, HAL should act as closed after device error
// But mStatus can be set to error by framework as well, so just log
// a warning here.
ALOGW("%s: received capture result in error state.", __FUNCTION__);
}
sp<NotificationListener> listener;
{
std::lock_guard<std::mutex> l(mOutputLock);
listener = mListener.promote();
}
if (mProcessCaptureResultLock.tryLock() != OK) {
// This should never happen; it indicates a wrong client implementation
// that doesn't follow the contract. But, we can be tolerant here.
ALOGE("%s: callback overlapped! waiting 1s...",
__FUNCTION__);
if (mProcessCaptureResultLock.timedLock(1000000000 /* 1s */) != OK) {
ALOGE("%s: cannot acquire lock in 1s, dropping results",
__FUNCTION__);
// really don't know what to do, so bail out.
return ::ndk::ScopedAStatus::ok();
}
}
AidlCaptureOutputStates states {
{
mId,
mInFlightLock, mLastCompletedRegularFrameNumber,
mLastCompletedReprocessFrameNumber, mLastCompletedZslFrameNumber,
mInFlightMap, mOutputLock, mResultQueue, mResultSignal,
mNextShutterFrameNumber,
mNextReprocessShutterFrameNumber, mNextZslStillShutterFrameNumber,
mNextResultFrameNumber,
mNextReprocessResultFrameNumber, mNextZslStillResultFrameNumber,
mUseHalBufManager, mHalBufManagedStreamIds, mUsePartialResult, mNeedFixupMonochromeTags,
mNumPartialResults, mVendorTagId, mDeviceInfo, mPhysicalDeviceInfoMap,
mDistortionMappers, mZoomRatioMappers, mRotateAndCropMappers,
mTagMonitor, mInputStream, mOutputStreams, mSessionStatsBuilder, listener, *this,
*this, *(mInterface), mLegacyClient, mMinExpectedDuration, mIsFixedFps,
mOverrideToPortrait, mActivePhysicalId}, mResultMetadataQueue
};
for (const auto& result : results) {
processOneCaptureResultLocked(states, result, result.physicalCameraMetadata);
}
mProcessCaptureResultLock.unlock();
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus AidlCamera3Device::notify(
const std::vector<camera::device::NotifyMsg>& msgs) {
// Ideally we should grab mLock, but that can lead to deadlock, and
// it's not super important to get up to date value of mStatus for this
// warning print, hence skipping the lock here
if (mStatus == STATUS_ERROR) {
// Per API contract, HAL should act as closed after device error
// But mStatus can be set to error by framework as well, so just log
// a warning here.
ALOGW("%s: received notify message in error state.", __FUNCTION__);
}
sp<NotificationListener> listener;
{
std::lock_guard<std::mutex> l(mOutputLock);
listener = mListener.promote();
}
AidlCaptureOutputStates states {
{ mId,
mInFlightLock, mLastCompletedRegularFrameNumber,
mLastCompletedReprocessFrameNumber, mLastCompletedZslFrameNumber,
mInFlightMap, mOutputLock, mResultQueue, mResultSignal,
mNextShutterFrameNumber,
mNextReprocessShutterFrameNumber, mNextZslStillShutterFrameNumber,
mNextResultFrameNumber,
mNextReprocessResultFrameNumber, mNextZslStillResultFrameNumber,
mUseHalBufManager, mHalBufManagedStreamIds, mUsePartialResult, mNeedFixupMonochromeTags,
mNumPartialResults, mVendorTagId, mDeviceInfo, mPhysicalDeviceInfoMap,
mDistortionMappers, mZoomRatioMappers, mRotateAndCropMappers,
mTagMonitor, mInputStream, mOutputStreams, mSessionStatsBuilder, listener, *this,
*this, *(mInterface), mLegacyClient, mMinExpectedDuration, mIsFixedFps,
mOverrideToPortrait, mActivePhysicalId}, mResultMetadataQueue
};
for (const auto& msg : msgs) {
camera3::notify(states, msg, mSensorReadoutTimestampSupported);
}
return ::ndk::ScopedAStatus::ok();
}
status_t AidlCamera3Device::switchToOffline(
const std::vector<int32_t>& streamsToKeep,
/*out*/ sp<CameraOfflineSessionBase>* session) {
ATRACE_CALL();
if (session == nullptr) {
ALOGE("%s: session must not be null", __FUNCTION__);
return BAD_VALUE;
}
Mutex::Autolock il(mInterfaceLock);
bool hasInputStream = mInputStream != nullptr;
int32_t inputStreamId = hasInputStream ? mInputStream->getId() : -1;
bool inputStreamSupportsOffline = hasInputStream ?
mInputStream->getOfflineProcessingSupport() : false;
auto outputStreamIds = mOutputStreams.getStreamIds();
auto streamIds = outputStreamIds;
if (hasInputStream) {
streamIds.push_back(mInputStream->getId());
}
// Check all streams in streamsToKeep supports offline mode
for (auto id : streamsToKeep) {
if (std::find(streamIds.begin(), streamIds.end(), id) == streamIds.end()) {
ALOGE("%s: Unknown stream ID %d", __FUNCTION__, id);
return BAD_VALUE;
} else if (id == inputStreamId) {
if (!inputStreamSupportsOffline) {
ALOGE("%s: input stream %d cannot be switched to offline",
__FUNCTION__, id);
return BAD_VALUE;
}
} else {
sp<camera3::Camera3OutputStreamInterface> stream = mOutputStreams.get(id);
if (!stream->getOfflineProcessingSupport()) {
ALOGE("%s: output stream %d cannot be switched to offline",
__FUNCTION__, id);
return BAD_VALUE;
}
}
}
// TODO: block surface sharing and surface group streams until we can support them
// Stop repeating request, wait until all remaining requests are submitted, then call into
// HAL switchToOffline
camera::device::CameraOfflineSessionInfo offlineSessionInfo;
std::shared_ptr<camera::device::ICameraOfflineSession> offlineSession;
camera3::BufferRecords bufferRecords;
status_t ret = static_cast<AidlRequestThread *>(mRequestThread.get())->switchToOffline(
streamsToKeep, &offlineSessionInfo, &offlineSession, &bufferRecords);
if (ret != OK) {
SET_ERR("Switch to offline failed: %s (%d)", strerror(-ret), ret);
return ret;
}
bool succ = mRequestBufferSM.onSwitchToOfflineSuccess();
if (!succ) {
SET_ERR("HAL must not be calling requestStreamBuffers call");
// TODO: block ALL callbacks from HAL till app configured new streams?
return UNKNOWN_ERROR;
}
// Verify offlineSessionInfo
std::vector<int32_t> offlineStreamIds;
offlineStreamIds.reserve(offlineSessionInfo.offlineStreams.size());
for (auto offlineStream : offlineSessionInfo.offlineStreams) {
// verify stream IDs
int32_t id = offlineStream.id;
if (std::find(streamIds.begin(), streamIds.end(), id) == streamIds.end()) {
SET_ERR("stream ID %d not found!", id);
return UNKNOWN_ERROR;
}
// When not using HAL buf manager, only allow streams requested by app to be preserved
if (!isHalBufferManagedStream(id)) {
if (std::find(streamsToKeep.begin(), streamsToKeep.end(), id) == streamsToKeep.end()) {
SET_ERR("stream ID %d must not be switched to offline!", id);
return UNKNOWN_ERROR;
}
}
offlineStreamIds.push_back(id);
sp<Camera3StreamInterface> stream = (id == inputStreamId) ?
static_cast<sp<Camera3StreamInterface>>(mInputStream) :
static_cast<sp<Camera3StreamInterface>>(mOutputStreams.get(id));
// Verify number of outstanding buffers
if (stream->getOutstandingBuffersCount() != (uint32_t)offlineStream.numOutstandingBuffers) {
SET_ERR("Offline stream %d # of remaining buffer mismatch: (%zu,%d) (service/HAL)",
id, stream->getOutstandingBuffersCount(), offlineStream.numOutstandingBuffers);
return UNKNOWN_ERROR;
}
}
// Verify all streams to be deleted don't have any outstanding buffers
if (hasInputStream && std::find(offlineStreamIds.begin(), offlineStreamIds.end(),
inputStreamId) == offlineStreamIds.end()) {
if (mInputStream->hasOutstandingBuffers()) {
SET_ERR("Input stream %d still has %zu outstanding buffer!",
inputStreamId, mInputStream->getOutstandingBuffersCount());
return UNKNOWN_ERROR;
}
}
for (const auto& outStreamId : outputStreamIds) {
if (std::find(offlineStreamIds.begin(), offlineStreamIds.end(),
outStreamId) == offlineStreamIds.end()) {
auto outStream = mOutputStreams.get(outStreamId);
if (outStream->hasOutstandingBuffers()) {
SET_ERR("Output stream %d still has %zu outstanding buffer!",
outStreamId, outStream->getOutstandingBuffersCount());
return UNKNOWN_ERROR;
}
}
}
InFlightRequestMap offlineReqs;
// Verify inflight requests and their pending buffers
{
std::lock_guard<std::mutex> l(mInFlightLock);
for (auto offlineReq : offlineSessionInfo.offlineRequests) {
int idx = mInFlightMap.indexOfKey(offlineReq.frameNumber);
if (idx == NAME_NOT_FOUND) {
SET_ERR("Offline request frame number %d not found!", offlineReq.frameNumber);
return UNKNOWN_ERROR;
}
const auto& inflightReq = mInFlightMap.valueAt(idx);
// TODO: check specific stream IDs
size_t numBuffersLeft = static_cast<size_t>(inflightReq.numBuffersLeft);
if (numBuffersLeft != offlineReq.pendingStreams.size()) {
SET_ERR("Offline request # of remaining buffer mismatch: (%d,%d) (service/HAL)",
inflightReq.numBuffersLeft, offlineReq.pendingStreams.size());
return UNKNOWN_ERROR;
}
offlineReqs.add(offlineReq.frameNumber, inflightReq);
}
}
// Create Camera3OfflineSession and transfer object ownership
// (streams, inflight requests, buffer caches)
camera3::StreamSet offlineStreamSet;
sp<camera3::Camera3Stream> inputStream;
for (auto offlineStream : offlineSessionInfo.offlineStreams) {
int32_t id = offlineStream.id;
if (mInputStream != nullptr && id == mInputStream->getId()) {
inputStream = mInputStream;
} else {
offlineStreamSet.add(id, mOutputStreams.get(id));
}
}
// TODO: check if we need to lock before copying states
// though technically no other thread should be talking to Camera3Device at this point
Camera3OfflineStates offlineStates(
mTagMonitor, mVendorTagId, mUseHalBufManager, mHalBufManagedStreamIds,
mNeedFixupMonochromeTags, mUsePartialResult, mNumPartialResults,
mLastCompletedRegularFrameNumber, mLastCompletedReprocessFrameNumber,
mLastCompletedZslFrameNumber, mNextResultFrameNumber,
mNextReprocessResultFrameNumber, mNextZslStillResultFrameNumber,
mNextShutterFrameNumber, mNextReprocessShutterFrameNumber,
mNextZslStillShutterFrameNumber, mDeviceInfo, mPhysicalDeviceInfoMap,
mDistortionMappers, mZoomRatioMappers, mRotateAndCropMappers);
*session = new AidlCamera3OfflineSession(mId, inputStream, offlineStreamSet,
std::move(bufferRecords), offlineReqs, offlineStates,
offlineSession, mSensorReadoutTimestampSupported);
// Delete all streams that has been transferred to offline session
Mutex::Autolock l(mLock);
for (auto offlineStream : offlineSessionInfo.offlineStreams) {
int32_t id = offlineStream.id;
if (mInputStream != nullptr && id == mInputStream->getId()) {
mInputStream.clear();
} else {
mOutputStreams.remove(id);
}
}
// disconnect all other streams and switch to UNCONFIGURED state
if (mInputStream != nullptr) {
ret = mInputStream->disconnect();
if (ret != OK) {
SET_ERR_L("disconnect input stream failed!");
return UNKNOWN_ERROR;
}
}
for (auto streamId : mOutputStreams.getStreamIds()) {
sp<Camera3StreamInterface> stream = mOutputStreams.get(streamId);
ret = stream->disconnect();
if (ret != OK) {
SET_ERR_L("disconnect output stream %d failed!", streamId);
return UNKNOWN_ERROR;
}
}
mInputStream.clear();
mOutputStreams.clear();
mNeedConfig = true;
internalUpdateStatusLocked(STATUS_UNCONFIGURED);
mOperatingMode = NO_MODE;
mIsConstrainedHighSpeedConfiguration = false;
mRequestThread->clearPreviousRequest();
return OK;
// TO be done by CameraDeviceClient/Camera3OfflineSession
// register the offline client to camera service
// Setup result passthing threads etc
// Initialize offline session so HAL can start sending callback to it (result Fmq)
// TODO: check how many onIdle callback will be sent
// Java side to make sure the CameraCaptureSession is properly closed
}
::ndk::ScopedAStatus AidlCamera3Device::AidlCameraDeviceCallbacks::requestStreamBuffers(
const std::vector<camera::device::BufferRequest>& bufReqs,
std::vector<aidl::android::hardware::camera::device::StreamBufferRet>* outBuffers,
aidl::android::hardware::camera::device::BufferRequestStatus* status) {
sp<AidlCamera3Device> p = mParent.promote();
if (p == nullptr) {
ALOGE("%s Parent AidlCameraDevice not alive, can't process callbacks", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return p->requestStreamBuffers(bufReqs, outBuffers, status);
}
::ndk::ScopedAStatus AidlCamera3Device::requestStreamBuffers(
const std::vector<camera::device::BufferRequest>& bufReqs,
std::vector<aidl::android::hardware::camera::device::StreamBufferRet>* outBuffers,
aidl::android::hardware::camera::device::BufferRequestStatus* status) {
RequestBufferStates states {
mId, mRequestBufferInterfaceLock, mUseHalBufManager, mHalBufManagedStreamIds,
mOutputStreams, mSessionStatsBuilder, *this, *(mInterface), *this};
camera3::requestStreamBuffers(states, bufReqs, outBuffers, status);
return ::ndk::ScopedAStatus::ok();
}
::ndk::ScopedAStatus AidlCamera3Device::AidlCameraDeviceCallbacks::returnStreamBuffers(
const std::vector<camera::device::StreamBuffer>& buffers) {
sp<AidlCamera3Device> p = mParent.promote();
if (p == nullptr) {
ALOGE("%s Parent AidlCameraDevice not alive, can't process callbacks", __FUNCTION__);
return ::ndk::ScopedAStatus::ok();
}
return p->returnStreamBuffers(buffers);
}
::ndk::SpAIBinder AidlCamera3Device::AidlCameraDeviceCallbacks::createBinder() {
auto binder = BnCameraDeviceCallback::createBinder();
AIBinder_setInheritRt(binder.get(), /*inheritRt*/ true);
return binder;
}
::ndk::ScopedAStatus AidlCamera3Device::returnStreamBuffers(
const std::vector<camera::device::StreamBuffer>& buffers) {
ReturnBufferStates states {
mId, mUseHalBufManager, mHalBufManagedStreamIds, mOutputStreams, mSessionStatsBuilder,
*(mInterface)};
camera3::returnStreamBuffers(states, buffers);
return ::ndk::ScopedAStatus::ok();
}
AidlCamera3Device::AidlHalInterface::AidlHalInterface(
std::shared_ptr<aidl::android::hardware::camera::device::ICameraDeviceSession> &session,
std::shared_ptr<AidlRequestMetadataQueue> queue,
bool useHalBufManager, bool supportOfflineProcessing,
bool supportSessionHalBufManager) :
HalInterface(useHalBufManager, supportOfflineProcessing),
mAidlSession(session),
mRequestMetadataQueue(queue),
mSupportSessionHalBufManager(supportSessionHalBufManager) { }
AidlCamera3Device::AidlHalInterface::AidlHalInterface(
std::shared_ptr<aidl::android::hardware::camera::device::ICameraDeviceSession>
&deviceSession,
std::shared_ptr<aidl::android::hardware::camera::device::ICameraInjectionSession>
&injectionSession, std::shared_ptr<AidlRequestMetadataQueue> queue,
bool useHalBufManager, bool supportOfflineProcessing,
bool supportSessionHalBufManager) :
HalInterface(useHalBufManager, supportOfflineProcessing),
mAidlSession(deviceSession),
mAidlInjectionSession(injectionSession),
mRequestMetadataQueue(queue),
mSupportSessionHalBufManager(supportSessionHalBufManager) { }
bool AidlCamera3Device::AidlHalInterface::valid() {
return (mAidlSession != nullptr);
}
void AidlCamera3Device::AidlHalInterface::clear() {
mAidlSession.reset();
}
status_t AidlCamera3Device::AidlHalInterface::flush() {
ATRACE_NAME("CameraHal::flush");
if (!valid()) return INVALID_OPERATION;
status_t res = OK;
auto err = mAidlSession->flush();
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
res = AidlProviderInfo::mapToStatusT(err);
}
return res;
}
status_t AidlCamera3Device::AidlHalInterface::dump(int /*fd*/) {
ATRACE_NAME("CameraHal::dump");
if (!valid()) return INVALID_OPERATION;
// Handled by CameraProviderManager::dump
return OK;
}
status_t AidlCamera3Device::AidlHalInterface::repeatingRequestEnd(uint32_t frameNumber,
const std::vector<int32_t> &streamIds) {
ATRACE_NAME("AidlCameraHal::repeatingRequestEnd");
if (!valid()) return INVALID_OPERATION;
auto err = mAidlSession->repeatingRequestEnd(frameNumber, streamIds);
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
return OK;
}
status_t AidlCamera3Device::AidlHalInterface::close() {
ATRACE_NAME("CameraHal::close()");
if (!valid()) return INVALID_OPERATION;
status_t res = OK;
auto err = mAidlSession->close();
// Interface will be dead shortly anyway, so don't log errors
if (!err.isOk()) {
res = DEAD_OBJECT;
}
return res;
}
void AidlCamera3Device::AidlHalInterface::signalPipelineDrain(const std::vector<int>& streamIds) {
ATRACE_NAME("CameraHal::signalPipelineDrain");
if (!valid()) {
ALOGE("%s called on invalid camera!", __FUNCTION__);
return;
}
auto err = mAidlSession->signalStreamFlush(streamIds, mNextStreamConfigCounter - 1);
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return;
}
}
bool AidlCamera3Device::AidlHalInterface::isReconfigurationRequired(
CameraMetadata& oldSessionParams, CameraMetadata& newSessionParams) {
// We do reconfiguration by default;
bool required = true;
if (mIsReconfigurationQuerySupported) {
aidl::android::hardware::camera::device::CameraMetadata oldParams, newParams;
camera_metadata_t* oldSessionMeta = const_cast<camera_metadata_t*>(
oldSessionParams.getAndLock());
uint8_t *oldSessionByteP = reinterpret_cast<uint8_t*>(oldSessionMeta);
camera_metadata_t* newSessionMeta = const_cast<camera_metadata_t*>(
newSessionParams.getAndLock());
uint8_t *newSessionByteP = reinterpret_cast<uint8_t*>(newSessionMeta);
// std::vector has no setToExternal, so we hacve to copy
oldParams.metadata.assign(oldSessionByteP,
oldSessionByteP + get_camera_metadata_size(oldSessionMeta));
newParams.metadata.assign(newSessionByteP,
newSessionByteP + get_camera_metadata_size(newSessionMeta));
auto err = mAidlSession->isReconfigurationRequired(oldParams, newParams, &required);
oldSessionParams.unlock(oldSessionMeta);
newSessionParams.unlock(newSessionMeta);
if (!err.isOk()) {
ALOGE("%s: Unexpected binder error: %s", __FUNCTION__, err.getMessage());
return true;
}
}
return required;
}
status_t AidlCamera3Device::AidlHalInterface::constructDefaultRequestSettings(
camera_request_template_t templateId,
/*out*/ camera_metadata_t **requestTemplate) {
ATRACE_NAME("CameraAidlHal::constructDefaultRequestSettings");
using aidl::android::hardware::camera::device::RequestTemplate;
if (!valid()) return INVALID_OPERATION;
RequestTemplate id;
status_t res = SessionConfigurationUtils::mapRequestTemplateToAidl(
templateId, &id);
if (res != OK) {
return res;
}
aidl::android::hardware::camera::device::CameraMetadata request;
auto err = mAidlSession->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) {
*requestTemplate = clone_camera_metadata(r);
if (*requestTemplate == 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 AidlCamera3Device::AidlHalInterface::configureStreams(
const camera_metadata_t *sessionParams,
camera_stream_configuration *config, const std::vector<uint32_t>& bufferSizes,
int64_t logId) {
using camera::device::StreamType;
using camera::device::StreamConfigurationMode;
ATRACE_NAME("CameraHal::configureStreams");
if (!valid()) return INVALID_OPERATION;
status_t res = OK;
// Convert stream config to AIDL
std::set<int> activeStreams;
camera::device::StreamConfiguration requestedConfiguration;
requestedConfiguration.streams.resize(config->num_streams);
for (size_t i = 0; i < config->num_streams; i++) {
camera::device::Stream &dst = requestedConfiguration.streams[i];
camera3::camera_stream_t *src = config->streams[i];
Camera3Stream* cam3stream = Camera3Stream::cast(src);
// For stream configurations with multi res streams, hal buffer manager has to be used.
if (!flags::session_hal_buf_manager() && cam3stream->getHalStreamGroupId() != -1 &&
src->stream_type != CAMERA_STREAM_INPUT) {
mUseHalBufManager = true;
config->use_hal_buf_manager = true;
}
cam3stream->setBufferFreedListener(this);
int streamId = cam3stream->getId();
StreamType streamType;
switch (src->stream_type) {
case CAMERA_STREAM_OUTPUT:
streamType = StreamType::OUTPUT;
break;
case CAMERA_STREAM_INPUT:
streamType = StreamType::INPUT;
break;
default:
ALOGE("%s: Stream %d: Unsupported stream type %d",
__FUNCTION__, streamId, config->streams[i]->stream_type);
return BAD_VALUE;
}
dst.id = streamId;
dst.streamType = streamType;
dst.width = src->width;
dst.height = src->height;
dst.usage = mapToAidlConsumerUsage(cam3stream->getUsage());
dst.rotation = mapToAidlStreamRotation((camera_stream_rotation_t) src->rotation);
dst.format = mapToAidlPixelFormat(cam3stream->isFormatOverridden() ?
cam3stream->getOriginalFormat() : src->format);
dst.dataSpace = mapToAidlDataspace(cam3stream->isDataSpaceOverridden() ?
cam3stream->getOriginalDataSpace() : src->data_space);
dst.colorSpace = src->color_space;
dst.bufferSize = bufferSizes[i];
if (!src->physical_camera_id.empty()) {
dst.physicalCameraId = src->physical_camera_id;
}
dst.groupId = cam3stream->getHalStreamGroupId();
dst.sensorPixelModesUsed.resize(src->sensor_pixel_modes_used.size());
size_t j = 0;
for (int mode : src->sensor_pixel_modes_used) {
dst.sensorPixelModesUsed[j++] = static_cast<SensorPixelMode>(mode);
}
dst.dynamicRangeProfile = mapToAidlDynamicProfile(src->dynamic_range_profile);
dst.useCase = static_cast<ScalerAvailableStreamUseCases>(src->use_case);
activeStreams.insert(streamId);
// Create Buffer ID map if necessary
mBufferRecords.tryCreateBufferCache(streamId);
}
// remove BufferIdMap for deleted streams
mBufferRecords.removeInactiveBufferCaches(activeStreams);
StreamConfigurationMode operationMode;
res = mapToAidlStreamConfigurationMode(
(camera_stream_configuration_mode_t) config->operation_mode,
/*out*/ &operationMode);
if (res != OK) {
return res;
}
requestedConfiguration.operationMode = operationMode;
size_t sessionParamSize = get_camera_metadata_size(sessionParams);
uint8_t *sessionParamP =
reinterpret_cast<uint8_t*>(const_cast<camera_metadata_t*>(sessionParams));
// std::vector has no setToExternal, so we have to copy
requestedConfiguration.sessionParams.metadata.assign(
sessionParamP, sessionParamP + sessionParamSize);
requestedConfiguration.operationMode = operationMode;
// Invoke configureStreams
std::vector<camera::device::HalStream> finalConfiguration;
requestedConfiguration.streamConfigCounter = mNextStreamConfigCounter++;
requestedConfiguration.multiResolutionInputImage = config->input_is_multi_resolution;
requestedConfiguration.logId = logId;
ndk::ScopedAStatus err = ndk::ScopedAStatus::ok();
int32_t interfaceVersion = 0;
camera::device::ConfigureStreamsRet configureStreamsRet;
err = mAidlSession->getInterfaceVersion(&interfaceVersion);
if (!err.isOk()) {
ALOGE("%s: Transaction error getting interface version: %s", __FUNCTION__,
err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
if (flags::session_hal_buf_manager() && interfaceVersion >= AIDL_DEVICE_SESSION_V3
&& mSupportSessionHalBufManager) {
err = mAidlSession->configureStreamsV2(requestedConfiguration, &configureStreamsRet);
finalConfiguration = std::move(configureStreamsRet.halStreams);
} else {
err = mAidlSession->configureStreams(requestedConfiguration, &finalConfiguration);
}
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
if (flags::session_hal_buf_manager()) {
std::set<int32_t> halBufferManagedStreamIds;
for (const auto &halStream: finalConfiguration) {
if ((interfaceVersion >= AIDL_DEVICE_SESSION_V3 &&
mSupportSessionHalBufManager && halStream.enableHalBufferManager)
|| mUseHalBufManager) {
halBufferManagedStreamIds.insert(halStream.id);
}
}
mHalBufManagedStreamIds = std::move(halBufferManagedStreamIds);
config->hal_buffer_managed_streams = mHalBufManagedStreamIds;
}
// And convert output stream configuration from AIDL
for (size_t i = 0; i < config->num_streams; i++) {
camera3::camera_stream_t *dst = config->streams[i];
int streamId = Camera3Stream::cast(dst)->getId();
// Start scan at i, with the assumption that the stream order matches
size_t realIdx = i;
bool found = false;
size_t halStreamCount = finalConfiguration.size();
for (size_t idx = 0; idx < halStreamCount; idx++) {
if (finalConfiguration[realIdx].id == streamId) {
found = true;
break;
}
realIdx = (realIdx >= halStreamCount - 1) ? 0 : realIdx + 1;
}
if (!found) {
ALOGE("%s: Stream %d not found in stream configuration response from HAL",
__FUNCTION__, streamId);
return INVALID_OPERATION;
}
camera::device::HalStream &src = finalConfiguration[realIdx];
Camera3Stream* dstStream = Camera3Stream::cast(dst);
int overrideFormat = mapToFrameworkFormat(src.overrideFormat);
android_dataspace overrideDataSpace = mapToFrameworkDataspace(src.overrideDataSpace);
dstStream->setOfflineProcessingSupport(src.supportOffline);
if (dstStream->getOriginalFormat() != HAL_PIXEL_FORMAT_IMPLEMENTATION_DEFINED) {
dstStream->setFormatOverride(false);
dstStream->setDataSpaceOverride(false);
if (dst->format != overrideFormat) {
ALOGE("%s: Stream %d: Format override not allowed for format 0x%x", __FUNCTION__,
streamId, dst->format);
}
if (dst->data_space != overrideDataSpace) {
ALOGE("%s: Stream %d: DataSpace override not allowed for format 0x%x", __FUNCTION__,
streamId, dst->format);
}
} else {
bool needFormatOverride =
requestedConfiguration.streams[i].format != src.overrideFormat;
bool needDataspaceOverride =
requestedConfiguration.streams[i].dataSpace != src.overrideDataSpace;
// Override allowed with IMPLEMENTATION_DEFINED
dstStream->setFormatOverride(needFormatOverride);
dstStream->setDataSpaceOverride(needDataspaceOverride);
dst->format = overrideFormat;
dst->data_space = overrideDataSpace;
}
if (dst->stream_type == CAMERA_STREAM_INPUT) {
if (static_cast<int64_t>(src.producerUsage) != 0) {
ALOGE("%s: Stream %d: INPUT streams must have 0 for producer usage",
__FUNCTION__, streamId);
return INVALID_OPERATION;
}
dstStream->setUsage(
mapConsumerToFrameworkUsage(src.consumerUsage));
} else {
// OUTPUT
if (static_cast<int64_t>(src.consumerUsage) != 0) {
ALOGE("%s: Stream %d: OUTPUT streams must have 0 for consumer usage",
__FUNCTION__, streamId);
return INVALID_OPERATION;
}
dstStream->setUsage(
mapProducerToFrameworkUsage(src.producerUsage));
if (flags::session_hal_buf_manager()) {
dstStream->setHalBufferManager(
contains(config->hal_buffer_managed_streams, streamId));
}
}
dst->max_buffers = src.maxBuffers;
}
return res;
}
status_t AidlCamera3Device::AidlHalInterface::configureInjectedStreams(
const camera_metadata_t* sessionParams, camera_stream_configuration* config,
const std::vector<uint32_t>& bufferSizes,
const CameraMetadata& cameraCharacteristics) {
using camera::device::StreamType;
using camera::device::StreamConfigurationMode;
ATRACE_NAME("InjectionCameraHal::configureStreams");
if (!valid()) return INVALID_OPERATION;
status_t res = OK;
if (config->input_is_multi_resolution) {
ALOGE("%s: Injection camera device doesn't support multi-resolution input "
"stream", __FUNCTION__);
return BAD_VALUE;
}
// Convert stream config to AIDL
std::set<int> activeStreams;
camera::device::StreamConfiguration requestedConfiguration;
requestedConfiguration.streams.resize(config->num_streams);
for (size_t i = 0; i < config->num_streams; i++) {
camera::device::Stream& dst = requestedConfiguration.streams[i];
camera3::camera_stream_t* src = config->streams[i];
Camera3Stream* cam3stream = Camera3Stream::cast(src);
cam3stream->setBufferFreedListener(this);
int streamId = cam3stream->getId();
StreamType streamType;
switch (src->stream_type) {
case CAMERA_STREAM_OUTPUT:
streamType = StreamType::OUTPUT;
break;
case CAMERA_STREAM_INPUT:
streamType = StreamType::INPUT;
break;
default:
ALOGE("%s: Stream %d: Unsupported stream type %d", __FUNCTION__,
streamId, config->streams[i]->stream_type);
return BAD_VALUE;
}
dst.id = streamId;
dst.streamType = streamType;
dst.width = src->width;
dst.height = src->height;
dst.usage = mapToAidlConsumerUsage(cam3stream->getUsage());
dst.rotation = mapToAidlStreamRotation((camera_stream_rotation_t)src->rotation);
dst.format =
mapToAidlPixelFormat(cam3stream->isFormatOverridden() ? cam3stream->getOriginalFormat()
: src->format);
dst.dataSpace =
mapToAidlDataspace(cam3stream->isDataSpaceOverridden() ?
cam3stream->getOriginalDataSpace() : src->data_space);
dst.bufferSize = bufferSizes[i];
if (!src->physical_camera_id.empty()) {
dst.physicalCameraId = src->physical_camera_id;
}
dst.groupId = cam3stream->getHalStreamGroupId();
dst.sensorPixelModesUsed.resize(src->sensor_pixel_modes_used.size());
size_t j = 0;
for (int mode : src->sensor_pixel_modes_used) {
dst.sensorPixelModesUsed[j++] = static_cast<SensorPixelMode>(mode);
}
activeStreams.insert(streamId);
// Create Buffer ID map if necessary
mBufferRecords.tryCreateBufferCache(streamId);
}
// remove BufferIdMap for deleted streams
mBufferRecords.removeInactiveBufferCaches(activeStreams);
StreamConfigurationMode operationMode;
res = mapToAidlStreamConfigurationMode(
(camera_stream_configuration_mode_t)config->operation_mode,
/*out*/ &operationMode);
if (res != OK) {
return res;
}
requestedConfiguration.operationMode = operationMode;
size_t sessionParamSize = get_camera_metadata_size(sessionParams);
uint8_t *sessionParamP =
reinterpret_cast<uint8_t*>(const_cast<camera_metadata_t*>(sessionParams));
requestedConfiguration.operationMode = operationMode;
requestedConfiguration.sessionParams.metadata.assign(
sessionParamP, sessionParamP + sessionParamSize);
// See which version of HAL we have
if (mAidlInjectionSession != nullptr) {
requestedConfiguration.streamConfigCounter = mNextStreamConfigCounter++;
requestedConfiguration.multiResolutionInputImage = config->input_is_multi_resolution;
const camera_metadata_t* rawMetadata = cameraCharacteristics.getAndLock();
uint8_t *aidlCharsP =
reinterpret_cast<uint8_t*>(const_cast<camera_metadata_t*>(rawMetadata));
aidl::android::hardware::camera::device::CameraMetadata aidlChars;
aidlChars.metadata.assign(aidlCharsP, aidlCharsP + get_camera_metadata_size(rawMetadata));
cameraCharacteristics.unlock(rawMetadata);
auto err = mAidlInjectionSession->configureInjectionStreams(requestedConfiguration,
aidlChars);
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
} else {
ALOGE("%s: mAidlInjectionSession == nullptr, the injection not supported ", __FUNCTION__);
return INVALID_OPERATION;
}
return res;
}
status_t AidlCamera3Device::AidlHalInterface::processBatchCaptureRequests(
std::vector<camera_capture_request_t*>& requests,/*out*/uint32_t* numRequestProcessed) {
ATRACE_NAME("CameraHal::processBatchCaptureRequests");
if (!valid()) return INVALID_OPERATION;
std::vector<camera::device::CaptureRequest> captureRequests;
size_t batchSize = requests.size();
if (batchSize > INT_MAX) {
ALOGE("%s batchSize %zu > INT_MAX, aidl interface cannot handle batch size", __FUNCTION__,
batchSize);
return BAD_VALUE;
}
captureRequests.resize(batchSize);
std::vector<native_handle_t*> handlesCreated;
std::vector<std::pair<int32_t, int32_t>> inflightBuffers;
status_t res = OK;
for (size_t i = 0; i < batchSize; i++) {
res = wrapAsAidlRequest(requests[i], /*out*/&captureRequests[i],
/*out*/&handlesCreated, /*out*/&inflightBuffers);
if (res != OK) {
mBufferRecords.popInflightBuffers(inflightBuffers);
cleanupNativeHandles(&handlesCreated);
return res;
}
}
std::vector<camera::device::BufferCache> cachesToRemove;
{
std::lock_guard<std::mutex> lock(mFreedBuffersLock);
for (auto& pair : mFreedBuffers) {
// The stream might have been removed since onBufferFreed
if (mBufferRecords.isStreamCached(pair.first)) {
cachesToRemove.push_back({pair.first, static_cast<int64_t>(pair.second)});
}
}
mFreedBuffers.clear();
}
*numRequestProcessed = 0;
// Write metadata to FMQ.
for (size_t i = 0; i < batchSize; i++) {
camera_capture_request_t* request = requests[i];
camera::device::CaptureRequest* captureRequest;
captureRequest = &captureRequests[i];
if (request->settings != nullptr) {
size_t settingsSize = get_camera_metadata_size(request->settings);
if (mRequestMetadataQueue != nullptr && mRequestMetadataQueue->write(
reinterpret_cast<const int8_t*>(request->settings), settingsSize)) {
captureRequest->settings.metadata.resize(0);
captureRequest->fmqSettingsSize = settingsSize;
} else {
if (mRequestMetadataQueue != nullptr) {
ALOGW("%s: couldn't utilize fmq, fallback to hwbinder", __FUNCTION__);
}
uint8_t *settingsP =
reinterpret_cast<uint8_t*>(
const_cast<camera_metadata_t*>(request->settings));
size_t settingsSize = get_camera_metadata_size(request->settings);
captureRequest->settings.metadata.assign(settingsP, settingsP + settingsSize);
captureRequest->fmqSettingsSize = 0u;
}
} else {
// A null request settings maps to a size-0 CameraMetadata
captureRequest->settings.metadata.resize(0);
captureRequest->fmqSettingsSize = 0u;
}
captureRequest ->inputWidth = request->input_width;
captureRequest->inputHeight = request->input_height;
std::vector<camera::device::PhysicalCameraSetting>& physicalCameraSettings =
captureRequest->physicalCameraSettings;
physicalCameraSettings.resize(request->num_physcam_settings);
for (size_t j = 0; j < request->num_physcam_settings; j++) {
if (request->physcam_settings != nullptr) {
size_t settingsSize = get_camera_metadata_size(request->physcam_settings[j]);
if (mRequestMetadataQueue != nullptr && mRequestMetadataQueue->write(
reinterpret_cast<const int8_t*>(request->physcam_settings[j]),
settingsSize)) {
physicalCameraSettings[j].settings.metadata.resize(0);
physicalCameraSettings[j].fmqSettingsSize = settingsSize;
} else {
if (mRequestMetadataQueue != nullptr) {
ALOGW("%s: couldn't utilize fmq, fallback to hwbinder", __FUNCTION__);
}
uint8_t *physicalSettingsP =
reinterpret_cast<uint8_t*>(const_cast<camera_metadata_t*>(
request->physcam_settings[j]));
physicalCameraSettings[j].settings.metadata.assign(physicalSettingsP,
physicalSettingsP + settingsSize);
physicalCameraSettings[j].fmqSettingsSize = 0u;
}
} else {
physicalCameraSettings[j].fmqSettingsSize = 0u;
physicalCameraSettings[j].settings.metadata.resize(0);
}
physicalCameraSettings[j].physicalCameraId = request->physcam_id[j];
}
}
int32_t numRequests = 0;
auto retS = mAidlSession->processCaptureRequest(captureRequests, cachesToRemove,
&numRequests);
if (!retS.isOk()) {
res = AidlProviderInfo::mapToStatusT(retS);
}
if (res == OK) {
if (numRequests < 0) {
res = INVALID_OPERATION;
} else {
*numRequestProcessed = static_cast<uint32_t>(numRequests);
}
}
if (res == OK && *numRequestProcessed == batchSize) {
if (mAidlSession->isRemote()) {
// Only close acquire fence FDs when the AIDL transaction succeeds (so the FDs have been
// sent to camera HAL processes)
cleanupNativeHandles(&handlesCreated, /*closeFd*/true);
} else {
// In passthrough mode the FDs are now owned by HAL
cleanupNativeHandles(&handlesCreated);
}
} else {
ALOGE("%s Error with processCaptureRequest %s ", __FUNCTION__, retS.getMessage());
mBufferRecords.popInflightBuffers(inflightBuffers);
cleanupNativeHandles(&handlesCreated);
}
return res;
}
status_t AidlCamera3Device::AidlHalInterface::wrapAsAidlRequest(camera_capture_request_t* request,
/*out*/camera::device::CaptureRequest* captureRequest,
/*out*/std::vector<native_handle_t*>* handlesCreated,
/*out*/std::vector<std::pair<int32_t, int32_t>>* inflightBuffers) {
using camera::device::BufferStatus;
using camera::device::StreamBuffer;
ATRACE_CALL();
if (captureRequest == nullptr || handlesCreated == nullptr || inflightBuffers == nullptr) {
ALOGE("%s: captureRequest (%p), handlesCreated (%p), and inflightBuffers(%p) "
"must not be null", __FUNCTION__, captureRequest, handlesCreated, inflightBuffers);
return BAD_VALUE;
}
captureRequest->frameNumber = request->frame_number;
captureRequest->fmqSettingsSize = 0;
{
if (request->input_buffer != nullptr) {
int32_t streamId = Camera3Stream::cast(request->input_buffer->stream)->getId();
buffer_handle_t buf = *(request->input_buffer->buffer);
auto pair = getBufferId(buf, streamId);
bool isNewBuffer = pair.first;
uint64_t bufferId = pair.second;
captureRequest->inputBuffer.streamId = streamId;
captureRequest->inputBuffer.bufferId = bufferId;
captureRequest->inputBuffer.buffer =
(isNewBuffer) ? camera3::dupToAidlIfNotNull(buf) :
aidl::android::hardware::common::NativeHandle();
captureRequest->inputBuffer.status = BufferStatus::OK;
native_handle_t *acquireFence = nullptr;
if (request->input_buffer->acquire_fence != -1) {
acquireFence = native_handle_create(1,0);
acquireFence->data[0] = request->input_buffer->acquire_fence;
handlesCreated->push_back(acquireFence);
}
// duping here is okay, in aidl ownership is not given to aidl_handle
captureRequest->inputBuffer.acquireFence = camera3::dupToAidlIfNotNull(acquireFence);
captureRequest->inputBuffer.releaseFence =
aidl::android::hardware::common::NativeHandle();
mBufferRecords.pushInflightBuffer(captureRequest->frameNumber, streamId,
request->input_buffer->buffer);
inflightBuffers->push_back(std::make_pair(captureRequest->frameNumber, streamId));
} else {
captureRequest->inputBuffer.streamId = -1;
captureRequest->inputBuffer.bufferId = BUFFER_ID_NO_BUFFER;
}
captureRequest->outputBuffers.resize(request->num_output_buffers);
for (size_t i = 0; i < request->num_output_buffers; i++) {
const camera_stream_buffer_t *src = request->output_buffers + i;
StreamBuffer &dst = captureRequest->outputBuffers[i];
int32_t streamId = Camera3Stream::cast(src->stream)->getId();
if (src->buffer != nullptr) {
buffer_handle_t buf = *(src->buffer);
auto pair = getBufferId(buf, streamId);
bool isNewBuffer = pair.first;
dst.bufferId = pair.second;
dst.buffer = isNewBuffer ?
camera3::dupToAidlIfNotNull(buf) :
aidl::android::hardware::common::NativeHandle();
native_handle_t *acquireFence = nullptr;
if (src->acquire_fence != -1) {
acquireFence = native_handle_create(1,0);
acquireFence->data[0] = src->acquire_fence;
handlesCreated->push_back(acquireFence);
}
dst.acquireFence = camera3::dupToAidlIfNotNull(acquireFence);
} else if (isHalBufferManagedStream(streamId)) {
// HAL buffer management path
dst.bufferId = BUFFER_ID_NO_BUFFER;
dst.buffer = aidl::android::hardware::common::NativeHandle();
dst.acquireFence = aidl::android::hardware::common::NativeHandle();
} else {
ALOGE("%s: cannot send a null buffer in capture request!", __FUNCTION__);
return BAD_VALUE;
}
dst.streamId = streamId;
dst.status = BufferStatus::OK;
dst.releaseFence = aidl::android::hardware::common::NativeHandle();
// Output buffers are empty when using HAL buffer manager
if (!isHalBufferManagedStream(streamId)) {
mBufferRecords.pushInflightBuffer(
captureRequest->frameNumber, streamId, src->buffer);
inflightBuffers->push_back(std::make_pair(captureRequest->frameNumber, streamId));
}
}
}
return OK;
}
status_t AidlCamera3Device::AidlHalInterface::switchToOffline(
const std::vector<int32_t>& streamsToKeep,
/*out*/aidl::android::hardware::camera::device::CameraOfflineSessionInfo*
offlineSessionInfo,
/*out*/std::shared_ptr<aidl::android::hardware::camera::device::ICameraOfflineSession>*
offlineSession,
/*out*/camera3::BufferRecords* bufferRecords) {
ATRACE_NAME("CameraHal::switchToOffline");
if (!valid()) {
ALOGE("%s called on invalid camera!", __FUNCTION__);
return INVALID_OPERATION;
}
if (offlineSessionInfo == nullptr || offlineSession == nullptr || bufferRecords == nullptr) {
ALOGE("%s: output arguments must not be null!", __FUNCTION__);
return INVALID_OPERATION;
}
auto err = mAidlSession->switchToOffline(streamsToKeep, offlineSessionInfo, offlineSession);
if (!err.isOk()) {
ALOGE("%s: Transaction error: %s", __FUNCTION__, err.getMessage());
return AidlProviderInfo::mapToStatusT(err);
}
return verifyBufferCaches(offlineSessionInfo, bufferRecords);
}
AidlCamera3Device::AidlRequestThread::AidlRequestThread(wp<Camera3Device> parent,
sp<camera3::StatusTracker> statusTracker,
sp<HalInterface> interface,
const Vector<int32_t>& sessionParamKeys,
bool useHalBufManager,
bool supportCameraMute,
bool overrideToPortrait,
bool supportSettingsOverride) :
RequestThread(parent, statusTracker, interface, sessionParamKeys,
useHalBufManager, supportCameraMute, overrideToPortrait,
supportSettingsOverride) {}
status_t AidlCamera3Device::AidlRequestThread::switchToOffline(
const std::vector<int32_t>& streamsToKeep,
/*out*/camera::device::CameraOfflineSessionInfo* offlineSessionInfo,
/*out*/std::shared_ptr<camera::device::ICameraOfflineSession>* offlineSession,
/*out*/camera3::BufferRecords* bufferRecords) {
Mutex::Autolock l(mRequestLock);
clearRepeatingRequestsLocked(/*lastFrameNumber*/nullptr);
// Wait until request thread is fully stopped
// TBD: check if request thread is being paused by other APIs (shouldn't be)
// We could also check for mRepeatingRequests.empty(), but the API interface
// is serialized by Camera3Device::mInterfaceLock so no one should be able to submit any
// new requests during the call; hence skip that check.
bool queueEmpty = mNextRequests.empty() && mRequestQueue.empty();
while (!queueEmpty) {
status_t res = mRequestSubmittedSignal.waitRelative(mRequestLock, kRequestSubmitTimeout);
if (res == TIMED_OUT) {
ALOGE("%s: request thread failed to submit one request within timeout!", __FUNCTION__);
return res;
} else if (res != OK) {
ALOGE("%s: request thread failed to submit a request: %s (%d)!",
__FUNCTION__, strerror(-res), res);
return res;
}
queueEmpty = mNextRequests.empty() && mRequestQueue.empty();
}
return (static_cast<AidlHalInterface *>(mInterface.get()))->switchToOffline(
streamsToKeep, offlineSessionInfo, offlineSession, bufferRecords);
}
status_t AidlCamera3Device::AidlCamera3DeviceInjectionMethods::injectionInitialize(
const std::string& injectedCamId, sp<CameraProviderManager> manager,
const std::shared_ptr<camera::device::ICameraDeviceCallback>&callback) {
ATRACE_CALL();
Mutex::Autolock lock(mInjectionLock);
if (manager == nullptr) {
ALOGE("%s: manager does not exist!", __FUNCTION__);
return INVALID_OPERATION;
}
sp<Camera3Device> parent = mParent.promote();
if (parent == nullptr) {
ALOGE("%s: parent does not exist!", __FUNCTION__);
return INVALID_OPERATION;
}
if (parent->getTransportType() != IPCTransport::AIDL) {
ALOGE("%s Parent transport not AIDL for injected camera id %s, aborting", __FUNCTION__,
injectedCamId.c_str());
return INVALID_OPERATION;
}
mInjectedCamId = injectedCamId;
std::shared_ptr<camera::device::ICameraInjectionSession> injectionSession;
ATRACE_BEGIN("Injection CameraHal::openSession");
status_t res = manager->openAidlInjectionSession(injectedCamId, callback,
/*out*/ &injectionSession);
ATRACE_END();
if (res != OK) {
ALOGE("Injection camera could not open camera session: %s (%d)",
strerror(-res), res);
return res;
}
std::shared_ptr<camera::device::ICameraDeviceSession> deviceSession = nullptr;
auto ret = injectionSession->getCameraDeviceSession(&deviceSession);
if (!ret.isOk() || deviceSession == nullptr) {
ALOGE("%s Camera injection session couldn't return ICameraDeviceSession", __FUNCTION__);
return AidlProviderInfo::mapToStatusT(ret);
}
std::shared_ptr<AidlRequestMetadataQueue> queue;
::aidl::android::hardware::common::fmq::MQDescriptor<
int8_t, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite> desc;
::ndk::ScopedAStatus requestQueueRet = deviceSession->getCaptureRequestMetadataQueue(&desc);
if (!requestQueueRet.isOk()) {
ALOGE("Injection camera transaction error when getting result metadata queue from camera"
" session: %s", requestQueueRet.getMessage());
return AidlProviderInfo::mapToStatusT(requestQueueRet);
}
queue = std::make_unique<AidlRequestMetadataQueue>(desc);
if (!queue->isValid() || queue->availableToWrite() <= 0) {
ALOGE("HAL returns empty result metadata fmq, not use it");
queue = nullptr;
// Don't use resQueue onwards.
}
std::unique_ptr<AidlResultMetadataQueue>& resQueue = mInjectionResultMetadataQueue;
::aidl::android::hardware::common::fmq::MQDescriptor<
int8_t, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite> resDesc;
::ndk::ScopedAStatus resultQueueRet = deviceSession->getCaptureResultMetadataQueue(&resDesc);
if (!resultQueueRet.isOk()) {
ALOGE("Transaction error when getting result metadata queue from camera session: %s",
resultQueueRet.getMessage());
return AidlProviderInfo::mapToStatusT(resultQueueRet);
}
resQueue = std::make_unique<AidlResultMetadataQueue>(resDesc);
if (!resQueue->isValid() || resQueue->availableToWrite() <= 0) {
ALOGE("HAL returns empty result metadata fmq, not use it");
resQueue = nullptr;
// Don't use resQueue onwards.
}
ALOGV("%s: Injection camera interface = new HalInterface()", __FUNCTION__);
mInjectedCamHalInterface =
new AidlHalInterface(deviceSession, injectionSession, queue, parent->mUseHalBufManager,
parent->mSupportOfflineProcessing, parent->mSessionHalBufManager);
if (mInjectedCamHalInterface == nullptr) {
ALOGE("%s: mInjectedCamHalInterface does not exist!", __FUNCTION__);
return DEAD_OBJECT;
}
return OK;
}
status_t AidlCamera3Device::AidlCamera3DeviceInjectionMethods::replaceHalInterface(
sp<HalInterface> newHalInterface, bool keepBackup) {
Mutex::Autolock lock(mInjectionLock);
if (newHalInterface.get() == nullptr) {
ALOGE("%s: The newHalInterface does not exist, to stop replacing.",
__FUNCTION__);
return DEAD_OBJECT;
}
sp<Camera3Device> parent = mParent.promote();
if (parent == nullptr) {
ALOGE("%s: parent does not exist!", __FUNCTION__);
return INVALID_OPERATION;
}
if (parent->getTransportType() != IPCTransport::AIDL ||
newHalInterface->getTransportType() != IPCTransport::AIDL) {
ALOGE("%s Parent transport not AIDL for replacing hal interface", __FUNCTION__);
return INVALID_OPERATION;
}
AidlCamera3Device *aidlParent = static_cast<AidlCamera3Device *>(parent.get());
if (keepBackup) {
if (mBackupHalInterface == nullptr) {
mBackupHalInterface = parent->mInterface;
}
if (mBackupResultMetadataQueue == nullptr) {
mBackupResultMetadataQueue = std::move(aidlParent->mResultMetadataQueue);
aidlParent->mResultMetadataQueue = std::move(mInjectionResultMetadataQueue);
}
} else {
mBackupHalInterface = nullptr;
aidlParent->mResultMetadataQueue = std::move(mBackupResultMetadataQueue);
mBackupResultMetadataQueue = nullptr;
}
parent->mInterface = newHalInterface;
return OK;
}
void AidlCamera3Device::applyMaxBatchSizeLocked(
RequestList* requestList, const sp<camera3::Camera3OutputStreamInterface>& stream) {
int batchSize = requestList->size();
if (!mBatchSizeLimitEnabled) {
(*requestList->begin())->mBatchSize = batchSize;
stream->setBatchSize(batchSize);
return;
}
const auto& metadata = (*requestList->begin())->mSettingsList.begin()->metadata;
uint32_t tag = ANDROID_CONTROL_AVAILABLE_HIGH_SPEED_VIDEO_CONFIGURATIONS;
auto sensorPixelModeEntry = metadata.find(ANDROID_SENSOR_PIXEL_MODE);
if (sensorPixelModeEntry.count != 0) {
if (ANDROID_SENSOR_PIXEL_MODE_MAXIMUM_RESOLUTION == sensorPixelModeEntry.data.u8[0]) {
tag = ANDROID_CONTROL_AVAILABLE_HIGH_SPEED_VIDEO_CONFIGURATIONS_MAXIMUM_RESOLUTION;
}
}
const auto fpsRange = metadata.find(ANDROID_CONTROL_AE_TARGET_FPS_RANGE);
if (fpsRange.count > 1) {
auto configEntry = mDeviceInfo.find(tag);
for (size_t index = 4; index < configEntry.count; index += 5) {
if (stream->getWidth() == static_cast<uint32_t>(configEntry.data.i32[index - 4]) &&
stream->getHeight() == static_cast<uint32_t>(configEntry.data.i32[index - 3]) &&
fpsRange.data.i32[0] == configEntry.data.i32[index - 2] &&
fpsRange.data.i32[1] == configEntry.data.i32[index - 1]) {
const int maxBatchSize = configEntry.data.i32[index - 1] / 30;
const int reportedSize = configEntry.data.i32[index];
if (maxBatchSize % reportedSize == 0 && requestList->size() % reportedSize == 0) {
batchSize = reportedSize;
ALOGVV("Matching high speed configuration found. Limit batch size to %d",
batchSize);
} else if (maxBatchSize % reportedSize == 0 &&
reportedSize % requestList->size() == 0) {
ALOGVV("Matching high speed configuration found, but requested batch size is "
"divisor of batch_size_max. No need to limit batch size.");
} else {
ALOGW("Matching high speed configuration found, but batch_size_max is not a "
"divisor of corresponding fps_max/30 or requested batch size is not a "
"divisor of batch_size_max, (fps_max %d, batch_size_max %d, requested "
"batch size %zu)",
configEntry.data.i32[index - 1], reportedSize, requestList->size());
}
break;
}
}
}
for (auto request = requestList->begin(); request != requestList->end(); request++) {
if (requestList->distance(requestList->begin(), request) % batchSize == 0) {
(*request)->mBatchSize = batchSize;
}
}
stream->setBatchSize(batchSize);
}
status_t AidlCamera3Device::injectionCameraInitialize(const std::string &injectedCamId,
sp<CameraProviderManager> manager) {
return (static_cast<AidlCamera3DeviceInjectionMethods *>
(mInjectionMethods.get()))->injectionInitialize(injectedCamId, manager,
std::shared_ptr<camera::device::ICameraDeviceCallback>(mCallbacks));
};
sp<Camera3Device::RequestThread> AidlCamera3Device::createNewRequestThread(
wp<Camera3Device> parent, sp<camera3::StatusTracker> statusTracker,
sp<Camera3Device::HalInterface> interface,
const Vector<int32_t>& sessionParamKeys,
bool useHalBufManager,
bool supportCameraMute,
bool overrideToPortrait,
bool supportSettingsOverride) {
return new AidlRequestThread(parent, statusTracker, interface, sessionParamKeys,
useHalBufManager, supportCameraMute, overrideToPortrait,
supportSettingsOverride);
};
sp<Camera3Device::Camera3DeviceInjectionMethods>
AidlCamera3Device::createCamera3DeviceInjectionMethods(wp<Camera3Device> parent) {
return new AidlCamera3DeviceInjectionMethods(parent);
}
}; // namespace android