/*
* Copyright (C) 2018 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.
*/
//#define LOG_NDEBUG 0
#define LOG_TAG "codec2_hidl_hal_component_test"
#include <android-base/logging.h>
#include <gtest/gtest.h>
#include <C2Config.h>
#include <codec2/hidl/client.h>
#include <VtsHalHidlTargetTestBase.h>
#include "media_c2_hidl_test_common.h"
static ComponentTestEnvironment* gEnv = nullptr;
namespace {
// google.codec2 Component test setup
class Codec2ComponentHidlTest : public ::testing::VtsHalHidlTargetTestBase {
private:
typedef ::testing::VtsHalHidlTargetTestBase Super;
public:
virtual void SetUp() override {
Super::SetUp();
mEos = false;
mClient = android::Codec2Client::CreateFromService(
gEnv->getInstance().c_str());
ASSERT_NE(mClient, nullptr);
mListener.reset(new CodecListener(
[this](std::list<std::unique_ptr<C2Work>>& workItems) {
handleWorkDone(workItems);
}));
ASSERT_NE(mListener, nullptr);
mClient->createComponent(gEnv->getComponent().c_str(), mListener,
&mComponent);
ASSERT_NE(mComponent, nullptr);
for (int i = 0; i < MAX_INPUT_BUFFERS; ++i) {
mWorkQueue.emplace_back(new C2Work);
}
}
virtual void TearDown() override {
if (mComponent != nullptr) {
// If you have encountered a fatal failure, it is possible that
// freeNode() will not go through. Instead of hanging the app.
// let it pass through and report errors
if (::testing::Test::HasFatalFailure()) return;
mComponent->release();
mComponent = nullptr;
}
Super::TearDown();
}
// callback function to process onWorkDone received by Listener
void handleWorkDone(std::list<std::unique_ptr<C2Work>>& workItems) {
for (std::unique_ptr<C2Work>& work : workItems) {
if (!work->worklets.empty()) {
bool mCsd = false;
uint32_t mFramesReceived = 0;
std::list<uint64_t> mFlushedIndices;
workDone(mComponent, work, mFlushedIndices, mQueueLock, mQueueCondition,
mWorkQueue, mEos, mCsd, mFramesReceived);
}
}
}
bool mEos;
std::mutex mQueueLock;
std::condition_variable mQueueCondition;
std::list<std::unique_ptr<C2Work>> mWorkQueue;
std::shared_ptr<android::Codec2Client> mClient;
std::shared_ptr<android::Codec2Client::Listener> mListener;
std::shared_ptr<android::Codec2Client::Component> mComponent;
protected:
static void description(const std::string& description) {
RecordProperty("description", description);
}
};
// Test Empty Flush
TEST_F(Codec2ComponentHidlTest, EmptyFlush) {
ALOGV("Empty Flush Test");
c2_status_t err = mComponent->start();
ASSERT_EQ(err, C2_OK);
std::list<std::unique_ptr<C2Work>> flushedWork;
err = mComponent->flush(C2Component::FLUSH_COMPONENT, &flushedWork);
ASSERT_EQ(err, C2_OK);
err = mComponent->stop();
ASSERT_EQ(err, C2_OK);
// Empty Flush should not return any work
ASSERT_EQ(flushedWork.size(), 0u);
}
// Test Queue Empty Work
TEST_F(Codec2ComponentHidlTest, QueueEmptyWork) {
ALOGV("Queue Empty Work Test");
c2_status_t err = mComponent->start();
ASSERT_EQ(err, C2_OK);
// Queueing an empty WorkBundle
std::list<std::unique_ptr<C2Work>> workList;
err = mComponent->queue(&workList);
ASSERT_EQ(err, C2_OK);
err = mComponent->reset();
ASSERT_EQ(err, C2_OK);
}
// Test Component Configuration
TEST_F(Codec2ComponentHidlTest, Config) {
ALOGV("Configuration Test");
C2String name = mComponent->getName();
EXPECT_NE(name.empty(), true) << "Invalid Component Name";
c2_status_t err = C2_OK;
std::vector<std::unique_ptr<C2Param>> queried;
std::vector<std::unique_ptr<C2SettingResult>> failures;
// Query supported params by the component
std::vector<std::shared_ptr<C2ParamDescriptor>> params;
err = mComponent->querySupportedParams(¶ms);
ASSERT_EQ(err, C2_OK);
ALOGV("Number of total params - %zu", params.size());
// Query and config all the supported params
for (std::shared_ptr<C2ParamDescriptor> p : params) {
ALOGD("Querying index %d", (int)p->index());
err = mComponent->query({}, {p->index()}, C2_DONT_BLOCK, &queried);
EXPECT_NE(queried.size(), 0u);
EXPECT_EQ(err, C2_OK);
err = mComponent->config({queried[0].get()}, C2_DONT_BLOCK, &failures);
ASSERT_EQ(err, C2_OK);
ASSERT_EQ(failures.size(), 0u);
}
}
// Test Multiple Start Stop Reset Test
TEST_F(Codec2ComponentHidlTest, MultipleStartStopReset) {
ALOGV("Multiple Start Stop and Reset Test");
c2_status_t err = C2_OK;
for (size_t i = 0; i < MAX_RETRY; i++) {
err = mComponent->start();
ASSERT_EQ(err, C2_OK);
err = mComponent->stop();
ASSERT_EQ(err, C2_OK);
}
err = mComponent->start();
ASSERT_EQ(err, C2_OK);
for (size_t i = 0; i < MAX_RETRY; i++) {
err = mComponent->reset();
ASSERT_EQ(err, C2_OK);
}
err = mComponent->start();
ASSERT_EQ(err, C2_OK);
err = mComponent->stop();
ASSERT_EQ(err, C2_OK);
// Second stop should return error
err = mComponent->stop();
ASSERT_NE(err, C2_OK);
}
// Test Component Release API
TEST_F(Codec2ComponentHidlTest, MultipleRelease) {
ALOGV("Multiple Release Test");
c2_status_t err = mComponent->start();
ASSERT_EQ(err, C2_OK);
// Query Component Domain Type
std::vector<std::unique_ptr<C2Param>> queried;
err = mComponent->query({}, {C2PortMediaTypeSetting::input::PARAM_TYPE},
C2_DONT_BLOCK, &queried);
EXPECT_NE(queried.size(), 0u);
// Configure Component Domain
std::vector<std::unique_ptr<C2SettingResult>> failures;
C2PortMediaTypeSetting::input* portMediaType =
C2PortMediaTypeSetting::input::From(queried[0].get());
err = mComponent->config({portMediaType}, C2_DONT_BLOCK, &failures);
ASSERT_EQ(err, C2_OK);
ASSERT_EQ(failures.size(), 0u);
for (size_t i = 0; i < MAX_RETRY; i++) {
err = mComponent->release();
ASSERT_EQ(err, C2_OK);
}
}
class Codec2ComponentInputTests : public Codec2ComponentHidlTest,
public ::testing::WithParamInterface<std::pair<uint32_t, bool> > {
};
TEST_P(Codec2ComponentInputTests, InputBufferTest) {
description("Tests for different inputs");
uint32_t flags = GetParam().first;
bool isNullBuffer = GetParam().second;
if (isNullBuffer) ALOGD("Testing for null input buffer with flag : %u", flags);
else ALOGD("Testing for empty input buffer with flag : %u", flags);
mEos = false;
ASSERT_EQ(mComponent->start(), C2_OK);
ASSERT_NO_FATAL_FAILURE(testInputBuffer(
mComponent, mQueueLock, mWorkQueue, flags, isNullBuffer));
ALOGD("Waiting for input consumption");
ASSERT_NO_FATAL_FAILURE(
waitOnInputConsumption(mQueueLock, mQueueCondition, mWorkQueue));
if (flags == C2FrameData::FLAG_END_OF_STREAM) ASSERT_EQ(mEos, true);
ASSERT_EQ(mComponent->stop(), C2_OK);
ASSERT_EQ(mComponent->reset(), C2_OK);
}
INSTANTIATE_TEST_CASE_P(NonStdInputs, Codec2ComponentInputTests, ::testing::Values(
std::make_pair(0, true),
std::make_pair(C2FrameData::FLAG_END_OF_STREAM, true),
std::make_pair(0, false),
std::make_pair(C2FrameData::FLAG_CODEC_CONFIG, false),
std::make_pair(C2FrameData::FLAG_END_OF_STREAM, false)));
} // anonymous namespace
// TODO: Add test for Invalid work,
// TODO: Add test for Invalid states
int main(int argc, char** argv) {
gEnv = new ComponentTestEnvironment();
::testing::AddGlobalTestEnvironment(gEnv);
::testing::InitGoogleTest(&argc, argv);
gEnv->init(&argc, argv);
int status = gEnv->initFromOptions(argc, argv);
if (status == 0) {
status = RUN_ALL_TESTS();
LOG(INFO) << "C2 Test result = " << status;
}
return status;
}