Browse code

Rename ffplay to avplay.

Anton Khirnov authored on 2011/07/27 00:51:34
Showing 17 changed files
... ...
@@ -12,7 +12,7 @@ doc/*.html
12 12
 doc/*.pod
13 13
 doxy
14 14
 ffmpeg
15
-ffplay
15
+avplay
16 16
 ffprobe
17 17
 ffserver
18 18
 libavcodec/*_tablegen
... ...
@@ -5,6 +5,7 @@ releases are sorted from youngest to oldest.
5 5
 version <next>:
6 6
 - BWF muxer
7 7
 - Flash Screen Video 2 decoder
8
+- ffplay renamed to avplay
8 9
 
9 10
 
10 11
 version 0.7:
... ...
@@ -53,7 +53,7 @@ COMPILE_S = $(call COMPILE,AS)
53 53
 %.c %.h: TAG = GEN
54 54
 
55 55
 PROGS-$(CONFIG_FFMPEG)   += ffmpeg
56
-PROGS-$(CONFIG_FFPLAY)   += ffplay
56
+PROGS-$(CONFIG_AVPLAY)   += avplay
57 57
 PROGS-$(CONFIG_FFPROBE)  += ffprobe
58 58
 PROGS-$(CONFIG_FFSERVER) += ffserver
59 59
 
... ...
@@ -64,7 +64,7 @@ HOSTPROGS  := $(TESTTOOLS:%=tests/%)
64 64
 TOOLS       = qt-faststart trasher
65 65
 TOOLS-$(CONFIG_ZLIB) += cws2fws
66 66
 
67
-BASENAMES   = ffmpeg ffplay ffprobe ffserver
67
+BASENAMES   = ffmpeg avplay ffprobe ffserver
68 68
 ALLPROGS    = $(BASENAMES:%=%$(EXESUF))
69 69
 ALLMANPAGES = $(BASENAMES:%=%.1)
70 70
 
... ...
@@ -116,8 +116,8 @@ endef
116 116
 
117 117
 $(foreach D,$(FFLIBS),$(eval $(call DOSUBDIR,lib$(D))))
118 118
 
119
-ffplay.o: CFLAGS += $(SDL_CFLAGS)
120
-ffplay$(EXESUF): FF_EXTRALIBS += $(SDL_LIBS)
119
+avplay.o: CFLAGS += $(SDL_CFLAGS)
120
+avplay$(EXESUF): FF_EXTRALIBS += $(SDL_LIBS)
121 121
 ffserver$(EXESUF): LDFLAGS += $(FFSERVERLDFLAGS)
122 122
 
123 123
 $(PROGS): %$(EXESUF): %.o cmdutils.o $(FF_DEP_LIBS)
124 124
new file mode 100644
... ...
@@ -0,0 +1,3071 @@
0
+/*
1
+ * avplay : Simple Media Player based on the Libav libraries
2
+ * Copyright (c) 2003 Fabrice Bellard
3
+ *
4
+ * This file is part of Libav.
5
+ *
6
+ * Libav is free software; you can redistribute it and/or
7
+ * modify it under the terms of the GNU Lesser General Public
8
+ * License as published by the Free Software Foundation; either
9
+ * version 2.1 of the License, or (at your option) any later version.
10
+ *
11
+ * Libav is distributed in the hope that it will be useful,
12
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
13
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14
+ * Lesser General Public License for more details.
15
+ *
16
+ * You should have received a copy of the GNU Lesser General Public
17
+ * License along with Libav; if not, write to the Free Software
18
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19
+ */
20
+
21
+#include "config.h"
22
+#include <inttypes.h>
23
+#include <math.h>
24
+#include <limits.h>
25
+#include "libavutil/avstring.h"
26
+#include "libavutil/colorspace.h"
27
+#include "libavutil/mathematics.h"
28
+#include "libavutil/pixdesc.h"
29
+#include "libavutil/imgutils.h"
30
+#include "libavutil/dict.h"
31
+#include "libavutil/parseutils.h"
32
+#include "libavutil/samplefmt.h"
33
+#include "libavformat/avformat.h"
34
+#include "libavdevice/avdevice.h"
35
+#include "libswscale/swscale.h"
36
+#include "libavcodec/audioconvert.h"
37
+#include "libavutil/opt.h"
38
+#include "libavcodec/avfft.h"
39
+
40
+#if CONFIG_AVFILTER
41
+# include "libavfilter/avfilter.h"
42
+# include "libavfilter/avfiltergraph.h"
43
+#endif
44
+
45
+#include "cmdutils.h"
46
+
47
+#include <SDL.h>
48
+#include <SDL_thread.h>
49
+
50
+#ifdef __MINGW32__
51
+#undef main /* We don't want SDL to override our main() */
52
+#endif
53
+
54
+#include <unistd.h>
55
+#include <assert.h>
56
+
57
+const char program_name[] = "avplay";
58
+const int program_birth_year = 2003;
59
+
60
+#define MAX_QUEUE_SIZE (15 * 1024 * 1024)
61
+#define MIN_AUDIOQ_SIZE (20 * 16 * 1024)
62
+#define MIN_FRAMES 5
63
+
64
+/* SDL audio buffer size, in samples. Should be small to have precise
65
+   A/V sync as SDL does not have hardware buffer fullness info. */
66
+#define SDL_AUDIO_BUFFER_SIZE 1024
67
+
68
+/* no AV sync correction is done if below the AV sync threshold */
69
+#define AV_SYNC_THRESHOLD 0.01
70
+/* no AV correction is done if too big error */
71
+#define AV_NOSYNC_THRESHOLD 10.0
72
+
73
+#define FRAME_SKIP_FACTOR 0.05
74
+
75
+/* maximum audio speed change to get correct sync */
76
+#define SAMPLE_CORRECTION_PERCENT_MAX 10
77
+
78
+/* we use about AUDIO_DIFF_AVG_NB A-V differences to make the average */
79
+#define AUDIO_DIFF_AVG_NB   20
80
+
81
+/* NOTE: the size must be big enough to compensate the hardware audio buffersize size */
82
+#define SAMPLE_ARRAY_SIZE (2*65536)
83
+
84
+static int sws_flags = SWS_BICUBIC;
85
+
86
+typedef struct PacketQueue {
87
+    AVPacketList *first_pkt, *last_pkt;
88
+    int nb_packets;
89
+    int size;
90
+    int abort_request;
91
+    SDL_mutex *mutex;
92
+    SDL_cond *cond;
93
+} PacketQueue;
94
+
95
+#define VIDEO_PICTURE_QUEUE_SIZE 2
96
+#define SUBPICTURE_QUEUE_SIZE 4
97
+
98
+typedef struct VideoPicture {
99
+    double pts;                                  ///<presentation time stamp for this picture
100
+    double target_clock;                         ///<av_gettime() time at which this should be displayed ideally
101
+    int64_t pos;                                 ///<byte position in file
102
+    SDL_Overlay *bmp;
103
+    int width, height; /* source height & width */
104
+    int allocated;
105
+    enum PixelFormat pix_fmt;
106
+
107
+#if CONFIG_AVFILTER
108
+    AVFilterBufferRef *picref;
109
+#endif
110
+} VideoPicture;
111
+
112
+typedef struct SubPicture {
113
+    double pts; /* presentation time stamp for this picture */
114
+    AVSubtitle sub;
115
+} SubPicture;
116
+
117
+enum {
118
+    AV_SYNC_AUDIO_MASTER, /* default choice */
119
+    AV_SYNC_VIDEO_MASTER,
120
+    AV_SYNC_EXTERNAL_CLOCK, /* synchronize to an external clock */
121
+};
122
+
123
+typedef struct VideoState {
124
+    SDL_Thread *parse_tid;
125
+    SDL_Thread *video_tid;
126
+    SDL_Thread *refresh_tid;
127
+    AVInputFormat *iformat;
128
+    int no_background;
129
+    int abort_request;
130
+    int paused;
131
+    int last_paused;
132
+    int seek_req;
133
+    int seek_flags;
134
+    int64_t seek_pos;
135
+    int64_t seek_rel;
136
+    int read_pause_return;
137
+    AVFormatContext *ic;
138
+    int dtg_active_format;
139
+
140
+    int audio_stream;
141
+
142
+    int av_sync_type;
143
+    double external_clock; /* external clock base */
144
+    int64_t external_clock_time;
145
+
146
+    double audio_clock;
147
+    double audio_diff_cum; /* used for AV difference average computation */
148
+    double audio_diff_avg_coef;
149
+    double audio_diff_threshold;
150
+    int audio_diff_avg_count;
151
+    AVStream *audio_st;
152
+    PacketQueue audioq;
153
+    int audio_hw_buf_size;
154
+    /* samples output by the codec. we reserve more space for avsync
155
+       compensation */
156
+    DECLARE_ALIGNED(16,uint8_t,audio_buf1)[(AVCODEC_MAX_AUDIO_FRAME_SIZE * 3) / 2];
157
+    DECLARE_ALIGNED(16,uint8_t,audio_buf2)[(AVCODEC_MAX_AUDIO_FRAME_SIZE * 3) / 2];
158
+    uint8_t *audio_buf;
159
+    unsigned int audio_buf_size; /* in bytes */
160
+    int audio_buf_index; /* in bytes */
161
+    AVPacket audio_pkt_temp;
162
+    AVPacket audio_pkt;
163
+    enum AVSampleFormat audio_src_fmt;
164
+    AVAudioConvert *reformat_ctx;
165
+
166
+    int show_audio; /* if true, display audio samples */
167
+    int16_t sample_array[SAMPLE_ARRAY_SIZE];
168
+    int sample_array_index;
169
+    int last_i_start;
170
+    RDFTContext *rdft;
171
+    int rdft_bits;
172
+    FFTSample *rdft_data;
173
+    int xpos;
174
+
175
+    SDL_Thread *subtitle_tid;
176
+    int subtitle_stream;
177
+    int subtitle_stream_changed;
178
+    AVStream *subtitle_st;
179
+    PacketQueue subtitleq;
180
+    SubPicture subpq[SUBPICTURE_QUEUE_SIZE];
181
+    int subpq_size, subpq_rindex, subpq_windex;
182
+    SDL_mutex *subpq_mutex;
183
+    SDL_cond *subpq_cond;
184
+
185
+    double frame_timer;
186
+    double frame_last_pts;
187
+    double frame_last_delay;
188
+    double video_clock;                          ///<pts of last decoded frame / predicted pts of next decoded frame
189
+    int video_stream;
190
+    AVStream *video_st;
191
+    PacketQueue videoq;
192
+    double video_current_pts;                    ///<current displayed pts (different from video_clock if frame fifos are used)
193
+    double video_current_pts_drift;              ///<video_current_pts - time (av_gettime) at which we updated video_current_pts - used to have running video pts
194
+    int64_t video_current_pos;                   ///<current displayed file pos
195
+    VideoPicture pictq[VIDEO_PICTURE_QUEUE_SIZE];
196
+    int pictq_size, pictq_rindex, pictq_windex;
197
+    SDL_mutex *pictq_mutex;
198
+    SDL_cond *pictq_cond;
199
+#if !CONFIG_AVFILTER
200
+    struct SwsContext *img_convert_ctx;
201
+#endif
202
+
203
+    //    QETimer *video_timer;
204
+    char filename[1024];
205
+    int width, height, xleft, ytop;
206
+
207
+    PtsCorrectionContext pts_ctx;
208
+
209
+#if CONFIG_AVFILTER
210
+    AVFilterContext *out_video_filter;          ///<the last filter in the video chain
211
+#endif
212
+
213
+    float skip_frames;
214
+    float skip_frames_index;
215
+    int refresh;
216
+} VideoState;
217
+
218
+static void show_help(void);
219
+
220
+/* options specified by the user */
221
+static AVInputFormat *file_iformat;
222
+static const char *input_filename;
223
+static const char *window_title;
224
+static int fs_screen_width;
225
+static int fs_screen_height;
226
+static int screen_width = 0;
227
+static int screen_height = 0;
228
+static int audio_disable;
229
+static int video_disable;
230
+static int wanted_stream[AVMEDIA_TYPE_NB]={
231
+    [AVMEDIA_TYPE_AUDIO]=-1,
232
+    [AVMEDIA_TYPE_VIDEO]=-1,
233
+    [AVMEDIA_TYPE_SUBTITLE]=-1,
234
+};
235
+static int seek_by_bytes=-1;
236
+static int display_disable;
237
+static int show_status = 1;
238
+static int av_sync_type = AV_SYNC_AUDIO_MASTER;
239
+static int64_t start_time = AV_NOPTS_VALUE;
240
+static int64_t duration = AV_NOPTS_VALUE;
241
+static int debug = 0;
242
+static int debug_mv = 0;
243
+static int step = 0;
244
+static int thread_count = 1;
245
+static int workaround_bugs = 1;
246
+static int fast = 0;
247
+static int genpts = 0;
248
+static int lowres = 0;
249
+static int idct = FF_IDCT_AUTO;
250
+static enum AVDiscard skip_frame= AVDISCARD_DEFAULT;
251
+static enum AVDiscard skip_idct= AVDISCARD_DEFAULT;
252
+static enum AVDiscard skip_loop_filter= AVDISCARD_DEFAULT;
253
+static int error_recognition = FF_ER_CAREFUL;
254
+static int error_concealment = 3;
255
+static int decoder_reorder_pts= -1;
256
+static int autoexit;
257
+static int exit_on_keydown;
258
+static int exit_on_mousedown;
259
+static int loop=1;
260
+static int framedrop=1;
261
+
262
+static int rdftspeed=20;
263
+#if CONFIG_AVFILTER
264
+static char *vfilters = NULL;
265
+#endif
266
+
267
+/* current context */
268
+static int is_full_screen;
269
+static VideoState *cur_stream;
270
+static int64_t audio_callback_time;
271
+
272
+static AVPacket flush_pkt;
273
+
274
+#define FF_ALLOC_EVENT   (SDL_USEREVENT)
275
+#define FF_REFRESH_EVENT (SDL_USEREVENT + 1)
276
+#define FF_QUIT_EVENT    (SDL_USEREVENT + 2)
277
+
278
+static SDL_Surface *screen;
279
+
280
+static int packet_queue_put(PacketQueue *q, AVPacket *pkt);
281
+
282
+/* packet queue handling */
283
+static void packet_queue_init(PacketQueue *q)
284
+{
285
+    memset(q, 0, sizeof(PacketQueue));
286
+    q->mutex = SDL_CreateMutex();
287
+    q->cond = SDL_CreateCond();
288
+    packet_queue_put(q, &flush_pkt);
289
+}
290
+
291
+static void packet_queue_flush(PacketQueue *q)
292
+{
293
+    AVPacketList *pkt, *pkt1;
294
+
295
+    SDL_LockMutex(q->mutex);
296
+    for(pkt = q->first_pkt; pkt != NULL; pkt = pkt1) {
297
+        pkt1 = pkt->next;
298
+        av_free_packet(&pkt->pkt);
299
+        av_freep(&pkt);
300
+    }
301
+    q->last_pkt = NULL;
302
+    q->first_pkt = NULL;
303
+    q->nb_packets = 0;
304
+    q->size = 0;
305
+    SDL_UnlockMutex(q->mutex);
306
+}
307
+
308
+static void packet_queue_end(PacketQueue *q)
309
+{
310
+    packet_queue_flush(q);
311
+    SDL_DestroyMutex(q->mutex);
312
+    SDL_DestroyCond(q->cond);
313
+}
314
+
315
+static int packet_queue_put(PacketQueue *q, AVPacket *pkt)
316
+{
317
+    AVPacketList *pkt1;
318
+
319
+    /* duplicate the packet */
320
+    if (pkt!=&flush_pkt && av_dup_packet(pkt) < 0)
321
+        return -1;
322
+
323
+    pkt1 = av_malloc(sizeof(AVPacketList));
324
+    if (!pkt1)
325
+        return -1;
326
+    pkt1->pkt = *pkt;
327
+    pkt1->next = NULL;
328
+
329
+
330
+    SDL_LockMutex(q->mutex);
331
+
332
+    if (!q->last_pkt)
333
+
334
+        q->first_pkt = pkt1;
335
+    else
336
+        q->last_pkt->next = pkt1;
337
+    q->last_pkt = pkt1;
338
+    q->nb_packets++;
339
+    q->size += pkt1->pkt.size + sizeof(*pkt1);
340
+    /* XXX: should duplicate packet data in DV case */
341
+    SDL_CondSignal(q->cond);
342
+
343
+    SDL_UnlockMutex(q->mutex);
344
+    return 0;
345
+}
346
+
347
+static void packet_queue_abort(PacketQueue *q)
348
+{
349
+    SDL_LockMutex(q->mutex);
350
+
351
+    q->abort_request = 1;
352
+
353
+    SDL_CondSignal(q->cond);
354
+
355
+    SDL_UnlockMutex(q->mutex);
356
+}
357
+
358
+/* return < 0 if aborted, 0 if no packet and > 0 if packet.  */
359
+static int packet_queue_get(PacketQueue *q, AVPacket *pkt, int block)
360
+{
361
+    AVPacketList *pkt1;
362
+    int ret;
363
+
364
+    SDL_LockMutex(q->mutex);
365
+
366
+    for(;;) {
367
+        if (q->abort_request) {
368
+            ret = -1;
369
+            break;
370
+        }
371
+
372
+        pkt1 = q->first_pkt;
373
+        if (pkt1) {
374
+            q->first_pkt = pkt1->next;
375
+            if (!q->first_pkt)
376
+                q->last_pkt = NULL;
377
+            q->nb_packets--;
378
+            q->size -= pkt1->pkt.size + sizeof(*pkt1);
379
+            *pkt = pkt1->pkt;
380
+            av_free(pkt1);
381
+            ret = 1;
382
+            break;
383
+        } else if (!block) {
384
+            ret = 0;
385
+            break;
386
+        } else {
387
+            SDL_CondWait(q->cond, q->mutex);
388
+        }
389
+    }
390
+    SDL_UnlockMutex(q->mutex);
391
+    return ret;
392
+}
393
+
394
+static inline void fill_rectangle(SDL_Surface *screen,
395
+                                  int x, int y, int w, int h, int color)
396
+{
397
+    SDL_Rect rect;
398
+    rect.x = x;
399
+    rect.y = y;
400
+    rect.w = w;
401
+    rect.h = h;
402
+    SDL_FillRect(screen, &rect, color);
403
+}
404
+
405
+#define ALPHA_BLEND(a, oldp, newp, s)\
406
+((((oldp << s) * (255 - (a))) + (newp * (a))) / (255 << s))
407
+
408
+#define RGBA_IN(r, g, b, a, s)\
409
+{\
410
+    unsigned int v = ((const uint32_t *)(s))[0];\
411
+    a = (v >> 24) & 0xff;\
412
+    r = (v >> 16) & 0xff;\
413
+    g = (v >> 8) & 0xff;\
414
+    b = v & 0xff;\
415
+}
416
+
417
+#define YUVA_IN(y, u, v, a, s, pal)\
418
+{\
419
+    unsigned int val = ((const uint32_t *)(pal))[*(const uint8_t*)(s)];\
420
+    a = (val >> 24) & 0xff;\
421
+    y = (val >> 16) & 0xff;\
422
+    u = (val >> 8) & 0xff;\
423
+    v = val & 0xff;\
424
+}
425
+
426
+#define YUVA_OUT(d, y, u, v, a)\
427
+{\
428
+    ((uint32_t *)(d))[0] = (a << 24) | (y << 16) | (u << 8) | v;\
429
+}
430
+
431
+
432
+#define BPP 1
433
+
434
+static void blend_subrect(AVPicture *dst, const AVSubtitleRect *rect, int imgw, int imgh)
435
+{
436
+    int wrap, wrap3, width2, skip2;
437
+    int y, u, v, a, u1, v1, a1, w, h;
438
+    uint8_t *lum, *cb, *cr;
439
+    const uint8_t *p;
440
+    const uint32_t *pal;
441
+    int dstx, dsty, dstw, dsth;
442
+
443
+    dstw = av_clip(rect->w, 0, imgw);
444
+    dsth = av_clip(rect->h, 0, imgh);
445
+    dstx = av_clip(rect->x, 0, imgw - dstw);
446
+    dsty = av_clip(rect->y, 0, imgh - dsth);
447
+    lum = dst->data[0] + dsty * dst->linesize[0];
448
+    cb = dst->data[1] + (dsty >> 1) * dst->linesize[1];
449
+    cr = dst->data[2] + (dsty >> 1) * dst->linesize[2];
450
+
451
+    width2 = ((dstw + 1) >> 1) + (dstx & ~dstw & 1);
452
+    skip2 = dstx >> 1;
453
+    wrap = dst->linesize[0];
454
+    wrap3 = rect->pict.linesize[0];
455
+    p = rect->pict.data[0];
456
+    pal = (const uint32_t *)rect->pict.data[1];  /* Now in YCrCb! */
457
+
458
+    if (dsty & 1) {
459
+        lum += dstx;
460
+        cb += skip2;
461
+        cr += skip2;
462
+
463
+        if (dstx & 1) {
464
+            YUVA_IN(y, u, v, a, p, pal);
465
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
466
+            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
467
+            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
468
+            cb++;
469
+            cr++;
470
+            lum++;
471
+            p += BPP;
472
+        }
473
+        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
474
+            YUVA_IN(y, u, v, a, p, pal);
475
+            u1 = u;
476
+            v1 = v;
477
+            a1 = a;
478
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
479
+
480
+            YUVA_IN(y, u, v, a, p + BPP, pal);
481
+            u1 += u;
482
+            v1 += v;
483
+            a1 += a;
484
+            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
485
+            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
486
+            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
487
+            cb++;
488
+            cr++;
489
+            p += 2 * BPP;
490
+            lum += 2;
491
+        }
492
+        if (w) {
493
+            YUVA_IN(y, u, v, a, p, pal);
494
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
495
+            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
496
+            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
497
+            p++;
498
+            lum++;
499
+        }
500
+        p += wrap3 - dstw * BPP;
501
+        lum += wrap - dstw - dstx;
502
+        cb += dst->linesize[1] - width2 - skip2;
503
+        cr += dst->linesize[2] - width2 - skip2;
504
+    }
505
+    for(h = dsth - (dsty & 1); h >= 2; h -= 2) {
506
+        lum += dstx;
507
+        cb += skip2;
508
+        cr += skip2;
509
+
510
+        if (dstx & 1) {
511
+            YUVA_IN(y, u, v, a, p, pal);
512
+            u1 = u;
513
+            v1 = v;
514
+            a1 = a;
515
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
516
+            p += wrap3;
517
+            lum += wrap;
518
+            YUVA_IN(y, u, v, a, p, pal);
519
+            u1 += u;
520
+            v1 += v;
521
+            a1 += a;
522
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
523
+            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
524
+            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
525
+            cb++;
526
+            cr++;
527
+            p += -wrap3 + BPP;
528
+            lum += -wrap + 1;
529
+        }
530
+        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
531
+            YUVA_IN(y, u, v, a, p, pal);
532
+            u1 = u;
533
+            v1 = v;
534
+            a1 = a;
535
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
536
+
537
+            YUVA_IN(y, u, v, a, p + BPP, pal);
538
+            u1 += u;
539
+            v1 += v;
540
+            a1 += a;
541
+            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
542
+            p += wrap3;
543
+            lum += wrap;
544
+
545
+            YUVA_IN(y, u, v, a, p, pal);
546
+            u1 += u;
547
+            v1 += v;
548
+            a1 += a;
549
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
550
+
551
+            YUVA_IN(y, u, v, a, p + BPP, pal);
552
+            u1 += u;
553
+            v1 += v;
554
+            a1 += a;
555
+            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
556
+
557
+            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 2);
558
+            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 2);
559
+
560
+            cb++;
561
+            cr++;
562
+            p += -wrap3 + 2 * BPP;
563
+            lum += -wrap + 2;
564
+        }
565
+        if (w) {
566
+            YUVA_IN(y, u, v, a, p, pal);
567
+            u1 = u;
568
+            v1 = v;
569
+            a1 = a;
570
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
571
+            p += wrap3;
572
+            lum += wrap;
573
+            YUVA_IN(y, u, v, a, p, pal);
574
+            u1 += u;
575
+            v1 += v;
576
+            a1 += a;
577
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
578
+            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
579
+            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
580
+            cb++;
581
+            cr++;
582
+            p += -wrap3 + BPP;
583
+            lum += -wrap + 1;
584
+        }
585
+        p += wrap3 + (wrap3 - dstw * BPP);
586
+        lum += wrap + (wrap - dstw - dstx);
587
+        cb += dst->linesize[1] - width2 - skip2;
588
+        cr += dst->linesize[2] - width2 - skip2;
589
+    }
590
+    /* handle odd height */
591
+    if (h) {
592
+        lum += dstx;
593
+        cb += skip2;
594
+        cr += skip2;
595
+
596
+        if (dstx & 1) {
597
+            YUVA_IN(y, u, v, a, p, pal);
598
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
599
+            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
600
+            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
601
+            cb++;
602
+            cr++;
603
+            lum++;
604
+            p += BPP;
605
+        }
606
+        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
607
+            YUVA_IN(y, u, v, a, p, pal);
608
+            u1 = u;
609
+            v1 = v;
610
+            a1 = a;
611
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
612
+
613
+            YUVA_IN(y, u, v, a, p + BPP, pal);
614
+            u1 += u;
615
+            v1 += v;
616
+            a1 += a;
617
+            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
618
+            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u, 1);
619
+            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v, 1);
620
+            cb++;
621
+            cr++;
622
+            p += 2 * BPP;
623
+            lum += 2;
624
+        }
625
+        if (w) {
626
+            YUVA_IN(y, u, v, a, p, pal);
627
+            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
628
+            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
629
+            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
630
+        }
631
+    }
632
+}
633
+
634
+static void free_subpicture(SubPicture *sp)
635
+{
636
+    avsubtitle_free(&sp->sub);
637
+}
638
+
639
+static void video_image_display(VideoState *is)
640
+{
641
+    VideoPicture *vp;
642
+    SubPicture *sp;
643
+    AVPicture pict;
644
+    float aspect_ratio;
645
+    int width, height, x, y;
646
+    SDL_Rect rect;
647
+    int i;
648
+
649
+    vp = &is->pictq[is->pictq_rindex];
650
+    if (vp->bmp) {
651
+#if CONFIG_AVFILTER
652
+         if (vp->picref->video->pixel_aspect.num == 0)
653
+             aspect_ratio = 0;
654
+         else
655
+             aspect_ratio = av_q2d(vp->picref->video->pixel_aspect);
656
+#else
657
+
658
+        /* XXX: use variable in the frame */
659
+        if (is->video_st->sample_aspect_ratio.num)
660
+            aspect_ratio = av_q2d(is->video_st->sample_aspect_ratio);
661
+        else if (is->video_st->codec->sample_aspect_ratio.num)
662
+            aspect_ratio = av_q2d(is->video_st->codec->sample_aspect_ratio);
663
+        else
664
+            aspect_ratio = 0;
665
+#endif
666
+        if (aspect_ratio <= 0.0)
667
+            aspect_ratio = 1.0;
668
+        aspect_ratio *= (float)vp->width / (float)vp->height;
669
+
670
+        if (is->subtitle_st)
671
+        {
672
+            if (is->subpq_size > 0)
673
+            {
674
+                sp = &is->subpq[is->subpq_rindex];
675
+
676
+                if (vp->pts >= sp->pts + ((float) sp->sub.start_display_time / 1000))
677
+                {
678
+                    SDL_LockYUVOverlay (vp->bmp);
679
+
680
+                    pict.data[0] = vp->bmp->pixels[0];
681
+                    pict.data[1] = vp->bmp->pixels[2];
682
+                    pict.data[2] = vp->bmp->pixels[1];
683
+
684
+                    pict.linesize[0] = vp->bmp->pitches[0];
685
+                    pict.linesize[1] = vp->bmp->pitches[2];
686
+                    pict.linesize[2] = vp->bmp->pitches[1];
687
+
688
+                    for (i = 0; i < sp->sub.num_rects; i++)
689
+                        blend_subrect(&pict, sp->sub.rects[i],
690
+                                      vp->bmp->w, vp->bmp->h);
691
+
692
+                    SDL_UnlockYUVOverlay (vp->bmp);
693
+                }
694
+            }
695
+        }
696
+
697
+
698
+        /* XXX: we suppose the screen has a 1.0 pixel ratio */
699
+        height = is->height;
700
+        width = ((int)rint(height * aspect_ratio)) & ~1;
701
+        if (width > is->width) {
702
+            width = is->width;
703
+            height = ((int)rint(width / aspect_ratio)) & ~1;
704
+        }
705
+        x = (is->width - width) / 2;
706
+        y = (is->height - height) / 2;
707
+        is->no_background = 0;
708
+        rect.x = is->xleft + x;
709
+        rect.y = is->ytop  + y;
710
+        rect.w = width;
711
+        rect.h = height;
712
+        SDL_DisplayYUVOverlay(vp->bmp, &rect);
713
+    }
714
+}
715
+
716
+/* get the current audio output buffer size, in samples. With SDL, we
717
+   cannot have a precise information */
718
+static int audio_write_get_buf_size(VideoState *is)
719
+{
720
+    return is->audio_buf_size - is->audio_buf_index;
721
+}
722
+
723
+static inline int compute_mod(int a, int b)
724
+{
725
+    a = a % b;
726
+    if (a >= 0)
727
+        return a;
728
+    else
729
+        return a + b;
730
+}
731
+
732
+static void video_audio_display(VideoState *s)
733
+{
734
+    int i, i_start, x, y1, y, ys, delay, n, nb_display_channels;
735
+    int ch, channels, h, h2, bgcolor, fgcolor;
736
+    int16_t time_diff;
737
+    int rdft_bits, nb_freq;
738
+
739
+    for(rdft_bits=1; (1<<rdft_bits)<2*s->height; rdft_bits++)
740
+        ;
741
+    nb_freq= 1<<(rdft_bits-1);
742
+
743
+    /* compute display index : center on currently output samples */
744
+    channels = s->audio_st->codec->channels;
745
+    nb_display_channels = channels;
746
+    if (!s->paused) {
747
+        int data_used= s->show_audio==1 ? s->width : (2*nb_freq);
748
+        n = 2 * channels;
749
+        delay = audio_write_get_buf_size(s);
750
+        delay /= n;
751
+
752
+        /* to be more precise, we take into account the time spent since
753
+           the last buffer computation */
754
+        if (audio_callback_time) {
755
+            time_diff = av_gettime() - audio_callback_time;
756
+            delay -= (time_diff * s->audio_st->codec->sample_rate) / 1000000;
757
+        }
758
+
759
+        delay += 2*data_used;
760
+        if (delay < data_used)
761
+            delay = data_used;
762
+
763
+        i_start= x = compute_mod(s->sample_array_index - delay * channels, SAMPLE_ARRAY_SIZE);
764
+        if(s->show_audio==1){
765
+            h= INT_MIN;
766
+            for(i=0; i<1000; i+=channels){
767
+                int idx= (SAMPLE_ARRAY_SIZE + x - i) % SAMPLE_ARRAY_SIZE;
768
+                int a= s->sample_array[idx];
769
+                int b= s->sample_array[(idx + 4*channels)%SAMPLE_ARRAY_SIZE];
770
+                int c= s->sample_array[(idx + 5*channels)%SAMPLE_ARRAY_SIZE];
771
+                int d= s->sample_array[(idx + 9*channels)%SAMPLE_ARRAY_SIZE];
772
+                int score= a-d;
773
+                if(h<score && (b^c)<0){
774
+                    h= score;
775
+                    i_start= idx;
776
+                }
777
+            }
778
+        }
779
+
780
+        s->last_i_start = i_start;
781
+    } else {
782
+        i_start = s->last_i_start;
783
+    }
784
+
785
+    bgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0x00);
786
+    if(s->show_audio==1){
787
+        fill_rectangle(screen,
788
+                       s->xleft, s->ytop, s->width, s->height,
789
+                       bgcolor);
790
+
791
+        fgcolor = SDL_MapRGB(screen->format, 0xff, 0xff, 0xff);
792
+
793
+        /* total height for one channel */
794
+        h = s->height / nb_display_channels;
795
+        /* graph height / 2 */
796
+        h2 = (h * 9) / 20;
797
+        for(ch = 0;ch < nb_display_channels; ch++) {
798
+            i = i_start + ch;
799
+            y1 = s->ytop + ch * h + (h / 2); /* position of center line */
800
+            for(x = 0; x < s->width; x++) {
801
+                y = (s->sample_array[i] * h2) >> 15;
802
+                if (y < 0) {
803
+                    y = -y;
804
+                    ys = y1 - y;
805
+                } else {
806
+                    ys = y1;
807
+                }
808
+                fill_rectangle(screen,
809
+                               s->xleft + x, ys, 1, y,
810
+                               fgcolor);
811
+                i += channels;
812
+                if (i >= SAMPLE_ARRAY_SIZE)
813
+                    i -= SAMPLE_ARRAY_SIZE;
814
+            }
815
+        }
816
+
817
+        fgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0xff);
818
+
819
+        for(ch = 1;ch < nb_display_channels; ch++) {
820
+            y = s->ytop + ch * h;
821
+            fill_rectangle(screen,
822
+                           s->xleft, y, s->width, 1,
823
+                           fgcolor);
824
+        }
825
+        SDL_UpdateRect(screen, s->xleft, s->ytop, s->width, s->height);
826
+    }else{
827
+        nb_display_channels= FFMIN(nb_display_channels, 2);
828
+        if(rdft_bits != s->rdft_bits){
829
+            av_rdft_end(s->rdft);
830
+            av_free(s->rdft_data);
831
+            s->rdft = av_rdft_init(rdft_bits, DFT_R2C);
832
+            s->rdft_bits= rdft_bits;
833
+            s->rdft_data= av_malloc(4*nb_freq*sizeof(*s->rdft_data));
834
+        }
835
+        {
836
+            FFTSample *data[2];
837
+            for(ch = 0;ch < nb_display_channels; ch++) {
838
+                data[ch] = s->rdft_data + 2*nb_freq*ch;
839
+                i = i_start + ch;
840
+                for(x = 0; x < 2*nb_freq; x++) {
841
+                    double w= (x-nb_freq)*(1.0/nb_freq);
842
+                    data[ch][x]= s->sample_array[i]*(1.0-w*w);
843
+                    i += channels;
844
+                    if (i >= SAMPLE_ARRAY_SIZE)
845
+                        i -= SAMPLE_ARRAY_SIZE;
846
+                }
847
+                av_rdft_calc(s->rdft, data[ch]);
848
+            }
849
+            //least efficient way to do this, we should of course directly access it but its more than fast enough
850
+            for(y=0; y<s->height; y++){
851
+                double w= 1/sqrt(nb_freq);
852
+                int a= sqrt(w*sqrt(data[0][2*y+0]*data[0][2*y+0] + data[0][2*y+1]*data[0][2*y+1]));
853
+                int b= (nb_display_channels == 2 ) ? sqrt(w*sqrt(data[1][2*y+0]*data[1][2*y+0]
854
+                       + data[1][2*y+1]*data[1][2*y+1])) : a;
855
+                a= FFMIN(a,255);
856
+                b= FFMIN(b,255);
857
+                fgcolor = SDL_MapRGB(screen->format, a, b, (a+b)/2);
858
+
859
+                fill_rectangle(screen,
860
+                            s->xpos, s->height-y, 1, 1,
861
+                            fgcolor);
862
+            }
863
+        }
864
+        SDL_UpdateRect(screen, s->xpos, s->ytop, 1, s->height);
865
+        s->xpos++;
866
+        if(s->xpos >= s->width)
867
+            s->xpos= s->xleft;
868
+    }
869
+}
870
+
871
+static int video_open(VideoState *is){
872
+    int flags = SDL_HWSURFACE|SDL_ASYNCBLIT|SDL_HWACCEL;
873
+    int w,h;
874
+
875
+    if(is_full_screen) flags |= SDL_FULLSCREEN;
876
+    else               flags |= SDL_RESIZABLE;
877
+
878
+    if (is_full_screen && fs_screen_width) {
879
+        w = fs_screen_width;
880
+        h = fs_screen_height;
881
+    } else if(!is_full_screen && screen_width){
882
+        w = screen_width;
883
+        h = screen_height;
884
+#if CONFIG_AVFILTER
885
+    }else if (is->out_video_filter && is->out_video_filter->inputs[0]){
886
+        w = is->out_video_filter->inputs[0]->w;
887
+        h = is->out_video_filter->inputs[0]->h;
888
+#else
889
+    }else if (is->video_st && is->video_st->codec->width){
890
+        w = is->video_st->codec->width;
891
+        h = is->video_st->codec->height;
892
+#endif
893
+    } else {
894
+        w = 640;
895
+        h = 480;
896
+    }
897
+    if(screen && is->width == screen->w && screen->w == w
898
+       && is->height== screen->h && screen->h == h)
899
+        return 0;
900
+
901
+#ifndef __APPLE__
902
+    screen = SDL_SetVideoMode(w, h, 0, flags);
903
+#else
904
+    /* setting bits_per_pixel = 0 or 32 causes blank video on OS X */
905
+    screen = SDL_SetVideoMode(w, h, 24, flags);
906
+#endif
907
+    if (!screen) {
908
+        fprintf(stderr, "SDL: could not set video mode - exiting\n");
909
+        return -1;
910
+    }
911
+    if (!window_title)
912
+        window_title = input_filename;
913
+    SDL_WM_SetCaption(window_title, window_title);
914
+
915
+    is->width = screen->w;
916
+    is->height = screen->h;
917
+
918
+    return 0;
919
+}
920
+
921
+/* display the current picture, if any */
922
+static void video_display(VideoState *is)
923
+{
924
+    if(!screen)
925
+        video_open(cur_stream);
926
+    if (is->audio_st && is->show_audio)
927
+        video_audio_display(is);
928
+    else if (is->video_st)
929
+        video_image_display(is);
930
+}
931
+
932
+static int refresh_thread(void *opaque)
933
+{
934
+    VideoState *is= opaque;
935
+    while(!is->abort_request){
936
+        SDL_Event event;
937
+        event.type = FF_REFRESH_EVENT;
938
+        event.user.data1 = opaque;
939
+        if(!is->refresh){
940
+            is->refresh=1;
941
+            SDL_PushEvent(&event);
942
+        }
943
+        usleep(is->audio_st && is->show_audio ? rdftspeed*1000 : 5000); //FIXME ideally we should wait the correct time but SDLs event passing is so slow it would be silly
944
+    }
945
+    return 0;
946
+}
947
+
948
+/* get the current audio clock value */
949
+static double get_audio_clock(VideoState *is)
950
+{
951
+    double pts;
952
+    int hw_buf_size, bytes_per_sec;
953
+    pts = is->audio_clock;
954
+    hw_buf_size = audio_write_get_buf_size(is);
955
+    bytes_per_sec = 0;
956
+    if (is->audio_st) {
957
+        bytes_per_sec = is->audio_st->codec->sample_rate *
958
+            2 * is->audio_st->codec->channels;
959
+    }
960
+    if (bytes_per_sec)
961
+        pts -= (double)hw_buf_size / bytes_per_sec;
962
+    return pts;
963
+}
964
+
965
+/* get the current video clock value */
966
+static double get_video_clock(VideoState *is)
967
+{
968
+    if (is->paused) {
969
+        return is->video_current_pts;
970
+    } else {
971
+        return is->video_current_pts_drift + av_gettime() / 1000000.0;
972
+    }
973
+}
974
+
975
+/* get the current external clock value */
976
+static double get_external_clock(VideoState *is)
977
+{
978
+    int64_t ti;
979
+    ti = av_gettime();
980
+    return is->external_clock + ((ti - is->external_clock_time) * 1e-6);
981
+}
982
+
983
+/* get the current master clock value */
984
+static double get_master_clock(VideoState *is)
985
+{
986
+    double val;
987
+
988
+    if (is->av_sync_type == AV_SYNC_VIDEO_MASTER) {
989
+        if (is->video_st)
990
+            val = get_video_clock(is);
991
+        else
992
+            val = get_audio_clock(is);
993
+    } else if (is->av_sync_type == AV_SYNC_AUDIO_MASTER) {
994
+        if (is->audio_st)
995
+            val = get_audio_clock(is);
996
+        else
997
+            val = get_video_clock(is);
998
+    } else {
999
+        val = get_external_clock(is);
1000
+    }
1001
+    return val;
1002
+}
1003
+
1004
+/* seek in the stream */
1005
+static void stream_seek(VideoState *is, int64_t pos, int64_t rel, int seek_by_bytes)
1006
+{
1007
+    if (!is->seek_req) {
1008
+        is->seek_pos = pos;
1009
+        is->seek_rel = rel;
1010
+        is->seek_flags &= ~AVSEEK_FLAG_BYTE;
1011
+        if (seek_by_bytes)
1012
+            is->seek_flags |= AVSEEK_FLAG_BYTE;
1013
+        is->seek_req = 1;
1014
+    }
1015
+}
1016
+
1017
+/* pause or resume the video */
1018
+static void stream_pause(VideoState *is)
1019
+{
1020
+    if (is->paused) {
1021
+        is->frame_timer += av_gettime() / 1000000.0 + is->video_current_pts_drift - is->video_current_pts;
1022
+        if(is->read_pause_return != AVERROR(ENOSYS)){
1023
+            is->video_current_pts = is->video_current_pts_drift + av_gettime() / 1000000.0;
1024
+        }
1025
+        is->video_current_pts_drift = is->video_current_pts - av_gettime() / 1000000.0;
1026
+    }
1027
+    is->paused = !is->paused;
1028
+}
1029
+
1030
+static double compute_target_time(double frame_current_pts, VideoState *is)
1031
+{
1032
+    double delay, sync_threshold, diff;
1033
+
1034
+    /* compute nominal delay */
1035
+    delay = frame_current_pts - is->frame_last_pts;
1036
+    if (delay <= 0 || delay >= 10.0) {
1037
+        /* if incorrect delay, use previous one */
1038
+        delay = is->frame_last_delay;
1039
+    } else {
1040
+        is->frame_last_delay = delay;
1041
+    }
1042
+    is->frame_last_pts = frame_current_pts;
1043
+
1044
+    /* update delay to follow master synchronisation source */
1045
+    if (((is->av_sync_type == AV_SYNC_AUDIO_MASTER && is->audio_st) ||
1046
+         is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
1047
+        /* if video is slave, we try to correct big delays by
1048
+           duplicating or deleting a frame */
1049
+        diff = get_video_clock(is) - get_master_clock(is);
1050
+
1051
+        /* skip or repeat frame. We take into account the
1052
+           delay to compute the threshold. I still don't know
1053
+           if it is the best guess */
1054
+        sync_threshold = FFMAX(AV_SYNC_THRESHOLD, delay);
1055
+        if (fabs(diff) < AV_NOSYNC_THRESHOLD) {
1056
+            if (diff <= -sync_threshold)
1057
+                delay = 0;
1058
+            else if (diff >= sync_threshold)
1059
+                delay = 2 * delay;
1060
+        }
1061
+    }
1062
+    is->frame_timer += delay;
1063
+
1064
+    av_dlog(NULL, "video: delay=%0.3f pts=%0.3f A-V=%f\n",
1065
+            delay, frame_current_pts, -diff);
1066
+
1067
+    return is->frame_timer;
1068
+}
1069
+
1070
+/* called to display each frame */
1071
+static void video_refresh_timer(void *opaque)
1072
+{
1073
+    VideoState *is = opaque;
1074
+    VideoPicture *vp;
1075
+
1076
+    SubPicture *sp, *sp2;
1077
+
1078
+    if (is->video_st) {
1079
+retry:
1080
+        if (is->pictq_size == 0) {
1081
+            //nothing to do, no picture to display in the que
1082
+        } else {
1083
+            double time= av_gettime()/1000000.0;
1084
+            double next_target;
1085
+            /* dequeue the picture */
1086
+            vp = &is->pictq[is->pictq_rindex];
1087
+
1088
+            if(time < vp->target_clock)
1089
+                return;
1090
+            /* update current video pts */
1091
+            is->video_current_pts = vp->pts;
1092
+            is->video_current_pts_drift = is->video_current_pts - time;
1093
+            is->video_current_pos = vp->pos;
1094
+            if(is->pictq_size > 1){
1095
+                VideoPicture *nextvp= &is->pictq[(is->pictq_rindex+1)%VIDEO_PICTURE_QUEUE_SIZE];
1096
+                assert(nextvp->target_clock >= vp->target_clock);
1097
+                next_target= nextvp->target_clock;
1098
+            }else{
1099
+                next_target= vp->target_clock + is->video_clock - vp->pts; //FIXME pass durations cleanly
1100
+            }
1101
+            if(framedrop && time > next_target){
1102
+                is->skip_frames *= 1.0 + FRAME_SKIP_FACTOR;
1103
+                if(is->pictq_size > 1 || time > next_target + 0.5){
1104
+                    /* update queue size and signal for next picture */
1105
+                    if (++is->pictq_rindex == VIDEO_PICTURE_QUEUE_SIZE)
1106
+                        is->pictq_rindex = 0;
1107
+
1108
+                    SDL_LockMutex(is->pictq_mutex);
1109
+                    is->pictq_size--;
1110
+                    SDL_CondSignal(is->pictq_cond);
1111
+                    SDL_UnlockMutex(is->pictq_mutex);
1112
+                    goto retry;
1113
+                }
1114
+            }
1115
+
1116
+            if(is->subtitle_st) {
1117
+                if (is->subtitle_stream_changed) {
1118
+                    SDL_LockMutex(is->subpq_mutex);
1119
+
1120
+                    while (is->subpq_size) {
1121
+                        free_subpicture(&is->subpq[is->subpq_rindex]);
1122
+
1123
+                        /* update queue size and signal for next picture */
1124
+                        if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
1125
+                            is->subpq_rindex = 0;
1126
+
1127
+                        is->subpq_size--;
1128
+                    }
1129
+                    is->subtitle_stream_changed = 0;
1130
+
1131
+                    SDL_CondSignal(is->subpq_cond);
1132
+                    SDL_UnlockMutex(is->subpq_mutex);
1133
+                } else {
1134
+                    if (is->subpq_size > 0) {
1135
+                        sp = &is->subpq[is->subpq_rindex];
1136
+
1137
+                        if (is->subpq_size > 1)
1138
+                            sp2 = &is->subpq[(is->subpq_rindex + 1) % SUBPICTURE_QUEUE_SIZE];
1139
+                        else
1140
+                            sp2 = NULL;
1141
+
1142
+                        if ((is->video_current_pts > (sp->pts + ((float) sp->sub.end_display_time / 1000)))
1143
+                                || (sp2 && is->video_current_pts > (sp2->pts + ((float) sp2->sub.start_display_time / 1000))))
1144
+                        {
1145
+                            free_subpicture(sp);
1146
+
1147
+                            /* update queue size and signal for next picture */
1148
+                            if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
1149
+                                is->subpq_rindex = 0;
1150
+
1151
+                            SDL_LockMutex(is->subpq_mutex);
1152
+                            is->subpq_size--;
1153
+                            SDL_CondSignal(is->subpq_cond);
1154
+                            SDL_UnlockMutex(is->subpq_mutex);
1155
+                        }
1156
+                    }
1157
+                }
1158
+            }
1159
+
1160
+            /* display picture */
1161
+            if (!display_disable)
1162
+                video_display(is);
1163
+
1164
+            /* update queue size and signal for next picture */
1165
+            if (++is->pictq_rindex == VIDEO_PICTURE_QUEUE_SIZE)
1166
+                is->pictq_rindex = 0;
1167
+
1168
+            SDL_LockMutex(is->pictq_mutex);
1169
+            is->pictq_size--;
1170
+            SDL_CondSignal(is->pictq_cond);
1171
+            SDL_UnlockMutex(is->pictq_mutex);
1172
+        }
1173
+    } else if (is->audio_st) {
1174
+        /* draw the next audio frame */
1175
+
1176
+        /* if only audio stream, then display the audio bars (better
1177
+           than nothing, just to test the implementation */
1178
+
1179
+        /* display picture */
1180
+        if (!display_disable)
1181
+            video_display(is);
1182
+    }
1183
+    if (show_status) {
1184
+        static int64_t last_time;
1185
+        int64_t cur_time;
1186
+        int aqsize, vqsize, sqsize;
1187
+        double av_diff;
1188
+
1189
+        cur_time = av_gettime();
1190
+        if (!last_time || (cur_time - last_time) >= 30000) {
1191
+            aqsize = 0;
1192
+            vqsize = 0;
1193
+            sqsize = 0;
1194
+            if (is->audio_st)
1195
+                aqsize = is->audioq.size;
1196
+            if (is->video_st)
1197
+                vqsize = is->videoq.size;
1198
+            if (is->subtitle_st)
1199
+                sqsize = is->subtitleq.size;
1200
+            av_diff = 0;
1201
+            if (is->audio_st && is->video_st)
1202
+                av_diff = get_audio_clock(is) - get_video_clock(is);
1203
+            printf("%7.2f A-V:%7.3f s:%3.1f aq=%5dKB vq=%5dKB sq=%5dB f=%"PRId64"/%"PRId64"   \r",
1204
+                   get_master_clock(is), av_diff, FFMAX(is->skip_frames-1, 0), aqsize / 1024, vqsize / 1024, sqsize, is->pts_ctx.num_faulty_dts, is->pts_ctx.num_faulty_pts);
1205
+            fflush(stdout);
1206
+            last_time = cur_time;
1207
+        }
1208
+    }
1209
+}
1210
+
1211
+static void stream_close(VideoState *is)
1212
+{
1213
+    VideoPicture *vp;
1214
+    int i;
1215
+    /* XXX: use a special url_shutdown call to abort parse cleanly */
1216
+    is->abort_request = 1;
1217
+    SDL_WaitThread(is->parse_tid, NULL);
1218
+    SDL_WaitThread(is->refresh_tid, NULL);
1219
+
1220
+    /* free all pictures */
1221
+    for(i=0;i<VIDEO_PICTURE_QUEUE_SIZE; i++) {
1222
+        vp = &is->pictq[i];
1223
+#if CONFIG_AVFILTER
1224
+        if (vp->picref) {
1225
+            avfilter_unref_buffer(vp->picref);
1226
+            vp->picref = NULL;
1227
+        }
1228
+#endif
1229
+        if (vp->bmp) {
1230
+            SDL_FreeYUVOverlay(vp->bmp);
1231
+            vp->bmp = NULL;
1232
+        }
1233
+    }
1234
+    SDL_DestroyMutex(is->pictq_mutex);
1235
+    SDL_DestroyCond(is->pictq_cond);
1236
+    SDL_DestroyMutex(is->subpq_mutex);
1237
+    SDL_DestroyCond(is->subpq_cond);
1238
+#if !CONFIG_AVFILTER
1239
+    if (is->img_convert_ctx)
1240
+        sws_freeContext(is->img_convert_ctx);
1241
+#endif
1242
+    av_free(is);
1243
+}
1244
+
1245
+static void do_exit(void)
1246
+{
1247
+    if (cur_stream) {
1248
+        stream_close(cur_stream);
1249
+        cur_stream = NULL;
1250
+    }
1251
+    uninit_opts();
1252
+#if CONFIG_AVFILTER
1253
+    avfilter_uninit();
1254
+#endif
1255
+    if (show_status)
1256
+        printf("\n");
1257
+    SDL_Quit();
1258
+    av_log(NULL, AV_LOG_QUIET, "");
1259
+    exit(0);
1260
+}
1261
+
1262
+/* allocate a picture (needs to do that in main thread to avoid
1263
+   potential locking problems */
1264
+static void alloc_picture(void *opaque)
1265
+{
1266
+    VideoState *is = opaque;
1267
+    VideoPicture *vp;
1268
+
1269
+    vp = &is->pictq[is->pictq_windex];
1270
+
1271
+    if (vp->bmp)
1272
+        SDL_FreeYUVOverlay(vp->bmp);
1273
+
1274
+#if CONFIG_AVFILTER
1275
+    if (vp->picref)
1276
+        avfilter_unref_buffer(vp->picref);
1277
+    vp->picref = NULL;
1278
+
1279
+    vp->width   = is->out_video_filter->inputs[0]->w;
1280
+    vp->height  = is->out_video_filter->inputs[0]->h;
1281
+    vp->pix_fmt = is->out_video_filter->inputs[0]->format;
1282
+#else
1283
+    vp->width   = is->video_st->codec->width;
1284
+    vp->height  = is->video_st->codec->height;
1285
+    vp->pix_fmt = is->video_st->codec->pix_fmt;
1286
+#endif
1287
+
1288
+    vp->bmp = SDL_CreateYUVOverlay(vp->width, vp->height,
1289
+                                   SDL_YV12_OVERLAY,
1290
+                                   screen);
1291
+    if (!vp->bmp || vp->bmp->pitches[0] < vp->width) {
1292
+        /* SDL allocates a buffer smaller than requested if the video
1293
+         * overlay hardware is unable to support the requested size. */
1294
+        fprintf(stderr, "Error: the video system does not support an image\n"
1295
+                        "size of %dx%d pixels. Try using -lowres or -vf \"scale=w:h\"\n"
1296
+                        "to reduce the image size.\n", vp->width, vp->height );
1297
+        do_exit();
1298
+    }
1299
+
1300
+    SDL_LockMutex(is->pictq_mutex);
1301
+    vp->allocated = 1;
1302
+    SDL_CondSignal(is->pictq_cond);
1303
+    SDL_UnlockMutex(is->pictq_mutex);
1304
+}
1305
+
1306
+/**
1307
+ *
1308
+ * @param pts the dts of the pkt / pts of the frame and guessed if not known
1309
+ */
1310
+static int queue_picture(VideoState *is, AVFrame *src_frame, double pts, int64_t pos)
1311
+{
1312
+    VideoPicture *vp;
1313
+#if CONFIG_AVFILTER
1314
+    AVPicture pict_src;
1315
+#else
1316
+    int dst_pix_fmt = PIX_FMT_YUV420P;
1317
+#endif
1318
+    /* wait until we have space to put a new picture */
1319
+    SDL_LockMutex(is->pictq_mutex);
1320
+
1321
+    if(is->pictq_size>=VIDEO_PICTURE_QUEUE_SIZE && !is->refresh)
1322
+        is->skip_frames= FFMAX(1.0 - FRAME_SKIP_FACTOR, is->skip_frames * (1.0-FRAME_SKIP_FACTOR));
1323
+
1324
+    while (is->pictq_size >= VIDEO_PICTURE_QUEUE_SIZE &&
1325
+           !is->videoq.abort_request) {
1326
+        SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1327
+    }
1328
+    SDL_UnlockMutex(is->pictq_mutex);
1329
+
1330
+    if (is->videoq.abort_request)
1331
+        return -1;
1332
+
1333
+    vp = &is->pictq[is->pictq_windex];
1334
+
1335
+    /* alloc or resize hardware picture buffer */
1336
+    if (!vp->bmp ||
1337
+#if CONFIG_AVFILTER
1338
+        vp->width  != is->out_video_filter->inputs[0]->w ||
1339
+        vp->height != is->out_video_filter->inputs[0]->h) {
1340
+#else
1341
+        vp->width != is->video_st->codec->width ||
1342
+        vp->height != is->video_st->codec->height) {
1343
+#endif
1344
+        SDL_Event event;
1345
+
1346
+        vp->allocated = 0;
1347
+
1348
+        /* the allocation must be done in the main thread to avoid
1349
+           locking problems */
1350
+        event.type = FF_ALLOC_EVENT;
1351
+        event.user.data1 = is;
1352
+        SDL_PushEvent(&event);
1353
+
1354
+        /* wait until the picture is allocated */
1355
+        SDL_LockMutex(is->pictq_mutex);
1356
+        while (!vp->allocated && !is->videoq.abort_request) {
1357
+            SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1358
+        }
1359
+        SDL_UnlockMutex(is->pictq_mutex);
1360
+
1361
+        if (is->videoq.abort_request)
1362
+            return -1;
1363
+    }
1364
+
1365
+    /* if the frame is not skipped, then display it */
1366
+    if (vp->bmp) {
1367
+        AVPicture pict;
1368
+#if CONFIG_AVFILTER
1369
+        if(vp->picref)
1370
+            avfilter_unref_buffer(vp->picref);
1371
+        vp->picref = src_frame->opaque;
1372
+#endif
1373
+
1374
+        /* get a pointer on the bitmap */
1375
+        SDL_LockYUVOverlay (vp->bmp);
1376
+
1377
+        memset(&pict,0,sizeof(AVPicture));
1378
+        pict.data[0] = vp->bmp->pixels[0];
1379
+        pict.data[1] = vp->bmp->pixels[2];
1380
+        pict.data[2] = vp->bmp->pixels[1];
1381
+
1382
+        pict.linesize[0] = vp->bmp->pitches[0];
1383
+        pict.linesize[1] = vp->bmp->pitches[2];
1384
+        pict.linesize[2] = vp->bmp->pitches[1];
1385
+
1386
+#if CONFIG_AVFILTER
1387
+        pict_src.data[0] = src_frame->data[0];
1388
+        pict_src.data[1] = src_frame->data[1];
1389
+        pict_src.data[2] = src_frame->data[2];
1390
+
1391
+        pict_src.linesize[0] = src_frame->linesize[0];
1392
+        pict_src.linesize[1] = src_frame->linesize[1];
1393
+        pict_src.linesize[2] = src_frame->linesize[2];
1394
+
1395
+        //FIXME use direct rendering
1396
+        av_picture_copy(&pict, &pict_src,
1397
+                        vp->pix_fmt, vp->width, vp->height);
1398
+#else
1399
+        sws_flags = av_get_int(sws_opts, "sws_flags", NULL);
1400
+        is->img_convert_ctx = sws_getCachedContext(is->img_convert_ctx,
1401
+            vp->width, vp->height, vp->pix_fmt, vp->width, vp->height,
1402
+            dst_pix_fmt, sws_flags, NULL, NULL, NULL);
1403
+        if (is->img_convert_ctx == NULL) {
1404
+            fprintf(stderr, "Cannot initialize the conversion context\n");
1405
+            exit(1);
1406
+        }
1407
+        sws_scale(is->img_convert_ctx, src_frame->data, src_frame->linesize,
1408
+                  0, vp->height, pict.data, pict.linesize);
1409
+#endif
1410
+        /* update the bitmap content */
1411
+        SDL_UnlockYUVOverlay(vp->bmp);
1412
+
1413
+        vp->pts = pts;
1414
+        vp->pos = pos;
1415
+
1416
+        /* now we can update the picture count */
1417
+        if (++is->pictq_windex == VIDEO_PICTURE_QUEUE_SIZE)
1418
+            is->pictq_windex = 0;
1419
+        SDL_LockMutex(is->pictq_mutex);
1420
+        vp->target_clock= compute_target_time(vp->pts, is);
1421
+
1422
+        is->pictq_size++;
1423
+        SDL_UnlockMutex(is->pictq_mutex);
1424
+    }
1425
+    return 0;
1426
+}
1427
+
1428
+/**
1429
+ * compute the exact PTS for the picture if it is omitted in the stream
1430
+ * @param pts1 the dts of the pkt / pts of the frame
1431
+ */
1432
+static int output_picture2(VideoState *is, AVFrame *src_frame, double pts1, int64_t pos)
1433
+{
1434
+    double frame_delay, pts;
1435
+
1436
+    pts = pts1;
1437
+
1438
+    if (pts != 0) {
1439
+        /* update video clock with pts, if present */
1440
+        is->video_clock = pts;
1441
+    } else {
1442
+        pts = is->video_clock;
1443
+    }
1444
+    /* update video clock for next frame */
1445
+    frame_delay = av_q2d(is->video_st->codec->time_base);
1446
+    /* for MPEG2, the frame can be repeated, so we update the
1447
+       clock accordingly */
1448
+    frame_delay += src_frame->repeat_pict * (frame_delay * 0.5);
1449
+    is->video_clock += frame_delay;
1450
+
1451
+    return queue_picture(is, src_frame, pts, pos);
1452
+}
1453
+
1454
+static int get_video_frame(VideoState *is, AVFrame *frame, int64_t *pts, AVPacket *pkt)
1455
+{
1456
+    int got_picture, i;
1457
+
1458
+    if (packet_queue_get(&is->videoq, pkt, 1) < 0)
1459
+        return -1;
1460
+
1461
+    if (pkt->data == flush_pkt.data) {
1462
+        avcodec_flush_buffers(is->video_st->codec);
1463
+
1464
+        SDL_LockMutex(is->pictq_mutex);
1465
+        //Make sure there are no long delay timers (ideally we should just flush the que but thats harder)
1466
+        for (i = 0; i < VIDEO_PICTURE_QUEUE_SIZE; i++) {
1467
+            is->pictq[i].target_clock= 0;
1468
+        }
1469
+        while (is->pictq_size && !is->videoq.abort_request) {
1470
+            SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1471
+        }
1472
+        is->video_current_pos = -1;
1473
+        SDL_UnlockMutex(is->pictq_mutex);
1474
+
1475
+        init_pts_correction(&is->pts_ctx);
1476
+        is->frame_last_pts = AV_NOPTS_VALUE;
1477
+        is->frame_last_delay = 0;
1478
+        is->frame_timer = (double)av_gettime() / 1000000.0;
1479
+        is->skip_frames = 1;
1480
+        is->skip_frames_index = 0;
1481
+        return 0;
1482
+    }
1483
+
1484
+    avcodec_decode_video2(is->video_st->codec, frame, &got_picture, pkt);
1485
+
1486
+    if (got_picture) {
1487
+        if (decoder_reorder_pts == -1) {
1488
+            *pts = guess_correct_pts(&is->pts_ctx, frame->pkt_pts, frame->pkt_dts);
1489
+        } else if (decoder_reorder_pts) {
1490
+            *pts = frame->pkt_pts;
1491
+        } else {
1492
+            *pts = frame->pkt_dts;
1493
+        }
1494
+
1495
+        if (*pts == AV_NOPTS_VALUE) {
1496
+            *pts = 0;
1497
+        }
1498
+
1499
+        is->skip_frames_index += 1;
1500
+        if(is->skip_frames_index >= is->skip_frames){
1501
+            is->skip_frames_index -= FFMAX(is->skip_frames, 1.0);
1502
+            return 1;
1503
+        }
1504
+
1505
+    }
1506
+    return 0;
1507
+}
1508
+
1509
+#if CONFIG_AVFILTER
1510
+typedef struct {
1511
+    VideoState *is;
1512
+    AVFrame *frame;
1513
+    int use_dr1;
1514
+} FilterPriv;
1515
+
1516
+static int input_get_buffer(AVCodecContext *codec, AVFrame *pic)
1517
+{
1518
+    AVFilterContext *ctx = codec->opaque;
1519
+    AVFilterBufferRef  *ref;
1520
+    int perms = AV_PERM_WRITE;
1521
+    int i, w, h, stride[4];
1522
+    unsigned edge;
1523
+    int pixel_size;
1524
+
1525
+    if (codec->codec->capabilities & CODEC_CAP_NEG_LINESIZES)
1526
+        perms |= AV_PERM_NEG_LINESIZES;
1527
+
1528
+    if(pic->buffer_hints & FF_BUFFER_HINTS_VALID) {
1529
+        if(pic->buffer_hints & FF_BUFFER_HINTS_READABLE) perms |= AV_PERM_READ;
1530
+        if(pic->buffer_hints & FF_BUFFER_HINTS_PRESERVE) perms |= AV_PERM_PRESERVE;
1531
+        if(pic->buffer_hints & FF_BUFFER_HINTS_REUSABLE) perms |= AV_PERM_REUSE2;
1532
+    }
1533
+    if(pic->reference) perms |= AV_PERM_READ | AV_PERM_PRESERVE;
1534
+
1535
+    w = codec->width;
1536
+    h = codec->height;
1537
+    avcodec_align_dimensions2(codec, &w, &h, stride);
1538
+    edge = codec->flags & CODEC_FLAG_EMU_EDGE ? 0 : avcodec_get_edge_width();
1539
+    w += edge << 1;
1540
+    h += edge << 1;
1541
+
1542
+    if(!(ref = avfilter_get_video_buffer(ctx->outputs[0], perms, w, h)))
1543
+        return -1;
1544
+
1545
+    pixel_size = av_pix_fmt_descriptors[ref->format].comp[0].step_minus1+1;
1546
+    ref->video->w = codec->width;
1547
+    ref->video->h = codec->height;
1548
+    for(i = 0; i < 4; i ++) {
1549
+        unsigned hshift = (i == 1 || i == 2) ? av_pix_fmt_descriptors[ref->format].log2_chroma_w : 0;
1550
+        unsigned vshift = (i == 1 || i == 2) ? av_pix_fmt_descriptors[ref->format].log2_chroma_h : 0;
1551
+
1552
+        if (ref->data[i]) {
1553
+            ref->data[i]    += ((edge * pixel_size) >> hshift) + ((edge * ref->linesize[i]) >> vshift);
1554
+        }
1555
+        pic->data[i]     = ref->data[i];
1556
+        pic->linesize[i] = ref->linesize[i];
1557
+    }
1558
+    pic->opaque = ref;
1559
+    pic->age    = INT_MAX;
1560
+    pic->type   = FF_BUFFER_TYPE_USER;
1561
+    pic->reordered_opaque = codec->reordered_opaque;
1562
+    if(codec->pkt) pic->pkt_pts = codec->pkt->pts;
1563
+    else           pic->pkt_pts = AV_NOPTS_VALUE;
1564
+    return 0;
1565
+}
1566
+
1567
+static void input_release_buffer(AVCodecContext *codec, AVFrame *pic)
1568
+{
1569
+    memset(pic->data, 0, sizeof(pic->data));
1570
+    avfilter_unref_buffer(pic->opaque);
1571
+}
1572
+
1573
+static int input_reget_buffer(AVCodecContext *codec, AVFrame *pic)
1574
+{
1575
+    AVFilterBufferRef *ref = pic->opaque;
1576
+
1577
+    if (pic->data[0] == NULL) {
1578
+        pic->buffer_hints |= FF_BUFFER_HINTS_READABLE;
1579
+        return codec->get_buffer(codec, pic);
1580
+    }
1581
+
1582
+    if ((codec->width != ref->video->w) || (codec->height != ref->video->h) ||
1583
+        (codec->pix_fmt != ref->format)) {
1584
+        av_log(codec, AV_LOG_ERROR, "Picture properties changed.\n");
1585
+        return -1;
1586
+    }
1587
+
1588
+    pic->reordered_opaque = codec->reordered_opaque;
1589
+    if(codec->pkt) pic->pkt_pts = codec->pkt->pts;
1590
+    else           pic->pkt_pts = AV_NOPTS_VALUE;
1591
+    return 0;
1592
+}
1593
+
1594
+static int input_init(AVFilterContext *ctx, const char *args, void *opaque)
1595
+{
1596
+    FilterPriv *priv = ctx->priv;
1597
+    AVCodecContext *codec;
1598
+    if(!opaque) return -1;
1599
+
1600
+    priv->is = opaque;
1601
+    codec    = priv->is->video_st->codec;
1602
+    codec->opaque = ctx;
1603
+    if(codec->codec->capabilities & CODEC_CAP_DR1) {
1604
+        priv->use_dr1 = 1;
1605
+        codec->get_buffer     = input_get_buffer;
1606
+        codec->release_buffer = input_release_buffer;
1607
+        codec->reget_buffer   = input_reget_buffer;
1608
+        codec->thread_safe_callbacks = 1;
1609
+    }
1610
+
1611
+    priv->frame = avcodec_alloc_frame();
1612
+
1613
+    return 0;
1614
+}
1615
+
1616
+static void input_uninit(AVFilterContext *ctx)
1617
+{
1618
+    FilterPriv *priv = ctx->priv;
1619
+    av_free(priv->frame);
1620
+}
1621
+
1622
+static int input_request_frame(AVFilterLink *link)
1623
+{
1624
+    FilterPriv *priv = link->src->priv;
1625
+    AVFilterBufferRef *picref;
1626
+    int64_t pts = 0;
1627
+    AVPacket pkt;
1628
+    int ret;
1629
+
1630
+    while (!(ret = get_video_frame(priv->is, priv->frame, &pts, &pkt)))
1631
+        av_free_packet(&pkt);
1632
+    if (ret < 0)
1633
+        return -1;
1634
+
1635
+    if(priv->use_dr1) {
1636
+        picref = avfilter_ref_buffer(priv->frame->opaque, ~0);
1637
+    } else {
1638
+        picref = avfilter_get_video_buffer(link, AV_PERM_WRITE, link->w, link->h);
1639
+        av_image_copy(picref->data, picref->linesize,
1640
+                      priv->frame->data, priv->frame->linesize,
1641
+                      picref->format, link->w, link->h);
1642
+    }
1643
+    av_free_packet(&pkt);
1644
+
1645
+    picref->pts = pts;
1646
+    picref->pos = pkt.pos;
1647
+    picref->video->pixel_aspect = priv->is->video_st->codec->sample_aspect_ratio;
1648
+    avfilter_start_frame(link, picref);
1649
+    avfilter_draw_slice(link, 0, link->h, 1);
1650
+    avfilter_end_frame(link);
1651
+
1652
+    return 0;
1653
+}
1654
+
1655
+static int input_query_formats(AVFilterContext *ctx)
1656
+{
1657
+    FilterPriv *priv = ctx->priv;
1658
+    enum PixelFormat pix_fmts[] = {
1659
+        priv->is->video_st->codec->pix_fmt, PIX_FMT_NONE
1660
+    };
1661
+
1662
+    avfilter_set_common_formats(ctx, avfilter_make_format_list(pix_fmts));
1663
+    return 0;
1664
+}
1665
+
1666
+static int input_config_props(AVFilterLink *link)
1667
+{
1668
+    FilterPriv *priv  = link->src->priv;
1669
+    AVCodecContext *c = priv->is->video_st->codec;
1670
+
1671
+    link->w = c->width;
1672
+    link->h = c->height;
1673
+    link->time_base = priv->is->video_st->time_base;
1674
+
1675
+    return 0;
1676
+}
1677
+
1678
+static AVFilter input_filter =
1679
+{
1680
+    .name      = "avplay_input",
1681
+
1682
+    .priv_size = sizeof(FilterPriv),
1683
+
1684
+    .init      = input_init,
1685
+    .uninit    = input_uninit,
1686
+
1687
+    .query_formats = input_query_formats,
1688
+
1689
+    .inputs    = (AVFilterPad[]) {{ .name = NULL }},
1690
+    .outputs   = (AVFilterPad[]) {{ .name = "default",
1691
+                                    .type = AVMEDIA_TYPE_VIDEO,
1692
+                                    .request_frame = input_request_frame,
1693
+                                    .config_props  = input_config_props, },
1694
+                                  { .name = NULL }},
1695
+};
1696
+
1697
+static int configure_video_filters(AVFilterGraph *graph, VideoState *is, const char *vfilters)
1698
+{
1699
+    char sws_flags_str[128];
1700
+    int ret;
1701
+    FFSinkContext ffsink_ctx = { .pix_fmt = PIX_FMT_YUV420P };
1702
+    AVFilterContext *filt_src = NULL, *filt_out = NULL;
1703
+    snprintf(sws_flags_str, sizeof(sws_flags_str), "flags=%d", sws_flags);
1704
+    graph->scale_sws_opts = av_strdup(sws_flags_str);
1705
+
1706
+    if ((ret = avfilter_graph_create_filter(&filt_src, &input_filter, "src",
1707
+                                            NULL, is, graph)) < 0)
1708
+        return ret;
1709
+    if ((ret = avfilter_graph_create_filter(&filt_out, &ffsink, "out",
1710
+                                            NULL, &ffsink_ctx, graph)) < 0)
1711
+        return ret;
1712
+
1713
+    if(vfilters) {
1714
+        AVFilterInOut *outputs = av_malloc(sizeof(AVFilterInOut));
1715
+        AVFilterInOut *inputs  = av_malloc(sizeof(AVFilterInOut));
1716
+
1717
+        outputs->name    = av_strdup("in");
1718
+        outputs->filter_ctx = filt_src;
1719
+        outputs->pad_idx = 0;
1720
+        outputs->next    = NULL;
1721
+
1722
+        inputs->name    = av_strdup("out");
1723
+        inputs->filter_ctx = filt_out;
1724
+        inputs->pad_idx = 0;
1725
+        inputs->next    = NULL;
1726
+
1727
+        if ((ret = avfilter_graph_parse(graph, vfilters, inputs, outputs, NULL)) < 0)
1728
+            return ret;
1729
+        av_freep(&vfilters);
1730
+    } else {
1731
+        if ((ret = avfilter_link(filt_src, 0, filt_out, 0)) < 0)
1732
+            return ret;
1733
+    }
1734
+
1735
+    if ((ret = avfilter_graph_config(graph, NULL)) < 0)
1736
+        return ret;
1737
+
1738
+    is->out_video_filter = filt_out;
1739
+
1740
+    return ret;
1741
+}
1742
+
1743
+#endif  /* CONFIG_AVFILTER */
1744
+
1745
+static int video_thread(void *arg)
1746
+{
1747
+    VideoState *is = arg;
1748
+    AVFrame *frame= avcodec_alloc_frame();
1749
+    int64_t pts_int;
1750
+    double pts;
1751
+    int ret;
1752
+
1753
+#if CONFIG_AVFILTER
1754
+    AVFilterGraph *graph = avfilter_graph_alloc();
1755
+    AVFilterContext *filt_out = NULL;
1756
+    int64_t pos;
1757
+
1758
+    if ((ret = configure_video_filters(graph, is, vfilters)) < 0)
1759
+        goto the_end;
1760
+    filt_out = is->out_video_filter;
1761
+#endif
1762
+
1763
+    for(;;) {
1764
+#if !CONFIG_AVFILTER
1765
+        AVPacket pkt;
1766
+#else
1767
+        AVFilterBufferRef *picref;
1768
+        AVRational tb;
1769
+#endif
1770
+        while (is->paused && !is->videoq.abort_request)
1771
+            SDL_Delay(10);
1772
+#if CONFIG_AVFILTER
1773
+        ret = get_filtered_video_frame(filt_out, frame, &picref, &tb);
1774
+        if (picref) {
1775
+            pts_int = picref->pts;
1776
+            pos     = picref->pos;
1777
+            frame->opaque = picref;
1778
+        }
1779
+
1780
+        if (av_cmp_q(tb, is->video_st->time_base)) {
1781
+            av_unused int64_t pts1 = pts_int;
1782
+            pts_int = av_rescale_q(pts_int, tb, is->video_st->time_base);
1783
+            av_dlog(NULL, "video_thread(): "
1784
+                    "tb:%d/%d pts:%"PRId64" -> tb:%d/%d pts:%"PRId64"\n",
1785
+                    tb.num, tb.den, pts1,
1786
+                    is->video_st->time_base.num, is->video_st->time_base.den, pts_int);
1787
+        }
1788
+#else
1789
+        ret = get_video_frame(is, frame, &pts_int, &pkt);
1790
+#endif
1791
+
1792
+        if (ret < 0) goto the_end;
1793
+
1794
+        if (!ret)
1795
+            continue;
1796
+
1797
+        pts = pts_int*av_q2d(is->video_st->time_base);
1798
+
1799
+#if CONFIG_AVFILTER
1800
+        ret = output_picture2(is, frame, pts, pos);
1801
+#else
1802
+        ret = output_picture2(is, frame, pts,  pkt.pos);
1803
+        av_free_packet(&pkt);
1804
+#endif
1805
+        if (ret < 0)
1806
+            goto the_end;
1807
+
1808
+        if (step)
1809
+            if (cur_stream)
1810
+                stream_pause(cur_stream);
1811
+    }
1812
+ the_end:
1813
+#if CONFIG_AVFILTER
1814
+    avfilter_graph_free(&graph);
1815
+#endif
1816
+    av_free(frame);
1817
+    return 0;
1818
+}
1819
+
1820
+static int subtitle_thread(void *arg)
1821
+{
1822
+    VideoState *is = arg;
1823
+    SubPicture *sp;
1824
+    AVPacket pkt1, *pkt = &pkt1;
1825
+    int got_subtitle;
1826
+    double pts;
1827
+    int i, j;
1828
+    int r, g, b, y, u, v, a;
1829
+
1830
+    for(;;) {
1831
+        while (is->paused && !is->subtitleq.abort_request) {
1832
+            SDL_Delay(10);
1833
+        }
1834
+        if (packet_queue_get(&is->subtitleq, pkt, 1) < 0)
1835
+            break;
1836
+
1837
+        if(pkt->data == flush_pkt.data){
1838
+            avcodec_flush_buffers(is->subtitle_st->codec);
1839
+            continue;
1840
+        }
1841
+        SDL_LockMutex(is->subpq_mutex);
1842
+        while (is->subpq_size >= SUBPICTURE_QUEUE_SIZE &&
1843
+               !is->subtitleq.abort_request) {
1844
+            SDL_CondWait(is->subpq_cond, is->subpq_mutex);
1845
+        }
1846
+        SDL_UnlockMutex(is->subpq_mutex);
1847
+
1848
+        if (is->subtitleq.abort_request)
1849
+            return 0;
1850
+
1851
+        sp = &is->subpq[is->subpq_windex];
1852
+
1853
+       /* NOTE: ipts is the PTS of the _first_ picture beginning in
1854
+           this packet, if any */
1855
+        pts = 0;
1856
+        if (pkt->pts != AV_NOPTS_VALUE)
1857
+            pts = av_q2d(is->subtitle_st->time_base)*pkt->pts;
1858
+
1859
+        avcodec_decode_subtitle2(is->subtitle_st->codec, &sp->sub,
1860
+                                 &got_subtitle, pkt);
1861
+
1862
+        if (got_subtitle && sp->sub.format == 0) {
1863
+            sp->pts = pts;
1864
+
1865
+            for (i = 0; i < sp->sub.num_rects; i++)
1866
+            {
1867
+                for (j = 0; j < sp->sub.rects[i]->nb_colors; j++)
1868
+                {
1869
+                    RGBA_IN(r, g, b, a, (uint32_t*)sp->sub.rects[i]->pict.data[1] + j);
1870
+                    y = RGB_TO_Y_CCIR(r, g, b);
1871
+                    u = RGB_TO_U_CCIR(r, g, b, 0);
1872
+                    v = RGB_TO_V_CCIR(r, g, b, 0);
1873
+                    YUVA_OUT((uint32_t*)sp->sub.rects[i]->pict.data[1] + j, y, u, v, a);
1874
+                }
1875
+            }
1876
+
1877
+            /* now we can update the picture count */
1878
+            if (++is->subpq_windex == SUBPICTURE_QUEUE_SIZE)
1879
+                is->subpq_windex = 0;
1880
+            SDL_LockMutex(is->subpq_mutex);
1881
+            is->subpq_size++;
1882
+            SDL_UnlockMutex(is->subpq_mutex);
1883
+        }
1884
+        av_free_packet(pkt);
1885
+    }
1886
+    return 0;
1887
+}
1888
+
1889
+/* copy samples for viewing in editor window */
1890
+static void update_sample_display(VideoState *is, short *samples, int samples_size)
1891
+{
1892
+    int size, len;
1893
+
1894
+    size = samples_size / sizeof(short);
1895
+    while (size > 0) {
1896
+        len = SAMPLE_ARRAY_SIZE - is->sample_array_index;
1897
+        if (len > size)
1898
+            len = size;
1899
+        memcpy(is->sample_array + is->sample_array_index, samples, len * sizeof(short));
1900
+        samples += len;
1901
+        is->sample_array_index += len;
1902
+        if (is->sample_array_index >= SAMPLE_ARRAY_SIZE)
1903
+            is->sample_array_index = 0;
1904
+        size -= len;
1905
+    }
1906
+}
1907
+
1908
+/* return the new audio buffer size (samples can be added or deleted
1909
+   to get better sync if video or external master clock) */
1910
+static int synchronize_audio(VideoState *is, short *samples,
1911
+                             int samples_size1, double pts)
1912
+{
1913
+    int n, samples_size;
1914
+    double ref_clock;
1915
+
1916
+    n = 2 * is->audio_st->codec->channels;
1917
+    samples_size = samples_size1;
1918
+
1919
+    /* if not master, then we try to remove or add samples to correct the clock */
1920
+    if (((is->av_sync_type == AV_SYNC_VIDEO_MASTER && is->video_st) ||
1921
+         is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
1922
+        double diff, avg_diff;
1923
+        int wanted_size, min_size, max_size, nb_samples;
1924
+
1925
+        ref_clock = get_master_clock(is);
1926
+        diff = get_audio_clock(is) - ref_clock;
1927
+
1928
+        if (diff < AV_NOSYNC_THRESHOLD) {
1929
+            is->audio_diff_cum = diff + is->audio_diff_avg_coef * is->audio_diff_cum;
1930
+            if (is->audio_diff_avg_count < AUDIO_DIFF_AVG_NB) {
1931
+                /* not enough measures to have a correct estimate */
1932
+                is->audio_diff_avg_count++;
1933
+            } else {
1934
+                /* estimate the A-V difference */
1935
+                avg_diff = is->audio_diff_cum * (1.0 - is->audio_diff_avg_coef);
1936
+
1937
+                if (fabs(avg_diff) >= is->audio_diff_threshold) {
1938
+                    wanted_size = samples_size + ((int)(diff * is->audio_st->codec->sample_rate) * n);
1939
+                    nb_samples = samples_size / n;
1940
+
1941
+                    min_size = ((nb_samples * (100 - SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
1942
+                    max_size = ((nb_samples * (100 + SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
1943
+                    if (wanted_size < min_size)
1944
+                        wanted_size = min_size;
1945
+                    else if (wanted_size > max_size)
1946
+                        wanted_size = max_size;
1947
+
1948
+                    /* add or remove samples to correction the synchro */
1949
+                    if (wanted_size < samples_size) {
1950
+                        /* remove samples */
1951
+                        samples_size = wanted_size;
1952
+                    } else if (wanted_size > samples_size) {
1953
+                        uint8_t *samples_end, *q;
1954
+                        int nb;
1955
+
1956
+                        /* add samples */
1957
+                        nb = (samples_size - wanted_size);
1958
+                        samples_end = (uint8_t *)samples + samples_size - n;
1959
+                        q = samples_end + n;
1960
+                        while (nb > 0) {
1961
+                            memcpy(q, samples_end, n);
1962
+                            q += n;
1963
+                            nb -= n;
1964
+                        }
1965
+                        samples_size = wanted_size;
1966
+                    }
1967
+                }
1968
+                av_dlog(NULL, "diff=%f adiff=%f sample_diff=%d apts=%0.3f vpts=%0.3f %f\n",
1969
+                        diff, avg_diff, samples_size - samples_size1,
1970
+                        is->audio_clock, is->video_clock, is->audio_diff_threshold);
1971
+            }
1972
+        } else {
1973
+            /* too big difference : may be initial PTS errors, so
1974
+               reset A-V filter */
1975
+            is->audio_diff_avg_count = 0;
1976
+            is->audio_diff_cum = 0;
1977
+        }
1978
+    }
1979
+
1980
+    return samples_size;
1981
+}
1982
+
1983
+/* decode one audio frame and returns its uncompressed size */
1984
+static int audio_decode_frame(VideoState *is, double *pts_ptr)
1985
+{
1986
+    AVPacket *pkt_temp = &is->audio_pkt_temp;
1987
+    AVPacket *pkt = &is->audio_pkt;
1988
+    AVCodecContext *dec= is->audio_st->codec;
1989
+    int n, len1, data_size;
1990
+    double pts;
1991
+
1992
+    for(;;) {
1993
+        /* NOTE: the audio packet can contain several frames */
1994
+        while (pkt_temp->size > 0) {
1995
+            data_size = sizeof(is->audio_buf1);
1996
+            len1 = avcodec_decode_audio3(dec,
1997
+                                        (int16_t *)is->audio_buf1, &data_size,
1998
+                                        pkt_temp);
1999
+            if (len1 < 0) {
2000
+                /* if error, we skip the frame */
2001
+                pkt_temp->size = 0;
2002
+                break;
2003
+            }
2004
+
2005
+            pkt_temp->data += len1;
2006
+            pkt_temp->size -= len1;
2007
+            if (data_size <= 0)
2008
+                continue;
2009
+
2010
+            if (dec->sample_fmt != is->audio_src_fmt) {
2011
+                if (is->reformat_ctx)
2012
+                    av_audio_convert_free(is->reformat_ctx);
2013
+                is->reformat_ctx= av_audio_convert_alloc(AV_SAMPLE_FMT_S16, 1,
2014
+                                                         dec->sample_fmt, 1, NULL, 0);
2015
+                if (!is->reformat_ctx) {
2016
+                    fprintf(stderr, "Cannot convert %s sample format to %s sample format\n",
2017
+                        av_get_sample_fmt_name(dec->sample_fmt),
2018
+                        av_get_sample_fmt_name(AV_SAMPLE_FMT_S16));
2019
+                        break;
2020
+                }
2021
+                is->audio_src_fmt= dec->sample_fmt;
2022
+            }
2023
+
2024
+            if (is->reformat_ctx) {
2025
+                const void *ibuf[6]= {is->audio_buf1};
2026
+                void *obuf[6]= {is->audio_buf2};
2027
+                int istride[6]= {av_get_bytes_per_sample(dec->sample_fmt)};
2028
+                int ostride[6]= {2};
2029
+                int len= data_size/istride[0];
2030
+                if (av_audio_convert(is->reformat_ctx, obuf, ostride, ibuf, istride, len)<0) {
2031
+                    printf("av_audio_convert() failed\n");
2032
+                    break;
2033
+                }
2034
+                is->audio_buf= is->audio_buf2;
2035
+                /* FIXME: existing code assume that data_size equals framesize*channels*2
2036
+                          remove this legacy cruft */
2037
+                data_size= len*2;
2038
+            }else{
2039
+                is->audio_buf= is->audio_buf1;
2040
+            }
2041
+
2042
+            /* if no pts, then compute it */
2043
+            pts = is->audio_clock;
2044
+            *pts_ptr = pts;
2045
+            n = 2 * dec->channels;
2046
+            is->audio_clock += (double)data_size /
2047
+                (double)(n * dec->sample_rate);
2048
+#ifdef DEBUG
2049
+            {
2050
+                static double last_clock;
2051
+                printf("audio: delay=%0.3f clock=%0.3f pts=%0.3f\n",
2052
+                       is->audio_clock - last_clock,
2053
+                       is->audio_clock, pts);
2054
+                last_clock = is->audio_clock;
2055
+            }
2056
+#endif
2057
+            return data_size;
2058
+        }
2059
+
2060
+        /* free the current packet */
2061
+        if (pkt->data)
2062
+            av_free_packet(pkt);
2063
+
2064
+        if (is->paused || is->audioq.abort_request) {
2065
+            return -1;
2066
+        }
2067
+
2068
+        /* read next packet */
2069
+        if (packet_queue_get(&is->audioq, pkt, 1) < 0)
2070
+            return -1;
2071
+        if(pkt->data == flush_pkt.data){
2072
+            avcodec_flush_buffers(dec);
2073
+            continue;
2074
+        }
2075
+
2076
+        pkt_temp->data = pkt->data;
2077
+        pkt_temp->size = pkt->size;
2078
+
2079
+        /* if update the audio clock with the pts */
2080
+        if (pkt->pts != AV_NOPTS_VALUE) {
2081
+            is->audio_clock = av_q2d(is->audio_st->time_base)*pkt->pts;
2082
+        }
2083
+    }
2084
+}
2085
+
2086
+/* prepare a new audio buffer */
2087
+static void sdl_audio_callback(void *opaque, Uint8 *stream, int len)
2088
+{
2089
+    VideoState *is = opaque;
2090
+    int audio_size, len1;
2091
+    double pts;
2092
+
2093
+    audio_callback_time = av_gettime();
2094
+
2095
+    while (len > 0) {
2096
+        if (is->audio_buf_index >= is->audio_buf_size) {
2097
+           audio_size = audio_decode_frame(is, &pts);
2098
+           if (audio_size < 0) {
2099
+                /* if error, just output silence */
2100
+               is->audio_buf = is->audio_buf1;
2101
+               is->audio_buf_size = 1024;
2102
+               memset(is->audio_buf, 0, is->audio_buf_size);
2103
+           } else {
2104
+               if (is->show_audio)
2105
+                   update_sample_display(is, (int16_t *)is->audio_buf, audio_size);
2106
+               audio_size = synchronize_audio(is, (int16_t *)is->audio_buf, audio_size,
2107
+                                              pts);
2108
+               is->audio_buf_size = audio_size;
2109
+           }
2110
+           is->audio_buf_index = 0;
2111
+        }
2112
+        len1 = is->audio_buf_size - is->audio_buf_index;
2113
+        if (len1 > len)
2114
+            len1 = len;
2115
+        memcpy(stream, (uint8_t *)is->audio_buf + is->audio_buf_index, len1);
2116
+        len -= len1;
2117
+        stream += len1;
2118
+        is->audio_buf_index += len1;
2119
+    }
2120
+}
2121
+
2122
+/* open a given stream. Return 0 if OK */
2123
+static int stream_component_open(VideoState *is, int stream_index)
2124
+{
2125
+    AVFormatContext *ic = is->ic;
2126
+    AVCodecContext *avctx;
2127
+    AVCodec *codec;
2128
+    SDL_AudioSpec wanted_spec, spec;
2129
+    AVDictionary *opts;
2130
+    AVDictionaryEntry *t = NULL;
2131
+
2132
+    if (stream_index < 0 || stream_index >= ic->nb_streams)
2133
+        return -1;
2134
+    avctx = ic->streams[stream_index]->codec;
2135
+
2136
+    opts = filter_codec_opts(codec_opts, avctx->codec_id, 0);
2137
+
2138
+    /* prepare audio output */
2139
+    if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
2140
+        if (avctx->channels > 0) {
2141
+            avctx->request_channels = FFMIN(2, avctx->channels);
2142
+        } else {
2143
+            avctx->request_channels = 2;
2144
+        }
2145
+    }
2146
+
2147
+    codec = avcodec_find_decoder(avctx->codec_id);
2148
+    avctx->debug_mv = debug_mv;
2149
+    avctx->debug = debug;
2150
+    avctx->workaround_bugs = workaround_bugs;
2151
+    avctx->lowres = lowres;
2152
+    if(lowres) avctx->flags |= CODEC_FLAG_EMU_EDGE;
2153
+    avctx->idct_algo= idct;
2154
+    if(fast) avctx->flags2 |= CODEC_FLAG2_FAST;
2155
+    avctx->skip_frame= skip_frame;
2156
+    avctx->skip_idct= skip_idct;
2157
+    avctx->skip_loop_filter= skip_loop_filter;
2158
+    avctx->error_recognition= error_recognition;
2159
+    avctx->error_concealment= error_concealment;
2160
+    avctx->thread_count= thread_count;
2161
+
2162
+    if (!codec ||
2163
+        avcodec_open2(avctx, codec, &opts) < 0)
2164
+        return -1;
2165
+    if ((t = av_dict_get(opts, "", NULL, AV_DICT_IGNORE_SUFFIX))) {
2166
+        av_log(NULL, AV_LOG_ERROR, "Option %s not found.\n", t->key);
2167
+        return AVERROR_OPTION_NOT_FOUND;
2168
+    }
2169
+
2170
+    /* prepare audio output */
2171
+    if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
2172
+        wanted_spec.freq = avctx->sample_rate;
2173
+        wanted_spec.format = AUDIO_S16SYS;
2174
+        wanted_spec.channels = avctx->channels;
2175
+        wanted_spec.silence = 0;
2176
+        wanted_spec.samples = SDL_AUDIO_BUFFER_SIZE;
2177
+        wanted_spec.callback = sdl_audio_callback;
2178
+        wanted_spec.userdata = is;
2179
+        if (SDL_OpenAudio(&wanted_spec, &spec) < 0) {
2180
+            fprintf(stderr, "SDL_OpenAudio: %s\n", SDL_GetError());
2181
+            return -1;
2182
+        }
2183
+        is->audio_hw_buf_size = spec.size;
2184
+        is->audio_src_fmt= AV_SAMPLE_FMT_S16;
2185
+    }
2186
+
2187
+    ic->streams[stream_index]->discard = AVDISCARD_DEFAULT;
2188
+    switch(avctx->codec_type) {
2189
+    case AVMEDIA_TYPE_AUDIO:
2190
+        is->audio_stream = stream_index;
2191
+        is->audio_st = ic->streams[stream_index];
2192
+        is->audio_buf_size = 0;
2193
+        is->audio_buf_index = 0;
2194
+
2195
+        /* init averaging filter */
2196
+        is->audio_diff_avg_coef = exp(log(0.01) / AUDIO_DIFF_AVG_NB);
2197
+        is->audio_diff_avg_count = 0;
2198
+        /* since we do not have a precise anough audio fifo fullness,
2199
+           we correct audio sync only if larger than this threshold */
2200
+        is->audio_diff_threshold = 2.0 * SDL_AUDIO_BUFFER_SIZE / avctx->sample_rate;
2201
+
2202
+        memset(&is->audio_pkt, 0, sizeof(is->audio_pkt));
2203
+        packet_queue_init(&is->audioq);
2204
+        SDL_PauseAudio(0);
2205
+        break;
2206
+    case AVMEDIA_TYPE_VIDEO:
2207
+        is->video_stream = stream_index;
2208
+        is->video_st = ic->streams[stream_index];
2209
+
2210
+        packet_queue_init(&is->videoq);
2211
+        is->video_tid = SDL_CreateThread(video_thread, is);
2212
+        break;
2213
+    case AVMEDIA_TYPE_SUBTITLE:
2214
+        is->subtitle_stream = stream_index;
2215
+        is->subtitle_st = ic->streams[stream_index];
2216
+        packet_queue_init(&is->subtitleq);
2217
+
2218
+        is->subtitle_tid = SDL_CreateThread(subtitle_thread, is);
2219
+        break;
2220
+    default:
2221
+        break;
2222
+    }
2223
+    return 0;
2224
+}
2225
+
2226
+static void stream_component_close(VideoState *is, int stream_index)
2227
+{
2228
+    AVFormatContext *ic = is->ic;
2229
+    AVCodecContext *avctx;
2230
+
2231
+    if (stream_index < 0 || stream_index >= ic->nb_streams)
2232
+        return;
2233
+    avctx = ic->streams[stream_index]->codec;
2234
+
2235
+    switch(avctx->codec_type) {
2236
+    case AVMEDIA_TYPE_AUDIO:
2237
+        packet_queue_abort(&is->audioq);
2238
+
2239
+        SDL_CloseAudio();
2240
+
2241
+        packet_queue_end(&is->audioq);
2242
+        if (is->reformat_ctx)
2243
+            av_audio_convert_free(is->reformat_ctx);
2244
+        is->reformat_ctx = NULL;
2245
+        break;
2246
+    case AVMEDIA_TYPE_VIDEO:
2247
+        packet_queue_abort(&is->videoq);
2248
+
2249
+        /* note: we also signal this mutex to make sure we deblock the
2250
+           video thread in all cases */
2251
+        SDL_LockMutex(is->pictq_mutex);
2252
+        SDL_CondSignal(is->pictq_cond);
2253
+        SDL_UnlockMutex(is->pictq_mutex);
2254
+
2255
+        SDL_WaitThread(is->video_tid, NULL);
2256
+
2257
+        packet_queue_end(&is->videoq);
2258
+        break;
2259
+    case AVMEDIA_TYPE_SUBTITLE:
2260
+        packet_queue_abort(&is->subtitleq);
2261
+
2262
+        /* note: we also signal this mutex to make sure we deblock the
2263
+           video thread in all cases */
2264
+        SDL_LockMutex(is->subpq_mutex);
2265
+        is->subtitle_stream_changed = 1;
2266
+
2267
+        SDL_CondSignal(is->subpq_cond);
2268
+        SDL_UnlockMutex(is->subpq_mutex);
2269
+
2270
+        SDL_WaitThread(is->subtitle_tid, NULL);
2271
+
2272
+        packet_queue_end(&is->subtitleq);
2273
+        break;
2274
+    default:
2275
+        break;
2276
+    }
2277
+
2278
+    ic->streams[stream_index]->discard = AVDISCARD_ALL;
2279
+    avcodec_close(avctx);
2280
+    switch(avctx->codec_type) {
2281
+    case AVMEDIA_TYPE_AUDIO:
2282
+        is->audio_st = NULL;
2283
+        is->audio_stream = -1;
2284
+        break;
2285
+    case AVMEDIA_TYPE_VIDEO:
2286
+        is->video_st = NULL;
2287
+        is->video_stream = -1;
2288
+        break;
2289
+    case AVMEDIA_TYPE_SUBTITLE:
2290
+        is->subtitle_st = NULL;
2291
+        is->subtitle_stream = -1;
2292
+        break;
2293
+    default:
2294
+        break;
2295
+    }
2296
+}
2297
+
2298
+/* since we have only one decoding thread, we can use a global
2299
+   variable instead of a thread local variable */
2300
+static VideoState *global_video_state;
2301
+
2302
+static int decode_interrupt_cb(void)
2303
+{
2304
+    return (global_video_state && global_video_state->abort_request);
2305
+}
2306
+
2307
+/* this thread gets the stream from the disk or the network */
2308
+static int decode_thread(void *arg)
2309
+{
2310
+    VideoState *is = arg;
2311
+    AVFormatContext *ic = NULL;
2312
+    int err, i, ret;
2313
+    int st_index[AVMEDIA_TYPE_NB];
2314
+    AVPacket pkt1, *pkt = &pkt1;
2315
+    int eof=0;
2316
+    int pkt_in_play_range = 0;
2317
+    AVDictionaryEntry *t;
2318
+    AVDictionary **opts;
2319
+    int orig_nb_streams;
2320
+
2321
+    memset(st_index, -1, sizeof(st_index));
2322
+    is->video_stream = -1;
2323
+    is->audio_stream = -1;
2324
+    is->subtitle_stream = -1;
2325
+
2326
+    global_video_state = is;
2327
+    avio_set_interrupt_cb(decode_interrupt_cb);
2328
+
2329
+    err = avformat_open_input(&ic, is->filename, is->iformat, &format_opts);
2330
+    if (err < 0) {
2331
+        print_error(is->filename, err);
2332
+        ret = -1;
2333
+        goto fail;
2334
+    }
2335
+    if ((t = av_dict_get(format_opts, "", NULL, AV_DICT_IGNORE_SUFFIX))) {
2336
+        av_log(NULL, AV_LOG_ERROR, "Option %s not found.\n", t->key);
2337
+        ret = AVERROR_OPTION_NOT_FOUND;
2338
+        goto fail;
2339
+    }
2340
+    is->ic = ic;
2341
+
2342
+    if(genpts)
2343
+        ic->flags |= AVFMT_FLAG_GENPTS;
2344
+
2345
+    opts = setup_find_stream_info_opts(ic, codec_opts);
2346
+    orig_nb_streams = ic->nb_streams;
2347
+
2348
+    err = avformat_find_stream_info(ic, opts);
2349
+    if (err < 0) {
2350
+        fprintf(stderr, "%s: could not find codec parameters\n", is->filename);
2351
+        ret = -1;
2352
+        goto fail;
2353
+    }
2354
+    for (i = 0; i < orig_nb_streams; i++)
2355
+        av_dict_free(&opts[i]);
2356
+    av_freep(&opts);
2357
+
2358
+    if(ic->pb)
2359
+        ic->pb->eof_reached= 0; //FIXME hack, avplay maybe should not use url_feof() to test for the end
2360
+
2361
+    if(seek_by_bytes<0)
2362
+        seek_by_bytes= !!(ic->iformat->flags & AVFMT_TS_DISCONT);
2363
+
2364
+    /* if seeking requested, we execute it */
2365
+    if (start_time != AV_NOPTS_VALUE) {
2366
+        int64_t timestamp;
2367
+
2368
+        timestamp = start_time;
2369
+        /* add the stream start time */
2370
+        if (ic->start_time != AV_NOPTS_VALUE)
2371
+            timestamp += ic->start_time;
2372
+        ret = avformat_seek_file(ic, -1, INT64_MIN, timestamp, INT64_MAX, 0);
2373
+        if (ret < 0) {
2374
+            fprintf(stderr, "%s: could not seek to position %0.3f\n",
2375
+                    is->filename, (double)timestamp / AV_TIME_BASE);
2376
+        }
2377
+    }
2378
+
2379
+    for (i = 0; i < ic->nb_streams; i++)
2380
+        ic->streams[i]->discard = AVDISCARD_ALL;
2381
+    if (!video_disable)
2382
+        st_index[AVMEDIA_TYPE_VIDEO] =
2383
+            av_find_best_stream(ic, AVMEDIA_TYPE_VIDEO,
2384
+                                wanted_stream[AVMEDIA_TYPE_VIDEO], -1, NULL, 0);
2385
+    if (!audio_disable)
2386
+        st_index[AVMEDIA_TYPE_AUDIO] =
2387
+            av_find_best_stream(ic, AVMEDIA_TYPE_AUDIO,
2388
+                                wanted_stream[AVMEDIA_TYPE_AUDIO],
2389
+                                st_index[AVMEDIA_TYPE_VIDEO],
2390
+                                NULL, 0);
2391
+    if (!video_disable)
2392
+        st_index[AVMEDIA_TYPE_SUBTITLE] =
2393
+            av_find_best_stream(ic, AVMEDIA_TYPE_SUBTITLE,
2394
+                                wanted_stream[AVMEDIA_TYPE_SUBTITLE],
2395
+                                (st_index[AVMEDIA_TYPE_AUDIO] >= 0 ?
2396
+                                 st_index[AVMEDIA_TYPE_AUDIO] :
2397
+                                 st_index[AVMEDIA_TYPE_VIDEO]),
2398
+                                NULL, 0);
2399
+    if (show_status) {
2400
+        av_dump_format(ic, 0, is->filename, 0);
2401
+    }
2402
+
2403
+    /* open the streams */
2404
+    if (st_index[AVMEDIA_TYPE_AUDIO] >= 0) {
2405
+        stream_component_open(is, st_index[AVMEDIA_TYPE_AUDIO]);
2406
+    }
2407
+
2408
+    ret=-1;
2409
+    if (st_index[AVMEDIA_TYPE_VIDEO] >= 0) {
2410
+        ret= stream_component_open(is, st_index[AVMEDIA_TYPE_VIDEO]);
2411
+    }
2412
+    is->refresh_tid = SDL_CreateThread(refresh_thread, is);
2413
+    if(ret<0) {
2414
+        if (!display_disable)
2415
+            is->show_audio = 2;
2416
+    }
2417
+
2418
+    if (st_index[AVMEDIA_TYPE_SUBTITLE] >= 0) {
2419
+        stream_component_open(is, st_index[AVMEDIA_TYPE_SUBTITLE]);
2420
+    }
2421
+
2422
+    if (is->video_stream < 0 && is->audio_stream < 0) {
2423
+        fprintf(stderr, "%s: could not open codecs\n", is->filename);
2424
+        ret = -1;
2425
+        goto fail;
2426
+    }
2427
+
2428
+    for(;;) {
2429
+        if (is->abort_request)
2430
+            break;
2431
+        if (is->paused != is->last_paused) {
2432
+            is->last_paused = is->paused;
2433
+            if (is->paused)
2434
+                is->read_pause_return= av_read_pause(ic);
2435
+            else
2436
+                av_read_play(ic);
2437
+        }
2438
+#if CONFIG_RTSP_DEMUXER
2439
+        if (is->paused && !strcmp(ic->iformat->name, "rtsp")) {
2440
+            /* wait 10 ms to avoid trying to get another packet */
2441
+            /* XXX: horrible */
2442
+            SDL_Delay(10);
2443
+            continue;
2444
+        }
2445
+#endif
2446
+        if (is->seek_req) {
2447
+            int64_t seek_target= is->seek_pos;
2448
+            int64_t seek_min= is->seek_rel > 0 ? seek_target - is->seek_rel + 2: INT64_MIN;
2449
+            int64_t seek_max= is->seek_rel < 0 ? seek_target - is->seek_rel - 2: INT64_MAX;
2450
+//FIXME the +-2 is due to rounding being not done in the correct direction in generation
2451
+//      of the seek_pos/seek_rel variables
2452
+
2453
+            ret = avformat_seek_file(is->ic, -1, seek_min, seek_target, seek_max, is->seek_flags);
2454
+            if (ret < 0) {
2455
+                fprintf(stderr, "%s: error while seeking\n", is->ic->filename);
2456
+            }else{
2457
+                if (is->audio_stream >= 0) {
2458
+                    packet_queue_flush(&is->audioq);
2459
+                    packet_queue_put(&is->audioq, &flush_pkt);
2460
+                }
2461
+                if (is->subtitle_stream >= 0) {
2462
+                    packet_queue_flush(&is->subtitleq);
2463
+                    packet_queue_put(&is->subtitleq, &flush_pkt);
2464
+                }
2465
+                if (is->video_stream >= 0) {
2466
+                    packet_queue_flush(&is->videoq);
2467
+                    packet_queue_put(&is->videoq, &flush_pkt);
2468
+                }
2469
+            }
2470
+            is->seek_req = 0;
2471
+            eof= 0;
2472
+        }
2473
+
2474
+        /* if the queue are full, no need to read more */
2475
+        if (   is->audioq.size + is->videoq.size + is->subtitleq.size > MAX_QUEUE_SIZE
2476
+            || (   (is->audioq   .size  > MIN_AUDIOQ_SIZE || is->audio_stream<0)
2477
+                && (is->videoq   .nb_packets > MIN_FRAMES || is->video_stream<0)
2478
+                && (is->subtitleq.nb_packets > MIN_FRAMES || is->subtitle_stream<0))) {
2479
+            /* wait 10 ms */
2480
+            SDL_Delay(10);
2481
+            continue;
2482
+        }
2483
+        if(eof) {
2484
+            if(is->video_stream >= 0){
2485
+                av_init_packet(pkt);
2486
+                pkt->data=NULL;
2487
+                pkt->size=0;
2488
+                pkt->stream_index= is->video_stream;
2489
+                packet_queue_put(&is->videoq, pkt);
2490
+            }
2491
+            SDL_Delay(10);
2492
+            if(is->audioq.size + is->videoq.size + is->subtitleq.size ==0){
2493
+                if(loop!=1 && (!loop || --loop)){
2494
+                    stream_seek(cur_stream, start_time != AV_NOPTS_VALUE ? start_time : 0, 0, 0);
2495
+                }else if(autoexit){
2496
+                    ret=AVERROR_EOF;
2497
+                    goto fail;
2498
+                }
2499
+            }
2500
+            continue;
2501
+        }
2502
+        ret = av_read_frame(ic, pkt);
2503
+        if (ret < 0) {
2504
+            if (ret == AVERROR_EOF || (ic->pb && ic->pb->eof_reached))
2505
+                eof=1;
2506
+            if (ic->pb && ic->pb->error)
2507
+                break;
2508
+            SDL_Delay(100); /* wait for user event */
2509
+            continue;
2510
+        }
2511
+        /* check if packet is in play range specified by user, then queue, otherwise discard */
2512
+        pkt_in_play_range = duration == AV_NOPTS_VALUE ||
2513
+                (pkt->pts - ic->streams[pkt->stream_index]->start_time) *
2514
+                av_q2d(ic->streams[pkt->stream_index]->time_base) -
2515
+                (double)(start_time != AV_NOPTS_VALUE ? start_time : 0)/1000000
2516
+                <= ((double)duration/1000000);
2517
+        if (pkt->stream_index == is->audio_stream && pkt_in_play_range) {
2518
+            packet_queue_put(&is->audioq, pkt);
2519
+        } else if (pkt->stream_index == is->video_stream && pkt_in_play_range) {
2520
+            packet_queue_put(&is->videoq, pkt);
2521
+        } else if (pkt->stream_index == is->subtitle_stream && pkt_in_play_range) {
2522
+            packet_queue_put(&is->subtitleq, pkt);
2523
+        } else {
2524
+            av_free_packet(pkt);
2525
+        }
2526
+    }
2527
+    /* wait until the end */
2528
+    while (!is->abort_request) {
2529
+        SDL_Delay(100);
2530
+    }
2531
+
2532
+    ret = 0;
2533
+ fail:
2534
+    /* disable interrupting */
2535
+    global_video_state = NULL;
2536
+
2537
+    /* close each stream */
2538
+    if (is->audio_stream >= 0)
2539
+        stream_component_close(is, is->audio_stream);
2540
+    if (is->video_stream >= 0)
2541
+        stream_component_close(is, is->video_stream);
2542
+    if (is->subtitle_stream >= 0)
2543
+        stream_component_close(is, is->subtitle_stream);
2544
+    if (is->ic) {
2545
+        av_close_input_file(is->ic);
2546
+        is->ic = NULL; /* safety */
2547
+    }
2548
+    avio_set_interrupt_cb(NULL);
2549
+
2550
+    if (ret != 0) {
2551
+        SDL_Event event;
2552
+
2553
+        event.type = FF_QUIT_EVENT;
2554
+        event.user.data1 = is;
2555
+        SDL_PushEvent(&event);
2556
+    }
2557
+    return 0;
2558
+}
2559
+
2560
+static VideoState *stream_open(const char *filename, AVInputFormat *iformat)
2561
+{
2562
+    VideoState *is;
2563
+
2564
+    is = av_mallocz(sizeof(VideoState));
2565
+    if (!is)
2566
+        return NULL;
2567
+    av_strlcpy(is->filename, filename, sizeof(is->filename));
2568
+    is->iformat = iformat;
2569
+    is->ytop = 0;
2570
+    is->xleft = 0;
2571
+
2572
+    /* start video display */
2573
+    is->pictq_mutex = SDL_CreateMutex();
2574
+    is->pictq_cond = SDL_CreateCond();
2575
+
2576
+    is->subpq_mutex = SDL_CreateMutex();
2577
+    is->subpq_cond = SDL_CreateCond();
2578
+
2579
+    is->av_sync_type = av_sync_type;
2580
+    is->parse_tid = SDL_CreateThread(decode_thread, is);
2581
+    if (!is->parse_tid) {
2582
+        av_free(is);
2583
+        return NULL;
2584
+    }
2585
+    return is;
2586
+}
2587
+
2588
+static void stream_cycle_channel(VideoState *is, int codec_type)
2589
+{
2590
+    AVFormatContext *ic = is->ic;
2591
+    int start_index, stream_index;
2592
+    AVStream *st;
2593
+
2594
+    if (codec_type == AVMEDIA_TYPE_VIDEO)
2595
+        start_index = is->video_stream;
2596
+    else if (codec_type == AVMEDIA_TYPE_AUDIO)
2597
+        start_index = is->audio_stream;
2598
+    else
2599
+        start_index = is->subtitle_stream;
2600
+    if (start_index < (codec_type == AVMEDIA_TYPE_SUBTITLE ? -1 : 0))
2601
+        return;
2602
+    stream_index = start_index;
2603
+    for(;;) {
2604
+        if (++stream_index >= is->ic->nb_streams)
2605
+        {
2606
+            if (codec_type == AVMEDIA_TYPE_SUBTITLE)
2607
+            {
2608
+                stream_index = -1;
2609
+                goto the_end;
2610
+            } else
2611
+                stream_index = 0;
2612
+        }
2613
+        if (stream_index == start_index)
2614
+            return;
2615
+        st = ic->streams[stream_index];
2616
+        if (st->codec->codec_type == codec_type) {
2617
+            /* check that parameters are OK */
2618
+            switch(codec_type) {
2619
+            case AVMEDIA_TYPE_AUDIO:
2620
+                if (st->codec->sample_rate != 0 &&
2621
+                    st->codec->channels != 0)
2622
+                    goto the_end;
2623
+                break;
2624
+            case AVMEDIA_TYPE_VIDEO:
2625
+            case AVMEDIA_TYPE_SUBTITLE:
2626
+                goto the_end;
2627
+            default:
2628
+                break;
2629
+            }
2630
+        }
2631
+    }
2632
+ the_end:
2633
+    stream_component_close(is, start_index);
2634
+    stream_component_open(is, stream_index);
2635
+}
2636
+
2637
+
2638
+static void toggle_full_screen(void)
2639
+{
2640
+    is_full_screen = !is_full_screen;
2641
+    video_open(cur_stream);
2642
+}
2643
+
2644
+static void toggle_pause(void)
2645
+{
2646
+    if (cur_stream)
2647
+        stream_pause(cur_stream);
2648
+    step = 0;
2649
+}
2650
+
2651
+static void step_to_next_frame(void)
2652
+{
2653
+    if (cur_stream) {
2654
+        /* if the stream is paused unpause it, then step */
2655
+        if (cur_stream->paused)
2656
+            stream_pause(cur_stream);
2657
+    }
2658
+    step = 1;
2659
+}
2660
+
2661
+static void toggle_audio_display(void)
2662
+{
2663
+    if (cur_stream) {
2664
+        int bgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0x00);
2665
+        cur_stream->show_audio = (cur_stream->show_audio + 1) % 3;
2666
+        fill_rectangle(screen,
2667
+                    cur_stream->xleft, cur_stream->ytop, cur_stream->width, cur_stream->height,
2668
+                    bgcolor);
2669
+        SDL_UpdateRect(screen, cur_stream->xleft, cur_stream->ytop, cur_stream->width, cur_stream->height);
2670
+    }
2671
+}
2672
+
2673
+/* handle an event sent by the GUI */
2674
+static void event_loop(void)
2675
+{
2676
+    SDL_Event event;
2677
+    double incr, pos, frac;
2678
+
2679
+    for(;;) {
2680
+        double x;
2681
+        SDL_WaitEvent(&event);
2682
+        switch(event.type) {
2683
+        case SDL_KEYDOWN:
2684
+            if (exit_on_keydown) {
2685
+                do_exit();
2686
+                break;
2687
+            }
2688
+            switch(event.key.keysym.sym) {
2689
+            case SDLK_ESCAPE:
2690
+            case SDLK_q:
2691
+                do_exit();
2692
+                break;
2693
+            case SDLK_f:
2694
+                toggle_full_screen();
2695
+                break;
2696
+            case SDLK_p:
2697
+            case SDLK_SPACE:
2698
+                toggle_pause();
2699
+                break;
2700
+            case SDLK_s: //S: Step to next frame
2701
+                step_to_next_frame();
2702
+                break;
2703
+            case SDLK_a:
2704
+                if (cur_stream)
2705
+                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_AUDIO);
2706
+                break;
2707
+            case SDLK_v:
2708
+                if (cur_stream)
2709
+                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_VIDEO);
2710
+                break;
2711
+            case SDLK_t:
2712
+                if (cur_stream)
2713
+                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_SUBTITLE);
2714
+                break;
2715
+            case SDLK_w:
2716
+                toggle_audio_display();
2717
+                break;
2718
+            case SDLK_LEFT:
2719
+                incr = -10.0;
2720
+                goto do_seek;
2721
+            case SDLK_RIGHT:
2722
+                incr = 10.0;
2723
+                goto do_seek;
2724
+            case SDLK_UP:
2725
+                incr = 60.0;
2726
+                goto do_seek;
2727
+            case SDLK_DOWN:
2728
+                incr = -60.0;
2729
+            do_seek:
2730
+                if (cur_stream) {
2731
+                    if (seek_by_bytes) {
2732
+                        if (cur_stream->video_stream >= 0 && cur_stream->video_current_pos>=0){
2733
+                            pos= cur_stream->video_current_pos;
2734
+                        }else if(cur_stream->audio_stream >= 0 && cur_stream->audio_pkt.pos>=0){
2735
+                            pos= cur_stream->audio_pkt.pos;
2736
+                        }else
2737
+                            pos = avio_tell(cur_stream->ic->pb);
2738
+                        if (cur_stream->ic->bit_rate)
2739
+                            incr *= cur_stream->ic->bit_rate / 8.0;
2740
+                        else
2741
+                            incr *= 180000.0;
2742
+                        pos += incr;
2743
+                        stream_seek(cur_stream, pos, incr, 1);
2744
+                    } else {
2745
+                        pos = get_master_clock(cur_stream);
2746
+                        pos += incr;
2747
+                        stream_seek(cur_stream, (int64_t)(pos * AV_TIME_BASE), (int64_t)(incr * AV_TIME_BASE), 0);
2748
+                    }
2749
+                }
2750
+                break;
2751
+            default:
2752
+                break;
2753
+            }
2754
+            break;
2755
+        case SDL_MOUSEBUTTONDOWN:
2756
+            if (exit_on_mousedown) {
2757
+                do_exit();
2758
+                break;
2759
+            }
2760
+        case SDL_MOUSEMOTION:
2761
+            if(event.type ==SDL_MOUSEBUTTONDOWN){
2762
+                x= event.button.x;
2763
+            }else{
2764
+                if(event.motion.state != SDL_PRESSED)
2765
+                    break;
2766
+                x= event.motion.x;
2767
+            }
2768
+            if (cur_stream) {
2769
+                if(seek_by_bytes || cur_stream->ic->duration<=0){
2770
+                    uint64_t size=  avio_size(cur_stream->ic->pb);
2771
+                    stream_seek(cur_stream, size*x/cur_stream->width, 0, 1);
2772
+                }else{
2773
+                    int64_t ts;
2774
+                    int ns, hh, mm, ss;
2775
+                    int tns, thh, tmm, tss;
2776
+                    tns = cur_stream->ic->duration/1000000LL;
2777
+                    thh = tns/3600;
2778
+                    tmm = (tns%3600)/60;
2779
+                    tss = (tns%60);
2780
+                    frac = x/cur_stream->width;
2781
+                    ns = frac*tns;
2782
+                    hh = ns/3600;
2783
+                    mm = (ns%3600)/60;
2784
+                    ss = (ns%60);
2785
+                    fprintf(stderr, "Seek to %2.0f%% (%2d:%02d:%02d) of total duration (%2d:%02d:%02d)       \n", frac*100,
2786
+                            hh, mm, ss, thh, tmm, tss);
2787
+                    ts = frac*cur_stream->ic->duration;
2788
+                    if (cur_stream->ic->start_time != AV_NOPTS_VALUE)
2789
+                        ts += cur_stream->ic->start_time;
2790
+                    stream_seek(cur_stream, ts, 0, 0);
2791
+                }
2792
+            }
2793
+            break;
2794
+        case SDL_VIDEORESIZE:
2795
+            if (cur_stream) {
2796
+                screen = SDL_SetVideoMode(event.resize.w, event.resize.h, 0,
2797
+                                          SDL_HWSURFACE|SDL_RESIZABLE|SDL_ASYNCBLIT|SDL_HWACCEL);
2798
+                screen_width = cur_stream->width = event.resize.w;
2799
+                screen_height= cur_stream->height= event.resize.h;
2800
+            }
2801
+            break;
2802
+        case SDL_QUIT:
2803
+        case FF_QUIT_EVENT:
2804
+            do_exit();
2805
+            break;
2806
+        case FF_ALLOC_EVENT:
2807
+            video_open(event.user.data1);
2808
+            alloc_picture(event.user.data1);
2809
+            break;
2810
+        case FF_REFRESH_EVENT:
2811
+            video_refresh_timer(event.user.data1);
2812
+            cur_stream->refresh=0;
2813
+            break;
2814
+        default:
2815
+            break;
2816
+        }
2817
+    }
2818
+}
2819
+
2820
+static int opt_frame_size(const char *opt, const char *arg)
2821
+{
2822
+    av_log(NULL, AV_LOG_ERROR,
2823
+           "Option '%s' has been removed, use private format options instead\n", opt);
2824
+    return AVERROR(EINVAL);
2825
+}
2826
+
2827
+static int opt_width(const char *opt, const char *arg)
2828
+{
2829
+    screen_width = parse_number_or_die(opt, arg, OPT_INT64, 1, INT_MAX);
2830
+    return 0;
2831
+}
2832
+
2833
+static int opt_height(const char *opt, const char *arg)
2834
+{
2835
+    screen_height = parse_number_or_die(opt, arg, OPT_INT64, 1, INT_MAX);
2836
+    return 0;
2837
+}
2838
+
2839
+static int opt_format(const char *opt, const char *arg)
2840
+{
2841
+    file_iformat = av_find_input_format(arg);
2842
+    if (!file_iformat) {
2843
+        fprintf(stderr, "Unknown input format: %s\n", arg);
2844
+        return AVERROR(EINVAL);
2845
+    }
2846
+    return 0;
2847
+}
2848
+
2849
+static int opt_frame_pix_fmt(const char *opt, const char *arg)
2850
+{
2851
+    av_log(NULL, AV_LOG_ERROR,
2852
+           "Option '%s' has been removed, use private format options instead\n", opt);
2853
+    return AVERROR(EINVAL);
2854
+}
2855
+
2856
+static int opt_sync(const char *opt, const char *arg)
2857
+{
2858
+    if (!strcmp(arg, "audio"))
2859
+        av_sync_type = AV_SYNC_AUDIO_MASTER;
2860
+    else if (!strcmp(arg, "video"))
2861
+        av_sync_type = AV_SYNC_VIDEO_MASTER;
2862
+    else if (!strcmp(arg, "ext"))
2863
+        av_sync_type = AV_SYNC_EXTERNAL_CLOCK;
2864
+    else {
2865
+        fprintf(stderr, "Unknown value for %s: %s\n", opt, arg);
2866
+        exit(1);
2867
+    }
2868
+    return 0;
2869
+}
2870
+
2871
+static int opt_seek(const char *opt, const char *arg)
2872
+{
2873
+    start_time = parse_time_or_die(opt, arg, 1);
2874
+    return 0;
2875
+}
2876
+
2877
+static int opt_duration(const char *opt, const char *arg)
2878
+{
2879
+    duration = parse_time_or_die(opt, arg, 1);
2880
+    return 0;
2881
+}
2882
+
2883
+static int opt_debug(const char *opt, const char *arg)
2884
+{
2885
+    av_log_set_level(99);
2886
+    debug = parse_number_or_die(opt, arg, OPT_INT64, 0, INT_MAX);
2887
+    return 0;
2888
+}
2889
+
2890
+static int opt_vismv(const char *opt, const char *arg)
2891
+{
2892
+    debug_mv = parse_number_or_die(opt, arg, OPT_INT64, INT_MIN, INT_MAX);
2893
+    return 0;
2894
+}
2895
+
2896
+static int opt_thread_count(const char *opt, const char *arg)
2897
+{
2898
+    thread_count= parse_number_or_die(opt, arg, OPT_INT64, 0, INT_MAX);
2899
+#if !HAVE_THREADS
2900
+    fprintf(stderr, "Warning: not compiled with thread support, using thread emulation\n");
2901
+#endif
2902
+    return 0;
2903
+}
2904
+
2905
+static const OptionDef options[] = {
2906
+#include "cmdutils_common_opts.h"
2907
+    { "x", HAS_ARG, {(void*)opt_width}, "force displayed width", "width" },
2908
+    { "y", HAS_ARG, {(void*)opt_height}, "force displayed height", "height" },
2909
+    { "s", HAS_ARG | OPT_VIDEO, {(void*)opt_frame_size}, "set frame size (WxH or abbreviation)", "size" },
2910
+    { "fs", OPT_BOOL, {(void*)&is_full_screen}, "force full screen" },
2911
+    { "an", OPT_BOOL, {(void*)&audio_disable}, "disable audio" },
2912
+    { "vn", OPT_BOOL, {(void*)&video_disable}, "disable video" },
2913
+    { "ast", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_AUDIO]}, "select desired audio stream", "stream_number" },
2914
+    { "vst", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_VIDEO]}, "select desired video stream", "stream_number" },
2915
+    { "sst", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_SUBTITLE]}, "select desired subtitle stream", "stream_number" },
2916
+    { "ss", HAS_ARG, {(void*)&opt_seek}, "seek to a given position in seconds", "pos" },
2917
+    { "t", HAS_ARG, {(void*)&opt_duration}, "play  \"duration\" seconds of audio/video", "duration" },
2918
+    { "bytes", OPT_INT | HAS_ARG, {(void*)&seek_by_bytes}, "seek by bytes 0=off 1=on -1=auto", "val" },
2919
+    { "nodisp", OPT_BOOL, {(void*)&display_disable}, "disable graphical display" },
2920
+    { "f", HAS_ARG, {(void*)opt_format}, "force format", "fmt" },
2921
+    { "pix_fmt", HAS_ARG | OPT_EXPERT | OPT_VIDEO, {(void*)opt_frame_pix_fmt}, "set pixel format", "format" },
2922
+    { "stats", OPT_BOOL | OPT_EXPERT, {(void*)&show_status}, "show status", "" },
2923
+    { "debug", HAS_ARG | OPT_EXPERT, {(void*)opt_debug}, "print specific debug info", "" },
2924
+    { "bug", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&workaround_bugs}, "workaround bugs", "" },
2925
+    { "vismv", HAS_ARG | OPT_EXPERT, {(void*)opt_vismv}, "visualize motion vectors", "" },
2926
+    { "fast", OPT_BOOL | OPT_EXPERT, {(void*)&fast}, "non spec compliant optimizations", "" },
2927
+    { "genpts", OPT_BOOL | OPT_EXPERT, {(void*)&genpts}, "generate pts", "" },
2928
+    { "drp", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&decoder_reorder_pts}, "let decoder reorder pts 0=off 1=on -1=auto", ""},
2929
+    { "lowres", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&lowres}, "", "" },
2930
+    { "skiploop", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_loop_filter}, "", "" },
2931
+    { "skipframe", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_frame}, "", "" },
2932
+    { "skipidct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_idct}, "", "" },
2933
+    { "idct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&idct}, "set idct algo",  "algo" },
2934
+    { "er", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_recognition}, "set error detection threshold (0-4)",  "threshold" },
2935
+    { "ec", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_concealment}, "set error concealment options",  "bit_mask" },
2936
+    { "sync", HAS_ARG | OPT_EXPERT, {(void*)opt_sync}, "set audio-video sync. type (type=audio/video/ext)", "type" },
2937
+    { "threads", HAS_ARG | OPT_EXPERT, {(void*)opt_thread_count}, "thread count", "count" },
2938
+    { "autoexit", OPT_BOOL | OPT_EXPERT, {(void*)&autoexit}, "exit at the end", "" },
2939
+    { "exitonkeydown", OPT_BOOL | OPT_EXPERT, {(void*)&exit_on_keydown}, "exit on key down", "" },
2940
+    { "exitonmousedown", OPT_BOOL | OPT_EXPERT, {(void*)&exit_on_mousedown}, "exit on mouse down", "" },
2941
+    { "loop", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&loop}, "set number of times the playback shall be looped", "loop count" },
2942
+    { "framedrop", OPT_BOOL | OPT_EXPERT, {(void*)&framedrop}, "drop frames when cpu is too slow", "" },
2943
+    { "window_title", OPT_STRING | HAS_ARG, {(void*)&window_title}, "set window title", "window title" },
2944
+#if CONFIG_AVFILTER
2945
+    { "vf", OPT_STRING | HAS_ARG, {(void*)&vfilters}, "video filters", "filter list" },
2946
+#endif
2947
+    { "rdftspeed", OPT_INT | HAS_ARG| OPT_AUDIO | OPT_EXPERT, {(void*)&rdftspeed}, "rdft speed", "msecs" },
2948
+    { "default", HAS_ARG | OPT_AUDIO | OPT_VIDEO | OPT_EXPERT, {(void*)opt_default}, "generic catch all option", "" },
2949
+    { "i", 0, {NULL}, "ffmpeg compatibility dummy option", ""},
2950
+    { NULL, },
2951
+};
2952
+
2953
+static void show_usage(void)
2954
+{
2955
+    printf("Simple media player\n");
2956
+    printf("usage: %s [options] input_file\n", program_name);
2957
+    printf("\n");
2958
+}
2959
+
2960
+static void show_help(void)
2961
+{
2962
+    av_log_set_callback(log_callback_help);
2963
+    show_usage();
2964
+    show_help_options(options, "Main options:\n",
2965
+                      OPT_EXPERT, 0);
2966
+    show_help_options(options, "\nAdvanced options:\n",
2967
+                      OPT_EXPERT, OPT_EXPERT);
2968
+    printf("\n");
2969
+    av_opt_show2(avcodec_opts[0], NULL,
2970
+                 AV_OPT_FLAG_DECODING_PARAM, 0);
2971
+    printf("\n");
2972
+    av_opt_show2(avformat_opts, NULL,
2973
+                 AV_OPT_FLAG_DECODING_PARAM, 0);
2974
+#if !CONFIG_AVFILTER
2975
+    printf("\n");
2976
+    av_opt_show2(sws_opts, NULL,
2977
+                 AV_OPT_FLAG_ENCODING_PARAM, 0);
2978
+#endif
2979
+    printf("\nWhile playing:\n"
2980
+           "q, ESC              quit\n"
2981
+           "f                   toggle full screen\n"
2982
+           "p, SPC              pause\n"
2983
+           "a                   cycle audio channel\n"
2984
+           "v                   cycle video channel\n"
2985
+           "t                   cycle subtitle channel\n"
2986
+           "w                   show audio waves\n"
2987
+           "s                   activate frame-step mode\n"
2988
+           "left/right          seek backward/forward 10 seconds\n"
2989
+           "down/up             seek backward/forward 1 minute\n"
2990
+           "mouse click         seek to percentage in file corresponding to fraction of width\n"
2991
+           );
2992
+}
2993
+
2994
+static void opt_input_file(const char *filename)
2995
+{
2996
+    if (input_filename) {
2997
+        fprintf(stderr, "Argument '%s' provided as input filename, but '%s' was already specified.\n",
2998
+                filename, input_filename);
2999
+        exit(1);
3000
+    }
3001
+    if (!strcmp(filename, "-"))
3002
+        filename = "pipe:";
3003
+    input_filename = filename;
3004
+}
3005
+
3006
+/* Called from the main */
3007
+int main(int argc, char **argv)
3008
+{
3009
+    int flags;
3010
+
3011
+    av_log_set_flags(AV_LOG_SKIP_REPEATED);
3012
+
3013
+    /* register all codecs, demux and protocols */
3014
+    avcodec_register_all();
3015
+#if CONFIG_AVDEVICE
3016
+    avdevice_register_all();
3017
+#endif
3018
+#if CONFIG_AVFILTER
3019
+    avfilter_register_all();
3020
+#endif
3021
+    av_register_all();
3022
+
3023
+    init_opts();
3024
+
3025
+    show_banner();
3026
+
3027
+    parse_options(argc, argv, options, opt_input_file);
3028
+
3029
+    if (!input_filename) {
3030
+        show_usage();
3031
+        fprintf(stderr, "An input file must be specified\n");
3032
+        fprintf(stderr, "Use -h to get full help or, even better, run 'man %s'\n", program_name);
3033
+        exit(1);
3034
+    }
3035
+
3036
+    if (display_disable) {
3037
+        video_disable = 1;
3038
+    }
3039
+    flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER;
3040
+#if !defined(__MINGW32__) && !defined(__APPLE__)
3041
+    flags |= SDL_INIT_EVENTTHREAD; /* Not supported on Windows or Mac OS X */
3042
+#endif
3043
+    if (SDL_Init (flags)) {
3044
+        fprintf(stderr, "Could not initialize SDL - %s\n", SDL_GetError());
3045
+        exit(1);
3046
+    }
3047
+
3048
+    if (!display_disable) {
3049
+#if HAVE_SDL_VIDEO_SIZE
3050
+        const SDL_VideoInfo *vi = SDL_GetVideoInfo();
3051
+        fs_screen_width = vi->current_w;
3052
+        fs_screen_height = vi->current_h;
3053
+#endif
3054
+    }
3055
+
3056
+    SDL_EventState(SDL_ACTIVEEVENT, SDL_IGNORE);
3057
+    SDL_EventState(SDL_SYSWMEVENT, SDL_IGNORE);
3058
+    SDL_EventState(SDL_USEREVENT, SDL_IGNORE);
3059
+
3060
+    av_init_packet(&flush_pkt);
3061
+    flush_pkt.data= "FLUSH";
3062
+
3063
+    cur_stream = stream_open(input_filename, file_iformat);
3064
+
3065
+    event_loop();
3066
+
3067
+    /* never returns */
3068
+
3069
+    return 0;
3070
+}
... ...
@@ -81,7 +81,7 @@ Configuration options:
81 81
                            and binaries will be unredistributable [no]
