blob: afa27f56bc82610faa133f822ad0286104b92833 [file] [log] [blame]
/*
* Copyright 2017 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.
*/
#ifndef ANDROID_C2_SOFT_AVC_DEC_H_
#define ANDROID_C2_SOFT_AVC_DEC_H_
#include <sys/time.h>
#include <media/stagefright/foundation/ColorUtils.h>
#include "MediaH264Decoder.h"
#include "GoldfishH264Helper.h"
#include <SimpleC2Component.h>
#include <atomic>
#include <map>
namespace android {
#define ALIGN2(x) ((((x) + 1) >> 1) << 1)
#define ALIGN8(x) ((((x) + 7) >> 3) << 3)
#define ALIGN16(x) ((((x) + 15) >> 4) << 4)
#define ALIGN32(x) ((((x) + 31) >> 5) << 5)
#define MAX_NUM_CORES 4
#define MIN(a, b) (((a) < (b)) ? (a) : (b))
#define GETTIME(a, b) gettimeofday(a, b);
#define TIME_DIFF(start, end, diff) \
diff = (((end).tv_sec - (start).tv_sec) * 1000000) + \
((end).tv_usec - (start).tv_usec);
class C2GoldfishAvcDec : public SimpleC2Component {
public:
class IntfImpl;
C2GoldfishAvcDec(const char *name, c2_node_id_t id,
const std::shared_ptr<IntfImpl> &intfImpl);
virtual ~C2GoldfishAvcDec();
// From SimpleC2Component
c2_status_t onInit() override;
c2_status_t onStop() override;
void onReset() override;
void onRelease() override;
c2_status_t onFlush_sm() override;
void process(const std::unique_ptr<C2Work> &work,
const std::shared_ptr<C2BlockPool> &pool) override;
c2_status_t drain(uint32_t drainMode,
const std::shared_ptr<C2BlockPool> &pool) override;
private:
std::unique_ptr<MediaH264Decoder> mContext;
bool mEnableAndroidNativeBuffers{true};
void checkMode(const std::shared_ptr<C2BlockPool> &pool);
// status_t createDecoder();
status_t createDecoder();
status_t setParams(size_t stride);
status_t initDecoder();
bool setDecodeArgs(C2ReadView *inBuffer, C2GraphicView *outBuffer,
size_t inOffset, size_t inSize, uint32_t tsMarker);
c2_status_t ensureDecoderState(const std::shared_ptr<C2BlockPool> &pool);
void finishWork(uint64_t index, const std::unique_ptr<C2Work> &work);
status_t setFlushMode();
c2_status_t drainInternal(uint32_t drainMode,
const std::shared_ptr<C2BlockPool> &pool,
const std::unique_ptr<C2Work> &work);
status_t resetDecoder();
void resetPlugin();
void deleteContext();
std::shared_ptr<IntfImpl> mIntf;
void removePts(uint64_t pts);
void insertPts(uint32_t work_index, uint64_t pts);
uint64_t getWorkIndex(uint64_t pts);
// there are same pts matching to different work indices
// this happen during csd0/csd1 switching
std::map<uint64_t, uint64_t> mOldPts2Index;
std::map<uint64_t, uint64_t> mPts2Index;
std::map<uint64_t, uint64_t> mIndex2Pts;
uint64_t mPts {0};
// TODO:This is not the right place for this enum. These should
// be part of c2-vndk so that they can be accessed by all video plugins
// until then, make them feel at home
enum {
kNotSupported,
kPreferBitstream,
kPreferContainer,
};
std::shared_ptr<C2GraphicBlock> mOutBlock;
uint8_t *mOutBufferFlush;
int mHostColorBufferId{-1};
void getVuiParams(h264_image_t &img);
void copyImageData(h264_image_t &img);
h264_image_t mImg{};
uint32_t mConsumedBytes{0};
uint8_t *mInPBuffer{nullptr};
uint32_t mInPBufferSize;
uint32_t mInTsMarker;
// size_t mNumCores;
// uint32_t mOutputDelay;
uint32_t mWidth;
uint32_t mHeight;
uint32_t mStride;
bool mSignalledOutputEos;
bool mSignalledError;
bool mHeaderDecoded;
std::atomic_uint64_t mOutIndex;
// Color aspects. These are ISO values and are meant to detect changes in
// aspects to avoid converting them to C2 values for each frame
struct VuiColorAspects {
uint8_t primaries;
uint8_t transfer;
uint8_t coeffs;
uint8_t fullRange;
// default color aspects
VuiColorAspects()
: primaries(2), transfer(2), coeffs(2), fullRange(0) {}
bool operator==(const VuiColorAspects &o) {
return primaries == o.primaries && transfer == o.transfer &&
coeffs == o.coeffs && fullRange == o.fullRange;
}
} mBitstreamColorAspects;
// profile
struct timeval mTimeStart;
struct timeval mTimeEnd;
#ifdef FILE_DUMP_ENABLE
char mInFile[200];
#endif /* FILE_DUMP_ENABLE */
std::vector<uint8_t> mCsd0;
std::vector<uint8_t> mCsd1;
void decodeHeaderAfterFlush();
std::unique_ptr<GoldfishH264Helper> mH264Helper;
C2_DO_NOT_COPY(C2GoldfishAvcDec);
};
} // namespace android
#endif // ANDROID_C2_SOFT_AVC_DEC_H_