* Copyright (c) 2022 Huawei Device Co., Ltd.
* 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 "hstream_common.h"
#include <atomic>
#include <cstdint>
#include <mutex>
#include <set>
#include <string>
#include "camera_log.h"
#include "camera_util.h"
#include "display/graphic/common/v2_1/cm_color_space.h"
#include "display/composer/v1_1/display_composer_type.h"
#include "ipc_skeleton.h"
#include "camera_report_uitls.h"
#include "rotate_plugin/camera_rotate_plugin.h"
#include "camera_util.h"
#include "image_receiver.h"
namespace OHOS {
namespace CameraStandard {
using namespace OHOS::HDI::Camera::V1_0;
using namespace OHOS::HDI::Display::Graphic::Common::V2_1;
using namespace OHOS::HDI::Display::Composer::V1_1;
using CM_ColorSpaceType_V2_1 = OHOS::HDI::Display::Graphic::Common::V2_1::CM_ColorSpaceType;
static const std::map<ColorSpace, CM_ColorSpaceType_V2_1> g_fwkToMetaColorSpaceMap_ = {
{COLOR_SPACE_UNKNOWN, CM_ColorSpaceType_V2_1::CM_COLORSPACE_NONE},
{DISPLAY_P3, CM_ColorSpaceType_V2_1::CM_P3_FULL},
{SRGB, CM_ColorSpaceType_V2_1::CM_SRGB_FULL},
{BT709, CM_ColorSpaceType_V2_1::CM_BT709_FULL},
{BT2020_HLG, CM_ColorSpaceType_V2_1::CM_BT2020_HLG_FULL},
{BT2020_PQ, CM_ColorSpaceType_V2_1::CM_BT2020_PQ_FULL},
{P3_HLG, CM_ColorSpaceType_V2_1::CM_P3_HLG_FULL},
{P3_PQ, CM_ColorSpaceType_V2_1::CM_P3_PQ_FULL},
{DISPLAY_P3_LIMIT, CM_ColorSpaceType_V2_1::CM_P3_LIMIT},
{SRGB_LIMIT, CM_ColorSpaceType_V2_1::CM_SRGB_LIMIT},
{BT709_LIMIT, CM_ColorSpaceType_V2_1::CM_BT709_LIMIT},
{BT2020_HLG_LIMIT, CM_ColorSpaceType_V2_1::CM_BT2020_HLG_LIMIT},
{BT2020_PQ_LIMIT, CM_ColorSpaceType_V2_1::CM_BT2020_PQ_LIMIT},
{P3_HLG_LIMIT, CM_ColorSpaceType_V2_1::CM_P3_HLG_LIMIT},
{P3_PQ_LIMIT, CM_ColorSpaceType_V2_1::CM_P3_PQ_LIMIT},
{H_LOG, CM_ColorSpaceType_V2_1::CM_BT2020_LOG_FULL},
};
namespace {
static const int32_t STREAMID_BEGIN = 1;
static const int32_t CAPTUREID_BEGIN = 1;
static const int32_t STREAMID_MAX = INT32_MAX - 1000;
static const int32_t CAPTUREID_MAX = INT32_MAX - 1000;
static std::atomic<int32_t> g_currentStreamId = STREAMID_BEGIN;
static std::atomic_int32_t g_currentCaptureId = CAPTUREID_BEGIN;
static int32_t GenerateStreamId()
{
int newId = g_currentStreamId++;
if (newId == STREAMID_MAX) {
g_currentStreamId = STREAMID_BEGIN;
}
return newId;
}
static int32_t GenerateCaptureId()
{
int32_t newId = g_currentCaptureId++;
if (newId == CAPTUREID_MAX) {
g_currentCaptureId = CAPTUREID_BEGIN;
}
return newId;
}
}
HStreamCommon::HStreamCommon(
StreamType streamType, sptr<OHOS::IBufferProducer> producer, int32_t format, int32_t width, int32_t height)
: format_(format), width_(width), height_(height), producer_(producer), streamType_(streamType)
{
MEDIA_DEBUG_LOG("Enter Into HStreamCommon::HStreamCommon");
callerToken_ = IPCSkeleton::GetCallingTokenID();
const int32_t metaStreamId = -1;
fwkStreamId_ = streamType == StreamType::METADATA ? metaStreamId : GenerateStreamId();
MEDIA_DEBUG_LOG("HStreamCommon Create streamId:%{public}d type:%{public}d width:%{public}d height:%{public}d"
" format:%{public}d ",
fwkStreamId_, streamType_, width_, height_, format_);
}
HStreamCommon::HStreamCommon(
StreamType streamType, int32_t format, int32_t width, int32_t height)
: format_(format), width_(width), height_(height), streamType_(streamType)
{
MEDIA_DEBUG_LOG("Enter Into HStreamCommon::HStreamCommon");
callerToken_ = IPCSkeleton::GetCallingTokenID();
const int32_t metaStreamId = -1;
fwkStreamId_ = streamType == StreamType::METADATA ? metaStreamId : GenerateStreamId();
MEDIA_DEBUG_LOG("HStreamCommon Create streamId:%{public}d type:%{public}d width:%{public}d height:%{public}d"
" format:%{public}d surfaceId:%{public}s",
fwkStreamId_, streamType_, width_, height_, format_, surfaceId_.c_str());
}
HStreamCommon::~HStreamCommon()
{
MEDIA_DEBUG_LOG("Enter Into HStreamCommon::~HStreamCommon streamId is:%{public}d, streamType is:%{public}d",
fwkStreamId_, streamType_);
streamOperator_ = nullptr;
}
void HStreamCommon::SetColorSpace(ColorSpace colorSpace)
{
auto itr = g_fwkToMetaColorSpaceMap_.find(colorSpace);
if (itr != g_fwkToMetaColorSpaceMap_.end()) {
dataSpace_ = itr->second;
MEDIA_INFO_LOG("HStreamCommon::SetColorSpace fwk colorSpace:%{public}d, HDI colorSpace: %{public}d", colorSpace,
dataSpace_);
} else {
MEDIA_ERR_LOG("HStreamCommon::SetColorSpace, %{public}d failed", static_cast<int32_t>(colorSpace));
}
}
ColorSpace HStreamCommon::GetColorSpace()
{
ColorSpace colorSpace = COLOR_SPACE_UNKNOWN;
for (const auto& iter: g_fwkToMetaColorSpaceMap_) {
if (iter.second == dataSpace_) {
colorSpace = iter.first;
break;
}
}
MEDIA_INFO_LOG("HStreamCommon::GetColorSpace fwk colorSpace:%{public}d", colorSpace);
return colorSpace;
}
int32_t HStreamCommon::LinkInput(wptr<OHOS::HDI::Camera::V1_0::IStreamOperator> streamOperator,
std::shared_ptr<OHOS::Camera::CameraMetadata> cameraAbility)
{
CHECK_RETURN_RET_ELOG(streamOperator == nullptr || cameraAbility == nullptr, CAMERA_INVALID_ARG,
"HStreamCommon::LinkInput streamOperator is null");
SetStreamOperator(streamOperator);
std::lock_guard<std::mutex> lock(cameraAbilityLock_);
cameraAbility_ = cameraAbility;
return CAMERA_OK;
}
int32_t HStreamCommon::UnlinkInput()
{
MEDIA_INFO_LOG("HStreamCommon::UnlinkInput streamType:%{public}d, streamId:%{public}d, hidStreamId:%{public}d",
streamType_, fwkStreamId_, hdiStreamId_);
StopStream();
SetStreamOperator(nullptr);
hdiStreamId_ = STREAM_ID_UNSET;
return CAMERA_OK;
}
int32_t HStreamCommon::StopStream()
{
CAMERA_SYNC_TRACE;
MEDIA_DEBUG_LOG("HStreamCommon::StopStream streamType:%{public}d, streamId:%{public}d, hdiStreamId:%{public}d, "
"captureId:%{public}d", streamType_, fwkStreamId_, hdiStreamId_, curCaptureID_);
auto streamOperator = GetStreamOperator();
CHECK_RETURN_RET_DLOG(streamOperator == nullptr, CAMERA_OK, "HStreamCommon::StopStream streamOperator is nullptr");
if (curCaptureID_ != CAPTURE_ID_UNSET) {
CamRetCode rc = (CamRetCode)(streamOperator->CancelCapture(curCaptureID_));
CHECK_PRINT_ELOG(rc != CamRetCode::NO_ERROR,
"HStreamCommon::StopStream streamOperator->CancelCapture get error code:%{public}d", rc);
ResetCaptureId();
return HdiToServiceError(rc);
}
return CAMERA_OK;
}
int32_t HStreamCommon::PrepareCaptureId()
{
curCaptureID_ = GenerateCaptureId();
captureIdForConfirmCapture_ = curCaptureID_;
return CAMERA_OK;
}
void HStreamCommon::ResetCaptureId()
{
curCaptureID_ = CAPTURE_ID_UNSET;
}
int32_t HStreamCommon::GetPreparedCaptureId()
{
return curCaptureID_;
}
void HStreamCommon::SetStreamInfo(StreamInfo_V1_5 &streamInfo)
{
int32_t pixelFormat = OHOS::HDI::Display::Composer::V1_1::PIXEL_FMT_YCRCB_420_SP;
auto it = g_cameraToPixelFormat.find(format_);
if (it != g_cameraToPixelFormat.end()) {
pixelFormat = it->second;
} else {
MEDIA_ERR_LOG("HStreamCommon::SetStreamInfo find format error, pixelFormat use default format");
}
MEDIA_DEBUG_LOG("HStreamCommon::SetStreamInfo pixelFormat:%{public}d type:%{public}d colorSpace:%{public}d",
pixelFormat, streamType_, dataSpace_);
streamInfo.v1_0.streamId_ = hdiStreamId_;
streamInfo.v1_0.width_ = width_;
streamInfo.v1_0.height_ = height_;
streamInfo.v1_0.format_ = pixelFormat;
streamInfo.v1_0.dataspace_ = dataSpace_;
streamInfo.v1_0.minFrameDuration_ = 0;
streamInfo.v1_0.tunneledMode_ = true;
{
std::lock_guard<std::mutex> lock(producerLock_);
if (producer_ != nullptr) {
MEDIA_DEBUG_LOG("HStreamCommon:producer is not null");
streamInfo.v1_0.bufferQueue_ = new BufferProducerSequenceable(producer_);
} else if (surface_!= nullptr && surface_->GetProducer() != nullptr) {
MEDIA_DEBUG_LOG("HStreamCommon:surface & producer is not null");
streamInfo.v1_0.bufferQueue_ = new BufferProducerSequenceable(surface_->GetProducer());
} else {
streamInfo.v1_0.bufferQueue_ = nullptr;
}
}
streamInfo.extendedStreamInfos = {};
streamInfo.settings = {};
}
int32_t HStreamCommon::ReleaseStream(bool isDelay)
{
MEDIA_DEBUG_LOG("Enter Into HStreamCommon::Release streamId is:%{public}d, hdiStreamId is:%{public}d, streamType "
"is:%{public}d, isDelay:%{public}d",
fwkStreamId_, hdiStreamId_, streamType_, isDelay);
StopStream();
bool isReleaseStream = !isDelay && hdiStreamId_ != STREAM_ID_UNSET;
if (isReleaseStream) {
auto streamOperator = GetStreamOperator();
if (streamOperator != nullptr) {
streamOperator->ReleaseStreams({ hdiStreamId_ });
}
}
fwkStreamId_ = STREAM_ID_UNSET;
hdiStreamId_ = STREAM_ID_UNSET;
SetStreamOperator(nullptr);
{
std::lock_guard<std::mutex> lock(cameraAbilityLock_);
cameraAbility_ = nullptr;
}
{
std::lock_guard<std::mutex> lock(producerLock_);
producer_ = nullptr;
}
return CAMERA_OK;
}
void HStreamCommon::DumpStreamInfo(CameraInfoDumper& infoDumper)
{
StreamInfo_V1_5 curStreamInfo;
SetStreamInfo(curStreamInfo);
std::string streamInfo = "Buffer producer Id:";
{
std::lock_guard<std::mutex> lock(producerLock_);
bool isProducerValid = curStreamInfo.v1_0.bufferQueue_ && curStreamInfo.v1_0.bufferQueue_->producer_;
if (isProducerValid) {
streamInfo.append("[" + std::to_string(curStreamInfo.v1_0.bufferQueue_->producer_->GetUniqueId()) + "]");
} else {
streamInfo.append("[empty]");
}
}
streamInfo.append(" stream Id:[" + std::to_string(curStreamInfo.v1_0.streamId_) + "]");
std::map<int, std::string>::const_iterator iter = g_cameraFormat.find(format_);
if (iter != g_cameraFormat.end()) {
streamInfo.append(" format:[" + iter->second + "]");
}
streamInfo.append(" width:[" + std::to_string(curStreamInfo.v1_0.width_) + "]");
streamInfo.append(" height:[" + std::to_string(curStreamInfo.v1_0.height_) + "]");
streamInfo.append(" dataspace:[" + std::to_string(curStreamInfo.v1_0.dataspace_) + "]");
streamInfo.append(" StreamType:[" + std::to_string(curStreamInfo.v1_0.intent_) + "]");
streamInfo.append(" TunnelMode:[" + std::to_string(curStreamInfo.v1_0.tunneledMode_) + "]");
streamInfo.append(" Encoding Type:[" + std::to_string(curStreamInfo.v1_0.encodeType_) + "]");
infoDumper.Msg(streamInfo);
infoDumper.Push();
infoDumper.Title("Stream Extened Info");
for (auto& extInfo : curStreamInfo.extendedStreamInfos) {
auto bufferQueue = extInfo.bufferQueue;
infoDumper.Msg("type:" + std::to_string(static_cast<int32_t>(extInfo.type)) +
" width:" + std::to_string(static_cast<int32_t>(extInfo.width)) +
" height:" + std::to_string(static_cast<int32_t>(extInfo.height)) +
" format:" + std::to_string(static_cast<int32_t>(extInfo.format)) +
" dataspace:" + std::to_string(static_cast<int32_t>(extInfo.dataspace)) +
" producer:" + ((bufferQueue == nullptr || bufferQueue->producer_ == nullptr)
? "empty"
: std::to_string(bufferQueue->producer_->GetUniqueId())));
}
infoDumper.Pop();
}
int32_t HStreamCommon::SetUsePhysicalCameraOrientation(bool isUsed)
{
std::lock_guard<std::mutex> lock(usePhysicalCameraOrientationMutex_);
usePhysicalCameraOrientation_ = isUsed;
MEDIA_INFO_LOG("HStreamCommon::SetUsePhysicalCameraOrientation isUsed %{public}d", isUsed);
return CAMERA_OK;
}
bool HStreamCommon::GetUsePhysicalCameraOrientation()
{
std::lock_guard<std::mutex> lock(usePhysicalCameraOrientationMutex_);
return usePhysicalCameraOrientation_;
}
void HStreamCommon::SetClientName(const std::string clientName)
{
std::lock_guard<std::mutex> lock(clientNameMutex_);
clientName_ = clientName;
}
std::string HStreamCommon::GetClientName()
{
std::lock_guard<std::mutex> lock(clientNameMutex_);
return clientName_;
}
void HStreamCommon::PrintCaptureDebugLog(const std::shared_ptr<OHOS::Camera::CameraMetadata> &captureMetadataSetting_)
{
CHECK_RETURN_ELOG(
captureMetadataSetting_ == nullptr, "HStreamCapture::PrintCaptureDebugLog captureMetadataSetting_ is nullptr");
camera_metadata_item_t item;
int result = OHOS::Camera::FindCameraMetadataItem(captureMetadataSetting_->get(), OHOS_JPEG_QUALITY, &item);
if (result != CAM_META_SUCCESS) {
MEDIA_DEBUG_LOG("HStreamCapture::Failed to find OHOS_JPEG_QUALITY tag");
} else {
MEDIA_DEBUG_LOG("HStreamCapture::find OHOS_JPEG_QUALITY value = %{public}d", item.data.u8[0]);
CameraReportUtils::GetInstance().UpdateImagingInfo(DFX_PHOTO_SETTING_QUALITY, std::to_string(item.data.u8[0]));
}
result = OHOS::Camera::FindCameraMetadataItem(captureMetadataSetting_->get(), OHOS_CONTROL_CAPTURE_MIRROR, &item);
if (result != CAM_META_SUCCESS) {
MEDIA_DEBUG_LOG("HStreamCapture::Failed to find OHOS_CONTROL_CAPTURE_MIRROR tag");
} else {
MEDIA_DEBUG_LOG("HStreamCapture::find OHOS_CONTROL_CAPTURE_MIRROR value = %{public}d", item.data.u8[0]);
CameraReportUtils::GetInstance().UpdateImagingInfo(DFX_PHOTO_SETTING_MIRROR, std::to_string(item.data.u8[0]));
}
result = OHOS::Camera::FindCameraMetadataItem(captureMetadataSetting_->get(), OHOS_JPEG_ORIENTATION, &item);
if (result != CAM_META_SUCCESS) {
MEDIA_DEBUG_LOG("HStreamCapture::Failed to find OHOS_JPEG_ORIENTATION tag");
} else {
MEDIA_DEBUG_LOG("HStreamCapture::find OHOS_JPEG_ORIENTATION value = %{public}d", item.data.i32[0]);
CameraReportUtils::GetInstance().UpdateImagingInfo(DFX_PHOTO_SETTING_ROTATION,
std::to_string(item.data.i32[0]));
}
CallerInfo caller = CameraReportUtils::GetCallerInfo();
result = OHOS::Camera::FindCameraMetadataItem(captureMetadataSetting_->get(), OHOS_CONTROL_FPS_RANGES, &item);
if (result != CAM_META_SUCCESS) {
MEDIA_DEBUG_LOG("HStreamCapture::Failed to find OHOS_CONTROL_FPS_RANGES tag");
} else {
MEDIA_DEBUG_LOG("HStreamCapture::find OHOS_CONTROL_FPS_RANGES value = %{public}d",
item.data.i32[0]);
CameraReportUtils::GetInstance().ReportUserBehavior(DFX_UB_SET_FRAMERATERANGE,
std::to_string(item.data.i32[0]), caller);
}
}
void HStreamCommon::SetBasicInfo(std::map<int32_t, std::string> basicParam)
{
std::lock_guard<std::mutex> lock(basicInfoLock_);
param = basicParam;
}
void HStreamCommon::SetCameraPermissionUsedRecordFunction(RecordFunction recordFunction)
{
recordFunction_ = recordFunction;
}
void HStreamCommon::AddCameraPermissionUsedRecord()
{
if (recordFunction_) {
recordFunction_();
} else {
MEDIA_ERR_LOG("HStreamCommon::AddCameraPermissionUsedRecord failed, recordFunction is null");
}
}
}
}