82 82
   --disable-doc            do not build documentation
83 83
   --disable-ffmpeg         disable ffmpeg build
84
-  --disable-ffplay         disable ffplay build
84
+  --disable-avplay         disable avplay build
85 85
   --disable-ffprobe        disable ffprobe build
86 86
   --disable-ffserver       disable ffserver build
87 87
   --disable-avdevice       disable libavdevice build
... ...
@@ -913,7 +913,7 @@ CONFIG_LIST="
913 913
     dxva2
914 914
     fastdiv
915 915
     ffmpeg
916
-    ffplay
916
+    avplay
917 917
     ffprobe
918 918
     ffserver
919 919
     fft
... ...
@@ -1489,8 +1489,8 @@ postproc_deps="gpl"
1489 1489
 # programs
1490 1490
 ffmpeg_deps="avcodec avformat swscale"
1491 1491
 ffmpeg_select="buffer_filter"
1492
-ffplay_deps="avcodec avformat swscale sdl"
1493
-ffplay_select="rdft"
1492
+avplay_deps="avcodec avformat swscale sdl"
1493
+avplay_select="rdft"
1494 1494
 ffprobe_deps="avcodec avformat"
1495 1495
 ffserver_deps="avformat ffm_muxer fork rtp_protocol rtsp_demuxer"
