2024-07-12
한어Русский языкEnglishFrançaisIndonesianSanskrit日本語DeutschPortuguêsΕλληνικάespañolItalianoSuomalainenLatina
Edellisessä osiossa "Äänen ja videon käytön aloittamisen perusteet: H.264-aihe (11) - kaava videon resoluution laskemiseen "", kuvaa kaavan H.264-koodatun videon resoluution laskemiseksi SPS:n attribuuttien avulla. Tämä artikkeli selittää videon resoluution laskemisen FFmpeg-lähdekoodissa.
Artikkelista "Äänen ja videon käytön aloittamisen perusteet: H.264-aihe (10) - Analyysi rakenteesta, joka tallentaa SPS-attribuutteja ja SPS:n dekoodauksen funktiota FFmpeg-lähdekoodissa》, voimme tietää, että FFmpeg-lähdekoodi purkaa SPS:n ff_h264_decode_seq_parameter_set-funktion kautta saadakseen attribuutit SPS:ssä.
Funktiossa ff_h264_decode_seq_parameter_set on seuraava koodi. Seuraavan koodin osan kautta saadaan videoresoluutio laskemiseen tarvittavat attribuutit:
- int ff_h264_decode_seq_parameter_set(GetBitContext *gb, AVCodecContext *avctx,
- H264ParamSets *ps, int ignore_truncation)
- {
- //...
-
- sps->gaps_in_frame_num_allowed_flag = get_bits1(gb);
- sps->mb_width = get_ue_golomb(gb) + 1;
- sps->mb_height = get_ue_golomb(gb) + 1;
-
- sps->frame_mbs_only_flag = get_bits1(gb);
-
- if (sps->mb_height >= INT_MAX / 2U) {
- av_log(avctx, AV_LOG_ERROR, "height overflown");
- goto fail;
- }
- sps->mb_height *= 2 - sps->frame_mbs_only_flag;
-
- //...
-
- sps->crop = get_bits1(gb);
- if (sps->crop) {
- unsigned int crop_left = get_ue_golomb(gb);
- unsigned int crop_right = get_ue_golomb(gb);
- unsigned int crop_top = get_ue_golomb(gb);
- unsigned int crop_bottom = get_ue_golomb(gb);
- int width = 16 * sps->mb_width;
- int height = 16 * sps->mb_height;
-
- if (avctx->flags2 & AV_CODEC_FLAG2_IGNORE_CROP) {
- av_log(avctx, AV_LOG_DEBUG, "discarding sps cropping, original "
- "values are l:%d r:%d t:%d b:%dn",
- crop_left, crop_right, crop_top, crop_bottom);
-
- sps->crop_left =
- sps->crop_right =
- sps->crop_top =
- sps->crop_bottom = 0;
- } else {
- int vsub = (sps->chroma_format_idc == 1) ? 1 : 0;
- int hsub = (sps->chroma_format_idc == 1 ||
- sps->chroma_format_idc == 2) ? 1 : 0;
- int step_x = 1 << hsub;
- int step_y = (2 - sps->frame_mbs_only_flag) << vsub;
-
- if (crop_left > (unsigned)INT_MAX / 4 / step_x ||
- crop_right > (unsigned)INT_MAX / 4 / step_x ||
- crop_top > (unsigned)INT_MAX / 4 / step_y ||
- crop_bottom> (unsigned)INT_MAX / 4 / step_y ||
- (crop_left + crop_right ) * step_x >= width ||
- (crop_top + crop_bottom) * step_y >= height
- ) {
- av_log(avctx, AV_LOG_ERROR, "crop values invalid %d %d %d %d / %d %dn",
- crop_left, crop_right, crop_top, crop_bottom, width, height);
- goto fail;
- }
-
- sps->crop_left = crop_left * step_x;
- sps->crop_right = crop_right * step_x;
- sps->crop_top = crop_top * step_y;
- sps->crop_bottom = crop_bottom * step_y;
- }
- } else {
- sps->crop_left =
- sps->crop_right =
- sps->crop_top =
- sps->crop_bottom =
- sps->crop = 0;
- }
-
- //...
- }
Sitten FFmpeg-lähdekoodin lähdetiedoston libavcodec/h264_parser.c funktiossa parse_nal_units on seuraava koodi:
- static inline int parse_nal_units(AVCodecParserContext *s,
- AVCodecContext *avctx,
- const uint8_t * const buf, int buf_size)
- {
- //...
-
- for (;;) {
- switch (nal.type) {
- case H264_NAL_SPS:
- ff_h264_decode_seq_parameter_set(&nal.gb, avctx, &p->ps, 0);
- break;
-
- //...
-
- case H264_NAL_IDR_SLICE:
-
- //...
-
- s->coded_width = 16 * sps->mb_width;
- s->coded_height = 16 * sps->mb_height;
- s->width = s->coded_width - (sps->crop_right + sps->crop_left);
- s->height = s->coded_height - (sps->crop_top + sps->crop_bottom);
- if (s->width <= 0 || s->height <= 0) {
- s->width = s->coded_width;
- s->height = s->coded_height;
- }
- //...
- }
- //...
- }
- }
Voit nähdä, että parse_nal_units-funktiossa videon resoluutio saadaan lopulta seuraavan lauseen avulla:
- s->width = s->coded_width - (sps->crop_right + sps->crop_left);
- s->height = s->coded_height - (sps->crop_top + sps->crop_bottom);
Voit nähdä videon resoluution laskemisen toteutuksen FFmpeg-lähdekoodissa ja artikkelissa "Äänen ja videon käytön aloittamisen perusteet: H.264-aihe (11) - kaava videon resoluution laskemiseenKohdassa kuvatut kaavat ovat johdonmukaisia.