You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
336 lines
12 KiB
336 lines
12 KiB
/*
|
|
* Copyright (C) 2020 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.
|
|
*/
|
|
|
|
// Unit Test for MediaTrackTranscoder
|
|
|
|
// #define LOG_NDEBUG 0
|
|
#define LOG_TAG "MediaTrackTranscoderTests"
|
|
|
|
#include <android-base/logging.h>
|
|
#include <android/binder_process.h>
|
|
#include <fcntl.h>
|
|
#include <gtest/gtest.h>
|
|
#include <media/MediaSampleReaderNDK.h>
|
|
#include <media/MediaTrackTranscoder.h>
|
|
#include <media/PassthroughTrackTranscoder.h>
|
|
#include <media/VideoTrackTranscoder.h>
|
|
|
|
#include "TranscoderTestUtils.h"
|
|
|
|
namespace android {
|
|
|
|
/** TrackTranscoder types to test. */
|
|
enum TrackTranscoderType {
|
|
VIDEO,
|
|
PASSTHROUGH,
|
|
};
|
|
|
|
class MediaTrackTranscoderTests : public ::testing::TestWithParam<TrackTranscoderType> {
|
|
public:
|
|
MediaTrackTranscoderTests() { LOG(DEBUG) << "MediaTrackTranscoderTests created"; }
|
|
|
|
void SetUp() override {
|
|
LOG(DEBUG) << "MediaTrackTranscoderTests set up";
|
|
|
|
// Need to start a thread pool to prevent AMediaExtractor binder calls from starving
|
|
// (b/155663561).
|
|
ABinderProcess_startThreadPool();
|
|
|
|
mCallback = std::make_shared<TestTrackTranscoderCallback>();
|
|
|
|
switch (GetParam()) {
|
|
case VIDEO:
|
|
mTranscoder = VideoTrackTranscoder::create(mCallback);
|
|
break;
|
|
case PASSTHROUGH:
|
|
mTranscoder = std::make_shared<PassthroughTrackTranscoder>(mCallback);
|
|
break;
|
|
}
|
|
ASSERT_NE(mTranscoder, nullptr);
|
|
|
|
initSampleReader("/data/local/tmp/TranscodingTestAssets/cubicle_avc_480x240_aac_24KHz.mp4");
|
|
}
|
|
|
|
void initSampleReader(const char* sourcePath) {
|
|
const int sourceFd = open(sourcePath, O_RDONLY);
|
|
ASSERT_GT(sourceFd, 0);
|
|
|
|
const size_t fileSize = lseek(sourceFd, 0, SEEK_END);
|
|
lseek(sourceFd, 0, SEEK_SET);
|
|
|
|
mMediaSampleReader = MediaSampleReaderNDK::createFromFd(sourceFd, 0 /* offset */, fileSize);
|
|
ASSERT_NE(mMediaSampleReader, nullptr);
|
|
close(sourceFd);
|
|
|
|
for (size_t trackIndex = 0; trackIndex < mMediaSampleReader->getTrackCount();
|
|
++trackIndex) {
|
|
AMediaFormat* trackFormat = mMediaSampleReader->getTrackFormat(trackIndex);
|
|
ASSERT_NE(trackFormat, nullptr);
|
|
|
|
const char* mime = nullptr;
|
|
AMediaFormat_getString(trackFormat, AMEDIAFORMAT_KEY_MIME, &mime);
|
|
ASSERT_NE(mime, nullptr);
|
|
|
|
if (GetParam() == VIDEO && strncmp(mime, "video/", 6) == 0) {
|
|
mTrackIndex = trackIndex;
|
|
|
|
mSourceFormat = std::shared_ptr<AMediaFormat>(trackFormat, &AMediaFormat_delete);
|
|
ASSERT_NE(mSourceFormat, nullptr);
|
|
|
|
mDestinationFormat =
|
|
TrackTranscoderTestUtils::getDefaultVideoDestinationFormat(trackFormat);
|
|
ASSERT_NE(mDestinationFormat, nullptr);
|
|
break;
|
|
} else if (GetParam() == PASSTHROUGH && strncmp(mime, "audio/", 6) == 0) {
|
|
// TODO(lnilsson): Test metadata track passthrough after hkuang@ provides sample.
|
|
mTrackIndex = trackIndex;
|
|
|
|
mSourceFormat = std::shared_ptr<AMediaFormat>(trackFormat, &AMediaFormat_delete);
|
|
ASSERT_NE(mSourceFormat, nullptr);
|
|
break;
|
|
}
|
|
|
|
AMediaFormat_delete(trackFormat);
|
|
}
|
|
|
|
ASSERT_NE(mSourceFormat, nullptr);
|
|
EXPECT_EQ(mMediaSampleReader->selectTrack(mTrackIndex), AMEDIA_OK);
|
|
}
|
|
|
|
// Drains the transcoder's output queue in a loop.
|
|
void drainOutputSamples(int numSamplesToSave = 0) {
|
|
mTranscoder->setSampleConsumer(
|
|
[this, numSamplesToSave](const std::shared_ptr<MediaSample>& sample) {
|
|
ASSERT_NE(sample, nullptr);
|
|
|
|
mGotEndOfStream = (sample->info.flags & SAMPLE_FLAG_END_OF_STREAM) != 0;
|
|
|
|
if (mSavedSamples.size() < numSamplesToSave) {
|
|
mSavedSamples.push_back(sample);
|
|
}
|
|
|
|
if (mSavedSamples.size() == numSamplesToSave || mGotEndOfStream) {
|
|
mSamplesSavedSemaphore.signal();
|
|
}
|
|
});
|
|
}
|
|
|
|
void TearDown() override { LOG(DEBUG) << "MediaTrackTranscoderTests tear down"; }
|
|
|
|
~MediaTrackTranscoderTests() { LOG(DEBUG) << "MediaTrackTranscoderTests destroyed"; }
|
|
|
|
protected:
|
|
std::shared_ptr<MediaTrackTranscoder> mTranscoder;
|
|
std::shared_ptr<TestTrackTranscoderCallback> mCallback;
|
|
|
|
std::shared_ptr<MediaSampleReader> mMediaSampleReader;
|
|
int mTrackIndex;
|
|
|
|
std::shared_ptr<AMediaFormat> mSourceFormat;
|
|
std::shared_ptr<AMediaFormat> mDestinationFormat;
|
|
|
|
std::vector<std::shared_ptr<MediaSample>> mSavedSamples;
|
|
OneShotSemaphore mSamplesSavedSemaphore;
|
|
bool mGotEndOfStream = false;
|
|
};
|
|
|
|
TEST_P(MediaTrackTranscoderTests, WaitNormalOperation) {
|
|
LOG(DEBUG) << "Testing WaitNormalOperation";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
ASSERT_TRUE(mTranscoder->start());
|
|
drainOutputSamples();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
EXPECT_TRUE(mCallback->transcodingFinished());
|
|
EXPECT_TRUE(mGotEndOfStream);
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, StopNormalOperation) {
|
|
LOG(DEBUG) << "Testing StopNormalOperation";
|
|
|
|
// Use a longer test asset to make sure that transcoding can be stopped.
|
|
initSampleReader("/data/local/tmp/TranscodingTestAssets/longtest_15s.mp4");
|
|
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
EXPECT_TRUE(mTranscoder->start());
|
|
mCallback->waitUntilTrackFormatAvailable();
|
|
mTranscoder->stop();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
EXPECT_TRUE(mCallback->transcodingWasStopped());
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, StartWithoutConfigure) {
|
|
LOG(DEBUG) << "Testing StartWithoutConfigure";
|
|
EXPECT_FALSE(mTranscoder->start());
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, StopWithoutStart) {
|
|
LOG(DEBUG) << "Testing StopWithoutStart";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
mTranscoder->stop();
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, DoubleStartStop) {
|
|
LOG(DEBUG) << "Testing DoubleStartStop";
|
|
|
|
// Use a longer test asset to make sure that transcoding can be stopped.
|
|
initSampleReader("/data/local/tmp/TranscodingTestAssets/longtest_15s.mp4");
|
|
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
EXPECT_TRUE(mTranscoder->start());
|
|
EXPECT_FALSE(mTranscoder->start());
|
|
mTranscoder->stop();
|
|
mTranscoder->stop();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
EXPECT_TRUE(mCallback->transcodingWasStopped());
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, DoubleConfigure) {
|
|
LOG(DEBUG) << "Testing DoubleConfigure";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_ERROR_UNSUPPORTED);
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, ConfigureAfterFail) {
|
|
LOG(DEBUG) << "Testing ConfigureAfterFail";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, -1, mDestinationFormat),
|
|
AMEDIA_ERROR_INVALID_PARAMETER);
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, RestartAfterStop) {
|
|
LOG(DEBUG) << "Testing RestartAfterStop";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
EXPECT_TRUE(mTranscoder->start());
|
|
mTranscoder->stop();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
EXPECT_FALSE(mTranscoder->start());
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, RestartAfterFinish) {
|
|
LOG(DEBUG) << "Testing RestartAfterFinish";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
ASSERT_TRUE(mTranscoder->start());
|
|
drainOutputSamples();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
mTranscoder->stop();
|
|
EXPECT_FALSE(mTranscoder->start());
|
|
EXPECT_TRUE(mGotEndOfStream);
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, HoldSampleAfterTranscoderRelease) {
|
|
LOG(DEBUG) << "Testing HoldSampleAfterTranscoderRelease";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
ASSERT_TRUE(mTranscoder->start());
|
|
drainOutputSamples(1 /* numSamplesToSave */);
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
mTranscoder->stop();
|
|
EXPECT_TRUE(mGotEndOfStream);
|
|
|
|
mTranscoder.reset();
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(20));
|
|
mSavedSamples.clear();
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, HoldSampleAfterTranscoderStop) {
|
|
LOG(DEBUG) << "Testing HoldSampleAfterTranscoderStop";
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
ASSERT_TRUE(mTranscoder->start());
|
|
drainOutputSamples(1 /* numSamplesToSave */);
|
|
mSamplesSavedSemaphore.wait();
|
|
mTranscoder->stop();
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(20));
|
|
mSavedSamples.clear();
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, NullSampleReader) {
|
|
LOG(DEBUG) << "Testing NullSampleReader";
|
|
std::shared_ptr<MediaSampleReader> nullSampleReader;
|
|
EXPECT_NE(mTranscoder->configure(nullSampleReader, mTrackIndex, mDestinationFormat), AMEDIA_OK);
|
|
ASSERT_FALSE(mTranscoder->start());
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, InvalidTrackIndex) {
|
|
LOG(DEBUG) << "Testing InvalidTrackIndex";
|
|
EXPECT_NE(mTranscoder->configure(mMediaSampleReader, -1, mDestinationFormat), AMEDIA_OK);
|
|
EXPECT_NE(mTranscoder->configure(mMediaSampleReader, mMediaSampleReader->getTrackCount(),
|
|
mDestinationFormat),
|
|
AMEDIA_OK);
|
|
}
|
|
|
|
TEST_P(MediaTrackTranscoderTests, StopOnSync) {
|
|
LOG(DEBUG) << "Testing StopOnSync";
|
|
|
|
// Use a longer test asset to make sure there is a GOP to finish.
|
|
initSampleReader("/data/local/tmp/TranscodingTestAssets/longtest_15s.mp4");
|
|
|
|
EXPECT_EQ(mTranscoder->configure(mMediaSampleReader, mTrackIndex, mDestinationFormat),
|
|
AMEDIA_OK);
|
|
|
|
bool lastSampleWasEos = false;
|
|
bool lastRealSampleWasSync = false;
|
|
OneShotSemaphore samplesReceivedSemaphore;
|
|
uint32_t sampleCount = 0;
|
|
|
|
mTranscoder->setSampleConsumer([&](const std::shared_ptr<MediaSample>& sample) {
|
|
ASSERT_NE(sample, nullptr);
|
|
|
|
if ((lastSampleWasEos = sample->info.flags & SAMPLE_FLAG_END_OF_STREAM)) {
|
|
samplesReceivedSemaphore.signal();
|
|
return;
|
|
}
|
|
lastRealSampleWasSync = sample->info.flags & SAMPLE_FLAG_SYNC_SAMPLE;
|
|
|
|
if (++sampleCount >= 10) { // Wait for a few samples before stopping.
|
|
samplesReceivedSemaphore.signal();
|
|
}
|
|
});
|
|
|
|
ASSERT_TRUE(mTranscoder->start());
|
|
samplesReceivedSemaphore.wait();
|
|
mTranscoder->stop(true /* stopOnSync */);
|
|
EXPECT_EQ(mCallback->waitUntilFinished(), AMEDIA_OK);
|
|
|
|
EXPECT_TRUE(lastSampleWasEos);
|
|
EXPECT_TRUE(lastRealSampleWasSync);
|
|
EXPECT_TRUE(mCallback->transcodingWasStopped());
|
|
}
|
|
|
|
}; // namespace android
|
|
|
|
using namespace android;
|
|
|
|
INSTANTIATE_TEST_SUITE_P(MediaTrackTranscoderTestsAll, MediaTrackTranscoderTests,
|
|
::testing::Values(VIDEO, PASSTHROUGH));
|
|
|
|
int main(int argc, char** argv) {
|
|
::testing::InitGoogleTest(&argc, argv);
|
|
return RUN_ALL_TESTS();
|
|
}
|