1496 1496
 ffserver_extralibs='$ldl'
... ...
@@ -1635,7 +1635,7 @@ enable debug
1635 1635
 enable doc
1636 1636
 enable fastdiv
1637 1637
 enable ffmpeg
1638
-enable ffplay
1638
+enable avplay
1639 1639
 enable ffprobe
1640 1640
 enable ffserver
1641 1641
 enable network
1642 1642
new file mode 100644
... ...
@@ -0,0 +1,182 @@
0
+\input texinfo @c -*- texinfo -*-
1
+
2
+@settitle avplay Documentation
3
+@titlepage
4
+@center @titlefont{avplay Documentation}
5
+@end titlepage
6
+
7
+@top
8
+
9
+@contents
10
+
11
+@chapter Synopsis
12
+
13
+@example
14
+@c man begin SYNOPSIS
15
+avplay [options] @file{input_file}
16
+@c man end
17
+@end example
18
+
19
+@chapter Description
20
+@c man begin DESCRIPTION
21
+
22
+AVplay is a very simple and portable media player using the Libav
23
+libraries and the SDL library. It is mostly used as a testbed for the
24
+various Libav APIs.
25
+@c man end
26
+
27
+@chapter Options
28
+@c man begin OPTIONS
29
+
30
+@include fftools-common-opts.texi
31
+
32
+@section Main options
33
+
34
+@table @option
35
+@item -x @var{width}
36
+Force displayed width.
37
+@item -y @var{height}
38
+Force displayed height.
39
+@item -s @var{size}
40
+This option has been removed. Use private format options for specifying the
41
+input video size.
42
+@item -an
43
+Disable audio.
44
+@item -vn
45
+Disable video.
46
+@item -ss @var{pos}
47
+Seek to a given position in seconds.
48
+@item -t @var{duration}
49
+play <duration> seconds of audio/video
50
+@item -bytes
51
+Seek by bytes.
52
+@item -nodisp
53
+Disable graphical display.
54
+@item -f @var{fmt}
55
+Force format.
56
+@item -window_title @var{title}
57
+Set window title (default is the input filename).
58
+@item -loop @var{number}
59
+Loops movie playback <number> times. 0 means forever.
60
+@item -vf @var{filter_graph}
61
+@var{filter_graph} is a description of the filter graph to apply to
62
+the input video.
63
+Use the option "-filters" to show all the available filters (including
64
+also sources and sinks).
65
+
66
+@end table
67
+
68
+@section Advanced options
69
+@table @option
70
+@item -pix_fmt @var{format}
71
+This option has been removed. Use private options for specifying the
72
+input pixel format.
73
+@item -stats
74
+Show the stream duration, the codec parameters, the current position in
75
+the stream and the audio/video synchronisation drift.
76
+@item -debug
77
+Print specific debug info.
78
+@item -bug
79
+Work around bugs.
80
+@item -vismv
81
+Visualize motion vectors.
82
+@item -fast
83
+Non-spec-compliant optimizations.
84
+@item -genpts
85
+Generate pts.
86
+@item -rtp_tcp
87
+Force RTP/TCP protocol usage instead of RTP/UDP. It is only meaningful
88
+if you are streaming with the RTSP protocol.
89
+@item -sync @var{type}
90
+Set the master clock to audio (@code{type=audio}), video
91
+(@code{type=video}) or external (@code{type=ext}). Default is audio. The
92
+master clock is used to control audio-video synchronization. Most media
93
+players use audio as master clock, but in some cases (streaming or high
94
+quality broadcast) it is necessary to change that. This option is mainly
95
+used for debugging purposes.
96
+@item -threads @var{count}
97
+Set the thread count.
98
+@item -ast @var{audio_stream_number}
99
+Select the desired audio stream number, counting from 0. The number
100
+refers to the list of all the input audio streams. If it is greater
101
+than the number of audio streams minus one, then the last one is
102
+selected, if it is negative the audio playback is disabled.
103
+@item -vst @var{video_stream_number}
104
+Select the desired video stream number, counting from 0. The number
105
+refers to the list of all the input video streams. If it is greater
106
+than the number of video streams minus one, then the last one is
107
+selected, if it is negative the video playback is disabled.
108
+@item -sst @var{subtitle_stream_number}
109
+Select the desired subtitle stream number, counting from 0. The number
110
+refers to the list of all the input subtitle streams. If it is greater
111
+than the number of subtitle streams minus one, then the last one is
112
+selected, if it is negative the subtitle rendering is disabled.
113
+@item -autoexit
114
+Exit when video is done playing.
115
+@item -exitonkeydown
116
+Exit if any key is pressed.
117
+@item -exitonmousedown
118
+Exit if any mouse button is pressed.
119
+@end table
120
+
121
+@section While playing
122
+
123
+@table @key
124
+@item q, ESC
125
+Quit.
126
+
127
+@item f
128
+Toggle full screen.
129
+
130
+@item p, SPC
131
+Pause.
132
+
133
+@item a
134
+Cycle audio channel.
135
+
136
+@item v
137
+Cycle video channel.
138
+
139
+@item t
140
+Cycle subtitle channel.
141
+
142
+@item w
143
+Show audio waves.
144
+
145
+@item left/right
146
+Seek backward/forward 10 seconds.
147
+
148
+@item down/up
149
+Seek backward/forward 1 minute.
150
+
151
+@item mouse click
152
+Seek to percentage in file corresponding to fraction of width.
153
+
154
+@end table
155
+
156
+@c man end
157
+
158
+@include eval.texi
159
+@include demuxers.texi
160
+@include muxers.texi
161
+@include indevs.texi
162
+@include outdevs.texi
163
+@include protocols.texi
164
+@include filters.texi
165
+
166
+@ignore
167
+
168
+@setfilename avplay
169
+@settitle AVplay media player
170
+
171
+@c man begin SEEALSO
172
+ffmpeg(1), ffprobe(1), ffserver(1) and the Libav HTML documentation
173
+@c man end
174
+
175
+@c man begin AUTHORS
176
+The Libav developers
177
+@c man end
178
+
179
+@end ignore
180
+
181
+@bye
... ...
@@ -70,7 +70,7 @@ Apple HTTP Live Streaming demuxer.
70 70
 
