1/* 2 * Copyright (C) 2023 Huawei Device Co., Ltd. 3 * Licensed under the Apache License, Version 2.0 (the "License"); 4 * you may not use this file except in compliance with the License. 5 * You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software 10 * distributed under the License is distributed on an "AS IS" BASIS, 11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 * See the License for the specific language governing permissions and 13 * limitations under the License. 14 */ 15 16#include "avmuxer_impl.h" 17#include <unistd.h> 18#include <fcntl.h> 19#include "securec.h" 20#include "avcodec_trace.h" 21#include "avcodec_log.h" 22#include "avcodec_errors.h" 23 24namespace { 25constexpr OHOS::HiviewDFX::HiLogLabel LABEL = {LOG_CORE, LOG_DOMAIN_MUXER, "AVMuxerImpl"}; 26} 27 28namespace OHOS { 29namespace MediaAVCodec { 30std::shared_ptr<AVMuxer> AVMuxerFactory::CreateAVMuxer(int32_t fd, Plugins::OutputFormat format) 31{ 32 AVCODEC_SYNC_TRACE; 33 CHECK_AND_RETURN_RET_LOG(fd >= 0, nullptr, "fd %{public}d is error!", fd); 34 uint32_t fdPermission = static_cast<uint32_t>(fcntl(fd, F_GETFL, 0)); 35 CHECK_AND_RETURN_RET_LOG((fdPermission & O_WRONLY) == O_WRONLY || (fdPermission & O_RDWR) == O_RDWR, 36 nullptr, "No permission to write fd."); 37 CHECK_AND_RETURN_RET_LOG(lseek(fd, 0, SEEK_CUR) != -1, nullptr, "The fd is not seekable"); 38 39 std::shared_ptr<AVMuxerImpl> impl = std::make_shared<AVMuxerImpl>(); 40 41 int32_t ret = impl->Init(fd, format); 42 CHECK_AND_RETURN_RET_LOG(ret == AVCS_ERR_OK, nullptr, "Init avmuxer implementation failed"); 43 return impl; 44} 45 46AVMuxerImpl::AVMuxerImpl() 47{ 48 AVCODEC_LOGD("AVMuxerImpl:0x%{public}06" PRIXPTR " Instances create", FAKE_POINTER(this)); 49} 50 51AVMuxerImpl::~AVMuxerImpl() 52{ 53 AVCODEC_LOGD("AVMuxerImpl:0x%{public}06" PRIXPTR " Instances destroy", FAKE_POINTER(this)); 54} 55 56int32_t AVMuxerImpl::Init(int32_t fd, Plugins::OutputFormat format) 57{ 58 AVCODEC_SYNC_TRACE; 59 muxerEngine_ = std::make_shared<Media::MediaMuxer>(getuid(), getprocpid()); 60 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_NO_MEMORY, "Create AVMuxer Engine failed"); 61 return StatusConvert(muxerEngine_->Init(fd, format)); 62} 63 64int32_t AVMuxerImpl::SetParameter(const std::shared_ptr<Meta> ¶m) 65{ 66 AVCODEC_SYNC_TRACE; 67 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 68 CHECK_AND_RETURN_RET_LOG(param != nullptr, AVCS_ERR_INVALID_VAL, "Invalid parameter"); 69 return StatusConvert(muxerEngine_->SetParameter(param)); 70} 71 72int32_t AVMuxerImpl::SetUserMeta(const std::shared_ptr<Meta> &userMeta) 73{ 74 AVCODEC_SYNC_TRACE; 75 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 76 CHECK_AND_RETURN_RET_LOG(userMeta != nullptr, AVCS_ERR_INVALID_VAL, "Invalid parameter"); 77 return StatusConvert(muxerEngine_->SetUserMeta(userMeta)); 78} 79 80int32_t AVMuxerImpl::AddTrack(int32_t &trackIndex, const std::shared_ptr<Meta> &trackDesc) 81{ 82 AVCODEC_SYNC_TRACE; 83 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 84 CHECK_AND_RETURN_RET_LOG(trackDesc != nullptr, AVCS_ERR_INVALID_VAL, "Invalid track format"); 85 return StatusConvert(muxerEngine_->AddTrack(trackIndex, trackDesc)); 86} 87 88sptr<AVBufferQueueProducer> AVMuxerImpl::GetInputBufferQueue(uint32_t trackIndex) 89{ 90 AVCODEC_SYNC_TRACE; 91 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, nullptr, "AVMuxer Engine does not exist"); 92 return muxerEngine_->GetInputBufferQueue(trackIndex); 93} 94 95int32_t AVMuxerImpl::Start() 96{ 97 AVCODEC_SYNC_TRACE; 98 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 99 return StatusConvert(muxerEngine_->Start()); 100} 101 102int32_t AVMuxerImpl::WriteSample(uint32_t trackIndex, const std::shared_ptr<AVBuffer> &sample) 103{ 104 AVCODEC_SYNC_TRACE; 105 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 106 CHECK_AND_RETURN_RET_LOG(sample != nullptr && sample->memory_ != nullptr && 107 sample->memory_->GetSize() >= 0, AVCS_ERR_INVALID_VAL, "Invalid memory"); 108 return StatusConvert(muxerEngine_->WriteSample(trackIndex, sample)); 109} 110 111int32_t AVMuxerImpl::Stop() 112{ 113 AVCODEC_SYNC_TRACE; 114 CHECK_AND_RETURN_RET_LOG(muxerEngine_ != nullptr, AVCS_ERR_INVALID_OPERATION, "AVMuxer Engine does not exist"); 115 return StatusConvert(muxerEngine_->Stop()); 116} 117 118int32_t AVMuxerImpl::StatusConvert(Media::Status status) 119{ 120 const static std::unordered_map<Media::Status, int32_t> table = { 121 {Status::END_OF_STREAM, AVCodecServiceErrCode::AVCS_ERR_OK}, 122 {Status::OK, AVCodecServiceErrCode::AVCS_ERR_OK}, 123 {Status::NO_ERROR, AVCodecServiceErrCode::AVCS_ERR_OK}, 124 {Status::ERROR_UNKNOWN, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 125 {Status::ERROR_PLUGIN_ALREADY_EXISTS, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 126 {Status::ERROR_INCOMPATIBLE_VERSION, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 127 {Status::ERROR_NO_MEMORY, AVCodecServiceErrCode::AVCS_ERR_NO_MEMORY}, 128 {Status::ERROR_WRONG_STATE, AVCodecServiceErrCode::AVCS_ERR_INVALID_OPERATION}, 129 {Status::ERROR_UNIMPLEMENTED, AVCodecServiceErrCode::AVCS_ERR_UNSUPPORT}, 130 {Status::ERROR_INVALID_PARAMETER, AVCodecServiceErrCode::AVCS_ERR_INVALID_VAL}, 131 {Status::ERROR_INVALID_DATA, AVCodecServiceErrCode::AVCS_ERR_INVALID_VAL}, 132 {Status::ERROR_MISMATCHED_TYPE, AVCodecServiceErrCode::AVCS_ERR_INVALID_VAL}, 133 {Status::ERROR_TIMED_OUT, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 134 {Status::ERROR_UNSUPPORTED_FORMAT, AVCodecServiceErrCode::AVCS_ERR_UNSUPPORT_FILE_TYPE}, 135 {Status::ERROR_NOT_ENOUGH_DATA, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 136 {Status::ERROR_NOT_EXISTED, AVCodecServiceErrCode::AVCS_ERR_OPEN_FILE_FAILED}, 137 {Status::ERROR_AGAIN, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 138 {Status::ERROR_PERMISSION_DENIED, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 139 {Status::ERROR_NULL_POINTER, AVCodecServiceErrCode::AVCS_ERR_INVALID_VAL}, 140 {Status::ERROR_INVALID_OPERATION, AVCodecServiceErrCode::AVCS_ERR_INVALID_OPERATION}, 141 {Status::ERROR_CLIENT, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 142 {Status::ERROR_SERVER, AVCodecServiceErrCode::AVCS_ERR_UNKNOWN}, 143 {Status::ERROR_DELAY_READY, AVCodecServiceErrCode::AVCS_ERR_OK}, 144 {Status::ERROR_INVALID_BUFFER_SIZE, AVCodecServiceErrCode::AVCS_ERR_INVALID_VAL}, 145 }; 146 auto ite = table.find(status); 147 if (ite == table.end()) { 148 return AVCodecServiceErrCode::AVCS_ERR_UNKNOWN; 149 } 150 return ite->second; 151} 152} // namespace MediaAVCodec 153} // namespace OHOS