mirror of
https://github.com/xenia-project/FFmpeg.git
synced 2024-11-27 05:20:48 +00:00
avformat/avc, mxfenc: Avoid allocation of H264 SPS structure, fix memleak
Up until now, ff_avc_decode_sps would parse a SPS and return some properties from it in a freshly allocated structure. Yet said structure is very small and completely internal to libavformat, so there is no reason to use the heap for it. This commit therefore changes the function to return an int and to modify a caller-provided structure. This will also allow ff_avc_decode_sps to return better error codes in the future. It also fixes a memleak in mxfenc: If a packet contained multiple SPS, only the SPS structure belonging to the last SPS would be freed, the other ones would leak when the pointer is overwritten to point to the new SPS structure. Of course, without allocations there are no leaks. This is Coverity issue #1445194. Furthermore, the SPS structure has been renamed from H264SequenceParameterSet to H264SPS in order to avoid overlong lines. Reviewed-by: Tomas Härdin <tjoppen@acc.umu.se> Signed-off-by: Andreas Rheinhardt <andreas.rheinhardt@gmail.com>
This commit is contained in:
parent
264ba34808
commit
a0b6df0a39
@ -196,18 +196,17 @@ int ff_isom_write_avcc(AVIOContext *pb, const uint8_t *data, int len)
|
|||||||
avio_write(pb, pps, pps_size);
|
avio_write(pb, pps, pps_size);
|
||||||
|
|
||||||
if (sps[3] != 66 && sps[3] != 77 && sps[3] != 88) {
|
if (sps[3] != 66 && sps[3] != 77 && sps[3] != 88) {
|
||||||
H264SequenceParameterSet *seq = ff_avc_decode_sps(sps + 3, sps_size - 3);
|
H264SPS seq;
|
||||||
if (!seq) {
|
ret = ff_avc_decode_sps(&seq, sps + 3, sps_size - 3);
|
||||||
ret = AVERROR(ENOMEM);
|
if (ret < 0)
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
|
||||||
avio_w8(pb, 0xfc | seq->chroma_format_idc); /* 6 bits reserved (111111) + chroma_format_idc */
|
avio_w8(pb, 0xfc | seq.chroma_format_idc); /* 6 bits reserved (111111) + chroma_format_idc */
|
||||||
avio_w8(pb, 0xf8 | (seq->bit_depth_luma - 8)); /* 5 bits reserved (11111) + bit_depth_luma_minus8 */
|
avio_w8(pb, 0xf8 | (seq.bit_depth_luma - 8)); /* 5 bits reserved (11111) + bit_depth_luma_minus8 */
|
||||||
avio_w8(pb, 0xf8 | (seq->bit_depth_chroma - 8)); /* 5 bits reserved (11111) + bit_depth_chroma_minus8 */
|
avio_w8(pb, 0xf8 | (seq.bit_depth_chroma - 8)); /* 5 bits reserved (11111) + bit_depth_chroma_minus8 */
|
||||||
avio_w8(pb, nb_sps_ext); /* number of sps ext */
|
avio_w8(pb, nb_sps_ext); /* number of sps ext */
|
||||||
if (nb_sps_ext)
|
if (nb_sps_ext)
|
||||||
avio_write(pb, sps_ext, sps_ext_size);
|
avio_write(pb, sps_ext, sps_ext_size);
|
||||||
av_free(seq);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fail:
|
fail:
|
||||||
@ -332,27 +331,24 @@ static inline int get_se_golomb(GetBitContext *gb) {
|
|||||||
return ((v >> 1) ^ sign) - sign;
|
return ((v >> 1) ^ sign) - sign;
|
||||||
}
|
}
|
||||||
|
|
||||||
H264SequenceParameterSet *ff_avc_decode_sps(const uint8_t *buf, int buf_size)
|
int ff_avc_decode_sps(H264SPS *sps, const uint8_t *buf, int buf_size)
|
||||||
{
|
{
|
||||||
int i, j, ret, rbsp_size, aspect_ratio_idc, pic_order_cnt_type;
|
int i, j, ret, rbsp_size, aspect_ratio_idc, pic_order_cnt_type;
|
||||||
int num_ref_frames_in_pic_order_cnt_cycle;
|
int num_ref_frames_in_pic_order_cnt_cycle;
|
||||||
int delta_scale, lastScale = 8, nextScale = 8;
|
int delta_scale, lastScale = 8, nextScale = 8;
|
||||||
int sizeOfScalingList;
|
int sizeOfScalingList;
|
||||||
H264SequenceParameterSet *sps = NULL;
|
|
||||||
GetBitContext gb;
|
GetBitContext gb;
|
||||||
uint8_t *rbsp_buf;
|
uint8_t *rbsp_buf;
|
||||||
|
|
||||||
rbsp_buf = ff_nal_unit_extract_rbsp(buf, buf_size, &rbsp_size, 0);
|
rbsp_buf = ff_nal_unit_extract_rbsp(buf, buf_size, &rbsp_size, 0);
|
||||||
if (!rbsp_buf)
|
if (!rbsp_buf)
|
||||||
return NULL;
|
return AVERROR(ENOMEM);
|
||||||
|
|
||||||
ret = init_get_bits8(&gb, rbsp_buf, rbsp_size);
|
ret = init_get_bits8(&gb, rbsp_buf, rbsp_size);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto end;
|
goto end;
|
||||||
|
|
||||||
sps = av_mallocz(sizeof(*sps));
|
memset(sps, 0, sizeof(*sps));
|
||||||
if (!sps)
|
|
||||||
goto end;
|
|
||||||
|
|
||||||
sps->profile_idc = get_bits(&gb, 8);
|
sps->profile_idc = get_bits(&gb, 8);
|
||||||
sps->constraint_set_flags |= get_bits1(&gb) << 0; // constraint_set0_flag
|
sps->constraint_set_flags |= get_bits1(&gb) << 0; // constraint_set0_flag
|
||||||
@ -448,7 +444,8 @@ H264SequenceParameterSet *ff_avc_decode_sps(const uint8_t *buf, int buf_size)
|
|||||||
sps->sar.den = 1;
|
sps->sar.den = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ret = 0;
|
||||||
end:
|
end:
|
||||||
av_free(rbsp_buf);
|
av_free(rbsp_buf);
|
||||||
return sps;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -46,8 +46,8 @@ typedef struct {
|
|||||||
uint8_t bit_depth_chroma;
|
uint8_t bit_depth_chroma;
|
||||||
uint8_t frame_mbs_only_flag;
|
uint8_t frame_mbs_only_flag;
|
||||||
AVRational sar;
|
AVRational sar;
|
||||||
} H264SequenceParameterSet;
|
} H264SPS;
|
||||||
|
|
||||||
H264SequenceParameterSet *ff_avc_decode_sps(const uint8_t *src, int src_len);
|
int ff_avc_decode_sps(H264SPS *sps, const uint8_t *buf, int buf_size);
|
||||||
|
|
||||||
#endif /* AVFORMAT_AVC_H */
|
#endif /* AVFORMAT_AVC_H */
|
||||||
|
@ -2171,14 +2171,14 @@ static int mxf_parse_h264_frame(AVFormatContext *s, AVStream *st,
|
|||||||
{
|
{
|
||||||
MXFContext *mxf = s->priv_data;
|
MXFContext *mxf = s->priv_data;
|
||||||
MXFStreamContext *sc = st->priv_data;
|
MXFStreamContext *sc = st->priv_data;
|
||||||
H264SequenceParameterSet *sps = NULL;
|
H264SPS seq, *const sps = &seq;
|
||||||
GetBitContext gb;
|
GetBitContext gb;
|
||||||
const uint8_t *buf = pkt->data;
|
const uint8_t *buf = pkt->data;
|
||||||
const uint8_t *buf_end = pkt->data + pkt->size;
|
const uint8_t *buf_end = pkt->data + pkt->size;
|
||||||
const uint8_t *nal_end;
|
const uint8_t *nal_end;
|
||||||
uint32_t state = -1;
|
uint32_t state = -1;
|
||||||
int extra_size = 512; // support AVC Intra files without SPS/PPS header
|
int extra_size = 512; // support AVC Intra files without SPS/PPS header
|
||||||
int i, frame_size, slice_type, intra_only = 0;
|
int i, frame_size, slice_type, has_sps = 0, intra_only = 0, ret;
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
buf = avpriv_find_start_code(buf, buf_end, &state);
|
buf = avpriv_find_start_code(buf, buf_end, &state);
|
||||||
@ -2193,11 +2193,12 @@ static int mxf_parse_h264_frame(AVFormatContext *s, AVStream *st,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
nal_end = ff_avc_find_startcode(buf, buf_end);
|
nal_end = ff_avc_find_startcode(buf, buf_end);
|
||||||
sps = ff_avc_decode_sps(buf, nal_end - buf);
|
ret = ff_avc_decode_sps(sps, buf, nal_end - buf);
|
||||||
if (!sps) {
|
if (ret < 0) {
|
||||||
av_log(s, AV_LOG_ERROR, "error parsing sps\n");
|
av_log(s, AV_LOG_ERROR, "error parsing sps\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
has_sps = 1;
|
||||||
|
|
||||||
sc->aspect_ratio.num = st->codecpar->width * sps->sar.num;
|
sc->aspect_ratio.num = st->codecpar->width * sps->sar.num;
|
||||||
sc->aspect_ratio.den = st->codecpar->height * sps->sar.den;
|
sc->aspect_ratio.den = st->codecpar->height * sps->sar.den;
|
||||||
@ -2243,7 +2244,7 @@ static int mxf_parse_h264_frame(AVFormatContext *s, AVStream *st,
|
|||||||
if (mxf->header_written)
|
if (mxf->header_written)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
if (!sps)
|
if (!has_sps)
|
||||||
sc->interlaced = st->codecpar->field_order != AV_FIELD_PROGRESSIVE ? 1 : 0;
|
sc->interlaced = st->codecpar->field_order != AV_FIELD_PROGRESSIVE ? 1 : 0;
|
||||||
sc->codec_ul = NULL;
|
sc->codec_ul = NULL;
|
||||||
frame_size = pkt->size + extra_size;
|
frame_size = pkt->size + extra_size;
|
||||||
@ -2260,7 +2261,7 @@ static int mxf_parse_h264_frame(AVFormatContext *s, AVStream *st,
|
|||||||
if (sc->interlaced)
|
if (sc->interlaced)
|
||||||
sc->field_dominance = 1; // top field first is mandatory for AVC Intra
|
sc->field_dominance = 1; // top field first is mandatory for AVC Intra
|
||||||
break;
|
break;
|
||||||
} else if (sps && mxf_h264_codec_uls[i].frame_size == 0 &&
|
} else if (has_sps && mxf_h264_codec_uls[i].frame_size == 0 &&
|
||||||
mxf_h264_codec_uls[i].profile == sps->profile_idc &&
|
mxf_h264_codec_uls[i].profile == sps->profile_idc &&
|
||||||
(mxf_h264_codec_uls[i].intra_only < 0 ||
|
(mxf_h264_codec_uls[i].intra_only < 0 ||
|
||||||
mxf_h264_codec_uls[i].intra_only == intra_only)) {
|
mxf_h264_codec_uls[i].intra_only == intra_only)) {
|
||||||
@ -2271,8 +2272,6 @@ static int mxf_parse_h264_frame(AVFormatContext *s, AVStream *st,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
av_free(sps);
|
|
||||||
|
|
||||||
if (!sc->codec_ul) {
|
if (!sc->codec_ul) {
|
||||||
av_log(s, AV_LOG_ERROR, "h264 profile not supported\n");
|
av_log(s, AV_LOG_ERROR, "h264 profile not supported\n");
|
||||||
return 0;
|
return 0;
|
||||||
|
Loading…
Reference in New Issue
Block a user