* Copyright (C) 2023 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 "avmuxer_demo.h"
#include <iostream>
#include <fstream>
#include <cstdio>
#include <unistd.h>
#include <fcntl.h>
#include <thread>
#include <vector>
#include "avcodec_errors.h"
namespace OHOS {
namespace MediaAVCodec {
int AVMuxerDemo::DoWriteSample(uint32_t trackIndex, std::shared_ptr<AVBuffer> sample)
{
if (avmuxer_ != nullptr &&
avmuxer_->WriteSample(trackIndex, sample) == AVCS_ERR_OK) {
return 0;
}
return -1;
}
int AVMuxerDemo::DoAddTrack(int32_t &trackIndex, std::shared_ptr<Meta> trackDesc)
{
int ret;
if ((ret = avmuxer_->AddTrack(trackIndex, trackDesc)) != AVCS_ERR_OK) {
std::cout<<"AVMuxerDemo::DoAddTrack failed! ret:"<<ret<<std::endl;
return -1;
}
return 0;
}
sptr<AVBufferQueueProducer> AVMuxerDemo::DoGetInputBufferQueue(uint32_t trackIndex)
{
std::cout<<"AVMuxerDemo::DoGetInputBufferQueue "<<trackIndex<<std::endl;
return avmuxer_->GetInputBufferQueue(trackIndex);
}
void AVMuxerDemo::DoRunMuxer(const std::string &runMode)
{
std::string outFileName = GetOutputFileName("inner_mux_" + runMode);
outFd_ = open(outFileName.c_str(), O_CREAT | O_RDWR | O_TRUNC, S_IRUSR | S_IWUSR);
if (outFd_ < 0) {
std::cout << "Open file failed! filePath is: " << outFileName << std::endl;
return;
}
std::cout<<"==== open success! =====\noutputFileName: "<<outFileName<<"\n============"<<std::endl;
long long testTimeStart = GetTimestamp();
avmuxer_ = AVMuxerFactory::CreateAVMuxer(outFd_, outputFormat_);
if (avmuxer_ == nullptr) {
std::cout << "avmuxer_ is null" << std::endl;
return;
}
std::cout << "create muxer success " << avmuxer_ << std::endl;
SetParameter();
AddAudioTrack(audioParams_);
AddVideoTrack(videoParams_);
AddCoverTrack(coverParams_);
std::cout << "add track success" << std::endl;
if (avmuxer_->Start() != AVCS_ERR_OK) {
return;
}
std::cout << "start muxer success" << std::endl;
WriteCoverSample();
std::cout<<"AVMuxerDemo::DoRunMuxer runMode is : "<<runMode<<std::endl;
if (runMode.compare(RUN_NORMAL) == 0) {
WriteTrackSample();
} else if (runMode.compare(RUN_MUL_THREAD) == 0) {
std::vector<std::thread> vecThread;
vecThread.emplace_back(MulThdWriteTrackSample, this, audioTrackId_, audioFile_);
vecThread.emplace_back(MulThdWriteTrackSample, this, videoTrackId_, videoFile_);
for (uint32_t i = 0; i < vecThread.size(); ++i) {
vecThread[i].join();
}
}
std::cout << "write muxer success" << std::endl;
if (avmuxer_->Stop() != AVCS_ERR_OK) {
return;
}
std::cout << "stop muxer success" << std::endl;
long long testTimeEnd = GetTimestamp();
std::cout << "muxer used time: " << testTimeEnd - testTimeStart << "us" << std::endl;
}
void AVMuxerDemo::DoRunMuxer()
{
DoRunMuxer(std::string(RUN_NORMAL));
}
void AVMuxerDemo::DoRunMultiThreadCase()
{
DoRunMuxer(std::string(RUN_MUL_THREAD));
}
void AVMuxerDemo::SetParameter()
{
std::shared_ptr<Meta> param = std::make_shared<Meta>();
param->Set<Tag::VIDEO_ROTATION>(Plugins::VideoRotation::VIDEO_ROTATION_0);
param->Set<Tag::MEDIA_CREATION_TIME>("2023-12-19T03:16:00.000Z");
param->Set<Tag::MEDIA_LATITUDE>(22.67f);
param->Set<Tag::MEDIA_LONGITUDE>(114.06f);
param->Set<Tag::MEDIA_TITLE>("ohos muxer");
param->Set<Tag::MEDIA_ARTIST>("ohos muxer");
param->Set<Tag::MEDIA_COMPOSER>("ohos muxer");
param->Set<Tag::MEDIA_DATE>("2023-12-19");
param->Set<Tag::MEDIA_ALBUM>("ohos muxer");
param->Set<Tag::MEDIA_ALBUM_ARTIST>("ohos muxer");
param->Set<Tag::MEDIA_COPYRIGHT>("ohos muxer");
if (avmuxer_->SetParameter(param) != AVCS_ERR_OK) {
std::cout<<"set parameter failed!"<<std::endl;
}
}
}
}