Line | Branch | Exec | Source |
---|---|---|---|
1 | /* | ||
2 | * Copyright (c) 2015 Ludmila Glinskih | ||
3 | * | ||
4 | * Permission is hereby granted, free of charge, to any person obtaining a copy | ||
5 | * of this software and associated documentation files (the "Software"), to deal | ||
6 | * in the Software without restriction, including without limitation the rights | ||
7 | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||
8 | * copies of the Software, and to permit persons to whom the Software is | ||
9 | * furnished to do so, subject to the following conditions: | ||
10 | * | ||
11 | * The above copyright notice and this permission notice shall be included in | ||
12 | * all copies or substantial portions of the Software. | ||
13 | * | ||
14 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||
15 | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||
16 | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | ||
17 | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||
18 | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||
19 | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||
20 | * THE SOFTWARE. | ||
21 | */ | ||
22 | |||
23 | /** | ||
24 | * Seek test. | ||
25 | */ | ||
26 | |||
27 | #include "libavutil/adler32.h" | ||
28 | #include "libavutil/mem.h" | ||
29 | #include "libavcodec/avcodec.h" | ||
30 | #include "libavformat/avformat.h" | ||
31 | #include "libavutil/imgutils.h" | ||
32 | |||
33 | int64_t *pts_array; | ||
34 | uint32_t *crc_array; | ||
35 | int size_of_array; | ||
36 | int number_of_elements; | ||
37 | |||
38 | 25 | static int add_crc_to_array(uint32_t crc, int64_t pts) | |
39 | { | ||
40 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 23 times.
|
25 | if (size_of_array <= number_of_elements) { |
41 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 1 times.
|
2 | if (size_of_array == 0) |
42 | 1 | size_of_array = 10; | |
43 | 2 | size_of_array *= 2; | |
44 | 2 | crc_array = av_realloc_f(crc_array, size_of_array, sizeof(uint32_t)); | |
45 | 2 | pts_array = av_realloc_f(pts_array, size_of_array, sizeof(int64_t)); | |
46 |
2/4✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
|
2 | if ((crc_array == NULL) || (pts_array == NULL)) { |
47 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't allocate array to store crcs\n"); | |
48 | ✗ | return AVERROR(ENOMEM); | |
49 | } | ||
50 | } | ||
51 | 25 | crc_array[number_of_elements] = crc; | |
52 | 25 | pts_array[number_of_elements] = pts; | |
53 | 25 | number_of_elements++; | |
54 | 25 | return 0; | |
55 | } | ||
56 | |||
57 | 19 | static int compare_crc_in_array(uint32_t crc, int64_t pts) | |
58 | { | ||
59 | int i; | ||
60 |
1/2✓ Branch 0 taken 163 times.
✗ Branch 1 not taken.
|
163 | for (i = 0; i < number_of_elements; i++) { |
61 |
2/2✓ Branch 0 taken 19 times.
✓ Branch 1 taken 144 times.
|
163 | if (pts_array[i] == pts) { |
62 |
1/2✓ Branch 0 taken 19 times.
✗ Branch 1 not taken.
|
19 | if (crc_array[i] == crc) { |
63 | 19 | printf("Comparing 0x%08"PRIx32" %"PRId64" %d is OK\n", crc, pts, i); | |
64 | 19 | return 0; | |
65 | } | ||
66 | else { | ||
67 | ✗ | av_log(NULL, AV_LOG_ERROR, "Incorrect crc of a frame after seeking\n"); | |
68 | ✗ | return -1; | |
69 | } | ||
70 | } | ||
71 | } | ||
72 | ✗ | av_log(NULL, AV_LOG_ERROR, "Incorrect pts of a frame after seeking\n"); | |
73 | ✗ | return -1; | |
74 | } | ||
75 | |||
76 | 29 | static int compute_crc_of_packets(AVFormatContext *fmt_ctx, int video_stream, | |
77 | AVCodecContext *ctx, AVPacket *pkt, AVFrame *fr, | ||
78 | uint64_t ts_start, uint64_t ts_end, int no_seeking) | ||
79 | { | ||
80 | int number_of_written_bytes; | ||
81 | int result; | ||
82 | int byte_buffer_size; | ||
83 | uint8_t *byte_buffer; | ||
84 | uint32_t crc; | ||
85 | |||
86 | 29 | byte_buffer_size = av_image_get_buffer_size(ctx->pix_fmt, ctx->width, ctx->height, 16); | |
87 | 29 | byte_buffer = av_malloc(byte_buffer_size); | |
88 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 29 times.
|
29 | if (!byte_buffer) { |
89 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't allocate buffer\n"); | |
90 | ✗ | return AVERROR(ENOMEM); | |
91 | } | ||
92 | |||
93 |
2/2✓ Branch 0 taken 28 times.
✓ Branch 1 taken 1 times.
|
29 | if (!no_seeking) { |
94 | 28 | result = av_seek_frame(fmt_ctx, video_stream, ts_start, AVSEEK_FLAG_ANY); | |
95 | 28 | printf("Seeking to %"PRId64", computing crc for frames with pts < %"PRId64"\n", ts_start, ts_end); | |
96 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 28 times.
|
28 | if (result < 0) { |
97 | ✗ | av_log(NULL, AV_LOG_ERROR, "Error in seeking\n"); | |
98 | ✗ | return result; | |
99 | } | ||
100 | 28 | avcodec_flush_buffers(ctx); | |
101 | } | ||
102 | |||
103 | do { | ||
104 | 54 | result = av_read_frame(fmt_ctx, pkt); | |
105 |
3/4✓ Branch 0 taken 53 times.
✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 53 times.
|
54 | if (result >= 0 && pkt->stream_index != video_stream) { |
106 | ✗ | av_packet_unref(pkt); | |
107 | ✗ | continue; | |
108 | } | ||
109 | |||
110 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 53 times.
|
54 | if (result < 0) |
111 | 1 | result = avcodec_send_packet(ctx, NULL); | |
112 | else { | ||
113 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 53 times.
|
53 | if (pkt->pts == AV_NOPTS_VALUE) { |
114 | ✗ | av_log(NULL, AV_LOG_ERROR, "Error: frames doesn't have pts values\n"); | |
115 | ✗ | return -1; | |
116 | } | ||
117 | 53 | result = avcodec_send_packet(ctx, pkt); | |
118 | } | ||
119 | |||
120 | 54 | av_packet_unref(pkt); | |
121 | |||
122 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 54 times.
|
54 | if (result < 0) { |
123 | ✗ | av_log(NULL, AV_LOG_ERROR, "Error submitting a packet for decoding\n"); | |
124 | ✗ | return result; | |
125 | } | ||
126 | |||
127 |
1/2✓ Branch 0 taken 98 times.
✗ Branch 1 not taken.
|
98 | while (result >= 0) { |
128 | 98 | result = avcodec_receive_frame(ctx, fr); | |
129 |
2/2✓ Branch 0 taken 1 times.
✓ Branch 1 taken 97 times.
|
98 | if (result == AVERROR_EOF) |
130 | 1 | goto finish; | |
131 |
2/2✓ Branch 0 taken 44 times.
✓ Branch 1 taken 53 times.
|
97 | else if (result == AVERROR(EAGAIN)) { |
132 | 44 | result = 0; | |
133 | 44 | break; | |
134 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 53 times.
|
53 | } else if (result < 0) { |
135 | ✗ | av_log(NULL, AV_LOG_ERROR, "Error decoding frame\n"); | |
136 | ✗ | return result; | |
137 | } | ||
138 | |||
139 | 53 | number_of_written_bytes = av_image_copy_to_buffer(byte_buffer, byte_buffer_size, | |
140 | 53 | (const uint8_t* const *)fr->data, (const int*) fr->linesize, | |
141 | ctx->pix_fmt, ctx->width, ctx->height, 1); | ||
142 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 53 times.
|
53 | if (number_of_written_bytes < 0) { |
143 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't copy image to buffer\n"); | |
144 | ✗ | return number_of_written_bytes; | |
145 | } | ||
146 |
4/4✓ Branch 0 taken 28 times.
✓ Branch 1 taken 25 times.
✓ Branch 2 taken 9 times.
✓ Branch 3 taken 19 times.
|
53 | if ((!no_seeking) && (fr->pts > ts_end)) |
147 | 9 | break; | |
148 | 44 | crc = av_adler32_update(0, (const uint8_t*)byte_buffer, number_of_written_bytes); | |
149 | 44 | printf("%10"PRId64", 0x%08"PRIx32"\n", fr->pts, crc); | |
150 |
2/2✓ Branch 0 taken 25 times.
✓ Branch 1 taken 19 times.
|
44 | if (no_seeking) { |
151 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 25 times.
|
25 | if (add_crc_to_array(crc, fr->pts) < 0) |
152 | ✗ | return -1; | |
153 | } | ||
154 | else { | ||
155 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 19 times.
|
19 | if (compare_crc_in_array(crc, fr->pts) < 0) |
156 | ✗ | return -1; | |
157 | } | ||
158 | 44 | av_frame_unref(fr); | |
159 | } | ||
160 |
4/6✗ Branch 0 not taken.
✓ Branch 1 taken 53 times.
✓ Branch 2 taken 25 times.
✓ Branch 3 taken 28 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 28 times.
|
53 | } while (result >= 0 && (no_seeking || (fr->pts + fr->duration <= ts_end))); |
161 | |||
162 | 28 | finish: | |
163 | 29 | av_freep(&byte_buffer); | |
164 | |||
165 | 29 | return 0; | |
166 | } | ||
167 | |||
168 | 2 | static long int read_seek_range(const char *string_with_number) | |
169 | { | ||
170 | long int number; | ||
171 | 2 | char *end_of_string = NULL; | |
172 | 2 | number = strtol(string_with_number, &end_of_string, 10); | |
173 |
2/4✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
|
2 | if ((strlen(string_with_number) != end_of_string - string_with_number) || (number < 0)) { |
174 | ✗ | av_log(NULL, AV_LOG_ERROR, "Incorrect input ranges of seeking\n"); | |
175 | ✗ | return -1; | |
176 | } | ||
177 |
2/4✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 2 times.
|
2 | else if ((number == LONG_MAX) || (number == LONG_MIN)) { |
178 | ✗ | if (errno == ERANGE) { | |
179 | ✗ | av_log(NULL, AV_LOG_ERROR, "Incorrect input ranges of seeking\n"); | |
180 | ✗ | return -1; | |
181 | } | ||
182 | } | ||
183 | 2 | return number; | |
184 | } | ||
185 | |||
186 | 1 | static int seek_test(const char *input_filename, const char *start, const char *end) | |
187 | { | ||
188 | 1 | const AVCodec *codec = NULL; | |
189 | 1 | AVCodecContext *ctx= NULL; | |
190 | 1 | AVCodecParameters *origin_par = NULL; | |
191 | 1 | AVPacket *pkt = NULL; | |
192 | 1 | AVFrame *fr = NULL; | |
193 | 1 | AVFormatContext *fmt_ctx = NULL; | |
194 | int video_stream; | ||
195 | int result; | ||
196 | int i, j; | ||
197 | long int start_ts, end_ts; | ||
198 | |||
199 | 1 | size_of_array = 0; | |
200 | 1 | number_of_elements = 0; | |
201 | 1 | crc_array = NULL; | |
202 | 1 | pts_array = NULL; | |
203 | |||
204 | 1 | result = avformat_open_input(&fmt_ctx, input_filename, NULL, NULL); | |
205 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (result < 0) { |
206 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't open file\n"); | |
207 | ✗ | return result; | |
208 | } | ||
209 | |||
210 | 1 | result = avformat_find_stream_info(fmt_ctx, NULL); | |
211 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (result < 0) { |
212 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't get stream info\n"); | |
213 | ✗ | goto end; | |
214 | } | ||
215 | |||
216 | 1 | start_ts = read_seek_range(start); | |
217 | 1 | end_ts = read_seek_range(end); | |
218 |
2/4✓ Branch 0 taken 1 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 1 times.
|
1 | if ((start_ts < 0) || (end_ts < 0)) { |
219 | ✗ | result = -1; | |
220 | ✗ | goto end; | |
221 | } | ||
222 | |||
223 | //TODO: add ability to work with audio format | ||
224 | 1 | video_stream = av_find_best_stream(fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0); | |
225 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (video_stream < 0) { |
226 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't find video stream in input file\n"); | |
227 | ✗ | result = video_stream; | |
228 | ✗ | goto end; | |
229 | } | ||
230 | |||
231 | 1 | origin_par = fmt_ctx->streams[video_stream]->codecpar; | |
232 | |||
233 | 1 | codec = avcodec_find_decoder(origin_par->codec_id); | |
234 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (!codec) { |
235 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't find decoder\n"); | |
236 | ✗ | result = AVERROR_DECODER_NOT_FOUND; | |
237 | ✗ | goto end; | |
238 | } | ||
239 | |||
240 | 1 | ctx = avcodec_alloc_context3(codec); | |
241 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (!ctx) { |
242 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't allocate decoder context\n"); | |
243 | ✗ | result = AVERROR(ENOMEM); | |
244 | ✗ | goto end; | |
245 | } | ||
246 | |||
247 | 1 | result = avcodec_parameters_to_context(ctx, origin_par); | |
248 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (result) { |
249 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't copy decoder context\n"); | |
250 | ✗ | goto end; | |
251 | } | ||
252 | |||
253 | 1 | result = avcodec_open2(ctx, codec, NULL); | |
254 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (result < 0) { |
255 | ✗ | av_log(ctx, AV_LOG_ERROR, "Can't open decoder\n"); | |
256 | ✗ | goto end; | |
257 | } | ||
258 | |||
259 | 1 | fr = av_frame_alloc(); | |
260 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (!fr) { |
261 | ✗ | av_log(NULL, AV_LOG_ERROR, "Can't allocate frame\n"); | |
262 | ✗ | result = AVERROR(ENOMEM); | |
263 | ✗ | goto end; | |
264 | } | ||
265 | |||
266 | 1 | pkt = av_packet_alloc(); | |
267 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (!pkt) { |
268 | ✗ | av_log(NULL, AV_LOG_ERROR, "Cannot allocate packet\n"); | |
269 | ✗ | result = AVERROR(ENOMEM); | |
270 | ✗ | goto end; | |
271 | } | ||
272 | |||
273 | 1 | result = compute_crc_of_packets(fmt_ctx, video_stream, ctx, pkt, fr, 0, 0, 1); | |
274 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (result != 0) |
275 | ✗ | goto end; | |
276 | |||
277 |
2/2✓ Branch 0 taken 8 times.
✓ Branch 1 taken 1 times.
|
9 | for (i = start_ts; i < end_ts; i += 100) { |
278 |
2/2✓ Branch 0 taken 28 times.
✓ Branch 1 taken 8 times.
|
36 | for (j = i + 100; j < end_ts; j += 100) { |
279 | 28 | result = compute_crc_of_packets(fmt_ctx, video_stream, ctx, pkt, fr, i, j, 0); | |
280 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 28 times.
|
28 | if (result != 0) |
281 | ✗ | break; | |
282 | } | ||
283 | } | ||
284 | |||
285 | 1 | end: | |
286 | 1 | av_freep(&crc_array); | |
287 | 1 | av_freep(&pts_array); | |
288 | 1 | av_packet_free(&pkt); | |
289 | 1 | av_frame_free(&fr); | |
290 | 1 | avformat_close_input(&fmt_ctx); | |
291 | 1 | avcodec_free_context(&ctx); | |
292 | 1 | return result; | |
293 | } | ||
294 | |||
295 | 1 | int main(int argc, char **argv) | |
296 | { | ||
297 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (argc < 4) { |
298 | ✗ | av_log(NULL, AV_LOG_ERROR, "Incorrect input\n"); | |
299 | ✗ | return 1; | |
300 | } | ||
301 | |||
302 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 1 times.
|
1 | if (seek_test(argv[1], argv[2], argv[3]) != 0) |
303 | ✗ | return 1; | |
304 | |||
305 | 1 | return 0; | |
306 | } | ||
307 |