71 71
 This demuxer presents all AVStreams from all variant streams.
72 72
 The id field is set to the bitrate variant index number. By setting
73
-the discard flags on AVStreams (by pressing 'a' or 'v' in ffplay),
73
+the discard flags on AVStreams (by pressing 'a' or 'v' in avplay),
74 74
 the caller can decide which variant streams to actually receive.
75 75
 The total bitrate of the variant that the stream belongs to is
76 76
 available in a metadata key named "variant_bitrate".
... ...
@@ -17,7 +17,7 @@
17 17
 decoding). Look at @file{libavcodec/apiexample.c} to see how to use it.
18 18
 
19 19
 @item libavformat is the library containing the file format handling (mux and
20
-demux code for several formats). Look at @file{ffplay.c} to use it in a
20
+demux code for several formats). Look at @file{avplay.c} to use it in a
21 21
 player. See @file{libavformat/output-example.c} to use it to generate
22 22
 audio or video streams.
23 23
 
... ...
@@ -1079,7 +1079,7 @@ file to which you want to add them.
1079 1079
 @settitle ffmpeg video converter
1080 1080
 
1081 1081
 @c man begin SEEALSO
1082
-ffplay(1), ffprobe(1), ffserver(1) and the Libav HTML documentation
1082
+avplay(1), ffprobe(1), ffserver(1) and the Libav HTML documentation
1083 1083
 @c man end
1084 1084
 
1085 1085
 @c man begin AUTHORS
