| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /* | ||
| 2 | * Audio and Video frame extraction | ||
| 3 | * Copyright (c) 2003 Fabrice Bellard | ||
| 4 | * Copyright (c) 2003 Michael Niedermayer | ||
| 5 | * | ||
| 6 | * This file is part of FFmpeg. | ||
| 7 | * | ||
| 8 | * FFmpeg is free software; you can redistribute it and/or | ||
| 9 | * modify it under the terms of the GNU Lesser General Public | ||
| 10 | * License as published by the Free Software Foundation; either | ||
| 11 | * version 2.1 of the License, or (at your option) any later version. | ||
| 12 | * | ||
| 13 | * FFmpeg is distributed in the hope that it will be useful, | ||
| 14 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 15 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 16 | * Lesser General Public License for more details. | ||
| 17 | * | ||
| 18 | * You should have received a copy of the GNU Lesser General Public | ||
| 19 | * License along with FFmpeg; if not, write to the Free Software | ||
| 20 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
| 21 | */ | ||
| 22 | |||
| 23 | #include <inttypes.h> | ||
| 24 | #include <stdint.h> | ||
| 25 | #include <string.h> | ||
| 26 | |||
| 27 | #include "libavutil/attributes.h" | ||
| 28 | #include "libavutil/avassert.h" | ||
| 29 | #include "libavutil/mem.h" | ||
| 30 | |||
| 31 | #include "codec_desc.h" | ||
| 32 | #include "parser.h" | ||
| 33 | #include "parser_internal.h" | ||
| 34 | |||
| 35 | #if FF_API_PARSER_CODECID | ||
| 36 | 10998 | av_cold AVCodecParserContext *av_parser_init(int codec_id) | |
| 37 | #else | ||
| 38 | av_cold AVCodecParserContext *av_parser_init(enum AVCodecID codec_id) | ||
| 39 | #endif | ||
| 40 | { | ||
| 41 | 10998 | AVCodecParserContext *s = NULL; | |
| 42 | const AVCodecParser *parser; | ||
| 43 | const FFCodecParser *ffparser; | ||
| 44 | 10998 | void *i = 0; | |
| 45 | int ret; | ||
| 46 | |||
| 47 |
2/2✓ Branch 0 taken 64 times.
✓ Branch 1 taken 10934 times.
|
10998 | if (codec_id == AV_CODEC_ID_NONE) |
| 48 | 64 | return NULL; | |
| 49 | |||
| 50 |
2/2✓ Branch 1 taken 564811 times.
✓ Branch 2 taken 3489 times.
|
568300 | while ((parser = av_parser_iterate(&i))) { |
| 51 |
2/2✓ Branch 0 taken 561444 times.
✓ Branch 1 taken 3367 times.
|
564811 | if (parser->codec_ids[0] == codec_id || |
| 52 |
2/2✓ Branch 0 taken 557547 times.
✓ Branch 1 taken 3897 times.
|
561444 | parser->codec_ids[1] == codec_id || |
| 53 |
2/2✓ Branch 0 taken 557386 times.
✓ Branch 1 taken 161 times.
|
557547 | parser->codec_ids[2] == codec_id || |
| 54 |
2/2✓ Branch 0 taken 557378 times.
✓ Branch 1 taken 8 times.
|
557386 | parser->codec_ids[3] == codec_id || |
| 55 |
2/2✓ Branch 0 taken 557370 times.
✓ Branch 1 taken 8 times.
|
557378 | parser->codec_ids[4] == codec_id || |
| 56 |
2/2✓ Branch 0 taken 557366 times.
✓ Branch 1 taken 4 times.
|
557370 | parser->codec_ids[5] == codec_id || |
| 57 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 557366 times.
|
557366 | parser->codec_ids[6] == codec_id) |
| 58 | 7445 | goto found; | |
| 59 | } | ||
| 60 | 3489 | return NULL; | |
| 61 | |||
| 62 | 7445 | found: | |
| 63 | 7445 | ffparser = ffcodecparser(parser); | |
| 64 | 7445 | s = av_mallocz(sizeof(AVCodecParserContext)); | |
| 65 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 7445 times.
|
7445 | if (!s) |
| 66 | ✗ | goto err_out; | |
| 67 | 7445 | s->parser = parser; | |
| 68 | 7445 | s->priv_data = av_mallocz(ffparser->priv_data_size); | |
| 69 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 7445 times.
|
7445 | if (!s->priv_data) |
| 70 | ✗ | goto err_out; | |
| 71 | 7445 | s->fetch_timestamp=1; | |
| 72 | 7445 | s->pict_type = AV_PICTURE_TYPE_I; | |
| 73 |
2/2✓ Branch 0 taken 2322 times.
✓ Branch 1 taken 5123 times.
|
7445 | if (ffparser->init) { |
| 74 | 2322 | ret = ffparser->init(s); | |
| 75 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 2322 times.
|
2322 | if (ret != 0) |
| 76 | ✗ | goto err_out; | |
| 77 | } | ||
| 78 | 7445 | s->key_frame = -1; | |
| 79 | 7445 | s->dts_sync_point = INT_MIN; | |
| 80 | 7445 | s->dts_ref_dts_delta = INT_MIN; | |
| 81 | 7445 | s->pts_dts_delta = INT_MIN; | |
| 82 | 7445 | s->format = -1; | |
| 83 | |||
| 84 | 7445 | return s; | |
| 85 | |||
| 86 | ✗ | err_out: | |
| 87 | ✗ | if (s) | |
| 88 | ✗ | av_freep(&s->priv_data); | |
| 89 | ✗ | av_free(s); | |
| 90 | ✗ | return NULL; | |
| 91 | } | ||
| 92 | |||
| 93 | 242456 | void ff_fetch_timestamp(AVCodecParserContext *s, int off, int remove, int fuzzy) | |
| 94 | { | ||
| 95 | int i; | ||
| 96 | |||
| 97 |
2/2✓ Branch 0 taken 241617 times.
✓ Branch 1 taken 839 times.
|
242456 | if (!fuzzy) { |
| 98 | 241617 | s->dts = | |
| 99 | 241617 | s->pts = AV_NOPTS_VALUE; | |
| 100 | 241617 | s->pos = -1; | |
| 101 | 241617 | s->offset = 0; | |
| 102 | } | ||
| 103 |
2/2✓ Branch 0 taken 750809 times.
✓ Branch 1 taken 97550 times.
|
848359 | for (i = 0; i < AV_PARSER_PTS_NB; i++) { |
| 104 |
2/2✓ Branch 0 taken 741066 times.
✓ Branch 1 taken 9743 times.
|
750809 | if (s->cur_offset + off >= s->cur_frame_offset[i] && |
| 105 |
2/2✓ Branch 0 taken 540791 times.
✓ Branch 1 taken 200275 times.
|
741066 | (s->frame_offset < s->cur_frame_offset[i] || |
| 106 |
4/4✓ Branch 0 taken 5954 times.
✓ Branch 1 taken 534837 times.
✓ Branch 2 taken 2192 times.
✓ Branch 3 taken 3762 times.
|
540791 | (!s->frame_offset && !s->next_frame_offset)) && // first field/frame |
| 107 | // check disabled since MPEG-TS does not send complete PES packets | ||
| 108 |
2/2✓ Branch 0 taken 201351 times.
✓ Branch 1 taken 1116 times.
|
202467 | /*s->next_frame_offset + off <*/ s->cur_frame_end[i]){ |
| 109 | |||
| 110 |
4/4✓ Branch 0 taken 1135 times.
✓ Branch 1 taken 200216 times.
✓ Branch 2 taken 262 times.
✓ Branch 3 taken 873 times.
|
201351 | if (!fuzzy || s->cur_frame_dts[i] != AV_NOPTS_VALUE) { |
| 111 | 200478 | s->dts = s->cur_frame_dts[i]; | |
| 112 | 200478 | s->pts = s->cur_frame_pts[i]; | |
| 113 | 200478 | s->pos = s->cur_frame_pos[i]; | |
| 114 | 200478 | s->offset = s->next_frame_offset - s->cur_frame_offset[i]; | |
| 115 | } | ||
| 116 |
2/2✓ Branch 0 taken 12690 times.
✓ Branch 1 taken 188661 times.
|
201351 | if (remove) |
| 117 | 12690 | s->cur_frame_offset[i] = INT64_MAX; | |
| 118 |
2/2✓ Branch 0 taken 144906 times.
✓ Branch 1 taken 56445 times.
|
201351 | if (s->cur_offset + off < s->cur_frame_end[i]) |
| 119 | 144906 | break; | |
| 120 | } | ||
| 121 | } | ||
| 122 | 242456 | } | |
| 123 | |||
| 124 | 872718 | int av_parser_parse2(AVCodecParserContext *s, AVCodecContext *avctx, | |
| 125 | uint8_t **poutbuf, int *poutbuf_size, | ||
| 126 | const uint8_t *buf, int buf_size, | ||
| 127 | int64_t pts, int64_t dts, int64_t pos) | ||
| 128 | { | ||
| 129 | const AVCodecDescriptor *desc; | ||
| 130 | int index, i; | ||
| 131 | uint8_t dummy_buf[AV_INPUT_BUFFER_PADDING_SIZE]; | ||
| 132 | |||
| 133 | av_assert1(avctx->codec_id != AV_CODEC_ID_NONE); | ||
| 134 | |||
| 135 | /* Parsers only work for the specified codec ids. */ | ||
| 136 | av_assert1(avctx->codec_id == s->parser->codec_ids[0] || | ||
| 137 | avctx->codec_id == s->parser->codec_ids[1] || | ||
| 138 | avctx->codec_id == s->parser->codec_ids[2] || | ||
| 139 | avctx->codec_id == s->parser->codec_ids[3] || | ||
| 140 | avctx->codec_id == s->parser->codec_ids[4] || | ||
| 141 | avctx->codec_id == s->parser->codec_ids[5] || | ||
| 142 | avctx->codec_id == s->parser->codec_ids[6]); | ||
| 143 | |||
| 144 | 872718 | desc = avcodec_descriptor_get(avctx->codec_id); | |
| 145 | |||
| 146 |
2/2✓ Branch 0 taken 3692 times.
✓ Branch 1 taken 869026 times.
|
872718 | if (!(s->flags & PARSER_FLAG_FETCHED_OFFSET)) { |
| 147 | 3692 | s->next_frame_offset = | |
| 148 | 3692 | s->cur_offset = pos; | |
| 149 | 3692 | s->flags |= PARSER_FLAG_FETCHED_OFFSET; | |
| 150 | } | ||
| 151 | |||
| 152 |
2/2✓ Branch 0 taken 3498 times.
✓ Branch 1 taken 869220 times.
|
872718 | if (buf_size == 0) { |
| 153 | /* padding is always necessary even if EOF, so we add it here */ | ||
| 154 | 3498 | memset(dummy_buf, 0, sizeof(dummy_buf)); | |
| 155 | 3498 | buf = dummy_buf; | |
| 156 |
2/2✓ Branch 0 taken 709243 times.
✓ Branch 1 taken 159977 times.
|
869220 | } else if (s->cur_offset + buf_size != s->cur_frame_end[s->cur_frame_start_index]) { /* skip remainder packets */ |
| 157 | /* add a new packet descriptor */ | ||
| 158 | 709243 | i = (s->cur_frame_start_index + 1) & (AV_PARSER_PTS_NB - 1); | |
| 159 | 709243 | s->cur_frame_start_index = i; | |
| 160 | 709243 | s->cur_frame_offset[i] = s->cur_offset; | |
| 161 | 709243 | s->cur_frame_end[i] = s->cur_offset + buf_size; | |
| 162 | 709243 | s->cur_frame_pts[i] = pts; | |
| 163 | 709243 | s->cur_frame_dts[i] = dts; | |
| 164 | 709243 | s->cur_frame_pos[i] = pos; | |
| 165 | } | ||
| 166 | |||
| 167 |
2/2✓ Branch 0 taken 235145 times.
✓ Branch 1 taken 637573 times.
|
872718 | if (s->fetch_timestamp) { |
| 168 | 235145 | s->fetch_timestamp = 0; | |
| 169 | 235145 | s->last_pts = s->pts; | |
| 170 | 235145 | s->last_dts = s->dts; | |
| 171 | 235145 | s->last_pos = s->pos; | |
| 172 | 235145 | ff_fetch_timestamp(s, 0, 0, 0); | |
| 173 | } | ||
| 174 | /* WARNING: the returned index can be negative */ | ||
| 175 | 872718 | index = ffcodecparser(s->parser)->parse(s, avctx, (const uint8_t **) poutbuf, | |
| 176 | poutbuf_size, buf, buf_size); | ||
| 177 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 872718 times.
|
872718 | av_assert0(index > -0x20000000); // The API does not allow returning AVERROR codes |
| 178 | #define FILL(name) if(s->name > 0 && avctx->name <= 0) avctx->name = s->name | ||
| 179 |
2/2✓ Branch 0 taken 634636 times.
✓ Branch 1 taken 238082 times.
|
872718 | if (avctx->codec_type == AVMEDIA_TYPE_VIDEO) { |
| 180 |
4/4✓ Branch 0 taken 240727 times.
✓ Branch 1 taken 393909 times.
✓ Branch 2 taken 497 times.
✓ Branch 3 taken 240230 times.
|
634636 | FILL(field_order); |
| 181 |
4/4✓ Branch 0 taken 340590 times.
✓ Branch 1 taken 294046 times.
✓ Branch 2 taken 907 times.
✓ Branch 3 taken 339683 times.
|
634636 | FILL(coded_width); |
| 182 |
4/4✓ Branch 0 taken 340590 times.
✓ Branch 1 taken 294046 times.
✓ Branch 2 taken 907 times.
✓ Branch 3 taken 339683 times.
|
634636 | FILL(coded_height); |
| 183 |
4/4✓ Branch 0 taken 344278 times.
✓ Branch 1 taken 290358 times.
✓ Branch 2 taken 565 times.
✓ Branch 3 taken 343713 times.
|
634636 | FILL(width); |
| 184 |
4/4✓ Branch 0 taken 344278 times.
✓ Branch 1 taken 290358 times.
✓ Branch 2 taken 565 times.
✓ Branch 3 taken 343713 times.
|
634636 | FILL(height); |
| 185 |
3/4✓ Branch 0 taken 634636 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 549 times.
✓ Branch 3 taken 634087 times.
|
634636 | if (desc && (desc->props & AV_CODEC_PROP_ENHANCEMENT) && |
| 186 |
4/4✓ Branch 0 taken 540 times.
✓ Branch 1 taken 9 times.
✓ Branch 2 taken 6 times.
✓ Branch 3 taken 534 times.
|
549 | s->format >= 0 && avctx->pix_fmt < 0) avctx->pix_fmt = s->format; |
| 187 | } | ||
| 188 | |||
| 189 | /* update the file pointer */ | ||
| 190 |
2/2✓ Branch 0 taken 232458 times.
✓ Branch 1 taken 640260 times.
|
872718 | if (*poutbuf_size) { |
| 191 | /* fill the data for the current frame */ | ||
| 192 | 232458 | s->frame_offset = s->next_frame_offset; | |
| 193 | |||
| 194 | /* offset of the next frame */ | ||
| 195 | 232458 | s->next_frame_offset = s->cur_offset + index; | |
| 196 | 232458 | s->fetch_timestamp = 1; | |
| 197 | } else { | ||
| 198 | /* Don't return a pointer to dummy_buf. */ | ||
| 199 | 640260 | *poutbuf = NULL; | |
| 200 | } | ||
| 201 |
2/2✓ Branch 0 taken 5007 times.
✓ Branch 1 taken 867711 times.
|
872718 | if (index < 0) |
| 202 | 5007 | index = 0; | |
| 203 | 872718 | s->cur_offset += index; | |
| 204 | 872718 | return index; | |
| 205 | } | ||
| 206 | |||
| 207 | 21543 | av_cold void av_parser_close(AVCodecParserContext *s) | |
| 208 | { | ||
| 209 |
2/2✓ Branch 0 taken 7445 times.
✓ Branch 1 taken 14098 times.
|
21543 | if (s) { |
| 210 |
2/2✓ Branch 1 taken 7062 times.
✓ Branch 2 taken 383 times.
|
7445 | if (ffcodecparser(s->parser)->close) |
| 211 | 7062 | ffcodecparser(s->parser)->close(s); | |
| 212 | 7445 | av_freep(&s->priv_data); | |
| 213 | 7445 | av_free(s); | |
| 214 | } | ||
| 215 | 21543 | } | |
| 216 | |||
| 217 | 809814 | int ff_combine_frame(ParseContext *pc, int next, | |
| 218 | const uint8_t **buf, int *buf_size) | ||
| 219 | { | ||
| 220 | 809814 | if (pc->overread) { | |
| 221 | ff_dlog(NULL, "overread %d, state:%"PRIX32" next:%d index:%d o_index:%d\n", | ||
| 222 | pc->overread, pc->state, next, pc->index, pc->overread_index); | ||
| 223 | ff_dlog(NULL, "%X %X %X %X\n", | ||
| 224 | (*buf)[0], (*buf)[1], (*buf)[2], (*buf)[3]); | ||
| 225 | } | ||
| 226 | |||
| 227 | /* Copy overread bytes from last frame into buffer. */ | ||
| 228 |
2/2✓ Branch 0 taken 1391 times.
✓ Branch 1 taken 809814 times.
|
811205 | for (; pc->overread > 0; pc->overread--) |
| 229 | 1391 | pc->buffer[pc->index++] = pc->buffer[pc->overread_index++]; | |
| 230 | |||
| 231 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 809814 times.
|
809814 | if (next > *buf_size) |
| 232 | ✗ | return AVERROR(EINVAL); | |
| 233 | |||
| 234 | /* flush remaining if EOF */ | ||
| 235 |
4/4✓ Branch 0 taken 2320 times.
✓ Branch 1 taken 807494 times.
✓ Branch 2 taken 1857 times.
✓ Branch 3 taken 463 times.
|
809814 | if (!*buf_size && next == END_NOT_FOUND) |
| 236 | 1857 | next = 0; | |
| 237 | |||
| 238 | 809814 | pc->last_index = pc->index; | |
| 239 | |||
| 240 | /* copy into buffer end return */ | ||
| 241 |
2/2✓ Branch 0 taken 629913 times.
✓ Branch 1 taken 179901 times.
|
809814 | if (next == END_NOT_FOUND) { |
| 242 | 629913 | void *new_buffer = av_fast_realloc(pc->buffer, &pc->buffer_size, | |
| 243 | 629913 | *buf_size + pc->index + | |
| 244 | AV_INPUT_BUFFER_PADDING_SIZE); | ||
| 245 | |||
| 246 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 629913 times.
|
629913 | if (!new_buffer) { |
| 247 | ✗ | av_log(NULL, AV_LOG_ERROR, "Failed to reallocate parser buffer to %d\n", *buf_size + pc->index + AV_INPUT_BUFFER_PADDING_SIZE); | |
| 248 | ✗ | pc->index = 0; | |
| 249 | ✗ | return AVERROR(ENOMEM); | |
| 250 | } | ||
| 251 | 629913 | pc->buffer = new_buffer; | |
| 252 | 629913 | memcpy(&pc->buffer[pc->index], *buf, *buf_size); | |
| 253 | 629913 | memset(&pc->buffer[pc->index + *buf_size], 0, AV_INPUT_BUFFER_PADDING_SIZE); | |
| 254 | 629913 | pc->index += *buf_size; | |
| 255 | 629913 | return -1; | |
| 256 | } | ||
| 257 | |||
| 258 |
3/4✓ Branch 0 taken 284 times.
✓ Branch 1 taken 179617 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 284 times.
|
179901 | av_assert0(next >= 0 || pc->buffer); |
| 259 | |||
| 260 | 179901 | *buf_size = | |
| 261 | 179901 | pc->overread_index = pc->index + next; | |
| 262 | |||
| 263 | /* append to buffer */ | ||
| 264 |
2/2✓ Branch 0 taken 63741 times.
✓ Branch 1 taken 116160 times.
|
179901 | if (pc->index) { |
| 265 | 63741 | void *new_buffer = av_fast_realloc(pc->buffer, &pc->buffer_size, | |
| 266 | 63741 | next + pc->index + | |
| 267 | AV_INPUT_BUFFER_PADDING_SIZE); | ||
| 268 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 63741 times.
|
63741 | if (!new_buffer) { |
| 269 | ✗ | av_log(NULL, AV_LOG_ERROR, "Failed to reallocate parser buffer to %d\n", next + pc->index + AV_INPUT_BUFFER_PADDING_SIZE); | |
| 270 | ✗ | *buf_size = | |
| 271 | ✗ | pc->overread_index = | |
| 272 | ✗ | pc->index = 0; | |
| 273 | ✗ | return AVERROR(ENOMEM); | |
| 274 | } | ||
| 275 | 63741 | pc->buffer = new_buffer; | |
| 276 |
2/2✓ Branch 0 taken 63736 times.
✓ Branch 1 taken 5 times.
|
63741 | if (next > -AV_INPUT_BUFFER_PADDING_SIZE) |
| 277 | 63736 | memcpy(&pc->buffer[pc->index], *buf, | |
| 278 | 63736 | next + AV_INPUT_BUFFER_PADDING_SIZE); | |
| 279 | 63741 | pc->index = 0; | |
| 280 | 63741 | *buf = pc->buffer; | |
| 281 | } | ||
| 282 | |||
| 283 |
2/2✓ Branch 0 taken 38 times.
✓ Branch 1 taken 179863 times.
|
179901 | if (next < -8) { |
| 284 | 38 | pc->overread += -8 - next; | |
| 285 | 38 | next = -8; | |
| 286 | } | ||
| 287 | /* store overread bytes */ | ||
| 288 |
2/2✓ Branch 0 taken 996 times.
✓ Branch 1 taken 179901 times.
|
180897 | for (; next < 0; next++) { |
| 289 | 996 | pc->state = pc->state << 8 | pc->buffer[pc->last_index + next]; | |
| 290 | 996 | pc->state64 = pc->state64 << 8 | pc->buffer[pc->last_index + next]; | |
| 291 | 996 | pc->overread++; | |
| 292 | } | ||
| 293 | |||
| 294 | 179901 | if (pc->overread) { | |
| 295 | ff_dlog(NULL, "overread %d, state:%"PRIX32" next:%d index:%d o_index:%d\n", | ||
| 296 | pc->overread, pc->state, next, pc->index, pc->overread_index); | ||
| 297 | ff_dlog(NULL, "%X %X %X %X\n", | ||
| 298 | (*buf)[0], (*buf)[1], (*buf)[2], (*buf)[3]); | ||
| 299 | } | ||
| 300 | |||
| 301 | 179901 | return 0; | |
| 302 | } | ||
| 303 | |||
| 304 | 5672 | av_cold void ff_parse_close(AVCodecParserContext *s) | |
| 305 | { | ||
| 306 | 5672 | ParseContext *pc = s->priv_data; | |
| 307 | |||
| 308 | 5672 | av_freep(&pc->buffer); | |
| 309 | 5672 | } | |
| 310 |