blob: 4b840fca1a2e468e72104dd5bc6ea68435f2d070 [file] [log] [blame]
/*
* Copyright (C) 2018 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 "Camera3-CompositeStream"
#define ATRACE_TAG ATRACE_TAG_CAMERA
//#define LOG_NDEBUG 0
#include <utils/Log.h>
#include <utils/Trace.h>
#include "common/CameraDeviceBase.h"
#include "CameraDeviceClient.h"
#include "CompositeStream.h"
namespace android {
namespace camera3 {
CompositeStream::CompositeStream(sp<CameraDeviceBase> device,
wp<hardware::camera2::ICameraDeviceCallbacks> cb) :
mDevice(device),
mRemoteCallback(cb),
mNumPartialResults(1),
mErrorState(false) {
if (device != nullptr) {
CameraMetadata staticInfo = device->info();
camera_metadata_entry_t entry = staticInfo.find(ANDROID_REQUEST_PARTIAL_RESULT_COUNT);
if (entry.count > 0) {
mNumPartialResults = entry.data.i32[0];
}
mStatusTracker = device->getStatusTracker();
}
}
status_t CompositeStream::createStream(const std::vector<sp<Surface>>& consumers,
bool hasDeferredConsumer, uint32_t width, uint32_t height, int format,
camera_stream_rotation_t rotation, int * id, const String8& physicalCameraId,
const std::unordered_set<int32_t> &sensorPixelModesUsed,
std::vector<int> * surfaceIds,
int streamSetId, bool isShared, bool isMultiResolution) {
if (hasDeferredConsumer) {
ALOGE("%s: Deferred consumers not supported in case of composite streams!",
__FUNCTION__);
return BAD_VALUE;
}
if (streamSetId != camera3::CAMERA3_STREAM_ID_INVALID) {
ALOGE("%s: Surface groups not supported in case of composite streams!",
__FUNCTION__);
return BAD_VALUE;
}
if (isShared) {
ALOGE("%s: Shared surfaces not supported in case of composite streams!",
__FUNCTION__);
return BAD_VALUE;
}
if (isMultiResolution) {
ALOGE("%s: Multi-resolution output not supported in case of composite streams!",
__FUNCTION__);
return BAD_VALUE;
}
return createInternalStreams(consumers, hasDeferredConsumer, width, height, format, rotation,
id, physicalCameraId, sensorPixelModesUsed, surfaceIds, streamSetId, isShared);
}
status_t CompositeStream::deleteStream() {
{
Mutex::Autolock l(mMutex);
mPendingCaptureResults.clear();
mCaptureResults.clear();
mFrameNumberMap.clear();
mErrorFrameNumbers.clear();
}
return deleteInternalStreams();
}
void CompositeStream::onBufferRequestForFrameNumber(uint64_t frameNumber, int streamId,
const CameraMetadata& /*settings*/) {
Mutex::Autolock l(mMutex);
if (!mErrorState && (streamId == getStreamId())) {
mPendingCaptureResults.emplace(frameNumber, CameraMetadata());
}
}
void CompositeStream::onBufferReleased(const BufferInfo& bufferInfo) {
Mutex::Autolock l(mMutex);
if (!mErrorState && !bufferInfo.mError) {
mFrameNumberMap.emplace(bufferInfo.mFrameNumber, bufferInfo.mTimestamp);
mInputReadyCondition.signal();
}
}
void CompositeStream::eraseResult(int64_t frameNumber) {
Mutex::Autolock l(mMutex);
auto it = mPendingCaptureResults.find(frameNumber);
if (it == mPendingCaptureResults.end()) {
return;
}
it = mPendingCaptureResults.erase(it);
}
void CompositeStream::onResultAvailable(const CaptureResult& result) {
bool resultError = false;
{
Mutex::Autolock l(mMutex);
uint64_t frameNumber = result.mResultExtras.frameNumber;
bool resultReady = false;
auto it = mPendingCaptureResults.find(frameNumber);
if (it != mPendingCaptureResults.end()) {
it->second.append(result.mMetadata);
if (result.mResultExtras.partialResultCount >= mNumPartialResults) {
auto entry = it->second.find(ANDROID_SENSOR_TIMESTAMP);
if (entry.count == 1) {
auto ts = entry.data.i64[0];
mCaptureResults.emplace(ts, std::make_tuple(frameNumber, it->second));
resultReady = true;
} else {
ALOGE("%s: Timestamp metadata entry missing for frameNumber: %" PRIu64,
__FUNCTION__, frameNumber);
resultError = true;
}
mPendingCaptureResults.erase(it);
}
}
if (resultReady) {
mInputReadyCondition.signal();
}
}
if (resultError) {
onResultError(result.mResultExtras);
}
}
void CompositeStream::flagAnErrorFrameNumber(int64_t frameNumber) {
Mutex::Autolock l(mMutex);
mErrorFrameNumbers.emplace(frameNumber);
mInputReadyCondition.signal();
}
status_t CompositeStream::registerCompositeStreamListener(int32_t streamId) {
sp<CameraDeviceBase> device = mDevice.promote();
if (device.get() == nullptr) {
return NO_INIT;
}
auto ret = device->addBufferListenerForStream(streamId, this);
if (ret != OK) {
ALOGE("%s: Failed to register composite stream listener!", __FUNCTION__);
}
return ret;
}
bool CompositeStream::onError(int32_t errorCode, const CaptureResultExtras& resultExtras) {
auto ret = false;
switch (errorCode) {
case hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_RESULT:
onResultError(resultExtras);
break;
case hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_BUFFER:
ret = onStreamBufferError(resultExtras);
break;
case hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_REQUEST:
onRequestError(resultExtras);
break;
default:
ALOGE("%s: Unrecoverable error: %d detected!", __FUNCTION__, errorCode);
Mutex::Autolock l(mMutex);
mErrorState = true;
break;
}
return ret;
}
void CompositeStream::notifyError(int64_t frameNumber, int32_t requestId) {
sp<hardware::camera2::ICameraDeviceCallbacks> remoteCb =
mRemoteCallback.promote();
if ((frameNumber >= 0) && (remoteCb.get() != nullptr)) {
CaptureResultExtras extras;
extras.errorStreamId = getStreamId();
extras.frameNumber = frameNumber;
extras.requestId = requestId;
remoteCb->onDeviceError(
hardware::camera2::ICameraDeviceCallbacks::ERROR_CAMERA_BUFFER,
extras);
}
}
void CompositeStream::switchToOffline() {
Mutex::Autolock l(mMutex);
mDevice.clear();
}
}; // namespace camera3
}; // namespace android