1086 1086
deleted file mode 100644
... ...
@@ -1,182 +0,0 @@
1
-\input texinfo @c -*- texinfo -*-
2
-
3
-@settitle ffplay Documentation
4
-@titlepage
5
-@center @titlefont{ffplay Documentation}
6
-@end titlepage
7
-
8
-@top
9
-
10
-@contents
11
-
12
-@chapter Synopsis
13
-
14
-@example
15
-@c man begin SYNOPSIS
16
-ffplay [options] @file{input_file}
17
-@c man end
18
-@end example
19
-
20
-@chapter Description
21
-@c man begin DESCRIPTION
22
-
23
-FFplay is a very simple and portable media player using the Libav
24
-libraries and the SDL library. It is mostly used as a testbed for the
25
-various Libav APIs.
26
-@c man end
27
-
28
-@chapter Options
29
-@c man begin OPTIONS
30
-
31
-@include fftools-common-opts.texi
32
-
33
-@section Main options
34
-
35
-@table @option
36
-@item -x @var{width}
37
-Force displayed width.
38
-@item -y @var{height}
39
-Force displayed height.
40
-@item -s @var{size}
41
-This option has been removed. Use private format options for specifying the
42
-input video size.
43
-@item -an
44
-Disable audio.
45
-@item -vn
46
-Disable video.
47
-@item -ss @var{pos}
48
-Seek to a given position in seconds.
49
-@item -t @var{duration}
50
-play <duration> seconds of audio/video
51
-@item -bytes
52
-Seek by bytes.
53
-@item -nodisp
54
-Disable graphical display.
55
-@item -f @var{fmt}
56
-Force format.
57
-@item -window_title @var{title}
58
-Set window title (default is the input filename).
59
-@item -loop @var{number}
60
-Loops movie playback <number> times. 0 means forever.
61
-@item -vf @var{filter_graph}
62
-@var{filter_graph} is a description of the filter graph to apply to
63
-the input video.
64
-Use the option "-filters" to show all the available filters (including
65
-also sources and sinks).
66
-
67
-@end table
68
-
69
-@section Advanced options
70
-@table @option
71
-@item -pix_fmt @var{format}
72
-This option has been removed. Use private options for specifying the
73
-input pixel format.
74
-@item -stats
75
-Show the stream duration, the codec parameters, the current position in
76
-the stream and the audio/video synchronisation drift.
77
-@item -debug
78
-Print specific debug info.
79
-@item -bug
80
-Work around bugs.
81
-@item -vismv
82
-Visualize motion vectors.
83
-@item -fast
84
-Non-spec-compliant optimizations.
85
-@item -genpts
86
-Generate pts.
87
-@item -rtp_tcp
88
-Force RTP/TCP protocol usage instead of RTP/UDP. It is only meaningful
89
-if you are streaming with the RTSP protocol.
90
-@item -sync @var{type}
91
-Set the master clock to audio (@code{type=audio}), video
92
-(@code{type=video}) or external (@code{type=ext}). Default is audio. The
93
-master clock is used to control audio-video synchronization. Most media
94
-players use audio as master clock, but in some cases (streaming or high
95
-quality broadcast) it is necessary to change that. This option is mainly
96
-used for debugging purposes.
97
-@item -threads @var{count}
98
-Set the thread count.
99
-@item -ast @var{audio_stream_number}
100
-Select the desired audio stream number, counting from 0. The number
101
-refers to the list of all the input audio streams. If it is greater
102
-than the number of audio streams minus one, then the last one is
103
-selected, if it is negative the audio playback is disabled.
104
-@item -vst @var{video_stream_number}
105
-Select the desired video stream number, counting from 0. The number
106
-refers to the list of all the input video streams. If it is greater
107
-than the number of video streams minus one, then the last one is
108
-selected, if it is negative the video playback is disabled.
109
-@item -sst @var{subtitle_stream_number}
110
-Select the desired subtitle stream number, counting from 0. The number
111
-refers to the list of all the input subtitle streams. If it is greater
112
-than the number of subtitle streams minus one, then the last one is
113
-selected, if it is negative the subtitle rendering is disabled.
114
-@item -autoexit
115
-Exit when video is done playing.
116
-@item -exitonkeydown
117
-Exit if any key is pressed.
118
-@item -exitonmousedown
119
-Exit if any mouse button is pressed.
120
-@end table
121
-
122
-@section While playing
123
-
124
-@table @key
125
-@item q, ESC
126
-Quit.
127
-
128
-@item f
129
-Toggle full screen.
130
-
131
-@item p, SPC
132
-Pause.
133
-
134
-@item a
135
-Cycle audio channel.
136
-
137
-@item v
138
-Cycle video channel.
139
-
140
-@item t
141
-Cycle subtitle channel.
142
-
143
-@item w
144
-Show audio waves.
145
-
146
-@item left/right
147
-Seek backward/forward 10 seconds.
148
-
149
-@item down/up
150
-Seek backward/forward 1 minute.
151
-
152
-@item mouse click
153
-Seek to percentage in file corresponding to fraction of width.
154
-
155
-@end table
156
-
157
-@c man end
158
-
159
-@include eval.texi
160
-@include demuxers.texi
161
-@include muxers.texi
162
-@include indevs.texi
163
-@include outdevs.texi
164
-@include protocols.texi
165
-@include filters.texi
166
-
167
-@ignore
168
-
169
-@setfilename ffplay
170
-@settitle FFplay media player
171
-
172
-@c man begin SEEALSO
173
-ffmpeg(1), ffprobe(1), ffserver(1) and the Libav HTML documentation
174
-@c man end
175
-
176
-@c man begin AUTHORS
177
-The Libav developers
178
-@c man end
179
-
180
-@end ignore
181
-
182
-@bye
... ...
@@ -122,7 +122,7 @@ with name "STREAM".
122 122
 @settitle ffprobe media prober
123 123
 
124 124
 @c man begin SEEALSO
125
-ffmpeg(1), ffplay(1), ffserver(1) and the Libav HTML documentation
125
+ffmpeg(1), avplay(1), ffserver(1) and the Libav HTML documentation
126 126
 @c man end
127 127
 
128 128
 @c man begin AUTHORS
... ...
@@ -265,7 +265,7 @@ rather than as a daemon.
265 265
 
266 266
 @c man begin SEEALSO
267 267
 
268
-ffmpeg(1), ffplay(1), ffprobe(1), the @file{ffmpeg/doc/ffserver.conf}
268
+ffmpeg(1), avplay(1), ffprobe(1), the @file{ffmpeg/doc/ffserver.conf}
269 269
 example and the Libav HTML documentation
270 270
 @c man end
271 271
 
... ...
@@ -837,7 +837,7 @@ speed up is close to non-existent for normal one-off builds and is only
837 837
 noticeable when running make for a second time (for example in
838 838
 @code{make install}).
839 839
 
