2014-01-14 15:48:20 +08:00
|
|
|
#include <fstream>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "codec_def.h"
|
|
|
|
#include "codec_app_def.h"
|
|
|
|
#include "utils/BufferedData.h"
|
|
|
|
#include "BaseDecoderTest.h"
|
|
|
|
|
|
|
|
static void ReadFrame(std::ifstream* file, BufferedData* buf) {
|
|
|
|
// start code of a frame is {0, 0, 0, 1}
|
|
|
|
int zeroCount = 0;
|
|
|
|
char b;
|
|
|
|
|
|
|
|
buf->Clear();
|
|
|
|
for (;;) {
|
|
|
|
file->read(&b, 1);
|
|
|
|
if (file->gcount() != 1) { // end of file
|
|
|
|
return;
|
|
|
|
}
|
2014-02-05 18:04:32 +08:00
|
|
|
if (!buf->PushBack(b)) {
|
2014-01-14 15:48:20 +08:00
|
|
|
FAIL() << "unable to allocate memory";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buf->Length() <= 4) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (zeroCount < 3) {
|
|
|
|
zeroCount = b != 0 ? 0 : zeroCount + 1;
|
|
|
|
} else {
|
|
|
|
if (b == 1) {
|
|
|
|
if (file->seekg(-4, file->cur).good()) {
|
|
|
|
buf->SetLength(buf->Length() - 4);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
FAIL() << "unable to seek file";
|
|
|
|
}
|
|
|
|
} else if (b == 0) {
|
|
|
|
zeroCount = 3;
|
|
|
|
} else {
|
|
|
|
zeroCount = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-18 19:31:54 +08:00
|
|
|
BaseDecoderTest::BaseDecoderTest()
|
|
|
|
: decoder_(NULL), decodeStatus_(OpenFile) {}
|
2014-01-14 15:48:20 +08:00
|
|
|
|
|
|
|
void BaseDecoderTest::SetUp() {
|
2014-03-25 17:36:47 +01:00
|
|
|
long rv = WelsCreateDecoder(&decoder_);
|
2014-01-14 15:48:20 +08:00
|
|
|
ASSERT_EQ(0, rv);
|
|
|
|
ASSERT_TRUE(decoder_ != NULL);
|
|
|
|
|
|
|
|
SDecodingParam decParam;
|
|
|
|
memset(&decParam, 0, sizeof(SDecodingParam));
|
|
|
|
decParam.iOutputColorFormat = videoFormatI420;
|
|
|
|
decParam.uiTargetDqLayer = UCHAR_MAX;
|
|
|
|
decParam.uiEcActiveFlag = 1;
|
|
|
|
decParam.sVideoProperty.eVideoBsType = VIDEO_BITSTREAM_DEFAULT;
|
|
|
|
|
2014-02-18 12:32:57 +02:00
|
|
|
rv = decoder_->Initialize(&decParam);
|
2014-01-14 15:48:20 +08:00
|
|
|
ASSERT_EQ(0, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BaseDecoderTest::TearDown() {
|
|
|
|
if (decoder_ != NULL) {
|
|
|
|
decoder_->Uninitialize();
|
2014-03-25 17:36:47 +01:00
|
|
|
WelsDestroyDecoder(decoder_);
|
2014-01-14 15:48:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BaseDecoderTest::DecodeFrame(const uint8_t* src, int sliceSize, Callback* cbk) {
|
|
|
|
void* data[3];
|
|
|
|
SBufferInfo bufInfo;
|
|
|
|
memset(data, 0, sizeof(data));
|
|
|
|
memset(&bufInfo, 0, sizeof(SBufferInfo));
|
|
|
|
|
2014-02-05 18:04:32 +08:00
|
|
|
DECODING_STATE rv = decoder_->DecodeFrame2(src, sliceSize, data, &bufInfo);
|
2014-01-14 15:48:20 +08:00
|
|
|
ASSERT_TRUE(rv == dsErrorFree);
|
|
|
|
|
|
|
|
if (bufInfo.iBufferStatus == 1 && cbk != NULL) {
|
|
|
|
const Frame frame = {
|
|
|
|
{ // y plane
|
|
|
|
static_cast<uint8_t*>(data[0]),
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iWidth,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iHeight,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iStride[0]
|
|
|
|
},
|
|
|
|
{ // u plane
|
|
|
|
static_cast<uint8_t*>(data[1]),
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iWidth / 2,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iHeight / 2,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iStride[1]
|
|
|
|
},
|
|
|
|
{ // v plane
|
|
|
|
static_cast<uint8_t*>(data[2]),
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iWidth / 2,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iHeight / 2,
|
|
|
|
bufInfo.UsrData.sSystemBuffer.iStride[1]
|
|
|
|
},
|
|
|
|
};
|
|
|
|
cbk->onDecodeFrame(frame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void BaseDecoderTest::DecodeFile(const char* fileName, Callback* cbk) {
|
|
|
|
std::ifstream file(fileName, std::ios::in | std::ios::binary);
|
|
|
|
ASSERT_TRUE(file.is_open());
|
|
|
|
|
2014-02-05 18:04:32 +08:00
|
|
|
BufferedData buf;
|
|
|
|
while (true) {
|
2014-01-14 15:48:20 +08:00
|
|
|
ReadFrame(&file, &buf);
|
|
|
|
if (::testing::Test::HasFatalFailure()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (buf.Length() == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
DecodeFrame(buf.data(), buf.Length(), cbk);
|
|
|
|
if (::testing::Test::HasFatalFailure()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t iEndOfStreamFlag = 1;
|
|
|
|
decoder_->SetOption(DECODER_OPTION_END_OF_STREAM, &iEndOfStreamFlag);
|
|
|
|
|
|
|
|
// Get pending last frame
|
|
|
|
DecodeFrame(NULL, 0, cbk);
|
|
|
|
}
|
2014-01-18 19:31:54 +08:00
|
|
|
|
|
|
|
bool BaseDecoderTest::Open(const char* fileName) {
|
|
|
|
if (decodeStatus_ == OpenFile) {
|
2014-02-06 10:50:31 +02:00
|
|
|
file_.open(fileName, std::ios_base::out | std::ios_base::binary);
|
2014-01-18 19:31:54 +08:00
|
|
|
if (file_.is_open()) {
|
|
|
|
decodeStatus_ = Decoding;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BaseDecoderTest::DecodeNextFrame(Callback* cbk) {
|
|
|
|
switch (decodeStatus_) {
|
|
|
|
case Decoding:
|
|
|
|
ReadFrame(&file_, &buf_);
|
|
|
|
if (::testing::Test::HasFatalFailure()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (buf_.Length() == 0) {
|
|
|
|
decodeStatus_ = EndOfStream;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
DecodeFrame(buf_.data(), buf_.Length(), cbk);
|
|
|
|
if (::testing::Test::HasFatalFailure()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
case EndOfStream: {
|
|
|
|
int32_t iEndOfStreamFlag = 1;
|
|
|
|
decoder_->SetOption(DECODER_OPTION_END_OF_STREAM, &iEndOfStreamFlag);
|
|
|
|
DecodeFrame(NULL, 0, cbk);
|
|
|
|
decodeStatus_ = End;
|
|
|
|
break;
|
|
|
|
}
|
2014-02-06 11:37:42 +08:00
|
|
|
case OpenFile:
|
|
|
|
case End:
|
|
|
|
break;
|
2014-01-18 19:31:54 +08:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|