--enable-encoder=libvpx_vp8 \
--enable-encoder=libvpx_vp9 \
--disable-decoder=*_crystalhd \
+ --enable-amf \
+ --enable-encoder=h264_amf \
+ --enable-encoder=hevc_amf \
--cc="$(FFMPEG.GCC.gcc)" \
--extra-ldflags="$(call fn.ARGS,FFMPEG.GCC,*archs *sysroot *minver ?extra) -L$(call fn.ABSOLUTE,$(CONTRIB.build/)lib)"
{ { "H.264 (x264)", "x264", "H.264 (libx264)", HB_VCODEC_X264_8BIT, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H264, },
{ { "H.264 10-bit (x264)", "x264_10bit", "H.264 10-bit (libx264)", HB_VCODEC_X264_10BIT, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H264, },
{ { "H.264 (Intel QSV)", "qsv_h264", "H.264 (Intel Media SDK)", HB_VCODEC_QSV_H264, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H264, },
+ { { "H.264 (AMD VCE)", "vce_h264", "H.264 (libavcodec)", HB_VCODEC_FFMPEG_VCE_H264, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H264, },
{ { "H.265 (x265)", "x265", "H.265 (libx265)", HB_VCODEC_X265_8BIT, HB_MUX_AV_MP4|HB_MUX_AV_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
{ { "H.265 10-bit (x265)", "x265_10bit", "H.265 10-bit (libx265)", HB_VCODEC_X265_10BIT, HB_MUX_AV_MP4|HB_MUX_AV_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
{ { "H.265 12-bit (x265)", "x265_12bit", "H.265 12-bit (libx265)", HB_VCODEC_X265_12BIT, HB_MUX_AV_MP4|HB_MUX_AV_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
{ { "H.265 16-bit (x265)", "x265_16bit", "H.265 16-bit (libx265)", HB_VCODEC_X265_16BIT, HB_MUX_AV_MP4|HB_MUX_AV_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
{ { "H.265 (Intel QSV)", "qsv_h265", "H.265 (Intel Media SDK)", HB_VCODEC_QSV_H265, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
{ { "H.265 10-bit (Intel QSV)", "qsv_h265_10bit", "H.265 10-bit (Intel Media SDK)", HB_VCODEC_QSV_H265_10BIT, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H265, },
+ { { "H.265 (AMD VCE)", "vce_h265", "H.265 (libavcodec)", HB_VCODEC_FFMPEG_VCE_H265, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_H264, },
{ { "MPEG-4", "mpeg4", "MPEG-4 (libavcodec)", HB_VCODEC_FFMPEG_MPEG4, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_MPEG4, },
{ { "MPEG-2", "mpeg2", "MPEG-2 (libavcodec)", HB_VCODEC_FFMPEG_MPEG2, HB_MUX_MASK_MP4|HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_MPEG2, },
{ { "VP8", "VP8", "VP8 (libvpx)", HB_VCODEC_FFMPEG_VP8, HB_MUX_MASK_MKV, }, NULL, 1, HB_GID_VCODEC_VP8, },
case HB_VCODEC_FFMPEG_MPEG2:
case HB_VCODEC_FFMPEG_VP8:
case HB_VCODEC_FFMPEG_VP9:
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ case HB_VCODEC_FFMPEG_VCE_H265:
return 1;
#ifdef USE_X265
case HB_VCODEC_X265_16BIT:
return hb_h265_profile_names_16bit;
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ return hb_h264_profile_names_8bit;
+ case HB_VCODEC_FFMPEG_VCE_H265:
+ return hb_h265_profile_names_8bit;
+
default:
return NULL;
}
cfr: 0 (vfr), 1 (cfr), 2 (pfr) [see render.c]
pass: 0, 1 or 2 (or -1 for scan)
areBframes: boolean to note if b-frames are used */
-#define HB_VCODEC_MASK 0x00FFFFF
+#define HB_VCODEC_MASK 0x08FFFFF
#define HB_VCODEC_INVALID 0x0000000
#define HB_VCODEC_THEORA 0x0000002
#define HB_VCODEC_FFMPEG_MPEG4 0x0000010
#define HB_VCODEC_FFMPEG_MPEG2 0x0000020
#define HB_VCODEC_FFMPEG_VP8 0x0000040
#define HB_VCODEC_FFMPEG_VP9 0x0000080
-#define HB_VCODEC_FFMPEG_MASK 0x00000F0
+#define HB_VCODEC_FFMPEG_VCE_H264 0x00040000
+#define HB_VCODEC_FFMPEG_VCE_H265 0x00080000
+#define HB_VCODEC_FFMPEG_MASK (0x00000F0|HB_VCODEC_FFMPEG_VCE_H264|HB_VCODEC_FFMPEG_VCE_H265)
#define HB_VCODEC_QSV_H264 0x0000100
#define HB_VCODEC_QSV_H265_8BIT 0x0000200
#define HB_VCODEC_QSV_H265_10BIT 0x0000400
#define HB_VCODEC_X264 HB_VCODEC_X264_8BIT
#define HB_VCODEC_X264_10BIT 0x0020000
#define HB_VCODEC_X264_MASK 0x0030000
-#define HB_VCODEC_H264_MASK (HB_VCODEC_X264_MASK|HB_VCODEC_QSV_H264)
+#define HB_VCODEC_H264_MASK (HB_VCODEC_X264_MASK|HB_VCODEC_QSV_H264|HB_VCODEC_FFMPEG_VCE_H264)
#define HB_VCODEC_X265_8BIT 0x0001000
#define HB_VCODEC_X265 HB_VCODEC_X265_8BIT
#define HB_VCODEC_X265_10BIT 0x0002000
#define HB_VCODEC_X265_12BIT 0x0004000
#define HB_VCODEC_X265_16BIT 0x0008000
#define HB_VCODEC_X265_MASK 0x000F000
-#define HB_VCODEC_H265_MASK (HB_VCODEC_X265_MASK|HB_VCODEC_QSV_H265_MASK)
+#define HB_VCODEC_H265_MASK (HB_VCODEC_X265_MASK|HB_VCODEC_QSV_H265_MASK|HB_VCODEC_FFMPEG_VCE_H265)
/* define an invalid CQ value compatible with all CQ-capable codecs */
#define HB_INVALID_VIDEO_QUALITY (-1000.)
#include "hb.h"
#include "hb_dict.h"
#include "hbffmpeg.h"
+#include "h264_common.h"
+#include "h265_common.h"
+#include "nal_units.h"
/*
* The frame info struct remembers information about each frame across calls
case AV_CODEC_ID_MPEG4:
{
hb_log("encavcodecInit: MPEG-4 ASP encoder");
+ codec = avcodec_find_encoder_by_name("mpeg4");
} break;
case AV_CODEC_ID_MPEG2VIDEO:
{
hb_log("encavcodecInit: MPEG-2 encoder");
+ codec = avcodec_find_encoder_by_name("mpeg2video");
} break;
case AV_CODEC_ID_VP8:
{
hb_log("encavcodecInit: VP8 encoder");
+ codec = avcodec_find_encoder_by_name("libvpx_vp8");
} break;
case AV_CODEC_ID_VP9:
{
hb_log("encavcodecInit: VP9 encoder");
+ codec = avcodec_find_encoder_by_name("libvpx_vp9");
} break;
+ case AV_CODEC_ID_H264:
+ {
+ hb_log("encavcodecInit: H.264 (AMD VCE)");
+ codec = avcodec_find_encoder_by_name("h264_amf");
+ }break;
+ case AV_CODEC_ID_HEVC:
+ {
+ hb_log("encavcodecInit: H.265 (AMD VCE)");
+ codec = avcodec_find_encoder_by_name("hevc_amf");
+ }break;
default:
{
hb_error("encavcodecInit: unsupported encoder!");
}
}
- codec = avcodec_find_encoder( w->codec_param );
if( !codec )
{
hb_log( "encavcodecInit: avcodec_find_encoder "
context->time_base.num = fps.den;
context->gop_size = ((double)job->orig_vrate.num / job->orig_vrate.den +
0.5) * 10;
+ if ((job->vcodec == HB_VCODEC_FFMPEG_VCE_H264) || (job->vcodec == HB_VCODEC_FFMPEG_VCE_H265))
+ {
+ // Set encoder preset
+ context->profile = FF_PROFILE_UNKNOWN;
+ if (job->encoder_preset != NULL && *job->encoder_preset)
+ {
+ if ((!strcasecmp(job->encoder_preset, "balanced"))
+ || (!strcasecmp(job->encoder_preset, "speed"))
+ || (!strcasecmp(job->encoder_preset, "quality")))
+ {
+ av_opt_set(context, "quality", job->encoder_preset, AV_OPT_SEARCH_CHILDREN);
+ }
+ }
+ }
/* place job->encoder_options in an hb_dict_t for convenience */
hb_dict_t * lavc_opts = NULL;
context->flags |= AV_CODEC_FLAG_GRAY;
}
+ if (job->vcodec == HB_VCODEC_FFMPEG_VCE_H264)
+ {
+ // Set profile and level
+ context->profile = FF_PROFILE_UNKNOWN;
+ if (job->encoder_profile != NULL && *job->encoder_profile)
+ {
+ if (!strcasecmp(job->encoder_profile, "baseline"))
+ context->profile = FF_PROFILE_H264_BASELINE;
+ else if (!strcasecmp(job->encoder_profile, "main"))
+ context->profile = FF_PROFILE_H264_MAIN;
+ else if (!strcasecmp(job->encoder_profile, "high"))
+ context->profile = FF_PROFILE_H264_HIGH;
+ }
+ context->level = FF_LEVEL_UNKNOWN;
+ if (job->encoder_level != NULL && *job->encoder_level)
+ {
+ int i = 1;
+ while (hb_h264_level_names[i] != NULL)
+ {
+ if (!strcasecmp(job->encoder_level, hb_h264_level_names[i]))
+ context->level = hb_h264_level_values[i];
+ ++i;
+ }
+ }
+ }
+
+ if (job->vcodec == HB_VCODEC_FFMPEG_VCE_H265)
+ {
+ // Set profile and level
+ context->profile = FF_PROFILE_UNKNOWN;
+ if (job->encoder_profile != NULL && *job->encoder_profile)
+ {
+ if (!strcasecmp(job->encoder_profile, "main"))
+ context->profile = FF_PROFILE_HEVC_MAIN;
+ }
+ context->level = FF_LEVEL_UNKNOWN;
+ if (job->encoder_level != NULL && *job->encoder_level)
+ {
+ int i = 1;
+ while (hb_h265_level_names[i] != NULL)
+ {
+ if (!strcasecmp(job->encoder_level, hb_h265_level_names[i]))
+ context->level = hb_h265_level_values[i];
+ ++i;
+ }
+ }
+ // FIXME
+ //context->tier = FF_TIER_UNKNOWN;
+ }
+
if( job->pass_id == HB_PASS_ENCODE_1ST ||
job->pass_id == HB_PASS_ENCODE_2ND )
{
if (hb_avcodec_open(context, codec, &av_opts, HB_FFMPEG_THREADS_AUTO))
{
hb_log( "encavcodecInit: avcodec_open failed" );
+ return 1;
}
if (job->pass_id == HB_PASS_ENCODE_1ST &&
{
job->areBframes = 1;
}
+
if( ( job->mux & HB_MUX_MASK_MP4 ) && job->pass_id != HB_PASS_ENCODE_1ST )
{
- w->config->mpeg4.length = context->extradata_size;
- memcpy( w->config->mpeg4.bytes, context->extradata,
- context->extradata_size );
+ if (w->codec_param == AV_CODEC_ID_H264) // FIXME: h265 as well?
+ {
+ // Scan extradata for the SPS/PPS headers
+ unsigned char *data = context->extradata;
+ unsigned char *dataEnd = context->extradata + context->extradata_size;
+ size_t len = dataEnd - data;
+
+ while ((data = hb_annexb_find_next_nalu(data, &len)) != NULL) {
+ if ((data[0] & 0x1f) == 7) {
+ // SPS found, copy into work object
+ w->config->h264.sps_length = len;
+ memcpy(w->config->h264.sps, data, len);
+ }
+ if ((data[0] & 0x1f) == 8) {
+ // PPS found, copy into work object
+ w->config->h264.pps_length = len;
+ memcpy(w->config->h264.pps, data, len);
+ }
+ len = dataEnd - data;
+ }
+ }
+ else
+ {
+ w->config->mpeg4.length = context->extradata_size;
+ memcpy( w->config->mpeg4.bytes, context->extradata,
+ context->extradata_size );
+ }
}
done:
static void get_packets( hb_work_object_t * w, hb_buffer_list_t * list )
{
hb_work_private_t * pv = w->private_data;
+ hb_job_t * job = pv->job;
while (1)
{
{
hb_log("encavcodec: avcodec_receive_packet failed");
}
- out = hb_buffer_init(pkt.size);
- memcpy(out->data, pkt.data, out->size);
+
+ if (job->vcodec == HB_VCODEC_FFMPEG_VCE_H264)
+ {
+ out = hb_nal_bitstream_annexb_to_mp4(pkt.data, pkt.size);
+ }
+ else
+ {
+ out = hb_buffer_init(pkt.size);
+ memcpy(out->data, pkt.data, out->size);
+ }
int64_t frameno = pkt.pts;
out->size = pkt.size;
case HB_VCODEC_FFMPEG_VP9:
return vpx_preset_names;
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ case HB_VCODEC_FFMPEG_VCE_H265:
+ return hb_vce_preset_names;
+
default:
return NULL;
}
For full terms see the file COPYING file or visit http://www.gnu.org/licenses/gpl-2.0.html
*/
+#ifndef HB_FFMPEG_H
+#define HB_FFMPEG_H
+
#include "libavcodec/avcodec.h"
#include "libavformat/avformat.h"
#include "libavutil/channel_layout.h"
int dstW, int dstH, enum AVPixelFormat dstFormat,
int flags, int colorspace);
+static const char* const hb_vce_preset_names[] = { "speed", "balanced", "quality", NULL, };
+
hb_buffer_t * hb_avframe_to_video_buffer(AVFrame *frame, AVRational time_base);
void hb_avframe_set_video_buffer_flags(hb_buffer_t * buf, AVFrame *frame,
AVRational time_base);
+#endif
case HB_VCODEC_X264_8BIT:
case HB_VCODEC_X264_10BIT:
case HB_VCODEC_QSV_H264:
+ case HB_VCODEC_FFMPEG_VCE_H264:
track->st->codecpar->codec_id = AV_CODEC_ID_H264;
if (job->mux == HB_MUX_AV_MP4 && job->inline_parameter_sets)
{
case HB_VCODEC_X265_16BIT:
case HB_VCODEC_QSV_H265:
case HB_VCODEC_QSV_H265_10BIT:
+ case HB_VCODEC_FFMPEG_VCE_H265:
track->st->codecpar->codec_id = AV_CODEC_ID_HEVC;
if (job->mux == HB_MUX_AV_MP4 && job->inline_parameter_sets)
{
w = hb_get_work(h, WORK_ENCX265);
break;
#endif
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ w = hb_get_work(h, WORK_ENCAVCODEC);
+ w->codec_param = AV_CODEC_ID_H264;
+ break;
+ case HB_VCODEC_FFMPEG_VCE_H265:
+ w = hb_get_work(h, WORK_ENCAVCODEC);
+ w->codec_param = AV_CODEC_ID_HEVC;
+ break;
default:
hb_error("Unknown video codec (0x%x)", vcodec );
}
case HB_VCODEC_QSV_H264:
case HB_VCODEC_QSV_H265:
case HB_VCODEC_QSV_H265_10BIT:
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ case HB_VCODEC_FFMPEG_VCE_H265:
hb_log(" + profile: %s", job->encoder_profile);
default:
break;
case HB_VCODEC_QSV_H264:
case HB_VCODEC_QSV_H265:
case HB_VCODEC_QSV_H265_10BIT:
+ case HB_VCODEC_FFMPEG_VCE_H264:
+ case HB_VCODEC_FFMPEG_VCE_H265:
hb_log(" + level: %s", job->encoder_level);
default:
break;