840
-@item In order to compile FFplay, you must have the MinGW development library
840
+@item In order to compile AVplay, you must have the MinGW development library
841 841
 of @uref{http://www.libsdl.org/, SDL}.
842 842
 Edit the @file{bin/sdl-config} script so that it points to the correct prefix
843 843
 where SDL was installed. Verify that @file{sdl-config} can be launched from
... ...
@@ -199,10 +199,10 @@ tools.
199 199
 @example
200 200
 # Grab and show the input of a video4linux device, frame rate is set
201 201
 # to the default of 25/1.
202
-ffplay -s 320x240 -f video4linux /dev/video0
202
+avplay -s 320x240 -f video4linux /dev/video0
203 203
 
204 204
 # Grab and show the input of a video4linux2 device, autoadjust size.
205
-ffplay -f video4linux2 /dev/video0
205
+avplay -f video4linux2 /dev/video0
206 206
 
207 207
 # Grab and record the input of a video4linux2 device, autoadjust size,
208 208
 # frame rate value defaults to 0/0 so it is read from the video4linux2
... ...
@@ -24,7 +24,7 @@ some directory of your choice by:
24 24
 @end example
25 25
 
26 26
 And then read the README file in the top directory to learn how to
27
-integrate it into ffmpeg and ffplay.
27
+integrate it into ffmpeg and avplay.
28 28
 
29 29
 But note that there may still be serious bugs in the code and its API
30 30
 and ABI should not be considered stable yet!
... ...
@@ -56,7 +56,7 @@ result will be that in output the top half of the video is mirrored
56 56
 onto the bottom half.
57 57
 
58 58
 Video filters are loaded using the @var{-vf} option passed to
59
-ffmpeg or to ffplay. Filters in the same linear chain are separated by
59
+ffmpeg or to avplay. Filters in the same linear chain are separated by
60 60
 commas. In our example, @var{split, fifo, overlay} are in one linear
61 61
 chain, and @var{fifo, crop, vflip} are in another. The points where
62 62
 the linear chains join are labeled by names enclosed in square
... ...
@@ -52,10 +52,10 @@ resource to be concatenated, each one possibly specifying a distinct
52 52
 protocol.
53 53
 
54 54
 For example to read a sequence of files @file{split1.mpeg},
55
-@file{split2.mpeg}, @file{split3.mpeg} with @file{ffplay} use the
55
+@file{split2.mpeg}, @file{split3.mpeg} with @file{avplay} use the
56 56
 command:
57 57
 @example
58
-ffplay concat:split1.mpeg\|split2.mpeg\|split3.mpeg
58
+avplay concat:split1.mpeg\|split2.mpeg\|split3.mpeg
59 59
 @end example
60 60
 
61 61
 Note that you may need to escape the character "|" which is special for
... ...
@@ -183,10 +183,10 @@ application specified in @var{app}, may be prefixed by "mp4:".
183 183
 
184 184
 @end table
185 185
 
186
-For example to read with @file{ffplay} a multimedia resource named
186
+For example to read with @file{avplay} a multimedia resource named
187 187
 "sample" from the application "vod" from an RTMP server "myserver":
188 188
 @example
189
-ffplay rtmp://myserver/vod/sample
189
+avplay rtmp://myserver/vod/sample
190 190
 @end example
191 191
 
192 192
 @section rtmp, rtmpe, rtmps, rtmpt, rtmpte
... ...
@@ -224,9 +224,9 @@ For example, to stream a file in real-time to an RTMP server using
224 224
 ffmpeg -re -i myfile -f flv rtmp://myserver/live/mystream
225 225
 @end example
226 226
 
227
-To play the same stream using @file{ffplay}:
227
+To play the same stream using @file{avplay}:
228 228
 @example
229
-ffplay "rtmp://myserver/live/mystream live=1"
229
+avplay "rtmp://myserver/live/mystream live=1"
230 230
 @end example
231 231
 
232 232
 @section rtp
... ...
@@ -281,7 +281,7 @@ When receiving data over UDP, the demuxer tries to reorder received packets
281 281
 order for this to be enabled, a maximum delay must be specified in the
282 282
 @code{max_delay} field of AVFormatContext.
283 283
 
284
-When watching multi-bitrate Real-RTSP streams with @file{ffplay}, the
284
+When watching multi-bitrate Real-RTSP streams with @file{avplay}, the
285 285
 streams to display can be chosen with @code{-vst} @var{n} and
286 286
 @code{-ast} @var{n} for video and audio respectively, and can be switched
287 287
 on the fly by pressing @code{v} and @code{a}.
... ...
@@ -291,13 +291,13 @@ Example command lines:
291 291
 To watch a stream over UDP, with a max reordering delay of 0.5 seconds:
292 292
 
293 293
 @example
294
-ffplay -max_delay 500000 rtsp://server/video.mp4?udp
294
+avplay -max_delay 500000 rtsp://server/video.mp4?udp
295 295
 @end example
296 296
 
297 297
 To watch a stream tunneled over HTTP:
298 298
 
299 299
 @example
300
-ffplay rtsp://server/video.mp4?http
300
+avplay rtsp://server/video.mp4?http
301 301
 @end example
302 302
 
303 303
 To send a stream in realtime to a RTSP server, for others to watch:
... ...
@@ -358,13 +358,13 @@ To broadcast a stream on the local subnet, for watching in VLC:
358 358
 ffmpeg -re -i @var{input} -f sap sap://224.0.0.255?same_port=1
359 359
 @end example
360 360
 
361
-Similarly, for watching in ffplay:
361
+Similarly, for watching in avplay:
362 362
 
363 363
 @example
364 364
 ffmpeg -re -i @var{input} -f sap sap://224.0.0.255
365 365
 @end example
366 366
 
367
-And for watching in ffplay, over IPv6:
367
+And for watching in avplay, over IPv6:
368 368
 
369 369
 @example
370 370
 ffmpeg -re -i @var{input} -f sap sap://[ff0e::1:2:3:4]
... ...
@@ -389,13 +389,13 @@ Example command lines follow.
389 389
 To play back the first stream announced on the normal SAP multicast address:
390 390
 
391 391
 @example
392
-ffplay sap://
392
+avplay sap://
393 393
 @end example
394 394
 
395 395
 To play back the first stream announced on one the default IPv6 SAP multicast address:
396 396
 
397 397
 @example
398
-ffplay sap://[ff0e::2:7ffe]
398
+avplay sap://[ff0e::2:7ffe]
399 399
 @end example
400 400
 
401 401
 @section tcp
... ...
@@ -414,7 +414,7 @@ Listen for an incoming connection
414 414
 
415 415
 @example
416 416
 ffmpeg -i @var{input} -f @var{format} tcp://@var{hostname}:@var{port}?listen
417
-ffplay tcp://@var{hostname}:@var{port}
417
+avplay tcp://@var{hostname}:@var{port}
418 418
 @end example
419 419
 
420 420
 @end table
421 421
deleted file mode 100644
... ...
@@ -1,3071 +0,0 @@
1
-/*
2
- * ffplay : Simple Media Player based on the Libav libraries
3
- * Copyright (c) 2003 Fabrice Bellard
4
- *
5
- * This file is part of Libav.
6
- *
7
- * Libav is free software; you can redistribute it and/or
8
- * modify it under the terms of the GNU Lesser General Public
9
- * License as published by the Free Software Foundation; either
10
- * version 2.1 of the License, or (at your option) any later version.
11
- *
12
- * Libav is distributed in the hope that it will be useful,
13
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
14
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15
- * Lesser General Public License for more details.
16
- *
17
- * You should have received a copy of the GNU Lesser General Public
18
- * License along with Libav; if not, write to the Free Software
19
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20
- */
21
-
22
-#include "config.h"
23
-#include <inttypes.h>
24
-#include <math.h>
25
-#include <limits.h>
26
-#include "libavutil/avstring.h"
27
-#include "libavutil/colorspace.h"
28
-#include "libavutil/mathematics.h"
29
-#include "libavutil/pixdesc.h"
30
-#include "libavutil/imgutils.h"
31
-#include "libavutil/dict.h"
32
-#include "libavutil/parseutils.h"
33
-#include "libavutil/samplefmt.h"
34
-#include "libavformat/avformat.h"
35
-#include "libavdevice/avdevice.h"
36
-#include "libswscale/swscale.h"
37
-#include "libavcodec/audioconvert.h"
38
-#include "libavutil/opt.h"
39
-#include "libavcodec/avfft.h"
40
-
41
-#if CONFIG_AVFILTER
42
-# include "libavfilter/avfilter.h"
43
-# include "libavfilter/avfiltergraph.h"
44
-#endif
45
-
46
-#include "cmdutils.h"
47
-
48
-#include <SDL.h>
49
-#include <SDL_thread.h>
50
-
51
-#ifdef __MINGW32__
52
-#undef main /* We don't want SDL to override our main() */
53
-#endif
54
-
55
-#include <unistd.h>
56
-#include <assert.h>
57
-
58
-const char program_name[] = "ffplay";
59
-const int program_birth_year = 2003;
60
-
61
-#define MAX_QUEUE_SIZE (15 * 1024 * 1024)
62
-#define MIN_AUDIOQ_SIZE (20 * 16 * 1024)
63
-#define MIN_FRAMES 5
64
-
65
-/* SDL audio buffer size, in samples. Should be small to have precise
66
-   A/V sync as SDL does not have hardware buffer fullness info. */
67
-#define SDL_AUDIO_BUFFER_SIZE 1024
68
-
69
-/* no AV sync correction is done if below the AV sync threshold */
70
-#define AV_SYNC_THRESHOLD 0.01
71
-/* no AV correction is done if too big error */
72
-#define AV_NOSYNC_THRESHOLD 10.0
73
-
74
-#define FRAME_SKIP_FACTOR 0.05
75
-
76
-/* maximum audio speed change to get correct sync */
77
-#define SAMPLE_CORRECTION_PERCENT_MAX 10
78
-
79
-/* we use about AUDIO_DIFF_AVG_NB A-V differences to make the average */
80
-#define AUDIO_DIFF_AVG_NB   20
81
-
82
-/* NOTE: the size must be big enough to compensate the hardware audio buffersize size */
83
-#define SAMPLE_ARRAY_SIZE (2*65536)
84
-
85
-static int sws_flags = SWS_BICUBIC;
86
-
87
-typedef struct PacketQueue {
88
-    AVPacketList *first_pkt, *last_pkt;
89
-    int nb_packets;
90
-    int size;
91
-    int abort_request;
92
-    SDL_mutex *mutex;
93
-    SDL_cond *cond;
94
-} PacketQueue;
95
-
96
-#define VIDEO_PICTURE_QUEUE_SIZE 2
97
-#define SUBPICTURE_QUEUE_SIZE 4
98
-
99
-typedef struct VideoPicture {
100
-    double pts;                                  ///<presentation time stamp for this picture
101
-    double target_clock;                         ///<av_gettime() time at which this should be displayed ideally
102
-    int64_t pos;                                 ///<byte position in file
103
-    SDL_Overlay *bmp;
104
-    int width, height; /* source height & width */
105
-    int allocated;
106
-    enum PixelFormat pix_fmt;
107
-
108
-#if CONFIG_AVFILTER
109
-    AVFilterBufferRef *picref;
110
-#endif
111
-} VideoPicture;
112
-
113
-typedef struct SubPicture {
114
-    double pts; /* presentation time stamp for this picture */
115
-    AVSubtitle sub;
116
-} SubPicture;
117
-
118
-enum {
119
-    AV_SYNC_AUDIO_MASTER, /* default choice */
120
-    AV_SYNC_VIDEO_MASTER,
121
-    AV_SYNC_EXTERNAL_CLOCK, /* synchronize to an external clock */
122
-};
123
-
124
-typedef struct VideoState {
125
-    SDL_Thread *parse_tid;
126
-    SDL_Thread *video_tid;
127
-    SDL_Thread *refresh_tid;
128
-    AVInputFormat *iformat;
129
-    int no_background;
130
-    int abort_request;
131
-    int paused;
132
-    int last_paused;
133
-    int seek_req;
134
-    int seek_flags;
135
-    int64_t seek_pos;
136
-    int64_t seek_rel;
137
-    int read_pause_return;
138
-    AVFormatContext *ic;
139
-    int dtg_active_format;
140
-
141
-    int audio_stream;
142
-
143
-    int av_sync_type;
144
-    double external_clock; /* external clock base */
145
-    int64_t external_clock_time;
146
-
147
-    double audio_clock;
148
-    double audio_diff_cum; /* used for AV difference average computation */
149
-    double audio_diff_avg_coef;
150
-    double audio_diff_threshold;
151
-    int audio_diff_avg_count;
152
-    AVStream *audio_st;
153
-    PacketQueue audioq;
154
-    int audio_hw_buf_size;
155
-    /* samples output by the codec. we reserve more space for avsync
156
-       compensation */
157
-    DECLARE_ALIGNED(16,uint8_t,audio_buf1)[(AVCODEC_MAX_AUDIO_FRAME_SIZE * 3) / 2];
158
-    DECLARE_ALIGNED(16,uint8_t,audio_buf2)[(AVCODEC_MAX_AUDIO_FRAME_SIZE * 3) / 2];
159
-    uint8_t *audio_buf;
160
-    unsigned int audio_buf_size; /* in bytes */
161
-    int audio_buf_index; /* in bytes */
162
-    AVPacket audio_pkt_temp;
163
-    AVPacket audio_pkt;
164
-    enum AVSampleFormat audio_src_fmt;
165
-    AVAudioConvert *reformat_ctx;
166
-
167
-    int show_audio; /* if true, display audio samples */
168
-    int16_t sample_array[SAMPLE_ARRAY_SIZE];
169
-    int sample_array_index;
170
-    int last_i_start;
171
-    RDFTContext *rdft;
172
-    int rdft_bits;
173
-    FFTSample *rdft_data;
174
-    int xpos;
175
-
176
-    SDL_Thread *subtitle_tid;
177
-    int subtitle_stream;
178
-    int subtitle_stream_changed;
179
-    AVStream *subtitle_st;
180
-    PacketQueue subtitleq;
181
-    SubPicture subpq[SUBPICTURE_QUEUE_SIZE];
182
-    int subpq_size, subpq_rindex, subpq_windex;
183
-    SDL_mutex *subpq_mutex;
184
-    SDL_cond *subpq_cond;
185
-
186
-    double frame_timer;
187
-    double frame_last_pts;
188
-    double frame_last_delay;
189
-    double video_clock;                          ///<pts of last decoded frame / predicted pts of next decoded frame
190
-    int video_stream;
191
-    AVStream *video_st;
192
-    PacketQueue videoq;
193
-    double video_current_pts;                    ///<current displayed pts (different from video_clock if frame fifos are used)
194
-    double video_current_pts_drift;              ///<video_current_pts - time (av_gettime) at which we updated video_current_pts - used to have running video pts
195
-    int64_t video_current_pos;                   ///<current displayed file pos
196
-    VideoPicture pictq[VIDEO_PICTURE_QUEUE_SIZE];
197
-    int pictq_size, pictq_rindex, pictq_windex;
198
-    SDL_mutex *pictq_mutex;
199
-    SDL_cond *pictq_cond;
200
-#if !CONFIG_AVFILTER
201
-    struct SwsContext *img_convert_ctx;
202
-#endif
203
-
204
-    //    QETimer *video_timer;
205
-    char filename[1024];
206
-    int width, height, xleft, ytop;
207
-
208
-    PtsCorrectionContext pts_ctx;
209
-
210
-#if CONFIG_AVFILTER
211
-    AVFilterContext *out_video_filter;          ///<the last filter in the video chain
212
-#endif
213
-
214
-    float skip_frames;
215
-    float skip_frames_index;
216
-    int refresh;
217
-} VideoState;
218
-
219
-static void show_help(void);
220
-
221
-/* options specified by the user */
222
-static AVInputFormat *file_iformat;
223
-static const char *input_filename;
224
-static const char *window_title;
225
-static int fs_screen_width;
226
-static int fs_screen_height;
227
-static int screen_width = 0;
228
-static int screen_height = 0;
229
-static int audio_disable;
230
-static int video_disable;
231
-static int wanted_stream[AVMEDIA_TYPE_NB]={
232
-    [AVMEDIA_TYPE_AUDIO]=-1,
233
-    [AVMEDIA_TYPE_VIDEO]=-1,
234
-    [AVMEDIA_TYPE_SUBTITLE]=-1,
235
-};
236
-static int seek_by_bytes=-1;
237
-static int display_disable;
238
-static int show_status = 1;
239
-static int av_sync_type = AV_SYNC_AUDIO_MASTER;
240
-static int64_t start_time = AV_NOPTS_VALUE;
241
-static int64_t duration = AV_NOPTS_VALUE;
242
-static int debug = 0;
243
-static int debug_mv = 0;
244
-static int step = 0;
245
-static int thread_count = 1;
246
-static int workaround_bugs = 1;
247
-static int fast = 0;
248
-static int genpts = 0;
249
-static int lowres = 0;
250
-static int idct = FF_IDCT_AUTO;
251
-static enum AVDiscard skip_frame= AVDISCARD_DEFAULT;
252
-static enum AVDiscard skip_idct= AVDISCARD_DEFAULT;
253
-static enum AVDiscard skip_loop_filter= AVDISCARD_DEFAULT;
254
-static int error_recognition = FF_ER_CAREFUL;
255
-static int error_concealment = 3;
256
-static int decoder_reorder_pts= -1;
257
-static int autoexit;
258
-static int exit_on_keydown;
259
-static int exit_on_mousedown;
260
-static int loop=1;
261
-static int framedrop=1;
262
-
263
-static int rdftspeed=20;
264
-#if CONFIG_AVFILTER
265
-static char *vfilters = NULL;
266
-#endif
267
-
268
-/* current context */
269
-static int is_full_screen;
270
-static VideoState *cur_stream;
271
-static int64_t audio_callback_time;
272
-
273
-static AVPacket flush_pkt;
274
-
275
-#define FF_ALLOC_EVENT   (SDL_USEREVENT)
276
-#define FF_REFRESH_EVENT (SDL_USEREVENT + 1)
277
-#define FF_QUIT_EVENT    (SDL_USEREVENT + 2)
278
-
279
-static SDL_Surface *screen;
280
-
281
-static int packet_queue_put(PacketQueue *q, AVPacket *pkt);
282
-
283
-/* packet queue handling */
284
-static void packet_queue_init(PacketQueue *q)
285
-{
286
-    memset(q, 0, sizeof(PacketQueue));
287
-    q->mutex = SDL_CreateMutex();
288
-    q->cond = SDL_CreateCond();
289
-    packet_queue_put(q, &flush_pkt);
290
-}
291
-
292
-static void packet_queue_flush(PacketQueue *q)
293
-{
294
-    AVPacketList *pkt, *pkt1;
295
-
296
-    SDL_LockMutex(q->mutex);
297
-    for(pkt = q->first_pkt; pkt != NULL; pkt = pkt1) {
298
-        pkt1 = pkt->next;
299
-        av_free_packet(&pkt->pkt);
300
-        av_freep(&pkt);
301
-    }
302
-    q->last_pkt = NULL;
303
-    q->first_pkt = NULL;
304
-    q->nb_packets = 0;
305
-    q->size = 0;
306
-    SDL_UnlockMutex(q->mutex);
307
-}
308
-
309
-static void packet_queue_end(PacketQueue *q)
310
-{
311
-    packet_queue_flush(q);
312
-    SDL_DestroyMutex(q->mutex);
313
-    SDL_DestroyCond(q->cond);
314
-}
315
-
316
-static int packet_queue_put(PacketQueue *q, AVPacket *pkt)
317
-{
318
-    AVPacketList *pkt1;
319
-
320
-    /* duplicate the packet */
321
-    if (pkt!=&flush_pkt && av_dup_packet(pkt) < 0)
322
-        return -1;
323
-
324
-    pkt1 = av_malloc(sizeof(AVPacketList));
325
-    if (!pkt1)
326
-        return -1;
327
-    pkt1->pkt = *pkt;
328
-    pkt1->next = NULL;
329
-
330
-
331
-    SDL_LockMutex(q->mutex);
332
-
333
-    if (!q->last_pkt)
334
-
335
-        q->first_pkt = pkt1;
336
-    else
337
-        q->last_pkt->next = pkt1;
338
-    q->last_pkt = pkt1;
339
-    q->nb_packets++;
340
-    q->size += pkt1->pkt.size + sizeof(*pkt1);
341
-    /* XXX: should duplicate packet data in DV case */
342
-    SDL_CondSignal(q->cond);
343
-
344
-    SDL_UnlockMutex(q->mutex);
345
-    return 0;
346
-}
347
-
348
-static void packet_queue_abort(PacketQueue *q)
349
-{
350
-    SDL_LockMutex(q->mutex);
351
-
352
-    q->abort_request = 1;
353
-
354
-    SDL_CondSignal(q->cond);
355
-
356
-    SDL_UnlockMutex(q->mutex);
357
-}
358
-
359
-/* return < 0 if aborted, 0 if no packet and > 0 if packet.  */
360
-static int packet_queue_get(PacketQueue *q, AVPacket *pkt, int block)
361
-{
362
-    AVPacketList *pkt1;
363
-    int ret;
364
-
365
-    SDL_LockMutex(q->mutex);
366
-
367
-    for(;;) {
368
-        if (q->abort_request) {
369
-            ret = -1;
370
-            break;
371
-        }
372
-
373
-        pkt1 = q->first_pkt;
374
-        if (pkt1) {
375
-            q->first_pkt = pkt1->next;
376
-            if (!q->first_pkt)
377
-                q->last_pkt = NULL;
378
-            q->nb_packets--;
379
-            q->size -= pkt1->pkt.size + sizeof(*pkt1);
380
-            *pkt = pkt1->pkt;
381
-            av_free(pkt1);
382
-            ret = 1;
383
-            break;
384
-        } else if (!block) {
385
-            ret = 0;
386
-            break;
387
-        } else {
388
-            SDL_CondWait(q->cond, q->mutex);
389
-        }
390
-    }
391
-    SDL_UnlockMutex(q->mutex);
392
-    return ret;
393
-}
394
-
395
-static inline void fill_rectangle(SDL_Surface *screen,
396
-                                  int x, int y, int w, int h, int color)
397
-{
398
-    SDL_Rect rect;
399
-    rect.x = x;
400
-    rect.y = y;
401
-    rect.w = w;
402
-    rect.h = h;
403
-    SDL_FillRect(screen, &rect, color);
404
-}
405
-
406
-#define ALPHA_BLEND(a, oldp, newp, s)\
407
-((((oldp << s) * (255 - (a))) + (newp * (a))) / (255 << s))
408
-
409
-#define RGBA_IN(r, g, b, a, s)\
410
-{\
411
-    unsigned int v = ((const uint32_t *)(s))[0];\
412
-    a = (v >> 24) & 0xff;\
413
-    r = (v >> 16) & 0xff;\
414
-    g = (v >> 8) & 0xff;\
415
-    b = v & 0xff;\
416
-}
417
-
418
-#define YUVA_IN(y, u, v, a, s, pal)\
419
-{\
420
-    unsigned int val = ((const uint32_t *)(pal))[*(const uint8_t*)(s)];\
421
-    a = (val >> 24) & 0xff;\
422
-    y = (val >> 16) & 0xff;\
423
-    u = (val >> 8) & 0xff;\
424
-    v = val & 0xff;\
425
-}
426
-
427
-#define YUVA_OUT(d, y, u, v, a)\
428
-{\
429
-    ((uint32_t *)(d))[0] = (a << 24) | (y << 16) | (u << 8) | v;\
430
-}
431
-
432
-
433
-#define BPP 1
434
-
435
-static void blend_subrect(AVPicture *dst, const AVSubtitleRect *rect, int imgw, int imgh)
436
-{
437
-    int wrap, wrap3, width2, skip2;
438
-    int y, u, v, a, u1, v1, a1, w, h;
439
-    uint8_t *lum, *cb, *cr;
440
-    const uint8_t *p;
441
-    const uint32_t *pal;
442
-    int dstx, dsty, dstw, dsth;
443
-
444
-    dstw = av_clip(rect->w, 0, imgw);
445
-    dsth = av_clip(rect->h, 0, imgh);
446
-    dstx = av_clip(rect->x, 0, imgw - dstw);
447
-    dsty = av_clip(rect->y, 0, imgh - dsth);
448
-    lum = dst->data[0] + dsty * dst->linesize[0];
449
-    cb = dst->data[1] + (dsty >> 1) * dst->linesize[1];
450
-    cr = dst->data[2] + (dsty >> 1) * dst->linesize[2];
451
-
452
-    width2 = ((dstw + 1) >> 1) + (dstx & ~dstw & 1);
453
-    skip2 = dstx >> 1;
454
-    wrap = dst->linesize[0];
455
-    wrap3 = rect->pict.linesize[0];
456
-    p = rect->pict.data[0];
457
-    pal = (const uint32_t *)rect->pict.data[1];  /* Now in YCrCb! */
458
-
459
-    if (dsty & 1) {
460
-        lum += dstx;
461
-        cb += skip2;
462
-        cr += skip2;
463
-
464
-        if (dstx & 1) {
465
-            YUVA_IN(y, u, v, a, p, pal);
466
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
467
-            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
468
-            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
469
-            cb++;
470
-            cr++;
471
-            lum++;
472
-            p += BPP;
473
-        }
474
-        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
475
-            YUVA_IN(y, u, v, a, p, pal);
476
-            u1 = u;
477
-            v1 = v;
478
-            a1 = a;
479
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
480
-
481
-            YUVA_IN(y, u, v, a, p + BPP, pal);
482
-            u1 += u;
483
-            v1 += v;
484
-            a1 += a;
485
-            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
486
-            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
487
-            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
488
-            cb++;
489
-            cr++;
490
-            p += 2 * BPP;
491
-            lum += 2;
492
-        }
493
-        if (w) {
494
-            YUVA_IN(y, u, v, a, p, pal);
495
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
496
-            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
497
-            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
498
-            p++;
499
-            lum++;
500
-        }
501
-        p += wrap3 - dstw * BPP;
502
-        lum += wrap - dstw - dstx;
503
-        cb += dst->linesize[1] - width2 - skip2;
504
-        cr += dst->linesize[2] - width2 - skip2;
505
-    }
506
-    for(h = dsth - (dsty & 1); h >= 2; h -= 2) {
507
-        lum += dstx;
508
-        cb += skip2;
509
-        cr += skip2;
510
-
511
-        if (dstx & 1) {
512
-            YUVA_IN(y, u, v, a, p, pal);
513
-            u1 = u;
514
-            v1 = v;
515
-            a1 = a;
516
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
517
-            p += wrap3;
518
-            lum += wrap;
519
-            YUVA_IN(y, u, v, a, p, pal);
520
-            u1 += u;
521
-            v1 += v;
522
-            a1 += a;
523
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
524
-            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
525
-            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
526
-            cb++;
527
-            cr++;
528
-            p += -wrap3 + BPP;
529
-            lum += -wrap + 1;
530
-        }
531
-        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
532
-            YUVA_IN(y, u, v, a, p, pal);
533
-            u1 = u;
534
-            v1 = v;
535
-            a1 = a;
536
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
537
-
538
-            YUVA_IN(y, u, v, a, p + BPP, pal);
539
-            u1 += u;
540
-            v1 += v;
541
-            a1 += a;
542
-            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
543
-            p += wrap3;
544
-            lum += wrap;
545
-
546
-            YUVA_IN(y, u, v, a, p, pal);
547
-            u1 += u;
548
-            v1 += v;
549
-            a1 += a;
550
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
551
-
552
-            YUVA_IN(y, u, v, a, p + BPP, pal);
553
-            u1 += u;
554
-            v1 += v;
555
-            a1 += a;
556
-            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
557
-
558
-            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 2);
559
-            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 2);
560
-
561
-            cb++;
562
-            cr++;
563
-            p += -wrap3 + 2 * BPP;
564
-            lum += -wrap + 2;
565
-        }
566
-        if (w) {
567
-            YUVA_IN(y, u, v, a, p, pal);
568
-            u1 = u;
569
-            v1 = v;
570
-            a1 = a;
571
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
572
-            p += wrap3;
573
-            lum += wrap;
574
-            YUVA_IN(y, u, v, a, p, pal);
575
-            u1 += u;
576
-            v1 += v;
577
-            a1 += a;
578
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
579
-            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u1, 1);
580
-            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v1, 1);
581
-            cb++;
582
-            cr++;
583
-            p += -wrap3 + BPP;
584
-            lum += -wrap + 1;
585
-        }
586
-        p += wrap3 + (wrap3 - dstw * BPP);
587
-        lum += wrap + (wrap - dstw - dstx);
588
-        cb += dst->linesize[1] - width2 - skip2;
589
-        cr += dst->linesize[2] - width2 - skip2;
590
-    }
591
-    /* handle odd height */
592
-    if (h) {
593
-        lum += dstx;
594
-        cb += skip2;
595
-        cr += skip2;
596
-
597
-        if (dstx & 1) {
598
-            YUVA_IN(y, u, v, a, p, pal);
599
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
600
-            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
601
-            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
602
-            cb++;
603
-            cr++;
604
-            lum++;
605
-            p += BPP;
606
-        }
607
-        for(w = dstw - (dstx & 1); w >= 2; w -= 2) {
608
-            YUVA_IN(y, u, v, a, p, pal);
609
-            u1 = u;
610
-            v1 = v;
611
-            a1 = a;
612
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
613
-
614
-            YUVA_IN(y, u, v, a, p + BPP, pal);
615
-            u1 += u;
616
-            v1 += v;
617
-            a1 += a;
618
-            lum[1] = ALPHA_BLEND(a, lum[1], y, 0);
619
-            cb[0] = ALPHA_BLEND(a1 >> 2, cb[0], u, 1);
620
-            cr[0] = ALPHA_BLEND(a1 >> 2, cr[0], v, 1);
621
-            cb++;
622
-            cr++;
623
-            p += 2 * BPP;
624
-            lum += 2;
625
-        }
626
-        if (w) {
627
-            YUVA_IN(y, u, v, a, p, pal);
628
-            lum[0] = ALPHA_BLEND(a, lum[0], y, 0);
629
-            cb[0] = ALPHA_BLEND(a >> 2, cb[0], u, 0);
630
-            cr[0] = ALPHA_BLEND(a >> 2, cr[0], v, 0);
631
-        }
632
-    }
633
-}
634
-
635
-static void free_subpicture(SubPicture *sp)
636
-{
637
-    avsubtitle_free(&sp->sub);
638
-}
639
-
640
-static void video_image_display(VideoState *is)
641
-{
642
-    VideoPicture *vp;
643
-    SubPicture *sp;
644
-    AVPicture pict;
645
-    float aspect_ratio;
646
-    int width, height, x, y;
647
-    SDL_Rect rect;
648
-    int i;
649
-
650
-    vp = &is->pictq[is->pictq_rindex];
651
-    if (vp->bmp) {
652
-#if CONFIG_AVFILTER
653
-         if (vp->picref->video->pixel_aspect.num == 0)
654
-             aspect_ratio = 0;
655
-         else
656
-             aspect_ratio = av_q2d(vp->picref->video->pixel_aspect);
657
-#else
658
-
659
-        /* XXX: use variable in the frame */
660
-        if (is->video_st->sample_aspect_ratio.num)
661
-            aspect_ratio = av_q2d(is->video_st->sample_aspect_ratio);
662
-        else if (is->video_st->codec->sample_aspect_ratio.num)
663
-            aspect_ratio = av_q2d(is->video_st->codec->sample_aspect_ratio);
664
-        else
665
-            aspect_ratio = 0;
666
-#endif
667
-        if (aspect_ratio <= 0.0)
668
-            aspect_ratio = 1.0;
669
-        aspect_ratio *= (float)vp->width / (float)vp->height;
670
-
671
-        if (is->subtitle_st)
672
-        {
673
-            if (is->subpq_size > 0)
674
-            {
675
-                sp = &is->subpq[is->subpq_rindex];
676
-
677
-                if (vp->pts >= sp->pts + ((float) sp->sub.start_display_time / 1000))
678
-                {
679
-                    SDL_LockYUVOverlay (vp->bmp);
680
-
681
-                    pict.data[0] = vp->bmp->pixels[0];
682
-                    pict.data[1] = vp->bmp->pixels[2];
683
-                    pict.data[2] = vp->bmp->pixels[1];
684
-
685
-                    pict.linesize[0] = vp->bmp->pitches[0];
686
-                    pict.linesize[1] = vp->bmp->pitches[2];
687
-                    pict.linesize[2] = vp->bmp->pitches[1];
688
-
689
-                    for (i = 0; i < sp->sub.num_rects; i++)
690
-                        blend_subrect(&pict, sp->sub.rects[i],
691
-                                      vp->bmp->w, vp->bmp->h);
692
-
693
-                    SDL_UnlockYUVOverlay (vp->bmp);
694
-                }
695
-            }
696
-        }
697
-
698
-
699
-        /* XXX: we suppose the screen has a 1.0 pixel ratio */
700
-        height = is->height;
701
-        width = ((int)rint(height * aspect_ratio)) & ~1;
702
-        if (width > is->width) {
703
-            width = is->width;
704
-            height = ((int)rint(width / aspect_ratio)) & ~1;
705
-        }
706
-        x = (is->width - width) / 2;
707
-        y = (is->height - height) / 2;
708
-        is->no_background = 0;
709
-        rect.x = is->xleft + x;
710
-        rect.y = is->ytop  + y;
711
-        rect.w = width;
712
-        rect.h = height;
713
-        SDL_DisplayYUVOverlay(vp->bmp, &rect);
714
-    }
715
-}
716
-
717
-/* get the current audio output buffer size, in samples. With SDL, we
718
-   cannot have a precise information */
719
-static int audio_write_get_buf_size(VideoState *is)
720
-{
721
-    return is->audio_buf_size - is->audio_buf_index;
722
-}
723
-
724
-static inline int compute_mod(int a, int b)
725
-{
726
-    a = a % b;
727
-    if (a >= 0)
728
-        return a;
729
-    else
730
-        return a + b;
731
-}
732
-
733
-static void video_audio_display(VideoState *s)
734
-{
735
-    int i, i_start, x, y1, y, ys, delay, n, nb_display_channels;
736
-    int ch, channels, h, h2, bgcolor, fgcolor;
737
-    int16_t time_diff;
738
-    int rdft_bits, nb_freq;
739
-
740
-    for(rdft_bits=1; (1<<rdft_bits)<2*s->height; rdft_bits++)
741
-        ;
742
-    nb_freq= 1<<(rdft_bits-1);
743
-
744
-    /* compute display index : center on currently output samples */
745
-    channels = s->audio_st->codec->channels;
746
-    nb_display_channels = channels;
747
-    if (!s->paused) {
748
-        int data_used= s->show_audio==1 ? s->width : (2*nb_freq);
749
-        n = 2 * channels;
750
-        delay = audio_write_get_buf_size(s);
751
-        delay /= n;
752
-
753
-        /* to be more precise, we take into account the time spent since
754
-           the last buffer computation */
755
-        if (audio_callback_time) {
756
-            time_diff = av_gettime() - audio_callback_time;
757
-            delay -= (time_diff * s->audio_st->codec->sample_rate) / 1000000;
758
-        }
759
-
760
-        delay += 2*data_used;
761
-        if (delay < data_used)
762
-            delay = data_used;
763
-
764
-        i_start= x = compute_mod(s->sample_array_index - delay * channels, SAMPLE_ARRAY_SIZE);
765
-        if(s->show_audio==1){
766
-            h= INT_MIN;
767
-            for(i=0; i<1000; i+=channels){
768
-                int idx= (SAMPLE_ARRAY_SIZE + x - i) % SAMPLE_ARRAY_SIZE;
769
-                int a= s->sample_array[idx];
770
-                int b= s->sample_array[(idx + 4*channels)%SAMPLE_ARRAY_SIZE];
771
-                int c= s->sample_array[(idx + 5*channels)%SAMPLE_ARRAY_SIZE];
772
-                int d= s->sample_array[(idx + 9*channels)%SAMPLE_ARRAY_SIZE];
773
-                int score= a-d;
774
-                if(h<score && (b^c)<0){
775
-                    h= score;
776
-                    i_start= idx;
777
-                }
778
-            }
779
-        }
780
-
781
-        s->last_i_start = i_start;
782
-    } else {
783
-        i_start = s->last_i_start;
784
-    }
785
-
786
-    bgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0x00);
787
-    if(s->show_audio==1){
788
-        fill_rectangle(screen,
789
-                       s->xleft, s->ytop, s->width, s->height,
790
-                       bgcolor);
791
-
792
-        fgcolor = SDL_MapRGB(screen->format, 0xff, 0xff, 0xff);
793
-
794
-        /* total height for one channel */
795
-        h = s->height / nb_display_channels;
796
-        /* graph height / 2 */
797
-        h2 = (h * 9) / 20;
798
-        for(ch = 0;ch < nb_display_channels; ch++) {
799
-            i = i_start + ch;
800
-            y1 = s->ytop + ch * h + (h / 2); /* position of center line */
801
-            for(x = 0; x < s->width; x++) {
802
-                y = (s->sample_array[i] * h2) >> 15;
803
-                if (y < 0) {
804
-                    y = -y;
805
-                    ys = y1 - y;
806
-                } else {
807
-                    ys = y1;
808
-                }
809
-                fill_rectangle(screen,
810
-                               s->xleft + x, ys, 1, y,
811
-                               fgcolor);
812
-                i += channels;
813
-                if (i >= SAMPLE_ARRAY_SIZE)
814
-                    i -= SAMPLE_ARRAY_SIZE;
815
-            }
816
-        }
817
-
818
-        fgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0xff);
819
-
820
-        for(ch = 1;ch < nb_display_channels; ch++) {
821
-            y = s->ytop + ch * h;
822
-            fill_rectangle(screen,
823
-                           s->xleft, y, s->width, 1,
824
-                           fgcolor);
825
-        }
826
-        SDL_UpdateRect(screen, s->xleft, s->ytop, s->width, s->height);
827
-    }else{
828
-        nb_display_channels= FFMIN(nb_display_channels, 2);
829
-        if(rdft_bits != s->rdft_bits){
830
-            av_rdft_end(s->rdft);
831
-            av_free(s->rdft_data);
832
-            s->rdft = av_rdft_init(rdft_bits, DFT_R2C);
833
-            s->rdft_bits= rdft_bits;
834
-            s->rdft_data= av_malloc(4*nb_freq*sizeof(*s->rdft_data));
835
-        }
836
-        {
837
-            FFTSample *data[2];
838
-            for(ch = 0;ch < nb_display_channels; ch++) {
839
-                data[ch] = s->rdft_data + 2*nb_freq*ch;
840
-                i = i_start + ch;
841
-                for(x = 0; x < 2*nb_freq; x++) {
842
-                    double w= (x-nb_freq)*(1.0/nb_freq);
843
-                    data[ch][x]= s->sample_array[i]*(1.0-w*w);
844
-                    i += channels;
845
-                    if (i >= SAMPLE_ARRAY_SIZE)
846
-                        i -= SAMPLE_ARRAY_SIZE;
847
-                }
848
-                av_rdft_calc(s->rdft, data[ch]);
849
-            }
850
-            //least efficient way to do this, we should of course directly access it but its more than fast enough
851
-            for(y=0; y<s->height; y++){
852
-                double w= 1/sqrt(nb_freq);
853
-                int a= sqrt(w*sqrt(data[0][2*y+0]*data[0][2*y+0] + data[0][2*y+1]*data[0][2*y+1]));
854
-                int b= (nb_display_channels == 2 ) ? sqrt(w*sqrt(data[1][2*y+0]*data[1][2*y+0]
855
-                       + data[1][2*y+1]*data[1][2*y+1])) : a;
856
-                a= FFMIN(a,255);
857
-                b= FFMIN(b,255);
858
-                fgcolor = SDL_MapRGB(screen->format, a, b, (a+b)/2);
859
-
860
-                fill_rectangle(screen,
861
-                            s->xpos, s->height-y, 1, 1,
862
-                            fgcolor);
863
-            }
864
-        }
865
-        SDL_UpdateRect(screen, s->xpos, s->ytop, 1, s->height);
866
-        s->xpos++;
867
-        if(s->xpos >= s->width)
868
-            s->xpos= s->xleft;
869
-    }
870
-}
871
-
872
-static int video_open(VideoState *is){
873
-    int flags = SDL_HWSURFACE|SDL_ASYNCBLIT|SDL_HWACCEL;
874
-    int w,h;
875
-
876
-    if(is_full_screen) flags |= SDL_FULLSCREEN;
877
-    else               flags |= SDL_RESIZABLE;
878
-
879
-    if (is_full_screen && fs_screen_width) {
880
-        w = fs_screen_width;
881
-        h = fs_screen_height;
882
-    } else if(!is_full_screen && screen_width){
883
-        w = screen_width;
884
-        h = screen_height;
885
-#if CONFIG_AVFILTER
886
-    }else if (is->out_video_filter && is->out_video_filter->inputs[0]){
887
-        w = is->out_video_filter->inputs[0]->w;
888
-        h = is->out_video_filter->inputs[0]->h;
889
-#else
890
-    }else if (is->video_st && is->video_st->codec->width){
891
-        w = is->video_st->codec->width;
892
-        h = is->video_st->codec->height;
893
-#endif
894
-    } else {
895
-        w = 640;
896
-        h = 480;
897
-    }
898
-    if(screen && is->width == screen->w && screen->w == w
899
-       && is->height== screen->h && screen->h == h)
900
-        return 0;
901
-
902
-#ifndef __APPLE__
903
-    screen = SDL_SetVideoMode(w, h, 0, flags);
904
-#else
905
-    /* setting bits_per_pixel = 0 or 32 causes blank video on OS X */
906
-    screen = SDL_SetVideoMode(w, h, 24, flags);
907
-#endif
908
-    if (!screen) {
909
-        fprintf(stderr, "SDL: could not set video mode - exiting\n");
910
-        return -1;
911
-    }
912
-    if (!window_title)
913
-        window_title = input_filename;
914
-    SDL_WM_SetCaption(window_title, window_title);
915
-
916
-    is->width = screen->w;
917
-    is->height = screen->h;
918
-
919
-    return 0;
920
-}
921
-
922
-/* display the current picture, if any */
923
-static void video_display(VideoState *is)
924
-{
925
-    if(!screen)
926
-        video_open(cur_stream);
927
-    if (is->audio_st && is->show_audio)
928
-        video_audio_display(is);
929
-    else if (is->video_st)
930
-        video_image_display(is);
931
-}
932
-
933
-static int refresh_thread(void *opaque)
934
-{
935
-    VideoState *is= opaque;
936
-    while(!is->abort_request){
937
-        SDL_Event event;
938
-        event.type = FF_REFRESH_EVENT;
939
-        event.user.data1 = opaque;
940
-        if(!is->refresh){
941
-            is->refresh=1;
942
-            SDL_PushEvent(&event);
943
-        }
944
-        usleep(is->audio_st && is->show_audio ? rdftspeed*1000 : 5000); //FIXME ideally we should wait the correct time but SDLs event passing is so slow it would be silly
945
-    }
946
-    return 0;
947
-}
948
-
949
-/* get the current audio clock value */
950
-static double get_audio_clock(VideoState *is)
951
-{
952
-    double pts;
953
-    int hw_buf_size, bytes_per_sec;
954
-    pts = is->audio_clock;
955
-    hw_buf_size = audio_write_get_buf_size(is);
956
-    bytes_per_sec = 0;
957
-    if (is->audio_st) {
958
-        bytes_per_sec = is->audio_st->codec->sample_rate *
959
-            2 * is->audio_st->codec->channels;
960
-    }
961
-    if (bytes_per_sec)
962
-        pts -= (double)hw_buf_size / bytes_per_sec;
963
-    return pts;
964
-}
965
-
966
-/* get the current video clock value */
967
-static double get_video_clock(VideoState *is)
968
-{
969
-    if (is->paused) {
970
-        return is->video_current_pts;
971
-    } else {
972
-        return is->video_current_pts_drift + av_gettime() / 1000000.0;
973
-    }
974
-}
975
-
976
-/* get the current external clock value */
977
-static double get_external_clock(VideoState *is)
978
-{
979
-    int64_t ti;
980
-    ti = av_gettime();
981
-    return is->external_clock + ((ti - is->external_clock_time) * 1e-6);
982
-}
983
-
984
-/* get the current master clock value */
985
-static double get_master_clock(VideoState *is)
986
-{
987
-    double val;
988
-
989
-    if (is->av_sync_type == AV_SYNC_VIDEO_MASTER) {
990
-        if (is->video_st)
991
-            val = get_video_clock(is);
992
-        else
993
-            val = get_audio_clock(is);
994
-    } else if (is->av_sync_type == AV_SYNC_AUDIO_MASTER) {
995
-        if (is->audio_st)
996
-            val = get_audio_clock(is);
997
-        else
998
-            val = get_video_clock(is);
999
-    } else {
1000
-        val = get_external_clock(is);
1001
-    }
1002
-    return val;
1003
-}
1004
-
1005
-/* seek in the stream */
1006
-static void stream_seek(VideoState *is, int64_t pos, int64_t rel, int seek_by_bytes)
1007
-{
1008
-    if (!is->seek_req) {
1009
-        is->seek_pos = pos;
1010
-        is->seek_rel = rel;
1011
-        is->seek_flags &= ~AVSEEK_FLAG_BYTE;
1012
-        if (seek_by_bytes)
1013
-            is->seek_flags |= AVSEEK_FLAG_BYTE;
1014
-        is->seek_req = 1;
1015
-    }
1016
-}
1017
-
1018
-/* pause or resume the video */
1019
-static void stream_pause(VideoState *is)
1020
-{
1021
-    if (is->paused) {
1022
-        is->frame_timer += av_gettime() / 1000000.0 + is->video_current_pts_drift - is->video_current_pts;
1023
-        if(is->read_pause_return != AVERROR(ENOSYS)){
1024
-            is->video_current_pts = is->video_current_pts_drift + av_gettime() / 1000000.0;
1025
-        }
1026
-        is->video_current_pts_drift = is->video_current_pts - av_gettime() / 1000000.0;
1027
-    }
1028
-    is->paused = !is->paused;
1029
-}
1030
-
1031
-static double compute_target_time(double frame_current_pts, VideoState *is)
1032
-{
1033
-    double delay, sync_threshold, diff;
1034
-
1035
-    /* compute nominal delay */
1036
-    delay = frame_current_pts - is->frame_last_pts;
1037
-    if (delay <= 0 || delay >= 10.0) {
1038
-        /* if incorrect delay, use previous one */
1039
-        delay = is->frame_last_delay;
1040
-    } else {
1041
-        is->frame_last_delay = delay;
1042
-    }
1043
-    is->frame_last_pts = frame_current_pts;
1044
-
1045
-    /* update delay to follow master synchronisation source */
1046
-    if (((is->av_sync_type == AV_SYNC_AUDIO_MASTER && is->audio_st) ||
1047
-         is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
1048
-        /* if video is slave, we try to correct big delays by
1049
-           duplicating or deleting a frame */
1050
-        diff = get_video_clock(is) - get_master_clock(is);
1051
-
1052
-        /* skip or repeat frame. We take into account the
1053
-           delay to compute the threshold. I still don't know
1054
-           if it is the best guess */
1055
-        sync_threshold = FFMAX(AV_SYNC_THRESHOLD, delay);
1056
-        if (fabs(diff) < AV_NOSYNC_THRESHOLD) {
1057
-            if (diff <= -sync_threshold)
1058
-                delay = 0;
1059
-            else if (diff >= sync_threshold)
1060
-                delay = 2 * delay;
1061
-        }
1062
-    }
1063
-    is->frame_timer += delay;
1064
-
1065
-    av_dlog(NULL, "video: delay=%0.3f pts=%0.3f A-V=%f\n",
1066
-            delay, frame_current_pts, -diff);
1067
-
1068
-    return is->frame_timer;
1069
-}
1070
-
1071
-/* called to display each frame */
1072
-static void video_refresh_timer(void *opaque)
1073
-{
1074
-    VideoState *is = opaque;
1075
-    VideoPicture *vp;
1076
-
1077
-    SubPicture *sp, *sp2;
1078
-
1079
-    if (is->video_st) {
1080
-retry:
1081
-        if (is->pictq_size == 0) {
1082
-            //nothing to do, no picture to display in the que
1083
-        } else {
1084
-            double time= av_gettime()/1000000.0;
1085
-            double next_target;
1086
-            /* dequeue the picture */
1087
-            vp = &is->pictq[is->pictq_rindex];
1088
-
1089
-            if(time < vp->target_clock)
1090
-                return;
1091
-            /* update current video pts */
1092
-            is->video_current_pts = vp->pts;
1093
-            is->video_current_pts_drift = is->video_current_pts - time;
1094
-            is->video_current_pos = vp->pos;
1095
-            if(is->pictq_size > 1){
1096
-                VideoPicture *nextvp= &is->pictq[(is->pictq_rindex+1)%VIDEO_PICTURE_QUEUE_SIZE];
1097
-                assert(nextvp->target_clock >= vp->target_clock);
1098
-                next_target= nextvp->target_clock;
1099
-            }else{
1100
-                next_target= vp->target_clock + is->video_clock - vp->pts; //FIXME pass durations cleanly
1101
-            }
1102
-            if(framedrop && time > next_target){
1103
-                is->skip_frames *= 1.0 + FRAME_SKIP_FACTOR;
1104
-                if(is->pictq_size > 1 || time > next_target + 0.5){
1105
-                    /* update queue size and signal for next picture */
1106
-                    if (++is->pictq_rindex == VIDEO_PICTURE_QUEUE_SIZE)
1107
-                        is->pictq_rindex = 0;
1108
-
1109
-                    SDL_LockMutex(is->pictq_mutex);
1110
-                    is->pictq_size--;
1111
-                    SDL_CondSignal(is->pictq_cond);
1112
-                    SDL_UnlockMutex(is->pictq_mutex);
1113
-                    goto retry;
1114
-                }
1115
-            }
1116
-
1117
-            if(is->subtitle_st) {
1118
-                if (is->subtitle_stream_changed) {
1119
-                    SDL_LockMutex(is->subpq_mutex);
1120
-
1121
-                    while (is->subpq_size) {
1122
-                        free_subpicture(&is->subpq[is->subpq_rindex]);
1123
-
1124
-                        /* update queue size and signal for next picture */
1125
-                        if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
1126
-                            is->subpq_rindex = 0;
1127
-
1128
-                        is->subpq_size--;
1129
-                    }
1130
-                    is->subtitle_stream_changed = 0;
1131
-
1132
-                    SDL_CondSignal(is->subpq_cond);
1133
-                    SDL_UnlockMutex(is->subpq_mutex);
1134
-                } else {
1135
-                    if (is->subpq_size > 0) {
1136
-                        sp = &is->subpq[is->subpq_rindex];
1137
-
1138
-                        if (is->subpq_size > 1)
1139
-                            sp2 = &is->subpq[(is->subpq_rindex + 1) % SUBPICTURE_QUEUE_SIZE];
1140
-                        else
1141
-                            sp2 = NULL;
1142
-
1143
-                        if ((is->video_current_pts > (sp->pts + ((float) sp->sub.end_display_time / 1000)))
1144
-                                || (sp2 && is->video_current_pts > (sp2->pts + ((float) sp2->sub.start_display_time / 1000))))
1145
-                        {
1146
-                            free_subpicture(sp);
1147
-
1148
-                            /* update queue size and signal for next picture */
1149
-                            if (++is->subpq_rindex == SUBPICTURE_QUEUE_SIZE)
1150
-                                is->subpq_rindex = 0;
1151
-
1152
-                            SDL_LockMutex(is->subpq_mutex);
1153
-                            is->subpq_size--;
1154
-                            SDL_CondSignal(is->subpq_cond);
1155
-                            SDL_UnlockMutex(is->subpq_mutex);
1156
-                        }
1157
-                    }
1158
-                }
1159
-            }
1160
-
1161
-            /* display picture */
1162
-            if (!display_disable)
1163
-                video_display(is);
1164
-
1165
-            /* update queue size and signal for next picture */
1166
-            if (++is->pictq_rindex == VIDEO_PICTURE_QUEUE_SIZE)
1167
-                is->pictq_rindex = 0;
1168
-
1169
-            SDL_LockMutex(is->pictq_mutex);
1170
-            is->pictq_size--;
1171
-            SDL_CondSignal(is->pictq_cond);
1172
-            SDL_UnlockMutex(is->pictq_mutex);
1173
-        }
1174
-    } else if (is->audio_st) {
1175
-        /* draw the next audio frame */
1176
-
1177
-        /* if only audio stream, then display the audio bars (better
1178
-           than nothing, just to test the implementation */
1179
-
1180
-        /* display picture */
1181
-        if (!display_disable)
1182
-            video_display(is);
1183
-    }
1184
-    if (show_status) {
1185
-        static int64_t last_time;
1186
-        int64_t cur_time;
1187
-        int aqsize, vqsize, sqsize;
1188
-        double av_diff;
1189
-
1190
-        cur_time = av_gettime();
1191
-        if (!last_time || (cur_time - last_time) >= 30000) {
1192
-            aqsize = 0;
1193
-            vqsize = 0;
1194
-            sqsize = 0;
1195
-            if (is->audio_st)
1196
-                aqsize = is->audioq.size;
1197
-            if (is->video_st)
1198
-                vqsize = is->videoq.size;
1199
-            if (is->subtitle_st)
1200
-                sqsize = is->subtitleq.size;
1201
-            av_diff = 0;
1202
-            if (is->audio_st && is->video_st)
1203
-                av_diff = get_audio_clock(is) - get_video_clock(is);
1204
-            printf("%7.2f A-V:%7.3f s:%3.1f aq=%5dKB vq=%5dKB sq=%5dB f=%"PRId64"/%"PRId64"   \r",
1205
-                   get_master_clock(is), av_diff, FFMAX(is->skip_frames-1, 0), aqsize / 1024, vqsize / 1024, sqsize, is->pts_ctx.num_faulty_dts, is->pts_ctx.num_faulty_pts);
1206
-            fflush(stdout);
1207
-            last_time = cur_time;
1208
-        }
1209
-    }
1210
-}
1211
-
1212
-static void stream_close(VideoState *is)
1213
-{
1214
-    VideoPicture *vp;
1215
-    int i;
1216
-    /* XXX: use a special url_shutdown call to abort parse cleanly */
1217
-    is->abort_request = 1;
1218
-    SDL_WaitThread(is->parse_tid, NULL);
1219
-    SDL_WaitThread(is->refresh_tid, NULL);
1220
-
1221
-    /* free all pictures */
1222
-    for(i=0;i<VIDEO_PICTURE_QUEUE_SIZE; i++) {
1223
-        vp = &is->pictq[i];
1224
-#if CONFIG_AVFILTER
1225
-        if (vp->picref) {
1226
-            avfilter_unref_buffer(vp->picref);
1227
-            vp->picref = NULL;
1228
-        }
1229
-#endif
1230
-        if (vp->bmp) {
1231
-            SDL_FreeYUVOverlay(vp->bmp);
1232
-            vp->bmp = NULL;
1233
-        }
1234
-    }
1235
-    SDL_DestroyMutex(is->pictq_mutex);
1236
-    SDL_DestroyCond(is->pictq_cond);
1237
-    SDL_DestroyMutex(is->subpq_mutex);
1238
-    SDL_DestroyCond(is->subpq_cond);
1239
-#if !CONFIG_AVFILTER
1240
-    if (is->img_convert_ctx)
1241
-        sws_freeContext(is->img_convert_ctx);
1242
-#endif
1243
-    av_free(is);
1244
-}
1245
-
1246
-static void do_exit(void)
1247
-{
1248
-    if (cur_stream) {
1249
-        stream_close(cur_stream);
1250
-        cur_stream = NULL;
1251
-    }
1252
-    uninit_opts();
1253
-#if CONFIG_AVFILTER
1254
-    avfilter_uninit();
1255
-#endif
1256
-    if (show_status)
1257
-        printf("\n");
1258
-    SDL_Quit();
1259
-    av_log(NULL, AV_LOG_QUIET, "");
1260
-    exit(0);
1261
-}
1262
-
1263
-/* allocate a picture (needs to do that in main thread to avoid
1264
-   potential locking problems */
1265
-static void alloc_picture(void *opaque)
1266
-{
1267
-    VideoState *is = opaque;
1268
-    VideoPicture *vp;
1269
-
1270
-    vp = &is->pictq[is->pictq_windex];
1271
-
1272
-    if (vp->bmp)
1273
-        SDL_FreeYUVOverlay(vp->bmp);
1274
-
1275
-#if CONFIG_AVFILTER
1276
-    if (vp->picref)
1277
-        avfilter_unref_buffer(vp->picref);
1278
-    vp->picref = NULL;
1279
-
1280
-    vp->width   = is->out_video_filter->inputs[0]->w;
1281
-    vp->height  = is->out_video_filter->inputs[0]->h;
1282
-    vp->pix_fmt = is->out_video_filter->inputs[0]->format;
1283
-#else
1284
-    vp->width   = is->video_st->codec->width;
1285
-    vp->height  = is->video_st->codec->height;
1286
-    vp->pix_fmt = is->video_st->codec->pix_fmt;
1287
-#endif
1288
-
1289
-    vp->bmp = SDL_CreateYUVOverlay(vp->width, vp->height,
1290
-                                   SDL_YV12_OVERLAY,
1291
-                                   screen);
1292
-    if (!vp->bmp || vp->bmp->pitches[0] < vp->width) {
1293
-        /* SDL allocates a buffer smaller than requested if the video
1294
-         * overlay hardware is unable to support the requested size. */
1295
-        fprintf(stderr, "Error: the video system does not support an image\n"
1296
-                        "size of %dx%d pixels. Try using -lowres or -vf \"scale=w:h\"\n"
1297
-                        "to reduce the image size.\n", vp->width, vp->height );
1298
-        do_exit();
1299
-    }
1300
-
1301
-    SDL_LockMutex(is->pictq_mutex);
1302
-    vp->allocated = 1;
1303
-    SDL_CondSignal(is->pictq_cond);
1304
-    SDL_UnlockMutex(is->pictq_mutex);
1305
-}
1306
-
1307
-/**
1308
- *
1309
- * @param pts the dts of the pkt / pts of the frame and guessed if not known
1310
- */
1311
-static int queue_picture(VideoState *is, AVFrame *src_frame, double pts, int64_t pos)
1312
-{
1313
-    VideoPicture *vp;
1314
-#if CONFIG_AVFILTER
1315
-    AVPicture pict_src;
1316
-#else
1317
-    int dst_pix_fmt = PIX_FMT_YUV420P;
1318
-#endif
1319
-    /* wait until we have space to put a new picture */
1320
-    SDL_LockMutex(is->pictq_mutex);
1321
-
1322
-    if(is->pictq_size>=VIDEO_PICTURE_QUEUE_SIZE && !is->refresh)
1323
-        is->skip_frames= FFMAX(1.0 - FRAME_SKIP_FACTOR, is->skip_frames * (1.0-FRAME_SKIP_FACTOR));
1324
-
1325
-    while (is->pictq_size >= VIDEO_PICTURE_QUEUE_SIZE &&
1326
-           !is->videoq.abort_request) {
1327
-        SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1328
-    }
1329
-    SDL_UnlockMutex(is->pictq_mutex);
1330
-
1331
-    if (is->videoq.abort_request)
1332
-        return -1;
1333
-
1334
-    vp = &is->pictq[is->pictq_windex];
1335
-
1336
-    /* alloc or resize hardware picture buffer */
1337
-    if (!vp->bmp ||
1338
-#if CONFIG_AVFILTER
1339
-        vp->width  != is->out_video_filter->inputs[0]->w ||
1340
-        vp->height != is->out_video_filter->inputs[0]->h) {
1341
-#else
1342
-        vp->width != is->video_st->codec->width ||
1343
-        vp->height != is->video_st->codec->height) {
1344
-#endif
1345
-        SDL_Event event;
1346
-
1347
-        vp->allocated = 0;
1348
-
1349
-        /* the allocation must be done in the main thread to avoid
1350
-           locking problems */
1351
-        event.type = FF_ALLOC_EVENT;
1352
-        event.user.data1 = is;
1353
-        SDL_PushEvent(&event);
1354
-
1355
-        /* wait until the picture is allocated */
1356
-        SDL_LockMutex(is->pictq_mutex);
1357
-        while (!vp->allocated && !is->videoq.abort_request) {
1358
-            SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1359
-        }
1360
-        SDL_UnlockMutex(is->pictq_mutex);
1361
-
1362
-        if (is->videoq.abort_request)
1363
-            return -1;
1364
-    }
1365
-
1366
-    /* if the frame is not skipped, then display it */
1367
-    if (vp->bmp) {
1368
-        AVPicture pict;
1369
-#if CONFIG_AVFILTER
1370
-        if(vp->picref)
1371
-            avfilter_unref_buffer(vp->picref);
1372
-        vp->picref = src_frame->opaque;
1373
-#endif
1374
-
1375
-        /* get a pointer on the bitmap */
1376
-        SDL_LockYUVOverlay (vp->bmp);
1377
-
1378
-        memset(&pict,0,sizeof(AVPicture));
1379
-        pict.data[0] = vp->bmp->pixels[0];
1380
-        pict.data[1] = vp->bmp->pixels[2];
1381
-        pict.data[2] = vp->bmp->pixels[1];
1382
-
1383
-        pict.linesize[0] = vp->bmp->pitches[0];
1384
-        pict.linesize[1] = vp->bmp->pitches[2];
1385
-        pict.linesize[2] = vp->bmp->pitches[1];
1386
-
1387
-#if CONFIG_AVFILTER
1388
-        pict_src.data[0] = src_frame->data[0];
1389
-        pict_src.data[1] = src_frame->data[1];
1390
-        pict_src.data[2] = src_frame->data[2];
1391
-
1392
-        pict_src.linesize[0] = src_frame->linesize[0];
1393
-        pict_src.linesize[1] = src_frame->linesize[1];
1394
-        pict_src.linesize[2] = src_frame->linesize[2];
1395
-
1396
-        //FIXME use direct rendering
1397
-        av_picture_copy(&pict, &pict_src,
1398
-                        vp->pix_fmt, vp->width, vp->height);
1399
-#else
1400
-        sws_flags = av_get_int(sws_opts, "sws_flags", NULL);
1401
-        is->img_convert_ctx = sws_getCachedContext(is->img_convert_ctx,
1402
-            vp->width, vp->height, vp->pix_fmt, vp->width, vp->height,
1403
-            dst_pix_fmt, sws_flags, NULL, NULL, NULL);
1404
-        if (is->img_convert_ctx == NULL) {
1405
-            fprintf(stderr, "Cannot initialize the conversion context\n");
1406
-            exit(1);
1407
-        }
1408
-        sws_scale(is->img_convert_ctx, src_frame->data, src_frame->linesize,
1409
-                  0, vp->height, pict.data, pict.linesize);
1410
-#endif
1411
-        /* update the bitmap content */
1412
-        SDL_UnlockYUVOverlay(vp->bmp);
1413
-
1414
-        vp->pts = pts;
1415
-        vp->pos = pos;
1416
-
1417
-        /* now we can update the picture count */
1418
-        if (++is->pictq_windex == VIDEO_PICTURE_QUEUE_SIZE)
1419
-            is->pictq_windex = 0;
1420
-        SDL_LockMutex(is->pictq_mutex);
1421
-        vp->target_clock= compute_target_time(vp->pts, is);
1422
-
1423
-        is->pictq_size++;
1424
-        SDL_UnlockMutex(is->pictq_mutex);
1425
-    }
1426
-    return 0;
1427
-}
1428
-
1429
-/**
1430
- * compute the exact PTS for the picture if it is omitted in the stream
1431
- * @param pts1 the dts of the pkt / pts of the frame
1432
- */
1433
-static int output_picture2(VideoState *is, AVFrame *src_frame, double pts1, int64_t pos)
1434
-{
1435
-    double frame_delay, pts;
1436
-
1437
-    pts = pts1;
1438
-
1439
-    if (pts != 0) {
1440
-        /* update video clock with pts, if present */
1441
-        is->video_clock = pts;
1442
-    } else {
1443
-        pts = is->video_clock;
1444
-    }
1445
-    /* update video clock for next frame */
1446
-    frame_delay = av_q2d(is->video_st->codec->time_base);
1447
-    /* for MPEG2, the frame can be repeated, so we update the
1448
-       clock accordingly */
1449
-    frame_delay += src_frame->repeat_pict * (frame_delay * 0.5);
1450
-    is->video_clock += frame_delay;
1451
-
1452
-    return queue_picture(is, src_frame, pts, pos);
1453
-}
1454
-
1455
-static int get_video_frame(VideoState *is, AVFrame *frame, int64_t *pts, AVPacket *pkt)
1456
-{
1457
-    int got_picture, i;
1458
-
1459
-    if (packet_queue_get(&is->videoq, pkt, 1) < 0)
1460
-        return -1;
1461
-
1462
-    if (pkt->data == flush_pkt.data) {
1463
-        avcodec_flush_buffers(is->video_st->codec);
1464
-
1465
-        SDL_LockMutex(is->pictq_mutex);
1466
-        //Make sure there are no long delay timers (ideally we should just flush the que but thats harder)
1467
-        for (i = 0; i < VIDEO_PICTURE_QUEUE_SIZE; i++) {
1468
-            is->pictq[i].target_clock= 0;
1469
-        }
1470
-        while (is->pictq_size && !is->videoq.abort_request) {
1471
-            SDL_CondWait(is->pictq_cond, is->pictq_mutex);
1472
-        }
1473
-        is->video_current_pos = -1;
1474
-        SDL_UnlockMutex(is->pictq_mutex);
1475
-
1476
-        init_pts_correction(&is->pts_ctx);
1477
-        is->frame_last_pts = AV_NOPTS_VALUE;
1478
-        is->frame_last_delay = 0;
1479
-        is->frame_timer = (double)av_gettime() / 1000000.0;
1480
-        is->skip_frames = 1;
1481
-        is->skip_frames_index = 0;
1482
-        return 0;
1483
-    }
1484
-
1485
-    avcodec_decode_video2(is->video_st->codec, frame, &got_picture, pkt);
1486
-
1487
-    if (got_picture) {
1488
-        if (decoder_reorder_pts == -1) {
1489
-            *pts = guess_correct_pts(&is->pts_ctx, frame->pkt_pts, frame->pkt_dts);
1490
-        } else if (decoder_reorder_pts) {
1491
-            *pts = frame->pkt_pts;
1492
-        } else {
1493
-            *pts = frame->pkt_dts;
1494
-        }
1495
-
1496
-        if (*pts == AV_NOPTS_VALUE) {
1497
-            *pts = 0;
1498
-        }
1499
-
1500
-        is->skip_frames_index += 1;
1501
-        if(is->skip_frames_index >= is->skip_frames){
1502
-            is->skip_frames_index -= FFMAX(is->skip_frames, 1.0);
1503
-            return 1;
1504
-        }
1505
-
1506
-    }
1507
-    return 0;
1508
-}
1509
-
1510
-#if CONFIG_AVFILTER
1511
-typedef struct {
1512
-    VideoState *is;
1513
-    AVFrame *frame;
1514
-    int use_dr1;
1515
-} FilterPriv;
1516
-
1517
-static int input_get_buffer(AVCodecContext *codec, AVFrame *pic)
1518
-{
1519
-    AVFilterContext *ctx = codec->opaque;
1520
-    AVFilterBufferRef  *ref;
1521
-    int perms = AV_PERM_WRITE;
1522
-    int i, w, h, stride[4];
1523
-    unsigned edge;
1524
-    int pixel_size;
1525
-
1526
-    if (codec->codec->capabilities & CODEC_CAP_NEG_LINESIZES)
1527
-        perms |= AV_PERM_NEG_LINESIZES;
1528
-
1529
-    if(pic->buffer_hints & FF_BUFFER_HINTS_VALID) {
1530
-        if(pic->buffer_hints & FF_BUFFER_HINTS_READABLE) perms |= AV_PERM_READ;
1531
-        if(pic->buffer_hints & FF_BUFFER_HINTS_PRESERVE) perms |= AV_PERM_PRESERVE;
1532
-        if(pic->buffer_hints & FF_BUFFER_HINTS_REUSABLE) perms |= AV_PERM_REUSE2;
1533
-    }
1534
-    if(pic->reference) perms |= AV_PERM_READ | AV_PERM_PRESERVE;
1535
-
1536
-    w = codec->width;
1537
-    h = codec->height;
1538
-    avcodec_align_dimensions2(codec, &w, &h, stride);
1539
-    edge = codec->flags & CODEC_FLAG_EMU_EDGE ? 0 : avcodec_get_edge_width();
1540
-    w += edge << 1;
1541
-    h += edge << 1;
1542
-
1543
-    if(!(ref = avfilter_get_video_buffer(ctx->outputs[0], perms, w, h)))
1544
-        return -1;
1545
-
1546
-    pixel_size = av_pix_fmt_descriptors[ref->format].comp[0].step_minus1+1;
1547
-    ref->video->w = codec->width;
1548
-    ref->video->h = codec->height;
1549
-    for(i = 0; i < 4; i ++) {
1550
-        unsigned hshift = (i == 1 || i == 2) ? av_pix_fmt_descriptors[ref->format].log2_chroma_w : 0;
1551
-        unsigned vshift = (i == 1 || i == 2) ? av_pix_fmt_descriptors[ref->format].log2_chroma_h : 0;
1552
-
1553
-        if (ref->data[i]) {
1554
-            ref->data[i]    += ((edge * pixel_size) >> hshift) + ((edge * ref->linesize[i]) >> vshift);
1555
-        }
1556
-        pic->data[i]     = ref->data[i];
1557
-        pic->linesize[i] = ref->linesize[i];
1558
-    }
1559
-    pic->opaque = ref;
1560
-    pic->age    = INT_MAX;
1561
-    pic->type   = FF_BUFFER_TYPE_USER;
1562
-    pic->reordered_opaque = codec->reordered_opaque;
1563
-    if(codec->pkt) pic->pkt_pts = codec->pkt->pts;
1564
-    else           pic->pkt_pts = AV_NOPTS_VALUE;
1565
-    return 0;
1566
-}
1567
-
1568
-static void input_release_buffer(AVCodecContext *codec, AVFrame *pic)
1569
-{
1570
-    memset(pic->data, 0, sizeof(pic->data));
1571
-    avfilter_unref_buffer(pic->opaque);
1572
-}
1573
-
1574
-static int input_reget_buffer(AVCodecContext *codec, AVFrame *pic)
1575
-{
1576
-    AVFilterBufferRef *ref = pic->opaque;
1577
-
1578
-    if (pic->data[0] == NULL) {
1579
-        pic->buffer_hints |= FF_BUFFER_HINTS_READABLE;
1580
-        return codec->get_buffer(codec, pic);
1581
-    }
1582
-
1583
-    if ((codec->width != ref->video->w) || (codec->height != ref->video->h) ||
1584
-        (codec->pix_fmt != ref->format)) {
1585
-        av_log(codec, AV_LOG_ERROR, "Picture properties changed.\n");
1586
-        return -1;
1587
-    }
1588
-
1589
-    pic->reordered_opaque = codec->reordered_opaque;
1590
-    if(codec->pkt) pic->pkt_pts = codec->pkt->pts;
1591
-    else           pic->pkt_pts = AV_NOPTS_VALUE;
1592
-    return 0;
1593
-}
1594
-
1595
-static int input_init(AVFilterContext *ctx, const char *args, void *opaque)
1596
-{
1597
-    FilterPriv *priv = ctx->priv;
1598
-    AVCodecContext *codec;
1599
-    if(!opaque) return -1;
1600
-
1601
-    priv->is = opaque;
1602
-    codec    = priv->is->video_st->codec;
1603
-    codec->opaque = ctx;
1604
-    if(codec->codec->capabilities & CODEC_CAP_DR1) {
1605
-        priv->use_dr1 = 1;
1606
-        codec->get_buffer     = input_get_buffer;
1607
-        codec->release_buffer = input_release_buffer;
1608
-        codec->reget_buffer   = input_reget_buffer;
1609
-        codec->thread_safe_callbacks = 1;
1610
-    }
1611
-
1612
-    priv->frame = avcodec_alloc_frame();
1613
-
1614
-    return 0;
1615
-}
1616
-
1617
-static void input_uninit(AVFilterContext *ctx)
1618
-{
1619
-    FilterPriv *priv = ctx->priv;
1620
-    av_free(priv->frame);
1621
-}
1622
-
1623
-static int input_request_frame(AVFilterLink *link)
1624
-{
1625
-    FilterPriv *priv = link->src->priv;
1626
-    AVFilterBufferRef *picref;
1627
-    int64_t pts = 0;
1628
-    AVPacket pkt;
1629
-    int ret;
1630
-
1631
-    while (!(ret = get_video_frame(priv->is, priv->frame, &pts, &pkt)))
1632
-        av_free_packet(&pkt);
1633
-    if (ret < 0)
1634
-        return -1;
1635
-
1636
-    if(priv->use_dr1) {
1637
-        picref = avfilter_ref_buffer(priv->frame->opaque, ~0);
1638
-    } else {
1639
-        picref = avfilter_get_video_buffer(link, AV_PERM_WRITE, link->w, link->h);
1640
-        av_image_copy(picref->data, picref->linesize,
1641
-                      priv->frame->data, priv->frame->linesize,
1642
-                      picref->format, link->w, link->h);
1643
-    }
1644
-    av_free_packet(&pkt);
1645
-
1646
-    picref->pts = pts;
1647
-    picref->pos = pkt.pos;
1648
-    picref->video->pixel_aspect = priv->is->video_st->codec->sample_aspect_ratio;
1649
-    avfilter_start_frame(link, picref);
1650
-    avfilter_draw_slice(link, 0, link->h, 1);
1651
-    avfilter_end_frame(link);
1652
-
1653
-    return 0;
1654
-}
1655
-
1656
-static int input_query_formats(AVFilterContext *ctx)
1657
-{
1658
-    FilterPriv *priv = ctx->priv;
1659
-    enum PixelFormat pix_fmts[] = {
1660
-        priv->is->video_st->codec->pix_fmt, PIX_FMT_NONE
1661
-    };
1662
-
1663
-    avfilter_set_common_formats(ctx, avfilter_make_format_list(pix_fmts));
1664
-    return 0;
1665
-}
1666
-
1667
-static int input_config_props(AVFilterLink *link)
1668
-{
1669
-    FilterPriv *priv  = link->src->priv;
1670
-    AVCodecContext *c = priv->is->video_st->codec;
1671
-
1672
-    link->w = c->width;
1673
-    link->h = c->height;
1674
-    link->time_base = priv->is->video_st->time_base;
1675
-
1676
-    return 0;
1677
-}
1678
-
1679
-static AVFilter input_filter =
1680
-{
1681
-    .name      = "ffplay_input",
1682
-
1683
-    .priv_size = sizeof(FilterPriv),
1684
-
1685
-    .init      = input_init,
1686
-    .uninit    = input_uninit,
1687
-
1688
-    .query_formats = input_query_formats,
1689
-
1690
-    .inputs    = (AVFilterPad[]) {{ .name = NULL }},
1691
-    .outputs   = (AVFilterPad[]) {{ .name = "default",
1692
-                                    .type = AVMEDIA_TYPE_VIDEO,
1693
-                                    .request_frame = input_request_frame,
1694
-                                    .config_props  = input_config_props, },
1695
-                                  { .name = NULL }},
1696
-};
1697
-
1698
-static int configure_video_filters(AVFilterGraph *graph, VideoState *is, const char *vfilters)
1699
-{
1700
-    char sws_flags_str[128];
1701
-    int ret;
1702
-    FFSinkContext ffsink_ctx = { .pix_fmt = PIX_FMT_YUV420P };
1703
-    AVFilterContext *filt_src = NULL, *filt_out = NULL;
1704
-    snprintf(sws_flags_str, sizeof(sws_flags_str), "flags=%d", sws_flags);
1705
-    graph->scale_sws_opts = av_strdup(sws_flags_str);
1706
-
1707
-    if ((ret = avfilter_graph_create_filter(&filt_src, &input_filter, "src",
1708
-                                            NULL, is, graph)) < 0)
1709
-        return ret;
1710
-    if ((ret = avfilter_graph_create_filter(&filt_out, &ffsink, "out",
1711
-                                            NULL, &ffsink_ctx, graph)) < 0)
1712
-        return ret;
1713
-
1714
-    if(vfilters) {
1715
-        AVFilterInOut *outputs = av_malloc(sizeof(AVFilterInOut));
1716
-        AVFilterInOut *inputs  = av_malloc(sizeof(AVFilterInOut));
1717
-
1718
-        outputs->name    = av_strdup("in");
1719
-        outputs->filter_ctx = filt_src;
1720
-        outputs->pad_idx = 0;
1721
-        outputs->next    = NULL;
1722
-
1723
-        inputs->name    = av_strdup("out");
1724
-        inputs->filter_ctx = filt_out;
1725
-        inputs->pad_idx = 0;
1726
-        inputs->next    = NULL;
1727
-
1728
-        if ((ret = avfilter_graph_parse(graph, vfilters, inputs, outputs, NULL)) < 0)
1729
-            return ret;
1730
-        av_freep(&vfilters);
1731
-    } else {
1732
-        if ((ret = avfilter_link(filt_src, 0, filt_out, 0)) < 0)
1733
-            return ret;
1734
-    }
1735
-
1736
-    if ((ret = avfilter_graph_config(graph, NULL)) < 0)
1737
-        return ret;
1738
-
1739
-    is->out_video_filter = filt_out;
1740
-
1741
-    return ret;
1742
-}
1743
-
1744
-#endif  /* CONFIG_AVFILTER */
1745
-
1746
-static int video_thread(void *arg)
1747
-{
1748
-    VideoState *is = arg;
1749
-    AVFrame *frame= avcodec_alloc_frame();
1750
-    int64_t pts_int;
1751
-    double pts;
1752
-    int ret;
1753
-
1754
-#if CONFIG_AVFILTER
1755
-    AVFilterGraph *graph = avfilter_graph_alloc();
1756
-    AVFilterContext *filt_out = NULL;
1757
-    int64_t pos;
1758
-
1759
-    if ((ret = configure_video_filters(graph, is, vfilters)) < 0)
1760
-        goto the_end;
1761
-    filt_out = is->out_video_filter;
1762
-#endif
1763
-
1764
-    for(;;) {
1765
-#if !CONFIG_AVFILTER
1766
-        AVPacket pkt;
1767
-#else
1768
-        AVFilterBufferRef *picref;
1769
-        AVRational tb;
1770
-#endif
1771
-        while (is->paused && !is->videoq.abort_request)
1772
-            SDL_Delay(10);
1773
-#if CONFIG_AVFILTER
1774
-        ret = get_filtered_video_frame(filt_out, frame, &picref, &tb);
1775
-        if (picref) {
1776
-            pts_int = picref->pts;
1777
-            pos     = picref->pos;
1778
-            frame->opaque = picref;
1779
-        }
1780
-
1781
-        if (av_cmp_q(tb, is->video_st->time_base)) {
1782
-            av_unused int64_t pts1 = pts_int;
1783
-            pts_int = av_rescale_q(pts_int, tb, is->video_st->time_base);
1784
-            av_dlog(NULL, "video_thread(): "
1785
-                    "tb:%d/%d pts:%"PRId64" -> tb:%d/%d pts:%"PRId64"\n",
1786
-                    tb.num, tb.den, pts1,
1787
-                    is->video_st->time_base.num, is->video_st->time_base.den, pts_int);
1788
-        }
1789
-#else
1790
-        ret = get_video_frame(is, frame, &pts_int, &pkt);
1791
-#endif
1792
-
1793
-        if (ret < 0) goto the_end;
1794
-
1795
-        if (!ret)
1796
-            continue;
1797
-
1798
-        pts = pts_int*av_q2d(is->video_st->time_base);
1799
-
1800
-#if CONFIG_AVFILTER
1801
-        ret = output_picture2(is, frame, pts, pos);
1802
-#else
1803
-        ret = output_picture2(is, frame, pts,  pkt.pos);
1804
-        av_free_packet(&pkt);
1805
-#endif
1806
-        if (ret < 0)
1807
-            goto the_end;
1808
-
1809
-        if (step)
1810
-            if (cur_stream)
1811
-                stream_pause(cur_stream);
1812
-    }
1813
- the_end:
1814
-#if CONFIG_AVFILTER
1815
-    avfilter_graph_free(&graph);
1816
-#endif
1817
-    av_free(frame);
1818
-    return 0;
1819
-}
1820
-
1821
-static int subtitle_thread(void *arg)
1822
-{
1823
-    VideoState *is = arg;
1824
-    SubPicture *sp;
1825
-    AVPacket pkt1, *pkt = &pkt1;
1826
-    int got_subtitle;
1827
-    double pts;
1828
-    int i, j;
1829
-    int r, g, b, y, u, v, a;
1830
-
1831
-    for(;;) {
1832
-        while (is->paused && !is->subtitleq.abort_request) {
1833
-            SDL_Delay(10);
1834
-        }
1835
-        if (packet_queue_get(&is->subtitleq, pkt, 1) < 0)
1836
-            break;
1837
-
1838
-        if(pkt->data == flush_pkt.data){
1839
-            avcodec_flush_buffers(is->subtitle_st->codec);
1840
-            continue;
1841
-        }
1842
-        SDL_LockMutex(is->subpq_mutex);
1843
-        while (is->subpq_size >= SUBPICTURE_QUEUE_SIZE &&
1844
-               !is->subtitleq.abort_request) {
1845
-            SDL_CondWait(is->subpq_cond, is->subpq_mutex);
1846
-        }
1847
-        SDL_UnlockMutex(is->subpq_mutex);
1848
-
1849
-        if (is->subtitleq.abort_request)
1850
-            return 0;
1851
-
1852
-        sp = &is->subpq[is->subpq_windex];
1853
-
1854
-       /* NOTE: ipts is the PTS of the _first_ picture beginning in
1855
-           this packet, if any */
1856
-        pts = 0;
1857
-        if (pkt->pts != AV_NOPTS_VALUE)
1858
-            pts = av_q2d(is->subtitle_st->time_base)*pkt->pts;
1859
-
1860
-        avcodec_decode_subtitle2(is->subtitle_st->codec, &sp->sub,
1861
-                                 &got_subtitle, pkt);
1862
-
1863
-        if (got_subtitle && sp->sub.format == 0) {
1864
-            sp->pts = pts;
1865
-
1866
-            for (i = 0; i < sp->sub.num_rects; i++)
1867
-            {
1868
-                for (j = 0; j < sp->sub.rects[i]->nb_colors; j++)
1869
-                {
1870
-                    RGBA_IN(r, g, b, a, (uint32_t*)sp->sub.rects[i]->pict.data[1] + j);
1871
-                    y = RGB_TO_Y_CCIR(r, g, b);
1872
-                    u = RGB_TO_U_CCIR(r, g, b, 0);
1873
-                    v = RGB_TO_V_CCIR(r, g, b, 0);
1874
-                    YUVA_OUT((uint32_t*)sp->sub.rects[i]->pict.data[1] + j, y, u, v, a);
1875
-                }
1876
-            }
1877
-
1878
-            /* now we can update the picture count */
1879
-            if (++is->subpq_windex == SUBPICTURE_QUEUE_SIZE)
1880
-                is->subpq_windex = 0;
1881
-            SDL_LockMutex(is->subpq_mutex);
1882
-            is->subpq_size++;
1883
-            SDL_UnlockMutex(is->subpq_mutex);
1884
-        }
1885
-        av_free_packet(pkt);
1886
-    }
1887
-    return 0;
1888
-}
1889
-
1890
-/* copy samples for viewing in editor window */
1891
-static void update_sample_display(VideoState *is, short *samples, int samples_size)
1892
-{
1893
-    int size, len;
1894
-
1895
-    size = samples_size / sizeof(short);
1896
-    while (size > 0) {
1897
-        len = SAMPLE_ARRAY_SIZE - is->sample_array_index;
1898
-        if (len > size)
1899
-            len = size;
1900
-        memcpy(is->sample_array + is->sample_array_index, samples, len * sizeof(short));
1901
-        samples += len;
1902
-        is->sample_array_index += len;
1903
-        if (is->sample_array_index >= SAMPLE_ARRAY_SIZE)
1904
-            is->sample_array_index = 0;
1905
-        size -= len;
1906
-    }
1907
-}
1908
-
1909
-/* return the new audio buffer size (samples can be added or deleted
1910
-   to get better sync if video or external master clock) */
1911
-static int synchronize_audio(VideoState *is, short *samples,
1912
-                             int samples_size1, double pts)
1913
-{
1914
-    int n, samples_size;
1915
-    double ref_clock;
1916
-
1917
-    n = 2 * is->audio_st->codec->channels;
1918
-    samples_size = samples_size1;
1919
-
1920
-    /* if not master, then we try to remove or add samples to correct the clock */
1921
-    if (((is->av_sync_type == AV_SYNC_VIDEO_MASTER && is->video_st) ||
1922
-         is->av_sync_type == AV_SYNC_EXTERNAL_CLOCK)) {
1923
-        double diff, avg_diff;
1924
-        int wanted_size, min_size, max_size, nb_samples;
1925
-
1926
-        ref_clock = get_master_clock(is);
1927
-        diff = get_audio_clock(is) - ref_clock;
1928
-
1929
-        if (diff < AV_NOSYNC_THRESHOLD) {
1930
-            is->audio_diff_cum = diff + is->audio_diff_avg_coef * is->audio_diff_cum;
1931
-            if (is->audio_diff_avg_count < AUDIO_DIFF_AVG_NB) {
1932
-                /* not enough measures to have a correct estimate */
1933
-                is->audio_diff_avg_count++;
1934
-            } else {
1935
-                /* estimate the A-V difference */
1936
-                avg_diff = is->audio_diff_cum * (1.0 - is->audio_diff_avg_coef);
1937
-
1938
-                if (fabs(avg_diff) >= is->audio_diff_threshold) {
1939
-                    wanted_size = samples_size + ((int)(diff * is->audio_st->codec->sample_rate) * n);
1940
-                    nb_samples = samples_size / n;
1941
-
1942
-                    min_size = ((nb_samples * (100 - SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
1943
-                    max_size = ((nb_samples * (100 + SAMPLE_CORRECTION_PERCENT_MAX)) / 100) * n;
1944
-                    if (wanted_size < min_size)
1945
-                        wanted_size = min_size;
1946
-                    else if (wanted_size > max_size)
1947
-                        wanted_size = max_size;
1948
-
1949
-                    /* add or remove samples to correction the synchro */
1950
-                    if (wanted_size < samples_size) {
1951
-                        /* remove samples */
1952
-                        samples_size = wanted_size;
1953
-                    } else if (wanted_size > samples_size) {
1954
-                        uint8_t *samples_end, *q;
1955
-                        int nb;
1956
-
1957
-                        /* add samples */
1958
-                        nb = (samples_size - wanted_size);
1959
-                        samples_end = (uint8_t *)samples + samples_size - n;
1960
-                        q = samples_end + n;
1961
-                        while (nb > 0) {
1962
-                            memcpy(q, samples_end, n);
1963
-                            q += n;
1964
-                            nb -= n;
1965
-                        }
1966
-                        samples_size = wanted_size;
1967
-                    }
1968
-                }
1969
-                av_dlog(NULL, "diff=%f adiff=%f sample_diff=%d apts=%0.3f vpts=%0.3f %f\n",
1970
-                        diff, avg_diff, samples_size - samples_size1,
1971
-                        is->audio_clock, is->video_clock, is->audio_diff_threshold);
1972
-            }
1973
-        } else {
1974
-            /* too big difference : may be initial PTS errors, so
1975
-               reset A-V filter */
1976
-            is->audio_diff_avg_count = 0;
1977
-            is->audio_diff_cum = 0;
1978
-        }
1979
-    }
1980
-
1981
-    return samples_size;
1982
-}
1983
-
1984
-/* decode one audio frame and returns its uncompressed size */
1985
-static int audio_decode_frame(VideoState *is, double *pts_ptr)
1986
-{
1987
-    AVPacket *pkt_temp = &is->audio_pkt_temp;
1988
-    AVPacket *pkt = &is->audio_pkt;
1989
-    AVCodecContext *dec= is->audio_st->codec;
1990
-    int n, len1, data_size;
1991
-    double pts;
1992
-
1993
-    for(;;) {
1994
-        /* NOTE: the audio packet can contain several frames */
1995
-        while (pkt_temp->size > 0) {
1996
-            data_size = sizeof(is->audio_buf1);
1997
-            len1 = avcodec_decode_audio3(dec,
1998
-                                        (int16_t *)is->audio_buf1, &data_size,
1999
-                                        pkt_temp);
2000
-            if (len1 < 0) {
2001
-                /* if error, we skip the frame */
2002
-                pkt_temp->size = 0;
2003
-                break;
2004
-            }
2005
-
2006
-            pkt_temp->data += len1;
2007
-            pkt_temp->size -= len1;
2008
-            if (data_size <= 0)
2009
-                continue;
2010
-
2011
-            if (dec->sample_fmt != is->audio_src_fmt) {
2012
-                if (is->reformat_ctx)
2013
-                    av_audio_convert_free(is->reformat_ctx);
2014
-                is->reformat_ctx= av_audio_convert_alloc(AV_SAMPLE_FMT_S16, 1,
2015
-                                                         dec->sample_fmt, 1, NULL, 0);
2016
-                if (!is->reformat_ctx) {
2017
-                    fprintf(stderr, "Cannot convert %s sample format to %s sample format\n",
2018
-                        av_get_sample_fmt_name(dec->sample_fmt),
2019
-                        av_get_sample_fmt_name(AV_SAMPLE_FMT_S16));
2020
-                        break;
2021
-                }
2022
-                is->audio_src_fmt= dec->sample_fmt;
2023
-            }
2024
-
2025
-            if (is->reformat_ctx) {
2026
-                const void *ibuf[6]= {is->audio_buf1};
2027
-                void *obuf[6]= {is->audio_buf2};
2028
-                int istride[6]= {av_get_bytes_per_sample(dec->sample_fmt)};
2029
-                int ostride[6]= {2};
2030
-                int len= data_size/istride[0];
2031
-                if (av_audio_convert(is->reformat_ctx, obuf, ostride, ibuf, istride, len)<0) {
2032
-                    printf("av_audio_convert() failed\n");
2033
-                    break;
2034
-                }
2035
-                is->audio_buf= is->audio_buf2;
2036
-                /* FIXME: existing code assume that data_size equals framesize*channels*2
2037
-                          remove this legacy cruft */
2038
-                data_size= len*2;
2039
-            }else{
2040
-                is->audio_buf= is->audio_buf1;
2041
-            }
2042
-
2043
-            /* if no pts, then compute it */
2044
-            pts = is->audio_clock;
2045
-            *pts_ptr = pts;
2046
-            n = 2 * dec->channels;
2047
-            is->audio_clock += (double)data_size /
2048
-                (double)(n * dec->sample_rate);
2049
-#ifdef DEBUG
2050
-            {
2051
-                static double last_clock;
2052
-                printf("audio: delay=%0.3f clock=%0.3f pts=%0.3f\n",
2053
-                       is->audio_clock - last_clock,
2054
-                       is->audio_clock, pts);
2055
-                last_clock = is->audio_clock;
2056
-            }
2057
-#endif
2058
-            return data_size;
2059
-        }
2060
-
2061
-        /* free the current packet */
2062
-        if (pkt->data)
2063
-            av_free_packet(pkt);
2064
-
2065
-        if (is->paused || is->audioq.abort_request) {
2066
-            return -1;
2067
-        }
2068
-
2069
-        /* read next packet */
2070
-        if (packet_queue_get(&is->audioq, pkt, 1) < 0)
2071
-            return -1;
2072
-        if(pkt->data == flush_pkt.data){
2073
-            avcodec_flush_buffers(dec);
2074
-            continue;
2075
-        }
2076
-
2077
-        pkt_temp->data = pkt->data;
2078
-        pkt_temp->size = pkt->size;
2079
-
2080
-        /* if update the audio clock with the pts */
2081
-        if (pkt->pts != AV_NOPTS_VALUE) {
2082
-            is->audio_clock = av_q2d(is->audio_st->time_base)*pkt->pts;
2083
-        }
2084
-    }
2085
-}
2086
-
2087
-/* prepare a new audio buffer */
2088
-static void sdl_audio_callback(void *opaque, Uint8 *stream, int len)
2089
-{
2090
-    VideoState *is = opaque;
2091
-    int audio_size, len1;
2092
-    double pts;
2093
-
2094
-    audio_callback_time = av_gettime();
2095
-
2096
-    while (len > 0) {
2097
-        if (is->audio_buf_index >= is->audio_buf_size) {
2098
-           audio_size = audio_decode_frame(is, &pts);
2099
-           if (audio_size < 0) {
2100
-                /* if error, just output silence */
2101
-               is->audio_buf = is->audio_buf1;
2102
-               is->audio_buf_size = 1024;
2103
-               memset(is->audio_buf, 0, is->audio_buf_size);
2104
-           } else {
2105
-               if (is->show_audio)
2106
-                   update_sample_display(is, (int16_t *)is->audio_buf, audio_size);
2107
-               audio_size = synchronize_audio(is, (int16_t *)is->audio_buf, audio_size,
2108
-                                              pts);
2109
-               is->audio_buf_size = audio_size;
2110
-           }
2111
-           is->audio_buf_index = 0;
2112
-        }
2113
-        len1 = is->audio_buf_size - is->audio_buf_index;
2114
-        if (len1 > len)
2115
-            len1 = len;
2116
-        memcpy(stream, (uint8_t *)is->audio_buf + is->audio_buf_index, len1);
2117
-        len -= len1;
2118
-        stream += len1;
2119
-        is->audio_buf_index += len1;
2120
-    }
2121
-}
2122
-
2123
-/* open a given stream. Return 0 if OK */
2124
-static int stream_component_open(VideoState *is, int stream_index)
2125
-{
2126
-    AVFormatContext *ic = is->ic;
2127
-    AVCodecContext *avctx;
2128
-    AVCodec *codec;
2129
-    SDL_AudioSpec wanted_spec, spec;
2130
-    AVDictionary *opts;
2131
-    AVDictionaryEntry *t = NULL;
2132
-
2133
-    if (stream_index < 0 || stream_index >= ic->nb_streams)
2134
-        return -1;
2135
-    avctx = ic->streams[stream_index]->codec;
2136
-
2137
-    opts = filter_codec_opts(codec_opts, avctx->codec_id, 0);
2138
-
2139
-    /* prepare audio output */
2140
-    if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
2141
-        if (avctx->channels > 0) {
2142
-            avctx->request_channels = FFMIN(2, avctx->channels);
2143
-        } else {
2144
-            avctx->request_channels = 2;
2145
-        }
2146
-    }
2147
-
2148
-    codec = avcodec_find_decoder(avctx->codec_id);
2149
-    avctx->debug_mv = debug_mv;
2150
-    avctx->debug = debug;
2151
-    avctx->workaround_bugs = workaround_bugs;
2152
-    avctx->lowres = lowres;
2153
-    if(lowres) avctx->flags |= CODEC_FLAG_EMU_EDGE;
2154
-    avctx->idct_algo= idct;
2155
-    if(fast) avctx->flags2 |= CODEC_FLAG2_FAST;
2156
-    avctx->skip_frame= skip_frame;
2157
-    avctx->skip_idct= skip_idct;
2158
-    avctx->skip_loop_filter= skip_loop_filter;
2159
-    avctx->error_recognition= error_recognition;
2160
-    avctx->error_concealment= error_concealment;
2161
-    avctx->thread_count= thread_count;
2162
-
2163
-    if (!codec ||
2164
-        avcodec_open2(avctx, codec, &opts) < 0)
2165
-        return -1;
2166
-    if ((t = av_dict_get(opts, "", NULL, AV_DICT_IGNORE_SUFFIX))) {
2167
-        av_log(NULL, AV_LOG_ERROR, "Option %s not found.\n", t->key);
2168
-        return AVERROR_OPTION_NOT_FOUND;
2169
-    }
2170
-
2171
-    /* prepare audio output */
2172
-    if (avctx->codec_type == AVMEDIA_TYPE_AUDIO) {
2173
-        wanted_spec.freq = avctx->sample_rate;
2174
-        wanted_spec.format = AUDIO_S16SYS;
2175
-        wanted_spec.channels = avctx->channels;
2176
-        wanted_spec.silence = 0;
2177
-        wanted_spec.samples = SDL_AUDIO_BUFFER_SIZE;
2178
-        wanted_spec.callback = sdl_audio_callback;
2179
-        wanted_spec.userdata = is;
2180
-        if (SDL_OpenAudio(&wanted_spec, &spec) < 0) {
2181
-            fprintf(stderr, "SDL_OpenAudio: %s\n", SDL_GetError());
2182
-            return -1;
2183
-        }
2184
-        is->audio_hw_buf_size = spec.size;
2185
-        is->audio_src_fmt= AV_SAMPLE_FMT_S16;
2186
-    }
2187
-
2188
-    ic->streams[stream_index]->discard = AVDISCARD_DEFAULT;
2189
-    switch(avctx->codec_type) {
2190
-    case AVMEDIA_TYPE_AUDIO:
2191
-        is->audio_stream = stream_index;
2192
-        is->audio_st = ic->streams[stream_index];
2193
-        is->audio_buf_size = 0;
2194
-        is->audio_buf_index = 0;
2195
-
2196
-        /* init averaging filter */
2197
-        is->audio_diff_avg_coef = exp(log(0.01) / AUDIO_DIFF_AVG_NB);
2198
-        is->audio_diff_avg_count = 0;
2199
-        /* since we do not have a precise anough audio fifo fullness,
2200
-           we correct audio sync only if larger than this threshold */
2201
-        is->audio_diff_threshold = 2.0 * SDL_AUDIO_BUFFER_SIZE / avctx->sample_rate;
2202
-
2203
-        memset(&is->audio_pkt, 0, sizeof(is->audio_pkt));
2204
-        packet_queue_init(&is->audioq);
2205
-        SDL_PauseAudio(0);
2206
-        break;
2207
-    case AVMEDIA_TYPE_VIDEO:
2208
-        is->video_stream = stream_index;
2209
-        is->video_st = ic->streams[stream_index];
2210
-
2211
-        packet_queue_init(&is->videoq);
2212
-        is->video_tid = SDL_CreateThread(video_thread, is);
2213
-        break;
2214
-    case AVMEDIA_TYPE_SUBTITLE:
2215
-        is->subtitle_stream = stream_index;
2216
-        is->subtitle_st = ic->streams[stream_index];
2217
-        packet_queue_init(&is->subtitleq);
2218
-
2219
-        is->subtitle_tid = SDL_CreateThread(subtitle_thread, is);
2220
-        break;
2221
-    default:
2222
-        break;
2223
-    }
2224
-    return 0;
2225
-}
2226
-
2227
-static void stream_component_close(VideoState *is, int stream_index)
2228
-{
2229
-    AVFormatContext *ic = is->ic;
2230
-    AVCodecContext *avctx;
2231
-
2232
-    if (stream_index < 0 || stream_index >= ic->nb_streams)
2233
-        return;
2234
-    avctx = ic->streams[stream_index]->codec;
2235
-
2236
-    switch(avctx->codec_type) {
2237
-    case AVMEDIA_TYPE_AUDIO:
2238
-        packet_queue_abort(&is->audioq);
2239
-
2240
-        SDL_CloseAudio();
2241
-
2242
-        packet_queue_end(&is->audioq);
2243
-        if (is->reformat_ctx)
2244
-            av_audio_convert_free(is->reformat_ctx);
2245
-        is->reformat_ctx = NULL;
2246
-        break;
2247
-    case AVMEDIA_TYPE_VIDEO:
2248
-        packet_queue_abort(&is->videoq);
2249
-
2250
-        /* note: we also signal this mutex to make sure we deblock the
2251
-           video thread in all cases */
2252
-        SDL_LockMutex(is->pictq_mutex);
2253
-        SDL_CondSignal(is->pictq_cond);
2254
-        SDL_UnlockMutex(is->pictq_mutex);
2255
-
2256
-        SDL_WaitThread(is->video_tid, NULL);
2257
-
2258
-        packet_queue_end(&is->videoq);
2259
-        break;
2260
-    case AVMEDIA_TYPE_SUBTITLE:
2261
-        packet_queue_abort(&is->subtitleq);
2262
-
2263
-        /* note: we also signal this mutex to make sure we deblock the
2264
-           video thread in all cases */
2265
-        SDL_LockMutex(is->subpq_mutex);
2266
-        is->subtitle_stream_changed = 1;
2267
-
2268
-        SDL_CondSignal(is->subpq_cond);
2269
-        SDL_UnlockMutex(is->subpq_mutex);
2270
-
2271
-        SDL_WaitThread(is->subtitle_tid, NULL);
2272
-
2273
-        packet_queue_end(&is->subtitleq);
2274
-        break;
2275
-    default:
2276
-        break;
2277
-    }
2278
-
2279
-    ic->streams[stream_index]->discard = AVDISCARD_ALL;
2280
-    avcodec_close(avctx);
2281
-    switch(avctx->codec_type) {
2282
-    case AVMEDIA_TYPE_AUDIO:
2283
-        is->audio_st = NULL;
2284
-        is->audio_stream = -1;
2285
-        break;
2286
-    case AVMEDIA_TYPE_VIDEO:
2287
-        is->video_st = NULL;
2288
-        is->video_stream = -1;
2289
-        break;
2290
-    case AVMEDIA_TYPE_SUBTITLE:
2291
-        is->subtitle_st = NULL;
2292
-        is->subtitle_stream = -1;
2293
-        break;
2294
-    default:
2295
-        break;
2296
-    }
2297
-}
2298
-
2299
-/* since we have only one decoding thread, we can use a global
2300
-   variable instead of a thread local variable */
2301
-static VideoState *global_video_state;
2302
-
2303
-static int decode_interrupt_cb(void)
2304
-{
2305
-    return (global_video_state && global_video_state->abort_request);
2306
-}
2307
-
2308
-/* this thread gets the stream from the disk or the network */
2309
-static int decode_thread(void *arg)
2310
-{
2311
-    VideoState *is = arg;
2312
-    AVFormatContext *ic = NULL;
2313
-    int err, i, ret;
2314
-    int st_index[AVMEDIA_TYPE_NB];
2315
-    AVPacket pkt1, *pkt = &pkt1;
2316
-    int eof=0;
2317
-    int pkt_in_play_range = 0;
2318
-    AVDictionaryEntry *t;
2319
-    AVDictionary **opts;
2320
-    int orig_nb_streams;
2321
-
2322
-    memset(st_index, -1, sizeof(st_index));
2323
-    is->video_stream = -1;
2324
-    is->audio_stream = -1;
2325
-    is->subtitle_stream = -1;
2326
-
2327
-    global_video_state = is;
2328
-    avio_set_interrupt_cb(decode_interrupt_cb);
2329
-
2330
-    err = avformat_open_input(&ic, is->filename, is->iformat, &format_opts);
2331
-    if (err < 0) {
2332
-        print_error(is->filename, err);
2333
-        ret = -1;
2334
-        goto fail;
2335
-    }
2336
-    if ((t = av_dict_get(format_opts, "", NULL, AV_DICT_IGNORE_SUFFIX))) {
2337
-        av_log(NULL, AV_LOG_ERROR, "Option %s not found.\n", t->key);
2338
-        ret = AVERROR_OPTION_NOT_FOUND;
2339
-        goto fail;
2340
-    }
2341
-    is->ic = ic;
2342
-
2343
-    if(genpts)
2344
-        ic->flags |= AVFMT_FLAG_GENPTS;
2345
-
2346
-    opts = setup_find_stream_info_opts(ic, codec_opts);
2347
-    orig_nb_streams = ic->nb_streams;
2348
-
2349
-    err = avformat_find_stream_info(ic, opts);
2350
-    if (err < 0) {
2351
-        fprintf(stderr, "%s: could not find codec parameters\n", is->filename);
2352
-        ret = -1;
2353
-        goto fail;
2354
-    }
2355
-    for (i = 0; i < orig_nb_streams; i++)
2356
-        av_dict_free(&opts[i]);
2357
-    av_freep(&opts);
2358
-
2359
-    if(ic->pb)
2360
-        ic->pb->eof_reached= 0; //FIXME hack, ffplay maybe should not use url_feof() to test for the end
2361
-
2362
-    if(seek_by_bytes<0)
2363
-        seek_by_bytes= !!(ic->iformat->flags & AVFMT_TS_DISCONT);
2364
-
2365
-    /* if seeking requested, we execute it */
2366
-    if (start_time != AV_NOPTS_VALUE) {
2367
-        int64_t timestamp;
2368
-
2369
-        timestamp = start_time;
2370
-        /* add the stream start time */
2371
-        if (ic->start_time != AV_NOPTS_VALUE)
2372
-            timestamp += ic->start_time;
2373
-        ret = avformat_seek_file(ic, -1, INT64_MIN, timestamp, INT64_MAX, 0);
2374
-        if (ret < 0) {
2375
-            fprintf(stderr, "%s: could not seek to position %0.3f\n",
2376
-                    is->filename, (double)timestamp / AV_TIME_BASE);
2377
-        }
2378
-    }
2379
-
2380
-    for (i = 0; i < ic->nb_streams; i++)
2381
-        ic->streams[i]->discard = AVDISCARD_ALL;
2382
-    if (!video_disable)
2383
-        st_index[AVMEDIA_TYPE_VIDEO] =
2384
-            av_find_best_stream(ic, AVMEDIA_TYPE_VIDEO,
2385
-                                wanted_stream[AVMEDIA_TYPE_VIDEO], -1, NULL, 0);
2386
-    if (!audio_disable)
2387
-        st_index[AVMEDIA_TYPE_AUDIO] =
2388
-            av_find_best_stream(ic, AVMEDIA_TYPE_AUDIO,
2389
-                                wanted_stream[AVMEDIA_TYPE_AUDIO],
2390
-                                st_index[AVMEDIA_TYPE_VIDEO],
2391
-                                NULL, 0);
2392
-    if (!video_disable)
2393
-        st_index[AVMEDIA_TYPE_SUBTITLE] =
2394
-            av_find_best_stream(ic, AVMEDIA_TYPE_SUBTITLE,
2395
-                                wanted_stream[AVMEDIA_TYPE_SUBTITLE],
2396
-                                (st_index[AVMEDIA_TYPE_AUDIO] >= 0 ?
2397
-                                 st_index[AVMEDIA_TYPE_AUDIO] :
2398
-                                 st_index[AVMEDIA_TYPE_VIDEO]),
2399
-                                NULL, 0);
2400
-    if (show_status) {
2401
-        av_dump_format(ic, 0, is->filename, 0);
2402
-    }
2403
-
2404
-    /* open the streams */
2405
-    if (st_index[AVMEDIA_TYPE_AUDIO] >= 0) {
2406
-        stream_component_open(is, st_index[AVMEDIA_TYPE_AUDIO]);
2407
-    }
2408
-
2409
-    ret=-1;
2410
-    if (st_index[AVMEDIA_TYPE_VIDEO] >= 0) {
2411
-        ret= stream_component_open(is, st_index[AVMEDIA_TYPE_VIDEO]);
2412
-    }
2413
-    is->refresh_tid = SDL_CreateThread(refresh_thread, is);
2414
-    if(ret<0) {
2415
-        if (!display_disable)
2416
-            is->show_audio = 2;
2417
-    }
2418
-
2419
-    if (st_index[AVMEDIA_TYPE_SUBTITLE] >= 0) {
2420
-        stream_component_open(is, st_index[AVMEDIA_TYPE_SUBTITLE]);
2421
-    }
2422
-
2423
-    if (is->video_stream < 0 && is->audio_stream < 0) {
2424
-        fprintf(stderr, "%s: could not open codecs\n", is->filename);
2425
-        ret = -1;
2426
-        goto fail;
2427
-    }
2428
-
2429
-    for(;;) {
2430
-        if (is->abort_request)
2431
-            break;
2432
-        if (is->paused != is->last_paused) {
2433
-            is->last_paused = is->paused;
2434
-            if (is->paused)
2435
-                is->read_pause_return= av_read_pause(ic);
2436
-            else
2437
-                av_read_play(ic);
2438
-        }
2439
-#if CONFIG_RTSP_DEMUXER
2440
-        if (is->paused && !strcmp(ic->iformat->name, "rtsp")) {
2441
-            /* wait 10 ms to avoid trying to get another packet */
2442
-            /* XXX: horrible */
2443
-            SDL_Delay(10);
2444
-            continue;
2445
-        }
2446
-#endif
2447
-        if (is->seek_req) {
2448
-            int64_t seek_target= is->seek_pos;
2449
-            int64_t seek_min= is->seek_rel > 0 ? seek_target - is->seek_rel + 2: INT64_MIN;
2450
-            int64_t seek_max= is->seek_rel < 0 ? seek_target - is->seek_rel - 2: INT64_MAX;
2451
-//FIXME the +-2 is due to rounding being not done in the correct direction in generation
2452
-//      of the seek_pos/seek_rel variables
2453
-
2454
-            ret = avformat_seek_file(is->ic, -1, seek_min, seek_target, seek_max, is->seek_flags);
2455
-            if (ret < 0) {
2456
-                fprintf(stderr, "%s: error while seeking\n", is->ic->filename);
2457
-            }else{
2458
-                if (is->audio_stream >= 0) {
2459
-                    packet_queue_flush(&is->audioq);
2460
-                    packet_queue_put(&is->audioq, &flush_pkt);
2461
-                }
2462
-                if (is->subtitle_stream >= 0) {
2463
-                    packet_queue_flush(&is->subtitleq);
2464
-                    packet_queue_put(&is->subtitleq, &flush_pkt);
2465
-                }
2466
-                if (is->video_stream >= 0) {
2467
-                    packet_queue_flush(&is->videoq);
2468
-                    packet_queue_put(&is->videoq, &flush_pkt);
2469
-                }
2470
-            }
2471
-            is->seek_req = 0;
2472
-            eof= 0;
2473
-        }
2474
-
2475
-        /* if the queue are full, no need to read more */
2476
-        if (   is->audioq.size + is->videoq.size + is->subtitleq.size > MAX_QUEUE_SIZE
2477
-            || (   (is->audioq   .size  > MIN_AUDIOQ_SIZE || is->audio_stream<0)
2478
-                && (is->videoq   .nb_packets > MIN_FRAMES || is->video_stream<0)
2479
-                && (is->subtitleq.nb_packets > MIN_FRAMES || is->subtitle_stream<0))) {
2480
-            /* wait 10 ms */
2481
-            SDL_Delay(10);
2482
-            continue;
2483
-        }
2484
-        if(eof) {
2485
-            if(is->video_stream >= 0){
2486
-                av_init_packet(pkt);
2487
-                pkt->data=NULL;
2488
-                pkt->size=0;
2489
-                pkt->stream_index= is->video_stream;
2490
-                packet_queue_put(&is->videoq, pkt);
2491
-            }
2492
-            SDL_Delay(10);
2493
-            if(is->audioq.size + is->videoq.size + is->subtitleq.size ==0){
2494
-                if(loop!=1 && (!loop || --loop)){
2495
-                    stream_seek(cur_stream, start_time != AV_NOPTS_VALUE ? start_time : 0, 0, 0);
2496
-                }else if(autoexit){
2497
-                    ret=AVERROR_EOF;
2498
-                    goto fail;
2499
-                }
2500
-            }
2501
-            continue;
2502
-        }
2503
-        ret = av_read_frame(ic, pkt);
2504
-        if (ret < 0) {
2505
-            if (ret == AVERROR_EOF || (ic->pb && ic->pb->eof_reached))
2506
-                eof=1;
2507
-            if (ic->pb && ic->pb->error)
2508
-                break;
2509
-            SDL_Delay(100); /* wait for user event */
2510
-            continue;
2511
-        }
2512
-        /* check if packet is in play range specified by user, then queue, otherwise discard */
2513
-        pkt_in_play_range = duration == AV_NOPTS_VALUE ||
2514
-                (pkt->pts - ic->streams[pkt->stream_index]->start_time) *
2515
-                av_q2d(ic->streams[pkt->stream_index]->time_base) -
2516
-                (double)(start_time != AV_NOPTS_VALUE ? start_time : 0)/1000000
2517
-                <= ((double)duration/1000000);
2518
-        if (pkt->stream_index == is->audio_stream && pkt_in_play_range) {
2519
-            packet_queue_put(&is->audioq, pkt);
2520
-        } else if (pkt->stream_index == is->video_stream && pkt_in_play_range) {
2521
-            packet_queue_put(&is->videoq, pkt);
2522
-        } else if (pkt->stream_index == is->subtitle_stream && pkt_in_play_range) {
2523
-            packet_queue_put(&is->subtitleq, pkt);
2524
-        } else {
2525
-            av_free_packet(pkt);
2526
-        }
2527
-    }
2528
-    /* wait until the end */
2529
-    while (!is->abort_request) {
2530
-        SDL_Delay(100);
2531
-    }
2532
-
2533
-    ret = 0;
2534
- fail:
2535
-    /* disable interrupting */
2536
-    global_video_state = NULL;
2537
-
2538
-    /* close each stream */
2539
-    if (is->audio_stream >= 0)
2540
-        stream_component_close(is, is->audio_stream);
2541
-    if (is->video_stream >= 0)
2542
-        stream_component_close(is, is->video_stream);
2543
-    if (is->subtitle_stream >= 0)
2544
-        stream_component_close(is, is->subtitle_stream);
2545
-    if (is->ic) {
2546
-        av_close_input_file(is->ic);
2547
-        is->ic = NULL; /* safety */
2548
-    }
2549
-    avio_set_interrupt_cb(NULL);
2550
-
2551
-    if (ret != 0) {
2552
-        SDL_Event event;
2553
-
2554
-        event.type = FF_QUIT_EVENT;
2555
-        event.user.data1 = is;
2556
-        SDL_PushEvent(&event);
2557
-    }
2558
-    return 0;
2559
-}
2560
-
2561
-static VideoState *stream_open(const char *filename, AVInputFormat *iformat)
2562
-{
2563
-    VideoState *is;
2564
-
2565
-    is = av_mallocz(sizeof(VideoState));
2566
-    if (!is)
2567
-        return NULL;
2568
-    av_strlcpy(is->filename, filename, sizeof(is->filename));
2569
-    is->iformat = iformat;
2570
-    is->ytop = 0;
2571
-    is->xleft = 0;
2572
-
2573
-    /* start video display */
2574
-    is->pictq_mutex = SDL_CreateMutex();
2575
-    is->pictq_cond = SDL_CreateCond();
2576
-
2577
-    is->subpq_mutex = SDL_CreateMutex();
2578
-    is->subpq_cond = SDL_CreateCond();
2579
-
2580
-    is->av_sync_type = av_sync_type;
2581
-    is->parse_tid = SDL_CreateThread(decode_thread, is);
2582
-    if (!is->parse_tid) {
2583
-        av_free(is);
2584
-        return NULL;
2585
-    }
2586
-    return is;
2587
-}
2588
-
2589
-static void stream_cycle_channel(VideoState *is, int codec_type)
2590
-{
2591
-    AVFormatContext *ic = is->ic;
2592
-    int start_index, stream_index;
2593
-    AVStream *st;
2594
-
2595
-    if (codec_type == AVMEDIA_TYPE_VIDEO)
2596
-        start_index = is->video_stream;
2597
-    else if (codec_type == AVMEDIA_TYPE_AUDIO)
2598
-        start_index = is->audio_stream;
2599
-    else
2600
-        start_index = is->subtitle_stream;
2601
-    if (start_index < (codec_type == AVMEDIA_TYPE_SUBTITLE ? -1 : 0))
2602
-        return;
2603
-    stream_index = start_index;
2604
-    for(;;) {
2605
-        if (++stream_index >= is->ic->nb_streams)
2606
-        {
2607
-            if (codec_type == AVMEDIA_TYPE_SUBTITLE)
2608
-            {
2609
-                stream_index = -1;
2610
-                goto the_end;
2611
-            } else
2612
-                stream_index = 0;
2613
-        }
2614
-        if (stream_index == start_index)
2615
-            return;
2616
-        st = ic->streams[stream_index];
2617
-        if (st->codec->codec_type == codec_type) {
2618
-            /* check that parameters are OK */
2619
-            switch(codec_type) {
2620
-            case AVMEDIA_TYPE_AUDIO:
2621
-                if (st->codec->sample_rate != 0 &&
2622
-                    st->codec->channels != 0)
2623
-                    goto the_end;
2624
-                break;
2625
-            case AVMEDIA_TYPE_VIDEO:
2626
-            case AVMEDIA_TYPE_SUBTITLE:
2627
-                goto the_end;
2628
-            default:
2629
-                break;
2630
-            }
2631
-        }
2632
-    }
2633
- the_end:
2634
-    stream_component_close(is, start_index);
2635
-    stream_component_open(is, stream_index);
2636
-}
2637
-
2638
-
2639
-static void toggle_full_screen(void)
2640
-{
2641
-    is_full_screen = !is_full_screen;
2642
-    video_open(cur_stream);
2643
-}
2644
-
2645
-static void toggle_pause(void)
2646
-{
2647
-    if (cur_stream)
2648
-        stream_pause(cur_stream);
2649
-    step = 0;
2650
-}
2651
-
2652
-static void step_to_next_frame(void)
2653
-{
2654
-    if (cur_stream) {
2655
-        /* if the stream is paused unpause it, then step */
2656
-        if (cur_stream->paused)
2657
-            stream_pause(cur_stream);
2658
-    }
2659
-    step = 1;
2660
-}
2661
-
2662
-static void toggle_audio_display(void)
2663
-{
2664
-    if (cur_stream) {
2665
-        int bgcolor = SDL_MapRGB(screen->format, 0x00, 0x00, 0x00);
2666
-        cur_stream->show_audio = (cur_stream->show_audio + 1) % 3;
2667
-        fill_rectangle(screen,
2668
-                    cur_stream->xleft, cur_stream->ytop, cur_stream->width, cur_stream->height,
2669
-                    bgcolor);
2670
-        SDL_UpdateRect(screen, cur_stream->xleft, cur_stream->ytop, cur_stream->width, cur_stream->height);
2671
-    }
2672
-}
2673
-
2674
-/* handle an event sent by the GUI */
2675
-static void event_loop(void)
2676
-{
2677
-    SDL_Event event;
2678
-    double incr, pos, frac;
2679
-
2680
-    for(;;) {
2681
-        double x;
2682
-        SDL_WaitEvent(&event);
2683
-        switch(event.type) {
2684
-        case SDL_KEYDOWN:
2685
-            if (exit_on_keydown) {
2686
-                do_exit();
2687
-                break;
2688
-            }
2689
-            switch(event.key.keysym.sym) {
2690
-            case SDLK_ESCAPE:
2691
-            case SDLK_q:
2692
-                do_exit();
2693
-                break;
2694
-            case SDLK_f:
2695
-                toggle_full_screen();
2696
-                break;
2697
-            case SDLK_p:
2698
-            case SDLK_SPACE:
2699
-                toggle_pause();
2700
-                break;
2701
-            case SDLK_s: //S: Step to next frame
2702
-                step_to_next_frame();
2703
-                break;
2704
-            case SDLK_a:
2705
-                if (cur_stream)
2706
-                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_AUDIO);
2707
-                break;
2708
-            case SDLK_v:
2709
-                if (cur_stream)
2710
-                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_VIDEO);
2711
-                break;
2712
-            case SDLK_t:
2713
-                if (cur_stream)
2714
-                    stream_cycle_channel(cur_stream, AVMEDIA_TYPE_SUBTITLE);
2715
-                break;
2716
-            case SDLK_w:
2717
-                toggle_audio_display();
2718
-                break;
2719
-            case SDLK_LEFT:
2720
-                incr = -10.0;
2721
-                goto do_seek;
2722
-            case SDLK_RIGHT:
2723
-                incr = 10.0;
2724
-                goto do_seek;
2725
-            case SDLK_UP:
2726
-                incr = 60.0;
2727
-                goto do_seek;
2728
-            case SDLK_DOWN:
2729
-                incr = -60.0;
2730
-            do_seek:
2731
-                if (cur_stream) {
2732
-                    if (seek_by_bytes) {
2733
-                        if (cur_stream->video_stream >= 0 && cur_stream->video_current_pos>=0){
2734
-                            pos= cur_stream->video_current_pos;
2735
-                        }else if(cur_stream->audio_stream >= 0 && cur_stream->audio_pkt.pos>=0){
2736
-                            pos= cur_stream->audio_pkt.pos;
2737
-                        }else
2738
-                            pos = avio_tell(cur_stream->ic->pb);
2739
-                        if (cur_stream->ic->bit_rate)
2740
-                            incr *= cur_stream->ic->bit_rate / 8.0;
2741
-                        else
2742
-                            incr *= 180000.0;
2743
-                        pos += incr;
2744
-                        stream_seek(cur_stream, pos, incr, 1);
2745
-                    } else {
2746
-                        pos = get_master_clock(cur_stream);
2747
-                        pos += incr;
2748
-                        stream_seek(cur_stream, (int64_t)(pos * AV_TIME_BASE), (int64_t)(incr * AV_TIME_BASE), 0);
2749
-                    }
2750
-                }
2751
-                break;
2752
-            default:
2753
-                break;
2754
-            }
2755
-            break;
2756
-        case SDL_MOUSEBUTTONDOWN:
2757
-            if (exit_on_mousedown) {
2758
-                do_exit();
2759
-                break;
2760
-            }
2761
-        case SDL_MOUSEMOTION:
2762
-            if(event.type ==SDL_MOUSEBUTTONDOWN){
2763
-                x= event.button.x;
2764
-            }else{
2765
-                if(event.motion.state != SDL_PRESSED)
2766
-                    break;
2767
-                x= event.motion.x;
2768
-            }
2769
-            if (cur_stream) {
2770
-                if(seek_by_bytes || cur_stream->ic->duration<=0){
2771
-                    uint64_t size=  avio_size(cur_stream->ic->pb);
2772
-                    stream_seek(cur_stream, size*x/cur_stream->width, 0, 1);
2773
-                }else{
2774
-                    int64_t ts;
2775
-                    int ns, hh, mm, ss;
2776
-                    int tns, thh, tmm, tss;
2777
-                    tns = cur_stream->ic->duration/1000000LL;
2778
-                    thh = tns/3600;
2779
-                    tmm = (tns%3600)/60;
2780
-                    tss = (tns%60);
2781
-                    frac = x/cur_stream->width;
2782
-                    ns = frac*tns;
2783
-                    hh = ns/3600;
2784
-                    mm = (ns%3600)/60;
2785
-                    ss = (ns%60);
2786
-                    fprintf(stderr, "Seek to %2.0f%% (%2d:%02d:%02d) of total duration (%2d:%02d:%02d)       \n", frac*100,
2787
-                            hh, mm, ss, thh, tmm, tss);
2788
-                    ts = frac*cur_stream->ic->duration;
2789
-                    if (cur_stream->ic->start_time != AV_NOPTS_VALUE)
2790
-                        ts += cur_stream->ic->start_time;
2791
-                    stream_seek(cur_stream, ts, 0, 0);
2792
-                }
2793
-            }
2794
-            break;
2795
-        case SDL_VIDEORESIZE:
2796
-            if (cur_stream) {
2797
-                screen = SDL_SetVideoMode(event.resize.w, event.resize.h, 0,
2798
-                                          SDL_HWSURFACE|SDL_RESIZABLE|SDL_ASYNCBLIT|SDL_HWACCEL);
2799
-                screen_width = cur_stream->width = event.resize.w;
2800
-                screen_height= cur_stream->height= event.resize.h;
2801
-            }
2802
-            break;
2803
-        case SDL_QUIT:
2804
-        case FF_QUIT_EVENT:
2805
-            do_exit();
2806
-            break;
2807
-        case FF_ALLOC_EVENT:
2808
-            video_open(event.user.data1);
2809
-            alloc_picture(event.user.data1);
2810
-            break;
2811
-        case FF_REFRESH_EVENT:
2812
-            video_refresh_timer(event.user.data1);
2813
-            cur_stream->refresh=0;
2814
-            break;
2815
-        default:
2816
-            break;
2817
-        }
2818
-    }
2819
-}
2820
-
2821
-static int opt_frame_size(const char *opt, const char *arg)
2822
-{
2823
-    av_log(NULL, AV_LOG_ERROR,
2824
-           "Option '%s' has been removed, use private format options instead\n", opt);
2825
-    return AVERROR(EINVAL);
2826
-}
2827
-
2828
-static int opt_width(const char *opt, const char *arg)
2829
-{
2830
-    screen_width = parse_number_or_die(opt, arg, OPT_INT64, 1, INT_MAX);
2831
-    return 0;
2832
-}
2833
-
2834
-static int opt_height(const char *opt, const char *arg)
2835
-{
2836
-    screen_height = parse_number_or_die(opt, arg, OPT_INT64, 1, INT_MAX);
2837
-    return 0;
2838
-}
2839
-
2840
-static int opt_format(const char *opt, const char *arg)
2841
-{
2842
-    file_iformat = av_find_input_format(arg);
2843
-    if (!file_iformat) {
2844
-        fprintf(stderr, "Unknown input format: %s\n", arg);
2845
-        return AVERROR(EINVAL);
2846
-    }
2847
-    return 0;
2848
-}
2849
-
2850
-static int opt_frame_pix_fmt(const char *opt, const char *arg)
2851
-{
2852
-    av_log(NULL, AV_LOG_ERROR,
2853
-           "Option '%s' has been removed, use private format options instead\n", opt);
2854
-    return AVERROR(EINVAL);
2855
-}
2856
-
2857
-static int opt_sync(const char *opt, const char *arg)
2858
-{
2859
-    if (!strcmp(arg, "audio"))
2860
-        av_sync_type = AV_SYNC_AUDIO_MASTER;
2861
-    else if (!strcmp(arg, "video"))
2862
-        av_sync_type = AV_SYNC_VIDEO_MASTER;
2863
-    else if (!strcmp(arg, "ext"))
2864
-        av_sync_type = AV_SYNC_EXTERNAL_CLOCK;
2865
-    else {
2866
-        fprintf(stderr, "Unknown value for %s: %s\n", opt, arg);
2867
-        exit(1);
2868
-    }
2869
-    return 0;
2870
-}
2871
-
2872
-static int opt_seek(const char *opt, const char *arg)
2873
-{
2874
-    start_time = parse_time_or_die(opt, arg, 1);
2875
-    return 0;
2876
-}
2877
-
2878
-static int opt_duration(const char *opt, const char *arg)
2879
-{
2880
-    duration = parse_time_or_die(opt, arg, 1);
2881
-    return 0;
2882
-}
2883
-
2884
-static int opt_debug(const char *opt, const char *arg)
2885
-{
2886
-    av_log_set_level(99);
2887
-    debug = parse_number_or_die(opt, arg, OPT_INT64, 0, INT_MAX);
2888
-    return 0;
2889
-}
2890
-
2891
-static int opt_vismv(const char *opt, const char *arg)
2892
-{
2893
-    debug_mv = parse_number_or_die(opt, arg, OPT_INT64, INT_MIN, INT_MAX);
2894
-    return 0;
2895
-}
2896
-
2897
-static int opt_thread_count(const char *opt, const char *arg)
2898
-{
2899
-    thread_count= parse_number_or_die(opt, arg, OPT_INT64, 0, INT_MAX);
2900
-#if !HAVE_THREADS
2901
-    fprintf(stderr, "Warning: not compiled with thread support, using thread emulation\n");
2902
-#endif
2903
-    return 0;
2904
-}
2905
-
2906
-static const OptionDef options[] = {
2907
-#include "cmdutils_common_opts.h"
2908
-    { "x", HAS_ARG, {(void*)opt_width}, "force displayed width", "width" },
2909
-    { "y", HAS_ARG, {(void*)opt_height}, "force displayed height", "height" },
2910
-    { "s", HAS_ARG | OPT_VIDEO, {(void*)opt_frame_size}, "set frame size (WxH or abbreviation)", "size" },
2911
-    { "fs", OPT_BOOL, {(void*)&is_full_screen}, "force full screen" },
2912
-    { "an", OPT_BOOL, {(void*)&audio_disable}, "disable audio" },
2913
-    { "vn", OPT_BOOL, {(void*)&video_disable}, "disable video" },
2914
-    { "ast", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_AUDIO]}, "select desired audio stream", "stream_number" },
2915
-    { "vst", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_VIDEO]}, "select desired video stream", "stream_number" },
2916
-    { "sst", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&wanted_stream[AVMEDIA_TYPE_SUBTITLE]}, "select desired subtitle stream", "stream_number" },
2917
-    { "ss", HAS_ARG, {(void*)&opt_seek}, "seek to a given position in seconds", "pos" },
2918
-    { "t", HAS_ARG, {(void*)&opt_duration}, "play  \"duration\" seconds of audio/video", "duration" },
2919
-    { "bytes", OPT_INT | HAS_ARG, {(void*)&seek_by_bytes}, "seek by bytes 0=off 1=on -1=auto", "val" },
2920
-    { "nodisp", OPT_BOOL, {(void*)&display_disable}, "disable graphical display" },
2921
-    { "f", HAS_ARG, {(void*)opt_format}, "force format", "fmt" },
2922
-    { "pix_fmt", HAS_ARG | OPT_EXPERT | OPT_VIDEO, {(void*)opt_frame_pix_fmt}, "set pixel format", "format" },
2923
-    { "stats", OPT_BOOL | OPT_EXPERT, {(void*)&show_status}, "show status", "" },
2924
-    { "debug", HAS_ARG | OPT_EXPERT, {(void*)opt_debug}, "print specific debug info", "" },
2925
-    { "bug", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&workaround_bugs}, "workaround bugs", "" },
2926
-    { "vismv", HAS_ARG | OPT_EXPERT, {(void*)opt_vismv}, "visualize motion vectors", "" },
2927
-    { "fast", OPT_BOOL | OPT_EXPERT, {(void*)&fast}, "non spec compliant optimizations", "" },
2928
-    { "genpts", OPT_BOOL | OPT_EXPERT, {(void*)&genpts}, "generate pts", "" },
2929
-    { "drp", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&decoder_reorder_pts}, "let decoder reorder pts 0=off 1=on -1=auto", ""},
2930
-    { "lowres", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&lowres}, "", "" },
2931
-    { "skiploop", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_loop_filter}, "", "" },
2932
-    { "skipframe", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_frame}, "", "" },
2933
-    { "skipidct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&skip_idct}, "", "" },
2934
-    { "idct", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&idct}, "set idct algo",  "algo" },
2935
-    { "er", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_recognition}, "set error detection threshold (0-4)",  "threshold" },
2936
-    { "ec", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&error_concealment}, "set error concealment options",  "bit_mask" },
2937
-    { "sync", HAS_ARG | OPT_EXPERT, {(void*)opt_sync}, "set audio-video sync. type (type=audio/video/ext)", "type" },
2938
-    { "threads", HAS_ARG | OPT_EXPERT, {(void*)opt_thread_count}, "thread count", "count" },
2939
-    { "autoexit", OPT_BOOL | OPT_EXPERT, {(void*)&autoexit}, "exit at the end", "" },
2940
-    { "exitonkeydown", OPT_BOOL | OPT_EXPERT, {(void*)&exit_on_keydown}, "exit on key down", "" },
2941
-    { "exitonmousedown", OPT_BOOL | OPT_EXPERT, {(void*)&exit_on_mousedown}, "exit on mouse down", "" },
2942
-    { "loop", OPT_INT | HAS_ARG | OPT_EXPERT, {(void*)&loop}, "set number of times the playback shall be looped", "loop count" },
2943
-    { "framedrop", OPT_BOOL | OPT_EXPERT, {(void*)&framedrop}, "drop frames when cpu is too slow", "" },
2944
-    { "window_title", OPT_STRING | HAS_ARG, {(void*)&window_title}, "set window title", "window title" },
2945
-#if CONFIG_AVFILTER
2946
-    { "vf", OPT_STRING | HAS_ARG, {(void*)&vfilters}, "video filters", "filter list" },
2947
-#endif
2948
-    { "rdftspeed", OPT_INT | HAS_ARG| OPT_AUDIO | OPT_EXPERT, {(void*)&rdftspeed}, "rdft speed", "msecs" },
2949
-    { "default", HAS_ARG | OPT_AUDIO | OPT_VIDEO | OPT_EXPERT, {(void*)opt_default}, "generic catch all option", "" },
2950
-    { "i", 0, {NULL}, "ffmpeg compatibility dummy option", ""},
2951
-    { NULL, },
2952
-};
2953
-
2954
-static void show_usage(void)
2955
-{
2956
-    printf("Simple media player\n");
2957
-    printf("usage: ffplay [options] input_file\n");
2958
-    printf("\n");
2959
-}
2960
-
2961
-static void show_help(void)
2962
-{
2963
-    av_log_set_callback(log_callback_help);
2964
-    show_usage();
2965
-    show_help_options(options, "Main options:\n",
2966
-                      OPT_EXPERT, 0);
2967
-    show_help_options(options, "\nAdvanced options:\n",
2968
-                      OPT_EXPERT, OPT_EXPERT);
2969
-    printf("\n");
2970
-    av_opt_show2(avcodec_opts[0], NULL,
2971
-                 AV_OPT_FLAG_DECODING_PARAM, 0);
2972
-    printf("\n");
2973
-    av_opt_show2(avformat_opts, NULL,
2974
-                 AV_OPT_FLAG_DECODING_PARAM, 0);
2975
-#if !CONFIG_AVFILTER
2976
-    printf("\n");
2977
-    av_opt_show2(sws_opts, NULL,
2978
-                 AV_OPT_FLAG_ENCODING_PARAM, 0);
2979
-#endif
2980
-    printf("\nWhile playing:\n"
2981
-           "q, ESC              quit\n"
2982
-           "f                   toggle full screen\n"
2983
-           "p, SPC              pause\n"
2984
-           "a                   cycle audio channel\n"
2985
-           "v                   cycle video channel\n"
2986
-           "t                   cycle subtitle channel\n"
2987
-           "w                   show audio waves\n"
2988
-           "s                   activate frame-step mode\n"
2989
-           "left/right          seek backward/forward 10 seconds\n"
2990
-           "down/up             seek backward/forward 1 minute\n"
2991
-           "mouse click         seek to percentage in file corresponding to fraction of width\n"
2992
-           );
2993
-}
2994
-
2995
-static void opt_input_file(const char *filename)
2996
-{
2997
-    if (input_filename) {
2998
-        fprintf(stderr, "Argument '%s' provided as input filename, but '%s' was already specified.\n",
2999
-                filename, input_filename);
3000
-        exit(1);
3001
-    }
3002
-    if (!strcmp(filename, "-"))
3003
-        filename = "pipe:";
3004
-    input_filename = filename;
3005
-}
3006
-
3007
-/* Called from the main */
3008
-int main(int argc, char **argv)
3009
-{
3010
-    int flags;
3011
-
3012
-    av_log_set_flags(AV_LOG_SKIP_REPEATED);
3013
-
3014
-    /* register all codecs, demux and protocols */
3015
-    avcodec_register_all();
3016
-#if CONFIG_AVDEVICE
3017
-    avdevice_register_all();
3018
-#endif
3019
-#if CONFIG_AVFILTER
3020
-    avfilter_register_all();
3021
-#endif
3022
-    av_register_all();
3023
-
3024
-    init_opts();
3025
-
3026
-    show_banner();
3027
-
3028
-    parse_options(argc, argv, options, opt_input_file);
3029
-
3030
-    if (!input_filename) {
3031
-        show_usage();
3032
-        fprintf(stderr, "An input file must be specified\n");
3033
-        fprintf(stderr, "Use -h to get full help or, even better, run 'man ffplay'\n");
3034
-        exit(1);
3035
-    }
3036
-
3037
-    if (display_disable) {
3038
-        video_disable = 1;
3039
-    }
3040
-    flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER;
3041
-#if !defined(__MINGW32__) && !defined(__APPLE__)
3042
-    flags |= SDL_INIT_EVENTTHREAD; /* Not supported on Windows or Mac OS X */
3043
-#endif
3044
-    if (SDL_Init (flags)) {
3045
-        fprintf(stderr, "Could not initialize SDL - %s\n", SDL_GetError());
3046
-        exit(1);
3047
-    }
3048
-
3049
-    if (!display_disable) {
3050
-#if HAVE_SDL_VIDEO_SIZE
3051
-        const SDL_VideoInfo *vi = SDL_GetVideoInfo();
3052
-        fs_screen_width = vi->current_w;
3053
-        fs_screen_height = vi->current_h;
3054
-#endif
3055
-    }
3056
-
3057
-    SDL_EventState(SDL_ACTIVEEVENT, SDL_IGNORE);
3058
-    SDL_EventState(SDL_SYSWMEVENT, SDL_IGNORE);
3059
-    SDL_EventState(SDL_USEREVENT, SDL_IGNORE);
3060
-
3061
-    av_init_packet(&flush_pkt);
3062
-    flush_pkt.data= "FLUSH";
3063
-
3064
-    cur_stream = stream_open(input_filename, file_iformat);
3065
-
3066
-    event_loop();
3067
-
3068
-    /* never returns */
3069
-
3070
-    return 0;
3071
-}