diff --git a/Makefile.am b/Makefile.am index d28175a..47590bc 100644 --- a/Makefile.am +++ b/Makefile.am @@ -38,6 +38,34 @@ if ENABLE_TESTS SUBDIRS += test endif +#libmebo and libaom_av1_rc support +if ENABLE_MEBO +MEBO_SUPPORT = yes +endif + +if ENABLE_AOM_AV1_RC +AOM_AV1_RC_SUPPORT = yes +endif + +if ENABLE_BRC +BRC_SUPPORT = yes +endif + +if ENABLE_BRC_CONTROLLER +BRC_CONTROLLER_SUPPORT = yes +endif + +export DL_LIBS + +MEBO_LOCAL_PATH = $(abs_top_srcdir)/encode/libmebo/libs +AOM_AV1_RC_LOCAL_PATH = $(abs_top_srcdir)/encode/aom_libs/ +SYSTEM_LIBS_DIR = /usr/local/lib/x86_64-linux-gnu/ + +export MEBO_LOCAL_PATH +export AOM_AV1_RC_LOCAL_PATH +export SYSTEM_LIBS_DIR + + # Extra clean files so that maintainer-clean removes *everything* MAINTAINERCLEANFILES = \ aclocal.m4 compile config.guess config.sub \ diff --git a/configure.ac b/configure.ac index c2c1be9..f174991 100644 --- a/configure.ac +++ b/configure.ac @@ -93,6 +93,156 @@ AC_ARG_ENABLE([tests], [build unit tests @<:@default=no@:>@])], [], [enable_tests="no"]) +AC_ARG_ENABLE(mebo, + [AC_HELP_STRING([--enable-mebo], + [Enable LibMebo Support @<:@default=yes@:>@])], + [], [enable_mebo="yes"]) + +AC_ARG_ENABLE(aom_av1_rc, + [AC_HELP_STRING([--enable-aom_av1_rc], + [Enable AOM AV1 RC Support @<:@default=yes@:>@])], + [], [enable_aom_av1_rc="yes"]) + +AC_ARG_ENABLE(brc, + [AC_HELP_STRING([--enable-brc], + [Enable Bitrate RC for VP9,VP8 Support + @<:@default=yes@:>@])], + [], [enable_brc="yes"]) + +AC_ARG_ENABLE(brc_controller, + [AC_HELP_STRING([--enable-brc_controller], + [Enable Bitrate RC_controller + to support libmebo @<:@default=yes@:>@])], + [], [enable_brc_controller="yes"]) + +#check for LibMebo library +MEBO_LIBDIR="" +abs_srcdir="$(cd "$srcdir" && pwd)" +if test "x$enable_mebo" != "xno" ; then + #local path + if test -f "$srcdir/encode/libmebo/libs/libmebo.so"; then + MEBO_LIBDIR="${abs_srcdir}/encode/libmebo/libs" + AC_DEFINE([HAVE_MEBO], [1], [Define if MEBO is available]) + AC_MSG_NOTICE([MEBO LIB INSTALLED in local path]) + else + AC_MSG_NOTICE([Mebo LIB NOT INSTALLED in local path]) + #system path + if test -f "/usr/local/lib/x86_64-linux-gnu/libmebo.so" ; then + MEBO_LIBDIR="/usr/local/lib/x86_64-linux-gnu" + AC_DEFINE([HAVE_MEBO], [1], [Define if MEBO is available]) + AC_MSG_NOTICE([MEBO LIB INSTALLED in SYSTEM path]) + else + AC_MSG_NOTICE([Mebo LIB NOT INSTALLED in system path]) + if test "x$enable_mebo"="xyes"; then + AC_MSG_ERROR( + [LibMebo NOT installed in local path nor in system paths]) + fi + fi + fi +fi +AC_SUBST([MEBO_LIBDIR]) + +#CHECK FOR AOM AV1 RC library +AOM_AV1_RC_LIBDIR="" +if test "x$enable_aom_av1_rc" != "xno" ; then + #local path + if test -f "$srcdir/encode/libmebo/libs/aom_libs/libaom_av1_rc.so"; then + AOM_AV1_RC_LIBDIR="${abs_srcdir}/encode/libmebo/libs/aom_libs/" + AC_DEFINE([HAVE_AOM_AV1_RC], [1], [Define if AOM AV1 RC is available]) + AC_MSG_NOTICE([AOM AV1 RC LIB INSTALLED in local path]) + else + AC_MSG_NOTICE([AOM AV1 RC LIB NOT INSTALLED in local path]) + #system path + if test -f "/usr/local/lib/x86_64-linux-gnu/libaom_av1_rc.so" ; then + AOM_AV1_RC_LIBDIR="/usr/local/lib/x86_64-linux-gnu" + AC_DEFINE([HAVE_AOM_AV1_RC], [1], [Define if AOM AV1 RC + is available]) + AC_MSG_NOTICE([AOM AV1 RC LIB INSTALLED in SYSTEM path]) + else + AC_MSG_NOTICE([AOM AV1 RC LIB NOT INSTALLED in system path]) + if test "x$enable_aom_av1_rc"="xyes"; then + AC_MSG_ERROR( + [AOM AV1 RC NOT installed in local path nor in system paths]) + fi + fi + fi +fi +AC_SUBST([AOM_AV1_RC_LIBDIR]) + + +#Check for brc library +BRC_LIBDIR="" +if test "x$enable_brc" != "xno" ; then + #local path + if test -f "$srcdir/encode/libmebo/libs/libbrc.so"; then + BRC_LIBDIR="${abs_srcdir}/encode/libmebo/libs" + AC_DEFINE([HAVE_BRC], [1], [Define if BRC is available]) + AC_MSG_NOTICE([BRC LIB INSTALLED in local path]) + else + AC_MSG_NOTICE([BRC LIB NOT INSTALLED in local path]) + #system path + if test -f "/usr/local/lib/x86_64-linux-gnu/libbrc.so" ; then + BRC_LIBDIR="/usr/local/lib/x86_64-linux-gnu" + AC_DEFINE([HAVE_BRC], [1], [Define if BRC is available]) + AC_MSG_NOTICE([BRC LIB INSTALLED in SYSTEM path]) + else + AC_MSG_NOTICE([BRC LIB NOT INSTALLED in system path]) + if test "x$enable_brc"="xyes"; then + AC_MSG_ERROR( + [BRC NOT installed in local path nor in system paths]) + fi + fi + fi +fi +AC_SUBST([BRC_LIBDIR]) + +#Check for brc_controller library +BRC_CONTROLLER_LIBDIR="" +if test "x$enable_brc_controller" != "xno" ; then + #local path + if test -f "$srcdir/encode/libmebo/libs/libbrc_controller.so"; then + BRC_CONTROLLER_LIBDIR="${abs_srcdir}/encode/libmebo/libs" + AC_DEFINE([HAVE_BRC_CONTROLLER], [1], [Define if BRC_CONTROLLER + is available]) + AC_MSG_NOTICE([BRC_CONTROLLER LIB INSTALLED in local path]) + else + AC_MSG_NOTICE([BRC_CONTROLLER LIB NOT INSTALLED in local path]) + #system path + if test -f "/usr/local/lib/x86_64-linux-gnu/libbrc_controller.so" ; then + BRC_CONTROLLER_LIBDIR="/usr/local/lib/x86_64-linux-gnu" + AC_DEFINE([HAVE_BRC_CONTROLLER], [1], [Define if BRC_CONTROLLER + is available]) + AC_MSG_NOTICE([BRC_CONTROLLER LIB INSTALLED in SYSTEM path]) + else + AC_MSG_NOTICE([BRC LIB NOT INSTALLED in system path]) + if test "x$enable_brc_controller"="xyes"; then + AC_MSG_ERROR( + [BRC_CONTROLLER NOT installed in local path nor + in system paths]) + fi + fi + fi +fi +AC_SUBST([BRC_CONTROLLER_LIBDIR]) + +# configure for ldl lib +AC_CHECK_LIB([dl], [dlopen], + [DL_LIBS= "-ldl", + AC_DEFINE([HAVE_DLOPEN], [1], [Define if dlopen is available])], + [AC_MSG_ERROR([Unable to find library support dlopen, dlsym])]) +AC_SUBST([DL_LIBS]) + +#libs path +AC_MSG_NOTICE([LIBRARY PATHS ARE set to]) +AC_MSG_NOTICE([MEBO_LIBDIR is set to :${MEBO_LIBDIR}]) +AC_MSG_NOTICE([AOM_AV1_RC_LIBDIR is set to :${AOM_AV1_RC_LIBDIR}]) +AC_MSG_NOTICE([BRC_LIBDIR is set to :${BRC_LIBDIR}]) +AC_MSG_NOTICE([BRC_CONTROLLER is set to :${BRC_CONTROLLER_LIBDIR}]) + +AM_CONDITIONAL([ENABLE_MEBO], [test "x$MEBO_LIBDIR"!="x"]) +AM_CONDITIONAL([ENABLE_AOM_AV1_RC], [test "x$AOM_AV1_RC_LIBDIR"!="x"]) +AM_CONDITIONAL([ENABLE_BRC], [test "x$BRC_LIBDIR"!="x"]) +AM_CONDITIONAL([ENABLE_BRC_CONTROLLER], [test "x$BRC_CONTROLLER_LIBDIR"!="x"]) LT_INIT AC_DISABLE_STATIC diff --git a/encode/Makefile.am b/encode/Makefile.am index 8f2fc09..ba8e588 100644 --- a/encode/Makefile.am +++ b/encode/Makefile.am @@ -89,12 +89,22 @@ vp8enc_LDADD = \ $(top_builddir)/common/libva-display.la \ -lpthread -lm +AUTOMAKE_OPTIONS =subdir-objects + av1encode_SOURCES = av1encode.c +av1encode_SOURCES += libmebo/libmebo_interface.cc av1encode_CFLAGS = -I$(top_srcdir)/common -g av1encode_LDADD = \ $(LIBVA_LIBS) \ $(top_builddir)/common/libva-display.la \ - -lpthread -lm + -lpthread -lm \ + -L$(MEBO_LIBDIR)-lmebo \ + -L$(BRC_LIBDIR)-lbrc \ + -L$(BRC_CONTROLLER_LIBDIR)-lbrc_controller \ + -L$(AOM_AV1_RC_LIBDIR)-laom_av1_rc \ + -ldl + + valgrind: $(bin_PROGRAMS) $(noinst_PROGRAMS) for a in $(bin_PROGRAMS) $(noinst_PROGRAMS); do \ diff --git a/encode/av1encode.c b/encode/av1encode.c index 04b4a58..beed211 100644 --- a/encode/av1encode.c +++ b/encode/av1encode.c @@ -17,6 +17,8 @@ #include #include #include "va_display.h" +#include "av1encode.h" +#include "libmebo/libmebo_interface.h" #define ALIGN16(x) ((x+15)&~15) #define CHECK_VASTATUS(va_status,func) \ @@ -441,43 +443,7 @@ struct BitOffsets uint32_t CDEFParamsSizeInBits; }; -struct Av1InputParameters -{ - char* srcyuv; - char* recyuv; - char* output; - uint32_t profile; - - uint32_t order_hint_bits; - uint32_t enable_cdef; - uint32_t width; - uint32_t height; - uint32_t LDB; - uint32_t frame_rate_extN; - uint32_t frame_rate_extD; - uint32_t level; - - // for brc - uint32_t bit_rate; - uint8_t MinBaseQIndex; - uint8_t MaxBaseQIndex; - - uint32_t intra_period; - uint32_t ip_period; - uint32_t RateControlMethod; - uint32_t BRefType; - int encode_syncmode; - int calc_psnr; - int frame_count; - int frame_width_aligned; - int frame_height_aligned; - uint32_t base_qindex; - int bit_depth; - int target_bitrate; - int vbr_max_bitrate; - int buffer_size; - int initial_buffer_fullness; -}; + static VADisplay va_dpy; @@ -498,7 +464,7 @@ static VAEncPictureParameterBufferAV1 pic_param; static VAEncTileGroupBufferAV1 tile_group_param; // sh fh ips -static struct Av1InputParameters ips; +struct Av1InputParameters ips; static FH fh; static SH sh; struct BitOffsets offsets; @@ -629,11 +595,12 @@ static void print_help() printf(" --base_q_idx 1-255\n"); printf(" --normal_mode select VAEntrypointEncSlice as entrypoint\n"); printf(" --low_power_mode select VAEntrypointEncSliceLP as entrypoint\n"); + printf(" --enswbrc 1 or 0 , 1 enable software brc, 0 use hw brc\n"); printf(" sample usage:\n"); - printf("./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode CQP --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --base_q_idx 128 -o ./out.av1 --LDB --low_power_mode\n" - "./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode CBR --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --target_bitrate 3360000 -o ./out.av1 --LDB --low_power_mode\n" - "./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode VBR --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --vbr_max_bitrate 3360000 -o ./out.av1 --LDB --low_power_mode\n"); + printf("./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode CQP --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --base_q_idx 128 -o ./out.av1 --LDB --low_power_mode --enswbrc 1\n" + "./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode CBR --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --target_bitrate 3360000 -o ./out.av1 --LDB --low_power_mode --enswbrc 0\n" + "./av1encode -n 8 -f 30 --intra_period 4 --ip_period 1 --rcmode VBR --srcyuv ./input.yuv --recyuv ./rec.yuv --fourcc IYUV --level 8 --width 1920 --height 1080 --vbr_max_bitrate 3360000 -o ./out.av1 --LDB --low_power_mode --enswbrc 0\n"); } @@ -659,6 +626,7 @@ static void process_cmdline(int argc, char *argv[]) {"low_power_mode", no_argument, NULL, 15}, {"target_bitrate", required_argument, NULL, 16}, {"vbr_max_bitrate", required_argument, NULL, 17}, + {"enswbrc", required_argument, NULL, 18}, {NULL, no_argument, NULL, 0 } }; @@ -732,6 +700,10 @@ static void process_cmdline(int argc, char *argv[]) case 17: ips.vbr_max_bitrate = atoi(optarg); break; + case 18: + ips.enable_swbrc = atoi(optarg); + printf("ips.enable_swbrc = %d\n", ips.enable_swbrc); + break; case 'u': ips.buffer_size = atoi(optarg) * 8000; break; @@ -2311,10 +2283,19 @@ build_pps_buffer(VAEncPictureParameterBufferAV1* pps) // //loop filter // auto& lf = fh.loop_filter_params; - pps->filter_level[0] = (uint8_t)(fh.loop_filter_params.loop_filter_level[0]); - pps->filter_level[1] = (uint8_t)(fh.loop_filter_params.loop_filter_level[1]); - pps->filter_level_u = (uint8_t)(fh.loop_filter_params.loop_filter_level[2]); - pps->filter_level_v = (uint8_t)(fh.loop_filter_params.loop_filter_level[3]); + if(ips.enable_swbrc) { + int lfdata[4] ; + getLoopfilterfromRc(lfdata); + pps->filter_level[0] = lfdata[0] ; + pps->filter_level[1] = lfdata[1] ; + pps->filter_level_u = lfdata[2] ; + pps->filter_level_v = lfdata[3] ; + } else { + pps->filter_level[0] = (uint8_t)(fh.loop_filter_params.loop_filter_level[0]); + pps->filter_level[1] = (uint8_t)(fh.loop_filter_params.loop_filter_level[1]); + pps->filter_level_u = (uint8_t)(fh.loop_filter_params.loop_filter_level[2]); + pps->filter_level_v = (uint8_t)(fh.loop_filter_params.loop_filter_level[3]); + } pps->loop_filter_flags.bits.sharpness_level = fh.loop_filter_params.loop_filter_sharpness; pps->loop_filter_flags.bits.mode_ref_delta_enabled = fh.loop_filter_params.loop_filter_delta_enabled; pps->loop_filter_flags.bits.mode_ref_delta_update = fh.loop_filter_params.loop_filter_delta_update; @@ -2573,6 +2554,9 @@ static int save_codeddata(unsigned long long display_order, unsigned long long e } printf("%08lld", encode_order); printf("(%06d bytes coded)\n", coded_size); + if(ips.enable_swbrc) { + post_encode_update(coded_size); + } fflush(coded_fp); @@ -2772,7 +2756,7 @@ static int encode_frames(void) memset(&seq_param, 0, sizeof(seq_param)); memset(&pic_param, 0, sizeof(pic_param)); memset(&tile_group_param, 0, sizeof(tile_group_param)); - + unsigned int baseqp =45; if (ips.encode_syncmode == 0) pthread_create(&encode_thread, NULL, storage_task_thread, NULL); @@ -2787,6 +2771,11 @@ static int encode_frames(void) } tmp = GetTickCount(); + if(ips.enable_swbrc) { + baseqp = getQPfromRatectrl(current_frame_type); + ips.base_qindex = baseqp; + } + va_status = vaBeginPicture(va_dpy, context_id, src_surface[current_slot]); CHECK_VASTATUS(va_status, "vaBeginPicture"); BeginPictureTicks += GetTickCount() - tmp; @@ -2963,6 +2952,14 @@ int main(int argc, char **argv) init_va(); setup_encode(); + if(ips.enable_swbrc) { + int result = CreateInitLibmebo(ips); + + if(result == 0) + printf("libs are opened correctly\n"); + else + printf("there is some problem in opening libs\n"); + } encode_frames(); diff --git a/encode/av1encode.h b/encode/av1encode.h new file mode 100644 index 0000000..972d53b --- /dev/null +++ b/encode/av1encode.h @@ -0,0 +1,63 @@ +#ifndef AV1_ENCODE_H_ +#define AV1_ENCODE_H_ + +#include +#include +#include +#include +#include +#include +#include +#ifdef __cplusplus +extern "C" { +#endif + +#define SWBRC 1 +struct Av1InputParameters +{ + char* srcyuv; + char* recyuv; + char* output; + uint32_t profile; + char *libpath; + + + uint32_t order_hint_bits; + uint32_t enable_cdef; + uint32_t width; + uint32_t height; + uint32_t LDB; + uint32_t frame_rate_extN; + uint32_t frame_rate_extD; + uint32_t level; + + // for brc + uint32_t bit_rate; + uint8_t MinBaseQIndex; + uint8_t MaxBaseQIndex; + + uint32_t intra_period; + uint32_t ip_period; + uint32_t RateControlMethod; + uint32_t BRefType; + int encode_syncmode; + int calc_psnr; + int frame_count; + int frame_width_aligned; + int frame_height_aligned; + uint32_t base_qindex; + int bit_depth; + int target_bitrate; + int vbr_max_bitrate; + int buffer_size; + int initial_buffer_fullness; + int enable_swbrc; +}; + + + +#ifdef __cplusplus +} +#endif + +#endif//AV1_ENCODE_H_ \ No newline at end of file diff --git a/encode/libmebo/AV1RateControlHandler.hpp b/encode/libmebo/AV1RateControlHandler.hpp new file mode 100644 index 0000000..d97b7f6 --- /dev/null +++ b/encode/libmebo/AV1RateControlHandler.hpp @@ -0,0 +1,73 @@ +#pragma once +#include + +#include "libmebo.hpp" +#include "ratectrl_rtc.h" +#include "LibMeboControlHandler.hpp" + +typedef enum ErrorsLoadingSymbols { + kNoError = 0, + kMainHandleLibError = -1, + kAv1CreateSymbLoadError = -2, + kAV1RateCtrlInitConfigSymbLoadError = -3, + kUpdateRateControlSymbLoadError = -4, + kCompueQPSymbLoadError = -5, + kPostEncodeSymbLoadError = -6, + kGetQpSymbLoadError = -7, + kGetLoopFilterSymbError = -8, +} ErrosLibmeboSymbols; + +class LibmeboBrc_AV1 : public LibMeboBrc { +public: + LibmeboBrc_AV1(LibMeboBrcAlgorithmID algo_id); + virtual ~LibmeboBrc_AV1() override; + LibMeboRateController *init(LibMeboRateController *rc, + LibMeboRateControllerConfig *rcConfig) override; + LibMeboStatus update_config(LibMeboRateController *rc, + LibMeboRateControllerConfig *rcConfig) override; + LibMeboStatus post_encode_update(LibMeboRateController *rc, + uint64_t encodedFrameSize) override; + LibMeboStatus compute_qp(LibMeboRateController *rc, + LibMeboRCFrameParams *rcFrameParams) override; + LibMeboStatus get_qp(LibMeboRateController *rc, int *qp) override; + LibMeboStatus get_loop_filter_level(LibMeboRateController *rc, + int *filterLevel) override; + int InitSymbolsFromLibrary(); + +private: + void *handle; + typedef void (*InitRateControlConfigFunc_AV1_t)( + struct AomAV1RateControlRtcConfig *rcConfig); + + typedef AomAV1RateControlRTC *(*CreateRateControl_AV1_t)( + const struct AomAV1RateControlRtcConfig *rcConfig); + + typedef bool (*UpdateRateControl_AV1_t)( + void *controller, struct AomAV1RateControlRtcConfig *rcConfig); + + typedef AomFrameDropDecision (*ComputeQP_AV1_t)( + void *controller, const AomAV1FrameParamsRTC &frameParams); + + typedef int (*GetQP_AV1_t)(void *controller); + + typedef AomAV1LoopfilterLevel (*GetLoopfilterLevel_AV1_t)(void *controller); + + typedef void (*PostEncodeUpdate_AV1_t)(void *controller, + uint64_t encodedFrameSize); + + typedef bool (*GetSegmentationData_AV1_t)( + void *controller, AomAV1SegmentationData *segmentation_data); + + typedef AomAV1CdefInfo (*GetCdefInfo_AV1_t)(void *controller); + + InitRateControlConfigFunc_AV1_t ptrInitConfigFunc; + CreateRateControl_AV1_t ptrCreateAV1Controller; + UpdateRateControl_AV1_t ptrUpdateRateControl_AV1; + ComputeQP_AV1_t ptrComputeQP_AV1; + GetQP_AV1_t ptrGetQP_AV1; + GetLoopfilterLevel_AV1_t ptrGetLoopfilterLevel_AV1; + PostEncodeUpdate_AV1_t ptrPostEncodeUpdate_AV1; + GetSegmentationData_AV1_t ptrGetSegmentationData_AV1; + GetCdefInfo_AV1_t ptrGetCdefInfo_AV1; + AomAV1RateControlRtcConfig *rcConfig; +}; \ No newline at end of file diff --git a/encode/libmebo/LibMeboControlHandler.hpp b/encode/libmebo/LibMeboControlHandler.hpp new file mode 100644 index 0000000..4f15679 --- /dev/null +++ b/encode/libmebo/LibMeboControlHandler.hpp @@ -0,0 +1,55 @@ + +#pragma once +extern "C" { +#include "libmebo.hpp" +} + +#define MaxSpatialLayers 3 +#define MaxTemporalLayers 3 + +struct EncParamsLibmebo { + unsigned int preset; + unsigned int id; + unsigned int bitrate; // in kbps + unsigned int framerate; + unsigned int width; + unsigned int height; + unsigned int framecount; // Number of Frames to be encoded + unsigned int num_sl; // Number of Spatial Layers + unsigned int num_tl; // Number of Temporal Layers + unsigned int dynamicRateChange; // dynamic rate change enablement flag + int64_t bufOptimalSz; +}; + +class LibMeboBrc { +public: + LibMeboBrc(LibMeboCodecType codecType, LibMeboBrcAlgorithmID algo_id); + virtual ~LibMeboBrc() = default; + + virtual LibMeboRateController * + init(LibMeboRateController *rc, LibMeboRateControllerConfig *rc_config) = 0; + virtual LibMeboStatus update_config(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_cfg) = 0; + virtual LibMeboStatus post_encode_update(LibMeboRateController *rc, + uint64_t encoded_frame_size) = 0; + virtual LibMeboStatus compute_qp(LibMeboRateController *rc, + LibMeboRCFrameParams *rc_frame_params) = 0; + virtual LibMeboStatus get_qp(LibMeboRateController *rc, int *qp) = 0; + virtual LibMeboStatus get_loop_filter_level(LibMeboRateController *rc, + int *filter_level) = 0; + + uint32_t MaxSizeOfKeyframeAsPercentage(uint32_t optimal_buffer_size, + uint32_t max_framerate); + int GetBitratekBps_l(int sl_id, int tl_id); + void InitLayeredFramerate(int num_tl, int framerate, int *tsRateDecimator); + void InitLayeredBitrateAlloc(int num_sl, int num_tl, int bitrate); + + LibMeboCodecType getCodecType() const { return codec_type; } + +protected: + LibMeboCodecType codec_type; + LibMeboBrcAlgorithmID algo_id; + EncParamsLibmebo encParamsLibMebo; + int layeredFrameRate[MaxTemporalLayers]; + int layered_bitrates[MaxSpatialLayers][MaxTemporalLayers]; +}; diff --git a/encode/libmebo/RateControlFactory.hpp b/encode/libmebo/RateControlFactory.hpp new file mode 100644 index 0000000..a8ef852 --- /dev/null +++ b/encode/libmebo/RateControlFactory.hpp @@ -0,0 +1,46 @@ +#pragma once + +#include "AV1RateControlHandler.hpp" +#include "LibMeboControlHandler.hpp" +#include "VP8RateControlHandler.hpp" +#include "VP9RateControlHandler.hpp" +#include +#include + +class Libmebo_brc_factory { +public: + static std::unique_ptr create(unsigned int id) { + LibMeboCodecType codecType; + switch (static_cast(id)) { + case LIBMEBO_CODEC_VP8: + codecType = LIBMEBO_CODEC_VP8; + break; + case LIBMEBO_CODEC_VP9: + codecType = LIBMEBO_CODEC_VP9; + break; + case LIBMEBO_CODEC_AV1: + codecType = LIBMEBO_CODEC_AV1; + break; + case LIBMEBO_CODEC_UNKNOWN: + codecType = LIBMEBO_CODEC_AV1; // defult to av1 -only av1 is implemented + break; + } + + switch (codecType) { + case LIBMEBO_CODEC_VP8: + return std::make_unique( + static_cast( + id)); // this is calling construcotr. + case LIBMEBO_CODEC_VP9: + return std::make_unique( + static_cast(id)); + case LIBMEBO_CODEC_AV1: + return std::make_unique( + static_cast(id)); + case LIBMEBO_CODEC_UNKNOWN: + return std::make_unique( + static_cast(id)); + break; + } + } +}; diff --git a/encode/libmebo/VP8RateControlHandler.hpp b/encode/libmebo/VP8RateControlHandler.hpp new file mode 100644 index 0000000..a7e42e4 --- /dev/null +++ b/encode/libmebo/VP8RateControlHandler.hpp @@ -0,0 +1,26 @@ +#pragma once + +#include "libmebo.hpp" + +#include "LibMeboControlHandler.hpp" + +class LibmeboBrc_VP8 : public LibMeboBrc { +public: + LibmeboBrc_VP8(LibMeboBrcAlgorithmID algo_id); + virtual ~LibmeboBrc_VP8() override = default; + LibMeboRateController *init(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_config) override; + LibMeboStatus update_config(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_cfg) override; + LibMeboStatus post_encode_update(LibMeboRateController *rc, + uint64_t encoded_frame_size) override; + LibMeboStatus compute_qp(LibMeboRateController *rc, + LibMeboRCFrameParams *rc_frame_params) override; + LibMeboStatus get_qp(LibMeboRateController *rc, int *qp) override; + LibMeboStatus get_loop_filter_level(LibMeboRateController *rc, + int *filter_level) override; + +private: + typedef void *BrcCodecEnginePtr; + BrcCodecEnginePtr brc_codec_handler; +}; diff --git a/encode/libmebo/VP9RateControlHandler.hpp b/encode/libmebo/VP9RateControlHandler.hpp new file mode 100644 index 0000000..ff9cb2f --- /dev/null +++ b/encode/libmebo/VP9RateControlHandler.hpp @@ -0,0 +1,26 @@ +#pragma once + +#include "libmebo.hpp" + +#include "LibMeboControlHandler.hpp" + +class LibmeboBrc_VP9 : public LibMeboBrc { +public: + LibmeboBrc_VP9(LibMeboBrcAlgorithmID algo_id); + virtual ~LibmeboBrc_VP9() override = default; + LibMeboRateController *init(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_config) override; + LibMeboStatus update_config(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_cfg) override; + LibMeboStatus post_encode_update(LibMeboRateController *rc, + uint64_t encoded_frame_size) override; + LibMeboStatus compute_qp(LibMeboRateController *rc, + LibMeboRCFrameParams *rc_frame_params) override; + LibMeboStatus get_qp(LibMeboRateController *rc, int *qp) override; + LibMeboStatus get_loop_filter_level(LibMeboRateController *rc, + int *filter_level) override; + +private: + typedef void *BrcCodecEnginePtr; + BrcCodecEnginePtr brc_codec_handler; +}; diff --git a/encode/libmebo/libmebo.hpp b/encode/libmebo/libmebo.hpp new file mode 100644 index 0000000..9fcfdd7 --- /dev/null +++ b/encode/libmebo/libmebo.hpp @@ -0,0 +1,432 @@ +/* + * Copyright (c) 2020 Intel Corporation. All Rights Reserved. + * Copyright (c) 2020 Sreerenj Balachandran + * Author: Sreerenj Balachandran + * + * Use of this source code is governed by a BSD-style license + * that can be found in the LICENSE file in the root of the source + * tree. An additional intellectual property rights grant can be found + * in the file PATENTS. All contributing project authors may + * be found in the AUTHORS file in the root of the source tree. + */ + +/* + * LibMebo API Specification + * + * Acknowledgements: + * Some concepts are borrowed from libvpx: + * Jerome Jiang and Marco Paniconi + * contributed to various aspects of the VP9 BRC solution. + */ + +/** + * \file libmebo.h + * \brief APIs exposed in LibMebo + * + * This file contains the \ref api_core "LibMebo API". + */ + +#ifndef __LIBMEBO_H__ +#define __LIBMEBO_H__ +#include + +/** + * \mainpage Library for Media Encode Bitrate-control-algorithm Orchestration + * (LibMebo) + * + * \section intro Introduction + * LibMebo is an open-source library to orchestrate the bitrate + * control algorithms for a video encoder pipeline.This library + * has a collection of algorithms derived from various video encoder + * libraries. + * + * The intention for Libmebo is to allow the video encoder implementations + * to enable customized bitrate control outside of the core pipeline. + * The encoder itself could be running on software or hardware. + * + * \section api_howto How to use the libmebo apis with an encoder + * + * \code + * #include + * + * LibMeboRateController *rc; + * LibMeboRateControllerConfig rc_config; + * LibMeboRCFrameParams rc_frame_param; + * LibMeboStatus status; + * + * //Initialize the rc_config with required parameters + * __InitializeConfig(rc_cfg); + * + * //Create an instance of libmebo + * rc = libmebo_rate_controller_new (LIBMEBO_CODEC_VP9, + * Libmebo_brc_ALGORITHM_DEFAULT); + * + * //Initialize the libmebo instance with the rc_config + * status = libmebo_rate_controller_init (rc, &rc_config); + * assert (status == LIBMEBO_STATUS_SUCCESS); + * + * //LibMebo in action + * while (frame_to_encode) { + * if (rc_config_changed) { + * status = libmebo_rate_controller_update_config (rc, &rc_config); + * assert (status == LIBMEBO_STATUS_SUCCESS); + * } + * + * //Initialize the per-frame parameters (frame_type & layer ids) + * __InitializeRCFrameParams (rc_frame_param); + * + * //Compute the QP + * status = libmebo_rate_controller_compute_qp (rc, rc_frame_param); + * assert (status == LIBMEBO_STATUS_SUCCESS); + * + * //Retrieve the QP for the next frame to be encoded + * status = libmebo_rate_controller_get_qp (rc, &qp); + * assert (status == LIBMEBO_STATUS_SUCCESS); + * + * // Optional:libmebo can also recommend the loop-filter strength + * status = libmebo_rate_controller_get_loop_filter_level (rc, &lf); + * + * // Ensure the status == LIBMEBO_STATUS_SUCCESS before using + * // the loop filter level since some algos are not supporting this API + * + * //Encoder implementation that running in CQP mode + * __EncodeBitStream(qp, [lf]) + * + * //Update libmebo instance with encoded frame size + * status = libmebo_rate_controller_post_encode_update (rc, EncFrameSize); + * assert (status == LIBMEBO_STATUS_SUCCESS); + * } + * \endcode + */ + +// #include "../src/Handlers/LibMeboControlHandler.hpp" + +#ifdef __cplusplus +extern "C" { +#endif + +//#define LIBMEBO_ENABLE_AV1 1 +#define LIBMEBO_ENABLE_VP9 1 +#define LIBMEBO_ENABLE_VP8 1 + +typedef void *BrcCodecEnginePtr; + +/** + * Codec Types + */ +typedef enum { + LIBMEBO_CODEC_VP8, + LIBMEBO_CODEC_VP9, + LIBMEBO_CODEC_AV1, + LIBMEBO_CODEC_UNKNOWN, +} LibMeboCodecType; + +/** + * Backend algorithm IDs + */ +typedef enum { + LIBMEBO_BRC_ALGORITHM_DEFAULT, + LIBMEBO_BRC_ALGORITHM_DERIVED_LIBVPX_VP8, + LIBMEBO_BRC_ALGORITHM_DERIVED_LIBVPX_VP9, + LIBMEBO_BRC_ALGORITHM_DERIVED_AOM_AV1, + LIBMEBO_BRC_ALGORITHM_UNKNOWN, +} LibMeboBrcAlgorithmID; + +/** + * Return status type of libmebo APIs + */ +typedef enum { + LIBMEBO_STATUS_SUCCESS, + LIBMEBO_STATUS_WARNING, + LIBMEBO_STATUS_ERROR, + LIBMEBO_STATUS_FAILED, + LIBMEBO_STATUS_INVALID_PARAM, + LIBMEBO_STATUS_UNSUPPORTED_CODEC, + LIBMEBO_STATUS_UNIMPLEMENTED, + LIBMEBO_STATUS_UNSUPPORTED_RC_MODE, + LIBMEBO_STATUS_UNKNOWN, +} LibMeboStatus; + +/** + * Rate Control Modes + */ +typedef enum { LIBMEBO_RC_CBR, LIBMEBO_RC_VBR } LibMeboRateControlMode; + +/** + * Frame prediction types + */ +typedef enum { + LIBMEBO_KEY_FRAME = 0, + LIBMEBO_INTER_FRAME = 1, + LIBMEBO_FRAME_TYPES, +} LibMeboFrameType; + +/** + * \biref Frame parameters + * + * This structure conveys frame level parameters and should be sent + * once per frame + */ +typedef struct _LibMeboRCFrameParams { + LibMeboFrameType frame_type; + int spatial_layer_id; + int temporal_layer_id; +} LibMeboRCFrameParams; + +/* Temporal Scalability: Maximum number of coding layers. + * Not all codecs are supporting the LIBMEBO_TS_MAX_LAYERS. The + * libmebo_rate_controller_init() will perform the codec specific + * layer number validation. + */ +#define LIBMEBO_TS_MAX_LAYERS 8 + +/* Spatial Scalability: Maximum number of Spatial coding layers. + * Not all codecs are supporting the LIBMEBO_SS_MAX_LAYERS. The + * libmebo_rate_controller_init() will perform the codec specific + * layer number validation. + */ +#define LIBMEBO_SS_MAX_LAYERS 4 + +/** + * Temporal + Spatial Scalability: Maximum number of coding layers + * Not all codecs are supporting the LIBMEBO_MAX_LAYERS. The + * libmebo_rate_controller_init() will perform the codec specific + * layer number validation. + */ +#define LIBMEBO_MAX_LAYERS 32 + +/** + * \biref LibMebo Rate Controller configuration structure + * + * This structure conveys the encoding parameters required + * for libmebo to configure the BRC instance. + */ +typedef struct _LibMeboRateControllerConfig { + /** \brief pixel width of the bitstream to be encoded */ + int width; + + /** \brief pixel height of the bitstream to be encoded */ + int height; + + /** + * \brief Maximum (Worst Quality) Quantizer: Range: 0 - 63 + * + * The quantizer is the most direct control over the quality of the + * encoded image. The actual range of the valid values for the quantizer + * is codec specific. Consult the documentation for the codec to determine + * the values to use(eg: VP8: 0-127, VP9: 0-255). + * + * LibMebo scales the max_quantizer between the codec specific limits. + */ + int max_quantizer; + + /* + * \brief Minimum (Best Quality) Quantizer: Range: 0 - 63 + * + * The quantizer is the most direct control over the quality of the + * encoded image. The actual ange of valid values for the quantizer + * is codec specific. Consult the documentation for the codec to determine + * the values to use (eg: VP8: 0-127, VP9: 0-255). + * + * LibMebo scales the min_quantizer between the codec specific limits. + */ + int min_quantizer; + + /** \brief target bandwidth (in kilo bits per second) of the bitstream */ + int64_t target_bandwidth; + + /** + * \brief Decoder(HRD) Buffer Initial Size + * + * This value indicates the amount of data that will be buffered by the + * decoding application prior to beginning playback for the encoded stream. + * This value is expressed in units of time (milliseconds). + */ + int64_t buf_initial_sz; + + /** + * \brief Decoder(HRD) Buffer Optimal Size + * + * This value indicates the amount of data that the encoder should try + * to maintain in the decoder's buffer. This value is expressed in units + * of time (milliseconds). + */ + int64_t buf_optimal_sz; + + /* + * \brief Decoder(HRD) Buffer Size + * + * This value indicates the amount of data that may be buffered by the + * decoding application. Note that this value is expressed in units of + * time (milliseconds). For example, a value of 5000 indicates that the + * client will buffer (at least) 5000ms worth of encoded data. + */ + int64_t buf_sz; + + /** + * \brief Rate control adaptation undershoot control + * + * Expressed as a percentage of the target bitrate, a threshold + * undershoot level (current rate vs target) beyond which more aggressive + * corrective measures are taken. + * + * This factor controls the maximum amount of bits that can + * be subtracted from the target bitrate in order to compensate + * for prior overshoot. + * + * Valid values in the range: 0-100 + */ + int undershoot_pct; + + /** + * \brief Rate control adaptation overshoot control + * + * Expressed as a percentage of the target bitrate, a threshold + * overshoot level (current rate vs target) beyond which more aggressive + * corrective measures are taken. + * + * This factor controls the maximum amount of bits that can + * be added to the target bitrate in order to compensate for + * prior undershoot. + * + * Valid values in the range: 0-100 + */ + int overshoot_pct; + + /** + * \brief Codec control attribute to set max data rate for Intra frames. + * + * This value controls additional clamping on the maximum size of a + * keyframe. It is expressed as a percentage of the average + * per-frame bitrate, with the special (and default) value 0 meaning + * unlimited, or no additional clamping beyond the codec's built-in + * algorithm. + * + * For example, to allocate no more than 4.5 frames worth of bitrate + * to a keyframe, set this to 450. + * + * It is not guaranteed that all brc algorithms will support this + * feature. The libmebo_rate_controller_init() is responsible for + * the codec specific parameter validation. + * + */ + int max_intra_bitrate_pct; + + /*\brief Codec control attribute to set max data rate for Inter frames. + * + * This value controls additional clamping on the maximum size of an + * inter frame. It is expressed as a percentage of the average + * per-frame bitrate, with the special (and default) value 0 meaning + * unlimited, or no additional clamping beyond the codec's built-in + * algorithm. + * + * For example, to allow no more than 4.5 frames worth of bitrate + * to an inter frame, set this to 450. + * + * It is not guaranteed that all brc algorithms will support this + * feature. The libmebo_rate_controller_init() is responsible for + * the codec specific parameter validation. + */ + int max_inter_bitrate_pct; + + /** \brief framerate of the stream */ + double framerate; + + /** + * \brief Number of spatial coding layers. + * + * This value specifies the number of spatial coding layers to be used. + */ + int ss_number_layers; + + /* + * \brief Number of temporal coding layers. + * + * This value specifies the number of temporal layers to be used. + */ + int ts_number_layers; + + /** + * \brief Maximum (Worst Quality) Quantizer for each layer + */ + int max_quantizers[LIBMEBO_MAX_LAYERS]; + + /* + * \brief Minimum (Best Quality) Quantizer + */ + int min_quantizers[LIBMEBO_MAX_LAYERS]; + + /* \brief Scaling factor numerator for each spatial layer */ + int scaling_factor_num[LIBMEBO_SS_MAX_LAYERS]; + + /* \brief Scaling factor denominator for each spatial layer */ + int scaling_factor_den[LIBMEBO_SS_MAX_LAYERS]; + + /* + * \brief Target bitrate for each spatial/temporal layer. + * + * These values specify the target coding bitrate to be used for each + * spatial/temporal layer. (in kbps) + * + */ + int layer_target_bitrate[LIBMEBO_MAX_LAYERS]; + + /* + * \brief Frame rate decimation factor for each temporal layer. + * + * These values specify the frame rate decimation factors to apply + * to each temporal layer. + */ + int ts_rate_decimator[LIBMEBO_TS_MAX_LAYERS]; + + /* + * \biref Rate control mode + */ + LibMeboRateControlMode rc_mode; + + /* Reserved bytes for future use, must be zero */ + uint32_t _libmebo_rc_config_reserved[32]; +} LibMeboRateControllerConfig; + +typedef struct _LibMeboRateController { + void *priv; + + /* \brief Indicates the codec in use */ + LibMeboCodecType codec_type; + + /* \brief currently configured rate control parameters */ + LibMeboRateControllerConfig rc_config; + + /* Reserved bytes for future use, must be zero */ + uint32_t _libmebo_reserved[32]; +} LibMeboRateController; + +LibMeboRateController * +libmebo_create_rate_controller(LibMeboCodecType CodecType, + LibMeboBrcAlgorithmID algo_id); +void libmebo_release_rate_controller(LibMeboRateController *rc); + +LibMeboRateController * +libmebo_init_rate_controller(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_config); + +LibMeboStatus +libmebo_update_rate_controller_config(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_cfg); +LibMeboStatus libmebo_post_encode_update(LibMeboRateController *rc, + uint64_t encoded_frame_size); +LibMeboStatus libmebo_compute_qp(LibMeboRateController *rc, + LibMeboRCFrameParams *rc_frame_params); + +LibMeboStatus libmebo_get_qp(LibMeboRateController *rc, int *qp); + +LibMeboStatus libmebo_get_loop_filter_level(LibMeboRateController *rc, + int *filter_level); + +void *create_brc_factory(unsigned int id); +void destroy_brc_factory(void *brc); +#ifdef __cplusplus +} +#endif + +#endif diff --git a/encode/libmebo/libmebo_interface.cc b/encode/libmebo/libmebo_interface.cc new file mode 100644 index 0000000..f477c55 --- /dev/null +++ b/encode/libmebo/libmebo_interface.cc @@ -0,0 +1,203 @@ +#include "libmebo_interface.h" +#include "RateControlFactory.hpp" +#include "av1encode.h" +#include +#include + +struct FunctionPtrstoLibmebo FnPtrsToLibmebo; +LibMeboRateController *libmebo_brc; + +void InitRcCofnig(struct Av1InputParameters ips, + LibMeboRateControllerConfig *rc_config) { + constexpr int kMinQP = 10; + constexpr int kMaxQP = 56; + rc_config->width = ips.width; + rc_config->height = ips.height; + rc_config->max_quantizer = kMaxQP; + rc_config->min_quantizer = kMinQP; + + rc_config->buf_initial_sz = 600; + rc_config->buf_optimal_sz = 500; + rc_config->target_bandwidth = ips.target_bitrate; + rc_config->buf_sz = 1000; + rc_config->undershoot_pct = 25; + rc_config->overshoot_pct = 50; + rc_config->max_intra_bitrate_pct = 300; + rc_config->max_inter_bitrate_pct = 50; + rc_config->framerate = 60; + rc_config->layer_target_bitrate[0] = ips.target_bitrate; + + rc_config->ts_rate_decimator[0] = 1; + rc_config->ss_number_layers = 1; + rc_config->ts_number_layers = 1; + rc_config->max_quantizers[0] = kMaxQP; + rc_config->min_quantizers[0] = kMinQP; + rc_config->scaling_factor_num[0] = 1; + rc_config->scaling_factor_den[0] = 1; +} + +int InitFuncPtrs(struct Av1InputParameters ips, + struct FunctionPtrstoLibmebo *pFnPtrsToLibmebo) { + + char path[] = "/usr/local/lib/libmebo.so"; + pFnPtrsToLibmebo->handle = dlopen(path, RTLD_LAZY); + + if (!pFnPtrsToLibmebo->handle) { + std::cerr << "Cannot open the library given path is :"<ptrCreateLibmeboController = + (createLibmeboRateController_t)dlsym(pFnPtrsToLibmebo->handle, + "libmebo_create_rate_controller"); + if (!pFnPtrsToLibmebo->ptrCreateLibmeboController) { + std::cerr << "Cannot load symbol 'libmebo_create_rate_controller' from " + "Libmebo.so\n"; + return kLibMeboControllerSymbolError; + } + + pFnPtrsToLibmebo->ptrInit_rate_controller = (init_rate_controller_t)dlsym( + pFnPtrsToLibmebo->handle, "libmebo_init_rate_controller"); + if (!pFnPtrsToLibmebo->ptrInit_rate_controller) { + std::cerr << "Cannot load symbol 'libmebo_init_rate_controller' from " + "Libmebo.so\n"; + return kInitRateControllerSymbolError; + } + + pFnPtrsToLibmebo->ptrUpdate_rate_control = (update_rate_control_t)dlsym( + pFnPtrsToLibmebo->handle, "libmebo_update_rate_controller_config"); + if (!pFnPtrsToLibmebo->ptrInit_rate_controller) { + std::cerr << "Cannot load symbol 'libmebo_update_rate_controller_config' " + "from Libmebo.so\n"; + return kUpdateRateControllerSymbolError; + } + + pFnPtrsToLibmebo->ptrPost_encode_update = (post_encode_update_t)dlsym( + pFnPtrsToLibmebo->handle, "libmebo_post_encode_update"); + if (!pFnPtrsToLibmebo->ptrPost_encode_update) { + std::cerr + << "Cannot load symbol 'libmebo_post_encode_update' from Libmebo.so\n"; + return kPostEncodeSymbolSymbolError; + } + + pFnPtrsToLibmebo->ptrCompute_qp = + (compute_qp_t)dlsym(pFnPtrsToLibmebo->handle, "libmebo_compute_qp"); + if (!pFnPtrsToLibmebo->ptrPost_encode_update) { + std::cerr << "Cannot load symbol 'libmebo_compute_qp' from Libmebo.so\n"; + return kComputeQpSybmolError; + } + + pFnPtrsToLibmebo->ptrGet_qp = + (get_qp_t)dlsym(pFnPtrsToLibmebo->handle, "libmebo_get_qp"); + if (!pFnPtrsToLibmebo->ptrGet_qp) { + std::cerr << "Cannot load symbol 'libmebo_get_qp' from Libmebo.so\n"; + return kGetQPSymbolError; + } + + pFnPtrsToLibmebo->ptrGet_loop_filter_level = (get_loop_filter_level_t)dlsym( + pFnPtrsToLibmebo->handle, "libmebo_get_loop_filter_level"); + if (!pFnPtrsToLibmebo->ptrGet_loop_filter_level) { + std::cerr << "Cannot load symbol 'libmebo_get_loop_filter_level' from " + "Libmebo.so\n"; + return kGetLoopFilterSymbolError; + } + return kNoError; +} + +int CreateInitLibmebo(struct Av1InputParameters ips) { + + LibMeboRateControllerConfig rc_config; + LibMeboCodecType CodecType = LIBMEBO_CODEC_AV1; + unsigned int algo_id = 2; + + int result = InitFuncPtrs(ips, &FnPtrsToLibmebo); + + if (result != kNoError) { + std::cerr << " Cannot load lib or symbol, error code is " << result << "-" + << dlerror() << std::endl; + } + + InitRcCofnig(ips, &rc_config); // creating rc config. + + libmebo_brc = reinterpret_cast( + FnPtrsToLibmebo.ptrCreateLibmeboController( + CodecType, static_cast(algo_id))); + + if (nullptr == libmebo_brc) { + std::cerr << "Libmebo_brc factory object creation failed \n"; + } + + std::cout + << "till this time, created the constructor to libmebo_brc and av1\n"; + + FnPtrsToLibmebo.ptrInit_rate_controller(libmebo_brc, &rc_config); + return 0; +} + +int myfunc_test(struct Av1InputParameters ips) { + + LibMeboRateControllerConfig rc_config; + LibMeboCodecType CodecType = LIBMEBO_CODEC_AV1; + unsigned int algo_id = 2; + uint64_t frame_size_bytes = 600; + LibMeboRCFrameParams rc_frame_params; + int qp; + int filter_level[4]; + + rc_frame_params.frame_type = LIBMEBO_KEY_FRAME; + + int result = InitFuncPtrs(ips, &FnPtrsToLibmebo); + + if (result != kNoError) { + std::cerr << " Cannot load lib or symbol, error code is " << result << "-" + << dlerror() << std::endl; + } + + InitRcCofnig(ips, &rc_config); // creating rc config. + + libmebo_brc = reinterpret_cast( + FnPtrsToLibmebo.ptrCreateLibmeboController( + CodecType, static_cast(algo_id))); + + if (nullptr == libmebo_brc) { + std::cerr << "Libmebo_brc factory object creation failed \n"; + } + + std::cout + << "till this time, created the construcotr for libmebo_brc and forav1\n"; + + FnPtrsToLibmebo.ptrInit_rate_controller(libmebo_brc, &rc_config); + FnPtrsToLibmebo.ptrUpdate_rate_control(libmebo_brc, &rc_config); + FnPtrsToLibmebo.ptrPost_encode_update(libmebo_brc, frame_size_bytes); + FnPtrsToLibmebo.ptrCompute_qp(libmebo_brc, &rc_frame_params); + FnPtrsToLibmebo.ptrGet_qp(libmebo_brc, &qp); + FnPtrsToLibmebo.ptrGet_loop_filter_level(libmebo_brc, filter_level); + return 0; +} + +int getQPfromRatectrl(int current_frame_type) { + LibMeboRCFrameParams rc_frame_params; + int qp; + + const bool is_keyframe = current_frame_type; + + rc_frame_params.frame_type = LIBMEBO_KEY_FRAME; // default. + if (rc_frame_params.frame_type == is_keyframe) { + rc_frame_params.frame_type = LIBMEBO_KEY_FRAME; + } else { + rc_frame_params.frame_type = LIBMEBO_INTER_FRAME; + } + rc_frame_params.spatial_layer_id = 0; + rc_frame_params.temporal_layer_id = 0; + FnPtrsToLibmebo.ptrCompute_qp(libmebo_brc, &rc_frame_params); + FnPtrsToLibmebo.ptrGet_qp(libmebo_brc, &qp); + return (qp); +} + +void getLoopfilterfromRc(int lfdata[4]) { + FnPtrsToLibmebo.ptrGet_loop_filter_level(libmebo_brc, lfdata); +} + +void post_encode_update(unsigned int consumed_bytes) { + FnPtrsToLibmebo.ptrPost_encode_update(libmebo_brc, consumed_bytes); +} \ No newline at end of file diff --git a/encode/libmebo/libmebo_interface.h b/encode/libmebo/libmebo_interface.h new file mode 100644 index 0000000..c531e9a --- /dev/null +++ b/encode/libmebo/libmebo_interface.h @@ -0,0 +1,53 @@ +#pragma once +#ifdef __cplusplus +extern "C" { +#endif +#include "av1encode.h" +#include "libmebo.hpp" +int CreateInitLibmebo(struct Av1InputParameters ips); +void getLoopfilterfromRc(int lfdata[4]); +void post_encode_update(unsigned int consumed_bytes); +int getQPfromRatectrl(int current_frame_type); + +typedef void *(*createLibmeboRateController_t)(LibMeboCodecType CodecType, + LibMeboBrcAlgorithmID algo_id); +typedef void *(*init_rate_controller_t)(LibMeboRateController *rc, + LibMeboRateControllerConfig *rc_config); +typedef LibMeboStatus (*update_rate_control_t)( + LibMeboRateController *rc, LibMeboRateControllerConfig *rc_cfg); +typedef LibMeboStatus (*post_encode_update_t)(LibMeboRateController *rc, + uint64_t encoded_frame_size); +typedef LibMeboStatus (*compute_qp_t)(LibMeboRateController *rc, + LibMeboRCFrameParams *rc_frame_params); +typedef LibMeboStatus (*get_qp_t)(LibMeboRateController *rc, int *qp); + +typedef LibMeboStatus (*get_loop_filter_level_t)(LibMeboRateController *rc, + int *filter_level); +struct FunctionPtrstoLibmebo { + void *handle; + LibMeboRateController *libmebo_brc; + createLibmeboRateController_t ptrCreateLibmeboController; + init_rate_controller_t ptrInit_rate_controller; + update_rate_control_t ptrUpdate_rate_control; + post_encode_update_t ptrPost_encode_update; + compute_qp_t ptrCompute_qp; + get_qp_t ptrGet_qp; + get_loop_filter_level_t ptrGet_loop_filter_level; +}; + +#if 1 +typedef enum ErrorsLoadingFunc { + kMainHandleLibError_interface = -1, + kLibMeboControllerSymbolError = -2, + kInitRateControllerSymbolError = -3, + kUpdateRateControllerSymbolError = -4, + kPostEncodeSymbolSymbolError = -5, + kComputeQpSybmolError = -6, + kGetQPSymbolError = -7, + kGetLoopFilterSymbolError = -8, +} ErrosLibmeboSymbols_interface; +#endif + +#ifdef __cplusplus +} +#endif diff --git a/encode/libmebo/libs/aom_libs/libaom_av1_rc.so b/encode/libmebo/libs/aom_libs/libaom_av1_rc.so new file mode 100755 index 0000000..9c9912f Binary files /dev/null and b/encode/libmebo/libs/aom_libs/libaom_av1_rc.so differ diff --git a/encode/libmebo/libs/libbrc.so b/encode/libmebo/libs/libbrc.so new file mode 100755 index 0000000..d0f7bf8 Binary files /dev/null and b/encode/libmebo/libs/libbrc.so differ diff --git a/encode/libmebo/libs/libbrc_controller.so b/encode/libmebo/libs/libbrc_controller.so new file mode 100755 index 0000000..e595466 Binary files /dev/null and b/encode/libmebo/libs/libbrc_controller.so differ diff --git a/encode/libmebo/libs/libmebo.so b/encode/libmebo/libs/libmebo.so new file mode 100755 index 0000000..6ee98c7 Binary files /dev/null and b/encode/libmebo/libs/libmebo.so differ diff --git a/encode/libmebo/ratectrl_rtc.h b/encode/libmebo/ratectrl_rtc.h new file mode 100644 index 0000000..3d0807b --- /dev/null +++ b/encode/libmebo/ratectrl_rtc.h @@ -0,0 +1,186 @@ +/* + * Copyright (c) 2021, Alliance for Open Media. All rights reserved. + * + * This source code is subject to the terms of the BSD 2 Clause License and + * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License + * was not distributed with this source code in the LICENSE file, you can + * obtain it at www.aomedia.org/license/software. If the Alliance for Open + * Media Patent License 1.0 was not distributed with this source code in the + * PATENTS file, you can obtain it at www.aomedia.org/license/patent. + */ + +#ifndef AOM_AV1_RATECTRL_RTC_H_ +#define AOM_AV1_RATECTRL_RTC_H_ + +#ifdef __cplusplus +#include +#include +#include +#else +#include +#include +#include +#endif // __cplusplus + +struct AV1_COMP; + +typedef struct AomAV1LoopfilterLevel { + int filter_level[2]; + int filter_level_u; + int filter_level_v; +} AomAV1LoopfilterLevel; + +typedef struct AomAV1CdefInfo { + int cdef_strength_y; + int cdef_strength_uv; + int damping; +} AomAV1CdefInfo; + +typedef struct AomAV1SegmentationData { + const uint8_t *segmentation_map; + size_t segmentation_map_size; + const int *delta_q; + size_t delta_q_size; +} AomAV1SegmentationData; + +typedef enum AomFrameType { kAomKeyFrame, kAomInterFrame } AomFrameType; + +typedef struct AomAV1FrameParamsRTC { + AomFrameType frame_type; + int spatial_layer_id; + int temporal_layer_id; +} AomAV1FrameParamsRTC; + +typedef enum AomFrameDropDecision { + kAomFrameDropDecisionOk, // Frame is encoded. + kAomFrameDropDecisionDrop, // Frame is dropped. +} AomFrameDropDecision; + +// These constants come from AV1 spec. +enum { + kAomAV1MaxLayers = 32, + kAomAV1MaxTemporalLayers = 8, + kAomAV1MaxSpatialLayers = 4, +}; + +typedef struct AomAV1RateControlRtcConfig { +#ifdef __cplusplus + AomAV1RateControlRtcConfig(); +#endif + + int width; + int height; + // Flag indicating if the content is screen or not. + bool is_screen; + // 0-63 + int max_quantizer; + int min_quantizer; + int64_t target_bandwidth; + int64_t buf_initial_sz; + int64_t buf_optimal_sz; + int64_t buf_sz; + int undershoot_pct; + int overshoot_pct; + int max_intra_bitrate_pct; + int max_inter_bitrate_pct; + int frame_drop_thresh; + int max_consec_drop_ms; + double framerate; + int layer_target_bitrate[kAomAV1MaxLayers]; + int ts_rate_decimator[kAomAV1MaxTemporalLayers]; + int aq_mode; + // Number of spatial layers + int ss_number_layers; + // Number of temporal layers + int ts_number_layers; + int max_quantizers[kAomAV1MaxLayers]; + int min_quantizers[kAomAV1MaxLayers]; + int scaling_factor_num[kAomAV1MaxSpatialLayers]; + int scaling_factor_den[kAomAV1MaxSpatialLayers]; +} AomAV1RateControlRtcConfig; + +struct AomAV1RateControlRTC; +typedef struct AomAV1RateControlRTC AomAV1RateControlRTC; + +#ifdef __cplusplus +namespace aom { + +using AV1LoopfilterLevel = AomAV1LoopfilterLevel; +using AV1CdefInfo = AomAV1CdefInfo; +using AV1SegmentationData = AomAV1SegmentationData; +using AV1FrameParamsRTC = AomAV1FrameParamsRTC; +using AV1RateControlRtcConfig = AomAV1RateControlRtcConfig; + +using FrameType = AomFrameType; +constexpr FrameType kKeyFrame = kAomKeyFrame; +constexpr FrameType kInterFrame = kAomInterFrame; + +using FrameDropDecision = AomFrameDropDecision; +constexpr FrameDropDecision kFrameDropDecisionOk = kAomFrameDropDecisionOk; +constexpr FrameDropDecision kFrameDropDecisionDrop = kAomFrameDropDecisionDrop; + +class AV1RateControlRTC { + public: + static std::unique_ptr Create( + const AV1RateControlRtcConfig &cfg); + ~AV1RateControlRTC(); + + bool UpdateRateControl(const AV1RateControlRtcConfig &rc_cfg); + // GetQP() needs to be called after ComputeQP() to get the latest QP + int GetQP() const; + // GetLoopfilterLevel() needs to be called after ComputeQP() + AV1LoopfilterLevel GetLoopfilterLevel() const; + // GetCdefInfo() needs to be called after ComputeQP() + AV1CdefInfo GetCdefInfo() const; + // Returns the segmentation map used for cyclic refresh, based on 4x4 blocks. + bool GetSegmentationData(AV1SegmentationData *segmentation_data) const; + // ComputeQP returns the QP if the frame is not dropped (kOk return), + // otherwise it returns kDrop and subsequent GetQP and PostEncodeUpdate + // are not to be called (av1_rc_postencode_update_drop_frame is already + // called via ComputeQP if drop is decided). + FrameDropDecision ComputeQP(const AV1FrameParamsRTC &frame_params); + // Feedback to rate control with the size of current encoded frame + void PostEncodeUpdate(uint64_t encoded_frame_size); + + private: + AV1RateControlRTC() = default; + bool InitRateControl(const AV1RateControlRtcConfig &cfg); + AV1_COMP *cpi_; + int initial_width_; + int initial_height_; +}; +} // namespace aom +#endif // __cplusplus + +#ifdef __cplusplus +extern "C" { +#endif +AomAV1RateControlRTC *av1_ratecontrol_rtc_create( + const AomAV1RateControlRtcConfig *rc_cfg); +void av1_ratecontrol_rtc_destroy(AomAV1RateControlRTC *controller); +bool av1_ratecontrol_rtc_update(AomAV1RateControlRTC *controller, + const AomAV1RateControlRtcConfig *rc_cfg); +int av1_ratecontrol_rtc_get_qp(const AomAV1RateControlRTC *controller); + +AomAV1LoopfilterLevel av1_ratecontrol_rtc_get_loop_filter_level( + const AomAV1RateControlRTC *controller); +AomFrameDropDecision av1_ratecontrol_rtc_compute_qp( + AomAV1RateControlRTC *controller, const AomAV1FrameParamsRTC *frame_params); + +void av1_ratecontrol_rtc_post_encode_update(AomAV1RateControlRTC *controller, + uint64_t encoded_frame_size); + +bool av1_ratecontrol_rtc_get_segmentation( + const AomAV1RateControlRTC *controller, + AomAV1SegmentationData *segmentation_data); + +AomAV1CdefInfo av1_ratecontrol_rtc_get_cdef_info( + const AomAV1RateControlRTC *controller); + +void av1_ratecontrol_rtc_init_ratecontrol_config( + AomAV1RateControlRtcConfig *config); +#ifdef __cplusplus +} // extern "C" +#endif + +#endif // AOM_AV1_RATECTRL_RTC_H_ diff --git a/encode/meson.build b/encode/meson.build index 65dcd4f..19b5bfc 100644 --- a/encode/meson.build +++ b/encode/meson.build @@ -24,8 +24,34 @@ executable('vp9enc', [ 'vp9enc.c' ], executable('vp8enc', [ 'vp8enc.c' ], dependencies: [libva_display_dep, threads, m ], install: true) + +c_compiler = meson.get_compiler('c') +cpp_compiler = meson.get_compiler('cpp') + +dl = c.find_library('dl') +dl_dep_cpp = cpp_compiler.find_library('dl', required:true) + + if libva_dep.version().version_compare('>= 1.14.0') - executable('av1encode', [ 'av1encode.c' ], - dependencies: [ libva_display_dep, threads, m ], + + av1_deps = [libva_display_dep, threads, m, dl, dl_dep_cpp] + if use_mebo and mebo_dep.found() + av1_deps += mebo_dep + endif + + if use_aom_av1_rc and aom_av1_rc_dep.found() + av1_deps += aom_av1_rc_dep + endif + + if use_brc and brc_dep.found() + av1_deps += brc_dep + endif + + if use_brc_controller and brc_controller_dep.found() + av1_deps += brc_controller_dep + endif + + executable('av1encode', [ 'av1encode.c', 'libmebo/libmebo_interface.cc' ], + dependencies: av1_deps, install: true) endif \ No newline at end of file diff --git a/meson.build b/meson.build index ef1df3b..5f227cf 100644 --- a/meson.build +++ b/meson.build @@ -19,6 +19,108 @@ libva_utils_flags = [ '-Wno-unused-parameter', backends = '' +c_compiler = meson.get_compiler('c') +cpp_compiler = meson.get_compiler('cpp') + +#Add Mebo support +use_mebo = false +if get_option('mebo') != 'disabled' + require_mebo = get_option('mebo') == 'enabled' + local_mebo_path = join_paths(meson.current_source_dir(), + 'encode', 'libmebo', 'libs') + mebo_dep = cpp_compiler.find_library('mebo', + dirs :[local_mebo_path, + '/usr/local/lib/x86_64-linux-gnu'], + required:false) + if not mebo_dep.found() and require_mebo + error('''LibMebo library not installed in local path or system: + local path=/encode/libmebo/libs''') + endif + + if mebo_dep.found() + use_mebo = true + message('mebo library found') + libva_utils_flags += ['-DHAVE_MEBO=1'] + endif + if not mebo_dep.found() + error('LibMebo not installed') + endif +endif + +use_brc = false +if get_option('brc') != 'disabled' + require_brc = get_option('brc') == 'enabled' + local_brc_path = join_paths(meson.current_source_dir(), + 'encode', 'libmebo', 'libs') + brc_dep = cpp_compiler.find_library('brc', + dirs :[local_brc_path, + '/usr/local/lib/x86_64-linux-gnu'], + required:false) + if not brc_dep.found() and require_brc + error('''Libbrc library not installed in local path or system: + local path=/encode/libmebo/libs''') + endif + + if brc_dep.found() + use_brc = true + message('brc library found') + libva_utils_flags += ['-DHAVE_BRC=1'] + endif + if not brc_dep.found() + error('LIBBRC not installed') + endif +endif + +use_brc_controller = false +if get_option('brc_controller') != 'disabled' + require_brc_controller = get_option('brc_controller') == 'enabled' + local_brc_controller_path = join_paths(meson.current_source_dir(), + 'encode', 'libmebo', 'libs') + brc_controller_dep = cpp_compiler.find_library('brc_controller', + dirs :[local_brc_controller_path, + '/usr/local/lib/x86_64-linux-gnu'], + required:false) + if not brc_controller_dep.found() and require_brc_controller + error('''Libbrc_controller library not installed in local path or system: + local path=/encode/libmebo/libs''') + endif + + if brc_controller_dep.found() + use_brc_controller = true + message('Libbrc_controller library found') + libva_utils_flags += ['-DHAVE_BRC=1'] + endif + if not brc_controller_dep.found() + error('Libbrc_controller not installed') + endif +endif + +#Add AOM_AV1_RC support +use_aom_av1_rc = false +if get_option('aom_av1_rc') != 'disabled' + require_aom_av1_rc = get_option('aom_av1_rc') == 'enabled' + local_aom_av1_rc_path = join_paths(meson.current_source_dir(), + 'encode', 'libs', 'aom_libs') + aom_av1_rc_dep = cpp_compiler.find_library('aom_av1_rc', + dirs :[local_aom_av1_rc_path, + '/usr/local/lib/x86_64-linux-gnu'], + required:false) + if not aom_av1_rc_dep.found() and require_aom_av1_rc + error('''libaom_av1_rc.so not installed in local path or system: + local path= encode/libmebo/libs/aom_libs''') + endif + + if aom_av1_rc_dep.found() + use_aom_av1_rc = true + message('aom_av1_rc library found ') + libva_utils_flags += ['-DHAVE_AOM_AV1_RC=1'] + endif + + if not aom_av1_rc_dep.found() + error('Lib AOM AV1 RC not installed') + endif +endif + # DRM use_drm = false drm_deps=[] diff --git a/meson_options.txt b/meson_options.txt index 4263b5d..92825d4 100644 --- a/meson_options.txt +++ b/meson_options.txt @@ -17,3 +17,14 @@ option('win32', option('tests', type : 'boolean', value : false) +option ('mebo', type: 'feature', value: 'enabled', + description: 'Enable LibMebo library support') + +option ('aom_av1_rc', type: 'feature', value: 'enabled', + description: 'Enable AV1 software BRC from AOM library') + +option ('brc', type: 'feature', value: 'enabled', + description: 'Enable brc to support libmebo library ') + +option ('brc_controller', type: 'feature', value: 'enabled', + description: 'Enable brc_controller to support libmebo')