1da853ecaSopenharmony_ci/*
2da853ecaSopenharmony_ci * Copyright (C) 2023 Huawei Device Co., Ltd.
3da853ecaSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4da853ecaSopenharmony_ci * you may not use this file except in compliance with the License.
5da853ecaSopenharmony_ci * You may obtain a copy of the License at
6da853ecaSopenharmony_ci *
7da853ecaSopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8da853ecaSopenharmony_ci *
9da853ecaSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10da853ecaSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11da853ecaSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12da853ecaSopenharmony_ci * See the License for the specific language governing permissions and
13da853ecaSopenharmony_ci * limitations under the License.
14da853ecaSopenharmony_ci */
15da853ecaSopenharmony_ci
16da853ecaSopenharmony_ci#include "avmuxer_demo.h"
17da853ecaSopenharmony_ci#include <iostream>
18da853ecaSopenharmony_ci#include <fstream>
19da853ecaSopenharmony_ci#include <cstdio>
20da853ecaSopenharmony_ci#include <unistd.h>
21da853ecaSopenharmony_ci#include <fcntl.h>
22da853ecaSopenharmony_ci#include <thread>
23da853ecaSopenharmony_ci#include <vector>
24da853ecaSopenharmony_ci#include "avcodec_errors.h"
25da853ecaSopenharmony_ci
26da853ecaSopenharmony_cinamespace OHOS {
27da853ecaSopenharmony_cinamespace MediaAVCodec {
28da853ecaSopenharmony_ciint AVMuxerDemo::DoWriteSample(uint32_t trackIndex, std::shared_ptr<AVBuffer> sample)
29da853ecaSopenharmony_ci{
30da853ecaSopenharmony_ci    if (avmuxer_ != nullptr &&
31da853ecaSopenharmony_ci        avmuxer_->WriteSample(trackIndex, sample) == AVCS_ERR_OK) {
32da853ecaSopenharmony_ci        return 0;
33da853ecaSopenharmony_ci    }
34da853ecaSopenharmony_ci    return -1;
35da853ecaSopenharmony_ci}
36da853ecaSopenharmony_ci
37da853ecaSopenharmony_ciint AVMuxerDemo::DoAddTrack(int32_t &trackIndex, std::shared_ptr<Meta> trackDesc)
38da853ecaSopenharmony_ci{
39da853ecaSopenharmony_ci    int ret;
40da853ecaSopenharmony_ci    if ((ret = avmuxer_->AddTrack(trackIndex, trackDesc)) != AVCS_ERR_OK) {
41da853ecaSopenharmony_ci        std::cout<<"AVMuxerDemo::DoAddTrack failed! ret:"<<ret<<std::endl;
42da853ecaSopenharmony_ci        return -1;
43da853ecaSopenharmony_ci    }
44da853ecaSopenharmony_ci    return 0;
45da853ecaSopenharmony_ci}
46da853ecaSopenharmony_ci
47da853ecaSopenharmony_cisptr<AVBufferQueueProducer> AVMuxerDemo::DoGetInputBufferQueue(uint32_t trackIndex)
48da853ecaSopenharmony_ci{
49da853ecaSopenharmony_ci    std::cout<<"AVMuxerDemo::DoGetInputBufferQueue "<<trackIndex<<std::endl;
50da853ecaSopenharmony_ci    return avmuxer_->GetInputBufferQueue(trackIndex);
51da853ecaSopenharmony_ci}
52da853ecaSopenharmony_ci
53da853ecaSopenharmony_civoid AVMuxerDemo::DoRunMuxer(const std::string &runMode)
54da853ecaSopenharmony_ci{
55da853ecaSopenharmony_ci    std::string outFileName = GetOutputFileName("inner_mux_" + runMode);
56da853ecaSopenharmony_ci    outFd_ = open(outFileName.c_str(), O_CREAT | O_RDWR | O_TRUNC, S_IRUSR | S_IWUSR);
57da853ecaSopenharmony_ci    if (outFd_ < 0) {
58da853ecaSopenharmony_ci        std::cout << "Open file failed! filePath is: " << outFileName << std::endl;
59da853ecaSopenharmony_ci        return;
60da853ecaSopenharmony_ci    }
61da853ecaSopenharmony_ci    std::cout<<"==== open success! =====\noutputFileName: "<<outFileName<<"\n============"<<std::endl;
62da853ecaSopenharmony_ci    long long testTimeStart = GetTimestamp();
63da853ecaSopenharmony_ci    avmuxer_ = AVMuxerFactory::CreateAVMuxer(outFd_, outputFormat_);
64da853ecaSopenharmony_ci    if (avmuxer_ == nullptr) {
65da853ecaSopenharmony_ci        std::cout << "avmuxer_ is null" << std::endl;
66da853ecaSopenharmony_ci        return;
67da853ecaSopenharmony_ci    }
68da853ecaSopenharmony_ci    std::cout << "create muxer success " << avmuxer_ << std::endl;
69da853ecaSopenharmony_ci
70da853ecaSopenharmony_ci    SetParameter();
71da853ecaSopenharmony_ci    AddAudioTrack(audioParams_);
72da853ecaSopenharmony_ci    AddVideoTrack(videoParams_);
73da853ecaSopenharmony_ci    AddCoverTrack(coverParams_);
74da853ecaSopenharmony_ci
75da853ecaSopenharmony_ci    std::cout << "add track success" << std::endl;
76da853ecaSopenharmony_ci
77da853ecaSopenharmony_ci    if (avmuxer_->Start() != AVCS_ERR_OK) {
78da853ecaSopenharmony_ci        return;
79da853ecaSopenharmony_ci    }
80da853ecaSopenharmony_ci
81da853ecaSopenharmony_ci    std::cout << "start muxer success" << std::endl;
82da853ecaSopenharmony_ci
83da853ecaSopenharmony_ci    WriteCoverSample();
84da853ecaSopenharmony_ci
85da853ecaSopenharmony_ci    std::cout<<"AVMuxerDemo::DoRunMuxer runMode is : "<<runMode<<std::endl;
86da853ecaSopenharmony_ci    if (runMode.compare(RUN_NORMAL) == 0) {
87da853ecaSopenharmony_ci        WriteTrackSample();
88da853ecaSopenharmony_ci    } else if (runMode.compare(RUN_MUL_THREAD) == 0) {
89da853ecaSopenharmony_ci        std::vector<std::thread> vecThread;
90da853ecaSopenharmony_ci        vecThread.emplace_back(MulThdWriteTrackSample, this, audioTrackId_, audioFile_);
91da853ecaSopenharmony_ci        vecThread.emplace_back(MulThdWriteTrackSample, this, videoTrackId_, videoFile_);
92da853ecaSopenharmony_ci        for (uint32_t i = 0; i < vecThread.size(); ++i) {
93da853ecaSopenharmony_ci            vecThread[i].join();
94da853ecaSopenharmony_ci        }
95da853ecaSopenharmony_ci    }
96da853ecaSopenharmony_ci
97da853ecaSopenharmony_ci    std::cout << "write muxer success" << std::endl;
98da853ecaSopenharmony_ci
99da853ecaSopenharmony_ci    if (avmuxer_->Stop() != AVCS_ERR_OK) {
100da853ecaSopenharmony_ci        return;
101da853ecaSopenharmony_ci    }
102da853ecaSopenharmony_ci    std::cout << "stop muxer success" << std::endl;
103da853ecaSopenharmony_ci    long long testTimeEnd = GetTimestamp();
104da853ecaSopenharmony_ci    std::cout << "muxer used time: " << testTimeEnd - testTimeStart << "us" << std::endl;
105da853ecaSopenharmony_ci}
106da853ecaSopenharmony_ci
107da853ecaSopenharmony_civoid AVMuxerDemo::DoRunMuxer()
108da853ecaSopenharmony_ci{
109da853ecaSopenharmony_ci    DoRunMuxer(std::string(RUN_NORMAL));
110da853ecaSopenharmony_ci}
111da853ecaSopenharmony_ci
112da853ecaSopenharmony_civoid AVMuxerDemo::DoRunMultiThreadCase()
113da853ecaSopenharmony_ci{
114da853ecaSopenharmony_ci    DoRunMuxer(std::string(RUN_MUL_THREAD));
115da853ecaSopenharmony_ci}
116da853ecaSopenharmony_ci
117da853ecaSopenharmony_civoid AVMuxerDemo::SetParameter()
118da853ecaSopenharmony_ci{
119da853ecaSopenharmony_ci    std::shared_ptr<Meta> param = std::make_shared<Meta>();
120da853ecaSopenharmony_ci    param->Set<Tag::VIDEO_ROTATION>(Plugins::VideoRotation::VIDEO_ROTATION_0);
121da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_CREATION_TIME>("2023-12-19T03:16:00.000Z");
122da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_LATITUDE>(22.67f);
123da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_LONGITUDE>(114.06f);
124da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_TITLE>("ohos muxer");
125da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_ARTIST>("ohos muxer");
126da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_COMPOSER>("ohos muxer");
127da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_DATE>("2023-12-19");
128da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_ALBUM>("ohos muxer");
129da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_ALBUM_ARTIST>("ohos muxer");
130da853ecaSopenharmony_ci    param->Set<Tag::MEDIA_COPYRIGHT>("ohos muxer");
131da853ecaSopenharmony_ci    if (avmuxer_->SetParameter(param) != AVCS_ERR_OK) {
132da853ecaSopenharmony_ci        std::cout<<"set parameter failed!"<<std::endl;
133da853ecaSopenharmony_ci    }
134da853ecaSopenharmony_ci}
135da853ecaSopenharmony_ci}  // namespace MediaAVCodec
136da853ecaSopenharmony_ci}  // namespace OHOS