You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

271 lines
8.4KB

  1. /*
  2. * This file is part of FFmpeg.
  3. *
  4. * FFmpeg is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Lesser General Public
  6. * License as published by the Free Software Foundation; either
  7. * version 2.1 of the License, or (at your option) any later version.
  8. *
  9. * FFmpeg is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * Lesser General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU Lesser General Public
  15. * License along with FFmpeg; if not, write to the Free Software
  16. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  17. */
  18. #include <va/va.h>
  19. #include <va/va_enc_vp8.h>
  20. #include "libavutil/avassert.h"
  21. #include "libavutil/common.h"
  22. #include "libavutil/internal.h"
  23. #include "libavutil/opt.h"
  24. #include "libavutil/pixfmt.h"
  25. #include "avcodec.h"
  26. #include "internal.h"
  27. #include "vaapi_encode.h"
  28. #include "vp8.h"
  29. typedef struct VAAPIEncodeVP8Context {
  30. int q_index_i;
  31. int q_index_p;
  32. } VAAPIEncodeVP8Context;
  33. typedef struct VAAPIEncodeVP8Options {
  34. int loop_filter_level;
  35. int loop_filter_sharpness;
  36. } VAAPIEncodeVP8Options;
  37. #define vseq_var(name) vseq->name, name
  38. #define vseq_field(name) vseq->seq_fields.bits.name, name
  39. #define vpic_var(name) vpic->name, name
  40. #define vpic_field(name) vpic->pic_fields.bits.name, name
  41. static int vaapi_encode_vp8_init_sequence_params(AVCodecContext *avctx)
  42. {
  43. VAAPIEncodeContext *ctx = avctx->priv_data;
  44. VAEncSequenceParameterBufferVP8 *vseq = ctx->codec_sequence_params;
  45. vseq->frame_width = avctx->width;
  46. vseq->frame_height = avctx->height;
  47. vseq->frame_width_scale = 0;
  48. vseq->frame_height_scale = 0;
  49. vseq->error_resilient = 0;
  50. vseq->kf_auto = 0;
  51. if (!(ctx->va_rc_mode & VA_RC_CQP)) {
  52. vseq->bits_per_second = avctx->bit_rate;
  53. vseq->intra_period = avctx->gop_size;
  54. }
  55. return 0;
  56. }
  57. static int vaapi_encode_vp8_init_picture_params(AVCodecContext *avctx,
  58. VAAPIEncodePicture *pic)
  59. {
  60. VAAPIEncodeContext *ctx = avctx->priv_data;
  61. VAEncPictureParameterBufferVP8 *vpic = pic->codec_picture_params;
  62. VAAPIEncodeVP8Options *opt = ctx->codec_options;
  63. int i;
  64. vpic->reconstructed_frame = pic->recon_surface;
  65. vpic->coded_buf = pic->output_buffer;
  66. switch (pic->type) {
  67. case PICTURE_TYPE_IDR:
  68. case PICTURE_TYPE_I:
  69. av_assert0(pic->nb_refs == 0);
  70. vpic->ref_flags.bits.force_kf = 1;
  71. vpic->ref_last_frame =
  72. vpic->ref_gf_frame =
  73. vpic->ref_arf_frame =
  74. VA_INVALID_SURFACE;
  75. break;
  76. case PICTURE_TYPE_P:
  77. av_assert0(pic->nb_refs == 1);
  78. vpic->ref_flags.bits.no_ref_last = 0;
  79. vpic->ref_flags.bits.no_ref_gf = 1;
  80. vpic->ref_flags.bits.no_ref_arf = 1;
  81. vpic->ref_last_frame =
  82. vpic->ref_gf_frame =
  83. vpic->ref_arf_frame =
  84. pic->refs[0]->recon_surface;
  85. break;
  86. default:
  87. av_assert0(0 && "invalid picture type");
  88. }
  89. vpic->pic_flags.bits.frame_type = (pic->type != PICTURE_TYPE_IDR);
  90. vpic->pic_flags.bits.show_frame = 1;
  91. vpic->pic_flags.bits.refresh_last = 1;
  92. vpic->pic_flags.bits.refresh_golden_frame = 1;
  93. vpic->pic_flags.bits.refresh_alternate_frame = 1;
  94. vpic->pic_flags.bits.version = 0;
  95. vpic->pic_flags.bits.loop_filter_type = 0;
  96. for (i = 0; i < 4; i++)
  97. vpic->loop_filter_level[i] = opt->loop_filter_level;
  98. vpic->sharpness_level = opt->loop_filter_sharpness;
  99. vpic->clamp_qindex_low = 0;
  100. vpic->clamp_qindex_high = 127;
  101. return 0;
  102. }
  103. static int vaapi_encode_vp8_write_quant_table(AVCodecContext *avctx,
  104. VAAPIEncodePicture *pic,
  105. int index, int *type,
  106. char *data, size_t *data_len)
  107. {
  108. VAAPIEncodeContext *ctx = avctx->priv_data;
  109. VAAPIEncodeVP8Context *priv = ctx->priv_data;
  110. VAQMatrixBufferVP8 quant;
  111. int i, q;
  112. if (index > 0)
  113. return AVERROR_EOF;
  114. if (*data_len < sizeof(quant))
  115. return AVERROR(EINVAL);
  116. *type = VAQMatrixBufferType;
  117. *data_len = sizeof(quant);
  118. if (pic->type == PICTURE_TYPE_P)
  119. q = priv->q_index_p;
  120. else
  121. q = priv->q_index_i;
  122. for (i = 0; i < 4; i++)
  123. quant.quantization_index[i] = q;
  124. for (i = 0; i < 5; i++)
  125. quant.quantization_index_delta[i] = 0;
  126. memcpy(data, &quant, sizeof(quant));
  127. return 0;
  128. }
  129. static av_cold int vaapi_encode_vp8_configure(AVCodecContext *avctx)
  130. {
  131. VAAPIEncodeContext *ctx = avctx->priv_data;
  132. VAAPIEncodeVP8Context *priv = ctx->priv_data;
  133. priv->q_index_p = av_clip(avctx->global_quality, 0, VP8_MAX_QUANT);
  134. if (avctx->i_quant_factor > 0.0)
  135. priv->q_index_i = av_clip((avctx->global_quality *
  136. avctx->i_quant_factor +
  137. avctx->i_quant_offset) + 0.5,
  138. 0, VP8_MAX_QUANT);
  139. else
  140. priv->q_index_i = priv->q_index_p;
  141. return 0;
  142. }
  143. static const VAAPIEncodeType vaapi_encode_type_vp8 = {
  144. .configure = &vaapi_encode_vp8_configure,
  145. .priv_data_size = sizeof(VAAPIEncodeVP8Context),
  146. .sequence_params_size = sizeof(VAEncSequenceParameterBufferVP8),
  147. .init_sequence_params = &vaapi_encode_vp8_init_sequence_params,
  148. .picture_params_size = sizeof(VAEncPictureParameterBufferVP8),
  149. .init_picture_params = &vaapi_encode_vp8_init_picture_params,
  150. .write_extra_buffer = &vaapi_encode_vp8_write_quant_table,
  151. };
  152. static av_cold int vaapi_encode_vp8_init(AVCodecContext *avctx)
  153. {
  154. VAAPIEncodeContext *ctx = avctx->priv_data;
  155. if (avctx->max_b_frames > 0) {
  156. av_log(avctx, AV_LOG_ERROR, "B-frames are not supported.\n");
  157. return AVERROR_PATCHWELCOME;
  158. }
  159. ctx->codec = &vaapi_encode_type_vp8;
  160. ctx->va_profile = VAProfileVP8Version0_3;
  161. ctx->va_entrypoint = VAEntrypointEncSlice;
  162. ctx->va_rt_format = VA_RT_FORMAT_YUV420;
  163. if (avctx->flags & AV_CODEC_FLAG_QSCALE) {
  164. ctx->va_rc_mode = VA_RC_CQP;
  165. } else if (avctx->bit_rate > 0) {
  166. if (avctx->rc_max_rate == avctx->bit_rate)
  167. ctx->va_rc_mode = VA_RC_CBR;
  168. else
  169. ctx->va_rc_mode = VA_RC_VBR;
  170. } else {
  171. ctx->va_rc_mode = VA_RC_CQP;
  172. }
  173. // Packed headers are not currently supported.
  174. ctx->va_packed_headers = 0;
  175. ctx->surface_width = FFALIGN(avctx->width, 16);
  176. ctx->surface_height = FFALIGN(avctx->height, 16);
  177. return ff_vaapi_encode_init(avctx);
  178. }
  179. #define OFFSET(x) (offsetof(VAAPIEncodeContext, codec_options_data) + \
  180. offsetof(VAAPIEncodeVP8Options, x))
  181. #define FLAGS (AV_OPT_FLAG_VIDEO_PARAM | AV_OPT_FLAG_ENCODING_PARAM)
  182. static const AVOption vaapi_encode_vp8_options[] = {
  183. { "loop_filter_level", "Loop filter level",
  184. OFFSET(loop_filter_level), AV_OPT_TYPE_INT, { .i64 = 16 }, 0, 63, FLAGS },
  185. { "loop_filter_sharpness", "Loop filter sharpness",
  186. OFFSET(loop_filter_sharpness), AV_OPT_TYPE_INT, { .i64 = 4 }, 0, 15, FLAGS },
  187. { NULL },
  188. };
  189. static const AVCodecDefault vaapi_encode_vp8_defaults[] = {
  190. { "b", "0" },
  191. { "bf", "0" },
  192. { "g", "120" },
  193. { "global_quality", "40" },
  194. { NULL },
  195. };
  196. static const AVClass vaapi_encode_vp8_class = {
  197. .class_name = "vp8_vaapi",
  198. .item_name = av_default_item_name,
  199. .option = vaapi_encode_vp8_options,
  200. .version = LIBAVUTIL_VERSION_INT,
  201. };
  202. AVCodec ff_vp8_vaapi_encoder = {
  203. .name = "vp8_vaapi",
  204. .long_name = NULL_IF_CONFIG_SMALL("VP8 (VAAPI)"),
  205. .type = AVMEDIA_TYPE_VIDEO,
  206. .id = AV_CODEC_ID_VP8,
  207. .priv_data_size = (sizeof(VAAPIEncodeContext) +
  208. sizeof(VAAPIEncodeVP8Options)),
  209. .init = &vaapi_encode_vp8_init,
  210. .encode2 = &ff_vaapi_encode2,
  211. .close = &ff_vaapi_encode_close,
  212. .priv_class = &vaapi_encode_vp8_class,
  213. .capabilities = AV_CODEC_CAP_DELAY | AV_CODEC_CAP_HARDWARE,
  214. .defaults = vaapi_encode_vp8_defaults,
  215. .pix_fmts = (const enum AVPixelFormat[]) {
  216. AV_PIX_FMT_VAAPI,
  217. AV_PIX_FMT_NONE,
  218. },
  219. .wrapper_name = "vaapi",
  220. };