Line | Branch | Exec | Source |
---|---|---|---|
1 | /* | ||
2 | * MPEG Audio parser | ||
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 "parser.h" | ||
24 | #include "mpegaudiodecheader.h" | ||
25 | #include "libavutil/common.h" | ||
26 | #include "libavformat/apetag.h" // for APE tag. | ||
27 | #include "libavformat/id3v1.h" // for ID3v1_TAG_SIZE | ||
28 | |||
29 | typedef struct MpegAudioParseContext { | ||
30 | ParseContext pc; | ||
31 | int frame_size; | ||
32 | uint32_t header; | ||
33 | int header_count; | ||
34 | int no_bitrate; | ||
35 | } MpegAudioParseContext; | ||
36 | |||
37 | #define MPA_HEADER_SIZE 4 | ||
38 | |||
39 | /* header + layer + freq + lsf/mpeg25 */ | ||
40 | #define SAME_HEADER_MASK \ | ||
41 | (0xffe00000 | (3 << 17) | (3 << 10) | (3 << 19)) | ||
42 | |||
43 | 21340 | static int mpegaudio_parse(AVCodecParserContext *s1, | |
44 | AVCodecContext *avctx, | ||
45 | const uint8_t **poutbuf, int *poutbuf_size, | ||
46 | const uint8_t *buf, int buf_size) | ||
47 | { | ||
48 | 21340 | MpegAudioParseContext *s = s1->priv_data; | |
49 | 21340 | ParseContext *pc = &s->pc; | |
50 | 21340 | uint32_t state= pc->state; | |
51 | int i; | ||
52 | 21340 | int next= END_NOT_FOUND; | |
53 | 21340 | int flush = !buf_size; | |
54 | |||
55 |
2/2✓ Branch 0 taken 39379 times.
✓ Branch 1 taken 3605 times.
|
42984 | for(i=0; i<buf_size; ){ |
56 |
2/2✓ Branch 0 taken 20950 times.
✓ Branch 1 taken 18429 times.
|
39379 | if(s->frame_size){ |
57 | 20950 | int inc= FFMIN(buf_size - i, s->frame_size); | |
58 | 20950 | i += inc; | |
59 | 20950 | s->frame_size -= inc; | |
60 | 20950 | state = 0; | |
61 | |||
62 |
2/2✓ Branch 0 taken 17735 times.
✓ Branch 1 taken 3215 times.
|
20950 | if(!s->frame_size){ |
63 | 17735 | next= i; | |
64 | 17735 | break; | |
65 | } | ||
66 | }else{ | ||
67 |
2/2✓ Branch 0 taken 222052 times.
✓ Branch 1 taken 263 times.
|
222315 | while(i<buf_size){ |
68 | int ret, sr, channels, bit_rate, frame_size; | ||
69 | 222052 | enum AVCodecID codec_id = avctx->codec_id; | |
70 | |||
71 | 222052 | state= (state<<8) + buf[i++]; | |
72 | |||
73 | 222052 | ret = ff_mpa_decode_header(state, &sr, &channels, &frame_size, &bit_rate, &codec_id); | |
74 |
2/2✓ Branch 0 taken 203886 times.
✓ Branch 1 taken 18166 times.
|
222052 | if (ret < 4) { |
75 |
2/2✓ Branch 0 taken 149486 times.
✓ Branch 1 taken 54400 times.
|
203886 | if (i > 4) |
76 | 149486 | s->header_count = -2; | |
77 | } else { | ||
78 |
3/4✓ Branch 0 taken 18166 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 94 times.
✓ Branch 3 taken 18072 times.
|
18166 | int header_threshold = avctx->codec_id != AV_CODEC_ID_NONE && avctx->codec_id != codec_id; |
79 |
4/4✓ Branch 0 taken 414 times.
✓ Branch 1 taken 17752 times.
✓ Branch 2 taken 64 times.
✓ Branch 3 taken 350 times.
|
18166 | if((state&SAME_HEADER_MASK) != (s->header&SAME_HEADER_MASK) && s->header) |
80 | 64 | s->header_count= -3; | |
81 | 18166 | s->header= state; | |
82 | 18166 | s->header_count++; | |
83 | 18166 | s->frame_size = ret-4; | |
84 | |||
85 |
2/2✓ Branch 0 taken 17822 times.
✓ Branch 1 taken 344 times.
|
18166 | if (s->header_count > header_threshold) { |
86 | 17822 | avctx->sample_rate= sr; | |
87 | 17822 | av_channel_layout_uninit(&avctx->ch_layout); | |
88 | 17822 | av_channel_layout_default(&avctx->ch_layout, channels); | |
89 | 17822 | s1->duration = frame_size; | |
90 | 17822 | avctx->codec_id = codec_id; | |
91 |
4/4✓ Branch 0 taken 5782 times.
✓ Branch 1 taken 12040 times.
✓ Branch 2 taken 72 times.
✓ Branch 3 taken 5710 times.
|
17822 | if (s->no_bitrate || !avctx->bit_rate) { |
92 | 12112 | s->no_bitrate = 1; | |
93 | 12112 | avctx->bit_rate += (bit_rate - avctx->bit_rate) / (s->header_count - header_threshold); | |
94 | } | ||
95 | } | ||
96 | |||
97 |
2/2✓ Branch 0 taken 302 times.
✓ Branch 1 taken 17864 times.
|
18166 | if (s1->flags & PARSER_FLAG_COMPLETE_FRAMES) { |
98 | 302 | s->frame_size = 0; | |
99 | 302 | next = buf_size; | |
100 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 17864 times.
|
17864 | } else if (codec_id == AV_CODEC_ID_MP3ADU) { |
101 | ✗ | avpriv_report_missing_feature(avctx, | |
102 | "MP3ADU full parser"); | ||
103 | ✗ | *poutbuf = NULL; | |
104 | ✗ | *poutbuf_size = 0; | |
105 | ✗ | return buf_size; /* parsers must not return error codes */ | |
106 | } | ||
107 | |||
108 | 18166 | break; | |
109 | } | ||
110 | } | ||
111 | } | ||
112 | } | ||
113 | |||
114 | 21340 | pc->state= state; | |
115 |
2/2✓ Branch 1 taken 3226 times.
✓ Branch 2 taken 18114 times.
|
21340 | if (ff_combine_frame(pc, next, &buf, &buf_size) < 0) { |
116 | 3226 | *poutbuf = NULL; | |
117 | 3226 | *poutbuf_size = 0; | |
118 | 3226 | return buf_size; | |
119 | } | ||
120 | |||
121 |
6/6✓ Branch 0 taken 124 times.
✓ Branch 1 taken 17990 times.
✓ Branch 2 taken 20 times.
✓ Branch 3 taken 104 times.
✓ Branch 4 taken 6 times.
✓ Branch 5 taken 14 times.
|
18114 | if (flush && buf_size >= ID3v1_TAG_SIZE && memcmp(buf, "TAG", 3) == 0) { |
122 | 6 | *poutbuf = NULL; | |
123 | 6 | *poutbuf_size = 0; | |
124 | 6 | return next; | |
125 | } | ||
126 | |||
127 |
5/6✓ Branch 0 taken 118 times.
✓ Branch 1 taken 17990 times.
✓ Branch 2 taken 15 times.
✓ Branch 3 taken 103 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 15 times.
|
18108 | if (flush && buf_size >= APE_TAG_FOOTER_BYTES && memcmp(buf, APE_TAG_PREAMBLE, 8) == 0) { |
128 | ✗ | *poutbuf = NULL; | |
129 | ✗ | *poutbuf_size = 0; | |
130 | ✗ | return next; | |
131 | } | ||
132 | |||
133 | 18108 | *poutbuf = buf; | |
134 | 18108 | *poutbuf_size = buf_size; | |
135 | 18108 | return next; | |
136 | } | ||
137 | |||
138 | |||
139 | const AVCodecParser ff_mpegaudio_parser = { | ||
140 | .codec_ids = { AV_CODEC_ID_MP1, AV_CODEC_ID_MP2, AV_CODEC_ID_MP3, AV_CODEC_ID_MP3ADU }, | ||
141 | .priv_data_size = sizeof(MpegAudioParseContext), | ||
142 | .parser_parse = mpegaudio_parse, | ||
143 | .parser_close = ff_parse_close, | ||
144 | }; | ||
145 |